guide to creating components
This commit is contained in:
		@@ -12,7 +12,7 @@ This question is best answered by tracing what happens when a user (you!) runs `
 | 
				
			|||||||
2. This file has a [shebang](<https://en.wikipedia.org/wiki/Shebang_(Unix)>) line at the top which tells npm to execute it using Node.
 | 
					2. This file has a [shebang](<https://en.wikipedia.org/wiki/Shebang_(Unix)>) line at the top which tells npm to execute it using Node.
 | 
				
			||||||
3. `bootstrap-cli.mjs` is responsible for a few things:
 | 
					3. `bootstrap-cli.mjs` is responsible for a few things:
 | 
				
			||||||
   1. Parsing the command-line arguments using [yargs](http://yargs.js.org/).
 | 
					   1. Parsing the command-line arguments using [yargs](http://yargs.js.org/).
 | 
				
			||||||
   2. Transpiling and bundling the rest of Quartz (which is in Typescript) to regular JavaScript using [esbuild](https://esbuild.github.io/). The `esbuild` configuration here is slightly special as it also handles `.scss` file imports using [esbuild-sass-plugin v2](https://www.npmjs.com/package/esbuild-sass-plugin). Additionally, we bundle 'inline' client-side scripts (any `.inline.ts` file) that components declare usiong a custom `esbuild` plugin that runs another instance of `esbuild` that bundles for the browser instead of `node`. Modules of both types are imported as plain text.
 | 
					   2. Transpiling and bundling the rest of Quartz (which is in Typescript) to regular JavaScript using [esbuild](https://esbuild.github.io/). The `esbuild` configuration here is slightly special as it also handles `.scss` file imports using [esbuild-sass-plugin v2](https://www.npmjs.com/package/esbuild-sass-plugin). Additionally, we bundle 'inline' client-side scripts (any `.inline.ts` file) that components declare using a custom `esbuild` plugin that runs another instance of `esbuild` that bundles for the browser instead of `node`. Modules of both types are imported as plain text.
 | 
				
			||||||
   3. Running the local preview server if `--serve` is set. This starts two servers:
 | 
					   3. Running the local preview server if `--serve` is set. This starts two servers:
 | 
				
			||||||
      1. A WebSocket server on port 3001 to handle hot-reload signals. This tracks all inbound connections and sends a 'rebuild' message a server-side change is detected (either content or configuration).
 | 
					      1. A WebSocket server on port 3001 to handle hot-reload signals. This tracks all inbound connections and sends a 'rebuild' message a server-side change is detected (either content or configuration).
 | 
				
			||||||
      2. An HTTP file-server on a user defined port (normally 8080) to serve the actual website files.
 | 
					      2. An HTTP file-server on a user defined port (normally 8080) to serve the actual website files.
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -20,9 +20,6 @@ However, HTML doesn't let you create reusable templates. If you wanted to create
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
In effect, components allow you to write a JavaScript function that takes some data and produces HTML as an output. **While Quartz doesn't use React, it uses the same component concept to allow you to easily express layout templates in your Quartz site.**
 | 
					In effect, components allow you to write a JavaScript function that takes some data and produces HTML as an output. **While Quartz doesn't use React, it uses the same component concept to allow you to easily express layout templates in your Quartz site.**
 | 
				
			||||||
 | 
					
 | 
				
			||||||
> [!hint]
 | 
					 | 
				
			||||||
> For those coming from React, Quartz components are different from React components in that it only uses JSX for templating and layout. Hooks like `useEffect`, `useState`, etc. are not rendered.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
## An Example Component
 | 
					## An Example Component
 | 
				
			||||||
 | 
					
 | 
				
			||||||
### Constructor
 | 
					### Constructor
 | 
				
			||||||
@@ -90,11 +87,11 @@ Note that inlined styles **must** be plain vanilla CSS.
 | 
				
			|||||||
```tsx {6-10} title="quartz/components/YourComponent.tsx"
 | 
					```tsx {6-10} title="quartz/components/YourComponent.tsx"
 | 
				
			||||||
export default (() => {
 | 
					export default (() => {
 | 
				
			||||||
  function YourComponent() {
 | 
					  function YourComponent() {
 | 
				
			||||||
    return <p>Example Component</p>
 | 
					    return <p class="red-text">Example Component</p>
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  YourComponent.css = `
 | 
					  YourComponent.css = `
 | 
				
			||||||
  p {
 | 
					  p.red-text {
 | 
				
			||||||
    color: red;
 | 
					    color: red;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
  `
 | 
					  `
 | 
				
			||||||
@@ -124,14 +121,114 @@ export default (() => {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
### Scripts and Interactivity
 | 
					### Scripts and Interactivity
 | 
				
			||||||
 | 
					
 | 
				
			||||||
- listening for the nav event
 | 
					What about interactivity? Suppose you want to add an-click handler for example. Like the `.css` property on the component, you can also declare `.beforeDOMLoaded` and `.afterDOMLoaded` properties that are strings that contain the script.
 | 
				
			||||||
  - best practice: anything here should unmount any existing event handlers to prevent memory leaks
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```tsx title="quartz/components/YourComponent.tsx"
 | 
				
			||||||
 | 
					export default (() => {
 | 
				
			||||||
 | 
					  function YourComponent() {
 | 
				
			||||||
 | 
					    return <button id="btn">Click me</button>
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  YourComponent.beforeDOM = `
 | 
				
			||||||
 | 
					  console.log("hello from before the page loads!")
 | 
				
			||||||
 | 
					  `
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  YourComponent.afterDOM = `
 | 
				
			||||||
 | 
					  document.getElementById('btn').onclick = () => {
 | 
				
			||||||
 | 
					    alert('button clicked!')
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  `
 | 
				
			||||||
 | 
					  return YourComponent
 | 
				
			||||||
 | 
					}) satisfies QuartzComponentConstructor
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					> [!hint]
 | 
				
			||||||
 | 
					> For those coming from React, Quartz components are different from React components in that it only uses JSX for templating and layout. Hooks like `useEffect`, `useState`, etc. are not rendered and other properties that accept functions like `onClick` handlers will not work. Instead, do it using a regular JS script that modifies the DOM element directly.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					As the names suggest, the `.beforeDOMLoaded` scripts are executed *before* the page is done loading so it doesn't have access to any elements on the page. This is mostly used to prefetch any critical data.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The `.afterDOMLoaded` script executes once the page has been completely loaded. This is a good place to setup anything that should last for the duration of a site visit (e.g. getting something saved from local storage).
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					If you need to create an `afterDOMLoaded` script that depends on *page specific* elements that may change when navigating to a new page, you can listen for the `"nav"` event that gets fired whenever a page loads (which may happen on navigation if [[SPA Routing]] is enabled).
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```ts
 | 
				
			||||||
 | 
					document.addEventListener("nav", () => {
 | 
				
			||||||
 | 
					  // do page specific logic here
 | 
				
			||||||
 | 
					  // e.g. attach event listeners
 | 
				
			||||||
 | 
					  const toggleSwitch = document.querySelector("#switch") as HTMLInputElement
 | 
				
			||||||
 | 
					  toggleSwitch.removeEventListener("change", switchTheme)
 | 
				
			||||||
 | 
					  toggleSwitch.addEventListener("change", switchTheme)
 | 
				
			||||||
 | 
					})
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					It is best practice to also unmount any existing event handlers to prevent memory leaks.
 | 
				
			||||||
 | 
					#### Importing Code
 | 
				
			||||||
 | 
					Of course, it isn't always practical (nor desired!) to write your code as a string literal in the component.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Quartz supports importing component code through `.inline.ts` files.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```tsx title="quartz/components/YourComponent.tsx"
 | 
				
			||||||
 | 
					// @ts-ignore: typescript doesn't know about our inline bundling system
 | 
				
			||||||
 | 
					// so we need to silence the error
 | 
				
			||||||
 | 
					import script from "./scripts/graph.inline"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default (() => {
 | 
				
			||||||
 | 
					  function YourComponent() {
 | 
				
			||||||
 | 
					    return <button id="btn">Click me</button>
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  YourComponent.afterDOM = script
 | 
				
			||||||
 | 
					  return YourComponent
 | 
				
			||||||
 | 
					}) satisfies QuartzComponentConstructor
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```ts title="quartz/components/scripts/graph.inline.ts"
 | 
				
			||||||
 | 
					// any imports here are bundled for the browser
 | 
				
			||||||
 | 
					import * as d3 from "d3"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					document.getElementById('btn').onclick = () => {
 | 
				
			||||||
 | 
					  alert('button clicked!')
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Additionally, like what is shown in the example above, you can import packages in `.inline.ts` files. This will be bundled by Quartz and included in the actual script.
 | 
				
			||||||
### Using a Component 
 | 
					### Using a Component 
 | 
				
			||||||
 | 
					After creating your custom component, re-export it in `quartz/components/index.ts`:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#### In a layout
 | 
					```ts title="quartz/components/index.ts" {4,10}
 | 
				
			||||||
 | 
					import ArticleTitle from "./ArticleTitle"
 | 
				
			||||||
 | 
					import Content from "./pages/Content"
 | 
				
			||||||
 | 
					import Darkmode from "./Darkmode"
 | 
				
			||||||
 | 
					import YourComponent from "./YourComponent"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#### In the configuration
 | 
					export {
 | 
				
			||||||
 | 
					  ArticleTitle,
 | 
				
			||||||
 | 
					  Content,
 | 
				
			||||||
 | 
					  Darkmode,
 | 
				
			||||||
 | 
					  YourComponent
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Then, you can use it like any other component in `quartz.layout.ts` via `Component.YourComponent()`. See the [[configuration#Layout|layout]] section for more details.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					As Quartz components are just functions that return React components, you can compositionally use them in other Quartz components.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```tsx title="quartz/components/AnotherComponent.tsx"
 | 
				
			||||||
 | 
					import YourComponent from "./YourComponent"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default (() => {
 | 
				
			||||||
 | 
					  function AnotherComponent(props: QuartzComponentProps) {
 | 
				
			||||||
 | 
					    return <div>
 | 
				
			||||||
 | 
						    <p>It's nested!</p>
 | 
				
			||||||
 | 
						    <YourComponent {...props} />
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  return AnotherComponent
 | 
				
			||||||
 | 
					}) satisfies QuartzComponentConstructor
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
> [!hint]
 | 
					> [!hint]
 | 
				
			||||||
> Look in `quartz/components` for more examples of components in Quartz as reference for your own components!
 | 
					> Look in `quartz/components` for more examples of components in Quartz as reference for your own components!
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user