platform/plugins/view-resources/src/components/ActionsPopup.svelte
Andrey Sobolev 8d7e3078e1
Keyboard popup (#1625)
Signed-off-by: Andrey Sobolev <haiodo@gmail.com>
2022-05-04 15:08:12 +07:00

275 lines
8.1 KiB
Svelte

<!--
// Copyright © 2022 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">
import { WithLookup } from '@anticrm/core'
import { getResource, translate } from '@anticrm/platform'
import { createQuery, getClient } from '@anticrm/presentation'
import { closePopup, Icon, IconArrowLeft, Label } from '@anticrm/ui'
import { Action, ViewContext } from '@anticrm/view'
import { onMount } from 'svelte'
import { filterActions, getSelection } from '../actions'
import view from '../plugin'
import { focusStore, selectionStore } from '../selection'
import ActionContext from './ActionContext.svelte'
import ListView from './ListView.svelte'
import ObjectPresenter from './ObjectPresenter.svelte'
export let viewContext: ViewContext
let search: string = ''
let actions: WithLookup<Action>[] = []
let input: HTMLInputElement
const query = createQuery()
query.query(
view.class.Action,
{},
(res) => {
actions = res
},
{
lookup: {
category: view.class.ActionCategory
}
}
)
const targetQuery = createQuery()
targetQuery.query(view.class.Action, {}, (res) => {
actions = res
})
let supportedActions: WithLookup<Action>[] = []
let filteredActions: WithLookup<Action>[] = []
const client = getClient()
$: {
let fActions: WithLookup<Action>[] = actions
const docs = getSelection($focusStore, $selectionStore)
for (const d of docs) {
fActions = filterActions(client, d, fActions)
}
if (docs.length === 0) {
fActions = fActions.filter((it) => it.input === 'none')
}
fActions = fActions.filter(
(it) =>
(it.$lookup?.category?.visible ?? true) &&
(it.context.application === viewContext.application || it.context.application === undefined)
)
// Sort by category.
supportedActions = fActions.sort((a, b) => a.category.localeCompare(b.category))
}
async function filterSearchActions (actions: WithLookup<Action>[], search: string): Promise<void> {
const res: WithLookup<Action>[] = []
search = search.trim().toLowerCase()
if (search.length > 0) {
for (const a of actions) {
const tr = await translate(a.label, {})
if (tr.toLowerCase().indexOf(search) !== -1) {
res.push(a)
}
}
filteredActions = res
} else {
filteredActions = actions
}
}
$: filterSearchActions(supportedActions, search)
let phTraslate: string = ''
$: translate(view.string.ActionPlaceholder, {}).then((res) => {
phTraslate = res
})
onMount(() => {
if (input) input.focus()
})
let selection = 0
let list: ListView
/* eslint-disable no-undef */
async function handleSelection (evt: Event, selection: number): Promise<void> {
const action = filteredActions[selection]
const docs = getSelection($focusStore, $selectionStore)
if (action.input === 'focus') {
const impl = await getResource(action.action)
if (impl !== undefined) {
closePopup()
impl(docs[0], evt, action.actionProps)
}
}
if (action.input === 'selection' || action.input === 'any' || action.input === 'none') {
const impl = await getResource(action.action)
if (impl !== undefined) {
closePopup()
impl(docs, evt, action.actionProps)
}
}
}
function onKeydown (key: KeyboardEvent): void {
if (key.code === 'ArrowUp') {
key.stopPropagation()
key.preventDefault()
list.select(selection - 1)
}
if (key.code === 'ArrowDown') {
key.stopPropagation()
key.preventDefault()
list.select(selection + 1)
}
if (key.code === 'Enter') {
key.preventDefault()
key.stopPropagation()
handleSelection(key, selection)
}
if (key.code === 'Escape') {
key.preventDefault()
key.stopPropagation()
closePopup()
}
}
function formatKey (key: string): string[][] {
const thens = key.split('->')
const result: string[][] = []
for (const r of thens) {
result.push(
r.split('+').map((it) =>
it
.replaceAll('key', '')
.replaceAll(/Meta|meta/g, '⌘')
.replaceAll('ArrowUp', '↑')
.replaceAll('ArrowDown', '↓')
.replaceAll('ArrowLeft', '←')
.replaceAll('ArrowRight', '→')
.replaceAll('Backspace', '⌫')
)
)
}
return result
}
</script>
<ActionContext
context={{
mode: 'none'
}}
/>
<div class="selectPopup width-40" style:width="15rem" on:keydown={onKeydown}>
<div class="mt-2 ml-2">
{#if $selectionStore.length > 0}
<div class="item-box">
{$selectionStore.length} items
</div>
{:else if $focusStore.focus !== undefined}
<div class="item-box">
<ObjectPresenter
objectId={$focusStore.focus._id}
_class={$focusStore.focus._class}
value={$focusStore.focus}
props={{ inline: true }}
/>
</div>
{/if}
</div>
<div class="header">
<input bind:this={input} type="text" bind:value={search} placeholder={phTraslate} />
</div>
<div class="scroll">
<div class="box">
<ListView
bind:this={list}
count={filteredActions.length}
bind:selection
on:click={(evt) => handleSelection(evt, evt.detail)}
>
<svelte:fragment slot="category" let:item>
{@const action = filteredActions[item]}
{#if item === 0 || (item > 0 && filteredActions[item - 1].$lookup?.category?.label !== action.$lookup?.category?.label)}
<!--Category for first item-->
{#if action.$lookup?.category}
<div class="category-box">
<Label label={action.$lookup.category.label} />
</div>
{/if}
{/if}
</svelte:fragment>
<svelte:fragment slot="item" let:item>
{@const action = filteredActions[item]}
<div class="flex-row-center flex-between flex-grow ml-2 p-3 text-base">
<div class="mr-4">
<Icon icon={action.icon ?? IconArrowLeft} size={'small'} />
</div>
<div class="flex-grow">
<Label label={action.label} />
</div>
<div class="mr-2 text-md flex-row-center">
{#if action.keyBinding}
{#each action.keyBinding as key, i}
{#if i !== 0}
<div class="ml-2 mr-2">or</div>
{/if}
<div class="flex-row-center" class:ml-2={i !== 0}>
{#each formatKey(key) as k, jj}
{#if jj !== 0}
<div class="ml-1 mr-1">then</div>
{/if}
{#each k as kk, j}
<div class="flex-center text-sm key-box" class:ml-2={j !== 0}>
{kk}
</div>
{/each}
{/each}
</div>
{/each}
{/if}
</div>
</div>
</svelte:fragment>
</ListView>
</div>
</div>
</div>
<style lang="scss">
.key-box {
background-color: var(--divider-color);
color: var(--caption-color);
min-width: 1.5rem;
padding: 0.25rem;
}
.item-box {
display: inline-block;
background-color: var(--divider-color);
color: var(--caption-color);
border-radius: 0.5rem;
padding: 0.5rem;
}
.category-box {
display: inline-block;
background-color: var(--divider-color);
color: var(--caption-color);
padding: 0.5rem;
}
</style>