#include "controller.h" #include "app.h" #include "../apps_container.h" #include "../global_preferences.h" extern "C" { #include } namespace Home { Controller::ContentView::ContentView(Controller * controller, SelectableTableViewDataSource * selectionDataSource) : m_selectableTableView(controller, controller, selectionDataSource, controller) { m_selectableTableView.setVerticalCellOverlap(0); m_selectableTableView.setMargins(0, k_sideMargin, k_bottomMargin, k_sideMargin); m_selectableTableView.setBackgroundColor(KDColorWhite); static_cast(m_selectableTableView.decorator())->verticalBar()->setMargin(k_indicatorMargin); } SelectableTableView * Controller::ContentView::selectableTableView() { return &m_selectableTableView; } void Controller::ContentView::drawRect(KDContext * ctx, KDRect rect) const { ctx->fillRect(bounds(), KDColorWhite); } void Controller::ContentView::reloadBottomRow(SimpleTableViewDataSource * dataSource, int numberOfIcons, int numberOfColumns) { if (numberOfIcons % numberOfColumns) { /* We mark the missing icons on the last row as dirty. */ for (int i = 0; i < numberOfColumns; i++) { if (i >= numberOfIcons % numberOfColumns) { markRectAsDirty(KDRect(dataSource->cellWidth()*i, dataSource->cellHeight(), dataSource->cellWidth(), dataSource->cellHeight())); } } } } int Controller::ContentView::numberOfSubviews() const { return 1; } View * Controller::ContentView::subviewAtIndex(int index) { assert(index == 0); return &m_selectableTableView; } void Controller::ContentView::layoutSubviews() { m_selectableTableView.setFrame(bounds()); } Controller::Controller(Responder * parentResponder, SelectableTableViewDataSource * selectionDataSource) : ViewController(parentResponder), m_view(this, selectionDataSource) { } bool Controller::handleEvent(Ion::Events::Event event) { if (event == Ion::Events::OK || event == Ion::Events::EXE) { AppsContainer * container = AppsContainer::sharedAppsContainer(); ::App::Snapshot * selectedSnapshot = container->appSnapshotAtIndex(selectionDataSource()->selectedRow()*k_numberOfColumns+selectionDataSource()->selectedColumn()+1); if (GlobalPreferences::sharedGlobalPreferences()->examMode() == GlobalPreferences::ExamMode::Dutch && selectedSnapshot->descriptor()->name() == I18n::Message::CodeApp) { App::app()->displayWarning(I18n::Message::ForbidenAppInExamMode1, I18n::Message::ForbidenAppInExamMode2); } else { bool switched = container->switchTo(selectedSnapshot); assert(switched); (void) switched; // Silence compilation warning about unused variable. } return true; } if (event == Ion::Events::Home || event == Ion::Events::Back) { return m_view.selectableTableView()->selectCellAtLocation(0,0); } if (event == Ion::Events::Right && selectionDataSource()->selectedRow() < numberOfRows()) { return m_view.selectableTableView()->selectCellAtLocation(0, selectionDataSource()->selectedRow()+1); } if (event == Ion::Events::Left && selectionDataSource()->selectedRow() > 0) { return m_view.selectableTableView()->selectCellAtLocation(numberOfColumns()-1, selectionDataSource()->selectedRow()-1); } return false; } void Controller::didBecomeFirstResponder() { if (selectionDataSource()->selectedRow() == -1) { selectionDataSource()->selectCellAtLocation(0, 0); } Container::activeApp()->setFirstResponder(m_view.selectableTableView()); } void Controller::viewWillAppear() { } View * Controller::view() { return &m_view; } int Controller::numberOfRows() const { return ((numberOfIcons()-1)/k_numberOfColumns)+1; } int Controller::numberOfColumns() const { return k_numberOfColumns; } KDCoordinate Controller::cellHeight() { return k_cellHeight; } KDCoordinate Controller::cellWidth() { return k_cellWidth; } HighlightCell * Controller::reusableCell(int index) { return &m_cells[index]; } int Controller::reusableCellCount() const { return k_maxNumberOfCells; } void Controller::willDisplayCellAtLocation(HighlightCell * cell, int i, int j) { AppCell * appCell = (AppCell *)cell; AppsContainer * container = AppsContainer::sharedAppsContainer(); int appIndex = (j*k_numberOfColumns+i)+1; if (appIndex >= container->numberOfApps()) { appCell->setVisible(false); } else { appCell->setVisible(true); ::App::Descriptor * descriptor = container->appSnapshotAtIndex(appIndex)->descriptor(); appCell->setAppDescriptor(descriptor); } } int Controller::numberOfIcons() const { AppsContainer * container = AppsContainer::sharedAppsContainer(); assert(container->numberOfApps() > 0); return container->numberOfApps() - 1; } void Controller::tableViewDidChangeSelection(SelectableTableView * t, int previousSelectedCellX, int previousSelectedCellY, bool withinTemporarySelection) { AppsContainer * container = AppsContainer::sharedAppsContainer(); if (withinTemporarySelection) { return; } /* If the number of apps (including home) is != 3*n+1, when we display the * lowest icons, the other(s) are empty. As no icon is thus redrawn on the * previous ones, the cell is not cleaned. We need to redraw a white rect on * the cells to hide the leftover icons. Ideally, we would have redrawn all * the background in white and then redraw visible cells. However, the * redrawing takes time and is visible at scrolling. Here, we avoid the * background complete redrawing but the code is a bit * clumsy. */ if (t->selectedRow() == numberOfRows()-1) { m_view.reloadBottomRow(this, container->numberOfApps()-1, k_numberOfColumns); } /* To prevent the selectable table view to select cells that are unvisible, * we reselect the previous selected cell as soon as the selected cell is * unvisible. This trick does not create an endless loop as we ensure not to * stay on a unvisible cell and to initialize the first cell on a visible one * (so the previous one is always visible). */ int appIndex = (t->selectedColumn()+t->selectedRow()*k_numberOfColumns)+1; if (appIndex >= container->numberOfApps()) { t->selectCellAtLocation(previousSelectedCellX, previousSelectedCellY); } } SelectableTableViewDataSource * Controller::selectionDataSource() const { return App::app()->snapshot(); } }