// SPDX-License-Identifier: GPL-2.0-or-later /** * @file * SVG stylesheets implementation. */ /* Authors: * Lauris Kaplinski * Peter Moulder * bulia byak * Abhishek Sharma * Tavmjong Bah * Kris De Gussem * * Copyright (C) 2001-2002 Lauris Kaplinski * Copyright (C) 2001 Ximian, Inc. * Copyright (C) 2005 Monash University * Copyright (C) 2012 Kris De Gussem * Copyright (C) 2014-2015 Tavmjong Bah * * Released under GNU GPL v2+, read the file 'COPYING' for more information. */ #include "style.h" #include #include #include #include #include #include #include "attributes.h" #include "bad-uri-exception.h" #include "document.h" #include "preferences.h" #include "3rdparty/libcroco/cr-sel-eng.h" #include "object/sp-paint-server.h" #include "object/uri-references.h" #include "object/uri.h" #include "svg/css-ostringstream.h" #include "svg/svg.h" #include "util/units.h" #include "xml/croco-node-iface.h" #include "xml/simple-document.h" #if !GLIB_CHECK_VERSION(2, 64, 0) #define g_warning_once g_warning #endif using Inkscape::CSSOStringStream; #define BMAX 8192 struct SPStyleEnum; int SPStyle::_count = 0; /*######################### ## FORWARD DECLARATIONS #########################*/ void sp_style_filter_ref_changed(SPObject *old_ref, SPObject *ref, SPStyle *style); void sp_style_fill_paint_server_ref_changed(SPObject *old_ref, SPObject *ref, SPStyle *style); void sp_style_stroke_paint_server_ref_changed(SPObject *old_ref, SPObject *ref, SPStyle *style); static void sp_style_object_release(SPObject *object, SPStyle *style); static CRSelEng *sp_repr_sel_eng(); /** * Helper class for SPStyle property member lookup by SPAttr or * by name, and for iterating over ordered members. */ class SPStylePropHelper { SPStylePropHelper() { #define REGISTER_PROPERTY(id, member, name) \ g_assert(decltype(SPStyle::member)::static_id() == id); \ _register(reinterpret_cast(&SPStyle::member), id) /* name unused */ // SVG 2: Attributes promoted to properties REGISTER_PROPERTY(SPAttr::D, d, "d"); // 'color' must be before 'fill', 'stroke', 'text-decoration-color', ... REGISTER_PROPERTY(SPAttr::COLOR, color, "color"); // 'font-size'/'font' must be before properties that need to know em, ex size (SPILength, // SPILengthOrNormal) REGISTER_PROPERTY(SPAttr::FONT_STYLE, font_style, "font-style"); REGISTER_PROPERTY(SPAttr::FONT_VARIANT, font_variant, "font-variant"); REGISTER_PROPERTY(SPAttr::FONT_WEIGHT, font_weight, "font-weight"); REGISTER_PROPERTY(SPAttr::FONT_STRETCH, font_stretch, "font-stretch"); REGISTER_PROPERTY(SPAttr::FONT_SIZE, font_size, "font-size"); REGISTER_PROPERTY(SPAttr::LINE_HEIGHT, line_height, "line-height"); REGISTER_PROPERTY(SPAttr::FONT_FAMILY, font_family, "font-family"); REGISTER_PROPERTY(SPAttr::FONT, font, "font"); REGISTER_PROPERTY(SPAttr::INKSCAPE_FONT_SPEC, font_specification, "-inkscape-font-specification"); // Font variants REGISTER_PROPERTY(SPAttr::FONT_VARIANT_LIGATURES, font_variant_ligatures, "font-variant-ligatures"); REGISTER_PROPERTY(SPAttr::FONT_VARIANT_POSITION, font_variant_position, "font-variant-position"); REGISTER_PROPERTY(SPAttr::FONT_VARIANT_CAPS, font_variant_caps, "font-variant-caps"); REGISTER_PROPERTY(SPAttr::FONT_VARIANT_NUMERIC, font_variant_numeric, "font-variant-numeric"); REGISTER_PROPERTY(SPAttr::FONT_VARIANT_ALTERNATES, font_variant_alternates, "font-variant-alternates"); REGISTER_PROPERTY(SPAttr::FONT_VARIANT_EAST_ASIAN, font_variant_east_asian, "font-variant-east-asian"); REGISTER_PROPERTY(SPAttr::FONT_FEATURE_SETTINGS, font_feature_settings, "font-feature-settings"); // Variable Fonts REGISTER_PROPERTY(SPAttr::FONT_VARIATION_SETTINGS, font_variation_settings, "font-variation-settings"); REGISTER_PROPERTY(SPAttr::TEXT_INDENT, text_indent, "text-indent"); REGISTER_PROPERTY(SPAttr::TEXT_ALIGN, text_align, "text-align"); REGISTER_PROPERTY(SPAttr::TEXT_DECORATION, text_decoration, "text-decoration"); REGISTER_PROPERTY(SPAttr::TEXT_DECORATION_LINE, text_decoration_line, "text-decoration-line"); REGISTER_PROPERTY(SPAttr::TEXT_DECORATION_STYLE, text_decoration_style, "text-decoration-style"); REGISTER_PROPERTY(SPAttr::TEXT_DECORATION_COLOR, text_decoration_color, "text-decoration-color"); REGISTER_PROPERTY(SPAttr::TEXT_DECORATION_FILL, text_decoration_fill, "text-decoration-fill"); REGISTER_PROPERTY(SPAttr::TEXT_DECORATION_STROKE, text_decoration_stroke, "text-decoration-stroke"); REGISTER_PROPERTY(SPAttr::LETTER_SPACING, letter_spacing, "letter-spacing"); REGISTER_PROPERTY(SPAttr::WORD_SPACING, word_spacing, "word-spacing"); REGISTER_PROPERTY(SPAttr::TEXT_TRANSFORM, text_transform, "text-transform"); REGISTER_PROPERTY(SPAttr::WRITING_MODE, writing_mode, "writing-mode"); REGISTER_PROPERTY(SPAttr::DIRECTION, direction, "direction"); REGISTER_PROPERTY(SPAttr::TEXT_ORIENTATION, text_orientation, "text-orientation"); REGISTER_PROPERTY(SPAttr::DOMINANT_BASELINE, dominant_baseline, "dominant-baseline"); REGISTER_PROPERTY(SPAttr::BASELINE_SHIFT, baseline_shift, "baseline-shift"); REGISTER_PROPERTY(SPAttr::TEXT_ANCHOR, text_anchor, "text-anchor"); REGISTER_PROPERTY(SPAttr::WHITE_SPACE, white_space, "white-space"); REGISTER_PROPERTY(SPAttr::SHAPE_INSIDE, shape_inside, "shape-inside"); REGISTER_PROPERTY(SPAttr::SHAPE_SUBTRACT, shape_subtract, "shape-subtract"); REGISTER_PROPERTY(SPAttr::SHAPE_PADDING, shape_padding, "shape-padding"); REGISTER_PROPERTY(SPAttr::SHAPE_MARGIN, shape_margin, "shape-margin"); REGISTER_PROPERTY(SPAttr::INLINE_SIZE, inline_size, "inline-size"); REGISTER_PROPERTY(SPAttr::CLIP_RULE, clip_rule, "clip-rule"); REGISTER_PROPERTY(SPAttr::DISPLAY, display, "display"); REGISTER_PROPERTY(SPAttr::OVERFLOW_, overflow, "overflow"); REGISTER_PROPERTY(SPAttr::VISIBILITY, visibility, "visibility"); REGISTER_PROPERTY(SPAttr::OPACITY, opacity, "opacity"); REGISTER_PROPERTY(SPAttr::ISOLATION, isolation, "isolation"); REGISTER_PROPERTY(SPAttr::MIX_BLEND_MODE, mix_blend_mode, "mix-blend-mode"); REGISTER_PROPERTY(SPAttr::COLOR_INTERPOLATION, color_interpolation, "color-interpolation"); REGISTER_PROPERTY(SPAttr::COLOR_INTERPOLATION_FILTERS, color_interpolation_filters, "color-interpolation-filters"); REGISTER_PROPERTY(SPAttr::SOLID_COLOR, solid_color, "solid-color"); REGISTER_PROPERTY(SPAttr::SOLID_OPACITY, solid_opacity, "solid-opacity"); REGISTER_PROPERTY(SPAttr::VECTOR_EFFECT, vector_effect, "vector-effect"); REGISTER_PROPERTY(SPAttr::FILL, fill, "fill"); REGISTER_PROPERTY(SPAttr::FILL_OPACITY, fill_opacity, "fill-opacity"); REGISTER_PROPERTY(SPAttr::FILL_RULE, fill_rule, "fill-rule"); REGISTER_PROPERTY(SPAttr::STROKE, stroke, "stroke"); REGISTER_PROPERTY(SPAttr::STROKE_WIDTH, stroke_width, "stroke-width"); REGISTER_PROPERTY(SPAttr::STROKE_LINECAP, stroke_linecap, "stroke-linecap"); REGISTER_PROPERTY(SPAttr::STROKE_LINEJOIN, stroke_linejoin, "stroke-linejoin"); REGISTER_PROPERTY(SPAttr::STROKE_MITERLIMIT, stroke_miterlimit, "stroke-miterlimit"); REGISTER_PROPERTY(SPAttr::STROKE_DASHARRAY, stroke_dasharray, "stroke-dasharray"); REGISTER_PROPERTY(SPAttr::STROKE_DASHOFFSET, stroke_dashoffset, "stroke-dashoffset"); REGISTER_PROPERTY(SPAttr::STROKE_OPACITY, stroke_opacity, "stroke-opacity"); REGISTER_PROPERTY(SPAttr::STROKE_EXTENSIONS, stroke_extensions, "-inkscape-stroke"); REGISTER_PROPERTY(SPAttr::MARKER, marker, "marker"); REGISTER_PROPERTY(SPAttr::MARKER_START, marker_start, "marker-start"); REGISTER_PROPERTY(SPAttr::MARKER_MID, marker_mid, "marker-mid"); REGISTER_PROPERTY(SPAttr::MARKER_END, marker_end, "marker-end"); REGISTER_PROPERTY(SPAttr::PAINT_ORDER, paint_order, "paint-order"); REGISTER_PROPERTY(SPAttr::FILTER, filter, "filter"); REGISTER_PROPERTY(SPAttr::COLOR_RENDERING, color_rendering, "color-rendering"); REGISTER_PROPERTY(SPAttr::IMAGE_RENDERING, image_rendering, "image-rendering"); REGISTER_PROPERTY(SPAttr::SHAPE_RENDERING, shape_rendering, "shape-rendering"); REGISTER_PROPERTY(SPAttr::TEXT_RENDERING, text_rendering, "text-rendering"); REGISTER_PROPERTY(SPAttr::ENABLE_BACKGROUND, enable_background, "enable-background"); REGISTER_PROPERTY(SPAttr::STOP_COLOR, stop_color, "stop-color"); REGISTER_PROPERTY(SPAttr::STOP_OPACITY, stop_opacity, "stop-opacity"); } // this is a singleton, copy not allowed SPStylePropHelper(SPStylePropHelper const&) = delete; public: /** * Singleton instance */ static SPStylePropHelper &instance() { static SPStylePropHelper _instance; return _instance; } /** * Get property pointer by enum */ SPIBase *get(SPStyle *style, SPAttr id) { auto it = m_id_map.find(id); if (it != m_id_map.end()) { return _get(style, it->second); } return nullptr; } /** * Get property pointer by name */ SPIBase *get(SPStyle *style, const std::string &name) { return get(style, sp_attribute_lookup(name.c_str())); } /** * Get a vector of property pointers * \todo provide iterator instead */ std::vector get_vector(SPStyle *style) { std::vector v; v.reserve(m_vector.size()); for (auto ptr : m_vector) { v.push_back(_get(style, ptr)); } return v; } private: SPIBase *_get(SPStyle *style, SPIBasePtr ptr) { return &(style->*ptr); } void _register(SPIBasePtr ptr, SPAttr id) { m_vector.push_back(ptr); if (id != SPAttr::INVALID) { m_id_map[id] = ptr; } } std::unordered_map m_id_map; std::vector m_vector; }; auto &_prop_helper = SPStylePropHelper::instance(); // C++11 allows one constructor to call another... might be useful. The original C code // had separate calls to create SPStyle, one with only SPDocument and the other with only // SPObject as parameters. SPStyle::SPStyle(SPDocument *document_in, SPObject *object_in) : // Unimplemented SVG 1.1: alignment-baseline, clip, clip-path, color-profile, cursor, // dominant-baseline, flood-color, flood-opacity, font-size-adjust, // glyph-orientation-horizontal, glyph-orientation-vertical, kerning, lighting-color, // pointer-events, unicode-bidi // 'font', 'font-size', and 'font-family' must come first as other properties depend on them // for calculated values (through 'em' and 'ex'). ('ex' is currently not read.) // The following properties can depend on 'em' and 'ex': // baseline-shift, kerning, letter-spacing, stroke-dash-offset, stroke-width, word-spacing, // Non-SVG 1.1: text-indent, line-spacing // Hidden in SPIFontStyle: (to be refactored) // font-family // font-specification // SVG 2 attributes promoted to properties. (When geometry properties are added, move after font.) d( false), // SPIString Not inherited! // Font related properties and 'font' shorthand font_style( SP_CSS_FONT_STYLE_NORMAL), font_variant( SP_CSS_FONT_VARIANT_NORMAL), font_weight( SP_CSS_FONT_WEIGHT_NORMAL), font_stretch( SP_CSS_FONT_STRETCH_NORMAL), font_size(), line_height( 1.25 ), // SPILengthOrNormal font_family( ), // SPIString w/default font(), // SPIFont font_specification( ), // SPIString // Font variants (Features) font_variant_ligatures( ), font_variant_position( SP_CSS_FONT_VARIANT_POSITION_NORMAL), font_variant_caps( SP_CSS_FONT_VARIANT_CAPS_NORMAL), font_variant_numeric( ), font_variant_alternates(SP_CSS_FONT_VARIANT_ALTERNATES_NORMAL), font_variant_east_asian(), font_feature_settings( ), // Variable Fonts font_variation_settings(), // SPIFontVariationSettings // Text related properties text_indent( ), // SPILength text_align( SP_CSS_TEXT_ALIGN_START), letter_spacing( ), // SPILengthOrNormal word_spacing( ), // SPILengthOrNormal text_transform( SP_CSS_TEXT_TRANSFORM_NONE), direction( SP_CSS_DIRECTION_LTR), writing_mode( SP_CSS_WRITING_MODE_LR_TB), text_orientation( SP_CSS_TEXT_ORIENTATION_MIXED), dominant_baseline( SP_CSS_BASELINE_AUTO), baseline_shift(), text_anchor( SP_CSS_TEXT_ANCHOR_START), white_space( SP_CSS_WHITE_SPACE_NORMAL), // SVG 2 Text Wrapping shape_inside( ), // SPIString shape_subtract( ), // SPIString shape_padding( ), // SPILength for now shape_margin( ), // SPILength for now inline_size( ), // SPILength for now text_decoration(), text_decoration_line(), text_decoration_style(), text_decoration_color( ), // SPIColor text_decoration_fill( ), // SPIPaint text_decoration_stroke( ), // SPIPaint // General visual properties clip_rule( SP_WIND_RULE_NONZERO), display( SP_CSS_DISPLAY_INLINE, false), overflow( SP_CSS_OVERFLOW_VISIBLE, false), visibility( SP_CSS_VISIBILITY_VISIBLE), opacity( false), isolation( SP_CSS_ISOLATION_AUTO), mix_blend_mode( SP_CSS_BLEND_NORMAL), paint_order(), // SPIPaintOrder // Color properties color( ), // SPIColor color_interpolation( SP_CSS_COLOR_INTERPOLATION_SRGB), color_interpolation_filters(SP_CSS_COLOR_INTERPOLATION_LINEARRGB), // Solid color properties solid_color( ), // SPIColor solid_opacity( ), // Vector effects vector_effect(), // Fill properties fill( ), // SPIPaint fill_opacity( ), fill_rule( SP_WIND_RULE_NONZERO), // Stroke properties stroke( ), // SPIPaint stroke_width( 1.0), // SPILength stroke_linecap( SP_STROKE_LINECAP_BUTT), stroke_linejoin( SP_STROKE_LINEJOIN_MITER), stroke_miterlimit( 4), // SPIFloat (only use of float!) stroke_dasharray(), // SPIDashArray stroke_dashoffset( ), // SPILength for now stroke_opacity( ), stroke_extensions( ), marker( ), // SPIString marker_start( ), // SPIString marker_mid( ), // SPIString marker_end( ), // SPIString // Filter properties filter(), enable_background( SP_CSS_BACKGROUND_ACCUMULATE, false), // Rendering hint properties color_rendering( SP_CSS_COLOR_RENDERING_AUTO), image_rendering( SP_CSS_IMAGE_RENDERING_AUTO), shape_rendering( SP_CSS_SHAPE_RENDERING_AUTO), text_rendering( SP_CSS_TEXT_RENDERING_AUTO), // Stop color and opacity stop_color( false), // SPIColor, does not inherit stop_opacity( false) // Does not inherit { // std::cout << "SPStyle::SPStyle( SPDocument ): Entrance: (" << _count << ")" << std::endl; // std::cout << " Document: " << (document_in?"present":"null") << std::endl; // std::cout << " Object: " // << (object_in?(object_in->getId()?object_in->getId():"id null"):"object null") << std::endl; // static bool first = true; // if( first ) { // std::cout << "Size of SPStyle: " << sizeof(SPStyle) << std::endl; // std::cout << " SPIBase: " << sizeof(SPIBase) << std::endl; // std::cout << " SPIFloat: " << sizeof(SPIFloat) << std::endl; // std::cout << " SPIScale24: " << sizeof(SPIScale24) << std::endl; // std::cout << " SPILength: " << sizeof(SPILength) << std::endl; // std::cout << " SPILengthOrNormal: " << sizeof(SPILengthOrNormal) << std::endl; // std::cout << " SPIColor: " << sizeof(SPIColor) << std::endl; // std::cout << " SPIPaint: " << sizeof(SPIPaint) << std::endl; // std::cout << " SPITextDecorationLine" << sizeof(SPITextDecorationLine) << std::endl; // std::cout << " Glib::ustring:" << sizeof(Glib::ustring) << std::endl; // std::cout << " SPColor: " << sizeof(SPColor) << std::endl; // first = false; // } ++_count; // Poor man's memory leak detector _refcount = 1; cloned = false; object = object_in; if( object ) { g_assert( SP_IS_OBJECT(object) ); document = object->document; release_connection = object->connectRelease(sigc::bind<1>(sigc::ptr_fun(&sp_style_object_release), this)); cloned = object->cloned; } else { document = document_in; } // 'font' shorthand requires access to included properties. font.setStylePointer( this ); // Properties that depend on 'font-size' for calculating lengths. baseline_shift.setStylePointer( this ); text_indent.setStylePointer( this ); line_height.setStylePointer( this ); letter_spacing.setStylePointer( this ); word_spacing.setStylePointer( this ); stroke_width.setStylePointer( this ); stroke_dashoffset.setStylePointer( this ); shape_padding.setStylePointer( this ); shape_margin.setStylePointer( this ); inline_size.setStylePointer( this ); // Properties that depend on 'color' text_decoration_color.setStylePointer(this); fill.setStylePointer(this); stroke.setStylePointer(this); color.setStylePointer(this); stop_color.setStylePointer(this); solid_color.setStylePointer(this); // 'text_decoration' shorthand requires access to included properties. text_decoration.setStylePointer( this ); // SPIPaint, SPIFilter needs access to 'this' (SPStyle) // for setting up signals... 'fill', 'stroke' already done filter.setStylePointer( this ); shape_inside.setStylePointer( this ); shape_subtract.setStylePointer( this ); // Used to iterate over markers marker_ptrs[SP_MARKER_LOC] = ▮ marker_ptrs[SP_MARKER_LOC_START] = &marker_start; marker_ptrs[SP_MARKER_LOC_MID] = &marker_mid; marker_ptrs[SP_MARKER_LOC_END] = &marker_end; // This might be too resource hungary... but for now it possible to loop over properties _properties = _prop_helper.get_vector(this); } SPStyle::~SPStyle() { // std::cout << "SPStyle::~SPStyle" << std::endl; --_count; // Poor man's memory leak detector. // Remove connections release_connection.disconnect(); fill_ps_changed_connection.disconnect(); stroke_ps_changed_connection.disconnect(); filter_changed_connection.disconnect(); // The following should be moved into SPIPaint and SPIFilter if (fill.value.href) { fill_ps_modified_connection.disconnect(); } if (stroke.value.href) { stroke_ps_modified_connection.disconnect(); } if (filter.href) { filter_modified_connection.disconnect(); } // Conjecture: all this SPStyle ref counting is not needed. SPObject creates an instance of // SPStyle when it is constructed and deletes it when it is destructed. The refcount is // incremented and decremented only in the files: display/drawing-item.cpp, // display/nr-filter-primitive.cpp, and libnrtype/Layout-TNG-Input.cpp. if( _refcount > 1 ) { std::cerr << "SPStyle::~SPStyle: ref count greater than 1! " << _refcount << std::endl; } // std::cout << "SPStyle::~SPStyle(): Exit\n" << std::endl; } const std::vector SPStyle::properties() { return this->_properties; } void SPStyle::clear(SPAttr id) { SPIBase *p = _prop_helper.get(this, id); if (p) { p->clear(); } else { g_warning("Unimplemented style property %d", (int)id); } } void SPStyle::clear() { for (auto * p : _properties) { p->clear(); } // Release connection to object, created in constructor. release_connection.disconnect(); // href->detach() called in fill->clear()... fill_ps_modified_connection.disconnect(); if (fill.value.href) { delete fill.value.href; fill.value.href = nullptr; } stroke_ps_modified_connection.disconnect(); if (stroke.value.href) { delete stroke.value.href; stroke.value.href = nullptr; } filter_modified_connection.disconnect(); if (filter.href) { delete filter.href; filter.href = nullptr; } if (document) { filter.href = new SPFilterReference(document); filter_changed_connection = filter.href->changedSignal().connect(sigc::bind(sigc::ptr_fun(sp_style_filter_ref_changed), this)); fill.value.href = new SPPaintServerReference(document); fill_ps_changed_connection = fill.value.href->changedSignal().connect(sigc::bind(sigc::ptr_fun(sp_style_fill_paint_server_ref_changed), this)); stroke.value.href = new SPPaintServerReference(document); stroke_ps_changed_connection = stroke.value.href->changedSignal().connect(sigc::bind(sigc::ptr_fun(sp_style_stroke_paint_server_ref_changed), this)); } cloned = false; } // Matches void sp_style_read(SPStyle *style, SPObject *object, Inkscape::XML::Node *repr) void SPStyle::read( SPObject *object, Inkscape::XML::Node *repr ) { // std::cout << "SPStyle::read( SPObject, Inkscape::XML::Node ): Entrance: " // << (object?(object->getId()?object->getId():"id null"):"object null") << " " // << (repr?(repr->name()?repr->name():"no name"):"repr null") // << std::endl; g_assert(repr != nullptr); g_assert(!object || (object->getRepr() == repr)); // // Uncomment to verify that we don't need to call clear. // std::cout << " Creating temp style for testing" << std::endl; // SPStyle *temp = new SPStyle(); // if( !(*temp == *this ) ) std::cout << "SPStyle::read: Need to clear" << std::endl; // delete temp; clear(); // FIXME, If this isn't here, EVERYTHING stops working! Why? if (object && object->cloned) { cloned = true; } /* 1. Style attribute */ // std::cout << " MERGING STYLE ATTRIBUTE" << std::endl; gchar const *val = repr->attribute("style"); if( val != nullptr && *val ) { _mergeString( val ); } /* 2 Style sheet */ if (object) { _mergeObjectStylesheet( object ); } /* 3 Presentation attributes */ for (auto * p : _properties) { // Shorthands are not allowed as presentation properties. Note: text-decoration and // font-variant are converted to shorthands in CSS 3 but can still be read as a // non-shorthand for compatibility with older renders, so they should not be in this list. if (p->id() != SPAttr::FONT && p->id() != SPAttr::MARKER) { p->readAttribute( repr ); } } /* 4 Cascade from parent */ if( object ) { if( object->parent ) { cascade( object->parent->style ); } } else if( repr->parent() ) { // When does this happen? // std::cout << "SPStyle::read(): reading via repr->parent()" << std::endl; SPStyle *parent = new SPStyle(); parent->read( nullptr, repr->parent() ); cascade( parent ); delete parent; } } /** * Read style properties from object's repr. * * 1. Reset existing object style * 2. Load current effective object style * 3. Load i attributes from immediate parent (which has to be up-to-date) */ void SPStyle::readFromObject( SPObject *object ) { // std::cout << "SPStyle::readFromObject: "<< (object->getId()?object->getId():"null") << std::endl; g_return_if_fail(object != nullptr); g_return_if_fail(SP_IS_OBJECT(object)); Inkscape::XML::Node *repr = object->getRepr(); g_return_if_fail(repr != nullptr); read( object, repr ); } /** * Read style properties from preferences. * @param path Preferences directory from which the style should be read */ void SPStyle::readFromPrefs(Glib::ustring const &path) { g_return_if_fail(!path.empty()); Inkscape::Preferences *prefs = Inkscape::Preferences::get(); // not optimal: we reconstruct the node based on the prefs, then pass it to // sp_style_read for actual processing. Inkscape::XML::SimpleDocument *tempdoc = new Inkscape::XML::SimpleDocument; Inkscape::XML::Node *tempnode = tempdoc->createElement("prefs"); std::vector attrs = prefs->getAllEntries(path); for (auto & attr : attrs) { tempnode->setAttribute(attr.getEntryName(), attr.getString()); } read( nullptr, tempnode ); Inkscape::GC::release(tempnode); Inkscape::GC::release(tempdoc); delete tempdoc; } // Matches sp_style_merge_property(SPStyle *style, gint id, gchar const *val) void SPStyle::readIfUnset(SPAttr id, gchar const *val, SPStyleSrc const &source ) { // std::cout << "SPStyle::readIfUnset: Entrance: " << sp_attribute_name(id) << ": " << (val?val:"null") << std::endl; // To Do: If it is not too slow, use std::map instead of std::vector inorder to remove switch() // (looking up SPAttr::xxxx already uses a hash). g_return_if_fail(val != nullptr); switch (id) { /* SVG */ /* Clip/Mask */ case SPAttr::CLIP_PATH: /** \todo * This is a workaround. Inkscape only supports 'clip-path' as SVG attribute, not as * style property. By having both CSS and SVG attribute set, editing of clip-path * will fail, since CSS always overwrites SVG attributes. * Fixes Bug #324849 */ g_warning_once("attribute 'clip-path' given as CSS"); //XML Tree being directly used here. if (object) { object->setAttribute("clip-path", val); } return; case SPAttr::MASK: /** \todo * See comment for SPAttr::CLIP_PATH */ g_warning_once("attribute 'mask' given as CSS"); //XML Tree being directly used here. if (object) { object->setAttribute("mask", val); } return; case SPAttr::FILTER: if( !filter.inherit ) filter.readIfUnset( val, source ); return; case SPAttr::COLOR_INTERPOLATION: // We read it but issue warning color_interpolation.readIfUnset( val, source ); if( color_interpolation.value != SP_CSS_COLOR_INTERPOLATION_SRGB ) { g_warning("Inkscape currently only supports color-interpolation = sRGB"); } return; } auto p = _prop_helper.get(this, id); if (p) { p->readIfUnset(val, source); } else { g_warning("Unimplemented style property %d", (int)id); } } // return if is seted property bool SPStyle::isSet(SPAttr id) { bool set = false; switch (id) { case SPAttr::CLIP_PATH: return set; case SPAttr::MASK: return set; case SPAttr::FILTER: if (!filter.inherit) set = filter.set; return set; case SPAttr::COLOR_INTERPOLATION: // We read it but issue warning return color_interpolation.set; } auto p = _prop_helper.get(this, id); if (p) { return p->set; } else { g_warning("Unimplemented style property %d", (int)id); return set; } } /** * Outputs the style to a CSS string. * * Use with SP_STYLE_FLAG_ALWAYS for copying an object's complete cascaded style to * style_clipboard. * * Use with SP_STYLE_FLAG_IFDIFF and a pointer to the parent class when you need a CSS string for * an object in the document tree. * * \pre flags in {IFSET, ALWAYS, IFDIFF}. * \pre base. * \post ret != NULL. */ Glib::ustring SPStyle::write( guint const flags, SPStyleSrc const style_src_req, SPStyle const *const base ) const { // std::cout << "SPStyle::write: flags: " << flags << std::endl; // If not excluding this case, we'd end up writing all non-inheritable properties. // Can happen when adding fallback s to text like this: // Hello if (base == this) { assert((flags & SP_STYLE_FLAG_IFDIFF) && !(flags & SP_STYLE_FLAG_ALWAYS)); return {}; } Glib::ustring style_string; for(std::vector::size_type i = 0; i != _properties.size(); ++i) { if( base != nullptr ) { style_string += _properties[i]->write( flags, style_src_req, base->_properties[i] ); } else { style_string += _properties[i]->write( flags, style_src_req, nullptr ); } } // Extended properties. Cascading not supported. for (auto const &pair : extended_properties) { // std::cout << "extended property: " << pair.first << " = " << pair.second << std::endl; style_string += pair.first + ":" + pair.second + ";"; } // Remove trailing ';' if( style_string.size() > 0 ) { style_string.erase( style_string.size() - 1 ); } return style_string; } /** * Get CSS string for set properties, or with SP_STYLE_FLAG_ALWAYS, for all properties. */ Glib::ustring SPStyle::write(unsigned int flags) const { assert(flags & (SP_STYLE_FLAG_IFSET | SP_STYLE_FLAG_ALWAYS)); return write(flags, SPStyleSrc::UNSET); } /** * Get CSS string for set properties from the requested source */ Glib::ustring SPStyle::write(SPStyleSrc style_src_req) const { assert(style_src_req != SPStyleSrc::UNSET); return write(SP_STYLE_FLAG_IFSRC | SP_STYLE_FLAG_IFSET, style_src_req); } /** * Get CSS string for set properties which are different from the given * base style. If base is NULL, all set flags are considered different. */ Glib::ustring SPStyle::writeIfDiff(SPStyle const *base) const { return write(SP_STYLE_FLAG_IFDIFF, SPStyleSrc::UNSET, base); } // Corresponds to sp_style_merge_from_parent() /** * Sets computed values in \a style, which may involve inheriting from (or in some other way * calculating from) corresponding computed values of \a parent. * * References: http://www.w3.org/TR/SVG11/propidx.html shows what properties inherit by default. * http://www.w3.org/TR/SVG11/styling.html#Inheritance gives general rules as to what it means to * inherit a value. http://www.w3.org/TR/REC-CSS2/cascade.html#computed-value is more precise * about what the computed value is (not obvious for lengths). * * \pre \a parent's computed values are already up-to-date. */ void SPStyle::cascade( SPStyle const *const parent ) { // std::cout << "SPStyle::cascade: " << (object->getId()?object->getId():"null") << std::endl; for(std::vector::size_type i = 0; i != _properties.size(); ++i) { _properties[i]->cascade( parent->_properties[i] ); } } // Corresponds to sp_style_merge_from_dying_parent() /** * Combine \a style and \a parent style specifications into a single style specification that * preserves (as much as possible) the effect of the existing \a style being a child of \a parent. * * Called when the parent repr is to be removed (e.g. the parent is a \ element that is being * unlinked), in which case we copy/adapt property values that are explicitly set in \a parent, * trying to retain the same visual appearance once the parent is removed. Interesting cases are * when there is unusual interaction with the parent's value (opacity, display) or when the value * can be specified as relative to the parent computed value (font-size, font-weight etc.). * * Doesn't update computed values of \a style. For correctness, you should subsequently call * sp_style_merge_from_parent against the new parent (presumably \a parent's parent) even if \a * style was previously up-to-date wrt \a parent. * * \pre \a parent's computed values are already up-to-date. * (\a style's computed values needn't be up-to-date.) */ void SPStyle::merge( SPStyle const *const parent ) { // std::cout << "SPStyle::merge" << std::endl; for(std::vector::size_type i = 0; i != _properties.size(); ++i) { _properties[i]->merge( parent->_properties[i] ); } } /** * Parses a style="..." string and merges it with an existing SPStyle. */ void SPStyle::mergeString( gchar const *const p ) { _mergeString( p ); } /** * Append an existing css statement into this style, used in css editing * always appends declarations as STYLE_SHEET properties. */ void SPStyle::mergeStatement( CRStatement *statement ) { if (statement->type != RULESET_STMT) { return; } CRDeclaration *decl_list = nullptr; cr_statement_ruleset_get_declarations (statement, &decl_list); if (decl_list) { _mergeDeclList(decl_list, SPStyleSrc::STYLE_SHEET); } } // Mostly for unit testing bool SPStyle::operator==(const SPStyle& rhs) { // Uncomment for testing // for(std::vector::size_type i = 0; i != _properties.size(); ++i) { // if( *_properties[i] != *rhs._properties[i]) // std::cout << _properties[i]->name << ": " // << _properties[i]->write(SP_STYLE_FLAG_ALWAYS,NULL) << " " // << rhs._properties[i]->write(SP_STYLE_FLAG_ALWAYS,NULL) // << (*_properties[i] == *rhs._properties[i]) << std::endl; // } for(std::vector::size_type i = 0; i != _properties.size(); ++i) { if( *_properties[i] != *rhs._properties[i]) return false; } return true; } void SPStyle::_mergeString( gchar const *const p ) { // std::cout << "SPStyle::_mergeString: " << (p?p:"null") << std::endl; CRDeclaration *const decl_list = cr_declaration_parse_list_from_buf(reinterpret_cast(p), CR_UTF_8); if (decl_list) { _mergeDeclList( decl_list, SPStyleSrc::STYLE_PROP ); cr_declaration_destroy(decl_list); } } void SPStyle::_mergeDeclList( CRDeclaration const *const decl_list, SPStyleSrc const &source ) { // std::cout << "SPStyle::_mergeDeclList" << std::endl; // In reverse order, as later declarations to take precedence over earlier ones. // (Properties are only set if not previously set. See: // Ref: http://www.w3.org/TR/REC-CSS2/cascade.html#cascading-order point 4.) if (decl_list->next) { _mergeDeclList( decl_list->next, source ); } _mergeDecl( decl_list, source ); } void SPStyle::_mergeDecl( CRDeclaration const *const decl, SPStyleSrc const &source ) { // std::cout << "SPStyle::_mergeDecl" << std::endl; auto prop_idx = sp_attribute_lookup(decl->property->stryng->str); if (prop_idx != SPAttr::INVALID) { /** \todo * effic: Test whether the property is already set before trying to * convert to string. Alternatively, set from CRTerm directly rather * than converting to string. */ if (!isSet(prop_idx) || decl->important) { guchar *const str_value_unsigned = cr_term_to_string(decl->value); gchar *const str_value = reinterpret_cast(str_value_unsigned); // Add "!important" rule if necessary as this is not handled by cr_term_to_string(). gchar const *important = decl->important ? " !important" : ""; Inkscape::CSSOStringStream os; os << str_value << important; readIfUnset(prop_idx, os.str().c_str(), source); g_free(str_value); } } else { gchar const *key = decl->property->stryng->str; auto value = reinterpret_cast(cr_term_to_string(decl->value)); if (g_str_has_prefix(key, "--")) { g_warning("Ignoring CSS variable: %s", key); } else if (g_str_has_prefix(key, "-")) { extended_properties[key] = value; } else { g_warning("Ignoring unrecognized CSS property: %s", key); } g_free(value); } } void SPStyle::_mergeProps( CRPropList *const props ) { // std::cout << "SPStyle::_mergeProps" << std::endl; // In reverse order, as later declarations to take precedence over earlier ones. if (props) { _mergeProps( cr_prop_list_get_next( props ) ); CRDeclaration *decl = nullptr; cr_prop_list_get_decl(props, &decl); _mergeDecl( decl, SPStyleSrc::STYLE_SHEET ); } } void SPStyle::_mergeObjectStylesheet( SPObject const *const object ) { // std::cout << "SPStyle::_mergeObjectStylesheet: " << (object->getId()?object->getId():"null") << std::endl; _mergeObjectStylesheet(object, object->document); } void SPStyle::_mergeObjectStylesheet( SPObject const *const object, SPDocument *const document ) { static CRSelEng *sel_eng = sp_repr_sel_eng(); if (auto *const parent = document->getParent()) { _mergeObjectStylesheet(object, parent); } else if (auto *const parent = document->get_reference_document()) { _mergeObjectStylesheet(object, parent); } CRPropList *props = nullptr; //XML Tree being directly used here while it shouldn't be. CRStatus status = cr_sel_eng_get_matched_properties_from_cascade(sel_eng, document->getStyleCascade(), object->getRepr(), &props); g_return_if_fail(status == CR_OK); /// \todo Check what errors can occur, and handle them properly. if (props) { _mergeProps(props); cr_prop_list_destroy(props); } } // Used for input into Pango. Must be computed value! std::string SPStyle::getFontFeatureString() { std::string feature_string; if ( !(font_variant_ligatures.computed & SP_CSS_FONT_VARIANT_LIGATURES_COMMON) ) feature_string += "liga 0, clig 0, "; if ( font_variant_ligatures.computed & SP_CSS_FONT_VARIANT_LIGATURES_DISCRETIONARY ) feature_string += "dlig, "; if ( font_variant_ligatures.computed & SP_CSS_FONT_VARIANT_LIGATURES_HISTORICAL ) feature_string += "hlig, "; if ( !(font_variant_ligatures.computed & SP_CSS_FONT_VARIANT_LIGATURES_CONTEXTUAL) ) feature_string += "calt 0, "; switch (font_variant_position.computed) { case SP_CSS_FONT_VARIANT_POSITION_SUB: feature_string += "subs, "; break; case SP_CSS_FONT_VARIANT_POSITION_SUPER: feature_string += "sups, "; } switch (font_variant_caps.computed) { case SP_CSS_FONT_VARIANT_CAPS_SMALL: feature_string += "smcp, "; break; case SP_CSS_FONT_VARIANT_CAPS_ALL_SMALL: feature_string += "smcp, c2sc, "; break; case SP_CSS_FONT_VARIANT_CAPS_PETITE: feature_string += "pcap, "; break; case SP_CSS_FONT_VARIANT_CAPS_ALL_PETITE: feature_string += "pcap, c2pc, "; break; case SP_CSS_FONT_VARIANT_CAPS_UNICASE: feature_string += "unic, "; break; case SP_CSS_FONT_VARIANT_CAPS_TITLING: feature_string += "titl, "; } if ( font_variant_numeric.computed & SP_CSS_FONT_VARIANT_NUMERIC_LINING_NUMS ) feature_string += "lnum, "; if ( font_variant_numeric.computed & SP_CSS_FONT_VARIANT_NUMERIC_OLDSTYLE_NUMS ) feature_string += "onum, "; if ( font_variant_numeric.computed & SP_CSS_FONT_VARIANT_NUMERIC_PROPORTIONAL_NUMS ) feature_string += "pnum, "; if ( font_variant_numeric.computed & SP_CSS_FONT_VARIANT_NUMERIC_TABULAR_NUMS ) feature_string += "tnum, "; if ( font_variant_numeric.computed & SP_CSS_FONT_VARIANT_NUMERIC_DIAGONAL_FRACTIONS ) feature_string += "frac, "; if ( font_variant_numeric.computed & SP_CSS_FONT_VARIANT_NUMERIC_STACKED_FRACTIONS ) feature_string += "afrc, "; if ( font_variant_numeric.computed & SP_CSS_FONT_VARIANT_NUMERIC_ORDINAL ) feature_string += "ordn, "; if ( font_variant_numeric.computed & SP_CSS_FONT_VARIANT_NUMERIC_SLASHED_ZERO ) feature_string += "zero, "; if( font_variant_east_asian.computed & SP_CSS_FONT_VARIANT_EAST_ASIAN_JIS78 ) feature_string += "jp78, "; if( font_variant_east_asian.computed & SP_CSS_FONT_VARIANT_EAST_ASIAN_JIS83 ) feature_string += "jp83, "; if( font_variant_east_asian.computed & SP_CSS_FONT_VARIANT_EAST_ASIAN_JIS90 ) feature_string += "jp90, "; if( font_variant_east_asian.computed & SP_CSS_FONT_VARIANT_EAST_ASIAN_JIS04 ) feature_string += "jp04, "; if( font_variant_east_asian.computed & SP_CSS_FONT_VARIANT_EAST_ASIAN_SIMPLIFIED ) feature_string += "smpl, "; if( font_variant_east_asian.computed & SP_CSS_FONT_VARIANT_EAST_ASIAN_TRADITIONAL ) feature_string += "trad, "; if( font_variant_east_asian.computed & SP_CSS_FONT_VARIANT_EAST_ASIAN_FULL_WIDTH ) feature_string += "fwid, "; if( font_variant_east_asian.computed & SP_CSS_FONT_VARIANT_EAST_ASIAN_PROPORTIONAL_WIDTH ) feature_string += "pwid, "; if( font_variant_east_asian.computed & SP_CSS_FONT_VARIANT_EAST_ASIAN_RUBY ) feature_string += "ruby, "; char const *val = font_feature_settings.value(); if (val[0] && strcmp(val, "normal")) { // We do no sanity checking... feature_string += val; feature_string += ", "; } if (feature_string.empty()) { feature_string = "normal"; } else { // Remove last ", " feature_string.resize(feature_string.size() - 2); } return feature_string; } // Internal /** * Release callback. */ static void sp_style_object_release(SPObject *object, SPStyle *style) { (void)object; // TODO style->object = nullptr; } // Internal /** * Emit style modified signal on style's object if the filter changed. */ static void sp_style_filter_ref_modified(SPObject *obj, guint flags, SPStyle *style) { (void)flags; // TODO SPFilter *filter=static_cast(obj); if (style->getFilter() == filter) { if (style->object) { style->object->requestModified(SP_OBJECT_MODIFIED_FLAG | SP_OBJECT_STYLE_MODIFIED_FLAG); } } } // Internal /** * Gets called when the filter is (re)attached to the style */ void sp_style_filter_ref_changed(SPObject *old_ref, SPObject *ref, SPStyle *style) { if (old_ref) { (dynamic_cast( old_ref ))->_refcount--; style->filter_modified_connection.disconnect(); } if ( SP_IS_FILTER(ref)) { (dynamic_cast( ref ))->_refcount++; style->filter_modified_connection = ref->connectModified(sigc::bind(sigc::ptr_fun(&sp_style_filter_ref_modified), style)); } sp_style_filter_ref_modified(ref, 0, style); } /** * Emit style modified signal on style's object if server is style's fill * or stroke paint server. */ static void sp_style_paint_server_ref_modified(SPObject *obj, guint flags, SPStyle *style) { (void)flags; // TODO SPPaintServer *server = static_cast(obj); if ((style->fill.isPaintserver()) && style->getFillPaintServer() == server) { if (style->object) { /** \todo * fixme: I do not know, whether it is optimal - we are * forcing reread of everything (Lauris) */ /** \todo * fixme: We have to use object_modified flag, because parent * flag is only available downstreams. */ style->object->requestModified(SP_OBJECT_MODIFIED_FLAG | SP_OBJECT_STYLE_MODIFIED_FLAG); } } else if ((style->stroke.isPaintserver()) && style->getStrokePaintServer() == server) { if (style->object) { /// \todo fixme: style->object->requestModified(SP_OBJECT_MODIFIED_FLAG | SP_OBJECT_STYLE_MODIFIED_FLAG); } } else if (server) { g_assert_not_reached(); } } /** * Gets called when the paintserver is (re)attached to the style */ void sp_style_fill_paint_server_ref_changed(SPObject *old_ref, SPObject *ref, SPStyle *style) { if (old_ref) { style->fill_ps_modified_connection.disconnect(); } if (SP_IS_PAINT_SERVER(ref)) { style->fill_ps_modified_connection = ref->connectModified(sigc::bind(sigc::ptr_fun(&sp_style_paint_server_ref_modified), style)); } style->signal_fill_ps_changed.emit(old_ref, ref); sp_style_paint_server_ref_modified(ref, 0, style); } /** * Gets called when the paintserver is (re)attached to the style */ void sp_style_stroke_paint_server_ref_changed(SPObject *old_ref, SPObject *ref, SPStyle *style) { if (old_ref) { style->stroke_ps_modified_connection.disconnect(); } if (SP_IS_PAINT_SERVER(ref)) { style->stroke_ps_modified_connection = ref->connectModified(sigc::bind(sigc::ptr_fun(&sp_style_paint_server_ref_modified), style)); } style->signal_stroke_ps_changed.emit(old_ref, ref); sp_style_paint_server_ref_modified(ref, 0, style); } // Called in display/drawing-item.cpp, display/nr-filter-primitive.cpp, libnrtype/Layout-TNG-Input.cpp /** * Increase refcount of style. */ SPStyle * sp_style_ref(SPStyle *style) { g_return_val_if_fail(style != nullptr, NULL); style->style_ref(); // Increase ref count return style; } // Called in display/drawing-item.cpp, display/nr-filter-primitive.cpp, libnrtype/Layout-TNG-Input.cpp /** * Decrease refcount of style with possible destruction. */ SPStyle * sp_style_unref(SPStyle *style) { g_return_val_if_fail(style != nullptr, NULL); if (style->style_unref() < 1) { delete style; return nullptr; } return style; } static CRSelEng * sp_repr_sel_eng() { CRSelEng *const ret = cr_sel_eng_new(&Inkscape::XML::croco_node_iface); /** \todo * Check whether we need to register any pseudo-class handlers. * libcroco has its own default handlers for first-child and lang. * * We probably want handlers for link and arguably visited (though * inkscape can't visit links at the time of writing). hover etc. * more useful in inkview than the editor inkscape. * * http://www.w3.org/TR/SVG11/styling.html#StylingWithCSS says that * the following should be honoured, at least by inkview: * :hover, :active, :focus, :visited, :link. */ g_assert(ret); return ret; } // The following functions should be incorporated into SPIPaint. FIXME // Called in: style.cpp, style-internal.cpp void sp_style_set_ipaint_to_uri(SPStyle *style, SPIPaint *paint, const Inkscape::URI *uri, SPDocument *document) { if (!paint->value.href) { if (style->object) { // Should not happen as href should have been created in SPIPaint. (TODO: Removed code duplication.) paint->value.href = new SPPaintServerReference(style->object); } else if (document) { // Used by desktop style (no object to attach to!). paint->value.href = new SPPaintServerReference(document); } else { std::cerr << "sp_style_set_ipaint_to_uri: No valid object or document!" << std::endl; return; } if (paint == &style->fill) { style->fill_ps_changed_connection = paint->value.href->changedSignal().connect(sigc::bind(sigc::ptr_fun(sp_style_fill_paint_server_ref_changed), style)); } else { style->stroke_ps_changed_connection = paint->value.href->changedSignal().connect(sigc::bind(sigc::ptr_fun(sp_style_stroke_paint_server_ref_changed), style)); } } if (paint->value.href){ if (paint->value.href->getObject()){ paint->value.href->detach(); } try { paint->value.href->attach(*uri); } catch (Inkscape::BadURIException &e) { g_warning("%s", e.what()); paint->value.href->detach(); } } } // Called in: style.cpp, style-internal.cpp void sp_style_set_ipaint_to_uri_string (SPStyle *style, SPIPaint *paint, const gchar *uri) { try { const Inkscape::URI IURI(uri); sp_style_set_ipaint_to_uri(style, paint, &IURI, style->document); } catch (...) { g_warning("URI failed to parse: %s", uri); } } // Called in: desktop-style.cpp void sp_style_set_to_uri(SPStyle *style, bool isfill, Inkscape::URI const *uri) { sp_style_set_ipaint_to_uri(style, style->getFillOrStroke(isfill), uri, style->document); } // Called in: widgets/font-selector.cpp, widgets/text-toolbar.cpp, ui/dialog/text-edit.cpp gchar const * sp_style_get_css_unit_string(int unit) { // specify px by default, see inkscape bug 1221626, mozilla bug 234789 // This is a problematic fix as some properties (e.g. 'line-height') have // different behaviour if there is no unit. switch (unit) { case SP_CSS_UNIT_NONE: return "px"; case SP_CSS_UNIT_PX: return "px"; case SP_CSS_UNIT_PT: return "pt"; case SP_CSS_UNIT_PC: return "pc"; case SP_CSS_UNIT_MM: return "mm"; case SP_CSS_UNIT_CM: return "cm"; case SP_CSS_UNIT_IN: return "in"; case SP_CSS_UNIT_EM: return "em"; case SP_CSS_UNIT_EX: return "ex"; case SP_CSS_UNIT_PERCENT: return "%"; default: return "px"; } return "px"; } // Called in: style-internal.cpp, widgets/text-toolbar.cpp, ui/dialog/text-edit.cpp /* * Convert a size in pixels into another CSS unit size */ double sp_style_css_size_px_to_units(double size, int unit, double font_size) { double unit_size = size; if (font_size == 0) { g_warning("sp_style_get_css_font_size_units: passed in zero font_size"); font_size = SP_CSS_FONT_SIZE_DEFAULT; } switch (unit) { case SP_CSS_UNIT_NONE: unit_size = size; break; case SP_CSS_UNIT_PX: unit_size = size; break; case SP_CSS_UNIT_PT: unit_size = Inkscape::Util::Quantity::convert(size, "px", "pt"); break; case SP_CSS_UNIT_PC: unit_size = Inkscape::Util::Quantity::convert(size, "px", "pc"); break; case SP_CSS_UNIT_MM: unit_size = Inkscape::Util::Quantity::convert(size, "px", "mm"); break; case SP_CSS_UNIT_CM: unit_size = Inkscape::Util::Quantity::convert(size, "px", "cm"); break; case SP_CSS_UNIT_IN: unit_size = Inkscape::Util::Quantity::convert(size, "px", "in"); break; case SP_CSS_UNIT_EM: unit_size = size / font_size; break; case SP_CSS_UNIT_EX: unit_size = size * 2.0 / font_size ; break; case SP_CSS_UNIT_PERCENT: unit_size = size * 100.0 / font_size; break; default: g_warning("sp_style_get_css_font_size_units conversion to %d not implemented.", unit); break; } return unit_size; } // Called in: widgets/text-toolbar.cpp, ui/dialog/text-edit.cpp /* * Convert a size in a CSS unit size to pixels */ double sp_style_css_size_units_to_px(double size, int unit, double font_size) { if (unit == SP_CSS_UNIT_PX) { return size; } //g_message("sp_style_css_size_units_to_px %f %d = %f px", size, unit, out); return size * (size / sp_style_css_size_px_to_units(size, unit, font_size));; } // FIXME: Everything below this line belongs in a different file - css-chemistry? void sp_style_set_property_url (SPObject *item, gchar const *property, SPObject *linked, bool recursive) { Inkscape::XML::Node *repr = item->getRepr(); if (repr == nullptr) return; SPCSSAttr *css = sp_repr_css_attr_new(); if (linked) { gchar *val = g_strdup_printf("url(#%s)", linked->getId()); sp_repr_css_set_property(css, property, val); g_free(val); } else { sp_repr_css_unset_property(css, "filter"); } if (recursive) { sp_repr_css_change_recursive(repr, css, "style"); } else { sp_repr_css_change(repr, css, "style"); } sp_repr_css_attr_unref(css); } /** * \pre style != NULL. * \pre flags in {IFSET, ALWAYS}. * Only used by sp_css_attr_from_object() and in splivarot.cpp - sp_item_path_outline(). */ SPCSSAttr * sp_css_attr_from_style(SPStyle const *const style, guint const flags) { g_return_val_if_fail(style != nullptr, NULL); g_return_val_if_fail(((flags & SP_STYLE_FLAG_IFSET) || (flags & SP_STYLE_FLAG_ALWAYS)), NULL); Glib::ustring style_str = style->write(flags); SPCSSAttr *css = sp_repr_css_attr_new(); sp_repr_css_attr_add_from_string(css, style_str.c_str()); return css; } // Called in: selection-chemistry.cpp, widgets/stroke-marker-selector.cpp, widgets/stroke-style.cpp, // ui/tools/freehand-base.cpp /** * \pre object != NULL * \pre flags in {IFSET, ALWAYS}. */ SPCSSAttr *sp_css_attr_from_object(SPObject *object, guint const flags) { g_return_val_if_fail(((flags == SP_STYLE_FLAG_IFSET) || (flags == SP_STYLE_FLAG_ALWAYS) ), NULL); SPCSSAttr * result = nullptr; if (object->style) { result = sp_css_attr_from_style(object->style, flags); } return result; } // Called in: selection-chemistry.cpp, ui/dialog/inkscape-preferences.cpp /** * Unset any text-related properties */ SPCSSAttr * sp_css_attr_unset_text(SPCSSAttr *css) { sp_repr_css_set_property(css, "font", nullptr); sp_repr_css_set_property(css, "-inkscape-font-specification", nullptr); sp_repr_css_set_property(css, "font-size", nullptr); sp_repr_css_set_property(css, "font-size-adjust", nullptr); // not implemented yet sp_repr_css_set_property(css, "font-style", nullptr); sp_repr_css_set_property(css, "font-variant", nullptr); sp_repr_css_set_property(css, "font-weight", nullptr); sp_repr_css_set_property(css, "font-stretch", nullptr); sp_repr_css_set_property(css, "font-family", nullptr); sp_repr_css_set_property(css, "text-indent", nullptr); sp_repr_css_set_property(css, "text-align", nullptr); sp_repr_css_set_property(css, "line-height", nullptr); sp_repr_css_set_property(css, "letter-spacing", nullptr); sp_repr_css_set_property(css, "word-spacing", nullptr); sp_repr_css_set_property(css, "text-transform", nullptr); sp_repr_css_set_property(css, "direction", nullptr); sp_repr_css_set_property(css, "writing-mode", nullptr); sp_repr_css_set_property(css, "text-orientation", nullptr); sp_repr_css_set_property(css, "text-anchor", nullptr); sp_repr_css_set_property(css, "white-space", nullptr); sp_repr_css_set_property(css, "shape-inside", nullptr); sp_repr_css_set_property(css, "shape-subtract", nullptr); sp_repr_css_set_property(css, "shape-padding", nullptr); sp_repr_css_set_property(css, "shape-margin", nullptr); sp_repr_css_set_property(css, "inline-size", nullptr); sp_repr_css_set_property(css, "kerning", nullptr); // not implemented yet sp_repr_css_set_property(css, "dominant-baseline", nullptr); // not implemented yet sp_repr_css_set_property(css, "alignment-baseline", nullptr); // not implemented yet sp_repr_css_set_property(css, "baseline-shift", nullptr); sp_repr_css_set_property(css, "text-decoration", nullptr); sp_repr_css_set_property(css, "text-decoration-line", nullptr); sp_repr_css_set_property(css, "text-decoration-color", nullptr); sp_repr_css_set_property(css, "text-decoration-style", nullptr); sp_repr_css_set_property(css, "font-variant-ligatures", nullptr); sp_repr_css_set_property(css, "font-variant-position", nullptr); sp_repr_css_set_property(css, "font-variant-caps", nullptr); sp_repr_css_set_property(css, "font-variant-numeric", nullptr); sp_repr_css_set_property(css, "font-variant-alternates", nullptr); sp_repr_css_set_property(css, "font-variant-east-asian", nullptr); sp_repr_css_set_property(css, "font-feature-settings", nullptr); return css; } // ui/dialog/inkscape-preferences.cpp /** * Unset properties that should not be set for default tool style. * This list needs to be reviewed. */ SPCSSAttr * sp_css_attr_unset_blacklist(SPCSSAttr *css) { sp_repr_css_set_property(css, "color", nullptr); sp_repr_css_set_property(css, "clip-rule", nullptr); sp_repr_css_set_property(css, "d", nullptr); sp_repr_css_set_property(css, "display", nullptr); sp_repr_css_set_property(css, "overflow", nullptr); sp_repr_css_set_property(css, "visibility", nullptr); sp_repr_css_set_property(css, "isolation", nullptr); sp_repr_css_set_property(css, "mix-blend-mode", nullptr); sp_repr_css_set_property(css, "color-interpolation", nullptr); sp_repr_css_set_property(css, "color-interpolation-filters", nullptr); sp_repr_css_set_property(css, "solid-color", nullptr); sp_repr_css_set_property(css, "solid-opacity", nullptr); sp_repr_css_set_property(css, "fill-rule", nullptr); sp_repr_css_set_property(css, "color-rendering", nullptr); sp_repr_css_set_property(css, "image-rendering", nullptr); sp_repr_css_set_property(css, "shape-rendering", nullptr); sp_repr_css_set_property(css, "text-rendering", nullptr); sp_repr_css_set_property(css, "enable-background", nullptr); return css; } // Called in style.cpp static bool is_url(char const *p) { if (p == nullptr) return false; /** \todo * FIXME: I'm not sure if this applies to SVG as well, but CSS2 says any URIs * in property values must start with 'url('. */ return (g_ascii_strncasecmp(p, "url(", 4) == 0); } // Called in: ui/dialog/inkscape-preferences.cpp, ui/tools/tweek-tool.cpp /** * Unset any properties that contain URI values. * * Used for storing style that will be reused across documents when carrying * the referenced defs is impractical. */ SPCSSAttr * sp_css_attr_unset_uris(SPCSSAttr *css) { // All properties that may hold or according to SVG 1.1 if (is_url(sp_repr_css_property(css, "clip-path", nullptr))) sp_repr_css_set_property(css, "clip-path", nullptr); if (is_url(sp_repr_css_property(css, "color-profile", nullptr))) sp_repr_css_set_property(css, "color-profile", nullptr); if (is_url(sp_repr_css_property(css, "cursor", nullptr))) sp_repr_css_set_property(css, "cursor", nullptr); if (is_url(sp_repr_css_property(css, "filter", nullptr))) sp_repr_css_set_property(css, "filter", nullptr); if (is_url(sp_repr_css_property(css, "marker", nullptr))) sp_repr_css_set_property(css, "marker", nullptr); if (is_url(sp_repr_css_property(css, "marker-start", nullptr))) sp_repr_css_set_property(css, "marker-start", nullptr); if (is_url(sp_repr_css_property(css, "marker-mid", nullptr))) sp_repr_css_set_property(css, "marker-mid", nullptr); if (is_url(sp_repr_css_property(css, "marker-end", nullptr))) sp_repr_css_set_property(css, "marker-end", nullptr); if (is_url(sp_repr_css_property(css, "mask", nullptr))) sp_repr_css_set_property(css, "mask", nullptr); if (is_url(sp_repr_css_property(css, "fill", nullptr))) sp_repr_css_set_property(css, "fill", nullptr); if (is_url(sp_repr_css_property(css, "stroke", nullptr))) sp_repr_css_set_property(css, "stroke", nullptr); return css; } // Called in style.cpp /** * Scale a single-value property. */ static void sp_css_attr_scale_property_single(SPCSSAttr *css, gchar const *property, double ex, bool only_with_units = false) { gchar const *w = sp_repr_css_property(css, property, nullptr); if (w) { gchar *units = nullptr; double wd = g_ascii_strtod(w, &units) * ex; if (w == units) {// nothing converted, non-numeric value return; } if (only_with_units && (units == nullptr || *units == '\0' || *units == '%' || *units == 'e')) { // only_with_units, but no units found, so do nothing. // 'e' matches 'em' or 'ex' return; } Inkscape::CSSOStringStream os; os << wd << units; // reattach units sp_repr_css_set_property(css, property, os.str().c_str()); } } // Called in style.cpp for stroke-dasharray /** * Scale a list-of-values property. */ static void sp_css_attr_scale_property_list(SPCSSAttr *css, gchar const *property, double ex) { gchar const *string = sp_repr_css_property(css, property, nullptr); if (string) { Inkscape::CSSOStringStream os; gchar **a = g_strsplit(string, ",", 10000); bool first = true; for (gchar **i = a; i != nullptr; i++) { gchar *w = *i; if (w == nullptr) break; gchar *units = nullptr; double wd = g_ascii_strtod(w, &units) * ex; if (w == units) {// nothing converted, non-numeric value ("none" or "inherit"); do nothing g_strfreev(a); return; } if (!first) { os << ","; } os << wd << units; // reattach units first = false; } sp_repr_css_set_property(css, property, os.str().c_str()); g_strfreev(a); } } // Called in: text-editing.cpp, /** * Scale any properties that may hold by ex. */ SPCSSAttr * sp_css_attr_scale(SPCSSAttr *css, double ex) { sp_css_attr_scale_property_single(css, "baseline-shift", ex); sp_css_attr_scale_property_single(css, "stroke-width", ex); sp_css_attr_scale_property_list (css, "stroke-dasharray", ex); sp_css_attr_scale_property_single(css, "stroke-dashoffset", ex); sp_css_attr_scale_property_single(css, "font-size", ex, true); sp_css_attr_scale_property_single(css, "kerning", ex); sp_css_attr_scale_property_single(css, "letter-spacing", ex); sp_css_attr_scale_property_single(css, "word-spacing", ex); sp_css_attr_scale_property_single(css, "line-height", ex, true); return css; } /** * Quote and/or escape string for writing to CSS, changing strings in place. * See: http://www.w3.org/TR/CSS21/syndata.html#value-def-identifier */ void css_quote(Glib::ustring &val) { Glib::ustring out; bool quote = false; // Can't wait for C++11! for( Glib::ustring::iterator it = val.begin(); it != val.end(); ++it) { if(g_ascii_isalnum(*it) || *it=='-' || *it=='_' || *it > 0xA0) { out += *it; } else if (*it == '\'') { // Single quotes require escaping and quotes. out += '\\'; out += *it; quote = true; } else { // Quote everything else including spaces. // (CSS Fonts Level 3 recommends quoting with spaces.) out += *it; quote = true; } if( it == val.begin() && !g_ascii_isalpha(*it) ) { // A non-ASCII/non-alpha initial value on any identifier needs quotes. // (Actually it's a bit more complicated but as it never hurts to quote...) quote = true; } } if( quote ) { out.insert( out.begin(), '\'' ); out += '\''; } val = out; } /** * Quote font names in font-family lists, changing string in place. * We use unquoted names internally but some need to be quoted in CSS. */ void css_font_family_quote(Glib::ustring &val) { std::vector tokens = Glib::Regex::split_simple("\\s*,\\s*", val ); val.erase(); for(auto & token : tokens) { css_quote( token ); val += token + ", "; } if( val.size() > 1 ) val.erase( val.size() - 2 ); // Remove trailing ", " } // Called in style-internal.cpp, xml/repr-css.cpp /** * Remove paired single and double quotes from a string, changing string in place. */ void css_unquote(Glib::ustring &val) { if( val.size() > 1 && ( (val[0] == '"' && val[val.size()-1] == '"' ) || (val[0] == '\'' && val[val.size()-1] == '\'' ) ) ) { val.erase( 0, 1 ); val.erase( val.size()-1 ); } } // Called in style-internal.cpp, text-toolbar.cpp /** * Remove paired single and double quotes from font names in font-family lists, * changing string in place. * We use unquoted family names internally but CSS sometimes uses quoted names. */ void css_font_family_unquote(Glib::ustring &val) { std::vector tokens = Glib::Regex::split_simple("\\s*,\\s*", val ); val.erase(); for(auto & token : tokens) { css_unquote( token ); val += token + ", "; } if( val.size() > 1 ) val.erase( val.size() - 2 ); // Remove trailing ", " } /* 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 :