2018-09-02 02:51:17 +08:00
|
|
|
|
<template>
|
2018-11-09 05:03:10 +08:00
|
|
|
|
<div>
|
|
|
|
|
|
|
|
|
|
<div class="editor">
|
2019-05-08 05:23:53 +08:00
|
|
|
|
<editor-menu-bar :editor="editor" v-slot="{ commands }">
|
|
|
|
|
<div class="menubar">
|
2018-11-23 05:55:07 +08:00
|
|
|
|
<button class="menubar__button" @click="commands.mention({ id: 1, label: 'Philipp Kühn' })">
|
|
|
|
|
<icon name="mention" />
|
|
|
|
|
<span>Insert Mention</span>
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
</editor-menu-bar>
|
2018-11-09 05:03:10 +08:00
|
|
|
|
<editor-content class="editor__content" :editor="editor" />
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<div class="suggestion-list" v-show="showSuggestions" ref="suggestions">
|
|
|
|
|
<template v-if="hasResults">
|
|
|
|
|
<div
|
|
|
|
|
v-for="(user, index) in filteredUsers"
|
|
|
|
|
:key="user.id"
|
|
|
|
|
class="suggestion-list__item"
|
|
|
|
|
:class="{ 'is-selected': navigatedUserIndex === index }"
|
|
|
|
|
@click="selectUser(user)"
|
|
|
|
|
>
|
|
|
|
|
{{ user.name }}
|
|
|
|
|
</div>
|
|
|
|
|
</template>
|
|
|
|
|
<div v-else class="suggestion-list__item is-empty">
|
|
|
|
|
No users found
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
</div>
|
2018-09-02 02:51:17 +08:00
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<script>
|
2018-09-25 14:37:39 +08:00
|
|
|
|
import Fuse from 'fuse.js'
|
2020-04-05 20:13:37 +08:00
|
|
|
|
import tippy, { roundArrow, sticky } from 'tippy.js'
|
2018-11-23 05:55:07 +08:00
|
|
|
|
import Icon from 'Components/Icon'
|
|
|
|
|
import { Editor, EditorContent, EditorMenuBar } from 'tiptap'
|
2018-09-02 02:51:17 +08:00
|
|
|
|
import {
|
2018-11-09 05:03:10 +08:00
|
|
|
|
HardBreak,
|
|
|
|
|
Heading,
|
|
|
|
|
Mention,
|
|
|
|
|
Code,
|
|
|
|
|
Bold,
|
|
|
|
|
Italic,
|
2018-09-02 02:51:17 +08:00
|
|
|
|
} from 'tiptap-extensions'
|
2020-03-29 03:18:14 +08:00
|
|
|
|
import 'tippy.js/dist/svg-arrow.css'
|
2018-09-02 02:51:17 +08:00
|
|
|
|
|
|
|
|
|
export default {
|
2018-09-29 18:33:18 +08:00
|
|
|
|
|
2018-11-09 05:03:10 +08:00
|
|
|
|
components: {
|
2018-11-23 05:55:07 +08:00
|
|
|
|
Icon,
|
2018-11-09 05:03:10 +08:00
|
|
|
|
EditorContent,
|
2018-11-23 05:55:07 +08:00
|
|
|
|
EditorMenuBar,
|
2018-11-09 05:03:10 +08:00
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
data() {
|
|
|
|
|
return {
|
|
|
|
|
editor: new Editor({
|
|
|
|
|
extensions: [
|
|
|
|
|
new HardBreak(),
|
|
|
|
|
new Heading({ levels: [1, 2, 3] }),
|
|
|
|
|
new Mention({
|
|
|
|
|
// a list of all suggested items
|
|
|
|
|
items: () => [
|
|
|
|
|
{ id: 1, name: 'Philipp Kühn' },
|
|
|
|
|
{ id: 2, name: 'Hans Pagel' },
|
|
|
|
|
{ id: 3, name: 'Kris Siepert' },
|
|
|
|
|
{ id: 4, name: 'Justin Schueler' },
|
|
|
|
|
],
|
|
|
|
|
// is called when a suggestion starts
|
|
|
|
|
onEnter: ({
|
|
|
|
|
items, query, range, command, virtualNode,
|
|
|
|
|
}) => {
|
|
|
|
|
this.query = query
|
|
|
|
|
this.filteredUsers = 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.filteredUsers = 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.filteredUsers = []
|
|
|
|
|
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
|
|
|
|
|
},
|
|
|
|
|
// is called when a suggestion has changed
|
|
|
|
|
// this function is optional because there is basic filtering built-in
|
|
|
|
|
// you can overwrite it if you prefer your own filtering
|
|
|
|
|
// in this example we use fuse.js with support for fuzzy search
|
|
|
|
|
onFilter: (items, query) => {
|
|
|
|
|
if (!query) {
|
|
|
|
|
return items
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const fuse = new Fuse(items, {
|
|
|
|
|
threshold: 0.2,
|
|
|
|
|
keys: ['name'],
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
return fuse.search(query)
|
|
|
|
|
},
|
|
|
|
|
}),
|
|
|
|
|
new Code(),
|
|
|
|
|
new Bold(),
|
|
|
|
|
new Italic(),
|
|
|
|
|
],
|
|
|
|
|
content: `
|
|
|
|
|
<h2>
|
|
|
|
|
Suggestions
|
|
|
|
|
</h2>
|
|
|
|
|
<p>
|
|
|
|
|
Sometimes it's useful to <strong>mention</strong> someone. That's a feature we're very used to. Under the hood this technique can also be used for other features likes <strong>hashtags</strong> and <strong>commands</strong> – lets call it <em>suggestions</em>.
|
|
|
|
|
</p>
|
|
|
|
|
<p>
|
|
|
|
|
This is an example how to mention some users like <span data-mention-id="1">Philipp Kühn</span> or <span data-mention-id="2">Hans Pagel</span>. Try to type <code>@</code> and a popup (rendered with tippy.js) will appear. You can navigate with arrow keys through a list of suggestions.
|
|
|
|
|
</p>
|
|
|
|
|
`,
|
|
|
|
|
}),
|
|
|
|
|
query: null,
|
|
|
|
|
suggestionRange: null,
|
|
|
|
|
filteredUsers: [],
|
|
|
|
|
navigatedUserIndex: 0,
|
|
|
|
|
insertMention: () => {},
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
computed: {
|
|
|
|
|
|
|
|
|
|
hasResults() {
|
|
|
|
|
return this.filteredUsers.length
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
showSuggestions() {
|
|
|
|
|
return this.query || this.hasResults
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
methods: {
|
|
|
|
|
|
|
|
|
|
// navigate to the previous item
|
|
|
|
|
// if it's the first item, navigate to the last one
|
|
|
|
|
upHandler() {
|
|
|
|
|
this.navigatedUserIndex = ((this.navigatedUserIndex + this.filteredUsers.length) - 1) % this.filteredUsers.length
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
// navigate to the next item
|
|
|
|
|
// if it's the last item, navigate to the first one
|
|
|
|
|
downHandler() {
|
|
|
|
|
this.navigatedUserIndex = (this.navigatedUserIndex + 1) % this.filteredUsers.length
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
enterHandler() {
|
|
|
|
|
const user = this.filteredUsers[this.navigatedUserIndex]
|
|
|
|
|
|
|
|
|
|
if (user) {
|
|
|
|
|
this.selectUser(user)
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
// we have to replace our suggestion text with a mention
|
|
|
|
|
// so it's important to pass also the position of your suggestion text
|
|
|
|
|
selectUser(user) {
|
|
|
|
|
this.insertMention({
|
|
|
|
|
range: this.suggestionRange,
|
|
|
|
|
attrs: {
|
|
|
|
|
id: user.id,
|
|
|
|
|
label: user.name,
|
|
|
|
|
},
|
|
|
|
|
})
|
|
|
|
|
this.editor.focus()
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
// renders a popup with suggestions
|
|
|
|
|
// tiptap provides a virtualNode object for using popper.js (or tippy.js) for popups
|
|
|
|
|
renderPopup(node) {
|
|
|
|
|
if (this.popup) {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-29 03:18:14 +08:00
|
|
|
|
// ref: https://atomiks.github.io/tippyjs/v6/all-props/
|
|
|
|
|
this.popup = tippy('.page', {
|
|
|
|
|
getReferenceClientRect: node.getBoundingClientRect,
|
|
|
|
|
appendTo: () => document.body,
|
2018-11-09 05:03:10 +08:00
|
|
|
|
interactive: true,
|
2020-03-29 03:18:14 +08:00
|
|
|
|
arrow: roundArrow,
|
2020-04-05 20:13:37 +08:00
|
|
|
|
sticky: true, // make sure position of tippy is updated when content changes
|
|
|
|
|
plugins: [sticky],
|
2020-03-29 03:18:14 +08:00
|
|
|
|
content: this.$refs.suggestions,
|
|
|
|
|
trigger: 'mouseenter', // manual
|
|
|
|
|
showOnCreate: true,
|
2018-11-09 05:03:10 +08:00
|
|
|
|
theme: 'dark',
|
|
|
|
|
placement: 'top-start',
|
|
|
|
|
inertia: true,
|
|
|
|
|
duration: [400, 200],
|
|
|
|
|
})
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
destroyPopup() {
|
|
|
|
|
if (this.popup) {
|
2020-03-29 03:18:14 +08:00
|
|
|
|
this.popup[0].destroy()
|
2018-11-09 05:03:10 +08:00
|
|
|
|
this.popup = null
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
},
|
2018-09-02 02:51:17 +08:00
|
|
|
|
}
|
2018-09-06 04:09:18 +08:00
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
<style lang="scss">
|
|
|
|
|
@import "~variables";
|
|
|
|
|
|
|
|
|
|
.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;
|
2018-11-09 05:03:10 +08:00
|
|
|
|
white-space: nowrap;
|
2018-09-06 04:09:18 +08:00
|
|
|
|
}
|
2018-09-25 13:43:21 +08:00
|
|
|
|
|
2018-09-28 19:46:39 +08:00
|
|
|
|
.mention-suggestion {
|
2018-11-09 05:03:10 +08:00
|
|
|
|
color: rgba($color-black, 0.6);
|
2018-09-28 19:46:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
.suggestion-list {
|
2018-11-09 05:03:10 +08:00
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-09-25 13:43:21 +08:00
|
|
|
|
}
|
2018-09-29 04:39:26 +08:00
|
|
|
|
|
2020-03-29 03:18:14 +08:00
|
|
|
|
.tippy-box[data-theme~=dark] {
|
2018-11-09 05:03:10 +08:00
|
|
|
|
background-color: $color-black;
|
|
|
|
|
padding: 0;
|
|
|
|
|
font-size: 1rem;
|
|
|
|
|
text-align: inherit;
|
|
|
|
|
color: $color-white;
|
|
|
|
|
border-radius: 5px;
|
|
|
|
|
|
2020-03-29 03:18:14 +08:00
|
|
|
|
&[data-placement^=top]>.tippy-arrow:before {
|
|
|
|
|
border-top-color: $color-black;
|
2018-11-09 05:03:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-03-29 03:18:14 +08:00
|
|
|
|
&[data-placement^=bottom]>.tippy-arrow:before {
|
|
|
|
|
border-bottom-color: $color-black;
|
2018-11-09 05:03:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-03-29 03:18:14 +08:00
|
|
|
|
&[data-placement^=left]>.tippy-arrow:before {
|
|
|
|
|
border-left-color: $color-black;
|
2018-11-09 05:03:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-03-29 03:18:14 +08:00
|
|
|
|
&[data-placement^=right]>.tippy-arrow:before {
|
|
|
|
|
border-right-color: $color-black;
|
2018-11-09 05:03:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-03-29 03:18:14 +08:00
|
|
|
|
&>.tippy-backdrop {
|
|
|
|
|
background-color: $color-black;
|
2018-11-09 05:03:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-03-29 03:18:14 +08:00
|
|
|
|
&>.tippy-svg-arrow {
|
|
|
|
|
fill: $color-black;
|
2018-11-09 05:03:10 +08:00
|
|
|
|
}
|
2018-09-29 04:39:26 +08:00
|
|
|
|
}
|
2018-09-06 04:09:18 +08:00
|
|
|
|
</style>
|