import { Button, Icon, TreeNode } from '@blueprintjs/core'; import { Popover2 } from '@blueprintjs/popover2'; import { normalizeSubtasks, readSubtasksFromFiles } from '@hydrooj/utils/lib/common'; import { TestCaseConfig } from 'hydrooj'; import { isEqual } from 'lodash'; import React from 'react'; import { DndProvider, useDrop } from 'react-dnd'; import { HTML5Backend } from 'react-dnd-html5-backend'; import { useDispatch, useSelector, useStore } from 'react-redux'; import { i18n } from 'vj/utils'; import { RootState } from './reducer'; import { AddTestcase } from './tree/AddTestcase'; import { SelectionManager } from './tree/SelectionManager'; import { GlobalSettings, SubtaskSettings } from './tree/SubtaskSettings'; interface TestcasesDndItem { cases: TestCaseConfig[]; subtaskId: number; } export function SubtaskNode(props: { subtaskId: number }) { const { subtaskId } = props; const subtaskIds = useSelector((s: RootState) => Object.values(s.config?.subtasks || []).map((i) => i.id), isEqual); const cases = useSelector((state: RootState) => (subtaskId === -1 ? state.config.__cases : state.config.subtasks.find((i) => i.id === subtaskId).cases || [])); const time = useSelector((s: RootState) => s.config?.time); const memory = useSelector((s: RootState) => s.config?.memory); const dispatch = useDispatch(); const [expand, setExpand] = React.useState(true); const [, drop] = useDrop(() => ({ accept: 'cases', canDrop(item) { return item.subtaskId !== subtaskId; }, drop(item) { dispatch({ type: 'problemconfig/moveTestcases', payload: { target: subtaskId, source: item.subtaskId, cases: item.cases, }, }); }, })); function deleteSubtask() { dispatch({ type: 'problemconfig/moveTestcases', payload: { target: -1, source: subtaskId, cases, }, }); dispatch({ type: 'problemconfig/deleteSubtask', id: subtaskId, }); } return (
  • {subtaskId !== -1 &&
    setExpand((e) => !e)}>   {i18n('Subtask {0}', subtaskId)} ev.stopPropagation()}> {i18n('Are you sure you want to delete this subtask?')}
    } > }
      {subtaskId !== -1 && } {expand ? : setExpand(false)} icon="layers" label={<> {cases.length} testcases.} path={[0]} />} {!cases.length && (
    • {subtaskId === -1 ? i18n('No testcase here') : i18n('Drag and drop testcases here:')}
    • )}
  • ); } export function SubtaskConfigTree() { const ids = useSelector((s: RootState) => Object.values(s.config?.subtasks || []).map((i) => i.id), isEqual); const dispatch = useDispatch(); const store = useStore(); function autoConfigure() { const state = store.getState(); const subtasks = readSubtasksFromFiles(state.testdata.map((t) => t.name), state.config); const cases = subtasks.reduce((a, b) => a.concat(b.cases), []); dispatch({ type: 'CONFIG_AUTOCASES_UPDATE', subtasks: normalizeSubtasks(subtasks, (i) => i, state.config.time, state.config.memory, true), }); dispatch({ type: 'problemconfig/delTestcases', cases, }); } return (
    •   {i18n('Auto configure')}
    • {ids.map((id) => )}
    • dispatch({ type: 'CONFIG_SUBTASK_UPDATE', id: 0, key: 'add' })} >
        {i18n('Add new subtask')}
    ); } export function ProblemConfigTree() { return (
    ); }