diff --git a/addons/web/static/lib/qweb/qweb-test-call.xml b/addons/web/static/lib/qweb/qweb-test-call.xml
index 45a65eabce981c46a22fcc1784df9cb24faeb477..b0c5deb9fe6b57658ae1c994bb41b56d915665de 100644
--- a/addons/web/static/lib/qweb/qweb-test-call.xml
+++ b/addons/web/static/lib/qweb/qweb-test-call.xml
@@ -57,7 +57,7 @@
     </result>
 
     <t t-name="basic-caller">
-        <t t-call="{{True and '_basic-callee' or 'other'}}"/>
+        <t t-call="{{1 and '_basic-callee' or 'other'}}"/>
     </t>
     <result id="basic-caller">ok</result>
 </templates>
diff --git a/addons/web/static/lib/qweb/qweb-test-global.xml b/addons/web/static/lib/qweb/qweb-test-global.xml
index 1db3e5920fb254c973e638c97dfe306e569e74ee..bfcca8c1eba3ffc3e09f2786dcdd7e943c248593 100644
--- a/addons/web/static/lib/qweb/qweb-test-global.xml
+++ b/addons/web/static/lib/qweb/qweb-test-global.xml
@@ -18,7 +18,7 @@
             </t>
         </t>
         <t t-call="_callee-asc-toto"/>
-        <t t-set="toto"><t t-set="truc" t-value="'bbb'"/><i t-att-notruc="not truc" t-att-truc="bool(truc)">i</i></t>
+        <t t-set="toto"><t t-set="truc" t-value="'bbb'"/><i t-att-notruc="not truc" t-att-truc="truc and 'ok'">i</i></t>
         <t t-call="_callee-asc-toto"/>
     </t>
 
@@ -49,6 +49,6 @@
         
         <div>toto default</div>
         
-        <div><i truc="True">i</i></div>
+        <div><i truc="ok">i</i></div>
     ]]></result>
 </templates>
diff --git a/addons/web/static/lib/qweb/qweb-test.js.html b/addons/web/static/lib/qweb/qweb-test.js.html
deleted file mode 100644
index 0ee5f27c3bdb05ddf86ba0952c3bb3b9da69f26e..0000000000000000000000000000000000000000
--- a/addons/web/static/lib/qweb/qweb-test.js.html
+++ /dev/null
@@ -1,73 +0,0 @@
-<!doctype html>
-<html>
-<head>
-    <script src="/web/static/lib/jquery/jquery.js"></script>
-    <link rel="stylesheet" href="/web/static/lib/qunit/qunit.css" type="text/css" media="screen"/>
-    <script type="text/javascript" src="/web/static/lib/qunit/qunit.js"></script>
-
-    <script type="text/javascript" src="qweb2.js"></script>
-
-    <script>
-        QWeb = new QWeb2.Engine();
-        function trim(s) {
-            return s.replace(/(^\s+|\s+$)/g, '');
-        }
-        function render(template, context) {
-            return trim(QWeb.render(template, context)).toLowerCase();
-        }
-
-        /**
-         * Loads the template file, and executes all the test template in a
-         * qunit module $title
-         */
-        function test(title, template) {
-            QUnit.module(title, {
-                setup: function () {
-                    var self = this;
-                    this.qweb = new QWeb2.Engine();
-                    QUnit.stop();
-                    this.qweb.add_template(template, function (_, doc) {
-                        self.doc = doc;
-                        QUnit.start();
-                    })
-                }
-            });
-            QUnit.test('autotest', function (assert) {
-                var templates = this.qweb.templates;
-                for (var template in templates) {
-                    if (!templates.hasOwnProperty(template)) { continue; }
-                    // ignore templates whose name starts with _, they're
-                    // helpers/internal
-                    if (/^_/.test(template)) { continue; }
-
-                    var params = this.doc.querySelector('params#' + template);
-                    var args = params ? JSON.parse(params.textContent) : {};
-
-                    var results = this.doc.querySelector('result#' + template);
-                    assert.equal(
-                        trim(this.qweb.render(template, args)),
-                        trim(results.textContent),
-                        template);
-                }
-            });
-        }
-        $(document).ready(function() {
-            test("Output", 'qweb-test-output.xml');
-            test("Context-setting", 'qweb-test-set.xml');
-            test("Conditionals", 'qweb-test-conditionals.xml');
-            test("Attributes manipulation", 'qweb-test-attributes.xml');
-            test("Template calling (to the faraway pages)",
-                 'qweb-test-call.xml');
-            test("Foreach", 'qweb-test-foreach.xml');
-            test("Global", 'qweb-test-global.xml');
-
-            test('Template inheritance', 'qweb-test-extend.xml');
-        });
-    </script>
-
-</head>
-<body>
-  <div id="qunit"></div>
-  <div id="qunit-fixture"></div>
-</body>
-</html>
diff --git a/addons/web/static/lib/qweb/qweb2.js b/addons/web/static/lib/qweb/qweb2.js
index c4bb7128e9bec9fadd91f3aa4efe38d762ab4883..8058cb7745f645e9b352ac96c9a6b2f020d48336 100644
--- a/addons/web/static/lib/qweb/qweb2.js
+++ b/addons/web/static/lib/qweb/qweb2.js
@@ -28,8 +28,10 @@ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 var QWeb2 = {
     expressions_cache: { },
     RESERVED_WORDS: 'true,false,NaN,null,undefined,debugger,console,window,in,instanceof,new,function,return,this,typeof,eval,void,Math,RegExp,Array,Object,Date'.split(','),
-    ACTIONS_PRECEDENCE: 'foreach,if,elif,else,call,set,esc,raw,js,debug,log'.split(','),
+    ACTIONS_PRECEDENCE: 'foreach,if,elif,else,call,set,js,debug,log'.split(','),
     WORD_REPLACEMENT: {
+        // FUCKING CHM FOR FUCK'S SAKE
+        'not': '!',
         'and': '&&',
         'or': '||',
         'gt': '>',
@@ -76,9 +78,9 @@ var QWeb2 = {
             return s;
         },
         gen_attribute: function(o) {
-            if (o !== null && o !== undefined) {
+            if (o) {
                 if (o.constructor === Array) {
-                    if (o[1] !== null && o[1] !== undefined) {
+                    if (o[1]) {
                         return this.format_attribute(o[0], o[1]);
                     }
                 } else if (typeof o === 'object') {
@@ -718,36 +720,52 @@ QWeb2.Element = (function() {
         },
         compile_element : function() {
             for (var i = 0, ilen = this.engine.actions_precedence.length; i < ilen; i++) {
-                var a = this.engine.actions_precedence[i];
-                if (a in this.actions) {
-                    var value = this.actions[a];
-                    var key = 'compile_action_' + a;
+                var compiled_action = this.engine.actions_precedence[i];
+                if (compiled_action in this.actions) {
+                    var value = this.actions[compiled_action];
+                    var key = 'compile_action_' + compiled_action;
                     if (this[key]) {
                         this[key](value);
                     } else if (this.engine[key]) {
                         this.engine[key].call(this, value);
                     } else {
-                        this.engine.tools.exception("No handler method for action '" + a + "'");
+                        this.engine.tools.exception("No handler method for action '" + compiled_action + "'");
                     }
                 }
             }
+            function print_contents(self) {
+                var expr, esc, raw;
+                if (esc = self.actions['esc']) {
+                    expr = "context.engine.tools.html_escape(" + self.format_str(esc) + ");";
+                } else if(raw = self.actions['raw']) {
+                    expr = self.format_str(raw);
+                }
+                // print non-empty result of t-esc or t-raw, otherwise
+                // print/process body
+                if (expr) {
+                    self.top('var expr_result = ' + expr + ';');
+                    self.top('if (expr_result) {');
+                    self.top('    r.push(expr_result);');
+                    self.top('} else {');
+                    self.bottom('}');
+                }
+             }
             if (this.tag.toLowerCase() !== this.engine.prefix) {
                 var tag = "<" + this.tag;
-                for (var a in this.attributes) {
-                    tag += this.engine.tools.gen_attribute([a, this.attributes[a]]);
+                for (var att in this.attributes) {
+                    tag += this.engine.tools.gen_attribute([att, this.attributes[att]]);
                 }
                 this.top_string(tag);
                 if (this.actions.att) {
-                    this.top("r.push(context.engine.tools.gen_attribute(" + (this.format_expression(this.actions.att)) + "));");
+
                 }
                 for (var a in this.actions) {
-                    var v = this.actions[a];
-                    var m = a.match(/att-(.+)/);
-                    if (m) {
+                    var m, v = this.actions[a];
+                    if (a === 'att') {
+                        this.top("r.push(context.engine.tools.gen_attribute(" + (this.format_expression(v)) + "));");
+                    } else if (m = a.match(/att-(.+)/)) {
                         this.top("r.push(context.engine.tools.gen_attribute(['" + m[1] + "', (" + (this.format_expression(v)) + ")]));");
-                    }
-                    var m = a.match(/attf-(.+)/);
-                    if (m) {
+                    } else if (m = a.match(/attf-(.+)/)) {
                         this.top("r.push(context.engine.tools.gen_attribute(['" + m[1] + "', (" + (this.string_interpolation(v)) + ")]));");
                     }
                 }
@@ -755,11 +773,13 @@ QWeb2.Element = (function() {
                     // We do not enforce empty content on void elements
                     // because QWeb rendering is not necessarily html.
                     this.top_string("/>");
+                    return; // ensure we skip processing content
                 } else {
                     this.top_string(">");
                     this.bottom_string("</" + this.tag + ">");
                 }
             }
+            print_contents(this);
         },
         compile_action_if : function(value) {
             this.top("if (" + (this.format_expression(value)) + ") {");
@@ -817,14 +837,6 @@ QWeb2.Element = (function() {
                 }
             }
         },
-        compile_action_esc : function(value) {
-            this.top("r.push(context.engine.tools.html_escape("
-                    + this.format_expression(value)
-                    + "));");
-        },
-        compile_action_raw : function(value) {
-            this.top("r.push(" + (this.format_str(value)) + ");");
-        },
         compile_action_js : function(value) {
             this.top("(function(" + value + ") {");
             this.bottom("})(dict);");
diff --git a/addons/web/static/lib/qweb/tests.js b/addons/web/static/lib/qweb/tests.js
new file mode 100644
index 0000000000000000000000000000000000000000..5b961acd427bdc0a32a2a458d1b896108d0c786f
--- /dev/null
+++ b/addons/web/static/lib/qweb/tests.js
@@ -0,0 +1,77 @@
+odoo.define('qweb.tests', function () {
+'use strict';
+
+function trim(s) {
+    return s.replace(/(^\s+|\s+$)/g, '');
+}
+
+/**
+ * Generates a QUnit.module hook object loading the specified test file
+ * (from /web/static/lib/qweb) and setting ``this.qweb`` (the qweb
+ * instance for this module) and ``this.doc`` (the loaded XML file).
+ *
+ * Note that test files mix template elements <t t-name>, params elements
+ * <params> and result elements <result>. A result and an optional params
+ * object are linked to the corresponding template via the ``id``
+ * attribute (result and params have the template name as id).
+ */
+function hooks(testfile) {
+    var template = '/web/static/lib/qweb/' + testfile;
+    return {
+        before: function () {
+            var self = this;
+            this.qweb = new QWeb2.Engine();
+            var p = $.Deferred();
+            this.qweb.add_template(template, function (_, doc) {
+                self.doc = doc;
+                p.resolve(doc);
+            });
+            return p.promise();
+        }
+    }
+}
+// can't get generative QUnit.test (e.g. QUnit.test in a for(;;)
+// or Array#forEach() loop) to work, so each test file has a single test,
+// that seems to work
+function runtest() {
+    QUnit.test('', function (assert) {
+    var templates = this.qweb.templates;
+    assert.expect(_.reduce(templates, function (acc, _, k) {
+        return acc + (/^_/.test(k) ? 0 : 1);
+    }, 0));
+    for (var template in templates) {
+        if (!templates.hasOwnProperty(template)) { continue; }
+        // ignore templates whose name starts with _, they're
+        // helpers/internal
+        if (/^_/.test(template)) { continue; }
+
+        var params = this.doc.querySelector('params#' + template);
+        var args = params ? JSON.parse(params.textContent) : {};
+
+        var results = this.doc.querySelector('result#' + template);
+        assert.equal(
+            trim(this.qweb.render(template, args)),
+            trim(results.textContent),
+            template);
+        }
+    });
+}
+
+var TEMPLATES = [
+    ["Output", 'qweb-test-output.xml'],
+    ["Context-setting", 'qweb-test-set.xml'],
+    ["Conditionals", 'qweb-test-conditionals.xml'],
+    ["Attributes manipulation", 'qweb-test-attributes.xml'],
+    ["Template calling (to the faraway pages)", 'qweb-test-call.xml'],
+    ["Foreach", 'qweb-test-foreach.xml'],
+    ["Global", 'qweb-test-global.xml'],
+    ['Template inheritance', 'qweb-test-extend.xml']
+];
+
+QUnit.module('qweb', {}, function () {
+    TEMPLATES.forEach(function (it) {
+        QUnit.module(it[0], hooks(it[1]), runtest);
+    })
+});
+
+});
diff --git a/addons/web/views/webclient_templates.xml b/addons/web/views/webclient_templates.xml
index 4f69ed286b33dd399f2b214f584153bc25b0463a..605003443d079d346473b9709b18fcefdc5cec29 100644
--- a/addons/web/views/webclient_templates.xml
+++ b/addons/web/views/webclient_templates.xml
@@ -448,6 +448,9 @@
                         display: inherit;
                     }
                 </style>
+
+                <script type="text/javascript" src="/web/static/lib/qweb/tests.js"></script>
+
                 <script type="text/javascript" src="/web/static/tests/helpers/test_utils.js"></script>
                 <script type="text/javascript" src="/web/static/tests/helpers/mock_server.js"></script>