Skip to content

Commit

Permalink
Merge pull request #971 from primer/variable-deprecations
Browse files Browse the repository at this point in the history
Add variable deprecation data and tests
  • Loading branch information
shawnbot authored Nov 4, 2019
2 parents c539fcd + c6be855 commit 4fec939
Show file tree
Hide file tree
Showing 6 changed files with 251 additions and 33 deletions.
49 changes: 47 additions & 2 deletions deprecations.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,36 @@
* array and a "message" string.
*/
const versionDeprecations = {
'14.0.0': [
{
selectors: ['.UnderlineNav-item.selected', '.UnderlineNav-item.selected .UnderlineNav-octicon'],
message: `Please use aria-selected="true" to indicate the selected state of an UnderlineNav item.`
},
{
variables: ['$status-pending'],
message: `This variable is deprecated.`
},
{
variables: ['$repo-private-text', '$repo-private-bg', '$repo-private-icon'],
message: `These variables are deprecated.`
},
{
variables: ['$marketingSpacers', '$allSpacers'],
message: `Please use the $marketing-spacers and $marketing-all-spacers variables.`
},
{
variables: ['$exploregrid-item-border-radius'],
message: `This variable is deprecated. Use "4px" instead.`
},
{
variables: ['$stats-switcher-py', '$stats-viewport-height'],
message: `These variables are deprecated.`
},
{
variables: ['$min_tab_size', '$max_tab_size'],
message: `These variables are deprecated.`
}
],
'13.0.0': [
{
selectors: [
Expand Down Expand Up @@ -68,11 +98,15 @@ const semver = require('semver')

// map selectors to the version and message of their deprecation
const selectorDeprecations = new Map()
const variableDeprecations = new Map()
for (const [version, deps] of Object.entries(versionDeprecations)) {
for (const {selectors, message} of deps) {
for (const {selectors = [], variables = [], message} of deps) {
for (const selector of selectors) {
selectorDeprecations.set(selector, {version, message})
}
for (const variable of variables) {
variableDeprecations.set(variable, {version, message})
}
}
}

Expand All @@ -81,4 +115,15 @@ function isSelectorDeprecated(selector, version = CURRENT_VERSION) {
return deprecation ? semver.gte(deprecation.version, version) : false
}

module.exports = {versionDeprecations, selectorDeprecations, isSelectorDeprecated}
function isVariableDeprecated(variable, version = CURRENT_VERSION) {
const deprecation = variableDeprecations.get(variable)
return deprecation ? semver.gte(deprecation.version, version) : false
}

module.exports = {
versionDeprecations,
selectorDeprecations,
variableDeprecations,
isSelectorDeprecated,
isVariableDeprecated
}
72 changes: 71 additions & 1 deletion docs/content/getting-started/contributing.md
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,11 @@ It's usually better to open an issue before investing time in spiking out a new
1. What the pattern is and how it's being used across the site - post screenshots and urls where possible. If you need help identifying where the pattern is being used, call that out here and cc the relevant team and/or cc `@product-design` to help.
2. Why you think a new pattern is needed (this should answer the relevant questions above).
3. If you intend to work on these new styles yourself, let us know what your timeline and next steps are. If you need help and this is a dependency for shipping another project, please make that clear here and what the timeline is.
4. Add the `type: new styles` label, or `type: refactor` where appropriate.
4. Add the appropriate label(s):
- `Type: Enhancement` for new styles
- `Type: Bug Fix` for—you guessed it!—bug fixes
- `Type: Polish` for refactors of existing styles
- `Type: Breaking Change` for any change that [removes CSS selectors or SCSS variables](#removing-styles-and-variables)

### Step 2: Design and build the new styles

Expand All @@ -64,6 +68,67 @@ If you get to this step you've helped contribute to a style guide that many of y

Let the [design systems team](https://github.com/github/design-systems) know if we can improve these guidelines and make following this process any easier.


## Removing styles and variables

Removing styles and SCSS variables can be scary. How do you know if the thing you're deleting (or just planning to delete) isn't used in other projects? [Semantic versioning] provides us with an answer: We **don't** know, but we can use "major" version increments (from, say, `13.4.0` to `14.0.0`) to signal that the release includes potentially breaking changes. The rule is simple:

**Whenever we delete a CSS selector or SCSS variable, we will increment to the next major version.**

When planning to delete a CSS selector or SCSS variable, you should:

1. Add a [TODO@version comment](#primer-csstodo) above the line in question:

```scss
// [email protected]: delete $some-unused-var
$some-unused-var: 15px !default;
```

1. Add it to [deprecations.js]:

```js
const versionDeprecations = {
'15.0.0': [
{
variables: ['$some-unused-var'],
message: '$some-unused-var is unused, and has been deprecated.'
}
]
}
```

We have several checks and tools in place to help us plan, track, and catch both expected and unexpected removals of both CSS selectors and SCSS variables:

### `deprecations.js`
[This file][deprecations.js] is where we document all of our current and _planned_ CSS selector and SCSS variable deprecations (removals), and is used to generate [deprecation data](../tools/deprecations) for other tools.

### `script/test-deprecations.js`
[This script][script/test-deprecations.js] compares the CSS stats and variable data between the latest release and the local code, and throws error messages if:

- A CSS selector has been deleted but was not listed in [deprecations.js]
- A CSS selector listed in [deprecations.js] was _not removed_ in the version it claims to have been deprecated
- An SCSS variable has been deleted but was not listed in [deprecations.js]
- An SCSS variable listed in [deprecations.js] was _not removed_ in the version it claims to have been deprecated

Run `script/test-deprecation.js --help` for more info and available options.

### `primer-css/TODO`
[This stylelint rule][lib/stylelint-todo.js] looks for comments in the form:

```scss
// TODO@<version>: <message>
```

and generates an error for each one whose `<version>` is less than or equal to the current version (in `package.json`). You can test this rule for future releases with:

```sh
PRIMER_VERSION=<version> npx stylelint-only primer-css/TODO -- src
```

where `<version>` is the future version you'd like to compare against. Assuming that the correctly formatted comments exist already, violations of this stylelint rule can be used to generate a checklist of lines to remove in a future release.

See [the deprecation data docs](../tools/deprecations) for more information.

## Documentation structure

- Our documentation site for Primer CSS is built using [Doctocat](https://primer.style/doctocat) and deployed with [Now](https://zeit.co/now). Our site is built from the `docs` folder and uses [MDX](https://mdxjs.com) to render markdown.
Expand Down Expand Up @@ -97,3 +162,8 @@ Check out Doctocat's [live code](https://primer.style/doctocat/usage/live-code)
Primer CSS follows [semantic versioning](http://semver.org/) conventions. This helps others know when a change is a patch, minor, or breaking change.

To understand what choice to make, you'll need to understand semver and know if one of the changes shown is a major, minor, or patch. Semver is confusing at first, so I recommend reviewing [semver](http://semver.org/) and/or ask in [#design-systems](https://github.slack.com/archives/design-systems) or and experienced open-source contributor.

[semantic versioning]: https://semver.org
[script/test-deprecations.js]: https://github.com/primer/css/tree/master/script/test-deprecations.js
[deprecations.js]: https://github.com/primer/css/tree/master/deprecations.js
[lib/stylelint-todo.js]: https://github.com/primer/css/tree/master/lib/stylelint-todo.js
58 changes: 46 additions & 12 deletions docs/content/tools/deprecations.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,28 +2,25 @@
title: Deprecation data
---

As of version 12.7.0, we publish CSS selector deprecation data with
`@primer/css`. You can access the data via the [Node API](#node) or as
[JSON](#json).
As of version 12.7.0, we publish CSS selector and SCSS variable deprecation data (as of 14.0.0) with `@primer/css`. You can access the data via the [Node API](#node) or as [JSON](#json).

Deprecation messages strings may include Markdown so that they can be included
in the [changelog].

**Keep in mind that this data includes both active and _planned_
deprecations.** You can determine whether a CSS selector is deprecated for the
version of `@primer/css` you've installed via the [Node API](#node), or by
comparing the version of a selector deprecation with the installed version in
your own environment.
**Keep in mind that this data includes both active and _planned_ deprecations.** The [Node API](#node) is the best way to determine whether a selector or variable is deprecated for the version of `@primer/css` you've installed.

## JSON

The JSON data is available in the unpacked node module's `dist/deprecations.json`, and is an object with the following structure:

* `versions` is an object whose keys are version numbers (e.g. `13.0.0`) and values are deprecation messages: objects with a `selectors` array and a `message`:
* `versions` is an object whose keys are version numbers (e.g. `13.0.0`) and values are deprecation messages, each of which has a `message` string and a `selectors` and/or `variables` array:

```json
{
"versions": {
"14.0.0": [
{
"variables": ["$min_tab_size", "$max_tab_size"],
"message": "These variables have been deprecated."
}
],
"13.0.0": [
{
"selectors": [".btn-purple"],
Expand All @@ -33,6 +30,8 @@ The JSON data is available in the unpacked node module's `dist/deprecations.json
}
}
```

Deprecation messages strings may include Markdown so that they can be included in the [changelog].

* `selectors` is an object mapping CSS selectors (e.g. `.btn-purple`) to the version in which they are _or will be_ deprecated:

Expand All @@ -46,6 +45,20 @@ The JSON data is available in the unpacked node module's `dist/deprecations.json
}
}
```

* `variables` is an object mapping SCSS variables (including the leading `$`, e.g. `$status-pending`) to the version in which they are or will be deprecated:

```json
{
"variables": {
"$status-pending": {
"version": "14.0.0",
"message": "This variable is unused in Primer, and is deprecated."
}
}
}
```


## Node

Expand Down Expand Up @@ -90,6 +103,27 @@ console.log(`Primary buttons are bad? ${isSelectorDeprecated('.btn-primary')}`)
// "Primary buttons are bad? false"
```

### `variableDeprecations`
This is a [Map] object with keys for each SCSS variable mapped to the deprecation info:

```js
const {selectorDeprecations} = require('@primer/css/deprecations')
console.log(`Will $status-pending be deprecated? ${variableDeprecations.has('$status-pending')}`)
// "Will $status-pending be deprecated? true"
```

### `isVariableDeprecated(variable[, version])`
Returns `true` if the named SCSS variable (including the leading `$`) will have been deprecated (removed) _by_ the specified [semver] version.

```js
const {isVariableDeprecated} = require('@primer/css/deprecations')
console.log(`$status-pending deprecated? ${isVariableDeprecated('$status-pending')}`)
// "$status-pending deprecated? true"
console.log(`$yellow-700 deprecated? ${isVariableDeprecated('$yellow-700')}`)
// "$yellow-700 deprecated false"
```


[semver]: https://npm.im/semver
[changelog]: https://github.com/primer/css/tree/master/CHANGELOG.md
[Map]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map
13 changes: 9 additions & 4 deletions script/dist.js
Original file line number Diff line number Diff line change
Expand Up @@ -88,15 +88,20 @@ function getPathName(path) {
}

function writeDeprecationData() {
const {versionDeprecations, selectorDeprecations} = require('../deprecations')
const {versionDeprecations, selectorDeprecations, variableDeprecations} = require('../deprecations')
const data = {
versions: versionDeprecations,
selectors: Array.from(selectorDeprecations.entries()).reduce((obj, [selector, deprecation]) => {
obj[selector] = deprecation
selectors: mapToObject(selectorDeprecations),
variables: mapToObject(variableDeprecations)
}
return writeFile(join(outDir, 'deprecations.json'), JSON.stringify(data, null, 2))

function mapToObject(map) {
return Array.from(map.entries()).reduce((obj, [key, value]) => {
obj[key] = value
return obj
}, {})
}
return writeFile(join(outDir, 'deprecations.json'), JSON.stringify(data, null, 2))
}

if (require.main === module) {
Expand Down
Loading

0 comments on commit 4fec939

Please sign in to comment.