From d3176e2a8da7fc746ad7b0f96e6ccdeab196a6fb Mon Sep 17 00:00:00 2001
From: Thomas Citharel <tcit@tcit.fr>
Date: Wed, 29 May 2019 16:46:23 +0200
Subject: [PATCH] Add mentions

Signed-off-by: Thomas Citharel <tcit@tcit.fr>
---
 js/package.json               |   1 +
 js/src/components/Editor.vue  | 280 +++++++++++++++++++++++++++++++++-
 js/src/graphql/search.ts      |  18 +++
 js/src/types/search.model.ts  |   7 +-
 js/src/views/Event/Create.vue |   6 -
 js/yarn.lock                  |  12 ++
 6 files changed, 309 insertions(+), 15 deletions(-)

diff --git a/js/package.json b/js/package.json
index 27f0fedad..f0b61c7f3 100644
--- a/js/package.json
+++ b/js/package.json
@@ -27,6 +27,7 @@
     "lodash": "^4.17.11",
     "ngeohash": "^0.6.3",
     "register-service-worker": "^1.6.2",
+    "tippy.js": "^4.3.1",
     "tiptap": "^1.20.1",
     "tiptap-extensions": "^1.20.1",
     "typeface-signika": "0.0.72",
diff --git a/js/src/components/Editor.vue b/js/src/components/Editor.vue
index fe8b43044..b16b616f7 100644
--- a/js/src/components/Editor.vue
+++ b/js/src/components/Editor.vue
@@ -1,4 +1,5 @@
 <template>
+    <div>
     <div class="editor">
         <editor-menu-bar :editor="editor" v-slot="{ commands, isActive, focused }">
             <div class="menubar bar-is-hidden" :class="{ 'is-focused': focused }">
@@ -122,6 +123,23 @@
 
         <editor-content class="editor__content" :editor="editor" />
     </div>
+        <div class="suggestion-list" v-show="showSuggestions" ref="suggestions">
+            <template v-if="hasResults">
+                <div
+                        v-for="(actor, index) in filteredActors"
+                        :key="actor.id"
+                        class="suggestion-list__item"
+                        :class="{ 'is-selected': navigatedUserIndex === index }"
+                        @click="selectActor(actor)"
+                >
+                    {{ actor.name }}
+                </div>
+            </template>
+            <div v-else class="suggestion-list__item is-empty">
+                No actors found
+            </div>
+        </div>
+    </div>
 </template>
 
 <script lang="ts">
@@ -142,8 +160,14 @@ import {
     Link,
     Underline,
     History,
-    Placeholder
+    Placeholder,
+    Mention,
+    Image,
 } from 'tiptap-extensions';
+import tippy, { Instance } from 'tippy.js';
+import { SEARCH_PERSONS } from '@/graphql/search';
+import { IActor } from '@/types/actor';
+
 
 @Component({
   components: { EditorContent, EditorMenuBar, EditorMenuBubble },
@@ -151,9 +175,33 @@ import {
 export default class CreateEvent extends Vue {
   @Prop({ required: true }) value!: String;
   editor: Editor = null;
-  linkUrl: string|null = null;
+
+    /**
+     * Editor Link
+     */
+  linkUrl!: string|null;
   linkMenuIsActive: boolean = false;
 
+    /**
+     * Editor Suggestions
+     */
+  query!: string|null;
+  filteredActors: IActor[] = [];
+  suggestionRange!: object|null;
+  navigatedUserIndex: number = 0;
+  popup!: Instance|null;
+
+  get hasResults() {
+    return this.filteredActors.length;
+  }
+  get showSuggestions() {
+    return this.query || this.hasResults;
+  }
+
+  insertMention: Function = () => {};
+  observer!: MutationObserver|null;
+
+
   mounted() {
     this.editor = new Editor({
       extensions: [
@@ -161,6 +209,77 @@ export default class CreateEvent extends Vue {
         new BulletList(),
         new HardBreak(),
         new Heading({ levels: [1, 2, 3] }),
+        new Mention({
+          items: () => [],
+          onEnter: ({ items, query, range, command, virtualNode }) => {
+            this.query = query;
+            this.filteredActors = items;
+            this.suggestionRange = range;
+            this.renderPopup(virtualNode);
+            // we save the command for inserting a selected mention
+            // this allows us to call it inside of our custom popup
+            // via keyboard navigation and on click
+            this.insertMention = command;
+          },
+          /**
+           * is called when a suggestion has changed
+           */
+          onChange: ({ items, query, range, virtualNode }) => {
+            this.query = query;
+            this.filteredActors = items;
+            this.suggestionRange = range;
+            this.navigatedUserIndex = 0;
+            this.renderPopup(virtualNode);
+          },
+
+          /**
+           * is called when a suggestion is cancelled
+           */
+          onExit: () => {
+            // reset all saved values
+            this.query = null;
+            this.filteredActors = [];
+            this.suggestionRange = null;
+            this.navigatedUserIndex = 0;
+            this.destroyPopup();
+          },
+
+          /**
+           * is called on every keyDown event while a suggestion is active
+           */
+          onKeyDown: ({ event }) => {
+            // pressing up arrow
+            if (event.keyCode === 38) {
+              this.upHandler();
+              return true;
+            }
+            // pressing down arrow
+            if (event.keyCode === 40) {
+              this.downHandler();
+              return true;
+            }
+            // pressing enter
+            if (event.keyCode === 13) {
+              this.enterHandler();
+              return true;
+            }
+            return false;
+          },
+          onFilter: async (items, query) => {
+            if (!query) {
+              return items;
+            }
+            const result = await this.$apollo.query({
+              query: SEARCH_PERSONS,
+              variables: {
+                searchText: query,
+              },
+            });
+            // TODO: TipTap doesn't handle async for onFilter, hence the following line.
+            this.filteredActors = result.data.searchPersons.elements;
+            return this.filteredActors;
+          },
+        }),
         new ListItem(),
         new OrderedList(),
         new TodoItem(),
@@ -172,10 +291,11 @@ export default class CreateEvent extends Vue {
         new Underline(),
         new History(),
         new Placeholder({
-            emptyClass: 'is-empty',
-            emptyNodeText: 'Write something …',
-            showOnlyWhenEditable: false,
-        })
+          emptyClass: 'is-empty',
+          emptyNodeText: 'Write something …',
+          showOnlyWhenEditable: false,
+        }),
+        new Image(),
       ],
       onUpdate: ({ getHTML }) => {
         this.$emit('input', getHTML());
@@ -191,7 +311,7 @@ export default class CreateEvent extends Vue {
     }
   }
 
-  showLinkMenu(attrs) {
+  showLinkMenu(attrs: any) {
     this.linkUrl = attrs.href;
     this.linkMenuIsActive = true;
     this.$nextTick(() => {
@@ -209,6 +329,87 @@ export default class CreateEvent extends Vue {
     this.editor.focus();
   }
 
+  upHandler() {
+    this.navigatedUserIndex = ((this.navigatedUserIndex + this.filteredActors.length) - 1) % this.filteredActors.length;
+  }
+
+  /**
+   * navigate to the next item
+   * if it's the last item, navigate to the first one
+   */
+  downHandler() {
+    this.navigatedUserIndex = (this.navigatedUserIndex + 1) % this.filteredActors.length;
+  }
+
+  enterHandler() {
+    const actor = this.filteredActors[this.navigatedUserIndex];
+    if (actor) {
+      this.selectActor(actor);
+    }
+  }
+
+  /**
+   * we have to replace our suggestion text with a mention
+   * so it's important to pass also the position of your suggestion text
+   * @param actor IActor
+   */
+  selectActor(actor: IActor) {
+    this.insertMention({
+      range: this.suggestionRange,
+      attrs: {
+        id: actor.id,
+        label: actor.name,
+      },
+    });
+    this.editor.focus();
+  }
+
+  /**
+   * renders a popup with suggestions
+   * tiptap provides a virtualNode object for using popper.js (or tippy.js) for popups
+   * @param node
+   */
+  renderPopup(node) {
+    if (this.popup) {
+      return;
+    }
+    this.popup = tippy(node, {
+      content: this.$refs.suggestions as HTMLElement,
+      trigger: 'mouseenter',
+      interactive: true,
+      theme: 'dark',
+      placement: 'top-start',
+      inertia: true,
+      duration: [400, 200],
+      showOnInit: true,
+      arrow: true,
+      arrowType: 'round',
+    }) as Instance;
+        // we have to update tippy whenever the DOM is updated
+    if (MutationObserver) {
+      this.observer = new MutationObserver(() => {
+        if (this.popup != null && this.popup.popperInstance) {
+          this.popup.popperInstance.scheduleUpdate();
+        }
+      });
+      this.observer.observe(this.$refs.suggestions as HTMLElement, {
+        childList: true,
+        subtree: true,
+        characterData: true,
+      });
+    }
+  }
+
+  destroyPopup() {
+    if (this.popup) {
+      this.popup.destroy();
+      this.popup = null;
+    }
+    if (this.observer) {
+      this.observer.disconnect();
+    }
+  }
+
   beforeDestroy() {
     this.editor.destroy();
   }
@@ -272,11 +473,11 @@ export default class CreateEvent extends Vue {
 
         &__content {
             div.ProseMirror {
-                background: #fff;
                 min-height: 10rem;
 
                 &:focus {
                     border-color: #3273dc;
+                    background: #fff;
                     box-shadow: 0 0 0 0.125em rgba(50, 115, 220, 0.25);
                 }
             }
@@ -386,5 +587,68 @@ export default class CreateEvent extends Vue {
             color: $color-white;
         }
     }
+    .mention {
+        background: rgba($color-black, 0.1);
+        color: rgba($color-black, 0.6);
+        font-size: 0.8rem;
+        font-weight: bold;
+        border-radius: 5px;
+        padding: 0.2rem 0.5rem;
+        white-space: nowrap;
+    }
+    .mention-suggestion {
+        color: rgba($color-black, 0.6);
+    }
+    .suggestion-list {
+        padding: 0.2rem;
+        border: 2px solid rgba($color-black, 0.1);
+        font-size: 0.8rem;
+        font-weight: bold;
+        &__no-results {
+            padding: 0.2rem 0.5rem;
+        }
+        &__item {
+            border-radius: 5px;
+            padding: 0.2rem 0.5rem;
+            margin-bottom: 0.2rem;
+            cursor: pointer;
+            &:last-child {
+                margin-bottom: 0;
+            }
+            &.is-selected,
+            &:hover {
+                background-color: rgba($color-white, 0.2);
+            }
+            &.is-empty {
+                opacity: 0.5;
+            }
+        }
+    }
+    .tippy-tooltip.dark-theme {
+        background-color: $color-black;
+        padding: 0;
+        font-size: 1rem;
+        text-align: inherit;
+        color: $color-white;
+        border-radius: 5px;
+        .tippy-backdrop {
+            display: none;
+        }
+        .tippy-roundarrow {
+            fill: $color-black;
+        }
+        .tippy-popper[x-placement^=top] & .tippy-arrow {
+            border-top-color: $color-black;
+        }
+        .tippy-popper[x-placement^=bottom] & .tippy-arrow {
+            border-bottom-color: $color-black;
+        }
+        .tippy-popper[x-placement^=left] & .tippy-arrow {
+            border-left-color: $color-black;
+        }
+        .tippy-popper[x-placement^=right] & .tippy-arrow {
+            border-right-color: $color-black;
+        }
+    }
 
 </style>
diff --git a/js/src/graphql/search.ts b/js/src/graphql/search.ts
index f5af5221b..562a8eab1 100644
--- a/js/src/graphql/search.ts
+++ b/js/src/graphql/search.ts
@@ -34,3 +34,21 @@ query SearchGroups($searchText: String!) {
   }
 }
 `;
+
+export const SEARCH_PERSONS = gql`
+  query SearchPersons($searchText: String!) {
+    searchPersons(search: $searchText) {
+      total,
+      elements {
+        id,
+        avatar {
+          url
+        },
+        domain,
+        preferredUsername,
+        name,
+        __typename
+      }
+    }
+  }
+`;
diff --git a/js/src/types/search.model.ts b/js/src/types/search.model.ts
index 1bd731439..afe27ba7b 100644
--- a/js/src/types/search.model.ts
+++ b/js/src/types/search.model.ts
@@ -1,4 +1,4 @@
-import { IGroup } from '@/types/actor';
+import { IGroup, IPerson } from '@/types/actor';
 import { IEvent } from '@/types/event.model';
 
 export interface SearchEvent {
@@ -10,3 +10,8 @@ export interface SearchGroup {
   total: number;
   elements: IGroup[];
 }
+
+export interface SearchPerson {
+  total: number;
+  elements: IPerson[];
+}
diff --git a/js/src/views/Event/Create.vue b/js/src/views/Event/Create.vue
index 414423877..4e3a8d08f 100644
--- a/js/src/views/Event/Create.vue
+++ b/js/src/views/Event/Create.vue
@@ -142,9 +142,3 @@ export default class CreateEvent extends Vue {
   // }
 }
 </script>
-
-<style>
-.markdown-render h1 {
-  font-size: 2em;
-}
-</style>
diff --git a/js/yarn.lock b/js/yarn.lock
index 41bd08b6b..0cd625ae1 100644
--- a/js/yarn.lock
+++ b/js/yarn.lock
@@ -7738,6 +7738,11 @@ pofile@^1.0.10:
   resolved "https://registry.yarnpkg.com/pofile/-/pofile-1.0.11.tgz#35aff58c17491d127a07336d5522ebc9df57c954"
   integrity sha512-Vy9eH1dRD9wHjYt/QqXcTz+RnX/zg53xK+KljFSX30PvdDMb2z+c6uDUeblUGqqJgz3QFsdlA0IJvHziPmWtQg==
 
+popper.js@^1.14.7:
+  version "1.15.0"
+  resolved "https://registry.yarnpkg.com/popper.js/-/popper.js-1.15.0.tgz#5560b99bbad7647e9faa475c6b8056621f5a4ff2"
+  integrity sha512-w010cY1oCUmI+9KwwlWki+r5jxKfTFDVoadl7MSrIujHU5MJ5OR6HTDj6Xo8aoR/QsA56x8jKjA59qGH4ELtrA==
+
 portfinder@^1.0.20:
   version "1.0.20"
   resolved "https://registry.yarnpkg.com/portfinder/-/portfinder-1.0.20.tgz#bea68632e54b2e13ab7b0c4775e9b41bf270e44a"
@@ -9903,6 +9908,13 @@ timsort@^0.3.0:
   resolved "https://registry.yarnpkg.com/timsort/-/timsort-0.3.0.tgz#405411a8e7e6339fe64db9a234de11dc31e02bd4"
   integrity sha1-QFQRqOfmM5/mTbmiNN4R3DHgK9Q=
 
+tippy.js@^4.3.1:
+  version "4.3.1"
+  resolved "https://registry.yarnpkg.com/tippy.js/-/tippy.js-4.3.1.tgz#ea38fa7e1a2e3448ac35faa5115ccbb8414f50aa"
+  integrity sha512-H09joePakSu6eDSL1wj5LjEWwvpEELyJQlgsts4wVH7223t4DlyzGCaZNDO8/MQAnSuic4JhKpXtgzSYGlobvg==
+  dependencies:
+    popper.js "^1.14.7"
+
 tiptap-commands@^1.10.5:
   version "1.10.5"
   resolved "https://registry.yarnpkg.com/tiptap-commands/-/tiptap-commands-1.10.5.tgz#e897b59debdddcbc20f8289c92f9e39c5d22e19a"