ESLint-Rules
Qwik comes with an own set of ESLint rules to help developers write better code.
โ Warn in 'recommended' ruleset
โ Error in 'recommended' ruleset
๐ Warn in 'strict' ruleset
๐ Error in 'strict' ruleset
Possible Problems
These rules are available.
use-method-usage
Object destructuring is not recommended for component$โ
๐
valid-lexical-scope
Used the tsc typechecker to detect the capture of unserializable data in dollar ($) scopes.โ
๐
loader-location
Detect declaration location of loader$โ
๐
no-react-props
Disallow usage of React-specific className
/htmlFor
props.โ
๐
prefer-classlist
Enforce using the classlist prop over importing a classnames helper. The classlist prop accepts an object { [class: string]: boolean }
just like classnames.โ
๐
jsx-no-script-url
Disallow javascript: URLs.โ
๐
jsx-key
Disallow missing key
props in iterators/collection literalsโ
๐
unused-server
Detect unused server$() functionsโ
๐
jsx-img
For performance reasons, always provide width and height attributes for <img> elements, it will help to prevent layout shifts.โ
๐
Details
use-method-usage
Object destructuring is not recommended for component$use-after-await
use-wrong-function
Examples of correct code for this rule:
export const Counter = component$(() => {
const count = useSignal(0);
});
Examples of incorrect code for this rule:
export const Counter = (() => {
const count = useSignal(0);
});
use*
methods can just be used in component$
functions.
export const Counter = (() => {
const count = useSignal(0);
});
use*
methods can just be used in component$
functions.
use-not-root
Examples of correct code for this rule:
export const Counter = component$(() => {
const count = useSignal(0);
});
Examples of incorrect code for this rule:
export const Counter = (() => {
const count = useSignal(0);
});
use*
methods can just be used in component$
functions.
export const Counter = (() => {
const count = useSignal(0);
});
use*
methods can just be used in component$
functions.
valid-lexical-scope
Used the tsc typechecker to detect the capture of unserializable data in dollar ($) scopes.referencesOutside
Examples of correct code for this rule:
import { component$, useTask$, $ } from '@builder.io/qwik';
export const HelloWorld = component$(() => {
const print = $((msg: string) => {
console.log(msg);
});
useTask$(() => {
print("Hello World");
});
return <h1>Hello</h1>;
});
Examples of incorrect code for this rule:
import { component$, useTask$ } from '@builder.io/qwik';
export const HelloWorld = component$(() => {
const print = (msg: string) => {
console.log(msg);
};
useTask$(() => {
print("Hello World");
});
return <h1>Hello</h1>;
});
Since Expressions are not serializable, they must be wrapped with $( ... )
so that the optimizer can split the code into small chunks.
invalidJsxDollar
Examples of correct code for this rule:
import { component$, $ } from '@builder.io/qwik';
export const HelloWorld = component$(() => {
const click = $(() => console.log());
return (
<button onClick$={click}>log it</button>
);
});
Examples of incorrect code for this rule:
import { component$ } from '@builder.io/qwik';
export const HelloWorld = component$(() => {
const click = () => console.log();
return (
<button onClick$={click}>log it</button>
);
});
Event handler must be wrapped with ${ ... }
.
mutableIdentifier
Examples of correct code for this rule:
import { component$ } from '@builder.io/qwik';
export const HelloWorld = component$(() => {
const person = { name: 'Bob' };
return (
<button onClick$={() => {
person.name = 'Alice';
}}>
{person.name}
</button>
);
});
Examples of incorrect code for this rule:
import { component$ } from '@builder.io/qwik';
export const HelloWorld = component$(() => {
let personName = 'Bob';
return (
<button onClick$={() => {
personName = 'Alice';
}}>
{personName}
</button>
);
});
Simple values are not allowed to be mutated. Use an Object instead and edit one of its property.
loader-location
Detect declaration location of loader$invalidLoaderLocation
Examples of correct code for this rule:
import { routeLoader$ } from '@builder.io/qwik-city';
export const useProductDetails = routeLoader$(async (requestEvent) => {
const res = await fetch(`https://.../products/${requestEvent.params.productId}`);
const product = await res.json();
return product as Product;
});
Examples of incorrect code for this rule:
import { routeLoader$ } from '@builder.io/qwik-city';
export const useProductDetails = routeLoader$(async (requestEvent) => {
const res = await fetch(`https://.../products/${requestEvent.params.productId}`);
const product = await res.json();
return product as Product;
});
This is not a valid location for a route loader. It only can be used inside the src/routes
folder, in a layout.tsx
or index.tsx
file.
missingExport
Examples of correct code for this rule:
import { routeLoader$ } from '@builder.io/qwik-city';
export const useProductDetails = routeLoader$(async (requestEvent) => {
const res = await fetch(`https://.../products/${requestEvent.params.productId}`);
const product = await res.json();
return product as Product;
});
Examples of incorrect code for this rule:
import { routeLoader$ } from '@builder.io/qwik-city';
const useProductDetails = routeLoader$(async (requestEvent) => {
const res = await fetch(`https://.../products/${requestEvent.params.productId}`);
const product = await res.json();
return product as Product;
});
The route loader function must be exported.
wrongName
Examples of correct code for this rule:
import { routeLoader$ } from '@builder.io/qwik-city';
export const useProductDetails = routeLoader$(async (requestEvent) => {
const res = await fetch(`https://.../products/${requestEvent.params.productId}`);
const product = await res.json();
return product as Product;
});
Examples of incorrect code for this rule:
import { routeLoader$ } from '@builder.io/qwik-city';
export const getProductDetails = routeLoader$(async (requestEvent) => {
const res = await fetch(`https://.../products/${requestEvent.params.productId}`);
const product = await res.json();
return product as Product;
});
The route loader function name must start with use
.
recommendedValue
Examples of correct code for this rule:
import { routeLoader$ } from '@builder.io/qwik-city';
export const useProductDetails = routeLoader$(async (requestEvent) => {
const res = await fetch(`https://.../products/${requestEvent.params.productId}`);
const product = await res.json();
return product as Product;
});
Examples of incorrect code for this rule:
import { routeLoader$ } from '@builder.io/qwik-city';
async function fetcher() {
const res = await fetch(`https://.../products/${requestEvent.params.productId}`);
const product = await res.json();
return product as Product;
}
export const useProductDetails = routeLoader$(fetcher);
It is recommended to inline the arrow function. This will help the optimizer make sure that no server code is leaked to the client build.
no-react-props
Disallow usage of React-specificclassName
/htmlFor
props.prefer
Examples of correct code for this rule:
<MyReactComponent class="foo" for="#password" />;
Examples of incorrect code for this rule:
<MyReactComponent className="foo" htmlFor="#password" />;
Prefer class
and for
props over className
and htmlFor
.
prefer-classlist
Enforce using the classlist prop over importing a classnames helper. The classlist prop accepts an object{ [class: string]: boolean }
just like classnames.preferClasslist
Examples of correct code for this rule:
import { component$ } from '@builder.io/qwik';
import styles from './MyComponent.module.css';
export default component$((props) => {
// Array syntax example
return <div class={[
styles.container,
'p-8',
props.isHighAttention ? 'text-green-500' : 'text-slate-500',
{ active: true}
]}>Hello world</div>;
// Object syntax example
return <div class={{
'text-green-500': props.isHighAttention,
'p-4': true
}}>Hello world</div>;
});
Examples of incorrect code for this rule:
import { component$ } from '@builder.io/qwik';
import classnames from 'classnames';
import styles from './MyComponent.module.css';
export default component$((props) => {
return <div class={classnames(
styles.container,
'p-8',
{
'text-green-500' : props.isHighAttention,
'text-slate-500' : !props.isHighAttention,
},
{ active: true}
)}>Hello world</div>;
});
The class prop should be used instead of any 3rd party lib to efficiently set classes based on an object.
jsx-no-script-url
Disallow javascript: URLs.noJSURL
Examples of correct code for this rule:
<button onClick$={() => alert('open the door please')>ring</button>
Examples of incorrect code for this rule:
<button onClick$="javascript:alert('open the door please')">ring</button>
jsx-key
Disallow missingkey
props in iterators/collection literalsmissingIterKey
Examples of correct code for this rule:
import { component$ } from '@builder.io/qwik';
export const Person = component$(() => {
const person = {
firstName: 'John',
lastName: 'Doe',
age: 32,
}
return (
<ul>
{Object.keys(person).map((color) => (
<li key={`person-${key}`}>{person[key]}</li>
)}
</ul>
);
});
Examples of incorrect code for this rule:
import { component$ } from '@builder.io/qwik';
export const Person = component$(() => {
const person = {
firstName: 'John',
lastName: 'Doe',
age: 32,
}
return (
<ul>
{Object.keys(person).map((color) => (
<li>{person[key]}</li>
)}
</ul>
);
});
Missing key
prop for element in iterator.
missingIterKeyUsePrag
Examples of correct code for this rule:
import { component$ } from '@builder.io/qwik';
import Card from './Card';
import Summary from './Summary';
export const Person = component$(() => {
const person = {
firstName: 'John',
lastName: 'Doe',
age: 32,
}
return (
{Object.keys(person).map((color) => (
<Fragment key={`person-${key}`}>
<Card value={person[key]} />
<Summary value={person[key]} />
</Fragment>
)}
);
});
Examples of incorrect code for this rule:
import { component$ } from '@builder.io/qwik';
import Card from './Card';
import Summary from './Summary';
export const Person = component$(() => {
const person = {
firstName: 'John',
lastName: 'Doe',
age: 32,
}
return (
{Object.keys(person).map((color) => (
< key={`person-${key}`}>
<Card value={person[key]} />
<Summary value={person[key]} />
</>
)}
);
});
Missing key
prop for element in iterator. The key prop allows for improved rendering performance. Shorthand fragment syntax does not support providing keys. Use <Fragment>
instead
missingArrayKey
Examples of correct code for this rule:
import { component$ } from '@builder.io/qwik';
export const ColorList = component$(() => {
const colors = ['red', 'green', 'blue'];
return (
<ul>
{colors.map((color) => (
<li key={`color-${color}`}>{color}</li>
)}
</ul>
);
});
Examples of incorrect code for this rule:
import { component$ } from '@builder.io/qwik';
export const ColorList = component$(() => {
const colors = ['red', 'green', 'blue'];
return (
<ul>
{colors.map((color) => (
<li>{color}</li>
)}
</ul>
);
});
Missing key
prop for element in array. The key prop allows for improved rendering performance.
missingArrayKeyUsePrag
Examples of correct code for this rule:
import { component$, Fragment } from '@builder.io/qwik';
export const ColorList = component$(() => {
const colors = ['red', 'green', 'blue'];
return (
{colors.map((color) => (
<Fragment key={`color-${color}`}>
<h2>{color}</h2>
<p>The color "${color}" is a great color.</p>
</Fragment>
)}
);
});
Examples of incorrect code for this rule:
import { component$ } from '@builder.io/qwik';
export const ColorList = component$(() => {
const colors = ['red', 'green', 'blue'];
return (
{colors.map((color) => (
< key={`color-${color}`}>
<h2>{color}</h2>
<p>The color "${color}" is a great color.</p>
</>
)}
);
});
Missing key
prop for element in array. The key prop allows for improved rendering performance. Shorthand fragment syntax does not support providing keys. Use <Fragment>
instead
nonUniqueKeys
Examples of correct code for this rule:
import { component$ } from '@builder.io/qwik';
export const ColorList = component$(() => {
const colors = ['red', 'green', 'blue'];
return (
<ul>
{colors.map((color) => (
<li key={`color-${color}`}>{color}</li>
)}
</ul>
);
});
Examples of incorrect code for this rule:
import { component$ } from '@builder.io/qwik';
export const ColorList = component$(() => {
const colors = ['red', 'green', 'blue'];
return (
<ul>
{colors.map((color) => (
<li key="not-a-good-idea">{color}</li>
)}
</ul>
);
});
The key
prop must be unique.
unused-server
Detect unused server$() functionsunusedServer
Examples of correct code for this rule:
import { component$ } from '@builder.io/qwik';
import { server$ } from '@builder.io/qwik-city';
const serverGreeter = server$((firstName: string, lastName: string) => {
const greeting = `Hello ${firstName} ${lastName}`;
return greeting;
});
export default component$(() => (
<button
onClick$={async () => {
const greeting = await serverGreeter('John', 'Doe');
alert(greeting);
}}
>
greet
</button>
);
);
Examples of incorrect code for this rule:
import { component$ } from '@builder.io/qwik';
import { server$ } from '@builder.io/qwik-city';
const serverGreeter = server$((firstName: string, lastName: string) => {
const greeting = `Hello ${firstName} ${lastName}`;
return greeting;
});
export default component$(() => (
<button
onClick$={async () => {
const greeting = 'not using the server$ function';
alert(greeting);
}}
>
greet
</button>
);
);
A server$
function is declared, but never used.
jsx-img
For performance reasons, always provide width and height attributes for <img> elements, it will help to prevent layout shifts.noLocalSrc
Examples of correct code for this rule:
import Image from '~/media/image.png';
<Image />
Examples of incorrect code for this rule:
<img src="/image.png">
Serving images from public are not optimized, nor cached. Import images using ESM instead.
noWidthHeight
Examples of correct code for this rule:
<img width="200" height="600" src="/static/images/portrait-01.webp">
Examples of incorrect code for this rule:
<img src="/static/images/portrait-01.webp">
For performance reasons, always provide width and height attributes for <img>
elements, it will help to prevent layout shifts.