summaryrefslogtreecommitdiffstats
path: root/src/ui/dialog/dialog-base.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/ui/dialog/dialog-base.cpp')
-rw-r--r--src/ui/dialog/dialog-base.cpp318
1 files changed, 318 insertions, 0 deletions
diff --git a/src/ui/dialog/dialog-base.cpp b/src/ui/dialog/dialog-base.cpp
new file mode 100644
index 0000000..154c811
--- /dev/null
+++ b/src/ui/dialog/dialog-base.cpp
@@ -0,0 +1,318 @@
+// SPDX-License-Identifier: GPL-2.0-or-later
+
+/** @file
+ * @brief A base class for all dialogs.
+ *
+ * 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-base.h"
+
+#include <glibmm/i18n.h>
+#include <glibmm/main.h>
+#include <glibmm/refptr.h>
+#include <gtkmm/cssprovider.h>
+#include <gtkmm/notebook.h>
+#include <gtkmm/scrolledwindow.h>
+#include <iostream>
+
+#include "inkscape.h"
+#include "desktop.h"
+#include "ui/dialog/dialog-data.h"
+#include "ui/dialog/dialog-notebook.h"
+#include "ui/dialog-events.h"
+// get_latin_keyval
+#include "ui/tools/tool-base.h"
+#include "widgets/spw-utilities.h"
+#include "ui/widget/canvas.h"
+#include "ui/util.h"
+
+namespace Inkscape {
+namespace UI {
+namespace Dialog {
+
+/**
+ * DialogBase constructor.
+ *
+ * @param prefs_path characteristic path to load/save dialog position.
+ * @param dialog_type is the "type" string for the dialog.
+ */
+DialogBase::DialogBase(gchar const *prefs_path, Glib::ustring dialog_type)
+ : Gtk::Box(Gtk::ORIENTATION_VERTICAL)
+ , desktop(nullptr)
+ , document(nullptr)
+ , selection(nullptr)
+ , _name("DialogBase")
+ , _prefs_path(prefs_path)
+ , _dialog_type(dialog_type)
+ , _app(InkscapeApplication::instance())
+{
+ auto const &dialog_data = get_dialog_data();
+
+ // Derive a pretty display name for the dialog.
+ auto it = dialog_data.find(dialog_type);
+ if (it != dialog_data.end()) {
+
+ _name = it->second.label; // Already translated
+
+ // remove ellipsis and mnemonics
+ int pos = _name.find("...", 0);
+ if (pos >= 0 && pos < _name.length() - 2) {
+ _name.erase(pos, 3);
+ }
+ pos = _name.find("…", 0);
+ if (pos >= 0 && pos < _name.length()) {
+ _name.erase(pos, 1);
+ }
+ pos = _name.find("_", 0);
+ if (pos >= 0 && pos < _name.length()) {
+ _name.erase(pos, 1);
+ }
+ }
+
+ set_name(_dialog_type); // Essential for dialog functionality
+ property_margin().set_value(1); // Essential for dialog UI
+}
+
+DialogBase::~DialogBase() {
+#ifdef _WIN32
+ // this is bad, but it supposedly fixes some resizng problem on Windows
+ ensure_size();
+#endif
+
+ unsetDesktop();
+};
+
+void DialogBase::ensure_size() {
+ if (desktop) {
+ resize_widget_children(desktop->getToplevel());
+ resize_widget_children(this);
+ }
+}
+
+void DialogBase::on_map() {
+ // Update asks the dialogs if they need their Gtk widgets updated.
+ update();
+ // Set the desktop on_map, although we might want to be smarter about this.
+ // Note: Inkscape::Application::instance().active_desktop() is used here, as it contains current desktop at
+ // the time of dialog creation. Formerly used _app.get_active_view() did not at application start-up.
+ setDesktop(Inkscape::Application::instance().active_desktop());
+ parent_type::on_map();
+ ensure_size();
+}
+
+bool DialogBase::on_key_press_event(GdkEventKey* key_event) {
+ switch (Inkscape::UI::Tools::get_latin_keyval(key_event)) {
+ case GDK_KEY_Escape:
+ defocus_dialog();
+ return true;
+ }
+
+ return parent_type::on_key_press_event(key_event);
+}
+
+
+/**
+ * Highlight notebook where dialog already exists.
+ */
+void DialogBase::blink()
+{
+ Gtk::Notebook *notebook = dynamic_cast<Gtk::Notebook *>(get_parent());
+ if (notebook && notebook->get_is_drawable()) {
+ // Switch notebook to this dialog.
+ notebook->set_current_page(notebook->page_num(*this));
+ notebook->get_style_context()->add_class("blink");
+
+ // Add timer to turn off blink.
+ sigc::slot<bool ()> slot = sigc::mem_fun(*this, &DialogBase::blink_off);
+ sigc::connection connection = Glib::signal_timeout().connect(slot, 1000); // msec
+ }
+}
+
+void DialogBase::focus_dialog() {
+ if (auto window = dynamic_cast<Gtk::Window*>(get_toplevel())) {
+ window->present();
+ }
+
+ // widget that had focus, if any
+ if (auto child = get_focus_child()) {
+ child->grab_focus();
+ }
+ else {
+ // find first focusable widget
+ if (auto child = sp_find_focusable_widget(this)) {
+ child->grab_focus();
+ }
+ }
+}
+
+void DialogBase::defocus_dialog() {
+ if (auto wnd = dynamic_cast<Gtk::Window*>(get_toplevel())) {
+ // defocus floating dialog:
+ sp_dialog_defocus_cpp(wnd);
+
+ // for docked dialogs, move focus to canvas
+ if (auto desktop = getDesktop()) {
+ desktop->getCanvas()->grab_focus();
+ }
+ }
+}
+
+/**
+ * Callback to reset the dialog highlight.
+ */
+bool DialogBase::blink_off()
+{
+ Gtk::Notebook *notebook = dynamic_cast<Gtk::Notebook *>(get_parent());
+ if (notebook && notebook->get_is_drawable()) {
+ notebook->get_style_context()->remove_class("blink");
+ }
+ return false;
+}
+
+/**
+ * Called when the desktop might have changed for this dialog.
+ */
+void DialogBase::setDesktop(SPDesktop *new_desktop)
+{
+ if (desktop == new_desktop) {
+ return;
+ }
+
+ unsetDesktop();
+
+ if (new_desktop) {
+ desktop = new_desktop;
+
+ if (auto sel = desktop->getSelection()) {
+ selection = sel;
+ _select_changed = selection->connectChanged([this](Inkscape::Selection *selection) {
+ _changed_while_hidden = !_showing;
+ if (_showing)
+ selectionChanged(selection);
+ });
+ _select_modified = selection->connectModified([this](Inkscape::Selection *selection, guint flags) {
+ _modified_while_hidden = !_showing;
+ _modified_flags = flags;
+ if (_showing)
+ selectionModified(selection, flags);
+ });
+ }
+
+ _doc_replaced = desktop->connectDocumentReplaced(sigc::hide<0>(sigc::mem_fun(*this, &DialogBase::setDocument)));
+ _desktop_destroyed = desktop->connectDestroy(sigc::mem_fun(*this, &DialogBase::desktopDestroyed));
+ this->setDocument(desktop->getDocument());
+
+ if (desktop->getSelection()) {
+ selectionChanged(selection);
+ }
+ set_sensitive(true);
+ }
+
+ desktopReplaced();
+}
+
+//
+void DialogBase::fix_inner_scroll(Gtk::Widget *scrollwindow)
+{
+ auto scrollwin = dynamic_cast<Gtk::ScrolledWindow *>(scrollwindow);
+ auto viewport = dynamic_cast<Gtk::ScrolledWindow *>(scrollwin->get_child());
+ Gtk::Widget *child = nullptr;
+ if (viewport) { //some widgets has viewportother not
+ child = viewport->get_child();
+ } else {
+ child = scrollwin->get_child();
+ }
+ if (child && scrollwin) {
+ Glib::RefPtr<Gtk::Adjustment> adjustment = scrollwin->get_vadjustment();
+ child->signal_scroll_event().connect([=](GdkEventScroll* event) {
+ auto container = dynamic_cast<Gtk::Container *>(this);
+ if (container) {
+ std::vector<Gtk::Widget*> widgets = container->get_children();
+ if (widgets.size()) {
+ auto parentscroll = dynamic_cast<Gtk::ScrolledWindow *>(widgets[0]);
+ if (parentscroll) {
+ if (event->delta_y > 0 && (adjustment->get_value() + adjustment->get_page_size()) == adjustment->get_upper()) {
+ parentscroll->event((GdkEvent*)event);
+ return true;
+ } else if (event->delta_y < 0 && adjustment->get_value() == adjustment->get_lower()) {
+ parentscroll->event((GdkEvent*)event);
+ return true;
+ }
+ }
+ }
+ }
+ return false;
+ });
+ }
+}
+
+/**
+ * function called from notebook dialog that performs an update of the dialog and sets the dialog showing state true
+ */
+void
+DialogBase::setShowing(bool showing) {
+ _showing = showing;
+ if (showing && _changed_while_hidden) {
+ selectionChanged(getSelection());
+ _changed_while_hidden = false;
+ }
+ if (showing && _modified_while_hidden) {
+ selectionModified(getSelection(), _modified_flags);
+ _modified_while_hidden = false;
+ }
+}
+
+/**
+ * Called to destruct desktops, must not call virtuals
+ */
+void DialogBase::unsetDesktop()
+{
+ desktop = nullptr;
+ document = nullptr;
+ selection = nullptr;
+ _desktop_destroyed.disconnect();
+ _doc_replaced.disconnect();
+ _select_changed.disconnect();
+ _select_modified.disconnect();
+}
+
+void DialogBase::desktopDestroyed(SPDesktop* old_desktop)
+{
+ if (old_desktop == desktop && desktop) {
+ unsetDesktop();
+ desktopReplaced();
+ set_sensitive(false);
+ }
+}
+
+/**
+ * Called when the document might have changed, called from setDesktop too.
+ */
+void DialogBase::setDocument(SPDocument *new_document)
+{
+ if (document != new_document) {
+ document = new_document;
+ documentReplaced();
+ }
+}
+
+} // 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 :