erp-platform/ui/src/views/list/GanttView.tsx

304 lines
9.1 KiB
TypeScript
Raw Normal View History

2025-12-01 14:45:23 +00:00
import Container from '@/components/shared/Container'
import { Dialog, Notification, toast } from '@/components/ui'
import { DX_CLASSNAMES } from '@/constants/app.constant'
import { GridDto } from '@/proxy/form/models'
import { useLocalization } from '@/utils/hooks/useLocalization'
import useResponsive from '@/utils/hooks/useResponsive'
import Gantt, {
Column,
Editing,
GanttRef,
2025-12-01 22:05:05 +00:00
Item,
2025-12-01 14:45:23 +00:00
Tasks,
2025-12-01 22:05:05 +00:00
Toolbar,
2025-12-01 14:45:23 +00:00
Validation,
} from 'devextreme-react/gantt'
2025-12-01 22:05:05 +00:00
import DataSource from 'devextreme/data/data_source'
2025-12-01 14:45:23 +00:00
import { useCallback, useEffect, useRef, useState } from 'react'
import { Helmet } from 'react-helmet'
import { getList } from '@/services/form.service'
import { useListFormCustomDataSource } from './useListFormCustomDataSource'
import { useToolbar } from './useToolbar'
import { useFilters } from './useFilters'
import { addCss, addJs } from './Utils'
import { layoutTypes } from '../admin/listForm/edit/types'
import WidgetGroup from '@/components/ui/Widget/WidgetGroup'
interface GanttViewProps {
listFormCode: string
searchParams?: URLSearchParams
isSubForm?: boolean
level?: number
refreshData?: () => Promise<void>
gridDto?: GridDto
}
const GanttView = (props: GanttViewProps) => {
const { listFormCode, searchParams, isSubForm, level, gridDto: extGridDto } = props
const { translate } = useLocalization()
const { smaller } = useResponsive()
const ganttRef = useRef<GanttRef>()
const refListFormCode = useRef('')
const widgetGroupRef = useRef<HTMLDivElement>(null)
2025-12-01 22:05:05 +00:00
const [ganttDataSource, setGanttDataSource] = useState<DataSource<any, any>>()
2025-12-01 14:45:23 +00:00
const [gridDto, setGridDto] = useState<GridDto>()
const [widgetGroupHeight, setWidgetGroupHeight] = useState(0)
useEffect(() => {
const initializeGantt = async () => {
const response = await getList({ listFormCode })
setGridDto(response.data)
}
if (extGridDto === undefined) {
initializeGantt()
} else {
setGridDto(extGridDto)
}
}, [listFormCode, extGridDto])
const layout = layoutTypes.gantt || 'gantt'
const { toolbarData, toolbarModalData, setToolbarModalData } = useToolbar({
gridDto,
listFormCode,
getSelectedRowKeys: () => Promise.resolve([]),
getSelectedRowsData: () => [],
refreshData,
getFilter: () => undefined,
layout,
})
const { filterToolbarData, ...filterData } = useFilters({
gridDto,
gridRef: ganttRef as any,
listFormCode,
})
const { createSelectDataSource } = useListFormCustomDataSource({ gridRef: ganttRef as any })
function refreshData() {
ganttRef.current?.instance()?.refresh()
}
function onTaskInserted() {
props.refreshData?.()
}
function onTaskUpdated() {
props.refreshData?.()
}
function onTaskDeleted() {
props.refreshData?.()
}
useEffect(() => {
if (ganttRef?.current) {
const instance = ganttRef?.current?.instance()
if (instance) {
instance.option('dataSource', undefined)
}
}
if (refListFormCode.current !== listFormCode) {
// Reset state if needed
}
}, [listFormCode])
useEffect(() => {
if (!gridDto) {
return
}
// Set js and css
const grdOpt = gridDto.gridOptions
if (grdOpt.customJsSources.length) {
for (const js of grdOpt.customJsSources) {
addJs(js)
}
}
if (grdOpt.customStyleSources.length) {
for (const css of grdOpt.customStyleSources) {
addCss(css)
}
}
}, [gridDto])
useEffect(() => {
if (!gridDto) return
2025-12-01 22:05:05 +00:00
const customStore = createSelectDataSource(
2025-12-01 14:45:23 +00:00
gridDto.gridOptions,
listFormCode,
searchParams,
layout,
undefined,
)
2025-12-01 22:05:05 +00:00
// Gantt için DataSource wrapper'ı oluştur
const dataSource = new DataSource({
store: customStore,
reshapeOnPush: true,
})
2025-12-01 14:45:23 +00:00
setGanttDataSource(dataSource)
2025-12-01 22:05:05 +00:00
}, [gridDto, searchParams, createSelectDataSource])
2025-12-01 14:45:23 +00:00
useEffect(() => {
refListFormCode.current = listFormCode
}, [listFormCode])
// WidgetGroup yüksekliğini hesapla
useEffect(() => {
const calculateWidgetHeight = () => {
if (widgetGroupRef.current) {
const height = widgetGroupRef.current.offsetHeight
setWidgetGroupHeight(height)
}
}
calculateWidgetHeight()
const resizeObserver = new ResizeObserver(calculateWidgetHeight)
if (widgetGroupRef.current) {
resizeObserver.observe(widgetGroupRef.current)
}
return () => {
resizeObserver.disconnect()
}
}, [gridDto?.widgets])
// Gantt için sütunları oluştur
const getGanttColumns = useCallback(() => {
if (!gridDto?.columnFormats) return []
return gridDto.columnFormats
.filter((col) => col.canRead && col.isActive && col.visible)
.map((col) => {
const column: any = {
dataField: col.fieldName,
caption: col.captionName ? translate('::' + col.captionName) : col.fieldName,
width: col.width > 0 ? col.width : undefined,
alignment: col.alignment,
format: col.format,
}
return column
})
}, [gridDto, translate])
const ganttColumns = getGanttColumns()
return (
<>
<div ref={widgetGroupRef}>
<WidgetGroup widgetGroups={gridDto?.widgets ?? []} />
</div>
<Container className={DX_CLASSNAMES}>
{!isSubForm && (
<Helmet
titleTemplate="%s | Erp Platform"
title={translate('::' + gridDto?.gridOptions.title)}
defaultTitle="Erp Platform"
2025-12-01 22:05:05 +00:00
>
<link rel="stylesheet" href="/css/gantt/dx-gantt.min.css" />
</Helmet>
2025-12-01 14:45:23 +00:00
)}
{!gridDto && <div className="p-4">Loading gantt configuration...</div>}
{gridDto && !ganttDataSource && <div className="p-4">Loading data source...</div>}
{gridDto && ganttDataSource && (
<>
<div className="p-1">
<Gantt
ref={ganttRef as any}
key={`Gantt-${listFormCode}-${ganttDataSource ? 'loaded' : 'loading'}`}
id={'Gantt-' + listFormCode}
taskListWidth={500}
scaleType="weeks"
2025-12-01 22:05:05 +00:00
rootValue={
gridDto.gridOptions.treeOptionDto?.rootValue === '' ||
gridDto.gridOptions.treeOptionDto?.rootValue === undefined
? null
: gridDto.gridOptions.treeOptionDto?.rootValue
}
2025-12-01 14:45:23 +00:00
height={
gridDto.gridOptions.height > 0
? gridDto.gridOptions.height
: gridDto.gridOptions.fullHeight
? `calc(100vh - ${170 + widgetGroupHeight}px)`
: 700
}
showResources={true}
showDependencies={true}
onTaskInserted={onTaskInserted}
onTaskUpdated={onTaskUpdated}
onTaskDeleted={onTaskDeleted}
>
2025-12-01 22:05:05 +00:00
<Tasks
dataSource={ganttDataSource}
keyExpr={gridDto.gridOptions.treeOptionDto?.keyExpr}
parentIdExpr={gridDto.gridOptions.treeOptionDto?.parentIdExpr}
titleExpr={gridDto.gridOptions.treeOptionDto?.titleExpr}
startExpr={gridDto.gridOptions.treeOptionDto?.startExpr}
endExpr={gridDto.gridOptions.treeOptionDto?.endExpr}
progressExpr={gridDto.gridOptions.treeOptionDto?.progressExpr}
/>
<Toolbar>
<Item name="undo" />
<Item name="redo" />
<Item name="separator" />
<Item name="collapseAll" />
<Item name="expandAll" />
<Item name="separator" />
<Item name="addTask" />
<Item name="deleteTask" />
<Item name="separator" />
<Item name="zoomIn" />
<Item name="zoomOut" />
</Toolbar>
2025-12-01 14:45:23 +00:00
<Editing
2025-12-01 22:05:05 +00:00
enabled={gridDto.gridOptions.editingOptionDto?.allowUpdating}
2025-12-01 14:45:23 +00:00
allowTaskAdding={gridDto.gridOptions.editingOptionDto?.allowAdding}
allowTaskUpdating={gridDto.gridOptions.editingOptionDto?.allowUpdating}
allowTaskDeleting={gridDto.gridOptions.editingOptionDto?.allowDeleting}
allowDependencyAdding={false}
allowDependencyDeleting={false}
allowResourceAdding={false}
allowResourceDeleting={false}
/>
<Validation autoUpdateParentTasks={true} />
{ganttColumns.map((col: any) => (
<Column
key={col.dataField}
dataField={col.dataField}
caption={col.caption}
width={col.width}
/>
))}
</Gantt>
</div>
</>
)}
<Dialog
isOpen={toolbarModalData?.open || false}
onClose={() => setToolbarModalData(undefined)}
onRequestClose={() => setToolbarModalData(undefined)}
>
{toolbarModalData?.content}
</Dialog>
</Container>
</>
)
}
export default GanttView