sledgehammer_bindings.rs 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285
  1. use js_sys::Function;
  2. use sledgehammer_bindgen::bindgen;
  3. use web_sys::Node;
  4. #[bindgen]
  5. mod js {
  6. const JS: &str = r#"
  7. class ListenerMap {
  8. constructor(root) {
  9. // bubbling events can listen at the root element
  10. this.global = {};
  11. // non bubbling events listen at the element the listener was created at
  12. this.local = {};
  13. this.root = null;
  14. this.handler = null;
  15. }
  16. create(event_name, element, bubbles) {
  17. if (bubbles) {
  18. if (this.global[event_name] === undefined) {
  19. this.global[event_name] = {};
  20. this.global[event_name].active = 1;
  21. this.root.addEventListener(event_name, this.handler);
  22. } else {
  23. this.global[event_name].active++;
  24. }
  25. }
  26. else {
  27. const id = element.getAttribute("data-dioxus-id");
  28. if (!this.local[id]) {
  29. this.local[id] = {};
  30. }
  31. element.addEventListener(event_name, this.handler);
  32. }
  33. }
  34. remove(element, event_name, bubbles) {
  35. if (bubbles) {
  36. this.global[event_name].active--;
  37. if (this.global[event_name].active === 0) {
  38. this.root.removeEventListener(event_name, this.global[event_name].callback);
  39. delete this.global[event_name];
  40. }
  41. }
  42. else {
  43. const id = element.getAttribute("data-dioxus-id");
  44. delete this.local[id][event_name];
  45. if (this.local[id].length === 0) {
  46. delete this.local[id];
  47. }
  48. element.removeEventListener(event_name, this.handler);
  49. }
  50. }
  51. removeAllNonBubbling(element) {
  52. const id = element.getAttribute("data-dioxus-id");
  53. delete this.local[id];
  54. }
  55. }
  56. function SetAttributeInner(node, field, value, ns) {
  57. const name = field;
  58. if (ns === "style") {
  59. // ????? why do we need to do this
  60. if (node.style === undefined) {
  61. node.style = {};
  62. }
  63. node.style[name] = value;
  64. } else if (ns !== null && ns !== undefined && ns !== "") {
  65. node.setAttributeNS(ns, name, value);
  66. } else {
  67. switch (name) {
  68. case "value":
  69. if (value !== node.value) {
  70. node.value = value;
  71. }
  72. break;
  73. case "initial_value":
  74. node.defaultValue = value;
  75. break;
  76. case "checked":
  77. node.checked = truthy(value);
  78. break;
  79. case "selected":
  80. node.selected = truthy(value);
  81. break;
  82. case "initial_selected":
  83. node.defaultSelected = truthy(value);
  84. break;
  85. case "dangerous_inner_html":
  86. node.innerHTML = value;
  87. break;
  88. default:
  89. // https://github.com/facebook/react/blob/8b88ac2592c5f555f315f9440cbb665dd1e7457a/packages/react-dom/src/shared/DOMProperty.js#L352-L364
  90. if (!truthy(value) && bool_attrs.hasOwnProperty(name)) {
  91. node.removeAttribute(name);
  92. } else {
  93. node.setAttribute(name, value);
  94. }
  95. }
  96. }
  97. }
  98. function LoadChild(ptr, len) {
  99. // iterate through each number and get that child
  100. node = stack[stack.length - 1];
  101. ptr_end = ptr + len;
  102. for (; ptr < ptr_end; ptr++) {
  103. end = m.getUint8(ptr);
  104. for (node = node.firstChild; end > 0; end--) {
  105. node = node.nextSibling;
  106. }
  107. }
  108. return node;
  109. }
  110. const listeners = new ListenerMap();
  111. let nodes = [];
  112. let stack = [];
  113. let root;
  114. const templates = {};
  115. let node, els, end, ptr_end, k;
  116. export function save_template(nodes, tmpl_id) {
  117. templates[tmpl_id] = nodes;
  118. }
  119. export function set_node(id, node) {
  120. nodes[id] = node;
  121. }
  122. export function get_node(id) {
  123. return nodes[id];
  124. }
  125. export function initilize(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. const bool_attrs = {
  139. allowfullscreen: true,
  140. allowpaymentrequest: true,
  141. async: true,
  142. autofocus: true,
  143. autoplay: true,
  144. checked: true,
  145. controls: true,
  146. default: true,
  147. defer: true,
  148. disabled: true,
  149. formnovalidate: true,
  150. hidden: true,
  151. ismap: true,
  152. itemscope: true,
  153. loop: true,
  154. multiple: true,
  155. muted: true,
  156. nomodule: true,
  157. novalidate: true,
  158. open: true,
  159. playsinline: true,
  160. readonly: true,
  161. required: true,
  162. reversed: true,
  163. selected: true,
  164. truespeed: true,
  165. webkitdirectory: true,
  166. };
  167. function truthy(val) {
  168. return val === "true" || val === true;
  169. }
  170. "#;
  171. extern "C" {
  172. #[wasm_bindgen]
  173. pub fn save_template(nodes: Vec<Node>, tmpl_id: u32);
  174. #[wasm_bindgen]
  175. pub fn set_node(id: u32, node: Node);
  176. #[wasm_bindgen]
  177. pub fn get_node(id: u32) -> Node;
  178. #[wasm_bindgen]
  179. pub fn initilize(root: Node, handler: &Function);
  180. }
  181. fn mount_to_root() {
  182. "{AppendChildren(root, stack.length-1);}"
  183. }
  184. fn push_root(root: u32) {
  185. "{stack.push(nodes[$root$]);}"
  186. }
  187. fn append_children(id: u32, many: u32) {
  188. "{AppendChildren($id$, $many$);}"
  189. }
  190. fn pop_root() {
  191. "{stack.pop();}"
  192. }
  193. fn replace_with(id: u32, n: u32) {
  194. "{root = nodes[$id$]; els = stack.splice(stack.length-$n$); if (root.listening) { listeners.removeAllNonBubbling(root); } root.replaceWith(...els);}"
  195. }
  196. fn insert_after(id: u32, n: u32) {
  197. "{nodes[$id$].after(...stack.splice(stack.length-$n$));}"
  198. }
  199. fn insert_before(id: u32, n: u32) {
  200. "{nodes[$id$].before(...stack.splice(stack.length-$n$));}"
  201. }
  202. fn remove(id: u32) {
  203. "{node = nodes[$id$]; if (node !== undefined) { if (node.listening) { listeners.removeAllNonBubbling(node); } node.remove(); }}"
  204. }
  205. fn create_raw_text(text: &str) {
  206. "{stack.push(document.createTextNode($text$));}"
  207. }
  208. fn create_text_node(text: &str, id: u32) {
  209. "{node = document.createTextNode($text$); nodes[$id$] = node; stack.push(node);}"
  210. }
  211. fn create_placeholder(id: u32) {
  212. "{node = document.createElement('pre'); node.hidden = true; stack.push(node); nodes[$id$] = node;}"
  213. }
  214. fn new_event_listener(event_name: &str<u8, evt>, id: u32, bubbles: u8) {
  215. 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$);"#
  216. }
  217. fn remove_event_listener(event_name: &str<u8, evt>, id: u32, bubbles: u8) {
  218. "{node = nodes[$id$]; node.listening -= 1; node.removeAttribute('data-dioxus-id'); listeners.remove(node, $event_name$, $bubbles$);}"
  219. }
  220. fn set_text(id: u32, text: &str) {
  221. "{nodes[$id$].textContent = $text$;}"
  222. }
  223. fn set_attribute(id: u32, field: &str<u8, attr>, value: &str, ns: &str<u8, ns_cache>) {
  224. "{node = nodes[$id$]; SetAttributeInner(node, $field$, $value$, $ns$);}"
  225. }
  226. fn remove_attribute(id: u32, field: &str<u8, attr>, ns: &str<u8, ns_cache>) {
  227. r#"{
  228. node = nodes[$id$];
  229. if (!ns) {
  230. switch (field) {
  231. case "value":
  232. node.value = "";
  233. break;
  234. case "checked":
  235. node.checked = false;
  236. break;
  237. case "selected":
  238. node.selected = false;
  239. break;
  240. case "dangerous_inner_html":
  241. node.innerHTML = "";
  242. break;
  243. default:
  244. node.removeAttribute(field);
  245. break;
  246. }
  247. } else if (ns == "style") {
  248. node.style.removeProperty(name);
  249. } else {
  250. node.removeAttributeNS(ns, field);
  251. }
  252. }"#
  253. }
  254. fn assign_id(ptr: u32, len: u8, id: u32) {
  255. "{nodes[$id$] = LoadChild($ptr$, $len$);}"
  256. }
  257. fn hydrate_text(ptr: u32, len: u8, value: &str, id: u32) {
  258. r#"{
  259. node = LoadChild($ptr$, $len$);
  260. if (node.nodeType == Node.TEXT_NODE) {
  261. node.textContent = value;
  262. } else {
  263. let text = document.createTextNode(value);
  264. node.replaceWith(text);
  265. node = text;
  266. }
  267. nodes[$id$] = node;
  268. }"#
  269. }
  270. fn replace_placeholder(ptr: u32, len: u8, n: u32) {
  271. "{els = stack.splice(stack.length - $n$); node = LoadChild($ptr$, $len$); node.replaceWith(...els);}"
  272. }
  273. fn load_template(tmpl_id: u32, index: u32, id: u32) {
  274. "{node = templates[$tmpl_id$][$index$].cloneNode(true); nodes[$id$] = node; stack.push(node);}"
  275. }
  276. }