Затем мы снова проходим по списку файла, на этот раз пользуясь индексацией массива. Для каждого файла мы создаем строку из амперсанда, номера файла (j + 1), пробела и имени файла (без пути). Для соответствующего пункта меню мы задаем этот текст. Например, если первым был файл С:\My Documents\tab04.sp, пункт меню первого недавно используемого файла будет иметь текст «&1 tab04.sp».
Рис. 3.11. Меню File со списком файлов, которые открывались недавно.
С каждым пунктом меню recentFileActions может быть связан элемент данных «data» типа QVariant. Тип QVariant может хранить многие типы С++ и Qt; он рассматривается в гл. 11. Здесь в элементе меню «data» мы храним полное имя файла, чтобы позже можно было легко его найти. Мы также делаем этот пункт меню видимым.
Если пунктов меню (массив recentFileActions) больше, чем недавно открытых файлов (массив recentFiles), мы просто не отображаем дополнительные пункты. Наконец, если существует по крайней мере один недавно используемый файл, мы делаем разделитель видимым.
01 void MainWindow::openRecentFile
02 {
03 if (okToContinue) {
04 QAction *action = qobject_cast(sender);
05 if (action)
06 loadFile(action->data. toString);
07 }
08 }
При выборе пользователем какого-нибудь недавно используемого файла вызывается слот openRecentFile. Функция okToContinue используется в том случае, когда имеются несохраненные изменения, и если пользователь не отменил сохранение изменений, мы определяем, какой конкретный пункт меню вызвал слот, используя функцию QObject::sender.
Функция qobject_cast выполняет динамическое приведение типов на основе мета―информации, сгенерированной moc — компилятором мета—объектов Qt. Она возвращает указатель на запрошенный подкласс QObject или 0, если нельзя объект привести к данному типу. В отличие от функции dynamic_cast стандартного С++, функция Qt qobject_cast работает правильно за пределами динамической библиотеки. В нашем примере мы используем qobject_cast для приведения указателя QObject в указатель QAction. Если приведение удачно (а оно должно быть удачным), мы вызываем функцию loadFile, задавая полное имя файла, которое мы извлекаем из элемента данных пункта меню.
Поскольку мы знаем, что слот вызывался объектом QAction, в данном случае программа все же правильно сработала бы при использовании функции static_cast или при традиционном приведении С—типов. (См. раздел «Преобразование типов» в приложении Б, где дается обзор различных методов приведения типов в С++.)
Применение диалоговых окон
В данном разделе мы рассмотрим способы применения диалоговых окон в Qt: как они создаются и инициализируются и как они реагируют на действия пользователя при работе с ними. Мы будем использовать диалоговые окна Find, Go-to-Cell и Sort (найти, перейти в ячейку и сортировать), которые были созданы нами в главе 2. Мы также создадим простое окно About (справка о программе).
Рис. 3.12. Диалоговое окно Find приложения Электронная таблица.
Мы начнем с диалогового окна Find. Поскольку мы хотим, чтобы пользователь имел возможность свободно переключаться с главного окна приложения Электронная таблица на диалоговое окно Find и обратно, это диалоговое окно должно быть немодальным. Немодальным называется окно, которое может работать независимо от других окон приложения.
При создании немодальных диалоговых окон они обычно имеют свои сигналы, соединенные со слотами, которые реагируют на действия пользователя:
01 void MainWindow::find
02 {
03 if (!findDialog) {
04 findDialog = new FindDialog(this);
05 connect(findDialog, SIGNAL (findNext(const QString &,
06 Qt::CaseSensitivity)),
07 spreadsheet, SLOT (findNext(const QString &,
08 Qt::CaseSensitivity)));
09 connect(findDialog, SIGNAL(findPrevious(const QString &,
10 Qt::CaseSensitivity)),