<template> <div class="sci-inline-edit" :class="{ 'editing': editing }" tabindex="0" @keyup.enter="enableEdit($event)"> <div class="sci-inline-edit__content" :class="{ 'error': error }"> <textarea ref="input" rows="1" v-if="editing" :placeholder="placeholder" v-model="newValue" @input="handleInput" @keydown="handleKeypress" @paste="handlePaste" @blur="handleBlur" @keyup.escape="cancelEdit" ></textarea> <div v-else @click="enableEdit($event)" class="sci-inline-edit__view" v-html="sa_value || value || placeholder" :class="{ 'blank': isBlank }"></div> <div v-if="editing && error" class="sci-inline-edit__error"> {{ error }} </div> </div> <template v-if="editing"> <div :class="{ 'btn-primary': !error, 'btn-disabled': error }" class="sci-inline-edit__control btn icon-btn" @click="update"> <i class="fas fa-check"></i> </div> <div class="sci-inline-edit__control btn btn-light icon-btn" @click="cancelEdit"> <i class="fas fa-times"></i> </div> </template> </div> </template> <script> export default { name: 'InlineEdit', props: { value: { type: String, default: '' }, sa_value: { type: String}, allowBlank: { type: Boolean, default: true }, attributeName: { type: String, required: true }, characterLimit: { type: Number }, placeholder: { type: String }, autofocus: { type: Boolean, default: false }, multilinePaste: { type: Boolean, default: false }, smartAnnotation: { type: Boolean, default: false }, editOnload: { type: Boolean, default: false } }, data() { return { editing: false, newValue: '' } }, created( ){ this.newValue = this.value || ''; }, mounted() { this.handleAutofocus(); if (this.editOnload) { this.enableEdit(); } }, watch: { autofocus() { this.handleAutofocus(); } }, computed: { isBlank() { return this.newValue.length === 0 }, error() { if(!this.allowBlank && this.isBlank) { return this.i18n.t('inline_edit.errors.blank', { attribute: this.attributeName }) } if(this.characterLimit && this.newValue.length > this.characterLimit) { return( this.i18n.t('inline_edit.errors.over_limit', { attribute: this.attributeName, limit: this.characterLimit } ) ) } return false } }, methods: { handleAutofocus() { if (this.autofocus || !this.placeholder && this.isBlank || this.editOnload && this.isBlank) { this.enableEdit(); setTimeout(this.focus, 50); } }, handleBlur() { if ($('.atwho-view:visible').length) return; if (this.allowBlank || !this.isBlank) { this.$nextTick(this.update); } else { this.$emit('delete'); } }, focus() { this.$nextTick(() => { if (!this.$refs.input) return; this.$refs.input.focus(); this.resize(); }); }, enableEdit(e) { if (e && $(e.target).hasClass('atwho-user-popover')) return this.editing = true; this.focus(); this.$nextTick(() => { if (this.smartAnnotation) { SmartAnnotation.init($(this.$refs.input)); } }) this.$emit('editingEnabled'); }, cancelEdit() { this.editing = false; this.newValue = this.value || ''; this.$emit('editingDisabled'); }, handlePaste(e) { if (!this.multilinePaste) return; e.clipboardData.items[0].getAsString((data) => { let lines = data.split(/[\n\r]/); if (lines.length > 1) { this.newValue = lines[0]; this.$emit('multilinePaste', lines); } }) }, handleInput() { this.newValue = this.newValue.replace(/^[\n\r]+|[\n\r]+$/g, ''); this.$nextTick(this.resize); }, handleKeypress(e) { switch(e.key) { case 'Escape': this.cancelEdit(); break; case 'Enter': this.update(); break; } }, resize() { this.$refs.input.style.height = "auto"; this.$refs.input.style.height = (this.$refs.input.scrollHeight) + "px"; }, update() { setTimeout(() => { if(this.error) return; if(!this.editing) return; this.newValue = this.$refs.input.value // Fix for smart annotation this.newValue = this.newValue.trim(); this.editing = false; this.$emit('editingDisabled'); this.$emit('update', this.newValue); }, 100) // due to clicking 'x' also triggering a blur event } } } </script>