Item.tsx 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. import React, {
  2. useCallback, useState, FC, useEffect,
  3. } from 'react';
  4. import nodePath from 'path';
  5. import { ItemNode } from './ItemNode';
  6. import { useSWRxPageChildren } from '../../../stores/page-listing';
  7. import { usePageId } from '../../../stores/context';
  8. interface ItemProps {
  9. itemNode: ItemNode
  10. isOpen?: boolean
  11. }
  12. // Utility to mark target
  13. const markTarget = (children: ItemNode[], targetId: string): void => {
  14. children.forEach((node) => {
  15. if (node.page._id === targetId) {
  16. node.page.isTarget = true;
  17. }
  18. return node;
  19. });
  20. return;
  21. };
  22. const Item: FC<ItemProps> = (props: ItemProps) => {
  23. const { itemNode, isOpen: _isOpen = false } = props;
  24. const { page, children } = itemNode;
  25. const [currentChildren, setCurrentChildren] = useState(children);
  26. const [isOpen, setIsOpen] = useState(_isOpen);
  27. const { data: targetId } = usePageId();
  28. const { data, error } = useSWRxPageChildren(isOpen ? page._id : null);
  29. const hasChildren = useCallback((): boolean => {
  30. return currentChildren != null && currentChildren.length > 0;
  31. }, [currentChildren]);
  32. const onClickLoadChildren = useCallback(async() => {
  33. setIsOpen(!isOpen);
  34. }, [isOpen]);
  35. // didMount
  36. useEffect(() => {
  37. if (hasChildren()) setIsOpen(true);
  38. }, []);
  39. /*
  40. * Make sure itemNode.children and currentChildren are synced
  41. */
  42. useEffect(() => {
  43. if (children.length > currentChildren.length) {
  44. markTarget(children, targetId);
  45. setCurrentChildren(children);
  46. }
  47. }, []);
  48. /*
  49. * When swr fetch succeeded
  50. */
  51. useEffect(() => {
  52. if (isOpen && error == null && data != null) {
  53. const newChildren = ItemNode.generateNodesFromPages(data.children);
  54. markTarget(newChildren, targetId);
  55. setCurrentChildren(newChildren);
  56. }
  57. }, [data]);
  58. // TODO: improve style
  59. const opacityStyle = { opacity: 1.0 };
  60. if (page.isTarget) opacityStyle.opacity = 0.7;
  61. if (isOpen) opacityStyle.opacity = 0.5;
  62. return (
  63. <div style={{ margin: '10px' }}>
  64. <div style={opacityStyle}>
  65. <button type="button" className="d-inline-block btn btn-light p-1 mr-1" onClick={onClickLoadChildren}>Load</button>
  66. <a href={page._id} className="d-inline-block">
  67. <p>{nodePath.basename(page.path as string) || '/'}</p>
  68. </a>
  69. </div>
  70. {
  71. isOpen && hasChildren() && currentChildren.map(node => (
  72. <Item
  73. key={node.page._id}
  74. itemNode={node}
  75. isOpen={false}
  76. />
  77. ))
  78. }
  79. </div>
  80. );
  81. };
  82. export default Item;