您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

strophe.emuc.js 25KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615
  1. /* jshint -W117 */
  2. /* a simple MUC connection plugin
  3. * can only handle a single MUC room
  4. */
  5. var bridgeIsDown = false;
  6. var Moderator = require("./moderator");
  7. var JingleSession = require("./JingleSession");
  8. module.exports = function(XMPP, eventEmitter) {
  9. Strophe.addConnectionPlugin('emuc', {
  10. connection: null,
  11. roomjid: null,
  12. myroomjid: null,
  13. members: {},
  14. list_members: [], // so we can elect a new focus
  15. presMap: {},
  16. preziMap: {},
  17. joined: false,
  18. isOwner: false,
  19. role: null,
  20. focusMucJid: null,
  21. ssrc2jid: {},
  22. init: function (conn) {
  23. this.connection = conn;
  24. },
  25. initPresenceMap: function (myroomjid) {
  26. this.presMap['to'] = myroomjid;
  27. this.presMap['xns'] = 'http://jabber.org/protocol/muc';
  28. },
  29. doJoin: function (jid, password) {
  30. this.myroomjid = jid;
  31. console.info("Joined MUC as " + this.myroomjid);
  32. this.initPresenceMap(this.myroomjid);
  33. if (!this.roomjid) {
  34. this.roomjid = Strophe.getBareJidFromJid(jid);
  35. // add handlers (just once)
  36. this.connection.addHandler(this.onPresence.bind(this), null, 'presence', null, null, this.roomjid, {matchBare: true});
  37. this.connection.addHandler(this.onPresenceUnavailable.bind(this), null, 'presence', 'unavailable', null, this.roomjid, {matchBare: true});
  38. this.connection.addHandler(this.onPresenceError.bind(this), null, 'presence', 'error', null, this.roomjid, {matchBare: true});
  39. this.connection.addHandler(this.onMessage.bind(this), null, 'message', null, null, this.roomjid, {matchBare: true});
  40. }
  41. if (password !== undefined) {
  42. this.presMap['password'] = password;
  43. }
  44. this.sendPresence();
  45. },
  46. doLeave: function () {
  47. console.log("do leave", this.myroomjid);
  48. var pres = $pres({to: this.myroomjid, type: 'unavailable' });
  49. this.presMap.length = 0;
  50. this.connection.send(pres);
  51. },
  52. createNonAnonymousRoom: function () {
  53. // http://xmpp.org/extensions/xep-0045.html#createroom-reserved
  54. var getForm = $iq({type: 'get', to: this.roomjid})
  55. .c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'})
  56. .c('x', {xmlns: 'jabber:x:data', type: 'submit'});
  57. var self = this;
  58. this.connection.sendIQ(getForm, function (form) {
  59. if (!$(form).find(
  60. '>query>x[xmlns="jabber:x:data"]' +
  61. '>field[var="muc#roomconfig_whois"]').length) {
  62. console.error('non-anonymous rooms not supported');
  63. return;
  64. }
  65. var formSubmit = $iq({to: this.roomjid, type: 'set'})
  66. .c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'});
  67. formSubmit.c('x', {xmlns: 'jabber:x:data', type: 'submit'});
  68. formSubmit.c('field', {'var': 'FORM_TYPE'})
  69. .c('value')
  70. .t('http://jabber.org/protocol/muc#roomconfig').up().up();
  71. formSubmit.c('field', {'var': 'muc#roomconfig_whois'})
  72. .c('value').t('anyone').up().up();
  73. self.connection.sendIQ(formSubmit);
  74. }, function (error) {
  75. console.error("Error getting room configuration form");
  76. });
  77. },
  78. onPresence: function (pres) {
  79. var from = pres.getAttribute('from');
  80. // What is this for? A workaround for something?
  81. if (pres.getAttribute('type')) {
  82. return true;
  83. }
  84. // Parse etherpad tag.
  85. var etherpad = $(pres).find('>etherpad');
  86. if (etherpad.length) {
  87. if (config.etherpad_base && !Moderator.isModerator()) {
  88. eventEmitter.emit(XMPPEvents.ETHERPAD, etherpad.text());
  89. }
  90. }
  91. // Parse prezi tag.
  92. var presentation = $(pres).find('>prezi');
  93. if (presentation.length) {
  94. var url = presentation.attr('url');
  95. var current = presentation.find('>current').text();
  96. console.log('presentation info received from', from, url);
  97. if (this.preziMap[from] == null) {
  98. this.preziMap[from] = url;
  99. $(document).trigger('presentationadded.muc', [from, url, current]);
  100. }
  101. else {
  102. $(document).trigger('gotoslide.muc', [from, url, current]);
  103. }
  104. }
  105. else if (this.preziMap[from] != null) {
  106. var url = this.preziMap[from];
  107. delete this.preziMap[from];
  108. $(document).trigger('presentationremoved.muc', [from, url]);
  109. }
  110. // Parse audio info tag.
  111. var audioMuted = $(pres).find('>audiomuted');
  112. if (audioMuted.length) {
  113. $(document).trigger('audiomuted.muc', [from, audioMuted.text()]);
  114. }
  115. // Parse video info tag.
  116. var videoMuted = $(pres).find('>videomuted');
  117. if (videoMuted.length) {
  118. $(document).trigger('videomuted.muc', [from, videoMuted.text()]);
  119. }
  120. var stats = $(pres).find('>stats');
  121. if (stats.length) {
  122. var statsObj = {};
  123. Strophe.forEachChild(stats[0], "stat", function (el) {
  124. statsObj[el.getAttribute("name")] = el.getAttribute("value");
  125. });
  126. eventEmitter.emit(XMPPEvents.REMOTE_STATS, from, statsObj);
  127. }
  128. // Parse status.
  129. if ($(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>status[code="201"]').length) {
  130. this.isOwner = true;
  131. this.createNonAnonymousRoom();
  132. }
  133. // Parse roles.
  134. var member = {};
  135. member.show = $(pres).find('>show').text();
  136. member.status = $(pres).find('>status').text();
  137. var tmp = $(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>item');
  138. member.affiliation = tmp.attr('affiliation');
  139. member.role = tmp.attr('role');
  140. // Focus recognition
  141. member.jid = tmp.attr('jid');
  142. member.isFocus = false;
  143. if (member.jid
  144. && member.jid.indexOf(Moderator.getFocusUserJid() + "/") == 0) {
  145. member.isFocus = true;
  146. }
  147. var nicktag = $(pres).find('>nick[xmlns="http://jabber.org/protocol/nick"]');
  148. member.displayName = (nicktag.length > 0 ? nicktag.html() : null);
  149. if (from == this.myroomjid) {
  150. if (member.affiliation == 'owner') this.isOwner = true;
  151. if (this.role !== member.role) {
  152. this.role = member.role;
  153. eventEmitter.emit(XMPPEvents.LOCALROLE_CHANGED,
  154. from, member, pres, Moderator.isModerator(),
  155. Moderator.isExternalAuthEnabled());
  156. }
  157. if (!this.joined) {
  158. this.joined = true;
  159. eventEmitter.emit(XMPPEvents.MUC_JOINED, from, member);
  160. this.list_members.push(from);
  161. }
  162. } else if (this.members[from] === undefined) {
  163. // new participant
  164. this.members[from] = member;
  165. this.list_members.push(from);
  166. console.log('entered', from, member);
  167. if (member.isFocus) {
  168. this.focusMucJid = from;
  169. console.info("Ignore focus: " + from + ", real JID: " + member.jid);
  170. }
  171. else {
  172. var id = $(pres).find('>userID').text();
  173. var email = $(pres).find('>email');
  174. if (email.length > 0) {
  175. id = email.text();
  176. }
  177. eventEmitter.emit(XMPPEvents.MUC_ENTER, from, id, member.displayName);
  178. }
  179. } else {
  180. // Presence update for existing participant
  181. // Watch role change:
  182. if (this.members[from].role != member.role) {
  183. this.members[from].role = member.role;
  184. eventEmitter.emit(XMPPEvents.MUC_ROLE_CHANGED,
  185. member.role, member.displayName);
  186. }
  187. }
  188. // Always trigger presence to update bindings
  189. this.parsePresence(from, member, pres);
  190. // Trigger status message update
  191. if (member.status) {
  192. eventEmitter.emit(XMPPEvents.PRESENCE_STATUS, from, member);
  193. }
  194. return true;
  195. },
  196. onPresenceUnavailable: function (pres) {
  197. var from = pres.getAttribute('from');
  198. // Status code 110 indicates that this notification is "self-presence".
  199. if (!$(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>status[code="110"]').length) {
  200. delete this.members[from];
  201. this.list_members.splice(this.list_members.indexOf(from), 1);
  202. this.onParticipantLeft(from);
  203. }
  204. // If the status code is 110 this means we're leaving and we would like
  205. // to remove everyone else from our view, so we trigger the event.
  206. else if (this.list_members.length > 1) {
  207. for (var i = 0; i < this.list_members.length; i++) {
  208. var member = this.list_members[i];
  209. delete this.members[i];
  210. this.list_members.splice(i, 1);
  211. this.onParticipantLeft(member);
  212. }
  213. }
  214. if ($(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>status[code="307"]').length) {
  215. $(document).trigger('kicked.muc', [from]);
  216. if (this.myroomjid === from) {
  217. XMPP.disposeConference(false);
  218. eventEmitter.emit(XMPPEvents.KICKED);
  219. }
  220. }
  221. return true;
  222. },
  223. onPresenceError: function (pres) {
  224. var from = pres.getAttribute('from');
  225. if ($(pres).find('>error[type="auth"]>not-authorized[xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"]').length) {
  226. console.log('on password required', from);
  227. var self = this;
  228. eventEmitter.emit(XMPPEvents.PASSWORD_REQUIRED, function (value) {
  229. self.doJoin(from, value);
  230. });
  231. } else if ($(pres).find(
  232. '>error[type="cancel"]>not-allowed[xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"]').length) {
  233. var toDomain = Strophe.getDomainFromJid(pres.getAttribute('to'));
  234. if (toDomain === config.hosts.anonymousdomain) {
  235. // enter the room by replying with 'not-authorized'. This would
  236. // result in reconnection from authorized domain.
  237. // We're either missing Jicofo/Prosody config for anonymous
  238. // domains or something is wrong.
  239. // XMPP.promptLogin();
  240. UI.messageHandler.openReportDialog(null,
  241. 'Oops ! We couldn`t join the conference.' +
  242. ' There might be some problem with security' +
  243. ' configuration. Please contact service' +
  244. ' administrator.', pres);
  245. } else {
  246. console.warn('onPresError ', pres);
  247. UI.messageHandler.openReportDialog(null,
  248. 'Oops! Something went wrong and we couldn`t connect to the conference.',
  249. pres);
  250. }
  251. } else {
  252. console.warn('onPresError ', pres);
  253. UI.messageHandler.openReportDialog(null,
  254. 'Oops! Something went wrong and we couldn`t connect to the conference.',
  255. pres);
  256. }
  257. return true;
  258. },
  259. sendMessage: function (body, nickname) {
  260. var msg = $msg({to: this.roomjid, type: 'groupchat'});
  261. msg.c('body', body).up();
  262. if (nickname) {
  263. msg.c('nick', {xmlns: 'http://jabber.org/protocol/nick'}).t(nickname).up().up();
  264. }
  265. this.connection.send(msg);
  266. eventEmitter.emit(XMPPEvents.SENDING_CHAT_MESSAGE, body);
  267. },
  268. setSubject: function (subject) {
  269. var msg = $msg({to: this.roomjid, type: 'groupchat'});
  270. msg.c('subject', subject);
  271. this.connection.send(msg);
  272. console.log("topic changed to " + subject);
  273. },
  274. onMessage: function (msg) {
  275. // FIXME: this is a hack. but jingle on muc makes nickchanges hard
  276. var from = msg.getAttribute('from');
  277. var nick =
  278. $(msg).find('>nick[xmlns="http://jabber.org/protocol/nick"]')
  279. .text() ||
  280. Strophe.getResourceFromJid(from);
  281. var txt = $(msg).find('>body').text();
  282. var type = msg.getAttribute("type");
  283. if (type == "error") {
  284. eventEmitter.emit(XMPPEvents.CHAT_ERROR_RECEIVED,
  285. $(msg).find('>text').text(), txt);
  286. return true;
  287. }
  288. var subject = $(msg).find('>subject');
  289. if (subject.length) {
  290. var subjectText = subject.text();
  291. if (subjectText || subjectText == "") {
  292. eventEmitter.emit(XMPPEvents.SUBJECT_CHANGED, subjectText);
  293. console.log("Subject is changed to " + subjectText);
  294. }
  295. }
  296. if (txt) {
  297. console.log('chat', nick, txt);
  298. eventEmitter.emit(XMPPEvents.MESSAGE_RECEIVED,
  299. from, nick, txt, this.myroomjid);
  300. }
  301. return true;
  302. },
  303. lockRoom: function (key, onSuccess, onError, onNotSupported) {
  304. //http://xmpp.org/extensions/xep-0045.html#roomconfig
  305. var ob = this;
  306. this.connection.sendIQ($iq({to: this.roomjid, type: 'get'}).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'}),
  307. function (res) {
  308. if ($(res).find('>query>x[xmlns="jabber:x:data"]>field[var="muc#roomconfig_roomsecret"]').length) {
  309. var formsubmit = $iq({to: ob.roomjid, type: 'set'}).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'});
  310. formsubmit.c('x', {xmlns: 'jabber:x:data', type: 'submit'});
  311. formsubmit.c('field', {'var': 'FORM_TYPE'}).c('value').t('http://jabber.org/protocol/muc#roomconfig').up().up();
  312. formsubmit.c('field', {'var': 'muc#roomconfig_roomsecret'}).c('value').t(key).up().up();
  313. // Fixes a bug in prosody 0.9.+ https://code.google.com/p/lxmppd/issues/detail?id=373
  314. formsubmit.c('field', {'var': 'muc#roomconfig_whois'}).c('value').t('anyone').up().up();
  315. // FIXME: is muc#roomconfig_passwordprotectedroom required?
  316. ob.connection.sendIQ(formsubmit,
  317. onSuccess,
  318. onError);
  319. } else {
  320. onNotSupported();
  321. }
  322. }, onError);
  323. },
  324. kick: function (jid) {
  325. var kickIQ = $iq({to: this.roomjid, type: 'set'})
  326. .c('query', {xmlns: 'http://jabber.org/protocol/muc#admin'})
  327. .c('item', {nick: Strophe.getResourceFromJid(jid), role: 'none'})
  328. .c('reason').t('You have been kicked.').up().up().up();
  329. this.connection.sendIQ(
  330. kickIQ,
  331. function (result) {
  332. console.log('Kick participant with jid: ', jid, result);
  333. },
  334. function (error) {
  335. console.log('Kick participant error: ', error);
  336. });
  337. },
  338. sendPresence: function () {
  339. var pres = $pres({to: this.presMap['to'] });
  340. pres.c('x', {xmlns: this.presMap['xns']});
  341. if (this.presMap['password']) {
  342. pres.c('password').t(this.presMap['password']).up();
  343. }
  344. pres.up();
  345. // Send XEP-0115 'c' stanza that contains our capabilities info
  346. if (this.connection.caps) {
  347. this.connection.caps.node = config.clientNode;
  348. pres.c('c', this.connection.caps.generateCapsAttrs()).up();
  349. }
  350. pres.c('user-agent', {xmlns: 'http://jitsi.org/jitmeet/user-agent'})
  351. .t(navigator.userAgent).up();
  352. if (this.presMap['bridgeIsDown']) {
  353. pres.c('bridgeIsDown').up();
  354. }
  355. if (this.presMap['email']) {
  356. pres.c('email').t(this.presMap['email']).up();
  357. }
  358. if (this.presMap['userId']) {
  359. pres.c('userId').t(this.presMap['userId']).up();
  360. }
  361. if (this.presMap['displayName']) {
  362. // XEP-0172
  363. pres.c('nick', {xmlns: 'http://jabber.org/protocol/nick'})
  364. .t(this.presMap['displayName']).up();
  365. }
  366. if (this.presMap['audions']) {
  367. pres.c('audiomuted', {xmlns: this.presMap['audions']})
  368. .t(this.presMap['audiomuted']).up();
  369. }
  370. if (this.presMap['videons']) {
  371. pres.c('videomuted', {xmlns: this.presMap['videons']})
  372. .t(this.presMap['videomuted']).up();
  373. }
  374. if (this.presMap['statsns']) {
  375. var stats = pres.c('stats', {xmlns: this.presMap['statsns']});
  376. for (var stat in this.presMap["stats"])
  377. if (this.presMap["stats"][stat] != null)
  378. stats.c("stat", {name: stat, value: this.presMap["stats"][stat]}).up();
  379. pres.up();
  380. }
  381. if (this.presMap['prezins']) {
  382. pres.c('prezi',
  383. {xmlns: this.presMap['prezins'],
  384. 'url': this.presMap['preziurl']})
  385. .c('current').t(this.presMap['prezicurrent']).up().up();
  386. }
  387. if (this.presMap['etherpadns']) {
  388. pres.c('etherpad', {xmlns: this.presMap['etherpadns']})
  389. .t(this.presMap['etherpadname']).up();
  390. }
  391. if (this.presMap['medians']) {
  392. pres.c('media', {xmlns: this.presMap['medians']});
  393. var sourceNumber = 0;
  394. Object.keys(this.presMap).forEach(function (key) {
  395. if (key.indexOf('source') >= 0) {
  396. sourceNumber++;
  397. }
  398. });
  399. if (sourceNumber > 0)
  400. for (var i = 1; i <= sourceNumber / 3; i++) {
  401. pres.c('source',
  402. {type: this.presMap['source' + i + '_type'],
  403. ssrc: this.presMap['source' + i + '_ssrc'],
  404. direction: this.presMap['source' + i + '_direction']
  405. || 'sendrecv' }
  406. ).up();
  407. }
  408. }
  409. pres.up();
  410. this.connection.send(pres);
  411. },
  412. addDisplayNameToPresence: function (displayName) {
  413. this.presMap['displayName'] = displayName;
  414. },
  415. addMediaToPresence: function (sourceNumber, mtype, ssrcs, direction) {
  416. if (!this.presMap['medians'])
  417. this.presMap['medians'] = 'http://estos.de/ns/mjs';
  418. this.presMap['source' + sourceNumber + '_type'] = mtype;
  419. this.presMap['source' + sourceNumber + '_ssrc'] = ssrcs;
  420. this.presMap['source' + sourceNumber + '_direction'] = direction;
  421. },
  422. clearPresenceMedia: function () {
  423. var self = this;
  424. Object.keys(this.presMap).forEach(function (key) {
  425. if (key.indexOf('source') != -1) {
  426. delete self.presMap[key];
  427. }
  428. });
  429. },
  430. addPreziToPresence: function (url, currentSlide) {
  431. this.presMap['prezins'] = 'http://jitsi.org/jitmeet/prezi';
  432. this.presMap['preziurl'] = url;
  433. this.presMap['prezicurrent'] = currentSlide;
  434. },
  435. removePreziFromPresence: function () {
  436. delete this.presMap['prezins'];
  437. delete this.presMap['preziurl'];
  438. delete this.presMap['prezicurrent'];
  439. },
  440. addCurrentSlideToPresence: function (currentSlide) {
  441. this.presMap['prezicurrent'] = currentSlide;
  442. },
  443. getPrezi: function (roomjid) {
  444. return this.preziMap[roomjid];
  445. },
  446. addEtherpadToPresence: function (etherpadName) {
  447. this.presMap['etherpadns'] = 'http://jitsi.org/jitmeet/etherpad';
  448. this.presMap['etherpadname'] = etherpadName;
  449. },
  450. addAudioInfoToPresence: function (isMuted) {
  451. this.presMap['audions'] = 'http://jitsi.org/jitmeet/audio';
  452. this.presMap['audiomuted'] = isMuted.toString();
  453. },
  454. addVideoInfoToPresence: function (isMuted) {
  455. this.presMap['videons'] = 'http://jitsi.org/jitmeet/video';
  456. this.presMap['videomuted'] = isMuted.toString();
  457. },
  458. addConnectionInfoToPresence: function (stats) {
  459. this.presMap['statsns'] = 'http://jitsi.org/jitmeet/stats';
  460. this.presMap['stats'] = stats;
  461. },
  462. findJidFromResource: function (resourceJid) {
  463. if (resourceJid &&
  464. resourceJid === Strophe.getResourceFromJid(this.myroomjid)) {
  465. return this.myroomjid;
  466. }
  467. var peerJid = null;
  468. Object.keys(this.members).some(function (jid) {
  469. peerJid = jid;
  470. return Strophe.getResourceFromJid(jid) === resourceJid;
  471. });
  472. return peerJid;
  473. },
  474. addBridgeIsDownToPresence: function () {
  475. this.presMap['bridgeIsDown'] = true;
  476. },
  477. addEmailToPresence: function (email) {
  478. this.presMap['email'] = email;
  479. },
  480. addUserIdToPresence: function (userId) {
  481. this.presMap['userId'] = userId;
  482. },
  483. isModerator: function () {
  484. return this.role === 'moderator';
  485. },
  486. getMemberRole: function (peerJid) {
  487. if (this.members[peerJid]) {
  488. return this.members[peerJid].role;
  489. }
  490. return null;
  491. },
  492. onParticipantLeft: function (jid) {
  493. eventEmitter.emit(XMPPEvents.MUC_LEFT, jid);
  494. this.connection.jingle.terminateByJid(jid);
  495. if (this.getPrezi(jid)) {
  496. $(document).trigger('presentationremoved.muc',
  497. [jid, this.getPrezi(jid)]);
  498. }
  499. Moderator.onMucLeft(jid);
  500. },
  501. parsePresence: function (from, memeber, pres) {
  502. if($(pres).find(">bridgeIsDown").length > 0 && !bridgeIsDown) {
  503. bridgeIsDown = true;
  504. eventEmitter.emit(XMPPEvents.BRIDGE_DOWN);
  505. }
  506. if(memeber.isFocus)
  507. return;
  508. var self = this;
  509. // Remove old ssrcs coming from the jid
  510. Object.keys(this.ssrc2jid).forEach(function (ssrc) {
  511. if (self.ssrc2jid[ssrc] == from) {
  512. delete self.ssrc2jid[ssrc];
  513. }
  514. });
  515. var changedStreams = [];
  516. $(pres).find('>media[xmlns="http://estos.de/ns/mjs"]>source').each(function (idx, ssrc) {
  517. //console.log(jid, 'assoc ssrc', ssrc.getAttribute('type'), ssrc.getAttribute('ssrc'));
  518. var ssrcV = ssrc.getAttribute('ssrc');
  519. self.ssrc2jid[ssrcV] = from;
  520. JingleSession.notReceivedSSRCs.push(ssrcV);
  521. var type = ssrc.getAttribute('type');
  522. var direction = ssrc.getAttribute('direction');
  523. changedStreams.push({type: type, direction: direction});
  524. });
  525. eventEmitter.emit(XMPPEvents.CHANGED_STREAMS, from, changedStreams);
  526. var displayName = !config.displayJids
  527. ? memeber.displayName : Strophe.getResourceFromJid(from);
  528. if (displayName && displayName.length > 0)
  529. {
  530. eventEmitter.emit(XMPPEvents.DISPLAY_NAME_CHANGED, from, displayName);
  531. }
  532. var id = $(pres).find('>userID').text();
  533. var email = $(pres).find('>email');
  534. if(email.length > 0) {
  535. id = email.text();
  536. }
  537. eventEmitter.emit(XMPPEvents.USER_ID_CHANGED, from, id);
  538. }
  539. });
  540. };