@@ -255,34 +255,57 @@ void MainWindow::setController(MainWindowController* new_controller, bool has_fi
255
255
createHelpMenu ();
256
256
257
257
#if defined(Q_OS_MACOS)
258
- // Defeat Qt's menu text heuristic, as a workaround for QTBUG-30812.
259
- // Changing an action's menu role (to QAction::NoRole) after it was
260
- // added to the menu is unsupported and triggers crashes (#1077).
261
- // Instead, we defeat the heuristic by adding a zero width space at the
262
- // beginning and the end of the text of every action in the menus.
263
- const auto menubar_actions = menuBar ()->actions ();
264
- for (auto action : menubar_actions)
265
- {
266
- if (const auto menu = action->menu ())
267
- {
268
- const auto menu_actions = menu->actions ();
269
- for (auto action : menu_actions)
270
- {
271
- static const auto zwsp = QString::fromUtf8 (" \u200B " );
272
- action->setText (zwsp + action->text () + zwsp);
273
- }
274
- }
275
- }
276
-
277
- // Needed to activate the menu bar changes
278
258
if (isVisible () && qApp->activeWindow () == this )
279
259
{
280
260
// Force a menu synchronisation,
281
261
// QCocoaMenuBar::updateMenuBarImmediately(),
282
262
// via QCocoaNativeInterface::onAppFocusWindowChanged().
263
+ // / \todo Review in Qt > 5.6
283
264
qApp->focusWindowChanged (qApp->focusWindow ());
284
265
}
285
- #endif
266
+
267
+ # if defined(MAPPER_DEVELOPMENT_BUILD)
268
+ {
269
+ // Qt's menu text heuristic can assign unexpected platform specific roles,
270
+ // which resulted in Mapper issue #1067. The only supported solution is
271
+ // assigning QAction::NoRole) before adding items to the menubar.
272
+ // Cf. QTBUG-30812.
273
+ // However, the heuristic is required for some platform-specific items.
274
+ // Cf. detectMenuRole() in qtbase/src/plugins/platforms/cocoa/messages.cpp
275
+ const auto platform_keywords = {
276
+ QCoreApplication::translate (" QCocoaMenuItem" , " Cut" ),
277
+ QCoreApplication::translate (" QCocoaMenuItem" , " Copy" ),
278
+ QCoreApplication::translate (" QCocoaMenuItem" , " Paste" ),
279
+ QCoreApplication::translate (" QCocoaMenuItem" , " Select All" )
280
+ };
281
+ const auto menubar_actions = menuBar ()->actions ();
282
+ for (auto menubar_action : menubar_actions)
283
+ {
284
+ if (const auto menu = menubar_action->menu ())
285
+ {
286
+ const auto menu_actions = menu->actions ();
287
+ for (auto action : menu_actions)
288
+ {
289
+ if (action->menuRole () != QAction::TextHeuristicRole
290
+ || action->isSeparator ())
291
+ continue ;
292
+ const auto text = action->text ().remove (QLatin1Char (' &' ));
293
+ if (std::none_of (begin (platform_keywords), end (platform_keywords), [&text](const auto & keyword) {
294
+ return keyword.compare (text, Qt::CaseInsensitive) == 0 ;
295
+ }))
296
+ {
297
+ // Such warnings may indiciate missing setting of QAction::NoRole
298
+ // on a (new) item, or incomplete translations for Mapper or Qt.
299
+ qDebug (" Unexpected TextHeuristicRole for \" %s > %s\" " ,
300
+ qUtf8Printable (menubar_action->text ()),
301
+ qUtf8Printable (action->text ()));
302
+ }
303
+ }
304
+ }
305
+ }
306
+ }
307
+ # endif // MAPPER_DEVELOPMENT_BUILD
308
+ #endif // Q_OS_MACOS
286
309
287
310
setHasAutosaveConflict (false );
288
311
setHasUnsavedChanges (false );
@@ -291,18 +314,21 @@ void MainWindow::setController(MainWindowController* new_controller, bool has_fi
291
314
void MainWindow::createFileMenu ()
292
315
{
293
316
QAction* new_act = new QAction (QIcon (QString::fromLatin1 (" :/images/new.png" )), tr (" &New" ), this );
317
+ new_act->setMenuRole (QAction::NoRole);
294
318
new_act->setShortcuts (QKeySequence::New);
295
319
new_act->setStatusTip (tr (" Create a new map" ));
296
320
new_act->setWhatsThis (Util::makeWhatThis (" file_menu.html" ));
297
321
connect (new_act, &QAction::triggered, this , &MainWindow::showNewMapWizard);
298
322
299
323
QAction* open_act = new QAction (QIcon (QString::fromLatin1 (" :/images/open.png" )), tr (" &Open..." ), this );
324
+ open_act->setMenuRole (QAction::NoRole);
300
325
open_act->setShortcuts (QKeySequence::Open);
301
326
open_act->setStatusTip (tr (" Open an existing file" ));
302
327
open_act->setWhatsThis (Util::makeWhatThis (" file_menu.html" ));
303
328
connect (open_act, &QAction::triggered, this , &MainWindow::showOpenDialog);
304
329
305
330
open_recent_menu = new QMenu (tr (" Open &recent" ), this );
331
+ open_recent_menu->menuAction ()->setMenuRole (QAction::NoRole);
306
332
open_recent_menu->setWhatsThis (Util::makeWhatThis (" file_menu.html" ));
307
333
for (auto & action : recent_file_act)
308
334
{
@@ -314,11 +340,13 @@ void MainWindow::createFileMenu()
314
340
// NOTE: if you insert something between open_recent_menu and save_act, adjust updateRecentFileActions()!
315
341
316
342
save_act = new QAction (QIcon (QString::fromLatin1 (" :/images/save.png" )), tr (" &Save" ), this );
343
+ save_act->setMenuRole (QAction::NoRole);
317
344
save_act->setShortcuts (QKeySequence::Save);
318
345
save_act->setWhatsThis (Util::makeWhatThis (" file_menu.html" ));
319
346
connect (save_act, &QAction::triggered, this , &MainWindow::save);
320
347
321
348
auto save_as_act = new QAction (tr (" Save &as..." ), this );
349
+ save_as_act->setMenuRole (QAction::NoRole);
322
350
if (QKeySequence::keyBindings (QKeySequence::SaveAs).empty ())
323
351
save_as_act->setShortcut (tr (" Ctrl+Shift+S" ));
324
352
else
@@ -327,20 +355,21 @@ void MainWindow::createFileMenu()
327
355
connect (save_as_act, &QAction::triggered, this , &MainWindow::showSaveAsDialog);
328
356
329
357
settings_act = new QAction (tr (" Settings..." ), this );
330
- settings_act->setShortcut (QKeySequence::Preferences);
331
358
settings_act->setMenuRole (QAction::PreferencesRole);
359
+ settings_act->setShortcut (QKeySequence::Preferences);
332
360
connect (settings_act, &QAction::triggered, this , &MainWindow::showSettings);
333
361
334
362
close_act = new QAction (QIcon (QString::fromLatin1 (" :/images/close.png" )), tr (" Close" ), this );
363
+ close_act->setMenuRole (QAction::NoRole);
335
364
close_act->setShortcut (QKeySequence::Close);
336
365
close_act->setStatusTip (tr (" Close this file" ));
337
366
close_act->setWhatsThis (Util::makeWhatThis (" file_menu.html" ));
338
367
connect (close_act, &QAction::triggered, this , &MainWindow::closeFile);
339
368
340
369
QAction* exit_act = new QAction (tr (" E&xit" ), this );
370
+ exit_act->setMenuRole (QAction::QuitRole);
341
371
exit_act->setShortcuts (QKeySequence::Quit);
342
372
exit_act->setStatusTip (tr (" Exit the application" ));
343
- exit_act->setMenuRole (QAction::QuitRole);
344
373
exit_act->setWhatsThis (Util::makeWhatThis (" file_menu.html" ));
345
374
connect (exit_act, &QAction::triggered, qApp, &QApplication::closeAllWindows);
346
375
@@ -381,25 +410,30 @@ void MainWindow::createHelpMenu()
381
410
{
382
411
// Help menu
383
412
QAction* manualAct = new QAction (QIcon (QString::fromLatin1 (" :/images/help.png" )), tr (" Open &Manual" ), this );
413
+ manualAct->setMenuRole (QAction::NoRole);
384
414
manualAct->setStatusTip (tr (" Show the help file for this application" ));
385
415
manualAct->setShortcut (QKeySequence::HelpContents);
386
416
connect (manualAct, &QAction::triggered, this , &MainWindow::showHelp);
387
417
388
418
QAction* aboutAct = new QAction (tr (" &About %1" ).arg (appName ()), this );
389
- aboutAct->setStatusTip (tr (" Show information about this application" ));
390
419
aboutAct->setMenuRole (QAction::AboutRole);
420
+ aboutAct->setStatusTip (tr (" Show information about this application" ));
391
421
connect (aboutAct, &QAction::triggered, this , &MainWindow::showAbout);
392
422
393
423
QAction* aboutQtAct = new QAction (tr (" About &Qt" ), this );
394
- aboutQtAct->setStatusTip (tr (" Show information about Qt" ));
395
424
aboutQtAct->setMenuRole (QAction::AboutQtRole);
425
+ aboutQtAct->setStatusTip (tr (" Show information about Qt" ));
396
426
connect (aboutQtAct, &QAction::triggered, qApp, QApplication::aboutQt);
397
427
398
428
if (show_menu)
399
429
{
400
430
QMenu* helpMenu = menuBar ()->addMenu (tr (" &Help" ));
401
431
helpMenu->addAction (manualAct);
402
- helpMenu->addAction (QWhatsThis::createAction (this ));
432
+ helpMenu->addAction ([this ] {
433
+ auto action = QWhatsThis::createAction (this );
434
+ action->setMenuRole (QAction::NoRole);
435
+ return action;
436
+ }());
403
437
helpMenu->addSeparator ();
404
438
helpMenu->addAction (aboutAct);
405
439
helpMenu->addAction (aboutQtAct);
0 commit comments