Skip to content

Latest commit

 

History

History
173 lines (146 loc) · 7.4 KB

CODE_STYLE.md

File metadata and controls

173 lines (146 loc) · 7.4 KB

Code style

The vast majority of base code style is enforced by ESLint and Prettier. The linter runs in a pre-commit hook on staged files with an auto-fix flag set to address any trivial issues.

In addition to the lint rules, please also where possible, stick to the contribution guidelines below. These rules should be kept in mind when reviewing Pull Requests.

Contribution guidelines

  • Code should be functional in style rather than Object Orientated or Imperative unless there are no clean alternatives.

    • Use pure functions where possible to make them testable and modular.
    • Avoid mutating variables and the let keyword.
    • Prefer functional & immutable array methods .ie map/filter/reduce/some/every over any types of mutable for loop.
    • Prefer "return early" coding style, more about it here.
    • Avoid classes and stateful modules where possible.
    • Don't Repeat Yourself. Make extensive use of the constants and utils files for re-usable strings and methods.
    • Don't obsess over performance of code, obsess over making it clear.
    • The above rules can be relaxed for test scripts.
  • React components should be simple and composable and cater to real life UI design problems:

    • Simplicity: Strive to keep the component API fairly simple and show real-world scenarios of using the component.

    • Representational: React components should be templates, free of logic, and purely presentational. It aims to make our components shareable and easy to test.

    • Composition: Break down components into smaller parts with minimal props to keep complexity low, and compose them together. This will ensure that the styles and functionality are flexible and extensible.

    • Accessibility: When creating a component, keep accessibility top of mind. This includes keyboard navigation, focus management, color contrast, voice over, and the correct aria-* attributes.

    • Naming Props: We all know naming is the hardest thing in this industry. Generally, ensure a prop name is indicative of what it does. Boolean props should be named using auxiliary verbs such as does, has, is and should. For example, Button uses isDisabled, isLoading, etc.

  • When doing styling, strive to use antd based components for consitentcy and painless maintenance.

    • Avoid hard coding colors. All colors from the design guideline should be pre-defined in variables.less. Please use them with respect.

    • When building grid layouts, try to avoid injecting styles via style attribute because it's impossible to build responsiveness with inline styles.

      // Don't
      ;<div style={{ diplay: 'flex' }}>
        <div style={{ width: '50%' }}>...</div>
        <div style={{ width: '50%' }}>...</div>
      </div>
      
      // Do
      import { Row, Col } from 'antd'
      
      ReactDOM.render(
        <>
          <Row>
            <Col span={12}>col-12</Col>
            <Col span={12}>col-12</Col>
          </Row>
          <Row>
            <Col span={8}>col-8</Col>
            <Col span={8}>col-8</Col>
            <Col span={8}>col-8</Col>
          </Row>
          <Row>
            <Col span={6}>col-6</Col>
            <Col span={6}>col-6</Col>
            <Col span={6}>col-6</Col>
            <Col span={6}>col-6</Col>
          </Row>
        </>,
        mountNode,
      )
  • Maintain the separation of concerns in the folder structure laid out in the initial scaffold:

    • constants/ contains shared variables used across the app.
    • components/ contains shared ui components used across the app.
    • utils/ contains shared functions used across the app.
    • hooks/ contains shared hooks used across the app.
    • types/ contains common TypeScript types or interfaces.
  • When adding third party libraries to the project consider carefully the following;

    • Is this a trivial package we can easily write in-house? If so, take the time to roll your own.
    • Is this library well supported, under active development and widely used in the community? If not, do not use it.
    • Do we use a similar but different library for the same task elsewhere in the company? If so, use this library for developer familiarity and consistency.
    • Will this library impact significantly performance or bundle size eg Lodash or Moment? If so, consider carefully if it is necessary or if there is a better alternative.
  • TypeScript helps write code better but should be used with care to not lose its strengths:

    • Avoid using any when possible. Using any is sometimes valid, but should rearely be used, even if to make quicker progress. Even Unknown is better than using any if you aren't sure of an input parameter.
    • Pay attention when using the non-null assertion operator !. Only use if you know that a variable cannot be null right now rather than blindly using it to pass the validation step.
  • File naming should be PascalCase for all React components files and kebab-case for the rest.

  • Exports from files can be either as variable or default exports, but please stick to naming the object before using export default to avoid anonymous module names in stack traces and React dev tools.

  • Your app should be fast but also remember "Premature optimization is the root of all evil". If you think it’s slow, prove it with a benchmark., the profiler of React Developer Tools (Chrome extension) is your friend! Once you find the root cause, try to follow the suggestions to fix it:

    • Use useMemo mostly just for expensive calculations.
    • Use React.memo, useMemo, and useCallback for reducing re-renders, they shouldn't have many dependencies and the dependencies should be mostly primitive types.
    • Make sure your React.memo, useCallback or useMemo is doing what you think it's doing (is it really preventing rerendering?).
    • Stop punching yourself every time you blink (fix slow renders before fixing rerenders).
    • Putting your state as close as possible to where it's being used will not only make your code so much easier to read but It would also make your app faster (state colocation).
    • Context should be logically separated, do not add to many values in one context provider. If any of the values of your context changes, all components consuming that context also rerenders even if those components don't use the specific value that was actually changed.
    • You can optimize context by separating the state and the dispatch function.
    • Know the terms lazy loading and bundle/code splitting
    • Window large lists (with tannerlinsley/react-virtual or similar).
    • A smaller bundle size usually also means a faster app. You can visualize the code bundles you've generated with @next/bundle-analyzer.

Read on: