summaryrefslogtreecommitdiffstats
path: root/src/object/sp-page.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/object/sp-page.cpp')
-rw-r--r--src/object/sp-page.cpp433
1 files changed, 433 insertions, 0 deletions
diff --git a/src/object/sp-page.cpp b/src/object/sp-page.cpp
new file mode 100644
index 0000000..7107c10
--- /dev/null
+++ b/src/object/sp-page.cpp
@@ -0,0 +1,433 @@
+// SPDX-License-Identifier: GPL-2.0-or-later
+/*
+ * Inkscape pages implementation
+ *
+ * Authors:
+ * Martin Owens <doctormo@geek-2.com>
+ *
+ * Copyright (C) 2021 Martin Owens
+ *
+ * Released under GNU GPL v2+, read the file 'COPYING' for more information.
+ */
+
+#include <glibmm/i18n.h>
+
+#include "sp-page.h"
+
+#include "attributes.h"
+#include "desktop.h"
+#include "display/control/canvas-page.h"
+#include "inkscape.h"
+#include "object/object-set.h"
+#include "sp-namedview.h"
+#include "sp-root.h"
+
+using Inkscape::DocumentUndo;
+
+SPPage::SPPage()
+ : SPObject()
+{
+ _canvas_item = new Inkscape::CanvasPage();
+}
+
+SPPage::~SPPage()
+{
+ delete _canvas_item;
+ _canvas_item = nullptr;
+}
+
+void SPPage::build(SPDocument *document, Inkscape::XML::Node *repr)
+{
+ SPObject::build(document, repr);
+
+ this->readAttr(SPAttr::INKSCAPE_LABEL);
+ this->readAttr(SPAttr::X);
+ this->readAttr(SPAttr::Y);
+ this->readAttr(SPAttr::WIDTH);
+ this->readAttr(SPAttr::HEIGHT);
+
+ /* Register */
+ document->addResource("page", this);
+}
+
+void SPPage::release()
+{
+ if (this->document) {
+ // Unregister ourselves
+ this->document->removeResource("page", this);
+ }
+
+ SPObject::release();
+}
+
+void SPPage::set(SPAttr key, const gchar *value)
+{
+ switch (key) {
+ case SPAttr::X:
+ this->x.readOrUnset(value);
+ break;
+ case SPAttr::Y:
+ this->y.readOrUnset(value);
+ break;
+ case SPAttr::WIDTH:
+ this->width.readOrUnset(value);
+ break;
+ case SPAttr::HEIGHT:
+ this->height.readOrUnset(value);
+ break;
+ default:
+ SPObject::set(key, value);
+ break;
+ }
+ this->requestDisplayUpdate(SP_OBJECT_MODIFIED_FLAG);
+}
+
+/**
+ * Gets the rectangle in document units
+ */
+Geom::Rect SPPage::getRect() const
+{
+ return Geom::Rect(this->x.computed, this->y.computed, this->x.computed + this->width.computed,
+ this->y.computed + this->height.computed);
+}
+
+/**
+ * Get the rectangle of the page, in desktop units
+ */
+Geom::Rect SPPage::getDesktopRect() const
+{
+ auto rect = getDocumentRect();
+ rect *= document->dt2doc();
+ return rect;
+}
+
+/**
+ * Get the rectangle of the page, scaled to the document.
+ */
+Geom::Rect SPPage::getDocumentRect() const
+{
+ return getRect() * document->getDocumentScale();
+}
+
+/**
+ * Like getDesktopRect but returns a slightly shrunken rectangle
+ * so interactions don't confuse the border with the object.
+ */
+Geom::Rect SPPage::getSensitiveRect() const
+{
+ auto rect = getDesktopRect();
+ rect.expandBy(-0.1);
+ return rect;
+}
+
+/**
+ * Set the page rectangle in it's native units.
+ */
+void SPPage::setRect(Geom::Rect rect)
+{
+ this->x = rect.left();
+ this->y = rect.top();
+ this->width = rect.width();
+ this->height = rect.height();
+
+ // This is needed to update the xml
+ this->updateRepr();
+
+ // This eventually calls the ::update below while idle
+ this->requestModified(SP_OBJECT_MODIFIED_FLAG);
+}
+
+/**
+ * Set the page rectangle is desktop coordinates.
+ */
+void SPPage::setDesktopRect(Geom::Rect rect)
+{
+ rect *= document->doc2dt();
+ setRect(rect * document->getDocumentScale().inverse());
+}
+
+/**
+ * Set just the height and width from a predefined size.
+ */
+void SPPage::setDesktopSize(double width, double height)
+{
+ auto rect = getDesktopRect();
+ rect.setMax(rect.corner(0) + Geom::Point(width, height));
+ setDesktopRect(rect);
+}
+
+/**
+ * Get the items which are ONLY on this page and don't overlap.
+ *
+ * This ignores layers so items in the same layer which are shared
+ * between pages are not moved around or exported into pages they
+ * shouldn't be.
+ *
+ * @param hidden - Return hidden items (default: true)
+ */
+std::vector<SPItem *> SPPage::getExclusiveItems(bool hidden) const
+{
+ return document->getItemsInBox(0, getDocumentRect(), hidden, true, true, false);
+}
+
+/**
+ * Like ExcludiveItems above but get all the items which are inside or overlapping.
+ *
+ * @param hidden - Return hidden items (default: true)
+ */
+std::vector<SPItem *> SPPage::getOverlappingItems(bool hidden) const
+{
+ return document->getItemsPartiallyInBox(0, getDocumentRect(), hidden, true, true, false);
+}
+
+/**
+ * Return true if this item is contained within the page boundary.
+ */
+bool SPPage::itemOnPage(SPItem *item, bool contains) const
+{
+ if (auto box = item->desktopGeometricBounds()) {
+ if (contains) {
+ return getDesktopRect().contains(*box);
+ }
+ return getDesktopRect().intersects(*box);
+ }
+ return false;
+}
+
+/**
+ * Returns true if this page is the same as the viewport.
+ */
+bool SPPage::isViewportPage() const
+{
+ auto rect = document->preferredBounds();
+ return getDesktopRect().corner(0) == rect->corner(0);
+}
+
+/**
+ * Shows the page in the given canvas item group.
+ */
+void SPPage::showPage(Inkscape::CanvasItemGroup *fg, Inkscape::CanvasItemGroup *bg)
+{
+ _canvas_item->add(getDesktopRect(), fg, bg);
+ // The final steps are completed in an update cycle
+ this->requestDisplayUpdate(SP_OBJECT_MODIFIED_FLAG);
+}
+
+/**
+ * Sets the default attributes from the namedview.
+ */
+bool SPPage::setDefaultAttributes()
+{
+ if (document->getPageManager().setDefaultAttributes(_canvas_item)) {
+ this->requestDisplayUpdate(SP_OBJECT_MODIFIED_FLAG);
+ return true;
+ }
+ return false;
+}
+
+/**
+ * Set the selected high-light for this page.
+ */
+void SPPage::setSelected(bool sel)
+{
+ this->_canvas_item->is_selected = sel;
+ this->requestDisplayUpdate(SP_OBJECT_MODIFIED_FLAG);
+}
+
+/**
+ * Returns the page number (order of pages) starting at 1
+ */
+int SPPage::getPageIndex() const
+{
+ return document->getPageManager().getPageIndex(this);
+}
+
+/**
+ * Set this page to a new order in the page stack.
+ *
+ * @param index - Placement of page in the stack, starting at '0'
+ * @param swap_page - Swap the rectangle position
+ *
+ * @returns true if page has been moved.
+ */
+bool SPPage::setPageIndex(int index, bool swap_page)
+{
+ int current = getPageIndex();
+
+ if (current != index) {
+ auto &page_manager = document->getPageManager();
+
+ // The page we're going to be shifting to
+ auto sibling = page_manager.getPage(index);
+
+ // Insertions are done to the right of the sibling
+ if (index < current) {
+ index -= 1;
+ }
+ auto insert_after = page_manager.getPage(index);
+
+ // We may have selected an index off the end, so attach it after the last page.
+ if (!insert_after && index > 0) {
+ insert_after = page_manager.getLastPage();
+ sibling = nullptr; // disable swap
+ }
+
+ if (insert_after) {
+ if (this == insert_after) {
+ g_warning("Page is already at this index. Not moving.");
+ return false;
+ }
+ // Attach after the given page
+ getRepr()->parent()->changeOrder(getRepr(), insert_after->getRepr());
+ } else {
+ // Attach to before any existing page
+ sibling = page_manager.getFirstPage();
+ getRepr()->parent()->changeOrder(getRepr(), nullptr);
+ }
+ if (sibling && swap_page) {
+ swapPage(sibling, true);
+ }
+ return true;
+ }
+ return false;
+}
+
+/**
+ * Returns the sibling page next to this one in the stack order.
+ */
+SPPage *SPPage::getNextPage()
+{
+ SPObject *item = this;
+ while ((item = item->getNext())) {
+ if (auto next = dynamic_cast<SPPage *>(item)) {
+ return next;
+ }
+ }
+ return nullptr;
+}
+
+/**
+ * Returns the sibling page previous to this one in the stack order.
+ */
+SPPage *SPPage::getPreviousPage()
+{
+ SPObject *item = this;
+ while ((item = item->getPrev())) {
+ if (auto prev = dynamic_cast<SPPage *>(item)) {
+ return prev;
+ }
+ }
+ return nullptr;
+}
+
+/**
+ * Move the page by the given affine, in desktop units.
+ *
+ * @param translate - The positional translation to apply.
+ * @param with_objects - Flag to request that connected objects also move.
+ */
+void SPPage::movePage(Geom::Affine translate, bool with_objects)
+{
+ if (translate.isTranslation()) {
+ if (with_objects) {
+ // Move each item that is overlapping this page too
+ moveItems(translate * document->dt2doc(), getOverlappingItems());
+ }
+ setDesktopRect(getDesktopRect() * translate);
+ }
+}
+
+/**
+ * Move the given items by the given translation in document units.
+ *
+ * @param translate - The movement to be applied
+ * @param objects - a vector of SPItems to move
+ */
+void SPPage::moveItems(Geom::Affine translate, std::vector<SPItem *> const &objects)
+{
+ for (auto &item : objects) {
+ if (item->isLocked()) {
+ continue;
+ }
+ if (auto parent_item = dynamic_cast<SPItem *>(item->parent)) {
+ auto move = item->i2dt_affine() * (translate * parent_item->i2doc_affine().inverse());
+ item->doWriteTransform(move, &move, false);
+ }
+ }
+}
+
+/**
+ * Swap the locations of this page with another page (see movePage)
+ *
+ * @param other - The other page to swap with
+ * @param with_objects - Should the page objects move too.
+ */
+void SPPage::swapPage(SPPage *other, bool with_objects)
+{
+ // Swapping with the viewport page must be handled gracefully.
+ if (this->isViewportPage()) {
+ auto other_rect = other->getDesktopRect();
+ auto new_rect = Geom::Rect(Geom::Point(0, 0),
+ Geom::Point(other_rect.width(), other_rect.height()));
+ this->document->fitToRect(new_rect, false);
+ } else if (other->isViewportPage()) {
+ other->swapPage(this, with_objects);
+ return;
+ }
+
+ auto this_affine = Geom::Translate(getDesktopRect().corner(0));
+ auto other_affine = Geom::Translate(other->getDesktopRect().corner(0));
+ movePage(this_affine.inverse() * other_affine, with_objects);
+ other->movePage(other_affine.inverse() * this_affine, with_objects);
+}
+
+void SPPage::update(SPCtx * /*ctx*/, unsigned int /*flags*/)
+{
+ // This is manual because this is not an SPItem, but it's own visual identity.
+ _canvas_item->update(getDesktopRect(), this->label());
+}
+
+/**
+ * Write out the page's data into it's xml structure.
+ */
+Inkscape::XML::Node *SPPage::write(Inkscape::XML::Document *xml_doc, Inkscape::XML::Node *repr, guint flags)
+{
+ if ((flags & SP_OBJECT_WRITE_BUILD) && !repr) {
+ repr = xml_doc->createElement("inkscape:page");
+ }
+
+ repr->setAttributeSvgDouble("x", this->x.computed);
+ repr->setAttributeSvgDouble("y", this->y.computed);
+ repr->setAttributeSvgDouble("width", this->width.computed);
+ repr->setAttributeSvgDouble("height", this->height.computed);
+
+ return SPObject::write(xml_doc, repr, flags);
+}
+
+std::string SPPage::getDefaultLabel() const
+{
+ gchar *format = g_strdup_printf(_("Page %d"), getPagePosition());
+ auto ret = std::string(format);
+ g_free(format);
+ return ret;
+}
+
+std::string SPPage::getLabel() const
+{
+ auto ret = label();
+ if (!ret) {
+ return getDefaultLabel();
+ }
+ return std::string(ret);
+}
+
+/*
+ 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 :