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.

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