You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

strophe.emuc.js 27KB

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