Просмотр исходного кода

- modified the Bookmark manager to be installed as a widget tab in Plot Area; fixed the drag & drop function for the table rows that have CellWidgets inside
- marked in gray color the rows in the Bookmark Manager table that will populate the BookMark menu
- made sure that only one instance of the BookmarkManager class is active at one time

Marius Stanciu 6 лет назад
Родитель
Сommit
1ad7b7716b
6 измененных файлов с 516 добавлено и 360 удалено
  1. 52 299
      FlatCAMApp.py
  2. 2 1
      FlatCAMTranslation.py
  3. 3 0
      README.md
  4. 336 0
      flatcamGUI/FlatCAMGUI.py
  5. 112 60
      flatcamGUI/GUIElements.py
  6. 11 0
      flatcamGUI/PreferencesUI.py

+ 52 - 299
FlatCAMApp.py

@@ -474,6 +474,8 @@ class App(QtCore.QObject):
             "global_compression_level": self.ui.general_defaults_form.general_app_group.compress_spinner,
             "global_compression_level": self.ui.general_defaults_form.general_app_group.compress_spinner,
             "global_save_compressed": self.ui.general_defaults_form.general_app_group.save_type_cb,
             "global_save_compressed": self.ui.general_defaults_form.general_app_group.save_type_cb,
 
 
+            "global_bookmarks_limit": self.ui.general_defaults_form.general_app_group.bm_limit_spinner,
+
             # General GUI Preferences
             # General GUI Preferences
             "global_gridx": self.ui.general_defaults_form.general_gui_group.gridx_entry,
             "global_gridx": self.ui.general_defaults_form.general_gui_group.gridx_entry,
             "global_gridy": self.ui.general_defaults_form.general_gui_group.gridy_entry,
             "global_gridy": self.ui.general_defaults_form.general_gui_group.gridy_entry,
@@ -922,6 +924,7 @@ class App(QtCore.QObject):
             "global_recent_limit": 10,  # Max. items in recent list.
             "global_recent_limit": 10,  # Max. items in recent list.
 
 
             "global_bookmarks": dict(),
             "global_bookmarks": dict(),
+            "global_bookmarks_limit": 10,
 
 
             "fit_key": 'V',
             "fit_key": 'V',
             "zoom_out_key": '-',
             "zoom_out_key": '-',
@@ -2436,8 +2439,13 @@ class App(QtCore.QObject):
         # always install tools only after the shell is initialized because the self.inform.emit() depends on shell
         # always install tools only after the shell is initialized because the self.inform.emit() depends on shell
         self.install_tools()
         self.install_tools()
 
 
+        # ##################################################################################
+        # ########################### BookMarks Manager ####################################
+        # ##################################################################################
+
         # install Bookmark Manager and populate bookmarks in the Help -> Bookmarks
         # install Bookmark Manager and populate bookmarks in the Help -> Bookmarks
         self.install_bookmarks()
         self.install_bookmarks()
+        self.book_dialog_tab = BookmarkManager(app=self, storage=self.defaults["global_bookmarks"], parent=self.ui)
 
 
         # ### System Font Parsing ###
         # ### System Font Parsing ###
         # self.f_parse = ParseFont(self)
         # self.f_parse = ParseFont(self)
@@ -4611,13 +4619,23 @@ class App(QtCore.QObject):
 
 
         AboutDialog(self.ui).exec_()
         AboutDialog(self.ui).exec_()
 
 
-    def install_bookmarks(self):
-        # self.ui.menuhelp_bookmarks_manager.triggered.connect(lambda: webbrowser.open(self.app_url))
-        self.defaults["global_bookmarks"].update(
-            {
-                'FlatCAM': "http://flatcam.org"
-            }
-        )
+    def install_bookmarks(self, book_dict=None):
+        """
+        Install the bookmarks actions in the Help menu -> Bookmarks
+
+        :param book_dict: a dict having the actions text as keys and the weblinks as the values
+        :return: None
+        """
+
+        if book_dict is None:
+            self.defaults["global_bookmarks"].update(
+                {
+                    'FlatCAM': "http://flatcam.org"
+                }
+            )
+        else:
+            self.defaults["global_bookmarks"].clear()
+            self.defaults["global_bookmarks"].update(book_dict)
 
 
         # first try to disconnect if somehow they get connected from elsewhere
         # first try to disconnect if somehow they get connected from elsewhere
         for act in self.ui.menuhelp_bookmarks.actions():
         for act in self.ui.menuhelp_bookmarks.actions():
@@ -4626,313 +4644,44 @@ class App(QtCore.QObject):
             except TypeError:
             except TypeError:
                 pass
                 pass
 
 
+            # clear all actions except the last one who is the Bookmark manager
+            if act is self.ui.menuhelp_bookmarks.actions()[-1]:
+                pass
+            else:
+                self.ui.menuhelp_bookmarks.removeAction(act)
+
+        bm_limit = int(self.defaults["global_bookmarks_limit"])
         if self.defaults["global_bookmarks"]:
         if self.defaults["global_bookmarks"]:
-            for title, weblink in self.defaults["global_bookmarks"].items():
+            for title, weblink in list(self.defaults["global_bookmarks"].items())[:bm_limit]:
                 act = QtWidgets.QAction(parent=self.ui.menuhelp_bookmarks)
                 act = QtWidgets.QAction(parent=self.ui.menuhelp_bookmarks)
                 act.setText(title)
                 act.setText(title)
 
 
                 act.setIcon(QtGui.QIcon('share/link16.png'))
                 act.setIcon(QtGui.QIcon('share/link16.png'))
-                act.triggered.connect(lambda: webbrowser.open(weblink))
+                # from here: https://stackoverflow.com/questions/20390323/pyqt-dynamic-generate-qmenu-action-and-connect
+                act.triggered.connect(lambda sig, link=weblink: webbrowser.open(link))
                 self.ui.menuhelp_bookmarks.insertAction(self.ui.menuhelp_bookmarks_manager, act)
                 self.ui.menuhelp_bookmarks.insertAction(self.ui.menuhelp_bookmarks_manager, act)
 
 
         self.ui.menuhelp_bookmarks_manager.triggered.connect(self.on_bookmarks_manager)
         self.ui.menuhelp_bookmarks_manager.triggered.connect(self.on_bookmarks_manager)
 
 
     def on_bookmarks_manager(self):
     def on_bookmarks_manager(self):
 
 
-        class BookDialog(QtWidgets.QDialog):
-            def __init__(self, app, storage, parent=None):
-                super(BookDialog, self).__init__(parent)
-
-                self.app = app
-
-                assert isinstance(storage, dict), "Storage argument is not a dictionary"
-
-                self.bm_dict = storage
-
-                # Icon and title
-                self.setWindowIcon(parent.app_icon)
-                self.setWindowTitle(_("Bookmark Manager"))
-                self.resize(600, 400)
-
-                # title = QtWidgets.QLabel(
-                #     "<font size=8><B>FlatCAM</B></font><BR>"
-                # )
-                # title.setOpenExternalLinks(True)
-
-                # layouts
-                layout = QtWidgets.QVBoxLayout()
-                self.setLayout(layout)
-
-                table_hlay = QtWidgets.QHBoxLayout()
-                layout.addLayout(table_hlay)
-
-                self.table_widget = FCTable()
-                self.table_widget.setSelectionBehavior(QtWidgets.QAbstractItemView.SelectRows)
-                table_hlay.addWidget(self.table_widget)
-
-                self.table_widget.setColumnCount(3)
-                self.table_widget.setColumnWidth(0, 20)
-                self.table_widget.setHorizontalHeaderLabels(
-                    [
-                        '#',
-                        _('Title'),
-                        _('Web Link')
-                    ]
-                )
-                self.table_widget.horizontalHeaderItem(0).setToolTip(
-                    _("Index"))
-                self.table_widget.horizontalHeaderItem(1).setToolTip(
-                    _("Description of the link that is set as an menu action.\n"
-                      "Try to keep it short because it is installed as a menu item."))
-                self.table_widget.horizontalHeaderItem(2).setToolTip(
-                    _("Web Link. E.g: https://your_website.org "))
-
-                # pal = QtGui.QPalette()
-                # pal.setColor(QtGui.QPalette.Background, Qt.white)
-
-                # New Bookmark
-                new_vlay = QtWidgets.QVBoxLayout()
-                layout.addLayout(new_vlay)
-
-                new_title_lbl = QtWidgets.QLabel(_("<b>New Bookmark</b>"))
-                new_vlay.addWidget(new_title_lbl)
-
-                form0 = QtWidgets.QFormLayout()
-                new_vlay.addLayout(form0)
-
-                title_lbl = QtWidgets.QLabel('%s:' % _("Title"))
-                self.title_entry = FCEntry()
-                form0.addRow(title_lbl, self.title_entry)
-
-                link_lbl = QtWidgets.QLabel('%s:' % _("Web Link"))
-                self.link_entry = FCEntry()
-                self.link_entry.set_value('http://')
-                form0.addRow(link_lbl, self.link_entry)
-
-                # Buttons Layout
-                button_hlay = QtWidgets.QHBoxLayout()
-                layout.addLayout(button_hlay)
-
-                add_entry_btn = FCButton(_("Add Entry"))
-                remove_entry_btn = FCButton(_("Remove Entry"))
-                export_list_btn = FCButton(_("Export List"))
-                import_list_btn = FCButton(_("Import List"))
-                closebtn = QtWidgets.QPushButton(_("Close"))
-
-                # button_hlay.addStretch()
-                button_hlay.addWidget(add_entry_btn)
-                button_hlay.addWidget(remove_entry_btn)
-
-                button_hlay.addWidget(export_list_btn)
-                button_hlay.addWidget(import_list_btn)
-                button_hlay.addWidget(closebtn)
-                # ##############################################################################
-                # ######################## SIGNALS #############################################
-                # ##############################################################################
-
-                add_entry_btn.clicked.connect(self.on_add_entry)
-                remove_entry_btn.clicked.connect(self.on_remove_entry)
-                export_list_btn.clicked.connect(self.on_export_bookmarks)
-                import_list_btn.clicked.connect(self.on_import_bookmarks)
-                closebtn.clicked.connect(self.accept)
-
-                self.build_bm_ui()
-
-            def build_bm_ui(self):
-
-                self.table_widget.setRowCount(len(self.bm_dict))
-
-                nr_crt = 0
-                for title, weblink in self.bm_dict.items():
-                    row = nr_crt
-                    nr_crt += 1
-                    id_item = QtWidgets.QTableWidgetItem('%d' % int(nr_crt))
-                    # id.setFlags(QtCore.Qt.ItemIsSelectable | QtCore.Qt.ItemIsEnabled)
-                    self.table_widget.setItem(row, 0, id_item)  # Tool name/id
-
-                    title_item = QtWidgets.QTableWidgetItem(title)
-
-                    self.table_widget.setItem(row, 1, title_item)
-
-                    weblink_txt = QtWidgets.QTextBrowser()
-                    weblink_txt.setOpenExternalLinks(True)
-                    weblink_txt.setFrameStyle(QtWidgets.QFrame.NoFrame)
-                    weblink_txt.document().setDefaultStyleSheet("a{ text-decoration: none; }")
-
-                    weblink_txt.setHtml('<a href=%s>%s</a>' % (weblink, weblink))
-
-                    self.table_widget.setCellWidget(row, 2, weblink_txt)
-
-                    vertical_header = self.table_widget.verticalHeader()
-                    vertical_header.hide()
-
-                    horizontal_header = self.table_widget.horizontalHeader()
-                    horizontal_header.setMinimumSectionSize(10)
-                    horizontal_header.setDefaultSectionSize(70)
-                    horizontal_header.setSectionResizeMode(0, QtWidgets.QHeaderView.Fixed)
-                    horizontal_header.resizeSection(0, 20)
-                    horizontal_header.setSectionResizeMode(1, QtWidgets.QHeaderView.ResizeToContents)
-                    horizontal_header.setSectionResizeMode(2, QtWidgets.QHeaderView.Stretch)
-
-            def on_add_entry(self, **kwargs):
-                """
-                Add a entry in the Bookmark Table and in the menu actions
-                :return: None
-                """
-                if 'title' in kwargs:
-                    title = kwargs['title']
-                else:
-                    title = self.title_entry.get_value()
-                if title == '':
-                    self.app.inform.emit(f'[ERROR_NOTCL] {_("Title entry is empty.")}')
-                    return 'fail'
-
-                if 'link' is kwargs:
-                    link = kwargs['link']
-                else:
-                    link = self.link_entry.get_value()
-                if link == '':
-                    self.app.inform.emit(f'[ERROR_NOTCL] {_("Web link entry is empty.")}')
-                    return 'fail'
-
-                # if 'http' not in link or 'https' not in link:
-                #     link = 'http://' + link
-
-                if title in self.bm_dict.keys() or link in self.bm_dict.values():
-                    self.app.inform.emit(f'[ERROR_NOTCL] {_("Either the Title or the Weblink already in the table.")}')
-                    return 'fail'
-
-                # for some reason if the last char in the weblink is a slash it does not make the link clickable
-                # so I remove it
-                if link[-1] == '/':
-                    link = link[:-1]
-                # add the new entry to storage
-                self.bm_dict[title] = link
-
-                # add the link to the menu
-                act = QtWidgets.QAction(parent=self.app.ui.menuhelp_bookmarks)
-                act.setText(title)
-                act.setIcon(QtGui.QIcon('share/link16.png'))
-                act.triggered.connect(lambda: webbrowser.open(link))
-                self.app.ui.menuhelp_bookmarks.insertAction(self.app.ui.menuhelp_bookmarks_manager, act)
-
-                # add the new entry to the bookmark manager table
-                self.build_bm_ui()
-
-            def on_remove_entry(self):
-                """
-                Remove an Entry in the Bookmark table and from the menu actions
-                :return:
-                """
-                index_list = []
-                for model_index in self.table_widget.selectionModel().selectedRows():
-                    index = QtCore.QPersistentModelIndex(model_index)
-                    index_list.append(index)
-                    title_to_remove = self.table_widget.item(model_index.row(), 1).text()
-
-                    if title_to_remove in list(self.bm_dict.keys()):
-                        # remove from the storage
-                        self.bm_dict.pop(title_to_remove, None)
-
-                        for act in self.app.ui.menuhelp_bookmarks.actions():
-                            if act.text() == title_to_remove:
-                                # disconnect the signal
-                                try:
-                                    act.triggered.disconnect()
-                                except TypeError:
-                                    pass
-                                # remove the action from the menu
-                                self.app.ui.menuhelp_bookmarks.removeAction(act)
-
-                for index in index_list:
-                    self.table_widget.model().removeRow(index.row())
-
-            def on_export_bookmarks(self):
-                self.app.report_usage("on_export_bookmarks")
-                App.log.debug("on_export_bookmarks()")
-
-                date = str(datetime.today()).rpartition('.')[0]
-                date = ''.join(c for c in date if c not in ':-')
-                date = date.replace(' ', '_')
-
-                filter__ = "Text File (*.TXT);;All Files (*.*)"
-                filename, _f = QtWidgets.QFileDialog.getSaveFileName(caption=_("Export FlatCAM Preferences"),
-                                                                     filter=filter__)
-
-                filename = str(filename)
-
-                if filename == "":
-                    self.inform.emit('[WARNING_NOTCL] %s' %
-                                     _("FlatCAM bookmarks export cancelled."))
-                    return
-                else:
-                    try:
-                        f = open(filename, 'w')
-                        f.close()
-                    except PermissionError:
-                        self.app.inform.emit('[WARNING] %s' %
-                                             _("Permission denied, saving not possible.\n"
-                                               "Most likely another app is holding the file open and not accessible."))
-                        return
-                    except IOError:
-                        App.log.debug('Creating a new bookmarks file ...')
-                        f = open(filename, 'w')
-                        f.close()
-                    except:
-                        e = sys.exc_info()[0]
-                        App.log.error("Could not load defaults file.")
-                        App.log.error(str(e))
-                        self.app.inform.emit('[ERROR_NOTCL] %s' %
-                                             _("Could not load bookamrks file."))
-                        return
-
-                    # Save update options
-                    try:
-                        with open(filename, "w") as f:
-                            for title, link in self.bm_dict.items():
-                                line2write = str(title) + ':' + str(link) + '\n'
-                                print(line2write)
-                                f.write(line2write)
-                    except:
-                        self.app.inform.emit('[ERROR_NOTCL] %s' %
-                                             _("Failed to write bookmarks to file."))
-                        return
-                self.app.inform.emit('[success] %s: %s' %
-                                     (_("Exported bookmarks to"), filename))
-
-            def on_import_bookmarks(self):
-                App.log.debug("on_import_bookmarks()")
-
-                filter_ = "Text File (*.txt);;All Files (*.*)"
-                filename, _f = QtWidgets.QFileDialog.getOpenFileName(caption=_("Import FlatCAM Bookmarks"),
-                                                                     filter=filter_)
-
-                filename = str(filename)
-
-                if filename == "":
-                    self.app.inform.emit('[WARNING_NOTCL] %s' %
-                                         _("FlatCAM bookmarks import cancelled."))
-                else:
-                    try:
-                        with open(filename) as f:
-                            bookmarks = f.readlines()
-                    except IOError:
-                        self.app.log.error("Could not load bookamrks file.")
-                        self.app.inform.emit('[ERROR_NOTCL] %s' %
-                                             _("Could not load bookmarks file."))
-                        return
+        for idx in range(self.ui.plot_tab_area.count()):
+            if self.ui.plot_tab_area.tabText(idx) == _("Bookmarks Manager"):
+                # there can be only one instance of Bookmark Manager at one time
+                return
 
 
-                    for line in bookmarks:
-                        proc_line = line.replace(' ', '').partition(':')
-                        self.on_add_entry(title=proc_line[0], link=proc_line[2])
+        # BookDialog(app=self, storage=self.defaults["global_bookmarks"], parent=self.ui).exec_()
+        self.book_dialog_tab = BookmarkManager(app=self, storage=self.defaults["global_bookmarks"], parent=self.ui)
 
 
-                    self.app.inform.emit('[success] %s: %s' %
-                                         (_("Imported Bookmarks from"), filename))
+        # add the tab if it was closed
+        self.ui.plot_tab_area.addTab(self.book_dialog_tab, _("Bookmarks Manager"))
 
 
-            def closeEvent(self, QCloseEvent):
-                super().closeEvent(QCloseEvent)
+        # delete the absolute and relative position and messages in the infobar
+        self.ui.position_label.setText("")
+        self.ui.rel_position_label.setText("")
 
 
-        BookDialog(app=self, storage=self.defaults["global_bookmarks"], parent=self.ui).exec_()
+        # Switch plot_area to preferences page
+        self.ui.plot_tab_area.setCurrentWidget(self.book_dialog_tab)
 
 
     def on_file_savedefaults(self):
     def on_file_savedefaults(self):
         """
         """
@@ -7703,6 +7452,10 @@ class App(QtCore.QObject):
         if title == _("Code Editor"):
         if title == _("Code Editor"):
             self.toggle_codeeditor = False
             self.toggle_codeeditor = False
 
 
+        if title == _("Bookmarks Manager"):
+            self.book_dialog_tab.rebuild_actions()
+            self.book_dialog_tab.deleteLater()
+
     def on_flipy(self):
     def on_flipy(self):
         self.report_usage("on_flipy()")
         self.report_usage("on_flipy()")
 
 

+ 2 - 1
FlatCAMTranslation.py

@@ -8,14 +8,15 @@
 
 
 import os
 import os
 import sys
 import sys
+import logging
 from pathlib import Path
 from pathlib import Path
 
 
 from PyQt5 import QtWidgets, QtGui
 from PyQt5 import QtWidgets, QtGui
 from PyQt5.QtCore import QSettings
 from PyQt5.QtCore import QSettings
 
 
-from flatcamGUI.GUIElements import log
 import gettext
 import gettext
 
 
+log = logging.getLogger('base')
 
 
 # import builtins
 # import builtins
 #
 #

+ 3 - 0
README.md

@@ -14,6 +14,9 @@ CAD program, and create G-Code for Isolation routing.
 - added a Bookmark Manager and a Bookmark menu in the Help Menu
 - added a Bookmark Manager and a Bookmark menu in the Help Menu
 - added an initial support for rows drag and drop in FCTable in GUIElements; it crashes for CellWidgets for now, if CellWidgetsare in the table rows
 - added an initial support for rows drag and drop in FCTable in GUIElements; it crashes for CellWidgets for now, if CellWidgetsare in the table rows
 - fixed some issues in the Bookmark Manager
 - fixed some issues in the Bookmark Manager
+- modified the Bookmark manager to be installed as a widget tab in Plot Area; fixed the drag & drop function for the table rows that have CellWidgets inside
+- marked in gray color the rows in the Bookmark Manager table that will populate the BookMark menu
+- made sure that only one instance of the BookmarkManager class is active at one time
 
 
 10.10.2019
 10.10.2019
 
 

+ 336 - 0
flatcamGUI/FlatCAMGUI.py

@@ -3672,4 +3672,340 @@ class FlatCAMSystemTray(QtWidgets.QSystemTrayIcon):
 
 
         exitAction.triggered.connect(self.app.final_save)
         exitAction.triggered.connect(self.app.final_save)
 
 
+
+class BookmarkManager(QtWidgets.QWidget):
+
+    mark_rows = pyqtSignal()
+
+    def __init__(self, app, storage, parent=None):
+        super(BookmarkManager, self).__init__(parent)
+
+        self.app = app
+
+        assert isinstance(storage, dict), "Storage argument is not a dictionary"
+
+        self.bm_dict = deepcopy(storage)
+
+        # Icon and title
+        # self.setWindowIcon(parent.app_icon)
+        # self.setWindowTitle(_("Bookmark Manager"))
+        # self.resize(600, 400)
+
+        # title = QtWidgets.QLabel(
+        #     "<font size=8><B>FlatCAM</B></font><BR>"
+        # )
+        # title.setOpenExternalLinks(True)
+
+        # layouts
+        layout = QtWidgets.QVBoxLayout()
+        self.setLayout(layout)
+
+        table_hlay = QtWidgets.QHBoxLayout()
+        layout.addLayout(table_hlay)
+
+        self.table_widget = FCTable(drag_drop=True)
+        self.table_widget.setSelectionBehavior(QtWidgets.QAbstractItemView.SelectRows)
+        table_hlay.addWidget(self.table_widget)
+
+        self.table_widget.setColumnCount(3)
+        self.table_widget.setColumnWidth(0, 20)
+        self.table_widget.setHorizontalHeaderLabels(
+            [
+                '#',
+                _('Title'),
+                _('Web Link')
+            ]
+        )
+        self.table_widget.horizontalHeaderItem(0).setToolTip(
+            _("Index.\n"
+              "The rows in gray color will populate the Bookmarks menu.\n"
+              "The number of gray colored rows is set in Preferences."))
+        self.table_widget.horizontalHeaderItem(1).setToolTip(
+            _("Description of the link that is set as an menu action.\n"
+              "Try to keep it short because it is installed as a menu item."))
+        self.table_widget.horizontalHeaderItem(2).setToolTip(
+            _("Web Link. E.g: https://your_website.org "))
+
+        # pal = QtGui.QPalette()
+        # pal.setColor(QtGui.QPalette.Background, Qt.white)
+
+        # New Bookmark
+        new_vlay = QtWidgets.QVBoxLayout()
+        layout.addLayout(new_vlay)
+
+        new_title_lbl = QtWidgets.QLabel(_("<b>New Bookmark</b>"))
+        new_vlay.addWidget(new_title_lbl)
+
+        form0 = QtWidgets.QFormLayout()
+        new_vlay.addLayout(form0)
+
+        title_lbl = QtWidgets.QLabel('%s:' % _("Title"))
+        self.title_entry = FCEntry()
+        form0.addRow(title_lbl, self.title_entry)
+
+        link_lbl = QtWidgets.QLabel('%s:' % _("Web Link"))
+        self.link_entry = FCEntry()
+        self.link_entry.set_value('http://')
+        form0.addRow(link_lbl, self.link_entry)
+
+        # Buttons Layout
+        button_hlay = QtWidgets.QHBoxLayout()
+        layout.addLayout(button_hlay)
+
+        add_entry_btn = FCButton(_("Add Entry"))
+        remove_entry_btn = FCButton(_("Remove Entry"))
+        export_list_btn = FCButton(_("Export List"))
+        import_list_btn = FCButton(_("Import List"))
+        closebtn = QtWidgets.QPushButton(_("Close"))
+
+        # button_hlay.addStretch()
+        button_hlay.addWidget(add_entry_btn)
+        button_hlay.addWidget(remove_entry_btn)
+
+        button_hlay.addWidget(export_list_btn)
+        button_hlay.addWidget(import_list_btn)
+        # button_hlay.addWidget(closebtn)
+        # ##############################################################################
+        # ######################## SIGNALS #############################################
+        # ##############################################################################
+
+        add_entry_btn.clicked.connect(self.on_add_entry)
+        remove_entry_btn.clicked.connect(self.on_remove_entry)
+        export_list_btn.clicked.connect(self.on_export_bookmarks)
+        import_list_btn.clicked.connect(self.on_import_bookmarks)
+        self.title_entry.editingFinished.connect(self.on_add_entry)
+        self.link_entry.editingFinished.connect(self.on_add_entry)
+        # closebtn.clicked.connect(self.accept)
+
+        self.table_widget.drag_drop_sig.connect(self.mark_table_rows_for_actions)
+        self.build_bm_ui()
+
+    def build_bm_ui(self):
+
+        self.table_widget.setRowCount(len(self.bm_dict))
+
+        nr_crt = 0
+        for title, weblink in self.bm_dict.items():
+            row = nr_crt
+            nr_crt += 1
+            id_item = QtWidgets.QTableWidgetItem('%d' % int(nr_crt))
+            # id.setFlags(QtCore.Qt.ItemIsSelectable | QtCore.Qt.ItemIsEnabled)
+            self.table_widget.setItem(row, 0, id_item)  # Tool name/id
+
+            title_item = QtWidgets.QTableWidgetItem(title)
+
+            self.table_widget.setItem(row, 1, title_item)
+
+            weblink_txt = QtWidgets.QTextBrowser()
+            weblink_txt.setOpenExternalLinks(True)
+            weblink_txt.setFrameStyle(QtWidgets.QFrame.NoFrame)
+            weblink_txt.document().setDefaultStyleSheet("a{ text-decoration: none; }")
+
+            weblink_txt.setHtml('<a href=%s>%s</a>' % (weblink, weblink))
+
+            self.table_widget.setCellWidget(row, 2, weblink_txt)
+
+            vertical_header = self.table_widget.verticalHeader()
+            vertical_header.hide()
+
+            horizontal_header = self.table_widget.horizontalHeader()
+            horizontal_header.setMinimumSectionSize(10)
+            horizontal_header.setDefaultSectionSize(70)
+            horizontal_header.setSectionResizeMode(0, QtWidgets.QHeaderView.Fixed)
+            horizontal_header.resizeSection(0, 20)
+            horizontal_header.setSectionResizeMode(1, QtWidgets.QHeaderView.ResizeToContents)
+            horizontal_header.setSectionResizeMode(2, QtWidgets.QHeaderView.Stretch)
+
+        self.mark_table_rows_for_actions()
+
+    def on_add_entry(self, **kwargs):
+        """
+        Add a entry in the Bookmark Table and in the menu actions
+        :return: None
+        """
+        if 'title' in kwargs:
+            title = kwargs['title']
+        else:
+            title = self.title_entry.get_value()
+        if title == '':
+            self.app.inform.emit(f'[ERROR_NOTCL] {_("Title entry is empty.")}')
+            return 'fail'
+
+        if 'link' is kwargs:
+            link = kwargs['link']
+        else:
+            link = self.link_entry.get_value()
+        if link == 'http://':
+            self.app.inform.emit(f'[ERROR_NOTCL] {_("Web link entry is empty.")}')
+            return 'fail'
+
+        # if 'http' not in link or 'https' not in link:
+        #     link = 'http://' + link
+
+        if title in self.bm_dict.keys() or link in self.bm_dict.values():
+            self.app.inform.emit(f'[ERROR_NOTCL] {_("Either the Title or the Weblink already in the table.")}')
+            return 'fail'
+
+        # for some reason if the last char in the weblink is a slash it does not make the link clickable
+        # so I remove it
+        if link[-1] == '/':
+            link = link[:-1]
+        # add the new entry to storage
+        self.bm_dict[title] = link
+
+        # add the link to the menu
+        act = QtWidgets.QAction(parent=self.app.ui.menuhelp_bookmarks)
+        act.setText(title)
+        act.setIcon(QtGui.QIcon('share/link16.png'))
+        act.triggered.connect(lambda: webbrowser.open(link))
+        self.app.ui.menuhelp_bookmarks.insertAction(self.app.ui.menuhelp_bookmarks_manager, act)
+
+        # add the new entry to the bookmark manager table
+        self.build_bm_ui()
+
+    def on_remove_entry(self):
+        """
+        Remove an Entry in the Bookmark table and from the menu actions
+        :return:
+        """
+        index_list = []
+        for model_index in self.table_widget.selectionModel().selectedRows():
+            index = QtCore.QPersistentModelIndex(model_index)
+            index_list.append(index)
+            title_to_remove = self.table_widget.item(model_index.row(), 1).text()
+
+            if title_to_remove in list(self.bm_dict.keys()):
+                # remove from the storage
+                self.bm_dict.pop(title_to_remove, None)
+
+                for act in self.app.ui.menuhelp_bookmarks.actions():
+                    if act.text() == title_to_remove:
+                        # disconnect the signal
+                        try:
+                            act.triggered.disconnect()
+                        except TypeError:
+                            pass
+                        # remove the action from the menu
+                        self.app.ui.menuhelp_bookmarks.removeAction(act)
+
+        # for index in index_list:
+        #     self.table_widget.model().removeRow(index.row())
+        self.build_bm_ui()
+
+    def on_export_bookmarks(self):
+        self.app.report_usage("on_export_bookmarks")
+        self.app.log.debug("on_export_bookmarks()")
+
+        date = str(datetime.today()).rpartition('.')[0]
+        date = ''.join(c for c in date if c not in ':-')
+        date = date.replace(' ', '_')
+
+        filter__ = "Text File (*.TXT);;All Files (*.*)"
+        filename, _f = QtWidgets.QFileDialog.getSaveFileName(caption=_("Export FlatCAM Preferences"),
+                                                             directory=_('{l_save}/FlatCAM_Bookmarks_{date}').format(
+                                                                 l_save=str(self.app.get_last_save_folder()),
+                                                                 date=date),
+                                                             filter=filter__)
+
+        filename = str(filename)
+
+        if filename == "":
+            self.app.inform.emit('[WARNING_NOTCL] %s' % _("FlatCAM bookmarks export cancelled."))
+            return
+        else:
+            try:
+                f = open(filename, 'w')
+                f.close()
+            except PermissionError:
+                self.app.inform.emit('[WARNING] %s' %
+                                     _("Permission denied, saving not possible.\n"
+                                       "Most likely another app is holding the file open and not accessible."))
+                return
+            except IOError:
+                self.app.log.debug('Creating a new bookmarks file ...')
+                f = open(filename, 'w')
+                f.close()
+            except:
+                e = sys.exc_info()[0]
+                self.app.log.error("Could not load defaults file.")
+                self.app.log.error(str(e))
+                self.app.inform.emit('[ERROR_NOTCL] %s' %
+                                     _("Could not load bookamrks file."))
+                return
+
+            # Save update options
+            try:
+                with open(filename, "w") as f:
+                    for title, link in self.bm_dict.items():
+                        line2write = str(title) + ':' + str(link) + '\n'
+                        f.write(line2write)
+            except:
+                self.app.inform.emit('[ERROR_NOTCL] %s' %
+                                     _("Failed to write bookmarks to file."))
+                return
+        self.app.inform.emit('[success] %s: %s' %
+                             (_("Exported bookmarks to"), filename))
+
+    def on_import_bookmarks(self):
+        self.app.log.debug("on_import_bookmarks()")
+
+        filter_ = "Text File (*.txt);;All Files (*.*)"
+        filename, _f = QtWidgets.QFileDialog.getOpenFileName(caption=_("Import FlatCAM Bookmarks"),
+                                                             filter=filter_)
+
+        filename = str(filename)
+
+        if filename == "":
+            self.app.inform.emit('[WARNING_NOTCL] %s' %
+                                 _("FlatCAM bookmarks import cancelled."))
+        else:
+            try:
+                with open(filename) as f:
+                    bookmarks = f.readlines()
+            except IOError:
+                self.app.log.error("Could not load bookamrks file.")
+                self.app.inform.emit('[ERROR_NOTCL] %s' %
+                                     _("Could not load bookmarks file."))
+                return
+
+            for line in bookmarks:
+                proc_line = line.replace(' ', '').partition(':')
+                self.on_add_entry(title=proc_line[0], link=proc_line[2])
+
+            self.app.inform.emit('[success] %s: %s' %
+                                 (_("Imported Bookmarks from"), filename))
+
+    def mark_table_rows_for_actions(self):
+        for row in range(self.table_widget.rowCount()):
+            item_to_paint = self.table_widget.item(row, 0)
+            if row < self.app.defaults["global_bookmarks_limit"]:
+                item_to_paint.setBackground(QtGui.QColor('gray'))
+                # item_to_paint.setForeground(QtGui.QColor('black'))
+            else:
+                item_to_paint.setBackground(QtGui.QColor('white'))
+                # item_to_paint.setForeground(QtGui.QColor('black'))
+
+    def rebuild_actions(self):
+        # rebuild the storage to reflect the order of the lines
+        self.bm_dict.clear()
+        for row in range(self.table_widget.rowCount()):
+            title = self.table_widget.item(row, 1).text()
+            wlink = self.table_widget.cellWidget(row, 2).toPlainText()
+
+            self.bm_dict.update(
+                {
+                    title: wlink
+                }
+            )
+
+        self.app.install_bookmarks(book_dict=self.bm_dict)
+
+    # def accept(self):
+    #     self.rebuild_actions()
+    #     super().accept()
+
+    def closeEvent(self, QCloseEvent):
+        self.rebuild_actions()
+        super().closeEvent(QCloseEvent)
+
 # end of file
 # end of file

+ 112 - 60
flatcamGUI/GUIElements.py

@@ -20,7 +20,10 @@ from copy import copy
 import re
 import re
 import logging
 import logging
 import html
 import html
+import webbrowser
 from copy import deepcopy
 from copy import deepcopy
+import sys
+from datetime import datetime
 
 
 log = logging.getLogger('base')
 log = logging.getLogger('base')
 
 
@@ -1705,6 +1708,9 @@ class OptionalHideInputSection:
 
 
 
 
 class FCTable(QtWidgets.QTableWidget):
 class FCTable(QtWidgets.QTableWidget):
+
+    drag_drop_sig = pyqtSignal()
+
     def __init__(self, drag_drop=False, parent=None):
     def __init__(self, drag_drop=False, parent=None):
         super(FCTable, self).__init__(parent)
         super(FCTable, self).__init__(parent)
 
 
@@ -1763,71 +1769,117 @@ class FCTable(QtWidgets.QTableWidget):
         self.addAction(action)
         self.addAction(action)
         action.triggered.connect(call_function)
         action.triggered.connect(call_function)
 
 
-    def dropEvent(self, event: QtGui.QDropEvent):
-        if not event.isAccepted() and event.source() == self:
-            drop_row = self.drop_on(event)
-
-            rows = sorted(set(item.row() for item in self.selectedItems()))
-            # rows_to_move = [
-            #     [QtWidgets.QTableWidgetItem(self.item(row_index, column_index))
-            #      for column_index in range(self.columnCount())] for row_index in rows
-            # ]
-            self.rows_to_move[:] = []
-            for row_index in rows:
-                row_items = list()
-                for column_index in range(self.columnCount()):
-                    r_item = self.item(row_index, column_index)
-                    w_item = self.cellWidget(row_index, column_index)
-
-                    if r_item is not None:
-                        row_items.append(QtWidgets.QTableWidgetItem(r_item))
-                    elif w_item is not None:
-                        row_items.append(w_item)
-
-                self.rows_to_move.append(row_items)
-
-            for row_index in reversed(rows):
-                self.removeRow(row_index)
-                if row_index < drop_row:
-                    drop_row -= 1
-
-            for row_index, data in enumerate(self.rows_to_move):
-                row_index += drop_row
-                self.insertRow(row_index)
-
-                for column_index, column_data in enumerate(data):
-                    if isinstance(column_data, QtWidgets.QTableWidgetItem):
-                        self.setItem(row_index, column_index, column_data)
-                    else:
-                        self.setCellWidget(row_index, column_index, column_data)
-
-            event.accept()
-            for row_index in range(len(self.rows_to_move)):
-                self.item(drop_row + row_index, 0).setSelected(True)
-                self.item(drop_row + row_index, 1).setSelected(True)
+    # def dropEvent(self, event: QtGui.QDropEvent):
+    #     if not event.isAccepted() and event.source() == self:
+    #         drop_row = self.drop_on(event)
+    #
+    #         rows = sorted(set(item.row() for item in self.selectedItems()))
+    #         # rows_to_move = [
+    #         #     [QtWidgets.QTableWidgetItem(self.item(row_index, column_index))
+    #         #      for column_index in range(self.columnCount())] for row_index in rows
+    #         # ]
+    #         self.rows_to_move[:] = []
+    #         for row_index in rows:
+    #             row_items = list()
+    #             for column_index in range(self.columnCount()):
+    #                 r_item = self.item(row_index, column_index)
+    #                 w_item = self.cellWidget(row_index, column_index)
+    #
+    #                 if r_item is not None:
+    #                     row_items.append(QtWidgets.QTableWidgetItem(r_item))
+    #                 elif w_item is not None:
+    #                     row_items.append(w_item)
+    #
+    #             self.rows_to_move.append(row_items)
+    #
+    #         for row_index in reversed(rows):
+    #             self.removeRow(row_index)
+    #             if row_index < drop_row:
+    #                 drop_row -= 1
+    #
+    #         for row_index, data in enumerate(self.rows_to_move):
+    #             row_index += drop_row
+    #             self.insertRow(row_index)
+    #
+    #             for column_index, column_data in enumerate(data):
+    #                 if isinstance(column_data, QtWidgets.QTableWidgetItem):
+    #                     self.setItem(row_index, column_index, column_data)
+    #                 else:
+    #                     self.setCellWidget(row_index, column_index, column_data)
+    #
+    #         event.accept()
+    #         for row_index in range(len(self.rows_to_move)):
+    #             self.item(drop_row + row_index, 0).setSelected(True)
+    #             self.item(drop_row + row_index, 1).setSelected(True)
+    #
+    #     super().dropEvent(event)
+    #
+    # def drop_on(self, event):
+    #     ret_val = False
+    #     index = self.indexAt(event.pos())
+    #     if not index.isValid():
+    #         return self.rowCount()
+    #
+    #     ret_val = index.row() + 1 if self.is_below(event.pos(), index) else index.row()
+    #
+    #     return ret_val
+    #
+    # def is_below(self, pos, index):
+    #     rect = self.visualRect(index)
+    #     margin = 2
+    #     if pos.y() - rect.top() < margin:
+    #         return False
+    #     elif rect.bottom() - pos.y() < margin:
+    #         return True
+    #     # noinspection PyTypeChecker
+    #     return rect.contains(pos, True) and not (
+    #                 int(self.model().flags(index)) & Qt.ItemIsDropEnabled) and pos.y() >= rect.center().y()
+
+    def dropEvent(self, event):
+        """
+        From here: https://stackoverflow.com/questions/26227885/drag-and-drop-rows-within-qtablewidget
+        :param event:
+        :return:
+        """
+        if event.source() == self:
+            rows = set([mi.row() for mi in self.selectedIndexes()])
+            targetRow = self.indexAt(event.pos()).row()
+            rows.discard(targetRow)
+            rows = sorted(rows)
 
 
-        super().dropEvent(event)
+            if not rows:
+                return
+            if targetRow == -1:
+                targetRow = self.rowCount()
 
 
-    def drop_on(self, event):
-        ret_val = False
-        index = self.indexAt(event.pos())
-        if not index.isValid():
-            return self.rowCount()
+            for _ in range(len(rows)):
+                self.insertRow(targetRow)
 
 
-        ret_val = index.row() + 1 if self.is_below(event.pos(), index) else index.row()
+            rowMapping = dict()  # Src row to target row.
+            for idx, row in enumerate(rows):
+                if row < targetRow:
+                    rowMapping[row] = targetRow + idx
+                else:
+                    rowMapping[row + len(rows)] = targetRow + idx
+
+            colCount = self.columnCount()
+            for srcRow, tgtRow in sorted(rowMapping.items()):
+                for col in range(0, colCount):
+                    new_item = self.item(srcRow, col)
+                    if new_item is None:
+                        new_item = self.cellWidget(srcRow, col)
+                    if isinstance(new_item, QtWidgets.QTableWidgetItem):
+                        new_item = self.takeItem(srcRow, col)
+                        self.setItem(tgtRow, col, new_item)
+                    else:
+                        self.setCellWidget(tgtRow, col, new_item)
 
 
-        return ret_val
+            for row in reversed(sorted(rowMapping.keys())):
+                self.removeRow(row)
+            event.accept()
+            self.drag_drop_sig.emit()
 
 
-    def is_below(self, pos, index):
-        rect = self.visualRect(index)
-        margin = 2
-        if pos.y() - rect.top() < margin:
-            return False
-        elif rect.bottom() - pos.y() < margin:
-            return True
-        # noinspection PyTypeChecker
-        return rect.contains(pos, True) and not (
-                    int(self.model().flags(index)) & Qt.ItemIsDropEnabled) and pos.y() >= rect.center().y()
+            return
 
 
 
 
 class SpinBoxDelegate(QtWidgets.QItemDelegate):
 class SpinBoxDelegate(QtWidgets.QItemDelegate):

+ 11 - 0
flatcamGUI/PreferencesUI.py

@@ -1123,6 +1123,17 @@ class GeneralAppPrefGroupUI(OptionsGroupUI):
 
 
         self.proj_ois = OptionalInputSection(self.save_type_cb, [self.compress_label, self.compress_spinner], True)
         self.proj_ois = OptionalInputSection(self.save_type_cb, [self.compress_label, self.compress_spinner], True)
 
 
+        self.bm_limit_spinner = FCSpinner()
+        self.bm_limit_label = QtWidgets.QLabel('%s:' % _('Bookmarks limit'))
+        self.bm_limit_label.setToolTip(
+            _("The maximum number of bookmarks that may be installed in the menu.\n"
+              "The number of bookmarks in the bookmark manager may be greater\n"
+              "but the menu will hold only so much.")
+        )
+
+        grid0.addWidget(self.bm_limit_label, 18, 0)
+        grid0.addWidget(self.bm_limit_spinner, 18, 1)
+
         self.layout.addStretch()
         self.layout.addStretch()
 
 
         if sys.platform != 'win32':
         if sys.platform != 'win32':