2017-05-23 03:20:15 +02:00
|
|
|
var noop = function () {};
|
|
|
|
|
|
|
|
var exports = {};
|
|
|
|
|
2017-05-25 01:49:29 +02:00
|
|
|
exports.make_zjquery = function () {
|
|
|
|
|
|
|
|
var elems = {};
|
|
|
|
|
2018-04-12 14:45:06 +02:00
|
|
|
// Our fn structure helps us simulate extending jQuery.
|
|
|
|
var fn = {};
|
|
|
|
|
|
|
|
function add_extensions(obj) {
|
|
|
|
_.each(fn, (v, k) => {
|
|
|
|
obj[k] = v;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-05-25 01:49:29 +02:00
|
|
|
function new_elem(selector) {
|
|
|
|
var html = 'never-been-set';
|
|
|
|
var text = 'never-been-set';
|
|
|
|
var value;
|
|
|
|
var shown = false;
|
|
|
|
var focused = false;
|
2017-07-08 14:31:18 +02:00
|
|
|
var find_results = new Dict();
|
2017-05-25 01:49:29 +02:00
|
|
|
var my_parent;
|
2017-07-08 14:49:09 +02:00
|
|
|
var parents_result = new Dict();
|
2017-06-16 14:45:02 +02:00
|
|
|
var properties = new Dict();
|
2017-06-16 20:55:07 +02:00
|
|
|
var attrs = new Dict();
|
2017-05-25 01:49:29 +02:00
|
|
|
var classes = new Dict();
|
2017-06-14 20:28:55 +02:00
|
|
|
var on_functions = new Dict();
|
2017-07-06 00:18:15 +02:00
|
|
|
var child_on_functions = new Dict();
|
2017-06-22 23:40:38 +02:00
|
|
|
|
|
|
|
function generic_event(event_name, arg) {
|
|
|
|
if (typeof(arg) === 'function') {
|
|
|
|
on_functions.set(event_name, arg);
|
|
|
|
} else {
|
|
|
|
var handler = on_functions.get(event_name);
|
2018-03-01 12:43:42 +01:00
|
|
|
if (!handler) {
|
|
|
|
var error = 'Cannot find ' + event_name + ' handler for ' + selector;
|
|
|
|
throw Error(error);
|
|
|
|
}
|
2017-06-22 23:40:38 +02:00
|
|
|
handler(arg);
|
|
|
|
}
|
|
|
|
}
|
2017-05-25 01:49:29 +02:00
|
|
|
|
|
|
|
var self = {
|
2017-07-08 14:31:18 +02:00
|
|
|
add_child: function () {
|
|
|
|
// TODO: Remove this once some in-flight PRs
|
|
|
|
// get merged.
|
|
|
|
assert(false, 'Use set_find_results instead.');
|
2017-05-25 01:49:29 +02:00
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
addClass: function (class_name) {
|
|
|
|
classes.set(class_name, true);
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-06-16 20:55:07 +02:00
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
attr: function (name, val) {
|
|
|
|
if (val === undefined) {
|
|
|
|
return attrs.get(name);
|
|
|
|
}
|
|
|
|
attrs.set(name, val);
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-06-14 20:28:55 +02:00
|
|
|
},
|
2017-05-25 01:49:29 +02:00
|
|
|
blur: function () {
|
|
|
|
focused = false;
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-05-25 01:49:29 +02:00
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
click: function (arg) {
|
|
|
|
generic_event('click', arg);
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-05-25 01:49:29 +02:00
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
css: noop,
|
|
|
|
data: noop,
|
2017-05-25 01:49:29 +02:00
|
|
|
debug: function () {
|
|
|
|
return {
|
|
|
|
value: value,
|
|
|
|
shown: shown,
|
|
|
|
selector: selector,
|
|
|
|
};
|
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
empty: noop,
|
2017-08-18 01:23:55 +02:00
|
|
|
eq: function () {
|
|
|
|
return self;
|
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
expectOne: function () {
|
|
|
|
// silently do nothing
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-05-25 01:49:29 +02:00
|
|
|
},
|
2017-06-27 22:06:28 +02:00
|
|
|
fadeTo: noop,
|
2017-05-25 01:49:29 +02:00
|
|
|
find: function (child_selector) {
|
2017-07-08 14:31:18 +02:00
|
|
|
var child = find_results.get(child_selector);
|
2017-05-25 01:49:29 +02:00
|
|
|
if (child) {
|
|
|
|
return child;
|
|
|
|
}
|
|
|
|
|
2017-05-31 17:03:18 +02:00
|
|
|
throw Error("Cannot find " + child_selector + " in " + selector);
|
2017-05-25 01:49:29 +02:00
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
focus: function () {
|
|
|
|
focused = true;
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-05-25 01:49:29 +02:00
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
get: function (idx) {
|
|
|
|
// We have some legacy code that does $('foo').get(0).
|
|
|
|
assert.equal(idx, 0);
|
|
|
|
return selector;
|
2017-05-25 01:49:29 +02:00
|
|
|
},
|
2017-07-06 01:38:28 +02:00
|
|
|
get_on_handler: function (name, child_selector) {
|
|
|
|
var funcs = self.get_on_handlers(name, child_selector);
|
|
|
|
assert.equal(funcs.length, 1, 'We expected to have exactly one handler here.');
|
|
|
|
return funcs[0];
|
|
|
|
},
|
2017-07-05 23:11:59 +02:00
|
|
|
get_on_handlers: function (name, child_selector) {
|
|
|
|
if (child_selector === undefined) {
|
|
|
|
return on_functions.get(name) || [];
|
|
|
|
}
|
|
|
|
|
|
|
|
var child_on = child_on_functions.get(child_selector) || {};
|
|
|
|
if (!child_on) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
return child_on.get(name) || [];
|
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
hasClass: function (class_name) {
|
|
|
|
return classes.has(class_name);
|
2017-05-25 01:49:29 +02:00
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
height: noop,
|
|
|
|
hide: function () {
|
|
|
|
shown = false;
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-05-25 01:49:29 +02:00
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
html: function (arg) {
|
|
|
|
if (arg !== undefined) {
|
|
|
|
html = arg;
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-06-27 22:02:51 +02:00
|
|
|
}
|
|
|
|
return html;
|
|
|
|
},
|
2018-03-15 12:10:07 +01:00
|
|
|
is: function (arg) {
|
|
|
|
if (arg === ':visible') {
|
|
|
|
return shown;
|
|
|
|
}
|
|
|
|
return self;
|
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
is_focused: function () {
|
|
|
|
// is_focused is not a jQuery thing; this is
|
|
|
|
// for our testing
|
|
|
|
return focused;
|
|
|
|
},
|
|
|
|
keydown: function (arg) {
|
|
|
|
generic_event('keydown', arg);
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-06-27 22:02:51 +02:00
|
|
|
},
|
|
|
|
keyup: function (arg) {
|
|
|
|
generic_event('keyup', arg);
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-05-25 01:49:29 +02:00
|
|
|
},
|
2018-03-25 15:28:49 +02:00
|
|
|
off: function () {
|
|
|
|
var event_name;
|
|
|
|
|
|
|
|
if (arguments.length === 2) {
|
|
|
|
event_name = arguments[0];
|
|
|
|
on_functions[event_name] = [];
|
|
|
|
} else if (arguments.length === 3) {
|
|
|
|
event_name = arguments[0];
|
|
|
|
var sel = arguments[1];
|
|
|
|
var child_on = child_on_functions.setdefault(sel, new Dict());
|
|
|
|
child_on[event_name] = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
return self;
|
|
|
|
},
|
2017-07-06 00:18:15 +02:00
|
|
|
on: function () {
|
|
|
|
// parameters will either be
|
|
|
|
// (event_name, handler) or
|
|
|
|
// (event_name, sel, handler)
|
|
|
|
var event_name;
|
|
|
|
var sel;
|
|
|
|
var handler;
|
|
|
|
|
|
|
|
// For each event_name (or event_name/sel combo), we will store an
|
|
|
|
// array of functions that are mapped to the event (or event/selector).
|
|
|
|
//
|
|
|
|
// Usually funcs is an array of just one element, but not always.
|
|
|
|
var funcs;
|
|
|
|
|
|
|
|
if (arguments.length === 2) {
|
|
|
|
event_name = arguments[0];
|
|
|
|
handler = arguments[1];
|
|
|
|
funcs = on_functions.setdefault(event_name, []);
|
|
|
|
funcs.push(handler);
|
|
|
|
} else if (arguments.length === 3) {
|
|
|
|
event_name = arguments[0];
|
|
|
|
sel = arguments[1];
|
|
|
|
handler = arguments[2];
|
|
|
|
assert.equal(typeof(sel), 'string', 'String selectors expected here.');
|
|
|
|
assert.equal(typeof(handler), 'function', 'An handler function expected here.');
|
|
|
|
var child_on = child_on_functions.setdefault(sel, new Dict());
|
|
|
|
funcs = child_on.setdefault(event_name, []);
|
|
|
|
funcs.push(handler);
|
|
|
|
}
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-05-25 01:49:29 +02:00
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
parent: function () {
|
|
|
|
return my_parent;
|
2017-06-20 00:31:00 +02:00
|
|
|
},
|
2017-07-08 14:49:09 +02:00
|
|
|
parents: function (parents_selector) {
|
|
|
|
var result = parents_result.get(parents_selector);
|
|
|
|
assert(result, 'You need to call set_parents_result for ' +
|
|
|
|
parents_selector + ' in ' + selector);
|
|
|
|
return result;
|
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
prop: function (name, val) {
|
|
|
|
if (val === undefined) {
|
|
|
|
return properties.get(name);
|
|
|
|
}
|
|
|
|
properties.set(name, val);
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-06-22 23:40:39 +02:00
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
removeAttr: function (name) {
|
|
|
|
attrs.del(name);
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-06-22 23:40:39 +02:00
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
removeClass: function (class_name) {
|
|
|
|
classes.del(class_name);
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-06-22 23:40:40 +02:00
|
|
|
},
|
2017-05-25 01:49:29 +02:00
|
|
|
remove: function () {
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-05-25 01:49:29 +02:00
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
removeData: noop,
|
|
|
|
select: function (arg) {
|
|
|
|
generic_event('select', arg);
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-06-14 21:08:32 +02:00
|
|
|
},
|
2017-07-08 14:31:18 +02:00
|
|
|
set_find_results: function (find_selector, jquery_object) {
|
|
|
|
find_results.set(find_selector, jquery_object);
|
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
show: function () {
|
|
|
|
shown = true;
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-06-27 22:02:51 +02:00
|
|
|
},
|
|
|
|
set_parent: function (parent_elem) {
|
|
|
|
my_parent = parent_elem;
|
|
|
|
},
|
2017-07-08 14:49:09 +02:00
|
|
|
set_parents_result: function (selector, result) {
|
|
|
|
parents_result.set(selector, result);
|
|
|
|
},
|
2017-06-27 22:08:11 +02:00
|
|
|
stop: function () {
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-06-27 22:08:11 +02:00
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
text: function (arg) {
|
|
|
|
if (arg !== undefined) {
|
|
|
|
text = arg;
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-06-16 20:55:07 +02:00
|
|
|
}
|
2017-06-27 22:02:51 +02:00
|
|
|
return text;
|
|
|
|
},
|
|
|
|
trigger: function (ev) {
|
2018-01-07 18:47:55 +01:00
|
|
|
var ev_name = typeof ev === 'string' ? ev : ev.name;
|
|
|
|
var funcs = on_functions.get(ev_name) || [];
|
2017-07-06 15:47:25 +02:00
|
|
|
// The following assertion is temporary. It can be
|
|
|
|
// legitimate for code to trigger multiple handlers.
|
|
|
|
// But up until now, we haven't needed this, and if
|
|
|
|
// you come across this assertion, it's possible that
|
2018-04-12 17:46:21 +02:00
|
|
|
// you can simplify your tests by just doing your own
|
2017-07-06 15:47:25 +02:00
|
|
|
// mocking of trigger(). If you really know what you
|
|
|
|
// are doing, you can remove this limitation.
|
|
|
|
assert(funcs.length <= 1, 'multiple functions set up');
|
|
|
|
|
2017-06-27 22:02:51 +02:00
|
|
|
_.each(funcs, function (f) {
|
|
|
|
f(ev.data);
|
|
|
|
});
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-06-16 20:55:07 +02:00
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
val: function () {
|
|
|
|
if (arguments.length === 0) {
|
|
|
|
return value || '';
|
2017-06-16 14:45:02 +02:00
|
|
|
}
|
2017-06-27 22:02:51 +02:00
|
|
|
value = arguments[0];
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-06-16 14:45:02 +02:00
|
|
|
},
|
2017-06-27 22:02:51 +02:00
|
|
|
visible: function () {
|
|
|
|
return shown;
|
|
|
|
},
|
2017-05-25 01:49:29 +02:00
|
|
|
};
|
2017-06-14 14:17:03 +02:00
|
|
|
|
|
|
|
if (selector[0] === '<') {
|
|
|
|
self.html(selector);
|
|
|
|
}
|
|
|
|
|
2017-07-08 17:41:53 +02:00
|
|
|
self[0] = 'you-must-set-the-child-yourself';
|
2017-05-25 01:49:29 +02:00
|
|
|
|
2018-04-12 14:45:06 +02:00
|
|
|
add_extensions(self);
|
|
|
|
|
2018-04-12 17:50:18 +02:00
|
|
|
self.selector = selector;
|
2017-07-08 17:41:53 +02:00
|
|
|
return self;
|
2017-05-25 01:49:29 +02:00
|
|
|
}
|
2017-05-23 03:20:15 +02:00
|
|
|
|
2018-04-05 16:46:45 +02:00
|
|
|
var zjquery = function (arg, arg2) {
|
2017-05-25 01:49:29 +02:00
|
|
|
if (typeof arg === "function") {
|
|
|
|
// If somebody is passing us a function, we emulate
|
|
|
|
// jQuery's behavior of running this function after
|
|
|
|
// page load time. But there are no pages to load,
|
|
|
|
// so we just call it right away.
|
|
|
|
arg();
|
|
|
|
return;
|
2017-06-22 23:40:40 +02:00
|
|
|
} else if (typeof arg === "object") {
|
|
|
|
// If somebody is passing us an element, we return
|
|
|
|
// the element itself if it's been created with
|
|
|
|
// zjquery.
|
|
|
|
// This may happen in cases like $(this).
|
2018-04-12 17:50:18 +02:00
|
|
|
if (arg.selector) {
|
|
|
|
if (elems[arg.selector]) {
|
2017-06-22 23:40:40 +02:00
|
|
|
return arg;
|
|
|
|
}
|
|
|
|
}
|
2018-04-12 17:37:34 +02:00
|
|
|
|
|
|
|
// We occasionally create stub objects that know
|
|
|
|
// they want to be wrapped by jQuery (so they can
|
|
|
|
// in turn return stubs). The convention is that
|
|
|
|
// they provide a to_$ attribute.
|
|
|
|
if (arg.to_$) {
|
|
|
|
assert(typeof arg.to_$ === "function");
|
|
|
|
return arg.to_$();
|
|
|
|
}
|
2017-05-25 01:49:29 +02:00
|
|
|
}
|
2017-05-24 03:34:18 +02:00
|
|
|
|
2018-04-05 16:46:45 +02:00
|
|
|
if (arg2 !== undefined) {
|
|
|
|
throw Error("We only use one-argument variations of $(...) in Zulip code.");
|
|
|
|
}
|
|
|
|
|
2017-05-25 01:49:29 +02:00
|
|
|
var selector = arg;
|
2017-07-08 15:16:19 +02:00
|
|
|
|
|
|
|
var valid_selector =
|
|
|
|
('<#.'.indexOf(selector[0]) >= 0) ||
|
|
|
|
(selector === 'window-stub') ||
|
|
|
|
(selector === 'document-stub') ||
|
2018-04-12 01:01:00 +02:00
|
|
|
(selector === 'body') ||
|
2017-07-08 15:16:19 +02:00
|
|
|
(selector === 'html') ||
|
|
|
|
(selector.location) ||
|
|
|
|
(selector.indexOf('#') >= 0) ||
|
2018-01-07 18:49:16 +01:00
|
|
|
(selector.indexOf('.') >= 0) ||
|
|
|
|
(selector.indexOf('[') >= 0 && selector.indexOf(']') >= selector.indexOf('['));
|
2017-07-08 15:16:19 +02:00
|
|
|
|
|
|
|
assert(valid_selector,
|
|
|
|
'Invalid selector: ' + selector +
|
|
|
|
' Use $.create() maybe?');
|
|
|
|
|
|
|
|
|
2017-05-25 01:49:29 +02:00
|
|
|
if (elems[selector] === undefined) {
|
|
|
|
var elem = new_elem(selector);
|
2017-07-08 17:41:53 +02:00
|
|
|
elems[selector] = elem;
|
2017-05-24 03:34:18 +02:00
|
|
|
}
|
2017-05-25 01:49:29 +02:00
|
|
|
return elems[selector];
|
|
|
|
};
|
2017-05-24 03:34:18 +02:00
|
|
|
|
2017-07-08 15:16:19 +02:00
|
|
|
zjquery.create = function (name) {
|
|
|
|
assert(!elems[name],
|
|
|
|
'You already created an object with this name!!');
|
|
|
|
var elem = new_elem(name);
|
2017-07-08 17:41:53 +02:00
|
|
|
elems[name] = elem;
|
2017-07-08 15:16:19 +02:00
|
|
|
return elems[name];
|
|
|
|
};
|
2017-05-24 03:05:29 +02:00
|
|
|
|
2017-05-25 01:49:29 +02:00
|
|
|
zjquery.stub_selector = function (selector, stub) {
|
|
|
|
elems[selector] = stub;
|
|
|
|
};
|
2017-05-24 22:31:12 +02:00
|
|
|
|
2017-05-25 01:49:29 +02:00
|
|
|
zjquery.trim = function (s) { return s; };
|
2017-05-23 03:20:15 +02:00
|
|
|
|
2017-05-25 01:49:29 +02:00
|
|
|
zjquery.state = function () {
|
|
|
|
// useful for debugging
|
|
|
|
var res = _.map(elems, function (v) {
|
|
|
|
return v.debug();
|
|
|
|
});
|
2017-05-23 03:20:15 +02:00
|
|
|
|
2017-05-25 01:49:29 +02:00
|
|
|
res = _.map(res, function (v) {
|
|
|
|
return [v.selector, v.value, v.shown];
|
|
|
|
});
|
2017-05-23 03:20:15 +02:00
|
|
|
|
2017-05-25 01:49:29 +02:00
|
|
|
res.sort();
|
2017-05-23 03:20:15 +02:00
|
|
|
|
2017-05-25 01:49:29 +02:00
|
|
|
return res;
|
|
|
|
};
|
2017-05-23 03:20:15 +02:00
|
|
|
|
2017-06-14 20:28:55 +02:00
|
|
|
zjquery.Event = function (name, data) {
|
|
|
|
return {
|
|
|
|
name: name,
|
|
|
|
data: data,
|
|
|
|
};
|
|
|
|
};
|
2017-05-23 03:20:15 +02:00
|
|
|
|
2017-06-22 23:40:39 +02:00
|
|
|
zjquery.extend = function (content, container) {
|
|
|
|
return _.extend(content, container);
|
|
|
|
};
|
|
|
|
|
2018-04-12 14:45:06 +02:00
|
|
|
zjquery.fn = fn;
|
|
|
|
|
2017-05-25 01:49:29 +02:00
|
|
|
return zjquery;
|
|
|
|
};
|
2017-05-23 03:20:15 +02:00
|
|
|
|
2017-05-25 01:49:29 +02:00
|
|
|
module.exports = exports;
|