Skip to content

Latest commit

 

History

History
158 lines (116 loc) · 5.68 KB

CODE_STYLE.md

File metadata and controls

158 lines (116 loc) · 5.68 KB

Tegel Design System - Code conventions

Welcome to the code conventions section of Tegel, a comprehensive design system that aims to provide a unified and consistent visual language for all of your digital products. In this section, we'll cover the coding standards and conventions that are used to build and maintain Tegel components.

General

  • Use camelCase for variables and functions.
  • Use PascalCase for classes and interfaces.
  • Use template literals instead of concatenation.
  • Use meaningful variable and function names: Name your variables and functions in a way that clearly communicates their purpose. Avoid using single-letter variable names, instead of i for index, be descriptive and name the variable index.
  • Use single quotes (') for strings.
  • If applicable, add unit test.

Branching

  • Branch out of develop branch for day-to-day work
  • Branch named main is updated once release is done by merging develop to it
  • Use task type as to separate branches in different categories (fix, feat, chore, build, docs, release...)
  • In case you are IXIB team member, start the name of the branch with Jira ticket number followed by short description of task
  • example: fix/CDEP-1369-banner-color-correction
  • If Jira ticket number is not available to you, use Github ticket number as prefix instead. If no issue exists, create that first before creating the branch.

Commit messages

  • Tegel is using Conventional Commits to manage commit messages.
  • Use the commit message to explain what the commit does and which component it affects.
  • example: fix(banner): correct color
  • There is commit lint configured to enforce conventional commit format. If you are not following the format, the commit will be rejected.
  • In case you need help, there is a script that will generate a conventional commit message for you. You can use it by running npm run commit and following the prompts.

Unit testing approach

For our unit testing, we've adopted Playwright. Within our project structure, any component subject to testing is accompanied by a dedicated test folder located within its specific component directory. For instance, you'll find this under src/components/button/test/disabled/ for testing the disabled state of a button component. Within this folder, you should create a test file named button.e2e.ts alongside an index.html file that outlines the structure of the component being tested.

Folder structure

All components are located in the src/components directory. Each component in this folder has a dedicated folder named as the component minus the tds-prefix. For our Button the file path is src/components/button. The components tsx/scss files are all named without the tds-prefix. E.g. button.tsx, button.scss, button-vars.scss.

Example - folder structure for Button

├── src
│   ├── components
│   │   ├── button
│   │   │   ├──button-vars.scss
│   │   │   ├──button.tsx
│   │   │   ├──button.scss

Component structure

Our component files (e.g. button.tsx) should follow the following structure, in order:

  1. Props, state and variables
    • @Component decorator with the component specific arguments.
    • Host element for the component (@Element).
    • Component props.
    • Component state.
    • Private variables.
  2. Event, emitters and listeners
    • Events emitters (@Event).
    • Event listeners (@Listen)
  3. Methods
    • Props and state watchers (@Watch)
    • Public methods (@Method)
    • Private methods
    • Lifecycle methods
    • Render method.

Example:

 
@Component({
  tag: 'tds-component',
  styleUrl: 'tds-component.scss',
  shadow: true,
})
export class TdsComponent {
  @Element() host: HTMLElement;

  /** Comment explaining the use of the the prop */
  @Prop() prop: string;
 
  /** Comment explaining the use of the the prop */
  @Prop() secondProp: string;

  @State() state: boolean;

  variable: string

  /** Comment explaining the event. */
  @Event({
    eventName: 'tdsEvent',
    composed: true ,
    cancelable: true,
    bubbles: true,
  })
  tdsEvent: EventEmitter<{}>;

  @Listen('tdsEvent', { target: 'body' })
  handleListener(){
  }

  @Watch('prop')
  handlePropChange(){
  }

  /** Comment explaining the method. */
  @Method()
  async handleComponent(){
  }

  handleClick(){
  }

  connectedCallback(){
  }

  componentWillLoad() {
  }

  render() {
    return (
      <Host>
      </Host>
    );
  }
}

Slots

Slots are used throughout many of our components. Some of these are named slots, and in that case they should follow the following convention: The slot should always be named to represent its position/use case. For example this could be 'bottom', 'label' or 'end'.

Events

The tegel components emit custom events to allow the users to repond to changes/updates in the components. These are all named using the tds-prefix. This is done in order to not have conflicting events and to make it clear to the user the specified event is something that is emitted from a tegel component.

The events are named according to our naming convention: 'tds' + event. For a click event this would result in the event being called tdsClick.

Internal events

Some of our components are using events to communicate with its parent/child. These events are not recommended to use in any way since they might change without notice. Their payload might also be changed based on refactoring of components. These events are prefixed with 'internal'. This is to make it as clear as possible to a user that this is an internal event that the components are using, but the user should not interact with it. E.g. internalTdsPropsChange.