2018-01-16 11:34:37 +01:00
|
|
|
flectra.define('web.widget_tests', function (require) {
|
2018-01-16 06:58:15 +01:00
|
|
|
"use strict";
|
|
|
|
|
|
|
|
var concurrency = require('web.concurrency');
|
|
|
|
var core = require('web.core');
|
|
|
|
var QWeb = require('web.QWeb');
|
|
|
|
var Widget = require('web.Widget');
|
|
|
|
|
|
|
|
QUnit.module('core', {}, function () {
|
|
|
|
|
|
|
|
QUnit.module('Widget');
|
|
|
|
|
|
|
|
QUnit.test('proxy (String)', function (assert) {
|
|
|
|
assert.expect(1);
|
|
|
|
|
|
|
|
var W = Widget.extend({
|
|
|
|
exec: function () {
|
|
|
|
this.executed = true;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
var w = new W();
|
|
|
|
var fn = w.proxy('exec');
|
|
|
|
fn();
|
|
|
|
assert.ok(w.executed, 'should execute the named method in the right context');
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test('proxy (String)(*args)', function (assert) {
|
|
|
|
assert.expect(2);
|
|
|
|
|
|
|
|
var W = Widget.extend({
|
|
|
|
exec: function (arg) {
|
|
|
|
this.executed = arg;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
var w = new W();
|
|
|
|
var fn = w.proxy('exec');
|
|
|
|
fn(42);
|
|
|
|
assert.ok(w.executed, "should execute the named method in the right context");
|
|
|
|
assert.strictEqual(w.executed, 42, "should be passed the proxy's arguments");
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test('proxy (String), include', function (assert) {
|
|
|
|
assert.expect(1);
|
|
|
|
|
|
|
|
// the proxy function should handle methods being changed on the class
|
|
|
|
// and should always proxy "by name", to the most recent one
|
|
|
|
var W = Widget.extend({
|
|
|
|
exec: function () {
|
|
|
|
this.executed = 1;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
var w = new W();
|
|
|
|
var fn = w.proxy('exec');
|
|
|
|
W.include({
|
|
|
|
exec: function () { this.executed = 2; }
|
|
|
|
});
|
|
|
|
|
|
|
|
fn();
|
|
|
|
assert.strictEqual(w.executed, 2, "should be lazily resolved");
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test('proxy (Function)', function (assert) {
|
|
|
|
assert.expect(1);
|
|
|
|
|
|
|
|
var w = new (Widget.extend({ }))();
|
|
|
|
|
|
|
|
var fn = w.proxy(function () { this.executed = true; });
|
|
|
|
fn();
|
|
|
|
assert.ok(w.executed, "should set the function's context (like Function#bind)");
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test('proxy (Function)(*args)', function (assert) {
|
|
|
|
assert.expect(1);
|
|
|
|
|
|
|
|
var w = new (Widget.extend({ }))();
|
|
|
|
|
|
|
|
var fn = w.proxy(function (arg) { this.executed = arg; });
|
|
|
|
fn(42);
|
|
|
|
assert.strictEqual(w.executed, 42, "should be passed the proxy's arguments");
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
QUnit.test('renderElement, no template, default', function (assert) {
|
|
|
|
assert.expect(8);
|
|
|
|
|
|
|
|
var widget = new (Widget.extend({ }))();
|
|
|
|
|
|
|
|
var $original = widget.$el;
|
|
|
|
assert.ok($original, "should initially have a root element");
|
|
|
|
|
|
|
|
widget.renderElement();
|
|
|
|
|
|
|
|
assert.ok(widget.$el, "should have generated a root element");
|
|
|
|
assert.ok($original !== widget.$el, "should have generated a new root element");
|
|
|
|
assert.strictEqual(widget.$el, widget.$el, "should provide $el alias");
|
|
|
|
assert.ok(widget.$el.is(widget.el), "should provide raw DOM alias");
|
|
|
|
|
|
|
|
assert.strictEqual(widget.el.nodeName, 'DIV', "should have generated the default element");
|
|
|
|
assert.strictEqual(widget.el.attributes.length, 0, "should not have generated any attribute");
|
|
|
|
assert.ok(_.isEmpty(widget.$el.html(), "should not have generated any content"));
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test('no template, custom tag', function (assert) {
|
|
|
|
assert.expect(1);
|
|
|
|
|
|
|
|
|
|
|
|
var widget = new (Widget.extend({
|
|
|
|
tagName: 'ul'
|
|
|
|
}))();
|
|
|
|
widget.renderElement();
|
|
|
|
|
|
|
|
assert.strictEqual(widget.el.nodeName, 'UL', "should have generated the custom element tag");
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test('no template, @id', function (assert) {
|
|
|
|
assert.expect(3);
|
|
|
|
|
|
|
|
var widget = new (Widget.extend({
|
|
|
|
id: 'foo'
|
|
|
|
}))();
|
|
|
|
widget.renderElement();
|
|
|
|
|
|
|
|
assert.strictEqual(widget.el.attributes.length, 1, "should have one attribute");
|
|
|
|
assert.strictEqual(widget.$el.attr('id'), 'foo', "should have generated the id attribute");
|
|
|
|
assert.strictEqual(widget.el.id, 'foo', "should also be available via property");
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test('no template, @className', function (assert) {
|
|
|
|
assert.expect(2);
|
|
|
|
|
|
|
|
var widget = new (Widget.extend({
|
|
|
|
className: 'oe_some_class'
|
|
|
|
}))();
|
|
|
|
widget.renderElement();
|
|
|
|
|
|
|
|
assert.strictEqual(widget.el.className, 'oe_some_class', "should have the right property");
|
|
|
|
assert.strictEqual(widget.$el.attr('class'), 'oe_some_class', "should have the right attribute");
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test('no template, bunch of attributes', function (assert) {
|
|
|
|
assert.expect(9);
|
|
|
|
|
|
|
|
var widget = new (Widget.extend({
|
|
|
|
attributes: {
|
|
|
|
'id': 'some_id',
|
|
|
|
'class': 'some_class',
|
|
|
|
'data-foo': 'data attribute',
|
|
|
|
'clark': 'gable',
|
|
|
|
'spoiler': // don't read the next line if you care about Harry Potter...
|
|
|
|
'snape kills dumbledore'
|
|
|
|
}
|
|
|
|
}))();
|
|
|
|
widget.renderElement();
|
|
|
|
|
|
|
|
assert.strictEqual(widget.el.attributes.length, 5, "should have all the specified attributes");
|
|
|
|
|
|
|
|
assert.strictEqual(widget.el.id, 'some_id');
|
|
|
|
assert.strictEqual(widget.$el.attr('id'), 'some_id');
|
|
|
|
|
|
|
|
assert.strictEqual(widget.el.className, 'some_class');
|
|
|
|
assert.strictEqual(widget.$el.attr('class'), 'some_class');
|
|
|
|
|
|
|
|
assert.strictEqual(widget.$el.attr('data-foo'), 'data attribute');
|
|
|
|
assert.strictEqual(widget.$el.data('foo'), 'data attribute');
|
|
|
|
|
|
|
|
assert.strictEqual(widget.$el.attr('clark'), 'gable');
|
|
|
|
assert.strictEqual(widget.$el.attr('spoiler'), 'snape kills dumbledore');
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test('template', function (assert) {
|
|
|
|
assert.expect(3);
|
|
|
|
|
|
|
|
core.qweb.add_template(
|
|
|
|
'<no>' +
|
|
|
|
'<t t-name="test.widget.template">' +
|
|
|
|
'<ol>' +
|
|
|
|
'<li t-foreach="5" t-as="counter" ' +
|
|
|
|
't-attf-class="class-#{counter}">' +
|
|
|
|
'<input/>' +
|
|
|
|
'<t t-esc="counter"/>' +
|
|
|
|
'</li>' +
|
|
|
|
'</ol>' +
|
|
|
|
'</t>' +
|
|
|
|
'</no>'
|
|
|
|
);
|
|
|
|
|
|
|
|
var widget = new (Widget.extend({
|
|
|
|
template: 'test.widget.template'
|
|
|
|
}))();
|
|
|
|
widget.renderElement();
|
|
|
|
|
|
|
|
assert.strictEqual(widget.el.nodeName, 'OL');
|
|
|
|
assert.strictEqual(widget.$el.children().length, 5);
|
|
|
|
assert.strictEqual(widget.el.textContent, '01234');
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test('repeated', function (assert) {
|
|
|
|
assert.expect(4);
|
|
|
|
var $fix = $( "#qunit-fixture");
|
|
|
|
|
|
|
|
core.qweb.add_template(
|
|
|
|
'<no>' +
|
|
|
|
'<t t-name="test.widget.template">' +
|
|
|
|
'<p><t t-esc="widget.value"/></p>' +
|
|
|
|
'</t>' +
|
|
|
|
'</no>'
|
|
|
|
);
|
|
|
|
var widget = new (Widget.extend({
|
|
|
|
template: 'test.widget.template'
|
|
|
|
}))();
|
|
|
|
widget.value = 42;
|
|
|
|
|
|
|
|
return widget.appendTo($fix)
|
|
|
|
.done(function () {
|
|
|
|
assert.strictEqual($fix.find('p').text(), '42', "DOM fixture should contain initial value");
|
|
|
|
assert.strictEqual(widget.$el.text(), '42', "should set initial value");
|
|
|
|
widget.value = 36;
|
|
|
|
widget.renderElement();
|
|
|
|
assert.strictEqual($fix.find('p').text(), '36', "DOM fixture should use new value");
|
|
|
|
assert.strictEqual(widget.$el.text(), '36', "should set new value");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
QUnit.module('Widgets, with QWeb', {
|
|
|
|
beforeEach: function() {
|
|
|
|
this.oldQWeb = core.qweb;
|
|
|
|
core.qweb = new QWeb();
|
|
|
|
core.qweb.add_template(
|
|
|
|
'<no>' +
|
|
|
|
'<t t-name="test.widget.template">' +
|
|
|
|
'<ol>' +
|
|
|
|
'<li t-foreach="5" t-as="counter" ' +
|
|
|
|
't-attf-class="class-#{counter}">' +
|
|
|
|
'<input/>' +
|
|
|
|
'<t t-esc="counter"/>' +
|
|
|
|
'</li>' +
|
|
|
|
'</ol>' +
|
|
|
|
'</t>' +
|
|
|
|
'</no>'
|
|
|
|
);
|
|
|
|
},
|
|
|
|
afterEach: function() {
|
|
|
|
core.qweb = this.oldQWeb;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test('basic-alias', function (assert) {
|
|
|
|
assert.expect(1);
|
|
|
|
|
|
|
|
|
|
|
|
var widget = new (Widget.extend({
|
|
|
|
template: 'test.widget.template'
|
|
|
|
}))();
|
|
|
|
widget.renderElement();
|
|
|
|
|
|
|
|
assert.ok(widget.$('li:eq(3)').is(widget.$el.find('li:eq(3)')),
|
|
|
|
"should do the same thing as calling find on the widget root");
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
QUnit.test('delegate', function (assert) {
|
|
|
|
assert.expect(5);
|
|
|
|
|
|
|
|
var a = [];
|
|
|
|
var widget = new (Widget.extend({
|
|
|
|
template: 'test.widget.template',
|
|
|
|
events: {
|
|
|
|
'click': function () {
|
|
|
|
a[0] = true;
|
|
|
|
assert.strictEqual(this, widget, "should trigger events in widget");
|
|
|
|
},
|
|
|
|
'click li.class-3': 'class3',
|
|
|
|
'change input': function () { a[2] = true; }
|
|
|
|
},
|
|
|
|
class3: function () { a[1] = true; }
|
|
|
|
}))();
|
|
|
|
widget.renderElement();
|
|
|
|
|
|
|
|
widget.$el.click();
|
|
|
|
widget.$('li:eq(3)').click();
|
|
|
|
widget.$('input:last').val('foo').change();
|
|
|
|
|
|
|
|
for(var i=0; i<3; ++i) {
|
|
|
|
assert.ok(a[i], "should pass test " + i);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test('undelegate', function (assert) {
|
|
|
|
assert.expect(4);
|
|
|
|
|
|
|
|
var clicked = false;
|
|
|
|
var newclicked = false;
|
|
|
|
|
|
|
|
var widget = new (Widget.extend({
|
|
|
|
template: 'test.widget.template',
|
|
|
|
events: { 'click li': function () { clicked = true; } }
|
|
|
|
}))();
|
|
|
|
|
|
|
|
widget.renderElement();
|
|
|
|
widget.$el.on('click', 'li', function () { newclicked = true; });
|
|
|
|
|
|
|
|
widget.$('li').click();
|
|
|
|
assert.ok(clicked, "should trigger bound events");
|
|
|
|
assert.ok(newclicked, "should trigger bound events");
|
|
|
|
|
|
|
|
clicked = newclicked = false;
|
|
|
|
widget.undelegateEvents();
|
|
|
|
widget.$('li').click();
|
|
|
|
assert.ok(!clicked, "undelegate should unbind events delegated");
|
|
|
|
assert.ok(newclicked, "undelegate should only unbind events it created");
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.module('Widget, and async stuff');
|
|
|
|
|
|
|
|
QUnit.test("alive(alive)", function (assert) {
|
|
|
|
assert.expect(1);
|
|
|
|
|
|
|
|
var widget = new (Widget.extend({}));
|
|
|
|
|
|
|
|
return concurrency.asyncWhen(widget.start())
|
|
|
|
.then(function () { return widget.alive(concurrency.asyncWhen()); })
|
|
|
|
.then(function () { assert.ok(true); });
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test("alive(dead)", function (assert) {
|
|
|
|
assert.expect(1);
|
|
|
|
var widget = new (Widget.extend({}));
|
|
|
|
|
|
|
|
return $.Deferred(function (d) {
|
|
|
|
concurrency.asyncWhen(widget.start())
|
|
|
|
.then(function () {
|
|
|
|
// destroy widget
|
|
|
|
widget.destroy();
|
|
|
|
var promise = concurrency.asyncWhen();
|
|
|
|
// leave time for alive() to do its stuff
|
|
|
|
promise.then(function () {
|
|
|
|
return concurrency.asyncWhen();
|
|
|
|
}).then(function () {
|
|
|
|
assert.ok(true);
|
|
|
|
d.resolve();
|
|
|
|
});
|
|
|
|
// ensure that widget.alive() refuses to resolve or reject
|
|
|
|
return widget.alive(promise);
|
|
|
|
}).always(function () {
|
|
|
|
d.reject();
|
|
|
|
assert.ok(false, "alive() should not terminate by default");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test("alive(alive, true)", function (assert) {
|
|
|
|
assert.expect(1);
|
|
|
|
var widget = new (Widget.extend({}));
|
|
|
|
return concurrency.asyncWhen(widget.start())
|
|
|
|
.then(function () { return widget.alive(concurrency.asyncWhen(), true) })
|
|
|
|
.then(function () { assert.ok(true); });
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test("alive(dead, true)", function (assert) {
|
|
|
|
assert.expect(1);
|
|
|
|
var done = assert.async();
|
|
|
|
|
|
|
|
var widget = new (Widget.extend({}));
|
|
|
|
|
|
|
|
concurrency.asyncWhen(widget.start())
|
|
|
|
.then(function () {
|
|
|
|
// destroy widget
|
|
|
|
widget.destroy();
|
|
|
|
return widget.alive(concurrency.asyncWhen(), true);
|
|
|
|
}).then(function () {
|
|
|
|
assert.ok(false, "alive(p, true) should fail its promise");
|
|
|
|
done();
|
|
|
|
}, function () {
|
|
|
|
assert.ok(true, "alive(p, true) should fail its promise");
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
QUnit.test("calling _rpc on destroyed widgets", function (assert) {
|
|
|
|
assert.expect(1);
|
|
|
|
|
|
|
|
var widget = new Widget();
|
|
|
|
widget.destroy();
|
|
|
|
widget._rpc({route: '/a/route'}).always(function () {
|
|
|
|
throw Error("Calling _rpc on a destroyed widget should return a " +
|
|
|
|
"deferred that is never resolved nor rejected");
|
|
|
|
});
|
|
|
|
assert.ok(true,
|
|
|
|
"there should be no crash when calling _rpc on a destroyed widget");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2018-01-16 11:34:37 +01:00
|
|
|
});
|