Mocoapp Textarea Fix

Allows to add line breaks in the description textarea of the Moco booking tool

// ==UserScript==
// @name         Mocoapp Textarea Fix
// @namespace    http://tampermonkey.net/
// @version      2025-04-08-03
// @description  Allows to add line breaks in the description textarea of the Moco booking tool
// @author       opctim
// @license      MIT
// @match        https://*.mocoapp.com/*
// @icon         
// @grant        none
// ==/UserScript==

(function() {
    'use strict';

    window.addEventListener('load', function () {
        const style = document.createElement('style');
        style.innerHTML = '.activity-row .third > .flex > div, .tst-timesheet-activity span[title] { white-space: pre-wrap }';
        document.head.appendChild(style);

        const attachListener = (textarea) => {
            // Avoid attaching the listener multiple times
            if (textarea._shiftEnterHandled) return;
            textarea._shiftEnterHandled = true;

            textarea.addEventListener('keydown', function (event) {
                if (event.key === 'Enter' && event.shiftKey) {
                    event.stopPropagation();
                    console.log('Shift + Enter stopped on:', textarea);
                }
            });
        };

        // Attach to existing elements
        document.querySelectorAll('textarea[name="description"]').forEach(attachListener);

        // Observe DOM changes to catch newly added textareas
        const observer = new MutationObserver((mutations) => {
            for (const mutation of mutations) {
                for (const node of mutation.addedNodes) {
                    if (node.nodeType !== Node.ELEMENT_NODE) continue;

                    if (node.matches?.('textarea[name="description"]')) {
                        attachListener(node);
                    }

                    node.querySelectorAll?.('textarea[name="description"]').forEach(attachListener);
                }
            }
        });

        observer.observe(document.body, {
            childList: true,
            subtree: true,
        });
    });
})();