| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125 | /* jshint quotmark: false */'use strict';exports.type = 'perItem';exports.active = true;exports.description = 'converts style to attributes';exports.params = {    keepImportant: false};var stylingProps = require('./_collections').attrsGroups.presentation,    rEscape = '\\\\(?:[0-9a-f]{1,6}\\s?|\\r\\n|.)',                 // Like \" or \2051. Code points consume one space.    rAttr = '\\s*(' + g('[^:;\\\\]', rEscape) + '*?)\\s*',          // attribute name like ‘fill’    rSingleQuotes = "'(?:[^'\\n\\r\\\\]|" + rEscape + ")*?(?:'|$)", // string in single quotes: 'smth'    rQuotes = '"(?:[^"\\n\\r\\\\]|' + rEscape + ')*?(?:"|$)',       // string in double quotes: "smth"    rQuotedString = new RegExp('^' + g(rSingleQuotes, rQuotes) + '$'),    // Parentheses, E.g.: url(...).    // ':' and ';' inside of it should be threated as is. (Just like in strings.)    rParenthesis = '\\(' + g('[^\'"()\\\\]+', rEscape, rSingleQuotes, rQuotes) + '*?' + '\\)',    // The value. It can have strings and parentheses (see above). Fallbacks to anything in case of unexpected input.    rValue = '\\s*(' + g('[^!\'"();\\\\]+?', rEscape, rSingleQuotes, rQuotes, rParenthesis, '[^;]*?') + '*?' + ')',    // End of declaration. Spaces outside of capturing groups help to do natural trimming.    rDeclEnd = '\\s*(?:;\\s*|$)',    // Important rule    rImportant = '(\\s*!important(?![-(\w]))?',    // Final RegExp to parse CSS declarations.    regDeclarationBlock = new RegExp(rAttr + ':' + rValue + rImportant + rDeclEnd, 'ig'),    // Comments expression. Honors escape sequences and strings.    regStripComments = new RegExp(g(rEscape, rSingleQuotes, rQuotes, '/\\*[^]*?\\*/'), 'ig');/** * Convert style in attributes. Cleanups comments and illegal declarations (without colon) as a side effect. * * @example * <g style="fill:#000; color: #fff;"> *             ⬇ * <g fill="#000" color="#fff"> * * @example * <g style="fill:#000; color: #fff; -webkit-blah: blah"> *             ⬇ * <g fill="#000" color="#fff" style="-webkit-blah: blah"> * * @param {Object} item current iteration item * @return {Boolean} if false, item will be filtered out * * @author Kir Belevich */exports.fn = function(item, params) {    /* jshint boss: true */    if (item.elem && item.hasAttr('style')) {            // ['opacity: 1', 'color: #000']        var styleValue = item.attr('style').value,            styles = [],            attrs = {};        // Strip CSS comments preserving escape sequences and strings.        styleValue = styleValue.replace(regStripComments, function(match) {            return match[0] == '/' ? '' :                match[0] == '\\' && /[-g-z]/i.test(match[1]) ? match[1] : match;        });        regDeclarationBlock.lastIndex = 0;        for (var rule; rule = regDeclarationBlock.exec(styleValue);) {            if (!params.keepImportant || !rule[3]) {                styles.push([rule[1], rule[2]]);            }        }        if (styles.length) {            styles = styles.filter(function(style) {                if (style[0]) {                    var prop = style[0].toLowerCase(),                        val = style[1];                    if (rQuotedString.test(val)) {                        val = val.slice(1, -1);                    }                    if (stylingProps.indexOf(prop) > -1) {                        attrs[prop] = {                            name: prop,                            value: val,                            local: prop,                            prefix: ''                        };                        return false;                    }                }                return true;            });            Object.assign(item.attrs, attrs);            if (styles.length) {                item.attr('style').value = styles                    .map(function(declaration) { return declaration.join(':') })                    .join(';');            } else {                item.removeAttr('style');            }        }    }};function g() {    return '(?:' + Array.prototype.join.call(arguments, '|') + ')';}
 |