fix(avatar): Avatar properties not updated before local user join

Replaces changeAvatarID, changeAvatarURL and changeEmail with
participantUpdated action.
participantUpdated can be fired for local user without id. This
fixes the problem with updating the local user before the user
join the conference which results in fix for failing to execute
commands for avatarID, avatarURL and email right after the iframe
api creates the iframe with Jitsi Meet.
This commit is contained in:
hristoterezov 2017-03-23 13:01:33 -05:00
parent 0ed39dad63
commit 4ab4aa04da
7 changed files with 91 additions and 96 deletions

View File

@ -32,12 +32,10 @@ import {
isFatalJitsiConnectionError isFatalJitsiConnectionError
} from './react/features/base/lib-jitsi-meet'; } from './react/features/base/lib-jitsi-meet';
import { import {
changeParticipantAvatarID,
changeParticipantAvatarURL,
changeParticipantEmail,
participantJoined, participantJoined,
participantLeft, participantLeft,
participantRoleChanged participantRoleChanged,
participantUpdated
} from './react/features/base/participants'; } from './react/features/base/participants';
import { import {
mediaPermissionPromptVisibilityChanged, mediaPermissionPromptVisibilityChanged,
@ -161,6 +159,10 @@ function createInitialLocalTracksAndConnect(roomName) {
* @param {string} value new value * @param {string} value new value
*/ */
function sendData (command, value) { function sendData (command, value) {
if(!room) {
return;
}
room.removeCommand(command); room.removeCommand(command);
room.sendCommand(command, {value: value}); room.sendCommand(command, {value: value});
} }
@ -1469,7 +1471,10 @@ export default {
APP.UI.addListener(UIEvents.EMAIL_CHANGED, this.changeLocalEmail); APP.UI.addListener(UIEvents.EMAIL_CHANGED, this.changeLocalEmail);
room.addCommandListener(this.commands.defaults.EMAIL, (data, from) => { room.addCommandListener(this.commands.defaults.EMAIL, (data, from) => {
APP.store.dispatch(changeParticipantEmail(from, data.value)); APP.store.dispatch(participantUpdated({
id: from,
email: data.value
}));
APP.UI.setUserEmail(from, data.value); APP.UI.setUserEmail(from, data.value);
}); });
@ -1477,14 +1482,20 @@ export default {
this.commands.defaults.AVATAR_URL, this.commands.defaults.AVATAR_URL,
(data, from) => { (data, from) => {
APP.store.dispatch( APP.store.dispatch(
changeParticipantAvatarURL(from, data.value)); participantUpdated({
id: from,
avatarURL: data.value
}));
APP.UI.setUserAvatarUrl(from, data.value); APP.UI.setUserAvatarUrl(from, data.value);
}); });
room.addCommandListener(this.commands.defaults.AVATAR_ID, room.addCommandListener(this.commands.defaults.AVATAR_ID,
(data, from) => { (data, from) => {
APP.store.dispatch( APP.store.dispatch(
changeParticipantAvatarID(from, data.value)); participantUpdated({
id: from,
avatarID: data.value
}));
APP.UI.setUserAvatarID(from, data.value); APP.UI.setUserAvatarID(from, data.value);
}); });
@ -1896,10 +1907,17 @@ export default {
if (email === APP.settings.getEmail()) { if (email === APP.settings.getEmail()) {
return; return;
} }
APP.store.dispatch(changeParticipantEmail(room.myUserId(), email));
const localId = room ? room.myUserId() : undefined;
APP.store.dispatch(participantUpdated({
id: localId,
local: true,
email
}));
APP.settings.setEmail(email); APP.settings.setEmail(email);
APP.UI.setUserEmail(room.myUserId(), email); APP.UI.setUserEmail(localId, email);
sendData(commands.EMAIL, email); sendData(commands.EMAIL, email);
}, },
@ -1913,10 +1931,17 @@ export default {
if (url === APP.settings.getAvatarUrl()) { if (url === APP.settings.getAvatarUrl()) {
return; return;
} }
APP.store.dispatch(changeParticipantAvatarURL(room.myUserId(), url));
const localId = room ? room.myUserId() : undefined;
APP.store.dispatch(participantUpdated({
id: localId,
local: true,
avatarURL: url
}));
APP.settings.setAvatarUrl(url); APP.settings.setAvatarUrl(url);
APP.UI.setUserAvatarUrl(room.myUserId(), url); APP.UI.setUserAvatarUrl(localId, url);
sendData(commands.AVATAR_URL, url); sendData(commands.AVATAR_URL, url);
}, },

View File

@ -762,6 +762,9 @@ UI.setUserEmail = function (id, email) {
Avatar.setUserEmail(id, email); Avatar.setUserEmail(id, email);
changeAvatar(id, Avatar.getAvatarUrl(id)); changeAvatar(id, Avatar.getAvatarUrl(id));
if (APP.conference.isLocalId(id)) {
Profile.changeEmail(email);
}
}; };
/** /**

View File

@ -30,7 +30,7 @@ let users = {};
export default { export default {
/** /**
* Sets prop in users object. * Sets prop in users object.
* @param id {string} user id * @param id {string} user id or undefined for the local user.
* @param prop {string} name of the prop * @param prop {string} name of the prop
* @param val {string} value to be set * @param val {string} value to be set
*/ */
@ -38,7 +38,7 @@ export default {
// FIXME: Fixes the issue with not be able to return avatar for the // FIXME: Fixes the issue with not be able to return avatar for the
// local user when the conference has been left. Maybe there is beter // local user when the conference has been left. Maybe there is beter
// way to solve it. // way to solve it.
if(APP.conference.isLocalId(id)) { if(!id || APP.conference.isLocalId(id)) {
id = "local"; id = "local";
} }
if(!val || (users[id] && users[id][prop] === val)) if(!val || (users[id] && users[id][prop] === val))

View File

@ -15,10 +15,10 @@ const htmlStr = `
</div> </div>
<div class="sideToolbarBlock"> <div class="sideToolbarBlock">
<label data-i18n="profile.setEmailLabel"></label> <label data-i18n="profile.setEmailLabel"></label>
<input id="setEmail" type="text" class="input-control" <input id="setEmail" type="text" class="input-control"
data-i18n="[placeholder]profile.setEmailInput"> data-i18n="[placeholder]profile.setEmailInput">
</div> </div>
<div id="profile_auth_container" <div id="profile_auth_container"
class="sideToolbarBlock auth_container"> class="sideToolbarBlock auth_container">
<p data-i18n="toolbar.authenticate"></p> <p data-i18n="toolbar.authenticate"></p>
<ul> <ul>
@ -122,6 +122,14 @@ export default {
$('#avatar').attr('src', avatarUrl); $('#avatar').attr('src', avatarUrl);
}, },
/**
* Change the value of the field for the user email.
* @param {string} email the new value that will be displayed in the field.
*/
changeEmail (email) {
$('#setEmail').val(email);
},
/** /**
* Shows or hides authentication related buttons * Shows or hides authentication related buttons
* @param {boolean} show <tt>true</tt> to show or <tt>false</tt> to hide * @param {boolean} show <tt>true</tt> to show or <tt>false</tt> to hide

View File

@ -1,13 +1,11 @@
import { JitsiConferenceEvents } from '../lib-jitsi-meet'; import { JitsiConferenceEvents } from '../lib-jitsi-meet';
import { import {
changeParticipantAvatarID,
changeParticipantAvatarURL,
changeParticipantEmail,
dominantSpeakerChanged, dominantSpeakerChanged,
getLocalParticipant, getLocalParticipant,
participantJoined, participantJoined,
participantLeft, participantLeft,
participantRoleChanged participantRoleChanged,
participantUpdated
} from '../participants'; } from '../participants';
import { trackAdded, trackRemoved } from '../tracks'; import { trackAdded, trackRemoved } from '../tracks';
@ -78,13 +76,22 @@ function _addConferenceListeners(conference, dispatch) {
conference.addCommandListener( conference.addCommandListener(
AVATAR_ID_COMMAND, AVATAR_ID_COMMAND,
(data, id) => dispatch(changeParticipantAvatarID(id, data.value))); (data, id) => dispatch(participantUpdated({
id,
avatarID: data.value
})));
conference.addCommandListener( conference.addCommandListener(
AVATAR_URL_COMMAND, AVATAR_URL_COMMAND,
(data, id) => dispatch(changeParticipantAvatarURL(id, data.value))); (data, id) => dispatch(participantUpdated({
id,
avatarURL: data.value
})));
conference.addCommandListener( conference.addCommandListener(
EMAIL_COMMAND, EMAIL_COMMAND,
(data, id) => dispatch(changeParticipantEmail(id, data.value))); (data, id) => dispatch(participantUpdated({
id,
email: data.value
})));
} }
/** /**

View File

@ -8,75 +8,6 @@ import {
} from './actionTypes'; } from './actionTypes';
import { getLocalParticipant } from './functions'; import { getLocalParticipant } from './functions';
/**
* Action to update a participant's avatar ID.
*
* @param {string} id - Participant's ID.
* @param {string} avatarID - Participant's avatar ID.
* @returns {{
* type: PARTICIPANT_UPDATED,
* participant: {
* id: string,
* avatarID: string,
* }
* }}
*/
export function changeParticipantAvatarID(id, avatarID) {
return {
type: PARTICIPANT_UPDATED,
participant: {
id,
avatarID
}
};
}
/**
* Action to update a participant's avatar URL.
*
* @param {string} id - Participant's ID.
* @param {string} avatarURL - Participant's avatar URL.
* @returns {{
* type: PARTICIPANT_UPDATED,
* participant: {
* id: string,
* avatarURL: string,
* }
* }}
*/
export function changeParticipantAvatarURL(id, avatarURL) {
return {
type: PARTICIPANT_UPDATED,
participant: {
id,
avatarURL
}
};
}
/**
* Action to update a participant's email.
*
* @param {string} id - Participant's ID.
* @param {string} email - Participant's email.
* @returns {{
* type: PARTICIPANT_UPDATED,
* participant: {
* id: string,
* email: string
* }
* }}
*/
export function changeParticipantEmail(id, email) {
return {
type: PARTICIPANT_UPDATED,
participant: {
id,
email
}
};
}
/** /**
* Create an action for when dominant speaker changes. * Create an action for when dominant speaker changes.
* *
@ -203,12 +134,28 @@ export function participantLeft(id) {
* }} * }}
*/ */
export function participantRoleChanged(id, role) { export function participantRoleChanged(id, role) {
return participantUpdated({
id,
role
});
}
/**
* Action to signal that some of participant properties has been changed.
*
* @param {Participant} participant={} - Information about participant. To
* identify the participant the object should contain either property id with
* value the id of the participant or property local with value true (if the
* local participant hasn't joined the conference yet).
* @returns {{
* type: PARTICIPANT_UPDATED,
* participant: Participant
* }}
*/
export function participantUpdated(participant = {}) {
return { return {
type: PARTICIPANT_UPDATED, type: PARTICIPANT_UPDATED,
participant: { participant
id,
role
}
}; };
} }

View File

@ -112,7 +112,12 @@ function _participant(state, action) {
case PARTICIPANT_UPDATED: { case PARTICIPANT_UPDATED: {
const participant = action.participant; // eslint-disable-line no-shadow const participant = action.participant; // eslint-disable-line no-shadow
const { id } = participant; const { local } = participant;
let { id } = participant;
if (!id && local) {
id = LOCAL_PARTICIPANT_DEFAULT_ID;
}
if (state.id === id) { if (state.id === id) {
const newState = { ...state }; const newState = { ...state };