edit and remove images

This commit is contained in:
lightling 2025-03-03 21:22:45 -05:00
parent 4a6ad11d16
commit 0a9e4ebbd4
Signed by: lightling
GPG key ID: F1F29650D537C773
4 changed files with 59 additions and 20 deletions

View file

@ -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[]

View file

@ -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
) )

View file

@ -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'

View file

@ -0,0 +1,9 @@
import {
type Row,
} from 'src/types/data'
export interface EditingRow {
data: Row
index: number
field: string
}