🦉 Quick Overview 🦉

Owl components in an application are used to define a (dynamic) tree of components.

Root / \ A B / \ C D

State: each component can manage its own local state. It is a simple ES6 class, there are no special rules:

class Counter extends Component { static template = xml` <button t-on-click="increment"> Click Me! [<t t-esc="state.value"/>] </button>`; state = { value: 0 }; increment() { this.state.value++; this.render(); } }

The example above shows a component with a local state. Note that since there is nothing magical to the state object, we need to manually call the render function whenever we update it. This can quickly become annoying (and not efficient if we do it too much). There is a better way: using the useState hook, which transforms an object into a reactive version of itself:

const { useState } = owl.hooks; class Counter extends Component { static template = xml` <button t-on-click="increment"> Click Me! [<t t-esc="state.value"/>] </button>`; state = useState({ value: 0 }); increment() { this.state.value++; } }

Note that the t-on-click handler can even be replaced by an inline statement:

<button t-on-click="state.value++">

Props: sub components often needs some information from their parents. This is done by adding the required information to the template. This will then be accessible by the sub component in the props object. Note that there is an important rule here: the information contained in the props object is not owned by the sub component, and should never be modified.

class Child extends Component { static template = xml`<div>Hello <t t-esc="props.name"/></div>`; } class Parent extends Component { static template = xml` <div> <Child name="'Owl'" /> <Child name="'Framework'" /> </div>`; static components = { Child }; }

Communication: there are multiple ways to communicate information between components. However, the two most important ways are the following:

The following example illustrate both mechanisms:

class OrderLine extends Component { static template = xml` <div t-on-click="add"> <div><t t-esc="props.line.name"/></div> <div>Quantity: <t t-esc="props.line.quantity"/></div> </div>`; add() { this.trigger("add-to-order", { line: props.line }); } } class Parent extends Component { static template = xml` <div t-on-add-to-order="addToOrder"> <OrderLine t-foreach="orders" t-as="line" line="line" /> </div>`; static components = { OrderLine }; orders = useState([ { id: 1, name: "Coffee", quantity: 0 }, { id: 2, name: "Tea", quantity: 0 }, ]); addToOrder(event) { const line = event.detail.line; line.quantity++; } }

In this example, the OrderLine component trigger a add-to-order event. This will generate a DOM event which will bubble along the DOM tree. It will then be intercepted by the parent component, which will then get the line (from the detail key) and then increment its quantity. See the page on event handling for more details on how events work.

Note that this example would have also worked if the OrderLine component directly modifies the line object. However, this is not a good practice: this only works because the props object received by the child component is reactive, so the child component is then coupled to the parents implementation.