Nevar pievienot vairāk kā 25 tēmas Tēmai ir jāsākas ar burtu vai ciparu, tā var saturēt domu zīmes ('-') un var būt līdz 35 simboliem gara.

strophe.emuc.js 28KB

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