sledgehammer_bindings.rs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411
  1. #[cfg(feature = "web")]
  2. use js_sys::Function;
  3. #[cfg(feature = "web")]
  4. use sledgehammer_bindgen::bindgen;
  5. #[cfg(feature = "web")]
  6. use web_sys::Node;
  7. #[cfg(feature = "web")]
  8. pub const SLEDGEHAMMER_JS: &str = GENERATED_JS;
  9. #[cfg(feature = "web")]
  10. #[bindgen(module)]
  11. mod js {
  12. const JS_FILE: &str = "./src/common.js";
  13. const JS: &str = r#"
  14. class ListenerMap {
  15. constructor(root) {
  16. // bubbling events can listen at the root element
  17. this.global = {};
  18. // non bubbling events listen at the element the listener was created at
  19. this.local = {};
  20. this.root = null;
  21. this.handler = null;
  22. }
  23. create(event_name, element, bubbles) {
  24. if (bubbles) {
  25. if (this.global[event_name] === undefined) {
  26. this.global[event_name] = {};
  27. this.global[event_name].active = 1;
  28. this.root.addEventListener(event_name, this.handler);
  29. } else {
  30. this.global[event_name].active++;
  31. }
  32. }
  33. else {
  34. const id = element.getAttribute("data-dioxus-id");
  35. if (!this.local[id]) {
  36. this.local[id] = {};
  37. }
  38. element.addEventListener(event_name, this.handler);
  39. }
  40. }
  41. remove(element, event_name, bubbles) {
  42. if (bubbles) {
  43. this.global[event_name].active--;
  44. if (this.global[event_name].active === 0) {
  45. this.root.removeEventListener(event_name, this.global[event_name].callback);
  46. delete this.global[event_name];
  47. }
  48. }
  49. else {
  50. const id = element.getAttribute("data-dioxus-id");
  51. delete this.local[id][event_name];
  52. if (this.local[id].length === 0) {
  53. delete this.local[id];
  54. }
  55. element.removeEventListener(event_name, this.handler);
  56. }
  57. }
  58. removeAllNonBubbling(element) {
  59. const id = element.getAttribute("data-dioxus-id");
  60. delete this.local[id];
  61. }
  62. }
  63. function LoadChild(ptr, len) {
  64. // iterate through each number and get that child
  65. node = stack[stack.length - 1];
  66. ptr_end = ptr + len;
  67. for (; ptr < ptr_end; ptr++) {
  68. end = m.getUint8(ptr);
  69. for (node = node.firstChild; end > 0; end--) {
  70. node = node.nextSibling;
  71. }
  72. }
  73. return node;
  74. }
  75. const listeners = new ListenerMap();
  76. let nodes = [];
  77. let stack = [];
  78. let root;
  79. const templates = {};
  80. let node, els, end, ptr_end, k;
  81. export function save_template(nodes, tmpl_id) {
  82. templates[tmpl_id] = nodes;
  83. }
  84. export function hydrate(ids) {
  85. console.log("hydrating", ids);
  86. const hydrateNodes = document.querySelectorAll('[data-node-hydration]');
  87. for (let i = 0; i < hydrateNodes.length; i++) {
  88. const hydrateNode = hydrateNodes[i];
  89. const hydration = hydrateNode.getAttribute('data-node-hydration');
  90. const split = hydration.split(',');
  91. const id = ids[parseInt(split[0])];
  92. nodes[id] = hydrateNode;
  93. console.log("hydrating node", hydrateNode, id);
  94. if (split.length > 1) {
  95. hydrateNode.listening = split.length - 1;
  96. hydrateNode.setAttribute('data-dioxus-id', id);
  97. for (let j = 1; j < split.length; j++) {
  98. const listener = split[j];
  99. const split2 = listener.split(':');
  100. const event_name = split2[0];
  101. const bubbles = split2[1] === '1';
  102. console.log("hydrating listener", event_name, bubbles);
  103. listeners.create(event_name, hydrateNode, bubbles);
  104. }
  105. }
  106. }
  107. const treeWalker = document.createTreeWalker(
  108. document.body,
  109. NodeFilter.SHOW_COMMENT,
  110. );
  111. let currentNode = treeWalker.nextNode();
  112. while (currentNode) {
  113. const id = currentNode.textContent;
  114. const split = id.split('node-id');
  115. if (split.length > 1) {
  116. console.log("hydrating text", currentNode.nextSibling, id);
  117. nodes[ids[parseInt(split[1])]] = currentNode.nextSibling;
  118. }
  119. currentNode = treeWalker.nextNode();
  120. }
  121. }
  122. export function get_node(id) {
  123. return nodes[id];
  124. }
  125. export function initialize(root, handler) {
  126. listeners.handler = handler;
  127. nodes = [root];
  128. stack = [root];
  129. listeners.root = root;
  130. }
  131. function AppendChildren(id, many){
  132. root = nodes[id];
  133. els = stack.splice(stack.length-many);
  134. for (k = 0; k < many; k++) {
  135. root.appendChild(els[k]);
  136. }
  137. }
  138. "#;
  139. extern "C" {
  140. #[wasm_bindgen]
  141. pub fn save_template(nodes: Vec<Node>, tmpl_id: u16);
  142. #[wasm_bindgen]
  143. pub fn hydrate(ids: Vec<u32>);
  144. #[wasm_bindgen]
  145. pub fn get_node(id: u32) -> Node;
  146. #[wasm_bindgen]
  147. pub fn initialize(root: Node, handler: &Function);
  148. }
  149. fn mount_to_root() {
  150. "{AppendChildren(root, stack.length-1);}"
  151. }
  152. fn push_root(root: u32) {
  153. "{stack.push(nodes[$root$]);}"
  154. }
  155. fn append_children(id: u32, many: u16) {
  156. "{AppendChildren($id$, $many$);}"
  157. }
  158. fn pop_root() {
  159. "{stack.pop();}"
  160. }
  161. fn replace_with(id: u32, n: u16) {
  162. "{root = nodes[$id$]; els = stack.splice(stack.length-$n$); if (root.listening) { listeners.removeAllNonBubbling(root); } root.replaceWith(...els);}"
  163. }
  164. fn insert_after(id: u32, n: u16) {
  165. "{nodes[$id$].after(...stack.splice(stack.length-$n$));}"
  166. }
  167. fn insert_before(id: u32, n: u16) {
  168. "{nodes[$id$].before(...stack.splice(stack.length-$n$));}"
  169. }
  170. fn remove(id: u32) {
  171. "{node = nodes[$id$]; if (node !== undefined) { if (node.listening) { listeners.removeAllNonBubbling(node); } node.remove(); }}"
  172. }
  173. fn create_raw_text(text: &str) {
  174. "{stack.push(document.createTextNode($text$));}"
  175. }
  176. fn create_text_node(text: &str, id: u32) {
  177. "{node = document.createTextNode($text$); nodes[$id$] = node; stack.push(node);}"
  178. }
  179. fn create_placeholder(id: u32) {
  180. "{node = document.createElement('pre'); node.hidden = true; stack.push(node); nodes[$id$] = node;}"
  181. }
  182. fn new_event_listener(event_name: &str<u8, evt>, id: u32, bubbles: u8) {
  183. r#"node = nodes[id]; if(node.listening){node.listening += 1;}else{node.listening = 1;} node.setAttribute('data-dioxus-id', `\${id}`); listeners.create($event_name$, node, $bubbles$);"#
  184. }
  185. fn remove_event_listener(event_name: &str<u8, evt>, id: u32, bubbles: u8) {
  186. "{node = nodes[$id$]; node.listening -= 1; node.removeAttribute('data-dioxus-id'); listeners.remove(node, $event_name$, $bubbles$);}"
  187. }
  188. fn set_text(id: u32, text: &str) {
  189. "{nodes[$id$].textContent = $text$;}"
  190. }
  191. fn set_attribute(id: u32, field: &str<u8, attr>, value: &str, ns: &str<u8, ns_cache>) {
  192. "{node = nodes[$id$]; setAttributeInner(node, $field$, $value$, $ns$);}"
  193. }
  194. fn remove_attribute(id: u32, field: &str<u8, attr>, ns: &str<u8, ns_cache>) {
  195. r#"{
  196. node = nodes[$id$];
  197. if (!ns) {
  198. switch (field) {
  199. case "value":
  200. node.value = "";
  201. break;
  202. case "checked":
  203. node.checked = false;
  204. break;
  205. case "selected":
  206. node.selected = false;
  207. break;
  208. case "dangerous_inner_html":
  209. node.innerHTML = "";
  210. break;
  211. default:
  212. node.removeAttribute(field);
  213. break;
  214. }
  215. } else if (ns == "style") {
  216. node.style.removeProperty(name);
  217. } else {
  218. node.removeAttributeNS(ns, field);
  219. }
  220. }"#
  221. }
  222. fn assign_id(ptr: u32, len: u8, id: u32) {
  223. "{nodes[$id$] = LoadChild($ptr$, $len$);}"
  224. }
  225. fn hydrate_text(ptr: u32, len: u8, value: &str, id: u32) {
  226. r#"{
  227. node = LoadChild($ptr$, $len$);
  228. if (node.nodeType == Node.TEXT_NODE) {
  229. node.textContent = value;
  230. } else {
  231. let text = document.createTextNode(value);
  232. node.replaceWith(text);
  233. node = text;
  234. }
  235. nodes[$id$] = node;
  236. }"#
  237. }
  238. fn replace_placeholder(ptr: u32, len: u8, n: u16) {
  239. "{els = stack.splice(stack.length - $n$); node = LoadChild($ptr$, $len$); node.replaceWith(...els);}"
  240. }
  241. fn load_template(tmpl_id: u16, index: u16, id: u32) {
  242. "{node = templates[$tmpl_id$][$index$].cloneNode(true); nodes[$id$] = node; stack.push(node);}"
  243. }
  244. }
  245. #[cfg(feature = "binary-protocol")]
  246. pub mod binary_protocol {
  247. use sledgehammer_bindgen::bindgen;
  248. pub const SLEDGEHAMMER_JS: &str = GENERATED_JS;
  249. #[bindgen]
  250. mod protocol_js {
  251. const JS_FILE: &str = "./src/interpreter.js";
  252. const JS_FILE: &str = "./src/common.js";
  253. fn mount_to_root() {
  254. "{AppendChildren(root, stack.length-1);}"
  255. }
  256. fn push_root(root: u32) {
  257. "{stack.push(nodes[$root$]);}"
  258. }
  259. fn append_children(id: u32, many: u16) {
  260. "{AppendChildren($id$, $many$);}"
  261. }
  262. fn append_children_to_top(many: u16) {
  263. "{
  264. root = stack[stack.length-many-1];
  265. els = stack.splice(stack.length-many);
  266. for (k = 0; k < many; k++) {
  267. root.appendChild(els[k]);
  268. }
  269. }"
  270. }
  271. fn pop_root() {
  272. "{stack.pop();}"
  273. }
  274. fn replace_with(id: u32, n: u16) {
  275. "{root = nodes[$id$]; els = stack.splice(stack.length-$n$); if (root.listening) { listeners.removeAllNonBubbling(root); } root.replaceWith(...els);}"
  276. }
  277. fn insert_after(id: u32, n: u16) {
  278. "{nodes[$id$].after(...stack.splice(stack.length-$n$));}"
  279. }
  280. fn insert_before(id: u32, n: u16) {
  281. "{nodes[$id$].before(...stack.splice(stack.length-$n$));}"
  282. }
  283. fn remove(id: u32) {
  284. "{node = nodes[$id$]; if (node !== undefined) { if (node.listening) { listeners.removeAllNonBubbling(node); } node.remove(); }}"
  285. }
  286. fn create_raw_text(text: &str) {
  287. "{stack.push(document.createTextNode($text$));}"
  288. }
  289. fn create_text_node(text: &str, id: u32) {
  290. "{node = document.createTextNode($text$); nodes[$id$] = node; stack.push(node);}"
  291. }
  292. fn create_element(element: &'static str<u8, el>) {
  293. "{stack.push(document.createElement($element$))}"
  294. }
  295. fn create_element_ns(element: &'static str<u8, el>, ns: &'static str<u8, namespace>) {
  296. "{stack.push(document.createElementNS($ns$, $element$))}"
  297. }
  298. fn create_placeholder(id: u32) {
  299. "{node = document.createElement('pre'); node.hidden = true; stack.push(node); nodes[$id$] = node;}"
  300. }
  301. fn add_placeholder() {
  302. "{node = document.createElement('pre'); node.hidden = true; stack.push(node);}"
  303. }
  304. fn new_event_listener(event: &str<u8, evt>, id: u32, bubbles: u8) {
  305. r#"
  306. bubbles = bubbles == 1;
  307. node = nodes[id];
  308. if(node.listening){
  309. node.listening += 1;
  310. } else {
  311. node.listening = 1;
  312. }
  313. node.setAttribute('data-dioxus-id', `\${id}`);
  314. const event_name = $event$;
  315. // if this is a mounted listener, we send the event immediately
  316. if (event_name === "mounted") {
  317. window.ipc.postMessage(
  318. window.interpreter.serializeIpcMessage("user_event", {
  319. name: event_name,
  320. element: id,
  321. data: null,
  322. bubbles,
  323. })
  324. );
  325. } else {
  326. listeners.create(event_name, node, bubbles, (event) => {
  327. handler(event, event_name, bubbles, config);
  328. });
  329. }"#
  330. }
  331. fn remove_event_listener(event_name: &str<u8, evt>, id: u32, bubbles: u8) {
  332. "{node = nodes[$id$]; node.listening -= 1; node.removeAttribute('data-dioxus-id'); listeners.remove(node, $event_name$, $bubbles$);}"
  333. }
  334. fn set_text(id: u32, text: &str) {
  335. "{nodes[$id$].textContent = $text$;}"
  336. }
  337. fn set_attribute(id: u32, field: &str<u8, attr>, value: &str, ns: &str<u8, ns_cache>) {
  338. "{node = nodes[$id$]; setAttributeInner(node, $field$, $value$, $ns$);}"
  339. }
  340. fn set_top_attribute(field: &str<u8, attr>, value: &str, ns: &str<u8, ns_cache>) {
  341. "{setAttributeInner(stack[stack.length-1], $field$, $value$, $ns$);}"
  342. }
  343. fn remove_attribute(id: u32, field: &str<u8, attr>, ns: &str<u8, ns_cache>) {
  344. r#"{
  345. node = nodes[$id$];
  346. if (!ns) {
  347. switch (field) {
  348. case "value":
  349. node.value = "";
  350. break;
  351. case "checked":
  352. node.checked = false;
  353. break;
  354. case "selected":
  355. node.selected = false;
  356. break;
  357. case "dangerous_inner_html":
  358. node.innerHTML = "";
  359. break;
  360. default:
  361. node.removeAttribute(field);
  362. break;
  363. }
  364. } else if (ns == "style") {
  365. node.style.removeProperty(name);
  366. } else {
  367. node.removeAttributeNS(ns, field);
  368. }
  369. }"#
  370. }
  371. fn assign_id(array: &[u8], id: u32) {
  372. "{nodes[$id$] = LoadChild($array$);}"
  373. }
  374. fn hydrate_text(array: &[u8], value: &str, id: u32) {
  375. r#"{
  376. node = LoadChild($array$);
  377. if (node.nodeType == Node.TEXT_NODE) {
  378. node.textContent = value;
  379. } else {
  380. let text = document.createTextNode(value);
  381. node.replaceWith(text);
  382. node = text;
  383. }
  384. nodes[$id$] = node;
  385. }"#
  386. }
  387. fn replace_placeholder(array: &[u8], n: u16) {
  388. "{els = stack.splice(stack.length - $n$); node = LoadChild($array$); node.replaceWith(...els);}"
  389. }
  390. fn load_template(tmpl_id: u16, index: u16, id: u32) {
  391. "{node = templates[$tmpl_id$][$index$].cloneNode(true); nodes[$id$] = node; stack.push(node);}"
  392. }
  393. fn add_templates(tmpl_id: u16, len: u16) {
  394. "{templates[$tmpl_id$] = stack.splice(stack.length-$len$);}"
  395. }
  396. }
  397. }