UNPKG

react-saasify-chrisvxd

Version:

React components for Saasify web clients.

273 lines (236 loc) 6.79 kB
const Asset = require('../Asset'); const postcss = require('postcss'); const valueParser = require('postcss-value-parser'); const postcssTransform = require('../transforms/postcss'); const CssSyntaxError = require('postcss/lib/css-syntax-error'); const SourceMap = require('../SourceMap'); const loadSourceMap = require('../utils/loadSourceMap'); const path = require('path'); const urlJoin = require('../utils/urlJoin'); const isURL = require('../utils/is-url'); const URL_RE = /url\s*\("?(?![a-z]+:)/; const IMPORT_RE = /@import/; const COMPOSES_RE = /composes:.+from\s*("|').*("|')\s*;?/; const FROM_IMPORT_RE = /.+from\s*(?:"|')(.*)(?:"|')\s*;?/; const PROTOCOL_RE = /^[a-z]+:/; class CSSAsset extends Asset { constructor(name, options) { super(name, options); this.type = 'css'; this.previousSourceMap = this.options.rendition ? this.options.rendition.map : null; } mightHaveDependencies() { return ( !/\.css$/.test(this.name) || IMPORT_RE.test(this.contents) || COMPOSES_RE.test(this.contents) || URL_RE.test(this.contents) ); } parse(code) { let root = postcss.parse(code, { from: this.name }); return new CSSAst(code, root); } collectDependencies() { this.ast.root.walkAtRules('import', rule => { let params = valueParser(rule.params); let [name, ...media] = params.nodes; let dep; if ( name.type === 'function' && name.value === 'url' && name.nodes.length ) { name = name.nodes[0]; } dep = name.value; if (!dep) { throw new Error('Could not find import name for ' + rule); } if (PROTOCOL_RE.test(dep)) { return; } // If this came from an inline <style> tag, don't inline the imported file. Replace with the correct URL instead. // TODO: run CSSPackager on inline style tags. let inlineHTML = this.options.rendition && this.options.rendition.inlineHTML; if (inlineHTML) { name.value = this.addURLDependency(dep, {loc: rule.source.start}); rule.params = params.toString(); } else { media = valueParser.stringify(media).trim(); this.addDependency(dep, {media, loc: rule.source.start}); rule.remove(); } this.ast.dirty = true; }); this.ast.root.walkDecls(decl => { if (URL_RE.test(decl.value)) { let parsed = valueParser(decl.value); let dirty = false; parsed.walk(node => { if ( node.type === 'function' && node.value === 'url' && node.nodes.length ) { let url = this.addURLDependency(node.nodes[0].value, { loc: decl.source.start }); if (!isURL(url)) { url = urlJoin(this.options.publicURL, url); } dirty = node.nodes[0].value !== url; node.nodes[0].value = url; } }); if (dirty) { decl.value = parsed.toString(); this.ast.dirty = true; } } if (decl.prop === 'composes' && FROM_IMPORT_RE.test(decl.value)) { let parsed = valueParser(decl.value); parsed.walk(node => { if (node.type === 'string') { const [, importPath] = FROM_IMPORT_RE.exec(decl.value); this.addURLDependency(importPath, { dynamic: false, loc: decl.source.start }); } }); } }); } async pretransform() { if (this.options.sourceMaps && !this.previousSourceMap) { this.previousSourceMap = await loadSourceMap(this); } } async transform() { await postcssTransform(this); } getCSSAst() { // Converts the ast to a CSS ast if needed, so we can apply postcss transforms. if (!(this.ast instanceof CSSAst)) { this.ast = CSSAsset.prototype.parse.call( this, this.ast.render(this.name) ); } return this.ast.root; } async generate() { let css; if (this.ast) { let result = this.ast.render(this.name); css = result.css; if (result.map) this.sourceMap = result.map; } else { css = this.contents; } let js = ''; if (this.options.hmr) { this.addDependency('_css_loader'); js = ` var reloadCSS = require('_css_loader'); module.hot.dispose(reloadCSS); module.hot.accept(reloadCSS); `; } if (this.cssModules) { js += 'module.exports = ' + JSON.stringify(this.cssModules, null, 2) + ';'; } if (this.options.sourceMaps) { if (this.sourceMap) { this.sourceMap = await new SourceMap().addMap(this.sourceMap); } if (this.previousSourceMap) { this.previousSourceMap.sources = this.previousSourceMap.sources.map(v => path.join( path.dirname(this.relativeName), this.previousSourceMap.sourceRoot || '', v ) ); if (this.sourceMap) { this.sourceMap = await new SourceMap().extendSourceMap( this.previousSourceMap, this.sourceMap ); } else { this.sourceMap = await new SourceMap().addMap(this.previousSourceMap); } } else if (!this.sourceMap) { this.sourceMap = new SourceMap().generateEmptyMap( this.relativeName, css ); } } return [ { type: 'css', value: css, cssModules: this.cssModules, map: this.sourceMap }, { type: 'js', value: js, hasDependencies: false } ]; } generateErrorMessage(err) { // Wrap the error in a CssSyntaxError if needed so we can generate a code frame if (err.loc && !err.showSourceCode) { err = new CssSyntaxError( err.message, err.loc.line, err.loc.column, this.contents ); } err.message = err.reason || err.message; err.loc = { line: err.line, column: err.column }; if (err.showSourceCode) { err.codeFrame = err.showSourceCode(); err.highlightedCodeFrame = err.showSourceCode(true); } return err; } } class CSSAst { constructor(css, root) { this.css = css; this.root = root; this.dirty = false; } render(name) { if (this.dirty) { let {css, map} = this.root.toResult({ to: name, map: {inline: false, annotation: false, sourcesContent: true} }); this.css = css; return { css: this.css, map: map ? map.toJSON() : null }; } return { css: this.css }; } } module.exports = CSSAsset;