Fix disco hierarchy
Previously we kept all entities and their items (which are also instances of _converse.DiscoEntity) in a flat array. Instead, we should have a tree-like structure where items are stored on the relevant entity (and recursively on other items).
This commit is contained in:
parent
77a51cc2a6
commit
a9d2881888
@ -30,7 +30,7 @@
|
|||||||
"accessor-pairs": "error",
|
"accessor-pairs": "error",
|
||||||
"array-bracket-spacing": "off",
|
"array-bracket-spacing": "off",
|
||||||
"array-callback-return": "error",
|
"array-callback-return": "error",
|
||||||
"arrow-body-style": "error",
|
"arrow-body-style": "off",
|
||||||
"arrow-parens": "error",
|
"arrow-parens": "error",
|
||||||
"arrow-spacing": "error",
|
"arrow-spacing": "error",
|
||||||
"block-scoped-var": "off",
|
"block-scoped-var": "off",
|
||||||
|
178
spec/disco.js
178
spec/disco.js
@ -20,7 +20,7 @@
|
|||||||
var IQ_ids = _converse.connection.IQ_ids;
|
var IQ_ids = _converse.connection.IQ_ids;
|
||||||
test_utils.waitUntil(function () {
|
test_utils.waitUntil(function () {
|
||||||
return _.filter(IQ_stanzas, function (iq) {
|
return _.filter(IQ_stanzas, function (iq) {
|
||||||
return iq.nodeTree.querySelector('query[xmlns="http://jabber.org/protocol/disco#info"]');
|
return iq.nodeTree.querySelector('iq[to="localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]');
|
||||||
}).length > 0;
|
}).length > 0;
|
||||||
}, 300).then(function () {
|
}, 300).then(function () {
|
||||||
/* <iq type='result'
|
/* <iq type='result'
|
||||||
@ -49,8 +49,11 @@
|
|||||||
* </query>
|
* </query>
|
||||||
* </iq>
|
* </iq>
|
||||||
*/
|
*/
|
||||||
var info_IQ_id = IQ_ids[0];
|
var stanza = _.filter(IQ_stanzas, function (iq) {
|
||||||
var stanza = $iq({
|
return iq.nodeTree.querySelector('iq[to="localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]');
|
||||||
|
})[0];
|
||||||
|
var info_IQ_id = IQ_ids[IQ_stanzas.indexOf(stanza)];
|
||||||
|
stanza = $iq({
|
||||||
'type': 'result',
|
'type': 'result',
|
||||||
'from': 'localhost',
|
'from': 'localhost',
|
||||||
'to': 'dummy@localhost/resource',
|
'to': 'dummy@localhost/resource',
|
||||||
@ -79,92 +82,97 @@
|
|||||||
'var': 'jabber:iq:version'});
|
'var': 'jabber:iq:version'});
|
||||||
_converse.connection._dataRecv(test_utils.createRequest(stanza));
|
_converse.connection._dataRecv(test_utils.createRequest(stanza));
|
||||||
|
|
||||||
var entities = _converse.disco_entities;
|
_converse.api.disco.entities.get().then(function (entities) {
|
||||||
expect(entities.length).toBe(2); // We have an extra entity, which is the user's JID
|
expect(entities.length).toBe(2); // We have an extra entity, which is the user's JID
|
||||||
expect(entities.get(_converse.domain).features.length).toBe(5);
|
expect(entities.get(_converse.domain).features.length).toBe(5);
|
||||||
expect(entities.get(_converse.domain).identities.length).toBe(3);
|
expect(entities.get(_converse.domain).identities.length).toBe(3);
|
||||||
expect(entities.get('localhost').features.where({'var': 'jabber:iq:version'}).length).toBe(1);
|
expect(entities.get('localhost').features.where({'var': 'jabber:iq:version'}).length).toBe(1);
|
||||||
expect(entities.get('localhost').features.where({'var': 'jabber:iq:time'}).length).toBe(1);
|
expect(entities.get('localhost').features.where({'var': 'jabber:iq:time'}).length).toBe(1);
|
||||||
expect(entities.get('localhost').features.where({'var': 'jabber:iq:register'}).length).toBe(1);
|
expect(entities.get('localhost').features.where({'var': 'jabber:iq:register'}).length).toBe(1);
|
||||||
expect(entities.get('localhost').features.where(
|
expect(entities.get('localhost').features.where(
|
||||||
{'var': 'http://jabber.org/protocol/disco#items'}).length).toBe(1);
|
{'var': 'http://jabber.org/protocol/disco#items'}).length).toBe(1);
|
||||||
expect(entities.get('localhost').features.where(
|
expect(entities.get('localhost').features.where(
|
||||||
{'var': 'http://jabber.org/protocol/disco#info'}).length).toBe(1);
|
{'var': 'http://jabber.org/protocol/disco#info'}).length).toBe(1);
|
||||||
|
|
||||||
|
|
||||||
test_utils.waitUntil(function () {
|
test_utils.waitUntil(function () {
|
||||||
// Converse.js sees that the entity has a disco#items feature,
|
// Converse.js sees that the entity has a disco#items feature,
|
||||||
// so it will make a query for it.
|
// so it will make a query for it.
|
||||||
return _.filter(IQ_stanzas, function (iq) {
|
return _.filter(IQ_stanzas, function (iq) {
|
||||||
return iq.nodeTree.querySelector('query[xmlns="http://jabber.org/protocol/disco#items"]');
|
return iq.nodeTree.querySelector('query[xmlns="http://jabber.org/protocol/disco#items"]');
|
||||||
}).length > 0;
|
}).length > 0;
|
||||||
}, 300).then(function () {
|
}, 300).then(function () {
|
||||||
/* <iq type='result'
|
/* <iq type='result'
|
||||||
* from='catalog.shakespeare.lit'
|
* from='catalog.shakespeare.lit'
|
||||||
* to='romeo@montague.net/orchard'
|
* to='romeo@montague.net/orchard'
|
||||||
* id='items2'>
|
* id='items2'>
|
||||||
* <query xmlns='http://jabber.org/protocol/disco#items'>
|
* <query xmlns='http://jabber.org/protocol/disco#items'>
|
||||||
* <item jid='people.shakespeare.lit'
|
* <item jid='people.shakespeare.lit'
|
||||||
* name='Directory of Characters'/>
|
* name='Directory of Characters'/>
|
||||||
* <item jid='plays.shakespeare.lit'
|
* <item jid='plays.shakespeare.lit'
|
||||||
* name='Play-Specific Chatrooms'/>
|
* name='Play-Specific Chatrooms'/>
|
||||||
* <item jid='mim.shakespeare.lit'
|
* <item jid='mim.shakespeare.lit'
|
||||||
* name='Gateway to Marlowe IM'/>
|
* name='Gateway to Marlowe IM'/>
|
||||||
* <item jid='words.shakespeare.lit'
|
* <item jid='words.shakespeare.lit'
|
||||||
* name='Shakespearean Lexicon'/>
|
* name='Shakespearean Lexicon'/>
|
||||||
*
|
*
|
||||||
* <item jid='catalog.shakespeare.lit'
|
* <item jid='catalog.shakespeare.lit'
|
||||||
* node='books'
|
* node='books'
|
||||||
* name='Books by and about Shakespeare'/>
|
* name='Books by and about Shakespeare'/>
|
||||||
* <item jid='catalog.shakespeare.lit'
|
* <item jid='catalog.shakespeare.lit'
|
||||||
* node='clothing'
|
* node='clothing'
|
||||||
* name='Wear your literary taste with pride'/>
|
* name='Wear your literary taste with pride'/>
|
||||||
* <item jid='catalog.shakespeare.lit'
|
* <item jid='catalog.shakespeare.lit'
|
||||||
* node='music'
|
* node='music'
|
||||||
* name='Music from the time of Shakespeare'/>
|
* name='Music from the time of Shakespeare'/>
|
||||||
* </query>
|
* </query>
|
||||||
* </iq>
|
* </iq>
|
||||||
*/
|
*/
|
||||||
var items_IQ_id = IQ_ids.pop();
|
var stanza = _.filter(IQ_stanzas, function (iq) {
|
||||||
stanza = $iq({
|
return iq.nodeTree.querySelector('iq[to="localhost"] query[xmlns="http://jabber.org/protocol/disco#items"]');
|
||||||
'type': 'result',
|
})[0];
|
||||||
'from': 'localhost',
|
var items_IQ_id = IQ_ids[IQ_stanzas.indexOf(stanza)];
|
||||||
'to': 'dummy@localhost/resource',
|
stanza = $iq({
|
||||||
'id': items_IQ_id
|
'type': 'result',
|
||||||
}).c('query', {'xmlns': 'http://jabber.org/protocol/disco#items'})
|
'from': 'localhost',
|
||||||
.c('item', {
|
'to': 'dummy@localhost/resource',
|
||||||
'jid': 'people.shakespeare.lit',
|
'id': items_IQ_id
|
||||||
'name': 'Directory of Characters'}).up()
|
}).c('query', {'xmlns': 'http://jabber.org/protocol/disco#items'})
|
||||||
.c('item', {
|
.c('item', {
|
||||||
'jid': 'plays.shakespeare.lit',
|
'jid': 'people.shakespeare.lit',
|
||||||
'name': 'Play-Specific Chatrooms'}).up()
|
'name': 'Directory of Characters'}).up()
|
||||||
.c('item', {
|
.c('item', {
|
||||||
'jid': 'words.shakespeare.lit',
|
'jid': 'plays.shakespeare.lit',
|
||||||
'name': 'Gateway to Marlowe IM'}).up()
|
'name': 'Play-Specific Chatrooms'}).up()
|
||||||
.c('item', {
|
.c('item', {
|
||||||
'jid': 'localhost',
|
'jid': 'words.shakespeare.lit',
|
||||||
'name': 'Shakespearean Lexicon'}).up()
|
'name': 'Gateway to Marlowe IM'}).up()
|
||||||
|
|
||||||
.c('item', {
|
.c('item', {
|
||||||
'jid': 'localhost',
|
'jid': 'localhost',
|
||||||
'node': 'books',
|
'node': 'books',
|
||||||
'name': 'Books by and about Shakespeare'}).up()
|
'name': 'Books by and about Shakespeare'}).up()
|
||||||
.c('item', {
|
.c('item', {
|
||||||
'node': 'localhost',
|
'node': 'localhost',
|
||||||
'name': 'Wear your literary taste with pride'}).up()
|
'name': 'Wear your literary taste with pride'}).up()
|
||||||
.c('item', {
|
.c('item', {
|
||||||
'jid': 'localhost',
|
'jid': 'localhost',
|
||||||
'node': 'music',
|
'node': 'music',
|
||||||
'name': 'Music from the time of Shakespeare'
|
'name': 'Music from the time of Shakespeare'
|
||||||
});
|
});
|
||||||
_converse.connection._dataRecv(test_utils.createRequest(stanza));
|
_converse.connection._dataRecv(test_utils.createRequest(stanza));
|
||||||
|
|
||||||
entities = _converse.disco_entities;
|
entities = _converse.disco_entities;
|
||||||
expect(entities.length).toBe(5); // We have an extra entity, which is the user's JID
|
expect(entities.length).toBe(2); // We have an extra entity, which is the user's JID
|
||||||
expect(entities.get(_converse.domain).identities.where({'category': 'conference'}).length).toBe(1);
|
expect(entities.get(_converse.domain).items.length).toBe(3);
|
||||||
expect(entities.get(_converse.domain).identities.where({'category': 'directory'}).length).toBe(1);
|
expect(_.includes(entities.get(_converse.domain).items.pluck('jid'), 'people.shakespeare.lit')).toBeTruthy();
|
||||||
done();
|
expect(_.includes(entities.get(_converse.domain).items.pluck('jid'), 'plays.shakespeare.lit')).toBeTruthy();
|
||||||
});
|
expect(_.includes(entities.get(_converse.domain).items.pluck('jid'), 'words.shakespeare.lit')).toBeTruthy();
|
||||||
|
expect(entities.get(_converse.domain).identities.where({'category': 'conference'}).length).toBe(1);
|
||||||
|
expect(entities.get(_converse.domain).identities.where({'category': 'directory'}).length).toBe(1);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
@ -10,6 +10,7 @@
|
|||||||
var Strophe = converse.env.Strophe;
|
var Strophe = converse.env.Strophe;
|
||||||
var $iq = converse.env.$iq;
|
var $iq = converse.env.$iq;
|
||||||
var _ = converse.env._;
|
var _ = converse.env._;
|
||||||
|
var f = converse.env.f;
|
||||||
|
|
||||||
describe("XEP-0363: HTTP File Upload", function () {
|
describe("XEP-0363: HTTP File Upload", function () {
|
||||||
|
|
||||||
@ -18,135 +19,171 @@
|
|||||||
it("is done automatically", mock.initConverseWithAsync(function (done, _converse) {
|
it("is done automatically", mock.initConverseWithAsync(function (done, _converse) {
|
||||||
var IQ_stanzas = _converse.connection.IQ_stanzas;
|
var IQ_stanzas = _converse.connection.IQ_stanzas;
|
||||||
var IQ_ids = _converse.connection.IQ_ids;
|
var IQ_ids = _converse.connection.IQ_ids;
|
||||||
test_utils.waitUntil(function () {
|
|
||||||
return _.filter(IQ_stanzas, function (iq) {
|
test_utils.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, [], []).then(function () {
|
||||||
return iq.nodeTree.querySelector('query[xmlns="http://jabber.org/protocol/disco#info"]');
|
test_utils.waitUntil(function () {
|
||||||
}).length > 0;
|
return _.filter(IQ_stanzas, function (iq) {
|
||||||
}, 300).then(function () {
|
return iq.nodeTree.querySelector('iq[to="localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]');
|
||||||
/* <iq type='result'
|
}).length > 0;
|
||||||
* from='plays.shakespeare.lit'
|
}, 300).then(function () {
|
||||||
* to='romeo@montague.net/orchard'
|
/* <iq type='result'
|
||||||
* id='info1'>
|
* from='plays.shakespeare.lit'
|
||||||
* <query xmlns='http://jabber.org/protocol/disco#info'>
|
* to='romeo@montague.net/orchard'
|
||||||
* <identity
|
* id='info1'>
|
||||||
* category='server'
|
* <query xmlns='http://jabber.org/protocol/disco#info'>
|
||||||
* type='im'/>
|
* <identity
|
||||||
* <feature var='http://jabber.org/protocol/disco#info'/>
|
* category='server'
|
||||||
* <feature var='http://jabber.org/protocol/disco#items'/>
|
* type='im'/>
|
||||||
* </query>
|
* <feature var='http://jabber.org/protocol/disco#info'/>
|
||||||
* </iq>
|
* <feature var='http://jabber.org/protocol/disco#items'/>
|
||||||
*/
|
* </query>
|
||||||
var info_IQ_id = IQ_ids[0];
|
* </iq>
|
||||||
var stanza = $iq({
|
*/
|
||||||
|
var stanza = _.filter(IQ_stanzas, function (iq) {
|
||||||
|
return iq.nodeTree.querySelector('iq[to="localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]');
|
||||||
|
})[0];
|
||||||
|
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 () {
|
||||||
|
/* <iq from='montague.tld'
|
||||||
|
* id='step_01'
|
||||||
|
* to='romeo@montague.tld/garden'
|
||||||
|
* type='result'>
|
||||||
|
* <query xmlns='http://jabber.org/protocol/disco#items'>
|
||||||
|
* <item jid='upload.montague.tld' name='HTTP File Upload' />
|
||||||
|
* <item jid='conference.montague.tld' name='Chatroom Service' />
|
||||||
|
* </query>
|
||||||
|
* </iq>
|
||||||
|
*/
|
||||||
|
var stanza = _.filter(IQ_stanzas, function (iq) {
|
||||||
|
return iq.nodeTree.querySelector('iq[to="localhost"] query[xmlns="http://jabber.org/protocol/disco#items"]');
|
||||||
|
})[0];
|
||||||
|
var items_IQ_id = IQ_ids[IQ_stanzas.indexOf(stanza)];
|
||||||
|
stanza = $iq({
|
||||||
'type': 'result',
|
'type': 'result',
|
||||||
'from': 'localhost',
|
'from': 'localhost',
|
||||||
'to': 'dummy@localhost/resource',
|
'to': 'dummy@localhost/resource',
|
||||||
'id': info_IQ_id
|
'id': items_IQ_id
|
||||||
}).c('query', {'xmlns': 'http://jabber.org/protocol/disco#info'})
|
}).c('query', {'xmlns': 'http://jabber.org/protocol/disco#items'})
|
||||||
.c('identity', {
|
.c('item', {
|
||||||
'category': 'server',
|
'jid': 'upload.localhost',
|
||||||
'type': 'im'}).up()
|
'name': 'HTTP File Upload'});
|
||||||
.c('feature', {
|
_converse.connection._dataRecv(test_utils.createRequest(stanza));
|
||||||
'var': 'http://jabber.org/protocol/disco#info'}).up()
|
|
||||||
.c('feature', {
|
|
||||||
'var': 'http://jabber.org/protocol/disco#items'});
|
|
||||||
_converse.connection._dataRecv(test_utils.createRequest(stanza));
|
|
||||||
|
|
||||||
var entities = _converse.disco_entities;
|
_converse.api.disco.entities.get().then(function (entities) {
|
||||||
expect(entities.length).toBe(2); // We have an extra entity, which is the user's JID
|
expect(entities.length).toBe(2);
|
||||||
expect(entities.get(_converse.domain).features.length).toBe(2);
|
expect(entities.get('localhost').items.length).toBe(1);
|
||||||
expect(entities.get(_converse.domain).identities.length).toBe(1);
|
return test_utils.waitUntil(function () {
|
||||||
|
// Converse.js sees that the entity has a disco#info feature,
|
||||||
return test_utils.waitUntil(function () {
|
// so it will make a query for it.
|
||||||
// Converse.js sees that the entity has a disco#items feature,
|
return _.filter(IQ_stanzas, function (iq) {
|
||||||
// so it will make a query for it.
|
return iq.nodeTree.querySelector('iq[to="upload.localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]');
|
||||||
return _.filter(IQ_stanzas, function (iq) {
|
}).length > 0;
|
||||||
return iq.nodeTree.querySelector('query[xmlns="http://jabber.org/protocol/disco#items"]');
|
}, 300);
|
||||||
}).length > 0;
|
});
|
||||||
}, 300);
|
}).then(function () {
|
||||||
}).then(function () {
|
var stanza = _.filter(IQ_stanzas, function (iq) {
|
||||||
/* <iq from='montague.tld'
|
|
||||||
* id='step_01'
|
|
||||||
* to='romeo@montague.tld/garden'
|
|
||||||
* type='result'>
|
|
||||||
* <query xmlns='http://jabber.org/protocol/disco#items'>
|
|
||||||
* <item jid='upload.montague.tld' name='HTTP File Upload' />
|
|
||||||
* <item jid='conference.montague.tld' name='Chatroom Service' />
|
|
||||||
* </query>
|
|
||||||
* </iq>
|
|
||||||
*/
|
|
||||||
var items_IQ_id = IQ_ids[IQ_ids.length-1];
|
|
||||||
var stanza = $iq({
|
|
||||||
'type': 'result',
|
|
||||||
'from': 'localhost',
|
|
||||||
'to': 'dummy@localhost/resource',
|
|
||||||
'id': items_IQ_id
|
|
||||||
}).c('query', {'xmlns': 'http://jabber.org/protocol/disco#items'})
|
|
||||||
.c('item', {
|
|
||||||
'jid': 'upload.localhost',
|
|
||||||
'name': 'HTTP File Upload'}).up()
|
|
||||||
.c('item', {
|
|
||||||
'jid': 'conference.localhost',
|
|
||||||
'name': 'Chatrooms Service'});
|
|
||||||
_converse.connection._dataRecv(test_utils.createRequest(stanza));
|
|
||||||
|
|
||||||
var entities = _converse.disco_entities;
|
|
||||||
expect(entities.length).toBe(4); // We have an extra entity, which is the user's JID
|
|
||||||
|
|
||||||
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="upload.localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]');
|
return iq.nodeTree.querySelector('iq[to="upload.localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]');
|
||||||
}).length > 0;
|
})[0];
|
||||||
}, 300);
|
var IQ_id = IQ_ids[IQ_stanzas.indexOf(stanza)];
|
||||||
}).then(function () {
|
expect(stanza.toLocaleString()).toBe(
|
||||||
var stanza = _.filter(IQ_stanzas, function (iq) {
|
"<iq from='dummy@localhost/resource' to='upload.localhost' type='get' xmlns='jabber:client' id='"+IQ_id+"'>"+
|
||||||
return iq.nodeTree.querySelector('iq[to="upload.localhost"] query[xmlns="http://jabber.org/protocol/disco#info"]');
|
"<query xmlns='http://jabber.org/protocol/disco#info'/>"+
|
||||||
})[0];
|
"</iq>");
|
||||||
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
|
// Upload service responds and reports a maximum file size of 5MiB
|
||||||
/* <iq from='upload.montague.tld'
|
/* <iq from='upload.montague.tld'
|
||||||
* id='step_02'
|
* id='step_02'
|
||||||
* to='romeo@montague.tld/garden'
|
* to='romeo@montague.tld/garden'
|
||||||
* type='result'>
|
* type='result'>
|
||||||
* <query xmlns='http://jabber.org/protocol/disco#info'>
|
* <query xmlns='http://jabber.org/protocol/disco#info'>
|
||||||
* <identity category='store'
|
* <identity category='store'
|
||||||
* type='file'
|
* type='file'
|
||||||
* name='HTTP File Upload' />
|
* name='HTTP File Upload' />
|
||||||
* <feature var='urn:xmpp:http:upload:0' />
|
* <feature var='urn:xmpp:http:upload:0' />
|
||||||
* <x type='result' xmlns='jabber:x:data'>
|
* <x type='result' xmlns='jabber:x:data'>
|
||||||
* <field var='FORM_TYPE' type='hidden'>
|
* <field var='FORM_TYPE' type='hidden'>
|
||||||
* <value>urn:xmpp:http:upload:0</value>
|
* <value>urn:xmpp:http:upload:0</value>
|
||||||
* </field>
|
* </field>
|
||||||
* <field var='max-file-size'>
|
* <field var='max-file-size'>
|
||||||
* <value>5242880</value>
|
* <value>5242880</value>
|
||||||
* </field>
|
* </field>
|
||||||
* </x>
|
* </x>
|
||||||
* </query>
|
* </query>
|
||||||
* </iq>
|
* </iq>
|
||||||
*/
|
*/
|
||||||
stanza = $iq({'type': 'result', 'to': 'dummy@localhost/resource', 'id': IQ_id, 'from': 'upload.localhost'})
|
stanza = $iq({'type': 'result', 'to': 'dummy@localhost/resource', 'id': IQ_id, 'from': 'upload.localhost'})
|
||||||
.c('query', {'xmlns': 'http://jabber.org/protocol/disco#info'})
|
.c('query', {'xmlns': 'http://jabber.org/protocol/disco#info'})
|
||||||
.c('identity', {'category':'store', 'type':'file', 'name':'HTTP File Upload'}).up()
|
.c('identity', {'category':'store', 'type':'file', 'name':'HTTP File Upload'}).up()
|
||||||
.c('feature', {'var':'urn:xmpp:http:upload:0'}).up()
|
.c('feature', {'var':'urn:xmpp:http:upload:0'}).up()
|
||||||
.c('x', {'type':'result', 'xmlns':'jabber:x:data'})
|
.c('x', {'type':'result', 'xmlns':'jabber:x:data'})
|
||||||
.c('field', {'var':'FORM_TYPE', 'type':'hidden'})
|
.c('field', {'var':'FORM_TYPE', 'type':'hidden'})
|
||||||
.c('value').t('urn:xmpp:http:upload:0').up().up()
|
.c('value').t('urn:xmpp:http:upload:0').up().up()
|
||||||
.c('field', {'var':'max-file-size'})
|
.c('field', {'var':'max-file-size'})
|
||||||
.c('value').t('5242880');
|
.c('value').t('5242880');
|
||||||
_converse.connection._dataRecv(test_utils.createRequest(stanza));
|
_converse.connection._dataRecv(test_utils.createRequest(stanza));
|
||||||
|
|
||||||
var entities = _converse.disco_entities;
|
_converse.api.disco.entities.get().then(function (entities) {
|
||||||
expect(entities.get('upload.localhost').identities.where({'category': 'store'}).length).toBe(1);
|
expect(entities.get('localhost').items.get('upload.localhost').identities.where({'category': 'store'}).length).toBe(1);
|
||||||
done();
|
_converse.api.disco.supports(Strophe.NS.HTTPUPLOAD, _converse.domain).then(
|
||||||
|
function (result) {
|
||||||
|
expect(result.length).toBe(1);
|
||||||
|
expect(result[0].get('jid')).toBe('upload.localhost');
|
||||||
|
done();
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
|
||||||
|
})
|
||||||
})
|
})
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("When supported", function () {
|
||||||
|
|
||||||
|
describe("A file upload toolbar button", function () {
|
||||||
|
|
||||||
|
it("appears in private chats", mock.initConverseWithAsync(function (done, _converse) {
|
||||||
|
test_utils.createContacts(_converse, 'current');
|
||||||
|
var contact_jid = mock.cur_names[2].replace(/ /g,'.').toLowerCase() + '@localhost';
|
||||||
|
test_utils.openChatBoxFor(_converse, contact_jid);
|
||||||
|
done();
|
||||||
|
}));
|
||||||
|
|
||||||
|
it("appears in MUC chats", mock.initConverseWithAsync(function (done, _converse) {
|
||||||
|
done();
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
1044
spec/mam.js
1044
spec/mam.js
File diff suppressed because it is too large
Load Diff
@ -550,7 +550,7 @@
|
|||||||
_converse.api.disco.getIdentity('pubsub', 'pep', _converse.bare_jid),
|
_converse.api.disco.getIdentity('pubsub', 'pep', _converse.bare_jid),
|
||||||
_converse.api.disco.supports(Strophe.NS.PUBSUB+'#publish-options', _converse.bare_jid)
|
_converse.api.disco.supports(Strophe.NS.PUBSUB+'#publish-options', _converse.bare_jid)
|
||||||
]).then((args) => {
|
]).then((args) => {
|
||||||
resolve(args[0] && (args[1].supported || _converse.allow_public_bookmarks));
|
resolve(args[0] && (args[1].length || _converse.allow_public_bookmarks));
|
||||||
}).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
|
}).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
|
||||||
}).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
|
}).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
|
||||||
}
|
}
|
||||||
|
@ -341,8 +341,7 @@
|
|||||||
Promise.all(_.map(_.keys(resources), (resource) =>
|
Promise.all(_.map(_.keys(resources), (resource) =>
|
||||||
_converse.api.disco.supports(Strophe.NS.SPOILER, `${contact_jid}/${resource}`)
|
_converse.api.disco.supports(Strophe.NS.SPOILER, `${contact_jid}/${resource}`)
|
||||||
)).then((results) => {
|
)).then((results) => {
|
||||||
const supported = _.every(f.map(f.get('supported'))(results));
|
if (results.length) {
|
||||||
if (supported) {
|
|
||||||
const html = tpl_spoiler_button(this.model.toJSON());
|
const html = tpl_spoiler_button(this.model.toJSON());
|
||||||
if (_converse.visible_toolbar_buttons.emoji) {
|
if (_converse.visible_toolbar_buttons.emoji) {
|
||||||
this.el.querySelector('.toggle-smiley').insertAdjacentHTML('afterEnd', html);
|
this.el.querySelector('.toggle-smiley').insertAdjacentHTML('afterEnd', html);
|
||||||
|
@ -12,7 +12,7 @@
|
|||||||
define(["converse-core", "sizzle", "strophe.disco"], factory);
|
define(["converse-core", "sizzle", "strophe.disco"], factory);
|
||||||
}(this, function (converse, sizzle) {
|
}(this, function (converse, sizzle) {
|
||||||
|
|
||||||
const { Backbone, Promise, Strophe, b64_sha1, utils, _ } = converse.env;
|
const { Backbone, Promise, Strophe, b64_sha1, utils, _, f } = converse.env;
|
||||||
|
|
||||||
converse.plugins.add('converse-disco', {
|
converse.plugins.add('converse-disco', {
|
||||||
|
|
||||||
@ -22,24 +22,10 @@
|
|||||||
*/
|
*/
|
||||||
const { _converse } = this;
|
const { _converse } = this;
|
||||||
|
|
||||||
function onDiscoItems (stanza) {
|
|
||||||
_.each(stanza.querySelectorAll('query item'), (item) => {
|
|
||||||
if (item.getAttribute("node")) {
|
|
||||||
// XXX: ignore nodes for now.
|
|
||||||
// See: https://xmpp.org/extensions/xep-0030.html#items-nodes
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const jid = item.getAttribute('jid');
|
|
||||||
const entities = _converse.disco_entities;
|
|
||||||
if (_.isUndefined(entities.get(jid))) {
|
|
||||||
entities.create({'jid': jid});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Promises exposed by this plugin
|
// Promises exposed by this plugin
|
||||||
_converse.api.promises.add('discoInitialized');
|
_converse.api.promises.add('discoInitialized');
|
||||||
|
|
||||||
|
|
||||||
_converse.DiscoEntity = Backbone.Model.extend({
|
_converse.DiscoEntity = Backbone.Model.extend({
|
||||||
/* A Disco Entity is a JID addressable entity that can be queried
|
/* A Disco Entity is a JID addressable entity that can be queried
|
||||||
* for features.
|
* for features.
|
||||||
@ -63,6 +49,10 @@
|
|||||||
);
|
);
|
||||||
this.fetchFeatures();
|
this.fetchFeatures();
|
||||||
|
|
||||||
|
this.items = new _converse.DiscoEntities();
|
||||||
|
this.items.browserStorage = new Backbone.BrowserStorage[_converse.storage](
|
||||||
|
b64_sha1(`converse.disco-items-${this.get('jid')}`)
|
||||||
|
);
|
||||||
},
|
},
|
||||||
|
|
||||||
getIdentity (category, type) {
|
getIdentity (category, type) {
|
||||||
@ -98,17 +88,10 @@
|
|||||||
const entity = this;
|
const entity = this;
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
function fulfillPromise () {
|
function fulfillPromise () {
|
||||||
const model = entity.features.findWhere({'var': feature });
|
if (entity.features.findWhere({'var': feature})) {
|
||||||
if (model) {
|
resolve(entity);
|
||||||
resolve({
|
|
||||||
'supported': true,
|
|
||||||
'feature': model
|
|
||||||
});
|
|
||||||
} else {
|
} else {
|
||||||
resolve({
|
resolve();
|
||||||
'supported': false,
|
|
||||||
'feature': null
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
entity.waitUntilFeaturesDiscovered
|
entity.waitUntilFeaturesDiscovered
|
||||||
@ -141,13 +124,27 @@
|
|||||||
_converse.connection.disco.info(this.get('jid'), null, this.onInfo.bind(this));
|
_converse.connection.disco.info(this.get('jid'), null, this.onInfo.bind(this));
|
||||||
},
|
},
|
||||||
|
|
||||||
|
onDiscoItems (stanza) {
|
||||||
|
_.each(stanza.querySelectorAll('query item'), (item) => {
|
||||||
|
if (item.getAttribute("node")) {
|
||||||
|
// XXX: ignore nodes for now.
|
||||||
|
// See: https://xmpp.org/extensions/xep-0030.html#items-nodes
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const jid = item.getAttribute('jid');
|
||||||
|
if (_.isUndefined(this.items.get(jid))) {
|
||||||
|
this.items.create({'jid': jid});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
queryForItems () {
|
queryForItems () {
|
||||||
if (_.isEmpty(this.identities.where({'category': 'server'}))) {
|
if (_.isEmpty(this.identities.where({'category': 'server'}))) {
|
||||||
// Don't fetch features and items if this is not a
|
// Don't fetch features and items if this is not a
|
||||||
// server or a conference component.
|
// server or a conference component.
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
_converse.connection.disco.items(this.get('jid'), null, onDiscoItems);
|
_converse.connection.disco.items(this.get('jid'), null, this.onDiscoItems.bind(this));
|
||||||
},
|
},
|
||||||
|
|
||||||
onInfo (stanza) {
|
onInfo (stanza) {
|
||||||
@ -175,26 +172,11 @@
|
|||||||
_converse.DiscoEntities = Backbone.Collection.extend({
|
_converse.DiscoEntities = Backbone.Collection.extend({
|
||||||
model: _converse.DiscoEntity,
|
model: _converse.DiscoEntity,
|
||||||
|
|
||||||
initialize () {
|
|
||||||
this.browserStorage = new Backbone.BrowserStorage[_converse.storage](
|
|
||||||
b64_sha1(`converse.disco-entities-${_converse.bare_jid}`)
|
|
||||||
);
|
|
||||||
this.fetchEntities().then(
|
|
||||||
_.partial(_converse.emit, 'discoInitialized'),
|
|
||||||
_.partial(_converse.emit, 'discoInitialized')
|
|
||||||
).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
|
|
||||||
},
|
|
||||||
|
|
||||||
fetchEntities () {
|
fetchEntities () {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
this.fetch({
|
this.fetch({
|
||||||
add: true,
|
add: true,
|
||||||
success: function (collection) {
|
success: resolve,
|
||||||
if (collection.length === 0 || !collection.get(_converse.domain)) {
|
|
||||||
this.create({'jid': _converse.domain});
|
|
||||||
}
|
|
||||||
resolve();
|
|
||||||
}.bind(this),
|
|
||||||
error () {
|
error () {
|
||||||
reject (new Error("Could not fetch disco entities"));
|
reject (new Error("Could not fetch disco entities"));
|
||||||
}
|
}
|
||||||
@ -227,6 +209,18 @@
|
|||||||
function initializeDisco () {
|
function initializeDisco () {
|
||||||
addClientFeatures();
|
addClientFeatures();
|
||||||
_converse.disco_entities = new _converse.DiscoEntities();
|
_converse.disco_entities = new _converse.DiscoEntities();
|
||||||
|
_converse.disco_entities.browserStorage = new Backbone.BrowserStorage[_converse.storage](
|
||||||
|
b64_sha1(`converse.disco-entities-${_converse.bare_jid}`)
|
||||||
|
);
|
||||||
|
|
||||||
|
_converse.disco_entities.fetchEntities().then((collection) => {
|
||||||
|
if (collection.length === 0 || !collection.get(_converse.domain)) {
|
||||||
|
// If we don't have an entity for our own XMPP server,
|
||||||
|
// create one.
|
||||||
|
_converse.disco_entities.create({'jid': _converse.domain});
|
||||||
|
}
|
||||||
|
_converse.emit('discoInitialized');
|
||||||
|
}).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
|
||||||
}
|
}
|
||||||
_converse.api.listen.on('reconnected', initializeDisco);
|
_converse.api.listen.on('reconnected', initializeDisco);
|
||||||
_converse.api.listen.on('connected', initializeDisco);
|
_converse.api.listen.on('connected', initializeDisco);
|
||||||
@ -247,28 +241,49 @@
|
|||||||
'disco': {
|
'disco': {
|
||||||
'entities': {
|
'entities': {
|
||||||
'get' (entity_jid, create=false) {
|
'get' (entity_jid, create=false) {
|
||||||
const entity = _converse.disco_entities.get(entity_jid);
|
return _converse.api.waitUntil('discoInitialized').then(() => {
|
||||||
if (entity || !create) {
|
if (_.isNil(entity_jid)) {
|
||||||
return entity;
|
return _converse.disco_entities;
|
||||||
}
|
}
|
||||||
return _converse.disco_entities.create({'jid': entity_jid});
|
const entity = _converse.disco_entities.get(entity_jid);
|
||||||
|
if (entity || !create) {
|
||||||
|
return entity;
|
||||||
|
}
|
||||||
|
return _converse.disco_entities.create({'jid': entity_jid});
|
||||||
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
'supports' (feature, entity_jid) {
|
'supports' (feature, entity_jid) {
|
||||||
/* Returns a Promise which resolves with a map indicating
|
/* Returns a Promise which resolves with a list containing
|
||||||
* whether a given feature is supported.
|
* _converse.Entity instances representing the entity
|
||||||
|
* itself or those items associated with the entity if
|
||||||
|
* they support the given feature.
|
||||||
*
|
*
|
||||||
* Parameters:
|
* Parameters:
|
||||||
* (String) feature - The feature that might be
|
* (String) feature - The feature that might be
|
||||||
* supported. In the XML stanza, this is the `var`
|
* supported. In the XML stanza, this is the `var`
|
||||||
* attribute of the `<feature>` element. For
|
* attribute of the `<feature>` element. For
|
||||||
* example: 'http://jabber.org/protocol/muc'
|
* example: 'http://jabber.org/protocol/muc'
|
||||||
* (String) entity_jid - The JID of the entity which might support the feature.
|
* (String) entity_jid - The JID of the entity
|
||||||
|
* (and its associated items) which should be queried
|
||||||
*/
|
*/
|
||||||
return _converse.api.waitUntil('discoInitialized').then(() => {
|
if (_.isNil(entity_jid)) {
|
||||||
const entity = _converse.api.disco.entities.get(entity_jid, true);
|
throw new TypeError('disco.supports: You need to provide an entity JID');
|
||||||
return entity.hasFeature(feature);
|
}
|
||||||
|
return _converse.api.waitUntil('discoInitialized').then((entity) => {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
_converse.api.disco.entities.get(entity_jid, true).then((entity) => {
|
||||||
|
Promise.all(
|
||||||
|
_.concat(
|
||||||
|
entity.items.map((item) => item.hasFeature(feature)),
|
||||||
|
entity.hasFeature(feature)
|
||||||
|
)
|
||||||
|
).then((result) => {
|
||||||
|
resolve(f.filter(f.isObject, result));
|
||||||
|
}).catch(reject);
|
||||||
|
})
|
||||||
|
});
|
||||||
}).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
|
}).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -286,10 +301,13 @@
|
|||||||
* In the XML stanza, this is the `type`
|
* In the XML stanza, this is the `type`
|
||||||
* attribute of the `<identity>` element.
|
* attribute of the `<identity>` element.
|
||||||
* For example: 'pep'
|
* For example: 'pep'
|
||||||
|
* (String) entity_jid - The JID of the entity which might have the identity
|
||||||
*/
|
*/
|
||||||
return _converse.api.waitUntil('discoInitialized').then(() => {
|
return new Promise((resolve, reject) => {
|
||||||
const entity = _converse.api.disco.entities.get(entity_jid, true);
|
_converse.api.waitUntil('discoInitialized').then(() => {
|
||||||
return entity.getIdentity(category, type);
|
_converse.api.disco.entities.get(entity_jid, true)
|
||||||
|
.then((entity) => resolve(entity.getIdentity(category, type)));
|
||||||
|
})
|
||||||
}).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
|
}).catch(_.partial(_converse.log, _, Strophe.LogLevel.FATAL));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,16 +1,48 @@
|
|||||||
(function (root, factory) {
|
(function (root, factory) {
|
||||||
define(["converse-http-file-upload"], factory);
|
define(["converse-core"], factory);
|
||||||
}(this, function (converse) {
|
}(this, function (converse) {
|
||||||
"use strict";
|
"use strict";
|
||||||
|
|
||||||
|
const { Promise, Strophe, _ } = converse.env;
|
||||||
|
const u = converse.env.utils;
|
||||||
|
|
||||||
|
Strophe.addNamespace('HTTPUPLOAD', 'urn:xmpp:http:upload:0');
|
||||||
|
|
||||||
converse.plugins.add('converse-http-file-upload', {
|
converse.plugins.add('converse-http-file-upload', {
|
||||||
|
/* Plugin dependencies are other plugins which might be
|
||||||
|
* overridden or relied upon, and therefore need to be loaded before
|
||||||
|
* this plugin.
|
||||||
|
*
|
||||||
|
* If the setting "strict_plugin_dependencies" is set to true,
|
||||||
|
* an error will be raised if the plugin is not found. By default it's
|
||||||
|
* false, which means these plugins are only loaded opportunistically.
|
||||||
|
*
|
||||||
|
* NB: These plugins need to have already been loaded via require.js.
|
||||||
|
*/
|
||||||
|
dependencies: ["converse-chatview"],
|
||||||
|
|
||||||
|
overrides: {
|
||||||
|
|
||||||
|
ChatBoxView: {
|
||||||
|
addFileUploadButton (options) {
|
||||||
|
},
|
||||||
|
|
||||||
|
renderToolbar (toolbar, options) {
|
||||||
|
const { _converse } = this.__super__;
|
||||||
|
const result = this.__super__.renderToolbar.apply(this, arguments);
|
||||||
|
// TODO: check results.length
|
||||||
|
_converse.api.disco.supports(Strophe.NS.HTTPUPLOAD, _converse.domain)
|
||||||
|
.then(this.addFileUploadButton.bind(this));
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
initialize () {
|
initialize () {
|
||||||
/* The initialize function gets called as soon as the plugin is
|
/* The initialize function gets called as soon as the plugin is
|
||||||
* loaded by converse.js's plugin machinery.
|
* loaded by converse.js's plugin machinery.
|
||||||
*/
|
*/
|
||||||
const { _converse } = this;
|
const { _converse } = this;
|
||||||
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}));
|
}));
|
||||||
|
@ -154,8 +154,8 @@
|
|||||||
if (this.disable_mam) { return; }
|
if (this.disable_mam) { return; }
|
||||||
const { _converse } = this.__super__;
|
const { _converse } = this.__super__;
|
||||||
_converse.api.disco.supports(Strophe.NS.MAM, _converse.bare_jid).then(
|
_converse.api.disco.supports(Strophe.NS.MAM, _converse.bare_jid).then(
|
||||||
(result) => { // Success
|
(results) => { // Success
|
||||||
if (result.supported) {
|
if (result.length) {
|
||||||
const most_recent_msg = utils.getMostRecentMessage(this.model);
|
const most_recent_msg = utils.getMostRecentMessage(this.model);
|
||||||
if (_.isNil(most_recent_msg)) {
|
if (_.isNil(most_recent_msg)) {
|
||||||
this.fetchArchivedMessages();
|
this.fetchArchivedMessages();
|
||||||
@ -193,7 +193,7 @@
|
|||||||
const { _converse } = this.__super__;
|
const { _converse } = this.__super__;
|
||||||
_converse.api.disco.supports(Strophe.NS.MAM, _converse.bare_jid).then(
|
_converse.api.disco.supports(Strophe.NS.MAM, _converse.bare_jid).then(
|
||||||
(result) => { // Success
|
(result) => { // Success
|
||||||
if (result.supported) {
|
if (result.length) {
|
||||||
this.fetchArchivedMessages();
|
this.fetchArchivedMessages();
|
||||||
}
|
}
|
||||||
this.model.save({'mam_initialized': true});
|
this.model.save({'mam_initialized': true});
|
||||||
|
@ -186,7 +186,7 @@
|
|||||||
if (_.isNil(_converse.xmppstatus.get('vcard_updated'))) {
|
if (_.isNil(_converse.xmppstatus.get('vcard_updated'))) {
|
||||||
_converse.api.disco.supports(Strophe.NS.VCARD, _converse.domain)
|
_converse.api.disco.supports(Strophe.NS.VCARD, _converse.domain)
|
||||||
.then((result) => {
|
.then((result) => {
|
||||||
if (result.supported) {
|
if (result.length) {
|
||||||
_converse.api.vcard.get(_converse.bare_jid)
|
_converse.api.vcard.get(_converse.bare_jid)
|
||||||
.then((vcard) => _converse.xmppstatus.save(vcard));
|
.then((vcard) => _converse.xmppstatus.save(vcard));
|
||||||
}})
|
}})
|
||||||
|
@ -7,24 +7,25 @@ if (typeof define !== 'undefined') {
|
|||||||
* --------------------
|
* --------------------
|
||||||
* Any of the following components may be removed if they're not needed.
|
* Any of the following components may be removed if they're not needed.
|
||||||
*/
|
*/
|
||||||
"converse-chatview", // Renders standalone chat boxes for single user chat
|
"converse-bookmarks", // XEP-0048 Bookmarks
|
||||||
"converse-controlbox", // The control box
|
"converse-chatview", // Renders standalone chat boxes for single user chat
|
||||||
"converse-bookmarks", // XEP-0048 Bookmarks
|
"converse-controlbox", // The control box
|
||||||
"converse-roomslist", // Show currently open chat rooms
|
"converse-dragresize", // Allows chat boxes to be resized by dragging them
|
||||||
"converse-mam", // XEP-0313 Message Archive Management
|
"converse-fullscreen",
|
||||||
"converse-muc", // XEP-0045 Multi-user chat
|
"converse-headline", // Support for headline messages
|
||||||
"converse-muc-views", // Views related to MUC
|
"converse-http-file-upload",
|
||||||
|
"converse-mam", // XEP-0313 Message Archive Management
|
||||||
|
"converse-minimize", // Allows chat boxes to be minimized
|
||||||
|
"converse-muc", // XEP-0045 Multi-user chat
|
||||||
"converse-muc-embedded",
|
"converse-muc-embedded",
|
||||||
"converse-muc-views",
|
"converse-muc-views",
|
||||||
"converse-vcard", // XEP-0054 VCard-temp
|
"converse-muc-views", // Views related to MUC
|
||||||
"converse-otr", // Off-the-record encryption for one-on-one messages
|
"converse-notification", // HTML5 Notifications
|
||||||
"converse-register", // XEP-0077 In-band registration
|
"converse-otr", // Off-the-record encryption for one-on-one messages
|
||||||
"converse-ping", // XEP-0199 XMPP Ping
|
"converse-ping", // XEP-0199 XMPP Ping
|
||||||
"converse-notification",// HTML5 Notifications
|
"converse-register", // XEP-0077 In-band registration
|
||||||
"converse-minimize", // Allows chat boxes to be minimized
|
"converse-roomslist", // Show currently open chat rooms
|
||||||
"converse-dragresize", // Allows chat boxes to be resized by dragging them
|
"converse-vcard", // XEP-0054 VCard-temp
|
||||||
"converse-headline", // Support for headline messages
|
|
||||||
"converse-fullscreen"
|
|
||||||
/* END: Removable components */
|
/* END: Removable components */
|
||||||
], function (converse) {
|
], function (converse) {
|
||||||
return converse;
|
return converse;
|
||||||
|
Loading…
Reference in New Issue
Block a user