|
|
|
<template lang="pug">
|
|
|
|
div(v-intersect.once='onIntersect')
|
|
|
|
v-textarea#discussion-new(
|
|
|
|
outlined
|
|
|
|
flat
|
|
|
|
:placeholder='$t(`common:comments.newPlaceholder`)'
|
|
|
|
auto-grow
|
|
|
|
dense
|
|
|
|
rows='3'
|
|
|
|
hide-details
|
|
|
|
v-model='newcomment'
|
|
|
|
color='blue-grey darken-2'
|
|
|
|
:background-color='$vuetify.theme.dark ? `grey darken-5` : `white`'
|
|
|
|
v-if='permissions.write'
|
|
|
|
)
|
|
|
|
v-row.mt-2(dense, v-if='!isAuthenticated && permissions.write')
|
|
|
|
v-col(cols='12', lg='6')
|
|
|
|
v-text-field(
|
|
|
|
outlined
|
|
|
|
color='blue-grey darken-2'
|
|
|
|
:background-color='$vuetify.theme.dark ? `grey darken-5` : `white`'
|
|
|
|
:placeholder='$t(`common:comments.fieldName`)'
|
|
|
|
hide-details
|
|
|
|
dense
|
|
|
|
autocomplete='name'
|
|
|
|
v-model='guestName'
|
|
|
|
)
|
|
|
|
v-col(cols='12', lg='6')
|
|
|
|
v-text-field(
|
|
|
|
outlined
|
|
|
|
color='blue-grey darken-2'
|
|
|
|
:background-color='$vuetify.theme.dark ? `grey darken-5` : `white`'
|
|
|
|
:placeholder='$t(`common:comments.fieldEmail`)'
|
|
|
|
hide-details
|
|
|
|
type='email'
|
|
|
|
dense
|
|
|
|
autocomplete='email'
|
|
|
|
v-model='guestEmail'
|
|
|
|
)
|
|
|
|
.d-flex.align-center.pt-3(v-if='permissions.write')
|
|
|
|
v-icon.mr-1(color='blue-grey') mdi-language-markdown-outline
|
|
|
|
.caption.blue-grey--text {{$t('common:comments.markdownFormat')}}
|
|
|
|
v-spacer
|
|
|
|
.caption.mr-3(v-if='isAuthenticated')
|
|
|
|
i18next(tag='span', path='common:comments.postingAs')
|
|
|
|
strong(place='name') {{userDisplayName}}
|
|
|
|
v-btn(
|
|
|
|
dark
|
|
|
|
color='blue-grey darken-2'
|
|
|
|
@click='postComment'
|
|
|
|
depressed
|
|
|
|
)
|
|
|
|
v-icon(left) mdi-comment
|
|
|
|
span.text-none {{$t('common:comments.postComment')}}
|
|
|
|
v-divider.mt-3(v-if='permissions.write')
|
|
|
|
.pa-5.d-flex.align-center.justify-center(v-if='isLoading && !hasLoadedOnce')
|
|
|
|
v-progress-circular(
|
|
|
|
indeterminate
|
|
|
|
size='20'
|
|
|
|
width='1'
|
|
|
|
color='blue-grey'
|
|
|
|
)
|
|
|
|
.caption.blue-grey--text.pl-3: em {{$t('common:comments.loading')}}
|
|
|
|
v-timeline(
|
|
|
|
dense
|
|
|
|
v-else-if='comments && comments.length > 0'
|
|
|
|
)
|
|
|
|
v-timeline-item.comments-post(
|
|
|
|
color='pink darken-4'
|
|
|
|
large
|
|
|
|
v-for='cm of comments'
|
|
|
|
:key='`comment-` + cm.id'
|
|
|
|
:id='`comment-post-id-` + cm.id'
|
|
|
|
)
|
|
|
|
template(v-slot:icon)
|
|
|
|
v-avatar(color='blue-grey')
|
|
|
|
//- v-img(src='http://i.pravatar.cc/64')
|
|
|
|
span.white--text.title {{cm.initials}}
|
|
|
|
v-card.elevation-1
|
|
|
|
v-card-text
|
|
|
|
.comments-post-actions(v-if='permissions.manage && !isBusy && commentEditId === 0')
|
|
|
|
v-icon.mr-3(small, @click='editComment(cm)') mdi-pencil
|
|
|
|
v-icon(small, @click='deleteCommentConfirm(cm)') mdi-delete
|
|
|
|
.comments-post-name.caption: strong {{cm.authorName}}
|
|
|
|
.comments-post-date.overline.grey--text {{cm.createdAt | moment('from') }} #[em(v-if='cm.createdAt !== cm.updatedAt') - {{$t('common:comments.modified', { reldate: $options.filters.moment(cm.updatedAt, 'from') })}}]
|
|
|
|
.comments-post-content.mt-3(v-if='commentEditId !== cm.id', v-html='cm.render')
|
|
|
|
.comments-post-editcontent.mt-3(v-else)
|
|
|
|
v-textarea(
|
|
|
|
outlined
|
|
|
|
flat
|
|
|
|
auto-grow
|
|
|
|
dense
|
|
|
|
rows='3'
|
|
|
|
hide-details
|
|
|
|
v-model='commentEditContent'
|
|
|
|
color='blue-grey darken-2'
|
|
|
|
:background-color='$vuetify.theme.dark ? `grey darken-5` : `white`'
|
|
|
|
)
|
|
|
|
.d-flex.align-center.pt-3
|
|
|
|
v-spacer
|
|
|
|
v-btn.mr-3(
|
|
|
|
dark
|
|
|
|
color='blue-grey darken-2'
|
|
|
|
@click='editCommentCancel'
|
|
|
|
outlined
|
|
|
|
)
|
|
|
|
v-icon(left) mdi-close
|
|
|
|
span.text-none {{$t('common:action.cancel')}}
|
|
|
|
v-btn(
|
|
|
|
dark
|
|
|
|
color='blue-grey darken-2'
|
|
|
|
@click='updateComment'
|
|
|
|
depressed
|
|
|
|
)
|
|
|
|
v-icon(left) mdi-comment
|
|
|
|
span.text-none {{$t('common:comments.updateComment')}}
|
|
|
|
.pt-5.text-center.body-2.blue-grey--text(v-else-if='permissions.write') {{$t('common:comments.beFirst')}}
|
|
|
|
.text-center.body-2.blue-grey--text(v-else) {{$t('common:comments.none')}}
|
|
|
|
|
|
|
|
v-dialog(v-model='deleteCommentDialogShown', max-width='500')
|
|
|
|
v-card
|
|
|
|
.dialog-header.is-red {{$t('common:comments.deleteConfirmTitle')}}
|
|
|
|
v-card-text.pt-5
|
|
|
|
span {{$t('common:comments.deleteWarn')}}
|
|
|
|
.caption: strong {{$t('common:comments.deletePermanentWarn')}}
|
|
|
|
v-card-chin
|
|
|
|
v-spacer
|
|
|
|
v-btn(text, @click='deleteCommentDialogShown = false') {{$t('common:actions.cancel')}}
|
|
|
|
v-btn(color='red', dark, @click='deleteComment') {{$t('common:actions.delete')}}
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script>
|
|
|
|
import gql from 'graphql-tag'
|
|
|
|
import { get } from 'vuex-pathify'
|
|
|
|
import validate from 'validate.js'
|
|
|
|
import _ from 'lodash'
|
|
|
|
|
|
|
|
export default {
|
|
|
|
data () {
|
|
|
|
return {
|
|
|
|
newcomment: '',
|
|
|
|
isLoading: true,
|
|
|
|
hasLoadedOnce: false,
|
|
|
|
comments: [],
|
|
|
|
guestName: '',
|
|
|
|
guestEmail: '',
|
|
|
|
commentToDelete: {},
|
|
|
|
commentEditId: 0,
|
|
|
|
commentEditContent: null,
|
|
|
|
deleteCommentDialogShown: false,
|
|
|
|
isBusy: false,
|
|
|
|
scrollOpts: {
|
|
|
|
duration: 1500,
|
|
|
|
offset: 0,
|
|
|
|
easing: 'easeInOutCubic'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
computed: {
|
|
|
|
pageId: get('page/id'),
|
|
|
|
permissions: get('page/effectivePermissions@comments'),
|
|
|
|
isAuthenticated: get('user/authenticated'),
|
|
|
|
userDisplayName: get('user/name')
|
|
|
|
},
|
|
|
|
methods: {
|
|
|
|
onIntersect (entries, observer, isIntersecting) {
|
|
|
|
if (isIntersecting) {
|
|
|
|
this.fetch(true)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
async fetch (silent = false) {
|
|
|
|
this.isLoading = true
|
|
|
|
try {
|
|
|
|
const results = await this.$apollo.query({
|
|
|
|
query: gql`
|
|
|
|
query ($locale: String!, $path: String!) {
|
|
|
|
comments {
|
|
|
|
list(locale: $locale, path: $path) {
|
|
|
|
id
|
|
|
|
render
|
|
|
|
authorName
|
|
|
|
createdAt
|
|
|
|
updatedAt
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
variables: {
|
|
|
|
locale: this.$store.get('page/locale'),
|
|
|
|
path: this.$store.get('page/path')
|
|
|
|
},
|
|
|
|
fetchPolicy: 'network-only'
|
|
|
|
})
|
|
|
|
this.comments = _.get(results, 'data.comments.list', []).map(c => {
|
|
|
|
const nameParts = c.authorName.toUpperCase().split(' ')
|
|
|
|
let initials = _.head(nameParts).charAt(0)
|
|
|
|
if (nameParts.length > 1) {
|
|
|
|
initials += _.last(nameParts).charAt(0)
|
|
|
|
}
|
|
|
|
c.initials = initials
|
|
|
|
return c
|
|
|
|
})
|
|
|
|
} catch (err) {
|
|
|
|
console.warn(err)
|
|
|
|
if (!silent) {
|
|
|
|
this.$store.commit('showNotification', {
|
|
|
|
style: 'red',
|
|
|
|
message: err.message,
|
|
|
|
icon: 'alert'
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.isLoading = false
|
|
|
|
this.hasLoadedOnce = true
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Post New Comment
|
|
|
|
*/
|
|
|
|
async postComment () {
|
|
|
|
let rules = {
|
|
|
|
comment: {
|
|
|
|
presence: {
|
|
|
|
allowEmpty: false
|
|
|
|
},
|
|
|
|
length: {
|
|
|
|
minimum: 2
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!this.isAuthenticated && this.permissions.write) {
|
|
|
|
rules.name = {
|
|
|
|
presence: {
|
|
|
|
allowEmpty: false
|
|
|
|
},
|
|
|
|
length: {
|
|
|
|
minimum: 2,
|
|
|
|
maximum: 255
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rules.email = {
|
|
|
|
presence: {
|
|
|
|
allowEmpty: false
|
|
|
|
},
|
|
|
|
email: true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const validationResults = validate({
|
|
|
|
comment: this.newcomment,
|
|
|
|
name: this.guestName,
|
|
|
|
email: this.guestEmail
|
|
|
|
}, rules, { format: 'flat' })
|
|
|
|
|
|
|
|
if (validationResults) {
|
|
|
|
this.$store.commit('showNotification', {
|
|
|
|
style: 'red',
|
|
|
|
message: validationResults[0],
|
|
|
|
icon: 'alert'
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
const resp = await this.$apollo.mutate({
|
|
|
|
mutation: gql`
|
|
|
|
mutation (
|
|
|
|
$pageId: Int!
|
|
|
|
$replyTo: Int
|
|
|
|
$content: String!
|
|
|
|
$guestName: String
|
|
|
|
$guestEmail: String
|
|
|
|
) {
|
|
|
|
comments {
|
|
|
|
create (
|
|
|
|
pageId: $pageId
|
|
|
|
replyTo: $replyTo
|
|
|
|
content: $content
|
|
|
|
guestName: $guestName
|
|
|
|
guestEmail: $guestEmail
|
|
|
|
) {
|
|
|
|
responseResult {
|
|
|
|
succeeded
|
|
|
|
errorCode
|
|
|
|
slug
|
|
|
|
message
|
|
|
|
}
|
|
|
|
id
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
variables: {
|
|
|
|
pageId: this.pageId,
|
|
|
|
replyTo: 0,
|
|
|
|
content: this.newcomment,
|
|
|
|
guestName: this.guestName,
|
|
|
|
guestEmail: this.guestEmail
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
if (_.get(resp, 'data.comments.create.responseResult.succeeded', false)) {
|
|
|
|
this.$store.commit('showNotification', {
|
|
|
|
style: 'success',
|
|
|
|
message: this.$t('common:comments.postSuccess'),
|
|
|
|
icon: 'check'
|
|
|
|
})
|
|
|
|
|
|
|
|
this.newcomment = ''
|
|
|
|
await this.fetch()
|
|
|
|
this.$nextTick(() => {
|
|
|
|
this.$vuetify.goTo(`#comment-post-id-${_.get(resp, 'data.comments.create.id', 0)}`, this.scrollOpts)
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
throw new Error(_.get(resp, 'data.comments.create.responseResult.message', 'An unexpected error occured.'))
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
this.$store.commit('showNotification', {
|
|
|
|
style: 'red',
|
|
|
|
message: err.message,
|
|
|
|
icon: 'alert'
|
|
|
|
})
|
|
|
|
}
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Show Comment Editing Form
|
|
|
|
*/
|
|
|
|
async editComment (cm) {
|
|
|
|
this.$store.commit(`loadingStart`, 'comments-edit')
|
|
|
|
this.isBusy = true
|
|
|
|
try {
|
|
|
|
const results = await this.$apollo.query({
|
|
|
|
query: gql`
|
|
|
|
query ($id: Int!) {
|
|
|
|
comments {
|
|
|
|
single(id: $id) {
|
|
|
|
content
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
variables: {
|
|
|
|
id: cm.id
|
|
|
|
},
|
|
|
|
fetchPolicy: 'network-only'
|
|
|
|
})
|
|
|
|
this.commentEditContent = _.get(results, 'data.comments.single.content', null)
|
|
|
|
if (this.commentEditContent === null) {
|
|
|
|
throw new Error('Failed to load comment content.')
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
console.warn(err)
|
|
|
|
this.$store.commit('showNotification', {
|
|
|
|
style: 'red',
|
|
|
|
message: err.message,
|
|
|
|
icon: 'alert'
|
|
|
|
})
|
|
|
|
}
|
|
|
|
this.commentEditId = cm.id
|
|
|
|
this.isBusy = false
|
|
|
|
this.$store.commit(`loadingStop`, 'comments-edit')
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Cancel Comment Edit
|
|
|
|
*/
|
|
|
|
editCommentCancel () {
|
|
|
|
this.commentEditId = 0
|
|
|
|
this.commentEditContent = null
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Update Comment with new content
|
|
|
|
*/
|
|
|
|
async updateComment () {
|
|
|
|
this.$store.commit(`loadingStart`, 'comments-edit')
|
|
|
|
this.isBusy = true
|
|
|
|
try {
|
|
|
|
if (this.commentEditContent.length < 2) {
|
|
|
|
throw new Error(this.$t('common:comments.contentMissingError'))
|
|
|
|
}
|
|
|
|
const resp = await this.$apollo.mutate({
|
|
|
|
mutation: gql`
|
|
|
|
mutation (
|
|
|
|
$id: Int!
|
|
|
|
$content: String!
|
|
|
|
) {
|
|
|
|
comments {
|
|
|
|
update (
|
|
|
|
id: $id,
|
|
|
|
content: $content
|
|
|
|
) {
|
|
|
|
responseResult {
|
|
|
|
succeeded
|
|
|
|
errorCode
|
|
|
|
slug
|
|
|
|
message
|
|
|
|
}
|
|
|
|
render
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
variables: {
|
|
|
|
id: this.commentEditId,
|
|
|
|
content: this.commentEditContent
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
if (_.get(resp, 'data.comments.update.responseResult.succeeded', false)) {
|
|
|
|
this.$store.commit('showNotification', {
|
|
|
|
style: 'success',
|
|
|
|
message: this.$t('common:comments.updateSuccess'),
|
|
|
|
icon: 'check'
|
|
|
|
})
|
|
|
|
|
|
|
|
const cm = _.find(this.comments, ['id', this.commentEditId])
|
|
|
|
cm.render = _.get(resp, 'data.comments.update.render', '-- Failed to load updated comment --')
|
|
|
|
cm.updatedAt = (new Date()).toISOString()
|
|
|
|
|
|
|
|
this.editCommentCancel()
|
|
|
|
} else {
|
|
|
|
throw new Error(_.get(resp, 'data.comments.delete.responseResult.message', 'An unexpected error occured.'))
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
console.warn(err)
|
|
|
|
this.$store.commit('showNotification', {
|
|
|
|
style: 'red',
|
|
|
|
message: err.message,
|
|
|
|
icon: 'alert'
|
|
|
|
})
|
|
|
|
}
|
|
|
|
this.isBusy = false
|
|
|
|
this.$store.commit(`loadingStop`, 'comments-edit')
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Show Delete Comment Confirmation Dialog
|
|
|
|
*/
|
|
|
|
deleteCommentConfirm (cm) {
|
|
|
|
this.commentToDelete = cm
|
|
|
|
this.deleteCommentDialogShown = true
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* Delete Comment
|
|
|
|
*/
|
|
|
|
async deleteComment () {
|
|
|
|
this.$store.commit(`loadingStart`, 'comments-delete')
|
|
|
|
this.isBusy = true
|
|
|
|
this.deleteCommentDialogShown = false
|
|
|
|
|
|
|
|
try {
|
|
|
|
const resp = await this.$apollo.mutate({
|
|
|
|
mutation: gql`
|
|
|
|
mutation (
|
|
|
|
$id: Int!
|
|
|
|
) {
|
|
|
|
comments {
|
|
|
|
delete (
|
|
|
|
id: $id
|
|
|
|
) {
|
|
|
|
responseResult {
|
|
|
|
succeeded
|
|
|
|
errorCode
|
|
|
|
slug
|
|
|
|
message
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`,
|
|
|
|
variables: {
|
|
|
|
id: this.commentToDelete.id
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
if (_.get(resp, 'data.comments.delete.responseResult.succeeded', false)) {
|
|
|
|
this.$store.commit('showNotification', {
|
|
|
|
style: 'success',
|
|
|
|
message: this.$t('common:comments.deleteSuccess'),
|
|
|
|
icon: 'check'
|
|
|
|
})
|
|
|
|
|
|
|
|
this.comments = _.reject(this.comments, ['id', this.commentToDelete.id])
|
|
|
|
} else {
|
|
|
|
throw new Error(_.get(resp, 'data.comments.delete.responseResult.message', 'An unexpected error occured.'))
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
this.$store.commit('showNotification', {
|
|
|
|
style: 'red',
|
|
|
|
message: err.message,
|
|
|
|
icon: 'alert'
|
|
|
|
})
|
|
|
|
}
|
|
|
|
this.isBusy = false
|
|
|
|
this.$store.commit(`loadingStop`, 'comments-delete')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style lang="scss">
|
|
|
|
.comments-post {
|
|
|
|
position: relative;
|
|
|
|
|
|
|
|
&:hover {
|
|
|
|
.comments-post-actions {
|
|
|
|
opacity: 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
&-actions {
|
|
|
|
position: absolute;
|
|
|
|
top: 16px;
|
|
|
|
right: 16px;
|
|
|
|
opacity: 0;
|
|
|
|
transition: opacity .4s ease;
|
|
|
|
}
|
|
|
|
|
|
|
|
&-content {
|
|
|
|
> p:first-child {
|
|
|
|
padding-top: 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
p {
|
|
|
|
padding-top: 1rem;
|
|
|
|
margin-bottom: 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
img {
|
|
|
|
max-width: 100%;
|
|
|
|
border-radius: 5px;
|
|
|
|
}
|
|
|
|
|
|
|
|
code {
|
|
|
|
background-color: rgba(mc('pink', '500'), .1);
|
|
|
|
box-shadow: none;
|
|
|
|
}
|
|
|
|
|
|
|
|
pre > code {
|
|
|
|
margin-top: 1rem;
|
|
|
|
padding: 12px;
|
|
|
|
background-color: #111;
|
|
|
|
box-shadow: none;
|
|
|
|
border-radius: 5px;
|
|
|
|
width: 100%;
|
|
|
|
color: #FFF;
|
|
|
|
font-weight: 400;
|
|
|
|
font-size: .85rem;
|
|
|
|
font-family: Roboto Mono, monospace;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
</style>
|