Exemplo n.º 1
0
class TestUIActions(tests.TestCase):
    def setUp(self):
        window = EmptyWindowObject()
        self.notebook = self.setUpNotebook(content={
            'Test': 'Test 123',
            'ExistingPage': 'Exists !'
        })
        self.page = self.notebook.get_page(Path('Test'))
        self.navigation = MockNavigation()
        self.uiactions = UIActions(
            window,
            self.notebook,
            self.page,
            self.navigation,
        )

    def testCreateNewPage(self):
        page = self.notebook.get_page(Path('NewPage'))
        self.assertFalse(page.exists())

        def open_new_page(dialog):
            dialog.set_input(page='NewPage')
            dialog.assert_response_ok()

        with tests.DialogContext(open_new_page):
            self.uiactions.new_page()

        self.assertTrue(page.exists())
        self.assertEqual(self.navigation.opened, page)

    def testCreateNewPageFailsForExistingPage(self):
        from zim.notebook import PageExistsError

        def open_new_page(dialog):
            dialog.set_input(page='ExistingPage')
            self.assertRaises(PageExistsError, dialog.assert_response_ok)

        with tests.DialogContext(open_new_page):
            self.uiactions.new_page()

    def testCreateNewPageWithRelativePaths(self):
        self.uiactions.page = self.notebook.get_page(Path('Test:SubPage'))

        for string, result in (
            ('NewPage', Path('Test:NewPage')),
            (':NewPage', Path('NewPage')),
            ('+NewPage', Path('Test:SubPage:NewPage')),
        ):
            page = self.notebook.get_page(result)
            self.assertFalse(page.exists())

            def open_new_page(dialog):
                dialog.set_input(page=string)
                dialog.assert_response_ok()

            with tests.DialogContext(open_new_page):
                self.uiactions.new_page()

            self.assertTrue(page.exists())
            self.assertEqual(self.navigation.opened, page)

    def testCreateNewChildPage(self):
        page = self.notebook.get_page(Path('Test:Child'))
        self.assertFalse(page.exists())

        def open_new_page(dialog):
            dialog.set_input(page='Child')
            dialog.assert_response_ok()

        with tests.DialogContext(open_new_page):
            self.uiactions.new_sub_page()

        self.assertTrue(page.exists())
        self.assertEqual(self.navigation.opened, page)

    def testOpenAnotherNotebook(self):
        from zim.gui.notebookdialog import NotebookDialog

        def check_dialog_shown(dialog):
            assert isinstance(dialog, NotebookDialog)

        with tests.DialogContext(check_dialog_shown):
            self.uiactions.show_open_notebook()

        # See tests/notebookdialog.py for more testing of the dialog itself

    def testImportPageFromFile(self):
        folder = self.setUpFolder(mock=tests.MOCK_ALWAYS_REAL)
        file = folder.file('TestImport.txt')
        file.write('import test 123')

        def import_file(dialog):
            dialog.set_file(file)
            dialog.assert_response_ok()

        with tests.DialogContext(import_file):
            self.uiactions.import_page()

        page = self.notebook.get_page(Path('TestImport'))
        self.assertTrue(page.exists())
        self.assertEqual(page.dump('plain'), ['import test 123\n'])

    def testOpenNewWindow(self):
        self.uiactions.widget = setUpMainWindow(self.notebook)

        def pagewindow(window):
            window.pageview.page == self.page

        with tests.DialogContext(pagewindow):
            self.uiactions.open_new_window()

    def testOpenNewWindowWithPage(self):
        self.uiactions.widget = setUpMainWindow(self.notebook)

        page = self.notebook.get_page(Path('OtherPage'))
        self.assertNotEqual(page, self.page)

        def pagewindow(window):
            window.pageview.page == page

        with tests.DialogContext(pagewindow):
            self.uiactions.open_new_window(page)

    def testSaveCopyDialog(self):
        folder = self.setUpFolder(mock=tests.MOCK_ALWAYS_REAL)
        file = folder.file('test.txt')

        def savecopy(dialog):
            dialog.set_file(file)
            dialog.assert_response_ok()

        with tests.DialogContext(savecopy):
            self.uiactions.save_copy()

        self.assertEqual(file.read(), 'Test 123\n')

    def testShowExport(self):
        from zim.gui.exportdialog import ExportDialog

        def check_dialog_shown(dialog):
            assert isinstance(dialog, ExportDialog)

        with tests.DialogContext(check_dialog_shown):
            self.uiactions.show_export()

        # See tests/export.py for more testing of the dialog itself

    def testEmailPage(self):
        def check_url(widget, url):
            self.assertEqual(url, 'mailto:?subject=Test&body=Test%20123%0A')

        self.uiactions.email_page(_callback=check_url)

    def testRenamePage(self):
        def renamepage(dialog):
            self.assertEqual(dialog.path, self.page)
            self.assertFalse(dialog.get_input('update'))
            self.assertFalse(dialog.get_input_enabled('update'))
            self.assertFalse(dialog.get_input('head'))  # no matching heading
            dialog.set_input(name='NewName')
            dialog.assert_response_ok()

        with tests.DialogContext(renamepage):
            self.uiactions.move_page()

        page = self.notebook.get_page(Path('Test'))
        self.assertFalse(page.exists())
        page = self.notebook.get_page(Path('NewName'))
        self.assertTrue(page.exists())

    def testRenamePageSameNameInvalidInput(self):
        self.assertEqual(self.page.basename, 'Test')

        def renamepage(dialog):
            dialog.set_input(name='Test')
            self.assertFalse(dialog.do_response_ok())

        with tests.DialogContext(renamepage):
            self.uiactions.move_page()

    def testRenamePageFailsForExistingPage(self):
        from zim.notebook import PageExistsError

        def renamepage(dialog):
            dialog.set_input(name='ExistingPage')
            self.assertRaises(PageExistsError, dialog.do_response_ok)

        with tests.DialogContext(renamepage):
            self.uiactions.move_page()

    def testRenamePageNonExistingPageFails(self):
        from zim.notebook import PageNotFoundError
        page = self.notebook.get_page(Path('NonExistingPage'))

        def renamepage(dialog):
            dialog.set_input(name='NewName')
            self.assertRaises(PageNotFoundError, dialog.do_response_ok)

        with tests.DialogContext(renamepage):
            self.uiactions.move_page(page)

    def testRenamePageWithPageUpdateHeading(self):
        page = self.notebook.get_page(Path('MyPage'))
        page.parse('wiki', ['======= MyPage =======\n', 'Test 123\n'])
        tree = page.get_parsetree()
        self.assertEqual(tree.get_heading_text(), 'MyPage')
        self.notebook.store_page(page)

        def renamepage(dialog):
            self.assertEqual(dialog.path, page)
            self.assertTrue(dialog.get_input(
                'head'))  # dialog should detect matching heading
            dialog.set_input(name='NewName')
            dialog.assert_response_ok()

        with tests.DialogContext(renamepage):
            self.uiactions.move_page(page)

        page = self.notebook.get_page(Path('NewName'))
        tree = page.get_parsetree()
        self.assertEqual(tree.get_heading_text(), 'NewName')

    def testRenamePageWithPageKeepHeading(self):
        page = self.notebook.get_page(Path('MyPage'))
        page.parse('wiki', ['======= MyPage =======\n', 'Test 123\n'])
        tree = page.get_parsetree()
        self.assertEqual(tree.get_heading_text(), 'MyPage')
        self.notebook.store_page(page)

        def renamepage(dialog):
            self.assertEqual(dialog.path, page)
            self.assertTrue(dialog.get_input(
                'head'))  # dialog should detect matching heading
            dialog.set_input(name='NewName', head=False)
            dialog.assert_response_ok()

        with tests.DialogContext(renamepage):
            self.uiactions.move_page(page)

        page = self.notebook.get_page(Path('NewName'))
        tree = page.get_parsetree()
        self.assertEqual(tree.get_heading_text(), 'MyPage')

    def testRenamePageAddHeading(self):
        # Default test page does not have an heading
        tree = self.page.get_parsetree()
        self.assertEqual(tree.get_heading_text(), '')

        def renamepage(dialog):
            dialog.set_input(name='NewName', head=True)
            dialog.assert_response_ok()

        with tests.DialogContext(renamepage):
            self.uiactions.move_page()

        page = self.notebook.get_page(Path('NewName'))
        tree = page.get_parsetree()
        self.assertEqual(tree.get_heading_text(), 'NewName')

    def testRenamePageUpdateLinks(self):
        referrer = self.notebook.get_page(Path('Referrer'))
        referrer.parse('wiki', 'Test [[Test]]\n')
        self.notebook.store_page(referrer)

        def renamepage(dialog):
            self.assertTrue(dialog.get_input_enabled('update'))
            self.assertTrue(dialog.get_input('update'))
            dialog.set_input(name='NewName')
            dialog.assert_response_ok()

        with tests.DialogContext(renamepage):
            self.uiactions.move_page()

        self.assertEqual(referrer.dump('wiki'), ['Test [[NewName]]\n'])

    def testRenamePageNoUpdateLinks(self):
        referrer = self.notebook.get_page(Path('Referrer'))
        referrer.parse('wiki', 'Test [[Test]]\n')
        self.notebook.store_page(referrer)

        def renamepage(dialog):
            self.assertTrue(dialog.get_input_enabled('update'))
            self.assertTrue(dialog.get_input('update'))
            dialog.set_input(name='NewName', update=False)
            dialog.assert_response_ok()

        with tests.DialogContext(renamepage):
            self.uiactions.move_page()

        self.assertEqual(referrer.dump('wiki'), ['Test [[Test]]\n'])

    def testMovePage(self):
        def movepage(dialog):
            self.assertEqual(dialog.path, self.page)
            self.assertFalse(dialog.get_input('update'))
            self.assertFalse(dialog.get_input_enabled('update'))
            dialog.set_input(parent='ExistingPage')
            dialog.assert_response_ok()

        with tests.DialogContext(movepage):
            self.uiactions.move_page()

        page = self.notebook.get_page(Path('Test'))
        self.assertFalse(page.exists())
        page = self.notebook.get_page(Path('ExistingPage:Test'))
        self.assertTrue(page.exists())

    def testMovePageNonExistingParentOK(self):
        page = self.notebook.get_page(Path('NonExistingPage'))
        self.assertFalse(page.exists())

        def movepage(dialog):
            dialog.set_input(parent='NonExistingPage')
            dialog.assert_response_ok()

        with tests.DialogContext(movepage):
            self.uiactions.move_page()

        page = self.notebook.get_page(Path('Test'))
        self.assertFalse(page.exists())
        page = self.notebook.get_page(Path('NonExistingPage:Test'))
        self.assertTrue(page.exists())

    def testMovePageToTopWithPage(self):
        page = self.notebook.get_page(Path('SomeParent:MyPage'))
        page.parse('wiki', 'test 123\n')
        self.notebook.store_page(page)

        def movepage(dialog):
            dialog.set_input(parent=':')
            dialog.assert_response_ok()

        with tests.DialogContext(movepage):
            self.uiactions.move_page(page)

        page = self.notebook.get_page(Path('SomeParent:MyPage'))
        self.assertFalse(page.exists())
        page = self.notebook.get_page(Path('MyPage'))
        self.assertTrue(page.exists())

    def testMovePageSameParentInvalidInput(self):
        page = self.notebook.get_page(Path('SomeParent:MyPage'))
        page.parse('wiki', 'test 123\n')
        self.notebook.store_page(page)

        def movepage(dialog):
            dialog.set_input(parent='SomeParent')
            self.assertFalse(dialog.do_response_ok())

        with tests.DialogContext(movepage):
            self.uiactions.move_page(page)

    def testMovePageFailsForExistingPage(self):
        from zim.notebook import PageExistsError

        page = self.notebook.get_page(Path('SomeParent:ExistingPage'))
        page.parse('wiki', 'test 123\n')
        self.notebook.store_page(page)

        def movepage(dialog):
            dialog.set_input(parent=':')
            self.assertRaises(PageExistsError, dialog.do_response_ok)

        with tests.DialogContext(movepage):
            self.uiactions.move_page(page)

    def testMovePageNonExistingPageFails(self):
        from zim.notebook import PageNotFoundError
        page = self.notebook.get_page(Path('NonExistingPage'))

        def movepage(dialog):
            dialog.set_input(parent='NewParent')
            self.assertRaises(PageNotFoundError, dialog.do_response_ok)

        with tests.DialogContext(movepage):
            self.uiactions.move_page(page)

    def testMovePageUpdateLinks(self):
        referrer = self.notebook.get_page(Path('Referrer'))
        referrer.parse('wiki', 'Test [[Test]]\n')
        self.notebook.store_page(referrer)

        def movepage(dialog):
            self.assertTrue(dialog.get_input_enabled('update'))
            self.assertTrue(dialog.get_input('update'))
            dialog.set_input(parent='NewParent')
            dialog.assert_response_ok()

        with tests.DialogContext(movepage):
            self.uiactions.move_page()

        self.assertEqual(referrer.dump('wiki'), ['Test [[NewParent:Test]]\n'])

    def testMovePageNoUpdateLinks(self):
        referrer = self.notebook.get_page(Path('Referrer'))
        referrer.parse('wiki', 'Test [[Test]]\n')
        self.notebook.store_page(referrer)

        def movepage(dialog):
            self.assertTrue(dialog.get_input_enabled('update'))
            self.assertTrue(dialog.get_input('update'))
            dialog.set_input(parent='NewParent', update=False)
            dialog.assert_response_ok()

        with tests.DialogContext(movepage):
            self.uiactions.move_page()

        self.assertEqual(referrer.dump('wiki'), ['Test [[Test]]\n'])

    def testEditProperties(self):
        from zim.gui.preferencesdialog import PreferencesDialog
        from zim.plugins import PluginManager

        self.uiactions.widget = Gtk.Window()

        def edit_properties(dialog):
            dialog.set_input(home='NewHome')
            dialog.assert_response_ok()

        with tests.DialogContext(edit_properties):
            self.uiactions.show_properties()

        self.assertEqual(self.notebook.config['Notebook']['home'],
                         Path('NewHome'))

    def testEditPropertiesReadOnly(self):
        from zim.gui.preferencesdialog import PreferencesDialog
        from zim.plugins import PluginManager

        self.uiactions.widget = Gtk.Window()

        self.assertFalse(self.notebook.readonly)  # implies attribute exists ..
        self.notebook.readonly = True

        def edit_properties(dialog):
            self.assertFalse(dialog.get_input_enabled('home'))
            dialog.assert_response_ok()

        with tests.DialogContext(edit_properties):
            self.uiactions.show_properties()

    def testPropertiesNotChangedOnCancel(self):
        from zim.gui.preferencesdialog import PreferencesDialog
        from zim.plugins import PluginManager

        self.uiactions.widget = Gtk.Window()

        # In fact this is testig the "cancel" button for all dialogs
        # which have one ..
        def edit_properties(dialog):
            dialog.set_input(home='NewHome')
            dialog.do_response_cancel()

        with tests.DialogContext(edit_properties):
            self.uiactions.show_properties()

        self.assertNotEqual(self.notebook.config['Notebook']['home'],
                            Path('NewHome'))

    def testCopyLocation(self):
        from zim.gui.clipboard import Clipboard

        Clipboard.set_text('XXX')
        self.assertEqual(Clipboard.get_text(), 'XXX')

        self.uiactions.copy_location()
        self.assertEqual(Clipboard.get_text(), 'Test')

    def testShowTemplateEditor(self):
        from zim.gui.templateeditordialog import TemplateEditorDialog
        with tests.DialogContext(TemplateEditorDialog):
            self.uiactions.show_templateeditor()

        # More tests in tests/templateeditordialog.py

    def testShowPreferencesDialog(self):
        from zim.gui.preferencesdialog import PreferencesDialog
        from zim.plugins import PluginManager

        self.uiactions.widget = Gtk.Window()

        with tests.DialogContext(PreferencesDialog):
            self.uiactions.show_preferences()

        # More tests in tests/preferencesdialog.py

    def testShowSearchDialog(self):
        from zim.gui.searchdialog import SearchDialog

        with tests.DialogContext(SearchDialog):
            self.uiactions.show_search()

        # More tests of dialog function in tests/searchdialog.py

    @tests.expectedFailure  # query given after present(), also check callback logic
    def testSearchSection(self):
        from zim.gui.searchdialog import SearchDialog

        for name, text in (
            (self.page.name + ':Child1', 'Test 123'),
            (self.page.name + ':Child2', 'Test 123'),
        ):
            page = self.notebook.get_page(Path(name))
            page.parse('plain', text)
            self.notebook.store_page(page)

        def check_section(dialog):
            results = dialog.results_treeview.get_model()
            self.assertEqual(len(results), 2)
            for row in results:
                self.assertTrue(row[-1].ischild(self.page))
            dialog.assert_response_ok()

        with tests.DialogContext(check_section):
            self.uiactions.show_search_section()

    @tests.expectedFailure  # query given after present(), also check callback logic
    def testSearchBacklinks(self):
        from zim.gui.searchdialog import SearchDialog

        for name, text in (
            ('link1', '[[%s]]\n' % self.page.name),
            ('link2', '[[%s]]\n' % self.page.name),
        ):
            page = self.notebook.get_page(Path(name))
            page.parse('wiki', text)
            self.notebook.store_page(page)

        def check_backlinks(dialog):
            results = dialog.results_treeview.get_model()
            self.assertEqual(len(results), 2)
            for row in results:
                self.assertIn(row[-1].name, ('link1', 'link2'))
            dialog.assert_response_ok()

        with tests.DialogContext(check_backlinks):
            self.uiactions.show_search_backlinks()

    def testShowRecentChangesDialog(self):
        def use_recent_changes(dialog):
            # Check view
            model = dialog.treeview.get_model()
            pages = set(r[0] for r in model)
            self.assertEqual(pages, {'Test', 'ExistingPage'})

            # TODO: how can we check rendering of date column ?

            # Check live update
            page = self.notebook.get_page(Path('NewPage'))
            page.parse('wiki', 'TEst 123')
            self.notebook.store_page(page)

            pages = set(r[0] for r in model)
            self.assertEqual(pages, {'NewPage', 'Test', 'ExistingPage'})

            # Check opening a page
            col = dialog.treeview.get_column(0)
            dialog.treeview.row_activated(Gtk.TreePath((0, )), col)

        with tests.DialogContext(use_recent_changes):
            self.uiactions.show_recent_changes()

        self.assertEqual(self.navigation.opened, Path('NewPage'))

    def testShowServerDialog(self):
        from zim.main import ZIM_APPLICATION
        ZIM_APPLICATION._running = True  # HACK

        from zim.gui.server import ServerWindow
        ServerWindow.show_all = tests.Counter()
        ServerWindow.present = tests.Counter()

        self.uiactions.show_server_gui()

        self.assertEqual(ServerWindow.present.count, 1)

    def testReloadIndex(self):
        self.uiactions.reload_index()

    def testReloadIndexWhileOngoing(self):
        from zim.notebook.operations import ongoing_operation
        from zim.notebook.index import IndexCheckAndUpdateOperation

        op = IndexCheckAndUpdateOperation(self.notebook)
        op_iter = iter(op)
        next(op_iter)
        self.assertEqual(ongoing_operation(self.notebook), op)

        self.uiactions.reload_index()

        self.assertIsNone(ongoing_operation(self.notebook))

    def testEnsureIndexWhileOutOfDate(self):
        self.notebook.index.flush()
        self.assertTrue(self.uiactions.ensure_index_uptodate())

    def testShowManageCustomTools(self):
        from zim.gui.customtools import CustomToolManagerDialog
        with tests.DialogContext(CustomToolManagerDialog):
            self.uiactions.manage_custom_tools()

        # more tests in tests/customtools.py

    def testOpenHelp(self, page=None):
        from zim.main import ZIM_APPLICATION
        ZIM_APPLICATION._running = True  # HACK

        def check_window(window):
            self.assertEqual(window.notebook.folder.basename, 'manual')
            if page:
                self.assertEqual(window.page, page)

        with tests.LoggingFilter('zim', 'Exception while loading plugin:'):
            with tests.WindowContext(
                    check_window,
                    check_window):  # window.present() called twice
                self.uiactions.show_help()

    @tests.expectedFailure  # page opened after window.present
    def testOpenHelpFAQ(self):
        self.testOpenHelp(page='FAQ')

    @tests.expectedFailure  # page opened after window.present
    def testOpenHelpKeys(self):
        self.testOpenHelp(page='Help:Key Bindings')

    @tests.expectedFailure  # page opened after window.present
    def testOpenHelpBugs(self):
        self.testOpenHelp(page='Bugs')

    def testOpenAboutDialog(self):
        from zim.gui.uiactions import MyAboutDialog
        MyAboutDialog.run = tests.Counter()
        self.uiactions.show_about()
        self.assertEqual(MyAboutDialog.run.count, 1)

    def testAccesActionsFromPopupMenu(self):
        # Test depends on first menu item being "new_page_here"
        from zim.gui.uiactions import NewPageDialog
        menu = Gtk.Menu()
        self.uiactions.populate_menu_with_actions(PAGE_EDIT_ACTIONS, menu)

        def open_new_page(dialog):
            self.assertIsInstance(dialog, NewPageDialog)
            dialog.set_input(page='Child')
            dialog.assert_response_ok()

        with tests.DialogContext(open_new_page):
            menu.get_children()[0].activate()

    def testAccesActionsFromPopupMenuForRoot(self):
        # Test depends on first menu item being "new_page_here"
        # When triggered from empty space in index, page will be root namespace
        from zim.gui.uiactions import NewPageDialog
        menu = Gtk.Menu()
        self.uiactions.page = Path(':')
        self.uiactions.populate_menu_with_actions(PAGE_EDIT_ACTIONS, menu)

        def open_new_page(dialog):
            self.assertIsInstance(dialog, NewPageDialog)
            dialog.set_input(page='Child')
            dialog.assert_response_ok()

        with tests.DialogContext(open_new_page):
            menu.get_children()[0].activate()
Exemplo n.º 2
0
class MainWindow(Window):

    # define signals we want to use - (closure type, return type and arg types)
    __gsignals__ = {
        'init-uistate': (GObject.SignalFlags.RUN_LAST, None, ()),
        'page-changed': (GObject.SignalFlags.RUN_LAST, None, (object, )),
        'readonly-changed': (GObject.SignalFlags.RUN_LAST, None, (bool, )),
        'close': (GObject.SignalFlags.RUN_LAST, None, ()),
    }

    def __init__(self, notebook, page=None, fullscreen=False, geometry=None):
        '''Constructor
		@param notebook: the L{Notebook} to show in this window
		@param page: a C{Path} object to open
		@param fullscreen: if C{True} the window is shown fullscreen,
		if C{None} the previous state is restored
		@param geometry: the window geometry as string in format
		"C{WxH+X+Y}", if C{None} the previous state is restored
		'''
        Window.__init__(self)
        self.notebook = notebook
        self.page = None  # will be set later by open_page
        self.navigation = NavigationModel(self)
        self.hideonclose = False

        self.preferences = ConfigManager.preferences['GtkInterface']
        self.preferences.define(
            toggle_on_ctrlspace=Boolean(False),
            remove_links_on_delete=Boolean(True),
            always_use_last_cursor_pos=Boolean(True),
        )
        self.preferences.connect('changed', self.do_preferences_changed)

        self.maximized = False
        self.isfullscreen = False
        self.connect_after('window-state-event',
                           self.__class__.on_window_state_event)

        # Hidden setting to force the gtk bell off. Otherwise it
        # can bell every time you reach the begin or end of the text
        # buffer. Especially specific gtk version on windows.
        # See bug lp:546920
        self.preferences.setdefault('gtk_bell', False)
        if not self.preferences['gtk_bell']:
            Gtk.rc_parse_string('gtk-error-bell = 0')

        self._block_toggle_panes = False
        self._sidepane_autoclose = False
        self._switch_focus_accelgroup = None

        # Catching this signal prevents the window to actually be destroyed
        # when the user tries to close it. The action for close should either
        # hide or destroy the window.
        def do_delete_event(*a):
            logger.debug('Action: close (delete-event)')
            self.close()
            return True  # Do not destroy - let close() handle it

        self.connect('delete-event', do_delete_event)

        # setup uistate
        self.uistate = notebook.state['MainWindow']
        self.uistate.setdefault('windowpos', None, check=value_is_coord)
        self.uistate.setdefault('windowsize', (600, 450), check=value_is_coord)
        self.uistate.setdefault('windowmaximized', False)
        self.uistate.setdefault('active_tabs', None, tuple)
        self.uistate.setdefault('show_toolbar', True)
        self.uistate.setdefault('show_statusbar', True)
        self.uistate.setdefault('readonly', False)

        self.history = History(notebook, notebook.state)

        # init uimanager
        self.uimanager = Gtk.UIManager()
        self.uimanager.add_ui_from_string('''
		<ui>
			<menubar name="menubar">
			</menubar>
			<toolbar name="toolbar">
			</toolbar>
		</ui>
		''')

        # setup menubar and toolbar
        self.add_accel_group(self.uimanager.get_accel_group())
        self.menubar = self.uimanager.get_widget('/menubar')
        self.toolbar = self.uimanager.get_widget('/toolbar')
        self.toolbar.connect('popup-context-menu', self.do_toolbar_popup)
        self.add_bar(self.menubar)
        self.add_bar(self.toolbar)

        self.pageview = PageView(self.notebook, self.navigation)
        self.connect_object('readonly-changed', PageView.set_readonly,
                            self.pageview)
        self.pageview.connect_after('textstyle-changed',
                                    self.on_textview_textstyle_changed)
        self.pageview.textview.connect_after('toggle-overwrite',
                                             self.on_textview_toggle_overwrite)
        self.pageview.textview.connect('link-enter', self.on_link_enter)
        self.pageview.textview.connect('link-leave', self.on_link_leave)

        self.add(self.pageview)

        # create statusbar
        self.statusbar = Gtk.Statusbar()
        self.statusbar.push(0, '<page>')
        self.add_bar(self.statusbar, start=False)
        self.statusbar.set_property('margin', 0)
        self.statusbar.set_property('spacing', 0)

        def statusbar_element(string, size):
            frame = Gtk.Frame()
            frame.set_shadow_type(Gtk.ShadowType.NONE)
            self.statusbar.pack_end(frame, False, True, 0)
            label = Gtk.Label(label=string)
            label.set_size_request(size, 10)
            label.set_alignment(0.1, 0.5)
            frame.add(label)
            return label

        # specify statusbar elements right-to-left
        self.statusbar_insert_label = statusbar_element('INS', 60)
        self.statusbar_style_label = statusbar_element('<style>', 110)

        # and build the widget for backlinks
        self.statusbar_backlinks_button = \
         BackLinksMenuButton(self.notebook, self.open_page, status_bar_style=True)
        frame = Gtk.Frame()
        frame.set_shadow_type(Gtk.ShadowType.NONE)
        self.statusbar.pack_end(Gtk.Separator(), False, True, 0)
        self.statusbar.pack_end(frame, False, True, 0)
        self.statusbar.pack_end(Gtk.Separator(), False, True, 0)
        frame.add(self.statusbar_backlinks_button)

        self.move_bottom_minimized_tabs_to_statusbar(self.statusbar)

        self.do_preferences_changed()

        self._geometry_set = False
        self._set_fullscreen = False
        if geometry:
            try:
                self.parse_geometry(geometry)
                self._geometry_set = True
            except:
                logger.exception('Parsing geometry string failed:')
        elif fullscreen:
            self._set_fullscreen = True

        # Init mouse settings
        self.preferences.setdefault('mouse_nav_button_back', 8)
        self.preferences.setdefault('mouse_nav_button_forw', 9)

        # Finish uimanager
        self._uiactions = UIActions(self, self.notebook, self.page,
                                    self.navigation)
        group = get_gtk_actiongroup(self._uiactions)
        self.uimanager.insert_action_group(group, 0)

        group = get_gtk_actiongroup(self.pageview)
        self.uimanager.insert_action_group(group, 0)

        group = get_gtk_actiongroup(self)
        # don't use mnemonics on macOS to allow alt-<letter> shortcuts
        global MENU_ACTIONS
        if sys.platform == "darwin":
            MENU_ACTIONS = tuple(
                (t[0], t[1], t[2].replace('_', '')) for t in MENU_ACTIONS)
        group.add_actions(MENU_ACTIONS)
        self.uimanager.insert_action_group(group, 0)

        group.get_action('open_page_back').set_sensitive(False)
        group.get_action('open_page_forward').set_sensitive(False)

        fname = 'menubar.xml'
        self.uimanager.add_ui_from_string(data_file(fname).read())
        self.pageview.emit(
            'ui-init')  # Needs to trigger after default menus are build

        # Do this last, else menu items show up in wrong place
        self._customtools = CustomToolManagerUI(self.uimanager, self.pageview)
        self._insertedobjects = InsertedObjectUI(self.uimanager, self.pageview)
        # XXX: would like to do this in PageView itself, but need access to uimanager

        # Setup notebook signals
        notebook.connect('page-info-changed', self.do_page_info_changed)

        def move_away(o, path):
            # Try several options to get awaay
            actions = [
                self.open_page_back, self.open_page_parent, self.open_page_home
            ]
            while (path == self.page or self.page.ischild(path)) and actions:
                action = actions.pop(0)
                action()

        notebook.connect('deleted-page', move_away)  # after action

        def follow(o, path, newpath):
            if path == self.page:
                self.open_page(newpath)
            elif self.page.ischild(path):
                newpath = newpath + self.page.relname(path)
                self.open_page(newpath)
            else:
                pass

        notebook.connect('moved-page', follow)  # after action

        # init page
        page = page or self.history.get_current()
        if page:
            page = notebook.get_page(page)
            self.open_page(page)
        else:
            self.open_page_home()

        self.pageview.grab_focus()

    @action(_('_Close'), '<Primary>W')  # T: Menu item
    def close(self):
        '''Menu action for close. Will hide when L{hideonclose} is set,
		otherwise destroys window, which could result in the application
		closing if there are no other toplevel windows.
		'''
        if self.hideonclose:  # XXX
            self._do_close()
        else:
            self.destroy()

    def _do_close(self):
        self.save_uistate()
        self.hide()
        self.emit('close')

    def destroy(self):
        self.pageview.save_changes()
        if self.page.modified:
            return  # Do not quit if page not saved
        self.pageview.page.set_ui_object(None)  # XXX

        self._do_close()

        while Gtk.events_pending():
            Gtk.main_iteration_do(False)

        self.notebook.index.stop_background_check()
        op = ongoing_operation(self.notebook)
        if op:
            op.wait()

        Window.destroy(self)  # gtk destroy & will also emit destroy signal

    def do_update_statusbar(self, *a):
        page = self.pageview.get_page()
        if not page:
            return
        label = page.name
        if page.modified:
            label += '*'
        if self.notebook.readonly or page.readonly:
            label += ' [' + _('readonly') + ']'  # T: page status in statusbar
        self.statusbar.pop(0)
        self.statusbar.push(0, label)

    def on_window_state_event(self, event):
        if bool(event.changed_mask & Gdk.WindowState.MAXIMIZED):
            self.maximized = bool(event.new_window_state
                                  & Gdk.WindowState.MAXIMIZED)

        if bool(event.changed_mask & Gdk.WindowState.FULLSCREEN):
            self.isfullscreen = bool(event.new_window_state
                                     & Gdk.WindowState.FULLSCREEN)
            self.__class__.toggle_fullscreen.set_toggleaction_state(
                self, self.isfullscreen)

        if bool(event.changed_mask & Gdk.WindowState.MAXIMIZED) \
         or bool(event.changed_mask & Gdk.WindowState.FULLSCREEN):
            schedule_on_idle(lambda: self.pageview.scroll_cursor_on_screen())

    def do_preferences_changed(self, *a):
        if self._switch_focus_accelgroup:
            self.remove_accel_group(self._switch_focus_accelgroup)

        space = Gdk.unicode_to_keyval(ord(' '))
        group = Gtk.AccelGroup()

        self.preferences.setdefault('toggle_on_altspace', False)
        if self.preferences['toggle_on_altspace']:
            # Hidden param, disabled because it causes problems with
            # several international layouts (space mistaken for alt-space,
            # see bug lp:620315)
            group.connect(  # <Alt><Space>
                space, Gdk.ModifierType.MOD1_MASK, Gtk.AccelFlags.VISIBLE,
                self.toggle_sidepane_focus)

        # Toggled by preference menu, also causes issues with international
        # layouts - esp. when switching input method on Meta-Space
        if self.preferences['toggle_on_ctrlspace']:
            group.connect(  # <Primary><Space>
                space, PRIMARY_MODIFIER_MASK, Gtk.AccelFlags.VISIBLE,
                self.toggle_sidepane_focus)

        self.add_accel_group(group)
        self._switch_focus_accelgroup = group

    @toggle_action(_('Menubar'),
                   init=True)  # T: label for View->Menubar menu item
    def toggle_menubar(self, show):
        '''Menu action to toggle the visibility of the menu bar
		@param show: when C{True} or C{False} force the visibility,
		when C{None} toggle based on current state
		'''
        if show:
            self.menubar.set_no_show_all(False)
            self.menubar.show()
        else:
            self.menubar.hide()
            self.menubar.set_no_show_all(True)

    @toggle_action(_('_Toolbar'), init=True)  # T: Menu item
    def toggle_toolbar(self, show):
        '''Menu action to toggle the visibility of the tool bar'''
        if show:
            self.toolbar.set_no_show_all(False)
            self.toolbar.show()
        else:
            self.toolbar.hide()
            self.toolbar.set_no_show_all(True)

        self.uistate['show_toolbar'] = show

    def do_toolbar_popup(self, toolbar, x, y, button):
        '''Show the context menu for the toolbar'''
        menu = self.uimanager.get_widget('/toolbar_popup')
        gtk_popup_at_pointer(menu)

    @toggle_action(_('_Statusbar'), init=True)  # T: Menu item
    def toggle_statusbar(self, show):
        '''Menu action to toggle the visibility of the status bar'''
        if show:
            self.statusbar.set_no_show_all(False)
            self.statusbar.show()
        else:
            self.statusbar.hide()
            self.statusbar.set_no_show_all(True)

        self.uistate['show_statusbar'] = show

    @toggle_action(_('_Fullscreen'), 'F11', icon='gtk-fullscreen',
                   init=False)  # T: Menu item
    def toggle_fullscreen(self, show):
        '''Menu action to toggle the fullscreen state of the window'''
        if show:
            self.fullscreen()
        else:
            self.unfullscreen()

    def do_pane_state_changed(self, pane, *a):
        if not hasattr(self, 'actiongroup') \
        or self._block_toggle_panes:
            return

        action = self.actiongroup.get_action('toggle_panes')
        visible = bool(self.get_visible_panes())
        if visible != action.get_active():
            action.set_active(visible)

    @toggle_action(_('_Side Panes'), 'F9', icon='gtk-index',
                   init=True)  # T: Menu item
    def toggle_panes(self, show):
        '''Menu action to toggle the visibility of the all panes
		@param show: when C{True} or C{False} force the visibility,
		when C{None} toggle based on current state
		'''
        self._block_toggle_panes = True
        Window.toggle_panes(self, show)
        self._block_toggle_panes = False

        if show:
            self.focus_sidepane()
        else:
            self.pageview.grab_focus()

        self._sidepane_autoclose = False
        self.save_uistate()

    def do_set_focus(self, widget):
        Window.do_set_focus(self, widget)
        if widget == self.pageview.textview \
        and self._sidepane_autoclose:
            # Sidepane open and should close automatically
            self.toggle_panes(False)

    def toggle_sidepane_focus(self, *a):
        '''Switch focus between the textview and the page index.
		Automatically opens the sidepane if it is closed
		(but sets a property to automatically close it again).
		This method is used for the (optional) <Primary><Space> keybinding.
		'''
        action = self.actiongroup.get_action('toggle_panes')
        if action.get_active():
            # side pane open
            if self.pageview.textview.is_focus():
                self.focus_sidepane()
            else:
                if self._sidepane_autoclose:
                    self.toggle_panes(False)
                else:
                    self.pageview.grab_focus()
        else:
            # open the pane
            self.toggle_panes(True)
            self._sidepane_autoclose = True

        return True  # stop

    @radio_action(
        None,
        radio_option(TOOLBAR_ICONS_AND_TEXT,
                     _('Icons _And Text')),  # T: Menu item
        radio_option(TOOLBAR_ICONS_ONLY, _('_Icons Only')),  # T: Menu item
        radio_option(TOOLBAR_TEXT_ONLY, _('_Text Only')),  # T: Menu item
    )
    def set_toolbar_style(self, style):
        '''Set the toolbar style
		@param style: can be either:
			- C{TOOLBAR_ICONS_AND_TEXT}
			- C{TOOLBAR_ICONS_ONLY}
			- C{TOOLBAR_TEXT_ONLY}
		'''
        if style == TOOLBAR_ICONS_AND_TEXT:
            self.toolbar.set_style(Gtk.ToolbarStyle.BOTH)
        elif style == TOOLBAR_ICONS_ONLY:
            self.toolbar.set_style(Gtk.ToolbarStyle.ICONS)
        elif style == TOOLBAR_TEXT_ONLY:
            self.toolbar.set_style(Gtk.ToolbarStyle.TEXT)
        else:
            assert False, 'BUG: Unkown toolbar style: %s' % style

        self.preferences['toolbar_style'] = style

    @radio_action(
        None,
        radio_option(TOOLBAR_ICONS_LARGE, _('_Large Icons')),  # T: Menu item
        radio_option(TOOLBAR_ICONS_SMALL, _('_Small Icons')),  # T: Menu item
        radio_option(TOOLBAR_ICONS_TINY, _('_Tiny Icons')),  # T: Menu item
    )
    def set_toolbar_icon_size(self, size):
        '''Set the toolbar style
		@param size: can be either:
			- C{TOOLBAR_ICONS_LARGE}
			- C{TOOLBAR_ICONS_SMALL}
			- C{TOOLBAR_ICONS_TINY}
		'''
        if size == TOOLBAR_ICONS_LARGE:
            self.toolbar.set_icon_size(Gtk.IconSize.LARGE_TOOLBAR)
        elif size == TOOLBAR_ICONS_SMALL:
            self.toolbar.set_icon_size(Gtk.IconSize.SMALL_TOOLBAR)
        elif size == TOOLBAR_ICONS_TINY:
            self.toolbar.set_icon_size(Gtk.IconSize.MENU)
        else:
            assert False, 'BUG: Unkown toolbar size: %s' % size

        self.preferences['toolbar_size'] = size

    @toggle_action(_('Notebook _Editable'), icon='gtk-edit',
                   init=True)  # T: menu item
    def toggle_editable(self, editable):
        '''Menu action to toggle the read-only state of the application
		@emits: readonly-changed
		'''
        readonly = not editable
        if readonly and self.page and self.page.modified:
            # Save any modification now - will not be allowed after switch
            self.pageview.save_changes()

        for group in self.uimanager.get_action_groups():
            for action in group.list_actions():
                if hasattr(action, 'zim_readonly') \
                and not action.zim_readonly:
                    action.set_sensitive(not readonly)

        self.uistate['readonly'] = readonly
        self.emit('readonly-changed', readonly)

    def init_uistate(self):
        # Initialize all the uistate parameters
        # delayed till show or show_all because all this needs real
        # uistate to be in place and plugins to be loaded
        # Run between loading plugins and actually presenting the window to the user

        if not self._geometry_set:
            # Ignore this if an explicit geometry was specified to the constructor
            if self.uistate['windowpos'] is not None:
                x, y = self.uistate['windowpos']
                self.move(x, y)

            w, h = self.uistate['windowsize']
            self.set_default_size(w, h)

            if self.uistate['windowmaximized']:
                self.maximize()

        # For these two "None" means system default, but we don't know what that default is :(
        self.preferences.setdefault(
            'toolbar_style', None,
            (TOOLBAR_ICONS_ONLY, TOOLBAR_ICONS_AND_TEXT, TOOLBAR_TEXT_ONLY))
        self.preferences.setdefault(
            'toolbar_size', None,
            (TOOLBAR_ICONS_TINY, TOOLBAR_ICONS_SMALL, TOOLBAR_ICONS_LARGE))

        self.toggle_toolbar(self.uistate['show_toolbar'])
        self.toggle_statusbar(self.uistate['show_statusbar'])

        Window.init_uistate(self)  # takes care of sidepane positions etc

        if self.preferences['toolbar_style'] is not None:
            self.set_toolbar_style(self.preferences['toolbar_style'])

        if self.preferences['toolbar_size'] is not None:
            self.set_toolbar_icon_size(self.preferences['toolbar_size'])

        self.toggle_fullscreen(self._set_fullscreen)

        if self.notebook.readonly:
            self.toggle_editable(False)
            action = self.actiongroup.get_action('toggle_editable')
            action.set_sensitive(False)
        else:
            self.toggle_editable(not self.uistate['readonly'])

        # And hook to notebook properties
        self.on_notebook_properties_changed(self.notebook.properties)
        self.notebook.properties.connect('changed',
                                         self.on_notebook_properties_changed)

        # Hook up the statusbar
        self.connect('page-changed', self.do_update_statusbar)
        self.connect('readonly-changed', self.do_update_statusbar)
        self.pageview.connect('modified-changed', self.do_update_statusbar)
        self.notebook.connect_after('stored-page', self.do_update_statusbar)

        # Notify plugins
        self.emit('init-uistate')

        # Update menus etc.
        self.uimanager.ensure_update()
        # Prevent flashing when the toolbar is loaded after showing the window
        # and do this before connecting signal below for accelmap.

        # Add search bar onec toolbar is loaded
        space = Gtk.SeparatorToolItem()
        space.set_draw(False)
        space.set_expand(True)
        self.toolbar.insert(space, -1)

        from zim.gui.widgets import InputEntry
        entry = InputEntry(placeholder_text=_('Search'))
        entry.set_icon_from_stock(Gtk.EntryIconPosition.SECONDARY,
                                  Gtk.STOCK_FIND)
        entry.set_icon_activatable(Gtk.EntryIconPosition.SECONDARY, True)
        entry.set_icon_tooltip_text(Gtk.EntryIconPosition.SECONDARY,
                                    _('Search Pages...'))
        # T: label in search entry
        inline_search = lambda e, *a: self._uiactions.show_search(
            query=e.get_text() or None)
        entry.connect('activate', inline_search)
        entry.connect('icon-release', inline_search)
        entry.show()
        item = Gtk.ToolItem()
        item.add(entry)
        self.toolbar.insert(item, -1)

        # Load accelmap config and setup saving it
        # TODO - this probably belongs in the application class, not here
        accelmap = ConfigManager.get_config_file('accelmap').file
        logger.debug('Accelmap: %s', accelmap.path)
        if accelmap.exists():
            Gtk.AccelMap.load(accelmap.path)

        def on_accel_map_changed(o, path, key, mod):
            logger.info('Accelerator changed for %s', path)
            Gtk.AccelMap.save(accelmap.path)

        Gtk.AccelMap.get().connect('changed', on_accel_map_changed)

        self.do_update_statusbar()

    def save_uistate(self):
        if self.is_visible() and not self.isfullscreen:
            self.uistate['windowpos'] = tuple(self.get_position())
            self.uistate['windowsize'] = tuple(self.get_size())
            self.uistate['windowmaximized'] = self.maximized

        Window.save_uistate(self)  # takes care of sidepane positions etc.

        if self.notebook.state.modified:
            self.notebook.state.write()

    def on_notebook_properties_changed(self, properties):
        self.set_title(self.notebook.name + ' - Zim')
        if self.notebook.icon:
            try:
                self.set_icon_from_file(self.notebook.icon)
            except (GObject.GError, GLib.Error):
                logger.exception('Could not load icon %s', self.notebook.icon)

    def on_textview_toggle_overwrite(self, view):
        state = view.get_overwrite()
        if state:
            text = 'OVR'
        else:
            text = 'INS'
        self.statusbar_insert_label.set_text(text)

    def on_textview_textstyle_changed(self, view, styles):
        label = ", ".join([s.title() for s in styles
                           if s]) if styles else 'None'
        self.statusbar_style_label.set_text(label)

    def on_link_enter(self, view, link):
        self.statusbar.push(1, 'Go to "%s"' % link['href'])

    def on_link_leave(self, view, link):
        self.statusbar.pop(1)

    def do_button_press_event(self, event):
        ## Try to capture buttons for navigation
        if event.button > 3:
            if event.button == self.preferences['mouse_nav_button_back']:
                self.open_page_back()
            elif event.button == self.preferences['mouse_nav_button_forw']:
                self.open_page_forward()
            else:
                logger.debug("Unused mouse button %i", event.button)
        #~ return Window.do_button_press_event(self, event)

    def open_page(self, path):
        '''Method to open a page in the mainwindow, and menu action for
		the "jump to" menu item.

		Fails silently when saving current page failed (which is usually the
		result of pressing "cancel" in the error dialog shown when saving
		fails). Check return value for success if you want to be sure.

		@param path: a L{path} for the page to open.
		@raises PageNotFound: if C{path} can not be opened
		@emits: page-changed
		@returns: C{True} for success
		'''
        assert isinstance(path, Path)
        if isinstance(path, Page) and path.valid:
            page = path
        else:
            page = self.notebook.get_page(path)  # can raise

        if self.page and id(self.page) == id(page):
            # XXX: Check ID to enable reload_page but catch all other
            # redundant calls.
            return
        elif self.page:
            self.pageview.save_changes(
            )  # XXX - should connect to signal instead of call here
            self.notebook.wait_for_store_page_async(
            )  # XXX - should not be needed - hide in notebook/page class - how?
            if self.page.modified:
                return False  # Assume SavePageErrorDialog was shown and cancelled

            old_cursor = self.pageview.get_cursor_pos()
            old_scroll = self.pageview.get_scroll_pos()
            self.history.set_state(self.page, old_cursor, old_scroll)

            self.save_uistate()

        logger.info('Open page: %s (%s)', page, path)
        self.page = page
        self._uiactions.page = page

        self.notebook.index.touch_current_page_placeholder(path)

        paths = [page] + list(page.parents())
        self.notebook.index.check_async(self.notebook, paths, recursive=False)

        if isinstance(path, HistoryPath):
            self.history.set_current(path)
            cursor = path.cursor  # can still be None
        else:
            self.history.append(page)
            cursor = None

        if cursor is None and self.preferences['always_use_last_cursor_pos']:
            cursor, _ = self.history.get_state(page)

        self.pageview.set_page(page, cursor)

        self.emit('page-changed', page)

        self.pageview.grab_focus()

    def do_page_changed(self, page):
        #TODO: set toggle_editable() insensitive when page is readonly
        self.update_buttons_history()
        self.update_buttons_hierarchy()
        self.statusbar_backlinks_button.set_page(self.page)

    def do_page_info_changed(self, notebook, page):
        if page == self.page:
            self.update_buttons_hierarchy()

    def update_buttons_history(self):
        historyrecord = self.history.get_current()

        back = self.actiongroup.get_action('open_page_back')
        back.set_sensitive(not historyrecord.is_first)

        forward = self.actiongroup.get_action('open_page_forward')
        forward.set_sensitive(not historyrecord.is_last)

    def update_buttons_hierarchy(self):
        parent = self.actiongroup.get_action('open_page_parent')
        child = self.actiongroup.get_action('open_page_child')
        parent.set_sensitive(len(self.page.namespace) > 0)
        child.set_sensitive(self.page.haschildren)

        previous = self.actiongroup.get_action('open_page_previous')
        next = self.actiongroup.get_action('open_page_next')
        has_prev, has_next = self.notebook.pages.get_has_previous_has_next(
            self.page)
        previous.set_sensitive(has_prev)
        next.set_sensitive(has_next)

    @action(_('_Jump To...'), '<Primary>J')  # T: Menu item
    def show_jump_to(self):
        return OpenPageDialog(self, self.page, self.open_page).run()

    @action(
        _('_Back'),
        verb_icon='gtk-go-back',  # T: Menu item
        accelerator='<alt>Left',
        alt_accelerator='XF86Back')
    def open_page_back(self):
        '''Menu action to open the previous page from the history
		@returns: C{True} if successfull
		'''
        record = self.history.get_previous()
        if not record is None:
            self.open_page(record)

    @action(
        _('_Forward'),
        verb_icon='gtk-go-forward',  # T: Menu item
        accelerator='<alt>Right',
        alt_accelerator='XF86Forward')
    def open_page_forward(self):
        '''Menu action to open the next page from the history
		@returns: C{True} if successfull
		'''
        record = self.history.get_next()
        if not record is None:
            self.open_page(record)

    @action(_('_Parent'), '<alt>Up')  # T: Menu item
    def open_page_parent(self):
        '''Menu action to open the parent page
		@returns: C{True} if successful
		'''
        namespace = self.page.namespace
        if namespace:
            self.open_page(Path(namespace))

    @action(_('_Child'), '<alt>Down')  # T: Menu item
    def open_page_child(self):
        '''Menu action to open a child page. Either takes the last child
		from the history, or the first child.
		@returns: C{True} if successfull
		'''
        path = self.notebook.pages.lookup_by_pagename(self.page)
        # Force refresh "haschildren" ...
        if path.haschildren:
            record = self.history.get_child(path)
            if not record is None:
                self.open_page(record)
            else:
                child = self.notebook.pages.get_next(path)
                self.open_page(child)

    @action(_('_Previous in index'),
            accelerator='<alt>Page_Up')  # T: Menu item
    def open_page_previous(self):
        '''Menu action to open the previous page from the index
		@returns: C{True} if successfull
		'''
        path = self.notebook.pages.get_previous(self.page)
        if not path is None:
            self.open_page(path)

    @action(_('_Next in index'), accelerator='<alt>Page_Down')  # T: Menu item
    def open_page_next(self):
        '''Menu action to open the next page from the index
		@returns: C{True} if successfull
		'''
        path = self.notebook.pages.get_next(self.page)
        if not path is None:
            self.open_page(path)

    @action(_('_Home'), '<alt>Home', icon='gtk-home')  # T: Menu item
    def open_page_home(self):
        '''Menu action to open the home page'''
        self.open_page(self.notebook.get_home_page())

    @action(_('_Reload'), '<Primary>R')  # T: Menu item
    def reload_page(self):
        '''Menu action to reload the current page. Will first try
		to save any unsaved changes, then reload the page from disk.
		'''
        # TODO: this is depending on behavior of open_page(), should be more robust
        pos = self.pageview.get_cursor_pos()
        self.pageview.save_changes()  # XXX
        self.notebook.flush_page_cache(self.page)
        if self.open_page(self.notebook.get_page(self.page)):
            self.pageview.set_cursor_pos(pos)