From JSDoc to TypeScript:
JSDoc: Structuring creativity
JavaScript, a language of freedom, where creativity knows no bounds 😆
Let's channel this creativity with JSDoc for clarity and guidance 🤓
# CHAPTER 1
What is JSDoc?
- JSDoc is a markup language for documenting JavaScript code.
- Allows generation of HTML documentation from source code comments.
- Supports tags to detail code's structure, purpose, and behavior.
- Enhances readability, maintainability, and collaboration among developers.
- Facilitates code insights in IDEs and supports type checking.
The Significance of JSDoc
Facilitating Onboarding
Accelerates the onboarding process for new developers, enabling them to grasp the functionality and architecture of large-scale JavaScript projects rapidly.
1.
2.
Enhancing Team Collaboration
Fosters better understanding and collaboration within development teams.
3.
Sustainable Development
Supports long-term project sustainability by making the codebase more navigable and easier to maintain, even as team members change over time.
# CHAPTER 1
// Normal Javascript Comment 1
/* Normal Javascript Comment 2 */
/**
JSDoc containing two asterisks
*/
# PRESENTING CODE
JSDoc comment
Two types of tags
# CHAPTER 1
-
Block Tags: Sit at the top level of a JSDoc comment. Start with
@
and are followed by a newline, except for the last block tag in a comment.
-
Inline Tags: Embedded within the text of a block tag or description. Begin with
@
and are enclosed in curly braces{}
. Use a backslash\
to escape curly braces within the tag text. Newlines are not required after inline tags.
/**
* @type {number}
*/
var FOO = 1
/**
* @const {number}
*/
const FOO = 1
/**
* @type {Array<string>}
*/
const colours = ['red', 'blue', 'green']
# JSDoc cheatsheet
Variables
/**
* This is a function.
*
* @param {string} n - A string param
* @param {string} [o] - A optional string param
* @param {string} [d=DefaultValue] - A optional string param
* @return {string} A good string
*
* @example
*
* foo('hello')
*/
function foo(n, o, d) {
return n
}
# JSDoc cheatsheet
Functions
# JSDoc cheatsheet
Types
/**
* A song
* @typedef {Object} Song
* @property {string} title - The title
* @property {string} artist - The artist
* @property {number} year - The year
*/
/**
* Plays a song
* @param {Song} song - The {@link Song} to be played
*/
function play(song) {}
# JSDoc cheatsheet
Typedef
/**
* A song
* @typedef {{
* title: string,
* artist: string,
* year: number
* }} Song
*/
/**
* Plays a song
* @param {Song} song - The {@link Song} to be played
*/
function play(song) {}
# JSDoc cheatsheet
Typedef Shorthand
/**
* @typedef {import('./Foo').default} Bar
*/
/**
* @param {Bar} x
*/
function test(x) {}
# JSDoc cheatsheet
Importing types
/**
* Adds two numbers
* @param {number} a - The first number
* @param {number} b - The second number
*/
function add(a, b) {
return a + b;
}
/**
* @param {number} a - The first number
* @param {number} b - The second number
* @description Adds two numbers
*/
function add(a, b) {
return a + b;
}
# JSDoc cheatsheet
Tag @description
// The @param tag is used to specify the name (required),
// type, and description of a function's parameter.
// The syntax is:
// @param {Type} Name - Description.
// The parameter type can be a native JavaScript type,
// and expressions can be used to indicate if a parameter
// is non-nullable or an array.
/**
* Greets a user
* @param {string} name - The user's name
*/
function sayHello(name) {
alert('Hello ' + name);
}
# JSDoc cheatsheet
Tag @param
// The @returns tag is used to describe
// the value that a function returns.
//
// The syntax is:
// @returns {Type} Description.
/**
* Adds two numbers
* @param {number} a - The first number
* @param {number} b - The second number
* @returns {number} The sum of numbers a and b
*/
function add(a, b) {
return a + b;
}
# JSDoc cheatsheet
Tag @returns
/**
* Divides two numbers
* @param {number} a - The numerator
* @param {number} b - The denominator, must not be 0
* @throws {Error} If the denominator is 0,
* an error is thrown indicating division by zero
* is not allowed.
* @returns {number} The result of the division a / b
*/
function divide(a, b) {
if (b === 0) {
throw new Error('Division by zero is not allowed.');
}
return a / b;
}
# JSDoc cheatsheet
Tag @throws
/**
* Fetches data from a URL asynchronously
* @async
* @param {string} url - The URL to fetch data from
* @returns {Promise<Object>} A promise resolving to
* the data fetched from the URL
*/
async function fetchData(url) {
const response = await fetch(url);
return response.json();
}
# JSDoc cheatsheet
Tag @async
/**
* Possible title for this talk
* @type {string}
* @author Exiliot [exiliot@example.com]
*/
const talkTitle = "Demystifying JSDoc"
# JSDoc cheatsheet
Tag @author
/**
* Sums of the square of two numbers a**2 + b**2
* @example <caption>How to use the sumSquares function</caption>
* // returns 13
* sumSquares(2, 3)
* @example
* // returns 41
* sumSquares(4, 5)
* // Typing the function
* @param {number} a - The first number
* @param {number} b - The second number
* @returns {Number} Returns the sum of the squares
* */
const sumSquares = function(a, b){
return a**2 + b**2
}
# JSDoc cheatsheet
Tag @example
/**
* @version 1.0.0
* @type {number}
**/
const meaningOfLife = 42
# JSDoc cheatsheet
Tag @version
/**
* How to use the link tags
* Also see the {@link https://jsdoc.app/tags-inline-link.html official docs} for more information
* @tutorial getting-started
**/
function myFunction (){
}
# JSDoc cheatsheet
Tag @link
Work with React
/**
* Represents a greeting component that displays a welcome message.
*
* @component
* @param {Object} props - The properties passed to the component.
* @param {string} props.name - The name of the person to greet.
* @returns {JSX.Element} A greeting message wrapped in a <div>.
*/
const Greeting = ({ name }) => <div>Hello, {name}!</div>;
# JSDoc cheatsheet
Functional component
import React from 'react';
/**
* @typedef Props
* @prop {React.ReactNode} title
* @prop {React.ReactNode} [desc]
* @prop {React.ReactNode} [selectFlightPrompt]
* @prop {()=>void} [onLogin]
*/
/**
* @param {Props} props
*/
const InflowHeader = ({ title, desc, selectFlightPrompt, onLogin }) => (
...
);
# JSDoc cheatsheet
Functional component
/**
* @typedef Props
* @prop {any} resource
*/
/**
* @typedef State
* @prop {boolean} isDrawerOpen
*/
# JSDoc cheatsheet
Class component
/**
* @extends Component<Props, State>
*/
class SampleContainer extends Component {
/** @type {State} */
state = {
isDrawerOpen: false,
};
constructor(props){
super(props);
}
render() {
return <>...</>;
}
}
# JSDoc cheatsheet
Class component
/**
* Custom hook for managing and toggling boolean state.
*
* @returns {[boolean, () => void]} Current state and
* a function to toggle the state.
*
* @example
* const [isOpen, toggleOpen] = useToggle(false);
*/
const useToggle = (initialState = false) => {
const [state, setState] = useState(initialState);
const toggle = () => setState(!state);
return [state, toggle];
};
# JSDoc cheatsheet
Hook
/**
* Custom hook for managing and toggling boolean state.
*
* @typedef {Object} HookResult - The result of the hook.
* @property {boolean} state - The current boolean state.
* @property {Function} toggle - Function to toggle the
* state.
*
* @param {boolean} [initialState=false] - The initial state.
* @returns {HookResult} Current state and a function to
* toggle the state.
*/
const useToggle = (initialState = false) => {
const [state, setState] = useState(initialState);
const toggle = () => setState(!state);
return [state, toggle];
};
# JSDoc cheatsheet
Hook
JSDoc + IntelliSense = ♥️
# JSDOC + IntelliSense
Typehinting in VSCode with JSDoc help
# JSDOC + IntelliSense
Using @typedef to define custom type definitions
JSDoc vs TypeScript
JSDoc and TypeScript both solve the problems with writing and maintaining plain JavaScript code.
However, they used different approaches, which have their benefits and drawbacks.
Benefits of JSDoc over TypeScript
Flexibility & Compatability
JSDoc being just JS comments, means it could be added to any JS codebase regardless of the language version, and it is not tied to a compiler like TypeScript is.
1.
2.
Code Annotation
JSDoc could be used for more than just type checking. It could be used to add more documentation, describe how functions work, and generate a documentationwebsite, all providing value to enhance code maintainability and understanding.
3.
No Compilation Step
TypeScript requires compilation to change the TypeScript code into JS so the engine can understand it, while JSDoc does not require any other step as they are just "comments", which is supported feature of JS itself.
# CHAPTER 2
Drawbacks of using JSDoc
Stronger Static Typing
TypeScript provides a strong model for types and catches these errors at compile time. Unlike JSDoc, where these typing end in the code itself and isn't enforced.
1.
2.
Type Inference
TypeScript can infer the type from its value. This helps reduce explicit type annotations and makes the codebase less verbose.
3.
Transpilation
TypeScript can adopt the latest and future features of the JS language with its polyfill feature. It effectively transpiles these codes down to understandable versions for browsers that would not have support for the features yet.
# CHAPTER 2
// TypeScript
function stringsStringStrings(p1: string, p2?: string, p3?: string, p4 = "test"): string {
// ...
}
// JSDoc
/**
* @param {string} p1
* @param {string=} p2
* @param {string} [p3]
* @param {string} [p4="test"]
* @return {string}
*/
function stringsStringStrings(p1, p2, p3, p4) {
// ...
}
# PRESENTING CODE
Readability
const thing = document.querySelector('#my-element')
addStuff(thing); // Error: type undefined is not assignable to type Thing
// or
const data = await fetch('https://my-url.dev');
return data.results // error: data is possibly undefined
// TypeScript
const thing = document.querySelector('#my-element')
addStuff(thing!); // although probably don't do this
addStuff(thing as Thing) // still not great but it works
// JSDoc
const thing = document.querySelector('#my-element')
addStuff(
// thing is definitely defined by this point, see line 120
// @ts-ignore-next-line
thing
);
# PRESENTING CODE
Flexibility
- Complementary Documentation
- Richer IDE Support
- Usage Examples
- Descriptive Annotations
JSDoc in TypeScript:
Enhancing Documentation
Improving DX with Better Tooling
# PRESENTING CODE
# PRESENTING CODE
- Highlight lines containing diagnostics
- Append diagnostic as text to the end of the line
- Show icons in gutter
- Show message in status bar
# PRESENTING CODE
The End :)
From JSDoc to TypeScript
By Nikita Malik
From JSDoc to TypeScript
- 169