|  | @@ -1,214 +1,214 @@
 | 
											
												
													
														|  | -import React from "react";
 |  | 
 | 
											
												
													
														|  | -import toast from "react-hot-toast";
 |  | 
 | 
											
												
													
														|  | -import Link from "next/link";
 |  | 
 | 
											
												
													
														|  | -import styled from "styled-components";
 |  | 
 | 
											
												
													
														|  | -import { CanvasDirection } from "reaflow";
 |  | 
 | 
											
												
													
														|  | -import { TiFlowMerge } from "react-icons/ti";
 |  | 
 | 
											
												
													
														|  | -import { CgArrowsMergeAltH, CgArrowsShrinkH } from "react-icons/cg";
 |  | 
 | 
											
												
													
														|  | -import {
 |  | 
 | 
											
												
													
														|  | -  AiOutlineDelete,
 |  | 
 | 
											
												
													
														|  | -  AiFillGithub,
 |  | 
 | 
											
												
													
														|  | -  AiOutlineTwitter,
 |  | 
 | 
											
												
													
														|  | -  AiOutlineSave,
 |  | 
 | 
											
												
													
														|  | -  AiOutlineFileAdd,
 |  | 
 | 
											
												
													
														|  | -  AiOutlineLink,
 |  | 
 | 
											
												
													
														|  | -} from "react-icons/ai";
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -import { Tooltip } from "src/components/Tooltip";
 |  | 
 | 
											
												
													
														|  | -import { useRouter } from "next/router";
 |  | 
 | 
											
												
													
														|  | -import { ImportModal } from "src/containers/Modals/ImportModal";
 |  | 
 | 
											
												
													
														|  | -import { ClearModal } from "src/containers/Modals/ClearModal";
 |  | 
 | 
											
												
													
														|  | -import { ShareModal } from "src/containers/Modals/ShareModal";
 |  | 
 | 
											
												
													
														|  | -import useConfig from "src/hooks/store/useConfig";
 |  | 
 | 
											
												
													
														|  | -import { getNextLayout } from "src/containers/Editor/LiveEditor/helpers";
 |  | 
 | 
											
												
													
														|  | -import { HiHeart } from "react-icons/hi";
 |  | 
 | 
											
												
													
														|  | -import shallow from "zustand/shallow";
 |  | 
 | 
											
												
													
														|  | -import { IoAlertCircleSharp } from "react-icons/io5";
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -const StyledSidebar = styled.div`
 |  | 
 | 
											
												
													
														|  | -  display: flex;
 |  | 
 | 
											
												
													
														|  | -  justify-content: space-between;
 |  | 
 | 
											
												
													
														|  | -  flex-direction: column;
 |  | 
 | 
											
												
													
														|  | -  align-items: center;
 |  | 
 | 
											
												
													
														|  | -  width: fit-content;
 |  | 
 | 
											
												
													
														|  | -  background: ${({ theme }) => theme.BACKGROUND_TERTIARY};
 |  | 
 | 
											
												
													
														|  | -  padding: 4px;
 |  | 
 | 
											
												
													
														|  | -  border-right: 1px solid ${({ theme }) => theme.BACKGROUND_MODIFIER_ACCENT};
 |  | 
 | 
											
												
													
														|  | -`;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -const StyledElement = styled.div<{ beta?: boolean }>`
 |  | 
 | 
											
												
													
														|  | -  position: relative;
 |  | 
 | 
											
												
													
														|  | -  display: flex;
 |  | 
 | 
											
												
													
														|  | -  justify-content: center;
 |  | 
 | 
											
												
													
														|  | -  text-align: center;
 |  | 
 | 
											
												
													
														|  | -  font-size: 26px;
 |  | 
 | 
											
												
													
														|  | -  font-weight: 600;
 |  | 
 | 
											
												
													
														|  | -  width: 100%;
 |  | 
 | 
											
												
													
														|  | -  color: ${({ theme }) => theme.INTERACTIVE_NORMAL};
 |  | 
 | 
											
												
													
														|  | -  cursor: pointer;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -  svg {
 |  | 
 | 
											
												
													
														|  | -    padding: 8px;
 |  | 
 | 
											
												
													
														|  | -    vertical-align: middle;
 |  | 
 | 
											
												
													
														|  | -  }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -  a {
 |  | 
 | 
											
												
													
														|  | -    display: flex;
 |  | 
 | 
											
												
													
														|  | -  }
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -  &:hover :is(a, svg) {
 |  | 
 | 
											
												
													
														|  | -    color: ${({ theme }) => theme.INTERACTIVE_HOVER};
 |  | 
 | 
											
												
													
														|  | -  }
 |  | 
 | 
											
												
													
														|  | -`;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -const StyledText = styled.span<{ secondary?: boolean }>`
 |  | 
 | 
											
												
													
														|  | -  color: ${({ theme, secondary }) =>
 |  | 
 | 
											
												
													
														|  | -    secondary ? theme.INTERACTIVE_NORMAL : theme.ORANGE};
 |  | 
 | 
											
												
													
														|  | -`;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -const StyledFlowIcon = styled(TiFlowMerge)<{ rotate: number }>`
 |  | 
 | 
											
												
													
														|  | -  transform: rotate(${({ rotate }) => `${rotate}deg`});
 |  | 
 | 
											
												
													
														|  | -`;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -const StyledTopWrapper = styled.nav`
 |  | 
 | 
											
												
													
														|  | -  display: flex;
 |  | 
 | 
											
												
													
														|  | -  justify-content: space-between;
 |  | 
 | 
											
												
													
														|  | -  flex-direction: column;
 |  | 
 | 
											
												
													
														|  | -  align-items: center;
 |  | 
 | 
											
												
													
														|  | -  width: 100%;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -  & > div:nth-child(n + 1) {
 |  | 
 | 
											
												
													
														|  | -    border-bottom: 1px solid ${({ theme }) => theme.BACKGROUND_MODIFIER_ACCENT};
 |  | 
 | 
											
												
													
														|  | -  }
 |  | 
 | 
											
												
													
														|  | -`;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -const StyledBottomWrapper = styled.nav`
 |  | 
 | 
											
												
													
														|  | -  display: flex;
 |  | 
 | 
											
												
													
														|  | -  justify-content: space-between;
 |  | 
 | 
											
												
													
														|  | -  flex-direction: column;
 |  | 
 | 
											
												
													
														|  | -  align-items: center;
 |  | 
 | 
											
												
													
														|  | -  width: 100%;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -  & > div,
 |  | 
 | 
											
												
													
														|  | -  a:nth-child(0) {
 |  | 
 | 
											
												
													
														|  | -    border-top: 1px solid ${({ theme }) => theme.BACKGROUND_MODIFIER_ACCENT};
 |  | 
 | 
											
												
													
														|  | -  }
 |  | 
 | 
											
												
													
														|  | -`;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -const StyledLogo = styled.div`
 |  | 
 | 
											
												
													
														|  | -  color: ${({ theme }) => theme.FULL_WHITE};
 |  | 
 | 
											
												
													
														|  | -`;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -function rotateLayout(layout: CanvasDirection) {
 |  | 
 | 
											
												
													
														|  | -  if (layout === "LEFT") return 90;
 |  | 
 | 
											
												
													
														|  | -  if (layout === "UP") return 180;
 |  | 
 | 
											
												
													
														|  | -  if (layout === "RIGHT") return 270;
 |  | 
 | 
											
												
													
														|  | -  return 360;
 |  | 
 | 
											
												
													
														|  | -}
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -export const Sidebar: React.FC = () => {
 |  | 
 | 
											
												
													
														|  | -  const getJson = useConfig((state) => state.getJson);
 |  | 
 | 
											
												
													
														|  | -  const setConfig = useConfig((state) => state.setConfig);
 |  | 
 | 
											
												
													
														|  | -  const [uploadVisible, setUploadVisible] = React.useState(false);
 |  | 
 | 
											
												
													
														|  | -  const [clearVisible, setClearVisible] = React.useState(false);
 |  | 
 | 
											
												
													
														|  | -  const [shareVisible, setShareVisible] = React.useState(false);
 |  | 
 | 
											
												
													
														|  | -  const { push } = useRouter();
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -  const [expand, layout] = useConfig(
 |  | 
 | 
											
												
													
														|  | -    (state) => [state.expand, state.layout],
 |  | 
 | 
											
												
													
														|  | -    shallow
 |  | 
 | 
											
												
													
														|  | -  );
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -  const handleSave = () => {
 |  | 
 | 
											
												
													
														|  | -    const a = document.createElement("a");
 |  | 
 | 
											
												
													
														|  | -    const file = new Blob([getJson()], { type: "text/plain" });
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -    a.href = window.URL.createObjectURL(file);
 |  | 
 | 
											
												
													
														|  | -    a.download = "jsonvisio.json";
 |  | 
 | 
											
												
													
														|  | -    a.click();
 |  | 
 | 
											
												
													
														|  | -  };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -  const toggleExpandCollapse = () => {
 |  | 
 | 
											
												
													
														|  | -    setConfig("expand", !expand);
 |  | 
 | 
											
												
													
														|  | -    toast(`${expand ? "Collapsed" : "Expanded"} nodes.`);
 |  | 
 | 
											
												
													
														|  | -  };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -  const toggleLayout = () => {
 |  | 
 | 
											
												
													
														|  | -    const nextLayout = getNextLayout(layout);
 |  | 
 | 
											
												
													
														|  | -    setConfig("layout", nextLayout);
 |  | 
 | 
											
												
													
														|  | -  };
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -  return (
 |  | 
 | 
											
												
													
														|  | -    <StyledSidebar>
 |  | 
 | 
											
												
													
														|  | -      <StyledTopWrapper>
 |  | 
 | 
											
												
													
														|  | -        <Link passHref href="/">
 |  | 
 | 
											
												
													
														|  | -          <StyledElement onClick={() => push("/")}>
 |  | 
 | 
											
												
													
														|  | -            <StyledLogo>
 |  | 
 | 
											
												
													
														|  | -              <StyledText>J</StyledText>
 |  | 
 | 
											
												
													
														|  | -              <StyledText secondary>V</StyledText>
 |  | 
 | 
											
												
													
														|  | -            </StyledLogo>
 |  | 
 | 
											
												
													
														|  | -          </StyledElement>
 |  | 
 | 
											
												
													
														|  | -        </Link>
 |  | 
 | 
											
												
													
														|  | -        <Tooltip title="Import File">
 |  | 
 | 
											
												
													
														|  | -          <StyledElement onClick={() => setUploadVisible(true)}>
 |  | 
 | 
											
												
													
														|  | -            <AiOutlineFileAdd />
 |  | 
 | 
											
												
													
														|  | -          </StyledElement>
 |  | 
 | 
											
												
													
														|  | -        </Tooltip>
 |  | 
 | 
											
												
													
														|  | -        <Tooltip title="Rotate Layout">
 |  | 
 | 
											
												
													
														|  | -          <StyledElement onClick={toggleLayout}>
 |  | 
 | 
											
												
													
														|  | -            <StyledFlowIcon rotate={rotateLayout(layout)} />
 |  | 
 | 
											
												
													
														|  | -          </StyledElement>
 |  | 
 | 
											
												
													
														|  | -        </Tooltip>
 |  | 
 | 
											
												
													
														|  | -        <Tooltip title={expand ? "Shrink Nodes" : "Expand Nodes"}>
 |  | 
 | 
											
												
													
														|  | -          <StyledElement
 |  | 
 | 
											
												
													
														|  | -            title="Toggle Expand/Collapse"
 |  | 
 | 
											
												
													
														|  | -            onClick={toggleExpandCollapse}
 |  | 
 | 
											
												
													
														|  | -          >
 |  | 
 | 
											
												
													
														|  | -            {expand ? <CgArrowsMergeAltH /> : <CgArrowsShrinkH />}
 |  | 
 | 
											
												
													
														|  | -          </StyledElement>
 |  | 
 | 
											
												
													
														|  | -        </Tooltip>
 |  | 
 | 
											
												
													
														|  | -        <Tooltip title="Save JSON">
 |  | 
 | 
											
												
													
														|  | -          <StyledElement onClick={handleSave}>
 |  | 
 | 
											
												
													
														|  | -            <AiOutlineSave />
 |  | 
 | 
											
												
													
														|  | -          </StyledElement>
 |  | 
 | 
											
												
													
														|  | -        </Tooltip>
 |  | 
 | 
											
												
													
														|  | -        <Tooltip title="Clear JSON">
 |  | 
 | 
											
												
													
														|  | -          <StyledElement onClick={() => setClearVisible(true)}>
 |  | 
 | 
											
												
													
														|  | -            <AiOutlineDelete />
 |  | 
 | 
											
												
													
														|  | -          </StyledElement>
 |  | 
 | 
											
												
													
														|  | -        </Tooltip>
 |  | 
 | 
											
												
													
														|  | -        <Tooltip title="Share">
 |  | 
 | 
											
												
													
														|  | -          <StyledElement onClick={() => setShareVisible(true)}>
 |  | 
 | 
											
												
													
														|  | -            <AiOutlineLink />
 |  | 
 | 
											
												
													
														|  | -          </StyledElement>
 |  | 
 | 
											
												
													
														|  | -        </Tooltip>
 |  | 
 | 
											
												
													
														|  | -      </StyledTopWrapper>
 |  | 
 | 
											
												
													
														|  | -      <StyledBottomWrapper>
 |  | 
 | 
											
												
													
														|  | -        <StyledElement>
 |  | 
 | 
											
												
													
														|  | -          <Link href="https://twitter.com/aykutsarach">
 |  | 
 | 
											
												
													
														|  | -            <a aria-label="Twitter" rel="me" target="_blank">
 |  | 
 | 
											
												
													
														|  | -              <AiOutlineTwitter />
 |  | 
 | 
											
												
													
														|  | -            </a>
 |  | 
 | 
											
												
													
														|  | -          </Link>
 |  | 
 | 
											
												
													
														|  | -        </StyledElement>
 |  | 
 | 
											
												
													
														|  | -        <StyledElement>
 |  | 
 | 
											
												
													
														|  | -          <Link href="https://github.com/AykutSarac/jsonvisio.com">
 |  | 
 | 
											
												
													
														|  | -            <a aria-label="GitHub" rel="me" target="_blank">
 |  | 
 | 
											
												
													
														|  | -              <AiFillGithub />
 |  | 
 | 
											
												
													
														|  | -            </a>
 |  | 
 | 
											
												
													
														|  | -          </Link>
 |  | 
 | 
											
												
													
														|  | -        </StyledElement>
 |  | 
 | 
											
												
													
														|  | -        <StyledElement>
 |  | 
 | 
											
												
													
														|  | -          <Link href="https://github.com/sponsors/AykutSarac">
 |  | 
 | 
											
												
													
														|  | -            <a aria-label="GitHub Sponsors" rel="me" target="_blank">
 |  | 
 | 
											
												
													
														|  | -              <HiHeart />
 |  | 
 | 
											
												
													
														|  | -            </a>
 |  | 
 | 
											
												
													
														|  | -          </Link>
 |  | 
 | 
											
												
													
														|  | -        </StyledElement>
 |  | 
 | 
											
												
													
														|  | -      </StyledBottomWrapper>
 |  | 
 | 
											
												
													
														|  | -      <ImportModal visible={uploadVisible} setVisible={setUploadVisible} />
 |  | 
 | 
											
												
													
														|  | -      <ClearModal visible={clearVisible} setVisible={setClearVisible} />
 |  | 
 | 
											
												
													
														|  | -      <ShareModal visible={shareVisible} setVisible={setShareVisible} />
 |  | 
 | 
											
												
													
														|  | -    </StyledSidebar>
 |  | 
 | 
											
												
													
														|  | -  );
 |  | 
 | 
											
												
													
														|  | -};
 |  | 
 | 
											
												
													
														|  | 
 |  | +import React from "react";
 | 
											
												
													
														|  | 
 |  | +import toast from "react-hot-toast";
 | 
											
												
													
														|  | 
 |  | +import Link from "next/link";
 | 
											
												
													
														|  | 
 |  | +import styled from "styled-components";
 | 
											
												
													
														|  | 
 |  | +import { CanvasDirection } from "reaflow";
 | 
											
												
													
														|  | 
 |  | +import { TiFlowMerge } from "react-icons/ti";
 | 
											
												
													
														|  | 
 |  | +import { CgArrowsMergeAltH, CgArrowsShrinkH } from "react-icons/cg";
 | 
											
												
													
														|  | 
 |  | +import {
 | 
											
												
													
														|  | 
 |  | +  AiOutlineDelete,
 | 
											
												
													
														|  | 
 |  | +  AiFillGithub,
 | 
											
												
													
														|  | 
 |  | +  AiOutlineTwitter,
 | 
											
												
													
														|  | 
 |  | +  AiOutlineSave,
 | 
											
												
													
														|  | 
 |  | +  AiOutlineFileAdd,
 | 
											
												
													
														|  | 
 |  | +  AiOutlineLink,
 | 
											
												
													
														|  | 
 |  | +} from "react-icons/ai";
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +import { Tooltip } from "src/components/Tooltip";
 | 
											
												
													
														|  | 
 |  | +import { useRouter } from "next/router";
 | 
											
												
													
														|  | 
 |  | +import { ImportModal } from "src/containers/Modals/ImportModal";
 | 
											
												
													
														|  | 
 |  | +import { ClearModal } from "src/containers/Modals/ClearModal";
 | 
											
												
													
														|  | 
 |  | +import { ShareModal } from "src/containers/Modals/ShareModal";
 | 
											
												
													
														|  | 
 |  | +import useConfig from "src/hooks/store/useConfig";
 | 
											
												
													
														|  | 
 |  | +import { getNextLayout } from "src/containers/Editor/LiveEditor/helpers";
 | 
											
												
													
														|  | 
 |  | +import { HiHeart } from "react-icons/hi";
 | 
											
												
													
														|  | 
 |  | +import shallow from "zustand/shallow";
 | 
											
												
													
														|  | 
 |  | +import { IoAlertCircleSharp } from "react-icons/io5";
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +const StyledSidebar = styled.div`
 | 
											
												
													
														|  | 
 |  | +  display: flex;
 | 
											
												
													
														|  | 
 |  | +  justify-content: space-between;
 | 
											
												
													
														|  | 
 |  | +  flex-direction: column;
 | 
											
												
													
														|  | 
 |  | +  align-items: center;
 | 
											
												
													
														|  | 
 |  | +  width: fit-content;
 | 
											
												
													
														|  | 
 |  | +  background: ${({ theme }) => theme.BACKGROUND_TERTIARY};
 | 
											
												
													
														|  | 
 |  | +  padding: 4px;
 | 
											
												
													
														|  | 
 |  | +  border-right: 1px solid ${({ theme }) => theme.BACKGROUND_MODIFIER_ACCENT};
 | 
											
												
													
														|  | 
 |  | +`;
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +const StyledElement = styled.div<{ beta?: boolean }>`
 | 
											
												
													
														|  | 
 |  | +  position: relative;
 | 
											
												
													
														|  | 
 |  | +  display: flex;
 | 
											
												
													
														|  | 
 |  | +  justify-content: center;
 | 
											
												
													
														|  | 
 |  | +  text-align: center;
 | 
											
												
													
														|  | 
 |  | +  font-size: 26px;
 | 
											
												
													
														|  | 
 |  | +  font-weight: 600;
 | 
											
												
													
														|  | 
 |  | +  width: 100%;
 | 
											
												
													
														|  | 
 |  | +  color: ${({ theme }) => theme.INTERACTIVE_NORMAL};
 | 
											
												
													
														|  | 
 |  | +  cursor: pointer;
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +  svg {
 | 
											
												
													
														|  | 
 |  | +    padding: 8px;
 | 
											
												
													
														|  | 
 |  | +    vertical-align: middle;
 | 
											
												
													
														|  | 
 |  | +  }
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +  a {
 | 
											
												
													
														|  | 
 |  | +    display: flex;
 | 
											
												
													
														|  | 
 |  | +  }
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +  &:hover :is(a, svg) {
 | 
											
												
													
														|  | 
 |  | +    color: ${({ theme }) => theme.INTERACTIVE_HOVER};
 | 
											
												
													
														|  | 
 |  | +  }
 | 
											
												
													
														|  | 
 |  | +`;
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +const StyledText = styled.span<{ secondary?: boolean }>`
 | 
											
												
													
														|  | 
 |  | +  color: ${({ theme, secondary }) =>
 | 
											
												
													
														|  | 
 |  | +    secondary ? theme.INTERACTIVE_NORMAL : theme.ORANGE};
 | 
											
												
													
														|  | 
 |  | +`;
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +const StyledFlowIcon = styled(TiFlowMerge)<{ rotate: number }>`
 | 
											
												
													
														|  | 
 |  | +  transform: rotate(${({ rotate }) => `${rotate}deg`});
 | 
											
												
													
														|  | 
 |  | +`;
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +const StyledTopWrapper = styled.nav`
 | 
											
												
													
														|  | 
 |  | +  display: flex;
 | 
											
												
													
														|  | 
 |  | +  justify-content: space-between;
 | 
											
												
													
														|  | 
 |  | +  flex-direction: column;
 | 
											
												
													
														|  | 
 |  | +  align-items: center;
 | 
											
												
													
														|  | 
 |  | +  width: 100%;
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +  & > div:nth-child(n + 1) {
 | 
											
												
													
														|  | 
 |  | +    border-bottom: 1px solid ${({ theme }) => theme.BACKGROUND_MODIFIER_ACCENT};
 | 
											
												
													
														|  | 
 |  | +  }
 | 
											
												
													
														|  | 
 |  | +`;
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +const StyledBottomWrapper = styled.nav`
 | 
											
												
													
														|  | 
 |  | +  display: flex;
 | 
											
												
													
														|  | 
 |  | +  justify-content: space-between;
 | 
											
												
													
														|  | 
 |  | +  flex-direction: column;
 | 
											
												
													
														|  | 
 |  | +  align-items: center;
 | 
											
												
													
														|  | 
 |  | +  width: 100%;
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +  & > div,
 | 
											
												
													
														|  | 
 |  | +  a:nth-child(0) {
 | 
											
												
													
														|  | 
 |  | +    border-top: 1px solid ${({ theme }) => theme.BACKGROUND_MODIFIER_ACCENT};
 | 
											
												
													
														|  | 
 |  | +  }
 | 
											
												
													
														|  | 
 |  | +`;
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +const StyledLogo = styled.div`
 | 
											
												
													
														|  | 
 |  | +  color: ${({ theme }) => theme.FULL_WHITE};
 | 
											
												
													
														|  | 
 |  | +`;
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +function rotateLayout(layout: CanvasDirection) {
 | 
											
												
													
														|  | 
 |  | +  if (layout === "LEFT") return 90;
 | 
											
												
													
														|  | 
 |  | +  if (layout === "UP") return 180;
 | 
											
												
													
														|  | 
 |  | +  if (layout === "RIGHT") return 270;
 | 
											
												
													
														|  | 
 |  | +  return 360;
 | 
											
												
													
														|  | 
 |  | +}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +export const Sidebar: React.FC = () => {
 | 
											
												
													
														|  | 
 |  | +  const getJson = useConfig((state) => state.getJson);
 | 
											
												
													
														|  | 
 |  | +  const setConfig = useConfig((state) => state.setConfig);
 | 
											
												
													
														|  | 
 |  | +  const [uploadVisible, setUploadVisible] = React.useState(false);
 | 
											
												
													
														|  | 
 |  | +  const [clearVisible, setClearVisible] = React.useState(false);
 | 
											
												
													
														|  | 
 |  | +  const [shareVisible, setShareVisible] = React.useState(false);
 | 
											
												
													
														|  | 
 |  | +  const { push } = useRouter();
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +  const [expand, layout] = useConfig(
 | 
											
												
													
														|  | 
 |  | +    (state) => [state.expand, state.layout],
 | 
											
												
													
														|  | 
 |  | +    shallow
 | 
											
												
													
														|  | 
 |  | +  );
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +  const handleSave = () => {
 | 
											
												
													
														|  | 
 |  | +    const a = document.createElement("a");
 | 
											
												
													
														|  | 
 |  | +    const file = new Blob([getJson()], { type: "text/plain" });
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +    a.href = window.URL.createObjectURL(file);
 | 
											
												
													
														|  | 
 |  | +    a.download = "jsonvisio.json";
 | 
											
												
													
														|  | 
 |  | +    a.click();
 | 
											
												
													
														|  | 
 |  | +  };
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +  const toggleExpandCollapse = () => {
 | 
											
												
													
														|  | 
 |  | +    setConfig("expand", !expand);
 | 
											
												
													
														|  | 
 |  | +    toast(`${expand ? "Collapsed" : "Expanded"} nodes.`);
 | 
											
												
													
														|  | 
 |  | +  };
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +  const toggleLayout = () => {
 | 
											
												
													
														|  | 
 |  | +    const nextLayout = getNextLayout(layout);
 | 
											
												
													
														|  | 
 |  | +    setConfig("layout", nextLayout);
 | 
											
												
													
														|  | 
 |  | +  };
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +  return (
 | 
											
												
													
														|  | 
 |  | +    <StyledSidebar>
 | 
											
												
													
														|  | 
 |  | +      <StyledTopWrapper>
 | 
											
												
													
														|  | 
 |  | +        <Link passHref href="/">
 | 
											
												
													
														|  | 
 |  | +          <StyledElement onClick={() => push("/")}>
 | 
											
												
													
														|  | 
 |  | +            <StyledLogo>
 | 
											
												
													
														|  | 
 |  | +              <StyledText>J</StyledText>
 | 
											
												
													
														|  | 
 |  | +              <StyledText secondary>V</StyledText>
 | 
											
												
													
														|  | 
 |  | +            </StyledLogo>
 | 
											
												
													
														|  | 
 |  | +          </StyledElement>
 | 
											
												
													
														|  | 
 |  | +        </Link>
 | 
											
												
													
														|  | 
 |  | +        <Tooltip title="Import File">
 | 
											
												
													
														|  | 
 |  | +          <StyledElement onClick={() => setUploadVisible(true)}>
 | 
											
												
													
														|  | 
 |  | +            <AiOutlineFileAdd />
 | 
											
												
													
														|  | 
 |  | +          </StyledElement>
 | 
											
												
													
														|  | 
 |  | +        </Tooltip>
 | 
											
												
													
														|  | 
 |  | +        <Tooltip title="Rotate Layout">
 | 
											
												
													
														|  | 
 |  | +          <StyledElement onClick={toggleLayout}>
 | 
											
												
													
														|  | 
 |  | +            <StyledFlowIcon rotate={rotateLayout(layout)} />
 | 
											
												
													
														|  | 
 |  | +          </StyledElement>
 | 
											
												
													
														|  | 
 |  | +        </Tooltip>
 | 
											
												
													
														|  | 
 |  | +        <Tooltip title={expand ? "Shrink Nodes" : "Expand Nodes"}>
 | 
											
												
													
														|  | 
 |  | +          <StyledElement
 | 
											
												
													
														|  | 
 |  | +            title="Toggle Expand/Collapse"
 | 
											
												
													
														|  | 
 |  | +            onClick={toggleExpandCollapse}
 | 
											
												
													
														|  | 
 |  | +          >
 | 
											
												
													
														|  | 
 |  | +            {expand ? <CgArrowsMergeAltH /> : <CgArrowsShrinkH />}
 | 
											
												
													
														|  | 
 |  | +          </StyledElement>
 | 
											
												
													
														|  | 
 |  | +        </Tooltip>
 | 
											
												
													
														|  | 
 |  | +        <Tooltip title="Save JSON">
 | 
											
												
													
														|  | 
 |  | +          <StyledElement onClick={handleSave}>
 | 
											
												
													
														|  | 
 |  | +            <AiOutlineSave />
 | 
											
												
													
														|  | 
 |  | +          </StyledElement>
 | 
											
												
													
														|  | 
 |  | +        </Tooltip>
 | 
											
												
													
														|  | 
 |  | +        <Tooltip title="Clear JSON">
 | 
											
												
													
														|  | 
 |  | +          <StyledElement onClick={() => setClearVisible(true)}>
 | 
											
												
													
														|  | 
 |  | +            <AiOutlineDelete />
 | 
											
												
													
														|  | 
 |  | +          </StyledElement>
 | 
											
												
													
														|  | 
 |  | +        </Tooltip>
 | 
											
												
													
														|  | 
 |  | +        <Tooltip title="Share">
 | 
											
												
													
														|  | 
 |  | +          <StyledElement onClick={() => setShareVisible(true)}>
 | 
											
												
													
														|  | 
 |  | +            <AiOutlineLink />
 | 
											
												
													
														|  | 
 |  | +          </StyledElement>
 | 
											
												
													
														|  | 
 |  | +        </Tooltip>
 | 
											
												
													
														|  | 
 |  | +      </StyledTopWrapper>
 | 
											
												
													
														|  | 
 |  | +      <StyledBottomWrapper>
 | 
											
												
													
														|  | 
 |  | +        <StyledElement>
 | 
											
												
													
														|  | 
 |  | +          <Link href="https://twitter.com/aykutsarach">
 | 
											
												
													
														|  | 
 |  | +            <a aria-label="Twitter" rel="me" target="_blank">
 | 
											
												
													
														|  | 
 |  | +              <AiOutlineTwitter />
 | 
											
												
													
														|  | 
 |  | +            </a>
 | 
											
												
													
														|  | 
 |  | +          </Link>
 | 
											
												
													
														|  | 
 |  | +        </StyledElement>
 | 
											
												
													
														|  | 
 |  | +        <StyledElement>
 | 
											
												
													
														|  | 
 |  | +          <Link href="https://github.com/AykutSarac/jsonvisio.com">
 | 
											
												
													
														|  | 
 |  | +            <a aria-label="GitHub" rel="me" target="_blank">
 | 
											
												
													
														|  | 
 |  | +              <AiFillGithub />
 | 
											
												
													
														|  | 
 |  | +            </a>
 | 
											
												
													
														|  | 
 |  | +          </Link>
 | 
											
												
													
														|  | 
 |  | +        </StyledElement>
 | 
											
												
													
														|  | 
 |  | +        <StyledElement>
 | 
											
												
													
														|  | 
 |  | +          <Link href="https://github.com/sponsors/AykutSarac">
 | 
											
												
													
														|  | 
 |  | +            <a aria-label="GitHub Sponsors" rel="me" target="_blank">
 | 
											
												
													
														|  | 
 |  | +              <HiHeart />
 | 
											
												
													
														|  | 
 |  | +            </a>
 | 
											
												
													
														|  | 
 |  | +          </Link>
 | 
											
												
													
														|  | 
 |  | +        </StyledElement>
 | 
											
												
													
														|  | 
 |  | +      </StyledBottomWrapper>
 | 
											
												
													
														|  | 
 |  | +      <ImportModal visible={uploadVisible} setVisible={setUploadVisible} />
 | 
											
												
													
														|  | 
 |  | +      <ClearModal visible={clearVisible} setVisible={setClearVisible} />
 | 
											
												
													
														|  | 
 |  | +      <ShareModal visible={shareVisible} setVisible={setShareVisible} />
 | 
											
												
													
														|  | 
 |  | +    </StyledSidebar>
 | 
											
												
													
														|  | 
 |  | +  );
 | 
											
												
													
														|  | 
 |  | +};
 |