UNPKG

parcel-bundler

Version:

<p align="center"> <a href="https://parceljs.org/" target="_blank"> <img alt="Parcel" src="https://user-images.githubusercontent.com/19409/31321658-f6aed0f2-ac3d-11e7-8100-1587e676e0ec.png" width="749"> </a> </p>

99 lines (83 loc) 3.42 kB
const CSSAsset = require('./CSSAsset'); const config = require('../utils/config'); const localRequire = require('../utils/localRequire'); const Resolver = require('../Resolver'); const URL_RE = /^(?:url\s*\(\s*)?['"]?(?:[#/]|(?:https?:)?\/\/)/i; class StylusAsset extends CSSAsset { async parse(code) { // stylus should be installed locally in the module that's being required let stylus = await localRequire('stylus', this.name); let opts = this.package.stylus || (await config.load(this.name, ['.stylusrc', '.stylusrc.js'])); let style = stylus(code, opts); style.set('filename', this.name); style.set('include css', true); style.set('Evaluator', await createEvaluator(this)); // Setup a handler for the URL function so we add dependencies for linked assets. style.define('url', node => { let filename = this.addURLDependency(node.val, node.filename); return new stylus.nodes.Literal(`url(${JSON.stringify(filename)})`); }); return style; } collectDependencies() { // Do nothing. Dependencies are collected by our custom evaluator. } generateErrorMessage(err) { let index = err.message.indexOf('\n'); err.codeFrame = err.message.slice(index + 1); err.message = err.message.slice(0, index); return err; } } async function createEvaluator(asset) { const Evaluator = await localRequire( 'stylus/lib/visitor/evaluator', asset.name ); const utils = await localRequire('stylus/lib/utils', asset.name); const resolver = new Resolver(asset.options); // This is a custom stylus evaluator that extends stylus with support for the node // require resolution algorithm. It also adds all dependencies to the parcel asset // tree so the file watcher works correctly, etc. class CustomEvaluator extends Evaluator { visitImport(imported) { let node = this.visit(imported.path).first; let path = node.string; if (node.name !== 'url' && path && !URL_RE.test(path)) { try { // First try resolving using the node require resolution algorithm. // This allows stylus files in node_modules to be resolved properly. // If we find something, update the AST so stylus gets the absolute path to load later. node.string = resolver.resolveSync(path, imported.filename).path; asset.addDependency(node.string, {includedInParent: true}); } catch (err) { // If we couldn't resolve, try the normal stylus resolver. // We just need to do this to keep track of the dependencies - stylus does the real work. // support optional .styl if (!/\.styl$/i.test(path)) { path += '.styl'; } let found = utils.find(path, this.paths, this.filename); if (!found) { found = utils.lookupIndex(node.string, this.paths, this.filename); } if (!found) { let nodeName = imported.once ? 'require' : 'import'; throw new Error( 'failed to locate @' + nodeName + ' file ' + node.string ); } for (let file of found) { asset.addDependency(file, {includedInParent: true}); } } } // Done. Let stylus do its thing. return super.visitImport(imported); } } return CustomEvaluator; } module.exports = StylusAsset;