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 25KB

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