diff --git a/_config.yml b/_config.yml
index a54aa29..2490b6a 100644
--- a/_config.yml
+++ b/_config.yml
@@ -160,6 +160,12 @@ defaults:
values:
layout: "network"
category: "network"
+ -
+ scope:
+ type: "roles"
+ values:
+ layout: "role"
+ category: "roles"
#liquid:
@@ -199,7 +205,7 @@ ignore_theme_config: true
#
header_pages:
- about.html
- - posts.html
+ - roles.html
- networks.html
#
diff --git a/_layouts/role.html b/_layouts/role.html
new file mode 100644
index 0000000..c489629
--- /dev/null
+++ b/_layouts/role.html
@@ -0,0 +1,48 @@
+---
+layout: default
+---
+
+
+
+
+
+
+ {{ page.title | escape }}
+
+
{{ page.description | escape }}
+
+
+
+
+
+
+
Attributes
+
+
+
+ Title |
+ Value |
+
+
+
+
+ Cost |
+ {{ page.cost | upcase }} |
+
+
+ Throughput |
+ {{ page.throughput | upcase }} |
+
+
+
+
+
+
+
+
diff --git a/_roles/ceo.html b/_roles/ceo.html
index 92af860..4a4a536 100644
--- a/_roles/ceo.html
+++ b/_roles/ceo.html
@@ -1,7 +1,8 @@
---
-layout: page
title: Chief Executive Officer
+abbreviation: CEO
permalink: /roles/ceo/
----
-TODO
+cost: 400.0
+throughput: 400000.0
+---
diff --git a/_roles/cto.html b/_roles/cto.html
index fd56ded..01f00c6 100644
--- a/_roles/cto.html
+++ b/_roles/cto.html
@@ -1,7 +1,8 @@
---
-layout: page
title: Chief Technology Officer
+abbreviation: CTO
permalink: /roles/cto/
----
-TODO
+cost: 400.0
+throughput: 40000.0
+---
diff --git a/_roles/it.html b/_roles/it.html
index b5eb35e..4b2bfbf 100644
--- a/_roles/it.html
+++ b/_roles/it.html
@@ -1,7 +1,8 @@
---
-layout: page
-title: IT Support
+title: Information Technology Support
+abbreviation: IT
permalink: /roles/it/
----
-TODO
+cost: 400.0
+throughput: 40.0
+---
diff --git a/_roles/spo.html b/_roles/spo.html
index b014cca..7153999 100644
--- a/_roles/spo.html
+++ b/_roles/spo.html
@@ -1,8 +1,8 @@
---
-layout: page
title: Stake Pool Operator
+abbreviation: SPO
permalink: /roles/spo/
----
-
-TODO
+cost: 400.0
+throughput: 5.0
+---
diff --git a/assets/js/calculator.js b/assets/js/calculator.js
new file mode 100644
index 0000000..4cf3b29
--- /dev/null
+++ b/assets/js/calculator.js
@@ -0,0 +1,145 @@
+(function() {
+ const { Component } = owl;
+ const { xml } = owl.tags;
+ const { whenReady } = owl.utils;
+
+ const { useRef, useSubEnv } = owl.hooks;
+
+ // -------------------------------------------------------------------------
+ // Model
+ // -------------------------------------------------------------------------
+ class TaskModel extends owl.core.EventBus {
+ nextId = 1
+ tasks = [];
+
+ constructor(tasks) {
+ super()
+ for (let task of tasks) {
+ this.tasks.push(task);
+ this.nextId = Math.max(this.nextId, task.id + 1);
+ }
+ }
+
+ addTask(title) {
+ const newTask = {
+ id: this.nextId++,
+ title: title,
+ isCompleted: false,
+ };
+ this.tasks.unshift(newTask);
+ this.trigger('update');
+ }
+
+ toggleTask(id) {
+ const task = this.tasks.find(t => t.id === id);
+ task.isCompleted = !task.isCompleted;
+ this.tasks.sort(function(a, b) {
+ if (a.isCompleted) {
+ if (b.isCompleted) {
+ a.title.localeCompare(b.title)
+ } else {
+ return 1;
+ }
+ } else {
+ if (b.isCompleted) {
+ return -1;
+ } else {
+ a.title.localeCompare(b.title)
+ }
+ }
+ });
+ this.trigger('update')
+ }
+
+ deleteTask(id) {
+ const index = this.tasks.findIndex(t => t.id === id);
+ this.tasks.splice(index, 1);
+ this.trigger('update');
+ }
+ }
+
+ class StoredTaskModel extends TaskModel {
+ constructor(storage) {
+ const tasks = storage.getItem("todoapp");
+ super(tasks ? JSON.parse(tasks) : []);
+ this.on('update', this, () => {
+ storage.setItem("todoapp", JSON.stringify(this.tasks))
+ });
+ }
+ }
+
+ // -------------------------------------------------------------------------
+ // Task Component
+ // -------------------------------------------------------------------------
+ const TASK_TEMPLATE = xml /* xml */`
+
+
+
+ đź—‘
+
`;
+
+ class Task extends Component {
+ static template = TASK_TEMPLATE;
+
+ toggleTask() {
+ this.env.model.toggleTask(this.props.task.id);
+ }
+
+ deleteTask() {
+ this.env.model.deleteTask(this.props.task.id);
+ }
+ }
+
+ // -------------------------------------------------------------------------
+ // App Component
+ // -------------------------------------------------------------------------
+ const APP_TEMPLATE = xml /* xml */`
+ `;
+
+ class App extends Component {
+ static template = APP_TEMPLATE;
+ static components = { Task };
+
+ inputRef = useRef("add-input");
+
+ constructor() {
+ super();
+
+ const model = new StoredTaskModel(this.env.localStorage);
+ model.on('update', this, this.render);
+ useSubEnv({ model });
+ }
+
+ mounted() {
+ this.inputRef.el.focus();
+ }
+
+ addTask(ev) {
+ // 13 is keycode for ENTER
+ if (ev.keyCode === 13) {
+ const title = ev.target.value.trim();
+ ev.target.value = "";
+ if (title) {
+ this.env.model.addTask(title);
+ }
+ }
+ }
+
+ }
+
+ // Setup code
+ function setup() {
+ App.env.localStorage = window.localStorage;
+ const app = new App();
+ app.mount(document.getElementById('calculatorBody'));
+ }
+
+ whenReady(setup);
+})();
diff --git a/assets/js/owl.js b/assets/js/owl.js
new file mode 100644
index 0000000..3cbd68b
--- /dev/null
+++ b/assets/js/owl.js
@@ -0,0 +1,5227 @@
+(function (exports) {
+ 'use strict';
+
+ /**
+ * We define here a simple event bus: it can
+ * - emit events
+ * - add/remove listeners.
+ *
+ * This is a useful pattern of communication in many cases. For OWL, each
+ * components and stores are event buses.
+ */
+ //------------------------------------------------------------------------------
+ // EventBus
+ //------------------------------------------------------------------------------
+ class EventBus {
+ constructor() {
+ this.subscriptions = {};
+ }
+ /**
+ * Add a listener for the 'eventType' events.
+ *
+ * Note that the 'owner' of this event can be anything, but will more likely
+ * be a component or a class. The idea is that the callback will be called with
+ * the proper owner bound.
+ *
+ * Also, the owner should be kind of unique. This will be used to remove the
+ * listener.
+ */
+ on(eventType, owner, callback) {
+ if (!callback) {
+ throw new Error("Missing callback");
+ }
+ if (!this.subscriptions[eventType]) {
+ this.subscriptions[eventType] = [];
+ }
+ this.subscriptions[eventType].push({
+ owner,
+ callback,
+ });
+ }
+ /**
+ * Remove a listener
+ */
+ off(eventType, owner) {
+ const subs = this.subscriptions[eventType];
+ if (subs) {
+ this.subscriptions[eventType] = subs.filter((s) => s.owner !== owner);
+ }
+ }
+ /**
+ * Emit an event of type 'eventType'. Any extra arguments will be passed to
+ * the listeners callback.
+ */
+ trigger(eventType, ...args) {
+ const subs = this.subscriptions[eventType] || [];
+ for (let i = 0, iLen = subs.length; i < iLen; i++) {
+ const sub = subs[i];
+ sub.callback.call(sub.owner, ...args);
+ }
+ }
+ /**
+ * Remove all subscriptions.
+ */
+ clear() {
+ this.subscriptions = {};
+ }
+ }
+
+ /**
+ * Owl Observer
+ *
+ * This code contains the logic that allows Owl to observe and react to state
+ * changes.
+ *
+ * This is a Observer class that can observe any JS values. The way it works
+ * can be summarized thusly:
+ * - primitive values are not observed at all
+ * - Objects and arrays are observed by replacing them with a Proxy
+ * - each object/array metadata are tracked in a weakmap, and keep a revision
+ * number
+ *
+ * Note that this code is loosely inspired by Vue.
+ */
+ //------------------------------------------------------------------------------
+ // Observer
+ //------------------------------------------------------------------------------
+ class Observer {
+ constructor() {
+ this.rev = 1;
+ this.allowMutations = true;
+ this.weakMap = new WeakMap();
+ }
+ notifyCB() { }
+ observe(value, parent) {
+ if (value === null || typeof value !== "object" || value instanceof Date) {
+ // fun fact: typeof null === 'object'
+ return value;
+ }
+ let metadata = this.weakMap.get(value) || this._observe(value, parent);
+ return metadata.proxy;
+ }
+ revNumber(value) {
+ const metadata = this.weakMap.get(value);
+ return metadata ? metadata.rev : 0;
+ }
+ _observe(value, parent) {
+ var self = this;
+ const proxy = new Proxy(value, {
+ get(target, k) {
+ const targetValue = target[k];
+ return self.observe(targetValue, value);
+ },
+ set(target, key, newVal) {
+ const value = target[key];
+ if (newVal !== value) {
+ if (!self.allowMutations) {
+ throw new Error(`Observed state cannot be changed here! (key: "${key}", val: "${newVal}")`);
+ }
+ self._updateRevNumber(target);
+ target[key] = newVal;
+ self.notifyCB();
+ }
+ return true;
+ },
+ deleteProperty(target, key) {
+ if (key in target) {
+ delete target[key];
+ self._updateRevNumber(target);
+ self.notifyCB();
+ }
+ return true;
+ },
+ });
+ const metadata = {
+ value,
+ proxy,
+ rev: this.rev,
+ parent,
+ };
+ this.weakMap.set(value, metadata);
+ this.weakMap.set(metadata.proxy, metadata);
+ return metadata;
+ }
+ _updateRevNumber(target) {
+ this.rev++;
+ let metadata = this.weakMap.get(target);
+ let parent = target;
+ do {
+ metadata = this.weakMap.get(parent);
+ metadata.rev++;
+ } while ((parent = metadata.parent) && parent !== target);
+ }
+ }
+
+ //------------------------------------------------------------------------------
+ // module/props.ts
+ //------------------------------------------------------------------------------
+ function updateProps(oldVnode, vnode) {
+ var key, cur, old, elm = vnode.elm, oldProps = oldVnode.data.props, props = vnode.data.props;
+ if (!oldProps && !props)
+ return;
+ if (oldProps === props)
+ return;
+ oldProps = oldProps || {};
+ props = props || {};
+ for (key in oldProps) {
+ if (!props[key]) {
+ delete elm[key];
+ }
+ }
+ for (key in props) {
+ cur = props[key];
+ old = oldProps[key];
+ if (old !== cur && (key !== "value" || elm[key] !== cur)) {
+ elm[key] = cur;
+ }
+ }
+ }
+ const propsModule = {
+ create: updateProps,
+ update: updateProps,
+ };
+ //------------------------------------------------------------------------------
+ // module/eventlisteners.ts
+ //------------------------------------------------------------------------------
+ function invokeHandler(handler, vnode, event) {
+ if (typeof handler === "function") {
+ // call function handler
+ handler.call(vnode, event, vnode);
+ }
+ else if (typeof handler === "object") {
+ // call handler with arguments
+ if (typeof handler[0] === "function") {
+ // special case for single argument for performance
+ if (handler.length === 2) {
+ handler[0].call(vnode, handler[1], event, vnode);
+ }
+ else {
+ var args = handler.slice(1);
+ args.push(event);
+ args.push(vnode);
+ handler[0].apply(vnode, args);
+ }
+ }
+ else {
+ // call multiple handlers
+ for (let i = 0, iLen = handler.length; i < iLen; i++) {
+ invokeHandler(handler[i], vnode, event);
+ }
+ }
+ }
+ }
+ function handleEvent(event, vnode) {
+ var name = event.type, on = vnode.data.on;
+ // call event handler(s) if exists
+ if (on) {
+ if (on[name]) {
+ invokeHandler(on[name], vnode, event);
+ }
+ else if (on["!" + name]) {
+ invokeHandler(on["!" + name], vnode, event);
+ }
+ }
+ }
+ function createListener() {
+ return function handler(event) {
+ handleEvent(event, handler.vnode);
+ };
+ }
+ function updateEventListeners(oldVnode, vnode) {
+ var oldOn = oldVnode.data.on, oldListener = oldVnode.listener, oldElm = oldVnode.elm, on = vnode && vnode.data.on, elm = (vnode && vnode.elm), name;
+ // optimization for reused immutable handlers
+ if (oldOn === on) {
+ return;
+ }
+ // remove existing listeners which no longer used
+ if (oldOn && oldListener) {
+ // if element changed or deleted we remove all existing listeners unconditionally
+ if (!on) {
+ for (name in oldOn) {
+ // remove listener if element was changed or existing listeners removed
+ const capture = name.charAt(0) === "!";
+ name = capture ? name.slice(1) : name;
+ oldElm.removeEventListener(name, oldListener, capture);
+ }
+ }
+ else {
+ for (name in oldOn) {
+ // remove listener if existing listener removed
+ if (!on[name]) {
+ const capture = name.charAt(0) === "!";
+ name = capture ? name.slice(1) : name;
+ oldElm.removeEventListener(name, oldListener, capture);
+ }
+ }
+ }
+ }
+ // add new listeners which has not already attached
+ if (on) {
+ // reuse existing listener or create new
+ var listener = (vnode.listener = oldVnode.listener || createListener());
+ // update vnode for listener
+ listener.vnode = vnode;
+ // if element changed or added we add all needed listeners unconditionally
+ if (!oldOn) {
+ for (name in on) {
+ // add listener if element was changed or new listeners added
+ const capture = name.charAt(0) === "!";
+ name = capture ? name.slice(1) : name;
+ elm.addEventListener(name, listener, capture);
+ }
+ }
+ else {
+ for (name in on) {
+ // add listener if new listener added
+ if (!oldOn[name]) {
+ const capture = name.charAt(0) === "!";
+ name = capture ? name.slice(1) : name;
+ elm.addEventListener(name, listener, capture);
+ }
+ }
+ }
+ }
+ }
+ const eventListenersModule = {
+ create: updateEventListeners,
+ update: updateEventListeners,
+ destroy: updateEventListeners,
+ };
+ //------------------------------------------------------------------------------
+ // attributes.ts
+ //------------------------------------------------------------------------------
+ const xlinkNS = "http://www.w3.org/1999/xlink";
+ const xmlNS = "http://www.w3.org/XML/1998/namespace";
+ const colonChar = 58;
+ const xChar = 120;
+ function updateAttrs(oldVnode, vnode) {
+ var key, elm = vnode.elm, oldAttrs = oldVnode.data.attrs, attrs = vnode.data.attrs;
+ if (!oldAttrs && !attrs)
+ return;
+ if (oldAttrs === attrs)
+ return;
+ oldAttrs = oldAttrs || {};
+ attrs = attrs || {};
+ // update modified attributes, add new attributes
+ for (key in attrs) {
+ const cur = attrs[key];
+ const old = oldAttrs[key];
+ if (old !== cur) {
+ if (cur === true) {
+ elm.setAttribute(key, "");
+ }
+ else if (cur === false) {
+ elm.removeAttribute(key);
+ }
+ else {
+ if (key.charCodeAt(0) !== xChar) {
+ elm.setAttribute(key, cur);
+ }
+ else if (key.charCodeAt(3) === colonChar) {
+ // Assume xml namespace
+ elm.setAttributeNS(xmlNS, key, cur);
+ }
+ else if (key.charCodeAt(5) === colonChar) {
+ // Assume xlink namespace
+ elm.setAttributeNS(xlinkNS, key, cur);
+ }
+ else {
+ elm.setAttribute(key, cur);
+ }
+ }
+ }
+ }
+ // remove removed attributes
+ // use `in` operator since the previous `for` iteration uses it (.i.e. add even attributes with undefined value)
+ // the other option is to remove all attributes with value == undefined
+ for (key in oldAttrs) {
+ if (!(key in attrs)) {
+ elm.removeAttribute(key);
+ }
+ }
+ }
+ const attrsModule = {
+ create: updateAttrs,
+ update: updateAttrs,
+ };
+ //------------------------------------------------------------------------------
+ // class.ts
+ //------------------------------------------------------------------------------
+ function updateClass(oldVnode, vnode) {
+ var cur, name, elm, oldClass = oldVnode.data.class, klass = vnode.data.class;
+ if (!oldClass && !klass)
+ return;
+ if (oldClass === klass)
+ return;
+ oldClass = oldClass || {};
+ klass = klass || {};
+ elm = vnode.elm;
+ for (name in oldClass) {
+ if (name && !klass[name]) {
+ elm.classList.remove(name);
+ }
+ }
+ for (name in klass) {
+ cur = klass[name];
+ if (cur !== oldClass[name]) {
+ elm.classList[cur ? "add" : "remove"](name);
+ }
+ }
+ }
+ const classModule = { create: updateClass, update: updateClass };
+
+ /**
+ * Owl VDOM
+ *
+ * This file contains an implementation of a virtual DOM, which is a system that
+ * can generate in-memory representations of a DOM tree, compare them, and
+ * eventually change a concrete DOM tree to match its representation, in an
+ * hopefully efficient way.
+ *
+ * Note that this code is a fork of Snabbdom, slightly tweaked/optimized for our
+ * needs (see https://github.com/snabbdom/snabbdom).
+ *
+ * The main exported values are:
+ * - interface VNode
+ * - h function (a helper function to generate a vnode)
+ * - patch function (to apply a vnode to an actual DOM node)
+ */
+ function vnode(sel, data, children, text, elm) {
+ let key = data === undefined ? undefined : data.key;
+ return { sel, data, children, text, elm, key };
+ }
+ //------------------------------------------------------------------------------
+ // snabbdom.ts
+ //------------------------------------------------------------------------------
+ function isUndef(s) {
+ return s === undefined;
+ }
+ function isDef(s) {
+ return s !== undefined;
+ }
+ const emptyNode = vnode("", {}, [], undefined, undefined);
+ function sameVnode(vnode1, vnode2) {
+ return vnode1.key === vnode2.key && vnode1.sel === vnode2.sel;
+ }
+ function isVnode(vnode) {
+ return vnode.sel !== undefined;
+ }
+ function createKeyToOldIdx(children, beginIdx, endIdx) {
+ let i, map = {}, key, ch;
+ for (i = beginIdx; i <= endIdx; ++i) {
+ ch = children[i];
+ if (ch != null) {
+ key = ch.key;
+ if (key !== undefined)
+ map[key] = i;
+ }
+ }
+ return map;
+ }
+ const hooks = ["create", "update", "remove", "destroy", "pre", "post"];
+ function init(modules, domApi) {
+ let i, j, cbs = {};
+ const api = domApi !== undefined ? domApi : htmlDomApi;
+ for (i = 0; i < hooks.length; ++i) {
+ cbs[hooks[i]] = [];
+ for (j = 0; j < modules.length; ++j) {
+ const hook = modules[j][hooks[i]];
+ if (hook !== undefined) {
+ cbs[hooks[i]].push(hook);
+ }
+ }
+ }
+ function emptyNodeAt(elm) {
+ const id = elm.id ? "#" + elm.id : "";
+ const c = elm.className ? "." + elm.className.split(" ").join(".") : "";
+ return vnode(api.tagName(elm).toLowerCase() + id + c, {}, [], undefined, elm);
+ }
+ function createRmCb(childElm, listeners) {
+ return function rmCb() {
+ if (--listeners === 0) {
+ const parent = api.parentNode(childElm);
+ api.removeChild(parent, childElm);
+ }
+ };
+ }
+ function createElm(vnode, insertedVnodeQueue) {
+ let i, iLen, data = vnode.data;
+ if (data !== undefined) {
+ if (isDef((i = data.hook)) && isDef((i = i.init))) {
+ i(vnode);
+ data = vnode.data;
+ }
+ }
+ let children = vnode.children, sel = vnode.sel;
+ if (sel === "!") {
+ if (isUndef(vnode.text)) {
+ vnode.text = "";
+ }
+ vnode.elm = api.createComment(vnode.text);
+ }
+ else if (sel !== undefined) {
+ const elm = vnode.elm ||
+ (vnode.elm =
+ isDef(data) && isDef((i = data.ns))
+ ? api.createElementNS(i, sel)
+ : api.createElement(sel));
+ for (i = 0, iLen = cbs.create.length; i < iLen; ++i)
+ cbs.create[i](emptyNode, vnode);
+ if (array(children)) {
+ for (i = 0, iLen = children.length; i < iLen; ++i) {
+ const ch = children[i];
+ if (ch != null) {
+ api.appendChild(elm, createElm(ch, insertedVnodeQueue));
+ }
+ }
+ }
+ else if (primitive(vnode.text)) {
+ api.appendChild(elm, api.createTextNode(vnode.text));
+ }
+ i = vnode.data.hook; // Reuse variable
+ if (isDef(i)) {
+ if (i.create)
+ i.create(emptyNode, vnode);
+ if (i.insert)
+ insertedVnodeQueue.push(vnode);
+ }
+ }
+ else {
+ vnode.elm = api.createTextNode(vnode.text);
+ }
+ return vnode.elm;
+ }
+ function addVnodes(parentElm, before, vnodes, startIdx, endIdx, insertedVnodeQueue) {
+ for (; startIdx <= endIdx; ++startIdx) {
+ const ch = vnodes[startIdx];
+ if (ch != null) {
+ api.insertBefore(parentElm, createElm(ch, insertedVnodeQueue), before);
+ }
+ }
+ }
+ function invokeDestroyHook(vnode) {
+ let i, iLen, j, jLen, data = vnode.data;
+ if (data !== undefined) {
+ if (isDef((i = data.hook)) && isDef((i = i.destroy)))
+ i(vnode);
+ for (i = 0, iLen = cbs.destroy.length; i < iLen; ++i)
+ cbs.destroy[i](vnode);
+ if (vnode.children !== undefined) {
+ for (j = 0, jLen = vnode.children.length; j < jLen; ++j) {
+ i = vnode.children[j];
+ if (i != null && typeof i !== "string") {
+ invokeDestroyHook(i);
+ }
+ }
+ }
+ }
+ }
+ function removeVnodes(parentElm, vnodes, startIdx, endIdx) {
+ for (; startIdx <= endIdx; ++startIdx) {
+ let i, iLen, listeners, rm, ch = vnodes[startIdx];
+ if (ch != null) {
+ if (isDef(ch.sel)) {
+ invokeDestroyHook(ch);
+ listeners = cbs.remove.length + 1;
+ rm = createRmCb(ch.elm, listeners);
+ for (i = 0, iLen = cbs.remove.length; i < iLen; ++i)
+ cbs.remove[i](ch, rm);
+ if (isDef((i = ch.data)) && isDef((i = i.hook)) && isDef((i = i.remove))) {
+ i(ch, rm);
+ }
+ else {
+ rm();
+ }
+ }
+ else {
+ // Text node
+ api.removeChild(parentElm, ch.elm);
+ }
+ }
+ }
+ }
+ function updateChildren(parentElm, oldCh, newCh, insertedVnodeQueue) {
+ let oldStartIdx = 0, newStartIdx = 0;
+ let oldEndIdx = oldCh.length - 1;
+ let oldStartVnode = oldCh[0];
+ let oldEndVnode = oldCh[oldEndIdx];
+ let newEndIdx = newCh.length - 1;
+ let newStartVnode = newCh[0];
+ let newEndVnode = newCh[newEndIdx];
+ let oldKeyToIdx;
+ let idxInOld;
+ let elmToMove;
+ let before;
+ while (oldStartIdx <= oldEndIdx && newStartIdx <= newEndIdx) {
+ if (oldStartVnode == null) {
+ oldStartVnode = oldCh[++oldStartIdx]; // Vnode might have been moved left
+ }
+ else if (oldEndVnode == null) {
+ oldEndVnode = oldCh[--oldEndIdx];
+ }
+ else if (newStartVnode == null) {
+ newStartVnode = newCh[++newStartIdx];
+ }
+ else if (newEndVnode == null) {
+ newEndVnode = newCh[--newEndIdx];
+ }
+ else if (sameVnode(oldStartVnode, newStartVnode)) {
+ patchVnode(oldStartVnode, newStartVnode, insertedVnodeQueue);
+ oldStartVnode = oldCh[++oldStartIdx];
+ newStartVnode = newCh[++newStartIdx];
+ }
+ else if (sameVnode(oldEndVnode, newEndVnode)) {
+ patchVnode(oldEndVnode, newEndVnode, insertedVnodeQueue);
+ oldEndVnode = oldCh[--oldEndIdx];
+ newEndVnode = newCh[--newEndIdx];
+ }
+ else if (sameVnode(oldStartVnode, newEndVnode)) {
+ // Vnode moved right
+ patchVnode(oldStartVnode, newEndVnode, insertedVnodeQueue);
+ api.insertBefore(parentElm, oldStartVnode.elm, api.nextSibling(oldEndVnode.elm));
+ oldStartVnode = oldCh[++oldStartIdx];
+ newEndVnode = newCh[--newEndIdx];
+ }
+ else if (sameVnode(oldEndVnode, newStartVnode)) {
+ // Vnode moved left
+ patchVnode(oldEndVnode, newStartVnode, insertedVnodeQueue);
+ api.insertBefore(parentElm, oldEndVnode.elm, oldStartVnode.elm);
+ oldEndVnode = oldCh[--oldEndIdx];
+ newStartVnode = newCh[++newStartIdx];
+ }
+ else {
+ if (oldKeyToIdx === undefined) {
+ oldKeyToIdx = createKeyToOldIdx(oldCh, oldStartIdx, oldEndIdx);
+ }
+ idxInOld = oldKeyToIdx[newStartVnode.key];
+ if (isUndef(idxInOld)) {
+ // New element
+ api.insertBefore(parentElm, createElm(newStartVnode, insertedVnodeQueue), oldStartVnode.elm);
+ newStartVnode = newCh[++newStartIdx];
+ }
+ else {
+ elmToMove = oldCh[idxInOld];
+ if (elmToMove.sel !== newStartVnode.sel) {
+ api.insertBefore(parentElm, createElm(newStartVnode, insertedVnodeQueue), oldStartVnode.elm);
+ }
+ else {
+ patchVnode(elmToMove, newStartVnode, insertedVnodeQueue);
+ oldCh[idxInOld] = undefined;
+ api.insertBefore(parentElm, elmToMove.elm, oldStartVnode.elm);
+ }
+ newStartVnode = newCh[++newStartIdx];
+ }
+ }
+ }
+ if (oldStartIdx <= oldEndIdx || newStartIdx <= newEndIdx) {
+ if (oldStartIdx > oldEndIdx) {
+ before = newCh[newEndIdx + 1] == null ? null : newCh[newEndIdx + 1].elm;
+ addVnodes(parentElm, before, newCh, newStartIdx, newEndIdx, insertedVnodeQueue);
+ }
+ else {
+ removeVnodes(parentElm, oldCh, oldStartIdx, oldEndIdx);
+ }
+ }
+ }
+ function patchVnode(oldVnode, vnode, insertedVnodeQueue) {
+ let i, iLen, hook;
+ if (isDef((i = vnode.data)) && isDef((hook = i.hook)) && isDef((i = hook.prepatch))) {
+ i(oldVnode, vnode);
+ }
+ const elm = (vnode.elm = oldVnode.elm);
+ let oldCh = oldVnode.children;
+ let ch = vnode.children;
+ if (oldVnode === vnode)
+ return;
+ if (vnode.data !== undefined) {
+ for (i = 0, iLen = cbs.update.length; i < iLen; ++i)
+ cbs.update[i](oldVnode, vnode);
+ i = vnode.data.hook;
+ if (isDef(i) && isDef((i = i.update)))
+ i(oldVnode, vnode);
+ }
+ if (isUndef(vnode.text)) {
+ if (isDef(oldCh) && isDef(ch)) {
+ if (oldCh !== ch)
+ updateChildren(elm, oldCh, ch, insertedVnodeQueue);
+ }
+ else if (isDef(ch)) {
+ if (isDef(oldVnode.text))
+ api.setTextContent(elm, "");
+ addVnodes(elm, null, ch, 0, ch.length - 1, insertedVnodeQueue);
+ }
+ else if (isDef(oldCh)) {
+ removeVnodes(elm, oldCh, 0, oldCh.length - 1);
+ }
+ else if (isDef(oldVnode.text)) {
+ api.setTextContent(elm, "");
+ }
+ }
+ else if (oldVnode.text !== vnode.text) {
+ if (isDef(oldCh)) {
+ removeVnodes(elm, oldCh, 0, oldCh.length - 1);
+ }
+ api.setTextContent(elm, vnode.text);
+ }
+ if (isDef(hook) && isDef((i = hook.postpatch))) {
+ i(oldVnode, vnode);
+ }
+ }
+ return function patch(oldVnode, vnode) {
+ let i, iLen, elm, parent;
+ const insertedVnodeQueue = [];
+ for (i = 0, iLen = cbs.pre.length; i < iLen; ++i)
+ cbs.pre[i]();
+ if (!isVnode(oldVnode)) {
+ oldVnode = emptyNodeAt(oldVnode);
+ }
+ if (sameVnode(oldVnode, vnode)) {
+ patchVnode(oldVnode, vnode, insertedVnodeQueue);
+ }
+ else {
+ elm = oldVnode.elm;
+ parent = api.parentNode(elm);
+ createElm(vnode, insertedVnodeQueue);
+ if (parent !== null) {
+ api.insertBefore(parent, vnode.elm, api.nextSibling(elm));
+ removeVnodes(parent, [oldVnode], 0, 0);
+ }
+ }
+ for (i = 0, iLen = insertedVnodeQueue.length; i < iLen; ++i) {
+ insertedVnodeQueue[i].data.hook.insert(insertedVnodeQueue[i]);
+ }
+ for (i = 0, iLen = cbs.post.length; i < iLen; ++i)
+ cbs.post[i]();
+ return vnode;
+ };
+ }
+ //------------------------------------------------------------------------------
+ // is.ts
+ //------------------------------------------------------------------------------
+ const array = Array.isArray;
+ function primitive(s) {
+ return typeof s === "string" || typeof s === "number";
+ }
+ function createElement(tagName) {
+ return document.createElement(tagName);
+ }
+ function createElementNS(namespaceURI, qualifiedName) {
+ return document.createElementNS(namespaceURI, qualifiedName);
+ }
+ function createTextNode(text) {
+ return document.createTextNode(text);
+ }
+ function createComment(text) {
+ return document.createComment(text);
+ }
+ function insertBefore(parentNode, newNode, referenceNode) {
+ parentNode.insertBefore(newNode, referenceNode);
+ }
+ function removeChild(node, child) {
+ node.removeChild(child);
+ }
+ function appendChild(node, child) {
+ node.appendChild(child);
+ }
+ function parentNode(node) {
+ return node.parentNode;
+ }
+ function nextSibling(node) {
+ return node.nextSibling;
+ }
+ function tagName(elm) {
+ return elm.tagName;
+ }
+ function setTextContent(node, text) {
+ node.textContent = text;
+ }
+ const htmlDomApi = {
+ createElement,
+ createElementNS,
+ createTextNode,
+ createComment,
+ insertBefore,
+ removeChild,
+ appendChild,
+ parentNode,
+ nextSibling,
+ tagName,
+ setTextContent,
+ };
+ function addNS(data, children, sel) {
+ if (sel === "dummy") {
+ // we do not need to add the namespace on dummy elements, they come from a
+ // subcomponent, which will handle the namespace itself
+ return;
+ }
+ data.ns = "http://www.w3.org/2000/svg";
+ if (sel !== "foreignObject" && children !== undefined) {
+ for (let i = 0, iLen = children.length; i < iLen; ++i) {
+ const child = children[i];
+ let childData = child.data;
+ if (childData !== undefined) {
+ addNS(childData, child.children, child.sel);
+ }
+ }
+ }
+ }
+ function h(sel, b, c) {
+ var data = {}, children, text, i, iLen;
+ if (c !== undefined) {
+ data = b;
+ if (array(c)) {
+ children = c;
+ }
+ else if (primitive(c)) {
+ text = c;
+ }
+ else if (c && c.sel) {
+ children = [c];
+ }
+ }
+ else if (b !== undefined) {
+ if (array(b)) {
+ children = b;
+ }
+ else if (primitive(b)) {
+ text = b;
+ }
+ else if (b && b.sel) {
+ children = [b];
+ }
+ else {
+ data = b;
+ }
+ }
+ if (children !== undefined) {
+ for (i = 0, iLen = children.length; i < iLen; ++i) {
+ if (primitive(children[i]))
+ children[i] = vnode(undefined, undefined, undefined, children[i], undefined);
+ }
+ }
+ return vnode(sel, data, children, text, undefined);
+ }
+
+ const patch = init([eventListenersModule, attrsModule, propsModule, classModule]);
+
+ /**
+ * Owl QWeb Expression Parser
+ *
+ * Owl needs in various contexts to be able to understand the structure of a
+ * string representing a javascript expression. The usual goal is to be able
+ * to rewrite some variables. For example, if a template has
+ *
+ * ```xml
+ * ...
+ * ```
+ *
+ * this needs to be translated in something like this:
+ *
+ * ```js
+ * if (context["computeSomething"]({val: context["state"].val})) { ... }
+ * ```
+ *
+ * This file contains the implementation of an extremely naive tokenizer/parser
+ * and evaluator for javascript expressions. The supported grammar is basically
+ * only expressive enough to understand the shape of objects, of arrays, and
+ * various operators.
+ */
+ //------------------------------------------------------------------------------
+ // Misc types, constants and helpers
+ //------------------------------------------------------------------------------
+ const RESERVED_WORDS = "true,false,NaN,null,undefined,debugger,console,window,in,instanceof,new,function,return,this,eval,void,Math,RegExp,Array,Object,Date".split(",");
+ const WORD_REPLACEMENT = {
+ and: "&&",
+ or: "||",
+ gt: ">",
+ gte: ">=",
+ lt: "<",
+ lte: "<=",
+ };
+ const STATIC_TOKEN_MAP = {
+ "{": "LEFT_BRACE",
+ "}": "RIGHT_BRACE",
+ "[": "LEFT_BRACKET",
+ "]": "RIGHT_BRACKET",
+ ":": "COLON",
+ ",": "COMMA",
+ "(": "LEFT_PAREN",
+ ")": "RIGHT_PAREN",
+ };
+ // note that the space after typeof is relevant. It makes sure that the formatted
+ // expression has a space after typeof
+ const OPERATORS = "...,.,===,==,+,!==,!=,!,||,&&,>=,>,<=,<,?,-,*,/,%,typeof ,=>,=,;,in ".split(",");
+ let tokenizeString = function (expr) {
+ let s = expr[0];
+ let start = s;
+ if (s !== "'" && s !== '"') {
+ return false;
+ }
+ let i = 1;
+ let cur;
+ while (expr[i] && expr[i] !== start) {
+ cur = expr[i];
+ s += cur;
+ if (cur === "\\") {
+ i++;
+ cur = expr[i];
+ if (!cur) {
+ throw new Error("Invalid expression");
+ }
+ s += cur;
+ }
+ i++;
+ }
+ if (expr[i] !== start) {
+ throw new Error("Invalid expression");
+ }
+ s += start;
+ return { type: "VALUE", value: s };
+ };
+ let tokenizeNumber = function (expr) {
+ let s = expr[0];
+ if (s && s.match(/[0-9]/)) {
+ let i = 1;
+ while (expr[i] && expr[i].match(/[0-9]|\./)) {
+ s += expr[i];
+ i++;
+ }
+ return { type: "VALUE", value: s };
+ }
+ else {
+ return false;
+ }
+ };
+ let tokenizeSymbol = function (expr) {
+ let s = expr[0];
+ if (s && s.match(/[a-zA-Z_\$]/)) {
+ let i = 1;
+ while (expr[i] && expr[i].match(/\w/)) {
+ s += expr[i];
+ i++;
+ }
+ if (s in WORD_REPLACEMENT) {
+ return { type: "OPERATOR", value: WORD_REPLACEMENT[s], size: s.length };
+ }
+ return { type: "SYMBOL", value: s };
+ }
+ else {
+ return false;
+ }
+ };
+ const tokenizeStatic = function (expr) {
+ const char = expr[0];
+ if (char && char in STATIC_TOKEN_MAP) {
+ return { type: STATIC_TOKEN_MAP[char], value: char };
+ }
+ return false;
+ };
+ const tokenizeOperator = function (expr) {
+ for (let op of OPERATORS) {
+ if (expr.startsWith(op)) {
+ return { type: "OPERATOR", value: op };
+ }
+ }
+ return false;
+ };
+ const TOKENIZERS = [
+ tokenizeString,
+ tokenizeNumber,
+ tokenizeOperator,
+ tokenizeSymbol,
+ tokenizeStatic,
+ ];
+ /**
+ * Convert a javascript expression (as a string) into a list of tokens. For
+ * example: `tokenize("1 + b")` will return:
+ * ```js
+ * [
+ * {type: "VALUE", value: "1"},
+ * {type: "OPERATOR", value: "+"},
+ * {type: "SYMBOL", value: "b"}
+ * ]
+ * ```
+ */
+ function tokenize(expr) {
+ const result = [];
+ let token = true;
+ while (token) {
+ expr = expr.trim();
+ if (expr) {
+ for (let tokenizer of TOKENIZERS) {
+ token = tokenizer(expr);
+ if (token) {
+ result.push(token);
+ expr = expr.slice(token.size || token.value.length);
+ break;
+ }
+ }
+ }
+ else {
+ token = false;
+ }
+ }
+ if (expr.length) {
+ throw new Error(`Tokenizer error: could not tokenize "${expr}"`);
+ }
+ return result;
+ }
+ //------------------------------------------------------------------------------
+ // Expression "evaluator"
+ //------------------------------------------------------------------------------
+ /**
+ * This is the main function exported by this file. This is the code that will
+ * process an expression (given as a string) and returns another expression with
+ * proper lookups in the context.
+ *
+ * Usually, this kind of code would be very simple to do if we had an AST (so,
+ * if we had a javascript parser), since then, we would only need to find the
+ * variables and replace them. However, a parser is more complicated, and there
+ * are no standard builtin parser API.
+ *
+ * Since this method is applied to simple javasript expressions, and the work to
+ * be done is actually quite simple, we actually can get away with not using a
+ * parser, which helps with the code size.
+ *
+ * Here is the heuristic used by this method to determine if a token is a
+ * variable:
+ * - by default, all symbols are considered a variable
+ * - unless the previous token is a dot (in that case, this is a property: `a.b`)
+ * - or if the previous token is a left brace or a comma, and the next token is
+ * a colon (in that case, this is an object key: `{a: b}`)
+ *
+ * Some specific code is also required to support arrow functions. If we detect
+ * the arrow operator, then we add the current (or some previous tokens) token to
+ * the list of variables so it does not get replaced by a lookup in the context
+ */
+ function compileExprToArray(expr, scope) {
+ scope = Object.create(scope);
+ const tokens = tokenize(expr);
+ for (let i = 0; i < tokens.length; i++) {
+ let token = tokens[i];
+ let prevToken = tokens[i - 1];
+ let nextToken = tokens[i + 1];
+ let isVar = token.type === "SYMBOL" && !RESERVED_WORDS.includes(token.value);
+ if (token.type === "SYMBOL" && !RESERVED_WORDS.includes(token.value)) {
+ if (prevToken) {
+ if (prevToken.type === "OPERATOR" && prevToken.value === ".") {
+ isVar = false;
+ }
+ else if (prevToken.type === "LEFT_BRACE" || prevToken.type === "COMMA") {
+ if (nextToken && nextToken.type === "COLON") {
+ isVar = false;
+ }
+ }
+ }
+ }
+ if (nextToken && nextToken.type === "OPERATOR" && nextToken.value === "=>") {
+ if (token.type === "RIGHT_PAREN") {
+ let j = i - 1;
+ while (j > 0 && tokens[j].type !== "LEFT_PAREN") {
+ if (tokens[j].type === "SYMBOL" && tokens[j].originalValue) {
+ tokens[j].value = tokens[j].originalValue;
+ scope[tokens[j].value] = { id: tokens[j].value, expr: tokens[j].value };
+ }
+ j--;
+ }
+ }
+ else {
+ scope[token.value] = { id: token.value, expr: token.value };
+ }
+ }
+ if (isVar) {
+ token.varName = token.value;
+ if (token.value in scope && "id" in scope[token.value]) {
+ token.value = scope[token.value].expr;
+ }
+ else {
+ token.originalValue = token.value;
+ token.value = `scope['${token.value}']`;
+ }
+ }
+ }
+ return tokens;
+ }
+ function compileExpr(expr, scope) {
+ return compileExprToArray(expr, scope)
+ .map((t) => t.value)
+ .join("");
+ }
+
+ const INTERP_REGEXP = /\{\{.*?\}\}/g;
+ //------------------------------------------------------------------------------
+ // Compilation Context
+ //------------------------------------------------------------------------------
+ let CompilationContext = /** @class */ (() => {
+ class CompilationContext {
+ constructor(name) {
+ this.code = [];
+ this.variables = {};
+ this.escaping = false;
+ this.parentNode = null;
+ this.parentTextNode = null;
+ this.rootNode = null;
+ this.indentLevel = 0;
+ this.shouldDefineParent = false;
+ this.shouldDefineScope = false;
+ this.protectedScopeNumber = 0;
+ this.shouldDefineQWeb = false;
+ this.shouldDefineUtils = false;
+ this.shouldDefineRefs = false;
+ this.shouldDefineResult = true;
+ this.loopNumber = 0;
+ this.inPreTag = false;
+ this.allowMultipleRoots = false;
+ this.hasParentWidget = false;
+ this.hasKey0 = false;
+ this.keyStack = [];
+ this.rootContext = this;
+ this.templateName = name || "noname";
+ this.addLine("let h = this.h;");
+ }
+ generateID() {
+ return CompilationContext.nextID++;
+ }
+ /**
+ * This method generates a "template key", which is basically a unique key
+ * which depends on the currently set keys, and on the iteration numbers (if
+ * we are in a loop).
+ *
+ * Such a key is necessary when we need to associate an id to some element
+ * generated by a template (for example, a component)
+ */
+ generateTemplateKey(prefix = "") {
+ const id = this.generateID();
+ if (this.loopNumber === 0 && !this.hasKey0) {
+ return `'${prefix}__${id}__'`;
+ }
+ let key = `\`${prefix}__${id}__`;
+ let start = this.hasKey0 ? 0 : 1;
+ for (let i = start; i < this.loopNumber + 1; i++) {
+ key += `\${key${i}}__`;
+ }
+ this.addLine(`let k${id} = ${key}\`;`);
+ return `k${id}`;
+ }
+ generateCode() {
+ if (this.shouldDefineResult) {
+ this.code.unshift(" let result;");
+ }
+ if (this.shouldDefineScope) {
+ this.code.unshift(" let scope = Object.create(context);");
+ }
+ if (this.shouldDefineRefs) {
+ this.code.unshift(" context.__owl__.refs = context.__owl__.refs || {};");
+ }
+ if (this.shouldDefineParent) {
+ if (this.hasParentWidget) {
+ this.code.unshift(" let parent = extra.parent;");
+ }
+ else {
+ this.code.unshift(" let parent = context;");
+ }
+ }
+ if (this.shouldDefineQWeb) {
+ this.code.unshift(" let QWeb = this.constructor;");
+ }
+ if (this.shouldDefineUtils) {
+ this.code.unshift(" let utils = this.constructor.utils;");
+ }
+ return this.code;
+ }
+ withParent(node) {
+ if (!this.allowMultipleRoots &&
+ this === this.rootContext &&
+ (this.parentNode || this.parentTextNode)) {
+ throw new Error("A template should not have more than one root node");
+ }
+ if (!this.rootContext.rootNode) {
+ this.rootContext.rootNode = node;
+ }
+ if (!this.parentNode && this.rootContext.shouldDefineResult) {
+ this.addLine(`result = vn${node};`);
+ }
+ return this.subContext("parentNode", node);
+ }
+ subContext(key, value) {
+ const newContext = Object.create(this);
+ newContext[key] = value;
+ return newContext;
+ }
+ indent() {
+ this.rootContext.indentLevel++;
+ }
+ dedent() {
+ this.rootContext.indentLevel--;
+ }
+ addLine(line) {
+ const prefix = new Array(this.indentLevel + 2).join(" ");
+ this.code.push(prefix + line);
+ return this.code.length - 1;
+ }
+ addIf(condition) {
+ this.addLine(`if (${condition}) {`);
+ this.indent();
+ }
+ addElse() {
+ this.dedent();
+ this.addLine("} else {");
+ this.indent();
+ }
+ closeIf() {
+ this.dedent();
+ this.addLine("}");
+ }
+ getValue(val) {
+ return val in this.variables ? this.getValue(this.variables[val]) : val;
+ }
+ /**
+ * Prepare an expression for being consumed at render time. Its main job
+ * is to
+ * - replace unknown variables by a lookup in the context
+ * - replace already defined variables by their internal name
+ */
+ formatExpression(expr) {
+ this.rootContext.shouldDefineScope = true;
+ return compileExpr(expr, this.variables);
+ }
+ captureExpression(expr) {
+ this.rootContext.shouldDefineScope = true;
+ const argId = this.generateID();
+ const tokens = compileExprToArray(expr, this.variables);
+ const done = new Set();
+ return tokens
+ .map((tok) => {
+ if (tok.varName) {
+ if (!done.has(tok.varName)) {
+ done.add(tok.varName);
+ this.addLine(`const ${tok.varName}_${argId} = ${tok.value};`);
+ }
+ tok.value = `${tok.varName}_${argId}`;
+ }
+ return tok.value;
+ })
+ .join("");
+ }
+ /**
+ * Perform string interpolation on the given string. Note that if the whole
+ * string is an expression, it simply returns it (formatted and enclosed in
+ * parentheses).
+ * For instance:
+ * 'Hello {{x}}!' -> `Hello ${x}`
+ * '{{x ? 'a': 'b'}}' -> (x ? 'a' : 'b')
+ */
+ interpolate(s) {
+ let matches = s.match(INTERP_REGEXP);
+ if (matches && matches[0].length === s.length) {
+ return `(${this.formatExpression(s.slice(2, -2))})`;
+ }
+ let r = s.replace(/\{\{.*?\}\}/g, (s) => "${" + this.formatExpression(s.slice(2, -2)) + "}");
+ return "`" + r + "`";
+ }
+ startProtectScope(codeBlock) {
+ const protectID = this.generateID();
+ this.rootContext.protectedScopeNumber++;
+ this.rootContext.shouldDefineScope = true;
+ const scopeExpr = `Object.create(scope);`;
+ this.addLine(`let _origScope${protectID} = scope;`);
+ this.addLine(`scope = ${scopeExpr}`);
+ if (!codeBlock) {
+ this.addLine(`scope.__access_mode__ = 'ro';`);
+ }
+ return protectID;
+ }
+ stopProtectScope(protectID) {
+ this.rootContext.protectedScopeNumber--;
+ this.addLine(`scope = _origScope${protectID};`);
+ }
+ }
+ CompilationContext.nextID = 1;
+ return CompilationContext;
+ })();
+
+ const browser = {
+ setTimeout: window.setTimeout.bind(window),
+ clearTimeout: window.clearTimeout.bind(window),
+ setInterval: window.setInterval.bind(window),
+ clearInterval: window.clearInterval.bind(window),
+ requestAnimationFrame: window.requestAnimationFrame.bind(window),
+ random: Math.random,
+ Date: window.Date,
+ fetch: (window.fetch || (() => { })).bind(window),
+ localStorage: window.localStorage,
+ };
+
+ /**
+ * Owl Utils
+ *
+ * We have here a small collection of utility functions:
+ *
+ * - whenReady
+ * - loadJS
+ * - loadFile
+ * - escape
+ * - debounce
+ */
+ function whenReady(fn) {
+ return new Promise(function (resolve) {
+ if (document.readyState !== "loading") {
+ resolve();
+ }
+ else {
+ document.addEventListener("DOMContentLoaded", resolve, false);
+ }
+ }).then(fn || function () { });
+ }
+ const loadedScripts = {};
+ function loadJS(url) {
+ if (url in loadedScripts) {
+ return loadedScripts[url];
+ }
+ const promise = new Promise(function (resolve, reject) {
+ const script = document.createElement("script");
+ script.type = "text/javascript";
+ script.src = url;
+ script.onload = function () {
+ resolve();
+ };
+ script.onerror = function () {
+ reject(`Error loading file '${url}'`);
+ };
+ const head = document.head || document.getElementsByTagName("head")[0];
+ head.appendChild(script);
+ });
+ loadedScripts[url] = promise;
+ return promise;
+ }
+ async function loadFile(url) {
+ const result = await browser.fetch(url);
+ if (!result.ok) {
+ throw new Error("Error while fetching xml templates");
+ }
+ return await result.text();
+ }
+ function escape(str) {
+ if (str === undefined) {
+ return "";
+ }
+ if (typeof str === "number") {
+ return String(str);
+ }
+ const p = document.createElement("p");
+ p.textContent = str;
+ return p.innerHTML;
+ }
+ /**
+ * Returns a function, that, as long as it continues to be invoked, will not
+ * be triggered. The function will be called after it stops being called for
+ * N milliseconds. If `immediate` is passed, trigger the function on the
+ * leading edge, instead of the trailing.
+ *
+ * Inspired by https://davidwalsh.name/javascript-debounce-function
+ */
+ function debounce(func, wait, immediate) {
+ let timeout;
+ return function () {
+ const context = this;
+ const args = arguments;
+ function later() {
+ timeout = null;
+ if (!immediate) {
+ func.apply(context, args);
+ }
+ }
+ const callNow = immediate && !timeout;
+ browser.clearTimeout(timeout);
+ timeout = browser.setTimeout(later, wait);
+ if (callNow) {
+ func.apply(context, args);
+ }
+ };
+ }
+ function shallowEqual(p1, p2) {
+ for (let k in p1) {
+ if (p1[k] !== p2[k]) {
+ return false;
+ }
+ }
+ return true;
+ }
+
+ var _utils = /*#__PURE__*/Object.freeze({
+ __proto__: null,
+ whenReady: whenReady,
+ loadJS: loadJS,
+ loadFile: loadFile,
+ escape: escape,
+ debounce: debounce,
+ shallowEqual: shallowEqual
+ });
+
+ //------------------------------------------------------------------------------
+ // Const/global stuff/helpers
+ //------------------------------------------------------------------------------
+ const DISABLED_TAGS = ["input", "textarea", "button", "select", "option", "optgroup"];
+ const TRANSLATABLE_ATTRS = ["label", "title", "placeholder", "alt"];
+ const lineBreakRE = /[\r\n]/;
+ const whitespaceRE = /\s+/g;
+ const NODE_HOOKS_PARAMS = {
+ create: "(_, n)",
+ insert: "vn",
+ remove: "(vn, rm)",
+ destroy: "()",
+ };
+ function isComponent(obj) {
+ return obj && obj.hasOwnProperty("__owl__");
+ }
+ const UTILS = {
+ zero: Symbol("zero"),
+ toObj(expr) {
+ if (typeof expr === "string") {
+ expr = expr.trim();
+ if (!expr) {
+ return {};
+ }
+ let words = expr.split(/\s+/);
+ let result = {};
+ for (let i = 0; i < words.length; i++) {
+ result[words[i]] = true;
+ }
+ return result;
+ }
+ return expr;
+ },
+ shallowEqual,
+ addNameSpace(vnode) {
+ addNS(vnode.data, vnode.children, vnode.sel);
+ },
+ VDomArray: class VDomArray extends Array {
+ },
+ vDomToString: function (vdom) {
+ return vdom
+ .map((vnode) => {
+ if (vnode.sel) {
+ const node = document.createElement(vnode.sel);
+ const result = patch(node, vnode);
+ return result.elm.outerHTML;
+ }
+ else {
+ return vnode.text;
+ }
+ })
+ .join("");
+ },
+ getComponent(obj) {
+ while (obj && !isComponent(obj)) {
+ obj = obj.__proto__;
+ }
+ return obj;
+ },
+ getScope(obj, property) {
+ const obj0 = obj;
+ while (obj &&
+ !obj.hasOwnProperty(property) &&
+ !(obj.hasOwnProperty("__access_mode__") && obj.__access_mode__ === "ro")) {
+ const newObj = obj.__proto__;
+ if (!newObj || isComponent(newObj)) {
+ return obj0;
+ }
+ obj = newObj;
+ }
+ return obj;
+ },
+ };
+ function parseXML(xml) {
+ const parser = new DOMParser();
+ const doc = parser.parseFromString(xml, "text/xml");
+ if (doc.getElementsByTagName("parsererror").length) {
+ let msg = "Invalid XML in template.";
+ const parsererrorText = doc.getElementsByTagName("parsererror")[0].textContent;
+ if (parsererrorText) {
+ msg += "\nThe parser has produced the following error message:\n" + parsererrorText;
+ const re = /\d+/g;
+ const firstMatch = re.exec(parsererrorText);
+ if (firstMatch) {
+ const lineNumber = Number(firstMatch[0]);
+ const line = xml.split("\n")[lineNumber - 1];
+ const secondMatch = re.exec(parsererrorText);
+ if (line && secondMatch) {
+ const columnIndex = Number(secondMatch[0]) - 1;
+ if (line[columnIndex]) {
+ msg +=
+ `\nThe error might be located at xml line ${lineNumber} column ${columnIndex}\n` +
+ `${line}\n${"-".repeat(columnIndex - 1)}^`;
+ }
+ }
+ }
+ }
+ throw new Error(msg);
+ }
+ return doc;
+ }
+ function escapeQuotes(str) {
+ return str.replace(/\'/g, "\\'");
+ }
+ //------------------------------------------------------------------------------
+ // QWeb rendering engine
+ //------------------------------------------------------------------------------
+ let QWeb = /** @class */ (() => {
+ class QWeb extends EventBus {
+ constructor(config = {}) {
+ super();
+ this.h = h;
+ // subTemplates are stored in two objects: a (local) mapping from a name to an
+ // id, and a (global) mapping from an id to the compiled function. This is
+ // necessary to ensure that global templates can be called with more than one
+ // QWeb instance.
+ this.subTemplates = {};
+ this.isUpdating = false;
+ this.templates = Object.create(QWeb.TEMPLATES);
+ if (config.templates) {
+ this.addTemplates(config.templates);
+ }
+ if (config.translateFn) {
+ this.translateFn = config.translateFn;
+ }
+ }
+ static addDirective(directive) {
+ if (directive.name in QWeb.DIRECTIVE_NAMES) {
+ throw new Error(`Directive "${directive.name} already registered`);
+ }
+ QWeb.DIRECTIVES.push(directive);
+ QWeb.DIRECTIVE_NAMES[directive.name] = 1;
+ QWeb.DIRECTIVES.sort((d1, d2) => d1.priority - d2.priority);
+ if (directive.extraNames) {
+ directive.extraNames.forEach((n) => (QWeb.DIRECTIVE_NAMES[n] = 1));
+ }
+ }
+ static registerComponent(name, Component) {
+ if (QWeb.components[name]) {
+ throw new Error(`Component '${name}' has already been registered`);
+ }
+ QWeb.components[name] = Component;
+ }
+ /**
+ * Register globally a template. All QWeb instances will obtain their
+ * templates from their own template map, and then, from the global static
+ * TEMPLATES property.
+ */
+ static registerTemplate(name, template) {
+ if (QWeb.TEMPLATES[name]) {
+ throw new Error(`Template '${name}' has already been registered`);
+ }
+ const qweb = new QWeb();
+ qweb.addTemplate(name, template);
+ QWeb.TEMPLATES[name] = qweb.templates[name];
+ }
+ /**
+ * Add a template to the internal template map. Note that it is not
+ * immediately compiled.
+ */
+ addTemplate(name, xmlString, allowDuplicate) {
+ if (allowDuplicate && name in this.templates) {
+ return;
+ }
+ const doc = parseXML(xmlString);
+ if (!doc.firstChild) {
+ throw new Error("Invalid template (should not be empty)");
+ }
+ this._addTemplate(name, doc.firstChild);
+ }
+ /**
+ * Load templates from a xml (as a string or xml document). This will look up
+ * for the first tag, and will consider each child of this as a
+ * template, with the name given by the t-name attribute.
+ */
+ addTemplates(xmlstr) {
+ const doc = typeof xmlstr === "string" ? parseXML(xmlstr) : xmlstr;
+ const templates = doc.getElementsByTagName("templates")[0];
+ if (!templates) {
+ return;
+ }
+ for (let elem of templates.children) {
+ const name = elem.getAttribute("t-name");
+ this._addTemplate(name, elem);
+ }
+ }
+ _addTemplate(name, elem) {
+ if (name in this.templates) {
+ throw new Error(`Template ${name} already defined`);
+ }
+ this._processTemplate(elem);
+ const template = {
+ elem,
+ fn: function (context, extra) {
+ const compiledFunction = this._compile(name, elem);
+ template.fn = compiledFunction;
+ return compiledFunction.call(this, context, extra);
+ },
+ };
+ this.templates[name] = template;
+ }
+ _processTemplate(elem) {
+ let tbranch = elem.querySelectorAll("[t-elif], [t-else]");
+ for (let i = 0, ilen = tbranch.length; i < ilen; i++) {
+ let node = tbranch[i];
+ let prevElem = node.previousElementSibling;
+ let pattr = function (name) {
+ return prevElem.getAttribute(name);
+ };
+ let nattr = function (name) {
+ return +!!node.getAttribute(name);
+ };
+ if (prevElem && (pattr("t-if") || pattr("t-elif"))) {
+ if (pattr("t-foreach")) {
+ throw new Error("t-if cannot stay at the same level as t-foreach when using t-elif or t-else");
+ }
+ if (["t-if", "t-elif", "t-else"].map(nattr).reduce(function (a, b) {
+ return a + b;
+ }) > 1) {
+ throw new Error("Only one conditional branching directive is allowed per node");
+ }
+ // All text (with only spaces) and comment nodes (nodeType 8) between
+ // branch nodes are removed
+ let textNode;
+ while ((textNode = node.previousSibling) !== prevElem) {
+ if (textNode.nodeValue.trim().length && textNode.nodeType !== 8) {
+ throw new Error("text is not allowed between branching directives");
+ }
+ textNode.remove();
+ }
+ }
+ else {
+ throw new Error("t-elif and t-else directives must be preceded by a t-if or t-elif directive");
+ }
+ }
+ }
+ /**
+ * Render a template
+ *
+ * @param {string} name the template should already have been added
+ */
+ render(name, context = {}, extra = null) {
+ const template = this.templates[name];
+ if (!template) {
+ throw new Error(`Template ${name} does not exist`);
+ }
+ return template.fn.call(this, context, extra);
+ }
+ /**
+ * Render a template to a html string.
+ *
+ * Note that this is more limited than the `render` method: it is not suitable
+ * to render a full component tree, since this is an asynchronous operation.
+ * This method can only render templates without components.
+ */
+ renderToString(name, context = {}, extra) {
+ const vnode = this.render(name, context, extra);
+ if (vnode.sel === undefined) {
+ return vnode.text;
+ }
+ const node = document.createElement(vnode.sel);
+ const elem = patch(node, vnode).elm;
+ function escapeTextNodes(node) {
+ if (node.nodeType === 3) {
+ node.textContent = escape(node.textContent);
+ }
+ for (let n of node.childNodes) {
+ escapeTextNodes(n);
+ }
+ }
+ escapeTextNodes(elem);
+ return elem.outerHTML;
+ }
+ /**
+ * Force all widgets connected to this QWeb instance to rerender themselves.
+ *
+ * This method is mostly useful for external code that want to modify the
+ * application in some cases. For example, a router plugin.
+ */
+ forceUpdate() {
+ this.isUpdating = true;
+ Promise.resolve().then(() => {
+ if (this.isUpdating) {
+ this.isUpdating = false;
+ this.trigger("update");
+ }
+ });
+ }
+ _compile(name, elem, parentContext, defineKey) {
+ const isDebug = elem.attributes.hasOwnProperty("t-debug");
+ const ctx = new CompilationContext(name);
+ if (elem.tagName !== "t") {
+ ctx.shouldDefineResult = false;
+ }
+ if (parentContext) {
+ ctx.variables = Object.create(parentContext.variables);
+ ctx.parentNode = parentContext.parentNode || ctx.generateID();
+ ctx.allowMultipleRoots = true;
+ ctx.hasParentWidget = true;
+ ctx.shouldDefineResult = false;
+ ctx.addLine(`let c${ctx.parentNode} = extra.parentNode;`);
+ if (defineKey) {
+ ctx.addLine(`let key0 = extra.key || "";`);
+ ctx.hasKey0 = true;
+ }
+ }
+ this._compileNode(elem, ctx);
+ if (!parentContext) {
+ if (ctx.shouldDefineResult) {
+ ctx.addLine(`return result;`);
+ }
+ else {
+ if (!ctx.rootNode) {
+ throw new Error(`A template should have one root node (${ctx.templateName})`);
+ }
+ ctx.addLine(`return vn${ctx.rootNode};`);
+ }
+ }
+ let code = ctx.generateCode();
+ const templateName = ctx.templateName.replace(/`/g, "'").slice(0, 200);
+ code.unshift(` // Template name: "${templateName}"`);
+ let template;
+ try {
+ template = new Function("context, extra", code.join("\n"));
+ }
+ catch (e) {
+ console.groupCollapsed(`Invalid Code generated by ${templateName}`);
+ console.warn(code.join("\n"));
+ console.groupEnd();
+ throw new Error(`Invalid generated code while compiling template '${templateName}': ${e.message}`);
+ }
+ if (isDebug) {
+ const tpl = this.templates[name];
+ if (tpl) {
+ const msg = `Template: ${tpl.elem.outerHTML}\nCompiled code:\n${template.toString()}`;
+ console.log(msg);
+ }
+ }
+ return template;
+ }
+ /**
+ * Generate code from an xml node
+ *
+ */
+ _compileNode(node, ctx) {
+ if (!(node instanceof Element)) {
+ // this is a text node, there are no directive to apply
+ let text = node.textContent;
+ if (!ctx.inPreTag) {
+ if (lineBreakRE.test(text) && !text.trim()) {
+ return;
+ }
+ text = text.replace(whitespaceRE, " ");
+ }
+ if (this.translateFn) {
+ if (node.parentNode.getAttribute("t-translation") !== "off") {
+ text = this.translateFn(text);
+ }
+ }
+ if (ctx.parentNode) {
+ if (node.nodeType === 3) {
+ ctx.addLine(`c${ctx.parentNode}.push({text: \`${text}\`});`);
+ }
+ else if (node.nodeType === 8) {
+ ctx.addLine(`c${ctx.parentNode}.push(h('!', \`${text}\`));`);
+ }
+ }
+ else if (ctx.parentTextNode) {
+ ctx.addLine(`vn${ctx.parentTextNode}.text += \`${text}\`;`);
+ }
+ else {
+ // this is an unusual situation: this text node is the result of the
+ // template rendering.
+ let nodeID = ctx.generateID();
+ ctx.addLine(`let vn${nodeID} = {text: \`${text}\`};`);
+ ctx.addLine(`result = vn${nodeID};`);
+ ctx.rootContext.rootNode = nodeID;
+ ctx.rootContext.parentTextNode = nodeID;
+ }
+ return;
+ }
+ const firstLetter = node.tagName[0];
+ if (firstLetter === firstLetter.toUpperCase()) {
+ // this is a component, we modify in place the xml document to change
+ // to
+ node.setAttribute("t-component", node.tagName);
+ }
+ else if (node.tagName !== "t" && node.hasAttribute("t-component")) {
+ throw new Error(`Directive 't-component' can only be used on nodes (used on a <${node.tagName}>)`);
+ }
+ const attributes = node.attributes;
+ const validDirectives = [];
+ const finalizers = [];
+ // maybe this is not optimal: we iterate on all attributes here, and again
+ // just after for each directive.
+ for (let i = 0; i < attributes.length; i++) {
+ let attrName = attributes[i].name;
+ if (attrName.startsWith("t-")) {
+ let dName = attrName.slice(2).split(/-|\./)[0];
+ if (!(dName in QWeb.DIRECTIVE_NAMES)) {
+ throw new Error(`Unknown QWeb directive: '${attrName}'`);
+ }
+ if (node.tagName !== "t" && (attrName === "t-esc" || attrName === "t-raw")) {
+ const tNode = document.createElement("t");
+ tNode.setAttribute(attrName, node.getAttribute(attrName));
+ for (let child of Array.from(node.childNodes)) {
+ tNode.appendChild(child);
+ }
+ node.appendChild(tNode);
+ node.removeAttribute(attrName);
+ }
+ }
+ }
+ const DIR_N = QWeb.DIRECTIVES.length;
+ const ATTR_N = attributes.length;
+ let withHandlers = false;
+ for (let i = 0; i < DIR_N; i++) {
+ let directive = QWeb.DIRECTIVES[i];
+ let fullName;
+ let value;
+ for (let j = 0; j < ATTR_N; j++) {
+ const name = attributes[j].name;
+ if (name === "t-" + directive.name ||
+ name.startsWith("t-" + directive.name + "-") ||
+ name.startsWith("t-" + directive.name + ".")) {
+ fullName = name;
+ value = attributes[j].textContent;
+ validDirectives.push({ directive, value, fullName });
+ if (directive.name === "on" || directive.name === "model") {
+ withHandlers = true;
+ }
+ }
+ }
+ }
+ for (let { directive, value, fullName } of validDirectives) {
+ if (directive.finalize) {
+ finalizers.push({ directive, value, fullName });
+ }
+ if (directive.atNodeEncounter) {
+ const isDone = directive.atNodeEncounter({
+ node,
+ qweb: this,
+ ctx,
+ fullName,
+ value,
+ });
+ if (isDone) {
+ for (let { directive, value, fullName } of finalizers) {
+ directive.finalize({ node, qweb: this, ctx, fullName, value });
+ }
+ return;
+ }
+ }
+ }
+ if (node.nodeName !== "t") {
+ let nodeID = this._compileGenericNode(node, ctx, withHandlers);
+ ctx = ctx.withParent(nodeID);
+ let nodeHooks = {};
+ let addNodeHook = function (hook, handler) {
+ nodeHooks[hook] = nodeHooks[hook] || [];
+ nodeHooks[hook].push(handler);
+ };
+ for (let { directive, value, fullName } of validDirectives) {
+ if (directive.atNodeCreation) {
+ directive.atNodeCreation({
+ node,
+ qweb: this,
+ ctx,
+ fullName,
+ value,
+ nodeID,
+ addNodeHook,
+ });
+ }
+ }
+ if (Object.keys(nodeHooks).length) {
+ ctx.addLine(`p${nodeID}.hook = {`);
+ for (let hook in nodeHooks) {
+ ctx.addLine(` ${hook}: ${NODE_HOOKS_PARAMS[hook]} => {`);
+ for (let handler of nodeHooks[hook]) {
+ ctx.addLine(` ${handler}`);
+ }
+ ctx.addLine(` },`);
+ }
+ ctx.addLine(`};`);
+ }
+ }
+ if (node.nodeName === "pre") {
+ ctx = ctx.subContext("inPreTag", true);
+ }
+ this._compileChildren(node, ctx);
+ // svg support
+ // we hadd svg namespace if it is a svg or if it is a g, but only if it is
+ // the root node. This is the easiest way to support svg sub components:
+ // they need to have a g tag as root. Otherwise, we would need a complete
+ // list of allowed svg tags.
+ const shouldAddNS = node.nodeName === "svg" || (node.nodeName === "g" && ctx.rootNode === ctx.parentNode);
+ if (shouldAddNS) {
+ ctx.rootContext.shouldDefineUtils = true;
+ ctx.addLine(`utils.addNameSpace(vn${ctx.parentNode});`);
+ }
+ for (let { directive, value, fullName } of finalizers) {
+ directive.finalize({ node, qweb: this, ctx, fullName, value });
+ }
+ }
+ _compileGenericNode(node, ctx, withHandlers = true) {
+ // nodeType 1 is generic tag
+ if (node.nodeType !== 1) {
+ throw new Error("unsupported node type");
+ }
+ const attributes = node.attributes;
+ const attrs = [];
+ const props = [];
+ const tattrs = [];
+ function handleBooleanProps(key, val) {
+ let isProp = false;
+ if (node.nodeName === "input" && key === "checked") {
+ let type = node.getAttribute("type");
+ if (type === "checkbox" || type === "radio") {
+ isProp = true;
+ }
+ }
+ if (node.nodeName === "option" && key === "selected") {
+ isProp = true;
+ }
+ if (key === "disabled" && DISABLED_TAGS.indexOf(node.nodeName) > -1) {
+ isProp = true;
+ }
+ if ((key === "readonly" && node.nodeName === "input") || node.nodeName === "textarea") {
+ isProp = true;
+ }
+ if (isProp) {
+ props.push(`${key}: _${val}`);
+ }
+ }
+ let classObj = "";
+ for (let i = 0; i < attributes.length; i++) {
+ let name = attributes[i].name;
+ let value = attributes[i].textContent;
+ if (this.translateFn && TRANSLATABLE_ATTRS.includes(name)) {
+ value = this.translateFn(value);
+ }
+ // regular attributes
+ if (!name.startsWith("t-") && !node.getAttribute("t-attf-" + name)) {
+ const attID = ctx.generateID();
+ if (name === "class") {
+ if ((value = value.trim())) {
+ let classDef = value
+ .split(/\s+/)
+ .map((a) => `'${escapeQuotes(a)}':true`)
+ .join(",");
+ if (classObj) {
+ ctx.addLine(`Object.assign(${classObj}, {${classDef}})`);
+ }
+ else {
+ classObj = `_${ctx.generateID()}`;
+ ctx.addLine(`let ${classObj} = {${classDef}};`);
+ }
+ }
+ }
+ else {
+ ctx.addLine(`let _${attID} = '${escapeQuotes(value)}';`);
+ if (!name.match(/^[a-zA-Z]+$/)) {
+ // attribute contains 'non letters' => we want to quote it
+ name = '"' + name + '"';
+ }
+ attrs.push(`${name}: _${attID}`);
+ handleBooleanProps(name, attID);
+ }
+ }
+ // dynamic attributes
+ if (name.startsWith("t-att-")) {
+ let attName = name.slice(6);
+ const v = ctx.getValue(value);
+ let formattedValue = typeof v === "string" ? ctx.formatExpression(v) : `scope.${v.id}`;
+ if (attName === "class") {
+ ctx.rootContext.shouldDefineUtils = true;
+ formattedValue = `utils.toObj(${formattedValue})`;
+ if (classObj) {
+ ctx.addLine(`Object.assign(${classObj}, ${formattedValue})`);
+ }
+ else {
+ classObj = `_${ctx.generateID()}`;
+ ctx.addLine(`let ${classObj} = ${formattedValue};`);
+ }
+ }
+ else {
+ const attID = ctx.generateID();
+ if (!attName.match(/^[a-zA-Z]+$/)) {
+ // attribute contains 'non letters' => we want to quote it
+ attName = '"' + attName + '"';
+ }
+ // we need to combine dynamic with non dynamic attributes:
+ // class="a" t-att-class="'yop'" should be rendered as class="a yop"
+ const attValue = node.getAttribute(attName);
+ if (attValue) {
+ const attValueID = ctx.generateID();
+ ctx.addLine(`let _${attValueID} = ${formattedValue};`);
+ formattedValue = `'${attValue}' + (_${attValueID} ? ' ' + _${attValueID} : '')`;
+ const attrIndex = attrs.findIndex((att) => att.startsWith(attName + ":"));
+ attrs.splice(attrIndex, 1);
+ }
+ ctx.addLine(`let _${attID} = ${formattedValue};`);
+ attrs.push(`${attName}: _${attID}`);
+ handleBooleanProps(attName, attID);
+ }
+ }
+ if (name.startsWith("t-attf-")) {
+ let attName = name.slice(7);
+ if (!attName.match(/^[a-zA-Z]+$/)) {
+ // attribute contains 'non letters' => we want to quote it
+ attName = '"' + attName + '"';
+ }
+ const formattedExpr = ctx.interpolate(value);
+ const attID = ctx.generateID();
+ let staticVal = node.getAttribute(attName);
+ if (staticVal) {
+ ctx.addLine(`let _${attID} = '${staticVal} ' + ${formattedExpr};`);
+ }
+ else {
+ ctx.addLine(`let _${attID} = ${formattedExpr};`);
+ }
+ attrs.push(`${attName}: _${attID}`);
+ }
+ // t-att= attributes
+ if (name === "t-att") {
+ let id = ctx.generateID();
+ ctx.addLine(`let _${id} = ${ctx.formatExpression(value)};`);
+ tattrs.push(id);
+ }
+ }
+ let nodeID = ctx.generateID();
+ let key = ctx.loopNumber || ctx.hasKey0 ? `\`\${key${ctx.loopNumber}}_${nodeID}\`` : nodeID;
+ const parts = [`key:${key}`];
+ if (attrs.length + tattrs.length > 0) {
+ parts.push(`attrs:{${attrs.join(",")}}`);
+ }
+ if (props.length > 0) {
+ parts.push(`props:{${props.join(",")}}`);
+ }
+ if (classObj) {
+ parts.push(`class:${classObj}`);
+ }
+ if (withHandlers) {
+ parts.push(`on:{}`);
+ }
+ ctx.addLine(`let c${nodeID} = [], p${nodeID} = {${parts.join(",")}};`);
+ for (let id of tattrs) {
+ ctx.addIf(`_${id} instanceof Array`);
+ ctx.addLine(`p${nodeID}.attrs[_${id}[0]] = _${id}[1];`);
+ ctx.addElse();
+ ctx.addLine(`for (let key in _${id}) {`);
+ ctx.indent();
+ ctx.addLine(`p${nodeID}.attrs[key] = _${id}[key];`);
+ ctx.dedent();
+ ctx.addLine(`}`);
+ ctx.closeIf();
+ }
+ ctx.addLine(`let vn${nodeID} = h('${node.nodeName}', p${nodeID}, c${nodeID});`);
+ if (ctx.parentNode) {
+ ctx.addLine(`c${ctx.parentNode}.push(vn${nodeID});`);
+ }
+ else if (ctx.loopNumber || ctx.hasKey0) {
+ ctx.rootContext.shouldDefineResult = true;
+ ctx.addLine(`result = vn${nodeID};`);
+ }
+ return nodeID;
+ }
+ _compileChildren(node, ctx) {
+ if (node.childNodes.length > 0) {
+ for (let child of Array.from(node.childNodes)) {
+ this._compileNode(child, ctx);
+ }
+ }
+ }
+ }
+ QWeb.utils = UTILS;
+ QWeb.components = Object.create(null);
+ QWeb.DIRECTIVE_NAMES = {
+ name: 1,
+ att: 1,
+ attf: 1,
+ translation: 1,
+ };
+ QWeb.DIRECTIVES = [];
+ QWeb.TEMPLATES = {};
+ QWeb.nextId = 1;
+ // dev mode enables better error messages or more costly validations
+ QWeb.dev = false;
+ // slots contains sub templates defined with t-set inside t-component nodes, and
+ // are meant to be used by the t-slot directive.
+ QWeb.slots = {};
+ QWeb.nextSlotId = 1;
+ QWeb.subTemplates = {};
+ return QWeb;
+ })();
+
+ const parser = new DOMParser();
+ function htmlToVDOM(html) {
+ const doc = parser.parseFromString(html, "text/html");
+ const result = [];
+ for (let child of doc.body.childNodes) {
+ result.push(htmlToVNode(child));
+ }
+ return result;
+ }
+ function htmlToVNode(node) {
+ if (!(node instanceof Element)) {
+ return { text: node.textContent };
+ }
+ const attrs = {};
+ for (let attr of node.attributes) {
+ attrs[attr.name] = attr.textContent;
+ }
+ const children = [];
+ for (let c of node.childNodes) {
+ children.push(htmlToVNode(c));
+ }
+ const vnode = h(node.tagName, { attrs }, children);
+ if (vnode.sel === "svg") {
+ addNS(vnode.data, vnode.children, vnode.sel);
+ }
+ return vnode;
+ }
+
+ /**
+ * Owl QWeb Directives
+ *
+ * This file contains the implementation of most standard QWeb directives:
+ * - t-esc
+ * - t-raw
+ * - t-set/t-value
+ * - t-if/t-elif/t-else
+ * - t-call
+ * - t-foreach/t-as
+ * - t-debug
+ * - t-log
+ */
+ //------------------------------------------------------------------------------
+ // t-esc and t-raw
+ //------------------------------------------------------------------------------
+ QWeb.utils.htmlToVDOM = htmlToVDOM;
+ function compileValueNode(value, node, qweb, ctx) {
+ ctx.rootContext.shouldDefineScope = true;
+ if (value === "0") {
+ if (ctx.parentNode) {
+ // the 'zero' magical symbol is where we can find the result of the rendering
+ // of the body of the t-call.
+ ctx.rootContext.shouldDefineUtils = true;
+ const zeroArgs = ctx.escaping
+ ? `{text: utils.vDomToString(scope[utils.zero])}`
+ : `...scope[utils.zero]`;
+ ctx.addLine(`c${ctx.parentNode}.push(${zeroArgs});`);
+ }
+ return;
+ }
+ let exprID;
+ if (typeof value === "string") {
+ exprID = `_${ctx.generateID()}`;
+ ctx.addLine(`let ${exprID} = ${ctx.formatExpression(value)};`);
+ }
+ else {
+ exprID = `scope.${value.id}`;
+ }
+ ctx.addIf(`${exprID} != null`);
+ if (ctx.escaping) {
+ let protectID;
+ if (value.hasBody) {
+ ctx.rootContext.shouldDefineUtils = true;
+ protectID = ctx.startProtectScope();
+ ctx.addLine(`${exprID} = ${exprID} instanceof utils.VDomArray ? utils.vDomToString(${exprID}) : ${exprID};`);
+ }
+ if (ctx.parentTextNode) {
+ ctx.addLine(`vn${ctx.parentTextNode}.text += ${exprID};`);
+ }
+ else if (ctx.parentNode) {
+ ctx.addLine(`c${ctx.parentNode}.push({text: ${exprID}});`);
+ }
+ else {
+ let nodeID = ctx.generateID();
+ ctx.rootContext.rootNode = nodeID;
+ ctx.rootContext.parentTextNode = nodeID;
+ ctx.addLine(`let vn${nodeID} = {text: ${exprID}};`);
+ if (ctx.rootContext.shouldDefineResult) {
+ ctx.addLine(`result = vn${nodeID}`);
+ }
+ }
+ if (value.hasBody) {
+ ctx.stopProtectScope(protectID);
+ }
+ }
+ else {
+ ctx.rootContext.shouldDefineUtils = true;
+ if (value.hasBody) {
+ ctx.addLine(`const vnodeArray = ${exprID} instanceof utils.VDomArray ? ${exprID} : utils.htmlToVDOM(${exprID});`);
+ ctx.addLine(`c${ctx.parentNode}.push(...vnodeArray);`);
+ }
+ else {
+ ctx.addLine(`c${ctx.parentNode}.push(...utils.htmlToVDOM(${exprID}));`);
+ }
+ }
+ if (node.childNodes.length) {
+ ctx.addElse();
+ qweb._compileChildren(node, ctx);
+ }
+ ctx.closeIf();
+ }
+ QWeb.addDirective({
+ name: "esc",
+ priority: 70,
+ atNodeEncounter({ node, qweb, ctx }) {
+ let value = ctx.getValue(node.getAttribute("t-esc"));
+ compileValueNode(value, node, qweb, ctx.subContext("escaping", true));
+ return true;
+ },
+ });
+ QWeb.addDirective({
+ name: "raw",
+ priority: 80,
+ atNodeEncounter({ node, qweb, ctx }) {
+ let value = ctx.getValue(node.getAttribute("t-raw"));
+ compileValueNode(value, node, qweb, ctx);
+ return true;
+ },
+ });
+ //------------------------------------------------------------------------------
+ // t-set
+ //------------------------------------------------------------------------------
+ QWeb.addDirective({
+ name: "set",
+ extraNames: ["value"],
+ priority: 60,
+ atNodeEncounter({ node, qweb, ctx }) {
+ ctx.rootContext.shouldDefineScope = true;
+ const variable = node.getAttribute("t-set");
+ let value = node.getAttribute("t-value");
+ ctx.variables[variable] = ctx.variables[variable] || {};
+ let qwebvar = ctx.variables[variable];
+ const hasBody = node.hasChildNodes();
+ qwebvar.id = variable;
+ qwebvar.expr = `scope.${variable}`;
+ if (value) {
+ const formattedValue = ctx.formatExpression(value);
+ let scopeExpr = `scope`;
+ if (ctx.protectedScopeNumber) {
+ ctx.rootContext.shouldDefineUtils = true;
+ scopeExpr = `utils.getScope(scope, '${variable}')`;
+ }
+ ctx.addLine(`${scopeExpr}.${variable} = ${formattedValue};`);
+ qwebvar.value = formattedValue;
+ }
+ if (hasBody) {
+ ctx.rootContext.shouldDefineUtils = true;
+ if (value) {
+ ctx.addIf(`!(${qwebvar.expr})`);
+ }
+ const tempParentNodeID = ctx.generateID();
+ const _parentNode = ctx.parentNode;
+ ctx.parentNode = tempParentNodeID;
+ ctx.addLine(`let c${tempParentNodeID} = new utils.VDomArray();`);
+ const nodeCopy = node.cloneNode(true);
+ for (let attr of ["t-set", "t-value", "t-if", "t-else", "t-elif"]) {
+ nodeCopy.removeAttribute(attr);
+ }
+ qweb._compileNode(nodeCopy, ctx);
+ ctx.addLine(`${qwebvar.expr} = c${tempParentNodeID}`);
+ qwebvar.value = `c${tempParentNodeID}`;
+ qwebvar.hasBody = true;
+ ctx.parentNode = _parentNode;
+ if (value) {
+ ctx.closeIf();
+ }
+ }
+ return true;
+ },
+ });
+ //------------------------------------------------------------------------------
+ // t-if, t-elif, t-else
+ //------------------------------------------------------------------------------
+ QWeb.addDirective({
+ name: "if",
+ priority: 20,
+ atNodeEncounter({ node, ctx }) {
+ let cond = ctx.getValue(node.getAttribute("t-if"));
+ ctx.addIf(typeof cond === "string" ? ctx.formatExpression(cond) : `scope.${cond.id}`);
+ return false;
+ },
+ finalize({ ctx }) {
+ ctx.closeIf();
+ },
+ });
+ QWeb.addDirective({
+ name: "elif",
+ priority: 30,
+ atNodeEncounter({ node, ctx }) {
+ let cond = ctx.getValue(node.getAttribute("t-elif"));
+ ctx.addLine(`else if (${typeof cond === "string" ? ctx.formatExpression(cond) : `scope.${cond.id}`}) {`);
+ ctx.indent();
+ return false;
+ },
+ finalize({ ctx }) {
+ ctx.closeIf();
+ },
+ });
+ QWeb.addDirective({
+ name: "else",
+ priority: 40,
+ atNodeEncounter({ ctx }) {
+ ctx.addLine(`else {`);
+ ctx.indent();
+ return false;
+ },
+ finalize({ ctx }) {
+ ctx.closeIf();
+ },
+ });
+ //------------------------------------------------------------------------------
+ // t-call
+ //------------------------------------------------------------------------------
+ QWeb.addDirective({
+ name: "call",
+ priority: 50,
+ atNodeEncounter({ node, qweb, ctx }) {
+ // Step 1: sanity checks
+ // ------------------------------------------------
+ ctx.rootContext.shouldDefineScope = true;
+ ctx.rootContext.shouldDefineUtils = true;
+ if (node.nodeName !== "t") {
+ throw new Error("Invalid tag for t-call directive (should be 't')");
+ }
+ const subTemplate = node.getAttribute("t-call");
+ const nodeTemplate = qweb.templates[subTemplate];
+ if (!nodeTemplate) {
+ throw new Error(`Cannot find template "${subTemplate}" (t-call)`);
+ }
+ // Step 2: compile target template in sub templates
+ // ------------------------------------------------
+ let subId = qweb.subTemplates[subTemplate];
+ if (!subId) {
+ subId = QWeb.nextId++;
+ qweb.subTemplates[subTemplate] = subId;
+ const subTemplateFn = qweb._compile(subTemplate, nodeTemplate.elem, ctx, true);
+ QWeb.subTemplates[subId] = subTemplateFn;
+ }
+ // Step 3: compile t-call body if necessary
+ // ------------------------------------------------
+ let hasBody = node.hasChildNodes();
+ let protectID;
+ if (hasBody) {
+ // we add a sub scope to protect the ambient scope
+ ctx.addLine(`{`);
+ ctx.indent();
+ protectID = ctx.startProtectScope();
+ const nodeCopy = node.cloneNode(true);
+ for (let attr of ["t-if", "t-else", "t-elif", "t-call"]) {
+ nodeCopy.removeAttribute(attr);
+ }
+ const parentNode = ctx.parentNode;
+ ctx.parentNode = "__0";
+ // this local scope is intended to trap c__0
+ ctx.addLine(`{`);
+ ctx.indent();
+ ctx.addLine("let c__0 = [];");
+ qweb._compileNode(nodeCopy, ctx);
+ ctx.rootContext.shouldDefineUtils = true;
+ ctx.addLine("scope[utils.zero] = c__0;");
+ ctx.parentNode = parentNode;
+ ctx.dedent();
+ ctx.addLine(`}`);
+ }
+ // Step 4: add the appropriate function call to current component
+ // ------------------------------------------------
+ const callingScope = hasBody ? "scope" : "Object.assign(Object.create(context), scope)";
+ const parentComponent = `utils.getComponent(context)`;
+ const key = ctx.generateTemplateKey();
+ const parentNode = ctx.parentNode ? `c${ctx.parentNode}` : "result";
+ const extra = `Object.assign({}, extra, {parentNode: ${parentNode}, parent: ${parentComponent}, key: ${key}})`;
+ if (ctx.parentNode) {
+ ctx.addLine(`this.constructor.subTemplates['${subId}'].call(this, ${callingScope}, ${extra});`);
+ }
+ else {
+ // this is a t-call with no parentnode, we need to extract the result
+ ctx.rootContext.shouldDefineResult = true;
+ ctx.addLine(`result = []`);
+ ctx.addLine(`this.constructor.subTemplates['${subId}'].call(this, ${callingScope}, ${extra});`);
+ ctx.addLine(`result = result[0]`);
+ }
+ // Step 5: restore previous scope
+ // ------------------------------------------------
+ if (hasBody) {
+ ctx.stopProtectScope(protectID);
+ ctx.dedent();
+ ctx.addLine(`}`);
+ }
+ return true;
+ },
+ });
+ //------------------------------------------------------------------------------
+ // t-foreach
+ //------------------------------------------------------------------------------
+ QWeb.addDirective({
+ name: "foreach",
+ extraNames: ["as"],
+ priority: 10,
+ atNodeEncounter({ node, qweb, ctx }) {
+ ctx.rootContext.shouldDefineScope = true;
+ ctx = ctx.subContext("loopNumber", ctx.loopNumber + 1);
+ const elems = node.getAttribute("t-foreach");
+ const name = node.getAttribute("t-as");
+ let arrayID = ctx.generateID();
+ ctx.addLine(`let _${arrayID} = ${ctx.formatExpression(elems)};`);
+ ctx.addLine(`if (!_${arrayID}) { throw new Error('QWeb error: Invalid loop expression')}`);
+ let keysID = ctx.generateID();
+ let valuesID = ctx.generateID();
+ ctx.addLine(`let _${keysID} = _${valuesID} = _${arrayID};`);
+ ctx.addIf(`!(_${arrayID} instanceof Array)`);
+ ctx.addLine(`_${keysID} = Object.keys(_${arrayID});`);
+ ctx.addLine(`_${valuesID} = Object.values(_${arrayID});`);
+ ctx.closeIf();
+ ctx.addLine(`let _length${keysID} = _${keysID}.length;`);
+ let varsID = ctx.startProtectScope(true);
+ const loopVar = `i${ctx.loopNumber}`;
+ ctx.addLine(`for (let ${loopVar} = 0; ${loopVar} < _length${keysID}; ${loopVar}++) {`);
+ ctx.indent();
+ ctx.addLine(`scope.${name}_first = ${loopVar} === 0`);
+ ctx.addLine(`scope.${name}_last = ${loopVar} === _length${keysID} - 1`);
+ ctx.addLine(`scope.${name}_index = ${loopVar}`);
+ ctx.addLine(`scope.${name} = _${keysID}[${loopVar}]`);
+ ctx.addLine(`scope.${name}_value = _${valuesID}[${loopVar}]`);
+ const nodeCopy = node.cloneNode(true);
+ let shouldWarn = !nodeCopy.hasAttribute("t-key") &&
+ node.children.length === 1 &&
+ node.children[0].tagName !== "t" &&
+ !node.children[0].hasAttribute("t-key");
+ if (shouldWarn) {
+ console.warn(`Directive t-foreach should always be used with a t-key! (in template: '${ctx.templateName}')`);
+ }
+ if (nodeCopy.hasAttribute("t-key")) {
+ const expr = ctx.formatExpression(nodeCopy.getAttribute("t-key"));
+ ctx.addLine(`let key${ctx.loopNumber} = ${expr};`);
+ nodeCopy.removeAttribute("t-key");
+ }
+ else {
+ ctx.addLine(`let key${ctx.loopNumber} = i${ctx.loopNumber};`);
+ }
+ nodeCopy.removeAttribute("t-foreach");
+ qweb._compileNode(nodeCopy, ctx);
+ ctx.dedent();
+ ctx.addLine("}");
+ ctx.stopProtectScope(varsID);
+ return true;
+ },
+ });
+ //------------------------------------------------------------------------------
+ // t-debug
+ //------------------------------------------------------------------------------
+ QWeb.addDirective({
+ name: "debug",
+ priority: 1,
+ atNodeEncounter({ ctx }) {
+ ctx.addLine("debugger;");
+ },
+ });
+ //------------------------------------------------------------------------------
+ // t-log
+ //------------------------------------------------------------------------------
+ QWeb.addDirective({
+ name: "log",
+ priority: 1,
+ atNodeEncounter({ ctx, value }) {
+ const expr = ctx.formatExpression(value);
+ ctx.addLine(`console.log(${expr})`);
+ },
+ });
+
+ /**
+ * Owl QWeb Extensions
+ *
+ * This file contains the implementation of non standard QWeb directives, added
+ * by Owl and that will only work on Owl projects:
+ *
+ * - t-on
+ * - t-ref
+ * - t-transition
+ * - t-mounted
+ * - t-slot
+ * - t-model
+ */
+ //------------------------------------------------------------------------------
+ // t-on
+ //------------------------------------------------------------------------------
+ // these are pieces of code that will be injected into the event handler if
+ // modifiers are specified
+ const MODS_CODE = {
+ prevent: "e.preventDefault();",
+ self: "if (e.target !== this.elm) {return}",
+ stop: "e.stopPropagation();",
+ };
+ const FNAMEREGEXP = /^[$A-Z_][0-9A-Z_$]*$/i;
+ function makeHandlerCode(ctx, fullName, value, putInCache, modcodes = MODS_CODE) {
+ let [event, ...mods] = fullName.slice(5).split(".");
+ if (mods.includes("capture")) {
+ event = "!" + event;
+ }
+ if (!event) {
+ throw new Error("Missing event name with t-on directive");
+ }
+ let code;
+ // check if it is a method with no args, a method with args or an expression
+ let args = "";
+ const name = value.replace(/\(.*\)/, function (_args) {
+ args = _args.slice(1, -1);
+ return "";
+ });
+ const isMethodCall = name.match(FNAMEREGEXP);
+ // then generate code
+ if (isMethodCall) {
+ ctx.rootContext.shouldDefineUtils = true;
+ const comp = `utils.getComponent(context)`;
+ if (args) {
+ const argId = ctx.generateID();
+ ctx.addLine(`let args${argId} = [${ctx.formatExpression(args)}];`);
+ code = `${comp}['${name}'](...args${argId}, e);`;
+ putInCache = false;
+ }
+ else {
+ code = `${comp}['${name}'](e);`;
+ }
+ }
+ else {
+ // if we get here, then it is an expression
+ // we need to capture every variable in it
+ putInCache = false;
+ code = ctx.captureExpression(value);
+ }
+ const modCode = mods.map((mod) => modcodes[mod]).join("");
+ let handler = `function (e) {if (!context.__owl__.isMounted){return}${modCode}${code}}`;
+ if (putInCache) {
+ const key = ctx.generateTemplateKey(event);
+ ctx.addLine(`extra.handlers[${key}] = extra.handlers[${key}] || ${handler};`);
+ handler = `extra.handlers[${key}]`;
+ }
+ return { event, handler };
+ }
+ QWeb.addDirective({
+ name: "on",
+ priority: 90,
+ atNodeCreation({ ctx, fullName, value, nodeID }) {
+ const { event, handler } = makeHandlerCode(ctx, fullName, value, true);
+ ctx.addLine(`p${nodeID}.on['${event}'] = ${handler};`);
+ },
+ });
+ //------------------------------------------------------------------------------
+ // t-ref
+ //------------------------------------------------------------------------------
+ QWeb.addDirective({
+ name: "ref",
+ priority: 95,
+ atNodeCreation({ ctx, value, addNodeHook }) {
+ ctx.rootContext.shouldDefineRefs = true;
+ const refKey = `ref${ctx.generateID()}`;
+ ctx.addLine(`const ${refKey} = ${ctx.interpolate(value)};`);
+ addNodeHook("create", `context.__owl__.refs[${refKey}] = n.elm;`);
+ addNodeHook("destroy", `delete context.__owl__.refs[${refKey}];`);
+ },
+ });
+ //------------------------------------------------------------------------------
+ // t-transition
+ //------------------------------------------------------------------------------
+ QWeb.utils.nextFrame = function (cb) {
+ requestAnimationFrame(() => requestAnimationFrame(cb));
+ };
+ QWeb.utils.transitionInsert = function (vn, name) {
+ const elm = vn.elm;
+ // remove potential duplicated vnode that is currently being removed, to
+ // prevent from having twice the same node in the DOM during an animation
+ const dup = elm.parentElement && elm.parentElement.querySelector(`*[data-owl-key='${vn.key}']`);
+ if (dup) {
+ dup.remove();
+ }
+ elm.classList.add(name + "-enter");
+ elm.classList.add(name + "-enter-active");
+ elm.classList.remove(name + "-leave-active");
+ elm.classList.remove(name + "-leave-to");
+ const finalize = () => {
+ elm.classList.remove(name + "-enter-active");
+ elm.classList.remove(name + "-enter-to");
+ };
+ this.nextFrame(() => {
+ elm.classList.remove(name + "-enter");
+ elm.classList.add(name + "-enter-to");
+ whenTransitionEnd(elm, finalize);
+ });
+ };
+ QWeb.utils.transitionRemove = function (vn, name, rm) {
+ const elm = vn.elm;
+ elm.setAttribute("data-owl-key", vn.key);
+ elm.classList.add(name + "-leave");
+ elm.classList.add(name + "-leave-active");
+ const finalize = () => {
+ if (!elm.classList.contains(name + "-leave-active")) {
+ return;
+ }
+ elm.classList.remove(name + "-leave-active");
+ elm.classList.remove(name + "-leave-to");
+ rm();
+ };
+ this.nextFrame(() => {
+ elm.classList.remove(name + "-leave");
+ elm.classList.add(name + "-leave-to");
+ whenTransitionEnd(elm, finalize);
+ });
+ };
+ function getTimeout(delays, durations) {
+ /* istanbul ignore next */
+ while (delays.length < durations.length) {
+ delays = delays.concat(delays);
+ }
+ return Math.max.apply(null, durations.map((d, i) => {
+ return toMs(d) + toMs(delays[i]);
+ }));
+ }
+ // Old versions of Chromium (below 61.0.3163.100) formats floating pointer numbers
+ // in a locale-dependent way, using a comma instead of a dot.
+ // If comma is not replaced with a dot, the input will be rounded down (i.e. acting
+ // as a floor function) causing unexpected behaviors
+ function toMs(s) {
+ return Number(s.slice(0, -1).replace(",", ".")) * 1000;
+ }
+ function whenTransitionEnd(elm, cb) {
+ if (!elm.parentNode) {
+ // if we get here, this means that the element was removed for some other
+ // reasons, and in that case, we don't want to work on animation since nothing
+ // will be displayed anyway.
+ return;
+ }
+ const styles = window.getComputedStyle(elm);
+ const delays = (styles.transitionDelay || "").split(", ");
+ const durations = (styles.transitionDuration || "").split(", ");
+ const timeout = getTimeout(delays, durations);
+ if (timeout > 0) {
+ elm.addEventListener("transitionend", cb, { once: true });
+ }
+ else {
+ cb();
+ }
+ }
+ QWeb.addDirective({
+ name: "transition",
+ priority: 96,
+ atNodeCreation({ ctx, value, addNodeHook }) {
+ ctx.rootContext.shouldDefineUtils = true;
+ let name = value;
+ const hooks = {
+ insert: `utils.transitionInsert(vn, '${name}');`,
+ remove: `utils.transitionRemove(vn, '${name}', rm);`,
+ };
+ for (let hookName in hooks) {
+ addNodeHook(hookName, hooks[hookName]);
+ }
+ },
+ });
+ //------------------------------------------------------------------------------
+ // t-slot
+ //------------------------------------------------------------------------------
+ QWeb.addDirective({
+ name: "slot",
+ priority: 80,
+ atNodeEncounter({ ctx, value, node, qweb }) {
+ const slotKey = ctx.generateID();
+ ctx.addLine(`const slot${slotKey} = this.constructor.slots[context.__owl__.slotId + '_' + '${value}'];`);
+ ctx.addIf(`slot${slotKey}`);
+ let parentNode = `c${ctx.parentNode}`;
+ if (!ctx.parentNode) {
+ ctx.rootContext.shouldDefineResult = true;
+ ctx.rootContext.shouldDefineUtils = true;
+ parentNode = `children${ctx.generateID()}`;
+ ctx.addLine(`let ${parentNode}= []`);
+ ctx.addLine(`result = {}`);
+ }
+ ctx.addLine(`slot${slotKey}.call(this, context.__owl__.scope, Object.assign({}, extra, {parentNode: ${parentNode}, parent: extra.parent || context}));`);
+ if (!ctx.parentNode) {
+ ctx.addLine(`utils.defineProxy(result, ${parentNode}[0]);`);
+ }
+ if (node.hasChildNodes()) {
+ ctx.addElse();
+ const nodeCopy = node.cloneNode(true);
+ nodeCopy.removeAttribute("t-slot");
+ qweb._compileNode(nodeCopy, ctx);
+ }
+ ctx.closeIf();
+ return true;
+ },
+ });
+ //------------------------------------------------------------------------------
+ // t-model
+ //------------------------------------------------------------------------------
+ QWeb.utils.toNumber = function (val) {
+ const n = parseFloat(val);
+ return isNaN(n) ? val : n;
+ };
+ QWeb.addDirective({
+ name: "model",
+ priority: 42,
+ atNodeCreation({ ctx, nodeID, value, node, fullName, addNodeHook }) {
+ const type = node.getAttribute("type");
+ let handler;
+ let event = fullName.includes(".lazy") ? "change" : "input";
+ // we keep here a reference to the "base expression" (if the expression
+ // is `t-model="some.expr.value", then the base expression is "some.expr").
+ // This is necessary so we can capture it in the handler closure.
+ let expr = ctx.formatExpression(value);
+ const index = expr.lastIndexOf(".");
+ const baseExpr = expr.slice(0, index);
+ ctx.addLine(`let expr${nodeID} = ${baseExpr};`);
+ expr = `expr${nodeID}.${expr.slice(index + 1)}`;
+ const key = ctx.generateTemplateKey();
+ if (node.tagName === "select") {
+ ctx.addLine(`p${nodeID}.props = {value: ${expr}};`);
+ addNodeHook("create", `n.elm.value=${expr};`);
+ event = "change";
+ handler = `(ev) => {${expr} = ev.target.value}`;
+ }
+ else if (type === "checkbox") {
+ ctx.addLine(`p${nodeID}.props = {checked: ${expr}};`);
+ handler = `(ev) => {${expr} = ev.target.checked}`;
+ }
+ else if (type === "radio") {
+ const nodeValue = node.getAttribute("value");
+ ctx.addLine(`p${nodeID}.props = {checked:${expr} === '${nodeValue}'};`);
+ handler = `(ev) => {${expr} = ev.target.value}`;
+ event = "click";
+ }
+ else {
+ ctx.addLine(`p${nodeID}.props = {value: ${expr}};`);
+ const trimCode = fullName.includes(".trim") ? ".trim()" : "";
+ let valueCode = `ev.target.value${trimCode}`;
+ if (fullName.includes(".number")) {
+ ctx.rootContext.shouldDefineUtils = true;
+ valueCode = `utils.toNumber(${valueCode})`;
+ }
+ handler = `(ev) => {${expr} = ${valueCode}}`;
+ }
+ ctx.addLine(`extra.handlers[${key}] = extra.handlers[${key}] || (${handler});`);
+ ctx.addLine(`p${nodeID}.on['${event}'] = extra.handlers[${key}];`);
+ },
+ });
+ //------------------------------------------------------------------------------
+ // t-key
+ //------------------------------------------------------------------------------
+ QWeb.addDirective({
+ name: "key",
+ priority: 45,
+ atNodeEncounter({ ctx, value, node }) {
+ if (ctx.loopNumber === 0) {
+ ctx.keyStack.push(ctx.rootContext.hasKey0);
+ ctx.rootContext.hasKey0 = true;
+ }
+ ctx.addLine("{");
+ ctx.indent();
+ ctx.addLine(`let key${ctx.loopNumber} = ${ctx.formatExpression(value)};`);
+ },
+ finalize({ ctx }) {
+ ctx.dedent();
+ ctx.addLine("}");
+ if (ctx.loopNumber === 0) {
+ ctx.rootContext.hasKey0 = ctx.keyStack.pop();
+ }
+ },
+ });
+
+ const config = {};
+ Object.defineProperty(config, "mode", {
+ get() {
+ return QWeb.dev ? "dev" : "prod";
+ },
+ set(mode) {
+ QWeb.dev = mode === "dev";
+ if (QWeb.dev) {
+ const url = `https://github.com/odoo/owl/blob/master/doc/reference/config.md#mode`;
+ console.warn(`Owl is running in 'dev' mode. This is not suitable for production use. See ${url} for more information.`);
+ }
+ else {
+ console.log(`Owl is now running in 'prod' mode.`);
+ }
+ },
+ });
+
+ /**
+ * We define here OwlEvent, a subclass of CustomEvent, with an additional
+ * attribute:
+ * - originalComponent: the component that triggered the event
+ */
+ class OwlEvent extends CustomEvent {
+ constructor(component, eventType, options) {
+ super(eventType, options);
+ this.originalComponent = component;
+ }
+ }
+
+ //------------------------------------------------------------------------------
+ // t-component
+ //------------------------------------------------------------------------------
+ const T_COMPONENT_MODS_CODE = Object.assign({}, MODS_CODE, {
+ self: "if (e.target !== vn.elm) {return}",
+ });
+ QWeb.utils.defineProxy = function defineProxy(target, source) {
+ for (let k in source) {
+ Object.defineProperty(target, k, {
+ get() {
+ return source[k];
+ },
+ set(val) {
+ source[k] = val;
+ },
+ });
+ }
+ };
+ QWeb.utils.assignHooks = function assignHooks(dataObj, hooks) {
+ if ("hook" in dataObj) {
+ const hookObject = dataObj.hook;
+ for (let name in hooks) {
+ const current = hookObject[name];
+ const fn = hooks[name];
+ if (current) {
+ hookObject[name] = (...args) => {
+ current(...args);
+ fn(...args);
+ };
+ }
+ else {
+ hookObject[name] = fn;
+ }
+ }
+ }
+ else {
+ dataObj.hook = hooks;
+ }
+ };
+ /**
+ * The t-component directive is certainly a complicated and hard to maintain piece
+ * of code. To help you, fellow developer, if you have to maintain it, I offer
+ * you this advice: Good luck...
+ *
+ * Since it is not 'direct' code, but rather code that generates other code, it
+ * is not easy to understand. To help you, here is a detailed and commented
+ * explanation of the code generated by the t-component directive for the following
+ * situation:
+ * ```xml
+ *
+ * ```
+ *
+ * ```js
+ * // we assign utils on top of the function because it will be useful for
+ * // each components
+ * let utils = this.utils;
+ *
+ * // this is the virtual node representing the parent div
+ * let c1 = [], p1 = { key: 1 };
+ * var vn1 = h("div", p1, c1);
+ *
+ * // t-component directive: we start by evaluating the expression given by t-key:
+ * let key5 = "somestring";
+ *
+ * // def3 is the promise that will contain later either the new component
+ * // creation, or the props update...
+ * let def3;
+ *
+ * // this is kind of tricky: we need here to find if the component was already
+ * // created by a previous rendering. This is done by checking the internal
+ * // `cmap` (children map) of the parent component: it maps keys to component ids,
+ * // and, then, if there is an id, we look into the children list to get the
+ * // instance
+ * let w4 =
+ * key5 in context.__owl__.cmap
+ * ? context.__owl__.children[context.__owl__.cmap[key5]]
+ * : false;
+ *
+ * // We keep the index of the position of the component in the closure. We push
+ * // null to reserve the slot, and will replace it later by the component vnode,
+ * // when it will be ready (do not forget that preparing/rendering a component is
+ * // asynchronous)
+ * let _2_index = c1.length;
+ * c1.push(null);
+ *
+ * // we evaluate here the props given to the component. It is done here to be
+ * // able to easily reference it later, and also, it might be an expensive
+ * // computation, so it is certainly better to do it only once
+ * let props4 = { flag: context["state"].flag };
+ *
+ * // If we have a component, currently rendering, but not ready yet, we do not want
+ * // to wait for it to be ready if we can avoid it
+ * if (w4 && w4.__owl__.renderPromise && !w4.__owl__.vnode) {
+ * // we check if the props are the same. In that case, we can simply reuse
+ * // the previous rendering and skip all useless work
+ * if (utils.shallowEqual(props4, w4.__owl__.renderProps)) {
+ * def3 = w4.__owl__.renderPromise;
+ * } else {
+ * // if the props are not the same, we destroy the component and starts anew.
+ * // this will be faster than waiting for its rendering, then updating it
+ * w4.destroy();
+ * w4 = false;
+ * }
+ * }
+ *
+ * if (!w4) {
+ * // in this situation, we need to create a new component. First step is
+ * // to get a reference to the class, then create an instance with
+ * // current context as parent, and the props.
+ * let W4 = context.component && context.components[componentKey4] || QWeb.component[componentKey4];
+
+ * if (!W4) {
+ * throw new Error("Cannot find the definition of component 'child'");
+ * }
+ * w4 = new W4(owner, props4);
+ *
+ * // Whenever we rerender the parent component, we need to be sure that we
+ * // are able to find the component instance. To do that, we register it to
+ * // the parent cmap (children map). Note that the 'template' key is
+ * // used here, since this is what identify the component from the template
+ * // perspective.
+ * context.__owl__.cmap[key5] = w4.__owl__.id;
+ *
+ * // __prepare is called, to basically call willStart, then render the
+ * // component
+ * def3 = w4.__prepare();
+ *
+ * def3 = def3.then(vnode => {
+ * // we create here a virtual node for the parent (NOT the component). This
+ * // means that the vdom of the parent will be stopped here, and from
+ * // the parent's perspective, it simply is a vnode with no children.
+ * // However, it shares the same dom element with the component root
+ * // vnode.
+ * let pvnode = h(vnode.sel, { key: key5 });
+ *
+ * // we add hooks to the parent vnode so we can interact with the new
+ * // component at the proper time
+ * pvnode.data.hook = {
+ * insert(vn) {
+ * // the __mount method will patch the component vdom into the elm vn.elm,
+ * // then call the mounted hooks. However, suprisingly, the snabbdom
+ * // patch method actually replace the elm by a new elm, so we need
+ * // to synchronise the pvnode elm with the resulting elm
+ * let nvn = w4.__mount(vnode, vn.elm);
+ * pvnode.elm = nvn.elm;
+ * // what follows is only present if there are animations on the component
+ * utils.transitionInsert(vn, "fade");
+ * },
+ * remove() {
+ * // override with empty function to prevent from removing the node
+ * // directly. It will be removed when destroy is called anyway, which
+ * // delays the removal if there are animations.
+ * },
+ * destroy() {
+ * // if there are animations, we delay the call to destroy on the
+ * // component, if not, we call it directly.
+ * let finalize = () => {
+ * w4.destroy();
+ * };
+ * utils.transitionRemove(vn, "fade", finalize);
+ * }
+ * };
+ * // the pvnode is inserted at the correct position in the div's children
+ * c1[_2_index] = pvnode;
+ *
+ * // we keep here a reference to the parent vnode (representing the
+ * // component, so we can reuse it later whenever we update the component
+ * w4.__owl__.pvnode = pvnode;
+ * });
+ * } else {
+ * // this is the 'update' path of the directive.
+ * // the call to __updateProps is the actual component update
+ * // Note that we only update the props if we cannot reuse the previous
+ * // rendering work (in the case it was rendered with the same props)
+ * def3 = def3 || w4.__updateProps(props4, extra.forceUpdate, extra.patchQueue);
+ * def3 = def3.then(() => {
+ * // if component was destroyed in the meantime, we do nothing (so, this
+ * // means that the parent's element children list will have a null in
+ * // the component's position, which will cause the pvnode to be removed
+ * // when it is patched.
+ * if (w4.__owl__.isDestroyed) {
+ * return;
+ * }
+ * // like above, we register the pvnode to the children list, so it
+ * // will not be patched out of the dom.
+ * let pvnode = w4.__owl__.pvnode;
+ * c1[_2_index] = pvnode;
+ * });
+ * }
+ *
+ * // we register the deferred here so the parent can coordinate its patch operation
+ * // with all the children.
+ * extra.promises.push(def3);
+ * return vn1;
+ * ```
+ */
+ QWeb.addDirective({
+ name: "component",
+ extraNames: ["props"],
+ priority: 100,
+ atNodeEncounter({ ctx, value, node, qweb }) {
+ ctx.addLine(`// Component '${value}'`);
+ ctx.rootContext.shouldDefineQWeb = true;
+ ctx.rootContext.shouldDefineParent = true;
+ ctx.rootContext.shouldDefineUtils = true;
+ ctx.rootContext.shouldDefineScope = true;
+ let hasDynamicProps = node.getAttribute("t-props") ? true : false;
+ // t-on- events and t-transition
+ const events = [];
+ let transition = "";
+ const attributes = node.attributes;
+ const props = {};
+ for (let i = 0; i < attributes.length; i++) {
+ const name = attributes[i].name;
+ const value = attributes[i].textContent;
+ if (name.startsWith("t-on-")) {
+ events.push([name, value]);
+ }
+ else if (name === "t-transition") {
+ transition = value;
+ }
+ else if (!name.startsWith("t-")) {
+ if (name !== "class" && name !== "style") {
+ // this is a prop!
+ props[name] = ctx.formatExpression(value) || "undefined";
+ }
+ }
+ }
+ // computing the props string representing the props object
+ let propStr = Object.keys(props)
+ .map((k) => k + ":" + props[k])
+ .join(",");
+ let componentID = ctx.generateID();
+ const templateKey = ctx.generateTemplateKey();
+ let ref = node.getAttribute("t-ref");
+ let refExpr = "";
+ let refKey = "";
+ if (ref) {
+ ctx.rootContext.shouldDefineRefs = true;
+ refKey = `ref${ctx.generateID()}`;
+ ctx.addLine(`const ${refKey} = ${ctx.interpolate(ref)};`);
+ refExpr = `context.__owl__.refs[${refKey}] = w${componentID};`;
+ }
+ let finalizeComponentCode = `w${componentID}.destroy();`;
+ if (ref) {
+ finalizeComponentCode += `delete context.__owl__.refs[${refKey}];`;
+ }
+ if (transition) {
+ finalizeComponentCode = `let finalize = () => {
+ ${finalizeComponentCode}
+ };
+ delete w${componentID}.__owl__.transitionInserted;
+ utils.transitionRemove(vn, '${transition}', finalize);`;
+ }
+ let createHook = "";
+ let classAttr = node.getAttribute("class");
+ let tattClass = node.getAttribute("t-att-class");
+ let styleAttr = node.getAttribute("style");
+ let tattStyle = node.getAttribute("t-att-style");
+ if (tattStyle) {
+ const attVar = `_${ctx.generateID()}`;
+ ctx.addLine(`const ${attVar} = ${ctx.formatExpression(tattStyle)};`);
+ tattStyle = attVar;
+ }
+ let classObj = "";
+ if (classAttr || tattClass || styleAttr || tattStyle || events.length) {
+ if (classAttr) {
+ let classDef = classAttr
+ .trim()
+ .split(/\s+/)
+ .map((a) => `'${a}':true`)
+ .join(",");
+ classObj = `_${ctx.generateID()}`;
+ ctx.addLine(`let ${classObj} = {${classDef}};`);
+ }
+ if (tattClass) {
+ let tattExpr = ctx.formatExpression(tattClass);
+ if (tattExpr[0] !== "{" || tattExpr[tattExpr.length - 1] !== "}") {
+ tattExpr = `utils.toObj(${tattExpr})`;
+ }
+ if (classAttr) {
+ ctx.addLine(`Object.assign(${classObj}, ${tattExpr})`);
+ }
+ else {
+ classObj = `_${ctx.generateID()}`;
+ ctx.addLine(`let ${classObj} = ${tattExpr};`);
+ }
+ }
+ let eventsCode = events
+ .map(function ([name, value]) {
+ const capture = name.match(/\.capture/);
+ name = capture ? name.replace(/\.capture/, "") : name;
+ const { event, handler } = makeHandlerCode(ctx, name, value, false, T_COMPONENT_MODS_CODE);
+ if (capture) {
+ return `vn.elm.addEventListener('${event}', ${handler}, true);`;
+ }
+ return `vn.elm.addEventListener('${event}', ${handler});`;
+ })
+ .join("");
+ const styleExpr = tattStyle || (styleAttr ? `'${styleAttr}'` : false);
+ const styleCode = styleExpr ? `vn.elm.style = ${styleExpr};` : "";
+ createHook = `utils.assignHooks(vnode.data, {create(_, vn){${styleCode}${eventsCode}}});`;
+ }
+ ctx.addLine(`let w${componentID} = ${templateKey} in parent.__owl__.cmap ? parent.__owl__.children[parent.__owl__.cmap[${templateKey}]] : false;`);
+ let shouldProxy = !ctx.parentNode;
+ if (shouldProxy) {
+ let id = ctx.generateID();
+ ctx.rootContext.rootNode = id;
+ shouldProxy = true;
+ ctx.rootContext.shouldDefineResult = true;
+ ctx.addLine(`let vn${id} = {};`);
+ ctx.addLine(`result = vn${id};`);
+ }
+ if (hasDynamicProps) {
+ const dynamicProp = ctx.formatExpression(node.getAttribute("t-props"));
+ ctx.addLine(`let props${componentID} = Object.assign({${propStr}}, ${dynamicProp});`);
+ }
+ else {
+ ctx.addLine(`let props${componentID} = {${propStr}};`);
+ }
+ ctx.addIf(`w${componentID} && w${componentID}.__owl__.currentFiber && !w${componentID}.__owl__.vnode`);
+ ctx.addLine(`w${componentID}.destroy();`);
+ ctx.addLine(`w${componentID} = false;`);
+ ctx.closeIf();
+ let registerCode = "";
+ if (shouldProxy) {
+ registerCode = `utils.defineProxy(vn${ctx.rootNode}, pvnode);`;
+ }
+ // SLOTS
+ const hasSlots = node.childNodes.length;
+ let scope = hasSlots ? `Object.assign(Object.create(context), scope)` : "undefined";
+ ctx.addIf(`w${componentID}`);
+ // need to update component
+ let styleCode = "";
+ if (tattStyle) {
+ styleCode = `.then(()=>{if (w${componentID}.__owl__.isDestroyed) {return};w${componentID}.el.style=${tattStyle};});`;
+ }
+ ctx.addLine(`w${componentID}.__updateProps(props${componentID}, extra.fiber, ${scope})${styleCode};`);
+ ctx.addLine(`let pvnode = w${componentID}.__owl__.pvnode;`);
+ if (registerCode) {
+ ctx.addLine(registerCode);
+ }
+ if (ctx.parentNode) {
+ ctx.addLine(`c${ctx.parentNode}.push(pvnode);`);
+ }
+ ctx.addElse();
+ // new component
+ let dynamicFallback = "";
+ if (!value.match(INTERP_REGEXP)) {
+ dynamicFallback = `|| ${ctx.formatExpression(value)}`;
+ }
+ const interpValue = ctx.interpolate(value);
+ ctx.addLine(`let componentKey${componentID} = ${interpValue};`);
+ ctx.addLine(`let W${componentID} = context.constructor.components[componentKey${componentID}] || QWeb.components[componentKey${componentID}]${dynamicFallback};`);
+ // maybe only do this in dev mode...
+ ctx.addLine(`if (!W${componentID}) {throw new Error('Cannot find the definition of component "' + componentKey${componentID} + '"')}`);
+ ctx.addLine(`w${componentID} = new W${componentID}(parent, props${componentID});`);
+ if (transition) {
+ ctx.addLine(`const __patch${componentID} = w${componentID}.__patch;`);
+ ctx.addLine(`w${componentID}.__patch = (t, vn) => {__patch${componentID}.call(w${componentID}, t, vn); if(!w${componentID}.__owl__.transitionInserted){w${componentID}.__owl__.transitionInserted = true;utils.transitionInsert(w${componentID}.__owl__.vnode, '${transition}');}};`);
+ }
+ ctx.addLine(`parent.__owl__.cmap[${templateKey}] = w${componentID}.__owl__.id;`);
+ if (hasSlots) {
+ const clone = node.cloneNode(true);
+ const slotNodes = Array.from(clone.querySelectorAll("[t-set-slot]"));
+ // The next code is a fallback for compatibility reason. It accepts t-set
+ // elements that are direct children with a non empty body as nodes defining
+ // the content of a slot.
+ //
+ // This is wrong, but is necessary to prevent breaking all existing Owl
+ // code using slots. This will be removed in v2.0 someday. Meanwhile,
+ // please use t-set-slot everywhere you need to set the content of a
+ // slot.
+ for (let el of clone.children) {
+ if (el.getAttribute("t-set") && el.hasChildNodes()) {
+ slotNodes.push(el);
+ }
+ }
+ const slotId = QWeb.nextSlotId++;
+ ctx.addLine(`w${componentID}.__owl__.slotId = ${slotId};`);
+ if (slotNodes.length) {
+ for (let i = 0, length = slotNodes.length; i < length; i++) {
+ const slotNode = slotNodes[i];
+ slotNode.parentElement.removeChild(slotNode);
+ let key = slotNode.getAttribute("t-set-slot");
+ slotNode.removeAttribute("t-set-slot");
+ // here again, this code should be removed when we stop supporting
+ // using t-set to define the content of named slots.
+ if (!key) {
+ key = slotNode.getAttribute("t-set");
+ slotNode.removeAttribute("t-set");
+ }
+ const slotFn = qweb._compile(`slot_${key}_template`, slotNode, ctx);
+ QWeb.slots[`${slotId}_${key}`] = slotFn;
+ }
+ }
+ if (clone.childNodes.length) {
+ const t = clone.ownerDocument.createElement("t");
+ for (let child of Object.values(clone.childNodes)) {
+ t.appendChild(child);
+ }
+ const slotFn = qweb._compile(`slot_default_template`, t, ctx);
+ QWeb.slots[`${slotId}_default`] = slotFn;
+ }
+ }
+ ctx.addLine(`let fiber = w${componentID}.__prepare(extra.fiber, ${scope}, () => { const vnode = fiber.vnode; pvnode.sel = vnode.sel; ${createHook}});`);
+ // hack: specify empty remove hook to prevent the node from being removed from the DOM
+ const insertHook = refExpr ? `insert(vn) {${refExpr}},` : "";
+ ctx.addLine(`let pvnode = h('dummy', {key: ${templateKey}, hook: {${insertHook}remove() {},destroy(vn) {${finalizeComponentCode}}}});`);
+ if (registerCode) {
+ ctx.addLine(registerCode);
+ }
+ if (ctx.parentNode) {
+ ctx.addLine(`c${ctx.parentNode}.push(pvnode);`);
+ }
+ ctx.addLine(`w${componentID}.__owl__.pvnode = pvnode;`);
+ ctx.closeIf();
+ if (classObj) {
+ ctx.addLine(`w${componentID}.__owl__.classObj=${classObj};`);
+ }
+ ctx.addLine(`w${componentID}.__owl__.parentLastFiberId = extra.fiber.id;`);
+ return true;
+ },
+ });
+
+ class Scheduler {
+ constructor(requestAnimationFrame) {
+ this.tasks = [];
+ this.isRunning = false;
+ this.requestAnimationFrame = requestAnimationFrame;
+ }
+ start() {
+ this.isRunning = true;
+ this.scheduleTasks();
+ }
+ stop() {
+ this.isRunning = false;
+ }
+ addFiber(fiber) {
+ // if the fiber was remapped into a larger rendering fiber, it may not be a
+ // root fiber. But we only want to register root fibers
+ fiber = fiber.root;
+ return new Promise((resolve, reject) => {
+ if (fiber.error) {
+ return reject(fiber.error);
+ }
+ this.tasks.push({
+ fiber,
+ callback: () => {
+ if (fiber.error) {
+ return reject(fiber.error);
+ }
+ resolve();
+ },
+ });
+ if (!this.isRunning) {
+ this.start();
+ }
+ });
+ }
+ rejectFiber(fiber, reason) {
+ fiber = fiber.root;
+ const index = this.tasks.findIndex((t) => t.fiber === fiber);
+ if (index >= 0) {
+ const [task] = this.tasks.splice(index, 1);
+ fiber.cancel();
+ fiber.error = new Error(reason);
+ task.callback();
+ }
+ }
+ /**
+ * Process all current tasks. This only applies to the fibers that are ready.
+ * Other tasks are left unchanged.
+ */
+ flush() {
+ let tasks = this.tasks;
+ this.tasks = [];
+ tasks = tasks.filter((task) => {
+ if (task.fiber.isCompleted) {
+ task.callback();
+ return false;
+ }
+ if (task.fiber.counter === 0) {
+ if (!task.fiber.error) {
+ try {
+ task.fiber.complete();
+ }
+ catch (e) {
+ task.fiber.handleError(e);
+ }
+ }
+ task.callback();
+ return false;
+ }
+ return true;
+ });
+ this.tasks = tasks.concat(this.tasks);
+ if (this.tasks.length === 0) {
+ this.stop();
+ }
+ }
+ scheduleTasks() {
+ this.requestAnimationFrame(() => {
+ this.flush();
+ if (this.isRunning) {
+ this.scheduleTasks();
+ }
+ });
+ }
+ }
+ const scheduler = new Scheduler(browser.requestAnimationFrame);
+
+ /**
+ * Owl Fiber Class
+ *
+ * Fibers are small abstractions designed to contain all the internal state
+ * associated with a "rendering work unit", relative to a specific component.
+ *
+ * A rendering will cause the creation of a fiber for each impacted components.
+ *
+ * Fibers capture all that necessary information, which is critical to owl
+ * asynchronous rendering pipeline. Fibers can be cancelled, can be in different
+ * states and in general determine the state of the rendering.
+ */
+ let Fiber = /** @class */ (() => {
+ class Fiber {
+ constructor(parent, component, force, target, position) {
+ this.id = Fiber.nextId++;
+ // isCompleted means that the rendering corresponding to this fiber's work is
+ // done, either because the component has been mounted or patched, or because
+ // fiber has been cancelled.
+ this.isCompleted = false;
+ // the fibers corresponding to component updates (updateProps) need to call
+ // the willPatch and patched hooks from the corresponding component. However,
+ // fibers corresponding to a new component do not need to do that. So, the
+ // shouldPatch hook is the boolean that we check whenever we need to apply
+ // a patch.
+ this.shouldPatch = true;
+ // isRendered is the last state of a fiber. If true, this means that it has
+ // been rendered and is inert (so, it should not be taken into account when
+ // counting the number of active fibers).
+ this.isRendered = false;
+ // the counter number is a critical information. It is only necessary for a
+ // root fiber. For that fiber, this number counts the number of active sub
+ // fibers. When that number reaches 0, the fiber can be applied by the
+ // scheduler.
+ this.counter = 0;
+ this.vnode = null;
+ this.child = null;
+ this.sibling = null;
+ this.lastChild = null;
+ this.parent = null;
+ this.component = component;
+ this.force = force;
+ this.target = target;
+ this.position = position;
+ const __owl__ = component.__owl__;
+ this.scope = __owl__.scope;
+ this.root = parent ? parent.root : this;
+ this.parent = parent;
+ let oldFiber = __owl__.currentFiber;
+ if (oldFiber && !oldFiber.isCompleted) {
+ if (oldFiber.root === oldFiber && !parent) {
+ // both oldFiber and this fiber are root fibers
+ this._reuseFiber(oldFiber);
+ return oldFiber;
+ }
+ else {
+ this._remapFiber(oldFiber);
+ }
+ }
+ this.root.counter++;
+ __owl__.currentFiber = this;
+ }
+ /**
+ * When the oldFiber is not completed yet, and both oldFiber and this fiber
+ * are root fibers, we want to reuse the oldFiber instead of creating a new
+ * one. Doing so will guarantee that the initiator(s) of those renderings will
+ * be notified (the promise will resolve) when the last rendering will be done.
+ *
+ * This function thus assumes that oldFiber is a root fiber.
+ */
+ _reuseFiber(oldFiber) {
+ oldFiber.cancel(); // cancel children fibers
+ oldFiber.isCompleted = false; // keep the root fiber alive
+ oldFiber.isRendered = false; // the fiber has to be re-rendered
+ if (oldFiber.child) {
+ // remove relation to children
+ oldFiber.child.parent = null;
+ oldFiber.child = null;
+ oldFiber.lastChild = null;
+ }
+ oldFiber.counter = 1; // re-initialize counter
+ oldFiber.id = Fiber.nextId++;
+ }
+ /**
+ * In some cases, a rendering initiated at some component can detect that it
+ * should be part of a larger rendering initiated somewhere up the component
+ * tree. In that case, it needs to cancel the previous rendering and
+ * remap itself as a part of the current parent rendering.
+ */
+ _remapFiber(oldFiber) {
+ oldFiber.cancel();
+ this.shouldPatch = oldFiber.shouldPatch;
+ if (oldFiber === oldFiber.root) {
+ oldFiber.counter++;
+ }
+ if (oldFiber.parent && !this.parent) {
+ // re-map links
+ this.parent = oldFiber.parent;
+ this.root = this.parent.root;
+ this.sibling = oldFiber.sibling;
+ if (this.parent.lastChild === oldFiber) {
+ this.parent.lastChild = this;
+ }
+ if (this.parent.child === oldFiber) {
+ this.parent.child = this;
+ }
+ else {
+ let current = this.parent.child;
+ while (true) {
+ if (current.sibling === oldFiber) {
+ current.sibling = this;
+ break;
+ }
+ current = current.sibling;
+ }
+ }
+ }
+ }
+ /**
+ * This function has been taken from
+ * https://medium.com/react-in-depth/the-how-and-why-on-reacts-usage-of-linked-list-in-fiber-67f1014d0eb7
+ */
+ _walk(doWork) {
+ let root = this;
+ let current = this;
+ while (true) {
+ const child = doWork(current);
+ if (child) {
+ current = child;
+ continue;
+ }
+ if (current === root) {
+ return;
+ }
+ while (!current.sibling) {
+ if (!current.parent || current.parent === root) {
+ return;
+ }
+ current = current.parent;
+ }
+ current = current.sibling;
+ }
+ }
+ /**
+ * Successfully complete the work of the fiber: call the mount or patch hooks
+ * and patch the DOM. This function is called once the fiber and its children
+ * are ready, and the scheduler decides to process it.
+ */
+ complete() {
+ let component = this.component;
+ this.isCompleted = true;
+ if (!this.target && !component.__owl__.isMounted) {
+ return;
+ }
+ // build patchQueue
+ const patchQueue = [];
+ const doWork = function (f) {
+ patchQueue.push(f);
+ return f.child;
+ };
+ this._walk(doWork);
+ const patchLen = patchQueue.length;
+ // call willPatch hook on each fiber of patchQueue
+ for (let i = 0; i < patchLen; i++) {
+ const fiber = patchQueue[i];
+ if (fiber.shouldPatch) {
+ component = fiber.component;
+ if (component.__owl__.willPatchCB) {
+ component.__owl__.willPatchCB();
+ }
+ component.willPatch();
+ }
+ }
+ // call __patch on each fiber of (reversed) patchQueue
+ for (let i = patchLen - 1; i >= 0; i--) {
+ const fiber = patchQueue[i];
+ component = fiber.component;
+ if (fiber.target && i === 0) {
+ let target;
+ if (fiber.position === "self") {
+ target = fiber.target;
+ if (target.tagName.toLowerCase() !== fiber.vnode.sel) {
+ throw new Error(`Cannot attach '${component.constructor.name}' to target node (not same tag name)`);
+ }
+ // In self mode, we *know* we are to take possession of the target
+ // Hence we manually create the corresponding VNode and copy the "key" in data
+ const selfVnodeData = fiber.vnode.data ? { key: fiber.vnode.data.key } : {};
+ const selfVnode = h(fiber.vnode.sel, selfVnodeData);
+ selfVnode.elm = target;
+ target = selfVnode;
+ }
+ else {
+ target = component.__owl__.vnode || document.createElement(fiber.vnode.sel);
+ }
+ component.__patch(target, fiber.vnode);
+ }
+ else {
+ if (fiber.shouldPatch) {
+ component.__patch(component.__owl__.vnode, fiber.vnode);
+ }
+ else {
+ component.__patch(document.createElement(fiber.vnode.sel), fiber.vnode);
+ component.__owl__.pvnode.elm = component.__owl__.vnode.elm;
+ }
+ }
+ component.__owl__.currentFiber = null;
+ }
+ // insert into the DOM (mount case)
+ let inDOM = false;
+ if (this.target) {
+ switch (this.position) {
+ case "first-child":
+ this.target.prepend(this.component.el);
+ break;
+ case "last-child":
+ this.target.appendChild(this.component.el);
+ break;
+ }
+ inDOM = document.body.contains(this.component.el);
+ this.component.env.qweb.trigger("dom-appended");
+ }
+ // call patched/mounted hook on each fiber of (reversed) patchQueue
+ for (let i = patchLen - 1; i >= 0; i--) {
+ const fiber = patchQueue[i];
+ component = fiber.component;
+ if (fiber.shouldPatch && !this.target) {
+ component.patched();
+ if (component.__owl__.patchedCB) {
+ component.__owl__.patchedCB();
+ }
+ }
+ else if (this.target ? inDOM : true) {
+ component.__callMounted();
+ }
+ }
+ }
+ /**
+ * Cancel a fiber and all its children.
+ */
+ cancel() {
+ this._walk((f) => {
+ if (!f.isRendered) {
+ f.root.counter--;
+ }
+ f.isCompleted = true;
+ return f.child;
+ });
+ }
+ /**
+ * This is the global error handler for errors occurring in Owl main lifecycle
+ * methods. Caught errors are triggered on the QWeb instance, and are
+ * potentially given to some parent component which implements `catchError`.
+ *
+ * If there are no such component, we destroy everything. This is better than
+ * being in a corrupted state.
+ */
+ handleError(error) {
+ let component = this.component;
+ this.vnode = component.__owl__.vnode || h("div");
+ const qweb = component.env.qweb;
+ let root = component;
+ let canCatch = false;
+ while (component && !(canCatch = !!component.catchError)) {
+ root = component;
+ component = component.__owl__.parent;
+ }
+ qweb.trigger("error", error);
+ if (canCatch) {
+ component.catchError(error);
+ }
+ else {
+ // the 3 next lines aim to mark the root fiber as being in error, and
+ // to force it to end, without waiting for its children
+ this.root.counter = 0;
+ this.root.error = error;
+ scheduler.flush();
+ root.destroy();
+ }
+ }
+ }
+ Fiber.nextId = 1;
+ return Fiber;
+ })();
+
+ //------------------------------------------------------------------------------
+ // Prop validation helper
+ //------------------------------------------------------------------------------
+ /**
+ * Validate the component props (or next props) against the (static) props
+ * description. This is potentially an expensive operation: it may needs to
+ * visit recursively the props and all the children to check if they are valid.
+ * This is why it is only done in 'dev' mode.
+ */
+ QWeb.utils.validateProps = function (Widget, props) {
+ const propsDef = Widget.props;
+ if (propsDef instanceof Array) {
+ // list of strings (prop names)
+ for (let i = 0, l = propsDef.length; i < l; i++) {
+ const propName = propsDef[i];
+ if (propName[propName.length - 1] === "?") {
+ // optional prop
+ break;
+ }
+ if (!(propName in props)) {
+ throw new Error(`Missing props '${propsDef[i]}' (component '${Widget.name}')`);
+ }
+ }
+ for (let key in props) {
+ if (!propsDef.includes(key) && !propsDef.includes(key + "?")) {
+ throw new Error(`Unknown prop '${key}' given to component '${Widget.name}'`);
+ }
+ }
+ }
+ else if (propsDef) {
+ // propsDef is an object now
+ for (let propName in propsDef) {
+ if (props[propName] === undefined) {
+ if (propsDef[propName] && !propsDef[propName].optional) {
+ throw new Error(`Missing props '${propName}' (component '${Widget.name}')`);
+ }
+ else {
+ break;
+ }
+ }
+ let isValid;
+ try {
+ isValid = isValidProp(props[propName], propsDef[propName]);
+ }
+ catch (e) {
+ e.message = `Invalid prop '${propName}' in component ${Widget.name} (${e.message})`;
+ throw e;
+ }
+ if (!isValid) {
+ throw new Error(`Invalid Prop '${propName}' in component '${Widget.name}'`);
+ }
+ }
+ for (let propName in props) {
+ if (!(propName in propsDef)) {
+ throw new Error(`Unknown prop '${propName}' given to component '${Widget.name}'`);
+ }
+ }
+ }
+ };
+ /**
+ * Check if an invidual prop value matches its (static) prop definition
+ */
+ function isValidProp(prop, propDef) {
+ if (propDef === true) {
+ return true;
+ }
+ if (typeof propDef === "function") {
+ // Check if a value is constructed by some Constructor. Note that there is a
+ // slight abuse of language: we want to consider primitive values as well.
+ //
+ // So, even though 1 is not an instance of Number, we want to consider that
+ // it is valid.
+ if (typeof prop === "object") {
+ return prop instanceof propDef;
+ }
+ return typeof prop === propDef.name.toLowerCase();
+ }
+ else if (propDef instanceof Array) {
+ // If this code is executed, this means that we want to check if a prop
+ // matches at least one of its descriptor.
+ let result = false;
+ for (let i = 0, iLen = propDef.length; i < iLen; i++) {
+ result = result || isValidProp(prop, propDef[i]);
+ }
+ return result;
+ }
+ // propsDef is an object
+ if (propDef.optional && prop === undefined) {
+ return true;
+ }
+ let result = propDef.type ? isValidProp(prop, propDef.type) : true;
+ if (propDef.validate) {
+ result = result && propDef.validate(prop);
+ }
+ if (propDef.type === Array && propDef.element) {
+ for (let i = 0, iLen = prop.length; i < iLen; i++) {
+ result = result && isValidProp(prop[i], propDef.element);
+ }
+ }
+ if (propDef.type === Object && propDef.shape) {
+ const shape = propDef.shape;
+ for (let key in shape) {
+ result = result && isValidProp(prop[key], shape[key]);
+ }
+ if (result) {
+ for (let propName in prop) {
+ if (!(propName in shape)) {
+ throw new Error(`unknown prop '${propName}'`);
+ }
+ }
+ }
+ }
+ return result;
+ }
+
+ /**
+ * Owl Style System
+ *
+ * This files contains the Owl code related to processing (extended) css strings
+ * and creating/adding