| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100 |
- import React, {
- useCallback, useState, FC, useEffect,
- } from 'react';
- import nodePath from 'path';
- import { ItemNode } from './ItemNode';
- import { useSWRxPageChildren } from '../../../stores/page-listing';
- import { usePageId } from '../../../stores/context';
- interface ItemProps {
- itemNode: ItemNode
- isOpen?: boolean
- }
- // Utility to mark target
- const markTarget = (children: ItemNode[], targetId: string): void => {
- children.forEach((node) => {
- if (node.page._id === targetId) {
- node.page.isTarget = true;
- }
- return node;
- });
- return;
- };
- const Item: FC<ItemProps> = (props: ItemProps) => {
- const { itemNode, isOpen: _isOpen = false } = props;
- const { page, children } = itemNode;
- const [currentChildren, setCurrentChildren] = useState(children);
- const [isOpen, setIsOpen] = useState(_isOpen);
- const { data: targetId } = usePageId();
- const { data, error } = useSWRxPageChildren(isOpen ? page._id : null);
- const hasChildren = useCallback((): boolean => {
- return currentChildren != null && currentChildren.length > 0;
- }, [currentChildren]);
- const onClickLoadChildren = useCallback(async() => {
- setIsOpen(!isOpen);
- }, [isOpen]);
- // didMount
- useEffect(() => {
- if (hasChildren()) setIsOpen(true);
- }, []);
- /*
- * Make sure itemNode.children and currentChildren are synced
- */
- useEffect(() => {
- if (children.length > currentChildren.length) {
- markTarget(children, targetId);
- setCurrentChildren(children);
- }
- }, []);
- /*
- * When swr fetch succeeded
- */
- useEffect(() => {
- if (isOpen && error == null && data != null) {
- const newChildren = ItemNode.generateNodesFromPages(data.children);
- markTarget(newChildren, targetId);
- setCurrentChildren(newChildren);
- }
- }, [data]);
- // TODO: improve style
- const opacityStyle = { opacity: 1.0 };
- if (page.isTarget) opacityStyle.opacity = 0.7;
- if (isOpen) opacityStyle.opacity = 0.5;
- return (
- <div style={{ margin: '10px' }}>
- <div style={opacityStyle}>
- <button type="button" className="d-inline-block btn btn-light p-1 mr-1" onClick={onClickLoadChildren}>Load</button>
- <a href={page._id} className="d-inline-block">
- <p>{nodePath.basename(page.path as string) || '/'}</p>
- </a>
- </div>
- {
- isOpen && hasChildren() && currentChildren.map(node => (
- <Item
- key={node.page._id}
- itemNode={node}
- isOpen={false}
- />
- ))
- }
- </div>
- );
- };
- export default Item;
|