Below is a brief summary of code standards we'd like to keep with this project. For the most part this follows the Angular Style Guide, but in a more concise way so you have a quick to read reference guide.
If you have an IDE that supports a .editorconfig file, please make sure you take the necessary steps to install those files. There's more info on specific IDE installations here.
This will help with a few things: maintaining 2 space indentations, ensuring there is a newline at the end of each file, and removing unnecessary whitespace at the end of a line.
If the HTML & CSS are fewer than 12 lines total, please keep the template and style in the .component.ts file. Otherwise, you should move one or both to separate .component.html or .component.scss file(s).
If there are more than 12 lines, but the component has little to no TypeScript functionality, then an exception can be made if code reviewers agree to it.
We have a git hook to run TSLint before pushing code to the repository. Please install this hook to your local machine by running the following command:
npm run install-hooks
The only limitation with TSLint currently is that it is not very good at indentation rules. Please make sure to comply with 2 space indents and to indent the appropriate number of times.
We follow the standard Angular naming and would like to stick to simple names for files and classes. Each name should follow the format:
<feature><sub-feature?><type>
For files the different parts should be separated with a dash, for class names we should use camel case. Here are some examples:
File: courses.component.ts
Class name: CoursesComponent
File: resources-review.service.ts
Class name: ResourcesReviewService
We suggest using max one sub-feature word to keep file and class names from getting too long.
Please don't use the following name for variable since it is used by our entrypoint script in our production docker container.
| Name | Note |
|---|---|
| planet-db-host | used in the docker script |
| planet-db-port | used in the docker script |
Planet uses the il8n translation module to translate our content and make it accessible. This module can be included in an html tag with the keyword il8n. For example, <p il8n>Text to be translated</p>.
If there are no strings enclosed by the tag, or if only interpolation strings are enclosed, then the il8n attribute should not be used as it will raise errors during compilation. For example, <p>{{interpolatedString}}</p> needs no il8n tag.
If text of an attribute needs to be translated an il8n translation attribute can be added by including a label with the format il8n-x where x represents the label that needs to be translated. For example, <img src="oleImg" il8n-title title="Example image" />.
For more information, see the docs here.
Within the src/app directory, each feature should have its own directory. Within that directory as we add sub-features, if there are more than 9 files we should create a sub-directory with the same naming convention as files (i.e. Resources Review would be in a resources-review directory).
The src/app/shared directory is intended for files which are used across different features. Rather than creating more directories in the src/app directory, we can store these files here to reduce the number of files & directories in the main app directory.
Component files should be named after their page/feature and follow Angular's standard naming pattern:
feature.component.ts // Component logic
feature.component.html // Component template
feature.component.scss // Component styles
We follow a minimalist approach to code comments:
- Keep comments sparse and meaningful
- Only add comments when logic is complex and not immediately obvious from the code itself
- Use clear, descriptive method and variable names instead of comments when possible
- For complex algorithms or business rules that may not be intuitive, include a brief explanation
- Avoid redundant comments
For CSS class names, keep them descriptive and concise. Feature or role of the class should be a prefix separated with a dash from the rest of the name.
All color variables should be defined in /src/app/_variables.scss:
- Use Material's theme system (
$primary,$accent,$warn) - Define semantic variables (e.g.,
$light-greynot$color-1) - Use Material's
mat-color()function to access theme colors - Never hardcode color values
Default breakpoint variables in _variables.scss:
$screen-md: 1000px; // Medium screen breakpoint
$screen-sm: 780px; // Small screen breakpoint
$screen-xs: 480px; // Extra small screen breakpointComponents can override these for specific needs using the screen-sizes mixin:
@include screen-sizes($screen-md: 1200px, $screen-sm: 780px);We would like to make our classes reusable across components when possible. When creating a new class, make sure to consider if this can be used across current or in development components. If so, please create the class in the styles.scss file.
If, in the future, the styles.scss file becomes so large it is difficult to manage, we will break it up into different files.
For unit tests it is easier to locate tags with a specific unit test class that has a prefix km-. These should not be used for any CSS styling. By limiting these to unit test use it allows people working on testing to remove unused km- classes knowing that they are not affecting the site in any way.
Please put variables in the /src/app/_variables.scss.
Please use specific test classes to query the HTML elements when testing. These can be added directly to the HTML template and should have the prefix km- to let people working on the SCSS know that this class is for testing only. By using specific test classes we can ensure consistent testing even as the CSS changes.
Use Angular Material's color system with these conventions for buttons:
- Default: Primary color (blue) for standard actions
- Use
mat-buttonormat-raised-button color="primary"
- Use
- Accent color (yellow) for attention-drawing actions
- Use
mat-buttonormat-raised-button color="accent"
- Use
- Warning color (red) for destructive actions
- Use
mat-buttonormat-raised-button color="warn"
- Use
- Grey (default with no color attribute) for secondary actions
- Use
mat-buttonwithout color attribute
- Use
- Disabled state automatically applies grey
- Use [disabled]= "condition"
Common patterns from our components:
- Dialog submit:
<button [disabled]="!linkForm.valid"> - Bulk actions:
<button [disabled]="!selection.selected.length"> - Clear filters:
<button [disabled]="courses.filter.trim() === '' && tagFilter.value.length === 0"> - Step progression:
<button [disabled]="stepNum > 0 && !step.isPreviousTestTaken">
When designing component toolbars, follow these patterns:
- Primary (first) toolbar:
- Use filter lists and search inputs directly in the toolbar
- White background (default mat-toolbar)
- Secondary toolbars:
- Use kebab menu (three vertical dots) to contain actions on tablet and mobile views
- Primary color background with white text (
primary-colorclass) - Use white icons and text for buttons (
mat-icon,font-size-1classes)
For consistent loading states across the application, follow these standards:
-
For page-level or component-level loading:
- Use simple text indicators with "Loading [resource]..." format
- Example:
<span i18n>Loading courses...</span> - Always include an
*ngIf="isLoading"condition with anelsetemplate for loaded content
-
For action-based loading (form submissions, data operations):
- Use
DialogsLoadingServiceto show a loading wheel - Start loading with
dialogsLoadingService.start() - Stop loading with
dialogsLoadingService.stop()in both success and error cases - Always use within a
finalize()operator in RxJS pipes to ensure loading stops
- Use
-
Loading state variables:
- Initialize as
isLoading = truewhen data fetching begins - Set to
falsewhen data loading completes
- Initialize as
When using icons in the application, follow these guidelines:
- Use Material icons (
mat-icon) whenever possible - Prefer opaque icons over transparent ones for better visibility
- Keep icon sizes consistent within similar UI elements
- Use icon colors that maintain sufficient contrast with the background
- For custom icons, ensure they match Material Design style guidelines
When creating dialog boxes, follow these button placement rules:
- Primary action buttons (right side):
- Submit/OK button should be rightmost
- Use mat-raised-button with primary color
- Secondary action buttons (left side):
- Cancel button should be immediately left of primary action
- Additional actions (if any) go to left of Cancel
- Use mat-button without color
- Labels:
- Use "Cancel" and "Submit" for form dialogs
- Use "Close" and "OK" for confirmation dialogs
For consistency in our UI text, follow these capitalization and punctuation rules:
- Buttons, Headers, and Titles:
- Use Title Case (Capitalize Each Word)
- Do not use periods
- Example: "Submit Request", "User Profile", "Course Management"
- Messages and regular content:
- Use Sentence case (Only first letter capitalized)
- Include periods at the end of complete sentences
- Example: "Your request has been submitted.", "Please enter valid credentials."
- Exception: Short status, placeholder, or label text doesn't need periods ("No results found", "Required field")
To maintain clean UI layouts and improve readability, follow these guidelines for truncating text:
- Use
TruncateTextPipein templates to truncate text dynamically. - Use the
truncateTextutility function in TypeScript for programmatic truncation. - Avoid hardcoded or redundant truncation logic.
For consistency in forms across the application, follow these standards:
- Form Field Labels:
- Use Title Case for field labels
- Do not use colons after labels
- Placeholder Text:
- Use Sentence case
- Be concise and descriptive
- Error Messages:
- Use Sentence case
- End with a period
- Be specific about the error
For form validation, refer to these standard validator implementations:
- Built-in validators: Angular's
Validatorsclass - Custom validators: Found in
/src/app/validators/custom-validators.ts- Common use cases: time, date, password matching, link validation
- Async validators: Found in
/src/app/validators/validator.service.ts- Common use cases: unique field checking, password verification