support dropdown field option
This commit is contained in:
parent
8b191a93b8
commit
7a0b69c1c1
4 changed files with 140 additions and 10 deletions
67
src/components/InputChips.vue
Normal file
67
src/components/InputChips.vue
Normal file
|
@ -0,0 +1,67 @@
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { computed, ref } from 'vue'
|
||||||
|
import Button from 'primevue/button'
|
||||||
|
import Chip from 'primevue/chip'
|
||||||
|
import InputGroup from 'primevue/inputgroup'
|
||||||
|
import InputText from 'primevue/inputtext'
|
||||||
|
|
||||||
|
const model = defineModel<string[]>()
|
||||||
|
const entry = ref('')
|
||||||
|
const preventSubmit = computed(() => entry.value === '' || model.value?.includes(entry.value))
|
||||||
|
const emits = defineEmits<{
|
||||||
|
(e: 'dirty'): void
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const onRemove = (chip: string) => {
|
||||||
|
if (!model.value) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const index = model.value.findIndex(other => other === chip)
|
||||||
|
if (index > -1) {
|
||||||
|
model.value?.splice(index, 1)
|
||||||
|
emits('dirty')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const onSubmit = (event: Event) => {
|
||||||
|
event.preventDefault()
|
||||||
|
if (preventSubmit.value) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (!Array.isArray(model.value)) {
|
||||||
|
model.value = []
|
||||||
|
}
|
||||||
|
model.value.push(entry.value)
|
||||||
|
entry.value = ''
|
||||||
|
emits('dirty')
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template lang="pug">
|
||||||
|
.input-chips
|
||||||
|
.chips
|
||||||
|
Chip(
|
||||||
|
v-for='chip in model'
|
||||||
|
:label='chip'
|
||||||
|
:onRemove='(e) => onRemove(chip)'
|
||||||
|
removable
|
||||||
|
)
|
||||||
|
InputGroup
|
||||||
|
InputText(
|
||||||
|
v-model='entry'
|
||||||
|
v-on:keyup.enter='onSubmit'
|
||||||
|
fluid
|
||||||
|
)
|
||||||
|
Button(
|
||||||
|
icon='pi pi-plus'
|
||||||
|
label='Add Option'
|
||||||
|
type='submit'
|
||||||
|
:disabled='preventSubmit'
|
||||||
|
@click='onSubmit'
|
||||||
|
)
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style scoped lang="sass">
|
||||||
|
|
||||||
|
</style>
|
|
@ -1,3 +1,7 @@
|
||||||
|
/**
|
||||||
|
* Defines the available types of fields that can be edited by the data editor,
|
||||||
|
* set by a column's field type.
|
||||||
|
*/
|
||||||
export const FieldTypes = [
|
export const FieldTypes = [
|
||||||
'text',
|
'text',
|
||||||
'image',
|
'image',
|
||||||
|
@ -6,6 +10,9 @@ export const FieldTypes = [
|
||||||
'toggle',
|
'toggle',
|
||||||
] as const
|
] as const
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specifies the type of field for editing in the data editor.
|
||||||
|
*/
|
||||||
export type FieldType = typeof FieldTypes[number]
|
export type FieldType = typeof FieldTypes[number]
|
||||||
|
|
||||||
export interface Image {
|
export interface Image {
|
||||||
|
@ -13,18 +20,35 @@ export interface Image {
|
||||||
alt?: string
|
alt?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The structure of a column. Columns define the available fields to edit for every row
|
||||||
|
* and control how they are edited.
|
||||||
|
*/
|
||||||
export interface Column {
|
export interface Column {
|
||||||
|
/** the name of the field; this represents the key in the columns record */
|
||||||
name: string
|
name: string
|
||||||
|
/** the type of the field; this controls how it is edited in the data editor */
|
||||||
type: FieldType
|
type: FieldType
|
||||||
|
/** whether the field is secondary or not; if it is, it is hidden from the quick editing view in the data editor */
|
||||||
secondary?: boolean
|
secondary?: boolean
|
||||||
|
/** options for the field; applies only to certain field types such as dropdowns */
|
||||||
|
options?: any
|
||||||
}
|
}
|
||||||
|
|
||||||
export type Columns = Column[]
|
/**
|
||||||
|
* The structure of a Row. All rows at minimum should have a private _id field;
|
||||||
|
* beyond that, it is simply a record of key-value pairs
|
||||||
|
* where the key corresponds to a column by name
|
||||||
|
*/
|
||||||
export type Row = {
|
export type Row = {
|
||||||
_id: string
|
_id: string
|
||||||
} & Record<string, string | any>
|
} & Record<string, string | any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The structure of an inventory. An inventory is simply a set of columns (fields) and rows (data),
|
||||||
|
* where each row has key-value pairs where the keys correspond to a column by name,
|
||||||
|
* and the columns define how the rows can be edited
|
||||||
|
*/
|
||||||
export interface Inventory {
|
export interface Inventory {
|
||||||
columns: Column[]
|
columns: Column[]
|
||||||
rows: Row[]
|
rows: Row[]
|
||||||
|
|
|
@ -3,10 +3,11 @@ import { computed, ref } from 'vue'
|
||||||
import { v7 as uuidv7 } from 'uuid'
|
import { v7 as uuidv7 } from 'uuid'
|
||||||
import Button from 'primevue/button'
|
import Button from 'primevue/button'
|
||||||
import Checkbox from 'primevue/checkbox'
|
import Checkbox from 'primevue/checkbox'
|
||||||
import DataTable from 'primevue/datatable'
|
|
||||||
import Column from 'primevue/column'
|
import Column from 'primevue/column'
|
||||||
|
import DataTable from 'primevue/datatable'
|
||||||
import Image from 'primevue/image'
|
import Image from 'primevue/image'
|
||||||
import InputText from 'primevue/inputtext'
|
import InputText from 'primevue/inputtext'
|
||||||
|
import Select from 'primevue/select'
|
||||||
|
|
||||||
import {
|
import {
|
||||||
type Row as DataRow,
|
type Row as DataRow,
|
||||||
|
@ -143,7 +144,12 @@ DataTable.data-editor(
|
||||||
div(
|
div(
|
||||||
v-else
|
v-else
|
||||||
)
|
)
|
||||||
span {{ slotProps.data[col.name] }}
|
span(
|
||||||
|
v-if='slotProps.data[col.name]'
|
||||||
|
) {{ slotProps.data[col.name] }}
|
||||||
|
span.empty(
|
||||||
|
v-else
|
||||||
|
) not set
|
||||||
template(
|
template(
|
||||||
#editor='slotProps'
|
#editor='slotProps'
|
||||||
)
|
)
|
||||||
|
@ -168,6 +174,17 @@ DataTable.data-editor(
|
||||||
:name='`toggle for ${slotProps.data.name || slotProps.data._id} ${col.name}`'
|
:name='`toggle for ${slotProps.data.name || slotProps.data._id} ${col.name}`'
|
||||||
binary
|
binary
|
||||||
)
|
)
|
||||||
|
div(
|
||||||
|
v-else-if='col.type === "dropdown"'
|
||||||
|
)
|
||||||
|
Select(
|
||||||
|
v-model='slotProps.data[slotProps.field]'
|
||||||
|
:options='col.options'
|
||||||
|
placeholder='Select from Options'
|
||||||
|
filter
|
||||||
|
showClear
|
||||||
|
fluid
|
||||||
|
)
|
||||||
div(
|
div(
|
||||||
v-else
|
v-else
|
||||||
)
|
)
|
||||||
|
@ -228,7 +245,7 @@ DataTable.data-editor(
|
||||||
)
|
)
|
||||||
p.empty(
|
p.empty(
|
||||||
v-if='!slotProps.data[field]'
|
v-if='!slotProps.data[field]'
|
||||||
) {{ field }} is empty
|
) {{ field }} is not set
|
||||||
.info-image(
|
.info-image(
|
||||||
v-else-if='column.type === "image"'
|
v-else-if='column.type === "image"'
|
||||||
)
|
)
|
||||||
|
@ -262,13 +279,14 @@ Button(
|
||||||
img
|
img
|
||||||
max-width: 100%
|
max-width: 100%
|
||||||
max-height: 8rem
|
max-height: 8rem
|
||||||
:deep(td:last-of-type)
|
:deep(tr:not(.p-datatable-row-expansion) td:last-of-type)
|
||||||
padding-left: 0
|
padding-left: 0
|
||||||
button:first-of-type
|
button:first-of-type
|
||||||
margin-left: -16px
|
margin-left: -16px
|
||||||
.info :deep(.p-image) img
|
.info :deep(.p-image) img
|
||||||
max-width: 16rem
|
max-width: 16rem
|
||||||
max-height: 16rem
|
max-height: 16rem
|
||||||
p.empty
|
p.empty,
|
||||||
|
span.empty
|
||||||
font-style: italic
|
font-style: italic
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -12,12 +12,14 @@ import {
|
||||||
FieldTypes,
|
FieldTypes,
|
||||||
type Column as DataColumn,
|
type Column as DataColumn,
|
||||||
} from 'src/types/data'
|
} from 'src/types/data'
|
||||||
|
import InputChips from 'src/components/InputChips.vue'
|
||||||
|
|
||||||
const emits = defineEmits<{
|
const emits = defineEmits<{
|
||||||
(e: 'dirty'): void
|
(e: 'dirty'): void
|
||||||
}>()
|
}>()
|
||||||
const model = defineModel<DataColumn[]>({ required: true })
|
const model = defineModel<DataColumn[]>({ required: true })
|
||||||
const editingFields = ref([])
|
const editingFields = ref([])
|
||||||
|
const expandedRows = ref([])
|
||||||
|
|
||||||
const onFieldEditSave = (event: { newData: any, index: number }) => {
|
const onFieldEditSave = (event: { newData: any, index: number }) => {
|
||||||
let { newData, index } = event
|
let { newData, index } = event
|
||||||
|
@ -49,10 +51,15 @@ const onDeleteField = (event: Event, slotProps: { index: number }) => {
|
||||||
DataTable.field-editor(
|
DataTable.field-editor(
|
||||||
@row-edit-save='onFieldEditSave'
|
@row-edit-save='onFieldEditSave'
|
||||||
v-model:editingRows='editingFields'
|
v-model:editingRows='editingFields'
|
||||||
|
v-model:expandedRows='expandedRows'
|
||||||
:value='model'
|
:value='model'
|
||||||
editMode='row'
|
editMode='row'
|
||||||
tableStyle='min-width: 50rem'
|
tableStyle='min-width: 50rem'
|
||||||
)
|
)
|
||||||
|
Column(
|
||||||
|
expander
|
||||||
|
style='width:5rem;'
|
||||||
|
)
|
||||||
Column(
|
Column(
|
||||||
field='name'
|
field='name'
|
||||||
header='Name'
|
header='Name'
|
||||||
|
@ -156,6 +163,19 @@ DataTable.field-editor(
|
||||||
slotProps.editorCancelCallback(e)
|
slotProps.editorCancelCallback(e)
|
||||||
}`
|
}`
|
||||||
)
|
)
|
||||||
|
template(
|
||||||
|
#expansion='slotProps'
|
||||||
|
)
|
||||||
|
div(
|
||||||
|
v-if='slotProps.data.type === "dropdown"'
|
||||||
|
)
|
||||||
|
InputChips(
|
||||||
|
v-model='slotProps.data.options'
|
||||||
|
@dirty='$emit("dirty")'
|
||||||
|
)
|
||||||
|
p(
|
||||||
|
v-else
|
||||||
|
) there are no additional properties for this field type
|
||||||
Button(
|
Button(
|
||||||
icon='pi pi-plus'
|
icon='pi pi-plus'
|
||||||
aria-label='Add New Field'
|
aria-label='Add New Field'
|
||||||
|
@ -165,7 +185,8 @@ Button(
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped lang="sass">
|
<style scoped lang="sass">
|
||||||
:deep(td:last-of-type)
|
:deep(tr:not(.p-datatable-row-expansion) td:last-of-type)
|
||||||
display: flex
|
padding-left: 0
|
||||||
padding-left: 8px
|
button:first-of-type
|
||||||
|
margin-left: -16px
|
||||||
</style>
|
</style>
|
||||||
|
|
Loading…
Add table
Reference in a new issue