mirror of
https://github.com/ionic-team/ionic-framework.git
synced 2025-08-15 09:34:19 +08:00

<!-- Please do not submit updates to dependencies unless it fixes an issue. --> <!-- Please try to limit your pull request to one type (bugfix, feature, etc). Submit multiple pull requests if needed. --> ## What is the new behavior? <!-- Please describe the behavior or changes that are being added by this PR. --> Following the Design Doc for the new Ionic Design System, an architecture for Design Tokens was added. 1. Added new `foundations` folder inside the `src`, to hold all the tokens related files. 2. On this file, a json with all the tokens for the new Design System was added. The format followed was the one suggested on the [W3C draft](https://design-tokens.github.io/community-group/format/). 3. Added a dev dependency for [Styles Dictionary](https://amzn.github.io/style-dictionary/#/), version 3.9.2. 4. Added a tokens.js script that will take care of transforming the json tokens into the desired output. For now, three files are being generated: ionic.vars.css (:root with all CSS Variables); ionic.vars.css (scss variables for each token) and ionic.utility.css (a new utility-class for each token) 5. Added the script `npm run build.tokens` to package.json, that will generate the three files mentioned above. For now, all these changes bring no impact to the rest of the Framework, as these variables are not yet being consumed. The `margin`, `padding` and `border-radius` were removed from the prohibited properties on lint, to prevent lint errors with the new utility-classes. This is very open to discussion/feedback if it's seen as not ok. The `build` command now includes the `build.tokens` script, to make sure the files are always generated, in case someone forget to run the command, after changing the json file! ## Does this introduce a breaking change? - [ ] Yes - [x] No <!-- If this introduces a breaking change: 1. Describe the impact and migration path for existing applications below. 2. Update the BREAKING.md file with the breaking change. 6. Add "BREAKING CHANGE: [...]" to the commit description when merging. See https://github.com/ionic-team/ionic-framework/blob/main/.github/CONTRIBUTING.md#footer for more information. --> --------- Co-authored-by: Sean Perkins <13732623+sean-perkins@users.noreply.github.com>
159 lines
5.3 KiB
JavaScript
159 lines
5.3 KiB
JavaScript
/* For generating Design Tokens, we use Style Dictionary for several reasons:
|
|
- It's prepared to easily generate tokens for multiple types of outputs (CSS, SCSS, iOS, Android, documentation, etc.).
|
|
- It also works very well out of the box with any kind of Design Tokens formats, like Figma Tokens, as well as APIs to adjust to more custom ones.
|
|
- It is probably the most well-known and widely used Design Tokens tool. It has also been regularly maintained for a long time.
|
|
- It can easily scale to different necessities we might have in the future.
|
|
*/
|
|
// eslint-disable-next-line @typescript-eslint/no-var-requires
|
|
const StyleDictionary = require('style-dictionary');
|
|
|
|
const { fileHeader } = StyleDictionary.formatHelpers;
|
|
|
|
// Empty for as an example of how we can add some extra variables, not from the tokens JSON
|
|
const customVariables = ``;
|
|
|
|
// Prefix for all generated variables
|
|
const variablesPrefix = 'ionic';
|
|
|
|
// CSS vanilla :root format
|
|
StyleDictionary.registerFormat({
|
|
name: 'rootFormat',
|
|
formatter({ dictionary, file }) {
|
|
// Add a prefix to all variable names
|
|
const prefixedVariables = dictionary.allProperties.map((prop) => {
|
|
return ` --${variablesPrefix}-${prop.name}: ${prop.value};`;
|
|
});
|
|
|
|
return (
|
|
fileHeader({ file }) +
|
|
':root {\n' +
|
|
prefixedVariables.join('\n') + // Join all prefixed variables with a newline
|
|
customVariables +
|
|
'\n}\n'
|
|
);
|
|
},
|
|
});
|
|
|
|
// scss variables format
|
|
StyleDictionary.registerFormat({
|
|
name: 'scssVariablesFormat',
|
|
formatter({ dictionary, file }) {
|
|
// Add a prefix to all variable names
|
|
const prefixedVariables = dictionary.allProperties.map((prop) => {
|
|
return `$${variablesPrefix}-${prop.name}: var(--${variablesPrefix}-${prop.name}, ${prop.value});`;
|
|
});
|
|
|
|
return (
|
|
fileHeader({ file }) +
|
|
prefixedVariables.join('\n') + // Join all prefixed variables with a newline
|
|
customVariables +
|
|
'\n'
|
|
);
|
|
},
|
|
});
|
|
|
|
// Create utility-classes
|
|
StyleDictionary.registerFormat({
|
|
name: 'cssUtilityClassesFormat',
|
|
formatter({ dictionary, file }) {
|
|
const utilityClasses = dictionary.allProperties.map((prop) => {
|
|
let tokenType = prop.attributes.category;
|
|
const className = `${prop.name}`;
|
|
let utilityClass = '';
|
|
|
|
switch (tokenType) {
|
|
case 'color':
|
|
utilityClass = `.${variablesPrefix}-${className} {\n color: $ionic-${prop.name};\n}
|
|
.${variablesPrefix}-background-${className} {\n background-color: $ionic-${prop.name};\n}`;
|
|
break;
|
|
case 'border':
|
|
const borderAttribute = prop.attributes.type === 'radius' ? 'border-radius' : 'border-width';
|
|
utilityClass = `.${variablesPrefix}-${className} {\n ${borderAttribute}: $ionic-${prop.name};\n}`;
|
|
break;
|
|
case 'font':
|
|
const fontAttribute = prop.attributes.type === 'size' ? 'font-size' : 'font-weight';
|
|
utilityClass = `.${variablesPrefix}-${className} {\n ${fontAttribute}: $ionic-${prop.name};\n}`;
|
|
break;
|
|
case 'elevation':
|
|
utilityClass = `.${variablesPrefix}-${className} {\n box-shadow: $ionic-${prop.name};\n}`;
|
|
break;
|
|
case 'space':
|
|
utilityClass = `.${variablesPrefix}-margin-${className} {\n margin: $ionic-${prop.name};\n};
|
|
.${variablesPrefix}-padding-${className} {\n padding: $ionic-${prop.name};\n}`;
|
|
break;
|
|
default:
|
|
utilityClass = `.${variablesPrefix}-${className} {\n ${tokenType}: $ionic-${prop.name};\n}`;
|
|
}
|
|
|
|
return utilityClass;
|
|
});
|
|
|
|
return [fileHeader({ file }), '@import "./ionic.vars";\n', utilityClasses.join('\n')].join('\n');
|
|
},
|
|
});
|
|
|
|
// Make Style Dictionary comply with the $ format on properties from W3C Guidelines
|
|
const w3cTokenJsonParser = {
|
|
pattern: /\.json|\.tokens\.json|\.tokens$/,
|
|
parse(_a) {
|
|
var contents = _a.contents;
|
|
// replace $value with value so that style dictionary recognizes it
|
|
var preparedContent = (contents || '{}')
|
|
.replace(/"\$?value":/g, '"value":')
|
|
// convert $description to comment
|
|
.replace(/"\$?description":/g, '"comment":');
|
|
//
|
|
return JSON.parse(preparedContent);
|
|
},
|
|
};
|
|
|
|
StyleDictionary.registerParser(w3cTokenJsonParser);
|
|
|
|
// Generate Tokens
|
|
StyleDictionary.extend({
|
|
source: ['./src/foundations/*.json'],
|
|
platforms: {
|
|
css: {
|
|
buildPath: './src/foundations/',
|
|
transformGroup: 'css',
|
|
files: [
|
|
{
|
|
destination: 'ionic.root.scss',
|
|
format: 'rootFormat',
|
|
options: {
|
|
outputReferences: true,
|
|
fileHeader: `myFileHeader`,
|
|
},
|
|
},
|
|
],
|
|
},
|
|
scss: {
|
|
buildPath: './src/foundations/',
|
|
transformGroup: 'scss',
|
|
files: [
|
|
{
|
|
destination: 'ionic.vars.scss',
|
|
format: 'scssVariablesFormat',
|
|
options: {
|
|
outputReferences: true,
|
|
fileHeader: `myFileHeader`,
|
|
},
|
|
},
|
|
{
|
|
destination: 'ionic.utility.scss',
|
|
format: 'cssUtilityClassesFormat',
|
|
options: {
|
|
outputReferences: true,
|
|
fileHeader: `myFileHeader`,
|
|
},
|
|
},
|
|
],
|
|
},
|
|
},
|
|
fileHeader: {
|
|
myFileHeader: () => {
|
|
return [`This is an auto-generated file, please do not change it directly.`, `Ionic Design System`];
|
|
},
|
|
},
|
|
}).buildAllPlatforms();
|