|
<script lang="ts"> |
|
import { createEventDispatcher, onDestroy } from "svelte"; |
|
|
|
import { Color, type Gradient } from "@graphite/messages"; |
|
|
|
import { preventEscapeClosingParentFloatingMenu } from "@graphite/components/layout/FloatingMenu.svelte"; |
|
import LayoutCol from "@graphite/components/layout/LayoutCol.svelte"; |
|
import LayoutRow from "@graphite/components/layout/LayoutRow.svelte"; |
|
|
|
const BUTTON_LEFT = 0; |
|
const BUTTON_RIGHT = 2; |
|
|
|
const dispatch = createEventDispatcher<{ activeMarkerIndexChange: number | undefined; gradient: Gradient; dragging: boolean }>(); |
|
|
|
export let gradient: Gradient; |
|
export let activeMarkerIndex = 0 as number | undefined; |
|
|
|
|
|
|
|
let markerTrack: LayoutRow | undefined = undefined; |
|
let positionRestore: number | undefined = undefined; |
|
let deletionRestore: boolean | undefined = undefined; |
|
|
|
function markerPointerDown(e: PointerEvent, index: number) { |
|
// Left-click to select and begin potentially dragging |
|
if (e.button === BUTTON_LEFT) { |
|
activeMarkerIndex = index; |
|
dispatch("activeMarkerIndexChange", index); |
|
addEvents(); |
|
return; |
|
} |
|
|
|
|
|
if (e.button === BUTTON_RIGHT && deletionRestore === undefined) { |
|
deleteStopByIndex(index); |
|
return; |
|
} |
|
} |
|
|
|
function markerPosition(e: MouseEvent): number | undefined { |
|
const markerTrackRect = markerTrack?.div()?.getBoundingClientRect(); |
|
if (!markerTrackRect) return; |
|
|
|
const ratio = (e.clientX - markerTrackRect.left) / markerTrackRect.width; |
|
|
|
return Math.max(0, Math.min(1, ratio)); |
|
} |
|
|
|
function insertStop(e: MouseEvent) { |
|
if (e.button !== BUTTON_LEFT) return; |
|
|
|
let position = markerPosition(e); |
|
if (position === undefined) return; |
|
|
|
let before = gradient.stops.findLast((value) => value.position < position); |
|
let after = gradient.stops.find((value) => value.position > position); |
|
|
|
let color = Color.fromCSS("black") as Color; |
|
if (before && after) { |
|
let t = (position - before.position) / (after.position - before.position); |
|
color = before.color.lerp(after.color, t); |
|
} else if (before) { |
|
color = before.color; |
|
} else if (after) { |
|
color = after.color; |
|
} |
|
|
|
let index = gradient.stops.findIndex((value) => value.position > position); |
|
if (index === -1) index = gradient.stops.length; |
|
|
|
gradient.stops.splice(index, 0, { position, color }); |
|
activeMarkerIndex = index; |
|
deletionRestore = true; |
|
|
|
dispatch("activeMarkerIndexChange", index); |
|
dispatch("gradient", gradient); |
|
|
|
addEvents(); |
|
} |
|
|
|
function deleteStop(e: KeyboardEvent) { |
|
if (e.key !== "Delete" && e.key !== "Backspace") return; |
|
if (activeMarkerIndex === undefined) return; |
|
|
|
if (positionRestore !== undefined) stopDrag(); |
|
|
|
deleteStopByIndex(activeMarkerIndex); |
|
} |
|
|
|
function deleteStopByIndex(index: number) { |
|
if (gradient.stops.length <= 2) return; |
|
|
|
gradient.stops.splice(index, 1); |
|
if (gradient.stops.length === 0) { |
|
activeMarkerIndex = undefined; |
|
} else { |
|
activeMarkerIndex = Math.max(0, Math.min(gradient.stops.length - 1, index)); |
|
} |
|
deletionRestore = undefined; |
|
|
|
dispatch("activeMarkerIndexChange", activeMarkerIndex); |
|
dispatch("gradient", gradient); |
|
} |
|
|
|
function moveMarker(e: PointerEvent, index: number) { |
|
// Just in case the mouseup event is lost |
|
if (e.buttons === 0) stopDrag(); |
|
|
|
let position = markerPosition(e); |
|
if (position === undefined) return; |
|
|
|
if (positionRestore === undefined) positionRestore = position; |
|
if (deletionRestore === undefined) { |
|
deletionRestore = false; |
|
|
|
dispatch("dragging", true); |
|
} |
|
|
|
setPosition(index, position); |
|
} |
|
|
|
export function setPosition(index: number, position: number) { |
|
const active = gradient.stops[index]; |
|
active.position = position; |
|
gradient.stops.sort((a, b) => a.position - b.position); |
|
if (gradient.stops.indexOf(active) !== activeMarkerIndex) { |
|
activeMarkerIndex = gradient.stops.indexOf(active); |
|
dispatch("activeMarkerIndexChange", gradient.stops.indexOf(active)); |
|
} |
|
dispatch("gradient", gradient); |
|
} |
|
|
|
function abortDrag() { |
|
if (activeMarkerIndex === undefined) return; |
|
|
|
if (deletionRestore) { |
|
deleteStopByIndex(activeMarkerIndex); |
|
} else if (positionRestore !== undefined) { |
|
setPosition(activeMarkerIndex, positionRestore); |
|
} |
|
|
|
stopDrag(); |
|
} |
|
|
|
function stopDrag() { |
|
removeEvents(); |
|
|
|
positionRestore = undefined; |
|
deletionRestore = undefined; |
|
|
|
dispatch("dragging", false); |
|
} |
|
|
|
function onPointerMove(e: PointerEvent) { |
|
if (activeMarkerIndex !== undefined) moveMarker(e, activeMarkerIndex); |
|
} |
|
|
|
function onPointerUp() { |
|
stopDrag(); |
|
} |
|
|
|
function onMouseDown(e: MouseEvent) { |
|
const BUTTONS_RIGHT = 0b0000_0010; |
|
if (e.buttons & BUTTONS_RIGHT) abortDrag(); |
|
} |
|
|
|
function onKeyDown(e: KeyboardEvent) { |
|
if (e.key === "Escape") { |
|
const element = markerTrack?.div(); |
|
if (element) preventEscapeClosingParentFloatingMenu(element); |
|
|
|
abortDrag(); |
|
} |
|
} |
|
|
|
function addEvents() { |
|
document.addEventListener("pointermove", onPointerMove); |
|
document.addEventListener("pointerup", onPointerUp); |
|
document.addEventListener("mousedown", onMouseDown); |
|
document.addEventListener("keydown", onKeyDown); |
|
} |
|
|
|
function removeEvents() { |
|
document.removeEventListener("pointermove", onPointerMove); |
|
document.removeEventListener("pointerup", onPointerUp); |
|
document.removeEventListener("mousedown", onMouseDown); |
|
document.removeEventListener("keydown", onKeyDown); |
|
} |
|
|
|
document.addEventListener("keydown", deleteStop); |
|
onDestroy(() => { |
|
removeEvents(); |
|
document.removeEventListener("keydown", deleteStop); |
|
}); |
|
|
|
|
|
{ position, color }[], active) |
|
|
|
{ index, position, color }[]) |
|
|
|
|
|
|
|
{ index, position }[]) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
</script> |
|
|
|
<LayoutCol |
|
class="spectrum-input" |
|
styles={{ |
|
"--gradient-start": gradient.firstColor()?.toHexOptionalAlpha() || "black", |
|
"--gradient-end": gradient.lastColor()?.toHexOptionalAlpha() || "black", |
|
"--gradient-stops": gradient.toLinearGradientCSS(), |
|
}} |
|
> |
|
<LayoutRow class="gradient-strip" on:pointerdown={insertStop}></LayoutRow> |
|
<LayoutRow class="marker-track" bind:this={markerTrack}> |
|
{#each gradient.stops as marker, index} |
|
<svg |
|
class="marker" |
|
class:active={index === activeMarkerIndex} |
|
style:--marker-position={marker.position} |
|
style:--marker-color={marker.color.toRgbCSS()} |
|
on:pointerdown={(e) => markerPointerDown(e, index)} |
|
data-gradient-marker |
|
xmlns="http://www.w3.org/2000/svg" |
|
viewBox="0 0 12 12" |
|
> |
|
<path class="inner-fill" d="M10,11.5H2c-0.8,0-1.5-0.7-1.5-1.5V6.8c0-0.4,0.2-0.8,0.4-1.1L6,0.7l5.1,5.1c0.3,0.3,0.4,0.7,0.4,1.1V10C11.5,10.8,10.8,11.5,10,11.5z" /> |
|
<path |
|
class="outer-border" |
|
d="M6,1.4L1.3,6.1C1.1,6.3,1,6.6,1,6.8V10c0,0.6,0.4,1,1,1h8c0.6,0,1-0.4,1-1V6.8c0-0.3-0.1-0.5-0.3-0.7L6,1.4M6,0l5.4,5.4C11.8,5.8,12,6.3,12,6.8V10c0,1.1-0.9,2-2,2H2c-1.1,0-2-0.9-2-2V6.8c0-0.5,0.2-1,0.6-1.4L6,0z" |
|
/> |
|
</svg> |
|
{/each} |
|
</LayoutRow> |
|
</LayoutCol> |
|
|
|
<style lang="scss" global> |
|
.spectrum-input { |
|
--marker-half-width: 6px; |
|
|
|
.gradient-strip { |
|
flex: 0 0 auto; |
|
height: 16px; |
|
background-image: |
|
var(--gradient-stops), |
|
// Solid start/end colors on either side so the gradient begins at the center of a marker |
|
linear-gradient(var(--gradient-start), var(--gradient-start)), |
|
linear-gradient(var(--gradient-end), var(--gradient-end)), |
|
var(--color-transparent-checkered-background); |
|
background-size: |
|
calc(100% - 2 * var(--marker-half-width)) 100%, |
|
// TODO: Find a solution that avoids visual artifacts where these end colors meet the gradient that appear when viewing with a non-integer zoom or display scaling factor |
|
var(--marker-half-width) 100%, |
|
var(--marker-half-width) 100%, |
|
var(--color-transparent-checkered-background-size); |
|
background-position: |
|
var(--marker-half-width) 0, |
|
left 0, |
|
right 0, |
|
var(--color-transparent-checkered-background-position); |
|
background-repeat: no-repeat, no-repeat, no-repeat, var(--color-transparent-checkered-background-repeat); |
|
border-radius: 2px; |
|
} |
|
|
|
.marker-track { |
|
margin-top: calc(24px - 16px - 12px); |
|
margin-left: var(--marker-half-width); |
|
width: calc(100% - 2 * var(--marker-half-width)); |
|
position: relative; |
|
pointer-events: none; |
|
|
|
.marker { |
|
position: absolute; |
|
transform: translateX(-50%); |
|
left: calc(var(--marker-position) * 100%); |
|
width: 12px; |
|
height: 12px; |
|
pointer-events: auto; |
|
overflow: visible; |
|
padding-top: 12px; |
|
margin-top: -12px; |
|
|
|
.inner-fill { |
|
fill: var(--marker-color); |
|
} |
|
|
|
.outer-border { |
|
fill: var(--color-5-dullgray); |
|
} |
|
|
|
&:not(.active) { |
|
.inner-fill:hover + .outer-border, |
|
.outer-border:hover { |
|
fill: var(--color-6-lowergray); |
|
} |
|
} |
|
|
|
&.active { |
|
.inner-fill { |
|
filter: drop-shadow(0 0 1px var(--color-2-mildblack)) drop-shadow(0 0 1px var(--color-2-mildblack)); |
|
} |
|
|
|
// Outer border when active |
|
.outer-border { |
|
fill: var(--color-e-nearwhite); |
|
} |
|
|
|
.inner-fill:hover + .outer-border, |
|
.outer-border:hover { |
|
fill: var(--color-f-white); |
|
} |
|
} |
|
} |
|
} |
|
} |
|
</style> |
|
|