tiptap/examples/Components/Routes/Suggestions/index.vue

315 lines
7.0 KiB
Vue
Raw Normal View History

2018-09-02 02:51:17 +08:00
<template>
<div>
2018-09-29 18:33:18 +08:00
<editor class="editor" :extensions="extensions" ref="editor">
2018-09-02 02:51:17 +08:00
<div class="editor__content" slot="content" slot-scope="props">
<h2>
2018-09-29 05:05:34 +08:00
Suggestions
2018-09-02 02:51:17 +08:00
</h2>
<p>
2018-09-29 05:05:34 +08:00
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.
2018-09-02 02:51:17 +08:00
</p>
</div>
</editor>
2018-09-25 13:43:21 +08:00
2018-09-29 15:28:46 +08:00
<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>
2018-09-29 18:58:40 +08:00
<div v-else class="suggestion-list__item is-empty">
No users found
2018-09-25 14:37:39 +08:00
</div>
2018-09-25 13:43:21 +08:00
</div>
2018-09-29 18:33:18 +08:00
2018-09-02 02:51:17 +08:00
</div>
</template>
<script>
2018-09-25 14:37:39 +08:00
import Fuse from 'fuse.js'
2018-09-29 04:39:26 +08:00
import tippy from 'tippy.js'
2018-09-02 02:51:17 +08:00
import Icon from 'Components/Icon'
import { Editor } from 'tiptap'
import {
HardBreakNode,
HeadingNode,
MentionNode,
2018-09-29 05:05:34 +08:00
CodeMark,
BoldMark,
ItalicMark,
2018-09-02 02:51:17 +08:00
} from 'tiptap-extensions'
export default {
2018-09-29 18:33:18 +08:00
2018-09-02 02:51:17 +08:00
components: {
Editor,
Icon,
},
2018-09-29 18:33:18 +08:00
2018-09-02 02:51:17 +08:00
data() {
return {
extensions: [
new HardBreakNode(),
new HeadingNode({ maxLevel: 3 }),
2018-09-06 04:09:18 +08:00
new MentionNode({
2018-09-29 18:33:18 +08:00
// a list of all suggested items
items: () => [
2018-09-29 15:28:46 +08:00
{ id: 1, name: 'Philipp Kühn' },
{ id: 2, name: 'Hans Pagel' },
{ id: 3, name: 'Kris Siepert' },
{ id: 4, name: 'Justin Schueler' },
2018-09-27 17:35:32 +08:00
],
2018-09-29 18:33:18 +08:00
// is called when a suggestion starts
2018-11-08 23:38:33 +08:00
onEnter: ({
items, query, range, command, virtualNode,
}) => {
2018-09-28 00:53:23 +08:00
this.query = query
this.filteredUsers = items
2018-09-29 18:33:18 +08:00
this.suggestionRange = range
2018-09-29 05:05:34 +08:00
this.renderPopup(virtualNode)
2018-09-29 18:33:18 +08:00
// 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
2018-09-06 04:09:18 +08:00
},
2018-09-29 18:33:18 +08:00
// is called when a suggestion has changed
2018-11-08 23:38:33 +08:00
onChange: ({
items, query, range, virtualNode,
}) => {
2018-09-28 00:53:23 +08:00
this.query = query
this.filteredUsers = items
2018-09-29 18:33:18 +08:00
this.suggestionRange = range
2018-09-29 15:28:46 +08:00
this.navigatedUserIndex = 0
2018-09-29 05:05:34 +08:00
this.renderPopup(virtualNode)
2018-09-06 04:09:18 +08:00
},
2018-09-29 18:33:18 +08:00
// is called when a suggestion is cancelled
2018-09-28 00:53:23 +08:00
onExit: () => {
2018-09-29 18:33:18 +08:00
// reset all saved values
2018-09-25 14:37:39 +08:00
this.query = null
2018-09-28 00:53:23 +08:00
this.filteredUsers = []
2018-09-29 18:33:18 +08:00
this.suggestionRange = null
2018-09-29 15:28:46 +08:00
this.navigatedUserIndex = 0
2018-09-29 05:05:34 +08:00
this.destroyPopup()
2018-09-27 17:35:32 +08:00
},
2018-09-29 18:33:18 +08:00
// is called on every keyDown event while a suggestion is active
2018-09-27 19:06:24 +08:00
onKeyDown: ({ event }) => {
// pressing up arrow
if (event.keyCode === 38) {
this.upHandler()
2018-09-28 00:50:29 +08:00
return true
2018-09-27 19:06:24 +08:00
}
// pressing down arrow
if (event.keyCode === 40) {
this.downHandler()
2018-09-28 00:50:29 +08:00
return true
2018-09-27 19:06:24 +08:00
}
// pressing enter
if (event.keyCode === 13) {
this.enterHandler()
2018-09-28 00:50:29 +08:00
return true
2018-09-27 19:06:24 +08:00
}
2018-09-28 00:50:29 +08:00
return false
2018-09-27 19:06:24 +08:00
},
2018-09-29 18:33:18 +08:00
// 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
2018-09-27 17:35:32 +08:00
onFilter: (items, query) => {
if (!query) {
return items
}
const fuse = new Fuse(items, {
threshold: 0.2,
2018-09-29 18:33:18 +08:00
keys: ['name'],
2018-09-27 17:35:32 +08:00
})
return fuse.search(query)
2018-09-06 04:09:18 +08:00
},
}),
2018-09-29 05:05:34 +08:00
new CodeMark(),
new BoldMark(),
new ItalicMark(),
2018-09-02 02:51:17 +08:00
],
2018-09-25 14:37:39 +08:00
query: null,
2018-09-29 18:33:18 +08:00
suggestionRange: null,
2018-09-27 17:35:32 +08:00
filteredUsers: [],
2018-09-27 19:06:24 +08:00
navigatedUserIndex: 0,
2018-09-28 19:31:01 +08:00
insertMention: () => {},
2018-09-02 02:51:17 +08:00
}
},
2018-09-29 18:33:18 +08:00
2018-09-29 15:28:46 +08:00
computed: {
2018-09-29 18:33:18 +08:00
2018-09-29 15:28:46 +08:00
hasResults() {
return this.filteredUsers.length
},
2018-09-29 18:33:18 +08:00
2018-09-29 15:28:46 +08:00
showSuggestions() {
return this.query || this.hasResults
},
2018-09-29 18:33:18 +08:00
2018-09-29 15:28:46 +08:00
},
2018-09-29 18:33:18 +08:00
2018-09-25 13:43:21 +08:00
methods: {
2018-09-29 18:33:18 +08:00
// navigate to the previous item
// if it's the first item, navigate to the last one
2018-09-27 19:06:24 +08:00
upHandler() {
this.navigatedUserIndex = ((this.navigatedUserIndex + this.filteredUsers.length) - 1) % this.filteredUsers.length
},
2018-09-29 18:33:18 +08:00
// navigate to the next item
// if it's the last item, navigate to the first one
2018-09-27 19:06:24 +08:00
downHandler() {
this.navigatedUserIndex = (this.navigatedUserIndex + 1) % this.filteredUsers.length
},
2018-09-29 18:33:18 +08:00
2018-09-27 19:06:24 +08:00
enterHandler() {
const user = this.filteredUsers[this.navigatedUserIndex]
2018-09-28 19:33:40 +08:00
2018-09-28 19:32:36 +08:00
if (user) {
this.selectUser(user)
}
2018-09-27 19:06:24 +08:00
},
2018-09-29 18:33:18 +08:00
// we have to replace our suggestion text with a mention
// so it's important to pass also the position of your suggestion text
2018-09-25 13:43:21 +08:00
selectUser(user) {
2018-09-28 19:31:01 +08:00
this.insertMention({
2018-09-29 18:34:11 +08:00
range: this.suggestionRange,
2018-09-28 19:31:01 +08:00
attrs: {
id: user.id,
label: user.name,
},
2018-09-25 13:43:21 +08:00
})
this.$refs.editor.focus()
2018-09-25 13:43:21 +08:00
},
2018-09-29 18:33:18 +08:00
// renders a popup with suggestions
// tiptap provides a virtualNode object for using popper.js (or tippy.js) for popups
2018-09-29 05:05:34 +08:00
renderPopup(node) {
if (this.popup) {
2018-09-29 04:39:26 +08:00
return
}
2018-09-29 05:05:34 +08:00
this.popup = tippy(node, {
2018-09-29 04:39:26 +08:00
content: this.$refs.suggestions,
trigger: 'mouseenter',
interactive: true,
theme: 'dark',
placement: 'top-start',
performance: true,
inertia: true,
duration: [400, 200],
showOnInit: true,
arrow: true,
arrowType: 'round',
})
},
2018-09-29 18:33:18 +08:00
2018-09-29 05:05:34 +08:00
destroyPopup() {
if (this.popup) {
this.popup.destroyAll()
this.popup = null
2018-09-29 04:39:26 +08:00
}
},
2018-09-29 18:33:18 +08:00
2018-09-25 13:43:21 +08:00
},
2018-09-02 02:51:17 +08:00
}
2018-09-06 04:09:18 +08:00
</script>
<style lang="scss">
@import "~variables";
2018-09-29 04:39:26 +08:00
@import '~modules/tippy.js/dist/tippy.css';
2018-09-06 04:09:18 +08:00
.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-09-28 19:46:39 +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 {
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;
}
2018-09-27 19:06:24 +08:00
&__item {
2018-09-28 19:46:39 +08:00
border-radius: 5px;
padding: 0.2rem 0.5rem;
2018-09-29 15:40:36 +08:00
margin-bottom: 0.2rem;
cursor: pointer;
2018-09-28 19:46:39 +08:00
2018-09-29 15:40:36 +08:00
&:last-child {
margin-bottom: 0;
}
&.is-selected,
&:hover {
2018-09-29 04:39:26 +08:00
background-color: rgba($color-white, 0.2);
2018-09-27 19:06:24 +08:00
}
2018-09-29 18:58:40 +08:00
&.is-empty {
opacity: 0.5;
}
2018-09-27 19:06:24 +08:00
}
2018-09-25 13:43:21 +08:00
}
2018-09-29 04:39:26 +08:00
.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;
}
}
2018-09-06 04:09:18 +08:00
</style>