2023-12-21 10:14:53 +05:00
|
|
|
/*
|
|
|
|
|
This file is part of the Notesnook project (https://notesnook.com/)
|
|
|
|
|
|
|
|
|
|
Copyright (C) 2023 Streetwriters (Private) Limited
|
|
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
*/
|
|
|
|
|
import { useThemeColors } from "@notesnook/theme";
|
|
|
|
|
import React from "react";
|
|
|
|
|
import { View } from "react-native";
|
|
|
|
|
import { useDBItem } from "../../../hooks/use-db-item";
|
2023-12-25 15:02:34 +05:00
|
|
|
import { useTabStore } from "../../../screens/editor/tiptap/use-tab-store";
|
|
|
|
|
import { editorController } from "../../../screens/editor/tiptap/utils";
|
2023-12-21 10:14:53 +05:00
|
|
|
import { presentSheet } from "../../../services/event-manager";
|
|
|
|
|
import { SIZE } from "../../../utils/size";
|
|
|
|
|
import { Button } from "../../ui/button";
|
|
|
|
|
import { IconButton } from "../../ui/icon-button";
|
|
|
|
|
import { PressableButton } from "../../ui/pressable";
|
|
|
|
|
import Heading from "../../ui/typography/heading";
|
2023-12-25 15:02:34 +05:00
|
|
|
import Paragraph from "../../ui/typography/paragraph";
|
2023-12-21 10:14:53 +05:00
|
|
|
|
|
|
|
|
type TabItem = {
|
|
|
|
|
id: number;
|
|
|
|
|
noteId?: string;
|
2023-12-29 10:47:23 +05:00
|
|
|
previewTab?: boolean;
|
2023-12-21 10:14:53 +05:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const TabItemComponent = (props: {
|
|
|
|
|
tab: TabItem;
|
|
|
|
|
isFocused: boolean;
|
|
|
|
|
close?: (ctx?: string | undefined) => void;
|
|
|
|
|
}) => {
|
|
|
|
|
const { colors } = useThemeColors();
|
|
|
|
|
const [item] = useDBItem(props.tab.noteId, "note");
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<PressableButton
|
|
|
|
|
customStyle={{
|
|
|
|
|
alignItems: "center",
|
|
|
|
|
justifyContent: "space-between",
|
|
|
|
|
flexDirection: "row",
|
|
|
|
|
paddingLeft: 12,
|
|
|
|
|
height: 45
|
|
|
|
|
}}
|
|
|
|
|
type={props.isFocused ? "selected" : "transparent"}
|
|
|
|
|
onPress={() => {
|
|
|
|
|
if (!props.isFocused) {
|
|
|
|
|
useTabStore.getState().focusTab(props.tab.id);
|
2023-12-25 15:02:34 +05:00
|
|
|
props.close?.();
|
2023-12-21 10:14:53 +05:00
|
|
|
}
|
|
|
|
|
}}
|
2023-12-29 10:47:23 +05:00
|
|
|
onLongPress={() => {
|
|
|
|
|
useTabStore.getState().updateTab(props.tab.id, {
|
|
|
|
|
previewTab: false
|
|
|
|
|
});
|
|
|
|
|
}}
|
2023-12-21 10:14:53 +05:00
|
|
|
>
|
|
|
|
|
<View
|
|
|
|
|
style={{
|
|
|
|
|
flexDirection: "row",
|
|
|
|
|
alignItems: "center",
|
|
|
|
|
justifyContent: "flex-start"
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<Paragraph
|
|
|
|
|
color={
|
|
|
|
|
props.isFocused
|
|
|
|
|
? colors.selected.paragraph
|
|
|
|
|
: colors.primary.paragraph
|
|
|
|
|
}
|
2023-12-29 10:47:23 +05:00
|
|
|
style={{
|
|
|
|
|
fontStyle: props.tab.previewTab ? "italic" : "normal"
|
|
|
|
|
}}
|
2023-12-21 10:14:53 +05:00
|
|
|
size={SIZE.md}
|
|
|
|
|
>
|
|
|
|
|
{item?.title || "New note"}
|
|
|
|
|
</Paragraph>
|
|
|
|
|
</View>
|
|
|
|
|
|
|
|
|
|
<IconButton
|
|
|
|
|
name="close"
|
|
|
|
|
size={SIZE.lg}
|
|
|
|
|
color={colors.primary.icon}
|
|
|
|
|
onPress={() => {
|
2023-12-25 15:02:34 +05:00
|
|
|
const isLastTab = useTabStore.getState().tabs.length === 1;
|
2023-12-21 10:14:53 +05:00
|
|
|
useTabStore.getState().removeTab(props.tab.id);
|
2023-12-25 15:02:34 +05:00
|
|
|
// The last tab is not actually removed, it is just cleaned up.
|
|
|
|
|
if (isLastTab) {
|
|
|
|
|
editorController.current?.reset(props.tab.id, true, true);
|
|
|
|
|
props.close?.();
|
|
|
|
|
}
|
2023-12-21 10:14:53 +05:00
|
|
|
}}
|
|
|
|
|
top={0}
|
|
|
|
|
left={0}
|
|
|
|
|
right={20}
|
|
|
|
|
bottom={0}
|
|
|
|
|
/>
|
|
|
|
|
</PressableButton>
|
|
|
|
|
);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
export default function EditorTabs({
|
|
|
|
|
close
|
|
|
|
|
}: {
|
|
|
|
|
close?: (ctx?: string | undefined) => void;
|
|
|
|
|
}) {
|
|
|
|
|
const [tabs, currentTab] = useTabStore((state) => [
|
|
|
|
|
state.tabs,
|
|
|
|
|
state.currentTab
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<View
|
|
|
|
|
style={{
|
|
|
|
|
paddingHorizontal: 12,
|
|
|
|
|
gap: 12
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<View
|
|
|
|
|
style={{
|
|
|
|
|
flexDirection: "row",
|
|
|
|
|
justifyContent: "space-between",
|
|
|
|
|
width: "100%",
|
|
|
|
|
alignItems: "center"
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<Heading size={SIZE.lg}>Tabs</Heading>
|
|
|
|
|
<Button
|
|
|
|
|
onPress={() => {
|
2023-12-25 15:02:34 +05:00
|
|
|
close?.();
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
useTabStore.getState().newTab();
|
|
|
|
|
}, 300);
|
2023-12-21 10:14:53 +05:00
|
|
|
}}
|
|
|
|
|
title="New tab"
|
|
|
|
|
icon="plus"
|
|
|
|
|
style={{
|
|
|
|
|
flexDirection: "row",
|
|
|
|
|
justifyContent: "flex-start",
|
|
|
|
|
borderRadius: 100,
|
|
|
|
|
height: 35
|
|
|
|
|
}}
|
|
|
|
|
iconSize={SIZE.lg}
|
|
|
|
|
/>
|
|
|
|
|
</View>
|
|
|
|
|
|
|
|
|
|
{tabs.map((tab) => (
|
|
|
|
|
<TabItemComponent
|
|
|
|
|
key={tab.id}
|
|
|
|
|
tab={tab}
|
|
|
|
|
isFocused={tab.id === currentTab}
|
|
|
|
|
close={close}
|
|
|
|
|
/>
|
|
|
|
|
))}
|
|
|
|
|
</View>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
EditorTabs.present = () => {
|
|
|
|
|
presentSheet({
|
|
|
|
|
component: (ref, close, update) => <EditorTabs close={close} />
|
|
|
|
|
});
|
|
|
|
|
};
|