Corrected No date. Fix Status and Priority editors. Little fixes. (#1697)

Signed-off-by: Alexander Platov <sas_lord@mail.ru>
This commit is contained in:
Alexander Platov 2022-05-09 14:37:34 +03:00 committed by GitHub
parent 9266a61c98
commit b8e942a145
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
16 changed files with 239 additions and 289 deletions

View File

@ -133,6 +133,8 @@
placeholder={attribute?.label} placeholder={attribute?.label}
kind={'link'} kind={'link'}
size={'large'} size={'large'}
width={'100%'}
justify={'left'}
type={attribute?.type} type={attribute?.type}
{maxWidth} {maxWidth}
value={getAttribute(client, object, { key: attributeKey, attr: attribute })} value={getAttribute(client, object, { key: attributeKey, attr: attribute })}

View File

@ -69,7 +69,6 @@
&.vertical { &.vertical {
display: grid; display: grid;
grid-template-columns: 1fr 1.5fr; grid-template-columns: 1fr 1.5fr;
grid-template-rows: minmax(2rem, auto);
grid-auto-flow: row; grid-auto-flow: row;
justify-content: start; justify-content: start;
align-items: center; align-items: center;

View File

@ -427,7 +427,8 @@
// THead background-color in Tooltip and Popups // THead background-color in Tooltip and Popups
.popup-tooltip .antiTable .scroller-thead, .popup-tooltip .antiTable .scroller-thead,
.popup .antiTable .scroller-thead { background-color: var(--accent-bg-color); } .popup .antiTable .scroller-thead,
.antiDialogs .antiTable .scroller-thead { background-color: var(--accent-bg-color); }
// Hide row menu in Tooltip // Hide row menu in Tooltip
.popup-tooltip .antiTable .antiTable-body__row:hover .antiTable-cells__firstCell .antiTable-cells__firstCell-menuRow { visibility: hidden; } .popup-tooltip .antiTable .antiTable-body__row:hover .antiTable-cells__firstCell .antiTable-cells__firstCell-menuRow { visibility: hidden; }

View File

@ -22,7 +22,7 @@
import DPCalendarOver from './icons/DPCalendarOver.svelte' import DPCalendarOver from './icons/DPCalendarOver.svelte'
import DateRangePopup from './DateRangePopup.svelte' import DateRangePopup from './DateRangePopup.svelte'
export let value: number | null | undefined export let value: number | null | undefined = null
export let withTime: boolean = false export let withTime: boolean = false
export let editable: boolean = false export let editable: boolean = false
export let icon: 'normal' | 'warning' | 'overdue' = 'normal' export let icon: 'normal' | 'warning' | 'overdue' = 'normal'
@ -363,10 +363,10 @@
<div class="btn-icon {icon}"> <div class="btn-icon {icon}">
<Icon icon={icon === 'overdue' ? DPCalendarOver : DPCalendar} size={'full'} /> <Icon icon={icon === 'overdue' ? DPCalendarOver : DPCalendar} size={'full'} />
</div> </div>
{#if value !== null && value !== undefined} {#if value !== undefined && value !== null && value.toString() !== ''}
{#if labelOver !== undefined} {#if labelOver !== undefined}
<Label label={labelOver} /> <Label label={labelOver} />
{:else} {:else if value}
{new Date(value).getDate()} {new Date(value).getDate()}
{getMonthName(new Date(value), 'short')} {getMonthName(new Date(value), 'short')}
{#if new Date(value).getFullYear() !== today.getFullYear()} {#if new Date(value).getFullYear() !== today.getFullYear()}

View File

@ -240,16 +240,3 @@
{/if} {/if}
</svelte:fragment> </svelte:fragment>
</Card> </Card>
<style lang="scss">
.resume {
padding: 0.5rem 0.75rem;
background: var(--accent-bg-color);
border: 1px dashed var(--divider-color);
border-radius: 0.5rem;
&.solid {
border-style: solid;
}
}
</style>

View File

@ -14,28 +14,36 @@
--> -->
<script lang="ts"> <script lang="ts">
import type { IntlString } from '@anticrm/platform' import type { IntlString } from '@anticrm/platform'
import type { TooltipAlignment } from '@anticrm/ui' import { Label, Tooltip, Button } from '@anticrm/ui'
import { Label, Tooltip } from '@anticrm/ui' import type { TooltipAlignment, ButtonKind, ButtonSize } from '@anticrm/ui'
export let label: IntlString export let label: IntlString
export let tooltip: IntlString export let tooltip: IntlString
export let value: boolean | undefined export let value: boolean | undefined
export let disabled: boolean = false export let disabled: boolean = false
export let labelDirection: TooltipAlignment | undefined = undefined export let labelDirection: TooltipAlignment | undefined = undefined
export let kind: ButtonKind = 'no-border'
export let size: ButtonSize = 'small'
export let justify: 'left' | 'center' = 'center'
export let width: string | undefined = 'fit-content'
</script> </script>
<Tooltip direction={labelDirection} label={tooltip}> <Tooltip direction={labelDirection} label={tooltip}>
<button <Button
class="yesno-container" {kind}
{size}
{justify}
{width}
{disabled} {disabled}
class:yes={value === true}
class:no={value === false}
on:click={() => { on:click={() => {
if (value === true) value = false if (value === true) value = false
else if (value === false) value = undefined else if (value === false) value = undefined
else value = true else value = true
}} }}
> >
<svelte:fragment slot="content">
<div class="flex-row-center flex-no-wrap">
<span class="overflow-label"> <span class="overflow-label">
<Label {label} /> <Label {label} />
</span> </span>
@ -57,54 +65,24 @@
{/if} {/if}
</svg> </svg>
</div> </div>
</button> </div>
</svelte:fragment>
</Button>
</Tooltip> </Tooltip>
<style lang="scss"> <style lang="scss">
.yesno-container {
display: flex;
align-items: center;
flex-shrink: 0;
padding: 0 0.25rem 0 0.5rem;
min-width: 1.5rem;
height: 1.5rem;
font-weight: 400;
line-height: 1.5rem;
white-space: nowrap;
color: var(--accent-color);
background-color: var(--noborder-bg-color);
border: 1px solid transparent;
border-radius: 0.25rem;
box-shadow: var(--button-shadow);
transition-property: border, background-color, color, box-shadow;
transition-duration: 0.15s;
.btn-icon { .btn-icon {
color: var(--content-color); color: var(--content-color);
transition: color 0.15s; transition: color 0.15s;
pointer-events: none; pointer-events: none;
}
&:hover { &:hover {
color: var(--caption-color); color: var(--caption-color);
background-color: var(--noborder-bg-hover); }
transition-duration: 0; &:disabled:hover {
color: var(--content-color);
}
}
.btn-icon {
color: var(--caption-color);
}
}
&:disabled {
color: var(--content-color);
background-color: #30323655;
&:hover {
color: var(--content-color);
.btn-icon {
color: var(--content-color);
}
}
}
}
.yesno-svg { .yesno-svg {
width: 1rem; width: 1rem;
height: 1rem; height: 1rem;

View File

@ -15,7 +15,7 @@
<script lang="ts"> <script lang="ts">
import core from '@anticrm/core' import core from '@anticrm/core'
import { getClient, SpaceCreateCard } from '@anticrm/presentation' import { getClient, SpaceCreateCard } from '@anticrm/presentation'
import { EditBox, Grid } from '@anticrm/ui' import { EditBox } from '@anticrm/ui'
import { createEventDispatcher } from 'svelte' import { createEventDispatcher } from 'svelte'
import recruit from '../../plugin' import recruit from '../../plugin'
import Review from '../icons/Review.svelte' import Review from '../icons/Review.svelte'
@ -50,7 +50,6 @@
dispatch('close') dispatch('close')
}} }}
> >
<Grid column={1} rowGap={1.5}>
<EditBox <EditBox
label={recruit.string.ReviewCategoryName} label={recruit.string.ReviewCategoryName}
bind:value={name} bind:value={name}
@ -59,5 +58,4 @@
maxWidth={'16rem'} maxWidth={'16rem'}
focus focus
/> />
</Grid>
</SpaceCreateCard> </SpaceCreateCard>

View File

@ -27,7 +27,7 @@
let name: string let name: string
const done = false const done = false
let dueTo: number let dueTo: number | null = null
$: _space = space $: _space = space

View File

@ -46,7 +46,7 @@
"Title": "Заголовок", "Title": "Заголовок",
"Description": "", "Description": "",
"Status": "", "Status": "Статус",
"Number": "Number", "Number": "Number",
"Assignee": "Исполнитель", "Assignee": "Исполнитель",
"AssignTo": "Назначить...", "AssignTo": "Назначить...",

View File

@ -14,16 +14,21 @@
--> -->
<script lang="ts"> <script lang="ts">
import { IssuePriority } from '@anticrm/tracker' import { IssuePriority } from '@anticrm/tracker'
import { Button, showPopup, SelectPopup, Icon, Label, eventToHTMLElement } from '@anticrm/ui' import { Button, showPopup, SelectPopup, eventToHTMLElement } from '@anticrm/ui'
import type { ButtonKind, ButtonSize } from '@anticrm/ui'
import { issuePriorities } from '../utils' import { issuePriorities } from '../utils'
import tracker from '../plugin' import tracker from '../plugin'
export let priority: IssuePriority export let priority: IssuePriority
export let kind: 'button' | 'icon' = 'button'
export let shouldShowLabel: boolean = true export let shouldShowLabel: boolean = true
export let onPriorityChange: ((newPriority: IssuePriority | undefined) => void) | undefined = undefined export let onPriorityChange: ((newPriority: IssuePriority | undefined) => void) | undefined = undefined
export let isEditable: boolean = true export let isEditable: boolean = true
export let kind: ButtonKind = 'no-border'
export let size: ButtonSize = 'small'
export let justify: 'left' | 'center' = 'center'
export let width: string | undefined = 'min-content'
const prioritiesInfo = [ const prioritiesInfo = [
IssuePriority.NoPriority, IssuePriority.NoPriority,
IssuePriority.Urgent, IssuePriority.Urgent,
@ -45,44 +50,13 @@
} }
</script> </script>
{#if kind === 'button'} <Button
<Button
label={shouldShowLabel ? issuePriorities[priority].label : undefined} label={shouldShowLabel ? issuePriorities[priority].label : undefined}
icon={issuePriorities[priority].icon} icon={issuePriorities[priority].icon}
width="min-content" {justify}
size="small" {width}
kind="no-border" {size}
{kind}
disabled={!isEditable}
on:click={handlePriorityEditorOpened} on:click={handlePriorityEditorOpened}
/> />
{:else if kind === 'icon'}
<div class={isEditable ? 'flex-presenter' : 'presenter'} on:click={handlePriorityEditorOpened}>
<div class="priorityIcon" class:mPriorityIconEditable={isEditable}>
<Icon icon={issuePriorities[priority].icon} size={'small'} />
</div>
{#if shouldShowLabel}
<div class="label nowrap ml-2">
<Label label={issuePriorities[priority].label} />
</div>
{/if}
</div>
{/if}
<style lang="scss">
.presenter {
display: flex;
align-items: center;
flex-wrap: nowrap;
}
.priorityIcon {
width: 1rem;
height: 1rem;
color: var(--theme-content-dark-color);
&.mPriorityIconEditable {
&:hover {
color: var(--theme-caption-color);
}
}
}
</style>

View File

@ -16,16 +16,21 @@
import { Ref, WithLookup } from '@anticrm/core' import { Ref, WithLookup } from '@anticrm/core'
import { IssueStatus } from '@anticrm/tracker' import { IssueStatus } from '@anticrm/tracker'
import { Button, Icon, showPopup, SelectPopup, eventToHTMLElement } from '@anticrm/ui' import { Button, showPopup, SelectPopup, eventToHTMLElement } from '@anticrm/ui'
import type { ButtonKind, ButtonSize } from '@anticrm/ui'
import tracker from '../plugin' import tracker from '../plugin'
export let selectedStatusId: Ref<IssueStatus> export let selectedStatusId: Ref<IssueStatus>
export let statuses: WithLookup<IssueStatus>[] export let statuses: WithLookup<IssueStatus>[]
export let kind: 'button' | 'icon' = 'button'
export let shouldShowLabel: boolean = true export let shouldShowLabel: boolean = true
export let onStatusChange: ((newStatus: Ref<IssueStatus> | undefined) => void) | undefined = undefined export let onStatusChange: ((newStatus: Ref<IssueStatus> | undefined) => void) | undefined = undefined
export let isEditable: boolean = true export let isEditable: boolean = true
export let kind: ButtonKind = 'no-border'
export let size: ButtonSize = 'small'
export let justify: 'left' | 'center' = 'center'
export let width: string | undefined = 'min-content'
$: selectedStatus = statuses.find((status) => status._id === selectedStatusId) ?? statuses[0] $: selectedStatus = statuses.find((status) => status._id === selectedStatusId) ?? statuses[0]
$: selectedStatusIcon = selectedStatus?.$lookup?.category?.icon $: selectedStatusIcon = selectedStatus?.$lookup?.category?.icon
$: selectedStatusLabel = shouldShowLabel ? selectedStatus?.name : undefined $: selectedStatusLabel = shouldShowLabel ? selectedStatus?.name : undefined
@ -44,42 +49,18 @@
} }
</script> </script>
{#if kind === 'button'} <Button
<Button
icon={selectedStatusIcon} icon={selectedStatusIcon}
width="min-content" {justify}
size="small" {width}
kind="no-border" {size}
{kind}
disabled={!isEditable}
on:click={handleStatusEditorOpened} on:click={handleStatusEditorOpened}
> >
<svelte:fragment slot="content"> <svelte:fragment slot="content">
{#if selectedStatusLabel} {#if selectedStatusLabel}
<span class="nowrap">{selectedStatusLabel}</span> <span class="nowrap">{selectedStatusLabel}</span>
{/if} {/if}
</svelte:fragment> </svelte:fragment>
</Button> </Button>
{:else if kind === 'icon'}
<div class={isEditable ? 'flex-presenter' : 'presenter'} on:click={handleStatusEditorOpened}>
{#if selectedStatusIcon}
<div class="statusIcon">
<Icon icon={selectedStatusIcon} size={'small'} />
</div>
{/if}
{#if selectedStatusLabel}
<div class="label nowrap ml-2">{selectedStatusLabel}</div>
{/if}
</div>
{/if}
<style lang="scss">
.presenter {
display: flex;
align-items: center;
flex-wrap: nowrap;
}
.statusIcon {
width: 1rem;
height: 1rem;
}
</style>

View File

@ -169,23 +169,18 @@
<svelte:fragment slot="custom-attributes" let:direction> <svelte:fragment slot="custom-attributes" let:direction>
{#if issue && currentTeam && issueStatuses && direction === 'column'} {#if issue && currentTeam && issueStatuses && direction === 'column'}
<div class="content mt-4"> <div class="content">
<div class="flex-row-center mb-4"> <span class="label">
<span class="label w-24">
<Label label={tracker.string.Status} /> <Label label={tracker.string.Status} />
</span> </span>
<StatusEditor value={issue} statuses={issueStatuses} currentSpace={currentTeam._id} shouldShowLabel /> <StatusEditor value={issue} statuses={issueStatuses} currentSpace={currentTeam._id} shouldShowLabel />
</div>
<div class="flex-row-center mb-4"> <span class="label">
<span class="label w-24">
<Label label={tracker.string.Priority} /> <Label label={tracker.string.Priority} />
</span> </span>
<PriorityEditor value={issue} currentSpace={currentTeam._id} shouldShowLabel /> <PriorityEditor value={issue} currentSpace={currentTeam._id} shouldShowLabel />
</div>
<div class="flex-row-center mb-4"> <span class="label">
<span class="label w-24">
<Label label={tracker.string.Assignee} /> <Label label={tracker.string.Assignee} />
</span> </span>
<UserBox <UserBox
@ -195,53 +190,46 @@
bind:value={issue.assignee} bind:value={issue.assignee}
allowDeselect allowDeselect
titleDeselect={tracker.string.Unassigned} titleDeselect={tracker.string.Unassigned}
size="large" size={'large'}
kind="link" kind={'link'}
width={'100%'}
justify={'left'}
on:change={() => change('assignee', issue?.assignee)} on:change={() => change('assignee', issue?.assignee)}
/> />
</div>
<div class="flex-row-center mb-4"> <span class="label">
<span class="label w-24">
<Label label={tracker.string.Labels} /> <Label label={tracker.string.Labels} />
</span> </span>
<Button <Button
label={tracker.string.Labels} label={tracker.string.Labels}
icon={tracker.icon.Labels} icon={tracker.icon.Labels}
width="max-content" size={'large'}
size="large" kind={'link'}
kind="link" width={'100%'}
justify={'left'}
/> />
</div>
<div class="devider" /> <div class="divider" />
<div class="flex-row-center mb-4"> <span class="label">
<span class="label w-24">
<Label label={tracker.string.Project} /> <Label label={tracker.string.Project} />
</span> </span>
<Button <Button
label={tracker.string.Project} label={tracker.string.Project}
icon={tracker.icon.Projects} icon={tracker.icon.Projects}
width="fit-content" size={'large'}
size="large" kind={'link'}
kind="link" width={'100%'}
justify={'left'}
/> />
</div>
{#if issue.dueDate !== null} {#if issue.dueDate !== null}
<div class="devider" /> <div class="divider" />
<div class="flex-row-center mb-4"> <span class="label">
<span class="label w-24">
<Label label={tracker.string.DueDate} /> <Label label={tracker.string.DueDate} />
</span> </span>
<DatePresenter <DatePresenter bind:value={issue.dueDate} editable on:change={({ detail }) => change('dueDate', detail)} />
bind:value={issue.dueDate}
editable
on:change={({ detail }) => change('dueDate', detail)}
/>
</div>
{/if} {/if}
</div> </div>
{:else} {:else}
@ -268,22 +256,20 @@
<style lang="scss"> <style lang="scss">
.content { .content {
display: grid;
grid-template-columns: 1fr 1.5fr;
grid-auto-flow: row;
justify-content: start;
align-items: center;
gap: 1rem;
margin-top: 1rem;
width: 100%; width: 100%;
height: min-content;
} }
.content { .divider {
position: absolute; grid-column: 1 / 3;
inset: 2.5rem 0 0;
padding: 1.5rem 0.5rem 1.5rem 1.5rem;
.label {
margin: 0.625rem 0;
}
}
.devider {
height: 1px; height: 1px;
border-bottom: 1px solid var(--divider-color); background-color: var(--divider-color);
margin: 0.75rem 1.5rem 1.25rem 0;
} }
</style> </style>

View File

@ -17,6 +17,7 @@
import { Issue, IssuePriority, Team } from '@anticrm/tracker' import { Issue, IssuePriority, Team } from '@anticrm/tracker'
import { getClient } from '@anticrm/presentation' import { getClient } from '@anticrm/presentation'
import { Tooltip } from '@anticrm/ui' import { Tooltip } from '@anticrm/ui'
import type { ButtonKind, ButtonSize } from '@anticrm/ui'
import tracker from '../../plugin' import tracker from '../../plugin'
import PrioritySelector from '../PrioritySelector.svelte' import PrioritySelector from '../PrioritySelector.svelte'
@ -25,6 +26,11 @@
export let isEditable: boolean = true export let isEditable: boolean = true
export let shouldShowLabel: boolean = false export let shouldShowLabel: boolean = false
export let kind: ButtonKind = 'link'
export let size: ButtonSize = 'large'
export let justify: 'left' | 'center' = 'left'
export let width: string | undefined = '100%'
const client = getClient() const client = getClient()
const handlePriorityChanged = async (newPriority: IssuePriority | undefined) => { const handlePriorityChanged = async (newPriority: IssuePriority | undefined) => {
@ -44,9 +50,12 @@
{#if value} {#if value}
{#if isEditable} {#if isEditable}
<Tooltip direction={'bottom'} label={tracker.string.SetPriority}> <Tooltip label={tracker.string.SetPriority} fill>
<PrioritySelector <PrioritySelector
kind={'icon'} {kind}
{size}
{width}
{justify}
{isEditable} {isEditable}
{shouldShowLabel} {shouldShowLabel}
priority={value.priority} priority={value.priority}
@ -54,6 +63,6 @@
/> />
</Tooltip> </Tooltip>
{:else} {:else}
<PrioritySelector kind={'icon'} {isEditable} {shouldShowLabel} priority={value.priority} /> <PrioritySelector {kind} {size} {width} {justify} {isEditable} {shouldShowLabel} priority={value.priority} />
{/if} {/if}
{/if} {/if}

View File

@ -17,6 +17,7 @@
import { Issue, IssueStatus, Team } from '@anticrm/tracker' import { Issue, IssueStatus, Team } from '@anticrm/tracker'
import { getClient } from '@anticrm/presentation' import { getClient } from '@anticrm/presentation'
import { Tooltip } from '@anticrm/ui' import { Tooltip } from '@anticrm/ui'
import type { ButtonKind, ButtonSize } from '@anticrm/ui'
import tracker from '../../plugin' import tracker from '../../plugin'
import StatusSelector from '../StatusSelector.svelte' import StatusSelector from '../StatusSelector.svelte'
@ -26,6 +27,11 @@
export let isEditable: boolean = true export let isEditable: boolean = true
export let shouldShowLabel: boolean = false export let shouldShowLabel: boolean = false
export let kind: ButtonKind = 'link'
export let size: ButtonSize = 'large'
export let justify: 'left' | 'center' = 'left'
export let width: string | undefined = '100%'
const client = getClient() const client = getClient()
const handleStatusChanged = async (newStatus: Ref<IssueStatus> | undefined) => { const handleStatusChanged = async (newStatus: Ref<IssueStatus> | undefined) => {
@ -45,9 +51,12 @@
{#if value} {#if value}
{#if isEditable} {#if isEditable}
<Tooltip direction={'bottom'} label={tracker.string.SetStatus}> <Tooltip label={tracker.string.SetStatus} fill>
<StatusSelector <StatusSelector
kind={'icon'} {kind}
{size}
{width}
{justify}
{isEditable} {isEditable}
{shouldShowLabel} {shouldShowLabel}
{statuses} {statuses}
@ -57,7 +66,10 @@
</Tooltip> </Tooltip>
{:else} {:else}
<StatusSelector <StatusSelector
kind={'icon'} {kind}
{size}
{width}
{justify}
{isEditable} {isEditable}
{shouldShowLabel} {shouldShowLabel}
{statuses} {statuses}

View File

@ -229,13 +229,13 @@
{#if i !== 0} {#if i !== 0}
<div class="ml-2 mr-2">or</div> <div class="ml-2 mr-2">or</div>
{/if} {/if}
<div class="flex-row-center" class:ml-2={i !== 0}> <div class="flex-row-center">
{#each formatKey(key) as k, jj} {#each formatKey(key) as k, jj}
{#if jj !== 0} {#if jj !== 0}
<div class="ml-1 mr-1">then</div> <div class="ml-1 mr-1">then</div>
{/if} {/if}
{#each k as kk, j} {#each k as kk, j}
<div class="flex-center text-sm key-box" class:ml-2={j !== 0}> <div class="flex-center text-sm key-box">
{kk} {kk}
</div> </div>
{/each} {/each}
@ -253,10 +253,15 @@
<style lang="scss"> <style lang="scss">
.key-box { .key-box {
background-color: var(--divider-color); background-color: var(--button-bg-color);
color: var(--caption-color); color: var(--caption-color);
min-width: 1.5rem; min-width: 1.5rem;
padding: 0.25rem; padding: 0 0.5rem;
border: 1px solid var(--button-border-color);
border-radius: 0.25rem;
}
.key-box + .key-box {
margin-left: 0.5rem;
} }
.item-box { .item-box {
display: inline-block; display: inline-block;

View File

@ -15,21 +15,26 @@
--> -->
<script lang="ts"> <script lang="ts">
// import type { IntlString } from '@anticrm/platform' // import type { IntlString } from '@anticrm/platform'
import { Label } from '@anticrm/ui' import { Label, Button } from '@anticrm/ui'
import type { ButtonKind, ButtonSize } from '@anticrm/ui'
import { getBooleanLabel } from '../utils' import { getBooleanLabel } from '../utils'
export let value: any export let value: any
export let maxWidth: string | undefined = undefined export let maxWidth: string | undefined = undefined
export let withoutUndefined: boolean = false export let withoutUndefined: boolean = false
export let onChange: (value: any) => void export let onChange: (value: any) => void
export let kind: ButtonKind = 'no-border'
export let size: ButtonSize = 'small'
export let justify: 'left' | 'center' = 'center'
export let width: string | undefined = 'fit-content'
</script> </script>
<div <Button
class="flex-row-center yesno-container" {kind}
class:yes={value === true} {size}
class:no={value === false} {justify}
class:unknown={value === undefined} {width}
style={maxWidth ? `max-width: ${maxWidth};` : ''}
on:click={() => { on:click={() => {
if (value === true) value = false if (value === true) value = false
else if (value === false && !withoutUndefined) value = undefined else if (value === false && !withoutUndefined) value = undefined
@ -37,12 +42,23 @@
onChange(value) onChange(value)
}} }}
> >
<svelte:fragment slot="content">
<div
class="flex-row-center yesno-container"
class:yes={value === true}
class:no={value === false}
class:unknown={value === undefined}
style={maxWidth ? `max-width: ${maxWidth};` : ''}
>
<svg class="yesno-svg" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16"> <svg class="yesno-svg" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16">
<circle class:yes={value === true} class:no={value === false} cx="8" cy="8" r="6" /> <circle class:yes={value === true} class:no={value === false} cx="8" cy="8" r="6" />
{#if value === true} {#if value === true}
<polygon fill="#fff" points="7.4,10.9 4.9,8.4 5.7,7.6 7.3,9.1 10.2,5.6 11.1,6.4 " /> <polygon fill="#fff" points="7.4,10.9 4.9,8.4 5.7,7.6 7.3,9.1 10.2,5.6 11.1,6.4 " />
{:else if value === false} {:else if value === false}
<polygon fill="#fff" points="10.7,6 10,5.3 8,7.3 6,5.3 5.3,6 7.3,8 5.3,10 6,10.7 8,8.7 10,10.7 10.7,10 8.7,8 " /> <polygon
fill="#fff"
points="10.7,6 10,5.3 8,7.3 6,5.3 5.3,6 7.3,8 5.3,10 6,10.7 8,8.7 10,10.7 10.7,10 8.7,8 "
/>
{:else} {:else}
<path <path
fill="#fff" fill="#fff"
@ -51,7 +67,9 @@
{/if} {/if}
</svg> </svg>
<span><Label label={getBooleanLabel(value)} /></span> <span><Label label={getBooleanLabel(value)} /></span>
</div> </div>
</svelte:fragment>
</Button>
<style lang="scss"> <style lang="scss">
.yesno-container { .yesno-container {