@spectrum-css/component-builder
Version:
The Spectrum CSS component builder
183 lines (153 loc) • 4.84 kB
JavaScript
/*
Copyright 2019 Adobe. All rights reserved.
This file is licensed to you under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License. You may obtain a copy
of the License at http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software distributed under
the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
OF ANY KIND, either express or implied. See the License for the specific language
governing permissions and limitations under the License.
*/
const gulp = require('gulp');
const postcss = require('postcss');
const concat = require('gulp-concat');
const through = require('through2');
const fsp = require('fs').promises;
const logger = require('gulplog');
const path = require('path');
function getVarsFromCSS(css) {
let variableList = [];
let root = postcss.parse(css);
root.walkRules((rule, ruleIndex) => {
rule.walkDecls((decl) => {
let matches = decl.value.match(/var\(.*?\)/g);
if (matches) {
matches.forEach(function(match) {
let varName = match.replace(/var\((--[\w\-]+),?.*?\)/, '$1').trim();
if (variableList.indexOf(varName) === -1) {
variableList.push(varName);
}
});
}
});
});
return variableList;
}
function getVarValues(css) {
let root = postcss.parse(css);
let variables = {};
root.walkRules((rule, ruleIndex) => {
rule.walkDecls((decl) => {
variables[decl.prop] = decl.value;
});
});
return variables;
}
function getClassNames(contents, pkgName) {
let root = postcss.parse(contents);
let classNames = [];
function addClassname(className) {
if (classNames.indexOf(className) === -1) {
classNames.push(className);
}
}
let result = root.walkRules((rule, ruleIndex) => {
let selector = rule.selectors[0];
if (pkgName === 'page') {
className = '';
return 'false';
}
rule.selectors.forEach((fullSelector) => {
// Skip compound selectors, they may not start with the component itself
if (fullSelector.match(/~|\+/)) {
return true;
}
let selector = fullSelector.split(' ').shift();
if (rule.type === 'rule') {
let matches = selector.match(/^\.spectrum-[\w]+/);
if (matches) {
let modSelector = matches[0]
addClassname(modSelector);
}
}
});
});
if (classNames.length === 0) {
logger.error(`Could not find classNames for ${pkgName}, assuming no classNames`);
classNames.push('');
}
logger.debug(`Found classNames ${classNames.join(', ')} for ${pkgName}`);
return classNames;
}
function resolveValue(value, vars) {
if (value) {
let match = value.match(/var\((.+),?.*?\)/);
if (match) {
return match[1];
}
return value;
}
}
const varDir = path.join(path.dirname(require.resolve('@spectrum-css/vars')), '..');
async function readDNAVariables(file) {
let css = await fsp.readFile(path.join(varDir, 'css', file));
let vars = getVarValues(css);
return vars;
}
function getVariableDeclarations(classNames, vars) {
let varNames = Object.keys(vars);
if (varNames.length) {
return `
${classNames.map((className) => `${className}`).join(',\n')} {
${varNames.map((varName) => ` ${varName}: ${vars[varName]};`).join('\n')}
}
`;
}
return '';
}
function getAllVars() {
return new Promise((resolve, reject) => {
let variableList;
gulp.src([
`${varDir}/css/themes/*.css`,
`${varDir}/css/scales/*.css`,
`${varDir}/css/components/*.css`,
`${varDir}/css/globals/*.css`
])
.pipe(concat('everything.css'))
.pipe(through.obj(function getAllVars(file, enc, cb) {
variableList = getVarValues(file.contents.toString());
cb(null, file);
}))
.on('finish', () => {
resolve(variableList);
})
.on('error', reject);
});
}
function getAllComponentVars() {
return new Promise((resolve, reject) => {
let variableList;
gulp.src([
`${varDir}/css/components/*.css`,
`${varDir}/css/globals/*.css`
])
.pipe(concat('everything.css'))
.pipe(through.obj(function getAllVars(file, enc, cb) {
variableList = getVarValues(file.contents.toString());
cb(null, file);
}))
.on('finish', () => {
resolve(variableList);
})
.on('error', reject);
});
}
exports.getAllComponentVars = getAllComponentVars;
exports.getAllVars = getAllVars;
exports.getVarsFromCSS = getVarsFromCSS;
exports.getVarValues = getVarValues;
exports.getClassNames = getClassNames;
exports.resolveValue = resolveValue;
exports.readDNAVariables = readDNAVariables;
exports.getVariableDeclarations = getVariableDeclarations;