mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-10-23 23:08:57 -04:00
368 lines
12 KiB
Vue
368 lines
12 KiB
Vue
<template>
|
|
<div>
|
|
<trix-toolbar :id="toolbarId">
|
|
<div v-show="!disabledEditor" class="trix-button-row">
|
|
<span class="trix-button-group trix-button-group--text-tools" data-trix-button-group="text-tools">
|
|
<button type="button" class="trix-button trix-button--icon trix-button--icon-bold" data-trix-attribute="bold" data-trix-key="b" :title="$strings.LabelFontBold" tabindex="-1">{{ $strings.LabelFontBold }}</button>
|
|
<button type="button" class="trix-button trix-button--icon trix-button--icon-italic" data-trix-attribute="italic" data-trix-key="i" :title="$strings.LabelFontItalic" tabindex="-1">{{ $strings.LabelFontItalic }}</button>
|
|
<button type="button" class="trix-button trix-button--icon trix-button--icon-strike" data-trix-attribute="strike" :title="$strings.LabelFontStrikethrough" tabindex="-1">{{ $strings.LabelFontStrikethrough }}</button>
|
|
<button type="button" class="trix-button trix-button--icon trix-button--icon-link" data-trix-attribute="href" data-trix-action="link" data-trix-key="k" :title="$strings.LabelTextEditorLink" tabindex="-1">{{ $strings.LabelTextEditorLink }}</button>
|
|
</span>
|
|
<span class="trix-button-group trix-button-group--block-tools" data-trix-button-group="block-tools">
|
|
<button type="button" class="trix-button trix-button--icon trix-button--icon-bullet-list" data-trix-attribute="bullet" :title="$strings.LabelTextEditorBulletedList" tabindex="-1">{{ $strings.LabelTextEditorBulletedList }}</button>
|
|
<button type="button" class="trix-button trix-button--icon trix-button--icon-number-list" data-trix-attribute="number" :title="$strings.LabelTextEditorNumberedList" tabindex="-1">{{ $strings.LabelTextEditorNumberedList }}</button>
|
|
</span>
|
|
|
|
<span class="trix-button-group-spacer"></span>
|
|
<span class="trix-button-group trix-button-group--history-tools" data-trix-button-group="history-tools">
|
|
<button type="button" class="trix-button trix-button--icon trix-button--icon-undo" data-trix-action="undo" data-trix-key="z" :title="$strings.LabelUndo" tabindex="-1">{{ $strings.LabelUndo }}</button>
|
|
<button type="button" class="trix-button trix-button--icon trix-button--icon-redo" data-trix-action="redo" data-trix-key="shift+z" :title="$strings.LabelRedo" tabindex="-1">{{ $strings.LabelRedo }}</button>
|
|
</span>
|
|
</div>
|
|
<div class="trix-dialogs" data-trix-dialogs>
|
|
<div class="trix-dialog trix-dialog--link" data-trix-dialog="href" data-trix-dialog-attribute="href">
|
|
<div class="trix-dialog__link-fields">
|
|
<input type="url" name="href" class="trix-input trix-input--dialog" placeholder="" aria-label="URL" required data-trix-input />
|
|
<div class="trix-button-group">
|
|
<input type="button" class="trix-button trix-button--dialog" :value="$strings.LabelTextEditorLink" data-trix-method="setAttribute" />
|
|
<input type="button" class="trix-button trix-button--dialog" :value="$strings.LabelTextEditorUnlink" data-trix-method="removeAttribute" />
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</trix-toolbar>
|
|
<trix-editor :toolbar="toolbarId" :contenteditable="!disabledEditor" :class="['trix-content']" ref="trix" :input="computedId" :placeholder="placeholder" @trix-change="handleContentChange" @trix-initialize="handleInitialize" @trix-focus="processTrixFocus" @trix-blur="processTrixBlur" @trix-attachment-add="handleAttachmentAdd" />
|
|
<input type="hidden" :name="inputName" :id="computedId" :value="editorContent" />
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
/*
|
|
ORIGINAL SOURCE: https://github.com/hanhdt/vue-trix
|
|
|
|
modified for audiobookshelf
|
|
*/
|
|
import Trix from 'trix'
|
|
import '@/assets/trix.css'
|
|
|
|
function enableBreakParagraphOnReturn() {
|
|
// Trix works with divs by default, we want paragraphs instead
|
|
Trix.config.blockAttributes.default.tagName = 'p'
|
|
// Enable break paragraph on Enter (Shift + Enter will still create a line break)
|
|
Trix.config.blockAttributes.default.breakOnReturn = true
|
|
|
|
// Hack to fix buggy paragraph breaks
|
|
// Copied from https://github.com/basecamp/trix/issues/680#issuecomment-735742942
|
|
Trix.Block.prototype.breaksOnReturn = function () {
|
|
const attr = this.getLastAttribute()
|
|
const config = Trix.getBlockConfig(attr ? attr : 'default')
|
|
return config ? config.breakOnReturn : false
|
|
}
|
|
Trix.LineBreakInsertion.prototype.shouldInsertBlockBreak = function () {
|
|
if (this.block.hasAttributes() && this.block.isListItem() && !this.block.isEmpty()) {
|
|
return this.startLocation.offset > 0
|
|
} else {
|
|
return !this.shouldBreakFormattedBlock() ? this.breaksOnReturn : false
|
|
}
|
|
}
|
|
}
|
|
|
|
enableBreakParagraphOnReturn()
|
|
|
|
export default {
|
|
name: 'vue-trix',
|
|
model: {
|
|
prop: 'srcContent',
|
|
event: 'update'
|
|
},
|
|
props: {
|
|
/**
|
|
* This prop will put the editor in read-only mode
|
|
*/
|
|
disabledEditor: {
|
|
type: Boolean,
|
|
required: false,
|
|
default() {
|
|
return false
|
|
}
|
|
},
|
|
/**
|
|
* This is referenced `id` of the hidden input field defined.
|
|
* It is optional and will be a random string by default.
|
|
*/
|
|
inputId: {
|
|
type: String,
|
|
required: false,
|
|
default() {
|
|
return ''
|
|
}
|
|
},
|
|
/**
|
|
* This is referenced `name` of the hidden input field defined,
|
|
* default value is `content`.
|
|
*/
|
|
inputName: {
|
|
type: String,
|
|
required: false,
|
|
default() {
|
|
return 'content'
|
|
}
|
|
},
|
|
/**
|
|
* The placeholder attribute specifies a short hint
|
|
* that describes the expected value of a editor.
|
|
*/
|
|
placeholder: {
|
|
type: String,
|
|
required: false,
|
|
default() {
|
|
return ''
|
|
}
|
|
},
|
|
/**
|
|
* The source content is associcated to v-model directive.
|
|
*/
|
|
srcContent: {
|
|
type: String,
|
|
required: false,
|
|
default() {
|
|
return ''
|
|
}
|
|
},
|
|
/**
|
|
* The boolean attribute allows saving editor state into browser's localStorage
|
|
* (optional, default is `false`).
|
|
*/
|
|
localStorage: {
|
|
type: Boolean,
|
|
required: false,
|
|
default() {
|
|
return false
|
|
}
|
|
},
|
|
/**
|
|
* Focuses cursor in the editor when attached to the DOM
|
|
* (optional, default is `false`).
|
|
*/
|
|
autofocus: {
|
|
type: Boolean,
|
|
required: false,
|
|
default() {
|
|
return false
|
|
}
|
|
},
|
|
/**
|
|
* Object to override default editor configuration
|
|
*/
|
|
config: {
|
|
type: Object,
|
|
required: false,
|
|
default() {
|
|
return {}
|
|
}
|
|
}
|
|
},
|
|
data() {
|
|
return {
|
|
editorContent: this.srcContent,
|
|
isActived: null
|
|
}
|
|
},
|
|
watch: {
|
|
editorContent: {
|
|
handler: 'emitEditorState'
|
|
},
|
|
initialContent: {
|
|
handler: 'handleInitialContentChange'
|
|
},
|
|
isDisabled: {
|
|
handler: 'decorateDisabledEditor'
|
|
},
|
|
config: {
|
|
handler: 'overrideConfig',
|
|
immediate: true,
|
|
deep: true
|
|
}
|
|
},
|
|
computed: {
|
|
/**
|
|
* Compute a random id of hidden input
|
|
* when it haven't been specified.
|
|
*/
|
|
toolbarId() {
|
|
return `trix-toolbar-${this.generateId}`
|
|
},
|
|
generateId() {
|
|
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {
|
|
var r = (Math.random() * 16) | 0
|
|
var v = c === 'x' ? r : (r & 0x3) | 0x8
|
|
return v.toString(16)
|
|
})
|
|
},
|
|
computedId() {
|
|
return this.inputId || this.generateId
|
|
},
|
|
initialContent() {
|
|
return this.srcContent
|
|
},
|
|
isDisabled() {
|
|
return this.disabledEditor
|
|
}
|
|
},
|
|
methods: {
|
|
processTrixFocus(event) {
|
|
if (this.$refs.trix) {
|
|
this.isActived = true
|
|
this.$emit('trix-focus', this.$refs.trix.editor, event)
|
|
}
|
|
},
|
|
processTrixBlur(event) {
|
|
if (this.$refs.trix) {
|
|
this.isActived = false
|
|
this.$emit('trix-blur', this.$refs.trix.editor, event)
|
|
}
|
|
},
|
|
handleContentChange(event) {
|
|
this.editorContent = event.srcElement ? event.srcElement.value : event.target.value
|
|
this.$emit('input', this.editorContent)
|
|
},
|
|
handleInitialize(event) {
|
|
/**
|
|
* If autofocus is true, manually set focus to
|
|
* beginning of content (consistent with Trix behavior)
|
|
*/
|
|
if (this.autofocus) {
|
|
this.$refs.trix.editor.setSelectedRange(0)
|
|
}
|
|
this.$emit('trix-initialize', this.emitInitialize)
|
|
},
|
|
handleInitialContentChange(newContent, oldContent) {
|
|
newContent = newContent === undefined ? '' : newContent
|
|
if (this.$refs.trix.editor && this.$refs.trix.editor.innerHTML !== newContent) {
|
|
/* Update editor's content when initial content changed */
|
|
this.editorContent = newContent
|
|
/**
|
|
* If user are typing, then don't reload the editor,
|
|
* hence keep cursor's position after typing.
|
|
*/
|
|
if (!this.isActived) {
|
|
this.reloadEditorContent(this.editorContent)
|
|
}
|
|
}
|
|
},
|
|
emitEditorState(value) {
|
|
/**
|
|
* If localStorage is enabled,
|
|
* then save editor's content into storage
|
|
*/
|
|
if (this.localStorage) {
|
|
localStorage.setItem(this.storageId('VueTrix'), JSON.stringify(this.$refs.trix.editor))
|
|
}
|
|
this.$emit('update', this.editorContent)
|
|
},
|
|
storageId(component) {
|
|
if (this.inputId) {
|
|
return `${component}.${this.inputId}.content`
|
|
} else {
|
|
return `${component}.content`
|
|
}
|
|
},
|
|
reloadEditorContent(newContent) {
|
|
// Reload HTML content
|
|
this.$refs.trix.editor.loadHTML(newContent)
|
|
// Move cursor to end of new content updated
|
|
if (this.autofocus) {
|
|
this.$refs.trix.editor.setSelectedRange(this.getContentEndPosition())
|
|
}
|
|
},
|
|
getContentEndPosition() {
|
|
return this.$refs.trix.editor.getDocument().toString().length - 1
|
|
},
|
|
decorateDisabledEditor(editorState) {
|
|
/** Disable toolbar and editor by pointer events styling */
|
|
if (editorState) {
|
|
this.$refs.trix.disabled = true
|
|
this.$refs.trix.contentEditable = false
|
|
this.$refs.trix.style['pointer-events'] = 'none'
|
|
this.$refs.trix.style['background-color'] = '#444'
|
|
this.$refs.trix.style['color'] = '#bbb'
|
|
} else {
|
|
this.$refs.trix.disabled = false
|
|
this.$refs.trix.contentEditable = true
|
|
this.$refs.trix.style['pointer-events'] = 'unset'
|
|
this.$refs.trix.style['background-color'] = ''
|
|
this.$refs.trix.style['color'] = ''
|
|
}
|
|
},
|
|
overrideConfig(config) {
|
|
Trix.config = this.deepMerge(Trix.config, config)
|
|
},
|
|
deepMerge(target, override) {
|
|
// deep merge the object into the target object
|
|
for (let prop in override) {
|
|
if (override.hasOwnProperty(prop)) {
|
|
if (Object.prototype.toString.call(override[prop]) === '[object Object]') {
|
|
// if the property is a nested object
|
|
target[prop] = this.deepMerge(target[prop], override[prop])
|
|
} else {
|
|
// for regular property
|
|
target[prop] = override[prop]
|
|
}
|
|
}
|
|
}
|
|
return target
|
|
},
|
|
blur() {
|
|
if (this.$refs.trix && this.$refs.trix.blur) {
|
|
this.$refs.trix.blur()
|
|
}
|
|
},
|
|
handleAttachmentAdd(event) {
|
|
// Prevent pasting in images/any files from the browser
|
|
event.attachment.remove()
|
|
}
|
|
},
|
|
mounted() {
|
|
/** Override editor configuration */
|
|
this.overrideConfig(this.config)
|
|
/** Check if editor read-only mode is required */
|
|
this.decorateDisabledEditor(this.disabledEditor)
|
|
this.$nextTick(() => {
|
|
/**
|
|
* If localStorage is enabled,
|
|
* then load editor's content from the beginning.
|
|
*/
|
|
if (this.localStorage) {
|
|
const savedValue = localStorage.getItem(this.storageId('VueTrix'))
|
|
if (savedValue && !this.srcContent) {
|
|
this.$refs.trix.editor.loadJSON(JSON.parse(savedValue))
|
|
}
|
|
}
|
|
})
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<style lang="css" module>
|
|
.trix_container {
|
|
max-width: 100%;
|
|
height: auto;
|
|
}
|
|
.trix_container .trix-button-group {
|
|
background-color: white;
|
|
}
|
|
.trix_container .trix-content {
|
|
background-color: white;
|
|
}
|
|
trix-editor {
|
|
height: calc(4 * 1lh);
|
|
min-height: calc(4 * 1lh);
|
|
overflow-y: auto;
|
|
resize: vertical;
|
|
}
|
|
|
|
trix-editor * {
|
|
pointer-events: inherit;
|
|
}
|
|
</style>
|