// SPDX-License-Identifier: GPL-2.0-or-later /* * SVG conditional attribute evaluation * * Authors: * Andrius R. * Abhishek Sharma * * Copyright (C) 2006 authors * * Released under GNU GPL v2+, read the file 'COPYING' for more information. */ #include #include #include #include "conditions.h" #include "rdf.h" #include "object/sp-item.h" #include "xml/repr.h" typedef bool (* condition_evaluator)(SPItem const *item, gchar const *value); struct Condition { gchar const *attribute; condition_evaluator evaluator; }; static bool evaluateSystemLanguage(SPItem const *item, gchar const *value); static bool evaluateRequiredFeatures(SPItem const *item, gchar const *value); static bool evaluateRequiredExtensions(SPItem const *item, gchar const *value); /* define any conditional attributes and their handler functions in this array */ static Condition _condition_handlers[] = { { "systemLanguage", evaluateSystemLanguage }, { "requiredFeatures", evaluateRequiredFeatures }, { "requiredExtensions", evaluateRequiredExtensions }, }; // function which evaluates if item should be displayed bool sp_item_evaluate(SPItem const *item) { bool needDisplay = true; for ( unsigned int i = 0 ; needDisplay && (i < sizeof(_condition_handlers) / sizeof(_condition_handlers[0])) ; i++ ) { gchar const *value = item->getAttribute(_condition_handlers[i].attribute); if ( value && !_condition_handlers[i].evaluator(item, value) ) { needDisplay = false; } } return needDisplay; } #define ISALNUM(c) (((c) >= 'a' && (c) <= 'z') || ((c) >= 'A' && (c) <= 'Z') || ((c) >= '0' && (c) <= '9')) // TODO: review and modernize this function static gchar *preprocessLanguageCode(const gchar *language_code) { if ( nullptr == language_code || strcmp(language_code, "") == 0) return nullptr; gchar *lngcode = g_strdup(language_code); for ( unsigned int i = 0 ; i < strlen(lngcode) ; i++ ) { if ( lngcode[i] >= 'A' && lngcode[i] <= 'Z' ) { lngcode[i] = g_ascii_tolower(lngcode[i]); } else if ( '_' == lngcode[i] ) { lngcode[i] = '-'; } else if ( !ISALNUM(lngcode[i]) && '-' != lngcode[i] ) { // only alpha numeric characters and '-' may be contained in language code g_free(lngcode); return nullptr; } } return lngcode; } static bool evaluateSystemLanguage(SPItem const *item, gchar const *value) { if ( nullptr == value ) return true; std::set language_codes; gchar *str = nullptr; gchar **strlist = g_strsplit( value, ",", 0); for ( int i = 0 ; (str = strlist[i]) ; i++ ) { str = g_strstrip(str); gchar *lngcode = preprocessLanguageCode(str); if (lngcode == nullptr) continue; language_codes.insert(lngcode); gchar *pos = strchr (lngcode, '-'); if (pos) { // if subtag is used, primary tag is still a perfect match *pos = 0; if (strlen(lngcode) && language_codes.find(lngcode) == language_codes.end()) { language_codes.insert(lngcode); } } } g_strfreev(strlist); if (language_codes.empty()) return false; SPDocument *document = item->document; const auto document_languages = document->getLanguages(); if (document_languages.size() == 0) return false; for (const auto &language : document_languages) { gchar *lngcode = preprocessLanguageCode(language.c_str()); if (lngcode && (language_codes.find(lngcode) != language_codes.end())) { g_free(lngcode); return true; } g_free(lngcode); } return false; } static std::vector splitByWhitespace(gchar const *value) { std::vector parts; gchar *str = nullptr; gchar **strlist = g_strsplit( value, ",", 0); for ( int i = 0 ; (str = strlist[i]) ; i++ ) { gchar *part = g_strstrip(str); if ( 0 == *part ) continue; parts.emplace_back(part); } g_strfreev(strlist); return parts; } #define SVG11FEATURE "http://www.w3.org/TR/SVG11/feature#" #define SVG10FEATURE "org.w3c." static bool evaluateSVG11Feature(gchar const *feature) { static gchar const *_supported_features[] = { "SVG", // incomplete "SVG-static" - missing support for "Filter" /* SVG - user agent supports at least one of the following: "SVG-static", "SVG-animation", "SVG-dynamic" or "SVGDOM" */ // "SVGDOM", // not sure /* SVGDOM - user agent supports at least one of the following: "SVGDOM-static", "SVGDOM-animation" or "SVGDOM-dynamic" */ "SVG-static", // incomplete - missing support for "Filter" /* SVG-static - user agent supports the following features: "CoreAttribute", "Structure", "ContainerAttribute", "ConditionalProcessing", "Image", "Style", "ViewportAttribute", "Shape", "Text", "PaintAttribute", "OpacityAttribute", "GraphicsAttribute", "Marker", "ColorProfile", "Gradient", "Pattern", "Clip", "Mask", "Filter", "XlinkAttribute", "Font", "Extensibility" */ // "SVGDOM-static", // not sure /* SVGDOM-static - All of the DOM interfaces and methods that correspond to SVG-static */ // "SVG-animation", // no support /* SVG-animation - All of the language features from "SVG-static" plus the feature "feature#Animation" */ // "SVGDOM-animation", // no support /* SVGDOM-animation - All of the DOM interfaces and methods that correspond to SVG-animation */ // "SVG-dynamic", // no support /* SVG-dynamic - user agent supports all "SVG-animation" and the following features: "Hyperlinking", "Scripting", "View", "Cursor", "GraphicalEventsAttribute", "DocumentEventsAttribute", "AnimationEventsAttribute" */ // "SVGDOM-dynamic", // no support /* SVGDOM-dynamic - All of the DOM interfaces and methods that correspond to SVG-dynamic */ "CoreAttribute", "Structure", "BasicStructure", "ContainerAttribute", "ConditionalProcessing", "Image", "Style", "ViewportAttribute", // not sure "Shape", "Text", "BasicText", "PaintAttribute", "BasicPaintAttribute", "OpacityAttribute", "GraphicsAttribute", "BasicGraphicsAttribute", "Marker", "ColorProfile", "Gradient", "Pattern", "Clip", "BasicClip", "Mask", // "Filter", // "BasicFilter", // "DocumentEventsAttribute", // "GraphicalEventsAttribute", // "AnimationEventsAttribute", // "Cursor", // not sure "Hyperlinking", // not sure "XlinkAttribute", // not sure "ExternalResourcesRequired", // not sure "View", // "Script", // "Animation", "Font", "BasicFont", "Extensibility", // not sure }; for (auto & _supported_feature : _supported_features) { if ( 0 == strcasecmp(feature, _supported_feature) ) return true; } return false; } static bool evaluateSVG10Feature(gchar const *feature) { static gchar const *_supported_features[] = { "svg.static", // incomplete - no filter effects "dom.svg.static", // not sure // "svg.animation", // "dom.svg.animation", // "svg.dynamic", // "dom.svg.dynamic" // "svg.all", // "dom.svg.all" }; for (auto & _supported_feature : _supported_features) { if ( 0 == strcasecmp(feature, _supported_feature) ) return true; } return false; } static bool evaluateSingleFeature(gchar const *value) { if ( nullptr == value ) return false; gchar const *found; found = strstr(value, SVG11FEATURE); if ( value == found ) return evaluateSVG11Feature(found + strlen(SVG11FEATURE)); found = strstr(value, SVG10FEATURE); if ( value == found ) return evaluateSVG10Feature(found + strlen(SVG10FEATURE)); return false; } static bool evaluateRequiredFeatures(SPItem const */*item*/, gchar const *value) { if ( nullptr == value ) return true; std::vector parts = splitByWhitespace(value); if (parts.empty()) { return false; } for (auto & part : parts) { if (!evaluateSingleFeature(part.c_str())) { return false; } } return true; } static bool evaluateRequiredExtensions(SPItem const */*item*/, gchar const *value) { if ( nullptr == value ) return true; return false; } /* * Language codes and names: aa Afar ab Abkhazian af Afrikaans am Amharic ar Arabic as Assamese ay Aymara az Azerbaijani ba Bashkir be Byelorussian bg Bulgarian bh Bihari bi Bislama bn Bengali; Bangla bo Tibetan br Breton ca Catalan co Corsican cs Czech cy Welsh da Danish de German dz Bhutani el Greek en English eo Esperanto es Spanish et Estonian eu Basque fa Persian fi Finnish fj Fiji fo Faroese fr French fy Frisian ga Irish gd Scots Gaelic gl Galician gn Guarani gu Gujarati ha Hausa he Hebrew (formerly iw) hi Hindi hr Croatian hu Hungarian hy Armenian ia Interlingua id Indonesian (formerly in) ie Interlingue ik Inupiak is Icelandic it Italian iu Inuktitut ja Japanese jw Javanese ka Georgian kk Kazakh kl Greenlandic km Cambodian kn Kannada ko Korean ks Kashmiri ku Kurdish ky Kirghiz la Latin ln Lingala lo Laothian lt Lithuanian lv Latvian, Lettish mg Malagasy mi Maori mk Macedonian ml Malayalam mn Mongolian mo Moldavian mr Marathi ms Malay mt Maltese my Burmese na Nauru ne Nepali nl Dutch no Norwegian oc Occitan om (Afan) Oromo or Oriya pa Punjabi pl Polish ps Pashto, Pushto pt Portuguese qu Quechua rm Rhaeto-Romance rn Kirundi ro Romanian ru Russian rw Kinyarwanda sa Sanskrit sd Sindhi sg Sangho sh Serbo-Croatian si Sinhalese sk Slovak sl Slovenian sm Samoan sn Shona so Somali sq Albanian sr Serbian ss Siswati st Sesotho su Sundanese sv Swedish sw Swahili ta Tamil te Telugu tg Tajik th Thai ti Tigrinya tk Turkmen tl Tagalog tn Setswana to Tonga tr Turkish ts Tsonga tt Tatar tw Twi ug Uighur uk Ukrainian ur Urdu uz Uzbek vi Vietnamese vo Volapuk wo Wolof xh Xhosa yi Yiddish (formerly ji) yo Yoruba za Zhuang zh Chinese zu Zulu */ /* 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 :