Updated Enums in Settings, EditEnum (#4423)

Signed-off-by: Alexander Platov <sas_lord@mail.ru>
This commit is contained in:
Alexander Platov 2024-01-24 10:39:49 +03:00 committed by GitHub
parent b87bcf2ebe
commit 1024faa46e
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
30 changed files with 809 additions and 259 deletions

View File

@ -22,6 +22,7 @@
"NoMatchesInThis": "No matches in this {space}",
"NoMatchesFound": "No matches found",
"NotInThis": "Not in this {space}",
"Match": "Match",
"Add": "Add",
"Edit": "Edit",
"DocumentPreview": "Preview",

View File

@ -22,6 +22,7 @@
"NoMatchesInThis": "В этом {space} совпадения не обнаружены",
"NoMatchesFound": "Не найдено соответствий",
"NotInThis": "Не в этом {space}",
"Match": "Совпадение",
"Add": "Добавить",
"Edit": "Редактировать",
"DocumentPreview": "Предпросмотр",

View File

@ -62,6 +62,7 @@ export default plugin(presentationId, {
NoMatchesInThis: '' as IntlString,
NoMatchesFound: '' as IntlString,
NotInThis: '' as IntlString,
Match: '' as IntlString,
Add: '' as IntlString,
Edit: '' as IntlString,
DocumentPreview: '' as IntlString,

View File

@ -968,6 +968,9 @@ a.no-line {
.content-color { color: var(--theme-content-color); }
.caption-color { color: var(--theme-caption-color); }
.secondary-textColor { color: var(--global-secondary-TextColor) !important; }
.tertiary-textColor { color: var(--global-tertiary-TextColor) !important; }
.content-primary-color { color: var(--primary-button-color); }
.red-color { color: var(--highlight-red); }
.error-color { color: var(--theme-error-color); }

View File

@ -48,6 +48,7 @@
--global-accent-TextColor: #4D7FF5;
--global-focus-BorderColor: #2A59D6;
--global-popover-ShadowColor: #0E131E59;
--global-modal-ShadowColor: #0E131E73;
/** Buttons **/
--button-subtle-LabelColor: #fff;
@ -102,6 +103,7 @@
--global-accent-TextColor: #3566E2;
--global-focus-BorderColor: #204DC8;
--global-popover-ShadowColor: #0E131E1F;
--global-modal-ShadowColor: #0E131E14;
/** Buttons **/
--button-subtle-LabelColor: #000;

View File

@ -20,6 +20,7 @@
--spacing-0_5: 0.25rem;
--spacing-0_75: 0.375rem;
--spacing-1: 0.5rem;
--spacing-1_25: 0.625rem;
--spacing-1_5: 0.75rem;
--spacing-1_75: 0.875rem;
--spacing-2: 1rem;
@ -63,4 +64,8 @@
--global-popover-ShadowSpread: 0;
--global-popover-ShadowX: 0;
--global-popover-ShadowY: 0.5rem;
--global-modal-ShadowBlur: 1.5rem;
--global-modal-ShadowSpread: 0.25rem;
--global-modal-ShadowX: 0;
--global-modal-ShadowY: 1.5rem;
}

View File

@ -173,7 +173,6 @@
border-top: 1px solid transparent;
.hulyModal-content {
padding: var(--spacing-2) var(--spacing-1_5);
height: 100%;
&__titleGroup {
@ -222,14 +221,35 @@
flex-direction: row-reverse;
flex-shrink: 0;
gap: var(--spacing-1);
padding: var(--spacing-2) var(--spacing-2_5);
border-top: 1px solid var(--theme-divider-color); // var(--global-surface-01-BorderColor);
}
&.type-aside .hulyHeader-container {
border-radius: 0 var(--small-focus-BorderRadius) 0 0;
&.type-aside {
.hulyHeader-container {
border-radius: 0 var(--small-focus-BorderRadius) 0 0;
.hulyHeader-titleGroup {
.hulyHeader-buttonsGroup {
gap: var(--spacing-0_5);
}
}
.hulyModal-footer {
padding: var(--spacing-2) var(--spacing-2_5);
}
}
&.type-popup {
min-width: 45rem;
background-color: var(--theme-popup-color); // var(--global-surface-02-BackgroundColor);
border: 1px solid var(--theme-popup-divider); // var(--global-surface-02-BorderColor);
border-radius: var(--large-BorderRadius);
box-shadow: var(--global-modal-ShadowX) var(--global-modal-ShadowY) var(--global-modal-ShadowBlur) var(--global-modal-ShadowSpread) var(--global-popover-ShadowColor);
.hulyModal-footer {
padding: var(--spacing-1_5);
}
}
&.type-aside,
&.type-popup {
.hulyHeader-container .hulyHeader-titleGroup {
text-transform: uppercase;
font-family: var(--font-family);
font-weight: 500;
@ -238,9 +258,6 @@
line-height: 1rem;
color: var(--global-secondary-TextColor);
}
.hulyHeader-buttonsGroup {
gap: var(--spacing-0_5);
}
}
textarea {
font-weight: 400 !important;
@ -255,6 +272,10 @@
color: var(--global-tertiary-TextColor);
background-color: var(--global-ui-BackgroundColor);
border-radius: var(--extra-small-BorderRadius);
&.error {
color: var(--button-negative-loading-LabelColor);
}
}
.hulyHotKey-item {

View File

@ -64,8 +64,12 @@
&:not(.small) { font-size: .875rem; }
&.small { font-size: .75rem; }
&:not(.dark) { color: var(--global-on-accent-TextColor); }
&.dark { color: var(--global-secondary-TextColor); }
&:not(.dark) {
color: var(--global-primary-TextColor); // var(--global-on-accent-TextColor);
}
&.dark {
color: var(--theme-dark-color); // var(--global-secondary-TextColor);
}
}
&:hover {

View File

@ -41,6 +41,13 @@
width: var(--global-min-Size);
height: var(--global-min-Size);
}
.buttons-group {
display: flex;
align-items: center;
gap: var(--spacing-1);
min-width: 0;
min-height: 0;
}
}
.hulyTableAttr-content {
display: flex;
@ -108,6 +115,10 @@
border-radius: var(--extra-small-BorderRadius);
border: none;
outline: none;
&.drag {
cursor: grab !important;
}
}
&-icon {
width: var(--global-min-Size);
@ -173,10 +184,11 @@
color: var(--global-primary-LinkColor);
}
}
&.options .hulyTableAttr-content__row,
&.class .hulyTableAttr-content__row,
&.task .hulyTableAttr-content__row {
&.hovered,
&:hover {
&:not(.disableMouseOver):hover {
background-color: var(--theme-table-header-color); // var(--global-surface-03-hover-BackgroundColor);
}
&.selected {
@ -195,6 +207,7 @@
}
}
}
&.options,
&.class {
padding: var(--spacing-1);
@ -203,12 +216,31 @@
padding: var(--spacing-1) var(--spacing-2) var(--spacing-1) var(--spacing-1);
&.hovered .hulyTableAttr-content__row-arrow,
&:hover .hulyTableAttr-content__row-arrow,
&:not(.disableMouseOver):hover .hulyTableAttr-content__row-arrow,
&.selected .hulyTableAttr-content__row-arrow {
display: block;
}
}
}
&.options .hulyTableAttr-content__row {
padding: var(--spacing-1);
min-height: var(--global-large-Size);
&:not(.hovered) button.type-button-icon {
display: none;
}
&.disableMouseOver,
&-dragMenu {
cursor: default;
}
label.editbox-wrapper {
padding: 0 !important;
height: var(--global-extra-small-Size) !important;
}
&:hover button.type-button-icon {
display: inline-flex;
}
}
&.task {
.hulyTableAttr-content__row {
gap: var(--spacing-1);

View File

@ -14,7 +14,8 @@
-->
<script lang="ts">
import type { Asset, IntlString } from '@hcengineering/platform'
import { AnySvelteComponent } from '../types'
import { AnySvelteComponent, LabelAndProps } from '../types'
import { tooltip as tp } from '../tooltips'
import { ComponentType } from 'svelte'
import Spinner from './Spinner.svelte'
import Icon from './Icon.svelte'
@ -33,6 +34,7 @@
export let hasMenu: boolean = false
export let type: 'type-button' | 'type-button-icon'
export let inheritColor: boolean = false
export let tooltip: LabelAndProps | undefined = undefined
export let element: HTMLButtonElement | undefined = undefined
</script>
@ -44,6 +46,7 @@
class:inheritColor
class:menu={hasMenu}
disabled={loading || disabled}
use:tp={tooltip}
on:click
>
{#if loading}
@ -299,6 +302,9 @@
}
}
}
& > * {
pointer-events: none;
}
}
@keyframes rotate {

View File

@ -14,7 +14,7 @@
-->
<script lang="ts">
import type { Asset } from '@hcengineering/platform'
import { AnySvelteComponent } from '../types'
import { AnySvelteComponent, LabelAndProps } from '../types'
import { ComponentType } from 'svelte'
import ButtonBase from './ButtonBase.svelte'
@ -24,8 +24,10 @@
export let iconProps: any | undefined = undefined
export let disabled: boolean = false
export let pressed: boolean = false
export let hasMenu: boolean = false
export let loading: boolean = false
export let inheritColor: boolean = false
export let tooltip: LabelAndProps | undefined = undefined
</script>
<ButtonBase
@ -38,5 +40,7 @@
{loading}
{inheritColor}
{pressed}
{hasMenu}
{tooltip}
on:click
/>

View File

@ -16,7 +16,7 @@
import { createEventDispatcher } from 'svelte'
import { IconMaximize, IconMinimize, IconClose, ButtonIcon } from '..'
export let type: 'type-aside' | 'type-component' = 'type-component'
export let type: 'type-aside' | 'type-popup' | 'type-component' = 'type-component'
export let minimize: boolean = false
const dispatch = createEventDispatcher()
@ -41,8 +41,8 @@
<slot name="actions" />
</div>
{/if}
{#if type === 'type-aside'}
<div class="hulyHeader-divider" />
{#if type !== 'type-component'}
{#if type !== 'type-popup'}<div class="hulyHeader-divider" />{/if}
<div class="hulyHotKey-item">Esc</div>
<ButtonIcon icon={IconClose} kind={'tertiary'} size={'small'} on:click={() => dispatch('close')} />
{/if}

View File

@ -20,11 +20,9 @@
import ButtonIcon from './ButtonIcon.svelte'
import ButtonBase from './ButtonBase.svelte'
import Scroller from './Scroller.svelte'
import IconDelete from './icons/Delete.svelte'
import IconCopy from './icons/Copy.svelte'
import ui from '..'
export let type: 'type-aside' | 'type-component'
export let type: 'type-aside' | 'type-popup' | 'type-component'
export let label: IntlString
export let labelProps: any | undefined = undefined
export let okAction: () => Promise<void> | void
@ -49,26 +47,42 @@
<Header {type} on:close={close}>
<Label {label} params={labelProps} />
<svelte:fragment slot="actions">
<ButtonIcon icon={IconDelete} size={'small'} kind={'tertiary'} />
<ButtonIcon icon={IconCopy} size={'small'} kind={'tertiary'} />
<slot name="actions" />
</svelte:fragment>
</Header>
<div class="hulyModal-content">
<Scroller>
<Scroller
padding={type === 'type-popup'
? 'var(--spacing-2) var(--spacing-3) var(--spacing-4)'
: type === 'type-aside'
? 'var(--spacing-2) var(--spacing-1_5)'
: 'var(--spacing-3)'}
bottomPadding={type === 'type-popup'
? undefined
: type === 'type-aside'
? 'var(--spacing-2)'
: 'var(--spacing-3)'}
>
<slot />
</Scroller>
</div>
{#if type === 'type-aside'}
{#if type !== 'type-component'}
<div class="hulyModal-footer">
<ButtonBase
type={'type-button'}
kind={'primary'}
size={'large'}
size={type === 'type-aside' ? 'large' : 'medium'}
label={okLabel}
on:click={okAction}
disabled={!canSave}
/>
<ButtonBase type={'type-button'} kind={'secondary'} size={'large'} label={ui.string.Cancel} on:click={onCancel} />
<ButtonBase
type={'type-button'}
kind={'secondary'}
size={type === 'type-aside' ? 'large' : 'medium'}
label={ui.string.Cancel}
on:click={onCancel}
/>
</div>
{/if}
</div>

View File

@ -4,7 +4,7 @@
// Licensed under the Eclipse Public License v2.0 (SPDX: EPL-2.0).
//
import { createEventDispatcher } from 'svelte'
import { createEventDispatcher, onMount } from 'svelte'
import { IntlString, translate } from '@hcengineering/platform'
import Label from './Label.svelte'
import { themeStore } from '..'
@ -17,6 +17,9 @@
export let error: boolean = false
export let password: boolean = false
export let limit: number = 0
export let element: HTMLInputElement | undefined = undefined
export let autoFocus: boolean = false
export let width: string = ''
const dispatch = createEventDispatcher()
@ -28,14 +31,23 @@
})
$: labeled = kind === 'default' && size === 'large'
$: placeholder = labeled ? ' ' : placeholderStr
onMount(() => {
if (autoFocus && element) {
autoFocus = false
element.focus()
}
})
</script>
<label class="editbox-wrapper {kind} {size}" class:error class:disabled>
{#if password}
<input
bind:this={element}
type="password"
class="font-regular-14"
class:labeled
style:width
bind:value
autocomplete="off"
{placeholder}
@ -44,6 +56,7 @@
{maxlength}
on:change
on:keyup
on:keydown
on:input
on:blur={() => {
dispatch('blur', value)
@ -51,9 +64,11 @@
/>
{:else}
<input
bind:this={element}
type="text"
class="font-regular-14"
class:labeled
style:width
bind:value
autocomplete="off"
{placeholder}
@ -62,6 +77,7 @@
{maxlength}
on:change
on:keyup
on:keydown
on:input
on:blur={() => {
dispatch('blur', value)

View File

@ -1,6 +1,6 @@
<script lang="ts">
export let size: 'small' | 'medium' | 'large'
const fill: string = 'currentColor'
export let fill: string = 'currentColor'
</script>
<svg class="svg-{size}" {fill} viewBox="0 0 32 32" xmlns="http://www.w3.org/2000/svg">

View File

@ -0,0 +1,35 @@
<!--
// Copyright © 2024 Hardcore Engineering Inc.
//
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License. You may
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//
// See the License for the specific language governing permissions and
// limitations under the License.
-->
<script lang="ts">
export let size: 'small' | 'medium' | 'large'
const fill: string = 'currentColor'
</script>
<svg class="svg-{size}" {fill} viewBox="0 0 32 32" xmlns="http://www.w3.org/2000/svg">
<path
d="M26 11.0005C26 10.4482 25.5523 10.0005 25 10.0005H20C19.4477 10.0005 19 10.4482 19 11.0005C19 11.5528 19.4477 12.0005 20 12.0005H25C25.5523 12.0005 26 11.5528 26 11.0005Z"
/>
<path
d="M26 16.0005C26 15.4482 25.5523 15.0005 25 15.0005H20C19.4477 15.0005 19 15.4482 19 16.0005C19 16.5528 19.4477 17.0005 20 17.0005H25C25.5523 17.0005 26 16.5528 26 16.0005Z"
/>
<path
d="M19 21.0005C19 20.4482 19.4477 20.0005 20 20.0005H25C25.5523 20.0005 26 20.4482 26 21.0005C26 21.5528 25.5523 22.0005 25 22.0005H20C19.4477 22.0005 19 21.5528 19 21.0005Z"
/>
<path
fill-rule="evenodd"
clip-rule="evenodd"
d="M6 4.00012C3.79086 4.00012 2 5.79098 2 8.00012V24.0001C2 26.2093 3.79086 28.0001 6 28.0001H26C28.2091 28.0001 30 26.2093 30 24.0001V8.00012C30 5.79098 28.2091 4.00012 26 4.00012H6ZM15 6.00012H6C4.89543 6.00012 4 6.89555 4 8.00012V24.0001C4 25.1047 4.89543 26.0001 6 26.0001H15V6.00012ZM17 6.00012V26.0001H26C27.1046 26.0001 28 25.1047 28 24.0001V8.00012C28 6.89555 27.1046 6.00012 26 6.00012H17Z"
/>
</svg>

View File

@ -131,6 +131,7 @@ export { default as ButtonIcon } from './components/ButtonIcon.svelte'
export { default as ButtonMenu } from './components/ButtonMenu.svelte'
export { default as ModernButton } from './components/ModernButton.svelte'
export { default as ModernEditbox } from './components/ModernEditbox.svelte'
export { default as ModernPopup } from './components/ModernPopup.svelte'
export { default as NavItem } from './components/NavItem.svelte'
export { default as NavGroup } from './components/NavGroup.svelte'
export { default as Modal } from './components/Modal.svelte'
@ -201,6 +202,7 @@ export { default as IconDescription } from './components/icons/Description.svelt
export { default as IconSettings } from './components/icons/Settings.svelte'
export { default as IconSend } from './components/icons/Send.svelte'
export { default as IconSquareExpand } from './components/icons/SquareExpand.svelte'
export { default as IconTableOfContents } from './components/icons/TableOfContents.svelte'
export { default as PanelInstance } from './components/PanelInstance.svelte'
export { default as Panel } from './components/Panel.svelte'

View File

@ -49,7 +49,16 @@
"CreatingAttribute": "Creating an attribute",
"EditAttribute": "Edit attribute",
"CreateEnum": "Create enum",
"EditEnum": "Edit enum",
"Enums": "Enums",
"EnumsSettingHint": "A set or category of things having some property or attribute in common from others by kind, type, or quality.",
"EnumTitle": "Enum title",
"EnumsCount": "{count, plural, =1 {# option} other {# options}}",
"ProjectTypesCount": "{count, plural, =0 {No project types} =1 {# project type} other {# project types}}",
"Options": "Options",
"EnterOptionTitle": "Enter option title",
"NewEnumDialogClose": "Do you want to close this dialog?",
"NewEnumDialogCloseNote": "All changes will be lost",
"NewValue": "New value",
"Leave": "Leave workspace",
"LeaveDescr": "Are you sure you want to leave the workspace? This action cannot be undone.",

View File

@ -49,7 +49,16 @@
"CreatingAttribute": "Создание атрибута",
"EditAttribute": "Редактирование атрибута",
"CreateEnum": "Создать справочник",
"EditEnum": "Редактировать справочник",
"Enums": "Справочники",
"EnumsSettingHint": "Набор или категория вещей, обладающих каким-либо свойством или атрибутом, отличающимся от других по виду, типу или качеству.",
"EnumTitle": "Заголовок справочника",
"EnumsCount": "{count, plural, one {# вариант} other {# вариантов}}",
"ProjectTypesCount": "{count, plural, =0 {Нет типов проектов} =1 {# тип проекта} few {# типа проектов} other {# типов проектов}}",
"Options": "Варианты",
"EnterOptionTitle": "Введите название варианта",
"NewEnumDialogClose": "Вы действительно хотите закрыть окно?",
"NewEnumDialogCloseNote": "Все внесенные изменения будут потеряны",
"NewValue": "Новое значение",
"Leave": "Покинуть рабочее пространство",
"LeaveDescr": "Вы действительно хотите покинуть рабочее пространство? Отменить это действие невозможно",

View File

@ -26,7 +26,17 @@
} from '@hcengineering/core'
import { getEmbeddedLabel } from '@hcengineering/platform'
import presentation, { getClient } from '@hcengineering/presentation'
import { AnyComponent, Component, DropdownLabelsIntl, ModernEditbox, Label, Modal } from '@hcengineering/ui'
import {
AnyComponent,
Component,
DropdownLabelsIntl,
ModernEditbox,
Label,
Modal,
ButtonIcon,
IconDelete,
IconCopy
} from '@hcengineering/ui'
import { DropdownIntlItem } from '@hcengineering/ui/src/types'
import setting from '../plugin'
import view from '@hcengineering/view'
@ -108,6 +118,10 @@
clearSettingsStore()
}}
>
<svelte:fragment slot="actions">
<ButtonIcon icon={IconDelete} size={'small'} kind={'tertiary'} />
<ButtonIcon icon={IconCopy} size={'small'} kind={'tertiary'} />
</svelte:fragment>
<div class="hulyModal-content__titleGroup">
<div class="hulyChip-item font-medium-12">
<Label label={setting.string.Custom} />

View File

@ -25,7 +25,10 @@
ModernEditbox,
Label,
themeStore,
Modal
Modal,
ButtonIcon,
IconDelete,
IconCopy
} from '@hcengineering/ui'
import view from '@hcengineering/view-resources/src/plugin'
import { clearSettingsStore } from '../store'
@ -111,6 +114,10 @@
clearSettingsStore()
}}
>
<svelte:fragment slot="actions">
<ButtonIcon icon={IconDelete} size={'small'} kind={'tertiary'} />
<ButtonIcon icon={IconCopy} size={'small'} kind={'tertiary'} />
</svelte:fragment>
<div class="hulyModal-content__titleGroup">
{#if attribute.isCustom}
<div class="hulyChip-item font-medium-12">

View File

@ -14,22 +14,37 @@
-->
<script lang="ts">
import core, { Enum } from '@hcengineering/core'
import presentation, { Card, getClient, MessageBox } from '@hcengineering/presentation'
import { ActionIcon, EditBox, IconAdd, IconAttachment, IconDelete, ListView, showPopup } from '@hcengineering/ui'
import view from '@hcengineering/view-resources/src/plugin'
import presentation, { getClient, MessageBox } from '@hcengineering/presentation'
import {
IconAdd,
IconAttachment,
IconDelete,
showPopup,
Modal,
ModernEditbox,
Label,
ButtonIcon,
IconMoreV,
IconMoreV2,
eventToHTMLElement,
ModernPopup
} from '@hcengineering/ui'
import type { DropdownIntlItem } from '@hcengineering/ui'
import { createEventDispatcher } from 'svelte'
import setting from '../plugin'
import EnumValuesList from './EnumValuesList.svelte'
import Copy from './icons/Copy.svelte'
import IconBulletList from './icons/BulletList.svelte'
import Report from './icons/Report.svelte'
import { IntlString } from '@hcengineering/platform'
export let value: Enum | undefined
export let name: string = value?.name ?? ''
export let values: string[] = value?.enumValues ?? []
export let title: IntlString = setting.string.EditEnum
const client = getClient()
const dispatch = createEventDispatcher()
let list: ListView
async function save (): Promise<void> {
if (value === undefined) {
await client.createDoc(core.class.Enum, core.space.Model, {
@ -47,18 +62,29 @@
function add () {
newValue = newValue.trim()
if (newValue.length === 0) return
if (matched) return
if (values.includes(newValue)) return
values.push(newValue)
values = values
newValue = ''
}
function remove (value: string) {
values = values.filter((p) => p !== value)
}
const handleKeydown = (evt: KeyboardEvent) => {
if (evt.key === 'Enter') {
add()
}
if (evt.key === 'Escape') {
newItem = false
newValue = ''
evt.stopPropagation()
}
}
let newValue = ''
let newItem: boolean = false
let opened: boolean = false
let inputFile: HTMLInputElement
function processText (text: string): void {
@ -89,6 +115,7 @@
}
function fileDrop (e: DragEvent) {
dragover = false
const list = e.dataTransfer?.files
if (list === undefined || list.length === 0) return
for (let index = 0; index < list.length; index++) {
@ -114,40 +141,78 @@
const text = await navigator.clipboard.readText()
processText(text)
}
let dragover = false
const selection: number = 0
function onKeydown (key: KeyboardEvent): void {
if (key.code === 'ArrowUp') {
key.stopPropagation()
key.preventDefault()
list.select(selection - 1)
// $: filtered = newValue.length > 0 ? values.filter((it) => it.includes(newValue)) : values
$: matched = values.includes(newValue.trim())
// function onDelete () {
// showPopup(
// MessageBox,
// {
// label: view.string.DeleteObject,
// message: view.string.DeleteObjectConfirm,
// params: { count: filtered.length }
// },
// 'top',
// (result?: boolean) => {
// if (result === true) {
// values = values.filter((it) => !filtered.includes(it))
// newValue = ''
// }
// }
// )
// }
const items: (DropdownIntlItem & { action: () => void })[] = [
{
id: 'import',
icon: IconAttachment,
label: setting.string.ImportEnum,
action: () => {
inputFile.click()
}
},
{
id: 'paste',
icon: Report,
label: setting.string.ImportEnumCopy,
action: () => {
handleClipboard()
}
}
if (key.code === 'ArrowDown') {
key.stopPropagation()
key.preventDefault()
list.select(selection + 1)
]
const openPopup = (ev: MouseEvent): void => {
if (!opened) {
opened = true
showPopup(ModernPopup, { items }, eventToHTMLElement(ev), (result) => {
if (result) items.find((it) => it.id === result)?.action()
opened = false
})
}
}
$: filtered = newValue.length > 0 ? values.filter((it) => it.includes(newValue)) : values
async function showConfirmationDialog (): Promise<void> {
const isEnumEmpty = values.length === 0
function onDelete () {
showPopup(
MessageBox,
{
label: view.string.DeleteObject,
message: view.string.DeleteObjectConfirm,
params: { count: filtered.length }
},
'top',
(result?: boolean) => {
if (result === true) {
values = values.filter((it) => !filtered.includes(it))
newValue = ''
if (isEnumEmpty) {
dispatch('close')
} else {
showPopup(
MessageBox,
{
label: setting.string.NewEnumDialogClose,
message: setting.string.NewEnumDialogCloseNote
},
'top',
(result?: boolean) => {
if (result === true) dispatch('close')
}
}
)
)
}
}
</script>
@ -163,57 +228,22 @@
on:change={fileSelected}
/>
<Card
label={core.string.Enum}
<Modal
label={title}
type={'type-popup'}
okLabel={presentation.string.Save}
okAction={save}
canSave={name.trim().length > 0 && values.length > 0}
on:close={() => {
dispatch('close')
onCancel={() => {
showConfirmationDialog()
}}
on:changeContent
>
<!-- svelte-ignore a11y-no-static-element-interactions -->
<div class="flex-col" on:keydown={onKeydown}>
<EditBox bind:value={name} placeholder={core.string.Name} kind={'large-style'} fullSize />
<div class="flex-between my-4">
<EditBox placeholder={presentation.string.Search} kind={'large-style'} bind:value={newValue} fullSize />
<div class="flex-row-center flex-no-shrink gap-2 ml-4">
<ActionIcon icon={IconAdd} label={presentation.string.Add} action={add} size={'small'} />
<ActionIcon
icon={Copy}
label={setting.string.ImportEnumCopy}
action={() => {
handleClipboard()
}}
size={'small'}
/>
<ActionIcon
icon={IconDelete}
label={setting.string.Delete}
action={() => {
onDelete()
}}
size={'small'}
/>
</div>
</div>
<div class="scroll" style:margin={'0 -.5rem'}>
<div class="box flex max-h-125">
<EnumValuesList
bind:values
bind:filtered
on:remove={(e) => {
remove(e.detail)
}}
/>
</div>
</div>
</div>
<svelte:fragment slot="footer">
<div class="flex-col">
<ModernEditbox bind:value={name} label={setting.string.EnumTitle} kind={'ghost'} size={'large'} width={'100%'} />
<!-- svelte-ignore a11y-no-static-element-interactions -->
<div
class="resume flex-center"
class:solid={dragover}
class="hulyTableAttr-container mt-6"
class:dragDropZone={dragover}
on:dragover|preventDefault={() => {
dragover = true
}}
@ -222,14 +252,81 @@
}}
on:drop|preventDefault|stopPropagation={fileDrop}
>
<ActionIcon
icon={IconAttachment}
label={setting.string.ImportEnum}
action={() => {
inputFile.click()
}}
size={'small'}
/>
<div class="hulyTableAttr-header font-medium-12">
<IconBulletList size={'small'} />
<span><Label label={setting.string.Options} /></span>
<div class="buttons-group tertiary-textColor">
<ButtonIcon
kind={'tertiary'}
icon={IconMoreV}
size={'small'}
pressed={opened}
inheritColor
hasMenu
on:click={(ev) => {
openPopup(ev)
}}
/>
<ButtonIcon
kind={'primary'}
icon={IconAdd}
size={'small'}
tooltip={{ label: setting.string.Add }}
on:click={() => {
newItem = true
}}
/>
</div>
</div>
{#if values.length > 0 || newItem}
<div class="hulyTableAttr-content options">
<EnumValuesList
bind:values
disableMouseOver={newItem}
on:remove={(e) => {
remove(e.detail)
}}
/>
{#if newItem}
<div class="hulyTableAttr-content__row hovered">
<div class="hulyTableAttr-content__row-dragMenu">
<IconMoreV2 size={'small'} />
</div>
<div class="hulyTableAttr-content__row-label font-regular-14 accent grow">
<ModernEditbox
kind={'ghost'}
size={'small'}
label={setting.string.EnterOptionTitle}
on:keydown={handleKeydown}
bind:value={newValue}
width={'100%'}
autoFocus
/>
</div>
{#if matched}
<div class="hulyChip-item error font-medium-12">
<Label label={presentation.string.Match} />
</div>
{/if}
<ButtonIcon
kind={'tertiary'}
icon={IconDelete}
size={'small'}
on:click={() => {
newValue = ''
newItem = false
}}
/>
</div>
{/if}
</div>
{/if}
</div>
</svelte:fragment>
</Card>
</div>
</Modal>
<style lang="scss">
.dragDropZone {
border: 2px dashed var(--theme-popup-hover);
}
</style>

View File

@ -15,10 +15,8 @@
<script lang="ts">
import { createEventDispatcher } from 'svelte'
import core, { Enum } from '@hcengineering/core'
import { createQuery, getClient } from '@hcengineering/presentation'
import { createQuery } from '@hcengineering/presentation'
import {
CircleButton,
EditBox,
eventToHTMLElement,
IconAdd,
IconMoreH,
@ -29,7 +27,10 @@
defineSeparators,
settingsSeparators,
Separator,
Scroller
Scroller,
ButtonIcon,
IconTableOfContents,
ModernButton
} from '@hcengineering/ui'
import { ContextMenu } from '@hcengineering/view-resources'
import setting from '../plugin'
@ -44,7 +45,6 @@
let enums: Enum[] = []
let selected: Enum | undefined
let hovered: number | null = null
const client = getClient()
query.query(core.class.Enum, {}, (res) => {
enums = res
@ -54,32 +54,38 @@
})
function create () {
showPopup(setting.component.EditEnum, {}, 'top')
showPopup(setting.component.EditEnum, { title: setting.string.CreateEnum }, 'top')
}
async function update (value: Enum): Promise<void> {
await client.update(value, {
name: value.name
})
}
defineSeparators('workspaceSettings', settingsSeparators)
</script>
<div class="hulyComponent">
<Header minimize={!visibleNav} on:resize={(event) => dispatch('change', event.detail)}>
<Breadcrumb icon={setting.icon.Enums} label={setting.string.Enums} size={'large'} isCurrent />
<svelte:fragment slot="actions">
<ModernButton
kind={'primary'}
icon={IconAdd}
label={setting.string.CreateEnum}
size={'small'}
on:click={create}
/>
</svelte:fragment>
</Header>
<div class="hulyComponent-content__container columns">
<div class="hulyComponent-content__column">
<div class="flex-between trans-title m-3">
<Label label={setting.string.Enums} />
<CircleButton icon={IconAdd} size="medium" on:click={create} />
<div class="hulyComponent-content__navHeader">
<div class="hulyComponent-content__navHeader-menu">
<ButtonIcon kind={'tertiary'} icon={IconTableOfContents} size={'small'} inheritColor />
</div>
<div class="hulyComponent-content__navHeader-hint paragraph-regular-14">
<Label label={setting.string.EnumsSettingHint} />
</div>
</div>
<div class="overflow-y-auto">
<Scroller>
{#each enums as value, i}
<!-- svelte-ignore a11y-click-events-have-key-events -->
<!-- svelte-ignore a11y-no-static-element-interactions -->
<div
<button
class="enum__list-item"
class:hovered={hovered === i}
class:selected={selected === value}
@ -87,21 +93,27 @@
selected = value
}}
>
<EditBox bind:value={value.name} on:change={() => update(value)} />
<div
class="hover-trans"
on:click|stopPropagation={(ev) => {
<div class="flex-col">
<span class="font-regular-14 overflow-label">{value.name}</span>
<span class="font-regular-12 secondary-textColor overflow-label">
<Label label={setting.string.EnumsCount} params={{ count: value.enumValues.length }} />
</span>
</div>
<ButtonIcon
kind={'tertiary'}
icon={IconMoreH}
size={'small'}
pressed={hovered === i}
on:click={(ev) => {
hovered = i
showPopup(ContextMenu, { object: value }, eventToHTMLElement(ev), () => {
hovered = null
})
}}
>
<IconMoreH size={'medium'} />
</div>
</div>
/>
</button>
{/each}
</div>
</Scroller>
</div>
<Separator name={'workspaceSettings'} index={0} color={'var(--theme-divider-color)'} />
<div class="hulyComponent-content__column content">
@ -121,21 +133,27 @@
display: flex;
justify-content: space-between;
align-items: center;
min-height: 2.5rem;
margin: 0 0.75rem;
padding: 0 1.25rem;
border: 1px solid transparent;
border-radius: 12px;
cursor: pointer;
margin: 0 var(--spacing-1_5);
padding: var(--spacing-1) var(--spacing-1_25);
text-align: left;
border: none;
border-radius: var(--small-BorderRadius);
outline: none;
& :global(button.type-button-icon) {
visibility: hidden;
}
&.hovered,
&:hover {
background-color: var(--theme-button-hovered);
& :global(button.type-button-icon) {
visibility: visible;
}
}
&.selected {
background-color: var(--theme-button-default);
border-color: var(--theme-button-border);
cursor: auto;
cursor: default;
}
}
</style>

View File

@ -15,25 +15,43 @@
<script lang="ts">
import { Enum } from '@hcengineering/core'
import presentation, { getClient, MessageBox } from '@hcengineering/presentation'
import { ActionIcon, EditBox, IconAdd, IconAttachment, IconDelete, showPopup } from '@hcengineering/ui'
import view from '@hcengineering/view-resources/src/plugin'
import {
ModernEditbox,
IconAdd,
IconAttachment,
IconDelete,
showPopup,
ModernButton,
Label,
ButtonIcon,
IconMoreV,
IconMoreV2,
ModernPopup,
eventToHTMLElement
} from '@hcengineering/ui'
import type { DropdownIntlItem } from '@hcengineering/ui'
import setting from '../plugin'
import EnumValuesList from './EnumValuesList.svelte'
import Copy from './icons/Copy.svelte'
import IconCrossedArrows from './icons/CrossedArrows.svelte'
import IconBulletList from './icons/BulletList.svelte'
import Report from './icons/Report.svelte'
export let value: Enum
const client = getClient()
let newValue = ''
let newValue: string = ''
let newItem: boolean = false
let opened: boolean = false
async function add () {
if (newValue.trim().length === 0) return
if (value.enumValues.includes(newValue.trim())) return
if (matched) return
await client.update(value, {
$push: { enumValues: newValue }
})
newValue = ''
newItem = false
}
async function remove (target: string) {
@ -45,8 +63,13 @@
if (evt.key === 'Enter') {
add()
}
if (evt.key === 'Escape') {
newItem = false
newValue = ''
}
}
$: filtered = newValue.length > 0 ? value.enumValues.filter((it) => it.includes(newValue)) : value.enumValues
// $: filtered = newValue.length > 0 ? value.enumValues.filter((it) => it.includes(newValue)) : []
$: matched = value.enumValues.includes(newValue.trim())
async function handleClipboard (): Promise<void> {
const text = await navigator.clipboard.readText()
@ -81,31 +104,84 @@
}
inputFile.value = ''
}
function onDelete () {
showPopup(
MessageBox,
{
label: view.string.DeleteObject,
message: view.string.DeleteObjectConfirm,
params: { count: filtered.length }
},
undefined,
(result?: boolean) => {
if (result === true) {
client.update(value, {
$pull: { enumValues: { $in: filtered } }
})
newValue = ''
}
}
)
// function onDelete () {
// showPopup(
// MessageBox,
// {
// label: view.string.DeleteObject,
// message: view.string.DeleteObjectConfirm,
// params: { count: filtered.length }
// },
// undefined,
// (result?: boolean) => {
// if (result === true) {
// client.update(value, {
// $pull: { enumValues: { $in: filtered } }
// })
// newValue = ''
// }
// }
// )
// }
async function update (value: Enum): Promise<void> {
await client.update(value, {
name: value.name
})
}
async function onDrop () {
await client.update(value, { enumValues: value.enumValues })
}
const items: (DropdownIntlItem & { action: () => void })[] = [
{
id: 'import',
icon: IconAttachment,
label: setting.string.ImportEnum,
action: () => {
inputFile.click()
}
},
{
id: 'paste',
icon: Report,
label: setting.string.ImportEnumCopy,
action: () => {
handleClipboard()
}
}
]
const openPopup = (ev: MouseEvent): void => {
if (!opened) {
opened = true
showPopup(ModernPopup, { items }, eventToHTMLElement(ev), (result) => {
if (result) items.find((it) => it.id === result)?.action()
opened = false
})
}
}
</script>
<div class="flex-between flex-gap-2">
<ModernEditbox
bind:value={value.name}
label={setting.string.EnumTitle}
kind={'ghost'}
size={'large'}
on:change={() => update(value)}
/>
<ModernButton
icon={IconCrossedArrows}
label={setting.string.ProjectTypesCount}
labelParams={{ count: 0 }}
disabled
kind={'tertiary'}
size={'medium'}
hasMenu
/>
</div>
<input
bind:this={inputFile}
multiple
@ -115,56 +191,74 @@
style="display: none"
on:change={fileSelected}
/>
<div class="flex-grow">
<div class="flex-between mb-4">
<EditBox
placeholder={presentation.string.Search}
on:keydown={handleKeydown}
kind="large-style"
bind:value={newValue}
/>
<div class="flex-row-center gap-2">
<ActionIcon
<div class="hulyTableAttr-container mt-6">
<div class="hulyTableAttr-header font-medium-12">
<IconBulletList size={'small'} />
<span><Label label={setting.string.Options} /></span>
<div class="buttons-group tertiary-textColor">
<ButtonIcon
kind={'tertiary'}
icon={IconMoreV}
size={'small'}
pressed={opened}
inheritColor
hasMenu
on:click={(ev) => {
openPopup(ev)
}}
/>
<ButtonIcon
kind={'primary'}
icon={IconAdd}
label={setting.string.Add}
action={add}
size={'small'}
disabled={value.enumValues.includes(newValue.trim())}
/>
<ActionIcon
icon={IconAttachment}
label={setting.string.ImportEnum}
action={() => {
inputFile.click()
tooltip={{ label: setting.string.Add }}
on:click={() => {
newItem = true
}}
size={'small'}
/>
<ActionIcon
icon={Copy}
label={setting.string.ImportEnumCopy}
action={() => {
handleClipboard()
}}
size={'small'}
/>
<ActionIcon
icon={IconDelete}
label={setting.string.Delete}
action={() => {
onDelete()
}}
size={'small'}
/>
</div>
</div>
<div class="scroll">
<div class="box">
{#if value.enumValues.length > 0 || newItem}
<div class="hulyTableAttr-content options">
<EnumValuesList
bind:values={value.enumValues}
bind:filtered
disableMouseOver={newItem}
on:remove={(e) => remove(e.detail)}
on:drop={onDrop}
/>
{#if newItem}
<div class="hulyTableAttr-content__row hovered">
<div class="hulyTableAttr-content__row-dragMenu">
<IconMoreV2 size={'small'} />
</div>
<div class="hulyTableAttr-content__row-label font-regular-14 accent grow">
<ModernEditbox
kind={'ghost'}
size={'small'}
label={setting.string.EnterOptionTitle}
on:keydown={handleKeydown}
bind:value={newValue}
width={'100%'}
autoFocus
/>
</div>
{#if matched}
<div class="hulyChip-item error font-medium-12">
<Label label={presentation.string.Match} />
</div>
{/if}
<ButtonIcon
kind={'tertiary'}
icon={IconDelete}
size={'small'}
on:click={() => {
newValue = ''
newItem = false
}}
/>
</div>
{/if}
</div>
</div>
{/if}
</div>

View File

@ -13,20 +13,29 @@
// limitations under the License.
-->
<script lang="ts">
import presentation from '@hcengineering/presentation'
import { ActionIcon, IconCircles, IconDelete, Label, Scroller } from '@hcengineering/ui'
import {
ModernPopup,
IconDelete,
ButtonIcon,
IconMoreV,
IconMoreV2,
showPopup,
eventToHTMLElement
} from '@hcengineering/ui'
import type { DropdownIntlItem } from '@hcengineering/ui'
import { createEventDispatcher } from 'svelte'
import setting from '../plugin'
export let values: string[]
export let filtered: string[]
export let disableMouseOver: boolean = false
let selected: string | undefined
let opened: number | undefined = undefined
const elements: HTMLElement[] = []
function dragswap (ev: MouseEvent, item: string): boolean {
const s = filtered.findIndex((p) => p === selected)
const i = filtered.findIndex((p) => p === item)
const s = values.findIndex((p) => p === selected)
const i = values.findIndex((p) => p === item)
if (i < s) {
return ev.offsetY < elements[i].offsetHeight / 2
} else if (i > s) {
@ -52,45 +61,79 @@
async function onDrop () {
dispatch('drop')
}
const items: (DropdownIntlItem & { action: () => void })[] = [
{
id: 'delete',
icon: IconDelete,
label: setting.string.Delete,
action: () => {
if (opened !== undefined) {
remove(values[opened])
opened = undefined
}
}
}
]
function openPopup (ev: MouseEvent, n: number) {
if (opened === undefined) {
opened = n
showPopup(ModernPopup, { items }, eventToHTMLElement(ev), (result) => {
if (result) {
switch (result) {
case 'delete':
remove(values[n])
break
}
}
opened = undefined
})
}
}
</script>
<Scroller>
{#each filtered as item, i}
<!-- svelte-ignore a11y-no-static-element-interactions -->
<div
class="flex-between flex-nowrap item step-tb25"
draggable={true}
bind:this={elements[i]}
on:dragover|preventDefault={(ev) => {
dragover(ev, item)
}}
on:drop|preventDefault={onDrop}
on:dragstart={() => {
selected = item
}}
on:dragend={() => {
selected = undefined
}}
>
<div class="flex-row-center">
<div class="circles-mark"><IconCircles size={'small'} /></div>
<span class="overflow-label mx-2">{item}</span>
</div>
<ActionIcon
icon={IconDelete}
label={setting.string.Delete}
action={() => {
remove(item)
}}
size={'small'}
/>
{#each values as item, i}
<button
bind:this={elements[i]}
draggable={!disableMouseOver}
class="hulyTableAttr-content__row"
class:disableMouseOver
class:hovered={opened === i && !disableMouseOver}
class:selected={selected === item}
on:dragover|preventDefault={(ev) => {
dragover(ev, item)
}}
on:drop|preventDefault={onDrop}
on:dragstart={() => {
selected = item
}}
on:dragend={() => {
selected = undefined
}}
>
<button class="hulyTableAttr-content__row-dragMenu" class:drag={!disableMouseOver}>
<IconMoreV2 size={'small'} />
</button>
<div class="hulyTableAttr-content__row-label font-regular-14 accent">
{item}
</div>
{/each}
{#if filtered.length}<div class="antiVSpacer x4" />{/if}
</Scroller>
{#if filtered.length === 0}
<Label label={presentation.string.NoMatchesFound} />
{/if}
<div class="hulyTableAttr-content__row-label grow" />
{#if !disableMouseOver}
<ButtonIcon
kind={'tertiary'}
icon={IconMoreV}
iconProps={{ fill: 'var(--global-tertiary-TextColor)' }}
size={'small'}
pressed={opened === i}
hasMenu
on:click={(ev) => {
openPopup(ev, i)
}}
/>
{/if}
</button>
{/each}
<style lang="scss">
.item {

View File

@ -0,0 +1,37 @@
<!--
// Copyright © 2024 Hardcore Engineering Inc.
//
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License. You may
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//
// See the License for the specific language governing permissions and
// limitations under the License.
-->
<script lang="ts">
export let size: 'small' | 'medium' | 'large'
const fill: string = 'currentColor'
</script>
<svg class="svg-{size}" {fill} viewBox="0 0 32 32" xmlns="http://www.w3.org/2000/svg">
<path d="M4 9C5.10457 9 6 8.10457 6 7C6 5.89543 5.10457 5 4 5C2.89543 5 2 5.89543 2 7C2 8.10457 2.89543 9 4 9Z" />
<path
d="M9 7C9 6.44772 9.44772 6 10 6H29C29.5523 6 30 6.44772 30 7C30 7.55228 29.5523 8 29 8H10C9.44772 8 9 7.55228 9 7Z"
/>
<path
d="M9 16C9 15.4477 9.44772 15 10 15H29C29.5523 15 30 15.4477 30 16C30 16.5523 29.5523 17 29 17H10C9.44772 17 9 16.5523 9 16Z"
/>
<path
d="M10 24C9.44772 24 9 24.4477 9 25C9 25.5523 9.44772 26 10 26H29C29.5523 26 30 25.5523 30 25C30 24.4477 29.5523 24 29 24H10Z"
/>
<path
d="M6 16C6 17.1046 5.10457 18 4 18C2.89543 18 2 17.1046 2 16C2 14.8954 2.89543 14 4 14C5.10457 14 6 14.8954 6 16Z"
/>
<path
d="M4 27C5.10457 27 6 26.1046 6 25C6 23.8954 5.10457 23 4 23C2.89543 23 2 23.8954 2 25C2 26.1046 2.89543 27 4 27Z"
/>
</svg>

View File

@ -0,0 +1,24 @@
<!--
// Copyright © 2024 Hardcore Engineering Inc.
//
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License. You may
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//
// See the License for the specific language governing permissions and
// limitations under the License.
-->
<script lang="ts">
export let size: 'small' | 'medium' | 'large'
const fill: string = 'currentColor'
</script>
<svg class="svg-{size}" {fill} viewBox="0 0 32 32" xmlns="http://www.w3.org/2000/svg">
<path
d="M26.1696 9.00012L22.5898 12.5899L23.9998 13.9999L29.9998 7.99988L23.9998 1.99988L22.5898 3.40988L26.1698 6.99988L21.7712 7.00012C19.7025 7.00012 17.7797 8.06584 16.6832 9.82013L14 14.1133L11.3168 9.82013C10.2203 8.06584 8.2975 7.00012 6.22876 7.00012H3C2.44772 7.00012 2 7.44784 2 8.00012C2 8.55241 2.44772 9.00012 3 9.00012H6.22876C7.60792 9.00012 8.8898 9.7106 9.62075 10.8801L12.8208 16.0001L9.62075 21.1201C8.8898 22.2896 7.60792 23.0001 6.22876 23.0001H3C2.44772 23.0001 2 23.4478 2 24.0001C2 24.5524 2.44772 25.0001 3 25.0001H6.22876C8.2975 25.0001 10.2203 23.9344 11.3168 22.1801L14 17.8869L16.6832 22.1801C17.7797 23.9344 19.7025 25.0001 21.7712 25.0001H26.1696L22.5898 28.5899L23.9998 29.9999L29.9998 23.9999L23.9998 17.9999L22.5898 19.4099L26.1698 22.9999L21.7712 23.0001C20.3921 23.0001 19.1102 22.2896 18.3792 21.1201L15.1792 16.0001L18.3792 10.8801C19.1102 9.7106 20.3921 9.00012 21.7712 9.00012H26.1696Z"
/>
</svg>

View File

@ -0,0 +1,35 @@
<!--
// Copyright © 2024 Hardcore Engineering Inc.
//
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License. You may
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//
// See the License for the specific language governing permissions and
// limitations under the License.
-->
<script lang="ts">
export let size: 'small' | 'medium' | 'large'
const fill: string = 'currentColor'
</script>
<svg class="svg-{size}" {fill} viewBox="0 0 32 32" xmlns="http://www.w3.org/2000/svg">
<path
d="M10 14.0005C9.44772 14.0005 9 14.4482 9 15.0005C9 15.5528 9.44772 16.0005 10 16.0005H22C22.5523 16.0005 23 15.5528 23 15.0005C23 14.4482 22.5523 14.0005 22 14.0005H10Z"
/>
<path
d="M9 19.0005C9 18.4482 9.44772 18.0005 10 18.0005H18C18.5523 18.0005 19 18.4482 19 19.0005C19 19.5528 18.5523 20.0005 18 20.0005H10C9.44772 20.0005 9 19.5528 9 19.0005Z"
/>
<path
d="M10 22.0005C9.44772 22.0005 9 22.4482 9 23.0005C9 23.5528 9.44772 24.0005 10 24.0005H16C16.5523 24.0005 17 23.5528 17 23.0005C17 22.4482 16.5523 22.0005 16 22.0005H10Z"
/>
<path
fill-rule="evenodd"
clip-rule="evenodd"
d="M19 2.00049C20.3062 2.00049 21.4175 2.8353 21.8293 4.00049H23C25.2091 4.00049 27 5.79135 27 8.00049V26.0005C27 28.2096 25.2091 30.0005 23 30.0005H9C6.79086 30.0005 5 28.2096 5 26.0005V8.00049C5 5.79135 6.79086 4.00049 9 4.00049H10.1707C10.5825 2.8353 11.6938 2.00049 13 2.00049H19ZM12 5.00049C12 4.4482 12.4477 4.00049 13 4.00049H19C19.5523 4.00049 20 4.4482 20 5.00049C20 5.55277 19.5523 6.00049 19 6.00049H13C12.4477 6.00049 12 5.55277 12 5.00049ZM19 8.00049C20.3062 8.00049 21.4175 7.16568 21.8293 6.00049H23C24.1046 6.00049 25 6.89592 25 8.00049V26.0005C25 27.1051 24.1046 28.0005 23 28.0005H9C7.89543 28.0005 7 27.1051 7 26.0005V8.00049C7 6.89592 7.89543 6.00049 9 6.00049H10.1707C10.5825 7.16568 11.6938 8.00049 13 8.00049H19Z"
/>
</svg>

View File

@ -44,7 +44,16 @@ export default mergeIds(settingId, setting, {
CreatingAttribute: '' as IntlString,
EditAttribute: '' as IntlString,
CreateEnum: '' as IntlString,
EditEnum: '' as IntlString,
Enums: '' as IntlString,
EnumsSettingHint: '' as IntlString,
EnumTitle: '' as IntlString,
EnumsCount: '' as IntlString,
ProjectTypesCount: '' as IntlString,
Options: '' as IntlString,
EnterOptionTitle: '' as IntlString,
NewEnumDialogClose: '' as IntlString,
NewEnumDialogCloseNote: '' as IntlString,
NewValue: '' as IntlString,
Leave: '' as IntlString,
LeaveDescr: '' as IntlString,

View File

@ -27,7 +27,10 @@
TextArea,
getPlatformColorDef,
themeStore,
EmojiPopup
EmojiPopup,
ButtonIcon,
IconDelete,
IconCopy
} from '@hcengineering/ui'
import { statusStore, ColorsPopup } from '@hcengineering/view-resources'
import view from '@hcengineering/view-resources/src/plugin'
@ -137,6 +140,10 @@
clearSettingsStore()
}}
>
<svelte:fragment slot="actions">
<ButtonIcon icon={IconDelete} size={'small'} kind={'tertiary'} />
<ButtonIcon icon={IconCopy} size={'small'} kind={'tertiary'} />
</svelte:fragment>
<div class="hulyModal-content__titleGroup">
<ModernEditbox bind:value label={task.string.StatusName} size={'large'} kind={'ghost'} />
<TextArea