// SPDX-License-Identifier: GPL-2.0-or-later /* * SVG element implementation * * Section 20.8.3 of the W3C SVG 1.1 spec * available at: * http://www.w3.org/TR/SVG/fonts.html#FontFaceElement * * Author: * Felipe C. da S. Sanches * Abhishek Sharma * * Copyright (C) 2008, Felipe C. da S. Sanches * * Released under GNU GPL v2+, read the file 'COPYING' for more information. */ #include "xml/repr.h" #include "attributes.h" #include "sp-font-face.h" #include "document.h" #include static std::vector sp_read_fontFaceStyleType(gchar const *value){ std::vector v; if (!value){ v.push_back(SP_FONTFACE_STYLE_ALL); return v; } if (strncmp(value, "all", 3) == 0){ value += 3; while(value[0]==',' || value[0]==' ') value++; v.push_back(SP_FONTFACE_STYLE_ALL); return v; } while(value[0]!='\0'){ switch(value[0]){ case 'n': if (strncmp(value, "normal", 6) == 0){ v.push_back(SP_FONTFACE_STYLE_NORMAL); value += 6; } break; case 'i': if (strncmp(value, "italic", 6) == 0){ v.push_back(SP_FONTFACE_STYLE_ITALIC); value += 6; } break; case 'o': if (strncmp(value, "oblique", 7) == 0){ v.push_back(SP_FONTFACE_STYLE_OBLIQUE); value += 7; } break; } while(value[0]==',' || value[0]==' ') value++; } return v; } static std::vector sp_read_fontFaceVariantType(gchar const *value){ std::vector v; if (!value){ v.push_back(SP_FONTFACE_VARIANT_NORMAL); return v; } while(value[0]!='\0'){ switch(value[0]){ case 'n': if (strncmp(value, "normal", 6) == 0){ v.push_back(SP_FONTFACE_VARIANT_NORMAL); value += 6; } break; case 's': if (strncmp(value, "small-caps", 10) == 0){ v.push_back(SP_FONTFACE_VARIANT_SMALL_CAPS); value += 10; } break; } while(value[0]==',' || value[0]==' ') value++; } return v; } static std::vector sp_read_fontFaceWeightType(gchar const *value){ std::vector v; if (!value){ v.push_back(SP_FONTFACE_WEIGHT_ALL); return v; } if (strncmp(value, "all", 3) == 0){ value += 3; while(value[0]==',' || value[0]==' ') value++; v.push_back(SP_FONTFACE_WEIGHT_ALL); return v; } while(value[0]!='\0'){ switch(value[0]){ case 'n': if (strncmp(value, "normal", 6) == 0){ v.push_back(SP_FONTFACE_WEIGHT_NORMAL); value += 6; } break; case 'b': if (strncmp(value, "bold", 4) == 0){ v.push_back(SP_FONTFACE_WEIGHT_BOLD); value += 4; } break; case '1': if (strncmp(value, "100", 3) == 0){ v.push_back(SP_FONTFACE_WEIGHT_100); value += 3; } break; case '2': if (strncmp(value, "200", 3) == 0){ v.push_back(SP_FONTFACE_WEIGHT_200); value += 3; } break; case '3': if (strncmp(value, "300", 3) == 0){ v.push_back(SP_FONTFACE_WEIGHT_300); value += 3; } break; case '4': if (strncmp(value, "400", 3) == 0){ v.push_back(SP_FONTFACE_WEIGHT_400); value += 3; } break; case '5': if (strncmp(value, "500", 3) == 0){ v.push_back(SP_FONTFACE_WEIGHT_500); value += 3; } break; case '6': if (strncmp(value, "600", 3) == 0){ v.push_back(SP_FONTFACE_WEIGHT_600); value += 3; } break; case '7': if (strncmp(value, "700", 3) == 0){ v.push_back(SP_FONTFACE_WEIGHT_700); value += 3; } break; case '8': if (strncmp(value, "800", 3) == 0){ v.push_back(SP_FONTFACE_WEIGHT_800); value += 3; } break; case '9': if (strncmp(value, "900", 3) == 0){ v.push_back(SP_FONTFACE_WEIGHT_900); value += 3; } break; } while(value[0]==',' || value[0]==' ') value++; } return v; } static std::vector sp_read_fontFaceStretchType(gchar const *value){ std::vector v; if (!value){ v.push_back(SP_FONTFACE_STRETCH_NORMAL); return v; } if (strncmp(value, "all", 3) == 0){ value += 3; while(value[0]==',' || value[0]==' ') value++; v.push_back(SP_FONTFACE_STRETCH_ALL); return v; } while(value[0]!='\0'){ switch(value[0]){ case 'n': if (strncmp(value, "normal", 6) == 0){ v.push_back(SP_FONTFACE_STRETCH_NORMAL); value += 6; } break; case 'u': if (strncmp(value, "ultra-condensed", 15) == 0){ v.push_back(SP_FONTFACE_STRETCH_ULTRA_CONDENSED); value += 15; } if (strncmp(value, "ultra-expanded", 14) == 0){ v.push_back(SP_FONTFACE_STRETCH_ULTRA_EXPANDED); value += 14; } break; case 'e': if (strncmp(value, "expanded", 8) == 0){ v.push_back(SP_FONTFACE_STRETCH_EXPANDED); value += 8; } if (strncmp(value, "extra-condensed", 15) == 0){ v.push_back(SP_FONTFACE_STRETCH_EXTRA_CONDENSED); value += 15; } if (strncmp(value, "extra-expanded", 14) == 0){ v.push_back(SP_FONTFACE_STRETCH_EXTRA_EXPANDED); value += 14; } break; case 'c': if (strncmp(value, "condensed", 9) == 0){ v.push_back(SP_FONTFACE_STRETCH_CONDENSED); value += 9; } break; case 's': if (strncmp(value, "semi-condensed", 14) == 0){ v.push_back(SP_FONTFACE_STRETCH_SEMI_CONDENSED); value += 14; } if (strncmp(value, "semi-expanded", 13) == 0){ v.push_back(SP_FONTFACE_STRETCH_SEMI_EXPANDED); value += 13; } break; } while(value[0]==',' || value[0]==' ') value++; } return v; } SPFontFace::SPFontFace() : SPObject() { std::vector style; style.push_back(SP_FONTFACE_STYLE_ALL); this->font_style = style; std::vector variant; variant.push_back(SP_FONTFACE_VARIANT_NORMAL); this->font_variant = variant; std::vector weight; weight.push_back(SP_FONTFACE_WEIGHT_ALL); this->font_weight = weight; std::vector stretch; stretch.push_back(SP_FONTFACE_STRETCH_NORMAL); this->font_stretch = stretch; this->font_family = nullptr; //this->font_style = ; //this->font_variant = ; //this->font_weight = ; //this->font_stretch = ; this->font_size = nullptr; //this->unicode_range = ; this->units_per_em = 1000; //this->panose_1 = ; this->stemv = 0; this->stemh = 0; this->slope = 0; this->cap_height = 0; this->x_height = 0; this->accent_height = 0; this->ascent = 0; this->descent = 0; this->widths = nullptr; this->bbox = nullptr; this->ideographic = 0; this->alphabetic = 0; this->mathematical = 0; this->hanging = 0; this->v_ideographic = 0; this->v_alphabetic = 0; this->v_mathematical = 0; this->v_hanging = 0; this->underline_position = 0; this->underline_thickness = 0; this->strikethrough_position = 0; this->strikethrough_thickness = 0; this->overline_position = 0; this->overline_thickness = 0; } SPFontFace::~SPFontFace() = default; void SPFontFace::build(SPDocument *document, Inkscape::XML::Node *repr) { SPObject::build(document, repr); this->readAttr( "font-family" ); this->readAttr( "font-style" ); this->readAttr( "font-variant" ); this->readAttr( "font-weight" ); this->readAttr( "font-stretch" ); this->readAttr( "font-size" ); this->readAttr( "unicode-range" ); this->readAttr( "units-per-em" ); this->readAttr( "panose-1" ); this->readAttr( "stem-v" ); this->readAttr( "stem-h" ); this->readAttr( "slope" ); this->readAttr( "cap-height" ); this->readAttr( "x-height" ); this->readAttr( "accent-height" ); this->readAttr( "ascent" ); this->readAttr( "descent" ); this->readAttr( "widths" ); this->readAttr( "bbox" ); this->readAttr( "ideographic" ); this->readAttr( "alphabetic" ); this->readAttr( "mathematical" ); this->readAttr( "ranging" ); this->readAttr( "v-ideogaphic" ); this->readAttr( "v-alphabetic" ); this->readAttr( "v-mathematical" ); this->readAttr( "v-hanging" ); this->readAttr( "underline-position" ); this->readAttr( "underline-thickness" ); this->readAttr( "strikethrough-position" ); this->readAttr( "strikethrough-thickness" ); this->readAttr( "overline-position" ); this->readAttr( "overline-thickness" ); } /** * Callback for child_added event. */ void SPFontFace::child_added(Inkscape::XML::Node *child, Inkscape::XML::Node *ref) { SPObject::child_added(child, ref); this->parent->requestModified(SP_OBJECT_MODIFIED_FLAG); } /** * Callback for remove_child event. */ void SPFontFace::remove_child(Inkscape::XML::Node *child) { SPObject::remove_child(child); this->parent->requestModified(SP_OBJECT_MODIFIED_FLAG); } void SPFontFace::release() { SPObject::release(); } void SPFontFace::set(SPAttributeEnum key, const gchar *value) { std::vector style; std::vector variant; std::vector weight; std::vector stretch; switch (key) { case SP_PROP_FONT_FAMILY: if (this->font_family) { g_free(this->font_family); } this->font_family = g_strdup(value); this->requestModified(SP_OBJECT_MODIFIED_FLAG); break; case SP_PROP_FONT_STYLE: style = sp_read_fontFaceStyleType(value); if (this->font_style.size() != style.size()){ this->font_style = style; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } else { for (unsigned int i=0;ifont_style[i]){ this->font_style = style; this->requestModified(SP_OBJECT_MODIFIED_FLAG); break; } } } break; case SP_PROP_FONT_VARIANT: variant = sp_read_fontFaceVariantType(value); if (this->font_variant.size() != variant.size()){ this->font_variant = variant; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } else { for (unsigned int i=0;ifont_variant[i]){ this->font_variant = variant; this->requestModified(SP_OBJECT_MODIFIED_FLAG); break; } } } break; case SP_PROP_FONT_WEIGHT: weight = sp_read_fontFaceWeightType(value); if (this->font_weight.size() != weight.size()){ this->font_weight = weight; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } else { for (unsigned int i=0;ifont_weight[i]){ this->font_weight = weight; this->requestModified(SP_OBJECT_MODIFIED_FLAG); break; } } } break; case SP_PROP_FONT_STRETCH: stretch = sp_read_fontFaceStretchType(value); if (this->font_stretch.size() != stretch.size()){ this->font_stretch = stretch; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } else { for (unsigned int i=0;ifont_stretch[i]){ this->font_stretch = stretch; this->requestModified(SP_OBJECT_MODIFIED_FLAG); break; } } } break; case SP_ATTR_UNITS_PER_EM: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->units_per_em){ this->units_per_em = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_STEMV: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->stemv){ this->stemv = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_STEMH: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->stemh){ this->stemh = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_SLOPE: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->slope){ this->slope = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_CAP_HEIGHT: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->cap_height){ this->cap_height = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_X_HEIGHT: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->x_height){ this->x_height = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_ACCENT_HEIGHT: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->accent_height){ this->accent_height = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_ASCENT: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->ascent){ this->ascent = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_DESCENT: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->descent){ this->descent = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_IDEOGRAPHIC: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->ideographic){ this->ideographic = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_ALPHABETIC: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->alphabetic){ this->alphabetic = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_MATHEMATICAL: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->mathematical){ this->mathematical = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_HANGING: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->hanging){ this->hanging = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_V_IDEOGRAPHIC: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->v_ideographic){ this->v_ideographic = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_V_ALPHABETIC: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->v_alphabetic){ this->v_alphabetic = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_V_MATHEMATICAL: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->v_mathematical){ this->v_mathematical = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_V_HANGING: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->v_hanging){ this->v_hanging = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_UNDERLINE_POSITION: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->underline_position){ this->underline_position = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_UNDERLINE_THICKNESS: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->underline_thickness){ this->underline_thickness = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_STRIKETHROUGH_POSITION: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->strikethrough_position){ this->strikethrough_position = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_STRIKETHROUGH_THICKNESS: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->strikethrough_thickness){ this->strikethrough_thickness = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_OVERLINE_POSITION: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->overline_position){ this->overline_position = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } case SP_ATTR_OVERLINE_THICKNESS: { double number = value ? g_ascii_strtod(value, nullptr) : 0; if (number != this->overline_thickness){ this->overline_thickness = number; this->requestModified(SP_OBJECT_MODIFIED_FLAG); } break; } default: SPObject::set(key, value); break; } } /** * Receives update notifications. */ void SPFontFace::update(SPCtx *ctx, guint flags) { if (flags & (SP_OBJECT_MODIFIED_FLAG)) { this->readAttr( "font-family" ); this->readAttr( "font-style" ); this->readAttr( "font-variant" ); this->readAttr( "font-weight" ); this->readAttr( "font-stretch" ); this->readAttr( "font-size" ); this->readAttr( "unicode-range" ); this->readAttr( "units-per-em" ); this->readAttr( "panose-1" ); this->readAttr( "stemv" ); this->readAttr( "stemh" ); this->readAttr( "slope" ); this->readAttr( "cap-height" ); this->readAttr( "x-height" ); this->readAttr( "accent-height" ); this->readAttr( "ascent" ); this->readAttr( "descent" ); this->readAttr( "widths" ); this->readAttr( "bbox" ); this->readAttr( "ideographic" ); this->readAttr( "alphabetic" ); this->readAttr( "mathematical" ); this->readAttr( "hanging" ); this->readAttr( "v-ideographic" ); this->readAttr( "v-alphabetic" ); this->readAttr( "v-mathematical" ); this->readAttr( "v-hanging" ); this->readAttr( "underline-position" ); this->readAttr( "underline-thickness" ); this->readAttr( "strikethrough-position" ); this->readAttr( "strikethrough-thickness" ); this->readAttr( "overline-position" ); this->readAttr( "overline-thickness" ); } SPObject::update(ctx, flags); } #define COPY_ATTR(rd,rs,key) (rd)->setAttribute((key), rs->attribute(key)); Inkscape::XML::Node* SPFontFace::write(Inkscape::XML::Document *xml_doc, Inkscape::XML::Node *repr, guint flags) { if ((flags & SP_OBJECT_WRITE_BUILD) && !repr) { repr = xml_doc->createElement("svg:font-face"); } //TODO: //sp_repr_set_svg_double(repr, "font-family", face->font_family); //sp_repr_set_svg_double(repr, "font-style", face->font_style); //sp_repr_set_svg_double(repr, "font-variant", face->font_variant); //sp_repr_set_svg_double(repr, "font-weight", face->font_weight); //sp_repr_set_svg_double(repr, "font-stretch", face->font_stretch); //sp_repr_set_svg_double(repr, "font-size", face->font_size); //sp_repr_set_svg_double(repr, "unicode-range", face->unicode_range); sp_repr_set_svg_double(repr, "units-per-em", this->units_per_em); //sp_repr_set_svg_double(repr, "panose-1", face->panose_1); sp_repr_set_svg_double(repr, "stemv", this->stemv); sp_repr_set_svg_double(repr, "stemh", this->stemh); sp_repr_set_svg_double(repr, "slope", this->slope); sp_repr_set_svg_double(repr, "cap-height", this->cap_height); sp_repr_set_svg_double(repr, "x-height", this->x_height); sp_repr_set_svg_double(repr, "accent-height", this->accent_height); sp_repr_set_svg_double(repr, "ascent", this->ascent); sp_repr_set_svg_double(repr, "descent", this->descent); //sp_repr_set_svg_double(repr, "widths", face->widths); //sp_repr_set_svg_double(repr, "bbox", face->bbox); sp_repr_set_svg_double(repr, "ideographic", this->ideographic); sp_repr_set_svg_double(repr, "alphabetic", this->alphabetic); sp_repr_set_svg_double(repr, "mathematical", this->mathematical); sp_repr_set_svg_double(repr, "hanging", this->hanging); sp_repr_set_svg_double(repr, "v-ideographic", this->v_ideographic); sp_repr_set_svg_double(repr, "v-alphabetic", this->v_alphabetic); sp_repr_set_svg_double(repr, "v-mathematical", this->v_mathematical); sp_repr_set_svg_double(repr, "v-hanging", this->v_hanging); sp_repr_set_svg_double(repr, "underline-position", this->underline_position); sp_repr_set_svg_double(repr, "underline-thickness", this->underline_thickness); sp_repr_set_svg_double(repr, "strikethrough-position", this->strikethrough_position); sp_repr_set_svg_double(repr, "strikethrough-thickness", this->strikethrough_thickness); sp_repr_set_svg_double(repr, "overline-position", this->overline_position); sp_repr_set_svg_double(repr, "overline-thickness", this->overline_thickness); if (repr != this->getRepr()) { // In all COPY_ATTR given below the XML tree is // being used directly while it shouldn't be. COPY_ATTR(repr, this->getRepr(), "font-family"); COPY_ATTR(repr, this->getRepr(), "font-style"); COPY_ATTR(repr, this->getRepr(), "font-variant"); COPY_ATTR(repr, this->getRepr(), "font-weight"); COPY_ATTR(repr, this->getRepr(), "font-stretch"); COPY_ATTR(repr, this->getRepr(), "font-size"); COPY_ATTR(repr, this->getRepr(), "unicode-range"); COPY_ATTR(repr, this->getRepr(), "units-per-em"); COPY_ATTR(repr, this->getRepr(), "panose-1"); COPY_ATTR(repr, this->getRepr(), "stemv"); COPY_ATTR(repr, this->getRepr(), "stemh"); COPY_ATTR(repr, this->getRepr(), "slope"); COPY_ATTR(repr, this->getRepr(), "cap-height"); COPY_ATTR(repr, this->getRepr(), "x-height"); COPY_ATTR(repr, this->getRepr(), "accent-height"); COPY_ATTR(repr, this->getRepr(), "ascent"); COPY_ATTR(repr, this->getRepr(), "descent"); COPY_ATTR(repr, this->getRepr(), "widths"); COPY_ATTR(repr, this->getRepr(), "bbox"); COPY_ATTR(repr, this->getRepr(), "ideographic"); COPY_ATTR(repr, this->getRepr(), "alphabetic"); COPY_ATTR(repr, this->getRepr(), "mathematical"); COPY_ATTR(repr, this->getRepr(), "hanging"); COPY_ATTR(repr, this->getRepr(), "v-ideographic"); COPY_ATTR(repr, this->getRepr(), "v-alphabetic"); COPY_ATTR(repr, this->getRepr(), "v-mathematical"); COPY_ATTR(repr, this->getRepr(), "v-hanging"); COPY_ATTR(repr, this->getRepr(), "underline-position"); COPY_ATTR(repr, this->getRepr(), "underline-thickness"); COPY_ATTR(repr, this->getRepr(), "strikethrough-position"); COPY_ATTR(repr, this->getRepr(), "strikethrough-thickness"); COPY_ATTR(repr, this->getRepr(), "overline-position"); COPY_ATTR(repr, this->getRepr(), "overline-thickness"); } SPObject::write(xml_doc, repr, flags); return repr; } /* 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 :