|
<script lang="ts" context="module"> |
|
import Document from "@graphite/components/panels/Document.svelte"; |
|
import Layers from "@graphite/components/panels/Layers.svelte"; |
|
import Properties from "@graphite/components/panels/Properties.svelte"; |
|
import Spreadsheet from "@graphite/components/panels/Spreadsheet.svelte"; |
|
|
|
const PANEL_COMPONENTS = { |
|
Document, |
|
Layers, |
|
Properties, |
|
Spreadsheet, |
|
}; |
|
type PanelType = keyof typeof PANEL_COMPONENTS; |
|
</script> |
|
|
|
<script lang="ts"> |
|
import { getContext, tick } from "svelte"; |
|
|
|
import type { Editor } from "@graphite/editor"; |
|
import { type LayoutKeysGroup, type Key } from "@graphite/messages"; |
|
import { platformIsMac, isEventSupported } from "@graphite/utility-functions/platform"; |
|
|
|
import { extractPixelData } from "@graphite/utility-functions/rasterization"; |
|
|
|
import LayoutCol from "@graphite/components/layout/LayoutCol.svelte"; |
|
import LayoutRow from "@graphite/components/layout/LayoutRow.svelte"; |
|
import IconButton from "@graphite/components/widgets/buttons/IconButton.svelte"; |
|
import TextButton from "@graphite/components/widgets/buttons/TextButton.svelte"; |
|
import IconLabel from "@graphite/components/widgets/labels/IconLabel.svelte"; |
|
import TextLabel from "@graphite/components/widgets/labels/TextLabel.svelte"; |
|
import UserInputLabel from "@graphite/components/widgets/labels/UserInputLabel.svelte"; |
|
|
|
const BUTTON_MIDDLE = 1; |
|
|
|
const editor = getContext<Editor>("editor"); |
|
|
|
export let tabMinWidths = false; |
|
export let tabCloseButtons = false; |
|
export let tabLabels: { name: string; tooltip?: string }[]; |
|
export let tabActiveIndex: number; |
|
export let panelType: PanelType | undefined = undefined; |
|
export let clickAction: ((index: number) => void) | undefined = undefined; |
|
export let closeAction: ((index: number) => void) | undefined = undefined; |
|
|
|
let tabElements: (LayoutRow | undefined)[] = []; |
|
|
|
function platformModifiers(reservedKey: boolean): LayoutKeysGroup { |
|
// TODO: Remove this by properly feeding these keys from a layout provided by the backend |
|
|
|
const ALT: Key = { key: "Alt", label: "Alt" }; |
|
const COMMAND: Key = { key: "Command", label: "Command" }; |
|
const CONTROL: Key = { key: "Control", label: "Ctrl" }; |
|
|
|
if (platformIsMac()) return reservedKey ? [ALT, COMMAND] : [COMMAND]; |
|
return reservedKey ? [CONTROL, ALT] : [CONTROL]; |
|
} |
|
|
|
function dropFile(e: DragEvent) { |
|
if (!e.dataTransfer) return; |
|
|
|
e.preventDefault(); |
|
|
|
Array.from(e.dataTransfer.items).forEach(async (item) => { |
|
const file = item.getAsFile(); |
|
if (!file) return; |
|
|
|
if (file.type.includes("svg")) { |
|
const svgData = await file.text(); |
|
editor.handle.pasteSvg(file.name, svgData); |
|
return; |
|
} |
|
|
|
if (file.type.startsWith("image")) { |
|
const imageData = await extractPixelData(file); |
|
editor.handle.pasteImage(file.name, new Uint8Array(imageData.data), imageData.width, imageData.height); |
|
return; |
|
} |
|
|
|
if (file.name.endsWith(".graphite")) { |
|
const content = await file.text(); |
|
editor.handle.openDocumentFile(file.name, content); |
|
return; |
|
} |
|
}); |
|
} |
|
|
|
export async function scrollTabIntoView(newIndex: number) { |
|
await tick(); |
|
tabElements[newIndex]?.div?.()?.scrollIntoView(); |
|
} |
|
</script> |
|
|
|
<LayoutCol class="panel" on:pointerdown={() => panelType && editor.handle.setActivePanel(panelType)}> |
|
<LayoutRow class="tab-bar" classes={{ "min-widths": tabMinWidths }}> |
|
<LayoutRow class="tab-group" scrollableX={true}> |
|
{#each tabLabels as tabLabel, tabIndex} |
|
<LayoutRow |
|
class="tab" |
|
classes={{ active: tabIndex === tabActiveIndex }} |
|
tooltip={tabLabel.tooltip || undefined} |
|
on:click={(e) => { |
|
e.stopPropagation(); |
|
clickAction?.(tabIndex); |
|
}} |
|
on:auxclick={(e) => { |
|
// Middle mouse button click |
|
if (e.button === BUTTON_MIDDLE) { |
|
e.stopPropagation(); |
|
closeAction?.(tabIndex); |
|
} |
|
}} |
|
on:mouseup={(e) => { |
|
// Middle mouse button click fallback for Safari: |
|
// https://developer.mozilla.org/en-US/docs/Web/API/Element/auxclick_event#browser_compatibility |
|
// The downside of using mouseup is that the mousedown didn't have to originate in the same element. |
|
// A possible future improvement could save the target element during mousedown and check if it's the same here. |
|
if (!isEventSupported("auxclick") && e.button === BUTTON_MIDDLE) { |
|
e.stopPropagation(); |
|
closeAction?.(tabIndex); |
|
} |
|
}} |
|
bind:this={tabElements[tabIndex]} |
|
> |
|
<TextLabel>{tabLabel.name}</TextLabel> |
|
{#if tabCloseButtons} |
|
<IconButton |
|
action={(e) => { |
|
e?.stopPropagation(); |
|
closeAction?.(tabIndex); |
|
}} |
|
icon="CloseX" |
|
size={16} |
|
/> |
|
{/if} |
|
</LayoutRow> |
|
{/each} |
|
</LayoutRow> |
|
<!-- <PopoverButton style="VerticalEllipsis"> |
|
<TextLabel bold={true}>Panel Options</TextLabel> |
|
<TextLabel multiline={true}>Coming soon</TextLabel> |
|
</PopoverButton> --> |
|
</LayoutRow> |
|
<LayoutCol class="panel-body"> |
|
{#if panelType} |
|
<svelte:component this={PANEL_COMPONENTS[panelType]} /> |
|
{:else} |
|
<LayoutCol class="empty-panel" on:dragover={(e) => e.preventDefault()} on:drop={dropFile}> |
|
<LayoutCol class="content"> |
|
<LayoutRow class="logotype"> |
|
<IconLabel icon="GraphiteLogotypeSolid" /> |
|
</LayoutRow> |
|
<LayoutRow class="actions"> |
|
<table> |
|
<tr> |
|
<td> |
|
<TextButton label="New Document" icon="File" flush={true} action={() => editor.handle.newDocumentDialog()} /> |
|
</td> |
|
<td> |
|
<UserInputLabel keysWithLabelsGroups={[[...platformModifiers(true), { key: "KeyN", label: "N" }]]} /> |
|
</td> |
|
</tr> |
|
<tr> |
|
<td> |
|
<TextButton label="Open Document" icon="Folder" flush={true} action={() => editor.handle.openDocument()} /> |
|
</td> |
|
<td> |
|
<UserInputLabel keysWithLabelsGroups={[[...platformModifiers(false), { key: "KeyO", label: "O" }]]} /> |
|
</td> |
|
</tr> |
|
<tr> |
|
<td colspan="2"> |
|
<TextButton label="Open Demo Artwork" icon="Image" flush={true} action={() => editor.handle.demoArtworkDialog()} /> |
|
</td> |
|
</tr> |
|
<tr> |
|
<td colspan="2"> |
|
<TextButton label="Support the Development Fund" icon="Heart" flush={true} action={() => editor.handle.visitUrl("https://graphite.rs/donate/")} /> |
|
</td> |
|
</tr> |
|
</table> |
|
</LayoutRow> |
|
</LayoutCol> |
|
</LayoutCol> |
|
{/if} |
|
</LayoutCol> |
|
</LayoutCol> |
|
|
|
<style lang="scss" global> |
|
.panel { |
|
background: var(--color-1-nearblack); |
|
border-radius: 6px; |
|
overflow: hidden; |
|
|
|
.tab-bar { |
|
height: 28px; |
|
min-height: auto; |
|
|
|
&.min-widths .tab-group .tab { |
|
min-width: 120px; |
|
max-width: 360px; |
|
} |
|
|
|
.tab-group { |
|
flex: 1 1 100%; |
|
position: relative; |
|
|
|
// This always hangs out at the end of the last tab, providing 16px (15px plus the 1px reserved for the separator line) to the right of the tabs. |
|
// When the last tab is selected, its bottom rounded fillet adds 16px to the width, which stretches the scrollbar width allocation in only that situation. |
|
// This pseudo-element ensures we always reserve that space to prevent the scrollbar from jumping when the last tab is selected. |
|
// There is unfortunately no apparent way to remove that 16px gap from the end of the scroll container, since negative margin does not reduce the scrollbar allocation. |
|
&::after { |
|
content: ""; |
|
width: 15px; |
|
flex: 0 0 auto; |
|
} |
|
|
|
.tab { |
|
flex: 0 1 auto; |
|
height: 28px; |
|
padding: 0 8px; |
|
align-items: center; |
|
position: relative; |
|
|
|
&.active { |
|
background: var(--color-3-darkgray); |
|
border-radius: 6px 6px 0 0; |
|
position: relative; |
|
|
|
&:not |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|