@mui/lab#TreeView TypeScript Examples
The following examples show how to use
@mui/lab#TreeView.
You can vote up the ones you like or vote down the ones you don't like,
and go to the original project or source file by following the links above each example. You may check out the related API usage on the sidebar.
Example #1
Source File: Profiler.tsx From NekoMaid with MIT License | 5 votes |
Plugins: React.FC = React.memo(() => { const plugin = usePlugin() const [data, setData] = useState<[JSX.Element[], any[][]] | undefined>() useEffect(() => { const off = plugin.emit('profiler:fetchPlugins').on('profiler:plugins', (data: Record<string, [Record<string | number, [number, number]>]>) => { const pluginsTimes: any[][] = [[], [], []] const tree: [number, JSX.Element][] = [] for (const name in data) { let totalTypesTime = 0 let totalTypesCount = 0 const subTrees: JSX.Element[] = [] ;['events', 'tasks', 'commands'].forEach((type, i) => { const curKey = name + '/' + type const subTree: [number, JSX.Element][] = [] const cur = data[name][i] let totalTime = 0 let totalCount = 0 for (const key in cur) { const [count, time] = cur[key] totalCount += count totalTypesCount += count totalTime += time totalTypesTime += time const key2 = `${curKey}/${key}` subTree.push([time, <TreeItem nodeId={key2} key={key2} label={getLabel(key, time, count)} />]) } if (totalTime) pluginsTimes[i].push({ name, value: totalTime }) if (subTree.length) { subTrees.push(<TreeItem nodeId={curKey} key={curKey} label={getLabel((lang.profiler as any)[type], totalTime, totalCount)}> {subTree.sort((a, b) => b[0] - a[0]).map(it => it[1])} </TreeItem>) } }) if (totalTypesTime) { tree.push([totalTypesTime, <TreeItem nodeId={name} label={getLabel(name, totalTypesTime, totalTypesCount)} key={name} >{subTrees}</TreeItem>]) } } setData([ tree.sort((a, b) => b[0] - a[0]).map(it => it[1]), pluginsTimes.map(it => it.sort((a, b) => b.value - a.value)) ]) }) return () => { off() } }, []) return <Container maxWidth={false} sx={{ py: 3, position: 'relative', height: data ? undefined : '80vh' }}> <CircularLoading loading={!data} background={false} /> {data && <Grid container spacing={3}> <Grid item xs={12}> <Card> <CardHeader title={lang.profiler.pluginsTitle} sx={{ position: 'relative' }} /> <Divider /> {data[0].length ? <TreeView defaultCollapseIcon={<ExpandMore />} defaultExpandIcon={<ChevronRight />}>{data[0]}</TreeView> : <CardContent><Empty /></CardContent>} </Card> </Grid> <Pie title={lang.profiler.pluginsEventsTime} data={data[1][0]} formatter={nanoSecondFormatter} /> <Pie title={lang.profiler.pluginsTasksTime} data={data[1][1]} formatter={nanoSecondFormatter} /> <Pie title={lang.profiler.pluginsCommandsTime} data={data[1][2]} formatter={nanoSecondFormatter} /> </Grid>} </Container> })
Example #2
Source File: index.tsx From mui-toolpad with MIT License | 4 votes |
export default function HierarchyExplorer({ appId, className }: HierarchyExplorerProps) {
const dom = useDom();
const domApi = useDomApi();
const app = appDom.getApp(dom);
const {
apis = [],
codeComponents = [],
pages = [],
connections = [],
} = appDom.getChildNodes(dom, app);
const [expanded, setExpanded] = useLocalStorageState<string[]>(
`editor/${app.id}/hierarchy-expansion`,
[':connections', ':pages', ':codeComponents'],
);
const location = useLocation();
const match =
matchRoutes(
[
{ path: `/app/:appId/editor/pages/:activeNodeId` },
{ path: `/app/:appId/editor/apis/:activeNodeId` },
{ path: `/app/:appId/editor/codeComponents/:activeNodeId` },
{ path: `/app/:appId/editor/connections/:activeNodeId` },
],
location,
) || [];
const selected: NodeId[] = match.map((route) => route.params.activeNodeId as NodeId);
const handleToggle = (event: React.SyntheticEvent, nodeIds: string[]) => {
setExpanded(nodeIds as NodeId[]);
};
const navigate = useNavigate();
const handleSelect = (event: React.SyntheticEvent, nodeIds: string[]) => {
if (nodeIds.length <= 0) {
return;
}
const rawNodeId = nodeIds[0];
if (rawNodeId.startsWith(':')) {
return;
}
const selectedNodeId: NodeId = rawNodeId as NodeId;
const node = appDom.getNode(dom, selectedNodeId);
if (appDom.isElement(node)) {
// TODO: sort out in-page selection
const page = appDom.getPageAncestor(dom, node);
if (page) {
navigate(`/app/${appId}/editor/pages/${page.id}`);
}
}
if (appDom.isPage(node)) {
navigate(`/app/${appId}/editor/pages/${node.id}`);
}
if (appDom.isApi(node)) {
navigate(`/app/${appId}/editor/apis/${node.id}`);
}
if (appDom.isCodeComponent(node)) {
navigate(`/app/${appId}/editor/codeComponents/${node.id}`);
}
if (appDom.isConnection(node)) {
navigate(`/app/${appId}/editor/connections/${node.id}`);
}
};
const [createConnectionDialogOpen, setCreateConnectionDialogOpen] = React.useState(0);
const handleCreateConnectionDialogOpen = React.useCallback((event: React.MouseEvent) => {
event.stopPropagation();
setCreateConnectionDialogOpen(Math.random());
}, []);
const handleCreateConnectionDialogClose = React.useCallback(
() => setCreateConnectionDialogOpen(0),
[],
);
const [createPageDialogOpen, setCreatePageDialogOpen] = React.useState(0);
const handleCreatePageDialogOpen = React.useCallback((event: React.MouseEvent) => {
event.stopPropagation();
setCreatePageDialogOpen(Math.random());
}, []);
const handleCreatepageDialogClose = React.useCallback(() => setCreatePageDialogOpen(0), []);
const [createCodeComponentDialogOpen, setCreateCodeComponentDialogOpen] = React.useState(0);
const handleCreateCodeComponentDialogOpen = React.useCallback((event: React.MouseEvent) => {
event.stopPropagation();
setCreateCodeComponentDialogOpen(Math.random());
}, []);
const handleCreateCodeComponentDialogClose = React.useCallback(
() => setCreateCodeComponentDialogOpen(0),
[],
);
const [deletedNodeId, setDeletedNodeId] = React.useState<NodeId | null>(null);
const handleDeleteNodeDialogOpen = React.useCallback(
(nodeId: NodeId) => (event: React.MouseEvent) => {
event.stopPropagation();
setDeletedNodeId(nodeId);
},
[],
);
const handledeleteNodeDialogClose = React.useCallback(() => setDeletedNodeId(null), []);
const handleDeleteNode = React.useCallback(() => {
if (deletedNodeId) {
domApi.removeNode(deletedNodeId);
navigate(`/app/${appId}/editor/`);
handledeleteNodeDialogClose();
}
}, [deletedNodeId, domApi, navigate, appId, handledeleteNodeDialogClose]);
const deletedNode = deletedNodeId && appDom.getMaybeNode(dom, deletedNodeId);
const latestDeletedNode = useLatest(deletedNode);
return (
<HierarchyExplorerRoot className={className}>
<TreeView
aria-label="hierarchy explorer"
selected={selected}
onNodeSelect={handleSelect}
expanded={expanded}
onNodeToggle={handleToggle}
multiSelect
defaultCollapseIcon={<ExpandMoreIcon />}
defaultExpandIcon={<ChevronRightIcon />}
>
<HierarchyTreeItem
nodeId=":connections"
labelText="Connections"
onCreate={handleCreateConnectionDialogOpen}
>
{connections.map((connectionNode) => (
<HierarchyTreeItem
key={connectionNode.id}
nodeId={connectionNode.id}
labelText={connectionNode.name}
onDelete={handleDeleteNodeDialogOpen(connectionNode.id)}
/>
))}
</HierarchyTreeItem>
{apis.length > 0 ? (
<HierarchyTreeItem nodeId=":apis" labelText="Apis">
{apis.map((apiNode) => (
<HierarchyTreeItem
key={apiNode.id}
nodeId={apiNode.id}
labelText={apiNode.name}
onDelete={handleDeleteNodeDialogOpen(apiNode.id)}
/>
))}
</HierarchyTreeItem>
) : null}
<HierarchyTreeItem
nodeId=":codeComponents"
labelText="Components"
onCreate={handleCreateCodeComponentDialogOpen}
>
{codeComponents.map((codeComponent) => (
<HierarchyTreeItem
key={codeComponent.id}
nodeId={codeComponent.id}
labelText={codeComponent.name}
onDelete={handleDeleteNodeDialogOpen(codeComponent.id)}
/>
))}
</HierarchyTreeItem>
<HierarchyTreeItem nodeId=":pages" labelText="Pages" onCreate={handleCreatePageDialogOpen}>
{pages.map((page) => (
<HierarchyTreeItem
key={page.id}
nodeId={page.id}
labelText={page.name}
onDelete={handleDeleteNodeDialogOpen(page.id)}
/>
))}
</HierarchyTreeItem>
</TreeView>
<CreateConnectionNodeDialog
key={createConnectionDialogOpen || undefined}
appId={appId}
open={!!createConnectionDialogOpen}
onClose={handleCreateConnectionDialogClose}
/>
<CreatePageNodeDialog
key={createPageDialogOpen || undefined}
appId={appId}
open={!!createPageDialogOpen}
onClose={handleCreatepageDialogClose}
/>
<CreateCodeComponentNodeDialog
key={createCodeComponentDialogOpen || undefined}
appId={appId}
open={!!createCodeComponentDialogOpen}
onClose={handleCreateCodeComponentDialogClose}
/>
<Dialog open={!!deletedNode} onClose={handledeleteNodeDialogClose}>
<DialogTitle>
Delete {latestDeletedNode?.type} "{latestDeletedNode?.name}"?
</DialogTitle>
<DialogActions>
<Button
type="submit"
color="inherit"
variant="text"
onClick={handledeleteNodeDialogClose}
>
Cancel
</Button>
<Button type="submit" onClick={handleDeleteNode}>
Delete
</Button>
</DialogActions>
</Dialog>
</HierarchyExplorerRoot>
);
}
Example #3
Source File: Files.tsx From NekoMaid with MIT License | 4 votes |
Files: React.FC = () => {
const plugin = usePlugin()
const theme = useTheme()
const his = useHistory()
const loc = useLocation()
const drawerWidth = useDrawerWidth()
const tree = useRef<HTMLHRElement | null>(null)
const editor = useRef<UnControlled | null>(null)
const prevExpanded = useRef<string[]>([])
const dirs = useRef<Record<string, boolean>>({ })
// eslint-disable-next-line func-call-spacing
const loading = useRef<Record<string, () => Promise<void>> & { '!#LOADING'?: boolean }>({ })
const [id, setId] = useState(0)
const [curPath, setCurPath] = useState('')
const [progress, setProgress] = useState(-1)
const [copyPath, setCopyPath] = useState('')
const [expanded, setExpanded] = useState<string[]>([])
const [compressFile, setCompressFile] = useState<string | null>(null)
const [anchorEl, setAnchorEl] = React.useState<HTMLButtonElement | null>(null)
const isDir = !!dirs.current[curPath]
const dirPath = isDir ? curPath : curPath.substring(0, curPath.lastIndexOf('/'))
const spacing = theme.spacing(3)
const refresh = () => {
loading.current = { }
dirs.current = { }
prevExpanded.current = []
setCurPath('')
setExpanded([])
setId(id + 1)
}
useEffect(() => {
if (!tree.current) return
const resize = () => {
if (!tree.current) return
const height = tree.current.style.maxHeight = (window.innerHeight - tree.current.offsetTop - parseInt(spacing)) + 'px'
const style = (editor as any).current?.editor?.display?.wrapper?.style
if (style) style.height = height
}
resize()
window.addEventListener('resize', resize)
return window.removeEventListener('resize', resize)
}, [tree.current, spacing])
return <Box sx={{ height: '100vh', py: 3 }}>
<Toolbar />
<Container maxWidth={false}>
<Grid container spacing={3} sx={{ width: { sm: `calc(100vw - ${drawerWidth}px - ${theme.spacing(3)})` } }}>
<Grid item lg={4} md={12} xl={3} xs={12}>
<Card sx={{ minHeight: 400 }}>
<CardHeader
title={lang.files.filesList}
sx={{ position: 'relative' }}
action={<Box sx={{ position: 'absolute', right: theme.spacing(1), top: '50%', transform: 'translateY(-50%)' }}
>
<Tooltip title={lang.files.delete}><span>
<IconButton
disabled={!curPath}
size='small'
onClick={() => dialog({
okButton: { color: 'error' },
content: <>{lang.files.confirmDelete(<span className='bold'>{curPath}</span>)}
<span className='bold' style={{ color: theme.palette.error.main }}>({lang.unrecoverable})</span></>
}).then(it => it && plugin.emit('files:update', (res: boolean) => {
action(res)
if (!res) return
refresh()
if (loc.pathname.replace(/^\/NekoMaid\/files\/?/, '') === curPath) his.push('/NekoMaid/files')
}, curPath))}
><DeleteForever /></IconButton>
</span></Tooltip>
<Tooltip title={lang.files.createFile}>
<IconButton size='small' onClick={() => fileNameDialog(lang.files.createFile, curPath)
.then(it => it != null && his.push(`/NekoMaid/files/${dirPath ? dirPath + '/' : ''}${it}`))}>
<Description /></IconButton></Tooltip>
<Tooltip title={lang.files.createFolder}>
<IconButton size='small' onClick={() => fileNameDialog(lang.files.createFolder, curPath)
.then(it => it != null && plugin.emit('files:createDirectory', (res: boolean) => {
action(res)
if (res) refresh()
}, dirPath + '/' + it))}><CreateNewFolder /></IconButton></Tooltip>
<Tooltip title={lang.more}>
<IconButton size='small' onClick={e => setAnchorEl(anchorEl ? null : e.currentTarget)}><MoreHoriz /></IconButton>
</Tooltip>
</Box>} />
<Divider />
<TreeView
ref={tree}
defaultCollapseIcon={<ArrowDropDown />}
defaultExpandIcon={<ArrowRight />}
sx={{ flexGrow: 1, width: '100%', overflowY: 'auto' }}
expanded={expanded}
onNodeToggle={(_: any, it: string[]) => {
const l = loading.current
if (it.length < prevExpanded.current.length || !l[it[0]]) {
setExpanded(it)
prevExpanded.current = it
return
}
l[it[0]]().then(() => {
prevExpanded.current.unshift(it[0])
setExpanded([...prevExpanded.current])
delete l[it[0]]
})
delete l[it[0]]
}}
onNodeSelect={(_: any, it: string) => {
setCurPath(it[0] === '/' ? it.slice(1) : it)
if (dirs.current[it] || loading.current['!#LOADING']) return
if (it.startsWith('/')) it = it.slice(1)
his.push('/NekoMaid/files/' + it)
}}
>
<Item plugin={plugin} path='' loading={loading.current} dirs={dirs.current} key={id} />
</TreeView>
</Card>
</Grid>
<Grid item lg={8} md={12} xl={9} xs={12} sx={{ maxWidth: `calc(100vw - ${theme.spacing(1)})`, paddingBottom: 3 }}>
<Editor plugin={plugin} editorRef={editor} loading={loading.current} dirs={dirs.current} refresh={refresh} />
</Grid>
</Grid>
</Container>
<Menu
anchorEl={anchorEl}
open={Boolean(anchorEl)}
onClose={() => setAnchorEl(null)}
anchorOrigin={anchorOrigin}
transformOrigin={anchorOrigin}
>
<MenuItem onClick={() => {
refresh()
setAnchorEl(null)
}}><ListItemIcon><Refresh /></ListItemIcon>{lang.refresh}</MenuItem>
<MenuItem disabled={!curPath} onClick={() => {
setAnchorEl(null)
fileNameDialog(lang.files.rename, curPath).then(it => it != null && plugin.emit('files:rename', (res: boolean) => {
action(res)
if (res) refresh()
}, curPath, dirPath + '/' + it))
}}><ListItemIcon><DriveFileRenameOutline /></ListItemIcon>{lang.files.rename}</MenuItem>
<MenuItem disabled={!curPath} onClick={() => {
setAnchorEl(null)
setCopyPath(curPath)
}}>
<ListItemIcon><FileCopy /></ListItemIcon>{lang.files.copy}
</MenuItem>
<MenuItem disabled={!copyPath} onClick={() => {
setAnchorEl(null)
toast(lang.files.pasting)
plugin.emit('files:copy', (res: boolean) => {
action(res)
refresh()
}, copyPath, dirPath)
}}>
<ListItemIcon><ContentPaste /></ListItemIcon>{lang.files.paste}
</MenuItem>
<MenuItem disabled={progress !== -1} component='label' htmlFor='NekoMaid-files-upload-input' onClick={() => setAnchorEl(null)}>
<ListItemIcon><Upload /></ListItemIcon>{progress === -1 ? lang.files.upload : `${lang.files.uploading} (${progress.toFixed(2)}%)`}
</MenuItem>
<MenuItem disabled={isDir} onClick={() => {
setAnchorEl(null)
toast(lang.files.downloading)
plugin.emit('files:download', (res: ArrayBuffer | null) => {
if (res) window.open(address! + 'Download/' + res, '_blank')
else failed()
}, curPath)
}}><ListItemIcon><Download /></ListItemIcon>{lang.files.download}</MenuItem>
<MenuItem onClick={() => {
setAnchorEl(null)
setCompressFile(curPath)
}}><ListItemIcon><Inbox /></ListItemIcon>{lang.files.compress}</MenuItem>
<MenuItem onClick={() => {
setAnchorEl(null)
toast(lang.files.uncompressing)
plugin.emit('files:compress', (res: boolean) => {
action(res)
refresh()
}, curPath)
}}><ListItemIcon><Outbox /></ListItemIcon>{lang.files.decompress}</MenuItem>
</Menu>
<Input id='NekoMaid-files-upload-input' type='file' sx={{ display: 'none' }} onChange={e => {
const elm = e.target as HTMLInputElement
const file = elm.files?.[0]
elm.value = ''
if (!file) return
const size = file.size
if (size > 128 * 1024 * 1024) return failed(lang.files.uploadTooBig)
toast(lang.files.uploading)
const name = dirPath + '/' + file.name
if (dirs.current[name] != null) return failed(lang.files.exists)
plugin.emit('files:upload', (res: string | null) => {
if (!res) return failed(lang.files.exists)
const formdata = new FormData()
formdata.append('file', file)
const xhr = new XMLHttpRequest()
setProgress(0)
xhr.open('put', address! + 'Upload/' + res)
xhr.onreadystatechange = () => {
if (xhr.readyState !== 4) return
setProgress(-1)
action(xhr.status === 200)
refresh()
}
xhr.upload.onprogress = e => e.lengthComputable && setProgress(e.loaded / e.total * 100)
xhr.send(formdata)
}, name[0] === '/' ? name.slice(1) : name)
}} />
<CompressDialog file={compressFile} path={dirPath} dirs={dirs.current} onClose={() => setCompressFile(null)} refresh={refresh} plugin={plugin} />
</Box>
}
Example #4
Source File: Profiler.tsx From NekoMaid with MIT License | 4 votes |
Timings: React.FC = React.memo(() => {
const plugin = usePlugin()
const theme = useTheme()
const { isTimingsV1 } = useGlobalData()
const [status, setStatus] = useState(false)
const [data, setData] = useState<TimingsData | null>(null)
useEffect(() => {
const off = plugin.emit('profiler:timingsStatus', setStatus).on('profiler:timings', setData)
return () => { off() }
}, [])
const [tree, entitiesTick, tilesTick] = useMemo(() => {
if (!data) return []
const entitiesTickMap: Record<string, { value: number, name: string, count: number }> = {}
const tilesTickMap: Record<string, { value: number, name: string, count: number }> = {}
const map: Record<number, [number, number, number, [number, number, number][] | undefined] | undefined> = { }
data.data.forEach(it => (map[it[0]] = it))
const createNode = (id: number, percent: number) => {
const cur = map[id]
if (!cur) return
map[id] = undefined
const [, count, time] = cur
const handler = data.handlers[id] || [0, lang.unknown]
const handlerName = data.groups[handler[0]] || lang.unknown
const name = handler[1]
const children = cur[cur.length - 1]
if (isTimingsV1) {
if (name.startsWith('tickEntity - ')) {
const came = name.slice(13).replace(/^Entity(Mob)?/, '')
const entity = decamelize(came)
const node = entitiesTickMap[entity]
if (node) {
node.count += count
node.value += time
} else entitiesTickMap[entity] = { count, value: time, name: minecraft['entity.minecraft.' + entity] || came }
} else if (name.startsWith('tickTileEntity - ')) {
const came = name.slice(17).replace(/^TileEntity(Mob)?/, '')
const entity = decamelize(came)
const node = tilesTickMap[entity]
if (node) {
node.count += count
node.value += time
} else tilesTickMap[entity] = { count, value: time, name: minecraft['block.minecraft.' + entity] || came }
}
} else {
if (name.startsWith('tickEntity - ') && name.endsWith('ick')) {
const res = ENTITY_TYPE.exec(name)
if (res) {
const node = entitiesTickMap[res[1]]
if (node) {
node.count += count
node.value += time
} else entitiesTickMap[res[1]] = { count, value: time, name: minecraft['entity.minecraft.' + res[1]] || res[1] }
}
} else if (name.startsWith('tickTileEntity - ')) {
const arr = name.split('.')
const came = arr[arr.length - 1].replace(/^TileEntity(Mob)?/, '')
const tile = decamelize(came)
const node = tilesTickMap[tile]
if (node) {
node.count += count
node.value += time
} else tilesTickMap[tile] = { count, value: time, name: minecraft['block.minecraft.' + tile] || came }
}
}
return <TreeItem
key={id}
nodeId={id.toString()}
label={<Box sx={{
'& .info, .count': { color: 'transparent' },
'&:hover .count': { color: 'inherit' },
'&:hover .info': {
color: theme.palette.primary.contrastText,
textShadow: theme.palette.mode === 'light'
? '#000 1px 0 0, #000 0 1px 0, #000 -1px 0 0, #000 0 -1px 0'
: '#fff 1px 0 0, #fff 0 1px 0, #fff -1px 0 0, #fff 0 -1px 0'
}
}}>
<Box sx={{
position: 'relative',
zIndex: 2,
display: 'flex',
alignItems: 'center'
}}>
{handlerName !== 'Minecraft' && <><Typography color='primary' component='span'>
{isTimingsV1 ? 'Bukkit' : lang.plugin + ':' + handlerName}</Typography>::</>}
{name}
<Typography variant='caption' className='count'>({lang.profiler.timingsCount}: {count})</Typography>
</Box>
<Box className='info' sx={{
position: 'absolute',
height: 10,
right: 0,
top: '50%',
marginTop: '-5px',
minWidth: 40,
display: 'flex',
justifyContent: 'center',
alignItems: 'center'
}}>
<Typography variant='caption' sx={{ position: 'absolute' }}>({Math.round(100 * percent)}%)</Typography>
<div style={{ width: 100 * percent + 'px' }} className='bar' />
</Box>
</Box>}
>{Array.isArray(children) && children.sort((a, b) => b[2] - a[2]).map(it => createNode(it[0], percent * (it[2] / time)))}</TreeItem>
}
// eslint-disable-next-line react/jsx-key
return [<TreeView defaultCollapseIcon={<ExpandMore />} defaultExpandIcon={<ChevronRight />} defaultExpanded={['1']}>
{createNode(1, 1)}
</TreeView>, Object.values(entitiesTickMap), Object.values(tilesTickMap)]
}, [data])
return <Container maxWidth={false} sx={{ py: 3 }}>
<Grid container spacing={3}>
<Grid item xs={12}>
<Card>
<CardHeader title='Timings' sx={{ position: 'relative' }} action={<FormControlLabel
control={<Switch checked={status} onChange={e => plugin.emit('profiler:timingsStatus', setStatus, e.target.checked)} />}
label={minecraft['addServer.resourcePack.enabled']}
sx={cardActionStyles}
/>} />
<Divider />
{status
? <Box sx={{
position: 'relative',
minHeight: data ? undefined : 300,
'& .bar': { backgroundColor: theme.palette.primary.main, height: 10, marginLeft: 'auto', borderRadius: 2 }
}}>
<CircularLoading loading={!data} />
{tree}
</Box>
: <CardContent><Empty title={lang.profiler.timingsNotStarted} /></CardContent>}
</Card>
</Grid>
{data && <Pie title={lang.profiler.entitiesTick} data={entitiesTick!} formatter={countFormatter} />}
{data && <Pie title={lang.profiler.tilesTick} data={tilesTick!} formatter={countFormatter} />}
</Grid>
</Container>
})