mirror of
https://github.com/scinote-eln/scinote-web.git
synced 2025-01-01 21:21:50 +08:00
150 lines
5.1 KiB
Vue
150 lines
5.1 KiB
Vue
<template>
|
|
<div class="grid grid-cols-[1.5rem_auto] grid-rows-[1.5rem_auto] overflow-hidden">
|
|
<div class="z-10 bg-sn-super-light-grey"></div>
|
|
<div ref="columnsContainer" class="overflow-x-hidden">
|
|
<div :style="{'width': `${columnsList.length * 54}px`}">
|
|
<div v-for="column in columnsList" :key="column" class="uppercase float-left flex items-center justify-center w-[54px] ">
|
|
<span>{{ column }}</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div ref="rowContainer" class="overflow-y-hidden max-h-[70vh]">
|
|
<div v-for="row in rowsList" :key="row" class="uppercase flex items-center justify-center h-[54px]">
|
|
<span>{{ row }}</span>
|
|
</div>
|
|
</div>
|
|
<div ref="cellsContainer" class="overflow-hidden max-h-[70vh] relative">
|
|
<div class="grid" :style="{
|
|
'grid-template-columns': `repeat(${columnsList.length}, 1fr)`,
|
|
'width': `${columnsList.length * 54}px`
|
|
}">
|
|
<div v-for="cell in cellsList" :key="cell.row + cell.column" class="cell">
|
|
<div class="w-[54px] h-[54px] uppercase items-center flex justify-center p-1
|
|
border border-solid !border-transparent !border-b-sn-grey !border-r-sn-grey"
|
|
:class="{ '!border-t-sn-grey': cell.row === 0, '!border-l-sn-grey': cell.column === 0 }"
|
|
>
|
|
<div
|
|
class="h-full w-full rounded-full items-center flex justify-center"
|
|
@click="assignRow(cell)"
|
|
:class="{
|
|
'bg-sn-background-green': cellIsOccupied(cell),
|
|
'bg-sn-grey-100': cellIsHidden(cell),
|
|
'bg-white': cellIsAvailable(cell),
|
|
'bg-white border-sn-science-blue border-solid border-[1px]': cellIsSelected(cell),
|
|
'cursor-pointer': !cellIsHidden(cell)
|
|
}"
|
|
>
|
|
<template v-if="cellIsHidden(cell)">
|
|
<i class="sn-icon sn-icon-locked-task"></i>
|
|
</template>
|
|
<template v-else>
|
|
{{ rowsList[cell.row] }}{{ columnsList[cell.column] }}
|
|
</template>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
/* global PerfectScrollbar */
|
|
export default {
|
|
name: 'StorageLocationsGrid',
|
|
props: {
|
|
gridSize: {
|
|
type: Array,
|
|
required: true
|
|
},
|
|
assignedItems: {
|
|
type: Array,
|
|
default: () => []
|
|
},
|
|
selectedItems: {
|
|
type: Array,
|
|
default: () => []
|
|
}
|
|
},
|
|
data() {
|
|
return {
|
|
scrollBar: null
|
|
};
|
|
},
|
|
mounted() {
|
|
this.$refs.cellsContainer.addEventListener('scroll', this.handleScroll);
|
|
window.addEventListener('resize', this.handleScroll);
|
|
this.scrollBar = new PerfectScrollbar(this.$refs.cellsContainer, { wheelSpeed: 0.5, minScrollbarLength: 20 });
|
|
},
|
|
beforeUnmount() {
|
|
this.$refs.cellsContainer.removeEventListener('scroll', this.handleScroll);
|
|
window.removeEventListener('resize', this.handleScroll);
|
|
},
|
|
computed: {
|
|
columnsList() {
|
|
return Array.from({ length: this.gridSize[1] }, (v, i) => i + 1);
|
|
},
|
|
rowsList() {
|
|
return Array.from({ length: this.gridSize[0] }, (v, i) => String.fromCharCode(97 + i));
|
|
},
|
|
cellsList() {
|
|
const cells = [];
|
|
for (let i = 0; i < this.gridSize[0]; i++) {
|
|
for (let j = 0; j < this.gridSize[1]; j++) {
|
|
cells.push({ row: i, column: j });
|
|
}
|
|
}
|
|
return cells;
|
|
}
|
|
},
|
|
methods: {
|
|
cellObject(cell) {
|
|
return this.assignedItems.find((item) => item.position[0] === cell.row + 1 && item.position[1] === cell.column + 1);
|
|
},
|
|
cellIsOccupied(cell) {
|
|
return this.cellObject(cell) && !this.cellObject(cell)?.hidden;
|
|
},
|
|
cellIsHidden(cell) {
|
|
return this.cellObject(cell)?.hidden;
|
|
},
|
|
cellIsSelected(cell) {
|
|
return this.selectedItems.some((item) => item.position[0] === cell.row + 1 && item.position[1] === cell.column + 1);
|
|
},
|
|
cellIsAvailable(cell) {
|
|
return !this.cellIsOccupied(cell) && !this.cellIsHidden(cell);
|
|
},
|
|
assignRow(cell) {
|
|
if (this.cellIsOccupied(cell)) {
|
|
this.$emit('select', this.cellObject(cell));
|
|
return;
|
|
}
|
|
|
|
if (this.cellIsHidden(cell)) {
|
|
return;
|
|
}
|
|
|
|
this.$emit('assign', [cell.row + 1, cell.column + 1]);
|
|
},
|
|
handleScroll() {
|
|
this.$refs.columnsContainer.scrollLeft = this.$refs.cellsContainer.scrollLeft;
|
|
this.$refs.rowContainer.scrollTop = this.$refs.cellsContainer.scrollTop;
|
|
|
|
if (this.$refs.cellsContainer.scrollLeft > this.$refs.columnsContainer.scrollLeft) {
|
|
this.$refs.cellsContainer.scrollLeft = this.$refs.columnsContainer.scrollLeft;
|
|
}
|
|
|
|
if (this.$refs.rowContainer.scrollTop > 0) {
|
|
this.$refs.columnsContainer.style.boxShadow = '0px 0px 20px 0px rgba(16, 24, 40, 0.20)';
|
|
} else {
|
|
this.$refs.columnsContainer.style.boxShadow = 'none';
|
|
}
|
|
|
|
if (this.$refs.columnsContainer.scrollLeft > 0) {
|
|
this.$refs.rowContainer.style.boxShadow = '0px 0px 20px 0px rgba(16, 24, 40, 0.20)';
|
|
} else {
|
|
this.$refs.rowContainer.style.boxShadow = 'none';
|
|
}
|
|
}
|
|
}
|
|
};
|
|
</script>
|