zulip/frontend_tests/zjsunit/zjquery_element.js

409 lines
12 KiB
JavaScript

"use strict";
const {strict: assert} = require("assert");
const FakeEvent = require("./zjquery_event");
const noop = function () {};
// TODO: convert this to a true class
function FakeElement(selector, opts) {
let html = "never-been-set";
let text = "never-been-set";
let value;
let shown = false;
let height;
const find_results = new Map();
let $my_parent;
const parents_result = new Map();
const properties = new Map();
const attrs = new Map();
const classes = new Map();
const event_store = make_event_store(selector);
const $self = {
addClass(class_name) {
classes.set(class_name, true);
return $self;
},
append(arg) {
html = html + arg;
return $self;
},
attr(name, val) {
if (val === undefined) {
return attrs.get(name);
}
attrs.set(name, val);
return $self;
},
data(name, val) {
if (val === undefined) {
return attrs.get("data-" + name);
}
attrs.set("data-" + name, val);
return $self;
},
delay() {
return $self;
},
/* istanbul ignore next */
debug() {
return {
value,
shown,
selector,
};
},
empty(arg) {
if (arg === undefined) {
find_results.clear();
html = "";
}
return $self;
},
expectOne() {
// silently do nothing
return $self;
},
fadeTo: noop,
find(child_selector) {
const $child = find_results.get(child_selector);
if ($child) {
return $child;
}
if ($child === false) {
// This is deliberately set to simulate missing find results.
// Return an empty array, the most common check is
// if ($.find().length) { //success }
return [];
}
/* istanbul ignore next */
throw new Error(`
We need you to simulate the results of $(...).find(...)
by using set_find_results. You want something like this:
const $container = ...;
const $child = ...;
$container.set_find_results("${child_selector}", $child);
Then calling $container.find("${child_selector}") will return
the "$child" zjquery element.
`);
},
get_on_handler(name, child_selector) {
return event_store.get_on_handler(name, child_selector);
},
hasClass(class_name) {
return classes.has(class_name);
},
height() {
assert.notEqual(height, undefined, `Please call $("${selector}").set_height`);
return height;
},
hide() {
shown = false;
return $self;
},
html(arg) {
if (arg !== undefined) {
html = arg;
return $self;
}
return html;
},
is(arg) {
switch (arg) {
case ":visible":
return shown;
case ":focus":
return $self.is_focused();
/* istanbul ignore next */
default:
throw new Error("zjquery does not support this is() call");
}
},
is_focused() {
// is_focused is not a jQuery thing; this is
// for our testing
return event_store.is_focused();
},
off(...args) {
event_store.off(...args);
return $self;
},
offset() {
return {
top: 0,
left: 0,
};
},
on(...args) {
event_store.on(...args);
return $self;
},
/* istanbul ignore next */
one(...args) {
event_store.one(...args);
return $self;
},
parent() {
return $my_parent;
},
parents(parents_selector) {
const $result = parents_result.get(parents_selector);
assert.ok(
$result,
"You need to call set_parents_result for " + parents_selector + " in " + selector,
);
return $result;
},
prepend(arg) {
html = arg + html;
return $self;
},
prop(name, val) {
if (val === undefined) {
return properties.get(name);
}
properties.set(name, val);
return $self;
},
removeAttr(name) {
attrs.delete(name);
return $self;
},
removeClass(class_names) {
class_names = class_names.split(" ");
for (const class_name of class_names) {
classes.delete(class_name);
}
return $self;
},
/* istanbul ignore next */
remove() {
throw new Error(`
We don't support remove in zjquery.
You can do $(...).remove = ... if necessary.
But you are probably writing too deep a test
for node testing.
`);
},
removeData: noop,
set_find_results(find_selector, $jquery_object) {
assert.notEqual(
$jquery_object,
undefined,
"Please make the 'find result' be something like $.create('unused')",
);
find_results.set(find_selector, $jquery_object);
},
set_height(fake_height) {
height = fake_height;
},
set_parent($parent_elem) {
$my_parent = $parent_elem;
},
set_parents_result(selector, $result) {
parents_result.set(selector, $result);
},
show() {
shown = true;
return $self;
},
text(...args) {
if (args.length !== 0) {
if (args[0] !== undefined) {
text = args[0].toString();
}
return $self;
}
return text;
},
toggle(show) {
assert.ok([true, false].includes(show));
shown = show;
return $self;
},
trigger(ev) {
event_store.trigger($self, ev);
return $self;
},
val(...args) {
if (args.length === 0) {
return value || "";
}
[value] = args;
return $self;
},
visible() {
return shown;
},
};
if (opts.children) {
$self.map = (f) => opts.children.map((i, elem) => f(elem, i));
$self.each = (f) => {
for (const child of opts.children) {
f.call(child);
}
};
$self[Symbol.iterator] = function* () {
for (const child of opts.children) {
yield child;
}
};
for (const [i, child] of opts.children.entries()) {
$self[i] = child;
}
$self.length = opts.children.length;
}
if (selector[0] === "<") {
$self.html(selector);
}
$self.selector = selector;
$self.__zjquery = true;
return $self;
}
function make_event_store(selector) {
/*
This function returns an event_store object that
simulates the behavior of .on and .off from jQuery.
It also has methods to retrieve handlers that have
been set via .on (or similar methods), which can
be useful for tests that want to test the actual
handlers.
*/
const on_functions = new Map();
const child_on_functions = new Map();
let focused = false;
const self = {
get_on_handler(name, child_selector) {
let handler;
if (child_selector === undefined) {
handler = on_functions.get(name);
assert.ok(handler, `no ${name} handler for ${selector}`);
return handler;
}
const child_on = child_on_functions.get(child_selector);
if (child_on) {
handler = child_on.get(name);
}
assert.ok(handler, `no ${name} handler for ${selector} ${child_selector}`);
return handler;
},
off(event_name, ...args) {
if (args.length === 0) {
on_functions.delete(event_name);
return;
}
// In the Zulip codebase we never use this form of
// .off in code that we test: $(...).off('click', child_sel);
//
// So we don't support this for now.
/* istanbul ignore next */
throw new Error("zjquery does not support this call sequence");
},
on(event_name, ...args) {
// parameters will either be
// (event_name, handler) or
// (event_name, sel, handler)
if (args.length === 1) {
const [handler] = args;
/* istanbul ignore if */
if (on_functions.has(event_name)) {
console.info("\nEither the app or the test can be at fault here..");
console.info("(sometimes you just want to call $.clear_all_elements();)\n");
throw new Error("dup " + event_name + " handler for " + selector);
}
on_functions.set(event_name, handler);
return;
}
assert.equal(args.length, 2, "wrong number of arguments passed in");
const [sel, handler] = args;
assert.equal(typeof sel, "string", "String selectors expected here.");
assert.equal(typeof handler, "function", "An handler function expected here.");
if (!child_on_functions.has(sel)) {
child_on_functions.set(sel, new Map());
}
const child_on = child_on_functions.get(sel);
assert.ok(
!child_on.has(event_name),
`dup ${event_name} handler for ${selector} ${sel}`,
);
child_on.set(event_name, handler);
},
/* istanbul ignore next */
one(event_name, handler) {
self.on(event_name, function (ev) {
self.off(event_name);
return handler.call(this, ev);
});
},
trigger($element, ev, data) {
if (typeof ev === "string") {
ev = new FakeEvent(ev);
}
if (!ev.target) {
// FIXME: event.target should not be a jQuery object
ev.target = $element; // eslint-disable-line no-jquery/variable-pattern
}
const func = on_functions.get(ev.type);
if (func) {
// It's possible that test code will trigger events
// that haven't been set up yet, but we are trying to
// eventually deprecate trigger in our codebase, so for
// now we just let calls to trigger silently do nothing.
// (And I think actual jQuery would do the same thing.)
func.call($element, ev, data);
}
if (ev.type === "focus" || ev.type === "focusin") {
focused = true;
} else if (ev.type === "blur" || ev.type === "focusout") {
focused = false;
}
},
is_focused() {
return focused;
},
};
return self;
}
module.exports = FakeElement;