budibase/packages/bbui/src/IconPicker/IconPicker.svelte

185 lines
3.8 KiB
Svelte
Raw Normal View History

2022-06-14 15:56:09 +02:00
<script>
2022-06-14 15:57:34 +02:00
//import { createEventDispatcher } from "svelte"
2022-06-14 15:56:09 +02:00
import "@spectrum-css/popover/dist/index-vars.css"
import clickOutside from "../Actions/click_outside"
import { fly } from "svelte/transition"
import Icon from "../Icon/Icon.svelte"
2022-06-22 14:55:31 +02:00
import { createEventDispatcher } from "svelte"
2022-06-14 15:56:09 +02:00
2022-07-05 10:21:59 +02:00
export let value
2022-06-14 15:56:09 +02:00
export let size = "M"
export let alignRight = false
let open = false
2022-06-22 14:55:31 +02:00
const dispatch = createEventDispatcher()
2022-06-14 15:56:09 +02:00
const iconList = [
{
label: "Icons",
icons: [
"Apps",
"Actions",
"ConversionFunnel",
"App",
"Briefcase",
"Money",
"ShoppingCart",
"Form",
"Help",
"Monitoring",
"Sandbox",
"Project",
"Organisations",
"Magnify",
"Launch",
"Car",
"Camera",
"Bug",
"Channel",
"Calculator",
"Calendar",
"GraphDonut",
"GraphBarHorizontal",
"Demographic",
],
},
]
2022-06-22 14:55:31 +02:00
2022-06-14 15:56:09 +02:00
const onChange = value => {
dispatch("change", value)
open = false
}
const handleOutsideClick = event => {
if (open) {
event.stopPropagation()
open = false
}
}
2022-06-14 15:56:09 +02:00
</script>
<div class="container">
<div class="preview size--{size || 'M'}" on:click={() => (open = true)}>
<div
class="fill"
style={value ? `background: ${value};` : ""}
class:placeholder={!value}
>
2022-07-05 10:21:59 +02:00
<Icon name={value || "UserGroup"} />
2022-06-14 15:56:09 +02:00
</div>
</div>
{#if open}
<div
use:clickOutside={handleOutsideClick}
2022-06-14 15:56:09 +02:00
transition:fly={{ y: -20, duration: 200 }}
class="spectrum-Popover spectrum-Popover--bottom spectrum-Picker-popover is-open"
class:spectrum-Popover--align-right={alignRight}
>
{#each iconList as icon}
<div class="category">
<div class="heading">{icon.label}</div>
<div class="icons">
{#each icon.icons as icon}
<div
on:click={() => {
2022-06-22 14:55:31 +02:00
onChange(icon)
2022-06-14 15:56:09 +02:00
}}
>
<Icon name={icon} />
</div>
{/each}
</div>
</div>
{/each}
</div>
{/if}
</div>
<style>
.container {
position: relative;
}
.preview {
width: 32px;
height: 32px;
position: relative;
box-shadow: 0 0 0 1px var(--spectrum-global-color-gray-400);
}
.preview:hover {
cursor: pointer;
}
.fill {
width: 100%;
height: 100%;
position: absolute;
top: 0;
left: 0;
display: grid;
place-items: center;
}
.size--S {
width: 20px;
height: 20px;
}
.size--M {
width: 32px;
height: 32px;
}
.size--L {
width: 48px;
height: 48px;
}
.spectrum-Popover {
width: 210px;
z-index: 999;
top: 100%;
padding: var(--spacing-l) var(--spacing-xl);
display: flex;
flex-direction: column;
justify-content: flex-start;
align-items: stretch;
gap: var(--spacing-xl);
}
.spectrum-Popover--align-right {
right: 0;
}
.icons {
display: grid;
grid-template-columns: 1fr 1fr 1fr 1fr 1fr 1fr;
gap: var(--spacing-m);
}
.heading {
font-size: var(--font-size-s);
font-weight: 600;
letter-spacing: 0.14px;
flex: 1 1 auto;
text-transform: uppercase;
grid-column: 1 / 5;
margin-bottom: var(--spacing-s);
}
.icon {
height: 16px;
width: 16px;
border-radius: 100%;
box-shadow: 0 0 0 1px var(--spectrum-global-color-gray-300);
position: relative;
}
.icon:hover {
cursor: pointer;
box-shadow: 0 0 2px 2px var(--spectrum-global-color-gray-300);
}
.custom {
display: grid;
grid-template-columns: 1fr auto;
align-items: center;
gap: var(--spacing-m);
margin-right: var(--spacing-xs);
}
.spectrum-wrapper {
background-color: transparent;
}
</style>