feat: delete a page
This commit is contained in:
parent
fc764b7e4c
commit
516220bdda
@ -38,6 +38,7 @@ import historyComponent from './components/history.vue'
|
|||||||
import loadingSpinnerComponent from './components/loading-spinner.vue'
|
import loadingSpinnerComponent from './components/loading-spinner.vue'
|
||||||
import modalCreatePageComponent from './components/modal-create-page.vue'
|
import modalCreatePageComponent from './components/modal-create-page.vue'
|
||||||
import modalCreateUserComponent from './components/modal-create-user.vue'
|
import modalCreateUserComponent from './components/modal-create-user.vue'
|
||||||
|
import modalDeletePageComponent from './components/modal-delete-page.vue'
|
||||||
import modalDeleteUserComponent from './components/modal-delete-user.vue'
|
import modalDeleteUserComponent from './components/modal-delete-user.vue'
|
||||||
import modalDiscardPageComponent from './components/modal-discard-page.vue'
|
import modalDiscardPageComponent from './components/modal-discard-page.vue'
|
||||||
import modalMovePageComponent from './components/modal-move-page.vue'
|
import modalMovePageComponent from './components/modal-move-page.vue'
|
||||||
@ -86,6 +87,7 @@ Vue.component('history', historyComponent)
|
|||||||
Vue.component('loadingSpinner', loadingSpinnerComponent)
|
Vue.component('loadingSpinner', loadingSpinnerComponent)
|
||||||
Vue.component('modalCreatePage', modalCreatePageComponent)
|
Vue.component('modalCreatePage', modalCreatePageComponent)
|
||||||
Vue.component('modalCreateUser', modalCreateUserComponent)
|
Vue.component('modalCreateUser', modalCreateUserComponent)
|
||||||
|
Vue.component('modalDeletePage', modalDeletePageComponent)
|
||||||
Vue.component('modalDeleteUser', modalDeleteUserComponent)
|
Vue.component('modalDeleteUser', modalDeleteUserComponent)
|
||||||
Vue.component('modalDiscardPage', modalDiscardPageComponent)
|
Vue.component('modalDiscardPage', modalDiscardPageComponent)
|
||||||
Vue.component('modalMovePage', modalMovePageComponent)
|
Vue.component('modalMovePage', modalMovePageComponent)
|
||||||
|
66
client/js/components/modal-delete-page.vue
Normal file
66
client/js/components/modal-delete-page.vue
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
<template lang="pug">
|
||||||
|
transition(:duration="400")
|
||||||
|
.modal(v-show='isShown', v-cloak)
|
||||||
|
transition(name='modal-background')
|
||||||
|
.modal-background(v-show='isShown')
|
||||||
|
.modal-container
|
||||||
|
transition(name='modal-content')
|
||||||
|
.modal-content(v-show='isShown')
|
||||||
|
header.is-red
|
||||||
|
span {{ $t('modal.deletepagetitle') }}
|
||||||
|
p.modal-notify(v-bind:class='{ "is-active": isLoading }'): i
|
||||||
|
section
|
||||||
|
span {{ $t('modal.deletepagewarning') }}
|
||||||
|
footer
|
||||||
|
a.button.is-grey.is-outlined(v-on:click='discard') {{ $t('modal.discard') }}
|
||||||
|
a.button.is-red(v-on:click='deletePage') {{ $t('modal.delete') }}
|
||||||
|
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
name: 'modal-delete-page',
|
||||||
|
props: ['currentPath'],
|
||||||
|
data () {
|
||||||
|
return {
|
||||||
|
isLoading: false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
isShown () {
|
||||||
|
return this.$store.state.modalDeletePage.shown
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
discard () {
|
||||||
|
this.isLoading = false
|
||||||
|
this.$store.dispatch('modalDeletePage/close')
|
||||||
|
},
|
||||||
|
deletePage () {
|
||||||
|
let self = this
|
||||||
|
this.isLoading = true
|
||||||
|
this.$http.delete(window.location.href).then(resp => {
|
||||||
|
return resp.json()
|
||||||
|
}).then(resp => {
|
||||||
|
if (resp.ok) {
|
||||||
|
window.location.assign('/')
|
||||||
|
} else {
|
||||||
|
self.isLoading = false
|
||||||
|
self.$store.dispatch('alert', {
|
||||||
|
style: 'red',
|
||||||
|
icon: 'ui-2_square-remove-09',
|
||||||
|
msg: resp.msg
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}).catch(err => {
|
||||||
|
self.isLoading = false
|
||||||
|
self.$store.dispatch('alert', {
|
||||||
|
style: 'red',
|
||||||
|
icon: 'ui-2_square-remove-09',
|
||||||
|
msg: 'Error: ' + err.body.msg
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
@ -10,6 +10,7 @@ import editorVideo from './modules/editor-video'
|
|||||||
import modalCreatePage from './modules/modal-create-page'
|
import modalCreatePage from './modules/modal-create-page'
|
||||||
import modalCreateUser from './modules/modal-create-user'
|
import modalCreateUser from './modules/modal-create-user'
|
||||||
import modalDeleteUser from './modules/modal-delete-user'
|
import modalDeleteUser from './modules/modal-delete-user'
|
||||||
|
import modalDeletePage from './modules/modal-delete-page'
|
||||||
import modalDiscardPage from './modules/modal-discard-page'
|
import modalDiscardPage from './modules/modal-discard-page'
|
||||||
import modalMovePage from './modules/modal-move-page'
|
import modalMovePage from './modules/modal-move-page'
|
||||||
import modalProfile2fa from './modules/modal-profile-2fa'
|
import modalProfile2fa from './modules/modal-profile-2fa'
|
||||||
@ -39,6 +40,7 @@ export default new Vuex.Store({
|
|||||||
editorVideo,
|
editorVideo,
|
||||||
modalCreatePage,
|
modalCreatePage,
|
||||||
modalCreateUser,
|
modalCreateUser,
|
||||||
|
modalDeletePage,
|
||||||
modalDeleteUser,
|
modalDeleteUser,
|
||||||
modalDiscardPage,
|
modalDiscardPage,
|
||||||
modalMovePage,
|
modalMovePage,
|
||||||
|
14
client/js/store/modules/modal-delete-page.js
Normal file
14
client/js/store/modules/modal-delete-page.js
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
export default {
|
||||||
|
namespaced: true,
|
||||||
|
state: {
|
||||||
|
shown: false
|
||||||
|
},
|
||||||
|
getters: {},
|
||||||
|
mutations: {
|
||||||
|
shownChange: (state, shownState) => { state.shown = shownState }
|
||||||
|
},
|
||||||
|
actions: {
|
||||||
|
open({ commit }) { commit('shownChange', true) },
|
||||||
|
close({ commit }) { commit('shownChange', false) }
|
||||||
|
}
|
||||||
|
}
|
@ -173,7 +173,11 @@
|
|||||||
&.is-outlined {
|
&.is-outlined {
|
||||||
background-color: mc($primary, '500');
|
background-color: mc($primary, '500');
|
||||||
color: mc($primary, '100');
|
color: mc($primary, '100');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&.is-icon-only i {
|
||||||
|
margin-right: 0;
|
||||||
|
}
|
||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
background-color: mc($primary, '700');
|
background-color: mc($primary, '700');
|
||||||
|
@ -288,4 +288,29 @@ router.put('/*', (req, res, next) => {
|
|||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete document
|
||||||
|
*/
|
||||||
|
router.delete('/*', (req, res, next) => {
|
||||||
|
if (!res.locals.rights.write) {
|
||||||
|
return res.json({
|
||||||
|
ok: false,
|
||||||
|
error: lang.t('errors:forbidden')
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
let safePath = entryHelper.parsePath(req.path)
|
||||||
|
|
||||||
|
entries.remove(safePath, req.user).then(() => {
|
||||||
|
res.json({
|
||||||
|
ok: true
|
||||||
|
})
|
||||||
|
}).catch((err) => {
|
||||||
|
res.json({
|
||||||
|
ok: false,
|
||||||
|
error: err.message
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
module.exports = router
|
module.exports = router
|
||||||
|
@ -388,6 +388,32 @@ module.exports = {
|
|||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete a document
|
||||||
|
*
|
||||||
|
* @param {String} entryPath The current entry path
|
||||||
|
* @param {Object} author The author user object
|
||||||
|
* @return {Promise} Promise of the operation
|
||||||
|
*/
|
||||||
|
remove(entryPath, author) {
|
||||||
|
if (_.isEmpty(entryPath) || entryPath === 'home') {
|
||||||
|
return Promise.reject(new Error(lang.t('errors:invalidpath')))
|
||||||
|
}
|
||||||
|
|
||||||
|
return git.deleteDocument(entryPath, author).then(() => {
|
||||||
|
// Delete old cache version
|
||||||
|
|
||||||
|
let oldEntryCachePath = entryHelper.getCachePath(entryPath)
|
||||||
|
fs.unlinkAsync(oldEntryCachePath).catch((err) => { return true }) // eslint-disable-line handle-callback-err
|
||||||
|
|
||||||
|
// Delete old index entry
|
||||||
|
search.delete(entryPath)
|
||||||
|
|
||||||
|
// Delete entry
|
||||||
|
return db.Entry.deleteOne({ _id: entryPath })
|
||||||
|
})
|
||||||
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate a starter page content based on the entry path
|
* Generate a starter page content based on the entry path
|
||||||
*
|
*
|
||||||
|
@ -245,6 +245,29 @@ module.exports = {
|
|||||||
})
|
})
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete a document.
|
||||||
|
*
|
||||||
|
* @param {String} entryPath The entry path
|
||||||
|
* @return {Promise<Boolean>} Resolve on success
|
||||||
|
*/
|
||||||
|
deleteDocument(entryPath, author) {
|
||||||
|
let self = this
|
||||||
|
let gitFilePath = entryPath + '.md'
|
||||||
|
|
||||||
|
return this._git.exec('rm', [gitFilePath]).then((cProc) => {
|
||||||
|
let out = cProc.stdout.toString()
|
||||||
|
if (_.includes(out, 'fatal')) {
|
||||||
|
let errorMsg = _.capitalize(_.head(_.split(_.replace(out, 'fatal: ', ''), ',')))
|
||||||
|
throw new Error(errorMsg)
|
||||||
|
}
|
||||||
|
let commitUsr = securityHelper.sanitizeCommitUser(author)
|
||||||
|
return self._git.exec('commit', ['-m', lang.t('git:deleted', { path: gitFilePath }), '--author="' + commitUsr.name + ' <' + commitUsr.email + '>"']).catch((err) => {
|
||||||
|
if (_.includes(err.stdout, 'nothing to commit')) { return true }
|
||||||
|
})
|
||||||
|
})
|
||||||
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Commits uploads changes.
|
* Commits uploads changes.
|
||||||
*
|
*
|
||||||
|
@ -77,6 +77,8 @@
|
|||||||
"delete": "Delete",
|
"delete": "Delete",
|
||||||
"deletefiletitle": "Delete?",
|
"deletefiletitle": "Delete?",
|
||||||
"deletefilewarn": "Are you sure you want to delete",
|
"deletefilewarn": "Are you sure you want to delete",
|
||||||
|
"deletepagewarning": "Are you sure you want to delete this page? This action cannot be undone!",
|
||||||
|
"deletepagetitle": "Delete this page?",
|
||||||
"deleteusertitle": "Delete User Account?",
|
"deleteusertitle": "Delete User Account?",
|
||||||
"deleteuserwarning": "Are you sure you want to delete this user account? This action cannot be undone!",
|
"deleteuserwarning": "Are you sure you want to delete this user account? This action cannot be undone!",
|
||||||
"discard": "Discard",
|
"discard": "Discard",
|
||||||
@ -113,4 +115,4 @@
|
|||||||
"placeholder": "Search...",
|
"placeholder": "Search...",
|
||||||
"results": "Search Results"
|
"results": "Search Results"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -12,6 +12,8 @@ block rootNavRight
|
|||||||
loading-spinner
|
loading-spinner
|
||||||
.nav-item
|
.nav-item
|
||||||
if rights.write && pageData.meta.path !== 'home'
|
if rights.write && pageData.meta.path !== 'home'
|
||||||
|
a.button.is-outlined.is-icon-only(@click='$store.dispatch("modalDeletePage/open")')
|
||||||
|
i.nc-icon-outline.ui-1_trash
|
||||||
a.button.is-outlined(v-on:click='$store.dispatch("modalMovePage/open")')
|
a.button.is-outlined(v-on:click='$store.dispatch("modalMovePage/open")')
|
||||||
i.nc-icon-outline.arrows-1_shuffle-98
|
i.nc-icon-outline.arrows-1_shuffle-98
|
||||||
span= t('nav.move')
|
span= t('nav.move')
|
||||||
@ -83,4 +85,5 @@ block content
|
|||||||
|
|
||||||
modal-create-page(basepath=pageData.meta.path)
|
modal-create-page(basepath=pageData.meta.path)
|
||||||
modal-move-page(current-path=pageData.meta.path)
|
modal-move-page(current-path=pageData.meta.path)
|
||||||
|
modal-delete-page(current-path=pageData.meta.path)
|
||||||
anchor
|
anchor
|
||||||
|
@ -5137,7 +5137,7 @@ onetime@^2.0.0:
|
|||||||
dependencies:
|
dependencies:
|
||||||
mimic-fn "^1.0.0"
|
mimic-fn "^1.0.0"
|
||||||
|
|
||||||
opencollective@^1.0.3:
|
opencollective@^1.0.3, opencollective@~1.0.3:
|
||||||
version "1.0.3"
|
version "1.0.3"
|
||||||
resolved "https://registry.yarnpkg.com/opencollective/-/opencollective-1.0.3.tgz#aee6372bc28144583690c3ca8daecfc120dd0ef1"
|
resolved "https://registry.yarnpkg.com/opencollective/-/opencollective-1.0.3.tgz#aee6372bc28144583690c3ca8daecfc120dd0ef1"
|
||||||
dependencies:
|
dependencies:
|
||||||
|
Loading…
Reference in New Issue
Block a user