ConcernementMapPopup.vue 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158
  1. <script>
  2. // import { RouterLink, RouterView } from 'vue-router'
  3. import SvgIcon from '@jamescoyle/vue-icon';
  4. import { mdiHeadphones } from '@mdi/js';
  5. import { mapState, mapActions } from 'pinia'
  6. // import { UserStore } from '@/stores/user'
  7. import { ConcernementsStore } from '@/stores/concernements'
  8. export default {
  9. name: 'concernementMapPopup',
  10. props: ['infos'],
  11. data() {
  12. return {
  13. dom: null,
  14. type: null,
  15. concernement: null,
  16. entite: null,
  17. besoin: null,
  18. reponse: null,
  19. superposition: null,
  20. headphones_path: mdiHeadphones
  21. }
  22. },
  23. created () {
  24. // console.log(`popup created type: ${this.infos.type}`, this.infos);
  25. if (this.infos.type === 'concernement') {
  26. this.concernement = this.concernementsByID[this.infos.id];
  27. } else if(this.infos.type === 'entite' || this.infos.type === 'entite_action') {
  28. this.entite = this.allEntitesById[this.infos.id];
  29. } else if (this.infos.type === 'besoin') {
  30. this.besoin = this.allBesoinsById[this.infos.id];
  31. } else if (this.infos.type === 'reponse') {
  32. for (let i = 0; i < this.allBesoinsById[this.infos.bid].reponses.length; i++) {
  33. if (this.allBesoinsById[this.infos.bid].reponses[i].id === this.infos.id) {
  34. this.reponse = this.allBesoinsById[this.infos.bid].reponses[i];
  35. break;
  36. }
  37. }
  38. } else if (this.infos.type === 'superposition') {
  39. this.superposition = {
  40. concernement: this.concernementsByID[this.infos.cid],
  41. entite: this.allEntitesById[this.infos.eid]
  42. }
  43. }
  44. },
  45. mounted () {
  46. // console.log('APP onMounted')
  47. this.dom = this.$refs['map-popup'];
  48. window.addEventListener('mousemove', this.onMousemove);
  49. },
  50. computed: {
  51. ...mapState(ConcernementsStore,['concernements']),
  52. ...mapState(ConcernementsStore,['concernementsByID']),
  53. ...mapState(ConcernementsStore,['allEntitesById']),
  54. ...mapState(ConcernementsStore,['allBesoinsById'])
  55. },
  56. watch: {
  57. infos: {
  58. handler (n, o){
  59. if (n.type === 'concernement') {
  60. this.concernement = this.concernementsByID[n.id];
  61. } else if(n.type === 'entite' || n.type === 'entite_action') {
  62. this.entite = this.allEntitesById[this.infos.id];
  63. } else if (n.type === 'besoin') {
  64. this.besoin = this.allBesoinsById[this.infos.id];
  65. } else if (n.type === 'reponse') {
  66. }
  67. },
  68. deep: true
  69. },
  70. },
  71. methods: {
  72. onMousemove(e){
  73. // console.log(`popup move type: ${this.infos.type}`);
  74. let v = "top";
  75. let h = "right";
  76. if (e.clientX + this.dom.clientWidth > document.body.clientWidth) {
  77. h = "left";
  78. } else {
  79. h = "right";
  80. }
  81. if (e.clientY - this.dom.clientHeight < 0) {
  82. v = "bottom";
  83. } else {
  84. v = "top";
  85. }
  86. this.dom.setAttribute("pos", `${v}-${h}`);
  87. switch (h) {
  88. case "right":
  89. this.dom.style.left = `${e.clientX+2}px`;
  90. break;
  91. case "left":
  92. this.dom.style.left = `${e.clientX - this.dom.clientWidth-2}px`;
  93. break;
  94. }
  95. switch (v) {
  96. case "top":
  97. this.dom.style.top = `${e.clientY - this.dom.clientHeight-2}px`;
  98. break;
  99. case "bottom":
  100. this.dom.style.top = `${e.clientY+2}px`;
  101. break;
  102. }
  103. },
  104. truncate( str, n, useWordBoundary ){
  105. if (str.length <= n) { return str; }
  106. const subString = str.slice(0, n-1); // the original check
  107. return (useWordBoundary
  108. ? subString.slice(0, subString.lastIndexOf(" "))
  109. : subString) + " &hellip;";
  110. }
  111. },
  112. components: {
  113. SvgIcon
  114. }
  115. }
  116. </script>
  117. <template>
  118. <div id="map-popup" ref="map-popup">
  119. <div class="popup-content-wrapper">
  120. <section v-if="infos.type === 'concernement'" class="concernement-map-popup">
  121. <h1>{{ concernement.title }}</h1>
  122. <ul class="icons" v-if="concernement.has_puissancedagir || concernement.has_proximite || concernement.has_superposition || concernement.has_agissantes || concernement.has_doleance">
  123. <li v-if="concernement.has_puissancedagir" ><span class="icon puissancedagir"></span></li>
  124. <li v-if="concernement.has_proximite" ><span class="icon proximite"></span></li>
  125. <li v-if="concernement.has_superposition" ><span class="icon superposition"></span></li>
  126. <li v-if="concernement.has_agissantes" ><span class="icon action"></span></li>
  127. <li v-if="concernement.has_doleance" ><span class="icon doleancer"></span></li>
  128. </ul>
  129. </section>
  130. <section v-if="infos.type === 'concernement' && concernement.has_recit" class="concernement-map-popup-recit">
  131. <svg-icon type="mdi" :path="headphones_path"></svg-icon>
  132. </section>
  133. <section v-if="infos.type === 'entite' || infos.type === 'entite_action'" class="entite-map-popup">
  134. <h1>{{ entite.entite.title }}</h1>
  135. </section>
  136. <section v-if="infos.type === 'besoin'" class="besoin-map-popup">
  137. <div v-html="besoin.description"></div>
  138. </section>
  139. <section v-if="infos.type === 'reponse'" class="reponse-map-popup">
  140. <div v-if="reponse.qui"><label>Qui</label><p v-html="truncate(reponse.qui, 100, true)"/></div>
  141. <div v-if="reponse.quoi"><label>Quoi</label><p v-html="truncate(reponse.quoi, 100, true)"/></div>
  142. <div v-if="reponse.ou"><label>Où</label><p v-html="truncate(reponse.ou, 100, true)"/></div>
  143. <div v-if="reponse.avec"><label>Avec</label><p v-html="truncate(reponse.avec, 100, true)"/></div>
  144. </section>
  145. </div>
  146. </div>
  147. </template>
  148. <style lang="scss" scoped>
  149. </style>