summaryrefslogtreecommitdiff
path: root/client-web/source/chat.ts
blob: 6e74139c2b4be73d605e867d81905e6853a2a925 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
/*
    This file is part of keks-meet (https://codeberg.org/metamuffin/keks-meet)
    which is licensed under the GNU Affero General Public License (version 3); see /COPYING.
    Copyright (C) 2022 metamuffin <metamuffin@disroot.org>
*/
/// <reference lib="dom" />

import { ChatMessage } from "../../common/packets.d.ts";
import { ediv, esection, espan, image_view, notify } from "./helper.ts";
import { log } from "./logger.ts";
import { PREFS } from "./preferences/mod.ts";
import { Room } from "./room.ts";
import { LocalUser } from "./user/local.ts";
import { User } from "./user/mod.ts";

export class Chat {
    messages: HTMLElement
    controls: HTMLElement
    send_el: HTMLInputElement
    element: HTMLElement

    constructor(public room: Room) {
        const send = document.createElement("input")
        send.ariaLabel = "send message"
        send.type = "text"
        send.placeholder = "send a message..."

        const messages = ediv({ class: "messages", aria_live: "polite" })
        const controls = ediv({ class: "controls" })
        controls.append(send)
        messages.append(document.createElement("hr"))

        this.element = esection({ class: "chat", aria_label: "chat", role: "dialog" }, messages, controls)
        this.messages = messages
        this.controls = controls
        this.send_el = send

        send.onkeydown = (ev) => {
            if (ev.code == "Enter") {
                if (send.value.trim().length == 0) {
                    // keybind for toggle chat is Enter, so lets close here
                    // TODO! open
                    // this.control.check = false
                    return
                }
                this.send({ text: send.value })
                send.value = ""
            }
        }
        document.onpaste = (pasteEvent) => {
            // TODO will only work when pasting a single image
            const item = pasteEvent.clipboardData?.items[0];
            if (!item) return
            if (item.type.indexOf("image") === 0) {
                log("*", "image pasted")
                const blob = item.getAsFile()
                if (!blob) return
                const reader = new FileReader();
                reader.onload = (event) => {
                    if (!event.target) return
                    if (typeof event.target.result != "string") return
                    this.send({ image: event.target.result })
                };
                reader.readAsDataURL(blob);
            }
        }
    }

    on_show(): void { this.focus() }
    focus() { this.send_el.focus() }
    send(msg: ChatMessage) {
        this.room.local_user.chat(msg)
        this.add_message(this.room.local_user, msg)
    }

    add_message(sender: User, message: ChatMessage) {
        const els = []
        if (message.text) els.push(espan(message.text, { class: "text" }))
        if (message.image) els.push(image_view(message.image, { class: "image" }))

        // TODO! open
        // this.shown = true
        const e = ediv({ class: "message" }, espan(sender.display_name, { class: "author" }), ": ", ...els)
        this.messages.append(e)
        e.scrollIntoView({ block: "end", behavior: "smooth", inline: "end" })

        let body_str = "(empty message)"
        if (message.text) body_str = message.text
        if (message.image) body_str = "(image)"
        if (!(sender instanceof LocalUser) && PREFS.notify_chat) notify(body_str, sender.display_name)
    }
}