Set up your SSH Key GitHub account and install node.js 4 or higher.
- Generating SSH Keys - GitHub
nvm
(Node Version Manager) to use theNode 6
.
Contributing to carbon-components requires that you can run this repo locally on your computer.
Prefix all class names with #{$prefix}--
in SCSS, which is replaced with
bx--
by default, and design systems inheriting Carbon can override. This
prefix prevents potential conflicts with class names from the user.
HTML
<div
class="bx--inline-notification bx--inline-notification--error"
role="alert"
>
<div class="bx--inline-notification__details">...</div>
</div>
SCSS
.#{$prefix}--inline-notification {
...
}
.#{$prefix}--inline-notification__details {
...
}
Follow BEM naming convention for classes. Again, the only thing we do
differently is prefix all classes with #{$prefix}--
.
.#{$prefix}--block
.#{$prefix}--block__element
.#{$prefix}--block--modifier
Avoid nesting selectors, this will make it easier to maintain in the future.
// Don't do this
.#{$prefix}--inline-notification {
.#{$prefix}--btn {
&:hover {
svg {
...
}
}
}
}
// Do this instead
.#{$prefix}--inline-notification .#{$prefix}--btn {
&:hover svg {
...
}
}
}
SassDoc is used to document the Carbon Sass source.
SassDoc annotations start each line with ///
; do not use ///
in non-SassDoc
comments.
For consistency, capitalize types (used in @type
, @param
, @return
) and
descriptions (used in @param
, @return
, @deprecated
, @example
, @link
).
The following annotations are used:
Required annotations
-
Description - can be one line or multiple lines
-
@access
-public
orprivate
, where public items make up our public API -
@group
- typically a package or component name -
@type
- allowed on variables, (e.g.Map
,Color
,Number
) -
@param
- allowed on functions and mixins, include the type, name, and description, with a default value if there is one (e.g.@param {Map} $breakpoints [$carbon--grid-breakpoints] - A map of breakpoints where the key is the name
) -
@return
- allowed on functions, include the type and description (e.g.@return {Number} In px
) -
@alias
- do not include the$
if aliasing a variable -
@content
- allowed on mixins, describe the usage of content -
@deprecated
- context around possible replacements or when the item will no longer be availableOptional annotations
-
@example
- if the usage isn't straight forward or there are multiple use cases -
@link
- if there's a related link to referenceExamples
// Variable example
/// Primary interactive color; Primary buttons
/// @type Color
/// @access public
/// @group @carbon/themes
$interactive-01: map-get($carbon--theme, interactive-01) !default;
// Mixin example
/// Create the container for a grid. Will cause full-bleed for the grid unless
/// max-width properties are added with `make-container-max-widths`
/// @param {Map} $breakpoints [$carbon--grid-breakpoints] - A map of breakpoints where the key is the name
/// @access private
/// @group @carbon/grid
@mixin carbon--make-container($breakpoints: $carbon--grid-breakpoints) {
}
// Function example
/// Compute the type size for the given type scale step
/// @param {Number} $step - Type scale step
/// @return {Number} In px
/// @access public
/// @group @carbon/type
@function carbon--get-type-size($step) {
}
A nested element can use a new block name as long as the styles are independent of the parent.
<div class="bx--component">
<button class="bx--component-button">Button</button>
</div>
☝️ The #{$prefix}--component-button
class implies that this button has
independent styles from its parent. Generally, it's preferred to start a new
block.
Avoid names with multiple __element
names:
- ❌
.#{$prefix}--card__list__item
- ✅
.#{$prefix}--card-item
- ✅
.#{$prefix}--card__item
All components belong in src/components
in their own folder.
Name files and folders using singular form; not plural.
button
- button.hbs
- _button.scss
- button.js
- button.config.js
Also note that all variants of a component can live in a single HBS, SCSS and JS file respectively. For example, while there are many button variants (primary, secondary, etc.), they're all contained in those single source files in the button folder.
There are two ways to define markups for components and their variants:
- Defining markup with no conditional or data interpolations
- Defining markup with conditionals or data interpolations
Defining markup with no conditional or data interpolation is as easy as adding a
.hbs
file to the component directory. No .config.js
file in the component
directory is required in this case. One thing to note is that If there is a
.hbs
file whose basename is exactly the same as the component name, other
.hbs
files has to be in componentname--variantname.hbs
format.
Defining markup with conditionals or data interpolations requires creating
.config.js
file, which is a
JavaScript module format of Fractal configuration,
in component directory. .hbs
files are rendered with the data given via
context
property in variants[n]
(below).
Supported
properties in .config.js
are the following:
default
: The default variant namevariants
- An array of objects, supporting the following properties:name
: The variant namelabel
: The variant name shown in dev env UInotes
: A short explainer the variant shown in dev env UIcontext
: The data used for rendering.hbs
view
: The basename of the.hbs
file for variant markup (Unlike default Fractal environment, this property should point to the basename of a.hbs
file underdemo
directory orsrc
directory, without its path)preview
: The basename of the.hbs
file for the markup that lays out the variant markup, in "full render" mode (Unlike default Fractal environment, this property should point to the basename of a.hbs
file underdemo
directory orsrc
directory, without@
symbol)meta
: Some metadata. Carbon vanilla development environment reads the following ones specifically:linkOnly
: Only full-page demo is alloweduseIframe
: Use of<iframe>
for non full-page demoxVersionOnly
: Supports "experimental" theme onlyxVersionNotSupported
: "Experimental" theme is not supported
What .hbs
file is used for rendering a variant is determined by searching for
.hbs
files in demo
or src
directory and find one whose basename matches
one of the following (the priority is the following order):
view
property invariants[n]
- Variant handle, which takes a format of
componentname--variantname
format - Component handle, which is
componentname
JavaScript-framework-specific is not recommended as we strive to create styles that are framework-neutral. However, there are some rare cases where framework-specific cannot be avoided, and some of those make sense to be in maintained by core style library here.
There are a couple ways to work on framework-specific style.
This is the most straightforward way. When in the directory of your
carbon-components
folder, run the following command:
yarn link
You should see a success message similar to:
success Registered "carbon-components".
info You can now run `yarn link "carbon-components"` in the projects where you want to use this package and it will be used instead.
Now, go to the folder where carbon-components-angular
is located and run:
yarn link carbon-components
You should see a success message similar to:
success Using linked package for "carbon-components".
The yarn link
command will allow us to point the carbon-components
package
under node_modules
to the folder on our filesystem. So, if we make a change in
carbon-components
and re-compile the project it will update in the Storybook
environment for carbon-components-angular
.
In addition, if you would like to have your changes to styles automatically
compile and update Storybook you can run the following command in the
carbon-components
folder on your machine:
yarn gulp watch -s
This will execute the watch
command in gulpfile.js
. As a result, whenever
you make a change to the project styles it will automatically copy over into the
scss
folder which Storybook uses in carbon-components-angular
.
Though above approach is the most straightforward, it involves an overhead of
having to run build process at carbon-components
, in addition to one at
framework variant repo, upon every Sass code change.
To avoid such overhead, you can point NPM dependency of carbon-components
right to the source code, though there is a caveat that our future change to
directory structure, etc. may make such steps no longer work. Here are the
steps:
> cd /path/to/carbon-components-angular/node_modules/carbon-components
> mv scss scss.orig
> ln -s /path/to/carbon-components/src scss
Then edits of .scss
files in /path/to/carbon-components/src
will be
reflected to the development environment of your framework variant repository.
You don't need to do anything in carbon-components
side.
Go to carbon-components and click
the Fork
button in the top-right corner.
- Go to your GitHub Repositories.
- Click on
[your_github_username]/carbon-components
. - Click on the
Clone or Download
button and copy the URL from theClone with SSH
option. It should start with[email protected]...
In your terminal:
git clone [email protected]:[your_github_username]/carbon-components.git
cd carbon-components
See GitHub docs for more details.
When you clone your forked repo, doing a git remote -v
will show that the
origin
remote is set up for you already by default. This should be pointing to
your forked repo.
Add the IBM/carbon-components
repo to your remote (this can be useful to
update your fork of new changes down the road):
# Add the upstream remote to your repo
git remote add upstream [email protected]:IBM/carbon-components.git
# Verify the remote was added
git remote -v
When you do git remote -v
, you'll see these remotes:
origin
: connection to your forkupstream
: connection to the original repo.
- Always work in a branch.
- Submit pull requests from a branch.
- All commits must follow the convention outlined here.
npm run dev
# or
yarn dev
Once it's done building, you can start editing source code or creating new components. The system is set up to automatically bundle your changes/additions. Visit http://localhost:3000 to see the changes happen on the fly.
Options:
-b
: Enable breaking changes for the next release-e
: Enable experimental features
If you're contributing to our JavaScript code, test your changes by running our test commands:
gulp test:unit
If you add any features to our JavaScript code, make sure to add tests so that your code is covered. Tests are written in Mocha/Chai. You can see if your code is covered by looking at carbon-components/tests/coverage/*/index.html after running test.
If your change may hit some browser quirks, use -b
option, like:
gulp test:unit -b IE -b Firefox
(Other browsers tests can run with are: Safari
, Chrome
and ChromeHeadless
)
If you are very sure that your change affects a specific set of components, you
can use -f
option, like:
gulp test:unit -f tests/spec/fab_spec.js
Other options for testing are:
-d
/--debug
: Stop generating code coverage report. Useful to debug your code when running test.-k
/--keepalive
: Keep running test runner even after test ends. Test will restart running when you make changes to any test files or any files under test.-v
/--verbose
: Let Karma emit detailed log.
If you're contributing to our HTML/CSS code, a11y compliance of your code should be tested.
To do so, you can test your changes by running our test commands:
gulp test:a11y
If you are very sure that your change affects a specific set of components, you
can use --name
option, like:
gulp test:a11y --name dropdown
The a11y test may report potential issues that should be handled in
application-level, not in carbon-components code. In such case, you can ignore
those issues by adding an item to shouldIssueBeIgnoredForRule
table in
tests/a11y/global-ignore-aat-issues.js.
The table is keyed by something like wcag20.tech.h59.linkValid
which helps
identifying what RPT rule to ignore. You can specify true
to the value which
ignores all violations of the rule, or a function which takes the DOM element
violating the rule and returns true
if such violation should be ignored.
Note: Before you make a pull request, search the issues to see if a similar issue has already been submitted. If a similar issue has been submitted, assign yourself or ask to be assigned to the issue by posting a comment. If the issue does not exist, create a new issue.
When you're at a good stopping place and you're ready for feedback from other contributors and maintainers, push your commits to your fork:
Writing commit messages
<type>
indicates the type of commit that's being made. This can be:feat
,fix
,perf
,docs
,chore
,style
,refactor
<scope>
The scope could be anything specifying place of the commit change or the thing(s) that changed.Commit message format:
<type>(<scope>): <subject>
<BLANK LINE>
<body>
<BLANK LINE>
<footer>
Do not submit pull requests from the master
branch of your fork.
git checkout -b { YOUR_BRANCH_NAME }
git add .
git commit -m "fix(table): IE11 positioning error" -m "Fixes #34"
git push origin { YOUR_BRANCH_NAME }
In your browser, navigate to
IBM/carbon-components and click the
button that reads Compare & pull request
Is it a Breaking Change?
We want to respect semver. It's important to discern whether your pull request contains breaking changes or not. Sometimes, renaming or removing things in the code can result in breaking changes.
Here are some examples of breaking changes... changing, renaming or removing any of the following:
- HTML attributes
- Folders or Files
- Any SCSS
@mixin
,$variable
orfunction
- Any JS
function
orclass
We also practice graceful deprecation when something is slated to be removed -- we mark it as deprecated in the current version and remove it in the next major version.
Before you create a pull request, change the base branch depending on what kind of change you're submitting.
- Pull requests with non-breaking changes like patches and minor updates use
the
master
as the base branch. - Pull requests with breaking changes use the latest
major version number
branch as the base branch (i.e.7.0.0
or whatever the next major version is).
Write a title and description then click Create pull request
Stay up to date with the activity in your pull request. Maintainers from the Design System team will be reviewing your work and making comments, asking questions and suggesting changes to be made before they merge your code.
🎉 You no longer need to squash commits 🎉
When you need to make a change, add, commit and push to your branch normally.
Once all revisions to your pull request are complete, someone from Design Systems will squash and merge your commits for you.