diff options
Diffstat (limited to 'src/ui/dialog/dialog-container.cpp')
-rw-r--r-- | src/ui/dialog/dialog-container.cpp | 1153 |
1 files changed, 1153 insertions, 0 deletions
diff --git a/src/ui/dialog/dialog-container.cpp b/src/ui/dialog/dialog-container.cpp new file mode 100644 index 0000000..be5e587 --- /dev/null +++ b/src/ui/dialog/dialog-container.cpp @@ -0,0 +1,1153 @@ +// SPDX-License-Identifier: GPL-2.0-or-later + +/** @file + * @brief A widget that manages DialogNotebook's and other widgets inside a horizontal DialogMultipaned. + * + * Authors: see git history + * Tavmjong Bah + * + * Copyright (c) 2018 Tavmjong Bah, Authors + * + * Released under GNU GPL v2+, read the file 'COPYING' for more information. + */ + +#include "dialog-container.h" + +#include <glibmm/i18n.h> +#include <giomm/file.h> +#include <glibmm/keyfile.h> +#include <gtkmm/box.h> +#include <gtkmm/eventbox.h> +#include <gtkmm/image.h> + +#include "enums.h" +#include "inkscape-application.h" +#include "inkscape-window.h" +// #include "ui/dialog/align-and-distribute.h" +#include "ui/dialog/clonetiler.h" +#include "ui/dialog/dialog-data.h" +#include "ui/dialog/dialog-multipaned.h" +#include "ui/dialog/dialog-notebook.h" +#include "ui/dialog/dialog-window.h" +#include "ui/dialog/document-properties.h" +#include "ui/dialog/document-resources.h" +#include "ui/dialog/export.h" +#include "ui/dialog/fill-and-stroke.h" +#include "ui/dialog/filter-effects-dialog.h" +#include "ui/dialog/find.h" +#include "ui/dialog/font-collections-manager.h" +#include "ui/dialog/glyphs.h" +#include "ui/dialog/icon-preview.h" +#include "ui/dialog/inkscape-preferences.h" +#include "ui/dialog/input.h" +#include "ui/dialog/livepatheffect-editor.h" +#include "ui/dialog/memory.h" +#include "ui/dialog/messages.h" +#include "ui/dialog/object-attributes.h" +#include "ui/dialog/object-properties.h" +#include "ui/dialog/objects.h" +#include "ui/dialog/paint-servers.h" +#include "ui/dialog/selectorsdialog.h" +#if WITH_GSPELL +#include "ui/dialog/spellcheck.h" +#endif +#include "ui/dialog/svg-fonts-dialog.h" +#include "ui/dialog/swatches.h" +#include "ui/dialog/symbols.h" +#include "ui/dialog/text-edit.h" +#include "ui/dialog/tile.h" +#include "ui/dialog/tracedialog.h" +#include "ui/dialog/transformation.h" +#include "ui/dialog/undo-history.h" +#include "ui/dialog/xml-tree.h" +#include "ui/icon-names.h" +#include "ui/themes.h" +#include "ui/widget/canvas-grid.h" + +namespace Inkscape { +namespace UI { +namespace Dialog { + +DialogContainer::~DialogContainer() { + // delete columns; desktop widget deletes dialog container before it get "unrealized", + // so it doesn't get a chance to remove them + delete columns; +} + +DialogContainer::DialogContainer(InkscapeWindow* inkscape_window) + : _inkscape_window(inkscape_window) +{ + g_assert(_inkscape_window != nullptr); + + get_style_context()->add_class("DialogContainer"); + + // Setup main column + columns = Gtk::manage(new DialogMultipaned(Gtk::ORIENTATION_HORIZONTAL)); + + connections.emplace_back(columns->signal_prepend_drag_data().connect( + sigc::bind<DialogMultipaned *>(sigc::mem_fun(*this, &DialogContainer::prepend_drop), columns))); + + connections.emplace_back(columns->signal_append_drag_data().connect( + sigc::bind<DialogMultipaned *>(sigc::mem_fun(*this, &DialogContainer::append_drop), columns))); + + // Setup drop targets. + target_entries.emplace_back(Gtk::TargetEntry("GTK_NOTEBOOK_TAB")); + columns->set_target_entries(target_entries); + + add(*columns); + + // Should probably be moved to window. + // connections.emplace_back(signal_unmap().connect(sigc::mem_fun(*this, &DialogContainer::cb_on_unmap))); + + show_all_children(); +} + +DialogMultipaned *DialogContainer::create_column() +{ + DialogMultipaned *column = Gtk::manage(new DialogMultipaned(Gtk::ORIENTATION_VERTICAL)); + + connections.emplace_back(column->signal_prepend_drag_data().connect( + sigc::bind<DialogMultipaned *>(sigc::mem_fun(*this, &DialogContainer::prepend_drop), column))); + + connections.emplace_back(column->signal_append_drag_data().connect( + sigc::bind<DialogMultipaned *>(sigc::mem_fun(*this, &DialogContainer::append_drop), column))); + + connections.emplace_back(column->signal_now_empty().connect( + sigc::bind<DialogMultipaned *>(sigc::mem_fun(*this, &DialogContainer::column_empty), column))); + + column->set_target_entries(target_entries); + + return column; +} + +/** + * Get an instance of a DialogBase dialog using the associated dialog name. + */ +std::unique_ptr<DialogBase> DialogContainer::dialog_factory(Glib::ustring const &dialog_type) +{ + // clang-format off + if (dialog_type == "AlignDistribute") return std::make_unique<ArrangeDialog>(); + else if (dialog_type == "CloneTiler") return std::make_unique<CloneTiler>(); + else if (dialog_type == "DocumentProperties") return std::make_unique<DocumentProperties>(); + else if (dialog_type == "DocumentResources") return std::make_unique<DocumentResources>(); + else if (dialog_type == "Export") return std::make_unique<Export>(); + else if (dialog_type == "FillStroke") return std::make_unique<FillAndStroke>(); + else if (dialog_type == "FilterEffects") return std::make_unique<FilterEffectsDialog>(); + else if (dialog_type == "Find") return std::make_unique<Find>(); + else if (dialog_type == "FontCollections") return std::make_unique<FontCollectionsManager>(); + else if (dialog_type == "Glyphs") return std::make_unique<GlyphsPanel>(); + else if (dialog_type == "IconPreview") return std::make_unique<IconPreviewPanel>(); + else if (dialog_type == "Input") return InputDialog::create(); + else if (dialog_type == "LivePathEffect") return std::make_unique<LivePathEffectEditor>(); + else if (dialog_type == "Memory") return std::make_unique<Memory>(); + else if (dialog_type == "Messages") return std::make_unique<Messages>(); + else if (dialog_type == "ObjectAttributes") return std::make_unique<ObjectAttributes>(); + else if (dialog_type == "ObjectProperties") return std::make_unique<ObjectProperties>(); + else if (dialog_type == "Objects") return std::make_unique<ObjectsPanel>(); + else if (dialog_type == "PaintServers") return std::make_unique<PaintServersDialog>(); + else if (dialog_type == "Preferences") return std::make_unique<InkscapePreferences>(); + else if (dialog_type == "Selectors") return std::make_unique<SelectorsDialog>(); + else if (dialog_type == "SVGFonts") return std::make_unique<SvgFontsDialog>(); + else if (dialog_type == "Swatches") return std::make_unique<SwatchesPanel>(); + else if (dialog_type == "Symbols") return std::make_unique<SymbolsDialog>(); + else if (dialog_type == "Text") return std::make_unique<TextEdit>(); + else if (dialog_type == "Trace") return TraceDialog::create(); + else if (dialog_type == "Transform") return std::make_unique<Transformation>(); + else if (dialog_type == "UndoHistory") return std::make_unique<UndoHistory>(); + else if (dialog_type == "XMLEditor") return std::make_unique<XmlTree>(); +#if WITH_GSPELL + else if (dialog_type == "Spellcheck") return std::make_unique<SpellCheck>(); +#endif +#ifdef DEBUG + else if (dialog_type == "Prototype") return std::make_unique<Prototype>(); +#endif + else { + std::cerr << "DialogContainer::dialog_factory: Unhandled dialog: " << dialog_type.raw() << std::endl; + return nullptr; + } + // clang-format on +} + +// Create the notebook tab +Gtk::Widget *DialogContainer::create_notebook_tab(Glib::ustring label_str, Glib::ustring image_str, const Glib::ustring shortcut) +{ + Gtk::Label *label = Gtk::manage(new Gtk::Label(label_str)); + Gtk::Image *image = Gtk::manage(new Gtk::Image()); + Gtk::Button *close = Gtk::manage(new Gtk::Button()); + image->set_from_icon_name(image_str, Gtk::ICON_SIZE_MENU); + Gtk::Box *tab = Gtk::manage(new Gtk::Box(Gtk::ORIENTATION_HORIZONTAL, 2)); + close->set_image_from_icon_name("window-close"); + close->set_halign(Gtk::ALIGN_END); + close->set_tooltip_text(_("Close Tab")); + close->get_style_context()->add_class("close-button"); + Glib::ustring label_str_fix = label_str; + label_str_fix = Glib::Regex::create("\\W")->replace_literal(label_str_fix, 0, "-", (Glib::RegexMatchFlags)0); + tab->get_style_context()->add_class(label_str_fix); + tab->pack_start(*image); + tab->pack_end(*close); + tab->pack_end(*label); + tab->show_all(); + + // Workaround to the fact that Gtk::Box doesn't receive on_button_press event + Gtk::EventBox *cover = Gtk::manage(new Gtk::EventBox()); + cover->add(*tab); + + // Add shortcut tooltip + if (shortcut.size() > 0) { + auto tlabel = shortcut; + int pos = tlabel.find("&", 0); + if (pos >= 0 && pos < tlabel.length()) { + tlabel.replace(pos, 1, "&"); + } + tab->set_tooltip_markup(label_str + " (<b>" + tlabel + "</b>)"); + } else { + tab->set_tooltip_text(label_str); + } + + return cover; +} + +// find dialog's multipaned parent; is there a better way? +DialogMultipaned* get_dialog_parent(DialogBase* dialog) { + if (!dialog) return nullptr; + + // dialogs are nested inside Gtk::Notebook + if (auto notebook = dynamic_cast<Gtk::Notebook*>(dialog->get_parent())) { + // notebooks are inside viewport, inside scrolled window + if (auto viewport = dynamic_cast<Gtk::Viewport*>(notebook->get_parent())) { + if (auto scroll = dynamic_cast<Gtk::ScrolledWindow*>(viewport->get_parent())) { + // finally get the panel + if (auto panel = dynamic_cast<DialogMultipaned*>(scroll->get_parent())) { + return panel; + } + } + } + } + + return nullptr; +} + +/** + * Add new dialog to the current container or in a floating window, based on preferences. + */ +void DialogContainer::new_dialog(const Glib::ustring& dialog_type ) +{ + // Open all dialogs as floating, if set in preferences + Inkscape::Preferences *prefs = Inkscape::Preferences::get(); + if (prefs == nullptr) { + return; + } + + int dockable = prefs->getInt("/options/dialogtype/value", PREFS_DIALOGS_BEHAVIOR_DOCKABLE); + bool floating = DialogManager::singleton().should_open_floating(dialog_type); + if (dockable == PREFS_DIALOGS_BEHAVIOR_FLOATING || floating) { + new_floating_dialog(dialog_type); + } else { + new_dialog(dialog_type, nullptr); + } + + if (DialogBase* dialog = find_existing_dialog(dialog_type)) { + dialog->focus_dialog(); + } +} + + +DialogBase* DialogContainer::find_existing_dialog(const Glib::ustring& dialog_type) { + DialogBase *existing_dialog = get_dialog(dialog_type); + if (!existing_dialog) { + existing_dialog = DialogManager::singleton().find_floating_dialog(dialog_type); + } + return existing_dialog; +} + +/** + * Overloaded new_dialog + */ +void DialogContainer::new_dialog(const Glib::ustring& dialog_type, DialogNotebook *notebook) +{ + columns->ensure_multipaned_children(); + + // Limit each container to containing one of any type of dialog. + if (DialogBase* existing_dialog = find_existing_dialog(dialog_type)) { + // make sure parent window is not hidden/collapsed + if (auto panel = get_dialog_parent(existing_dialog)) { + panel->show(); + } + // found existing dialog; blink & exit + existing_dialog->blink(); + return; + } + + // Create the dialog widget + DialogBase *dialog = dialog_factory(dialog_type).release(); // Evil, but necessitated by GTK. + + if (!dialog) { + std::cerr << "DialogContainer::new_dialog(): couldn't find dialog for: " << dialog_type.raw() << std::endl; + return; + } + + // manage the dialog instance + dialog = Gtk::manage(dialog); + + // Create the notebook tab + auto const &dialog_data = get_dialog_data(); + Glib::ustring image("inkscape-logo"); + auto it = dialog_data.find(dialog_type); + if (it != dialog_data.end()) { + image = it->second.icon_name; + } + + Glib::ustring label; + Glib::ustring action_name = "win.dialog-open('" + dialog_type + "')"; + auto app = InkscapeApplication::instance(); + std::vector<Glib::ustring> accels = app->gtk_app()->get_accels_for_action(action_name); + if (accels.size() > 0) { + guint key = 0; + Gdk::ModifierType mods; + Gtk::AccelGroup::parse(accels[0], key, mods); + label = Gtk::AccelGroup::get_label(key, mods); + } + + Gtk::Widget *tab = create_notebook_tab(dialog->get_name(), image, label); + + // If not from notebook menu add at top of last column. + if (!notebook) { + // Look to see if last column contains a multipane. If not, add one. + DialogMultipaned *last_column = dynamic_cast<DialogMultipaned *>(columns->get_last_widget()); + if (!last_column) { + last_column = create_column(); + columns->append(last_column); + } + + // Look to see if first widget in column is notebook, if not add one. + notebook = dynamic_cast<DialogNotebook *>(last_column->get_first_widget()); + if (!notebook) { + notebook = Gtk::manage(new DialogNotebook(this)); + last_column->prepend(notebook); + } + } + + // Add dialog + notebook->add_page(*dialog, *tab, dialog->get_name()); + + if (auto panel = dynamic_cast<DialogMultipaned*>(notebook->get_parent())) { + // if panel is collapsed, show it now, or else new dialog will be mysteriously missing + panel->show_all(); + } +} + +// recreate dialogs hosted (docked) in a floating DialogWindow; window will be created +bool DialogContainer::recreate_dialogs_from_state(InkscapeWindow* inkscape_window, const Glib::KeyFile* keyfile) +{ + g_assert(inkscape_window != nullptr); + + bool restored = false; + // Step 1: check if we want to load the state + Inkscape::Preferences *prefs = Inkscape::Preferences::get(); + int save_state = prefs->getInt("/options/savedialogposition/value", PREFS_DIALOGS_STATE_SAVE); + if (save_state == PREFS_DIALOGS_STATE_NONE) { + return restored; // User has turned off this feature in Preferences + } + + // if it isn't dockable, all saved docked dialogs are made floating + bool is_dockable = + prefs->getInt("/options/dialogtype/value", PREFS_DIALOGS_BEHAVIOR_DOCKABLE) != PREFS_DIALOGS_BEHAVIOR_FLOATING; + + if (!is_dockable) + return false; // not applicable if docking is off + + // Step 2: get the number of windows; should be 1 + int windows_count = 0; + try { + // we may have no 'Windows' initially when recreating floating dialog state (state is empty) + if (keyfile->has_group("Windows") && keyfile->has_key("Windows", "Count")) { + windows_count = keyfile->get_integer("Windows", "Count"); + } + } catch (Glib::Error &error) { + std::cerr << G_STRFUNC << ": " << error.what().raw() << std::endl; + } + + // Step 3: for each window, load its state. + for (int window_idx = 0; window_idx < windows_count; ++window_idx) { + Glib::ustring group_name = "Window" + std::to_string(window_idx); + + bool has_position = keyfile->has_key(group_name, "Position") && keyfile->get_boolean(group_name, "Position"); + window_position_t pos; + if (has_position) { // floating window position recorded? + pos.x = keyfile->get_integer(group_name, "x"); + pos.y = keyfile->get_integer(group_name, "y"); + pos.width = keyfile->get_integer(group_name, "width"); + pos.height = keyfile->get_integer(group_name, "height"); + } + // Step 3.0: read the window parameters + int column_count = 0; + try { + column_count = keyfile->get_integer(group_name, "ColumnCount"); + } catch (Glib::Error &error) { + std::cerr << G_STRFUNC << ": " << error.what().raw() << std::endl; + } + + // Step 3.1: get the window's container columns where we want to create the dialogs + DialogWindow *dialog_window = new DialogWindow(inkscape_window, nullptr); + DialogContainer *active_container = dialog_window->get_container(); + DialogMultipaned *active_columns = active_container ? active_container->get_columns() : nullptr; + + if (!active_container || !active_columns) { + continue; + } + + // Step 3.2: for each column, load its state + for (int column_idx = 0; column_idx < column_count; ++column_idx) { + Glib::ustring column_group_name = group_name + "Column" + std::to_string(column_idx); + + // Step 3.2.0: read the column parameters + int notebook_count = 0; + bool before_canvas = false; + try { + notebook_count = keyfile->get_integer(column_group_name, "NotebookCount"); + if (keyfile->has_key(column_group_name, "BeforeCanvas")) { + before_canvas = keyfile->get_boolean(column_group_name, "BeforeCanvas"); + } + } catch (Glib::Error &error) { + std::cerr << G_STRFUNC << ": " << error.what().raw() << std::endl; + } + + // Step 3.2.1: create the column + DialogMultipaned *column = active_container->create_column(); + + before_canvas ? active_columns->prepend(column) : active_columns->append(column); + + // Step 3.2.2: for each noteboook, load its dialogs + for (int notebook_idx = 0; notebook_idx < notebook_count; ++notebook_idx) { + Glib::ustring key = "Notebook" + std::to_string(notebook_idx) + "Dialogs"; + + // Step 3.2.2.0 read the list of dialogs in the current notebook + std::vector<Glib::ustring> dialogs; + try { + dialogs = keyfile->get_string_list(column_group_name, key); + } catch (Glib::Error &error) { + std::cerr << G_STRFUNC << ": " << error.what().raw() << std::endl; + } + + if (!dialogs.size()) { + continue; + } + + DialogNotebook *notebook = nullptr; + auto const &dialog_data = get_dialog_data(); + + // Step 3.2.2.1 create each dialog in the current notebook + for (auto type : dialogs) { + if (DialogManager::singleton().find_floating_dialog(type)) { + // avoid duplicates + continue; + } + + if (dialog_data.find(type) != dialog_data.end()) { + if (!notebook) { + notebook = Gtk::manage(new DialogNotebook(active_container)); + column->append(notebook); + } + active_container->new_dialog(type, notebook); + } else { + std::cerr << "recreate_dialogs_from_state: invalid dialog type: " << type.raw() << std::endl; + } + } + } + } + + if (has_position) { + dm_restore_window_position(*dialog_window, pos); + } + else { + dialog_window->update_window_size_to_fit_children(); + } + dialog_window->show_all(); + // Set the style and icon theme of the new menu based on the desktop + INKSCAPE.themecontext->getChangeThemeSignal().emit(); + INKSCAPE.themecontext->add_gtk_css(true); + restored = true; + } + + return restored; +} + +/** + * Add a new floating dialog (or reuse existing one if it's already up) + */ +DialogWindow *DialogContainer::new_floating_dialog(const Glib::ustring& dialog_type) +{ + return create_new_floating_dialog(dialog_type, true); +} + +DialogWindow *DialogContainer::create_new_floating_dialog(const Glib::ustring& dialog_type, bool blink) +{ + // check if this dialog is already open + if (DialogBase* existing_dialog = find_existing_dialog(dialog_type)) { + // found existing dialog; blink & exit + if (blink) { + existing_dialog->blink(); + // show its window if it is hidden + if (auto window = DialogManager::singleton().find_floating_dialog_window(dialog_type)) { + DialogManager::singleton().set_floating_dialog_visibility(window, true); + } + } + return nullptr; + } + + // check if this dialog *was* open and floating; if so recreate its window + if (auto state = DialogManager::singleton().find_dialog_state(dialog_type)) { + if (recreate_dialogs_from_state(_inkscape_window, state.get())) { + return nullptr; + } + } + + // Create the dialog widget + DialogBase *dialog = dialog_factory(dialog_type).release(); // Evil, but necessitated by GTK. + + if (!dialog) { + std::cerr << "DialogContainer::new_dialog(): couldn't find dialog for: " << dialog_type.raw() << std::endl; + return nullptr; + } + + // manage the dialog instance + dialog = Gtk::manage(dialog); + + // Create the notebook tab + gchar* image = nullptr; + + Glib::ustring label; + Glib::ustring action_name = "win.dialog-open('" + dialog_type + "')"; + auto app = InkscapeApplication::instance(); + std::vector<Glib::ustring> accels = app->gtk_app()->get_accels_for_action(action_name); + if (accels.size() > 0) { + guint key = 0; + Gdk::ModifierType mods; + Gtk::AccelGroup::parse(accels[0], key, mods); + label = Gtk::AccelGroup::get_label(key, mods); + } + + Gtk::Widget *tab = + create_notebook_tab(dialog->get_name(), image ? Glib::ustring(image) : INKSCAPE_ICON("inkscape-logo"), label); + + // New temporary noteboook + DialogNotebook *notebook = Gtk::manage(new DialogNotebook(this)); + notebook->add_page(*dialog, *tab, dialog->get_name()); + + return notebook->pop_tab_callback(); +} + +// toggle dialogs (visibility) is invoked on a top container embedded in Inkscape window +void DialogContainer::toggle_dialogs() +{ + // check how many dialog panels are visible and how many are hidden + // we use this info to decide what it means to toggle visibility + int visible = 0; + int hidden = 0; + for (auto child : columns->get_children()) { + // only examine panels, skip drop zones and handles + if (auto panel = dynamic_cast<DialogMultipaned*>(child)) { + if (panel->is_visible()) { + ++visible; + } + else { + ++hidden; + } + } + } + + // next examine floating dialogs + auto windows = DialogManager::singleton().get_all_floating_dialog_windows(); + for (auto wnd : windows) { + if (wnd->is_visible()) { + ++visible; + } + else { + ++hidden; + } + } + + bool show_dialogs = true; + // if some dialogs are hidden, toggle will first show them; + // another option could be to hide all if some dialogs are visible + if (hidden > 0) { + show_dialogs = true; + } + else { + // if everything's visible, hide them + show_dialogs = false; + } + + // set visibility of floating dialogs + for (auto wnd : windows) { + DialogManager::singleton().set_floating_dialog_visibility(wnd, show_dialogs); + } + + // set visibility of docked dialogs + columns->toggle_multipaned_children(show_dialogs); +} + +// Update dialogs +void DialogContainer::update_dialogs() +{ + for_each(dialogs.begin(), dialogs.end(), [&](auto dialog) { dialog.second->update(); }); +} + +void DialogContainer::set_inkscape_window(InkscapeWindow* inkscape_window) +{ + g_assert(inkscape_window != nullptr); + _inkscape_window = inkscape_window; + auto desktop = _inkscape_window->get_desktop(); + for_each(dialogs.begin(), dialogs.end(), [&](auto dialog) { dialog.second->setDesktop(desktop); }); +} + +bool DialogContainer::has_dialog_of_type(DialogBase *dialog) +{ + return (dialogs.find(dialog->get_type()) != dialogs.end()); +} + +DialogBase *DialogContainer::get_dialog(const Glib::ustring& dialog_type) +{ + auto found = dialogs.find(dialog_type); + if (found != dialogs.end()) { + return found->second; + } + return nullptr; +} + +// Add dialog to list. +void DialogContainer::link_dialog(DialogBase *dialog) +{ + dialogs.insert(std::pair<Glib::ustring, DialogBase *>(dialog->get_type(), dialog)); + + DialogWindow *window = dynamic_cast<DialogWindow *>(get_toplevel()); + if (window) { + window->update_dialogs(); + } + else { + // dialog without DialogWindow has been docked; remove it's floating state + // so if user closes and reopens it, it shows up docked again, not floating + DialogManager::singleton().remove_dialog_floating_state(dialog->get_type()); + } +} + +// Remove dialog from list. +void DialogContainer::unlink_dialog(DialogBase *dialog) +{ + if (!dialog) { + return; + } + + auto found = dialogs.find(dialog->get_type()); + if (found != dialogs.end()) { + dialogs.erase(found); + } + + DialogWindow *window = dynamic_cast<DialogWindow *>(get_toplevel()); + if (window) { + window->update_dialogs(); + } +} + +/** + * Load last open window's dialog configuration state. + * + * For the keyfile format, check `save_container_state()`. + */ +void DialogContainer::load_container_state(Glib::KeyFile *keyfile, bool include_floating) +{ + // Step 1: check if we want to load the state + Inkscape::Preferences *prefs = Inkscape::Preferences::get(); + + // if it isn't dockable, all saved docked dialogs are made floating + bool is_dockable = + prefs->getInt("/options/dialogtype/value", PREFS_DIALOGS_BEHAVIOR_DOCKABLE) != PREFS_DIALOGS_BEHAVIOR_FLOATING; + + // Step 2: get the number of windows + int windows_count = keyfile->get_integer("Windows", "Count"); + + // Step 3: for each window, load its state. Only the first window is not floating (the others are DialogWindow) + for (int window_idx = 0; window_idx < windows_count; ++window_idx) { + if (window_idx > 0 && !include_floating) + break; + + Glib::ustring group_name = "Window" + std::to_string(window_idx); + + // Step 3.0: read the window parameters + int column_count = 0; + bool floating = window_idx != 0; + window_position_t pos; + bool has_position = false; + try { + column_count = keyfile->get_integer(group_name, "ColumnCount"); + floating = keyfile->get_boolean(group_name, "Floating"); + if (keyfile->has_key(group_name, "Position") && keyfile->get_boolean(group_name, "Position")) { + pos.x = keyfile->get_integer(group_name, "x"); + pos.y = keyfile->get_integer(group_name, "y"); + pos.width = keyfile->get_integer(group_name, "width"); + pos.height = keyfile->get_integer(group_name, "height"); + has_position = true; + } + } catch (Glib::Error &error) { + std::cerr << "DialogContainer::load_container_state: " << error.what().raw() << std::endl; + } + + // Step 3.1: get the window's container columns where we want to create the dialogs + DialogContainer *active_container = nullptr; + DialogMultipaned *active_columns = nullptr; + DialogWindow *dialog_window = nullptr; + + if (is_dockable) { + if (floating) { + dialog_window = new DialogWindow(_inkscape_window, nullptr); + if (dialog_window) { + active_container = dialog_window->get_container(); + active_columns = dialog_window->get_container()->get_columns(); + } + } else { + active_container = this; + active_columns = columns; + } + + if (!active_container || !active_columns) { + continue; + } + } + + // Step 3.2: for each column, load its state + for (int column_idx = 0; column_idx < column_count; ++column_idx) { + Glib::ustring column_group_name = group_name + "Column" + std::to_string(column_idx); + + // Step 3.2.0: read the column parameters + int notebook_count = 0; + bool before_canvas = false; + try { + notebook_count = keyfile->get_integer(column_group_name, "NotebookCount"); + before_canvas = keyfile->get_boolean(column_group_name, "BeforeCanvas"); + } catch (Glib::Error &error) { + std::cerr << "DialogContainer::load_container_state: " << error.what().raw() << std::endl; + } + + // Step 3.2.1: create the column + DialogMultipaned *column = nullptr; + if (is_dockable) { + column = active_container->create_column(); + if (!column) { + continue; + } + + if (keyfile->has_key(column_group_name, "ColumnWidth")) { + auto width = keyfile->get_integer(column_group_name, "ColumnWidth"); + column->set_restored_width(width); + } + + before_canvas ? active_columns->prepend(column) : active_columns->append(column); + } + + // Step 3.2.2: for each noteboook, load its dialogs + for (int notebook_idx = 0; notebook_idx < notebook_count; ++notebook_idx) { + Glib::ustring key = "Notebook" + std::to_string(notebook_idx) + "Dialogs"; + + // Step 3.2.2.0 read the list of dialogs in the current notebook + std::vector<Glib::ustring> dialogs; + try { + dialogs = keyfile->get_string_list(column_group_name, key); + } catch (Glib::Error &error) { + std::cerr << "DialogContainer::load_container_state: " << error.what().raw() << std::endl; + } + + if (!dialogs.size()) { + continue; + } + + DialogNotebook *notebook = nullptr; + if (is_dockable) { + notebook = Gtk::manage(new DialogNotebook(active_container)); + column->append(notebook); + } + + auto const &dialog_data = get_dialog_data(); + // Step 3.2.2.1 create each dialog in the current notebook + for (auto type : dialogs) { + + if (dialog_data.find(type) != dialog_data.end()) { + if (is_dockable) { + active_container->new_dialog(type, notebook); + } else { + dialog_window = create_new_floating_dialog(type, false); + } + } else { + std::cerr << "load_container_state: invalid dialog type: " << type.raw() << std::endl; + } + } + + if (notebook) { + Glib::ustring row = "Notebook" + std::to_string(notebook_idx) + "Height"; + if (keyfile->has_key(column_group_name, row)) { + auto height = keyfile->get_integer(column_group_name, row); + notebook->set_requested_height(height); + } + Glib::ustring tab = "Notebook" + std::to_string(notebook_idx) + "ActiveTab"; + if (keyfile->has_key(column_group_name, tab)) { + if (auto nb = notebook->get_notebook()) { + auto page = keyfile->get_integer(column_group_name, tab); + nb->set_current_page(page); + } + } + } + } + } + + if (dialog_window) { + if (has_position) { + dm_restore_window_position(*dialog_window, pos); + } + else { + dialog_window->update_window_size_to_fit_children(); + } + dialog_window->show_all(); + // Set the style and icon theme of the new menu based on the desktop + } + } + INKSCAPE.themecontext->getChangeThemeSignal().emit(); + INKSCAPE.themecontext->add_gtk_css(true); +} + +void save_wnd_position(Glib::KeyFile *keyfile, const Glib::ustring &group_name, const window_position_t *position) +{ + keyfile->set_boolean(group_name, "Position", position != nullptr); + if (position) { // floating window position? + keyfile->set_integer(group_name, "x", position->x); + keyfile->set_integer(group_name, "y", position->y); + keyfile->set_integer(group_name, "width", position->width); + keyfile->set_integer(group_name, "height", position->height); + } +} + +// get *this* container's state only; store window 'position' in the state if given +std::shared_ptr<Glib::KeyFile> DialogContainer::get_container_state(const window_position_t *position) const +{ + std::shared_ptr<Glib::KeyFile> keyfile = std::make_shared<Glib::KeyFile>(); + + DialogMultipaned *window = columns; + const int window_idx = 0; + + // Step 2: save the number of windows + keyfile->set_integer("Windows", "Count", 1); + + // Step 3.0: get all the multipanes of the window + std::vector<DialogMultipaned *> multipanes; + + for (auto const &column : window->get_children()) { + if (auto paned = dynamic_cast<DialogMultipaned *>(column)) { + multipanes.push_back(paned); + } + } + + // Step 3.1: for each non-empty column, save its data. + int column_count = 0; // non-empty columns count + for (size_t column_idx = 0; column_idx < multipanes.size(); ++column_idx) { + Glib::ustring group_name = "Window" + std::to_string(window_idx) + "Column" + std::to_string(column_idx); + int notebook_count = 0; // non-empty notebooks count + + // Step 3.1.0: for each notebook, get its dialogs + for (auto const &columns_widget : multipanes[column_idx]->get_children()) { + if (auto dialog_notebook = dynamic_cast<DialogNotebook *>(columns_widget)) { + std::vector<Glib::ustring> dialogs; + + for (auto const &widget : dialog_notebook->get_notebook()->get_children()) { + if (DialogBase *dialog = dynamic_cast<DialogBase *>(widget)) { + dialogs.push_back(dialog->get_type()); + } + } + + // save the dialogs type + Glib::ustring key = "Notebook" + std::to_string(notebook_count) + "Dialogs"; + keyfile->set_string_list(group_name, key, dialogs); + + // increase the notebook count + notebook_count++; + } + } + + // Step 3.1.1: increase the column count + if (notebook_count != 0) { + column_count++; + } + + // Step 3.1.2: Save the column's data + keyfile->set_integer(group_name, "NotebookCount", notebook_count); + } + + // Step 3.2: save the window group + Glib::ustring group_name = "Window" + std::to_string(window_idx); + keyfile->set_integer(group_name, "ColumnCount", column_count); + save_wnd_position(keyfile.get(), group_name, position); + + return keyfile; +} + +/** + * Save container state. The configuration of open dialogs and the relative positions of the notebooks are saved. + * + * The structure of such a KeyFile is: + * + * There is a "Windows" group that records the number of the windows: + * [Windows] + * Count=1 + * + * A "WindowX" group saves the number of columns the window's container has and whether the window is floating: + * + * [Window0] + * ColumnCount=1 + * Floating=false + * + * For each column, we have a "WindowWColumnX" group, where X is the index of the column. "BeforeCanvas" checks + * if the column is before the canvas or not. "NotebookCount" records how many notebooks are in each column and + * "NotebookXDialogs" records a list of the types for the dialogs in notebook X. + * + * [Window0Column0] + * Notebook0Dialogs=Text; + * NotebookCount=2 + * BeforeCanvas=false + * + */ +std::unique_ptr<Glib::KeyFile> DialogContainer::save_container_state() +{ + std::unique_ptr<Glib::KeyFile> keyfile = std::make_unique<Glib::KeyFile>(); + auto app = InkscapeApplication::instance(); + + // Step 1: get all the container columns (in order, from the current container and all DialogWindow containers) + std::vector<DialogMultipaned *> windows(1, columns); + std::vector<DialogWindow *> dialog_windows(1, nullptr); + + for (auto const &window : app->gtk_app()->get_windows()) { + DialogWindow *dialog_window = dynamic_cast<DialogWindow *>(window); + if (dialog_window) { + windows.push_back(dialog_window->get_container()->get_columns()); + dialog_windows.push_back(dialog_window); + } + } + + // Step 2: save the number of windows + keyfile->set_integer("Windows", "Count", windows.size()); + + // Step 3: for each window, save its data. Only the first window is not floating (the others are DialogWindow) + for (int window_idx = 0; window_idx < (int)windows.size(); ++window_idx) { + // Step 3.0: get all the multipanes of the window + std::vector<DialogMultipaned *> multipanes; + + // used to check if the column is before or after canvas + std::vector<DialogMultipaned *>::iterator multipanes_it = multipanes.begin(); + bool canvas_seen = window_idx != 0; // no floating windows (window_idx > 0) have a canvas + int before_canvas_columns_count = 0; + + for (auto const &column : windows[window_idx]->get_children()) { + if (!canvas_seen) { + UI::Widget::CanvasGrid *canvas = dynamic_cast<UI::Widget::CanvasGrid *>(column); + if (canvas) { + canvas_seen = true; + } else { + DialogMultipaned *paned = dynamic_cast<DialogMultipaned *>(column); + if (paned) { + multipanes_it = multipanes.insert(multipanes_it, paned); + before_canvas_columns_count++; + } + } + } else { + DialogMultipaned *paned = dynamic_cast<DialogMultipaned *>(column); + if (paned) { + multipanes.push_back(paned); + } + } + } + + // Step 3.1: for each non-empty column, save its data. + int column_count = 0; // non-empty columns count + for (int column_idx = 0; column_idx < (int)multipanes.size(); ++column_idx) { + Glib::ustring group_name = "Window" + std::to_string(window_idx) + "Column" + std::to_string(column_idx); + int notebook_count = 0; // non-empty notebooks count + int width = multipanes[column_idx]->get_allocated_width(); + + // Step 3.1.0: for each notebook, get its dialogs' types + for (auto const &columns_widget : multipanes[column_idx]->get_children()) { + DialogNotebook *dialog_notebook = dynamic_cast<DialogNotebook *>(columns_widget); + + if (dialog_notebook) { + std::vector<Glib::ustring> dialogs; + + for (auto const &widget : dialog_notebook->get_notebook()->get_children()) { + DialogBase *dialog = dynamic_cast<DialogBase *>(widget); + if (dialog) { + dialogs.push_back(dialog->get_type()); + } + } + + // save the dialogs type + Glib::ustring key = "Notebook" + std::to_string(notebook_count) + "Dialogs"; + keyfile->set_string_list(group_name, key, dialogs); + // save height; useful when there are multiple "rows" of docked dialogs + Glib::ustring row = "Notebook" + std::to_string(notebook_count) + "Height"; + keyfile->set_integer(group_name, row, dialog_notebook->get_allocated_height()); + if (auto notebook = dialog_notebook->get_notebook()) { + Glib::ustring row = "Notebook" + std::to_string(notebook_count) + "ActiveTab"; + keyfile->set_integer(group_name, row, notebook->get_current_page()); + } + + // increase the notebook count + notebook_count++; + } + } + + // Step 3.1.1: increase the column count + if (notebook_count != 0) { + column_count++; + } + + keyfile->set_integer(group_name, "ColumnWidth", width); + + // Step 3.1.2: Save the column's data + keyfile->set_integer(group_name, "NotebookCount", notebook_count); + keyfile->set_boolean(group_name, "BeforeCanvas", (column_idx < before_canvas_columns_count)); + } + + // Step 3.2: save the window group + Glib::ustring group_name = "Window" + std::to_string(window_idx); + keyfile->set_integer(group_name, "ColumnCount", column_count); + keyfile->set_boolean(group_name, "Floating", window_idx != 0); + if (window_idx != 0) { // floating? + if (auto wnd = dynamic_cast<DialogWindow *>(dialog_windows.at(window_idx))) { + // store window position + auto pos = dm_get_window_position(*wnd); + save_wnd_position(keyfile.get(), group_name, pos ? &*pos : nullptr); + } + } + } + + return keyfile; +} + +// Signals ----------------------------------------------------- + +/** + * No zombie windows. TODO: Need to work on this as it still leaves Gtk::Window! (?) + */ +void DialogContainer::on_unrealize() { + // Disconnect all signals + for_each(connections.begin(), connections.end(), [&](auto c) { c.disconnect(); }); + + delete columns; + columns = nullptr; + + parent_type::on_unrealize(); +} + +// Create a new notebook and move page. +DialogNotebook *DialogContainer::prepare_drop(const Glib::RefPtr<Gdk::DragContext> context) +{ + Gtk::Widget *source = Gtk::Widget::drag_get_source_widget(context); + + // Find source notebook and page + Gtk::Notebook *old_notebook = dynamic_cast<Gtk::Notebook *>(source); + if (!old_notebook) { + std::cerr << "DialogContainer::prepare_drop: notebook not found!" << std::endl; + return nullptr; + } + + // Find page + Gtk::Widget *page = old_notebook->get_nth_page(old_notebook->get_current_page()); + if (!page) { + std::cerr << "DialogContainer::prepare_drop: page not found!" << std::endl; + return nullptr; + } + + // Create new notebook and move page. + DialogNotebook *new_notebook = Gtk::manage(new DialogNotebook(this)); + new_notebook->move_page(*page); + + // move_page() takes care of updating dialog lists. + INKSCAPE.themecontext->getChangeThemeSignal().emit(); + INKSCAPE.themecontext->add_gtk_css(true); + return new_notebook; +} + +// Notebook page dropped on prepend target. Call function to create new notebook and then insert. +void DialogContainer::prepend_drop(const Glib::RefPtr<Gdk::DragContext> context, DialogMultipaned *multipane) +{ + DialogNotebook *new_notebook = prepare_drop(context); // Creates notebook, moves page. + if (!new_notebook) { + std::cerr << "DialogContainer::prepend_drop: no new notebook!" << std::endl; + return; + } + + if (multipane->get_orientation() == Gtk::ORIENTATION_HORIZONTAL) { + // Columns + // Create column + DialogMultipaned *column = create_column(); + column->prepend(new_notebook); + columns->prepend(column); + } else { + // Column + multipane->prepend(new_notebook); + } + + update_dialogs(); // Always update dialogs on Notebook change +} + +// Notebook page dropped on append target. Call function to create new notebook and then insert. +void DialogContainer::append_drop(const Glib::RefPtr<Gdk::DragContext> context, DialogMultipaned *multipane) +{ + DialogNotebook *new_notebook = prepare_drop(context); // Creates notebook, moves page. + if (!new_notebook) { + std::cerr << "DialogContainer::append_drop: no new notebook!" << std::endl; + return; + } + + if (multipane->get_orientation() == Gtk::ORIENTATION_HORIZONTAL) { + // Columns + // Create column + DialogMultipaned *column = create_column(); + column->append(new_notebook); + columns->append(column); + } else { + // Column + multipane->append(new_notebook); + } + + update_dialogs(); // Always update dialogs on Notebook change +} + +/** + * If a DialogMultipaned column is empty and it can be removed, remove it + */ +void DialogContainer::column_empty(DialogMultipaned *column) +{ + DialogMultipaned *parent = dynamic_cast<DialogMultipaned *>(column->get_parent()); + if (parent) { + parent->remove(*column); + } + + DialogWindow *window = dynamic_cast<DialogWindow *>(get_toplevel()); + if (window && parent) { + auto children = parent->get_children(); + // Close the DialogWindow if you're in an empty one + if (children.size() == 3 && parent->has_empty_widget()) { + window->close(); + } + } +} + +} // namespace Dialog +} // namespace UI +} // namespace Inkscape + +/* + Local Variables: + mode:c++ + c-file-style:"stroustrup" + c-file-offsets:((innamespace . 0)(inline-open . 0)(case-label . +)) + indent-tabs-mode:nil + fill-column:99 + End: +*/ +// vim: filetype=cpp:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:fileencoding=utf-8:textwidth=99 : |