edit and remove images
This commit is contained in:
parent
4a6ad11d16
commit
0a9e4ebbd4
4 changed files with 59 additions and 20 deletions
|
@ -22,7 +22,7 @@ export type Columns = Column[]
|
||||||
|
|
||||||
export type Row = {
|
export type Row = {
|
||||||
_id: string
|
_id: string
|
||||||
} & Record<string, string>
|
} & Record<string, string | any>
|
||||||
|
|
||||||
export interface Inventory {
|
export interface Inventory {
|
||||||
columns: Column[]
|
columns: Column[]
|
||||||
|
|
|
@ -12,12 +12,7 @@ import {
|
||||||
type Column as DataColumn,
|
type Column as DataColumn,
|
||||||
} from 'src/types/data'
|
} from 'src/types/data'
|
||||||
import ImageEditor from './ImageEditor.vue'
|
import ImageEditor from './ImageEditor.vue'
|
||||||
|
import { type EditingRow } from './types'
|
||||||
interface EditingRow {
|
|
||||||
data: DataRow
|
|
||||||
index: number
|
|
||||||
field: string
|
|
||||||
}
|
|
||||||
|
|
||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
columns: DataColumn[],
|
columns: DataColumn[],
|
||||||
|
@ -26,7 +21,7 @@ const emits = defineEmits<{
|
||||||
(e: 'dirty'): void
|
(e: 'dirty'): void
|
||||||
}>()
|
}>()
|
||||||
const model = defineModel<DataRow[]>({ required: true })
|
const model = defineModel<DataRow[]>({ required: true })
|
||||||
const editingRows = ref([])
|
const editingRows = ref<DataRow[]>([])
|
||||||
const editingImage = ref(false)
|
const editingImage = ref(false)
|
||||||
const editingImageRow = ref(null as EditingRow | null)
|
const editingImageRow = ref(null as EditingRow | null)
|
||||||
|
|
||||||
|
@ -55,18 +50,37 @@ const onDeleteRow = (event: Event, slotProps: { index: number }) => {
|
||||||
emits('dirty')
|
emits('dirty')
|
||||||
}
|
}
|
||||||
|
|
||||||
const onEditImage = (event: Event, slotProps: { data: DataRow, index: number, field: string }) => {
|
const onEditImage = (event: Event, slotProps: EditingRow) => {
|
||||||
event.preventDefault()
|
event.preventDefault()
|
||||||
editingImage.value = true
|
editingImage.value = true
|
||||||
editingImageRow.value = slotProps
|
editingImageRow.value = slotProps
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const onRemoveImage = (event: Event, slotProps: EditingRow) => {
|
||||||
|
event.preventDefault()
|
||||||
|
const row = { ...model.value[slotProps.index] } as any
|
||||||
|
delete row[slotProps.field]
|
||||||
|
model.value[slotProps.index] = row
|
||||||
|
const indexOfEditing = editingRows.value.findIndex(other => other._id === slotProps.data._id)
|
||||||
|
if (indexOfEditing > -1) {
|
||||||
|
let rows = editingRows.value
|
||||||
|
rows.splice(indexOfEditing, 1)
|
||||||
|
editingRows.value = rows
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const onSaveImage = (image: string) => {
|
const onSaveImage = (image: string) => {
|
||||||
const row = { ...model.value[editingImageRow.value!.index] } as any
|
const row = { ...model.value[editingImageRow.value!.index] } as any
|
||||||
row[editingImageRow.value!.field] = {
|
row[editingImageRow.value!.field] = {
|
||||||
src: image,
|
src: image,
|
||||||
}
|
}
|
||||||
model.value[editingImageRow.value!.index] = row
|
model.value[editingImageRow.value!.index] = row
|
||||||
|
const indexOfEditing = editingRows.value.findIndex(other => other._id === editingImageRow.value!.data._id)
|
||||||
|
if (indexOfEditing > -1) {
|
||||||
|
let rows = editingRows.value
|
||||||
|
rows.splice(indexOfEditing, 1)
|
||||||
|
editingRows.value = rows
|
||||||
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
@ -98,7 +112,7 @@ DataTable.data-editor(
|
||||||
v-if='col.type === "image"'
|
v-if='col.type === "image"'
|
||||||
)
|
)
|
||||||
Image(
|
Image(
|
||||||
v-if='slotProps.data[col.name]'
|
v-if='!!slotProps.data[col.name] && !!slotProps.data[col.name].src'
|
||||||
:src='slotProps.data[col.name].src'
|
:src='slotProps.data[col.name].src'
|
||||||
preview
|
preview
|
||||||
)
|
)
|
||||||
|
@ -119,6 +133,12 @@ DataTable.data-editor(
|
||||||
label='Edit Image'
|
label='Edit Image'
|
||||||
:onClick='(e) => onEditImage(e, slotProps)'
|
:onClick='(e) => onEditImage(e, slotProps)'
|
||||||
)
|
)
|
||||||
|
Button(
|
||||||
|
label='Remove Image'
|
||||||
|
severity='danger'
|
||||||
|
:disabled='!slotProps.data[col.name]'
|
||||||
|
:onClick='(e) => onRemoveImage(e, slotProps)'
|
||||||
|
)
|
||||||
div(
|
div(
|
||||||
v-else
|
v-else
|
||||||
)
|
)
|
||||||
|
|
|
@ -12,10 +12,8 @@ import InputGroup from 'primevue/inputgroup'
|
||||||
import InputGroupAddon from 'primevue/inputgroupaddon'
|
import InputGroupAddon from 'primevue/inputgroupaddon'
|
||||||
import InputText from 'primevue/inputtext'
|
import InputText from 'primevue/inputtext'
|
||||||
|
|
||||||
import {
|
|
||||||
type Row as DataRow,
|
|
||||||
} from 'src/types/data'
|
|
||||||
import { useAppStore } from 'src/store'
|
import { useAppStore } from 'src/store'
|
||||||
|
import { type EditingRow } from './types'
|
||||||
|
|
||||||
const store = useAppStore()
|
const store = useAppStore()
|
||||||
|
|
||||||
|
@ -23,10 +21,16 @@ const visible = defineModel<boolean>('visible', { default: false })
|
||||||
const emit = defineEmits<{
|
const emit = defineEmits<{
|
||||||
(e: 'saved', val: string): void
|
(e: 'saved', val: string): void
|
||||||
}>()
|
}>()
|
||||||
const row = defineModel<DataRow>('row')
|
const row = defineModel<EditingRow | null>('row')
|
||||||
const filePath = ref(store.currentInventory.filePath.replace('.json', '/'))
|
const filePath = ref(store.currentInventory.filePath.replace('.json', '/'))
|
||||||
const fileName = ref('')
|
|
||||||
const currentSrc = ref('')
|
const currentSrc = ref('')
|
||||||
|
const renderedSrc = computed(() => {
|
||||||
|
return currentSrc.value
|
||||||
|
? currentSrc.value
|
||||||
|
: !!row.value && !!row.value.data[row.value.field] && !!(row.value.data[row.value.field]).src
|
||||||
|
? (row.value.data[row.value.field]).src
|
||||||
|
: ''
|
||||||
|
})
|
||||||
const currentBlob = ref<Blob>()
|
const currentBlob = ref<Blob>()
|
||||||
|
|
||||||
const onBrowseForImage = async (e: Event) => {
|
const onBrowseForImage = async (e: Event) => {
|
||||||
|
@ -52,7 +56,7 @@ const onBrowseForImage = async (e: Event) => {
|
||||||
|
|
||||||
const onCancel = (e: Event) => {
|
const onCancel = (e: Event) => {
|
||||||
e.preventDefault()
|
e.preventDefault()
|
||||||
visible.value = false
|
reset()
|
||||||
}
|
}
|
||||||
|
|
||||||
function convertToDataUrlViaCanvas(url: string, callback: (val: string) => void, outputFormat: string) {
|
function convertToDataUrlViaCanvas(url: string, callback: (val: string) => void, outputFormat: string) {
|
||||||
|
@ -71,11 +75,17 @@ function convertToDataUrlViaCanvas(url: string, callback: (val: string) => void,
|
||||||
image.src = url
|
image.src = url
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const reset = () => {
|
||||||
|
visible.value = false
|
||||||
|
row.value = null
|
||||||
|
currentSrc.value = ''
|
||||||
|
}
|
||||||
|
|
||||||
const onSave = async (e: Event) => {
|
const onSave = async (e: Event) => {
|
||||||
e.preventDefault()
|
e.preventDefault()
|
||||||
convertToDataUrlViaCanvas(currentSrc.value, (val) => {
|
convertToDataUrlViaCanvas(currentSrc.value, (val) => {
|
||||||
emit('saved', val)
|
emit('saved', val)
|
||||||
visible.value = false
|
reset()
|
||||||
}, currentBlob.value!.type)
|
}, currentBlob.value!.type)
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
@ -87,7 +97,7 @@ Dialog(
|
||||||
header='Edit Image'
|
header='Edit Image'
|
||||||
)
|
)
|
||||||
.content(
|
.content(
|
||||||
v-if='!currentSrc'
|
v-if='!renderedSrc'
|
||||||
)
|
)
|
||||||
Button(
|
Button(
|
||||||
label='Browse For Image'
|
label='Browse For Image'
|
||||||
|
@ -98,7 +108,7 @@ Dialog(
|
||||||
)
|
)
|
||||||
InputGroup.image-group
|
InputGroup.image-group
|
||||||
ImageComponent(
|
ImageComponent(
|
||||||
:src='currentSrc'
|
:src='renderedSrc'
|
||||||
)
|
)
|
||||||
template(#footer)
|
template(#footer)
|
||||||
Button(
|
Button(
|
||||||
|
@ -109,7 +119,7 @@ Dialog(
|
||||||
Button(
|
Button(
|
||||||
label='Replace'
|
label='Replace'
|
||||||
@click='onBrowseForImage'
|
@click='onBrowseForImage'
|
||||||
:disabled='!currentSrc'
|
:disabled='!renderedSrc && !currentSrc'
|
||||||
)
|
)
|
||||||
Button(
|
Button(
|
||||||
label='Cancel'
|
label='Cancel'
|
||||||
|
|
9
src/views/Editor/types.ts
Normal file
9
src/views/Editor/types.ts
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
import {
|
||||||
|
type Row,
|
||||||
|
} from 'src/types/data'
|
||||||
|
|
||||||
|
export interface EditingRow {
|
||||||
|
data: Row
|
||||||
|
index: number
|
||||||
|
field: string
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue