import Aside, {AsideSection} from '~/components/Aside'; import {BlobResponse, blobFetcher} from '~/utils/fetch'; import type {Documentation, OpenedResult, Properties, SearchItem, SearchResult} from '~/resource/graph/types'; import GraphComponent, {GraphRef} from '~/components/GraphPage/Graph'; import React, {FunctionComponent, useCallback, useEffect, useMemo, useRef, useState} from 'react'; import Select, {SelectProps} from '~/components/Select'; import {primaryColor, rem, size} from '~/utils/style'; import Button from '~/components/Button'; import Checkbox from '~/components/Checkbox'; import Content from '~/components/Content'; import Field from '~/components/Field'; import HashLoader from 'react-spinners/HashLoader'; import ModelPropertiesDialog from '~/components/GraphPage/ModelPropertiesDialog'; import NodeDocumentationSidebar from '~/components/GraphPage/NodeDocumentationSidebar'; import NodePropertiesSidebar from '~/components/GraphPage/NodePropertiesSidebar'; import RadioButton from '~/components/RadioButton'; import RadioGroup from '~/components/RadioGroup'; import Search from '~/components/GraphPage/Search'; import Title from '~/components/Title'; import Uploader from '~/components/GraphPage/Uploader'; import styled from 'styled-components'; import useGlobalState from '~/hooks/useGlobalState'; import useRequest from '~/hooks/useRequest'; import {useTranslation} from 'react-i18next'; const FullWidthButton = styled(Button)` width: 100%; `; const FullWidthSelect = styled>>>(Select)` width: 100%; `; const ExportButtonWrapper = styled.div` display: flex; justify-content: space-between; > * { flex: 1 1 auto; &:not(:last-child) { margin-right: ${rem(20)}; } } `; // TODO: better way to auto fit height const SearchSection = styled(AsideSection)` max-height: calc(100% - ${rem(40)}); display: flex; flex-direction: column; &:not(:last-child) { padding-bottom: 0; } `; const Loading = styled.div` ${size('100%', '100%')} display: flex; flex-direction: column; justify-content: center; align-items: center; overscroll-behavior: none; cursor: progress; font-size: ${rem(16)}; line-height: ${rem(60)}; `; const Graph: FunctionComponent = () => { const {t} = useTranslation(['graph', 'common']); const [globalState, globalDispatch] = useGlobalState(); const graph = useRef(null); const file = useRef(null); const [files, setFiles] = useState(globalState.model); const onClickFile = useCallback(() => { if (file.current) { file.current.value = ''; file.current.click(); } }, []); const onChangeFile = useCallback( (e: React.ChangeEvent) => { const target = e.target; if (target && target.files && target.files.length) { globalDispatch({model: target.files}); setFiles(target.files); } }, [globalDispatch] ); const {data, loading} = useRequest(files ? null : '/graph/graph', blobFetcher); useEffect(() => { if (data?.data.size) { setFiles([new File([data.data], data.filename || 'unknwon_model')]); } }, [data]); const [modelGraphs, setModelGraphs] = useState([]); const [selectedGraph, setSelectedGraph] = useState>(''); const setOpenedModel = useCallback((data: OpenedResult) => { setModelGraphs(data.graphs); setSelectedGraph(data.selected || ''); }, []); const changeGraph = useCallback((name: string) => { setSelectedGraph(name); graph.current?.changeGraph(name); }, []); const [search, setSearch] = useState(''); const [searching, setSearching] = useState(false); const [searchResult, setSearchResult] = useState({text: '', result: []}); const onSearch = useCallback((value: string) => { setSearch(value); graph.current?.search(value); }, []); const onSelect = useCallback((item: SearchItem) => { setSearch(item.name); graph.current?.select(item); }, []); const [showAttributes, setShowAttributes] = useState(false); const [showInitializers, setShowInitializers] = useState(true); const [showNames, setShowNames] = useState(false); const [horizontal, setHorizontal] = useState(false); const [modelData, setModelData] = useState(null); const [nodeData, setNodeData] = useState(null); const [nodeDocumentation, setNodeDocumentation] = useState(null); useEffect(() => { setSearch(''); setSearchResult({text: '', result: []}); }, [files, showAttributes, showInitializers, showNames]); const bottom = useMemo( () => searching ? null : ( {t('graph:change-model')} ), [t, onClickFile, searching] ); const [rendered, setRendered] = useState(false); const aside = useMemo(() => { if (!rendered || loading) { return null; } if (nodeDocumentation) { return ( ); } if (nodeData) { return ( ); } return ( ); }, [ t, bottom, search, searching, searchResult, modelGraphs, selectedGraph, changeGraph, onSearch, onSelect, showAttributes, showInitializers, showNames, horizontal, rendered, loading, nodeData, nodeDocumentation ]); const uploader = useMemo(() => , [onClickFile]); return ( <> {t('common:graph')} setModelData(null)} /> {loading ? ( ) : ( setRendered(true)} onOpened={setOpenedModel} onSearch={data => setSearchResult(data)} onShowModelProperties={data => setModelData(data)} onShowNodeProperties={data => { setNodeData(data); setNodeDocumentation(null); }} onShowNodeDocumentation={data => setNodeDocumentation(data)} /> )} ); }; export default Graph;