|
|
@ -156,8 +156,8 @@ const getContainerScrollForElement = ( |
|
|
|
relativePos.right + (offset?.right || 0) > 0 |
|
|
|
relativePos.right + (offset?.right || 0) > 0 |
|
|
|
? container.scrollLeft + relativePos.right + (offset?.right || 0) |
|
|
|
? container.scrollLeft + relativePos.right + (offset?.right || 0) |
|
|
|
: relativePos.left - (offset?.left || 0) < 0 |
|
|
|
: relativePos.left - (offset?.left || 0) < 0 |
|
|
|
? container.scrollLeft + relativePos.left - (offset?.left || 0) |
|
|
|
? container.scrollLeft + relativePos.left - (offset?.left || 0) |
|
|
|
: container.scrollLeft |
|
|
|
: container.scrollLeft |
|
|
|
|
|
|
|
|
|
|
|
/* |
|
|
|
/* |
|
|
|
* If the element is below the container, scroll down (positive) |
|
|
|
* If the element is below the container, scroll down (positive) |
|
|
@ -167,134 +167,144 @@ const getContainerScrollForElement = ( |
|
|
|
relativePos.bottom + (offset?.bottom || 0) > 0 |
|
|
|
relativePos.bottom + (offset?.bottom || 0) > 0 |
|
|
|
? container.scrollTop + relativePos.bottom + (offset?.bottom || 0) |
|
|
|
? container.scrollTop + relativePos.bottom + (offset?.bottom || 0) |
|
|
|
: relativePos.top - (offset?.top || 0) < 0 |
|
|
|
: relativePos.top - (offset?.top || 0) < 0 |
|
|
|
? container.scrollTop + relativePos.top - (offset?.top || 0) |
|
|
|
? container.scrollTop + relativePos.top - (offset?.top || 0) |
|
|
|
: container.scrollTop |
|
|
|
: container.scrollTop |
|
|
|
|
|
|
|
|
|
|
|
return scroll |
|
|
|
return scroll |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
const { isCellSelected, activeCell, handleMouseDown, handleMouseOver, handleCellClick, clearSelectedRange, copyValue } = |
|
|
|
const { |
|
|
|
useMultiSelect( |
|
|
|
isCellSelected, |
|
|
|
meta, |
|
|
|
activeCell, |
|
|
|
fields, |
|
|
|
handleMouseDown, |
|
|
|
data, |
|
|
|
handleMouseOver, |
|
|
|
$$(editEnabled), |
|
|
|
handleCellClick, |
|
|
|
isPkAvail, |
|
|
|
clearSelectedRange, |
|
|
|
clearCell, |
|
|
|
copyValue, |
|
|
|
makeEditable, |
|
|
|
isCellActive, |
|
|
|
scrollToCell, |
|
|
|
} = useMultiSelect( |
|
|
|
(e: KeyboardEvent) => { |
|
|
|
meta, |
|
|
|
// ignore navigating if picker(Date, Time, DateTime, Year) |
|
|
|
fields, |
|
|
|
// or single/multi select options is open |
|
|
|
data, |
|
|
|
const activePickerOrDropdownEl = document.querySelector( |
|
|
|
$$(editEnabled), |
|
|
|
'.nc-picker-datetime.active,.nc-dropdown-single-select-cell.active,.nc-dropdown-multi-select-cell.active,.nc-picker-date.active,.nc-picker-year.active,.nc-picker-time.active', |
|
|
|
isPkAvail, |
|
|
|
) |
|
|
|
clearCell, |
|
|
|
if (activePickerOrDropdownEl) { |
|
|
|
makeEditable, |
|
|
|
|
|
|
|
scrollToCell, |
|
|
|
|
|
|
|
(e: KeyboardEvent) => { |
|
|
|
|
|
|
|
// ignore navigating if picker(Date, Time, DateTime, Year) |
|
|
|
|
|
|
|
// or single/multi select options is open |
|
|
|
|
|
|
|
const activePickerOrDropdownEl = document.querySelector( |
|
|
|
|
|
|
|
'.nc-picker-datetime.active,.nc-dropdown-single-select-cell.active,.nc-dropdown-multi-select-cell.active,.nc-picker-date.active,.nc-picker-year.active,.nc-picker-time.active', |
|
|
|
|
|
|
|
) |
|
|
|
|
|
|
|
if (activePickerOrDropdownEl) { |
|
|
|
|
|
|
|
e.preventDefault() |
|
|
|
|
|
|
|
return true |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// skip keyboard event handling if there is a drawer / modal |
|
|
|
|
|
|
|
if (isDrawerOrModalExist()) { |
|
|
|
|
|
|
|
return true |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const cmdOrCtrl = isMac() ? e.metaKey : e.ctrlKey |
|
|
|
|
|
|
|
const altOrOptionKey = e.altKey |
|
|
|
|
|
|
|
if (e.key === ' ') { |
|
|
|
|
|
|
|
if (isCellActive.value && !editEnabled && hasEditPermission) { |
|
|
|
e.preventDefault() |
|
|
|
e.preventDefault() |
|
|
|
|
|
|
|
clearSelectedRange() |
|
|
|
|
|
|
|
const row = data.value[activeCell.row] |
|
|
|
|
|
|
|
expandForm(row) |
|
|
|
return true |
|
|
|
return true |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
} else if (e.key === 'Escape') { |
|
|
|
// skip keyboard event handling if there is a drawer / modal |
|
|
|
if (editEnabled) { |
|
|
|
if (isDrawerOrModalExist()) { |
|
|
|
editEnabled = false |
|
|
|
|
|
|
|
return true |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} else if (e.key === 'Enter') { |
|
|
|
|
|
|
|
if (e.shiftKey) { |
|
|
|
|
|
|
|
// add a line break for types like LongText / JSON |
|
|
|
|
|
|
|
return true |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if (editEnabled) { |
|
|
|
|
|
|
|
editEnabled = false |
|
|
|
return true |
|
|
|
return true |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (cmdOrCtrl) { |
|
|
|
|
|
|
|
if (!isCellActive.value) return |
|
|
|
|
|
|
|
|
|
|
|
const cmdOrCtrl = isMac() ? e.metaKey : e.ctrlKey |
|
|
|
switch (e.key) { |
|
|
|
const altOrOptionKey = e.altKey |
|
|
|
case 'ArrowUp': |
|
|
|
if (e.key === ' ') { |
|
|
|
|
|
|
|
if (activeCell.row != null && !editEnabled && hasEditPermission) { |
|
|
|
|
|
|
|
e.preventDefault() |
|
|
|
e.preventDefault() |
|
|
|
clearSelectedRange() |
|
|
|
clearSelectedRange() |
|
|
|
const row = data.value[activeCell.row] |
|
|
|
activeCell.row = 0 |
|
|
|
expandForm(row) |
|
|
|
activeCell.col = activeCell.col ?? 0 |
|
|
|
|
|
|
|
scrollToCell?.() |
|
|
|
|
|
|
|
editEnabled = false |
|
|
|
return true |
|
|
|
return true |
|
|
|
} |
|
|
|
case 'ArrowDown': |
|
|
|
} else if (e.key === 'Escape') { |
|
|
|
e.preventDefault() |
|
|
|
if (editEnabled) { |
|
|
|
clearSelectedRange() |
|
|
|
|
|
|
|
activeCell.row = data.value.length - 1 |
|
|
|
|
|
|
|
activeCell.col = activeCell.col ?? 0 |
|
|
|
|
|
|
|
scrollToCell?.() |
|
|
|
editEnabled = false |
|
|
|
editEnabled = false |
|
|
|
return true |
|
|
|
return true |
|
|
|
} |
|
|
|
case 'ArrowRight': |
|
|
|
} else if (e.key === 'Enter') { |
|
|
|
e.preventDefault() |
|
|
|
if (e.shiftKey) { |
|
|
|
clearSelectedRange() |
|
|
|
// add a line break for types like LongText / JSON |
|
|
|
activeCell.row = activeCell.row ?? 0 |
|
|
|
|
|
|
|
activeCell.col = fields.value?.length - 1 |
|
|
|
|
|
|
|
scrollToCell?.() |
|
|
|
|
|
|
|
editEnabled = false |
|
|
|
return true |
|
|
|
return true |
|
|
|
} |
|
|
|
case 'ArrowLeft': |
|
|
|
if (editEnabled) { |
|
|
|
e.preventDefault() |
|
|
|
|
|
|
|
clearSelectedRange() |
|
|
|
|
|
|
|
activeCell.row = activeCell.row ?? 0 |
|
|
|
|
|
|
|
activeCell.col = 0 |
|
|
|
|
|
|
|
scrollToCell?.() |
|
|
|
editEnabled = false |
|
|
|
editEnabled = false |
|
|
|
return true |
|
|
|
return true |
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (cmdOrCtrl) { |
|
|
|
|
|
|
|
switch (e.key) { |
|
|
|
|
|
|
|
case 'ArrowUp': |
|
|
|
|
|
|
|
e.preventDefault() |
|
|
|
|
|
|
|
clearSelectedRange() |
|
|
|
|
|
|
|
activeCell.row = 0 |
|
|
|
|
|
|
|
activeCell.col = activeCell.col ?? 0 |
|
|
|
|
|
|
|
scrollToCell?.() |
|
|
|
|
|
|
|
editEnabled = false |
|
|
|
|
|
|
|
return true |
|
|
|
|
|
|
|
case 'ArrowDown': |
|
|
|
|
|
|
|
e.preventDefault() |
|
|
|
|
|
|
|
clearSelectedRange() |
|
|
|
|
|
|
|
activeCell.row = data.value.length - 1 |
|
|
|
|
|
|
|
activeCell.col = activeCell.col ?? 0 |
|
|
|
|
|
|
|
scrollToCell?.() |
|
|
|
|
|
|
|
editEnabled = false |
|
|
|
|
|
|
|
return true |
|
|
|
|
|
|
|
case 'ArrowRight': |
|
|
|
|
|
|
|
e.preventDefault() |
|
|
|
|
|
|
|
clearSelectedRange() |
|
|
|
|
|
|
|
activeCell.row = activeCell.row ?? 0 |
|
|
|
|
|
|
|
activeCell.col = fields.value?.length - 1 |
|
|
|
|
|
|
|
scrollToCell?.() |
|
|
|
|
|
|
|
editEnabled = false |
|
|
|
|
|
|
|
return true |
|
|
|
|
|
|
|
case 'ArrowLeft': |
|
|
|
|
|
|
|
e.preventDefault() |
|
|
|
|
|
|
|
clearSelectedRange() |
|
|
|
|
|
|
|
activeCell.row = activeCell.row ?? 0 |
|
|
|
|
|
|
|
activeCell.col = 0 |
|
|
|
|
|
|
|
scrollToCell?.() |
|
|
|
|
|
|
|
editEnabled = false |
|
|
|
|
|
|
|
return true |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
if (altOrOptionKey) { |
|
|
|
if (altOrOptionKey) { |
|
|
|
switch (e.keyCode) { |
|
|
|
switch (e.keyCode) { |
|
|
|
case 82: { |
|
|
|
case 82: { |
|
|
|
// ALT + R |
|
|
|
// ALT + R |
|
|
|
if (isAddingEmptyRowAllowed) { |
|
|
|
if (isAddingEmptyRowAllowed) { |
|
|
|
$e('c:shortcut', { key: 'ALT + R' }) |
|
|
|
$e('c:shortcut', { key: 'ALT + R' }) |
|
|
|
addEmptyRow() |
|
|
|
addEmptyRow() |
|
|
|
} |
|
|
|
|
|
|
|
break |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
case 67: { |
|
|
|
break |
|
|
|
// ALT + C |
|
|
|
} |
|
|
|
if (isAddingColumnAllowed) { |
|
|
|
case 67: { |
|
|
|
$e('c:shortcut', { key: 'ALT + C' }) |
|
|
|
// ALT + C |
|
|
|
addColumnDropdown.value = true |
|
|
|
if (isAddingColumnAllowed) { |
|
|
|
} |
|
|
|
$e('c:shortcut', { key: 'ALT + C' }) |
|
|
|
break |
|
|
|
addColumnDropdown.value = true |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
break |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
}, |
|
|
|
} |
|
|
|
async (ctx: { row: number; col?: number; updatedColumnTitle?: string }) => { |
|
|
|
}, |
|
|
|
const rowObj = data.value[ctx.row] |
|
|
|
async (ctx: { row: number; col?: number; updatedColumnTitle?: string }) => { |
|
|
|
const columnObj = ctx.col !== undefined ? fields.value[ctx.col] : null |
|
|
|
const rowObj = data.value[ctx.row] |
|
|
|
|
|
|
|
const columnObj = ctx.col !== undefined ? fields.value[ctx.col] : null |
|
|
|
|
|
|
|
|
|
|
|
if (!ctx.updatedColumnTitle && isVirtualCol(columnObj)) { |
|
|
|
if (!ctx.updatedColumnTitle && isVirtualCol(columnObj)) { |
|
|
|
return |
|
|
|
return |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// update/save cell value |
|
|
|
// update/save cell value |
|
|
|
await updateOrSaveRow(rowObj, ctx.updatedColumnTitle || columnObj.title) |
|
|
|
await updateOrSaveRow(rowObj, ctx.updatedColumnTitle || columnObj.title) |
|
|
|
}, |
|
|
|
}, |
|
|
|
) |
|
|
|
) |
|
|
|
|
|
|
|
|
|
|
|
function scrollToCell(row?: number | null, col?: number | null) { |
|
|
|
function scrollToCell(row?: number | null, col?: number | null) { |
|
|
|
row = row ?? activeCell.row |
|
|
|
row = row ?? activeCell.row |
|
|
@ -672,105 +682,104 @@ const closeAddColumnDropdown = () => { |
|
|
|
@contextmenu="showContextMenu" |
|
|
|
@contextmenu="showContextMenu" |
|
|
|
> |
|
|
|
> |
|
|
|
<thead ref="tableHead"> |
|
|
|
<thead ref="tableHead"> |
|
|
|
<tr class="nc-grid-header border-1 bg-gray-100 sticky top[-1px]"> |
|
|
|
<tr class="nc-grid-header border-1 bg-gray-100 sticky top[-1px]"> |
|
|
|
<th data-testid="grid-id-column"> |
|
|
|
<th data-testid="grid-id-column"> |
|
|
|
<div class="w-full h-full bg-gray-100 flex min-w-[70px] pl-5 pr-1 items-center" |
|
|
|
<div class="w-full h-full bg-gray-100 flex min-w-[70px] pl-5 pr-1 items-center" data-testid="nc-check-all"> |
|
|
|
data-testid="nc-check-all"> |
|
|
|
<template v-if="!readOnly"> |
|
|
|
<template v-if="!readOnly"> |
|
|
|
<div class="nc-no-label text-gray-500" :class="{ hidden: selectedAllRecords }">#</div> |
|
|
|
<div class="nc-no-label text-gray-500" :class="{ hidden: selectedAllRecords }">#</div> |
|
|
|
<div |
|
|
|
<div |
|
|
|
:class="{ hidden: !selectedAllRecords, flex: selectedAllRecords }" |
|
|
|
:class="{ hidden: !selectedAllRecords, flex: selectedAllRecords }" |
|
|
|
class="nc-check-all w-full items-center" |
|
|
|
class="nc-check-all w-full items-center" |
|
|
|
> |
|
|
|
> |
|
|
|
<a-checkbox v-model:checked="selectedAllRecords" /> |
|
|
|
<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"> |
|
|
|
|
|
|
|
<LazySmartsheetHeaderVirtualCell v-if="isVirtualCol(col)" :column="col" :hide-menu="readOnly" /> |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<LazySmartsheetHeaderCell v-else :column="col" :hide-menu="readOnly" /> |
|
|
|
<span class="flex-1" /> |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
</th> |
|
|
|
</template> |
|
|
|
<th |
|
|
|
<template v-else> |
|
|
|
v-if="isAddingColumnAllowed" |
|
|
|
<div class="text-gray-500">#</div> |
|
|
|
v-e="['c:column:add']" |
|
|
|
</template> |
|
|
|
class="cursor-pointer" |
|
|
|
</div> |
|
|
|
@click.stop="addColumnDropdown = true" |
|
|
|
</th> |
|
|
|
> |
|
|
|
<th |
|
|
|
<a-dropdown |
|
|
|
v-for="col in fields" |
|
|
|
v-model:visible="addColumnDropdown" |
|
|
|
:key="col.title" |
|
|
|
:trigger="['click']" |
|
|
|
v-xc-ver-resize |
|
|
|
overlay-class-name="nc-dropdown-grid-add-column" |
|
|
|
:data-col="col.id" |
|
|
|
|
|
|
|
:data-title="col.title" |
|
|
|
|
|
|
|
@xcresize="onresize(col.id, $event)" |
|
|
|
|
|
|
|
@xcresizing="onXcResizing(col.title, $event)" |
|
|
|
|
|
|
|
@xcresized="resizingCol = null" |
|
|
|
> |
|
|
|
> |
|
|
|
<div class="h-full w-[60px] flex items-center justify-center"> |
|
|
|
<div class="w-full h-full bg-gray-100 flex items-center"> |
|
|
|
<MdiPlus class="text-sm nc-column-add" /> |
|
|
|
<LazySmartsheetHeaderVirtualCell v-if="isVirtualCol(col)" :column="col" :hide-menu="readOnly" /> |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<LazySmartsheetHeaderCell v-else :column="col" :hide-menu="readOnly" /> |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
|
|
|
|
</th> |
|
|
|
|
|
|
|
<th |
|
|
|
|
|
|
|
v-if="isAddingColumnAllowed" |
|
|
|
|
|
|
|
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> |
|
|
|
<template #overlay> |
|
|
|
<SmartsheetColumnEditOrAddProvider |
|
|
|
<SmartsheetColumnEditOrAddProvider |
|
|
|
v-if="addColumnDropdown" |
|
|
|
v-if="addColumnDropdown" |
|
|
|
:column-position="columnOrder" |
|
|
|
:column-position="columnOrder" |
|
|
|
@submit="closeAddColumnDropdown" |
|
|
|
@submit="closeAddColumnDropdown" |
|
|
|
@cancel="closeAddColumnDropdown" |
|
|
|
@cancel="closeAddColumnDropdown" |
|
|
|
@click.stop |
|
|
|
@click.stop |
|
|
|
@keydown.stop |
|
|
|
@keydown.stop |
|
|
|
/> |
|
|
|
/> |
|
|
|
</template> |
|
|
|
</template> |
|
|
|
</a-dropdown> |
|
|
|
</a-dropdown> |
|
|
|
</th> |
|
|
|
</th> |
|
|
|
</tr> |
|
|
|
</tr> |
|
|
|
</thead> |
|
|
|
</thead> |
|
|
|
<tbody ref="tbodyEl"> |
|
|
|
<tbody ref="tbodyEl"> |
|
|
|
<LazySmartsheetRow v-for="(row, rowIndex) of data" ref="rowRefs" :key="rowIndex" :row="row"> |
|
|
|
<LazySmartsheetRow v-for="(row, rowIndex) of data" ref="rowRefs" :key="rowIndex" :row="row"> |
|
|
|
<template #default="{ state }"> |
|
|
|
<template #default="{ state }"> |
|
|
|
<tr class="nc-grid-row" :data-testid="`grid-row-${rowIndex}`"> |
|
|
|
<tr class="nc-grid-row" :data-testid="`grid-row-${rowIndex}`"> |
|
|
|
<td key="row-index" class="caption nc-grid-cell pl-5 pr-1" :data-testid="`cell-Id-${rowIndex}`"> |
|
|
|
<td key="row-index" class="caption nc-grid-cell pl-5 pr-1" :data-testid="`cell-Id-${rowIndex}`"> |
|
|
|
<div class="items-center flex gap-1 min-w-[55px]"> |
|
|
|
<div class="items-center flex gap-1 min-w-[55px]"> |
|
|
|
<div |
|
|
|
<div |
|
|
|
v-if="!readOnly || !isLocked" |
|
|
|
v-if="!readOnly || !isLocked" |
|
|
|
class="nc-row-no text-xs text-gray-500" |
|
|
|
class="nc-row-no text-xs text-gray-500" |
|
|
|
:class="{ toggle: !readOnly, hidden: row.rowMeta.selected }" |
|
|
|
:class="{ toggle: !readOnly, hidden: row.rowMeta.selected }" |
|
|
|
> |
|
|
|
> |
|
|
|
{{ rowIndex + 1 }} |
|
|
|
{{ rowIndex + 1 }} |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
<div |
|
|
|
<div |
|
|
|
v-if="!readOnly" |
|
|
|
v-if="!readOnly" |
|
|
|
:class="{ hidden: !row.rowMeta.selected, flex: row.rowMeta.selected }" |
|
|
|
:class="{ hidden: !row.rowMeta.selected, flex: row.rowMeta.selected }" |
|
|
|
class="nc-row-expand-and-checkbox" |
|
|
|
class="nc-row-expand-and-checkbox" |
|
|
|
> |
|
|
|
> |
|
|
|
<a-checkbox v-model:checked="row.rowMeta.selected" /> |
|
|
|
<a-checkbox v-model:checked="row.rowMeta.selected" /> |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
<span class="flex-1" /> |
|
|
|
<span class="flex-1" /> |
|
|
|
|
|
|
|
|
|
|
|
<div |
|
|
|
<div |
|
|
|
v-if="!readOnly || hasRole('commenter', true) || hasRole('viewer', true)" |
|
|
|
v-if="!readOnly || hasRole('commenter', true) || hasRole('viewer', true)" |
|
|
|
class="nc-expand" |
|
|
|
class="nc-expand" |
|
|
|
:data-testid="`nc-expand-${rowIndex}`" |
|
|
|
:data-testid="`nc-expand-${rowIndex}`" |
|
|
|
:class="{ 'nc-comment': row.rowMeta?.commentCount }" |
|
|
|
:class="{ 'nc-comment': row.rowMeta?.commentCount }" |
|
|
|
> |
|
|
|
> |
|
|
|
<a-spin |
|
|
|
<a-spin |
|
|
|
v-if="row.rowMeta.saving" |
|
|
|
v-if="row.rowMeta.saving" |
|
|
|
class="!flex items-center" |
|
|
|
class="!flex items-center" |
|
|
|
:data-testid="`row-save-spinner-${rowIndex}`" |
|
|
|
:data-testid="`row-save-spinner-${rowIndex}`" |
|
|
|
/> |
|
|
|
/> |
|
|
|
<template v-else> |
|
|
|
<template v-else> |
|
|
|
<span |
|
|
|
<span |
|
|
|
v-if="row.rowMeta?.commentCount" |
|
|
|
v-if="row.rowMeta?.commentCount" |
|
|
|
class="py-1 px-3 rounded-full text-xs cursor-pointer select-none transform hover:(scale-110)" |
|
|
|
class="py-1 px-3 rounded-full text-xs cursor-pointer select-none transform hover:(scale-110)" |
|
|
@ -779,86 +788,86 @@ const closeAddColumnDropdown = () => { |
|
|
|
> |
|
|
|
> |
|
|
|
{{ row.rowMeta.commentCount }} |
|
|
|
{{ row.rowMeta.commentCount }} |
|
|
|
</span> |
|
|
|
</span> |
|
|
|
<div |
|
|
|
<div |
|
|
|
v-else |
|
|
|
v-else |
|
|
|
class="cursor-pointer flex items-center border-1 active:ring rounded p-1 hover:(bg-primary bg-opacity-10)" |
|
|
|
class="cursor-pointer flex items-center border-1 active:ring rounded p-1 hover:(bg-primary bg-opacity-10)" |
|
|
|
> |
|
|
|
> |
|
|
|
<MdiArrowExpand |
|
|
|
<MdiArrowExpand |
|
|
|
v-e="['c:row-expand']" |
|
|
|
v-e="['c:row-expand']" |
|
|
|
class="select-none transform hover:(text-accent scale-120) nc-row-expand" |
|
|
|
class="select-none transform hover:(text-accent scale-120) nc-row-expand" |
|
|
|
@click="expandForm(row, state)" |
|
|
|
@click="expandForm(row, state)" |
|
|
|
/> |
|
|
|
/> |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
</template> |
|
|
|
</template> |
|
|
|
|
|
|
|
</div> |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
</td> |
|
|
|
</td> |
|
|
|
<SmartsheetTableDataCell |
|
|
|
<SmartsheetTableDataCell |
|
|
|
v-for="(columnObj, colIndex) of fields" |
|
|
|
v-for="(columnObj, colIndex) of fields" |
|
|
|
:key="columnObj.id" |
|
|
|
:key="columnObj.id" |
|
|
|
class="cell relative cursor-pointer nc-grid-cell" |
|
|
|
class="cell relative cursor-pointer nc-grid-cell" |
|
|
|
:class="{ |
|
|
|
:class="{ |
|
|
|
|
|
|
|
'active': hasEditPermission && isCellSelected(rowIndex, colIndex), |
|
|
|
'active': hasEditPermission && isCellSelected(rowIndex, colIndex), |
|
|
|
'nc-required-cell': isColumnRequiredAndNull(columnObj, row.row), |
|
|
|
'nc-required-cell': isColumnRequiredAndNull(columnObj, row.row), |
|
|
|
}" |
|
|
|
}" |
|
|
|
:data-testid="`cell-${columnObj.title}-${rowIndex}`" |
|
|
|
:data-testid="`cell-${columnObj.title}-${rowIndex}`" |
|
|
|
:data-key="rowIndex + columnObj.id" |
|
|
|
:data-key="rowIndex + columnObj.id" |
|
|
|
:data-col="columnObj.id" |
|
|
|
:data-col="columnObj.id" |
|
|
|
:data-title="columnObj.title" |
|
|
|
:data-title="columnObj.title" |
|
|
|
@mousedown="handleMouseDown($event, rowIndex, colIndex)" |
|
|
|
@mousedown="handleMouseDown($event, rowIndex, colIndex)" |
|
|
|
@mouseover="handleMouseOver(rowIndex, colIndex)" |
|
|
|
@mouseover="handleMouseOver(rowIndex, colIndex)" |
|
|
|
@click="handleCellClick($event, rowIndex, colIndex)" |
|
|
|
@click="handleCellClick($event, rowIndex, colIndex)" |
|
|
|
@dblclick="makeEditable(row, columnObj)" |
|
|
|
@dblclick="makeEditable(row, columnObj)" |
|
|
|
@contextmenu="showContextMenu($event, { row: rowIndex, col: colIndex })" |
|
|
|
@contextmenu="showContextMenu($event, { row: rowIndex, col: colIndex })" |
|
|
|
> |
|
|
|
> |
|
|
|
<div v-if="!switchingTab" class="w-full h-full"> |
|
|
|
<div v-if="!switchingTab" class="w-full h-full"> |
|
|
|
<LazySmartsheetVirtualCell |
|
|
|
<LazySmartsheetVirtualCell |
|
|
|
v-if="isVirtualCol(columnObj)" |
|
|
|
v-if="isVirtualCol(columnObj)" |
|
|
|
v-model="row.row[columnObj.title]" |
|
|
|
v-model="row.row[columnObj.title]" |
|
|
|
:column="columnObj" |
|
|
|
:column="columnObj" |
|
|
|
:active="activeCell.col === colIndex && activeCell.row === rowIndex" |
|
|
|
:active="activeCell.col === colIndex && activeCell.row === rowIndex" |
|
|
|
:row="row" |
|
|
|
:row="row" |
|
|
|
:read-only="readOnly" |
|
|
|
:read-only="readOnly" |
|
|
|
@navigate="onNavigate" |
|
|
|
@navigate="onNavigate" |
|
|
|
/> |
|
|
|
/> |
|
|
|
|
|
|
|
|
|
|
|
<LazySmartsheetCell |
|
|
|
<LazySmartsheetCell |
|
|
|
v-else |
|
|
|
v-else |
|
|
|
v-model="row.row[columnObj.title]" |
|
|
|
v-model="row.row[columnObj.title]" |
|
|
|
:column="columnObj" |
|
|
|
:column="columnObj" |
|
|
|
:edit-enabled=" |
|
|
|
:edit-enabled=" |
|
|
|
!!hasEditPermission && !!editEnabled && activeCell.col === colIndex && activeCell.row === rowIndex |
|
|
|
!!hasEditPermission && !!editEnabled && activeCell.col === colIndex && activeCell.row === rowIndex |
|
|
|
" |
|
|
|
" |
|
|
|
:row-index="rowIndex" |
|
|
|
:row-index="rowIndex" |
|
|
|
:active="activeCell.col === colIndex && activeCell.row === rowIndex" |
|
|
|
:active="activeCell.col === colIndex && activeCell.row === rowIndex" |
|
|
|
:read-only="readOnly" |
|
|
|
:read-only="readOnly" |
|
|
|
@update:edit-enabled="editEnabled = $event" |
|
|
|
@update:edit-enabled="editEnabled = $event" |
|
|
|
@save="updateOrSaveRow(row, columnObj.title, state)" |
|
|
|
@save="updateOrSaveRow(row, columnObj.title, state)" |
|
|
|
@navigate="onNavigate" |
|
|
|
@navigate="onNavigate" |
|
|
|
@cancel="editEnabled = false" |
|
|
|
@cancel="editEnabled = false" |
|
|
|
/> |
|
|
|
/> |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
</SmartsheetTableDataCell> |
|
|
|
</SmartsheetTableDataCell> |
|
|
|
</tr> |
|
|
|
</tr> |
|
|
|
</template> |
|
|
|
</template> |
|
|
|
</LazySmartsheetRow> |
|
|
|
</LazySmartsheetRow> |
|
|
|
|
|
|
|
|
|
|
|
<tr v-if="isAddingEmptyRowAllowed"> |
|
|
|
<tr v-if="isAddingEmptyRowAllowed"> |
|
|
|
<td |
|
|
|
<td |
|
|
|
v-e="['c:row:add:grid-bottom']" |
|
|
|
v-e="['c:row:add:grid-bottom']" |
|
|
|
:colspan="visibleColLength + 1" |
|
|
|
:colspan="visibleColLength + 1" |
|
|
|
class="text-left pointer nc-grid-add-new-cell cursor-pointer" |
|
|
|
class="text-left pointer nc-grid-add-new-cell cursor-pointer" |
|
|
|
@click="addEmptyRow()" |
|
|
|
@click="addEmptyRow()" |
|
|
|
> |
|
|
|
> |
|
|
|
<div class="px-2 w-full flex items-center text-gray-500"> |
|
|
|
<div class="px-2 w-full flex items-center text-gray-500"> |
|
|
|
<MdiPlus class="text-pint-500 text-xs ml-2 text-primary" /> |
|
|
|
<MdiPlus class="text-pint-500 text-xs ml-2 text-primary" /> |
|
|
|
|
|
|
|
|
|
|
|
<span class="ml-1"> |
|
|
|
<span class="ml-1"> |
|
|
|
{{ $t('activity.addRow') }} |
|
|
|
{{ $t('activity.addRow') }} |
|
|
|
</span> |
|
|
|
</span> |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
</td> |
|
|
|
</td> |
|
|
|
</tr> |
|
|
|
</tr> |
|
|
|
</tbody> |
|
|
|
</tbody> |
|
|
|
</table> |
|
|
|
</table> |
|
|
|
|
|
|
|
|
|
|
@ -897,8 +906,7 @@ const closeAddColumnDropdown = () => { |
|
|
|
</div> |
|
|
|
</div> |
|
|
|
</a-menu-item> |
|
|
|
</a-menu-item> |
|
|
|
|
|
|
|
|
|
|
|
<a-menu-item v-if="contextMenuTarget" data-testid="context-menu-item-copy" |
|
|
|
<a-menu-item v-if="contextMenuTarget" data-testid="context-menu-item-copy" @click="copyValue(contextMenuTarget)"> |
|
|
|
@click="copyValue(contextMenuTarget)"> |
|
|
|
|
|
|
|
<div v-e="['a:row:copy']" class="nc-project-menu-item"> |
|
|
|
<div v-e="['a:row:copy']" class="nc-project-menu-item"> |
|
|
|
<!-- Copy --> |
|
|
|
<!-- Copy --> |
|
|
|
{{ $t('general.copy') }} |
|
|
|
{{ $t('general.copy') }} |
|
|
|