Direction:
Contributing code
Before you dive into contributing code:
- Read the general contribution guidelines
- Check out the code design principles
Get the code
Clone the repository: git clone ssh://gerrit.wikimedia.org:29418/design/codex
First time using Gerrit?
See the Gerrit tutorial on mediawiki.org for a detailed guide.
Code contribution workflow
Task tracking
Tasks are tracked in Phabricator. We use three different Phabricator workboards, for three different purposes:
- Codex: used to indicate that a task is related to Codex. We do not track task status here.
- Design-Systems-Team: used to triage, sort, prioritize, and refine tasks that the Design Systems team and contributors will work on.
- Design-Systems-Sprint: used to track active works-in-progress from Research > Design > Development > Testing > Release. As a contributor, you are welcome to track your work related to the Design System on this board as well—just remember to keep the task in the appropriate column that reflects its status.
Create or claim a task as soon as you decide to work on it. This will help avoid overlapping duplicate, or out-of-order work. Note that very minor contributions (like build asset updates or icon code optimizations) do not require a corresponding task.
When adding a new component, developing the entire component to completely fulfill design criteria may be too much for a single patch. Consider creating a parent task for that component with sub-tasks for the minimum viable solution and additional features.
Component addition process
The following list represents the full process of adding a component, from conception as a task to a fully functional and documented component. These steps may be completed by a number of different contributors—a single developer does not need complete all of these steps on their own.
- Create a new component task epic in Phabricator. New Codex components must have a corresponding epic task in Phabricator. Use this component epic task template to create the task, filling in as much information as you can.
- Work with the design team to create a scope of work. Typically, a designated designer and developer work together to create a scope for the minimum viable product (using the component MVP task template), which will be added as a subtask of the component epic. Subtasks will be added for any other features that will be designed and developed after the MVP scope is complete.
- Await the Figma spec sheet. Before a component can be implemented in Codex, the design team will create a specification sheet for it in the Codex components Figma library. A new component task is considered ready for development only once design specifications have been linked in the component task.
- Ready for development. Once design specifications have been shared and the task has been refined by members of the Design Systems Team, the task will be moved to the "Codex Component Backlog" column on the Design-Systems-Team workboard. This indicates that implementation work can begin. You can assign the task to yourself and move it to the 'In Development' column on the Design-Systems-Sprint workboard.
- Build the component. Create the Vue component, applying the design tokens noted in the design specification. Visit writing styles for more details.
- Test the component. Write unit tests for the new code. Visit the unit tests section for more details.
- Demo the component. Create component demos in VitePress, following the specifications provided in the task and existing examples from other component demo pages. Visit the component demos section for more details.
- Handle feedback from design and product. Relevant designers and product managers will review the new component, provide feedback, and eventually sign-off on the component. Developers will respond to the feedback and implement any necessary changes, or open tasks so those changes can be done in the future.
Code contribution pathways
There are various options for contributing code for a new component based on the contributor's experience with our technologies and available time.
To be ready for public consumption, a component must fully implement the design spec and include unit tests, a simple demo in the Vite sandbox, and a demo page with a suite of comprehensive demos. However, these items do not all need to be included at once. Components that are in development but are not yet ready for public use are considered "work in progress" or "WIP" components. Our WIP components system allows submission of partially-completed components, which are not yet included in the main library distribution or on the live Codex docs site.
Here are some options based on different parts of the development process:
- Component completeness: You can build a component that implements every part of the design spec, or you can build part of a new component and submit it as a WIP component. More features can be added in subsequent patches by you or other developers.
- Unit tests: You can submit a new component without writing unit tests (although we do encourage at least adding snapshot tests), or you can help write unit tests for someone else's WIP component.
- Component demos: While a simple implementation of a component in the Vite Sandbox is required for all components, a full component demo page can be added later. Our component demo system is very specific to this project, so learning it might take more time than a contributor has to offer. You can submit a WIP component without adding a demo page, or you can help add a demo page for someone else's WIP component.
- Design/product feedback: While the original implementer of a new component can respond to feedback if they have the time and inclination, feedback can be handled by other developers, or added to the backlog to be completed later.
If you have questions about how you can contribute to a new component, or a proposal for how you would like to contribute to one, please add a comment to that component's Phabricator task and we will help develop a plan for who will do which parts of the component development process.
Patch requirements
Patches for new components should include all of the following. Patches that introduce changes should include or update the following as needed.
- Commit message: The first line of the commit message has to be prefixed by the component name (Button, Icon) or the type of work (build, tests, styles, docs, etc.), e.g.
docs: Add more docs.
If the commit covers multiple things, include them in a comma-delineated list, e.g.styles, docs, Button: Fix Button styles and document new convention
. Special cases are deprecating and breaking changes, to be prefixed with[DEPRECATING CHANGE]
or[BREAKING CHANGE]
, e.g.[BREAKING CHANGE] Button: Remove primary button functionality
. In general, Codex is following the MediaWiki commit message guidelines. - Documentation: Each component should have a docblock describing what the component is and the basics of how it works. Props, slots, and events should be documented in the code. If a component would benefit from additional documentation to communicate how it can be used, include it on the demo page.
- Vite Sandbox demo: Each component should have a simple demo in the Vite Sandbox for testing the basic functionality of the component locally and within MediaWiki.
- Jest snapshots: Include snapshots for all variations of props and slots.
- Unit tests (not required for WIP components): Attempt to meet the established coverage threshold, which will be calculated and output in the command line interface when you run tests.
- Demos (not required for WIP components): Each component should have a demo page on the VitePress site that shows realistic examples of different variations and uses of the component.
At a Codex release, but not at commit level, CHANGELOG.md
file in the root folder is updated with all the commit message subjects since last release – separated in "Breaking Changes", "Deprecating Changes", "Features", "Styles", "Code" and "Documentation" sections.
Development basics
Requirements
- Node: see
.nvmrc
in the root of the repository. To install and use the required version of Node, runnvm install "$(<.nvmrc)"
thennvm use
in the root of the repository. - NPM: v7.21 or greater is required to support workspaces; it is not included by default in older versions of Node (prior to v15) and will need to be upgraded manually.
Workspaces
The Codex codebase is configured as a monorepo using NPM. Sub-projects are defined in the packages/
directory with their own package.json
and an appropriate name
.
To run a command in a specific workspace, do the following:
bash
# Run the "build" command in the "@wikimedia/codex" workspace
npm run build --workspace @wikimedia/codex
# Install a dependency for one workspace
npm install vite --workspace @wikimedia/codex --save-dev
# This can be shortened to -w
npm install vue -w @wikimedia/codex --save-peer
To run a command for all workspaces, do this instead:
bash
# Run the "test" command in all workspaces
npm run test --workspaces
# This is equivalent
npm run test -ws
Note that the lint
command is global; it has to be run in the root directory. Running npm test
in the root directory will lint everything and run the tests in all workspaces.
Directory structure
Each component has its own subdirectory in packages/codex/src/components
, containing the Vue single file component and unit tests.
Functions designed for use with the composition API ("composables") are in src/composables
. Composables should have a name starting with use
, e.g. useModelWrapper
. Each composable should be in its own file, with the same name as the composable function, e.g. src/composables/useModelWrapper.ts
.
There are some special files at the top level of the src/
directory:
lib.ts
: The main entry point for the library. Only things that are exported here are accessible to external users of the library.types.ts
: Contains type definitions that are shared across multiple components, or need to be exported. This file should not export any functions or constants, only types.constants.ts
: Contains constants that need to be exported or that serve as the basis for types.utils.ts
: Utility functions shared across components, or for use with the types or constants intypes.ts
orconstants.ts
.