boussole.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331
  1. (function ($, Drupal, once) {
  2. var _boussole;
  3. Drupal.behaviors.boussole = {
  4. attach: function (context, settings) {
  5. console.log("Drupal Boussole behavior", context);
  6. let $boussole_wrapper = document.querySelector('#boussole-layout');
  7. once('boussole-class-behaviour', '#boussole-layout', context).forEach(element => {
  8. let tabs =
  9. // div.field-group-tabs-wrapper
  10. // input.horizontal-tabs-active-tab
  11. _boussole = new Boussole($boussole_wrapper);
  12. });
  13. let $form = $boussole_wrapper.querySelector(':scope>.form-item');
  14. console.log($form);
  15. let $tabledrag = $form.querySelector('table.field-multiple-table');
  16. console.log($tabledrag);
  17. let $form_entities = $tabledrag.querySelectorAll(':scope>tbody>tr>td:not(.field-multiple-drag):not(.delta-order)>div');
  18. console.log($form_entities);
  19. let entities = {};
  20. [...$form_entities].forEach($entity => {
  21. console.log($entity);
  22. let ajax_new_content = $entity.classList.contains('ajax-new-content');
  23. if(ajax_new_content){
  24. // its a new paragraph just added by ajax
  25. $entity = $entity.querySelector(':scope>div');
  26. }
  27. let id = $entity.getAttribute('id');
  28. let $af = $entity.querySelector('.paragraphs-subform .field--name-field-actuel-future');
  29. let $p = $entity.querySelector('.paragraphs-subform .field--name-field-prise input');
  30. let $mm = $entity.querySelector('.paragraphs-subform .field--name-field-menace-maintien-degres input');
  31. let e = {
  32. wrapper:{
  33. $dom: $entity,
  34. id: id
  35. },
  36. title: {
  37. $dom: null,
  38. value: null
  39. },
  40. af: {
  41. $dom: $af,
  42. value: parseInt($af.querySelector('input:checked').value)
  43. },
  44. p: {
  45. $dom: $p,
  46. value: parseFloat($p.value)
  47. },
  48. mm: {
  49. $dom: $mm,
  50. value: parseFloat($mm.value)
  51. }
  52. }
  53. // title does not exists when we add an "entité intégrée" as we also have to create the referenced node
  54. let $title = $entity.querySelector('.paragraphs-subform>.field--type-entity-reference.field--name-field-entite td.inline-entity-form-node-label');
  55. if ($title) {
  56. e.title= {
  57. $dom: $title,
  58. value: $title.textContent
  59. };
  60. }
  61. entities[id] = e;
  62. });
  63. console.log(entities);
  64. _boussole.updateCreateEntities(entities);
  65. }
  66. };
  67. class Boussole {
  68. constructor($boussole_wrapper) {
  69. this.$boussole_wrapper = $boussole_wrapper;
  70. this.$boussole = document.querySelector('#boussole-layout>.boussole-wrapper>.boussole');
  71. this.entities = {}
  72. this.initTabChecking();
  73. this.enableDragArea()
  74. }
  75. initTabChecking(){
  76. // debugger
  77. this.$field_group_tab = this.$boussole_wrapper.closest('details.field-group-tab');
  78. // watch for tabs opening and closing
  79. // hidden input do not triggers events
  80. // so we have to use mutationObserver tomake it trigger the change event
  81. this.$tabInput = this.$field_group_tab.parentElement.querySelector(':scope>input[type="hidden"]');
  82. let observer = new MutationObserver(function(mutations, observer) {
  83. if(mutations[0].attributeName == "value") {
  84. // console.log('mutation observed', mutations, observer);
  85. let event = new Event('change');
  86. mutations[0].target.dispatchEvent(event);
  87. }
  88. });
  89. observer.observe(this.$tabInput, {attributes: true});
  90. // then we listen to the event
  91. this.$tabInput.addEventListener('change', event => {
  92. // console.log('tab input changed', event, this);
  93. this.wrapperTabChanged(event);
  94. });
  95. this.checkTabIsOpen();
  96. }
  97. checkTabIsOpen(){
  98. this.tabisopen = this.$field_group_tab.getAttribute('id') === this.$tabInput.value;
  99. }
  100. wrapperTabChanged(event){
  101. this.checkTabIsOpen();
  102. if(this.tabisopen){
  103. this.updateEntities();
  104. }
  105. }
  106. enableDragArea(){
  107. this.$boussole.addEventListener("dragenter", (event) => {
  108. event.preventDefault();
  109. });
  110. this.$boussole.addEventListener("dragover", (event) => {
  111. event.preventDefault();
  112. });
  113. this.$boussole.addEventListener("drop", (event) => {
  114. event.preventDefault();
  115. let data = JSON.parse(event.dataTransfer.getData('text/plain'));
  116. console.log('boussole drop', event, data);
  117. data.clientX = event.clientX;
  118. data.clientY = event.clientY;
  119. // return on the entity class
  120. this.entities[data.entity_id].dropedOnBoussole(data);
  121. });
  122. }
  123. updateCreateEntities(entities){
  124. // console.log('Class Boussole update entities', entities);
  125. for (const [id, values] of Object.entries(entities)) {
  126. console.log(id, typeof this.entities[id], values);
  127. if (typeof this.entities[id] === 'undefined') {
  128. this.entities[id] = new Entity(this.$boussole, id, values);
  129. }else{
  130. this.entities[id].updateValuesFromForm(values);
  131. }
  132. }
  133. }
  134. updateEntities(){
  135. for (const [id, entity] of Object.entries(this.entities)) {
  136. entity.updatePos();
  137. }
  138. }
  139. }
  140. class Entity {
  141. constructor($container, id, values){
  142. console.log('Entity constructor', $container, id, values);
  143. this.$container = $container;
  144. this.updateSceneSize();
  145. this.id = id;
  146. this.values = values;
  147. this.listenToFormValue();
  148. this.createDomElemt();
  149. }
  150. updateSceneSize(){
  151. this.sceneSize = {
  152. w: this.$container.clientWidth,
  153. h: this.$container.clientHeight
  154. };
  155. }
  156. createDomElemt(){
  157. this.$e = document.createElement('div');
  158. this.$e.classList.add('entity');
  159. this.$e.setAttribute('id', `${this.id}-boussole-point`);
  160. if (this.values.title.value) {
  161. // the paragraph as an entity referenced
  162. this.$e.setAttribute('title', this.values.title.value);
  163. }else{
  164. // new paragraph with no entity referenced
  165. this.$e.classList.add('ajax-new');
  166. }
  167. this.$container.append(this.$e);
  168. console.log(this.$e);
  169. this.updatePos();
  170. this.initDraggable()
  171. }
  172. updateDomElemt(){
  173. if (this.values.title.value) {
  174. // the paragraph as an entity referenced
  175. this.$e.setAttribute('title', this.values.title.value);
  176. this.$e.classList.remove('ajax-new');
  177. }else{
  178. // new paragraph with no entity referenced
  179. this.$e.classList.add('ajax-new');
  180. }
  181. }
  182. initDraggable(){
  183. this.$e.setAttribute('draggable', true);
  184. this.$e.addEventListener('dragstart', event => {
  185. // console.log('dragstart', event);
  186. let data = {
  187. entity_id : this.id,
  188. dragging_start_x : event.target.offsetLeft - event.clientX,
  189. dragging_start_y : event.target.offsetTop - event.clientY
  190. };
  191. event.dataTransfer.setData('text/plain', JSON.stringify(data));
  192. // event.dataTransfer.effectAllowed = "move";
  193. setTimeout(() => {event.target.classList.add('hide');}, 0);
  194. });
  195. this.$e.addEventListener('dragend', event => {
  196. // console.log('dragend', event);
  197. event.target.classList.remove('hide');
  198. });
  199. // CAN'T DO THAT, EVENT IS FIRED FOR EACH ENTITIES
  200. // TAKING CARE OF THE DROP IN THE BOUSSOLE CLASS
  201. // this.$container.addEventListener('drop', event =>{
  202. // event.preventDefault();
  203. // console.log('drop', event);
  204. // })
  205. }
  206. dropedOnBoussole(data){
  207. this.$e.style.left = data.clientX + data.dragging_start_x + 'px';
  208. this.$e.style.top = data.clientY + data.dragging_start_y + 'px';
  209. this.updateFormValuesFromBoussole();
  210. }
  211. updateFormValuesFromBoussole(values){
  212. let compstyle = window.getComputedStyle(this.$e);
  213. // AF
  214. let top = parseInt(compstyle.top);
  215. let af = top >= this.sceneSize.h/2 ? 0 : 1;
  216. this.values.af.value = af;
  217. this.values.af.$dom.querySelector(`input[value="${af}"]`).checked = true;
  218. // (x, y)
  219. let x = parseInt(compstyle.left) - this.sceneSize.h/2;
  220. let y = Math.abs(this.sceneSize.h/2 - top);
  221. let a = parseFloat(Math.atan(x/y) * (180/Math.PI)).toFixed(2);
  222. let r = Math.sqrt(Math.pow(x,2)+Math.pow(y,2));
  223. this.values.mm.value = this.values.mm.$dom.value = a;
  224. // https://stackoverflow.com/questions/5731863/mapping-a-numeric-range-onto-another
  225. // from range 0 -> rayon_de_la_boussole to range 0 -> 100
  226. let r_max = this.sceneSize.w/2;
  227. let r_slope = (100 - 0) / (r_max - 0);
  228. this.values.p.value = this.values.p.$dom.value = 100 - parseFloat(0 + r_slope * (r - 0)).toFixed(2);
  229. // this.values.p.$dom.value = this.values.p.value;
  230. console.log('x:', x, 'r:', r, 'a:', a, "y:", y);
  231. }
  232. listenToFormValue(){
  233. ['af','p','mm'].forEach(v =>{
  234. this.values[v].$dom.addEventListener('change', function(v, event) {
  235. switch (v) {
  236. case 'af':
  237. this.values[v].value = parseInt(this.values[v].$dom.querySelector('input:checked').value);
  238. break;
  239. default:
  240. this.values[v].value = parseInt(this.values[v].$dom.value);
  241. break;
  242. }
  243. console.log(`new ${v}`, this.values[v].value);
  244. this.updatePos();
  245. }.bind(this, v));
  246. })
  247. }
  248. updateValuesFromForm(values){
  249. this.values = values;
  250. this.updateDomElemt();
  251. }
  252. updatePos(){
  253. console.log('update pos');
  254. this.updateSceneSize()
  255. // converte degres to radian, inverse it and rotate it
  256. let a = parseInt((this.values.mm.value) * -1 + 90) * (Math.PI/180);
  257. // https://stackoverflow.com/questions/5731863/mapping-a-numeric-range-onto-another
  258. // from range 0 -> 100 to range 0 -> diagonale de la scene
  259. // let diagonale = Math.sqrt(Math.pow(this.sceneSize.w, 2) + Math.pow(this.sceneSize.h, 2))/2;
  260. // let r_max = diagonale;
  261. // actualy we don't use diagonale, we stay on the circle
  262. let r_max = this.sceneSize.w/2;
  263. let r_slope = (r_max - 0) / (100 - 0)
  264. // we inverse p as the more p you have the closest you are from the center
  265. let r = parseInt(0 + r_slope * ((100 - this.values.p.value) - 0));
  266. let x = r * Math.cos(a);
  267. let y = r * Math.sin(a);
  268. // console.log('x:', x, 'r:', r, 'a:', a, "y:", y);
  269. let af = this.values.af.value;
  270. let left = this.sceneSize.w / 2 + x;
  271. let top = this.sceneSize.h / 2 - (af ? y : -y);
  272. // console.log('left:', left, "top:", top);
  273. this.$e.style.left = left+'px';
  274. this.$e.style.top = top+'px';
  275. }
  276. }
  277. })(jQuery, Drupal, once);