This repository has been archived on 2022-08-14. You can view files and clone it, but cannot push or open issues or pull requests.
outline/app/components/Sidebar/components/DocumentLink.js

141 lines
3.8 KiB
JavaScript
Raw Normal View History

2018-05-06 05:45:10 +00:00
// @flow
import * as React from "react";
import { observer } from "mobx-react";
import { observable } from "mobx";
import styled from "styled-components";
import Document from "models/Document";
import DocumentMenu from "menus/DocumentMenu";
import SidebarLink from "./SidebarLink";
import DropToImport from "components/DropToImport";
import Fade from "components/Fade";
import Collection from "models/Collection";
import DocumentsStore from "stores/DocumentsStore";
import Flex from "components/Flex";
import { type NavigationNode } from "types";
2018-05-06 05:45:10 +00:00
type Props = {
node: NavigationNode,
documents: DocumentsStore,
collection?: Collection,
2018-05-06 05:45:10 +00:00
activeDocument: ?Document,
activeDocumentRef?: (?HTMLElement) => void,
2018-05-06 05:45:10 +00:00
prefetchDocument: (documentId: string) => Promise<void>,
depth: number,
};
@observer
class DocumentLink extends React.Component<Props> {
@observable menuOpen = false;
2019-12-23 01:06:39 +00:00
componentDidMount() {
if (this.isActiveDocument() && this.hasChildDocuments()) {
this.props.documents.fetchChildDocuments(this.props.node.id);
}
}
componentDidUpdate(prevProps: Props) {
if (prevProps.activeDocument !== this.props.activeDocument) {
if (this.isActiveDocument() && this.hasChildDocuments()) {
this.props.documents.fetchChildDocuments(this.props.node.id);
}
}
}
handleMouseEnter = (ev: SyntheticEvent<>) => {
const { node, prefetchDocument } = this.props;
2018-05-06 05:45:10 +00:00
ev.stopPropagation();
ev.preventDefault();
prefetchDocument(node.id);
2018-05-06 05:45:10 +00:00
};
2019-12-23 01:06:39 +00:00
isActiveDocument = () => {
return (
this.props.activeDocument &&
this.props.activeDocument.id === this.props.node.id
);
};
hasChildDocuments = () => {
return !!this.props.node.children.length;
};
2018-05-06 05:45:10 +00:00
render() {
const {
node,
documents,
collection,
2018-05-06 05:45:10 +00:00
activeDocument,
activeDocumentRef,
prefetchDocument,
depth,
} = this.props;
const showChildren = !!(
activeDocument &&
collection &&
(collection
.pathToDocument(activeDocument)
.map((entry) => entry.id)
.includes(node.id) ||
2019-12-23 01:06:39 +00:00
this.isActiveDocument())
2018-05-06 05:45:10 +00:00
);
const document = documents.get(node.id);
2018-05-06 05:45:10 +00:00
return (
<Flex
column
key={node.id}
2019-12-23 01:06:39 +00:00
ref={this.isActiveDocument() ? activeDocumentRef : undefined}
2018-05-06 05:45:10 +00:00
onMouseEnter={this.handleMouseEnter}
>
<DropToImport documentId={node.id} activeClassName="activeDropZone">
2018-05-06 05:45:10 +00:00
<SidebarLink
2018-05-13 04:01:17 +00:00
to={{
pathname: node.url,
state: { title: node.title },
2018-05-13 04:01:17 +00:00
}}
expanded={showChildren ? true : undefined}
label={node.title || "Untitled"}
depth={depth}
exact={false}
menuOpen={this.menuOpen}
menu={
document ? (
<Fade>
<DocumentMenu
position="right"
document={document}
onOpen={() => (this.menuOpen = true)}
onClose={() => (this.menuOpen = false)}
/>
</Fade>
) : undefined
}
2018-05-06 05:45:10 +00:00
>
2019-12-23 01:06:39 +00:00
{this.hasChildDocuments() && (
<DocumentChildren column>
{node.children.map((childNode) => (
<DocumentLink
key={childNode.id}
collection={collection}
node={childNode}
documents={documents}
activeDocument={activeDocument}
prefetchDocument={prefetchDocument}
depth={depth + 1}
/>
))}
</DocumentChildren>
)}
2018-05-06 05:45:10 +00:00
</SidebarLink>
</DropToImport>
</Flex>
);
}
}
const DocumentChildren = styled(Flex)``;
2018-05-06 05:45:10 +00:00
export default DocumentLink;