summaryrefslogtreecommitdiffstats
path: root/src/ui/dialog/export.cpp
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-13 11:50:49 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-13 11:50:49 +0000
commitc853ffb5b2f75f5a889ed2e3ef89b818a736e87a (patch)
tree7d13a0883bb7936b84d6ecdd7bc332b41ed04bee /src/ui/dialog/export.cpp
parentInitial commit. (diff)
downloadinkscape-upstream.tar.xz
inkscape-upstream.zip
Adding upstream version 1.3+ds.upstream/1.3+dsupstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to '')
-rw-r--r--src/ui/dialog/export.cpp531
1 files changed, 531 insertions, 0 deletions
diff --git a/src/ui/dialog/export.cpp b/src/ui/dialog/export.cpp
new file mode 100644
index 0000000..522acf0
--- /dev/null
+++ b/src/ui/dialog/export.cpp
@@ -0,0 +1,531 @@
+// SPDX-License-Identifier: GPL-2.0-or-later
+/* Authors:
+ * Lauris Kaplinski <lauris@kaplinski.com>
+ * bulia byak <buliabyak@users.sf.net>
+ * Johan Engelen <j.b.c.engelen@ewi.utwente.nl>
+ * Peter Bostrom
+ * Jon A. Cruz <jon@joncruz.org>
+ * Abhishek Sharma
+ * Kris De Gussem <Kris.DeGussem@gmail.com>
+ * Anshudhar Kumar Singh <anshudhar2001@gmail.com>
+ *
+ * Copyright (C) 1999-2007, 2012, 2021 Authors
+ * Copyright (C) 2001-2002 Ximian, Inc.
+ *
+ * Released under GNU GPL v2+, read the file 'COPYING' for more information.
+ */
+
+// This has to be included prior to anything that includes setjmp.h, it croaks otherwise
+#include "export.h"
+
+#include <glibmm/convert.h>
+#include <glibmm/i18n.h>
+#include <glibmm/miscutils.h>
+#include <gtkmm.h>
+#include <png.h>
+
+#include "desktop.h"
+#include "document-undo.h"
+#include "document.h"
+#include "color/color-conv.h"
+#include "extension/db.h"
+#include "extension/output.h"
+#include "file.h"
+#include "helper/png-write.h"
+#include "inkscape-window.h"
+#include "inkscape.h"
+#include "io/resource.h"
+#include "io/sys.h"
+#include "message-stack.h"
+#include "object/object-set.h"
+#include "object/sp-namedview.h"
+#include "object/sp-root.h"
+#include "object/sp-page.h"
+#include "object/weakptr.h"
+#include "preferences.h"
+#include "selection-chemistry.h"
+#include "ui/dialog-events.h"
+#include "ui/dialog/export-single.h"
+#include "ui/dialog/export-batch.h"
+#include "ui/dialog/dialog-notebook.h"
+#include "ui/dialog/filedialog.h"
+#include "ui/interface.h"
+#include "ui/widget/color-picker.h"
+#include "ui/widget/scrollprotected.h"
+#include "ui/widget/unit-menu.h"
+
+#ifdef _WIN32
+
+#endif
+
+using Inkscape::Util::unit_table;
+namespace Inkscape {
+namespace UI {
+namespace Dialog {
+
+Export::Export()
+ : DialogBase("/dialogs/export/", "Export")
+{
+ std::string gladefile = get_filename_string(Inkscape::IO::Resource::UIS, "dialog-export.glade");
+
+ try {
+ builder = Gtk::Builder::create_from_file(gladefile);
+ } catch (const Glib::Error &ex) {
+ g_error("Glade file loading failed for export screen");
+ return;
+ }
+
+ prefs = Inkscape::Preferences::get();
+
+ builder->get_widget("export-box", container);
+ add(*container);
+ show_all_children();
+
+ builder->get_widget("export-notebook", export_notebook);
+
+ // Initialise Single Export and its objects
+ builder->get_widget_derived("single-image", single_image);
+
+ // Initialise Batch Export and its objects
+ builder->get_widget_derived("batch-export", batch_export);
+
+ container->signal_realize().connect([=]() {
+ setDefaultNotebookPage();
+ notebook_signal = export_notebook->signal_switch_page().connect(sigc::mem_fun(*this, &Export::onNotebookPageSwitch));
+ });
+ container->signal_unrealize().connect([=]() {
+ notebook_signal.disconnect();
+ });
+}
+
+// Set current page based on preference/last visited page
+void Export::setDefaultNotebookPage()
+{
+ pages[BATCH_EXPORT] = export_notebook->page_num(*batch_export->get_parent());
+ pages[SINGLE_IMAGE] = export_notebook->page_num(*single_image->get_parent());
+ export_notebook->set_current_page(pages[SINGLE_IMAGE]);
+}
+
+void Export::documentReplaced()
+{
+ single_image->setDocument(getDocument());
+ batch_export->setDocument(getDocument());
+}
+
+void Export::desktopReplaced()
+{
+ single_image->setDesktop(getDesktop());
+ single_image->setApp(getApp());
+ batch_export->setDesktop(getDesktop());
+ batch_export->setApp(getApp());
+ // Called previously, but we need post-desktop call too
+ documentReplaced();
+}
+
+void Export::selectionChanged(Inkscape::Selection *selection)
+{
+ auto current_page = export_notebook->get_current_page();
+ if (current_page == pages[SINGLE_IMAGE]) {
+ single_image->selectionChanged(selection);
+ }
+ if (current_page == pages[BATCH_EXPORT]) {
+ batch_export->selectionChanged(selection);
+ }
+}
+void Export::selectionModified(Inkscape::Selection *selection, guint flags)
+{
+ auto current_page = export_notebook->get_current_page();
+ if (current_page == pages[SINGLE_IMAGE]) {
+ single_image->selectionModified(selection, flags);
+ }
+ if (current_page == pages[BATCH_EXPORT]) {
+ batch_export->selectionModified(selection, flags);
+ }
+}
+
+void Export::onNotebookPageSwitch(Widget *page, guint page_number)
+{
+ auto desktop = getDesktop();
+ if (desktop) {
+ auto selection = desktop->getSelection();
+
+ if (page_number == pages[SINGLE_IMAGE]) {
+ single_image->selectionChanged(selection);
+ }
+ if (page_number == pages[BATCH_EXPORT]) {
+ batch_export->selectionChanged(selection);
+ }
+ }
+}
+
+std::string Export::absolutizePath(SPDocument *doc, const std::string &filename)
+{
+ std::string path;
+ // Make relative paths go from the document location, if possible:
+ if (!Glib::path_is_absolute(filename) && doc->getDocumentFilename()) {
+ auto dirname = Glib::path_get_dirname(doc->getDocumentFilename());
+ if (!dirname.empty()) {
+ path = Glib::build_filename(dirname, filename);
+ }
+ }
+ if (path.empty()) {
+ path = filename;
+ }
+ return path;
+}
+
+bool Export::unConflictFilename(SPDocument *doc, Glib::ustring &filename, Glib::ustring const extension)
+{
+ std::string path = absolutizePath(doc, Glib::filename_from_utf8(filename));
+ Glib::ustring test_filename = path + extension;
+ if (!Inkscape::IO::file_test(test_filename.c_str(), G_FILE_TEST_EXISTS)) {
+ filename = test_filename;
+ return true;
+ }
+ for (int i = 1; i <= 100; i++) {
+ test_filename = path + "_copy_" + std::to_string(i) + extension;
+ if (!Inkscape::IO::file_test(test_filename.c_str(), G_FILE_TEST_EXISTS)) {
+ filename = test_filename;
+ return true;
+ }
+ }
+ return false;
+}
+
+bool Export::exportRaster(
+ Geom::Rect const &area, unsigned long int const &width, unsigned long int const &height,
+ float const &dpi, guint32 bg_color, Glib::ustring const &filename, bool overwrite,
+ unsigned (*callback)(float, void *), void *data,
+ Inkscape::Extension::Output *extension, std::vector<SPItem *> *items)
+{
+ SPDesktop *desktop = SP_ACTIVE_DESKTOP;
+ if (!desktop)
+ return false;
+ SPDocument *doc = desktop->getDocument();
+
+ if (area.hasZeroArea() || width == 0 || height == 0) {
+ desktop->messageStack()->flash(Inkscape::ERROR_MESSAGE, _("The chosen area to be exported is invalid."));
+ sp_ui_error_dialog(_("The chosen area to be exported is invalid"));
+ return false;
+ }
+ if (filename.empty()) {
+ desktop->messageStack()->flash(Inkscape::ERROR_MESSAGE, _("You have to enter a filename."));
+ sp_ui_error_dialog(_("You have to enter a filename"));
+ return false;
+ }
+
+ if (!extension || !extension->is_raster()) {
+ desktop->messageStack()->flash(Inkscape::ERROR_MESSAGE, _("Raster Export Error"));
+ sp_ui_error_dialog(_("Raster export Method is used for NON RASTER EXTENSION"));
+ return false;
+ }
+
+ float pHYs = extension->get_param_float("png_phys", dpi);
+ if (pHYs < 0.01) pHYs = dpi;
+
+ bool use_interlacing = extension->get_param_bool("png_interlacing", false);
+ int antialiasing = extension->get_param_int("png_antialias", 2); // Cairo anti aliasing
+ int zlib = extension->get_param_int("png_compression", 1); // Default is 6 for png, but 1 for non-png
+ auto val = extension->get_param_int("png_bitdepth", 99); // corresponds to RGBA 8
+
+ int bit_depth = pow(2, (val & 0x0F));
+ int color_type = (val & 0xF0) >> 4;
+
+ std::string path = absolutizePath(doc, Glib::filename_from_utf8(filename));
+ Glib::ustring dirname = Glib::path_get_dirname(path);
+
+ if (dirname.empty() ||
+ !Inkscape::IO::file_test(dirname.c_str(), (GFileTest)(G_FILE_TEST_EXISTS | G_FILE_TEST_IS_DIR))) {
+ Glib::ustring safeDir = Inkscape::IO::sanitizeString(dirname.c_str());
+ Glib::ustring error =
+ g_strdup_printf(_("Directory <b>%s</b> does not exist or is not a directory.\n"), safeDir.c_str());
+
+ desktop->messageStack()->flash(Inkscape::ERROR_MESSAGE, error.c_str());
+ sp_ui_error_dialog(error.c_str());
+ return false;
+ }
+
+ // Do the over-write protection now, since the png is just a temp file.
+ if (!overwrite && !sp_ui_overwrite_file(path.c_str())) {
+ return false;
+ }
+
+ auto fn = Glib::path_get_basename(path);
+ auto png_filename = path;
+ {
+ // Select the extension and set the filename to a temporary file
+ int tempfd_out = Glib::file_open_tmp(png_filename, "ink_ext_");
+ close(tempfd_out);
+ }
+
+ // Export Start Here
+ std::vector<SPItem *> selected;
+ if (items && items->size() > 0) {
+ selected = *items;
+ }
+
+ ExportResult result = sp_export_png_file(desktop->getDocument(), png_filename.c_str(), area, width, height, pHYs,
+ pHYs, // previously xdpi, ydpi.
+ bg_color, callback, data, true, selected,
+ use_interlacing, color_type, bit_depth, zlib, antialiasing);
+
+ bool failed = result == EXPORT_ERROR; // || prog_dialog->get_stopped();
+
+ if (failed) {
+ Glib::ustring safeFile = Inkscape::IO::sanitizeString(path.c_str());
+ Glib::ustring error = g_strdup_printf(_("Could not export to filename <b>%s</b>.\n"), safeFile.c_str());
+
+ desktop->messageStack()->flash(Inkscape::ERROR_MESSAGE, error.c_str());
+ sp_ui_error_dialog(error.c_str());
+ return false;
+ } else if (result == EXPORT_OK) {
+ // Don't ask for preferences on every run.
+ try {
+ extension->export_raster(doc, png_filename, path.c_str(), false);
+ } catch (Inkscape::Extension::Output::save_failed &e) {
+ return false;
+ }
+ } else {
+ // Extensions have their own error popup, so this only tracks failures in the png step
+ desktop->messageStack()->flash(Inkscape::INFORMATION_MESSAGE, _("Export aborted."));
+ return false;
+ }
+
+ Glib::ustring safeFile = Inkscape::IO::sanitizeString(path.c_str());
+ desktop->messageStack()->flashF(Inkscape::INFORMATION_MESSAGE, _("Drawing exported to <b>%s</b>."),
+ safeFile.c_str());
+
+ unlink(png_filename.c_str());
+ return true;
+}
+
+bool Export::exportVector(
+ Inkscape::Extension::Output *extension, SPDocument *doc,
+ Glib::ustring const &filename,
+ bool overwrite, const std::vector<SPItem *> &items, SPPage *page)
+{
+ std::vector<SPPage *> pages;
+ if (page)
+ pages.push_back(page);
+ return exportVector(extension, doc, filename, overwrite, items, pages);
+}
+
+bool Export::exportVector(
+ Inkscape::Extension::Output *extension, SPDocument *copy_doc,
+ Glib::ustring const &filename,
+ bool overwrite, const std::vector<SPItem *> &items, const std::vector<SPPage *> &pages)
+{
+ SPDesktop *desktop = SP_ACTIVE_DESKTOP;
+ if (!desktop)
+ return false;
+
+ if (filename.empty()) {
+ desktop->messageStack()->flash(Inkscape::ERROR_MESSAGE, _("You have to enter a filename."));
+ sp_ui_error_dialog(_("You have to enter a filename"));
+ return false;
+ }
+
+ if (!extension || extension->is_raster()) {
+ desktop->messageStack()->flash(Inkscape::ERROR_MESSAGE, _("Vector Export Error"));
+ sp_ui_error_dialog(_("Vector export Method is used for RASTER EXTENSION"));
+ return false;
+ }
+
+ std::string path = absolutizePath(copy_doc, Glib::filename_from_utf8(filename));
+ Glib::ustring dirname = Glib::path_get_dirname(path);
+ Glib::ustring safeFile = Inkscape::IO::sanitizeString(path.c_str());
+ Glib::ustring safeDir = Inkscape::IO::sanitizeString(dirname.c_str());
+
+ if (dirname.empty() ||
+ !Inkscape::IO::file_test(dirname.c_str(), (GFileTest)(G_FILE_TEST_EXISTS | G_FILE_TEST_IS_DIR))) {
+ Glib::ustring error =
+ g_strdup_printf(_("Directory <b>%s</b> does not exist or is not a directory.\n"), safeDir.c_str());
+
+ desktop->messageStack()->flash(Inkscape::ERROR_MESSAGE, error.c_str());
+ sp_ui_error_dialog(error.c_str());
+
+ return false;
+ }
+
+ // Do the over-write protection now
+ if (!overwrite && !sp_ui_overwrite_file(path.c_str())) {
+ return false;
+ }
+ copy_doc->ensureUpToDate();
+
+ std::vector<SPItem *> objects = items;
+ std::set<std::string> obj_ids;
+ std::set<std::string> page_ids;
+ Geom::OptRect page_rect;
+ for (auto page : pages) {
+ // Save the first page rect, must be done before everything else
+ if (!page_rect)
+ page_rect = page->getDesktopRect();
+
+ if (auto _id = page->getId()) {
+ page_ids.insert(std::string(_id));
+ }
+ // If page then our item set is limited to the overlapping items
+ auto page_items = page->getOverlappingItems(true, true);
+
+ if (items.empty()) {
+ // Items is page_items, remove all items not in this page.
+ objects.insert(objects.end(), page_items.begin(), page_items.end());
+ } else {
+ for (auto &item : page_items) {
+ item->getIds(obj_ids);
+ }
+ }
+ }
+
+ // Delete any pages not specified, delete all pages if none specified
+ auto &pm = copy_doc->getPageManager();
+
+ // Make weak pointers to pages, since deletePage() can delete more than just the requested page.
+ std::vector<SPWeakPtr<SPPage>> copy_pages;
+ copy_pages.reserve(pm.getPageCount());
+ for (auto *page : pm.getPages()) {
+ copy_pages.emplace_back(page);
+ }
+
+ // We refuse to delete anything if everything would be deleted.
+ for (auto &page : copy_pages) {
+ if (page) {
+ auto _id = page->getId();
+ if (_id && page_ids.find(_id) == page_ids.end()) {
+ pm.deletePage(page.get(), false);
+ }
+ }
+ }
+
+ // Page export ALWAYS restricts, even if nothing would be on the page.
+ if (!objects.empty() || !pages.empty()) {
+ std::vector<SPObject *> objects_to_export;
+ Inkscape::ObjectSet object_set(copy_doc);
+ for (auto &object : objects) {
+ auto _id = object->getId();
+ if (!_id || (!obj_ids.empty() && obj_ids.find(_id) == obj_ids.end())) {
+ // This item is off the page so can be ignored for export
+ continue;
+ }
+
+ SPObject *obj = copy_doc->getObjectById(_id);
+ if (!obj) {
+ Glib::ustring error = g_strdup_printf(_("Could not export to filename <b>%s</b>. (missing object)\n"), safeFile.c_str());
+
+ desktop->messageStack()->flash(Inkscape::ERROR_MESSAGE, error.c_str());
+ sp_ui_error_dialog(error.c_str());
+
+ return false;
+ }
+ copy_doc->ensureUpToDate();
+
+ object_set.add(obj, true);
+ objects_to_export.push_back(obj);
+ }
+
+ copy_doc->getRoot()->cropToObjects(objects_to_export);
+
+ if (pages.empty()) {
+ object_set.fitCanvas(true, true);
+ }
+ }
+
+ // Remove all unused definitions
+ copy_doc->vacuumDocument();
+
+ try {
+ extension->save(copy_doc, path.c_str());
+ } catch (Inkscape::Extension::Output::save_failed &e) {
+ Glib::ustring safeFile = Inkscape::IO::sanitizeString(path.c_str());
+ Glib::ustring error = g_strdup_printf(_("Could not export to filename <b>%s</b>.\n"), safeFile.c_str());
+
+ desktop->messageStack()->flash(Inkscape::ERROR_MESSAGE, error.c_str());
+ sp_ui_error_dialog(error.c_str());
+
+ return false;
+ }
+
+ desktop->messageStack()->flashF(Inkscape::INFORMATION_MESSAGE, _("Drawing exported to <b>%s</b>."),
+ safeFile.c_str());
+ return true;
+}
+
+std::string Export::filePathFromObject(SPDocument *doc, SPObject *obj, const Glib::ustring &file_entry_text)
+{
+ Glib::ustring id = _("bitmap");
+ if (obj && obj->getId()) {
+ id = obj->getId();
+ }
+ return filePathFromId(doc, id, file_entry_text);
+}
+
+std::string Export::filePathFromId(SPDocument *doc, Glib::ustring id, const Glib::ustring &file_entry_text)
+{
+ g_assert(!id.empty());
+
+ std::string directory;
+
+ if (!file_entry_text.empty()) {
+ directory = Glib::path_get_dirname(Glib::filename_from_utf8(file_entry_text));
+ }
+
+ if (directory.empty()) {
+ /* Grab document directory */
+ const gchar *docFilename = doc->getDocumentFilename();
+ if (docFilename) {
+ directory = Glib::path_get_dirname(docFilename);
+ }
+ }
+
+ if (directory.empty()) {
+ directory = Inkscape::IO::Resource::homedir_path();
+ }
+
+ return Glib::build_filename(directory, Glib::filename_from_utf8(id));
+}
+
+Glib::ustring Export::defaultFilename(SPDocument *doc, Glib::ustring &filename_entry_text, Glib::ustring extension)
+{
+ Glib::ustring filename;
+ if (doc && doc->getDocumentFilename()) {
+ filename = doc->getDocumentFilename();
+ //appendExtensionToFilename(filename, extension);
+ } else if (doc) {
+ filename = filePathFromId(doc, _("bitmap"), filename_entry_text);
+ filename = filename + extension;
+ }
+ return filename;
+}
+
+void set_export_bg_color(SPObject* object, guint32 color) {
+ if (object) {
+ object->setAttribute("inkscape:export-bgcolor", Inkscape::Util::rgba_color_to_string(color).c_str());
+ }
+}
+
+guint32 get_export_bg_color(SPObject* object, guint32 default_color) {
+ if (object) {
+ if (auto color = Inkscape::Util::string_to_rgba_color(object->getAttribute("inkscape:export-bgcolor"))) {
+ return *color;
+ }
+ }
+ return default_color;
+}
+
+
+} // 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 :
+// vim: filetype=cpp:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:fileencoding=utf-8:textwidth=99 :