108 lines
2.9 KiB
Svelte
108 lines
2.9 KiB
Svelte
|
<!--
|
||
|
Copyright: Ankitects Pty Ltd and contributors
|
||
|
License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
||
|
-->
|
||
|
<script lang="ts">
|
||
|
import type { Placement } from "@floating-ui/dom";
|
||
|
import { onMount } from "svelte";
|
||
|
import { writable } from "svelte/store";
|
||
|
|
||
|
import isClosingClick from "../sveltelib/closing-click";
|
||
|
import isClosingKeyup from "../sveltelib/closing-keyup";
|
||
|
import { documentClick, documentKeyup } from "../sveltelib/event-store";
|
||
|
import portal from "../sveltelib/portal";
|
||
|
import position from "../sveltelib/position";
|
||
|
import subscribeTrigger from "../sveltelib/subscribe-trigger";
|
||
|
import { pageTheme } from "../sveltelib/theme";
|
||
|
import toggleable from "../sveltelib/toggleable";
|
||
|
|
||
|
/** TODO at the moment we only dropdowns which are placed actually below the reference */
|
||
|
const placement: Placement = "bottom";
|
||
|
|
||
|
export let closeOnInsideClick = false;
|
||
|
|
||
|
/** This may be passed in for more fine-grained control */
|
||
|
export let show = writable(false);
|
||
|
|
||
|
let reference: HTMLElement;
|
||
|
let floating: HTMLElement;
|
||
|
let arrow: HTMLElement;
|
||
|
|
||
|
const { toggle, on, off } = toggleable(show);
|
||
|
|
||
|
onMount(() =>
|
||
|
subscribeTrigger(
|
||
|
show,
|
||
|
isClosingClick(documentClick, {
|
||
|
reference,
|
||
|
floating,
|
||
|
inside: closeOnInsideClick,
|
||
|
outside: true,
|
||
|
}),
|
||
|
isClosingKeyup(documentKeyup, {
|
||
|
reference,
|
||
|
floating,
|
||
|
}),
|
||
|
),
|
||
|
);
|
||
|
</script>
|
||
|
|
||
|
<div
|
||
|
bind:this={reference}
|
||
|
class="reference"
|
||
|
use:position={{ floating: $show ? floating : null, placement, arrow }}
|
||
|
>
|
||
|
<slot name="reference" {show} {toggle} {on} {off} />
|
||
|
</div>
|
||
|
|
||
|
<div bind:this={floating} class="floating" hidden={!$show} use:portal>
|
||
|
<slot name="floating" />
|
||
|
|
||
|
<div bind:this={arrow} class="arrow" class:dark={$pageTheme.isDark} />
|
||
|
</div>
|
||
|
|
||
|
<style lang="scss">
|
||
|
@use "sass/elevation" as elevation;
|
||
|
|
||
|
.reference {
|
||
|
/* TODO This should not be necessary */
|
||
|
line-height: normal;
|
||
|
}
|
||
|
|
||
|
.floating {
|
||
|
position: absolute;
|
||
|
border-radius: 5px;
|
||
|
|
||
|
z-index: 90;
|
||
|
@include elevation.elevation(8);
|
||
|
}
|
||
|
|
||
|
.arrow {
|
||
|
position: absolute;
|
||
|
background-color: var(--frame-bg);
|
||
|
width: 10px;
|
||
|
height: 10px;
|
||
|
transform: rotate(45deg);
|
||
|
z-index: 60;
|
||
|
|
||
|
/* outer border */
|
||
|
border: 1px solid #b6b6b6;
|
||
|
|
||
|
&.dark {
|
||
|
border-color: #060606;
|
||
|
}
|
||
|
|
||
|
/* These are dependant on which edge the arrow is supposed to be */
|
||
|
border-right: none;
|
||
|
border-bottom: none;
|
||
|
|
||
|
/* inner border */
|
||
|
box-shadow: inset 1px 1px 0 0 #eeeeee;
|
||
|
/* lightmode box-shadow: inset 1px 1px 0 0 #eee; */
|
||
|
|
||
|
&.dark {
|
||
|
box-shadow: inset 0 0 0 1px #565656;
|
||
|
}
|
||
|
}
|
||
|
</style>
|