converse-roomslist.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287
  1. // Converse.js (A browser based XMPP chat client)
  2. // http://conversejs.org
  3. //
  4. // Copyright (c) 2013-2018, Jan-Carel Brand <jc@opkode.com>
  5. // Licensed under the Mozilla Public License (MPLv2)
  6. /* This is a non-core Converse.js plugin which shows a list of currently open
  7. * rooms in the "Rooms Panel" of the ControlBox.
  8. */
  9. import converse from "@converse/headless/converse-core";
  10. import muc from "@converse/headless/converse-muc";
  11. import tpl_rooms_list from "templates/rooms_list.html";
  12. import tpl_rooms_list_item from "templates/rooms_list_item.html"
  13. const { Backbone, Promise, Strophe, b64_sha1, sizzle, _ } = converse.env;
  14. const u = converse.env.utils;
  15. converse.plugins.add('converse-roomslist', {
  16. /* Optional dependencies are other plugins which might be
  17. * overridden or relied upon, and therefore need to be loaded before
  18. * this plugin. They are called "optional" because they might not be
  19. * available, in which case any overrides applicable to them will be
  20. * ignored.
  21. *
  22. * It's possible however to make optional dependencies non-optional.
  23. * If the setting "strict_plugin_dependencies" is set to true,
  24. * an error will be raised if the plugin is not found.
  25. *
  26. * NB: These plugins need to have already been loaded via require.js.
  27. */
  28. dependencies: ["converse-singleton", "converse-controlbox", "converse-muc", "converse-bookmarks"],
  29. initialize () {
  30. /* The initialize function gets called as soon as the plugin is
  31. * loaded by converse.js's plugin machinery.
  32. */
  33. const { _converse } = this,
  34. { __ } = _converse;
  35. _converse.OpenRooms = Backbone.Collection.extend({
  36. comparator (room) {
  37. if (room.get('bookmarked')) {
  38. const bookmark = _.head(_converse.bookmarksview.model.where({'jid': room.get('jid')}));
  39. return bookmark.get('name');
  40. } else {
  41. return room.get('name');
  42. }
  43. },
  44. initialize () {
  45. _converse.chatboxes.on('add', this.onChatBoxAdded, this);
  46. _converse.chatboxes.on('change:hidden', this.onChatBoxChanged, this);
  47. _converse.chatboxes.on('change:bookmarked', this.onChatBoxChanged, this);
  48. _converse.chatboxes.on('change:name', this.onChatBoxChanged, this);
  49. _converse.chatboxes.on('change:num_unread', this.onChatBoxChanged, this);
  50. _converse.chatboxes.on('change:num_unread_general', this.onChatBoxChanged, this);
  51. _converse.chatboxes.on('remove', this.onChatBoxRemoved, this);
  52. this.reset(_.map(_converse.chatboxes.where({'type': 'chatroom'}), 'attributes'));
  53. },
  54. onChatBoxAdded (item) {
  55. if (item.get('type') === 'chatroom') {
  56. this.create(item.attributes);
  57. }
  58. },
  59. onChatBoxChanged (item) {
  60. if (item.get('type') === 'chatroom') {
  61. const room = this.get(item.get('jid'));
  62. if (!_.isNil(room)) {
  63. room.set(item.attributes);
  64. }
  65. }
  66. },
  67. onChatBoxRemoved (item) {
  68. if (item.get('type') === 'chatroom') {
  69. const room = this.get(item.get('jid'))
  70. this.remove(room);
  71. }
  72. }
  73. });
  74. _converse.RoomsList = Backbone.Model.extend({
  75. defaults: {
  76. "toggle-state": _converse.OPENED
  77. }
  78. });
  79. _converse.RoomsListElementView = Backbone.VDOMView.extend({
  80. events: {
  81. 'click .room-info': 'showRoomDetailsModal'
  82. },
  83. initialize () {
  84. this.model.on('destroy', this.remove, this);
  85. this.model.on('remove', this.remove, this);
  86. this.model.on('change:bookmarked', this.render, this);
  87. this.model.on('change:hidden', this.render, this);
  88. this.model.on('change:name', this.render, this);
  89. this.model.on('change:num_unread', this.render, this);
  90. this.model.on('change:num_unread_general', this.render, this);
  91. },
  92. toHTML () {
  93. return tpl_rooms_list_item(
  94. _.extend(this.model.toJSON(), {
  95. // XXX: By the time this renders, the _converse.bookmarks
  96. // collection should already exist if bookmarks are
  97. // supported by the XMPP server. So we can use it
  98. // as a check for support (other ways of checking are async).
  99. 'allow_bookmarks': _converse.allow_bookmarks && _converse.bookmarks,
  100. 'currently_open': _converse.isSingleton() && !this.model.get('hidden'),
  101. 'info_leave_room': __('Leave this groupchat'),
  102. 'info_remove_bookmark': __('Unbookmark this groupchat'),
  103. 'info_add_bookmark': __('Bookmark this groupchat'),
  104. 'info_title': __('Show more information on this groupchat'),
  105. 'name': this.getRoomsListElementName(),
  106. 'open_title': __('Click to open this groupchat')
  107. }));
  108. },
  109. showRoomDetailsModal (ev) {
  110. const room = _converse.chatboxes.get(this.model.get('jid'));
  111. ev.preventDefault();
  112. if (_.isUndefined(room.room_details_modal)) {
  113. room.room_details_modal = new _converse.RoomDetailsModal({'model': room});
  114. }
  115. room.room_details_modal.show(ev);
  116. },
  117. getRoomsListElementName () {
  118. if (this.model.get('bookmarked') && _converse.bookmarksview) {
  119. const bookmark = _.head(_converse.bookmarksview.model.where({'jid': this.model.get('jid')}));
  120. return bookmark.get('name');
  121. } else {
  122. return this.model.get('name');
  123. }
  124. }
  125. });
  126. _converse.RoomsListView = Backbone.OrderedListView.extend({
  127. tagName: 'div',
  128. className: 'open-rooms-list list-container rooms-list-container',
  129. events: {
  130. 'click .add-bookmark': 'addBookmark',
  131. 'click .close-room': 'closeRoom',
  132. 'click .list-toggle': 'toggleRoomsList',
  133. 'click .remove-bookmark': 'removeBookmark',
  134. 'click .open-room': 'openRoom',
  135. },
  136. listSelector: '.rooms-list',
  137. ItemView: _converse.RoomsListElementView,
  138. subviewIndex: 'jid',
  139. initialize () {
  140. Backbone.OrderedListView.prototype.initialize.apply(this, arguments);
  141. this.model.on('add', this.showOrHide, this);
  142. this.model.on('remove', this.showOrHide, this);
  143. const storage = _converse.config.get('storage'),
  144. id = b64_sha1(`converse.roomslist${_converse.bare_jid}`);
  145. this.list_model = new _converse.RoomsList({'id': id});
  146. this.list_model.browserStorage = new Backbone.BrowserStorage[storage](id);
  147. this.list_model.fetch();
  148. this.render();
  149. this.sortAndPositionAllItems();
  150. },
  151. render () {
  152. this.el.innerHTML = tpl_rooms_list({
  153. 'toggle_state': this.list_model.get('toggle-state'),
  154. 'desc_rooms': __('Click to toggle the list of open groupchats'),
  155. 'label_rooms': __('Open Groupchats'),
  156. '_converse': _converse
  157. });
  158. if (this.list_model.get('toggle-state') !== _converse.OPENED) {
  159. this.el.querySelector('.open-rooms-list').classList.add('collapsed');
  160. }
  161. this.showOrHide();
  162. this.insertIntoControlBox();
  163. return this;
  164. },
  165. insertIntoControlBox () {
  166. const controlboxview = _converse.chatboxviews.get('controlbox');
  167. if (!_.isUndefined(controlboxview) && !u.rootContains(_converse.root, this.el)) {
  168. const el = controlboxview.el.querySelector('.open-rooms-list');
  169. if (!_.isNull(el)) {
  170. el.parentNode.replaceChild(this.el, el);
  171. }
  172. }
  173. },
  174. hide () {
  175. u.hideElement(this.el);
  176. },
  177. show () {
  178. u.showElement(this.el);
  179. },
  180. openRoom (ev) {
  181. ev.preventDefault();
  182. const name = ev.target.textContent;
  183. const jid = ev.target.getAttribute('data-room-jid');
  184. const data = {
  185. 'name': name || Strophe.unescapeNode(Strophe.getNodeFromJid(jid)) || jid
  186. }
  187. _converse.api.rooms.open(jid, data);
  188. },
  189. closeRoom (ev) {
  190. ev.preventDefault();
  191. const name = ev.target.getAttribute('data-room-name');
  192. const jid = ev.target.getAttribute('data-room-jid');
  193. if (confirm(__("Are you sure you want to leave the groupchat %1$s?", name))) {
  194. // TODO: replace with API call
  195. _converse.chatboxviews.get(jid).close();
  196. }
  197. },
  198. showOrHide (item) {
  199. if (!this.model.models.length) {
  200. u.hideElement(this.el);
  201. } else {
  202. u.showElement(this.el);
  203. }
  204. },
  205. removeBookmark: _converse.removeBookmarkViaEvent,
  206. addBookmark: _converse.addBookmarkViaEvent,
  207. toggleRoomsList (ev) {
  208. if (ev && ev.preventDefault) { ev.preventDefault(); }
  209. const icon_el = ev.target.querySelector('.fa');
  210. if (icon_el.classList.contains("fa-caret-down")) {
  211. u.slideIn(this.el.querySelector('.open-rooms-list')).then(() => {
  212. this.list_model.save({'toggle-state': _converse.CLOSED});
  213. icon_el.classList.remove("fa-caret-down");
  214. icon_el.classList.add("fa-caret-right");
  215. });
  216. } else {
  217. u.slideOut(this.el.querySelector('.open-rooms-list')).then(() => {
  218. this.list_model.save({'toggle-state': _converse.OPENED});
  219. icon_el.classList.remove("fa-caret-right");
  220. icon_el.classList.add("fa-caret-down");
  221. });
  222. }
  223. }
  224. });
  225. const initRoomsListView = function () {
  226. const storage = _converse.config.get('storage'),
  227. id = b64_sha1(`converse.open-rooms-{_converse.bare_jid}`),
  228. model = new _converse.OpenRooms();
  229. model.browserStorage = new Backbone.BrowserStorage[storage](id);
  230. _converse.rooms_list_view = new _converse.RoomsListView({'model': model});
  231. };
  232. if (_converse.allow_bookmarks) {
  233. u.onMultipleEvents([
  234. {'object': _converse, 'event': 'chatBoxesFetched'},
  235. {'object': _converse, 'event': 'roomsPanelRendered'},
  236. {'object': _converse, 'event': 'bookmarksInitialized'}
  237. ], initRoomsListView);
  238. } else {
  239. u.onMultipleEvents([
  240. {'object': _converse, 'event': 'chatBoxesFetched'},
  241. {'object': _converse, 'event': 'roomsPanelRendered'}
  242. ], initRoomsListView);
  243. }
  244. _converse.api.listen.on('reconnected', initRoomsListView);
  245. }
  246. });