mirror of
https://github.com/hcengineering/platform.git
synced 2025-04-23 16:56:07 +00:00
151 lines
4.0 KiB
Svelte
151 lines
4.0 KiB
Svelte
<!--
|
|
// Copyright © 2020, 2021 Anticrm Platform Contributors.
|
|
// Copyright © 2021 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 { createEventDispatcher, onMount } from 'svelte'
|
|
import { Doc, Ref, SortingOrder } from '@hcengineering/core'
|
|
import { createQuery } from '@hcengineering/presentation'
|
|
import activity from '@hcengineering/activity'
|
|
import chunter, { ChatMessage } from '@hcengineering/chunter'
|
|
import { closeTooltip, Label, Lazy, Spinner, resizeObserver, MiniToggle } from '@hcengineering/ui'
|
|
import { ObjectPresenter, DocNavLink } from '@hcengineering/view-resources'
|
|
import { loadSavedMessages } from '@hcengineering/activity-resources'
|
|
|
|
import ChatMessageInput from './ChatMessageInput.svelte'
|
|
import ChatMessagePresenter from './ChatMessagePresenter.svelte'
|
|
|
|
export let objectId: Ref<Doc>
|
|
export let object: Doc
|
|
export let withInput: boolean = true
|
|
|
|
const dispatch = createEventDispatcher()
|
|
const query = createQuery()
|
|
|
|
let loading = true
|
|
let messages: ChatMessage[] = []
|
|
|
|
let activityOrderNewestFirst = JSON.parse(localStorage.getItem('activity-newest-first') ?? 'false')
|
|
$: localStorage.setItem('activity-newest-first', JSON.stringify(activityOrderNewestFirst))
|
|
$: query.query(
|
|
chunter.class.ChatMessage,
|
|
{ attachedTo: objectId },
|
|
(res) => {
|
|
messages = res.sort((message) => (message?.isPinned ? -1 : 1))
|
|
loading = false
|
|
},
|
|
{ sort: { createdOn: activityOrderNewestFirst ? SortingOrder.Descending : SortingOrder.Ascending } }
|
|
)
|
|
|
|
let isTextMode = false
|
|
|
|
$: if (isTextMode) {
|
|
dispatch('tooltip', { kind: 'popup' })
|
|
}
|
|
|
|
onMount(() => {
|
|
loadSavedMessages()
|
|
})
|
|
</script>
|
|
|
|
<div class="commentPopup-container">
|
|
<!-- svelte-ignore a11y-no-static-element-interactions -->
|
|
<div
|
|
class="flex-between header"
|
|
use:resizeObserver={() => {
|
|
dispatch('changeContent')
|
|
}}
|
|
on:keydown={(evt) => {
|
|
console.log(evt)
|
|
if (isTextMode) {
|
|
evt.preventDefault()
|
|
evt.stopImmediatePropagation()
|
|
closeTooltip()
|
|
}
|
|
}}
|
|
>
|
|
<div class="fs-title mr-2">
|
|
<Label label={chunter.string.Comments} />
|
|
</div>
|
|
<MiniToggle bind:on={activityOrderNewestFirst} label={activity.string.NewestFirst} />
|
|
<DocNavLink {object}>
|
|
<ObjectPresenter _class={object._class} objectId={object._id} value={object} />
|
|
</DocNavLink>
|
|
</div>
|
|
<div class="messages">
|
|
{#if loading}
|
|
<div class="flex-center">
|
|
<Spinner />
|
|
</div>
|
|
{:else}
|
|
{#each messages as message}
|
|
<div class="item">
|
|
<Lazy>
|
|
<ChatMessagePresenter value={message} />
|
|
</Lazy>
|
|
</div>
|
|
{/each}
|
|
{/if}
|
|
</div>
|
|
{#if withInput}
|
|
<div class="input">
|
|
<ChatMessageInput
|
|
{object}
|
|
on:focus={() => {
|
|
isTextMode = true
|
|
}}
|
|
/>
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
|
|
<style lang="scss">
|
|
.commentPopup-container {
|
|
overflow: hidden;
|
|
display: flex;
|
|
flex-direction: column;
|
|
padding: 0;
|
|
min-width: 0;
|
|
min-height: 0;
|
|
max-height: 30rem;
|
|
|
|
.header {
|
|
flex-shrink: 0;
|
|
margin: 0 0.25rem 0.5rem;
|
|
padding: 0.5rem 1.25rem 1rem 0.75rem;
|
|
border-bottom: 1px solid var(--theme-divider-color);
|
|
}
|
|
|
|
.messages {
|
|
overflow: auto;
|
|
flex: 1;
|
|
padding: 0 1rem;
|
|
min-width: 0;
|
|
min-height: 0;
|
|
|
|
.item {
|
|
max-width: 30rem;
|
|
}
|
|
|
|
.item + .item {
|
|
margin-top: 0.75rem;
|
|
}
|
|
}
|
|
|
|
.input {
|
|
padding: 0.5rem 0.25rem 0.25rem;
|
|
}
|
|
}
|
|
</style>
|