Sfoglia il codice sorgente

Merge branch 'master' into converse-omemo

JC Brand 7 anni fa
parent
commit
4c3645c559

+ 4 - 0
CHANGES.md

@@ -30,6 +30,7 @@
 - Documentation includes utf-8 charset to make minfied versions compatible across platforms. #1017
 - Documentation includes utf-8 charset to make minfied versions compatible across platforms. #1017
 - #1026 Typing in MUC shows "Typing from another device"
 - #1026 Typing in MUC shows "Typing from another device"
 - #1039 Multi-option data form elements not shown and saved correctly
 - #1039 Multi-option data form elements not shown and saved correctly
+- #1143 Able to send blank message
 
 
 ### API changes
 ### API changes
 
 
@@ -39,6 +40,9 @@
 - New API method `_converse.api.vcard.update`.
 - New API method `_converse.api.vcard.update`.
 - The `contactStatusChanged` event has been renamed to `contactPresenceChanged`
 - The `contactStatusChanged` event has been renamed to `contactPresenceChanged`
   and a event `presenceChanged` is now also triggered on the contact.
   and a event `presenceChanged` is now also triggered on the contact.
+- `_converse.api.chats.open` and `_converse.api.rooms.open` now returns a 
+  `Presence` which resolves with the `Backbone.Model` representing the chat
+  object.
 
 
 ## UI changes
 ## UI changes
 
 

+ 1 - 0
Makefile

@@ -227,6 +227,7 @@ check: eslint
 html:
 html:
 	rm -rf $(BUILDDIR)/html
 	rm -rf $(BUILDDIR)/html
 	$(SPHINXBUILD) -b html $(ALLSPHINXOPTS) $(BUILDDIR)/html
 	$(SPHINXBUILD) -b html $(ALLSPHINXOPTS) $(BUILDDIR)/html
+	make apidoc
 	@echo
 	@echo
 	@echo "Build finished. The HTML pages are in $(BUILDDIR)/html."
 	@echo "Build finished. The HTML pages are in $(BUILDDIR)/html."
 
 

+ 218 - 115
dist/converse.js

@@ -4529,7 +4529,7 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
 /*! no static exports found */
 /*! no static exports found */
 /***/ (function(module, exports, __webpack_require__) {
 /***/ (function(module, exports, __webpack_require__) {
 
 
-/* WEBPACK VAR INJECTION */(function(global) {var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_DEFINE_RESULT__;// Native Javascript for Bootstrap 4 v2.0.22 | © dnp_theme | MIT-License
+/* WEBPACK VAR INJECTION */(function(global) {var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_DEFINE_RESULT__;// Native Javascript for Bootstrap 4 v2.0.23 | © dnp_theme | MIT-License
 (function (root, factory) {
 (function (root, factory) {
   if (true) {
   if (true) {
     // AMD support:
     // AMD support:
@@ -4612,7 +4612,7 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
     clickEvent    = 'click',
     clickEvent    = 'click',
     hoverEvent    = 'hover',
     hoverEvent    = 'hover',
     keydownEvent  = 'keydown',
     keydownEvent  = 'keydown',
-    keyupEvent    = 'keyup', 
+    keyupEvent    = 'keyup',
     resizeEvent   = 'resize',
     resizeEvent   = 'resize',
     scrollEvent   = 'scroll',
     scrollEvent   = 'scroll',
     // originalEvents
     // originalEvents
@@ -4632,18 +4632,20 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
     hasAttribute           = 'hasAttribute',
     hasAttribute           = 'hasAttribute',
     createElement          = 'createElement',
     createElement          = 'createElement',
     appendChild            = 'appendChild',
     appendChild            = 'appendChild',
-    innerHTML              = 'innerHTML',  
+    innerHTML              = 'innerHTML',
     getElementsByTagName   = 'getElementsByTagName',
     getElementsByTagName   = 'getElementsByTagName',
     preventDefault         = 'preventDefault',
     preventDefault         = 'preventDefault',
     getBoundingClientRect  = 'getBoundingClientRect',
     getBoundingClientRect  = 'getBoundingClientRect',
     querySelectorAll       = 'querySelectorAll',
     querySelectorAll       = 'querySelectorAll',
     getElementsByCLASSNAME = 'getElementsByClassName',
     getElementsByCLASSNAME = 'getElementsByClassName',
+    getComputedStyle       = 'getComputedStyle',  
   
   
     indexOf      = 'indexOf',
     indexOf      = 'indexOf',
     parentNode   = 'parentNode',
     parentNode   = 'parentNode',
     length       = 'length',
     length       = 'length',
     toLowerCase  = 'toLowerCase',
     toLowerCase  = 'toLowerCase',
     Transition   = 'Transition',
     Transition   = 'Transition',
+    Duration     = 'Duration',
     Webkit       = 'Webkit',
     Webkit       = 'Webkit',
     style        = 'style',
     style        = 'style',
     push         = 'push',
     push         = 'push',
@@ -4663,15 +4665,16 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
     // tooltip / popover
     // tooltip / popover
     mouseHover = ('onmouseleave' in DOC) ? [ 'mouseenter', 'mouseleave'] : [ 'mouseover', 'mouseout' ],
     mouseHover = ('onmouseleave' in DOC) ? [ 'mouseenter', 'mouseleave'] : [ 'mouseover', 'mouseout' ],
     tipPositions = /\b(top|bottom|left|right)+/,
     tipPositions = /\b(top|bottom|left|right)+/,
-    
+  
     // modal
     // modal
     modalOverlay = 0,
     modalOverlay = 0,
     fixedTop = 'fixed-top',
     fixedTop = 'fixed-top',
     fixedBottom = 'fixed-bottom',
     fixedBottom = 'fixed-bottom',
-    
+  
     // transitionEnd since 2.0.4
     // transitionEnd since 2.0.4
     supportTransitions = Webkit+Transition in HTML[style] || Transition[toLowerCase]() in HTML[style],
     supportTransitions = Webkit+Transition in HTML[style] || Transition[toLowerCase]() in HTML[style],
     transitionEndEvent = Webkit+Transition in HTML[style] ? Webkit[toLowerCase]()+Transition+'End' : Transition[toLowerCase]()+'end',
     transitionEndEvent = Webkit+Transition in HTML[style] ? Webkit[toLowerCase]()+Transition+'End' : Transition[toLowerCase]()+'end',
+    transitionDuration = Webkit+Duration in HTML[style] ? Webkit[toLowerCase]()+Transition+Duration : Transition[toLowerCase]()+Duration,
   
   
     // set new focus element since 2.0.3
     // set new focus element since 2.0.3
     setFocus = function(element){
     setFocus = function(element){
@@ -4725,9 +4728,16 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
         off(element, event, handlerWrapper);
         off(element, event, handlerWrapper);
       });
       });
     },
     },
+    getTransitionDurationFromElement = function(element) {
+      var duration = globalObject[getComputedStyle](element)[transitionDuration];
+      duration = parseFloat(duration);
+      duration = typeof duration === 'number' && !isNaN(duration) ? duration * 1000 : 0;
+      return duration + 50; // we take a short offset to make sure we fire on the next frame after animation
+    },
     emulateTransitionEnd = function(element,handler){ // emulateTransitionEnd since 2.0.4
     emulateTransitionEnd = function(element,handler){ // emulateTransitionEnd since 2.0.4
-      if (supportTransitions) { one(element, transitionEndEvent, function(e){ handler(e); }); }
-      else { handler(); }
+      var called = 0, duration = getTransitionDurationFromElement(element);
+      supportTransitions && one(element, transitionEndEvent, function(e){ handler(e); called = 1; });
+      setTimeout(function() { !called && handler(); }, duration);
     },
     },
     bootstrapCustomEvent = function (eventName, componentName, related) {
     bootstrapCustomEvent = function (eventName, componentName, related) {
       var OriginalCustomEvent = new CustomEvent( eventName + '.bs.' + componentName);
       var OriginalCustomEvent = new CustomEvent( eventName + '.bs.' + componentName);
@@ -4750,8 +4760,8 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
           scroll = parent === DOC[body] ? getScroll() : { x: parent[offsetLeft] + parent[scrollLeft], y: parent[offsetTop] + parent[scrollTop] },
           scroll = parent === DOC[body] ? getScroll() : { x: parent[offsetLeft] + parent[scrollLeft], y: parent[offsetTop] + parent[scrollTop] },
           linkDimensions = { w: rect[right] - rect[left], h: rect[bottom] - rect[top] },
           linkDimensions = { w: rect[right] - rect[left], h: rect[bottom] - rect[top] },
           isPopover = hasClass(element,'popover'),
           isPopover = hasClass(element,'popover'),
-          topPosition, leftPosition, 
-          
+          topPosition, leftPosition,
+  
           arrow = queryElement('.arrow',element),
           arrow = queryElement('.arrow',element),
           arrowTop, arrowLeft, arrowWidth, arrowHeight,
           arrowTop, arrowLeft, arrowWidth, arrowHeight,
   
   
@@ -4770,7 +4780,7 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
       position = position === bottom && bottomExceed ? top : position;
       position = position === bottom && bottomExceed ? top : position;
       position = position === left && leftExceed ? right : position;
       position = position === left && leftExceed ? right : position;
       position = position === right && rightExceed ? left : position;
       position = position === right && rightExceed ? left : position;
-      
+  
       // update tooltip/popover class
       // update tooltip/popover class
       element.className[indexOf](position) === -1 && (element.className = element.className.replace(tipPositions,position));
       element.className[indexOf](position) === -1 && (element.className = element.className.replace(tipPositions,position));
   
   
@@ -4823,7 +4833,7 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
       arrowLeft && (arrow[style][left] = arrowLeft + 'px');
       arrowLeft && (arrow[style][left] = arrowLeft + 'px');
     };
     };
   
   
-  BSN.version = '2.0.22';
+  BSN.version = '2.0.23';
   
   
   /* Native Javascript for Bootstrap 4 | Alert
   /* Native Javascript for Bootstrap 4 | Alert
   -------------------------------------------*/
   -------------------------------------------*/
@@ -4993,7 +5003,7 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
     // DATA API
     // DATA API
     var intervalAttribute = element[getAttribute](dataInterval),
     var intervalAttribute = element[getAttribute](dataInterval),
         intervalOption = options[interval],
         intervalOption = options[interval],
-        intervalData = intervalAttribute === 'false' ? 0 : parseInt(intervalAttribute) || 5000,  // bootstrap carousel default interval
+        intervalData = intervalAttribute === 'false' ? 0 : parseInt(intervalAttribute),  
         pauseData = element[getAttribute](dataPause) === hoverEvent || false,
         pauseData = element[getAttribute](dataPause) === hoverEvent || false,
         keyboardData = element[getAttribute](dataKeyboard) === 'true' || false,
         keyboardData = element[getAttribute](dataKeyboard) === 'true' || false,
       
       
@@ -5008,8 +5018,8 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
     this[pause] = (options[pause] === hoverEvent || pauseData) ? hoverEvent : false; // false / hover
     this[pause] = (options[pause] === hoverEvent || pauseData) ? hoverEvent : false; // false / hover
   
   
     this[interval] = typeof intervalOption === 'number' ? intervalOption
     this[interval] = typeof intervalOption === 'number' ? intervalOption
-                   : intervalData === 0 ? 0
-                   : intervalData;
+                   : intervalOption === false || intervalData === 0 || intervalData === false ? 0
+                   : 5000; // bootstrap carousel default interval
   
   
     // bind, event targets
     // bind, event targets
     var self = this, index = element.index = 0, timer = element.timer = 0, 
     var self = this, index = element.index = 0, timer = element.timer = 0, 
@@ -5128,10 +5138,10 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
         addClass(slides[next],carouselItem +'-'+ slideDirection);
         addClass(slides[next],carouselItem +'-'+ slideDirection);
         addClass(slides[activeItem],carouselItem +'-'+ slideDirection);
         addClass(slides[activeItem],carouselItem +'-'+ slideDirection);
   
   
-        one(slides[activeItem], transitionEndEvent, function(e) {
-          var timeout = e[target] !== slides[activeItem] ? e.elapsedTime*1000 : 0;
+        one(slides[next], transitionEndEvent, function(e) {
+          var timeout = e[target] !== slides[next] ? e.elapsedTime*1000+100 : 20;
           
           
-          setTimeout(function(){
+          isSliding && setTimeout(function(){
             isSliding = false;
             isSliding = false;
   
   
             addClass(slides[next],active);
             addClass(slides[next],active);
@@ -5146,7 +5156,7 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
             if ( !DOC.hidden && self[interval] && !hasClass(element,paused) ) {
             if ( !DOC.hidden && self[interval] && !hasClass(element,paused) ) {
               self.cycle();
               self.cycle();
             }
             }
-          },timeout+100);
+          }, timeout);
         });
         });
   
   
       } else {
       } else {
@@ -5211,23 +5221,24 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
   
   
     // event targets and constants
     // event targets and constants
     var accordion = null, collapse = null, self = this, 
     var accordion = null, collapse = null, self = this, 
-      isAnimating = false, // when true it will prevent click handlers
       accordionData = element[getAttribute]('data-parent'),
       accordionData = element[getAttribute]('data-parent'),
+      activeCollapse, activeElement,
   
   
       // component strings
       // component strings
       component = 'collapse',
       component = 'collapse',
       collapsed = 'collapsed',
       collapsed = 'collapsed',
+      isAnimating = 'isAnimating',
   
   
       // private methods
       // private methods
       openAction = function(collapseElement,toggle) {
       openAction = function(collapseElement,toggle) {
         bootstrapCustomEvent.call(collapseElement, showEvent, component);
         bootstrapCustomEvent.call(collapseElement, showEvent, component);
-        isAnimating = true;
+        collapseElement[isAnimating] = true;
         addClass(collapseElement,collapsing);
         addClass(collapseElement,collapsing);
         removeClass(collapseElement,component);
         removeClass(collapseElement,component);
         collapseElement[style][height] = collapseElement[scrollHeight] + 'px';
         collapseElement[style][height] = collapseElement[scrollHeight] + 'px';
         
         
         emulateTransitionEnd(collapseElement, function() {
         emulateTransitionEnd(collapseElement, function() {
-          isAnimating = false;
+          collapseElement[isAnimating] = false;
           collapseElement[setAttribute](ariaExpanded,'true');
           collapseElement[setAttribute](ariaExpanded,'true');
           toggle[setAttribute](ariaExpanded,'true');
           toggle[setAttribute](ariaExpanded,'true');
           removeClass(collapseElement,collapsing);
           removeClass(collapseElement,collapsing);
@@ -5239,7 +5250,7 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
       },
       },
       closeAction = function(collapseElement,toggle) {
       closeAction = function(collapseElement,toggle) {
         bootstrapCustomEvent.call(collapseElement, hideEvent, component);
         bootstrapCustomEvent.call(collapseElement, hideEvent, component);
-        isAnimating = true;
+        collapseElement[isAnimating] = true;
         collapseElement[style][height] = collapseElement[scrollHeight] + 'px'; // set height first
         collapseElement[style][height] = collapseElement[scrollHeight] + 'px'; // set height first
         removeClass(collapseElement,component);
         removeClass(collapseElement,component);
         removeClass(collapseElement,showClass);
         removeClass(collapseElement,showClass);
@@ -5248,7 +5259,7 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
         collapseElement[style][height] = '0px';
         collapseElement[style][height] = '0px';
         
         
         emulateTransitionEnd(collapseElement, function() {
         emulateTransitionEnd(collapseElement, function() {
-          isAnimating = false;
+          collapseElement[isAnimating] = false;
           collapseElement[setAttribute](ariaExpanded,'false');
           collapseElement[setAttribute](ariaExpanded,'false');
           toggle[setAttribute](ariaExpanded,'false');
           toggle[setAttribute](ariaExpanded,'false');
           removeClass(collapseElement,collapsing);
           removeClass(collapseElement,collapsing);
@@ -5267,29 +5278,29 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
     // public methods
     // public methods
     this.toggle = function(e) {
     this.toggle = function(e) {
       e[preventDefault]();
       e[preventDefault]();
-      if (isAnimating) return;
       if (!hasClass(collapse,showClass)) { self.show(); } 
       if (!hasClass(collapse,showClass)) { self.show(); } 
       else { self.hide(); }
       else { self.hide(); }
     };
     };
     this.hide = function() {
     this.hide = function() {
+      if ( collapse[isAnimating] ) return;    
       closeAction(collapse,element);
       closeAction(collapse,element);
       addClass(element,collapsed);
       addClass(element,collapsed);
     };
     };
     this.show = function() {
     this.show = function() {
       if ( accordion ) {
       if ( accordion ) {
-        var activeCollapse = queryElement('.'+component+'.'+showClass,accordion),
-            toggle = activeCollapse && (queryElement('['+dataToggle+'="'+component+'"]['+dataTarget+'="#'+activeCollapse.id+'"]',accordion)
-                   || queryElement('['+dataToggle+'="'+component+'"][href="#'+activeCollapse.id+'"]',accordion) ),
-            correspondingCollapse = toggle && (toggle[getAttribute](dataTarget) || toggle.href);
-        if ( activeCollapse && toggle && activeCollapse !== collapse ) {
-          closeAction(activeCollapse,toggle); 
-          if ( correspondingCollapse.split('#')[1] !== collapse.id ) { addClass(toggle,collapsed); } 
-          else { removeClass(toggle,collapsed); }
-        }
+        activeCollapse = queryElement('.'+component+'.'+showClass,accordion);
+        activeElement = activeCollapse && (queryElement('['+dataToggle+'="'+component+'"]['+dataTarget+'="#'+activeCollapse.id+'"]',accordion)
+                      || queryElement('['+dataToggle+'="'+component+'"][href="#'+activeCollapse.id+'"]',accordion) );
       }
       }
   
   
-      openAction(collapse,element);
-      removeClass(element,collapsed);
+      if ( !collapse[isAnimating] || activeCollapse && !activeCollapse[isAnimating] ) {
+        if ( activeElement && activeCollapse !== collapse ) {
+          closeAction(activeCollapse,activeElement); 
+          addClass(activeElement,collapsed);
+        }
+        openAction(collapse,element);
+        removeClass(element,collapsed);
+      }
     };
     };
   
   
     // init
     // init
@@ -5297,6 +5308,7 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
       on(element, clickEvent, self.toggle);
       on(element, clickEvent, self.toggle);
     }
     }
     collapse = getTarget();
     collapse = getTarget();
+    collapse[isAnimating] = false;  // when true it will prevent click handlers  
     accordion = queryElement(options.parent) || accordionData && getClosest(element, accordionData);
     accordion = queryElement(options.parent) || accordionData && getClosest(element, accordionData);
     element[stringCollapse] = self;
     element[stringCollapse] = self;
   };
   };
@@ -5454,6 +5466,7 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
     var btnCheck = element[getAttribute](dataTarget)||element[getAttribute]('href'),
     var btnCheck = element[getAttribute](dataTarget)||element[getAttribute]('href'),
       checkModal = queryElement( btnCheck ),
       checkModal = queryElement( btnCheck ),
       modal = hasClass(element,'modal') ? element : checkModal,
       modal = hasClass(element,'modal') ? element : checkModal,
+      overlayDelay,
   
   
       // strings
       // strings
       component = 'modal',
       component = 'modal',
@@ -5487,13 +5500,13 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
         return globalObject[innerWidth] || (htmlRect[right] - Math.abs(htmlRect[left]));
         return globalObject[innerWidth] || (htmlRect[right] - Math.abs(htmlRect[left]));
       },
       },
       setScrollbar = function () {
       setScrollbar = function () {
-        var bodyStyle = globalObject.getComputedStyle(DOC[body]),
+        var bodyStyle = globalObject[getComputedStyle](DOC[body]),
             bodyPad = parseInt((bodyStyle[paddingRight]), 10), itemPad;
             bodyPad = parseInt((bodyStyle[paddingRight]), 10), itemPad;
         if (bodyIsOverflowing) {
         if (bodyIsOverflowing) {
           DOC[body][style][paddingRight] = (bodyPad + scrollbarWidth) + 'px';
           DOC[body][style][paddingRight] = (bodyPad + scrollbarWidth) + 'px';
           if (fixedItems[length]){
           if (fixedItems[length]){
             for (var i = 0; i < fixedItems[length]; i++) {
             for (var i = 0; i < fixedItems[length]; i++) {
-              itemPad = globalObject.getComputedStyle(fixedItems[i])[paddingRight];
+              itemPad = globalObject[getComputedStyle](fixedItems[i])[paddingRight];
               fixedItems[i][style][paddingRight] = ( parseInt(itemPad) + scrollbarWidth) + 'px';
               fixedItems[i][style][paddingRight] = ( parseInt(itemPad) + scrollbarWidth) + 'px';
             }
             }
           }
           }
@@ -5635,6 +5648,7 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
   
   
       if ( overlay && modalOverlay && !hasClass(overlay,showClass)) {
       if ( overlay && modalOverlay && !hasClass(overlay,showClass)) {
         overlay[offsetWidth]; // force reflow to enable trasition
         overlay[offsetWidth]; // force reflow to enable trasition
+        overlayDelay = getTransitionDurationFromElement(overlay);              
         addClass(overlay, showClass);
         addClass(overlay, showClass);
       }
       }
   
   
@@ -5654,18 +5668,19 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
         keydownHandlerToggle();
         keydownHandlerToggle();
   
   
         hasClass(modal,'fade') ? emulateTransitionEnd(modal, triggerShow) : triggerShow();
         hasClass(modal,'fade') ? emulateTransitionEnd(modal, triggerShow) : triggerShow();
-      }, supportTransitions ? 150 : 0);
+      }, supportTransitions && overlay ? overlayDelay : 0);
     };
     };
     this.hide = function() {
     this.hide = function() {
       bootstrapCustomEvent.call(modal, hideEvent, component);
       bootstrapCustomEvent.call(modal, hideEvent, component);
       overlay = queryElement('.'+modalBackdropString);
       overlay = queryElement('.'+modalBackdropString);
+      overlayDelay = overlay && getTransitionDurationFromElement(overlay);    
   
   
       removeClass(modal,showClass);
       removeClass(modal,showClass);
       modal[setAttribute](ariaHidden, true);
       modal[setAttribute](ariaHidden, true);
   
   
-      (function(){
+      setTimeout(function(){
         hasClass(modal,'fade') ? emulateTransitionEnd(modal, triggerHide) : triggerHide();
         hasClass(modal,'fade') ? emulateTransitionEnd(modal, triggerHide) : triggerHide();
-      }());
+      }, supportTransitions && overlay ? overlayDelay : 0);
     };
     };
     this.setContent = function( content ) {
     this.setContent = function( content ) {
       queryElement('.'+component+'-content',modal)[innerHTML] = content;
       queryElement('.'+component+'-content',modal)[innerHTML] = content;
@@ -6021,7 +6036,7 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
               tabsContentContainer[style][height] = nextHeight + 'px'; // height animation
               tabsContentContainer[style][height] = nextHeight + 'px'; // height animation
               tabsContentContainer[offsetWidth];
               tabsContentContainer[offsetWidth];
               emulateTransitionEnd(tabsContentContainer, triggerEnd);
               emulateTransitionEnd(tabsContentContainer, triggerEnd);
-            },1);
+            },50);
           }
           }
         } else {
         } else {
           tabs[isAnimating] = false; 
           tabs[isAnimating] = false; 
@@ -6048,7 +6063,7 @@ backbone.nativeview = __webpack_require__(/*! backbone.nativeview */ "./node_mod
           tabsContentContainer[style][height] = containerHeight + 'px'; // height animation
           tabsContentContainer[style][height] = containerHeight + 'px'; // height animation
           tabsContentContainer[offsetHeight];
           tabsContentContainer[offsetHeight];
           activeContent[style][float] = '';
           activeContent[style][float] = '';
-          nextContent[style][float] = '';   
+          nextContent[style][float] = '';
         }
         }
   
   
         if ( hasClass(nextContent, 'fade') ) {
         if ( hasClass(nextContent, 'fade') ) {
@@ -62132,9 +62147,7 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
           return _converse.log(`Invalid JID "${jid}" provided in URL fragment`, Strophe.LogLevel.WARN);
           return _converse.log(`Invalid JID "${jid}" provided in URL fragment`, Strophe.LogLevel.WARN);
         }
         }
 
 
-        Promise.all([_converse.api.waitUntil('rosterContactsFetched'), _converse.api.waitUntil('chatBoxesFetched')]).then(() => {
-          _converse.api.chats.open(jid);
-        });
+        _converse.api.chats.open(jid);
       }
       }
 
 
       _converse.router.route('converse/chat?jid=:jid', openChat);
       _converse.router.route('converse/chat?jid=:jid', openChat);
@@ -62589,7 +62602,7 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
             attrs.from = stanza.getAttribute('from');
             attrs.from = stanza.getAttribute('from');
             attrs.nick = Strophe.unescapeNode(Strophe.getResourceFromJid(attrs.from));
             attrs.nick = Strophe.unescapeNode(Strophe.getResourceFromJid(attrs.from));
 
 
-            if (attrs.from === this.get('nick')) {
+            if (Strophe.getResourceFromJid(attrs.from) === this.get('nick')) {
               attrs.sender = 'me';
               attrs.sender = 'me';
             } else {
             } else {
               attrs.sender = 'them';
               attrs.sender = 'them';
@@ -63023,6 +63036,11 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
 
 
 
 
       _.extend(_converse.api, {
       _.extend(_converse.api, {
+        /**
+         * The "chats" grouping (used for one-on-one chats)
+         *
+         * @namespace
+         */
         'chats': {
         'chats': {
           'create'(jids, attrs) {
           'create'(jids, attrs) {
             if (_.isUndefined(jids)) {
             if (_.isUndefined(jids)) {
@@ -63053,21 +63071,78 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
             });
             });
           },
           },
 
 
+          /**
+           * Opens a new one-on-one chat.
+           *
+           * @function
+           *
+           * @param {String|string[]} name - e.g. 'buddy@example.com' or ['buddy1@example.com', 'buddy2@example.com']
+           * @returns {Promise} Promise which resolves with the Backbone.Model representing the chat.
+           *
+           * @example
+           * // To open a single chat, provide the JID of the contact you're chatting with in that chat:
+           * converse.plugins.add('myplugin', {
+           *     initialize: function() {
+           *         var _converse = this._converse;
+           *         // Note, buddy@example.org must be in your contacts roster!
+           *         _converse.api.chats.open('buddy@example.com').then((chat) => {
+           *             // Now you can do something with the chat model
+           *         });
+           *     }
+           * });
+           *
+           * @example
+           * // To open an array of chats, provide an array of JIDs:
+           * converse.plugins.add('myplugin', {
+           *     initialize: function () {
+           *         var _converse = this._converse;
+           *         // Note, these users must first be in your contacts roster!
+           *         _converse.api.chats.open(['buddy1@example.com', 'buddy2@example.com']).then((chats) => {
+           *             // Now you can do something with the chat models
+           *         });
+           *     }
+           * });
+           *
+           */
           'open'(jids, attrs) {
           'open'(jids, attrs) {
-            if (_.isUndefined(jids)) {
-              _converse.log("chats.open: You need to provide at least one JID", Strophe.LogLevel.ERROR);
+            return new Promise((resolve, reject) => {
+              Promise.all([_converse.api.waitUntil('rosterContactsFetched'), _converse.api.waitUntil('chatBoxesFetched')]).then(() => {
+                if (_.isUndefined(jids)) {
+                  const err_msg = "chats.open: You need to provide at least one JID";
 
 
-              return null;
-            } else if (_.isString(jids)) {
-              const chatbox = _converse.api.chats.create(jids, attrs);
+                  _converse.log(err_msg, Strophe.LogLevel.ERROR);
 
 
-              chatbox.trigger('show');
-              return chatbox;
-            }
-
-            return _.map(jids, jid => _converse.api.chats.create(jid, attrs).trigger('show'));
+                  reject(new Error(err_msg));
+                } else if (_.isString(jids)) {
+                  resolve(_converse.api.chats.create(jids, attrs).trigger('show'));
+                } else {
+                  resolve(_.map(jids, jid => _converse.api.chats.create(jid, attrs).trigger('show')));
+                }
+              }).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
+            });
           },
           },
 
 
+          /**
+           * Returns a chat model. The chat should already be open.
+           *
+           * @function
+           *
+           * @param {String|string[]} name - e.g. 'buddy@example.com' or ['buddy1@example.com', 'buddy2@example.com']
+           * @returns {Backbone.Model}
+           *
+           * @example
+           * // To return a single chat, provide the JID of the contact you're chatting with in that chat:
+           * const model = _converse.api.chats.get('buddy@example.com');
+           *
+           * @example
+           * // To return an array of chats, provide an array of JIDs:
+           * const models = _converse.api.chats.get(['buddy1@example.com', 'buddy2@example.com']);
+           *
+           * @example
+           * // To return all open chats, call the method without any parameters::
+           * const models = _converse.api.chats.get();
+           *
+           */
           'get'(jids) {
           'get'(jids) {
             if (_.isUndefined(jids)) {
             if (_.isUndefined(jids)) {
               const result = [];
               const result = [];
@@ -63975,6 +64050,11 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
           ev.preventDefault();
           ev.preventDefault();
           const textarea = this.el.querySelector('.chat-textarea'),
           const textarea = this.el.querySelector('.chat-textarea'),
                 message = textarea.value;
                 message = textarea.value;
+
+          if (!message.replace(/\s/g, '').length) {
+            return;
+          }
+
           let spoiler_hint;
           let spoiler_hint;
 
 
           if (this.model.get('composing_spoiler')) {
           if (this.model.get('composing_spoiler')) {
@@ -63989,12 +64069,9 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
           const event = document.createEvent('Event');
           const event = document.createEvent('Event');
           event.initEvent('input', true, true);
           event.initEvent('input', true, true);
           textarea.dispatchEvent(event);
           textarea.dispatchEvent(event);
+          this.onMessageSubmitted(message, spoiler_hint);
 
 
-          if (message !== '') {
-            this.onMessageSubmitted(message, spoiler_hint);
-
-            _converse.emit('messageSend', message);
-          }
+          _converse.emit('messageSend', message);
 
 
           this.setChatState(_converse.ACTIVE);
           this.setChatState(_converse.ACTIVE);
         },
         },
@@ -65155,7 +65232,11 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
   _.extend(_converse, Backbone.Events); // Core plugins are whitelisted automatically
   _.extend(_converse, Backbone.Events); // Core plugins are whitelisted automatically
 
 
 
 
-  _converse.core_plugins = ['converse-bookmarks', 'converse-caps', 'converse-chatboxes', 'converse-chatview', 'converse-controlbox', 'converse-core', 'converse-disco', 'converse-dragresize', 'converse-embedded', 'converse-fullscreen', 'converse-headline', 'converse-mam', 'converse-message-view', 'converse-minimize', 'converse-modal', 'converse-muc', 'converse-muc-views', 'converse-notification', 'converse-omemo', 'converse-oauth', 'converse-ping', 'converse-profile', 'converse-push', 'converse-register', 'converse-roomslist', 'converse-roster', 'converse-rosterview', 'converse-singleton', 'converse-spoilers', 'converse-vcard']; // Make converse pluggable
+  _converse.core_plugins = ['converse-bookmarks', 'converse-caps', 'converse-chatboxes', 'converse-chatview', 'converse-controlbox', 'converse-core', 'converse-disco', 'converse-dragresize', 'converse-embedded', 'converse-fullscreen', 'converse-headline', 'converse-mam', 'converse-message-view', 'converse-minimize', 'converse-modal', 'converse-muc', 'converse-muc-views', 'converse-notification', 'converse-omemo', 'converse-oauth', 'converse-ping', 'converse-profile', 'converse-push', 'converse-register', 'converse-roomslist', 'converse-roster', 'converse-rosterview', 'converse-singleton', 'converse-spoilers', 'converse-vcard']; // Setting wait to 59 instead of 60 to avoid timing conflicts with the
+  // webserver, which is often also set to 60 and might therefore sometimes
+  // return a 504 error page instead of passing through to the BOSH proxy.
+
+  const BOSH_WAIT = 59; // Make converse pluggable
 
 
   pluggable.enable(_converse, '_converse', 'pluggable'); // Module-level constants
   pluggable.enable(_converse, '_converse', 'pluggable'); // Module-level constants
 
 
@@ -66159,7 +66240,7 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
           this.connection.reset();
           this.connection.reset();
         }
         }
 
 
-        this.connection.connect(this.jid.toLowerCase(), null, this.onConnectStatusChanged);
+        this.connection.connect(this.jid.toLowerCase(), null, this.onConnectStatusChanged, BOSH_WAIT);
       } else if (this.authentication === _converse.LOGIN) {
       } else if (this.authentication === _converse.LOGIN) {
         const password = _.isNil(credentials) ? _converse.connection.pass || this.password : credentials.password;
         const password = _.isNil(credentials) ? _converse.connection.pass || this.password : credentials.password;
 
 
@@ -66187,7 +66268,7 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
           this.connection.reset();
           this.connection.reset();
         }
         }
 
 
-        this.connection.connect(this.jid, password, this.onConnectStatusChanged);
+        this.connection.connect(this.jid, password, this.onConnectStatusChanged, BOSH_WAIT);
       }
       }
     };
     };
 
 
@@ -66850,6 +66931,8 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
           });
           });
         }
         }
 
 
+        iqresult.c('query', attrs);
+
         _.each(plugin._identities, identity => {
         _.each(plugin._identities, identity => {
           const attrs = {
           const attrs = {
             'category': identity.category,
             'category': identity.category,
@@ -67605,8 +67688,8 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
         throw new Error("converse-embedded: auto_join_rooms must be an Array");
         throw new Error("converse-embedded: auto_join_rooms must be an Array");
       }
       }
 
 
-      if (_converse.auto_join_rooms.length !== 1 && _converse.auto_join_private_chats.length !== 1) {
-        throw new Error("converse-embedded: It doesn't make " + "sense to have the auto_join_rooms setting to zero or " + "more then one, since only one chat room can be open " + "at any time.");
+      if (_converse.auto_join_rooms.length > 1 && _converse.auto_join_private_chats.length > 1) {
+        throw new Error("converse-embedded: It doesn't make " + "sense to have the auto_join_rooms setting more then one, " + "since only one chat room can be open at any time.");
       }
       }
     }
     }
 
 
@@ -69911,17 +69994,7 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
           this.model.on('show', this.show, this);
           this.model.on('show', this.show, this);
           this.model.occupants.on('add', this.showJoinNotification, this);
           this.model.occupants.on('add', this.showJoinNotification, this);
           this.model.occupants.on('remove', this.showLeaveNotification, this);
           this.model.occupants.on('remove', this.showLeaveNotification, this);
-          this.model.occupants.on('change:show', occupant => {
-            if (!occupant.isMember() || _.includes(occupant.get('states'), '303')) {
-              return;
-            }
-
-            if (occupant.get('show') === 'offline') {
-              this.showLeaveNotification(occupant);
-            } else if (occupant.get('show') === 'online') {
-              this.showJoinNotification(occupant);
-            }
-          });
+          this.model.occupants.on('change:show', this.showJoinOrLeaveNotification, this);
           this.createEmojiPicker();
           this.createEmojiPicker();
           this.createOccupantsView();
           this.createOccupantsView();
           this.render().insertIntoDOM();
           this.render().insertIntoDOM();
@@ -69931,8 +70004,7 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
             const handler = () => {
             const handler = () => {
               if (!u.isPersistableModel(this.model)) {
               if (!u.isPersistableModel(this.model)) {
                 // Happens during tests, nothing to do if this
                 // Happens during tests, nothing to do if this
-                // is a hanging chatbox (i.e. not in the
-                // collection anymore).
+                // is a hanging chatbox (i.e. not in the collection anymore).
                 return;
                 return;
               }
               }
 
 
@@ -70809,6 +70881,18 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
           }
           }
         },
         },
 
 
+        showJoinOrLeaveNotification(occupant) {
+          if (!occupant.isMember() || _.includes(occupant.get('states'), '303')) {
+            return;
+          }
+
+          if (occupant.get('show') === 'offline') {
+            this.showLeaveNotification(occupant);
+          } else if (occupant.get('show') === 'online') {
+            this.showJoinNotification(occupant);
+          }
+        },
+
         showJoinNotification(occupant) {
         showJoinNotification(occupant) {
           if (this.model.get('connection_status') !== converse.ROOMSTATUS.ENTERED) {
           if (this.model.get('connection_status') !== converse.ROOMSTATUS.ENTERED) {
             return;
             return;
@@ -70856,10 +70940,9 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
         showLeaveNotification(occupant) {
         showLeaveNotification(occupant) {
           const nick = occupant.get('nick'),
           const nick = occupant.get('nick'),
                 stat = occupant.get('status'),
                 stat = occupant.get('status'),
-                last_el = this.content.lastElementChild,
-                last_msg_date = last_el.getAttribute('data-isodate');
+                last_el = this.content.lastElementChild;
 
 
-          if (_.includes(_.get(last_el, 'classList', []), 'chat-info') && moment(last_msg_date).isSame(new Date(), "day") && _.get(last_el, 'dataset', {}).join === `"${nick}"`) {
+          if (last_el && _.includes(_.get(last_el, 'classList', []), 'chat-info') && moment(last_el.getAttribute('data-isodate')).isSame(new Date(), "day") && _.get(last_el, 'dataset', {}).join === `"${nick}"`) {
             let message;
             let message;
 
 
             if (_.isNil(stat)) {
             if (_.isNil(stat)) {
@@ -70890,7 +70973,7 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
               'data': `data-leave="${nick}"`
               'data': `data-leave="${nick}"`
             };
             };
 
 
-            if (_.includes(_.get(last_el, 'classList', []), 'chat-info') && _.get(last_el, 'dataset', {}).leavejoin === `"${nick}"`) {
+            if (last_el && _.includes(_.get(last_el, 'classList', []), 'chat-info') && _.get(last_el, 'dataset', {}).leavejoin === `"${nick}"`) {
               last_el.outerHTML = tpl_info(data);
               last_el.outerHTML = tpl_info(data);
             } else {
             } else {
               const el = u.stringToElement(tpl_info(data));
               const el = u.stringToElement(tpl_info(data));
@@ -72789,13 +72872,21 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
           },
           },
 
 
           'open'(jids, attrs) {
           'open'(jids, attrs) {
-            if (_.isUndefined(jids)) {
-              throw new TypeError('rooms.open: You need to provide at least one JID');
-            } else if (_.isString(jids)) {
-              return _converse.api.rooms.create(jids, attrs).trigger('show');
-            }
+            return new Promise((resolve, reject) => {
+              _converse.api.waitUntil('chatBoxesFetched').then(() => {
+                if (_.isUndefined(jids)) {
+                  const err_msg = 'rooms.open: You need to provide at least one JID';
+
+                  _converse.log(err_msg, Strophe.LogLevel.ERROR);
 
 
-            return _.map(jids, jid => _converse.api.rooms.create(jid, attrs).trigger('show'));
+                  reject(new TypeError(err_msg));
+                } else if (_.isString(jids)) {
+                  resolve(_converse.api.rooms.create(jids, attrs).trigger('show'));
+                } else {
+                  resolve(_.map(jids, jid => _converse.api.rooms.create(jid, attrs).trigger('show')));
+                }
+              });
+            });
           },
           },
 
 
           'get'(jids, attrs, create) {
           'get'(jids, attrs, create) {
@@ -73262,7 +73353,12 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
         getBundlesAndBuildSessions() {
         getBundlesAndBuildSessions() {
           const _converse = this.__super__._converse;
           const _converse = this.__super__._converse;
           return new Promise((resolve, reject) => {
           return new Promise((resolve, reject) => {
-            _converse.getDevicesForContact(this.get('jid')).then(devices => {
+            _converse.getDevicesForContact(this.get('jid')).then(their_devices => {
+              const device_id = _converse.omemo_store.get('device_id'),
+                    devicelist = _converse.devicelists.get(_converse.bare_jid),
+                    own_devices = devicelist.devices.filter(device => device.get('id') !== device_id),
+                    devices = _.concat(own_devices, their_devices.models);
+
               Promise.all(devices.map(device => device.getBundle())).then(() => this.buildSessions(devices)).then(() => resolve(devices)).catch(_.partial(_converse.log, _, Strophe.LogLevel.ERROR));
               Promise.all(devices.map(device => device.getBundle())).then(() => this.buildSessions(devices)).then(() => resolve(devices)).catch(_.partial(_converse.log, _, Strophe.LogLevel.ERROR));
             }).catch(_.partial(_converse.log, _, Strophe.LogLevel.ERROR));
             }).catch(_.partial(_converse.log, _, Strophe.LogLevel.ERROR));
           });
           });
@@ -73329,32 +73425,34 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
           const _converse = this.__super__._converse,
           const _converse = this.__super__._converse,
                 address = new libsignal.SignalProtocolAddress(this.get('jid'), device.get('id')),
                 address = new libsignal.SignalProtocolAddress(this.get('jid'), device.get('id')),
                 sessionCipher = new window.libsignal.SessionCipher(_converse.omemo_store, address);
                 sessionCipher = new window.libsignal.SessionCipher(_converse.omemo_store, address);
-          return sessionCipher.encrypt(plaintext);
+          return new Promise((resolve, reject) => {
+            sessionCipher.encrypt(plaintext).then(payload => resolve({
+              'payload': payload,
+              'device': device
+            })).catch(_.partial(_converse.log, _, Strophe.LogLevel.ERROR));
+          });
         },
         },
 
 
-        addKeysToMessageStanza(stanza, devices, payloads) {
-          for (var i in payloads) {
-            if (Object.prototype.hasOwnProperty.call(payloads, i)) {
-              const payload = btoa(JSON.stringify(payloads[i]));
-              const prekey = 3 == parseInt(payloads[i].type, 10);
+        addKeysToMessageStanza(stanza, dicts, iv) {
+          for (var i in dicts) {
+            if (Object.prototype.hasOwnProperty.call(dicts, i)) {
+              const payload = dicts[i].payload,
+                    device = dicts[i].device,
+                    prekey = 3 == parseInt(payload.type, 10);
+              stanza.c('key', {
+                'rid': device.get('id')
+              }).t(btoa(JSON.stringify(dicts[i].payload)));
 
 
-              if (i == payloads.length - 1) {
-                stanza.c('key', {
-                  'rid': devices.get('id')
-                }).t(payload);
+              if (prekey) {
+                stanza.attrs({
+                  'prekey': prekey
+                });
+              }
 
 
-                if (prekey) {
-                  stanza.attrs({
-                    'prekey': prekey
-                  });
-                }
+              stanza.up();
 
 
-                stanza.up().c('iv').t(payloads[0].iv).up().up();
-              } else {
-                stanza.c('key', {
-                  prekey: prekey,
-                  rid: devices.get('id')
-                }).t(payload).up();
+              if (i == dicts.length - 1) {
+                stanza.c('iv').t(iv).up().up();
               }
               }
             }
             }
           }
           }
@@ -73389,8 +73487,8 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
             // concatenation is encrypted using the corresponding
             // concatenation is encrypted using the corresponding
             // long-standing SignalProtocol session.
             // long-standing SignalProtocol session.
             // TODO: need to include own devices here as well (and filter out distrusted devices)
             // TODO: need to include own devices here as well (and filter out distrusted devices)
-            const promises = devices.map(device => this.encryptKey(payload.key_str + payload.tag, device));
-            return Promise.all(promises).then(payloads => this.addKeysToMessageStanza(stanza, devices, payloads));
+            const promises = devices.filter(device => device.get('trusted') != UNTRUSTED).map(device => this.encryptKey(payload.key_str + payload.tag, device));
+            return Promise.all(promises).then(dicts => this.addKeysToMessageStanza(stanza, dicts, payload.iv)).catch(_.partial(_converse.log, _, Strophe.LogLevel.ERROR));
           });
           });
         },
         },
 
 
@@ -78295,8 +78393,13 @@ var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_
 
 
         xhr.onload = function () {
         xhr.onload = function () {
           if (xhr.status >= 200 && xhr.status < 400) {
           if (xhr.status >= 200 && xhr.status < 400) {
-            jed_instance = new Jed(window.JSON.parse(xhr.responseText));
-            resolve();
+            try {
+              const data = window.JSON.parse(xhr.responseText);
+              jed_instance = new Jed(data);
+              resolve();
+            } catch (e) {
+              xhr.onerror(e);
+            }
           } else {
           } else {
             xhr.onerror();
             xhr.onerror();
           }
           }

+ 8 - 8
docs/source/conf.py

@@ -1,6 +1,6 @@
 # -*- coding: utf-8 -*-
 # -*- coding: utf-8 -*-
 #
 #
-# Converse.js documentation build configuration file, created by
+# Converse documentation build configuration file, created by
 # sphinx-quickstart on Fri Apr 26 20:48:03 2013.
 # sphinx-quickstart on Fri Apr 26 20:48:03 2013.
 #
 #
 # This file is execfile()d with the current directory set to its containing dir.
 # This file is execfile()d with the current directory set to its containing dir.
@@ -40,8 +40,8 @@ source_suffix = '.rst'
 master_doc = 'index'
 master_doc = 'index'
 
 
 # General information about the project.
 # General information about the project.
-project = u'Converse.js'
-copyright = u'2017, JC Brand'
+project = u'Converse'
+copyright = u'2018, JC Brand'
 
 
 # The version info for the project you're documenting, acts as replacement for
 # The version info for the project you're documenting, acts as replacement for
 # |version| and |release|, also used in various other places throughout the
 # |version| and |release|, also used in various other places throughout the
@@ -108,7 +108,7 @@ html_logo = "_static/conversejs_small.png"
 # theme further.
 # theme further.
 html_theme_options = {
 html_theme_options = {
     # Navigation bar title. (Default: ``project`` value)
     # Navigation bar title. (Default: ``project`` value)
-    'navbar_title': "Converse.js",
+    'navbar_title': "Converse",
     # Tab name for entire site. (Default: "Site")
     # Tab name for entire site. (Default: "Site")
     'navbar_site_name': "Table of Contents",
     'navbar_site_name': "Table of Contents",
     # A list of tuples containing pages or urls to link to.
     # A list of tuples containing pages or urls to link to.
@@ -229,7 +229,7 @@ latex_elements = {
 # Grouping the document tree into LaTeX files. List of tuples
 # Grouping the document tree into LaTeX files. List of tuples
 # (source start file, target name, title, author, documentclass [howto/manual]).
 # (source start file, target name, title, author, documentclass [howto/manual]).
 latex_documents = [
 latex_documents = [
-  ('index', 'Conversejs.tex', u'Converse.js Documentation',
+  ('index', 'Conversejs.tex', u'Converse Documentation',
    u'JC Brand', 'manual'),
    u'JC Brand', 'manual'),
 ]
 ]
 
 
@@ -259,7 +259,7 @@ latex_documents = [
 # One entry per manual page. List of tuples
 # One entry per manual page. List of tuples
 # (source start file, name, description, authors, manual section).
 # (source start file, name, description, authors, manual section).
 man_pages = [
 man_pages = [
-    ('index', 'conversejs', u'Converse.js Documentation',
+    ('index', 'conversejs', u'Converse Documentation',
      [u'JC Brand'], 1)
      [u'JC Brand'], 1)
 ]
 ]
 
 
@@ -273,8 +273,8 @@ man_pages = [
 # (source start file, target name, title, author,
 # (source start file, target name, title, author,
 #  dir menu entry, description, category)
 #  dir menu entry, description, category)
 texinfo_documents = [
 texinfo_documents = [
-  ('index', 'Conversejs', u'Converse.js Documentation',
-   u'JC Brand', 'Conversejs', 'Open Source XMPP webchat',
+  ('index', 'Conversejs', u'Converse Documentation',
+   u'JC Brand', 'Converse', 'Open Source XMPP webchat',
    'Miscellaneous'),
    'Miscellaneous'),
 ]
 ]
 
 

+ 46 - 91
docs/source/developer_api.rst

@@ -2,12 +2,12 @@
 
 
     <div id="banner"><a href="https://github.com/jcbrand/converse.js/blob/master/docs/source/theming.rst">Edit me on GitHub</a></div>
     <div id="banner"><a href="https://github.com/jcbrand/converse.js/blob/master/docs/source/theming.rst">Edit me on GitHub</a></div>
 
 
-=============================
-The converse.js developer API
-=============================
+=========================
+The old API documentation
+=========================
 
 
-.. note:: The API documented here is available in Converse.js 0.8.4 and higher.
-        Earlier versions of Converse.js might have different API methods or none at all.
+.. note:: The API documented here is available in Converse 0.8.4 and higher.
+        Earlier versions of Converse might have different API methods or none at all.
 
 
 .. note:: From version 3.0.0 and onwards many API methods have been made
 .. note:: From version 3.0.0 and onwards many API methods have been made
         private and available to plugins only. This means that if you want to
         private and available to plugins only. This means that if you want to
@@ -15,7 +15,7 @@ The converse.js developer API
         access it. This change is done to avoid leakage of sensitive data to
         access it. This change is done to avoid leakage of sensitive data to
         malicious or non-whitelisted scripts.
         malicious or non-whitelisted scripts.
 
 
-The Converse.js API is broken up into different logical "groupings" (for
+The Converse API is broken up into different logical "groupings" (for
 example ``converse.plugins`` or ``converse.contacts``).
 example ``converse.plugins`` or ``converse.contacts``).
 
 
 There are some exceptions to this, like ``converse.initialize``, which aren't
 There are some exceptions to this, like ``converse.initialize``, which aren't
@@ -58,8 +58,8 @@ initialize
 
 
 .. note:: This method is the one exception of a method which is not logically grouped as explained above.
 .. note:: This method is the one exception of a method which is not logically grouped as explained above.
 
 
-Publich API method which initializes converse.js.
-This method must always be called when using converse.js.
+Publich API method which initializes Converse.
+This method must always be called when using Converse.
 
 
 The `initialize` method takes a map of :ref:`configuration-settings`.
 The `initialize` method takes a map of :ref:`configuration-settings`.
 
 
@@ -105,7 +105,7 @@ Registers a new plugin.
 
 
             // Inside this method, you have access to the closured
             // Inside this method, you have access to the closured
             // _converse object, which contains the core logic and data
             // _converse object, which contains the core logic and data
-            // structures of converse.js
+            // structures of Converse
         }
         }
     }
     }
     converse.plugins.add('myplugin', plugin);
     converse.plugins.add('myplugin', plugin);
@@ -182,7 +182,7 @@ two important ways:
 * A handler registered for a promise, will still fire *after* the promise has
 * A handler registered for a promise, will still fire *after* the promise has
   been resolved, which is not the case with an event handler.
   been resolved, which is not the case with an event handler.
 
 
-Converse.js has the following promises:
+Converse has the following promises:
 
 
 * :ref:`cachedRoster`
 * :ref:`cachedRoster`
 * :ref:`chatBoxesFetched`
 * :ref:`chatBoxesFetched`
@@ -210,7 +210,7 @@ already by that time.
 The **archive** grouping
 The **archive** grouping
 ------------------------
 ------------------------
 
 
-Converse.js supports the *Message Archive Management*
+Converse supports the *Message Archive Management*
 (`XEP-0313 <https://xmpp.org/extensions/xep-0313.html>`_) protocol,
 (`XEP-0313 <https://xmpp.org/extensions/xep-0313.html>`_) protocol,
 through which it is able to query an XMPP server for archived messages.
 through which it is able to query an XMPP server for archived messages.
 
 
@@ -263,12 +263,12 @@ the returned messages.
 
 
 **Waiting until server support has been determined**
 **Waiting until server support has been determined**
 
 
-The query method will only work if converse.js has been able to determine that
+The query method will only work if Converse has been able to determine that
 the server supports MAM queries, otherwise the following error will be raised:
 the server supports MAM queries, otherwise the following error will be raised:
 
 
 - *This server does not support XEP-0313, Message Archive Management*
 - *This server does not support XEP-0313, Message Archive Management*
 
 
-The very first time converse.js loads in a browser tab, if you call the query
+The very first time Converse loads in a browser tab, if you call the query
 API too quickly, the above error might appear because service discovery has not
 API too quickly, the above error might appear because service discovery has not
 yet been completed.
 yet been completed.
 
 
@@ -453,7 +453,7 @@ Paramters:
 get
 get
 ***
 ***
 
 
-Returns all of the identities registered for this client (i.e. instance of Converse.js).
+Returns all of the identities registered for this client (i.e. instance of Converse).
 
 
 .. code-block:: javascript
 .. code-block:: javascript
 
 
@@ -473,17 +473,17 @@ Paramters:
 * (String) name
 * (String) name
 * (String) lang
 * (String) lang
 
 
-Lets you add new identities for this client (i.e. instance of Converse.js).
+Lets you add new identities for this client (i.e. instance of Converse).
 
 
 .. code-block:: javascript
 .. code-block:: javascript
 
 
-    _converse.api.disco.own.identities.add('client', 'web', 'Converse.js');
+    _converse.api.disco.own.identities.add('client', 'web', 'Converse');
 
 
 
 
 get
 get
 ***
 ***
 
 
-Returns all of the identities registered for this client (i.e. instance of Converse.js).
+Returns all of the identities registered for this client (i.e. instance of Converse).
 
 
 .. code-block:: javascript
 .. code-block:: javascript
 
 
@@ -602,7 +602,7 @@ Logs the user in. This method can accept a map with the credentials, like this:
         }
         }
     });
     });
 
 
-or it can be called without any parameters, in which case converse.js will try
+or it can be called without any parameters, in which case Converse will try
 to log the user in by calling the `prebind_url` or `credentials_url` depending
 to log the user in by calling the `prebind_url` or `credentials_url` depending
 on whether prebinding is used or not.
 on whether prebinding is used or not.
 
 
@@ -816,112 +816,67 @@ Note, for MUC chatrooms, you need to use the "rooms" grouping instead.
 get
 get
 ~~~
 ~~~
 
 
-Returns an object representing a chatbox. The chatbox should already be open.
+Returns an object representing a chat. The chat should already be open.
 
 
-To return a single chatbox, provide the JID of the contact you're chatting
-with in that chatbox:
+To return a single chat, provide the JID of the contact you're chatting
+with in that chat:
 
 
 .. code-block:: javascript
 .. code-block:: javascript
 
 
     _converse.api.chats.get('buddy@example.com')
     _converse.api.chats.get('buddy@example.com')
 
 
-To return an array of chatboxes, provide an array of JIDs:
+To return an array of chats, provide an array of JIDs:
 
 
 .. code-block:: javascript
 .. code-block:: javascript
 
 
     _converse.api.chats.get(['buddy1@example.com', 'buddy2@example.com'])
     _converse.api.chats.get(['buddy1@example.com', 'buddy2@example.com'])
 
 
-To return all open chatboxes, call the method without any JIDs::
+To return all open chats, call the method without any JIDs::
 
 
     _converse.api.chats.get()
     _converse.api.chats.get()
 
 
 open
 open
 ~~~~
 ~~~~
 
 
-Opens a chatbox and returns a `Backbone.View <http://backbonejs.org/#View>`_ object
-representing a chatbox.
+Opens a new chat.
+
+It returns an promise which will resolve with a `Backbone.Model <https://backbonejs.org/#Model>`_ representing the chat.
 
 
 Note that converse doesn't allow opening chats with users who aren't in your roster
 Note that converse doesn't allow opening chats with users who aren't in your roster
 (unless you have set :ref:`allow_non_roster_messaging` to ``true``).
 (unless you have set :ref:`allow_non_roster_messaging` to ``true``).
 
 
-Before opening a chat, you should first wait until the roster has been populated.
-This is the :ref:`rosterContactsFetched` event/promise.
-
-Besides that, it's a good idea to also first wait until already opened chatboxes
-(which are cached in sessionStorage) have also been fetched from the cache.
-This is the :ref:`chatBoxesFetched` event/promise.
-
 These two events fire only once per session, so they're also available as promises.
 These two events fire only once per session, so they're also available as promises.
 
 
-So, to open a single chatbox:
+So, to open a single chat:
 
 
 .. code-block:: javascript
 .. code-block:: javascript
 
 
     converse.plugins.add('myplugin', {
     converse.plugins.add('myplugin', {
-      initialize: function() {
-        var _converse = this._converse;
-        Promise.all([
-            _converse.api.waitUntil('rosterContactsFetched'),
-            _converse.api.waitUntil('chatBoxesFetched')
-        ]).then(function() {
+        initialize: function() {
+            var _converse = this._converse;
+
             // Note, buddy@example.org must be in your contacts roster!
             // Note, buddy@example.org must be in your contacts roster!
-            _converse.api.chats.open('buddy@example.com')
-        });
-      }
+            _converse.api.chats.open('buddy@example.com').then((chat) => {
+                // Now you can do something with the chat model
+            });
+        }
     });
     });
 
 
-To return an array of chatboxes, provide an array of JIDs:
+To return an array of chats, provide an array of JIDs:
 
 
 .. code-block:: javascript
 .. code-block:: javascript
 
 
     converse.plugins.add('myplugin', {
     converse.plugins.add('myplugin', {
         initialize: function () {
         initialize: function () {
             var _converse = this._converse;
             var _converse = this._converse;
-            Promise.all([
-                _converse.api.waitUntil('rosterContactsFetched'),
-                _converse.api.waitUntil('chatBoxesFetched')
-            ]).then(function() {
-                // Note, these users must first be in your contacts roster!
-                _converse.api.chats.open(['buddy1@example.com', 'buddy2@example.com'])
+            // Note, these users must first be in your contacts roster!
+            _converse.api.chats.open(['buddy1@example.com', 'buddy2@example.com']).then((chats) => {
+                // Now you can do something with the chat models
             });
             });
         }
         }
     });
     });
 
 
 
 
-*The returned chatbox object contains the following methods:*
-
-+-------------------+------------------------------------------+
-| Method            | Description                              |
-+===================+==========================================+
-| close             | Close the chatbox.                       |
-+-------------------+------------------------------------------+
-| focus             | Focuses the chatbox textarea             |
-+-------------------+------------------------------------------+
-| model.endOTR      | End an OTR (Off-the-record) session.     |
-+-------------------+------------------------------------------+
-| model.get         | Get an attribute (i.e. accessor).        |
-+-------------------+------------------------------------------+
-| model.initiateOTR | Start an OTR (off-the-record) session.   |
-+-------------------+------------------------------------------+
-| model.maximize    | Minimize the chatbox.                    |
-+-------------------+------------------------------------------+
-| model.minimize    | Maximize the chatbox.                    |
-+-------------------+------------------------------------------+
-| model.set         | Set an attribute (i.e. mutator).         |
-+-------------------+------------------------------------------+
-| show              | Opens/shows the chatbox.                 |
-+-------------------+------------------------------------------+
-
-*The get and set methods can be used to retrieve and change the following attributes:*
-
-+-------------+-----------------------------------------------------+
-| Attribute   | Description                                         |
-+=============+=====================================================+
-| height      | The height of the chatbox.                          |
-+-------------+-----------------------------------------------------+
-| url         | The URL of the chatbox heading.                     |
-+-------------+-----------------------------------------------------+
-
 The **chatviews** grouping
 The **chatviews** grouping
 --------------------------
 --------------------------
 
 
@@ -952,7 +907,7 @@ To return an array of views, provide an array of JIDs:
 The **listen** grouping
 The **listen** grouping
 -----------------------
 -----------------------
 
 
-Converse.js emits events to which you can subscribe from your own JavaScript.
+Converse emits events to which you can subscribe from your own JavaScript.
 
 
 Concerning events, the following methods are available under the "listen"
 Concerning events, the following methods are available under the "listen"
 grouping:
 grouping:
@@ -1014,7 +969,7 @@ The **rooms** grouping
 get
 get
 ~~~
 ~~~
 
 
-Returns an object representing a multi user chatbox (room).
+Returns an object representing a multi user chat (room).
 It takes 3 parameters:
 It takes 3 parameters:
 
 
 * the room JID (if not specified, all rooms will be returned).
 * the room JID (if not specified, all rooms will be returned).
@@ -1046,7 +1001,7 @@ It takes 3 parameters:
 open
 open
 ~~~~
 ~~~~
 
 
-Opens a multi user chatbox and returns an object representing it.
+Opens a multi user chat and returns an object representing it.
 Similar to the ``chats.get`` API.
 Similar to the ``chats.get`` API.
 
 
 It takes 2 parameters:
 It takes 2 parameters:
@@ -1055,7 +1010,7 @@ It takes 2 parameters:
 * A map (object) containing any extra room attributes. For example, if you want
 * A map (object) containing any extra room attributes. For example, if you want
   to specify the nickname, use ``{'nick': 'bloodninja'}``.
   to specify the nickname, use ``{'nick': 'bloodninja'}``.
 
 
-To open a single multi user chatbox, provide the JID of the room:
+To open a single multi user chat, provide the JID of the room:
 
 
 .. code-block:: javascript
 .. code-block:: javascript
 
 
@@ -1150,7 +1105,7 @@ JIDs.
 The **promises** grouping
 The **promises** grouping
 -------------------------
 -------------------------
 
 
-Converse.js and its plugins emit various events which you can listen to via the
+Converse and its plugins emit various events which you can listen to via the
 :ref:`listen-grouping`.
 :ref:`listen-grouping`.
 
 
 Some of these events are also available as `ES2015 Promises <http://es6-features.org/#PromiseUsage>`_,
 Some of these events are also available as `ES2015 Promises <http://es6-features.org/#PromiseUsage>`_,
@@ -1209,7 +1164,7 @@ For example:
 The **settings** grouping
 The **settings** grouping
 -------------------------
 -------------------------
 
 
-This grouping allows access to the configuration settings of converse.js.
+This grouping allows access to the configuration settings of Converse.
 
 
 .. _`settings-update`:
 .. _`settings-update`:
 
 
@@ -1290,7 +1245,7 @@ or :
     });
     });
 
 
 Note, this is not an alternative to calling ``converse.initialize``, which still needs
 Note, this is not an alternative to calling ``converse.initialize``, which still needs
-to be called. Generally, you'd use this method after converse.js is already
+to be called. Generally, you'd use this method after Converse is already
 running and you want to change the configuration on-the-fly.
 running and you want to change the configuration on-the-fly.
 
 
 The **tokens** grouping
 The **tokens** grouping
@@ -1328,7 +1283,7 @@ Parameters:
 
 
 Returns a Promise which results with the VCard data for a particular JID or for
 Returns a Promise which results with the VCard data for a particular JID or for
 a `Backbone.Model` instance which represents an entity with a JID (such as a roster contact,
 a `Backbone.Model` instance which represents an entity with a JID (such as a roster contact,
-chatbox or chatroom occupant).
+chat or chatroom occupant).
 
 
 If a `Backbone.Model` instance is passed in, then it must have either a `jid`
 If a `Backbone.Model` instance is passed in, then it must have either a `jid`
 attribute or a `muc_jid` attribute.
 attribute or a `muc_jid` attribute.

+ 5 - 4
docs/source/development.rst

@@ -8,13 +8,13 @@
 Development
 Development
 ===========
 ===========
 
 
-Welcome to the developer documentation of converse.js. Read the documentation
+Welcome to the developer documentation of Converse. Read the documentation
 linked to below, if you want to add new features or create your own customized
 linked to below, if you want to add new features or create your own customized
-version of converse.js.
+version of Converse.
 
 
-Converse.js itself composed of plugins, and exposes an API with which you can
+Converse itself composed of plugins, and exposes an API with which you can
 create and register your own plugins. This is the recommended way to customize
 create and register your own plugins. This is the recommended way to customize
-or add new functionality to converse.js.
+or add new functionality to Converse.
 
 
 .. toctree::
 .. toctree::
    :maxdepth: 2
    :maxdepth: 2
@@ -22,6 +22,7 @@ or add new functionality to converse.js.
    developer_guidelines
    developer_guidelines
    style_guide
    style_guide
    plugin_development
    plugin_development
+   api/index
    developer_api
    developer_api
    events 
    events 
    other_frameworks
    other_frameworks

+ 47 - 21
docs/source/setup.rst

@@ -8,19 +8,19 @@
 Setup and integration
 Setup and integration
 =====================
 =====================
 
 
-This page documents what you'll need to do to be able to connect Converse.js with
+This page documents what you'll need to do to be able to connect Converse with
 your own XMPP server and to better integrate it into your website.
 your own XMPP server and to better integrate it into your website.
 
 
-At the very least you'll need Converse.js and an :ref:`XMPP server` with
+At the very least you'll need Converse and an :ref:`XMPP server` with
 :ref:`websocket-section` or :ref:`BOSH-section` enabled. That's definitely
 :ref:`websocket-section` or :ref:`BOSH-section` enabled. That's definitely
-enough to simply demo Converse.js or to do development work on it.
+enough to simply demo Converse or to do development work on it.
 
 
-However, if you want to more fully integrate it into a website or intranet,
+However, if you want to more fully integrate it into a website
 then you'll likely need to set up more services and components.
 then you'll likely need to set up more services and components.
 
 
 The diagram below shows a fairly common setup for a website or intranet:
 The diagram below shows a fairly common setup for a website or intranet:
 
 
-* Converse.js runs in the web-browser on the user's device.
+* Converse runs in the web-browser on the user's device.
 
 
 * It communicates with the XMPP server via BOSH or websocket which is usually
 * It communicates with the XMPP server via BOSH or websocket which is usually
   reverse-proxied by a web-server in order to overcome cross-site scripting
   reverse-proxied by a web-server in order to overcome cross-site scripting
@@ -34,13 +34,12 @@ The diagram below shows a fairly common setup for a website or intranet:
   the XMPP server is configured to use, so that users can log in with those
   the XMPP server is configured to use, so that users can log in with those
   credentials.
   credentials.
 
 
-
 * Usually (but optionally) there is a backend web application which hosts a
 * Usually (but optionally) there is a backend web application which hosts a
-  website in which Converse.js appears.
+  website in which Converse appears.
 
 
 .. figure:: images/diagram.png
 .. figure:: images/diagram.png
    :align: center
    :align: center
-   :alt: A diagram of a possible setup, consisting of Converse.js, a web server, a backend web application, an XMPP server, a user directory such as LDAP and an XMPP server.
+   :alt: A diagram of a possible setup, consisting of Converse, a web server, a backend web application, an XMPP server, a user directory such as LDAP and an XMPP server.
 
 
    This diagram shows the various services in a fairly common setup (image generated with `draw.io <https://draw.io>`_).
    This diagram shows the various services in a fairly common setup (image generated with `draw.io <https://draw.io>`_).
 
 
@@ -53,11 +52,11 @@ The various components
 An XMPP server
 An XMPP server
 ==============
 ==============
 
 
-*Converse.js* implements `XMPP <http://xmpp.org/about-xmpp/>`_ as its
+*Converse* uses `XMPP <http://xmpp.org/about-xmpp/>`_ as its
 messaging protocol, and therefore needs to connect to an XMPP/Jabber
 messaging protocol, and therefore needs to connect to an XMPP/Jabber
 server (Jabber® is an older and more user-friendly synonym for XMPP).
 server (Jabber® is an older and more user-friendly synonym for XMPP).
 
 
-You can connect to public XMPP servers like ``jabber.org`` but if you want to
+You can connect to public XMPP servers like ``conversejs.org`` but if you want to
 have :ref:`session support <session-support>` you'll have to set up your own XMPP server.
 have :ref:`session support <session-support>` you'll have to set up your own XMPP server.
 
 
 You can find a list of public XMPP servers/providers on `xmpp.net <https://list.jabber.at>`_
 You can find a list of public XMPP servers/providers on `xmpp.net <https://list.jabber.at>`_
@@ -79,7 +78,7 @@ stanzas to be sent over an HTTP connection.
 HTTP connections are stateless and usually shortlived.
 HTTP connections are stateless and usually shortlived.
 XMPP connections on the other hand are stateful and usually last much longer.
 XMPP connections on the other hand are stateful and usually last much longer.
 
 
-So to enable a web application like *Converse.js* to communicate with an XMPP
+So to enable a web application like *Converse* to communicate with an XMPP
 server, we need a proxy which acts as a bridge between these two protocols.
 server, we need a proxy which acts as a bridge between these two protocols.
 
 
 This is the job of a BOSH connection manager. BOSH (Bidirectional-streams Over
 This is the job of a BOSH connection manager. BOSH (Bidirectional-streams Over
@@ -87,7 +86,7 @@ Synchronous HTTP) is a protocol for allowing XMPP communication over HTTP. The
 protocol is defined in `XEP-0206: XMPP Over BOSH <http://xmpp.org/extensions/xep-0206.html>`_.
 protocol is defined in `XEP-0206: XMPP Over BOSH <http://xmpp.org/extensions/xep-0206.html>`_.
 
 
 Popular XMPP servers such as `Ejabberd <http://www.ejabberd.im>`_,
 Popular XMPP servers such as `Ejabberd <http://www.ejabberd.im>`_,
-prosody `(mod_bosh) <http://prosody.im/doc/setting_up_bosh>`_ and
+Prosody `(mod_bosh) <http://prosody.im/doc/setting_up_bosh>`_ and
 `OpenFire <http://www.igniterealtime.org/projects/openfire/>`_ all include
 `OpenFire <http://www.igniterealtime.org/projects/openfire/>`_ all include
 their own BOSH connection managers (but you usually have to enable them in the
 their own BOSH connection managers (but you usually have to enable them in the
 configuration).
 configuration).
@@ -98,14 +97,15 @@ https://conversejs.org does), then you'll need a standalone connection manager.
 For a standalone manager, see for example `Punjab <https://github.com/twonds/punjab>`_
 For a standalone manager, see for example `Punjab <https://github.com/twonds/punjab>`_
 and `node-xmpp-bosh <https://github.com/dhruvbird/node-xmpp-bosh>`_.
 and `node-xmpp-bosh <https://github.com/dhruvbird/node-xmpp-bosh>`_.
 
 
-The demo on the `Converse.js homepage <http://conversejs.org>`_ uses a connection
+The demo on the `Converse homepage <http://conversejs.org>`_ uses a connection
 manager located at https://conversejs.org/http-bind.
 manager located at https://conversejs.org/http-bind.
 
 
 This connection manager is available for testing purposes only, please don't
 This connection manager is available for testing purposes only, please don't
 use it in production.
 use it in production.
 
 
 Refer to the :ref:`bosh-service-url` configuration setting for information on
 Refer to the :ref:`bosh-service-url` configuration setting for information on
-how to configure Converse.js to connect to a BOSH URL.
+how to configure Converse to connect to a BOSH URL.
+
 
 
 .. _`websocket-section`:
 .. _`websocket-section`:
 
 
@@ -122,7 +122,7 @@ HTTP. Therefore BOSH, which operates over HTTP, doesn't apply to websockets.
 does the node-xmpp-bosh connection manager.
 does the node-xmpp-bosh connection manager.
 
 
 Refer to the :ref:`websocket-url` configuration setting for information on how to
 Refer to the :ref:`websocket-url` configuration setting for information on how to
-configure Converse.js to connect to a websocket URL.
+configure Converse to connect to a websocket URL.
 
 
 The Webserver
 The Webserver
 =============
 =============
@@ -133,7 +133,7 @@ Overcoming cross-domain request restrictions
 Lets say your domain is *example.org*, but the domain of your connection
 Lets say your domain is *example.org*, but the domain of your connection
 manager is *example.com*.
 manager is *example.com*.
 
 
-HTTP requests are made by *Converse.js* to the BOSH connection manager via
+HTTP requests are made by *Converse* to the BOSH connection manager via
 XmlHttpRequests (XHR). Until recently, it was not possible to make such
 XmlHttpRequests (XHR). Until recently, it was not possible to make such
 requests to a different domain than the one currently being served
 requests to a different domain than the one currently being served
 (to prevent XSS attacks).
 (to prevent XSS attacks).
@@ -159,7 +159,7 @@ Examples:
 Assuming your site is accessible on port ``80`` for the domain ``mysite.com``
 Assuming your site is accessible on port ``80`` for the domain ``mysite.com``
 and your connection manager manager is running at ``someothersite.com/http-bind``.
 and your connection manager manager is running at ``someothersite.com/http-bind``.
 
 
-The *bosh_service_url* value you want to give Converse.js to overcome
+The *bosh_service_url* value you want to give Converse to overcome
 the cross-domain restriction is ``mysite.com/http-bind`` and not
 the cross-domain restriction is ``mysite.com/http-bind`` and not
 ``someothersite.com/http-bind``.
 ``someothersite.com/http-bind``.
 
 
@@ -192,6 +192,32 @@ Apache
     </VirtualHost>
     </VirtualHost>
 
 
 
 
+.. note::
+
+    If you're getting XML parsing errors for your BOSH endpoint, for
+    example::
+
+        XML Parsing Error: mismatched tag. Expected: </hr>.
+        Location: https://example.org/http-bind/
+        Line Number 6, Column 3: bosh-anon:6:3
+        Also ERROR: request id 12.2 error 504 happened
+
+    Then your BOSH proxy is returning an HTML error page (for a 504 error in
+    the above example).
+
+    This might be because your webserver and BOSH proxy have the same timeout
+    for BOSH requests. Because the webserver receives the request slightly earlier, 
+    it gives up a few microseconds before the XMPP server’s empty result and thus returns a
+    504 error page containing HTML to browser, which then gets parsed as if its
+    XML.
+
+    To fix this, make sure that the webserver's timeout is slightly higher.
+    In Nginx you can do this by adding ``proxy_read_timeout 61;``;
+
+    From Converse 4.0.0 onwards the default ``wait`` time is set to 59 seconds, to avoid
+    this problem.
+
+
 .. _`session-support`:
 .. _`session-support`:
 
 
 Single Session Support
 Single Session Support
@@ -215,7 +241,7 @@ authenticated BOSH session with the XMPP server or a standalone `BOSH <http://xm
 connection manager.
 connection manager.
 
 
 Once authenticated, it receives RID and SID tokens which need to be passed
 Once authenticated, it receives RID and SID tokens which need to be passed
-on to converse.js upon pa. Converse.js will then attach to that same session using
+on to converse.js upon pa. Converse will then attach to that same session using
 those tokens.
 those tokens.
 
 
 It's called "prebind" because you bind to the BOSH session beforehand, and then
 It's called "prebind" because you bind to the BOSH session beforehand, and then
@@ -250,7 +276,7 @@ converse.js can then attach to.
     A BOSH server acts as a bridge between HTTP, the protocol of the web, and
     A BOSH server acts as a bridge between HTTP, the protocol of the web, and
     XMPP, the instant messaging protocol.
     XMPP, the instant messaging protocol.
 
 
-    Converse.js can only communicate via HTTP (or websocket, in which case BOSH can't be used).
+    Converse can only communicate via HTTP (or websocket, in which case BOSH can't be used).
     It cannot open TCP sockets to communicate to an XMPP server directly.
     It cannot open TCP sockets to communicate to an XMPP server directly.
 
 
     So the BOSH server acts as a middle man, translating our HTTP requests into XMPP stanzas and vice versa.
     So the BOSH server acts as a middle man, translating our HTTP requests into XMPP stanzas and vice versa.
@@ -303,7 +329,7 @@ authentication to external services. They are listed in the `Prosody community m
 page <https://modules.prosody.im/>`_. Other XMPP servers have similar plugin modules.
 page <https://modules.prosody.im/>`_. Other XMPP servers have similar plugin modules.
 
 
 If your web-application has access to the same credentials, it can send those
 If your web-application has access to the same credentials, it can send those
-credentials to Converse.js so that user's are automatically logged in when the
+credentials to Converse so that user's are automatically logged in when the
 page loads.
 page loads.
 
 
 This is can be done by setting :ref:`auto_login` to true and configuring the 
 This is can be done by setting :ref:`auto_login` to true and configuring the 
@@ -316,7 +342,7 @@ The first option has the drawback that your web-application needs to know the
 XMPP credentials of your users and that they need to be stored in the clear.
 XMPP credentials of your users and that they need to be stored in the clear.
 
 
 The second option has that same drawback and it also needs to pass those
 The second option has that same drawback and it also needs to pass those
-credentials to Converse.js.
+credentials to Converse.
 
 
 To avoid these drawbacks, you can instead let your backend web application
 To avoid these drawbacks, you can instead let your backend web application
 generate temporary authentication tokens which are then sent to the XMPP server
 generate temporary authentication tokens which are then sent to the XMPP server

File diff suppressed because it is too large
+ 0 - 0
locale/ar/LC_MESSAGES/converse.json


+ 48 - 77
locale/ar/LC_MESSAGES/converse.po

@@ -8,7 +8,7 @@ msgstr ""
 "Project-Id-Version: Converse.js 3.3.4\n"
 "Project-Id-Version: Converse.js 3.3.4\n"
 "Report-Msgid-Bugs-To: \n"
 "Report-Msgid-Bugs-To: \n"
 "POT-Creation-Date: 2018-07-22 11:17+0200\n"
 "POT-Creation-Date: 2018-07-22 11:17+0200\n"
-"PO-Revision-Date: 2018-07-22 11:49+0200\n"
+"PO-Revision-Date: 2018-08-02 05:34+0000\n"
 "Last-Translator: ButterflyOfFire <ButterflyOfFire@protonmail.com>\n"
 "Last-Translator: ButterflyOfFire <ButterflyOfFire@protonmail.com>\n"
 "Language-Team: Arabic <https://hosted.weblate.org/projects/conversejs/"
 "Language-Team: Arabic <https://hosted.weblate.org/projects/conversejs/"
 "translations/ar/>\n"
 "translations/ar/>\n"
@@ -18,28 +18,25 @@ msgstr ""
 "Content-Transfer-Encoding: 8bit\n"
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=6; plural=n==0 ? 0 : n==1 ? 1 : n==2 ? 2 : n%100>=3 "
 "Plural-Forms: nplurals=6; plural=n==0 ? 0 : n==1 ? 1 : n==2 ? 2 : n%100>=3 "
 "&& n%100<=10 ? 3 : n%100>=11 ? 4 : 5;\n"
 "&& n%100<=10 ? 3 : n%100>=11 ? 4 : 5;\n"
-"X-Generator: Weblate 3.0\n"
+"X-Generator: Weblate 3.1.1\n"
 
 
 #: dist/converse-no-dependencies.js:40690
 #: dist/converse-no-dependencies.js:40690
 #: dist/converse-no-dependencies.js:40775
 #: dist/converse-no-dependencies.js:40775
 #: dist/converse-no-dependencies.js:53689
 #: dist/converse-no-dependencies.js:53689
-#, fuzzy
 msgid "Bookmark this groupchat"
 msgid "Bookmark this groupchat"
-msgstr "إضافة هذه الغرفة إلى الفواصل المرجعية"
+msgstr "إضافة فريق المحادثة هذا إلى الفواصل المرجعية"
 
 
 #: dist/converse-no-dependencies.js:40776
 #: dist/converse-no-dependencies.js:40776
 msgid "The name for this bookmark:"
 msgid "The name for this bookmark:"
 msgstr "تسمية الفاصلة المرجعية :"
 msgstr "تسمية الفاصلة المرجعية :"
 
 
 #: dist/converse-no-dependencies.js:40777
 #: dist/converse-no-dependencies.js:40777
-#, fuzzy
 msgid "Would you like this groupchat to be automatically joined upon startup?"
 msgid "Would you like this groupchat to be automatically joined upon startup?"
-msgstr "هل تود الإلتحاق بهذه الغرفة آليا مباشَرةً بعد الإتصال ؟"
+msgstr "هل تريد الإلتحاق آليًا بفريق المحادثة هذا مباشَرةً بعد الإتصال ؟"
 
 
 #: dist/converse-no-dependencies.js:40778
 #: dist/converse-no-dependencies.js:40778
-#, fuzzy
 msgid "What should your nickname for this groupchat be?"
 msgid "What should your nickname for this groupchat be?"
-msgstr "ما هو الإسم المُستعار الذي تريد استخدامه في غرفة المحادثة هذه ؟"
+msgstr "ما هو الإسم المُستعار الذي تريد استخدامه في فريق المحادثة هذا ؟"
 
 
 #: dist/converse-no-dependencies.js:40780
 #: dist/converse-no-dependencies.js:40780
 #: dist/converse-no-dependencies.js:49483
 #: dist/converse-no-dependencies.js:49483
@@ -66,9 +63,8 @@ msgstr "المعذرة، لقد طرأ هناك خطأ أثناء محاولة 
 
 
 #: dist/converse-no-dependencies.js:41055
 #: dist/converse-no-dependencies.js:41055
 #: dist/converse-no-dependencies.js:53687
 #: dist/converse-no-dependencies.js:53687
-#, fuzzy
 msgid "Leave this groupchat"
 msgid "Leave this groupchat"
-msgstr "الخروج مِن هذه الغرفة"
+msgstr "مغادرة فريق المحادثة"
 
 
 #: dist/converse-no-dependencies.js:41056
 #: dist/converse-no-dependencies.js:41056
 msgid "Remove this bookmark"
 msgid "Remove this bookmark"
@@ -76,23 +72,20 @@ msgstr "إزالة هذه الفاصلة المرجعية"
 
 
 #: dist/converse-no-dependencies.js:41057
 #: dist/converse-no-dependencies.js:41057
 #: dist/converse-no-dependencies.js:53688
 #: dist/converse-no-dependencies.js:53688
-#, fuzzy
 msgid "Unbookmark this groupchat"
 msgid "Unbookmark this groupchat"
-msgstr "تنحية غرفة المحادثة مِن الفواصل المرجعية"
+msgstr "تنحية فريق المحادثة مِن الفواصل المرجعية"
 
 
 #: dist/converse-no-dependencies.js:41058
 #: dist/converse-no-dependencies.js:41058
 #: dist/converse-no-dependencies.js:48755
 #: dist/converse-no-dependencies.js:48755
 #: dist/converse-no-dependencies.js:53690
 #: dist/converse-no-dependencies.js:53690
-#, fuzzy
 msgid "Show more information on this groupchat"
 msgid "Show more information on this groupchat"
-msgstr "عرض المزيد مِن التفاصيل عن هذه الغرفة"
+msgstr "عرض المزيد مِن التفاصيل عن فريق المحادثة هذا"
 
 
 #: dist/converse-no-dependencies.js:41061
 #: dist/converse-no-dependencies.js:41061
 #: dist/converse-no-dependencies.js:48754
 #: dist/converse-no-dependencies.js:48754
 #: dist/converse-no-dependencies.js:53692
 #: dist/converse-no-dependencies.js:53692
-#, fuzzy
 msgid "Click to open this groupchat"
 msgid "Click to open this groupchat"
-msgstr "أنقر لفتح غرفة المحادثة هذه"
+msgstr "أنقر لفتح فريق المحادثة هذا"
 
 
 #: dist/converse-no-dependencies.js:41097
 #: dist/converse-no-dependencies.js:41097
 msgid "Click to toggle the bookmarks list"
 msgid "Click to toggle the bookmarks list"
@@ -111,11 +104,11 @@ msgid "Sorry, could not determine upload URL."
 msgstr ""
 msgstr ""
 
 
 #: dist/converse-no-dependencies.js:41573
 #: dist/converse-no-dependencies.js:41573
-#, fuzzy, javascript-format
+#, javascript-format
 msgid ""
 msgid ""
 "Sorry, could not succesfully upload your file. Your server’s response: \"%1$s"
 "Sorry, could not succesfully upload your file. Your server’s response: \"%1$s"
 "\""
 "\""
-msgstr "للأسف لم نتمكّن مِن القيام برفع ملفك بنجاح."
+msgstr "للأسف لم نتمكّن مِن القيام برفع ملفك بنجاح. أجاب خادومك : \"%1$s\""
 
 
 #: dist/converse-no-dependencies.js:41575
 #: dist/converse-no-dependencies.js:41575
 msgid "Sorry, could not succesfully upload your file."
 msgid "Sorry, could not succesfully upload your file."
@@ -163,9 +156,8 @@ msgid "Full Name"
 msgstr ""
 msgstr ""
 
 
 #: dist/converse-no-dependencies.js:42572
 #: dist/converse-no-dependencies.js:42572
-#, fuzzy
 msgid "Jabber ID"
 msgid "Jabber ID"
-msgstr "مُعرَّف حساب جابر :"
+msgstr "مُعرَّف حساب جابر"
 
 
 #: dist/converse-no-dependencies.js:42573
 #: dist/converse-no-dependencies.js:42573
 #: dist/converse-no-dependencies.js:49639
 #: dist/converse-no-dependencies.js:49639
@@ -174,9 +166,8 @@ msgid "Nickname"
 msgstr "الإسم المُستعار"
 msgstr "الإسم المُستعار"
 
 
 #: dist/converse-no-dependencies.js:42574
 #: dist/converse-no-dependencies.js:42574
-#, fuzzy
 msgid "Remove as contact"
 msgid "Remove as contact"
-msgstr "إضافة مراسل"
+msgstr "إزالة مِن المراسِلين"
 
 
 #: dist/converse-no-dependencies.js:42575
 #: dist/converse-no-dependencies.js:42575
 msgid "Refresh"
 msgid "Refresh"
@@ -248,7 +239,6 @@ msgid "Clear all messages"
 msgstr "تنظيف كافة الرسائل"
 msgstr "تنظيف كافة الرسائل"
 
 
 #: dist/converse-no-dependencies.js:42815
 #: dist/converse-no-dependencies.js:42815
-#, fuzzy
 msgid "Insert emojis"
 msgid "Insert emojis"
 msgstr "إدراج وجه مبتسم"
 msgstr "إدراج وجه مبتسم"
 
 
@@ -275,25 +265,25 @@ msgid "Are you sure you want to clear the messages from this conversation?"
 msgstr "هل أنت متأكد أنك تود مسح الرسائل مِن نافذة المحادثة هذه ؟"
 msgstr "هل أنت متأكد أنك تود مسح الرسائل مِن نافذة المحادثة هذه ؟"
 
 
 #: dist/converse-no-dependencies.js:43413
 #: dist/converse-no-dependencies.js:43413
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s has gone offline"
 msgid "%1$s has gone offline"
-msgstr "قد قطع الإتصال"
+msgstr "%1$s قد قطع الإتصال"
 
 
 #: dist/converse-no-dependencies.js:43415
 #: dist/converse-no-dependencies.js:43415
 #: dist/converse-no-dependencies.js:47662
 #: dist/converse-no-dependencies.js:47662
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s has gone away"
 msgid "%1$s has gone away"
-msgstr "قد غاب"
+msgstr "%1$s قد غاب"
 
 
 #: dist/converse-no-dependencies.js:43417
 #: dist/converse-no-dependencies.js:43417
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s is busy"
 msgid "%1$s is busy"
-msgstr "مشغول"
+msgstr "%1$s مشغول"
 
 
 #: dist/converse-no-dependencies.js:43419
 #: dist/converse-no-dependencies.js:43419
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s is online"
 msgid "%1$s is online"
-msgstr "متصل"
+msgstr "%1$s متصل"
 
 
 #: dist/converse-no-dependencies.js:44042
 #: dist/converse-no-dependencies.js:44042
 msgid "Username"
 msgid "Username"
@@ -346,18 +336,18 @@ msgid "Typing from another device"
 msgstr "يكتب عبر جهاز آخَر"
 msgstr "يكتب عبر جهاز آخَر"
 
 
 #: dist/converse-no-dependencies.js:47653
 #: dist/converse-no-dependencies.js:47653
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s is typing"
 msgid "%1$s is typing"
-msgstr "يكتب حاليا"
+msgstr "%1$s يكتب حاليا"
 
 
 #: dist/converse-no-dependencies.js:47657
 #: dist/converse-no-dependencies.js:47657
 msgid "Stopped typing on the other device"
 msgid "Stopped typing on the other device"
 msgstr "توقّف عن الكتابة عبر الجهاز الآخَر"
 msgstr "توقّف عن الكتابة عبر الجهاز الآخَر"
 
 
 #: dist/converse-no-dependencies.js:47659
 #: dist/converse-no-dependencies.js:47659
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s has stopped typing"
 msgid "%1$s has stopped typing"
-msgstr "توقّفَ عن الكتابة"
+msgstr "%1$s توقّفَ عن الكتابة"
 
 
 #: dist/converse-no-dependencies.js:47905
 #: dist/converse-no-dependencies.js:47905
 #: dist/converse-no-dependencies.js:47948
 #: dist/converse-no-dependencies.js:47948
@@ -373,29 +363,24 @@ msgid "Minimized"
 msgstr "تصغير"
 msgstr "تصغير"
 
 
 #: dist/converse-no-dependencies.js:48597
 #: dist/converse-no-dependencies.js:48597
-#, fuzzy
 msgid "This groupchat is not anonymous"
 msgid "This groupchat is not anonymous"
-msgstr "غرفة المحادثة هذه ليست مجهولة"
+msgstr "فريق المحادثة هذا ليس مجهولًا"
 
 
 #: dist/converse-no-dependencies.js:48598
 #: dist/converse-no-dependencies.js:48598
-#, fuzzy
 msgid "This groupchat now shows unavailable members"
 msgid "This groupchat now shows unavailable members"
-msgstr "هذه القاعة لا تقوم بعرض الأعضاء المشغولين"
+msgstr ""
 
 
 #: dist/converse-no-dependencies.js:48599
 #: dist/converse-no-dependencies.js:48599
-#, fuzzy
 msgid "This groupchat does not show unavailable members"
 msgid "This groupchat does not show unavailable members"
-msgstr "هذه القاعة لا تقوم بعرض الأعضاء المشغولين"
+msgstr "فريق المحادثة هذا لا يعرض الأعضاء المشغولين"
 
 
 #: dist/converse-no-dependencies.js:48600
 #: dist/converse-no-dependencies.js:48600
-#, fuzzy
 msgid "The groupchat configuration has changed"
 msgid "The groupchat configuration has changed"
-msgstr "تم تعديل إعدادات غرفة المحادثة"
+msgstr "تم تعديل خيارات فريق المحادثة"
 
 
 #: dist/converse-no-dependencies.js:48601
 #: dist/converse-no-dependencies.js:48601
-#, fuzzy
 msgid "groupchat logging is now enabled"
 msgid "groupchat logging is now enabled"
-msgstr "الإلتحاق بالغرفة مسموح للجميع الآن"
+msgstr "الإلتحاق بفريق المحادثة مسموح الآن للجميع"
 
 
 #: dist/converse-no-dependencies.js:48602
 #: dist/converse-no-dependencies.js:48602
 #, fuzzy
 #, fuzzy
@@ -403,34 +388,28 @@ msgid "groupchat logging is now disabled"
 msgstr "مُنِع الآن الإلتحاق بغرفة المحادثة"
 msgstr "مُنِع الآن الإلتحاق بغرفة المحادثة"
 
 
 #: dist/converse-no-dependencies.js:48603
 #: dist/converse-no-dependencies.js:48603
-#, fuzzy
 msgid "This groupchat is now no longer anonymous"
 msgid "This groupchat is now no longer anonymous"
-msgstr "لم تَعُد غرفة المحادثة مجهولة الآن"
+msgstr "لم يَعُد فريق المحادثة مجهولا بعد الآن"
 
 
 #: dist/converse-no-dependencies.js:48604
 #: dist/converse-no-dependencies.js:48604
-#, fuzzy
 msgid "This groupchat is now semi-anonymous"
 msgid "This groupchat is now semi-anonymous"
-msgstr "أصبحت غرفة المحادثة مجهولة نسبيًا"
+msgstr "أصبح فريق المحادثة مجهولا نسبيًا"
 
 
 #: dist/converse-no-dependencies.js:48605
 #: dist/converse-no-dependencies.js:48605
-#, fuzzy
 msgid "This groupchat is now fully-anonymous"
 msgid "This groupchat is now fully-anonymous"
-msgstr "أصبحت غرفة المحادثة الآن مجهولة تمامًا"
+msgstr "أصبح فريق المحادثة الآن مجهولا تمامًا"
 
 
 #: dist/converse-no-dependencies.js:48606
 #: dist/converse-no-dependencies.js:48606
-#, fuzzy
 msgid "A new groupchat has been created"
 msgid "A new groupchat has been created"
-msgstr "تم إنشاء غرفة محادثة جديدة"
+msgstr "تم إنشاء فريق محادثة جديد"
 
 
 #: dist/converse-no-dependencies.js:48609
 #: dist/converse-no-dependencies.js:48609
-#, fuzzy
 msgid "You have been banned from this groupchat"
 msgid "You have been banned from this groupchat"
-msgstr "لقد تم طردُك مِن غرفة المحادثة هذه"
+msgstr "لقد تم طردُك مِن فريق المحادثة هذا"
 
 
 #: dist/converse-no-dependencies.js:48610
 #: dist/converse-no-dependencies.js:48610
-#, fuzzy
 msgid "You have been kicked from this groupchat"
 msgid "You have been kicked from this groupchat"
-msgstr "لقد تم طردُك مؤقتًا مِن غرفة المحادثة هذه"
+msgstr "لقد تم طردُك مؤقتًا مِن فريق المحادثة هذا"
 
 
 #: dist/converse-no-dependencies.js:48611
 #: dist/converse-no-dependencies.js:48611
 msgid ""
 msgid ""
@@ -499,14 +478,12 @@ msgid "Description:"
 msgstr "التفاصيل :"
 msgstr "التفاصيل :"
 
 
 #: dist/converse-no-dependencies.js:48666
 #: dist/converse-no-dependencies.js:48666
-#, fuzzy
 msgid "Groupchat Address (JID):"
 msgid "Groupchat Address (JID):"
-msgstr "عنوان غرفة المحادثة (JID) :"
+msgstr "عنوان فريق المحادثة (JID) :"
 
 
 #: dist/converse-no-dependencies.js:48667
 #: dist/converse-no-dependencies.js:48667
-#, fuzzy
 msgid "Participants:"
 msgid "Participants:"
-msgstr "المستخدِمون المُقِيمون :"
+msgstr "المشتركون :"
 
 
 #: dist/converse-no-dependencies.js:48668
 #: dist/converse-no-dependencies.js:48668
 msgid "Features:"
 msgid "Features:"
@@ -543,9 +520,8 @@ msgid "Open"
 msgstr "مفتوحة"
 msgstr "مفتوحة"
 
 
 #: dist/converse-no-dependencies.js:48675
 #: dist/converse-no-dependencies.js:48675
-#, fuzzy
 msgid "Permanent"
 msgid "Permanent"
-msgstr "غرفة محادثة دائمة"
+msgstr "دائم"
 
 
 #: dist/converse-no-dependencies.js:48676
 #: dist/converse-no-dependencies.js:48676
 #: dist/converse-no-dependencies.js:57167
 #: dist/converse-no-dependencies.js:57167
@@ -570,32 +546,28 @@ msgid "Unmoderated"
 msgstr "ليست تحت الإشراف"
 msgstr "ليست تحت الإشراف"
 
 
 #: dist/converse-no-dependencies.js:48715
 #: dist/converse-no-dependencies.js:48715
-#, fuzzy
 msgid "Query for Groupchats"
 msgid "Query for Groupchats"
-msgstr "البحث عن قاعات"
+msgstr "البحث عن فِرق محادثة"
 
 
 #: dist/converse-no-dependencies.js:48716
 #: dist/converse-no-dependencies.js:48716
 msgid "Server address"
 msgid "Server address"
 msgstr "عنوان الخادوم"
 msgstr "عنوان الخادوم"
 
 
 #: dist/converse-no-dependencies.js:48717
 #: dist/converse-no-dependencies.js:48717
-#, fuzzy
 msgid "Show groupchats"
 msgid "Show groupchats"
-msgstr "عرض غُرف المحادثة"
+msgstr "عرض فِرَق المحادثة"
 
 
 #: dist/converse-no-dependencies.js:48718
 #: dist/converse-no-dependencies.js:48718
 msgid "conference.example.org"
 msgid "conference.example.org"
 msgstr "conference.example.org"
 msgstr "conference.example.org"
 
 
 #: dist/converse-no-dependencies.js:48767
 #: dist/converse-no-dependencies.js:48767
-#, fuzzy
 msgid "No groupchats found"
 msgid "No groupchats found"
-msgstr "لم يتم العثور على أية غُرفة محادثة"
+msgstr "لم يتم العثور على أي فريق محادثة"
 
 
 #: dist/converse-no-dependencies.js:48784
 #: dist/converse-no-dependencies.js:48784
-#, fuzzy
 msgid "groupchats found:"
 msgid "groupchats found:"
-msgstr "تم العثور على غرف المحادثة :"
+msgstr "تم العثور على فِرَق المحادثة :"
 
 
 #: dist/converse-no-dependencies.js:48836
 #: dist/converse-no-dependencies.js:48836
 #, fuzzy
 #, fuzzy
@@ -603,9 +575,8 @@ msgid "Enter a new Groupchat"
 msgstr "الدخول إلى غرفة محادثة جديدة"
 msgstr "الدخول إلى غرفة محادثة جديدة"
 
 
 #: dist/converse-no-dependencies.js:48837
 #: dist/converse-no-dependencies.js:48837
-#, fuzzy
 msgid "Groupchat address"
 msgid "Groupchat address"
-msgstr "عنوان غرفة المحادثة"
+msgstr "عنوان فريق المحادثة"
 
 
 #: dist/converse-no-dependencies.js:48838
 #: dist/converse-no-dependencies.js:48838
 #: dist/converse-no-dependencies.js:55005
 #: dist/converse-no-dependencies.js:55005
@@ -621,18 +592,18 @@ msgid "Join"
 msgstr "الإلتحاق بالغرفة"
 msgstr "الإلتحاق بالغرفة"
 
 
 #: dist/converse-no-dependencies.js:48884
 #: dist/converse-no-dependencies.js:48884
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "Groupchat info for %1$s"
 msgid "Groupchat info for %1$s"
-msgstr "إشعار مِن %1$s"
+msgstr ""
 
 
 #: dist/converse-no-dependencies.js:48990
 #: dist/converse-no-dependencies.js:48990
 msgid "Message"
 msgid "Message"
 msgstr "رسالة"
 msgstr "رسالة"
 
 
 #: dist/converse-no-dependencies.js:49036
 #: dist/converse-no-dependencies.js:49036
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s is no longer a moderator"
 msgid "%1$s is no longer a moderator"
-msgstr "لم يعُد %1$s مِن مُشْرِفي غرفة المحادثة"
+msgstr "لم يعُد %1$s مِن المُشْرِفين"
 
 
 #: dist/converse-no-dependencies.js:49040
 #: dist/converse-no-dependencies.js:49040
 #, fuzzy, javascript-format
 #, fuzzy, javascript-format

File diff suppressed because it is too large
+ 0 - 0
locale/de/LC_MESSAGES/converse.json


File diff suppressed because it is too large
+ 111 - 188
locale/de/LC_MESSAGES/converse.po


File diff suppressed because it is too large
+ 0 - 0
locale/fr/LC_MESSAGES/converse.json


+ 80 - 151
locale/fr/LC_MESSAGES/converse.po

@@ -7,7 +7,7 @@ msgstr ""
 "Project-Id-Version: Converse.js 0.4\n"
 "Project-Id-Version: Converse.js 0.4\n"
 "Report-Msgid-Bugs-To: \n"
 "Report-Msgid-Bugs-To: \n"
 "POT-Creation-Date: 2018-07-22 11:17+0200\n"
 "POT-Creation-Date: 2018-07-22 11:17+0200\n"
-"PO-Revision-Date: 2018-07-22 11:50+0200\n"
+"PO-Revision-Date: 2018-08-02 08:49+0000\n"
 "Last-Translator: Emmanuel Gil Peyrot <linkmauve@linkmauve.fr>\n"
 "Last-Translator: Emmanuel Gil Peyrot <linkmauve@linkmauve.fr>\n"
 "Language-Team: French <https://hosted.weblate.org/projects/conversejs/"
 "Language-Team: French <https://hosted.weblate.org/projects/conversejs/"
 "translations/fr/>\n"
 "translations/fr/>\n"
@@ -16,7 +16,7 @@ msgstr ""
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Type: text/plain; charset=UTF-8\n"
 "Content-Transfer-Encoding: 8bit\n"
 "Content-Transfer-Encoding: 8bit\n"
 "Plural-Forms: nplurals=2; plural=n > 1;\n"
 "Plural-Forms: nplurals=2; plural=n > 1;\n"
-"X-Generator: Weblate 3.1-dev\n"
+"X-Generator: Weblate 3.1.1\n"
 "plural_forms: nplurals=2; plural=(n != 1);\n"
 "plural_forms: nplurals=2; plural=(n != 1);\n"
 "lang: fr\n"
 "lang: fr\n"
 "Language-Code: fr\n"
 "Language-Code: fr\n"
@@ -27,9 +27,8 @@ msgstr ""
 #: dist/converse-no-dependencies.js:40690
 #: dist/converse-no-dependencies.js:40690
 #: dist/converse-no-dependencies.js:40775
 #: dist/converse-no-dependencies.js:40775
 #: dist/converse-no-dependencies.js:53689
 #: dist/converse-no-dependencies.js:53689
-#, fuzzy
 msgid "Bookmark this groupchat"
 msgid "Bookmark this groupchat"
-msgstr "Marquer ce salon"
+msgstr "Mettre ce salon en marque-page"
 
 
 #: dist/converse-no-dependencies.js:40776
 #: dist/converse-no-dependencies.js:40776
 msgid "The name for this bookmark:"
 msgid "The name for this bookmark:"
@@ -37,64 +36,60 @@ msgstr "Nom de ce marque-page :"
 
 
 #: dist/converse-no-dependencies.js:40777
 #: dist/converse-no-dependencies.js:40777
 msgid "Would you like this groupchat to be automatically joined upon startup?"
 msgid "Would you like this groupchat to be automatically joined upon startup?"
-msgstr ""
+msgstr "Voulez-vous que ce salon soit automatiquement rejoint au démarrage ?"
 
 
 #: dist/converse-no-dependencies.js:40778
 #: dist/converse-no-dependencies.js:40778
 msgid "What should your nickname for this groupchat be?"
 msgid "What should your nickname for this groupchat be?"
-msgstr ""
+msgstr "Que devrait être votre pseudo sur ce salon ?"
 
 
 #: dist/converse-no-dependencies.js:40780
 #: dist/converse-no-dependencies.js:40780
 #: dist/converse-no-dependencies.js:49483
 #: dist/converse-no-dependencies.js:49483
 #: dist/converse-no-dependencies.js:52484
 #: dist/converse-no-dependencies.js:52484
 #: dist/converse-no-dependencies.js:52568
 #: dist/converse-no-dependencies.js:52568
 msgid "Save"
 msgid "Save"
-msgstr ""
+msgstr "Sauvegarder"
 
 
 #: dist/converse-no-dependencies.js:40781
 #: dist/converse-no-dependencies.js:40781
 #: dist/converse-no-dependencies.js:49484
 #: dist/converse-no-dependencies.js:49484
 #: dist/converse-no-dependencies.js:52564
 #: dist/converse-no-dependencies.js:52564
 #: dist/converse-no-dependencies.js:58864
 #: dist/converse-no-dependencies.js:58864
 msgid "Cancel"
 msgid "Cancel"
-msgstr ""
+msgstr "Annuler"
 
 
 #: dist/converse-no-dependencies.js:40854
 #: dist/converse-no-dependencies.js:40854
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "Are you sure you want to remove the bookmark \"%1$s\"?"
 msgid "Are you sure you want to remove the bookmark \"%1$s\"?"
-msgstr "Voulez-vous vraiment quitter le salon « %1$s » ?"
+msgstr "Voulez-vous vraiment supprimer le marque-page « %1$s » ?"
 
 
 #: dist/converse-no-dependencies.js:40970
 #: dist/converse-no-dependencies.js:40970
-#, fuzzy
 msgid "Sorry, something went wrong while trying to save your bookmark."
 msgid "Sorry, something went wrong while trying to save your bookmark."
 msgstr ""
 msgstr ""
-"Désolé, quelque chose s’est mal passé pendant la sauvegarde de votre profil."
+"Désolé, quelque chose s’est mal passé pendant la sauvegarde de ce marque-"
+"page."
 
 
 #: dist/converse-no-dependencies.js:41055
 #: dist/converse-no-dependencies.js:41055
 #: dist/converse-no-dependencies.js:53687
 #: dist/converse-no-dependencies.js:53687
 msgid "Leave this groupchat"
 msgid "Leave this groupchat"
-msgstr ""
+msgstr "Quitter ce salon"
 
 
 #: dist/converse-no-dependencies.js:41056
 #: dist/converse-no-dependencies.js:41056
-#, fuzzy
 msgid "Remove this bookmark"
 msgid "Remove this bookmark"
-msgstr "Nom de ce marque-page :"
+msgstr "Supprimer ce marque-page"
 
 
 #: dist/converse-no-dependencies.js:41057
 #: dist/converse-no-dependencies.js:41057
 #: dist/converse-no-dependencies.js:53688
 #: dist/converse-no-dependencies.js:53688
-#, fuzzy
 msgid "Unbookmark this groupchat"
 msgid "Unbookmark this groupchat"
-msgstr "Marquer ce salon"
+msgstr "Retirer ce salon des marque-pages"
 
 
 #: dist/converse-no-dependencies.js:41058
 #: dist/converse-no-dependencies.js:41058
 #: dist/converse-no-dependencies.js:48755
 #: dist/converse-no-dependencies.js:48755
 #: dist/converse-no-dependencies.js:53690
 #: dist/converse-no-dependencies.js:53690
-#, fuzzy
 msgid "Show more information on this groupchat"
 msgid "Show more information on this groupchat"
 msgstr "Afficher davantage d’informations sur ce salon"
 msgstr "Afficher davantage d’informations sur ce salon"
 
 
 #: dist/converse-no-dependencies.js:41061
 #: dist/converse-no-dependencies.js:41061
 #: dist/converse-no-dependencies.js:48754
 #: dist/converse-no-dependencies.js:48754
 #: dist/converse-no-dependencies.js:53692
 #: dist/converse-no-dependencies.js:53692
-#, fuzzy
 msgid "Click to open this groupchat"
 msgid "Click to open this groupchat"
 msgstr "Cliquer pour ouvrir ce salon"
 msgstr "Cliquer pour ouvrir ce salon"
 
 
@@ -115,11 +110,11 @@ msgid "Sorry, could not determine upload URL."
 msgstr "Désolé, impossible de déterminer l’URL d’envoi de fichier."
 msgstr "Désolé, impossible de déterminer l’URL d’envoi de fichier."
 
 
 #: dist/converse-no-dependencies.js:41573
 #: dist/converse-no-dependencies.js:41573
-#, fuzzy, javascript-format
+#, javascript-format
 msgid ""
 msgid ""
 "Sorry, could not succesfully upload your file. Your server’s response: \"%1$s"
 "Sorry, could not succesfully upload your file. Your server’s response: \"%1$s"
 "\""
 "\""
-msgstr "Désolé, l’envoi de fichier a échoué."
+msgstr "Désolé, l’envoi de fichier a échoué. Votre serveur a répondu : « %1$s »"
 
 
 #: dist/converse-no-dependencies.js:41575
 #: dist/converse-no-dependencies.js:41575
 msgid "Sorry, could not succesfully upload your file."
 msgid "Sorry, could not succesfully upload your file."
@@ -140,7 +135,7 @@ msgstr ""
 
 
 #: dist/converse-no-dependencies.js:41996
 #: dist/converse-no-dependencies.js:41996
 msgid "Sorry, an error occurred:"
 msgid "Sorry, an error occurred:"
-msgstr ""
+msgstr "Désolé, une erreur s’est produite :"
 
 
 #: dist/converse-no-dependencies.js:42538
 #: dist/converse-no-dependencies.js:42538
 msgid "Close this chat box"
 msgid "Close this chat box"
@@ -280,25 +275,25 @@ msgid "Are you sure you want to clear the messages from this conversation?"
 msgstr "Voulez-vous vraiment effacer les messages de cette conversation ?"
 msgstr "Voulez-vous vraiment effacer les messages de cette conversation ?"
 
 
 #: dist/converse-no-dependencies.js:43413
 #: dist/converse-no-dependencies.js:43413
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s has gone offline"
 msgid "%1$s has gone offline"
-msgstr "s’est déconnecté"
+msgstr "%1$s s’est déconnecté"
 
 
 #: dist/converse-no-dependencies.js:43415
 #: dist/converse-no-dependencies.js:43415
 #: dist/converse-no-dependencies.js:47662
 #: dist/converse-no-dependencies.js:47662
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s has gone away"
 msgid "%1$s has gone away"
-msgstr "%1$s a été banni"
+msgstr "%1$s n’est plus disponible"
 
 
 #: dist/converse-no-dependencies.js:43417
 #: dist/converse-no-dependencies.js:43417
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s is busy"
 msgid "%1$s is busy"
-msgstr "est occupé"
+msgstr "%1$s est occupé"
 
 
 #: dist/converse-no-dependencies.js:43419
 #: dist/converse-no-dependencies.js:43419
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s is online"
 msgid "%1$s is online"
-msgstr "est en ligne"
+msgstr "%1$s est en ligne"
 
 
 #: dist/converse-no-dependencies.js:44042
 #: dist/converse-no-dependencies.js:44042
 msgid "Username"
 msgid "Username"
@@ -354,21 +349,21 @@ msgstr "Afficher plus"
 
 
 #: dist/converse-no-dependencies.js:47651
 #: dist/converse-no-dependencies.js:47651
 msgid "Typing from another device"
 msgid "Typing from another device"
-msgstr ""
+msgstr "En train d’écrire depuis un autre client"
 
 
 #: dist/converse-no-dependencies.js:47653
 #: dist/converse-no-dependencies.js:47653
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s is typing"
 msgid "%1$s is typing"
-msgstr "%1$s dit"
+msgstr "%1$s est en train d’écrire"
 
 
 #: dist/converse-no-dependencies.js:47657
 #: dist/converse-no-dependencies.js:47657
 msgid "Stopped typing on the other device"
 msgid "Stopped typing on the other device"
-msgstr ""
+msgstr "A arrêté d’écrire sur l’autre client"
 
 
 #: dist/converse-no-dependencies.js:47659
 #: dist/converse-no-dependencies.js:47659
 #, javascript-format
 #, javascript-format
 msgid "%1$s has stopped typing"
 msgid "%1$s has stopped typing"
-msgstr ""
+msgstr "%1$s a arrêté d’écrire"
 
 
 #: dist/converse-no-dependencies.js:47905
 #: dist/converse-no-dependencies.js:47905
 #: dist/converse-no-dependencies.js:47948
 #: dist/converse-no-dependencies.js:47948
@@ -384,73 +379,59 @@ msgid "Minimized"
 msgstr "Réduit(s)"
 msgstr "Réduit(s)"
 
 
 #: dist/converse-no-dependencies.js:48597
 #: dist/converse-no-dependencies.js:48597
-#, fuzzy
 msgid "This groupchat is not anonymous"
 msgid "This groupchat is not anonymous"
 msgstr "Ce salon n’est pas anonyme"
 msgstr "Ce salon n’est pas anonyme"
 
 
 #: dist/converse-no-dependencies.js:48598
 #: dist/converse-no-dependencies.js:48598
-#, fuzzy
 msgid "This groupchat now shows unavailable members"
 msgid "This groupchat now shows unavailable members"
 msgstr "Ce salon affiche maintenant les membres indisponibles"
 msgstr "Ce salon affiche maintenant les membres indisponibles"
 
 
 #: dist/converse-no-dependencies.js:48599
 #: dist/converse-no-dependencies.js:48599
-#, fuzzy
 msgid "This groupchat does not show unavailable members"
 msgid "This groupchat does not show unavailable members"
 msgstr "Ce salon n’affiche pas les membres indisponibles"
 msgstr "Ce salon n’affiche pas les membres indisponibles"
 
 
 #: dist/converse-no-dependencies.js:48600
 #: dist/converse-no-dependencies.js:48600
-#, fuzzy
 msgid "The groupchat configuration has changed"
 msgid "The groupchat configuration has changed"
 msgstr "Les paramètres de ce salon ont été modifiés"
 msgstr "Les paramètres de ce salon ont été modifiés"
 
 
 #: dist/converse-no-dependencies.js:48601
 #: dist/converse-no-dependencies.js:48601
-#, fuzzy
 msgid "groupchat logging is now enabled"
 msgid "groupchat logging is now enabled"
-msgstr "Le logging du salon est activé"
+msgstr "L’enregistrement des logs de ce salon est maintenant activé"
 
 
 #: dist/converse-no-dependencies.js:48602
 #: dist/converse-no-dependencies.js:48602
-#, fuzzy
 msgid "groupchat logging is now disabled"
 msgid "groupchat logging is now disabled"
-msgstr "Le logging du salon est désactivé"
+msgstr "L’enregistrement des logs de ce salon est maintenant désactivé"
 
 
 #: dist/converse-no-dependencies.js:48603
 #: dist/converse-no-dependencies.js:48603
-#, fuzzy
 msgid "This groupchat is now no longer anonymous"
 msgid "This groupchat is now no longer anonymous"
 msgstr "Ce salon n’est plus anonyme"
 msgstr "Ce salon n’est plus anonyme"
 
 
 #: dist/converse-no-dependencies.js:48604
 #: dist/converse-no-dependencies.js:48604
-#, fuzzy
 msgid "This groupchat is now semi-anonymous"
 msgid "This groupchat is now semi-anonymous"
 msgstr "Ce salon est maintenant semi-anonyme"
 msgstr "Ce salon est maintenant semi-anonyme"
 
 
 #: dist/converse-no-dependencies.js:48605
 #: dist/converse-no-dependencies.js:48605
-#, fuzzy
 msgid "This groupchat is now fully-anonymous"
 msgid "This groupchat is now fully-anonymous"
 msgstr "Ce salon est maintenant entièrement anonyme"
 msgstr "Ce salon est maintenant entièrement anonyme"
 
 
 #: dist/converse-no-dependencies.js:48606
 #: dist/converse-no-dependencies.js:48606
-#, fuzzy
 msgid "A new groupchat has been created"
 msgid "A new groupchat has been created"
 msgstr "Un nouveau salon a été créé"
 msgstr "Un nouveau salon a été créé"
 
 
 #: dist/converse-no-dependencies.js:48609
 #: dist/converse-no-dependencies.js:48609
-#, fuzzy
 msgid "You have been banned from this groupchat"
 msgid "You have been banned from this groupchat"
 msgstr "Vous avez été banni de ce salon"
 msgstr "Vous avez été banni de ce salon"
 
 
 #: dist/converse-no-dependencies.js:48610
 #: dist/converse-no-dependencies.js:48610
-#, fuzzy
 msgid "You have been kicked from this groupchat"
 msgid "You have been kicked from this groupchat"
 msgstr "Vous avez été expulsé de ce salon"
 msgstr "Vous avez été expulsé de ce salon"
 
 
 #: dist/converse-no-dependencies.js:48611
 #: dist/converse-no-dependencies.js:48611
-#, fuzzy
 msgid ""
 msgid ""
 "You have been removed from this groupchat because of an affiliation change"
 "You have been removed from this groupchat because of an affiliation change"
 msgstr "Vous avez été retiré de ce salon du fait d’un changement d’affiliation"
 msgstr "Vous avez été retiré de ce salon du fait d’un changement d’affiliation"
 
 
 #: dist/converse-no-dependencies.js:48612
 #: dist/converse-no-dependencies.js:48612
-#, fuzzy
 msgid ""
 msgid ""
 "You have been removed from this groupchat because the groupchat has changed "
 "You have been removed from this groupchat because the groupchat has changed "
 "to members-only and you're not a member"
 "to members-only and you're not a member"
@@ -459,13 +440,12 @@ msgstr ""
 "membres et vous n’êtes pas membre"
 "membres et vous n’êtes pas membre"
 
 
 #: dist/converse-no-dependencies.js:48613
 #: dist/converse-no-dependencies.js:48613
-#, fuzzy
 msgid ""
 msgid ""
 "You have been removed from this groupchat because the MUC (Multi-user chat) "
 "You have been removed from this groupchat because the MUC (Multi-user chat) "
 "service is being shut down"
 "service is being shut down"
 msgstr ""
 msgstr ""
-"Vous avez été retiré de ce salon parce que le service de chat multi-"
-"utilisateur (MUC) est en cours d’extinction"
+"Vous avez été retiré de ce salon parce que le service sur lequel il est "
+"hébergé est en cours d’extinction"
 
 
 #. XXX: Note the triple underscore function and not double
 #. XXX: Note the triple underscore function and not double
 #. * underscore.
 #. * underscore.
@@ -517,12 +497,10 @@ msgid "Description:"
 msgstr "Description :"
 msgstr "Description :"
 
 
 #: dist/converse-no-dependencies.js:48666
 #: dist/converse-no-dependencies.js:48666
-#, fuzzy
 msgid "Groupchat Address (JID):"
 msgid "Groupchat Address (JID):"
 msgstr "Adresse du salon (JID) :"
 msgstr "Adresse du salon (JID) :"
 
 
 #: dist/converse-no-dependencies.js:48667
 #: dist/converse-no-dependencies.js:48667
-#, fuzzy
 msgid "Participants:"
 msgid "Participants:"
 msgstr "Participants :"
 msgstr "Participants :"
 
 
@@ -561,9 +539,8 @@ msgid "Open"
 msgstr "Ouvert"
 msgstr "Ouvert"
 
 
 #: dist/converse-no-dependencies.js:48675
 #: dist/converse-no-dependencies.js:48675
-#, fuzzy
 msgid "Permanent"
 msgid "Permanent"
-msgstr "Salon permanent"
+msgstr "Permanent"
 
 
 #: dist/converse-no-dependencies.js:48676
 #: dist/converse-no-dependencies.js:48676
 #: dist/converse-no-dependencies.js:57167
 #: dist/converse-no-dependencies.js:57167
@@ -588,7 +565,6 @@ msgid "Unmoderated"
 msgstr "Non modéré"
 msgstr "Non modéré"
 
 
 #: dist/converse-no-dependencies.js:48715
 #: dist/converse-no-dependencies.js:48715
-#, fuzzy
 msgid "Query for Groupchats"
 msgid "Query for Groupchats"
 msgstr "Chercher un salon"
 msgstr "Chercher un salon"
 
 
@@ -597,7 +573,6 @@ msgid "Server address"
 msgstr "Adresse du serveur"
 msgstr "Adresse du serveur"
 
 
 #: dist/converse-no-dependencies.js:48717
 #: dist/converse-no-dependencies.js:48717
-#, fuzzy
 msgid "Show groupchats"
 msgid "Show groupchats"
 msgstr "Afficher les salons"
 msgstr "Afficher les salons"
 
 
@@ -606,22 +581,18 @@ msgid "conference.example.org"
 msgstr "chat.exemple.org"
 msgstr "chat.exemple.org"
 
 
 #: dist/converse-no-dependencies.js:48767
 #: dist/converse-no-dependencies.js:48767
-#, fuzzy
 msgid "No groupchats found"
 msgid "No groupchats found"
 msgstr "Aucun salon trouvé"
 msgstr "Aucun salon trouvé"
 
 
 #: dist/converse-no-dependencies.js:48784
 #: dist/converse-no-dependencies.js:48784
-#, fuzzy
 msgid "groupchats found:"
 msgid "groupchats found:"
 msgstr "Salons trouvés :"
 msgstr "Salons trouvés :"
 
 
 #: dist/converse-no-dependencies.js:48836
 #: dist/converse-no-dependencies.js:48836
-#, fuzzy
 msgid "Enter a new Groupchat"
 msgid "Enter a new Groupchat"
 msgstr "Entrer dans un nouveau salon"
 msgstr "Entrer dans un nouveau salon"
 
 
 #: dist/converse-no-dependencies.js:48837
 #: dist/converse-no-dependencies.js:48837
-#, fuzzy
 msgid "Groupchat address"
 msgid "Groupchat address"
 msgstr "Adresse du salon"
 msgstr "Adresse du salon"
 
 
@@ -639,9 +610,9 @@ msgid "Join"
 msgstr "Rejoindre"
 msgstr "Rejoindre"
 
 
 #: dist/converse-no-dependencies.js:48884
 #: dist/converse-no-dependencies.js:48884
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "Groupchat info for %1$s"
 msgid "Groupchat info for %1$s"
-msgstr "Notification depuis %1$s"
+msgstr "Informations sur le salon %1$s"
 
 
 #: dist/converse-no-dependencies.js:48990
 #: dist/converse-no-dependencies.js:48990
 msgid "Message"
 msgid "Message"
@@ -668,22 +639,18 @@ msgid "%1$s is now a moderator"
 msgstr "%1$s est désormais un modérateur"
 msgstr "%1$s est désormais un modérateur"
 
 
 #: dist/converse-no-dependencies.js:49056
 #: dist/converse-no-dependencies.js:49056
-#, fuzzy
 msgid "Close and leave this groupchat"
 msgid "Close and leave this groupchat"
 msgstr "Fermer et quitter ce salon"
 msgstr "Fermer et quitter ce salon"
 
 
 #: dist/converse-no-dependencies.js:49057
 #: dist/converse-no-dependencies.js:49057
-#, fuzzy
 msgid "Configure this groupchat"
 msgid "Configure this groupchat"
 msgstr "Configurer ce salon"
 msgstr "Configurer ce salon"
 
 
 #: dist/converse-no-dependencies.js:49058
 #: dist/converse-no-dependencies.js:49058
-#, fuzzy
 msgid "Show more details about this groupchat"
 msgid "Show more details about this groupchat"
 msgstr "Afficher davantage d’informations sur ce salon"
 msgstr "Afficher davantage d’informations sur ce salon"
 
 
 #: dist/converse-no-dependencies.js:49098
 #: dist/converse-no-dependencies.js:49098
-#, fuzzy
 msgid "Hide the list of participants"
 msgid "Hide the list of participants"
 msgstr "Cacher la liste des participants"
 msgstr "Cacher la liste des participants"
 
 
@@ -710,7 +677,6 @@ msgid "Change user's affiliation to admin"
 msgstr "Changer le rôle  de l’utilisateur en administrateur"
 msgstr "Changer le rôle  de l’utilisateur en administrateur"
 
 
 #: dist/converse-no-dependencies.js:49282
 #: dist/converse-no-dependencies.js:49282
-#, fuzzy
 msgid "Ban user from groupchat"
 msgid "Ban user from groupchat"
 msgstr "Bannir l’utilisateur du salon"
 msgstr "Bannir l’utilisateur du salon"
 
 
@@ -719,7 +685,6 @@ msgid "Change user role to participant"
 msgstr "Changer le rôle de l’utilisateur en participant"
 msgstr "Changer le rôle de l’utilisateur en participant"
 
 
 #: dist/converse-no-dependencies.js:49282
 #: dist/converse-no-dependencies.js:49282
-#, fuzzy
 msgid "Kick user from groupchat"
 msgid "Kick user from groupchat"
 msgstr "Expulser l’utilisateur du salon"
 msgstr "Expulser l’utilisateur du salon"
 
 
@@ -744,7 +709,6 @@ msgid "Grant moderator role to user"
 msgstr "Changer le rôle de l’utilisateur en modérateur"
 msgstr "Changer le rôle de l’utilisateur en modérateur"
 
 
 #: dist/converse-no-dependencies.js:49282
 #: dist/converse-no-dependencies.js:49282
-#, fuzzy
 msgid "Grant ownership of this groupchat"
 msgid "Grant ownership of this groupchat"
 msgstr "Accorder la propriété à ce salon"
 msgstr "Accorder la propriété à ce salon"
 
 
@@ -753,14 +717,12 @@ msgid "Revoke user's membership"
 msgstr "Révoquer l’utilisateur des membres"
 msgstr "Révoquer l’utilisateur des membres"
 
 
 #: dist/converse-no-dependencies.js:49282
 #: dist/converse-no-dependencies.js:49282
-#, fuzzy
 msgid "Set groupchat subject"
 msgid "Set groupchat subject"
-msgstr "Indiquer le sujet du salon"
+msgstr "Définir le sujet du salon"
 
 
 #: dist/converse-no-dependencies.js:49282
 #: dist/converse-no-dependencies.js:49282
-#, fuzzy
 msgid "Set groupchat subject (alias for /subject)"
 msgid "Set groupchat subject (alias for /subject)"
-msgstr "Définir le sujet de la salle (alias pour /subject)"
+msgstr "Définir le sujet du salon (alias pour /subject)"
 
 
 #: dist/converse-no-dependencies.js:49282
 #: dist/converse-no-dependencies.js:49282
 msgid "Allow muted user to post messages"
 msgid "Allow muted user to post messages"
@@ -779,12 +741,10 @@ msgid "Please choose your nickname"
 msgstr "Veuillez choisir votre alias"
 msgstr "Veuillez choisir votre alias"
 
 
 #: dist/converse-no-dependencies.js:49640
 #: dist/converse-no-dependencies.js:49640
-#, fuzzy
 msgid "Enter groupchat"
 msgid "Enter groupchat"
 msgstr "Entrer dans le salon"
 msgstr "Entrer dans le salon"
 
 
 #: dist/converse-no-dependencies.js:49661
 #: dist/converse-no-dependencies.js:49661
-#, fuzzy
 msgid "This groupchat requires a password"
 msgid "This groupchat requires a password"
 msgstr "Ce salon nécessite un mot de passe"
 msgstr "Ce salon nécessite un mot de passe"
 
 
@@ -808,47 +768,45 @@ msgid "The reason given is: \"%1$s\"."
 msgstr "La raison indiquée est : « %1$s »."
 msgstr "La raison indiquée est : « %1$s »."
 
 
 #: dist/converse-no-dependencies.js:49828
 #: dist/converse-no-dependencies.js:49828
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s has left and re-entered the groupchat"
 msgid "%1$s has left and re-entered the groupchat"
 msgstr "%1$s a quitté puis rejoint le salon"
 msgstr "%1$s a quitté puis rejoint le salon"
 
 
 #: dist/converse-no-dependencies.js:49834
 #: dist/converse-no-dependencies.js:49834
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s has entered the groupchat"
 msgid "%1$s has entered the groupchat"
 msgstr "%1$s a rejoint le salon"
 msgstr "%1$s a rejoint le salon"
 
 
 #: dist/converse-no-dependencies.js:49836
 #: dist/converse-no-dependencies.js:49836
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s has entered the groupchat. \"%2$s\""
 msgid "%1$s has entered the groupchat. \"%2$s\""
 msgstr "%1$s a rejoint le salon. « %2$s »"
 msgstr "%1$s a rejoint le salon. « %2$s »"
 
 
 #: dist/converse-no-dependencies.js:49867
 #: dist/converse-no-dependencies.js:49867
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s has entered and left the groupchat"
 msgid "%1$s has entered and left the groupchat"
 msgstr "%1$s a rejoint puis quitté le salon"
 msgstr "%1$s a rejoint puis quitté le salon"
 
 
 #: dist/converse-no-dependencies.js:49869
 #: dist/converse-no-dependencies.js:49869
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s has entered and left the groupchat. \"%2$s\""
 msgid "%1$s has entered and left the groupchat. \"%2$s\""
 msgstr "%1$s a rejoint puis quitté le salon. « %2$s »"
 msgstr "%1$s a rejoint puis quitté le salon. « %2$s »"
 
 
 #: dist/converse-no-dependencies.js:49882
 #: dist/converse-no-dependencies.js:49882
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s has left the groupchat"
 msgid "%1$s has left the groupchat"
 msgstr "%1$s a quitté le salon"
 msgstr "%1$s a quitté le salon"
 
 
 #: dist/converse-no-dependencies.js:49884
 #: dist/converse-no-dependencies.js:49884
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s has left the groupchat. \"%2$s\""
 msgid "%1$s has left the groupchat. \"%2$s\""
 msgstr "%1$s a quitté le salon. « %2$s »"
 msgstr "%1$s a quitté le salon. « %2$s »"
 
 
 #: dist/converse-no-dependencies.js:49930
 #: dist/converse-no-dependencies.js:49930
-#, fuzzy
 msgid "You are not on the member list of this groupchat."
 msgid "You are not on the member list of this groupchat."
 msgstr "Vous n’êtes pas dans la liste des membres de ce salon."
 msgstr "Vous n’êtes pas dans la liste des membres de ce salon."
 
 
 #: dist/converse-no-dependencies.js:49932
 #: dist/converse-no-dependencies.js:49932
-#, fuzzy
 msgid "You have been banned from this groupchat."
 msgid "You have been banned from this groupchat."
 msgstr "Vous avez été banni de ce salon."
 msgstr "Vous avez été banni de ce salon."
 
 
@@ -857,31 +815,27 @@ msgid "No nickname was specified."
 msgstr "Aucun alias n’a été indiqué."
 msgstr "Aucun alias n’a été indiqué."
 
 
 #: dist/converse-no-dependencies.js:49940
 #: dist/converse-no-dependencies.js:49940
-#, fuzzy
 msgid "You are not allowed to create new groupchats."
 msgid "You are not allowed to create new groupchats."
 msgstr "Vous n’êtes pas autorisé à créer des salons."
 msgstr "Vous n’êtes pas autorisé à créer des salons."
 
 
 #: dist/converse-no-dependencies.js:49942
 #: dist/converse-no-dependencies.js:49942
-#, fuzzy
 msgid "Your nickname doesn't conform to this groupchat's policies."
 msgid "Your nickname doesn't conform to this groupchat's policies."
-msgstr "Votre alias n’est pas conforme à la politique de ce salon."
+msgstr "Votre pseudo n’est pas conforme à la politique de ce salon."
 
 
 #: dist/converse-no-dependencies.js:49946
 #: dist/converse-no-dependencies.js:49946
-#, fuzzy
 msgid "This groupchat does not (yet) exist."
 msgid "This groupchat does not (yet) exist."
 msgstr "Ce salon n’existe pas (pour l’instant)."
 msgstr "Ce salon n’existe pas (pour l’instant)."
 
 
 #: dist/converse-no-dependencies.js:49948
 #: dist/converse-no-dependencies.js:49948
-#, fuzzy
 msgid "This groupchat has reached its maximum number of participants."
 msgid "This groupchat has reached its maximum number of participants."
 msgstr "Ce salon a atteint sa limite maximale d’occupants."
 msgstr "Ce salon a atteint sa limite maximale d’occupants."
 
 
 #: dist/converse-no-dependencies.js:49950
 #: dist/converse-no-dependencies.js:49950
 msgid "Remote server not found"
 msgid "Remote server not found"
-msgstr ""
+msgstr "Serveur distant introuvable"
 
 
 #: dist/converse-no-dependencies.js:49955
 #: dist/converse-no-dependencies.js:49955
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "The explanation given is: \"%1$s\"."
 msgid "The explanation given is: \"%1$s\"."
 msgstr "La raison indiquée est : « %1$s »."
 msgstr "La raison indiquée est : « %1$s »."
 
 
@@ -891,17 +845,14 @@ msgid "Topic set by %1$s"
 msgstr "Le sujet a été défini par %1$s"
 msgstr "Le sujet a été défini par %1$s"
 
 
 #: dist/converse-no-dependencies.js:50031
 #: dist/converse-no-dependencies.js:50031
-#, fuzzy
 msgid "Groupchats"
 msgid "Groupchats"
-msgstr "Groupes"
+msgstr "Salons"
 
 
 #: dist/converse-no-dependencies.js:50032
 #: dist/converse-no-dependencies.js:50032
-#, fuzzy
 msgid "Add a new groupchat"
 msgid "Add a new groupchat"
 msgstr "Ajouter un nouveau salon"
 msgstr "Ajouter un nouveau salon"
 
 
 #: dist/converse-no-dependencies.js:50033
 #: dist/converse-no-dependencies.js:50033
-#, fuzzy
 msgid "Query for groupchats"
 msgid "Query for groupchats"
 msgstr "Chercher un salon"
 msgstr "Chercher un salon"
 
 
@@ -915,12 +866,10 @@ msgid "This user is a moderator."
 msgstr "Cet utilisateur est un modérateur."
 msgstr "Cet utilisateur est un modérateur."
 
 
 #: dist/converse-no-dependencies.js:50073
 #: dist/converse-no-dependencies.js:50073
-#, fuzzy
 msgid "This user can send messages in this groupchat."
 msgid "This user can send messages in this groupchat."
 msgstr "Cet utilisateur peut envoyer des messages dans ce salon."
 msgstr "Cet utilisateur peut envoyer des messages dans ce salon."
 
 
 #: dist/converse-no-dependencies.js:50074
 #: dist/converse-no-dependencies.js:50074
-#, fuzzy
 msgid "This user can NOT send messages in this groupchat."
 msgid "This user can NOT send messages in this groupchat."
 msgstr "Cet utilisateur ne peut PAS envoyer de messages dans ce salon."
 msgstr "Cet utilisateur ne peut PAS envoyer de messages dans ce salon."
 
 
@@ -946,7 +895,7 @@ msgstr "Administrateur"
 
 
 #: dist/converse-no-dependencies.js:50121
 #: dist/converse-no-dependencies.js:50121
 msgid "Participants"
 msgid "Participants"
-msgstr ""
+msgstr "Participants"
 
 
 #: dist/converse-no-dependencies.js:50138
 #: dist/converse-no-dependencies.js:50138
 #: dist/converse-no-dependencies.js:50219
 #: dist/converse-no-dependencies.js:50219
@@ -954,31 +903,31 @@ msgid "Invite"
 msgstr "Inviter"
 msgstr "Inviter"
 
 
 #: dist/converse-no-dependencies.js:50196
 #: dist/converse-no-dependencies.js:50196
-#, fuzzy, javascript-format
+#, javascript-format
 msgid ""
 msgid ""
 "You are about to invite %1$s to the groupchat \"%2$s\". You may optionally "
 "You are about to invite %1$s to the groupchat \"%2$s\". You may optionally "
 "include a message, explaining the reason for the invitation."
 "include a message, explaining the reason for the invitation."
 msgstr ""
 msgstr ""
 "Vous allez inviter %1$s dans le salon %2$s. Vous pouvez facultativement "
 "Vous allez inviter %1$s dans le salon %2$s. Vous pouvez facultativement "
-"ajouter un message, expliquant la raison de cette invitation."
+"ajouter un message expliquant la raison de cette invitation."
 
 
 #: dist/converse-no-dependencies.js:50218
 #: dist/converse-no-dependencies.js:50218
 msgid "Please enter a valid XMPP username"
 msgid "Please enter a valid XMPP username"
 msgstr "Veuillez saisir un identifiant utilisateur XMPP valide"
 msgstr "Veuillez saisir un identifiant utilisateur XMPP valide"
 
 
 #: dist/converse-no-dependencies.js:51591
 #: dist/converse-no-dependencies.js:51591
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "%1$s has invited you to join a groupchat: %2$s"
 msgid "%1$s has invited you to join a groupchat: %2$s"
 msgstr "%1$s vous invite à rejoindre le salon : %2$s"
 msgstr "%1$s vous invite à rejoindre le salon : %2$s"
 
 
 #: dist/converse-no-dependencies.js:51593
 #: dist/converse-no-dependencies.js:51593
-#, fuzzy, javascript-format
+#, javascript-format
 msgid ""
 msgid ""
 "%1$s has invited you to join a groupchat: %2$s, and left the following "
 "%1$s has invited you to join a groupchat: %2$s, and left the following "
 "reason: \"%3$s\""
 "reason: \"%3$s\""
 msgstr ""
 msgstr ""
-"%1$s vous invite à rejoindre le salon : %2$s, avec le message suivant: "
-"« %3$s »"
+"%1$s vous invite à rejoindre le salon : %2$s, avec le message suivant: « %3$"
+"s »"
 
 
 #. workaround for Prosody which doesn't give type "headline"
 #. workaround for Prosody which doesn't give type "headline"
 #: dist/converse-no-dependencies.js:51974
 #: dist/converse-no-dependencies.js:51974
@@ -999,9 +948,8 @@ msgid "has gone offline"
 msgstr "s’est déconnecté"
 msgstr "s’est déconnecté"
 
 
 #: dist/converse-no-dependencies.js:52026
 #: dist/converse-no-dependencies.js:52026
-#, fuzzy
 msgid "has gone away"
 msgid "has gone away"
-msgstr "s’est déconnecté"
+msgstr "est absent"
 
 
 #: dist/converse-no-dependencies.js:52028
 #: dist/converse-no-dependencies.js:52028
 msgid "is busy"
 msgid "is busy"
@@ -1018,7 +966,7 @@ msgstr "veut être votre contact"
 #: dist/converse-no-dependencies.js:52229
 #: dist/converse-no-dependencies.js:52229
 #, javascript-format
 #, javascript-format
 msgid "Log in with %1$s"
 msgid "Log in with %1$s"
-msgstr ""
+msgstr "Se connecter avec %1$s"
 
 
 #: dist/converse-no-dependencies.js:52476
 #: dist/converse-no-dependencies.js:52476
 msgid "Your Profile"
 msgid "Your Profile"
@@ -1187,16 +1135,15 @@ msgstr ""
 "les données que vous avez fournies sont correctes."
 "les données que vous avez fournies sont correctes."
 
 
 #: dist/converse-no-dependencies.js:53748
 #: dist/converse-no-dependencies.js:53748
-#, fuzzy
 msgid "Click to toggle the list of open groupchats"
 msgid "Click to toggle the list of open groupchats"
-msgstr "Cliquer pour ouvrir la liste des salons"
+msgstr "Cliquer pour ouvrir la liste des salons ouverts"
 
 
 #: dist/converse-no-dependencies.js:53749
 #: dist/converse-no-dependencies.js:53749
 msgid "Open Groupchats"
 msgid "Open Groupchats"
-msgstr ""
+msgstr "Ouvrir les salons"
 
 
 #: dist/converse-no-dependencies.js:53793
 #: dist/converse-no-dependencies.js:53793
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "Are you sure you want to leave the groupchat %1$s?"
 msgid "Are you sure you want to leave the groupchat %1$s?"
 msgstr "Voulez-vous vraiment quitter le salon « %1$s » ?"
 msgstr "Voulez-vous vraiment quitter le salon « %1$s » ?"
 
 
@@ -1345,32 +1292,28 @@ msgid "Add a contact"
 msgstr "Ajouter un contact"
 msgstr "Ajouter un contact"
 
 
 #: dist/converse-no-dependencies.js:57111
 #: dist/converse-no-dependencies.js:57111
-#, fuzzy
 msgid "Name"
 msgid "Name"
-msgstr "Nom complet"
+msgstr "Nom"
 
 
 #: dist/converse-no-dependencies.js:57115
 #: dist/converse-no-dependencies.js:57115
-#, fuzzy
 msgid "Groupchat address (JID)"
 msgid "Groupchat address (JID)"
 msgstr "Adresse du salon (JID) :"
 msgstr "Adresse du salon (JID) :"
 
 
 #: dist/converse-no-dependencies.js:57119
 #: dist/converse-no-dependencies.js:57119
-#, fuzzy
 msgid "Description"
 msgid "Description"
-msgstr "Description :"
+msgstr "Description"
 
 
 #: dist/converse-no-dependencies.js:57125
 #: dist/converse-no-dependencies.js:57125
 msgid "Topic"
 msgid "Topic"
-msgstr ""
+msgstr "Sujet"
 
 
 #: dist/converse-no-dependencies.js:57129
 #: dist/converse-no-dependencies.js:57129
 msgid "Topic author"
 msgid "Topic author"
-msgstr ""
+msgstr "Auteur du sujet"
 
 
 #: dist/converse-no-dependencies.js:57135
 #: dist/converse-no-dependencies.js:57135
-#, fuzzy
 msgid "Online users"
 msgid "Online users"
-msgstr "En ligne"
+msgstr "Utilisateurs en ligne"
 
 
 #: dist/converse-no-dependencies.js:57139
 #: dist/converse-no-dependencies.js:57139
 #: dist/converse-no-dependencies.js:57291
 #: dist/converse-no-dependencies.js:57291
@@ -1384,30 +1327,25 @@ msgstr "Protégé par mot de passe"
 
 
 #: dist/converse-no-dependencies.js:57145
 #: dist/converse-no-dependencies.js:57145
 #: dist/converse-no-dependencies.js:57297
 #: dist/converse-no-dependencies.js:57297
-#, fuzzy
 msgid "This groupchat requires a password before entry"
 msgid "This groupchat requires a password before entry"
 msgstr "Ce salon nécessite un mot de passe pour y accéder"
 msgstr "Ce salon nécessite un mot de passe pour y accéder"
 
 
 #: dist/converse-no-dependencies.js:57151
 #: dist/converse-no-dependencies.js:57151
-#, fuzzy
 msgid "No password required"
 msgid "No password required"
-msgstr "Pas de mot de passe"
+msgstr "Pas de mot de passe nécessaire"
 
 
 #: dist/converse-no-dependencies.js:57153
 #: dist/converse-no-dependencies.js:57153
 #: dist/converse-no-dependencies.js:57305
 #: dist/converse-no-dependencies.js:57305
-#, fuzzy
 msgid "This groupchat does not require a password upon entry"
 msgid "This groupchat does not require a password upon entry"
 msgstr "Ce salon ne nécessite pas de mot de passe pour y accéder"
 msgstr "Ce salon ne nécessite pas de mot de passe pour y accéder"
 
 
 #: dist/converse-no-dependencies.js:57161
 #: dist/converse-no-dependencies.js:57161
 #: dist/converse-no-dependencies.js:57313
 #: dist/converse-no-dependencies.js:57313
-#, fuzzy
 msgid "This groupchat is not publicly searchable"
 msgid "This groupchat is not publicly searchable"
 msgstr "Ce salon ne peut pas être recherché publiquement"
 msgstr "Ce salon ne peut pas être recherché publiquement"
 
 
 #: dist/converse-no-dependencies.js:57169
 #: dist/converse-no-dependencies.js:57169
 #: dist/converse-no-dependencies.js:57321
 #: dist/converse-no-dependencies.js:57321
-#, fuzzy
 msgid "This groupchat is publicly searchable"
 msgid "This groupchat is publicly searchable"
 msgstr "Ce salon peut être recherché publiquement"
 msgstr "Ce salon peut être recherché publiquement"
 
 
@@ -1417,13 +1355,11 @@ msgid "Members only"
 msgstr "Membres uniquement"
 msgstr "Membres uniquement"
 
 
 #: dist/converse-no-dependencies.js:57177
 #: dist/converse-no-dependencies.js:57177
-#, fuzzy
 msgid "This groupchat is restricted to members only"
 msgid "This groupchat is restricted to members only"
 msgstr "Ce salon est restreint aux membres uniquement"
 msgstr "Ce salon est restreint aux membres uniquement"
 
 
 #: dist/converse-no-dependencies.js:57185
 #: dist/converse-no-dependencies.js:57185
 #: dist/converse-no-dependencies.js:57337
 #: dist/converse-no-dependencies.js:57337
-#, fuzzy
 msgid "Anyone can join this groupchat"
 msgid "Anyone can join this groupchat"
 msgstr "N’importe qui peut rejoindre ce salon"
 msgstr "N’importe qui peut rejoindre ce salon"
 
 
@@ -1434,25 +1370,21 @@ msgstr "Persistant"
 
 
 #: dist/converse-no-dependencies.js:57193
 #: dist/converse-no-dependencies.js:57193
 #: dist/converse-no-dependencies.js:57345
 #: dist/converse-no-dependencies.js:57345
-#, fuzzy
 msgid "This groupchat persists even if it's unoccupied"
 msgid "This groupchat persists even if it's unoccupied"
 msgstr "Ce salon persiste même s'il est inoccupé"
 msgstr "Ce salon persiste même s'il est inoccupé"
 
 
 #: dist/converse-no-dependencies.js:57201
 #: dist/converse-no-dependencies.js:57201
 #: dist/converse-no-dependencies.js:57353
 #: dist/converse-no-dependencies.js:57353
-#, fuzzy
 msgid "This groupchat will disappear once the last person leaves"
 msgid "This groupchat will disappear once the last person leaves"
 msgstr "Ce salon disparaîtra au départ de la dernière personne"
 msgstr "Ce salon disparaîtra au départ de la dernière personne"
 
 
 #: dist/converse-no-dependencies.js:57207
 #: dist/converse-no-dependencies.js:57207
 #: dist/converse-no-dependencies.js:57363
 #: dist/converse-no-dependencies.js:57363
-#, fuzzy
 msgid "Not anonymous"
 msgid "Not anonymous"
 msgstr "Non-anonyme"
 msgstr "Non-anonyme"
 
 
 #: dist/converse-no-dependencies.js:57209
 #: dist/converse-no-dependencies.js:57209
 #: dist/converse-no-dependencies.js:57361
 #: dist/converse-no-dependencies.js:57361
-#, fuzzy
 msgid "All other groupchat participants can see your XMPP username"
 msgid "All other groupchat participants can see your XMPP username"
 msgstr ""
 msgstr ""
 "Tous les autres occupants de ce salon peuvent voir votre nom d’utilisateur "
 "Tous les autres occupants de ce salon peuvent voir votre nom d’utilisateur "
@@ -1465,19 +1397,16 @@ msgstr "Seuls les modérateurs peuvent voir votre identifiant XMPP"
 
 
 #: dist/converse-no-dependencies.js:57225
 #: dist/converse-no-dependencies.js:57225
 #: dist/converse-no-dependencies.js:57377
 #: dist/converse-no-dependencies.js:57377
-#, fuzzy
 msgid "This groupchat is being moderated"
 msgid "This groupchat is being moderated"
 msgstr "Ce salon est modéré"
 msgstr "Ce salon est modéré"
 
 
 #: dist/converse-no-dependencies.js:57231
 #: dist/converse-no-dependencies.js:57231
 #: dist/converse-no-dependencies.js:57387
 #: dist/converse-no-dependencies.js:57387
-#, fuzzy
 msgid "Not moderated"
 msgid "Not moderated"
 msgstr "Non modéré"
 msgstr "Non modéré"
 
 
 #: dist/converse-no-dependencies.js:57233
 #: dist/converse-no-dependencies.js:57233
 #: dist/converse-no-dependencies.js:57385
 #: dist/converse-no-dependencies.js:57385
-#, fuzzy
 msgid "This groupchat is not being moderated"
 msgid "This groupchat is not being moderated"
 msgstr "Ce salon n’est pas modéré"
 msgstr "Ce salon n’est pas modéré"
 
 
@@ -1496,9 +1425,8 @@ msgid "No password"
 msgstr "Pas de mot de passe"
 msgstr "Pas de mot de passe"
 
 
 #: dist/converse-no-dependencies.js:57329
 #: dist/converse-no-dependencies.js:57329
-#, fuzzy
 msgid "this groupchat is restricted to members only"
 msgid "this groupchat is restricted to members only"
-msgstr "Ce salon est restreint aux membres uniquement"
+msgstr "ce salon est restreint aux membres uniquement"
 
 
 #: dist/converse-no-dependencies.js:58267
 #: dist/converse-no-dependencies.js:58267
 msgid "XMPP Username:"
 msgid "XMPP Username:"
@@ -1514,7 +1442,7 @@ msgstr "Mot de passe"
 
 
 #: dist/converse-no-dependencies.js:58283
 #: dist/converse-no-dependencies.js:58283
 msgid "This is a trusted device"
 msgid "This is a trusted device"
-msgstr ""
+msgstr "Ceci est un appareil de confiance"
 
 
 #: dist/converse-no-dependencies.js:58285
 #: dist/converse-no-dependencies.js:58285
 msgid ""
 msgid ""
@@ -1523,9 +1451,13 @@ msgid ""
 "log out. It's important that you explicitly log out, otherwise not all "
 "log out. It's important that you explicitly log out, otherwise not all "
 "cached data might be deleted."
 "cached data might be deleted."
 msgstr ""
 msgstr ""
+"Pour améliorer les performances, nous stockons vos données dans ce "
+"navigateur. Décochez ce bouton si vous êtes sur un ordinateur public, ou si "
+"vous voulez que vos données soient supprimées lorsque vous vous "
+"déconnecterez. Il est important que vous vous déconnectiez explicitement, "
+"sinon toutes les données stockées ne seront pas forcément supprimées."
 
 
 #: dist/converse-no-dependencies.js:58287
 #: dist/converse-no-dependencies.js:58287
-#, fuzzy
 msgid "Log in"
 msgid "Log in"
 msgstr "Se connecter"
 msgstr "Se connecter"
 
 
@@ -1534,19 +1466,16 @@ msgid "Click here to log in anonymously"
 msgstr "Cliquez ici pour se connecter anonymement"
 msgstr "Cliquez ici pour se connecter anonymement"
 
 
 #: dist/converse-no-dependencies.js:58376
 #: dist/converse-no-dependencies.js:58376
-#, fuzzy
 msgid "This message has been edited"
 msgid "This message has been edited"
-msgstr "Ce salon est modéré"
+msgstr "Ce message a été édité"
 
 
 #: dist/converse-no-dependencies.js:58402
 #: dist/converse-no-dependencies.js:58402
-#, fuzzy
 msgid "Edit this message"
 msgid "Edit this message"
-msgstr "Cacher le message caché"
+msgstr "Éditer ce message"
 
 
 #: dist/converse-no-dependencies.js:58427
 #: dist/converse-no-dependencies.js:58427
-#, fuzzy
 msgid "Message versions"
 msgid "Message versions"
-msgstr "Archivage des messages"
+msgstr "Versions du message"
 
 
 #: dist/converse-no-dependencies.js:58759
 #: dist/converse-no-dependencies.js:58759
 msgid "Don't have a chat account?"
 msgid "Don't have a chat account?"
@@ -1594,9 +1523,9 @@ msgid "Download"
 msgstr "Télécharger"
 msgstr "Télécharger"
 
 
 #: dist/converse-no-dependencies.js:59996
 #: dist/converse-no-dependencies.js:59996
-#, fuzzy, javascript-format
+#, javascript-format
 msgid "Download \"%1$s\""
 msgid "Download \"%1$s\""
-msgstr "Télécharger : \"%1$s\""
+msgstr "Télécharger « %1$s »"
 
 
 #: dist/converse-no-dependencies.js:60019
 #: dist/converse-no-dependencies.js:60019
 msgid "Download video file"
 msgid "Download video file"

+ 3 - 3
package-lock.json

@@ -2410,9 +2410,9 @@
       "dev": true
       "dev": true
     },
     },
     "bootstrap.native": {
     "bootstrap.native": {
-      "version": "2.0.22",
-      "resolved": "https://registry.npmjs.org/bootstrap.native/-/bootstrap.native-2.0.22.tgz",
-      "integrity": "sha512-eypi4y1eKJoRt8cTwkZCI3QQ7W04rbv4VU1nBjBshqNXkONI7jO6tG3qZTwq9Zd+gDoeaQASyk6V185y+Y7KHQ==",
+      "version": "2.0.23",
+      "resolved": "https://registry.npmjs.org/bootstrap.native/-/bootstrap.native-2.0.23.tgz",
+      "integrity": "sha512-bcbVgqIjRkyiHd6DN8Y18BYjJTKvvnN3Msb7Yh6K5vGGsjRT3gV0IFKR3rcEYgJ5Kvg1egL9exIfN/Hvwn4wNA==",
       "dev": true
       "dev": true
     },
     },
     "bourbon": {
     "bourbon": {

+ 1 - 1
package.json

@@ -44,7 +44,7 @@
     "backbone.overview": "1.0.2",
     "backbone.overview": "1.0.2",
     "backbone.vdomview": "1.0.1",
     "backbone.vdomview": "1.0.1",
     "bootstrap": "^4.0.0",
     "bootstrap": "^4.0.0",
-    "bootstrap.native": "^2.0.22",
+    "bootstrap.native": "^2.0.23",
     "bourbon": "^4.3.2",
     "bourbon": "^4.3.2",
     "bytebuffer": "^3.5.5",
     "bytebuffer": "^3.5.5",
     "clean-css-cli": "^4.0.10",
     "clean-css-cli": "^4.0.10",

+ 1 - 1
requirements.txt

@@ -1 +1 @@
-zc.buildout==2.11.4
+zc.buildout==2.12.1

+ 93 - 97
spec/bookmarks.js

@@ -34,24 +34,24 @@
                 });
                 });
                 spyOn(_converse.connection, 'getUniqueId').and.callThrough();
                 spyOn(_converse.connection, 'getUniqueId').and.callThrough();
 
 
-                test_utils.openChatRoom(_converse, 'theplay', 'conference.shakespeare.lit', 'JC');
-                var jid = 'theplay@conference.shakespeare.lit';
-                var view = _converse.chatboxviews.get(jid);
-                spyOn(view, 'renderBookmarkForm').and.callThrough();
-                spyOn(view, 'closeForm').and.callThrough();
-
-                test_utils.waitUntil(function () {
-                    return !_.isNull(view.el.querySelector('.toggle-bookmark'));
-                }, 300).then(function () {
-                    var $bookmark = $(view.el).find('.toggle-bookmark');
-                    $bookmark[0].click();
+                let view;
+                test_utils.openChatRoom(_converse, 'theplay', 'conference.shakespeare.lit', 'JC')
+                .then(() => {
+                    var jid = 'theplay@conference.shakespeare.lit';
+                    view = _converse.chatboxviews.get(jid);
+                    spyOn(view, 'renderBookmarkForm').and.callThrough();
+                    spyOn(view, 'closeForm').and.callThrough();
+                    return test_utils.waitUntil(() => !_.isNull(view.el.querySelector('.toggle-bookmark')));
+                }).then(() => {
+                    var bookmark = view.el.querySelector('.toggle-bookmark');
+                    bookmark.click();
                     expect(view.renderBookmarkForm).toHaveBeenCalled();
                     expect(view.renderBookmarkForm).toHaveBeenCalled();
 
 
                     view.el.querySelector('.button-cancel').click();
                     view.el.querySelector('.button-cancel').click();
                     expect(view.closeForm).toHaveBeenCalled();
                     expect(view.closeForm).toHaveBeenCalled();
-                    expect($bookmark.hasClass('on-button'), false);
+                    expect(u.hasClass('on-button', bookmark), false);
 
 
-                    $bookmark[0].click();
+                    bookmark.click();
                     expect(view.renderBookmarkForm).toHaveBeenCalled();
                     expect(view.renderBookmarkForm).toHaveBeenCalled();
 
 
                     /* Client uploads data:
                     /* Client uploads data:
@@ -93,7 +93,7 @@
                     view.el.querySelector('.btn-primary').click();
                     view.el.querySelector('.btn-primary').click();
 
 
                     expect(view.model.get('bookmarked')).toBeTruthy();
                     expect(view.model.get('bookmarked')).toBeTruthy();
-                    expect($bookmark.hasClass('on-button'), true);
+                    expect(u.hasClass('on-button', bookmark), true);
 
 
                     expect(sent_stanza.toLocaleString()).toBe(
                     expect(sent_stanza.toLocaleString()).toBe(
                         "<iq type='set' from='dummy@localhost/resource' xmlns='jabber:client' id='"+IQ_id+"'>"+
                         "<iq type='set' from='dummy@localhost/resource' xmlns='jabber:client' id='"+IQ_id+"'>"+
@@ -175,100 +175,96 @@
             it("displays that it's bookmarked through its bookmark icon", mock.initConverseWithPromises(
             it("displays that it's bookmarked through its bookmark icon", mock.initConverseWithPromises(
                 null, ['rosterGroupsFetched'], {}, function (done, _converse) {
                 null, ['rosterGroupsFetched'], {}, function (done, _converse) {
 
 
+                let view;
                 test_utils.waitUntilDiscoConfirmed(
                 test_utils.waitUntilDiscoConfirmed(
                     _converse, _converse.bare_jid,
                     _converse, _converse.bare_jid,
                     [{'category': 'pubsub', 'type': 'pep'}],
                     [{'category': 'pubsub', 'type': 'pep'}],
                     ['http://jabber.org/protocol/pubsub#publish-options']
                     ['http://jabber.org/protocol/pubsub#publish-options']
-                ).then(function () {
-                    test_utils.openChatRoom(_converse, 'lounge', 'localhost', 'dummy');
-                    var view = _converse.chatboxviews.get('lounge@localhost');
-
-                    test_utils.waitUntil(function () {
-                        return !_.isNull(view.el.querySelector('.toggle-bookmark'));
-                    }, 300).then(function () {
-                        var bookmark_icon = view.el.querySelector('.toggle-bookmark');
-                        expect(_.includes(bookmark_icon.classList, 'button-on')).toBeFalsy();
-                        view.model.set('bookmarked', true);
-                        expect(_.includes(bookmark_icon.classList, 'button-on')).toBeTruthy();
-                        view.model.set('bookmarked', false);
-                        expect(_.includes(bookmark_icon.classList, 'button-on')).toBeFalsy();
-                        done();
-                    });
+                ).then(() => test_utils.openChatRoom(_converse, 'lounge', 'localhost', 'dummy'))
+                .then(() => {
+                    view = _converse.chatboxviews.get('lounge@localhost');
+                    return test_utils.waitUntil(() => !_.isNull(view.el.querySelector('.toggle-bookmark')))
+                }).then(function () {
+                    var bookmark_icon = view.el.querySelector('.toggle-bookmark');
+                    expect(_.includes(bookmark_icon.classList, 'button-on')).toBeFalsy();
+                    view.model.set('bookmarked', true);
+                    expect(_.includes(bookmark_icon.classList, 'button-on')).toBeTruthy();
+                    view.model.set('bookmarked', false);
+                    expect(_.includes(bookmark_icon.classList, 'button-on')).toBeFalsy();
+                    done();
                 });
                 });
             }));
             }));
 
 
             it("can be unbookmarked", mock.initConverseWithPromises(
             it("can be unbookmarked", mock.initConverseWithPromises(
                 null, ['rosterGroupsFetched'], {}, function (done, _converse) {
                 null, ['rosterGroupsFetched'], {}, function (done, _converse) {
 
 
+                let sent_stanza, IQ_id, view, sendIQ;
+
                 test_utils.waitUntilDiscoConfirmed(
                 test_utils.waitUntilDiscoConfirmed(
                     _converse, _converse.bare_jid,
                     _converse, _converse.bare_jid,
                     [{'category': 'pubsub', 'type': 'pep'}],
                     [{'category': 'pubsub', 'type': 'pep'}],
                     ['http://jabber.org/protocol/pubsub#publish-options']
                     ['http://jabber.org/protocol/pubsub#publish-options']
-                ).then(function () {
-                    var sent_stanza, IQ_id;
-                    var sendIQ = _converse.connection.sendIQ;
-
-                    test_utils.openChatRoom(_converse, 'theplay', 'conference.shakespeare.lit', 'JC');
+                ).then(() => {
+                    sendIQ = _converse.connection.sendIQ;
+                    return test_utils.openChatRoom(_converse, 'theplay', 'conference.shakespeare.lit', 'JC');
+                }).then(() => {
                     var jid = 'theplay@conference.shakespeare.lit';
                     var jid = 'theplay@conference.shakespeare.lit';
-                    var view = _converse.chatboxviews.get(jid);
+                    view = _converse.chatboxviews.get(jid);
+                    return test_utils.waitUntil(() => !_.isNull(view.el.querySelector('.toggle-bookmark')));
+                }).then(function () {
+                    spyOn(view, 'toggleBookmark').and.callThrough();
+                    spyOn(_converse.bookmarks, 'sendBookmarkStanza').and.callThrough();
+                    view.delegateEvents();
 
 
-                    test_utils.waitUntil(function () {
-                        return !_.isNull(view.el.querySelector('.toggle-bookmark'));
-                    }, 300).then(function () {
-                        spyOn(view, 'toggleBookmark').and.callThrough();
-                        spyOn(_converse.bookmarks, 'sendBookmarkStanza').and.callThrough();
-                        view.delegateEvents();
-
-                        _converse.bookmarks.create({
-                            'jid': view.model.get('jid'),
-                            'autojoin': false,
-                            'name':  'The Play',
-                            'nick': ' Othello'
-                        });
-                        expect(_converse.bookmarks.length).toBe(1);
-                        expect(view.model.get('bookmarked')).toBeTruthy();
-                        var $bookmark_icon = $(view.el.querySelector('.toggle-bookmark'));
-                        expect($bookmark_icon.hasClass('button-on')).toBeTruthy();
-
-                        spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
-                            sent_stanza = iq;
-                            IQ_id = sendIQ.bind(this)(iq, callback, errback);
-                        });
-                        spyOn(_converse.connection, 'getUniqueId').and.callThrough();
-                        $bookmark_icon[0].click();
-                        expect(view.toggleBookmark).toHaveBeenCalled();
-                        expect($bookmark_icon.hasClass('button-on')).toBeFalsy();
-                        expect(_converse.bookmarks.length).toBe(0);
-
-                        // Check that an IQ stanza is sent out, containing no
-                        // conferences to bookmark (since we removed the one and
-                        // only bookmark).
-                        expect(sent_stanza.toLocaleString()).toBe(
-                            "<iq type='set' from='dummy@localhost/resource' xmlns='jabber:client' id='"+IQ_id+"'>"+
-                                "<pubsub xmlns='http://jabber.org/protocol/pubsub'>"+
-                                    "<publish node='storage:bookmarks'>"+
-                                        "<item id='current'>"+
-                                            "<storage xmlns='storage:bookmarks'/>"+
-                                        "</item>"+
-                                    "</publish>"+
-                                    "<publish-options>"+
-                                        "<x xmlns='jabber:x:data' type='submit'>"+
-                                            "<field var='FORM_TYPE' type='hidden'>"+
-                                                "<value>http://jabber.org/protocol/pubsub#publish-options</value>"+
-                                            "</field>"+
-                                            "<field var='pubsub#persist_items'>"+
-                                                "<value>true</value>"+
-                                            "</field>"+
-                                            "<field var='pubsub#access_model'>"+
-                                                "<value>whitelist</value>"+
-                                            "</field>"+
-                                        "</x>"+
-                                    "</publish-options>"+
-                                "</pubsub>"+
-                            "</iq>"
-                        );
-                        done();
+                    _converse.bookmarks.create({
+                        'jid': view.model.get('jid'),
+                        'autojoin': false,
+                        'name':  'The Play',
+                        'nick': ' Othello'
+                    });
+                    expect(_converse.bookmarks.length).toBe(1);
+                    expect(view.model.get('bookmarked')).toBeTruthy();
+                    var bookmark_icon = view.el.querySelector('.toggle-bookmark');
+                    expect(u.hasClass('button-on', bookmark_icon)).toBeTruthy();
+
+                    spyOn(_converse.connection, 'sendIQ').and.callFake(function (iq, callback, errback) {
+                        sent_stanza = iq;
+                        IQ_id = sendIQ.bind(this)(iq, callback, errback);
                     });
                     });
+                    spyOn(_converse.connection, 'getUniqueId').and.callThrough();
+                    bookmark_icon.click();
+                    expect(view.toggleBookmark).toHaveBeenCalled();
+                    expect(u.hasClass('button-on', bookmark_icon)).toBeFalsy();
+                    expect(_converse.bookmarks.length).toBe(0);
+
+                    // Check that an IQ stanza is sent out, containing no
+                    // conferences to bookmark (since we removed the one and
+                    // only bookmark).
+                    expect(sent_stanza.toLocaleString()).toBe(
+                        "<iq type='set' from='dummy@localhost/resource' xmlns='jabber:client' id='"+IQ_id+"'>"+
+                            "<pubsub xmlns='http://jabber.org/protocol/pubsub'>"+
+                                "<publish node='storage:bookmarks'>"+
+                                    "<item id='current'>"+
+                                        "<storage xmlns='storage:bookmarks'/>"+
+                                    "</item>"+
+                                "</publish>"+
+                                "<publish-options>"+
+                                    "<x xmlns='jabber:x:data' type='submit'>"+
+                                        "<field var='FORM_TYPE' type='hidden'>"+
+                                            "<value>http://jabber.org/protocol/pubsub#publish-options</value>"+
+                                        "</field>"+
+                                        "<field var='pubsub#persist_items'>"+
+                                            "<value>true</value>"+
+                                        "</field>"+
+                                        "<field var='pubsub#access_model'>"+
+                                            "<value>whitelist</value>"+
+                                        "</field>"+
+                                    "</x>"+
+                                "</publish-options>"+
+                            "</pubsub>"+
+                        "</iq>"
+                    );
+                    done();
                 });
                 });
             }));
             }));
         });
         });
@@ -587,9 +583,8 @@
                         'name':  'The Play',
                         'name':  'The Play',
                         'nick': ''
                         'nick': ''
                     });
                     });
-                    test_utils.waitUntil(function () {
-                        return $('#chatrooms .bookmarks.rooms-list .room-item:visible').length;
-                    }, 300).then(function () {
+                    test_utils.waitUntil(() => $('#chatrooms .bookmarks.rooms-list .room-item:visible').length
+                    ).then(function () {
                         expect($('#chatrooms .bookmarks.rooms-list').hasClass('collapsed')).toBeFalsy();
                         expect($('#chatrooms .bookmarks.rooms-list').hasClass('collapsed')).toBeFalsy();
                         expect($('#chatrooms .bookmarks.rooms-list .room-item:visible').length).toBe(1);
                         expect($('#chatrooms .bookmarks.rooms-list .room-item:visible').length).toBe(1);
                         expect(_converse.bookmarksview.list_model.get('toggle-state')).toBe(_converse.OPENED);
                         expect(_converse.bookmarksview.list_model.get('toggle-state')).toBe(_converse.OPENED);
@@ -614,6 +609,7 @@
             { hide_open_bookmarks: true },
             { hide_open_bookmarks: true },
             function (done, _converse) {
             function (done, _converse) {
 
 
+            const jid = 'room@conference.example.org';
             test_utils.waitUntilDiscoConfirmed(
             test_utils.waitUntilDiscoConfirmed(
                 _converse, _converse.bare_jid,
                 _converse, _converse.bare_jid,
                 [{'category': 'pubsub', 'type': 'pep'}],
                 [{'category': 'pubsub', 'type': 'pep'}],
@@ -627,14 +623,12 @@
                 _converse.emit('bookmarksInitialized');
                 _converse.emit('bookmarksInitialized');
 
 
                 // Check that it's there
                 // Check that it's there
-                var jid = 'room@conference.example.org';
                 _converse.bookmarks.create({
                 _converse.bookmarks.create({
                     'jid': jid,
                     'jid': jid,
                     'autojoin': false,
                     'autojoin': false,
                     'name':  'The Play',
                     'name':  'The Play',
                     'nick': ' Othello'
                     'nick': ' Othello'
                 });
                 });
-
                 expect(_converse.bookmarks.length).toBe(1);
                 expect(_converse.bookmarks.length).toBe(1);
                 var room_els = _converse.bookmarksview.el.querySelectorAll(".open-room");
                 var room_els = _converse.bookmarksview.el.querySelectorAll(".open-room");
                 expect(room_els.length).toBe(1);
                 expect(room_els.length).toBe(1);
@@ -642,9 +636,11 @@
                 // Check that it disappears once the room is opened
                 // Check that it disappears once the room is opened
                 var bookmark = _converse.bookmarksview.el.querySelector(".open-room");
                 var bookmark = _converse.bookmarksview.el.querySelector(".open-room");
                 bookmark.click();
                 bookmark.click();
+                return test_utils.waitUntil(() => _converse.chatboxviews.get(jid));
+            }).then(() => {
                 expect(u.hasClass('hidden', _converse.bookmarksview.el.querySelector(".available-chatroom"))).toBeTruthy();
                 expect(u.hasClass('hidden', _converse.bookmarksview.el.querySelector(".available-chatroom"))).toBeTruthy();
                 // Check that it reappears once the room is closed
                 // Check that it reappears once the room is closed
-                var view = _converse.chatboxviews.get(jid);
+                const view = _converse.chatboxviews.get(jid);
                 view.close();
                 view.close();
                 expect(u.hasClass('hidden', _converse.bookmarksview.el.querySelector(".available-chatroom"))).toBeFalsy();
                 expect(u.hasClass('hidden', _converse.bookmarksview.el.querySelector(".available-chatroom"))).toBeFalsy();
                 done();
                 done();

File diff suppressed because it is too large
+ 337 - 315
spec/chatbox.js


File diff suppressed because it is too large
+ 416 - 446
spec/chatroom.js


+ 38 - 35
spec/controlbox.js

@@ -66,47 +66,50 @@
 
 
             it("shows the number of unread mentions received",
             it("shows the number of unread mentions received",
                 mock.initConverseWithPromises(
                 mock.initConverseWithPromises(
-                    null, ['rosterGroupsFetched'], {},
+                    null, ['rosterGroupsFetched', 'chatBoxesFetched'], {},
                     function (done, _converse) {
                     function (done, _converse) {
 
 
                 test_utils.createContacts(_converse, 'all').openControlBox();
                 test_utils.createContacts(_converse, 'all').openControlBox();
+                _converse.emit('rosterContactsFetched');
 
 
-                var contacts_panel = _converse.chatboxviews.get('controlbox').contactspanel;
+                const sender_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
 
 
-                var sender_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
                 test_utils.openChatBoxFor(_converse, sender_jid);
                 test_utils.openChatBoxFor(_converse, sender_jid);
-                var chatview = _converse.chatboxviews.get(sender_jid);
-                chatview.model.set({'minimized': true});
-
-                expect(_.isNull(_converse.chatboxviews.el.querySelector('.restore-chat .message-count'))).toBeTruthy();
-                expect(_.isNull(_converse.rosterview.el.querySelector('.msgs-indicator'))).toBeTruthy();
-
-                var msg = $msg({
-                        from: sender_jid,
-                        to: _converse.connection.jid,
-                        type: 'chat',
-                        id: (new Date()).getTime()
-                    }).c('body').t('hello').up()
-                    .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree();
-                _converse.chatboxes.onMessage(msg);
-                expect(_converse.chatboxviews.el.querySelector('.restore-chat .message-count').textContent).toBe('1');
-                expect(_converse.rosterview.el.querySelector('.msgs-indicator').textContent).toBe('1');
-
-                msg = $msg({
-                        from: sender_jid,
-                        to: _converse.connection.jid,
-                        type: 'chat',
-                        id: (new Date()).getTime()
-                    }).c('body').t('hello again').up()
-                    .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree();
-                _converse.chatboxes.onMessage(msg);
-                expect(_converse.chatboxviews.el.querySelector('.restore-chat .message-count').textContent).toBe('2');
-                expect(_converse.rosterview.el.querySelector('.msgs-indicator').textContent).toBe('2');
-
-                chatview.model.set({'minimized': false});
-                expect(_.isNull(_converse.chatboxviews.el.querySelector('.restore-chat .message-count'))).toBeTruthy();
-                expect(_.isNull(_converse.rosterview.el.querySelector('.msgs-indicator'))).toBeTruthy();
-                done();
+                return test_utils.waitUntil(() => _converse.chatboxes.length).then(() => {
+
+                    const chatview = _converse.chatboxviews.get(sender_jid);
+                    chatview.model.set({'minimized': true});
+
+                    expect(_.isNull(_converse.chatboxviews.el.querySelector('.restore-chat .message-count'))).toBeTruthy();
+                    expect(_.isNull(_converse.rosterview.el.querySelector('.msgs-indicator'))).toBeTruthy();
+
+                    var msg = $msg({
+                            from: sender_jid,
+                            to: _converse.connection.jid,
+                            type: 'chat',
+                            id: (new Date()).getTime()
+                        }).c('body').t('hello').up()
+                        .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree();
+                    _converse.chatboxes.onMessage(msg);
+                    expect(_converse.chatboxviews.el.querySelector('.restore-chat .message-count').textContent).toBe('1');
+                    expect(_converse.rosterview.el.querySelector('.msgs-indicator').textContent).toBe('1');
+
+                    msg = $msg({
+                            from: sender_jid,
+                            to: _converse.connection.jid,
+                            type: 'chat',
+                            id: (new Date()).getTime()
+                        }).c('body').t('hello again').up()
+                        .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree();
+                    _converse.chatboxes.onMessage(msg);
+                    expect(_converse.chatboxviews.el.querySelector('.restore-chat .message-count').textContent).toBe('2');
+                    expect(_converse.rosterview.el.querySelector('.msgs-indicator').textContent).toBe('2');
+
+                    chatview.model.set({'minimized': false});
+                    expect(_.isNull(_converse.chatboxviews.el.querySelector('.restore-chat .message-count'))).toBeTruthy();
+                    expect(_.isNull(_converse.rosterview.el.querySelector('.msgs-indicator'))).toBeTruthy();
+                    done();
+                });
             }));
             }));
         });
         });
 
 

+ 57 - 44
spec/converse.js

@@ -1,12 +1,12 @@
 (function (root, factory) {
 (function (root, factory) {
     define([
     define([
-        "jquery",
         "jasmine",
         "jasmine",
         "mock",
         "mock",
         "test-utils"], factory);
         "test-utils"], factory);
-} (this, function ($, jasmine, mock, test_utils) {
-    var b64_sha1 = converse.env.b64_sha1;
-    var _ = converse.env._;
+} (this, function (jasmine, mock, test_utils) {
+    const b64_sha1 = converse.env.b64_sha1,
+          _ = converse.env._,
+          u = converse.env.utils;
 
 
     describe("Converse", function() {
     describe("Converse", function() {
         
         
@@ -274,59 +274,72 @@
 
 
         describe("The \"chats\" API", function() {
         describe("The \"chats\" API", function() {
 
 
-            it("has a method 'get' which returns the chatbox model", mock.initConverseWithPromises(
-                null, ['rosterInitialized'], {}, function (done, _converse) {
+            it("has a method 'get' which returns the promise that resolves to a chat model", mock.initConverseWithPromises(
+                null, ['rosterInitialized', 'chatBoxesInitialized'], {}, function (done, _converse) {
                     test_utils.openControlBox();
                     test_utils.openControlBox();
-                    test_utils.createContacts(_converse, 'current');
+                    test_utils.createContacts(_converse, 'current', 2);
+                    _converse.emit('rosterContactsFetched');
+
                     // Test on chat that doesn't exist.
                     // Test on chat that doesn't exist.
                     expect(_converse.api.chats.get('non-existing@jabber.org')).toBeFalsy();
                     expect(_converse.api.chats.get('non-existing@jabber.org')).toBeFalsy();
-                    var jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
+                    const jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
+                    const jid2 = mock.cur_names[1].replace(/ /g,'.').toLowerCase() + '@localhost';
+
                     // Test on chat that's not open
                     // Test on chat that's not open
-                    var box = _converse.api.chats.get(jid);
+                    let box = _converse.api.chats.get(jid);
                     expect(typeof box === 'undefined').toBeTruthy();
                     expect(typeof box === 'undefined').toBeTruthy();
-                    var chatboxview = _converse.chatboxviews.get(jid);
-                    // Test for single JID
+                    expect(_converse.chatboxes.length).toBe(1);
+
+                    // Test for one JID
                     test_utils.openChatBoxFor(_converse, jid);
                     test_utils.openChatBoxFor(_converse, jid);
-                    box = _converse.api.chats.get(jid);
+                    test_utils.waitUntil(() => _converse.chatboxes.length == 1).then(() => {
+                        box = _converse.api.chats.get(jid);
+                        expect(box instanceof Object).toBeTruthy();
+                        expect(box.get('box_id')).toBe(b64_sha1(jid));
+
+                        const chatboxview = _converse.chatboxviews.get(jid);
+                        expect(u.isVisible(chatboxview.el)).toBeTruthy();
+                        // Test for multiple JIDs
+                        test_utils.openChatBoxFor(_converse, jid2);
+                        return test_utils.waitUntil(() => _converse.chatboxes.length == 2);
+                    }).then(() => {
+                        const list = _converse.api.chats.get([jid, jid2]);
+                        expect(_.isArray(list)).toBeTruthy();
+                        expect(list[0].get('box_id')).toBe(b64_sha1(jid));
+                        expect(list[1].get('box_id')).toBe(b64_sha1(jid2));
+                        done();
+                    }).catch(_.partial(console.error, _));
+            }));
+
+            it("has a method 'open' which opens and returns a promise that resolves to a chat model", mock.initConverseWithPromises(
+                null, ['rosterGroupsFetched', 'chatBoxesInitialized'], {}, function (done, _converse) {
+
+                test_utils.openControlBox();
+                test_utils.createContacts(_converse, 'current', 2);
+                _converse.emit('rosterContactsFetched');
+
+                const jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
+                const jid2 = mock.cur_names[1].replace(/ /g,'.').toLowerCase() + '@localhost';
+                // Test on chat that doesn't exist.
+                expect(_converse.api.chats.get('non-existing@jabber.org')).toBeFalsy();
+
+                return _converse.api.chats.open(jid).then((box) => {
                     expect(box instanceof Object).toBeTruthy();
                     expect(box instanceof Object).toBeTruthy();
                     expect(box.get('box_id')).toBe(b64_sha1(jid));
                     expect(box.get('box_id')).toBe(b64_sha1(jid));
-                    chatboxview = _converse.chatboxviews.get(jid);
-                    expect($(chatboxview.el).is(':visible')).toBeTruthy();
+                    expect(
+                        _.keys(box),
+                        ['close', 'endOTR', 'focus', 'get', 'initiateOTR', 'is_chatroom', 'maximize', 'minimize', 'open', 'set']
+                    );
+                    const chatboxview = _converse.chatboxviews.get(jid);
+                    expect(u.isVisible(chatboxview.el)).toBeTruthy();
                     // Test for multiple JIDs
                     // Test for multiple JIDs
-                    var jid2 = mock.cur_names[1].replace(/ /g,'.').toLowerCase() + '@localhost';
-                    test_utils.openChatBoxFor(_converse, jid2);
-                    var list = _converse.api.chats.get([jid, jid2]);
+                    return _converse.api.chats.open([jid, jid2]);
+                }).then((list) => {
                     expect(_.isArray(list)).toBeTruthy();
                     expect(_.isArray(list)).toBeTruthy();
                     expect(list[0].get('box_id')).toBe(b64_sha1(jid));
                     expect(list[0].get('box_id')).toBe(b64_sha1(jid));
                     expect(list[1].get('box_id')).toBe(b64_sha1(jid2));
                     expect(list[1].get('box_id')).toBe(b64_sha1(jid2));
                     done();
                     done();
-            }));
-
-            it("has a method 'open' which opens and returns the chatbox model", mock.initConverseWithPromises(
-                null, ['rosterGroupsFetched'], {}, function (done, _converse) {
-
-                test_utils.openControlBox();
-                test_utils.createContacts(_converse, 'current');
-                var jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
-                var chatboxview;
-                // Test on chat that doesn't exist.
-                expect(_converse.api.chats.get('non-existing@jabber.org')).toBeFalsy();
-                var box = _converse.api.chats.open(jid);
-                expect(box instanceof Object).toBeTruthy();
-                expect(box.get('box_id')).toBe(b64_sha1(jid));
-                expect(
-                    _.keys(box),
-                    ['close', 'focus', 'get', 'is_chatroom', 'maximize', 'minimize', 'open', 'set']
-                );
-                chatboxview = _converse.chatboxviews.get(jid);
-                expect($(chatboxview.el).is(':visible')).toBeTruthy();
-                // Test for multiple JIDs
-                var jid2 = mock.cur_names[1].replace(/ /g,'.').toLowerCase() + '@localhost';
-                var list = _converse.api.chats.open([jid, jid2]);
-                expect(_.isArray(list)).toBeTruthy();
-                expect(list[0].get('box_id')).toBe(b64_sha1(jid));
-                expect(list[1].get('box_id')).toBe(b64_sha1(jid2));
-                done();
+                });
             }));
             }));
         });
         });
 
 

+ 295 - 298
spec/http-file-upload.js

@@ -225,25 +225,28 @@
                         [{'category': 'server', 'type':'IM'}],
                         [{'category': 'server', 'type':'IM'}],
                         ['http://jabber.org/protocol/disco#items'], [], 'info').then(function () {
                         ['http://jabber.org/protocol/disco#items'], [], 'info').then(function () {
 
 
-                        test_utils.waitUntilDiscoConfirmed(_converse, _converse.domain, [], [], ['upload.localhost'], 'items').then(function () {
-                            test_utils.waitUntilDiscoConfirmed(_converse, 'upload.localhost', [], [Strophe.NS.HTTPUPLOAD], []).then(function () {
-                                test_utils.createContacts(_converse, 'current');
-                                var contact_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost';
-                                test_utils.openChatBoxFor(_converse, contact_jid);
-                                var view = _converse.chatboxviews.get(contact_jid);
-                                test_utils.waitUntil(function () {
-                                    return view.el.querySelector('.upload-file');
-                                }, 150).then(function () {
-                                    expect(view.el.querySelector('.chat-toolbar .upload-file')).not.toBe(null);
-                                    done();
-                                });
-                            });
+                        let contact_jid, view;
+
+                        test_utils.waitUntilDiscoConfirmed(_converse, _converse.domain, [], [], ['upload.localhost'], 'items')
+                        .then(() => test_utils.waitUntilDiscoConfirmed(_converse, 'upload.localhost', [], [Strophe.NS.HTTPUPLOAD], []))
+                        .then(() => {
+                            test_utils.createContacts(_converse, 'current', 3);
+                            _converse.emit('rosterContactsFetched');
+
+                            contact_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost';
+                            return test_utils.openChatBoxFor(_converse, contact_jid);
+                        }).then(() => {
+                            view = _converse.chatboxviews.get(contact_jid);
+                            test_utils.waitUntil(() => view.el.querySelector('.upload-file'));
+                        }).then(() => {
+                            expect(view.el.querySelector('.chat-toolbar .upload-file')).not.toBe(null);
+                            done();
                         });
                         });
                     });
                     });
                 }));
                 }));
 
 
                 it("appears in MUC chats", mock.initConverseWithPromises(
                 it("appears in MUC chats", mock.initConverseWithPromises(
-                        null, ['rosterGroupsFetched'], {},
+                        null, ['rosterGroupsFetched', 'chatBoxesFetched'], {},
                         function (done, _converse) {
                         function (done, _converse) {
 
 
                     test_utils.waitUntilDiscoConfirmed(
                     test_utils.waitUntilDiscoConfirmed(
@@ -251,19 +254,15 @@
                         [{'category': 'server', 'type':'IM'}],
                         [{'category': 'server', 'type':'IM'}],
                         ['http://jabber.org/protocol/disco#items'], [], 'info').then(function () {
                         ['http://jabber.org/protocol/disco#items'], [], 'info').then(function () {
 
 
-                        test_utils.waitUntilDiscoConfirmed(_converse, _converse.domain, [], [], ['upload.localhost'], 'items').then(function () {
-                            test_utils.waitUntilDiscoConfirmed(_converse, 'upload.localhost', [], [Strophe.NS.HTTPUPLOAD], []).then(function () {
-                                test_utils.openAndEnterChatRoom(_converse, 'lounge', 'localhost', 'dummy').then(function () {
-                                    var view = _converse.chatboxviews.get('lounge@localhost');
-                                    test_utils.waitUntil(function () {
-                                        return view.el.querySelector('.upload-file');
-                                    }).then(function () {
-                                        expect(view.el.querySelector('.chat-toolbar .upload-file')).not.toBe(null);
-                                        done();
-                                    });
-                                });
-                            });
-                        });
+                        test_utils.waitUntilDiscoConfirmed(_converse, _converse.domain, [], [], ['upload.localhost'], 'items')
+                        .then(() => test_utils.waitUntilDiscoConfirmed(_converse, 'upload.localhost', [], [Strophe.NS.HTTPUPLOAD], []))
+                        .then(() => test_utils.openAndEnterChatRoom(_converse, 'lounge', 'localhost', 'dummy'))
+                        .then(() => test_utils.waitUntil(() => _converse.chatboxviews.get('lounge@localhost').el.querySelector('.upload-file')))
+                        .then(() => {
+                            const view = _converse.chatboxviews.get('lounge@localhost');
+                            expect(view.el.querySelector('.chat-toolbar .upload-file')).not.toBe(null);
+                            done();
+                        }).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
                     });
                     });
                 }));
                 }));
 
 
@@ -277,101 +276,104 @@
 
 
                             var send_backup = XMLHttpRequest.prototype.send;
                             var send_backup = XMLHttpRequest.prototype.send;
                             var IQ_stanzas = _converse.connection.IQ_stanzas;
                             var IQ_stanzas = _converse.connection.IQ_stanzas;
+                            let contact_jid;
 
 
-                            test_utils.waitUntilDiscoConfirmed(_converse, _converse.domain, [], [], ['upload.montague.tld'], 'items').then(function () {
-                                test_utils.waitUntilDiscoConfirmed(_converse, 'upload.montague.tld', [], [Strophe.NS.HTTPUPLOAD], []).then(function () {
-                                    test_utils.createContacts(_converse, 'current');
-                                    var contact_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost';
-                                    test_utils.openChatBoxFor(_converse, contact_jid);
-                                    var view = _converse.chatboxviews.get(contact_jid);
-                                    var file = {
-                                        'type': 'image/jpeg',
-                                        'size': '23456' ,
-                                        'lastModifiedDate': "",
-                                        'name': "my-juliet.jpg"
-                                    };
-                                    view.model.sendFiles([file]);
-                                    return test_utils.waitUntil(function () {
-                                        return _.filter(IQ_stanzas, function (iq) {
-                                            return iq.nodeTree.querySelector('iq[to="upload.montague.tld"] request');
-                                        }).length > 0;
-                                    }).then(function () {
-                                        var iq = IQ_stanzas.pop();
-                                        expect(iq.toLocaleString()).toBe(
-                                            "<iq from='dummy@localhost/resource' "+
-                                                "to='upload.montague.tld' "+
-                                                "type='get' "+
-                                                "xmlns='jabber:client' "+
-                                                "id='"+iq.nodeTree.getAttribute('id')+"'>"+
-                                            "<request xmlns='urn:xmpp:http:upload:0' "+
-                                                "filename='my-juliet.jpg' "+
-                                                "size='23456' "+
-                                                "content-type='image/jpeg'/>"+
-                                            "</iq>");
-
-                                        var base_url = document.URL.split(window.location.pathname)[0];
-                                        var message = base_url+"/logo/conversejs-filled.svg";
-
-                                        var stanza = Strophe.xmlHtmlNode(
-                                            "<iq from='upload.montague.tld'"+
-                                            "    id='"+iq.nodeTree.getAttribute('id')+"'"+
-                                            "    to='dummy@localhost/resource'"+
-                                            "    type='result'>"+
-                                            "<slot xmlns='urn:xmpp:http:upload:0'>"+
-                                            "    <put url='https://upload.montague.tld/4a771ac1-f0b2-4a4a-9700-f2a26fa2bb67/my-juliet.jpg'>"+
-                                            "    <header name='Authorization'>Basic Base64String==</header>"+
-                                            "    <header name='Cookie'>foo=bar; user=romeo</header>"+
-                                            "    </put>"+
-                                            "    <get url='"+message+"' />"+
-                                            "</slot>"+
-                                            "</iq>").firstElementChild;
-
-                                        spyOn(XMLHttpRequest.prototype, 'send').and.callFake(function () {
-                                            const message = view.model.messages.at(0);
-                                            expect(view.el.querySelector('.chat-content progress').getAttribute('value')).toBe('0');
-                                            message.set('progress', 0.5);
-                                            expect(view.el.querySelector('.chat-content progress').getAttribute('value')).toBe('0.5');
-                                            message.set('progress', 1);
-                                            expect(view.el.querySelector('.chat-content progress').getAttribute('value')).toBe('1');
-                                            message.save({
-                                                'upload': _converse.SUCCESS,
-                                                'oob_url': message.get('get'),
-                                                'message': message.get('get')
-                                            });
-                                        });
-                                        var sent_stanza;
-                                        spyOn(_converse.connection, 'send').and.callFake(function (stanza) {
-                                            sent_stanza = stanza;
+                           test_utils.waitUntilDiscoConfirmed(_converse, _converse.domain, [], [], ['upload.montague.tld'], 'items')
+                            .then(() => test_utils.waitUntilDiscoConfirmed(_converse, 'upload.montague.tld', [], [Strophe.NS.HTTPUPLOAD], []))
+                            .then(() => {
+                                test_utils.createContacts(_converse, 'current');
+                                _converse.emit('rosterContactsFetched');
+                                contact_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost';
+                                return test_utils.openChatBoxFor(_converse, contact_jid);
+                            }).then(() => {
+                                var view = _converse.chatboxviews.get(contact_jid);
+                                var file = {
+                                    'type': 'image/jpeg',
+                                    'size': '23456' ,
+                                    'lastModifiedDate': "",
+                                    'name': "my-juliet.jpg"
+                                };
+                                view.model.sendFiles([file]);
+                                return test_utils.waitUntil(function () {
+                                    return _.filter(IQ_stanzas, function (iq) {
+                                        return iq.nodeTree.querySelector('iq[to="upload.montague.tld"] request');
+                                    }).length > 0;
+                                }).then(function () {
+                                    var iq = IQ_stanzas.pop();
+                                    expect(iq.toLocaleString()).toBe(
+                                        "<iq from='dummy@localhost/resource' "+
+                                            "to='upload.montague.tld' "+
+                                            "type='get' "+
+                                            "xmlns='jabber:client' "+
+                                            "id='"+iq.nodeTree.getAttribute('id')+"'>"+
+                                        "<request xmlns='urn:xmpp:http:upload:0' "+
+                                            "filename='my-juliet.jpg' "+
+                                            "size='23456' "+
+                                            "content-type='image/jpeg'/>"+
+                                        "</iq>");
+
+                                    var base_url = document.URL.split(window.location.pathname)[0];
+                                    var message = base_url+"/logo/conversejs-filled.svg";
+
+                                    var stanza = Strophe.xmlHtmlNode(
+                                        "<iq from='upload.montague.tld'"+
+                                        "    id='"+iq.nodeTree.getAttribute('id')+"'"+
+                                        "    to='dummy@localhost/resource'"+
+                                        "    type='result'>"+
+                                        "<slot xmlns='urn:xmpp:http:upload:0'>"+
+                                        "    <put url='https://upload.montague.tld/4a771ac1-f0b2-4a4a-9700-f2a26fa2bb67/my-juliet.jpg'>"+
+                                        "    <header name='Authorization'>Basic Base64String==</header>"+
+                                        "    <header name='Cookie'>foo=bar; user=romeo</header>"+
+                                        "    </put>"+
+                                        "    <get url='"+message+"' />"+
+                                        "</slot>"+
+                                        "</iq>").firstElementChild;
+
+                                    spyOn(XMLHttpRequest.prototype, 'send').and.callFake(function () {
+                                        const message = view.model.messages.at(0);
+                                        expect(view.el.querySelector('.chat-content progress').getAttribute('value')).toBe('0');
+                                        message.set('progress', 0.5);
+                                        expect(view.el.querySelector('.chat-content progress').getAttribute('value')).toBe('0.5');
+                                        message.set('progress', 1);
+                                        expect(view.el.querySelector('.chat-content progress').getAttribute('value')).toBe('1');
+                                        message.save({
+                                            'upload': _converse.SUCCESS,
+                                            'oob_url': message.get('get'),
+                                            'message': message.get('get')
                                         });
                                         });
-                                        _converse.connection._dataRecv(test_utils.createRequest(stanza));
+                                    });
+                                    var sent_stanza;
+                                    spyOn(_converse.connection, 'send').and.callFake(function (stanza) {
+                                        sent_stanza = stanza;
+                                    });
+                                    _converse.connection._dataRecv(test_utils.createRequest(stanza));
 
 
+                                    return test_utils.waitUntil(function () {
+                                        return sent_stanza;
+                                    }, 1000).then(function () {
+                                        expect(sent_stanza.toLocaleString()).toBe(
+                                            "<message from='dummy@localhost/resource' "+
+                                                "to='irini.vlastuin@localhost' "+
+                                                "type='chat' "+
+                                                "id='"+sent_stanza.nodeTree.getAttribute('id')+"' xmlns='jabber:client'>"+
+                                                    "<body>"+message+"</body>"+
+                                                    "<active xmlns='http://jabber.org/protocol/chatstates'/>"+
+                                                    "<x xmlns='jabber:x:oob'>"+
+                                                        "<url>"+message+"</url>"+
+                                                    "</x>"+
+                                            "</message>");
                                         return test_utils.waitUntil(function () {
                                         return test_utils.waitUntil(function () {
-                                            return sent_stanza;
-                                        }, 1000).then(function () {
-                                            expect(sent_stanza.toLocaleString()).toBe(
-                                                "<message from='dummy@localhost/resource' "+
-                                                    "to='irini.vlastuin@localhost' "+
-                                                    "type='chat' "+
-                                                    "id='"+sent_stanza.nodeTree.getAttribute('id')+"' xmlns='jabber:client'>"+
-                                                        "<body>"+message+"</body>"+
-                                                        "<active xmlns='http://jabber.org/protocol/chatstates'/>"+
-                                                        "<x xmlns='jabber:x:oob'>"+
-                                                            "<url>"+message+"</url>"+
-                                                        "</x>"+
-                                                "</message>");
-                                            return test_utils.waitUntil(function () {
-                                                return view.el.querySelector('.chat-image');
-                                            }, 1000);
-                                        }).then(function () {
-                                            // Check that the image renders
-                                            expect(view.el.querySelector('.chat-msg .chat-msg__media').innerHTML.trim()).toEqual(
-                                                '<!-- src/templates/image.html -->\n'+
-                                                '<a href="http://localhost:8000/logo/conversejs-filled.svg" target="_blank" rel="noopener">'+
-                                                    '<img class="chat-image img-thumbnail" src="http://localhost:8000/logo/conversejs-filled.svg">'+
-                                                '</a>');
-                                            XMLHttpRequest.prototype.send = send_backup;
-                                            done();
-                                        });
+                                            return view.el.querySelector('.chat-image');
+                                        }, 1000);
+                                    }).then(function () {
+                                        // Check that the image renders
+                                        expect(view.el.querySelector('.chat-msg .chat-msg__media').innerHTML.trim()).toEqual(
+                                            `<!-- src/templates/image.html -->\n`+
+                                            `<a href="${window.location.origin}/logo/conversejs-filled.svg" target="_blank" rel="noopener">`+
+                                                `<img class="chat-image img-thumbnail" src="${window.location.origin}/logo/conversejs-filled.svg">`+
+                                            `</a>`);
+                                        XMLHttpRequest.prototype.send = send_backup;
+                                        done();
                                     });
                                     });
                                 });
                                 });
                             });
                             });
@@ -473,9 +475,10 @@
                                             }).then(function () {
                                             }).then(function () {
                                                 // Check that the image renders
                                                 // Check that the image renders
                                                 expect(view.el.querySelector('.chat-msg .chat-msg__media').innerHTML.trim()).toEqual(
                                                 expect(view.el.querySelector('.chat-msg .chat-msg__media').innerHTML.trim()).toEqual(
-                                                    '<!-- src/templates/image.html -->\n'+
-                                                    '<a href="http://localhost:8000/logo/conversejs-filled.svg" target="_blank" rel="noopener">'+
-                                                        '<img class="chat-image img-thumbnail" src="http://localhost:8000/logo/conversejs-filled.svg"></a>')
+                                                    `<!-- src/templates/image.html -->\n`+
+                                                    `<a href="${window.location.origin}/logo/conversejs-filled.svg" target="_blank" rel="noopener">`+
+                                                        `<img class="chat-image img-thumbnail" src="${window.location.origin}/logo/conversejs-filled.svg">`+
+                                                    `</a>`);
                                                 XMLHttpRequest.prototype.send = send_backup;
                                                 XMLHttpRequest.prototype.send = send_backup;
                                                 done();
                                                 done();
                                             });
                                             });
@@ -486,56 +489,53 @@
                         });
                         });
                     }));
                     }));
 
 
-                    it("shows and error message if the file is too large", mock.initConverseWithAsync(function (done, _converse) {
-                        var IQ_stanzas = _converse.connection.IQ_stanzas;
-                        var IQ_ids =  _converse.connection.IQ_ids;
-                        var send_backup = XMLHttpRequest.prototype.send;
+                    it("shows an error message if the file is too large", mock.initConverseWithAsync(function (done, _converse) {
+                        const IQ_stanzas = _converse.connection.IQ_stanzas;
+                        const IQ_ids =  _converse.connection.IQ_ids;
+                        const send_backup = XMLHttpRequest.prototype.send;
+                        let view, contact_jid;
+
+                        test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, [], [])
+                        .then(() => test_utils.waitUntil(() => _.filter(
+                            IQ_stanzas, (iq) => iq.nodeTree.querySelector('iq[to="localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]')).length
+                        )).then(() => {
+                            var stanza = _.find(IQ_stanzas, function (iq) {
+                                return iq.nodeTree.querySelector(
+                                    'iq[to="localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]');
+                            });
+                            var info_IQ_id = IQ_ids[IQ_stanzas.indexOf(stanza)];
 
 
-                        test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, [], []).then(function () {
-                            test_utils.waitUntil(function () {
+                            stanza = $iq({
+                                'type': 'result',
+                                'from': 'localhost',
+                                'to': 'dummy@localhost/resource',
+                                'id': info_IQ_id
+                            }).c('query', {'xmlns': 'http://jabber.org/protocol/disco#info'})
+                                .c('identity', {
+                                    'category': 'server',
+                                    'type': 'im'}).up()
+                                .c('feature', {
+                                    'var': 'http://jabber.org/protocol/disco#info'}).up()
+                                .c('feature', {
+                                    'var': 'http://jabber.org/protocol/disco#items'});
+                            _converse.connection._dataRecv(test_utils.createRequest(stanza));
+                            return _converse.api.disco.entities.get();
+                        }).then(function (entities) {
+                            expect(entities.length).toBe(2);
+                            expect(_.includes(entities.pluck('jid'), 'localhost')).toBe(true);
+                            expect(_.includes(entities.pluck('jid'), 'dummy@localhost')).toBe(true);
+
+                            expect(entities.get(_converse.domain).features.length).toBe(2);
+                            expect(entities.get(_converse.domain).identities.length).toBe(1);
+
+                            return test_utils.waitUntil(function () {
+                                // Converse.js sees that the entity has a disco#items feature,
+                                // so it will make a query for it.
                                 return _.filter(IQ_stanzas, function (iq) {
                                 return _.filter(IQ_stanzas, function (iq) {
-                                    return iq.nodeTree.querySelector(
-                                        'iq[to="localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]');
+                                    return iq.nodeTree.querySelector('iq[to="localhost"] query[xmlns="http://jabber.org/protocol/disco#items"]');
                                 }).length > 0;
                                 }).length > 0;
-                            }, 300).then(function () {
-                                var stanza = _.find(IQ_stanzas, function (iq) {
-                                    return iq.nodeTree.querySelector(
-                                        'iq[to="localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]');
-                                });
-                                var info_IQ_id = IQ_ids[IQ_stanzas.indexOf(stanza)];
-
-                                stanza = $iq({
-                                    'type': 'result',
-                                    'from': 'localhost',
-                                    'to': 'dummy@localhost/resource',
-                                    'id': info_IQ_id
-                                }).c('query', {'xmlns': 'http://jabber.org/protocol/disco#info'})
-                                    .c('identity', {
-                                        'category': 'server',
-                                        'type': 'im'}).up()
-                                    .c('feature', {
-                                        'var': 'http://jabber.org/protocol/disco#info'}).up()
-                                    .c('feature', {
-                                        'var': 'http://jabber.org/protocol/disco#items'});
-                                _converse.connection._dataRecv(test_utils.createRequest(stanza));
-
-                                _converse.api.disco.entities.get().then(function(entities) {
-                                    expect(entities.length).toBe(2);
-                                    expect(_.includes(entities.pluck('jid'), 'localhost')).toBe(true);
-                                    expect(_.includes(entities.pluck('jid'), 'dummy@localhost')).toBe(true);
-
-                                    expect(entities.get(_converse.domain).features.length).toBe(2);
-                                    expect(entities.get(_converse.domain).identities.length).toBe(1);
-
-                                    return test_utils.waitUntil(function () {
-                                        // Converse.js sees that the entity has a disco#items feature,
-                                        // so it will make a query for it.
-                                        return _.filter(IQ_stanzas, function (iq) {
-                                            return iq.nodeTree.querySelector('iq[to="localhost"] query[xmlns="http://jabber.org/protocol/disco#items"]');
-                                        }).length > 0;
-                                    }, 300);
-                                });
-                            }).then(function () {
+                            }, 300);
+                        }).then(function () {
                             var stanza = _.find(IQ_stanzas, function (iq) {
                             var stanza = _.find(IQ_stanzas, function (iq) {
                                 return iq.nodeTree.querySelector('iq[to="localhost"] query[xmlns="http://jabber.org/protocol/disco#items"]');
                                 return iq.nodeTree.querySelector('iq[to="localhost"] query[xmlns="http://jabber.org/protocol/disco#items"]');
                             });
                             });
@@ -549,77 +549,77 @@
                                 .c('item', {
                                 .c('item', {
                                     'jid': 'upload.localhost',
                                     'jid': 'upload.localhost',
                                     'name': 'HTTP File Upload'});
                                     'name': 'HTTP File Upload'});
-                                _converse.connection._dataRecv(test_utils.createRequest(stanza));
 
 
-                                _converse.api.disco.entities.get().then(function (entities) {
-                                    expect(entities.length).toBe(2);
-                                    expect(entities.get('localhost').items.length).toBe(1);
-                                    return test_utils.waitUntil(function () {
-                                        // Converse.js sees that the entity has a disco#info feature,
-                                        // so it will make a query for it.
-                                        return _.filter(IQ_stanzas, function (iq) {
-                                            return iq.nodeTree.querySelector('iq[to="upload.localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]');
-                                        }).length > 0;
-                                    }, 300);
-                                });
-                            }).then(function () {
-                                var stanza = _.find(IQ_stanzas, function (iq) {
-                                    return iq.nodeTree.querySelector('iq[to="upload.localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]');
-                                });
-                                var IQ_id = IQ_ids[IQ_stanzas.indexOf(stanza)];
-                                expect(stanza.toLocaleString()).toBe(
-                                    "<iq from='dummy@localhost/resource' to='upload.localhost' type='get' xmlns='jabber:client' id='"+IQ_id+"'>"+
-                                        "<query xmlns='http://jabber.org/protocol/disco#info'/>"+
-                                    "</iq>");
-
-                                // Upload service responds and reports a maximum file size of 5MiB
-                                stanza = $iq({'type': 'result', 'to': 'dummy@localhost/resource', 'id': IQ_id, 'from': 'upload.localhost'})
-                                    .c('query', {'xmlns': 'http://jabber.org/protocol/disco#info'})
-                                        .c('identity', {'category':'store', 'type':'file', 'name':'HTTP File Upload'}).up()
-                                        .c('feature', {'var':'urn:xmpp:http:upload:0'}).up()
-                                        .c('x', {'type':'result', 'xmlns':'jabber:x:data'})
-                                            .c('field', {'var':'FORM_TYPE', 'type':'hidden'})
-                                                .c('value').t('urn:xmpp:http:upload:0').up().up()
-                                            .c('field', {'var':'max-file-size'})
-                                                .c('value').t('5242880');
-                                _converse.connection._dataRecv(test_utils.createRequest(stanza));
-
-                                _converse.api.disco.entities.get().then(function (entities) {
-                                    expect(entities.get('localhost').items.get('upload.localhost').identities.where({'category': 'store'}).length).toBe(1);
-                                    _converse.api.disco.supports(Strophe.NS.HTTPUPLOAD, _converse.domain).then(
-                                        function (result) {
-                                            test_utils.createContacts(_converse, 'current');
-                                            var contact_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost';
-                                            test_utils.openChatBoxFor(_converse, contact_jid);
-                                            var view = _converse.chatboxviews.get(contact_jid);
-                                            var file = {
-                                                'type': 'image/jpeg',
-                                                'size': '5242881',
-                                                'lastModifiedDate': "",
-                                                'name': "my-juliet.jpg"
-                                            };
-                                            view.model.sendFiles([file]);
-                                            return test_utils.waitUntil(function () {
-                                                return view.el.querySelectorAll('.message').length;
-                                            }).then(function () {
-                                                const messages = view.el.querySelectorAll('.message.chat-error');
-                                                expect(messages.length).toBe(1);
-                                                expect(messages[0].textContent).toBe(
-                                                    'The size of your file, my-juliet.jpg, exceeds the maximum allowed by your server, which is 5 MB.');
-                                                done();
-                                            });
-                                        }
-                                    );
-                                }).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
-                            })
-                        });
+                            _converse.connection._dataRecv(test_utils.createRequest(stanza));
+
+                            _converse.api.disco.entities.get().then(function (entities) {
+                                expect(entities.length).toBe(2);
+                                expect(entities.get('localhost').items.length).toBe(1);
+                                return test_utils.waitUntil(function () {
+                                    // Converse.js sees that the entity has a disco#info feature,
+                                    // so it will make a query for it.
+                                    return _.filter(IQ_stanzas, function (iq) {
+                                        return iq.nodeTree.querySelector('iq[to="upload.localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]');
+                                    }).length > 0;
+                                }, 300);
+                            });
+                        }).then(function () {
+                            var stanza = _.find(IQ_stanzas, function (iq) {
+                                return iq.nodeTree.querySelector('iq[to="upload.localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]');
+                            });
+                            var IQ_id = IQ_ids[IQ_stanzas.indexOf(stanza)];
+                            expect(stanza.toLocaleString()).toBe(
+                                "<iq from='dummy@localhost/resource' to='upload.localhost' type='get' xmlns='jabber:client' id='"+IQ_id+"'>"+
+                                    "<query xmlns='http://jabber.org/protocol/disco#info'/>"+
+                                "</iq>");
+
+                            // Upload service responds and reports a maximum file size of 5MiB
+                            stanza = $iq({'type': 'result', 'to': 'dummy@localhost/resource', 'id': IQ_id, 'from': 'upload.localhost'})
+                                .c('query', {'xmlns': 'http://jabber.org/protocol/disco#info'})
+                                    .c('identity', {'category':'store', 'type':'file', 'name':'HTTP File Upload'}).up()
+                                    .c('feature', {'var':'urn:xmpp:http:upload:0'}).up()
+                                    .c('x', {'type':'result', 'xmlns':'jabber:x:data'})
+                                        .c('field', {'var':'FORM_TYPE', 'type':'hidden'})
+                                            .c('value').t('urn:xmpp:http:upload:0').up().up()
+                                        .c('field', {'var':'max-file-size'})
+                                            .c('value').t('5242880');
+                            _converse.connection._dataRecv(test_utils.createRequest(stanza));
+                            return _converse.api.disco.entities.get();
+                        }).then(function (entities) {
+                            expect(entities.get('localhost').items.get('upload.localhost').identities.where({'category': 'store'}).length).toBe(1);
+                            return _converse.api.disco.supports(Strophe.NS.HTTPUPLOAD, _converse.domain);
+                        }).then(function (result) {
+                            test_utils.createContacts(_converse, 'current');
+                            _converse.emit('rosterContactsFetched');
+
+                            contact_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost';
+                            return test_utils.openChatBoxFor(_converse, contact_jid);
+                        }).then(() => {
+                            view = _converse.chatboxviews.get(contact_jid);
+                            var file = {
+                                'type': 'image/jpeg',
+                                'size': '5242881',
+                                'lastModifiedDate': "",
+                                'name': "my-juliet.jpg"
+                            };
+                            view.model.sendFiles([file]);
+                            return test_utils.waitUntil(() => view.el.querySelectorAll('.message').length)
+                        }).then(function () {
+                            const messages = view.el.querySelectorAll('.message.chat-error');
+                            expect(messages.length).toBe(1);
+                            expect(messages[0].textContent).toBe(
+                                'The size of your file, my-juliet.jpg, exceeds the maximum allowed by your server, which is 5 MB.');
+                            done();
+                        }).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL))
                     }));
                     }));
                 });
                 });
             });
             });
 
 
             describe("While a file is being uploaded", function () {
             describe("While a file is being uploaded", function () {
 
 
-                it("shows a progress bar", mock.initConverseWithAsync(function (done, _converse) {
+                it("shows a progress bar", mock.initConverseWithPromises(
+                            null, ['rosterGroupsFetched', 'chatBoxesFetched'], {}, function (done, _converse) {
+
                     test_utils.waitUntilDiscoConfirmed(
                     test_utils.waitUntilDiscoConfirmed(
                         _converse, _converse.domain,
                         _converse, _converse.domain,
                         [{'category': 'server', 'type':'IM'}],
                         [{'category': 'server', 'type':'IM'}],
@@ -627,77 +627,74 @@
 
 
                         var send_backup = XMLHttpRequest.prototype.send;
                         var send_backup = XMLHttpRequest.prototype.send;
                         var IQ_stanzas = _converse.connection.IQ_stanzas;
                         var IQ_stanzas = _converse.connection.IQ_stanzas;
-
-                        test_utils.waitUntilDiscoConfirmed(_converse, _converse.domain, [], [], ['upload.montague.tld'], 'items').then(function () {
-                            test_utils.waitUntilDiscoConfirmed(_converse, 'upload.montague.tld', [], [Strophe.NS.HTTPUPLOAD], []).then(function () {
-                                test_utils.createContacts(_converse, 'current');
-                                var contact_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost';
-                                test_utils.openChatBoxFor(_converse, contact_jid);
-                                var view = _converse.chatboxviews.get(contact_jid);
-                                var file = {
-                                    'type': 'image/jpeg',
-                                    'size': '23456' ,
-                                    'lastModifiedDate': "",
-                                    'name': "my-juliet.jpg"
-                                };
-                                view.model.sendFiles([file]);
-                                return test_utils.waitUntil(function () {
-                                    return _.filter(IQ_stanzas, function (iq) {
-                                        return iq.nodeTree.querySelector('iq[to="upload.montague.tld"] request');
-                                    }).length > 0;
-                                }).then(function () {
-                                    var iq = IQ_stanzas.pop();
-                                    expect(iq.toLocaleString()).toBe(
-                                        "<iq from='dummy@localhost/resource' "+
-                                            "to='upload.montague.tld' "+
-                                            "type='get' "+
-                                            "xmlns='jabber:client' "+
-                                            "id='"+iq.nodeTree.getAttribute('id')+"'>"+
-                                        "<request xmlns='urn:xmpp:http:upload:0' "+
-                                            "filename='my-juliet.jpg' "+
-                                            "size='23456' "+
-                                            "content-type='image/jpeg'/>"+
-                                        "</iq>");
-
-                                    var base_url = document.URL.split(window.location.pathname)[0];
-                                    var message = base_url+"/logo/conversejs-filled.svg";
-
-                                    var stanza = Strophe.xmlHtmlNode(
-                                        "<iq from='upload.montague.tld'"+
-                                        "    id='"+iq.nodeTree.getAttribute('id')+"'"+
-                                        "    to='dummy@localhost/resource'"+
-                                        "    type='result'>"+
-                                        "<slot xmlns='urn:xmpp:http:upload:0'>"+
-                                        "    <put url='https://upload.montague.tld/4a771ac1-f0b2-4a4a-9700-f2a26fa2bb67/my-juliet.jpg'>"+
-                                        "    <header name='Authorization'>Basic Base64String==</header>"+
-                                        "    <header name='Cookie'>foo=bar; user=romeo</header>"+
-                                        "    </put>"+
-                                        "    <get url='"+message+"' />"+
-                                        "</slot>"+
-                                        "</iq>").firstElementChild;
-
-                                    spyOn(XMLHttpRequest.prototype, 'send').and.callFake(function () {
-                                        const message = view.model.messages.at(0);
-                                        expect(view.el.querySelector('.chat-content progress').getAttribute('value')).toBe('0');
-                                        message.set('progress', 0.5);
-                                        expect(view.el.querySelector('.chat-content progress').getAttribute('value')).toBe('0.5');
-                                        message.set('progress', 1);
-                                        expect(view.el.querySelector('.chat-content progress').getAttribute('value')).toBe('1');
-                                        expect(view.el.querySelector('.chat-content .chat-msg__text').textContent).toBe('Uploading file: my-juliet.jpg, 22.91 KB');
-                                        done();
-                                    });
-                                    var sent_stanza;
-                                    spyOn(_converse.connection, 'send').and.callFake(function (stanza) {
-                                        sent_stanza = stanza;
-                                    });
-                                    _converse.connection._dataRecv(test_utils.createRequest(stanza));
-                                });
+                        let view, contact_jid;
+
+                        test_utils.waitUntilDiscoConfirmed(_converse, _converse.domain, [], [], ['upload.montague.tld'], 'items')
+                        .then(() => test_utils.waitUntilDiscoConfirmed(_converse, 'upload.montague.tld', [], [Strophe.NS.HTTPUPLOAD], []))
+                        .then(() => {
+                            test_utils.createContacts(_converse, 'current');
+                            _converse.emit('rosterContactsFetched');
+
+                            contact_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost';
+                            return test_utils.openChatBoxFor(_converse, contact_jid);
+                        }).then(() => {
+                            view = _converse.chatboxviews.get(contact_jid);
+                            const file = {
+                                'type': 'image/jpeg',
+                                'size': '23456' ,
+                                'lastModifiedDate': "",
+                                'name': "my-juliet.jpg"
+                            };
+                            view.model.sendFiles([file]);
+                            return test_utils.waitUntil(() => _.filter(IQ_stanzas, (iq) => iq.nodeTree.querySelector('iq[to="upload.montague.tld"] request')).length)
+                        }).then(function () {
+                            const iq = IQ_stanzas.pop();
+                            expect(iq.toLocaleString()).toBe(
+                                "<iq from='dummy@localhost/resource' "+
+                                    "to='upload.montague.tld' "+
+                                    "type='get' "+
+                                    "xmlns='jabber:client' "+
+                                    "id='"+iq.nodeTree.getAttribute('id')+"'>"+
+                                "<request xmlns='urn:xmpp:http:upload:0' "+
+                                    "filename='my-juliet.jpg' "+
+                                    "size='23456' "+
+                                    "content-type='image/jpeg'/>"+
+                                "</iq>");
+
+                            const base_url = document.URL.split(window.location.pathname)[0];
+                            const message = base_url+"/logo/conversejs-filled.svg";
+                            const stanza = Strophe.xmlHtmlNode(
+                                "<iq from='upload.montague.tld'"+
+                                "    id='"+iq.nodeTree.getAttribute('id')+"'"+
+                                "    to='dummy@localhost/resource'"+
+                                "    type='result'>"+
+                                "<slot xmlns='urn:xmpp:http:upload:0'>"+
+                                "    <put url='https://upload.montague.tld/4a771ac1-f0b2-4a4a-9700-f2a26fa2bb67/my-juliet.jpg'>"+
+                                "    <header name='Authorization'>Basic Base64String==</header>"+
+                                "    <header name='Cookie'>foo=bar; user=romeo</header>"+
+                                "    </put>"+
+                                "    <get url='"+message+"' />"+
+                                "</slot>"+
+                                "</iq>").firstElementChild;
+                            spyOn(XMLHttpRequest.prototype, 'send').and.callFake(function () {
+                                const message = view.model.messages.at(0);
+                                expect(view.el.querySelector('.chat-content progress').getAttribute('value')).toBe('0');
+                                message.set('progress', 0.5);
+                                expect(view.el.querySelector('.chat-content progress').getAttribute('value')).toBe('0.5');
+                                message.set('progress', 1);
+                                expect(view.el.querySelector('.chat-content progress').getAttribute('value')).toBe('1');
+                                expect(view.el.querySelector('.chat-content .chat-msg__text').textContent).toBe('Uploading file: my-juliet.jpg, 22.91 KB');
+                                done();
                             });
                             });
+                            var sent_stanza;
+                            spyOn(_converse.connection, 'send').and.callFake(function (stanza) {
+                                sent_stanza = stanza;
+                            });
+                            _converse.connection._dataRecv(test_utils.createRequest(stanza));
                         });
                         });
                     });
                     });
                 }));
                 }));
             });
             });
-
         });
         });
     });
     });
 }));
 }));

File diff suppressed because it is too large
+ 514 - 500
spec/messages.js


+ 103 - 87
spec/minchats.js

@@ -1,8 +1,9 @@
 (function (root, factory) {
 (function (root, factory) {
     define(["jquery", "jasmine", "mock", "test-utils"], factory);
     define(["jquery", "jasmine", "mock", "test-utils"], factory);
 } (this, function ($, jasmine, mock, test_utils) {
 } (this, function ($, jasmine, mock, test_utils) {
-    var _ = converse.env._;
-    var $msg = converse.env.$msg;
+    const _ = converse.env._;
+    const  $msg = converse.env.$msg;
+    const u = converse.env.utils;
 
 
     describe("The Minimized Chats Widget", function () {
     describe("The Minimized Chats Widget", function () {
 
 
@@ -12,32 +13,37 @@
                 function (done, _converse) {
                 function (done, _converse) {
 
 
             test_utils.createContacts(_converse, 'current');
             test_utils.createContacts(_converse, 'current');
+            _converse.emit('rosterContactsFetched');
+
             test_utils.openControlBox();
             test_utils.openControlBox();
             _converse.minimized_chats.toggleview.model.browserStorage._clear();
             _converse.minimized_chats.toggleview.model.browserStorage._clear();
             _converse.minimized_chats.initToggle();
             _converse.minimized_chats.initToggle();
 
 
-            var contact_jid, chatview;
-            contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
-            test_utils.openChatBoxFor(_converse, contact_jid);
-            chatview = _converse.chatboxviews.get(contact_jid);
-            expect(chatview.model.get('minimized')).toBeFalsy();
-            expect($(_converse.minimized_chats.el).is(':visible')).toBeFalsy();
-            chatview.el.querySelector('.toggle-chatbox-button').click();
-            expect(chatview.model.get('minimized')).toBeTruthy();
-            expect($(_converse.minimized_chats.el).is(':visible')).toBeTruthy();
-            expect(_converse.minimized_chats.keys().length).toBe(1);
-            expect(_converse.minimized_chats.keys()[0]).toBe(contact_jid);
-
-            contact_jid = mock.cur_names[1].replace(/ /g,'.').toLowerCase() + '@localhost';
-            test_utils.openChatBoxFor(_converse, contact_jid);
-            chatview = _converse.chatboxviews.get(contact_jid);
-            expect(chatview.model.get('minimized')).toBeFalsy();
-            chatview.el.querySelector('.toggle-chatbox-button').click();
-            expect(chatview.model.get('minimized')).toBeTruthy();
-            expect($(_converse.minimized_chats.el).is(':visible')).toBeTruthy();
-            expect(_converse.minimized_chats.keys().length).toBe(2);
-            expect(_.includes(_converse.minimized_chats.keys(), contact_jid)).toBeTruthy();
-            done();
+            let contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
+            let chatview;
+            test_utils.openChatBoxFor(_converse, contact_jid)
+            .then(() => {
+                chatview = _converse.chatboxviews.get(contact_jid);
+                expect(chatview.model.get('minimized')).toBeFalsy();
+                expect($(_converse.minimized_chats.el).is(':visible')).toBeFalsy();
+                chatview.el.querySelector('.toggle-chatbox-button').click();
+                expect(chatview.model.get('minimized')).toBeTruthy();
+                expect($(_converse.minimized_chats.el).is(':visible')).toBeTruthy();
+                expect(_converse.minimized_chats.keys().length).toBe(1);
+                expect(_converse.minimized_chats.keys()[0]).toBe(contact_jid);
+
+                contact_jid = mock.cur_names[1].replace(/ /g,'.').toLowerCase() + '@localhost';
+                return test_utils.openChatBoxFor(_converse, contact_jid);
+            }).then(() => {
+                chatview = _converse.chatboxviews.get(contact_jid);
+                expect(chatview.model.get('minimized')).toBeFalsy();
+                chatview.el.querySelector('.toggle-chatbox-button').click();
+                expect(chatview.model.get('minimized')).toBeTruthy();
+                expect($(_converse.minimized_chats.el).is(':visible')).toBeTruthy();
+                expect(_converse.minimized_chats.keys().length).toBe(2);
+                expect(_.includes(_converse.minimized_chats.keys(), contact_jid)).toBeTruthy();
+                done();
+            });
         }));
         }));
 
 
         it("can be toggled to hide or show minimized chats",
         it("can be toggled to hide or show minimized chats",
@@ -46,25 +52,26 @@
                 function (done, _converse) {
                 function (done, _converse) {
 
 
             test_utils.createContacts(_converse, 'current');
             test_utils.createContacts(_converse, 'current');
+            _converse.emit('rosterContactsFetched');
+
             test_utils.openControlBox();
             test_utils.openControlBox();
             _converse.minimized_chats.toggleview.model.browserStorage._clear();
             _converse.minimized_chats.toggleview.model.browserStorage._clear();
             _converse.minimized_chats.initToggle();
             _converse.minimized_chats.initToggle();
 
 
-            var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
-            test_utils.openChatBoxFor(_converse, contact_jid);
-            var chatview = _converse.chatboxviews.get(contact_jid);
-            expect($(_converse.minimized_chats.el).is(':visible')).toBeFalsy();
-            chatview.model.set({'minimized': true});
-            expect($(_converse.minimized_chats.el).is(':visible')).toBeTruthy();
-            expect(_converse.minimized_chats.keys().length).toBe(1);
-            expect(_converse.minimized_chats.keys()[0]).toBe(contact_jid);
-            expect($(_converse.minimized_chats.el.querySelector('.minimized-chats-flyout')).is(':visible')).toBeTruthy();
-            expect(_converse.minimized_chats.toggleview.model.get('collapsed')).toBeFalsy();
-            _converse.minimized_chats.el.querySelector('#toggle-minimized-chats').click();
-
-            return test_utils.waitUntil(function () {
-                return $(_converse.minimized_chats.el.querySelector('.minimized-chats-flyout')).is(':visible');
-            }, 500).then(function () {
+            const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
+            test_utils.openChatBoxFor(_converse, contact_jid)
+            .then(() => {
+                const chatview = _converse.chatboxviews.get(contact_jid);
+                expect(u.isVisible(_converse.minimized_chats.el)).toBeFalsy();
+                chatview.model.set({'minimized': true});
+                expect(u.isVisible(_converse.minimized_chats.el)).toBeTruthy();
+                expect(_converse.minimized_chats.keys().length).toBe(1);
+                expect(_converse.minimized_chats.keys()[0]).toBe(contact_jid);
+                expect(u.isVisible(_converse.minimized_chats.el.querySelector('.minimized-chats-flyout'))).toBeTruthy();
+                expect(_converse.minimized_chats.toggleview.model.get('collapsed')).toBeFalsy();
+                _converse.minimized_chats.el.querySelector('#toggle-minimized-chats').click();
+                return test_utils.waitUntil(() => u.isVisible(_converse.minimized_chats.el.querySelector('.minimized-chats-flyout')));
+            }).then(() => {
                 expect(_converse.minimized_chats.toggleview.model.get('collapsed')).toBeTruthy();
                 expect(_converse.minimized_chats.toggleview.model.get('collapsed')).toBeTruthy();
                 done();
                 done();
             });
             });
@@ -72,70 +79,79 @@
 
 
         it("shows the number messages received to minimized chats",
         it("shows the number messages received to minimized chats",
             mock.initConverseWithPromises(
             mock.initConverseWithPromises(
-                null, ['rosterGroupsFetched'], {},
+                null, ['rosterGroupsFetched', 'chatBoxesFetched'], {},
                 function (done, _converse) {
                 function (done, _converse) {
 
 
             test_utils.createContacts(_converse, 'current');
             test_utils.createContacts(_converse, 'current');
+            _converse.emit('rosterContactsFetched');
+
             test_utils.openControlBox();
             test_utils.openControlBox();
             _converse.minimized_chats.toggleview.model.browserStorage._clear();
             _converse.minimized_chats.toggleview.model.browserStorage._clear();
             _converse.minimized_chats.initToggle();
             _converse.minimized_chats.initToggle();
 
 
             var i, contact_jid, chatview, msg;
             var i, contact_jid, chatview, msg;
             _converse.minimized_chats.toggleview.model.set({'collapsed': true});
             _converse.minimized_chats.toggleview.model.set({'collapsed': true});
-            expect($(_converse.minimized_chats.toggleview.el.querySelector('.unread-message-count')).is(':visible')).toBeFalsy();
+
+            const unread_el = _converse.minimized_chats.toggleview.el.querySelector('.unread-message-count');
+            expect(_.isNull(unread_el)).toBe(true);
+
             for (i=0; i<3; i++) {
             for (i=0; i<3; i++) {
                 contact_jid = mock.cur_names[i].replace(/ /g,'.').toLowerCase() + '@localhost';
                 contact_jid = mock.cur_names[i].replace(/ /g,'.').toLowerCase() + '@localhost';
                 test_utils.openChatBoxFor(_converse, contact_jid);
                 test_utils.openChatBoxFor(_converse, contact_jid);
-                chatview = _converse.chatboxviews.get(contact_jid);
-                chatview.model.set({'minimized': true});
-                msg = $msg({
+            }
+            return test_utils.waitUntil(() => _converse.chatboxes.length == 4).then(() => {
+                for (i=0; i<3; i++) {
+                    chatview = _converse.chatboxviews.get(contact_jid);
+                    chatview.model.set({'minimized': true});
+                    msg = $msg({
+                        from: contact_jid,
+                        to: _converse.connection.jid,
+                        type: 'chat',
+                        id: (new Date()).getTime()
+                    }).c('body').t('This message is sent to a minimized chatbox').up()
+                    .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree();
+                    _converse.chatboxes.onMessage(msg);
+                    expect($(_converse.minimized_chats.toggleview.el.querySelector('.unread-message-count')).is(':visible')).toBeTruthy();
+                    expect($(_converse.minimized_chats.toggleview.el.querySelector('.unread-message-count')).text()).toBe((i+1).toString());
+                }
+                // Chat state notifications don't increment the unread messages counter
+                // <composing> state
+                _converse.chatboxes.onMessage($msg({
                     from: contact_jid,
                     from: contact_jid,
                     to: _converse.connection.jid,
                     to: _converse.connection.jid,
                     type: 'chat',
                     type: 'chat',
                     id: (new Date()).getTime()
                     id: (new Date()).getTime()
-                }).c('body').t('This message is sent to a minimized chatbox').up()
-                .c('active', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree();
-                _converse.chatboxes.onMessage(msg);
-                expect($(_converse.minimized_chats.toggleview.el.querySelector('.unread-message-count')).is(':visible')).toBeTruthy();
-                expect($(_converse.minimized_chats.toggleview.el.querySelector('.unread-message-count')).text()).toBe((i+1).toString());
-            }
-            // Chat state notifications don't increment the unread messages counter
-            // <composing> state
-            _converse.chatboxes.onMessage($msg({
-                from: contact_jid,
-                to: _converse.connection.jid,
-                type: 'chat',
-                id: (new Date()).getTime()
-            }).c('composing', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree());
-            expect($(_converse.minimized_chats.toggleview.el.querySelector('.unread-message-count')).text()).toBe((i).toString());
-
-            // <paused> state
-            _converse.chatboxes.onMessage($msg({
-                from: contact_jid,
-                to: _converse.connection.jid,
-                type: 'chat',
-                id: (new Date()).getTime()
-            }).c('paused', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree());
-            expect($(_converse.minimized_chats.toggleview.el.querySelector('.unread-message-count')).text()).toBe((i).toString());
-
-            // <gone> state
-            _converse.chatboxes.onMessage($msg({
-                from: contact_jid,
-                to: _converse.connection.jid,
-                type: 'chat',
-                id: (new Date()).getTime()
-            }).c('gone', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree());
-            expect($(_converse.minimized_chats.toggleview.el.querySelector('.unread-message-count')).text()).toBe((i).toString());
-
-            // <inactive> state
-            _converse.chatboxes.onMessage($msg({
-                from: contact_jid,
-                to: _converse.connection.jid,
-                type: 'chat',
-                id: (new Date()).getTime()
-            }).c('inactive', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree());
-            expect($(_converse.minimized_chats.toggleview.el.querySelector('.unread-message-count')).text()).toBe((i).toString());
-            done();
+                }).c('composing', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree());
+                expect($(_converse.minimized_chats.toggleview.el.querySelector('.unread-message-count')).text()).toBe((i).toString());
+
+                // <paused> state
+                _converse.chatboxes.onMessage($msg({
+                    from: contact_jid,
+                    to: _converse.connection.jid,
+                    type: 'chat',
+                    id: (new Date()).getTime()
+                }).c('paused', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree());
+                expect($(_converse.minimized_chats.toggleview.el.querySelector('.unread-message-count')).text()).toBe((i).toString());
+
+                // <gone> state
+                _converse.chatboxes.onMessage($msg({
+                    from: contact_jid,
+                    to: _converse.connection.jid,
+                    type: 'chat',
+                    id: (new Date()).getTime()
+                }).c('gone', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree());
+                expect($(_converse.minimized_chats.toggleview.el.querySelector('.unread-message-count')).text()).toBe((i).toString());
+
+                // <inactive> state
+                _converse.chatboxes.onMessage($msg({
+                    from: contact_jid,
+                    to: _converse.connection.jid,
+                    type: 'chat',
+                    id: (new Date()).getTime()
+                }).c('inactive', {'xmlns': 'http://jabber.org/protocol/chatstates'}).tree());
+                expect($(_converse.minimized_chats.toggleview.el.querySelector('.unread-message-count')).text()).toBe((i).toString());
+                done();
+            });
         }));
         }));
 
 
         it("shows the number messages received to minimized groupchats",
         it("shows the number messages received to minimized groupchats",

+ 105 - 102
spec/roomslist.js

@@ -12,44 +12,46 @@
     describe("A list of open rooms", function () {
     describe("A list of open rooms", function () {
 
 
         it("is shown in the \"Rooms\" panel", mock.initConverseWithPromises(
         it("is shown in the \"Rooms\" panel", mock.initConverseWithPromises(
-            null, ['rosterGroupsFetched'],
+            null, ['rosterGroupsFetched', 'chatBoxesFetched'],
             { allow_bookmarks: false // Makes testing easier, otherwise we
             { allow_bookmarks: false // Makes testing easier, otherwise we
                                      // have to mock stanza traffic.
                                      // have to mock stanza traffic.
             },
             },
             function (done, _converse) {
             function (done, _converse) {
                 test_utils.openControlBox();
                 test_utils.openControlBox();
-                var controlbox = _converse.chatboxviews.get('controlbox');
-
-                var list = controlbox.el.querySelector('div.rooms-list-container');
+                const controlbox = _converse.chatboxviews.get('controlbox');
+                let list = controlbox.el.querySelector('div.rooms-list-container');
                 expect(_.includes(list.classList, 'hidden')).toBeTruthy();
                 expect(_.includes(list.classList, 'hidden')).toBeTruthy();
 
 
-                test_utils.openChatRoom(_converse, 'room', 'conference.shakespeare.lit', 'JC');
-
-                expect(_.isUndefined(_converse.rooms_list_view)).toBeFalsy();
-                var room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
-                expect(room_els.length).toBe(1);
-                expect(room_els[0].innerText).toBe('room@conference.shakespeare.lit');
-
-                test_utils.openChatRoom(_converse, 'lounge', 'localhost', 'dummy');
-                room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
-                expect(room_els.length).toBe(2);
-
-                var view = _converse.chatboxviews.get('room@conference.shakespeare.lit');
-                view.close();
-                room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
-                expect(room_els.length).toBe(1);
-                expect(room_els[0].innerText).toBe('lounge@localhost');
-                list = controlbox.el.querySelector('div.rooms-list-container');
-                expect(_.includes(list.classList, 'hidden')).toBeFalsy();
-
-                view = _converse.chatboxviews.get('lounge@localhost');
-                view.close();
-                room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
-                expect(room_els.length).toBe(0);
-
-                list = controlbox.el.querySelector('div.rooms-list-container');
-                expect(_.includes(list.classList, 'hidden')).toBeTruthy();
-                done();
+                let room_els;
+
+                test_utils.openChatRoom(_converse, 'room', 'conference.shakespeare.lit', 'JC')
+                .then(() => {
+                    expect(_.isUndefined(_converse.rooms_list_view)).toBeFalsy();
+                    room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
+                    expect(room_els.length).toBe(1);
+                    expect(room_els[0].innerText).toBe('room@conference.shakespeare.lit');
+                    return test_utils.openChatRoom(_converse, 'lounge', 'localhost', 'dummy');
+                }).then(() => {
+                    room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
+                    expect(room_els.length).toBe(2);
+
+                    var view = _converse.chatboxviews.get('room@conference.shakespeare.lit');
+                    view.close();
+                    room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
+                    expect(room_els.length).toBe(1);
+                    expect(room_els[0].innerText).toBe('lounge@localhost');
+                    list = controlbox.el.querySelector('div.rooms-list-container');
+                    expect(_.includes(list.classList, 'hidden')).toBeFalsy();
+
+                    view = _converse.chatboxviews.get('lounge@localhost');
+                    view.close();
+                    room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
+                    expect(room_els.length).toBe(0);
+
+                    list = controlbox.el.querySelector('div.rooms-list-container');
+                    expect(_.includes(list.classList, 'hidden')).toBeTruthy();
+                    done();
+                });
             }
             }
         ));
         ));
     });
     });
@@ -57,79 +59,81 @@
     describe("A groupchat shown in the groupchats list", function () {
     describe("A groupchat shown in the groupchats list", function () {
 
 
         it("is highlighted if its currently open", mock.initConverseWithPromises(
         it("is highlighted if its currently open", mock.initConverseWithPromises(
-            null, ['rosterGroupsFetched'],
+            null, ['rosterGroupsFetched', 'chatBoxesFetched'],
             { whitelisted_plugins: ['converse-roomslist'],
             { whitelisted_plugins: ['converse-roomslist'],
               allow_bookmarks: false // Makes testing easier, otherwise we
               allow_bookmarks: false // Makes testing easier, otherwise we
                                      // have to mock stanza traffic.
                                      // have to mock stanza traffic.
             }, function (done, _converse) {
             }, function (done, _converse) {
 
 
-            spyOn(_converse, 'isSingleton').and.callFake(function () {
-                return true;
-            });
+            spyOn(_converse, 'isSingleton').and.callFake(() => true);
 
 
+            let room_els, item;
             test_utils.openControlBox();
             test_utils.openControlBox();
-            _converse.api.rooms.open('coven@chat.shakespeare.lit', {'nick': 'some1'});
-            let room_els = _converse.rooms_list_view.el.querySelectorAll(".available-chatroom");
-            expect(room_els.length).toBe(1);
-
-            let item = room_els[0];
-            expect(u.hasClass('open', item)).toBe(true);
-            expect(item.textContent.trim()).toBe('coven@chat.shakespeare.lit');
+            _converse.api.rooms.open('coven@chat.shakespeare.lit', {'nick': 'some1'})
+            .then(() => {
+                room_els = _converse.rooms_list_view.el.querySelectorAll(".available-chatroom");
+                expect(room_els.length).toBe(1);
 
 
-            _converse.api.rooms.open('balcony@chat.shakespeare.lit', {'nick': 'some1'});
-            room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
-            expect(room_els.length).toBe(2);
+                item = room_els[0];
+                expect(u.hasClass('open', item)).toBe(true);
+                expect(item.textContent.trim()).toBe('coven@chat.shakespeare.lit');
+                return _converse.api.rooms.open('balcony@chat.shakespeare.lit', {'nick': 'some1'});
+            }).then(() => {
+                room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
+                expect(room_els.length).toBe(2);
 
 
-            room_els = _converse.rooms_list_view.el.querySelectorAll(".available-chatroom.open");
-            expect(room_els.length).toBe(1);
-            item = room_els[0];
-            expect(item.textContent.trim()).toBe('balcony@chat.shakespeare.lit');
-            done();
+                room_els = _converse.rooms_list_view.el.querySelectorAll(".available-chatroom.open");
+                expect(room_els.length).toBe(1);
+                item = room_els[0];
+                expect(item.textContent.trim()).toBe('balcony@chat.shakespeare.lit');
+                done();
+            });
         }));
         }));
 
 
         it("has an info icon which opens a details modal when clicked", mock.initConverseWithPromises(
         it("has an info icon which opens a details modal when clicked", mock.initConverseWithPromises(
-            null, ['rosterGroupsFetched'],
+            null, ['rosterGroupsFetched', 'chatBoxesFetched'],
             { whitelisted_plugins: ['converse-roomslist'],
             { whitelisted_plugins: ['converse-roomslist'],
               allow_bookmarks: false // Makes testing easier, otherwise we
               allow_bookmarks: false // Makes testing easier, otherwise we
                                      // have to mock stanza traffic.
                                      // have to mock stanza traffic.
             }, function (done, _converse) {
             }, function (done, _converse) {
 
 
+            let view;
             test_utils.openControlBox();
             test_utils.openControlBox();
-            _converse.api.rooms.open('coven@chat.shakespeare.lit', {'nick': 'some1'});
-            const view = _converse.chatboxviews.get('coven@chat.shakespeare.lit');
-            const last_stanza = _.last(_converse.connection.IQ_stanzas).nodeTree;
-            const IQ_id = last_stanza.getAttribute('id');
-            const features_stanza = $iq({
-                    'from': 'coven@chat.shakespeare.lit',
-                    'id': IQ_id,
-                    'to': 'dummy@localhost/desktop',
-                    'type': 'result'
-                })
-                .c('query', { 'xmlns': 'http://jabber.org/protocol/disco#info'})
-                    .c('identity', {
-                        'category': 'conference',
-                        'name': 'A Dark Cave',
-                        'type': 'text'
-                    }).up()
-                    .c('feature', {'var': 'http://jabber.org/protocol/muc'}).up()
-                    .c('feature', {'var': 'muc_passwordprotected'}).up()
-                    .c('feature', {'var': 'muc_hidden'}).up()
-                    .c('feature', {'var': 'muc_temporary'}).up()
-                    .c('feature', {'var': 'muc_open'}).up()
-                    .c('feature', {'var': 'muc_unmoderated'}).up()
-                    .c('feature', {'var': 'muc_nonanonymous'}).up()
-                    .c('feature', {'var': 'urn:xmpp:mam:0'}).up()
-                    .c('x', { 'xmlns':'jabber:x:data', 'type':'result'})
-                        .c('field', {'var':'FORM_TYPE', 'type':'hidden'})
-                            .c('value').t('http://jabber.org/protocol/muc#roominfo').up().up()
-                        .c('field', {'type':'text-single', 'var':'muc#roominfo_description', 'label':'Description'})
-                            .c('value').t('This is the description').up().up()
-                        .c('field', {'type':'text-single', 'var':'muc#roominfo_occupants', 'label':'Number of occupants'})
-                            .c('value').t(0);
-            _converse.connection._dataRecv(test_utils.createRequest(features_stanza));
-
-            test_utils.waitUntil(() => view.model.get('connection_status') === converse.ROOMSTATUS.CONNECTING)
-            .then(function () {
+            _converse.api.rooms.open('coven@chat.shakespeare.lit', {'nick': 'some1'})
+            .then(() => {
+                view = _converse.chatboxviews.get('coven@chat.shakespeare.lit');
+                const last_stanza = _.last(_converse.connection.IQ_stanzas).nodeTree;
+                const IQ_id = last_stanza.getAttribute('id');
+                const features_stanza = $iq({
+                        'from': 'coven@chat.shakespeare.lit',
+                        'id': IQ_id,
+                        'to': 'dummy@localhost/desktop',
+                        'type': 'result'
+                    })
+                    .c('query', { 'xmlns': 'http://jabber.org/protocol/disco#info'})
+                        .c('identity', {
+                            'category': 'conference',
+                            'name': 'A Dark Cave',
+                            'type': 'text'
+                        }).up()
+                        .c('feature', {'var': 'http://jabber.org/protocol/muc'}).up()
+                        .c('feature', {'var': 'muc_passwordprotected'}).up()
+                        .c('feature', {'var': 'muc_hidden'}).up()
+                        .c('feature', {'var': 'muc_temporary'}).up()
+                        .c('feature', {'var': 'muc_open'}).up()
+                        .c('feature', {'var': 'muc_unmoderated'}).up()
+                        .c('feature', {'var': 'muc_nonanonymous'}).up()
+                        .c('feature', {'var': 'urn:xmpp:mam:0'}).up()
+                        .c('x', { 'xmlns':'jabber:x:data', 'type':'result'})
+                            .c('field', {'var':'FORM_TYPE', 'type':'hidden'})
+                                .c('value').t('http://jabber.org/protocol/muc#roominfo').up().up()
+                            .c('field', {'type':'text-single', 'var':'muc#roominfo_description', 'label':'Description'})
+                                .c('value').t('This is the description').up().up()
+                            .c('field', {'type':'text-single', 'var':'muc#roominfo_occupants', 'label':'Number of occupants'})
+                                .c('value').t(0);
+                _converse.connection._dataRecv(test_utils.createRequest(features_stanza));
+                return test_utils.waitUntil(() => view.model.get('connection_status') === converse.ROOMSTATUS.CONNECTING)
+            }).then(function () {
                 var presence = $pres({
                 var presence = $pres({
                         to: _converse.connection.jid,
                         to: _converse.connection.jid,
                         from: 'coven@chat.shakespeare.lit/some1',
                         from: 'coven@chat.shakespeare.lit/some1',
@@ -201,23 +205,22 @@
             },
             },
             function (done, _converse) {
             function (done, _converse) {
 
 
-            spyOn(window, 'confirm').and.callFake(function () {
-                return true;
-            });
+            spyOn(window, 'confirm').and.callFake(() => true);
             expect(_converse.chatboxes.length).toBe(1);
             expect(_converse.chatboxes.length).toBe(1);
-            test_utils.openChatRoom(
-                _converse, 'lounge', 'conference.shakespeare.lit', 'JC');
-            expect(_converse.chatboxes.length).toBe(2);
-            var room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
-            expect(room_els.length).toBe(1);
-            var close_el = _converse.rooms_list_view.el.querySelector(".close-room");
-            close_el.click();
-            expect(window.confirm).toHaveBeenCalledWith(
-                'Are you sure you want to leave the groupchat lounge@conference.shakespeare.lit?');
-            room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
-            expect(room_els.length).toBe(0);
-            expect(_converse.chatboxes.length).toBe(1);
-            done();
+            test_utils.openChatRoom(_converse, 'lounge', 'conference.shakespeare.lit', 'JC')
+            .then(() => {
+                expect(_converse.chatboxes.length).toBe(2);
+                var room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
+                expect(room_els.length).toBe(1);
+                var close_el = _converse.rooms_list_view.el.querySelector(".close-room");
+                close_el.click();
+                expect(window.confirm).toHaveBeenCalledWith(
+                    'Are you sure you want to leave the groupchat lounge@conference.shakespeare.lit?');
+                room_els = _converse.rooms_list_view.el.querySelectorAll(".open-room");
+                expect(room_els.length).toBe(0);
+                expect(_converse.chatboxes.length).toBe(1);
+                done();
+            });
         }));
         }));
 
 
         it("shows unread messages directed at the user", mock.initConverseWithAsync(
         it("shows unread messages directed at the user", mock.initConverseWithAsync(

+ 26 - 22
spec/spoilers.js

@@ -92,12 +92,14 @@
 
 
         it("can be sent without a hint",
         it("can be sent without a hint",
             mock.initConverseWithPromises(
             mock.initConverseWithPromises(
-                null, ['rosterGroupsFetched'], {},
+                null, ['rosterGroupsFetched', 'chatBoxesFetched'], {},
                 function (done, _converse) {
                 function (done, _converse) {
 
 
-            test_utils.createContacts(_converse, 'current');
+            test_utils.createContacts(_converse, 'current', 1);
+            _converse.emit('rosterContactsFetched');
+
             test_utils.openControlBox();
             test_utils.openControlBox();
-            var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
+            const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
 
 
             // XXX: We need to send a presence from the contact, so that we
             // XXX: We need to send a presence from the contact, so that we
             // have a resource, that resource is then queried to see
             // have a resource, that resource is then queried to see
@@ -108,9 +110,9 @@
                 'to': 'dummy@localhost'
                 'to': 'dummy@localhost'
             });
             });
             _converse.connection._dataRecv(test_utils.createRequest(presence));
             _converse.connection._dataRecv(test_utils.createRequest(presence));
-            test_utils.openChatBoxFor(_converse, contact_jid);
-
-            test_utils.waitUntilDiscoConfirmed(_converse, contact_jid+'/phone', [], [Strophe.NS.SPOILER]).then(function () {
+            test_utils.openChatBoxFor(_converse, contact_jid)
+            .then(() => test_utils.waitUntilDiscoConfirmed(_converse, contact_jid+'/phone', [], [Strophe.NS.SPOILER]))
+            .then(() => {
                 var view = _converse.chatboxviews.get(contact_jid);
                 var view = _converse.chatboxviews.get(contact_jid);
                 spyOn(view, 'onMessageSubmitted').and.callThrough();
                 spyOn(view, 'onMessageSubmitted').and.callThrough();
                 spyOn(_converse.connection, 'send');
                 spyOn(_converse.connection, 'send');
@@ -167,10 +169,12 @@
 
 
         it("can be sent with a hint",
         it("can be sent with a hint",
             mock.initConverseWithPromises(
             mock.initConverseWithPromises(
-                null, ['rosterGroupsFetched'], {},
+                null, ['rosterGroupsFetched', 'chatBoxesFetched'], {},
                 function (done, _converse) {
                 function (done, _converse) {
 
 
-            test_utils.createContacts(_converse, 'current');
+            test_utils.createContacts(_converse, 'current', 1);
+            _converse.emit('rosterContactsFetched');
+
             test_utils.openControlBox();
             test_utils.openControlBox();
             var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
             var contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
 
 
@@ -183,9 +187,9 @@
                 'to': 'dummy@localhost'
                 'to': 'dummy@localhost'
             });
             });
             _converse.connection._dataRecv(test_utils.createRequest(presence));
             _converse.connection._dataRecv(test_utils.createRequest(presence));
-            test_utils.openChatBoxFor(_converse, contact_jid);
-
-            test_utils.waitUntilDiscoConfirmed(_converse, contact_jid+'/phone', [], [Strophe.NS.SPOILER]).then(function () {
+            test_utils.openChatBoxFor(_converse, contact_jid)
+            .then(() => test_utils.waitUntilDiscoConfirmed(_converse, contact_jid+'/phone', [], [Strophe.NS.SPOILER]))
+            .then(() => {
                 var view = _converse.chatboxviews.get(contact_jid);
                 var view = _converse.chatboxviews.get(contact_jid);
                 var spoiler_toggle = view.el.querySelector('.toggle-compose-spoiler');
                 var spoiler_toggle = view.el.querySelector('.toggle-compose-spoiler');
                 spoiler_toggle.click();
                 spoiler_toggle.click();
@@ -206,17 +210,17 @@
                 expect(view.onMessageSubmitted).toHaveBeenCalled();
                 expect(view.onMessageSubmitted).toHaveBeenCalled();
 
 
                 /* Test the XML stanza 
                 /* Test the XML stanza 
-                *
-                * <message from="dummy@localhost/resource"
-                *          to="max.frankfurter@localhost"
-                *          type="chat"
-                *          id="4547c38b-d98b-45a5-8f44-b4004dbc335e"
-                *          xmlns="jabber:client">
-                *    <body>This is the spoiler</body>
-                *    <active xmlns="http://jabber.org/protocol/chatstates"/>
-                *    <spoiler xmlns="urn:xmpp:spoiler:0">This is the hint</spoiler>
-                * </message>"
-                */
+                 *
+                 * <message from="dummy@localhost/resource"
+                 *          to="max.frankfurter@localhost"
+                 *          type="chat"
+                 *          id="4547c38b-d98b-45a5-8f44-b4004dbc335e"
+                 *          xmlns="jabber:client">
+                 *    <body>This is the spoiler</body>
+                 *    <active xmlns="http://jabber.org/protocol/chatstates"/>
+                 *    <spoiler xmlns="urn:xmpp:spoiler:0">This is the hint</spoiler>
+                 * </message>"
+                 */
                 var stanza = _converse.connection.send.calls.argsFor(0)[0].tree();
                 var stanza = _converse.connection.send.calls.argsFor(0)[0].tree();
                 var spoiler_el = stanza.querySelector('spoiler[xmlns="urn:xmpp:spoiler:0"]');
                 var spoiler_el = stanza.querySelector('spoiler[xmlns="urn:xmpp:spoiler:0"]');
 
 

+ 20 - 18
spec/user-details-modal.js

@@ -16,22 +16,23 @@
 
 
         it("can be used to remove a contact",
         it("can be used to remove a contact",
             mock.initConverseWithPromises(
             mock.initConverseWithPromises(
-                null, ['rosterGroupsFetched'], {},
+                null, ['rosterGroupsFetched', 'chatBoxesFetched'], {},
                 function (done, _converse) {
                 function (done, _converse) {
 
 
             test_utils.createContacts(_converse, 'current');
             test_utils.createContacts(_converse, 'current');
             _converse.emit('rosterContactsFetched');
             _converse.emit('rosterContactsFetched');
 
 
+            let view, show_modal_button, modal;
             const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
             const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
             test_utils.openChatBoxFor(_converse, contact_jid);
             test_utils.openChatBoxFor(_converse, contact_jid);
-
-            const view = _converse.chatboxviews.get(contact_jid);
-            const show_modal_button = view.el.querySelector('.show-user-details-modal');
-            expect(u.isVisible(show_modal_button)).toBeTruthy();
-            show_modal_button.click();
-            const modal = view.user_details_modal;
-            test_utils.waitUntil(() => u.isVisible(modal.el), 1000)
-            .then(function () {
+            return test_utils.waitUntil(() => _converse.chatboxes.length).then(() => {
+                view = _converse.chatboxviews.get(contact_jid);
+                show_modal_button = view.el.querySelector('.show-user-details-modal');
+                expect(u.isVisible(show_modal_button)).toBeTruthy();
+                show_modal_button.click();
+                modal = view.user_details_modal;
+                return test_utils.waitUntil(() => u.isVisible(modal.el), 1000);
+            }).then(function () {
                 spyOn(window, 'confirm').and.returnValue(true);
                 spyOn(window, 'confirm').and.returnValue(true);
                 spyOn(view.model.contact, 'removeFromRoster').and.callFake(function (callback) {
                 spyOn(view.model.contact, 'removeFromRoster').and.callFake(function (callback) {
                     callback();
                     callback();
@@ -57,16 +58,17 @@
             test_utils.createContacts(_converse, 'current');
             test_utils.createContacts(_converse, 'current');
             _converse.emit('rosterContactsFetched');
             _converse.emit('rosterContactsFetched');
 
 
+            let view, modal;
             const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
             const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@localhost';
-            test_utils.openChatBoxFor(_converse, contact_jid);
-
-            const view = _converse.chatboxviews.get(contact_jid);
-            const show_modal_button = view.el.querySelector('.show-user-details-modal');
-            expect(u.isVisible(show_modal_button)).toBeTruthy();
-            show_modal_button.click();
-            const modal = view.user_details_modal;
-            test_utils.waitUntil(() => u.isVisible(modal.el), 2000)
-            .then(function () {
+            test_utils.openChatBoxFor(_converse, contact_jid)
+            .then(() => {
+                view = _converse.chatboxviews.get(contact_jid);
+                const show_modal_button = view.el.querySelector('.show-user-details-modal');
+                expect(u.isVisible(show_modal_button)).toBeTruthy();
+                show_modal_button.click();
+                modal = view.user_details_modal;
+                return test_utils.waitUntil(() => u.isVisible(modal.el), 2000);
+            }).then(function () {
                 spyOn(window, 'confirm').and.returnValue(true);
                 spyOn(window, 'confirm').and.returnValue(true);
                 spyOn(view.model.contact, 'removeFromRoster').and.callFake(function (callback, errback) {
                 spyOn(view.model.contact, 'removeFromRoster').and.callFake(function (callback, errback) {
                     errback();
                     errback();

+ 79 - 17
src/converse-chatboxes.js

@@ -69,12 +69,7 @@
                         Strophe.LogLevel.WARN
                         Strophe.LogLevel.WARN
                     );
                     );
                 }
                 }
-                Promise.all([
-                    _converse.api.waitUntil('rosterContactsFetched'),
-                    _converse.api.waitUntil('chatBoxesFetched')
-                ]).then(() => {
-                    _converse.api.chats.open(jid);
-                });
+                _converse.api.chats.open(jid);
             }
             }
             _converse.router.route('converse/chat?jid=:jid', openChat);
             _converse.router.route('converse/chat?jid=:jid', openChat);
 
 
@@ -485,7 +480,7 @@
                     if (attrs.type === 'groupchat') {
                     if (attrs.type === 'groupchat') {
                         attrs.from = stanza.getAttribute('from');
                         attrs.from = stanza.getAttribute('from');
                         attrs.nick = Strophe.unescapeNode(Strophe.getResourceFromJid(attrs.from));
                         attrs.nick = Strophe.unescapeNode(Strophe.getResourceFromJid(attrs.from));
-                        if (attrs.from === this.get('nick')) {
+                        if (Strophe.getResourceFromJid(attrs.from) === this.get('nick')) {
                             attrs.sender = 'me';
                             attrs.sender = 'me';
                         } else {
                         } else {
                             attrs.sender = 'them';
                             attrs.sender = 'them';
@@ -876,6 +871,11 @@
 
 
             /************************ BEGIN API ************************/
             /************************ BEGIN API ************************/
             _.extend(_converse.api, {
             _.extend(_converse.api, {
+                /**
+                 * The "chats" grouping (used for one-on-one chats)
+                 *
+                 * @namespace
+                 */
                 'chats': {
                 'chats': {
                     'create' (jids, attrs) {
                     'create' (jids, attrs) {
                         if (_.isUndefined(jids)) {
                         if (_.isUndefined(jids)) {
@@ -885,7 +885,6 @@
                             );
                             );
                             return null;
                             return null;
                         }
                         }
-
                         if (_.isString(jids)) {
                         if (_.isString(jids)) {
                             if (attrs && !_.get(attrs, 'fullname')) {
                             if (attrs && !_.get(attrs, 'fullname')) {
                                 attrs.fullname = _.get(_converse.api.contacts.get(jids), 'attributes.fullname');
                                 attrs.fullname = _.get(_converse.api.contacts.get(jids), 'attributes.fullname');
@@ -902,17 +901,80 @@
                             return _converse.chatboxes.getChatBox(jid, attrs, true).trigger('show');
                             return _converse.chatboxes.getChatBox(jid, attrs, true).trigger('show');
                         });
                         });
                     },
                     },
+
+                    /**
+                     * Opens a new one-on-one chat.
+                     *
+                     * @function
+                     *
+                     * @param {String|string[]} name - e.g. 'buddy@example.com' or ['buddy1@example.com', 'buddy2@example.com']
+                     * @returns {Promise} Promise which resolves with the Backbone.Model representing the chat.
+                     *
+                     * @example
+                     * // To open a single chat, provide the JID of the contact you're chatting with in that chat:
+                     * converse.plugins.add('myplugin', {
+                     *     initialize: function() {
+                     *         var _converse = this._converse;
+                     *         // Note, buddy@example.org must be in your contacts roster!
+                     *         _converse.api.chats.open('buddy@example.com').then((chat) => {
+                     *             // Now you can do something with the chat model
+                     *         });
+                     *     }
+                     * });
+                     *
+                     * @example
+                     * // To open an array of chats, provide an array of JIDs:
+                     * converse.plugins.add('myplugin', {
+                     *     initialize: function () {
+                     *         var _converse = this._converse;
+                     *         // Note, these users must first be in your contacts roster!
+                     *         _converse.api.chats.open(['buddy1@example.com', 'buddy2@example.com']).then((chats) => {
+                     *             // Now you can do something with the chat models
+                     *         });
+                     *     }
+                     * });
+                     *
+                     */
                     'open' (jids, attrs) {
                     'open' (jids, attrs) {
-                        if (_.isUndefined(jids)) {
-                            _converse.log("chats.open: You need to provide at least one JID", Strophe.LogLevel.ERROR);
-                            return null;
-                        } else if (_.isString(jids)) {
-                            const chatbox = _converse.api.chats.create(jids, attrs);
-                            chatbox.trigger('show');
-                            return chatbox;
-                        }
-                        return _.map(jids, (jid) => _converse.api.chats.create(jid, attrs).trigger('show'));
+                        return new Promise((resolve, reject) => {
+                            Promise.all([
+                                _converse.api.waitUntil('rosterContactsFetched'),
+                                _converse.api.waitUntil('chatBoxesFetched')
+                            ]).then(() => {
+                                if (_.isUndefined(jids)) {
+                                    const err_msg = "chats.open: You need to provide at least one JID";
+                                    _converse.log(err_msg, Strophe.LogLevel.ERROR);
+                                    reject(new Error(err_msg));
+                                } else if (_.isString(jids)) {
+                                    resolve(_converse.api.chats.create(jids, attrs).trigger('show'));
+                                } else {
+                                    resolve(_.map(jids, (jid) => _converse.api.chats.create(jid, attrs).trigger('show')));
+                                }
+                            }).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
+                        });
                     },
                     },
+
+                    /**
+                     * Returns a chat model. The chat should already be open.
+                     *
+                     * @function
+                     *
+                     * @param {String|string[]} name - e.g. 'buddy@example.com' or ['buddy1@example.com', 'buddy2@example.com']
+                     * @returns {Backbone.Model}
+                     *
+                     * @example
+                     * // To return a single chat, provide the JID of the contact you're chatting with in that chat:
+                     * const model = _converse.api.chats.get('buddy@example.com');
+                     *
+                     * @example
+                     * // To return an array of chats, provide an array of JIDs:
+                     * const models = _converse.api.chats.get(['buddy1@example.com', 'buddy2@example.com']);
+                     *
+                     * @example
+                     * // To return all open chats, call the method without any parameters::
+                     * const models = _converse.api.chats.get();
+                     *
+                     */
                     'get' (jids) {
                     'get' (jids) {
                         if (_.isUndefined(jids)) {
                         if (_.isUndefined(jids)) {
                             const result = [];
                             const result = [];

+ 5 - 4
src/converse-chatview.js

@@ -888,6 +888,9 @@
                     const textarea = this.el.querySelector('.chat-textarea'),
                     const textarea = this.el.querySelector('.chat-textarea'),
                           message = textarea.value;
                           message = textarea.value;
 
 
+                    if (!message.replace(/\s/g, '').length) {
+                        return;
+                    }
                     let spoiler_hint;
                     let spoiler_hint;
                     if (this.model.get('composing_spoiler')) {
                     if (this.model.get('composing_spoiler')) {
                         const hint_el = this.el.querySelector('form.sendXMPPMessage input.spoiler-hint');
                         const hint_el = this.el.querySelector('form.sendXMPPMessage input.spoiler-hint');
@@ -901,10 +904,8 @@
                     event.initEvent('input', true, true);
                     event.initEvent('input', true, true);
                     textarea.dispatchEvent(event);
                     textarea.dispatchEvent(event);
 
 
-                    if (message !== '') {
-                        this.onMessageSubmitted(message, spoiler_hint);
-                        _converse.emit('messageSend', message);
-                    }
+                    this.onMessageSubmitted(message, spoiler_hint);
+                    _converse.emit('messageSend', message);
                     this.setChatState(_converse.ACTIVE);
                     this.setChatState(_converse.ACTIVE);
                 },
                 },
 
 

+ 7 - 2
src/converse-core.js

@@ -99,6 +99,11 @@
         'converse-vcard'
         'converse-vcard'
     ];
     ];
 
 
+    // Setting wait to 59 instead of 60 to avoid timing conflicts with the
+    // webserver, which is often also set to 60 and might therefore sometimes
+    // return a 504 error page instead of passing through to the BOSH proxy.
+    const BOSH_WAIT = 59;
+
     // Make converse pluggable
     // Make converse pluggable
     pluggable.enable(_converse, '_converse', 'pluggable');
     pluggable.enable(_converse, '_converse', 'pluggable');
 
 
@@ -1026,7 +1031,7 @@
                 if (!this.connection.reconnecting) {
                 if (!this.connection.reconnecting) {
                     this.connection.reset();
                     this.connection.reset();
                 }
                 }
-                this.connection.connect(this.jid.toLowerCase(), null, this.onConnectStatusChanged);
+                this.connection.connect(this.jid.toLowerCase(), null, this.onConnectStatusChanged, BOSH_WAIT);
             } else if (this.authentication === _converse.LOGIN) {
             } else if (this.authentication === _converse.LOGIN) {
                 const password = _.isNil(credentials) ? (_converse.connection.pass || this.password) : credentials.password;
                 const password = _.isNil(credentials) ? (_converse.connection.pass || this.password) : credentials.password;
                 if (!password) {
                 if (!password) {
@@ -1047,7 +1052,7 @@
                 if (!this.connection.reconnecting) {
                 if (!this.connection.reconnecting) {
                     this.connection.reset();
                     this.connection.reset();
                 }
                 }
-                this.connection.connect(this.jid, password, this.onConnectStatusChanged);
+                this.connection.connect(this.jid, password, this.onConnectStatusChanged, BOSH_WAIT);
             }
             }
         };
         };
 
 

+ 1 - 0
src/converse-disco.js

@@ -296,6 +296,7 @@
                 if (from !== null) {
                 if (from !== null) {
                     iqresult.attrs({'to': from});
                     iqresult.attrs({'to': from});
                 }
                 }
+                iqresult.c('query', attrs);
                 _.each(plugin._identities, (identity) => {
                 _.each(plugin._identities, (identity) => {
                     const attrs = {
                     const attrs = {
                         'category': identity.category,
                         'category': identity.category,

+ 3 - 4
src/converse-embedded.js

@@ -30,11 +30,10 @@
             if (!_.isArray(_converse.auto_join_rooms) && !_.isArray(_converse.auto_join_private_chats)) {
             if (!_.isArray(_converse.auto_join_rooms) && !_.isArray(_converse.auto_join_private_chats)) {
                 throw new Error("converse-embedded: auto_join_rooms must be an Array");
                 throw new Error("converse-embedded: auto_join_rooms must be an Array");
             }
             }
-            if (_converse.auto_join_rooms.length !== 1 && _converse.auto_join_private_chats.length !== 1) {
+            if (_converse.auto_join_rooms.length > 1 && _converse.auto_join_private_chats.length > 1) {
                 throw new Error("converse-embedded: It doesn't make "+
                 throw new Error("converse-embedded: It doesn't make "+
-                    "sense to have the auto_join_rooms setting to zero or "+
-                    "more then one, since only one chat room can be open "+
-                    "at any time.");
+                    "sense to have the auto_join_rooms setting more then one, "+
+                    "since only one chat room can be open at any time.");
             }
             }
         }
         }
     });
     });

+ 19 - 17
src/converse-muc-views.js

@@ -550,16 +550,7 @@
 
 
                     this.model.occupants.on('add', this.showJoinNotification, this);
                     this.model.occupants.on('add', this.showJoinNotification, this);
                     this.model.occupants.on('remove', this.showLeaveNotification, this);
                     this.model.occupants.on('remove', this.showLeaveNotification, this);
-                    this.model.occupants.on('change:show', (occupant) => {
-                        if (!occupant.isMember() || _.includes(occupant.get('states'), '303')) {
-                            return;
-                        }
-                        if (occupant.get('show') === 'offline') {
-                            this.showLeaveNotification(occupant);
-                        } else if (occupant.get('show') === 'online') {
-                            this.showJoinNotification(occupant);
-                        }
-                    });
+                    this.model.occupants.on('change:show', this.showJoinOrLeaveNotification, this);
 
 
                     this.createEmojiPicker();
                     this.createEmojiPicker();
                     this.createOccupantsView();
                     this.createOccupantsView();
@@ -570,8 +561,7 @@
                         const handler = () => {
                         const handler = () => {
                             if (!u.isPersistableModel(this.model)) {
                             if (!u.isPersistableModel(this.model)) {
                                 // Happens during tests, nothing to do if this
                                 // Happens during tests, nothing to do if this
-                                // is a hanging chatbox (i.e. not in the
-                                // collection anymore).
+                                // is a hanging chatbox (i.e. not in the collection anymore).
                                 return;
                                 return;
                             }
                             }
                             this.populateAndJoin();
                             this.populateAndJoin();
@@ -1383,6 +1373,17 @@
                     }
                     }
                 },
                 },
 
 
+                showJoinOrLeaveNotification (occupant) {
+                    if (!occupant.isMember() || _.includes(occupant.get('states'), '303')) {
+                        return;
+                    }
+                    if (occupant.get('show') === 'offline') {
+                        this.showLeaveNotification(occupant);
+                    } else if (occupant.get('show') === 'online') {
+                        this.showJoinNotification(occupant);
+                    }
+                },
+
                 showJoinNotification (occupant) {
                 showJoinNotification (occupant) {
                     if (this.model.get('connection_status') !==  converse.ROOMSTATUS.ENTERED) {
                     if (this.model.get('connection_status') !==  converse.ROOMSTATUS.ENTERED) {
                         return;
                         return;
@@ -1429,11 +1430,11 @@
                 showLeaveNotification (occupant) {
                 showLeaveNotification (occupant) {
                     const nick = occupant.get('nick'),
                     const nick = occupant.get('nick'),
                           stat = occupant.get('status'),
                           stat = occupant.get('status'),
-                          last_el = this.content.lastElementChild,
-                          last_msg_date = last_el.getAttribute('data-isodate');
+                          last_el = this.content.lastElementChild;
 
 
-                    if (_.includes(_.get(last_el, 'classList', []), 'chat-info') &&
-                            moment(last_msg_date).isSame(new Date(), "day") &&
+                    if (last_el &&
+                            _.includes(_.get(last_el, 'classList', []), 'chat-info') &&
+                            moment(last_el.getAttribute('data-isodate')).isSame(new Date(), "day") &&
                             _.get(last_el, 'dataset', {}).join === `"${nick}"`) {
                             _.get(last_el, 'dataset', {}).join === `"${nick}"`) {
 
 
                         let message;
                         let message;
@@ -1462,7 +1463,8 @@
                             'extra_classes': 'chat-event',
                             'extra_classes': 'chat-event',
                             'data': `data-leave="${nick}"`
                             'data': `data-leave="${nick}"`
                         }
                         }
-                        if (_.includes(_.get(last_el, 'classList', []), 'chat-info') &&
+                        if (last_el &&
+                            _.includes(_.get(last_el, 'classList', []), 'chat-info') &&
                             _.get(last_el, 'dataset', {}).leavejoin === `"${nick}"`) {
                             _.get(last_el, 'dataset', {}).leavejoin === `"${nick}"`) {
 
 
                             last_el.outerHTML = tpl_info(data);
                             last_el.outerHTML = tpl_info(data);

+ 15 - 6
src/converse-muc.js

@@ -1254,14 +1254,23 @@
                         }
                         }
                         return _.map(jids, _.partial(createChatRoom, _, attrs));
                         return _.map(jids, _.partial(createChatRoom, _, attrs));
                     },
                     },
+
                     'open' (jids, attrs) {
                     'open' (jids, attrs) {
-                        if (_.isUndefined(jids)) {
-                            throw new TypeError('rooms.open: You need to provide at least one JID');
-                        } else if (_.isString(jids)) {
-                            return _converse.api.rooms.create(jids, attrs).trigger('show');
-                        }
-                        return _.map(jids, (jid) => _converse.api.rooms.create(jid, attrs).trigger('show'));
+                        return new Promise((resolve, reject) => {
+                            _converse.api.waitUntil('chatBoxesFetched').then(() => {
+                                if (_.isUndefined(jids)) {
+                                    const err_msg = 'rooms.open: You need to provide at least one JID';
+                                    _converse.log(err_msg, Strophe.LogLevel.ERROR);
+                                    reject(new TypeError(err_msg));
+                                } else if (_.isString(jids)) {
+                                    resolve(_converse.api.rooms.create(jids, attrs).trigger('show'));
+                                } else {
+                                    resolve(_.map(jids, (jid) => _converse.api.rooms.create(jid, attrs).trigger('show')));
+                                }
+                            });
+                        });
                     },
                     },
+
                     'get' (jids, attrs, create) {
                     'get' (jids, attrs, create) {
                         if (_.isString(attrs)) {
                         if (_.isString(attrs)) {
                             attrs = {'nick': attrs};
                             attrs = {'nick': attrs};

+ 7 - 2
src/i18n.js

@@ -148,8 +148,13 @@
                 );
                 );
                 xhr.onload = function () {
                 xhr.onload = function () {
                     if (xhr.status >= 200 && xhr.status < 400) {
                     if (xhr.status >= 200 && xhr.status < 400) {
-                        jed_instance = new Jed(window.JSON.parse(xhr.responseText));
-                        resolve();
+                        try {
+                            const data = window.JSON.parse(xhr.responseText);
+                            jed_instance = new Jed(data);
+                            resolve();
+                        } catch (e) {
+                            xhr.onerror(e);
+                        }
                     } else {
                     } else {
                         xhr.onerror();
                         xhr.onerror();
                     }
                     }

+ 54 - 53
tests/utils.js

@@ -98,8 +98,9 @@
         return views;
         return views;
     };
     };
 
 
-    utils.openChatBoxFor = function (converse, jid) {
-        return converse.roster.get(jid).trigger("open");
+    utils.openChatBoxFor = function (_converse, jid) {
+        _converse.roster.get(jid).trigger("open");
+        return utils.waitUntil(() => _converse.chatboxviews.get(jid));
     };
     };
 
 
     utils.openChatRoomViaModal = function (_converse, jid, nick) {
     utils.openChatRoomViaModal = function (_converse, jid, nick) {
@@ -121,64 +122,64 @@
     };
     };
 
 
     utils.openChatRoom = function (_converse, room, server, nick) {
     utils.openChatRoom = function (_converse, room, server, nick) {
-        _converse.api.rooms.open(`${room}@${server}`);
+        return _converse.api.rooms.open(`${room}@${server}`);
     };
     };
 
 
     utils.openAndEnterChatRoom = function (_converse, room, server, nick) {
     utils.openAndEnterChatRoom = function (_converse, room, server, nick) {
         let last_stanza;
         let last_stanza;
 
 
-        return new Promise(function (resolve, reject) {
-            _converse.api.rooms.open(`${room}@${server}`);
-            const view = _converse.chatboxviews.get((room+'@'+server).toLowerCase());
-            // We pretend this is a new room, so no disco info is returned.
-            let last_stanza = _.last(_converse.connection.IQ_stanzas).nodeTree;
-            const IQ_id = last_stanza.getAttribute('id');
-            const features_stanza = $iq({
-                    'from': room+'@'+server,
-                    'id': IQ_id,
-                    'to': nick+'@'+server,
-                    'type': 'error'
-                }).c('error', {'type': 'cancel'})
-                    .c('item-not-found', {'xmlns': "urn:ietf:params:xml:ns:xmpp-stanzas"});
-            _converse.connection._dataRecv(utils.createRequest(features_stanza));
-
-            utils.waitUntil(() => {
-                return _.filter(
-                    _converse.connection.IQ_stanzas, (node) => {
-                        const query = node.nodeTree.querySelector('query');
-                        if (query && query.getAttribute('node') === 'x-roomuser-item') {
-                            last_stanza = node.nodeTree;
-                            return true;
-                        }
-                    }).length
-            }).then(function () {
-                // The XMPP server returns the reserved nick for this user.
+        return new Promise((resolve, reject) => {
+            return _converse.api.rooms.open(`${room}@${server}`).then(() => {
+                const view = _converse.chatboxviews.get((room+'@'+server).toLowerCase());
+                // We pretend this is a new room, so no disco info is returned.
+                let last_stanza = _.last(_converse.connection.IQ_stanzas).nodeTree;
                 const IQ_id = last_stanza.getAttribute('id');
                 const IQ_id = last_stanza.getAttribute('id');
-                const stanza = $iq({
-                    'type': 'result',
-                    'id': IQ_id,
-                    'from': view.model.get('jid'),
-                    'to': _converse.connection.jid 
-                }).c('query', {'xmlns': 'http://jabber.org/protocol/disco#info', 'node': 'x-roomuser-item'})
-                    .c('identity', {'category': 'conference', 'name': nick, 'type': 'text'});
-                _converse.connection._dataRecv(utils.createRequest(stanza));
-                // The user has just entered the room (because join was called)
-                // and receives their own presence from the server.
-                // See example 24: http://xmpp.org/extensions/xep-0045.html#enter-pres
-                var presence = $pres({
-                        to: _converse.connection.jid,
-                        from: room+'@'+server+'/'+nick,
-                        id: 'DC352437-C019-40EC-B590-AF29E879AF97'
-                }).c('x').attrs({xmlns:'http://jabber.org/protocol/muc#user'})
-                    .c('item').attrs({
-                        affiliation: 'member',
-                        jid: _converse.bare_jid,
-                        role: 'participant'
-                    }).up()
-                    .c('status').attrs({code:'110'});
-                _converse.connection._dataRecv(utils.createRequest(presence));
-                resolve();
+                const features_stanza = $iq({
+                        'from': room+'@'+server,
+                        'id': IQ_id,
+                        'to': nick+'@'+server,
+                        'type': 'error'
+                    }).c('error', {'type': 'cancel'})
+                        .c('item-not-found', {'xmlns': "urn:ietf:params:xml:ns:xmpp-stanzas"});
+                _converse.connection._dataRecv(utils.createRequest(features_stanza));
 
 
+                utils.waitUntil(() => {
+                    return _.filter(
+                        _converse.connection.IQ_stanzas, (node) => {
+                            const query = node.nodeTree.querySelector('query');
+                            if (query && query.getAttribute('node') === 'x-roomuser-item') {
+                                last_stanza = node.nodeTree;
+                                return true;
+                            }
+                        }).length
+                }).then(function () {
+                    // The XMPP server returns the reserved nick for this user.
+                    const IQ_id = last_stanza.getAttribute('id');
+                    const stanza = $iq({
+                        'type': 'result',
+                        'id': IQ_id,
+                        'from': view.model.get('jid'),
+                        'to': _converse.connection.jid 
+                    }).c('query', {'xmlns': 'http://jabber.org/protocol/disco#info', 'node': 'x-roomuser-item'})
+                        .c('identity', {'category': 'conference', 'name': nick, 'type': 'text'});
+                    _converse.connection._dataRecv(utils.createRequest(stanza));
+                    // The user has just entered the room (because join was called)
+                    // and receives their own presence from the server.
+                    // See example 24: http://xmpp.org/extensions/xep-0045.html#enter-pres
+                    var presence = $pres({
+                            to: _converse.connection.jid,
+                            from: room+'@'+server+'/'+nick,
+                            id: 'DC352437-C019-40EC-B590-AF29E879AF97'
+                    }).c('x').attrs({xmlns:'http://jabber.org/protocol/muc#user'})
+                        .c('item').attrs({
+                            affiliation: 'member',
+                            jid: _converse.bare_jid,
+                            role: 'participant'
+                        }).up()
+                        .c('status').attrs({code:'110'});
+                    _converse.connection._dataRecv(utils.createRequest(presence));
+                    resolve();
+                }).catch(_.partial(console.error, _));
             }).catch(_.partial(console.error, _));
             }).catch(_.partial(console.error, _));
         }).catch(_.partial(console.error, _));
         }).catch(_.partial(console.error, _));
     };
     };

Some files were not shown because too many files changed in this diff