Events
For a web application to be interactive, there needs to be a way to respond to user events. This is done by registering callback functions in the JSX template.
const Counter = component$(() => {
  const store = useStore({ count: 0 });
  return (
    <button
      onClick$={() => store.count++}
    >
      {store.count}
    </button>
  );
});
In the above example, the onClick$ attribute of the <button> element is used to let Qwik know that a callback () => store.count++ should be executed whenever the click event is fired by the <button>.
Notice that onClick$ ends with $. This is a hint to both the Qwik Optimizer and the developer that a special transformation occurs at this location. The presence of the $ suffix implies a lazy-loaded boundary here. The code associated with the click handler will not download until the user triggers the click event. See Optimizer Rules for more details.
In the above example, the click listener is trivial in implementation. But in real applications, the listener may refer to complex code. By creating a lazy-loaded boundary, Qwik can tree-shake all of the code behind the click listener and delay its loading until the user clicks the button.
Events and Components
So far, we have discussed how one can listen to DOM events in JSX. A similar mechanism exists with components. Let's assume we have two kinds of buttons to aid the discussion. An HTML button (<button>) and a component button <CmpButton>.
Let's create an example component using both <button> and <CmpButton>.
const Counter = component$(() => {
  const store = useStore({
    htmlCount: 0,
    cmpCount: 0,
  });
  return (
    <>
      <button onClick$={() => store.htmlCount++}>{store.htmlCount}</button>
      <CmpButton onClick$={() => store.cmpCount++}>{store.cmpCount}</CmpButton>
    </>
  );
});
Notice that both <button> and <CmpButton> use the same syntax for registering events. However, the resulting HTML is a bit different.
<div q:host>
  <button q:obj="1" on:click="./chunk-a.js#Counter_button_onClick[0]">0</button>
  <div q:host>...</div>
</div>
- For the HTML <button>the resulting HTML is<button>along withon:clickattribute for the event.
- For the Component <CmpButton>the resulting HTML is<div q:host>...</div>.- See Host Element for explanation of why components need host element.
- Notice that because the <CmpButton>is an element, there is no correspondingon:clickin the DOM. This makes sense because, in this case, theonClickevent is something that<CmpButton>emits and should not be confused with the browser'sclickevent. (It is likely that the implementation of the<CmpButton>will have an internalon:clicklistener which forwards the event, but that is an implementation detail of<CmpButton>)
 
The main point here is that while the syntax of the events is consistent between HTML elements and Components, the resulting HTML only has on:<event> attributes for the DOM events, not for the component props.
Prevent default
Because of the async nature of Qwik, event's handler execution might be delayed because the implementation is not downloaded yet. This introduces a problem when the event's handler needs to prevent the default behavior of the event. Traditional event.preventDefault() will not work, instead use the qwik's preventdefault:{eventName} attribute:
const Counter = component$(() => {
  return (
    <a
      href="/about"
      preventdefault:click // This will prevent the default behavior of the "click" event.
      onClick$={(event) => {
        // PreventDefault will not work here, because handle is dispatched asynchronously.
        // event.preventDefault();
        singlePageNavigate('/about');
      }}
    >
      Go to about page
    </a>
  );
});
onWindow and onDocument
So far, we have discussed how to listen to events that originate at elements. There are events (for example, scroll and mousemove) that require that we listen to them on window or document. For this reason, Qwik allows for the onWindow and onDocument prefixes when listening for events.
const EventExample = component$(() => {
  const store = useStore({
    scroll: 0,
    mouse: { x: 0, y: 0 },
    clickCount: 0,
  });
  return (
    <button
      window:onScroll$={(e) => (store.scroll = window.scrollY)}
      document:onMouseMove$={(e) => {
        store.mouse.x = e.x;
        store.mouse.y = e.y;
      }}
      onClick$={() => store.clickCount++}
    >
      scroll: {store.scroll}
      mouseMove: {store.mouse.x}, {store.mouse.y}
      click: {store.clickCount}
    </button>
  );
});
The purpose of the onWindow/onDocument is to register the event at a current DOM location of the component but have it receive events from the window/document. There are two advantages to it:
- The events can be registered declaratively in your JSX
- The events get automatically cleaned up when the component is destroyed. (No explicit bookkeeping and cleanup is needed.)
Declaring Custom Component Events
So far, we have ignored the implementation detail of <CmpButton> because we wanted to talk about its usage only. Now let's look at how one declares a child component that can be used with events.
interface CmpButtonProps {
  onClickQrl?: QRL<() => void>;
}
const CmpButton = component$((props: CmpButtonProps) => {
  return (
    <button onDblclickQrl={props.onClickQrl}>
      <Slot />
    </button>
  );
});
As far as Qwik is concerned, passing events to a component is equivalent to passing props. In our example, we declare all props in CmpButtonProps interface. Specifically, notice onClickQrl?: QRL<() => void> declaration.
<CmpButton> would like to receive an onClick closure which it invokes at some later point in time. Qwik mandates that all props to a component need to be serializable. For this reason, we can't ask for onClick?: () => void. Instead, we need to ask for a serializable and lazy loadable version of the closure in the form of onClickQrl?: QRL<() => void>. QRL<() => void> can be read as lazy-loadable reference to a () => void closure.
On the usage side, when referring to the <CmpButton>, it would be a lot more convenient to pass in a closure rather than QRL of the closure. The translation from closure toQRL closure is what Qwik Optimizer performs for us. For this reason, the usage is in the format where the closure is inlined like so:
<CmpButton onClick$={() => store.cmpCount++}>{store.cmpCount}</CmpButton>
Here the prop is onClick$ rather than onClickQrl. We rely on the Qwik Optimizer to perform the translation. The above is roughly translated to:
<CmpButton onClickQrl={qrl('./chunk-a.js', 'Counter_onRender_CmpButton_onClick', [state]}>
  {store.cmpCount}
</CmpButton>
Assume: chunk-a.js:
export const Counter_onRender_CmpButton_onClick = () => {
  const [store] = useLexicalScope();
  store.cmpCount++;
};
Notice that:
- onClick$was translated to- onClickQrl.
- The closure () => store.cmpCount++was replaced byqrl('./chunk-a.js', 'Counter_onRender_CmpButton_onClick', [state].
- The closure was exported as Counter_onRender_CmpButton_onClick.
- A const [store] = useLexicalScope();was generated to restore closure state.
Also, what is not immediately apparent is that TypeScript generates this interface for <CmpButton> that allows usage of both properties depending on convenience:
interface CmpButtonProps {
  onClickQrl?: QRL<() => void>;
  onClick$?: () => void;
}
Notice that TypeScript automatically creates a correct prop with $ suffix, which generates the parameterized T of QRL<T>. In our case T is () => void. This type information makes sure that you correctly pass QRL to on<event>Qrl suffix and closures to on<event>$ suffix.
Working with QRLs
Let's look at a variation of <CmpButton> implementation. In this example, we would like to demonstrate working with <prop>Qrl vs <prop>$. For this reason, we have created an additional listener onClick$
interface CmpButtonProps {
  onClickQrl?: QRL<() => void>;
}
const CmpButton = component$((props: CmpButtonProps) => {
  return (
    <button
      onDblclickQrl={props.onClickQrl}
      onClick$={async () => {
        await (props.onClickQrl && props.onClickQrl.invoke());
        console.log('clicked');
      }}
    >
      <Slot />
    </button>
  );
});
Notice that we can pass the props.onClickQrl directly to the onDblclickQrl as seen on <button>. (see attribute onDblclickQrl={props.onClickQrl}.) This is because both the inputting prop onClickQrl as well as JSX prop onDblclickQrl are of type QRL<?> (and both have Qrl suffix.)
However, it is not possible to pass props.onClickQrl to onClick$ because the types don't match. (This would result in type error: onClick$={props.onClickQrl}.) Instead, the $ is reserved for inlined closures. In our example, we would like to print the console.log("clicked") after we process the props.onClickQrl callback. We can do so with the props.onClickQrl.invoke() method. This 1) lazy-loads the code, 2) restores the closure state, and 3) invokes the closure. The operation is asynchronous and therefore returns a promise, which we can resolve using the await statement.
Advanced: Events and qwikloader
In the above example, the onClick$ is placed on <button>. This means that the listener needs to be registered with the DOM. The registration of the listener creates two problems in the context of the SSR/SSG that Qwik needs to solve. (For context, remember that Qwik is resumable, that is, it can continue executing the application from where the server paused without being forced to download and execute code eagerly.)
- listener location: Qwik needs to know where the events are in the HTML which came from the SSR/SSG.
- listener code: Qwik needs to know what code should run if the event is triggered.
Without the above information, Qwik would be forced to download the component template and execute it so that the listener location and closure can be recovered. This process is known as hydration, and Qwik explicitly tries to avoid hydration.
Qwik serializes the event listeners into DOM in the form of QRLs. For the above example, the resulting HTML would look something like this:
<div q:host>
  <button q:obj="1" on:click="./chunk-a.js#Counter_button_onClick[0]">0</button>
</div>
The critical thing to notice is that Qwik generated an on:click attribute, containing the value ./chunk-a.js#Counter_button_onClick[0]. In the above example the on:click attribute solves the listener location problem, and the attribute value solves the listener code location problem. By serializing the listeners into the HTML Qwik-applications do not need to perform hydration on startup.
Qwikloader
For the browser to understand the on:click attribute syntax, a small JavaScript known as Qwikloader is needed. The Qwikloader is small (about 1kb) and fast (about 5ms) to execute. The Qwikloader is inlined into the HTML so that it can be executed quickly.
When a user interacts with the application, the browser fires relevant events that bubble up the DOM. At the root of the DOM, Qwikloader listens for the events and then tries to locate the corresponding on:<event> attribute. If such an attribute is found, then the value of the attribute is used to resolve the location where code can be downloaded from and then executed.
State recovery
const Counter = component$(() => {
  const store = useStore({ count: 0 });
  return <button onClick$={() => store.count++}>{store.count}</button>;
});
At first sight, it may appear that Qwik simply lazy loads the onClick$ function. But upon closer inspection, it is important to realize that Qwik lazy loads a closure rather than a function. (A closure is a function that lexically captures the state inside its variables. In other words, closures carry state, whereas functions do not.) The capturing of the state is what allows the Qwik application to simply resume where the server left off because the recovered closure carries the state of the application with it.
In our case, the onClick$ closure captures store. Capturing of store allows the application to increment the count property on click without having to re-run the whole application. Let's look at how closure capturing works in Qwik.
The HTML generated by the above code is something like this:
<div q:host>
  <button q:obj="1" on:click="./chunk-a.js#Counter_button_onClick[0]">0</button>
</div>
Notice that on:click attribute contains three pieces of information:
- ./chunk-a.js: The file which needs to be lazy-loaded.
- Counter_button_onClick: The symbol which needs to be retrieved from the lazy-loaded chunk.
- [0]: An array of lexically captured variable references (State of the closure).
In our case () => store.count++ only captures store, and hence it contains only a single reference 0. 0 is an index into the q:obj attribute that contains a reference to the actual serialized object referring to store. (The exact mechanisms and syntax is an implementation detail that can change at any time.)
Comparison to import()
JavaScript supports dynamic import(). At first glance, it may seem that the same can be achieved by import(), but there are a few differences worth mentioning.
Dynamic import():
- Is relative to the file that contains it. This works great for file-a.jstrying to loadfile-b.jsasimport('./file-b.js'). However, when the./file-a.jsgets serialized into HTML then we lose its relative nature. It is the framework that reads the./file-b.jsfrom HTML and performs theimport(). This means that all imports now become relative to the framework, which is incorrect.
- Requires that the developer writes import('./file-a.js'), which means the developer is in charge of deciding where the lazy-loaded boundaries are. This limits our ability of the tooling to move code around in an automated way.
- Supports import of top-level functions only which don't capture the state. This is the biggest difference. Qwik allows the imported symbol to be a closure that carries all of its state with it.