Converse converse.js

Source: shared/chat/toolbar.js

  1. import './emoji-picker.js';
  2. import 'shared/chat/message-limit.js';
  3. import tplToolbar from './templates/toolbar.js';
  4. import { CustomElement } from 'shared/components/element.js';
  5. import { __ } from 'i18n';
  6. import { _converse, api, converse } from '@converse/headless/core';
  7. import { html } from 'lit';
  8. import { until } from 'lit/directives/until.js';
  9. import './styles/toolbar.scss';
  10. const Strophe = converse.env.Strophe
  11. export class ChatToolbar extends CustomElement {
  12. static get properties () {
  13. return {
  14. hidden_occupants: { type: Boolean },
  15. is_groupchat: { type: Boolean },
  16. message_limit: { type: Number },
  17. model: { type: Object },
  18. show_call_button: { type: Boolean },
  19. show_emoji_button: { type: Boolean },
  20. show_send_button: { type: Boolean },
  21. show_spoiler_button: { type: Boolean },
  22. }
  23. }
  24. connectedCallback () {
  25. super.connectedCallback();
  26. this.listenTo(this.model, 'change:composing_spoiler', () => this.requestUpdate());
  27. }
  28. render () {
  29. return tplToolbar(this);
  30. }
  31. firstUpdated () {
  32. /**
  33. * Triggered once the _converse.ChatBoxView's toolbar has been rendered
  34. * @event _converse#renderToolbar
  35. * @type { _converse.ChatBoxView }
  36. * @example _converse.api.listen.on('renderToolbar', this => { ... });
  37. */
  38. api.trigger('renderToolbar', this);
  39. }
  40. getButtons () {
  41. const buttons = [];
  42. if (this.show_emoji_button) {
  43. const chatview = _converse.chatboxviews.get(this.model.get('jid'));
  44. buttons.push(html`<converse-emoji-dropdown .chatview=${chatview}></converse-emoji-dropdown>`);
  45. }
  46. if (this.show_call_button) {
  47. const color = this.is_groupchat ? '--muc-toolbar-btn-color' : '--chat-toolbar-btn-color';
  48. const i18n_start_call = __('Start a call');
  49. buttons.push(html`
  50. <button class="toggle-call" @click=${this.toggleCall} title="${i18n_start_call}">
  51. <converse-icon color="var(${color})" class="fa fa-phone" size="1em"></converse-icon>
  52. </button>`
  53. );
  54. }
  55. const message_limit = api.settings.get('message_limit');
  56. if (message_limit) {
  57. buttons.push(html`
  58. <converse-message-limit-indicator .model=${this.model} class="right">
  59. </converse-message-limit-indicator>`
  60. );
  61. }
  62. if (this.show_spoiler_button) {
  63. buttons.push(this.getSpoilerButton());
  64. }
  65. const http_upload_promise = api.disco.supports(Strophe.NS.HTTPUPLOAD, _converse.domain);
  66. buttons.push(html`${until(http_upload_promise.then(is_supported => this.getHTTPUploadButton(is_supported)),'')}`);
  67. if (this.is_groupchat && api.settings.get('visible_toolbar_buttons')?.toggle_occupants) {
  68. const i18n_hide_occupants = __('Hide participants');
  69. const i18n_show_occupants = __('Show participants');
  70. buttons.push(html`
  71. <button class="toggle_occupants right"
  72. title="${this.hidden_occupants ? i18n_show_occupants : i18n_hide_occupants}"
  73. @click=${this.toggleOccupants}>
  74. <converse-icon
  75. color="var(--muc-toolbar-btn-color)"
  76. class="fa ${this.hidden_occupants ? `fa-angle-double-left` : `fa-angle-double-right`}"
  77. size="1em"></converse-icon>
  78. </button>`
  79. );
  80. }
  81. /**
  82. * *Hook* which allows plugins to add more buttons to a chat's toolbar
  83. * @event _converse#getToolbarButtons
  84. * @example
  85. * api.listen.on('getToolbarButtons', (toolbar_el, buttons) {
  86. * buttons.push(html`
  87. * <button @click=${() => alert('Foo!')}>Foo</button>`
  88. * );
  89. * return buttons;
  90. * }
  91. */
  92. return _converse.api.hook('getToolbarButtons', this, buttons);
  93. }
  94. getHTTPUploadButton (is_supported) {
  95. if (is_supported) {
  96. const i18n_choose_file = __('Choose a file to send')
  97. const color = this.is_groupchat ? '--muc-toolbar-btn-color' : '--chat-toolbar-btn-color';
  98. return html`
  99. <button title="${i18n_choose_file}" @click=${this.toggleFileUpload}>
  100. <converse-icon
  101. color="var(${color})"
  102. class="fa fa-paperclip"
  103. size="1em"></converse-icon>
  104. </button>
  105. <input type="file" @change=${this.onFileSelection} class="fileupload" multiple="" style="display:none"/>`;
  106. } else {
  107. return '';
  108. }
  109. }
  110. getSpoilerButton () {
  111. const model = this.model;
  112. if (!this.is_groupchat && !model.presence?.resources.length) {
  113. return;
  114. }
  115. let i18n_toggle_spoiler;
  116. if (model.get('composing_spoiler')) {
  117. i18n_toggle_spoiler = __("Click to write as a normal (non-spoiler) message");
  118. } else {
  119. i18n_toggle_spoiler = __("Click to write your message as a spoiler");
  120. }
  121. const color = this.is_groupchat ? '--muc-toolbar-btn-color' : '--chat-toolbar-btn-color';
  122. const markup = html`
  123. <button class="toggle-compose-spoiler"
  124. title="${i18n_toggle_spoiler}"
  125. @click=${this.toggleComposeSpoilerMessage}>
  126. <converse-icon
  127. color="var(${color})"
  128. class="fa ${model.get('composing_spoiler') ? 'fa-eye-slash' : 'fa-eye'}"
  129. size="1em"></converse-icon>
  130. </button>`;
  131. if (this.is_groupchat) {
  132. return markup;
  133. } else {
  134. const contact_jid = model.get('jid');
  135. const spoilers_promise = Promise.all(
  136. model.presence.resources.map(
  137. r => api.disco.supports(Strophe.NS.SPOILER, `${contact_jid}/${r.get('name')}`)
  138. )).then(results => results.reduce((acc, val) => (acc && val), true));
  139. return html`${until(spoilers_promise.then(() => markup), '')}`;
  140. }
  141. }
  142. toggleFileUpload (ev) {
  143. ev?.preventDefault?.();
  144. ev?.stopPropagation?.();
  145. this.querySelector('.fileupload').click();
  146. }
  147. onFileSelection (evt) {
  148. this.model.sendFiles(evt.target.files);
  149. }
  150. toggleComposeSpoilerMessage (ev) {
  151. ev?.preventDefault?.();
  152. ev?.stopPropagation?.();
  153. this.model.set('composing_spoiler', !this.model.get('composing_spoiler'));
  154. }
  155. toggleOccupants (ev) {
  156. ev?.preventDefault?.();
  157. ev?.stopPropagation?.();
  158. this.model.save({'hidden_occupants': !this.model.get('hidden_occupants')});
  159. }
  160. toggleCall (ev) {
  161. ev?.preventDefault?.();
  162. ev?.stopPropagation?.();
  163. /**
  164. * When a call button (i.e. with class .toggle-call) on a chatbox has been clicked.
  165. * @event _converse#callButtonClicked
  166. * @type { object }
  167. * @property { Strophe.Connection } _converse.connection - The XMPP Connection object
  168. * @property { _converse.ChatBox | _converse.ChatRoom } _converse.connection - The XMPP Connection object
  169. * @example _converse.api.listen.on('callButtonClicked', (connection, model) => { ... });
  170. */
  171. api.trigger('callButtonClicked', {
  172. connection: _converse.connection,
  173. model: this.model
  174. });
  175. }
  176. }
  177. api.elements.define('converse-chat-toolbar', ChatToolbar);