Mocoapp Textarea Fix

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

  1. // ==UserScript==
  2. // @name Mocoapp Textarea Fix
  3. // @namespace http://tampermonkey.net/
  4. // @version 2025-04-08-03
  5. // @description Allows to add line breaks in the description textarea of the Moco booking tool
  6. // @author opctim
  7. // @license MIT
  8. // @match https://*.mocoapp.com/*
  9. // @icon 
  10. // @grant none
  11. // ==/UserScript==
  12.  
  13. (function() {
  14. 'use strict';
  15.  
  16. window.addEventListener('load', function () {
  17. const style = document.createElement('style');
  18. style.innerHTML = '.activity-row .third > .flex > div, .tst-timesheet-activity span[title] { white-space: pre-wrap }';
  19. document.head.appendChild(style);
  20.  
  21. const attachListener = (textarea) => {
  22. // Avoid attaching the listener multiple times
  23. if (textarea._shiftEnterHandled) return;
  24. textarea._shiftEnterHandled = true;
  25.  
  26. textarea.addEventListener('keydown', function (event) {
  27. if (event.key === 'Enter' && event.shiftKey) {
  28. event.stopPropagation();
  29. console.log('Shift + Enter stopped on:', textarea);
  30. }
  31. });
  32. };
  33.  
  34. // Attach to existing elements
  35. document.querySelectorAll('textarea[name="description"]').forEach(attachListener);
  36.  
  37. // Observe DOM changes to catch newly added textareas
  38. const observer = new MutationObserver((mutations) => {
  39. for (const mutation of mutations) {
  40. for (const node of mutation.addedNodes) {
  41. if (node.nodeType !== Node.ELEMENT_NODE) continue;
  42.  
  43. if (node.matches?.('textarea[name="description"]')) {
  44. attachListener(node);
  45. }
  46.  
  47. node.querySelectorAll?.('textarea[name="description"]').forEach(attachListener);
  48. }
  49. }
  50. });
  51.  
  52. observer.observe(document.body, {
  53. childList: true,
  54. subtree: true,
  55. });
  56. });
  57. })();