diff --git a/test/webpack/node_modules/@folio/app1/icons/app.png b/test/webpack/node_modules/@folio/app1/icons/app.png deleted file mode 100644 index e69de29..0000000 diff --git a/test/webpack/node_modules/@folio/app1/icons/app.svg b/test/webpack/node_modules/@folio/app1/icons/app.svg deleted file mode 100644 index e69de29..0000000 diff --git a/test/webpack/node_modules/@folio/app1/package.json b/test/webpack/node_modules/@folio/app1/package.json deleted file mode 100644 index 3f0601f..0000000 --- a/test/webpack/node_modules/@folio/app1/package.json +++ /dev/null @@ -1,29 +0,0 @@ -{ - "name": "@folio/app1", - "version": "1.2.3", - "main": "src/index.js", - "stripes": { - "actsAs": [ - "app", - "settings" - ], - "displayName": "ui-app1.meta.title", - "route": "/app1", - "actionNames": [ - "stripesHome", - "app1SortByName" - ], - "icons": [ - { - "name": "app", - "alt": "Create, view and manage app1", - "title": "Application 1" - } - ], - "okapiInterfaces": { - "users": "15.0", - "configuration": "2.0" - }, - "stripesDeps": ["@folio/stripes-dep1", "@notfolio/stripes-dep2"] - } -} diff --git a/test/webpack/node_modules/@folio/app2/icons/app.svg b/test/webpack/node_modules/@folio/app2/icons/app.svg deleted file mode 100644 index e69de29..0000000 diff --git a/test/webpack/node_modules/@folio/app2/node_modules/@folio/stripes-dep1/icons/thing.png b/test/webpack/node_modules/@folio/app2/node_modules/@folio/stripes-dep1/icons/thing.png deleted file mode 100644 index e69de29..0000000 diff --git a/test/webpack/node_modules/@folio/app2/node_modules/@folio/stripes-dep1/icons/thing.svg b/test/webpack/node_modules/@folio/app2/node_modules/@folio/stripes-dep1/icons/thing.svg deleted file mode 100644 index e69de29..0000000 diff --git a/test/webpack/node_modules/@folio/app2/node_modules/@folio/stripes-dep1/package.json b/test/webpack/node_modules/@folio/app2/node_modules/@folio/stripes-dep1/package.json deleted file mode 100644 index 8b3b7d1..0000000 --- a/test/webpack/node_modules/@folio/app2/node_modules/@folio/stripes-dep1/package.json +++ /dev/null @@ -1,14 +0,0 @@ -{ - "name": "@folio/stripes-dep1", - "version": "1.2.3", - "main": "src/index.js", - "stripes": { - "icons": [ - { - "name": "thing", - "alt": "Do thing", - "title": "Thing" - } - ] - } -} diff --git a/test/webpack/node_modules/@folio/app2/package.json b/test/webpack/node_modules/@folio/app2/package.json deleted file mode 100644 index 6f00a55..0000000 --- a/test/webpack/node_modules/@folio/app2/package.json +++ /dev/null @@ -1,25 +0,0 @@ -{ - "name": "@folio/app2", - "version": "1.2.3", - "main": "src/index.js", - "stripes": { - "actsAs": [ - "app", - "settings" - ], - "displayName": "ui-app2.meta.title", - "route": "/app2", - "icons": [ - { - "name": "app", - "alt": "Create, view and manage app2", - "title": "Application 2" - } - ], - "okapiInterfaces": { - "users": "15.0", - "configuration": "2.0" - }, - "stripesDeps": ["@folio/stripes-dep1"] - } -} diff --git a/test/webpack/node_modules/@folio/stripes-dep1/icons/otherthing.png b/test/webpack/node_modules/@folio/stripes-dep1/icons/otherthing.png deleted file mode 100644 index e69de29..0000000 diff --git a/test/webpack/node_modules/@folio/stripes-dep1/icons/otherthing.svg b/test/webpack/node_modules/@folio/stripes-dep1/icons/otherthing.svg deleted file mode 100644 index e69de29..0000000 diff --git a/test/webpack/node_modules/@folio/stripes-dep1/icons/thing.png b/test/webpack/node_modules/@folio/stripes-dep1/icons/thing.png deleted file mode 100644 index e69de29..0000000 diff --git a/test/webpack/node_modules/@folio/stripes-dep1/icons/thing.svg b/test/webpack/node_modules/@folio/stripes-dep1/icons/thing.svg deleted file mode 100644 index e69de29..0000000 diff --git a/test/webpack/node_modules/@folio/stripes-dep1/package.json b/test/webpack/node_modules/@folio/stripes-dep1/package.json deleted file mode 100644 index 0bc5dda..0000000 --- a/test/webpack/node_modules/@folio/stripes-dep1/package.json +++ /dev/null @@ -1,19 +0,0 @@ -{ - "name": "@folio/stripes-dep1", - "version": "3.4.5", - "main": "src/index.js", - "stripes": { - "icons": [ - { - "name": "thing", - "alt": "Do thing", - "title": "Thingy" - }, - { - "name": "otherthing", - "alt": "Do other thing", - "title": "Other Thing" - } - ] - } -} diff --git a/test/webpack/node_modules/@notfolio/stripes-dep2/package.json b/test/webpack/node_modules/@notfolio/stripes-dep2/package.json deleted file mode 100644 index 8db0fd5..0000000 --- a/test/webpack/node_modules/@notfolio/stripes-dep2/package.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "name": "@notfolio/stripes-dep2", - "version": "1.2.3", - "main": "src/index.js" -} diff --git a/webpack.config.base.js b/webpack.config.base.js index c889ff4..3f03465 100644 --- a/webpack.config.base.js +++ b/webpack.config.base.js @@ -15,7 +15,6 @@ const specificReact = generateStripesAlias('react'); module.exports = { entry: [ - '@folio/stripes-components/lib/global.css', '@folio/stripes-core/src/index', //path.join(__dirname, 'src', 'index'), ], diff --git a/webpack.config.cli.dev.js b/webpack.config.cli.dev.js index d279009..9c2d556 100644 --- a/webpack.config.cli.dev.js +++ b/webpack.config.cli.dev.js @@ -3,31 +3,12 @@ const path = require('path'); const webpack = require('webpack'); -const postCssImport = require('postcss-import'); -const autoprefixer = require('autoprefixer'); -const postCssCustomProperties = require('postcss-custom-properties'); -const postCssCalc = require('postcss-calc'); -const postCssNesting = require('postcss-nesting'); -const postCssCustomMedia = require('postcss-custom-media'); -const postCssMediaMinMax = require('postcss-media-minmax'); -const postCssColorFunction = require('postcss-color-function'); -const { generateStripesAlias, tryResolve, getSharedStyles } = require('./webpack/module-paths'); + +const { tryResolve } = require('./webpack/module-paths'); const base = require('./webpack.config.base'); const cli = require('./webpack.config.cli'); - -const locateCssVariables = () => { - const variables = 'lib/variables.css'; - const localPath = path.join(path.resolve(), variables); - - // check if variables are present locally (in cases when stripes-components is - // being built directly) if not look for them via stripes aliases - return tryResolve(localPath) ? - localPath : - path.join(generateStripesAlias('@folio/stripes-components'), variables); -}; - const useBrowserMocha = () => { return tryResolve('mocha/mocha-es2018.js') ? 'mocha/mocha-es2018.js' : 'mocha'; }; @@ -62,45 +43,6 @@ devConfig.plugins = devConfig.plugins.concat([ devConfig.resolve.alias['react-dom'] = '@hot-loader/react-dom'; devConfig.resolve.alias.process = 'process/browser.js'; devConfig.resolve.alias['mocha'] = useBrowserMocha(); -devConfig.module.rules.push({ - test: /\.css$/, - use: [ - { - loader: 'style-loader' - }, - { - loader: 'css-loader', - options: { - modules: { - localIdentName: '[local]---[hash:base64:5]', - }, - sourceMap: true, - importLoaders: 1, - }, - }, - { - loader: 'postcss-loader', - options: { - postcssOptions: { - plugins: [ - postCssImport(), - autoprefixer(), - postCssCustomProperties({ - preserve: false, - importFrom: [locateCssVariables()] - }), - postCssCalc(), - postCssNesting(), - postCssCustomMedia(), - postCssMediaMinMax(), - postCssColorFunction(), - ], - }, - sourceMap: true, - }, - }, - ], -}); // add 'Buffer' global required for tests/reporting tools. devConfig.plugins.push( diff --git a/webpack.config.cli.prod.js b/webpack.config.cli.prod.js index 2b9a804..62df02b 100644 --- a/webpack.config.cli.prod.js +++ b/webpack.config.cli.prod.js @@ -3,7 +3,7 @@ const path = require('path'); const webpack = require('webpack'); -const MiniCssExtractPlugin = require('mini-css-extract-plugin'); + const CssMinimizerPlugin = require("css-minimizer-webpack-plugin") const postCssImport = require('postcss-import'); @@ -28,7 +28,6 @@ const prodConfig = Object.assign({}, base, cli, { }); prodConfig.plugins = prodConfig.plugins.concat([ - new MiniCssExtractPlugin({ filename: 'style.[contenthash].css' }), new webpack.ProvidePlugin({ process: 'process/browser.js', }), @@ -49,43 +48,43 @@ prodConfig.optimization = { minimize: true, } -prodConfig.module.rules.push({ - test: /\.css$/, - use: [ - { - loader: MiniCssExtractPlugin.loader, - }, - { - loader: 'css-loader', - options: { - modules: { - localIdentName: '[local]---[hash:base64:5]', - }, - importLoaders: 1, - }, - }, - { - loader: 'postcss-loader', - options: { - postcssOptions: { - plugins: [ - postCssImport(), - autoprefixer(), - postCssCustomProperties({ - preserve: false, - importFrom: [path.join(generateStripesAlias('@folio/stripes-components'), 'lib/variables.css')] - }), - postCssCalc(), - postCssNesting(), - postCssCustomMedia(), - postCssMediaMinMax(), - postCssColorFunction(), - ], - }, - }, - }, - ], -}); +// prodConfig.module.rules.push({ +// test: /\.css$/, +// use: [ +// { +// loader: MiniCssExtractPlugin.loader, +// }, +// { +// loader: 'css-loader', +// options: { +// modules: { +// localIdentName: '[local]---[hash:base64:5]', +// }, +// importLoaders: 1, +// }, +// }, +// { +// loader: 'postcss-loader', +// options: { +// postcssOptions: { +// plugins: [ +// postCssImport(), +// autoprefixer(), +// postCssCustomProperties({ +// preserve: false, +// importFrom: [path.join(generateStripesAlias('@folio/stripes-components'), 'lib/variables.css')] +// }), +// postCssCalc(), +// postCssNesting(), +// postCssCustomMedia(), +// postCssMediaMinMax(), +// postCssColorFunction(), +// ], +// }, +// }, +// }, +// ], +// }); prodConfig.module.rules.push( { diff --git a/webpack.config.css.js b/webpack.config.css.js new file mode 100644 index 0000000..ed6f654 --- /dev/null +++ b/webpack.config.css.js @@ -0,0 +1,129 @@ +const path = require('path'); +const postCssImport = require('postcss-import'); +const autoprefixer = require('autoprefixer'); +const postCssCustomProperties = require('postcss-custom-properties'); +const postCssCalc = require('postcss-calc'); +const postCssNesting = require('postcss-nesting'); +const postCssCustomMedia = require('postcss-custom-media'); +const postCssMediaMinMax = require('postcss-media-minmax'); +const postCssColorFunction = require('postcss-color-function'); +const postCssOmitImports = require('./webpack/postcss/postcss-omit-imports'); +const MiniCssExtractPlugin = require('mini-css-extract-plugin'); + +const { generateStripesAlias, tryResolve } = require('./webpack/module-paths'); + +const locateCssVariables = () => { + const variables = 'lib/variables.css'; + const localPath = path.join(path.resolve(), variables); + + // check if variables are present locally (in cases when stripes-components is + // being built directly) if not look for them via stripes aliases + return tryResolve(localPath) ? + localPath : + path.join(generateStripesAlias('@folio/stripes-components'), variables); +}; + +const locateTheme = (themePath) => { + return tryResolve(themePath) ? + themePath : path.join('..', themePath); +} + +const getCSSVariableSettings = (stripesConfig) => { + const settings = {}; + // preserve: false removes css variable entries, leaving only the baked CSS variables. + if (stripesConfig.legacyCSS) { + settings.preserve = false; + } + + settings.importFrom = []; + + // imports a theme from stripes-config. + if (stripesConfig.theme) { + settings.importFrom.push(locateTheme(stripesConfig.theme)) + } + + return settings; +} + +module.exports = (wpconfig, stripesConfig, context) => { + let production = false; + + let cssEntries = [ + locateCssVariables(), + '@folio/stripes-components/lib/global.css', + ]; + let themeEntry = []; + if (stripesConfig.theme) { + themeEntry = [locateTheme(stripesConfig.theme)]; + } + + wpconfig.entry = [ + ...cssEntries, + ...wpconfig.entry, + ...themeEntry + ]; + + let postCSSEntries = [postCssOmitImports({ contains: /variables/ })]; + +// since entries will be stripped out by karma-webpack, we have to import CSS via postcss-import (inline them); + if (context._.includes('karma')) { + postCSSEntries = [postCssImport()]; + } + + // default for development... style-loader. + let loaderEntries = [ + { + loader: 'style-loader' + }, + ]; + + // use CSS extraction for production instead of style-tag injection. + if (context._.includes('build')) { + production = true; + loaderEntries = [{ + loader: MiniCssExtractPlugin.loader, + }]; + + wpconfig.plugins.push(new MiniCssExtractPlugin({ filename: 'style.[contenthash].css' })); + } + + wpconfig.module.rules.push({ + test: /\.css$/, + use: [ + ...loaderEntries, + { + loader: 'css-loader', + options: { + modules: { + localIdentName: '[local]---[hash:base64:5]', + }, + sourceMap: !production, + importLoaders: 1, + }, + }, + { + loader: 'postcss-loader', + options: { + postcssOptions: { + plugins: [ + ...postCSSEntries, + autoprefixer(), + postCssCustomProperties( + getCSSVariableSettings(stripesConfig) + ), + postCssCalc(), + postCssNesting(), + postCssCustomMedia(), + postCssMediaMinMax(), + postCssColorFunction(), + ], + }, + sourceMap: !production, + }, + }, + ], + }); + + // console.log(JSON.stringify(wpconfig, null, 2)); + return wpconfig; +} diff --git a/webpack/build.js b/webpack/build.js index a175c64..d28ba12 100644 --- a/webpack/build.js +++ b/webpack/build.js @@ -11,6 +11,9 @@ module.exports = function build(stripesConfig, options) { return new Promise((resolve, reject) => { logger.log('starting build...'); let config = require('../webpack.config.cli.prod'); // eslint-disable-line global-require + let cssConfig = require('../webpack.config.css'); // eslint-disable-line global-require + + config = cssConfig(config, stripesConfig, options); if (!options.skipStripesBuild) { config.plugins.push(new StripesWebpackPlugin({ stripesConfig, createDll: options.createDll })); diff --git a/webpack/postcss/postcss-omit-imports/index.js b/webpack/postcss/postcss-omit-imports/index.js new file mode 100644 index 0000000..c844791 --- /dev/null +++ b/webpack/postcss/postcss-omit-imports/index.js @@ -0,0 +1,21 @@ +/* omit-imports will remove imports that match the supplied 'contains' regex */ + +module.exports = (opts = { contains: /variables/ }) => { + return { + postcssPlugin: 'postcss-omit-imports', + prepare (result) { + const removals = {} + return { + AtRuleExit: { + import: atRule => { + if (opts.contains.test(atRule.params)) { + atRule.remove(); + } + } + }, + } + } + } +}; + +module.exports.postcss = true; \ No newline at end of file diff --git a/webpack/postcss/postcss-omit-imports/omit.test.js b/webpack/postcss/postcss-omit-imports/omit.test.js new file mode 100644 index 0000000..2246ce1 --- /dev/null +++ b/webpack/postcss/postcss-omit-imports/omit.test.js @@ -0,0 +1,157 @@ +const postcss = require('postcss'); +const plugin = require('./'); +const mocha = require('mocha'); +const chai = require('chai'); + +const { + describe, + it +} = mocha; + +const { expect } = chai; + +function run(input, output, opts) { + return postcss([plugin(opts)]).process(input, { from: undefined }).then(result => { + expect(result.css).to.equal(output); + }) + .catch(err => {}); +} + +describe('omit imports', () => { + it('removes the import statement as expected',() => { + run(ex, exOut); + }) +}); + +describe('leaves other imports in place', () => { + it('removes the import statement as expected',() => { + run(ex2, ex2Out); + }) +}); + +describe('leaves other imports in place', () => { + it('removes the import statement as expected',() => { + run(ex3, ex3Out); + }) +}); + +const ex = ` +/** + * Shared form styles + */ + +@import "../variables.css"; +@import "some/other/file.css"; + +.inputGroup { + position: relative; +} + +/** + * Default input, textarea and select styling + */ +.formControl { + margin-bottom: var(--control-margin-bottom); + border: 1px solid var(--color-border-form); + color: var(--color-text); + background-color: var(--color-fill-form-element); + + &::-ms-clear { + display: none; + } +} +`; + +const exOut = ` +/** + * Shared form styles + */ + +@import "some/other/file.css"; + +.inputGroup { + position: relative; +} + +/** + * Default input, textarea and select styling + */ +.formControl { + margin-bottom: var(--control-margin-bottom); + border: 1px solid var(--color-border-form); + color: var(--color-text); + background-color: var(--color-fill-form-element); + + &::-ms-clear { + display: none; + } +} +`; + + +const ex2 = ` +@import '../variables.css'; +@import "some/other/file.css"; + +/** +* Default styling +*/ + +.button { + composes: interactionStyles from "../sharedStyles/interactionStyles.css"; + padding: 0 var(--gutter-static); + -webkit-appearance: none; + cursor: pointer; + border: transparent; + + &:visited { + color: inherit; + } + + &::before { + border-radius: 999px; + } +} +`; +const ex2Out =` +@import "some/other/file.css"; + +/** +* Default styling +*/ + +.button { + composes: interactionStyles from "../sharedStyles/interactionStyles.css"; + padding: 0 var(--gutter-static); + -webkit-appearance: none; + cursor: pointer; + border: transparent; + + &:visited { + color: inherit; + } + + &::before { + border-radius: 999px; + } +} +`; + +const ex3 = ` +@import "../variables"; +@import "./styles/padding"; +@import "./styles/margin"; +@import "./styles/flexbox"; +@import "./styles/width"; +@import "./styles/text"; +@import "./styles/float"; +` + +const ex3Out = ` +@import "./styles/padding"; +@import "./styles/margin"; +@import "./styles/flexbox"; +@import "./styles/width"; +@import "./styles/text"; +@import "./styles/float"; +` \ No newline at end of file diff --git a/webpack/serve.js b/webpack/serve.js index 4ce2986..6713dd9 100644 --- a/webpack/serve.js +++ b/webpack/serve.js @@ -18,14 +18,17 @@ module.exports = function serve(stripesConfig, options) { if (typeof stripesConfig.okapi !== 'object') throw new Error('Missing Okapi config'); if (typeof stripesConfig.okapi.url !== 'string') throw new Error('Missing Okapi URL'); if (stripesConfig.okapi.url.endsWith('/')) throw new Error('Trailing slash in Okapi URL will prevent Stripes from functioning'); + return new Promise((resolve) => { logger.log('starting serve...'); const app = express(); let config = require('../webpack.config.cli.dev'); // eslint-disable-line global-require let developmentConfig = require('../webpack.config.cli.dev.shared.styles'); + let cssConfig = require('../webpack.config.css'); if (process.env.NODE_ENV === 'development') { config = developmentConfig(config, {}); + config = cssConfig(config, stripesConfig, options); } config.plugins.push(new StripesWebpackPlugin({ stripesConfig }));