|
|
|
<script lang="ts" setup>
|
|
|
|
import type { ColumnType } from 'nocodb-sdk'
|
|
|
|
import { UITypes, isVirtualCol } from 'nocodb-sdk'
|
|
|
|
import { message } from 'ant-design-vue'
|
|
|
|
import {
|
|
|
|
ActiveViewInj,
|
|
|
|
CellUrlDisableOverlayInj,
|
|
|
|
ChangePageInj,
|
|
|
|
FieldsInj,
|
|
|
|
IsFormInj,
|
|
|
|
IsGalleryInj,
|
|
|
|
IsGridInj,
|
|
|
|
IsLockedInj,
|
|
|
|
MetaInj,
|
|
|
|
OpenNewRecordFormHookInj,
|
|
|
|
PaginationDataInj,
|
|
|
|
ReadonlyInj,
|
|
|
|
ReloadViewDataHookInj,
|
|
|
|
createEventHook,
|
|
|
|
extractPkFromRow,
|
|
|
|
inject,
|
|
|
|
onClickOutside,
|
|
|
|
onMounted,
|
|
|
|
provide,
|
|
|
|
reactive,
|
|
|
|
ref,
|
|
|
|
useCopy,
|
|
|
|
useEventListener,
|
|
|
|
useGridViewColumnWidth,
|
|
|
|
useI18n,
|
|
|
|
useRoute,
|
|
|
|
useSmartsheetStoreOrThrow,
|
|
|
|
useUIPermission,
|
|
|
|
useViewData,
|
|
|
|
watch,
|
|
|
|
} from '#imports'
|
|
|
|
import type { Row } from '~/composables'
|
|
|
|
import { NavigateDir } from '~/lib'
|
|
|
|
|
|
|
|
const { t } = useI18n()
|
|
|
|
|
|
|
|
const meta = inject(MetaInj, ref())
|
|
|
|
|
|
|
|
const view = inject(ActiveViewInj, ref())
|
|
|
|
|
|
|
|
// keep a root fields variable and will get modified from
|
|
|
|
// fields menu and get used in grid and gallery
|
|
|
|
const fields = inject(FieldsInj, ref([]))
|
|
|
|
const readOnly = inject(ReadonlyInj, false)
|
|
|
|
const isLocked = inject(IsLockedInj, ref(false))
|
|
|
|
|
|
|
|
const reloadViewDataHook = inject(ReloadViewDataHookInj, createEventHook())
|
|
|
|
const openNewRecordFormHook = inject(OpenNewRecordFormHookInj, createEventHook())
|
|
|
|
|
|
|
|
const { isUIAllowed } = useUIPermission()
|
|
|
|
const hasEditPermission = isUIAllowed('xcDatatableEditable')
|
|
|
|
|
|
|
|
const route = useRoute()
|
|
|
|
const router = useRouter()
|
|
|
|
|
|
|
|
// todo: get from parent ( inject or use prop )
|
|
|
|
const isView = false
|
|
|
|
|
|
|
|
const selected = reactive<{ row: number | null; col: number | null }>({ row: null, col: null })
|
|
|
|
|
|
|
|
let editEnabled = $ref(false)
|
|
|
|
|
|
|
|
const { xWhere, isPkAvail, cellRefs, isSqlView } = useSmartsheetStoreOrThrow()
|
|
|
|
|
|
|
|
const addColumnDropdown = ref(false)
|
|
|
|
|
|
|
|
const _contextMenu = ref(false)
|
|
|
|
const contextMenu = computed({
|
|
|
|
get: () => _contextMenu.value,
|
|
|
|
set: (val) => {
|
|
|
|
if (hasEditPermission) {
|
|
|
|
_contextMenu.value = val
|
|
|
|
}
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
const contextMenuTarget = ref<{ row: number; col: number } | null>(null)
|
|
|
|
const expandedFormDlg = ref(false)
|
|
|
|
const expandedFormRow = ref<Row>()
|
|
|
|
const expandedFormRowState = ref<Record<string, any>>()
|
|
|
|
|
|
|
|
const visibleColLength = $computed(() => fields.value?.length)
|
|
|
|
|
|
|
|
const {
|
|
|
|
isLoading,
|
|
|
|
loadData,
|
|
|
|
paginationData,
|
|
|
|
formattedData: data,
|
|
|
|
updateOrSaveRow,
|
|
|
|
changePage,
|
|
|
|
addEmptyRow,
|
|
|
|
deleteRow,
|
|
|
|
deleteSelectedRows,
|
|
|
|
selectedAllRecords,
|
|
|
|
removeRowIfNew,
|
|
|
|
} = useViewData(meta, view, xWhere)
|
|
|
|
|
|
|
|
const { loadGridViewColumns, updateWidth, resizingColWidth, resizingCol } = useGridViewColumnWidth(view)
|
|
|
|
|
|
|
|
const { copy } = useCopy()
|
|
|
|
|
|
|
|
onMounted(loadGridViewColumns)
|
|
|
|
|
|
|
|
provide(IsFormInj, ref(false))
|
|
|
|
|
|
|
|
provide(IsGalleryInj, ref(false))
|
|
|
|
|
|
|
|
provide(IsGridInj, ref(true))
|
|
|
|
|
|
|
|
provide(PaginationDataInj, paginationData)
|
|
|
|
|
|
|
|
provide(ChangePageInj, changePage)
|
|
|
|
|
|
|
|
provide(ReadonlyInj, !hasEditPermission)
|
|
|
|
|
|
|
|
const disableUrlOverlay = ref(false)
|
|
|
|
provide(CellUrlDisableOverlayInj, disableUrlOverlay)
|
|
|
|
|
|
|
|
const showLoading = ref(true)
|
|
|
|
|
|
|
|
reloadViewDataHook?.on(async (shouldShowLoading) => {
|
|
|
|
// set value if spinner should be hidden
|
|
|
|
showLoading.value = !!shouldShowLoading
|
|
|
|
await loadData()
|
|
|
|
|
|
|
|
// reset to default (showing spinner on load)
|
|
|
|
showLoading.value = true
|
|
|
|
})
|
|
|
|
|
|
|
|
const skipRowRemovalOnCancel = ref(false)
|
|
|
|
|
|
|
|
const expandForm = (row: Row, state?: Record<string, any>, fromToolbar = false) => {
|
|
|
|
if (!isUIAllowed('xcDatatableEditable')) return
|
|
|
|
|
|
|
|
const rowId = extractPkFromRow(row.row, meta.value?.columns as ColumnType[])
|
|
|
|
|
|
|
|
if (rowId) {
|
|
|
|
router.push({
|
|
|
|
query: {
|
|
|
|
...route.query,
|
|
|
|
rowId,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
expandedFormRow.value = row
|
|
|
|
expandedFormRowState.value = state
|
|
|
|
expandedFormDlg.value = true
|
|
|
|
skipRowRemovalOnCancel.value = !fromToolbar
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
openNewRecordFormHook?.on(async () => {
|
|
|
|
const newRow = await addEmptyRow()
|
|
|
|
expandForm(newRow, undefined, true)
|
|
|
|
})
|
|
|
|
|
|
|
|
const selectCell = (row: number, col: number) => {
|
|
|
|
selected.row = row
|
|
|
|
selected.col = col
|
|
|
|
}
|
|
|
|
|
|
|
|
watch(
|
|
|
|
() => view.value?.id,
|
|
|
|
async (next, old) => {
|
|
|
|
if (next && old && next !== old) {
|
|
|
|
await loadData()
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{ immediate: true },
|
|
|
|
)
|
|
|
|
|
|
|
|
const onresize = (colID: string, event: any) => {
|
|
|
|
updateWidth(colID, event.detail)
|
|
|
|
}
|
|
|
|
|
|
|
|
const onXcResizing = (cn: string, event: any) => {
|
|
|
|
resizingCol.value = cn
|
|
|
|
resizingColWidth.value = event.detail
|
|
|
|
}
|
|
|
|
|
|
|
|
defineExpose({
|
|
|
|
loadData,
|
|
|
|
})
|
|
|
|
|
|
|
|
// reset context menu target on hide
|
|
|
|
watch(contextMenu, () => {
|
|
|
|
if (!contextMenu.value) {
|
|
|
|
contextMenuTarget.value = null
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const clearCell = async (ctx: { row: number; col: number }) => {
|
|
|
|
const rowObj = data.value[ctx.row]
|
|
|
|
const columnObj = fields.value[ctx.col]
|
|
|
|
|
|
|
|
if (isVirtualCol(columnObj)) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
rowObj.row[columnObj.title] = null
|
|
|
|
// update/save cell value
|
|
|
|
await updateOrSaveRow(rowObj, columnObj.title)
|
|
|
|
}
|
|
|
|
|
|
|
|
const makeEditable = (row: Row, col: ColumnType) => {
|
|
|
|
if (!hasEditPermission || editEnabled || isView) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if (!isPkAvail.value && !row.rowMeta.new) {
|
|
|
|
// Update not allowed for table which doesn't have primary Key
|
|
|
|
message.info(t('msg.info.updateNotAllowedWithoutPK'))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if (col.ai) {
|
|
|
|
// Auto Increment field is not editable
|
|
|
|
message.info(t('msg.info.autoIncFieldNotEditable'))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if (col.pk && !row.rowMeta.new) {
|
|
|
|
// Editing primary key not supported
|
|
|
|
message.info(t('msg.info.editingPKnotSupported'))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
return (editEnabled = true)
|
|
|
|
}
|
|
|
|
|
|
|
|
/** handle keypress events */
|
|
|
|
const onKeyDown = async (e: KeyboardEvent) => {
|
|
|
|
if (e.key === 'Alt') {
|
|
|
|
disableUrlOverlay.value = true
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if (selected.row === null || selected.col === null) return
|
|
|
|
/** on tab key press navigate through cells */
|
|
|
|
switch (e.key) {
|
|
|
|
case 'Tab':
|
|
|
|
e.preventDefault()
|
|
|
|
if (e.shiftKey) {
|
|
|
|
if (selected.col > 0) {
|
|
|
|
selected.col--
|
|
|
|
} else if (selected.row > 0) {
|
|
|
|
selected.row--
|
|
|
|
selected.col = visibleColLength - 1
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (selected.col < visibleColLength - 1) {
|
|
|
|
selected.col++
|
|
|
|
} else if (selected.row < data.value.length - 1) {
|
|
|
|
selected.row++
|
|
|
|
selected.col = 0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break
|
|
|
|
/** on enter key press make cell editable */
|
|
|
|
case 'Enter':
|
|
|
|
e.preventDefault()
|
|
|
|
makeEditable(data.value[selected.row], fields.value[selected.col])
|
|
|
|
break
|
|
|
|
/** on delete key press clear cell */
|
|
|
|
case 'Delete':
|
|
|
|
if (!editEnabled) {
|
|
|
|
e.preventDefault()
|
|
|
|
await clearCell(selected as { row: number; col: number })
|
|
|
|
}
|
|
|
|
break
|
|
|
|
/** on arrow key press navigate through cells */
|
|
|
|
case 'ArrowRight':
|
|
|
|
e.preventDefault()
|
|
|
|
if (selected.col < visibleColLength - 1) selected.col++
|
|
|
|
break
|
|
|
|
case 'ArrowLeft':
|
|
|
|
e.preventDefault()
|
|
|
|
if (selected.col > 0) selected.col--
|
|
|
|
break
|
|
|
|
case 'ArrowUp':
|
|
|
|
e.preventDefault()
|
|
|
|
if (selected.row > 0) selected.row--
|
|
|
|
break
|
|
|
|
case 'ArrowDown':
|
|
|
|
e.preventDefault()
|
|
|
|
if (selected.row < data.value.length - 1) selected.row++
|
|
|
|
break
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
const rowObj = data.value[selected.row]
|
|
|
|
const columnObj = fields.value[selected.col]
|
|
|
|
|
|
|
|
if ((!editEnabled && e.metaKey) || e.ctrlKey) {
|
|
|
|
switch (e.keyCode) {
|
|
|
|
// copy - ctrl/cmd +c
|
|
|
|
case 67:
|
|
|
|
await copy(rowObj.row[columnObj.title] || '')
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (editEnabled || e.ctrlKey || e.altKey || e.metaKey) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
/** on letter key press make cell editable and empty */
|
|
|
|
if (e?.key?.length === 1) {
|
|
|
|
if (!isPkAvail && !rowObj.rowMeta.new) {
|
|
|
|
// Update not allowed for table which doesn't have primary Key
|
|
|
|
return message.info(t('msg.info.updateNotAllowedWithoutPK'))
|
|
|
|
}
|
|
|
|
if (makeEditable(rowObj, columnObj)) {
|
|
|
|
rowObj.row[columnObj.title] = ''
|
|
|
|
}
|
|
|
|
// editEnabled = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const onKeyUp = async (e: KeyboardEvent) => {
|
|
|
|
if (e.key === 'Alt') {
|
|
|
|
disableUrlOverlay.value = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
useEventListener(document, 'keydown', onKeyDown)
|
|
|
|
useEventListener(document, 'keyup', onKeyUp)
|
|
|
|
|
|
|
|
/** On clicking outside of table reset active cell */
|
|
|
|
const smartTable = ref(null)
|
|
|
|
onClickOutside(smartTable, () => {
|
|
|
|
if (selected.col === null) return
|
|
|
|
|
|
|
|
const activeCol = fields.value[selected.col]
|
|
|
|
|
|
|
|
if (editEnabled && (isVirtualCol(activeCol) || activeCol.uidt === UITypes.JSON)) return
|
|
|
|
|
|
|
|
selected.row = null
|
|
|
|
selected.col = null
|
|
|
|
})
|
|
|
|
|
|
|
|
const onNavigate = (dir: NavigateDir) => {
|
|
|
|
if (selected.row === null || selected.col === null) return
|
|
|
|
switch (dir) {
|
|
|
|
case NavigateDir.NEXT:
|
|
|
|
if (selected.row < data.value.length - 1) {
|
|
|
|
selected.row++
|
|
|
|
} else {
|
|
|
|
editEnabled = false
|
|
|
|
}
|
|
|
|
break
|
|
|
|
case NavigateDir.PREV:
|
|
|
|
if (selected.row > 0) {
|
|
|
|
selected.row--
|
|
|
|
} else {
|
|
|
|
editEnabled = false
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const showContextMenu = (e: MouseEvent, target?: { row: number; col: number }) => {
|
|
|
|
if (isSqlView.value) return
|
|
|
|
e.preventDefault()
|
|
|
|
if (target) {
|
|
|
|
contextMenuTarget.value = target
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const rowRefs = $ref<any[]>()
|
|
|
|
|
|
|
|
/** save/update records before unmounting the component */
|
|
|
|
onBeforeUnmount(async () => {
|
|
|
|
let index = -1
|
|
|
|
for (const currentRow of data.value) {
|
|
|
|
index++
|
|
|
|
/** if new record save row and save the LTAR cells */
|
|
|
|
if (currentRow.rowMeta.new) {
|
|
|
|
const syncLTARRefs = rowRefs[index]!.syncLTARRefs
|
|
|
|
const savedRow = await updateOrSaveRow(currentRow, '')
|
|
|
|
await syncLTARRefs(savedRow)
|
|
|
|
currentRow.rowMeta.changed = false
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
/** if existing row check updated cell and invoke update method */
|
|
|
|
if (currentRow.rowMeta.changed) {
|
|
|
|
currentRow.rowMeta.changed = false
|
|
|
|
for (const field of meta.value?.columns ?? []) {
|
|
|
|
if (isVirtualCol(field)) continue
|
|
|
|
if (currentRow.row[field.title!] !== currentRow.oldRow[field.title!]) {
|
|
|
|
await updateOrSaveRow(currentRow, field.title!)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const expandedFormOnRowIdDlg = computed({
|
|
|
|
get() {
|
|
|
|
return !!route.query.rowId
|
|
|
|
},
|
|
|
|
set(val) {
|
|
|
|
if (!val)
|
|
|
|
router.push({
|
|
|
|
query: {
|
|
|
|
...route.query,
|
|
|
|
rowId: undefined,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
// reload table data reload hook as fallback to rowdatareload
|
|
|
|
provide(ReloadRowDataHookInj, reloadViewDataHook)
|
|
|
|
|
|
|
|
// trigger initial data load in grid
|
|
|
|
reloadViewDataHook.trigger()
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<template>
|
|
|
|
<div class="flex flex-col h-full min-h-0 w-full">
|
|
|
|
<general-overlay :model-value="isLoading" inline transition class="!bg-opacity-15">
|
|
|
|
<div class="flex items-center justify-center h-full w-full">
|
|
|
|
<a-spin size="large" />
|
|
|
|
</div>
|
|
|
|
</general-overlay>
|
|
|
|
<div class="nc-grid-wrapper min-h-0 flex-1 scrollbar-thin-dull">
|
|
|
|
<a-dropdown
|
|
|
|
v-model:visible="contextMenu"
|
|
|
|
:trigger="isSqlView ? [] : ['contextmenu']"
|
|
|
|
overlay-class-name="nc-dropdown-grid-context-menu"
|
|
|
|
>
|
|
|
|
<table
|
|
|
|
ref="smartTable"
|
|
|
|
class="xc-row-table nc-grid backgroundColorDefault !h-auto bg-white"
|
|
|
|
@contextmenu="showContextMenu"
|
|
|
|
>
|
|
|
|
<thead>
|
|
|
|
<tr class="nc-grid-header border-1 bg-gray-100 sticky top[-1px]">
|
|
|
|
<th>
|
|
|
|
<div class="w-full h-full bg-gray-100 flex min-w-[70px] pl-5 pr-1 items-center">
|
|
|
|
<template v-if="!readOnly">
|
|
|
|
<div class="nc-no-label text-gray-500" :class="{ hidden: selectedAllRecords }">#</div>
|
|
|
|
<div
|
|
|
|
:class="{ hidden: !selectedAllRecords, flex: selectedAllRecords }"
|
|
|
|
class="nc-check-all w-full items-center"
|
|
|
|
>
|
|
|
|
<a-checkbox v-model:checked="selectedAllRecords" />
|
|
|
|
|
|
|
|
<span class="flex-1" />
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
<template v-else>
|
|
|
|
<div class="text-gray-500">#</div>
|
|
|
|
</template>
|
|
|
|
</div>
|
|
|
|
</th>
|
|
|
|
<th
|
|
|
|
v-for="col in fields"
|
|
|
|
:key="col.title"
|
|
|
|
v-xc-ver-resize
|
|
|
|
:data-col="col.id"
|
|
|
|
:data-title="col.title"
|
|
|
|
@xcresize="onresize(col.id, $event)"
|
|
|
|
@xcresizing="onXcResizing(col.title, $event)"
|
|
|
|
@xcresized="resizingCol = null"
|
|
|
|
>
|
|
|
|
<div class="w-full h-full bg-gray-100 flex items-center">
|
|
|
|
<SmartsheetHeaderVirtualCell v-if="isVirtualCol(col)" :column="col" :hide-menu="readOnly" />
|
|
|
|
|
|
|
|
<SmartsheetHeaderCell v-else :column="col" :hide-menu="readOnly" />
|
|
|
|
</div>
|
|
|
|
</th>
|
|
|
|
<th
|
|
|
|
v-if="!readOnly && !isLocked && isUIAllowed('add-column') && !isSqlView"
|
|
|
|
v-e="['c:column:add']"
|
|
|
|
class="cursor-pointer"
|
|
|
|
@click.stop="addColumnDropdown = true"
|
|
|
|
>
|
|
|
|
<a-dropdown
|
|
|
|
v-model:visible="addColumnDropdown"
|
|
|
|
:trigger="['click']"
|
|
|
|
overlay-class-name="nc-dropdown-grid-add-column"
|
|
|
|
>
|
|
|
|
<div class="h-full w-[60px] flex items-center justify-center">
|
|
|
|
<MdiPlus class="text-sm nc-column-add" />
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<template #overlay>
|
|
|
|
<SmartsheetColumnEditOrAddProvider
|
|
|
|
v-if="addColumnDropdown"
|
|
|
|
@submit="addColumnDropdown = false"
|
|
|
|
@cancel="addColumnDropdown = false"
|
|
|
|
@click.stop
|
|
|
|
@keydown.stop
|
|
|
|
/>
|
|
|
|
</template>
|
|
|
|
</a-dropdown>
|
|
|
|
</th>
|
|
|
|
</tr>
|
|
|
|
</thead>
|
|
|
|
<tbody>
|
|
|
|
<SmartsheetRow v-for="(row, rowIndex) of data" ref="rowRefs" :key="rowIndex" :row="row">
|
|
|
|
<template #default="{ state }">
|
|
|
|
<tr class="nc-grid-row">
|
|
|
|
<td key="row-index" class="caption nc-grid-cell pl-5 pr-1">
|
|
|
|
<div class="items-center flex gap-1 min-w-[55px]">
|
|
|
|
<div
|
|
|
|
v-if="!readOnly || !isLocked"
|
|
|
|
class="nc-row-no text-xs text-gray-500"
|
|
|
|
:class="{ toggle: !readOnly, hidden: row.rowMeta.selected }"
|
|
|
|
>
|
|
|
|
{{ rowIndex + 1 }}
|
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
v-if="!readOnly"
|
|
|
|
:class="{ hidden: !row.rowMeta.selected, flex: row.rowMeta.selected }"
|
|
|
|
class="nc-row-expand-and-checkbox"
|
|
|
|
>
|
|
|
|
<a-checkbox v-model:checked="row.rowMeta.selected" />
|
|
|
|
</div>
|
|
|
|
<span class="flex-1" />
|
|
|
|
<div v-if="!readOnly && !isLocked" class="nc-expand" :class="{ 'nc-comment': row.rowMeta?.commentCount }">
|
|
|
|
<span
|
|
|
|
v-if="row.rowMeta?.commentCount"
|
|
|
|
class="py-1 px-3 rounded-full text-xs cursor-pointer select-none transform hover:(scale-110)"
|
|
|
|
:style="{ backgroundColor: enumColor.light[row.rowMeta.commentCount % enumColor.light.length] }"
|
|
|
|
@click="expandForm(row, state)"
|
|
|
|
>
|
|
|
|
{{ row.rowMeta.commentCount }}
|
|
|
|
</span>
|
|
|
|
<div
|
|
|
|
v-else
|
|
|
|
class="cursor-pointer flex items-center border-1 active:ring rounded p-1 hover:(bg-primary bg-opacity-10)"
|
|
|
|
>
|
|
|
|
<MdiArrowExpand
|
|
|
|
v-e="['c:row-expand']"
|
|
|
|
class="select-none transform hover:(text-accent scale-120) nc-row-expand"
|
|
|
|
@click="expandForm(row, state)"
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</td>
|
|
|
|
<td
|
|
|
|
v-for="(columnObj, colIndex) of fields"
|
|
|
|
:ref="cellRefs.set"
|
|
|
|
:key="columnObj.id"
|
|
|
|
class="cell relative cursor-pointer nc-grid-cell"
|
|
|
|
:class="{
|
|
|
|
active: isUIAllowed('xcDatatableEditable') && selected.col === colIndex && selected.row === rowIndex,
|
|
|
|
}"
|
|
|
|
:data-key="rowIndex + columnObj.id"
|
|
|
|
:data-col="columnObj.id"
|
|
|
|
:data-title="columnObj.title"
|
|
|
|
@click="selectCell(rowIndex, colIndex)"
|
|
|
|
@dblclick="makeEditable(row, columnObj)"
|
|
|
|
@contextmenu="showContextMenu($event, { row: rowIndex, col: colIndex })"
|
|
|
|
>
|
|
|
|
<div class="w-full h-full">
|
|
|
|
<SmartsheetVirtualCell
|
|
|
|
v-if="isVirtualCol(columnObj)"
|
|
|
|
v-model="row.row[columnObj.title]"
|
|
|
|
:column="columnObj"
|
|
|
|
:active="selected.col === colIndex && selected.row === rowIndex"
|
|
|
|
:row="row"
|
|
|
|
@navigate="onNavigate"
|
|
|
|
/>
|
|
|
|
|
|
|
|
<SmartsheetCell
|
|
|
|
v-else
|
|
|
|
v-model="row.row[columnObj.title]"
|
|
|
|
:column="columnObj"
|
|
|
|
:edit-enabled="
|
|
|
|
isUIAllowed('xcDatatableEditable') &&
|
|
|
|
editEnabled &&
|
|
|
|
selected.col === colIndex &&
|
|
|
|
selected.row === rowIndex
|
|
|
|
"
|
|
|
|
:row-index="rowIndex"
|
|
|
|
:active="selected.col === colIndex && selected.row === rowIndex"
|
|
|
|
@update:edit-enabled="editEnabled = false"
|
|
|
|
@save="updateOrSaveRow(row, columnObj.title)"
|
|
|
|
@navigate="onNavigate"
|
|
|
|
@cancel="editEnabled = false"
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
</template>
|
|
|
|
</SmartsheetRow>
|
|
|
|
|
|
|
|
<tr v-if="!isView && !isLocked && isUIAllowed('xcDatatableEditable') && !isSqlView">
|
|
|
|
<td
|
|
|
|
v-e="['c:row:add:grid-bottom']"
|
|
|
|
:colspan="visibleColLength + 1"
|
|
|
|
class="text-left pointer nc-grid-add-new-cell cursor-pointer"
|
|
|
|
@click="addEmptyRow()"
|
|
|
|
>
|
|
|
|
<div class="px-2 w-full flex items-center text-gray-500">
|
|
|
|
<MdiPlus class="text-pint-500 text-xs ml-2 text-primary" />
|
|
|
|
|
|
|
|
<span class="ml-1">
|
|
|
|
{{ $t('activity.addRow') }}
|
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
</tbody>
|
|
|
|
</table>
|
|
|
|
|
|
|
|
<template v-if="!isLocked && isUIAllowed('xcDatatableEditable')" #overlay>
|
|
|
|
<a-menu class="shadow !rounded !py-0" @click="contextMenu = false">
|
|
|
|
<a-menu-item v-if="contextMenuTarget" @click="deleteRow(contextMenuTarget.row)">
|
|
|
|
<div v-e="['a:row:delete']" class="nc-project-menu-item">
|
|
|
|
<!-- Delete Row -->
|
|
|
|
{{ $t('activity.deleteRow') }}
|
|
|
|
</div>
|
|
|
|
</a-menu-item>
|
|
|
|
|
|
|
|
<a-menu-item @click="deleteSelectedRows">
|
|
|
|
<div v-e="['a:row:delete-bulk']" class="nc-project-menu-item">
|
|
|
|
<!-- Delete Selected Rows -->
|
|
|
|
{{ $t('activity.deleteSelectedRow') }}
|
|
|
|
</div>
|
|
|
|
</a-menu-item>
|
|
|
|
|
|
|
|
<!-- Clear cell -->
|
|
|
|
<a-menu-item v-if="contextMenuTarget" @click="clearCell(contextMenuTarget)">
|
|
|
|
<div v-e="['a:row:clear']" class="nc-project-menu-item">{{ $t('activity.clearCell') }}</div>
|
|
|
|
</a-menu-item>
|
|
|
|
|
|
|
|
<a-menu-item v-if="contextMenuTarget" @click="addEmptyRow(contextMenuTarget.row + 1)">
|
|
|
|
<div v-e="['a:row:insert']" class="nc-project-menu-item">
|
|
|
|
<!-- Insert New Row -->
|
|
|
|
{{ $t('activity.insertRow') }}
|
|
|
|
</div>
|
|
|
|
</a-menu-item>
|
|
|
|
</a-menu>
|
|
|
|
</template>
|
|
|
|
</a-dropdown>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<SmartsheetPagination />
|
|
|
|
|
|
|
|
<SmartsheetExpandedForm
|
|
|
|
v-if="expandedFormRow && expandedFormDlg"
|
|
|
|
v-model="expandedFormDlg"
|
|
|
|
:row="expandedFormRow"
|
|
|
|
:state="expandedFormRowState"
|
|
|
|
:meta="meta"
|
|
|
|
:view="view"
|
|
|
|
@update:model-value="
|
|
|
|
() => {
|
|
|
|
if (!skipRowRemovalOnCancel) removeRowIfNew(expandedFormRow)
|
|
|
|
}
|
|
|
|
"
|
|
|
|
/>
|
|
|
|
|
|
|
|
<SmartsheetExpandedForm
|
|
|
|
v-if="expandedFormOnRowIdDlg"
|
|
|
|
:key="route.query.rowId"
|
|
|
|
v-model="expandedFormOnRowIdDlg"
|
|
|
|
:row="{ row: {}, oldRow: {}, rowMeta: {} }"
|
|
|
|
:meta="meta"
|
|
|
|
:row-id="route.query.rowId"
|
|
|
|
:view="view"
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<style scoped lang="scss">
|
|
|
|
.nc-grid-wrapper {
|
|
|
|
@apply h-full w-full overflow-auto;
|
|
|
|
|
|
|
|
td,
|
|
|
|
th {
|
|
|
|
min-height: 41px !important;
|
|
|
|
height: 41px !important;
|
|
|
|
position: relative;
|
|
|
|
}
|
|
|
|
|
|
|
|
td:not(:first-child) > div {
|
|
|
|
overflow: hidden;
|
|
|
|
@apply flex items-center h-auto px-1;
|
|
|
|
}
|
|
|
|
|
|
|
|
table,
|
|
|
|
td,
|
|
|
|
th {
|
|
|
|
@apply !border-1;
|
|
|
|
border-collapse: collapse;
|
|
|
|
}
|
|
|
|
|
|
|
|
td {
|
|
|
|
text-overflow: ellipsis;
|
|
|
|
}
|
|
|
|
|
|
|
|
td.active::after,
|
|
|
|
td.active::before {
|
|
|
|
content: '';
|
|
|
|
position: absolute;
|
|
|
|
z-index: 3;
|
|
|
|
height: calc(100% + 2px);
|
|
|
|
width: calc(100% + 2px);
|
|
|
|
left: -1px;
|
|
|
|
top: -1px;
|
|
|
|
pointer-events: none;
|
|
|
|
}
|
|
|
|
|
|
|
|
// todo: replace with css variable
|
|
|
|
td.active::after {
|
|
|
|
@apply border-2 border-solid border-primary;
|
|
|
|
}
|
|
|
|
|
|
|
|
td.active::before {
|
|
|
|
@apply bg-primary bg-opacity-5;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
:deep {
|
|
|
|
.resizer:hover,
|
|
|
|
.resizer:active,
|
|
|
|
.resizer:focus {
|
|
|
|
// todo: replace with primary color
|
|
|
|
@apply bg-blue-500/50;
|
|
|
|
cursor: col-resize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.nc-grid-row {
|
|
|
|
.nc-row-expand-and-checkbox {
|
|
|
|
@apply w-full items-center justify-between;
|
|
|
|
}
|
|
|
|
|
|
|
|
.nc-expand {
|
|
|
|
&:not(.nc-comment) {
|
|
|
|
@apply hidden;
|
|
|
|
}
|
|
|
|
|
|
|
|
&.nc-comment {
|
|
|
|
display: flex;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
&:hover {
|
|
|
|
.nc-row-no.toggle {
|
|
|
|
@apply hidden;
|
|
|
|
}
|
|
|
|
|
|
|
|
.nc-expand {
|
|
|
|
@apply flex;
|
|
|
|
}
|
|
|
|
|
|
|
|
.nc-row-expand-and-checkbox {
|
|
|
|
@apply flex;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.nc-grid-header {
|
|
|
|
position: sticky;
|
|
|
|
top: -1px;
|
|
|
|
|
|
|
|
@apply z-1;
|
|
|
|
|
|
|
|
&:hover {
|
|
|
|
.nc-no-label {
|
|
|
|
@apply hidden;
|
|
|
|
}
|
|
|
|
|
|
|
|
.nc-check-all {
|
|
|
|
@apply flex;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tbody tr:hover {
|
|
|
|
@apply bg-gray-100 bg-opacity-50;
|
|
|
|
}
|
|
|
|
</style>
|