Merge pull request #517 from jitsi/fix-resizing

An additional fix should be added, which allow for more smooth resizing (we still see thumbnails going on a second row and then coming back up again), but this will be added in a separate commit.
This commit is contained in:
yanas 2016-02-29 13:20:19 -06:00
commit 1339b306e6
14 changed files with 241 additions and 184 deletions

View File

@ -29,5 +29,6 @@ var interfaceConfig = {
*/ */
filmStripOnly: false, filmStripOnly: false,
RANDOM_AVATAR_URL_PREFIX: false, RANDOM_AVATAR_URL_PREFIX: false,
RANDOM_AVATAR_URL_SUFFIX: false RANDOM_AVATAR_URL_SUFFIX: false,
FILM_STRIP_MAX_HEIGHT: 160
}; };

View File

@ -16,6 +16,7 @@ import PreziManager from './prezi/Prezi';
import EtherpadManager from './etherpad/Etherpad'; import EtherpadManager from './etherpad/Etherpad';
import VideoLayout from "./videolayout/VideoLayout"; import VideoLayout from "./videolayout/VideoLayout";
import FilmStrip from "./videolayout/FilmStrip";
import SettingsMenu from "./side_pannels/settings/SettingsMenu"; import SettingsMenu from "./side_pannels/settings/SettingsMenu";
import Settings from "./../settings/Settings"; import Settings from "./../settings/Settings";
import { reload } from '../util/helpers'; import { reload } from '../util/helpers';
@ -293,7 +294,7 @@ function registerListeners() {
function bindEvents() { function bindEvents() {
function onResize() { function onResize() {
PanelToggler.resizeChat(); PanelToggler.resizeChat();
VideoLayout.resizeLargeVideoContainer(PanelToggler.isVisible()); VideoLayout.resizeVideoArea(PanelToggler.isVisible());
} }
// Resize and reposition videos in full screen mode. // Resize and reposition videos in full screen mode.
@ -334,12 +335,13 @@ UI.start = function () {
registerListeners(); registerListeners();
BottomToolbar.init(); BottomToolbar.init();
FilmStrip.init();
VideoLayout.init(eventEmitter); VideoLayout.init(eventEmitter);
if (!interfaceConfig.filmStripOnly) { if (!interfaceConfig.filmStripOnly) {
VideoLayout.initLargeVideo(PanelToggler.isVisible()); VideoLayout.initLargeVideo(PanelToggler.isVisible());
} }
VideoLayout.resizeLargeVideoContainer(PanelToggler.isVisible(), true); VideoLayout.resizeVideoArea(PanelToggler.isVisible(), true, true);
ContactList.init(eventEmitter); ContactList.init(eventEmitter);
@ -367,9 +369,9 @@ UI.start = function () {
}); });
} else { } else {
$("#header").css("display", "none"); $("#header").css("display", "none");
$("#bottomToolbar").css("display", "none");
$("#downloadlog").css("display", "none"); $("#downloadlog").css("display", "none");
BottomToolbar.setupFilmStripOnly(); BottomToolbar.hide();
FilmStrip.setupFilmStripOnly();
messageHandler.disableNotifications(); messageHandler.disableNotifications();
$('body').popover("disable"); $('body').popover("disable");
JitsiPopover.enabled = false; JitsiPopover.enabled = false;
@ -598,7 +600,7 @@ UI.getSettings = function () {
* Toggles film strip. * Toggles film strip.
*/ */
UI.toggleFilmStrip = function () { UI.toggleFilmStrip = function () {
BottomToolbar.toggleFilmStrip(); FilmStrip.toggleFilmStrip();
}; };
/** /**

View File

@ -2,7 +2,7 @@
/* jshint -W101 */ /* jshint -W101 */
import CanvasUtil from './CanvasUtils'; import CanvasUtil from './CanvasUtils';
import BottomToolbar from '../toolbars/BottomToolbar'; import FilmStrip from '../videolayout/FilmStrip';
const LOCAL_LEVEL = 'local'; const LOCAL_LEVEL = 'local';
@ -228,7 +228,7 @@ const AudioLevels = {
let canvasWidth = thumbWidth + interfaceConfig.CANVAS_EXTRA; let canvasWidth = thumbWidth + interfaceConfig.CANVAS_EXTRA;
let canvasHeight = thumbHeight + interfaceConfig.CANVAS_EXTRA; let canvasHeight = thumbHeight + interfaceConfig.CANVAS_EXTRA;
BottomToolbar.getThumbs().children('canvas').each(function () { FilmStrip.getThumbs().children('canvas').each(function () {
$(this).attr('width', canvasWidth); $(this).attr('width', canvasWidth);
$(this).attr('height', canvasHeight); $(this).attr('height', canvasHeight);
}); });

View File

@ -4,7 +4,7 @@ import VideoLayout from "../videolayout/VideoLayout";
import LargeContainer from '../videolayout/LargeContainer'; import LargeContainer from '../videolayout/LargeContainer';
import UIUtil from "../util/UIUtil"; import UIUtil from "../util/UIUtil";
import SidePanelToggler from "../side_pannels/SidePanelToggler"; import SidePanelToggler from "../side_pannels/SidePanelToggler";
import BottomToolbar from '../toolbars/BottomToolbar'; import FilmStrip from '../videolayout/FilmStrip';
/** /**
* Etherpad options. * Etherpad options.
@ -101,7 +101,7 @@ class Etherpad extends LargeContainer {
} }
resize (containerWidth, containerHeight, animate) { resize (containerWidth, containerHeight, animate) {
let height = containerHeight - BottomToolbar.getFilmStripHeight(); let height = containerHeight - FilmStrip.getFilmStripHeight();
let width = containerWidth; let width = containerWidth;
$(this.iframe).width(width).height(height); $(this.iframe).width(width).height(height);

View File

@ -9,7 +9,7 @@ import UIEvents from '../../../service/UI/UIEvents';
import messageHandler from '../util/MessageHandler'; import messageHandler from '../util/MessageHandler';
import ToolbarToggler from "../toolbars/ToolbarToggler"; import ToolbarToggler from "../toolbars/ToolbarToggler";
import SidePanelToggler from "../side_pannels/SidePanelToggler"; import SidePanelToggler from "../side_pannels/SidePanelToggler";
import BottomToolbar from '../toolbars/BottomToolbar'; import FilmStrip from '../videolayout/FilmStrip';
/** /**
* Example of Prezi link. * Example of Prezi link.
@ -287,7 +287,7 @@ class PreziContainer extends LargeContainer {
} }
resize (containerWidth, containerHeight) { resize (containerWidth, containerHeight) {
let height = containerHeight - BottomToolbar.getFilmStripHeight(); let height = containerHeight - FilmStrip.getFilmStripHeight();
let width = containerWidth; let width = containerWidth;

View File

@ -96,11 +96,14 @@ var PanelToggler = {
$('#chatspace').trigger('shown'); $('#chatspace').trigger('shown');
}; };
VideoLayout.resizeVideoArea(!Chat.isVisible(), chatCompleteFunction); VideoLayout.resizeVideoArea(!Chat.isVisible(),
false,
true,
chatCompleteFunction);
toggle(Chat, toggle(Chat, //Object
'#chatspace', '#chatspace', // Selector
function () { function () { //onOpenComplete
// Request the focus in the nickname field or the chat input // Request the focus in the nickname field or the chat input
// field. // field.
if ($('#nickname').css('visibility') === 'visible') { if ($('#nickname').css('visibility') === 'visible') {
@ -109,9 +112,8 @@ var PanelToggler = {
$('#usermsg').focus(); $('#usermsg').focus();
} }
}, },
null, () => this.resizeChat(), //OnOpen
() => this.resizeChat(), null); //OnClose
null);
}, },
resizeChat () { resizeChat () {
@ -128,7 +130,11 @@ var PanelToggler = {
: function () { : function () {
$('#contactlist').trigger('shown'); $('#contactlist').trigger('shown');
}; };
VideoLayout.resizeVideoArea(!ContactList.isVisible(), completeFunction); VideoLayout.resizeVideoArea(
!ContactList.isVisible(),
false,
true,
completeFunction);
toggle(ContactList, toggle(ContactList,
'#contactlist', '#contactlist',
@ -143,7 +149,8 @@ var PanelToggler = {
* Opens / closes the settings menu * Opens / closes the settings menu
*/ */
toggleSettingsMenu () { toggleSettingsMenu () {
VideoLayout.resizeVideoArea(!SettingsMenu.isVisible(), function (){}); VideoLayout.resizeVideoArea(
!SettingsMenu.isVisible(), false, true, function (){});
toggle(SettingsMenu, toggle(SettingsMenu,
'#settingsmenu', '#settingsmenu',
null, null,

View File

@ -1,4 +1,4 @@
/* global $ */ /* global $, APP, interfaceConfig*/
import UIUtil from '../util/UIUtil'; import UIUtil from '../util/UIUtil';
import UIEvents from '../../../service/UI/UIEvents'; import UIEvents from '../../../service/UI/UIEvents';
import AnalyticsAdapter from '../../statistics/AnalyticsAdapter'; import AnalyticsAdapter from '../../statistics/AnalyticsAdapter';
@ -11,7 +11,6 @@ const defaultBottomToolbarButtons = {
const BottomToolbar = { const BottomToolbar = {
init () { init () {
this.filmStrip = $('#remoteVideos');
this.toolbar = $('#bottomToolbar'); this.toolbar = $('#bottomToolbar');
}, },
@ -38,71 +37,43 @@ const BottomToolbar = {
); );
}, },
toggleFilmStrip () {
this.filmStrip.toggleClass("hidden");
},
isFilmStripVisible () {
return !this.filmStrip.hasClass('hidden');
},
setupFilmStripOnly () {
this.filmStrip.css({
padding: "0px 0px 18px 0px",
right: 0
});
},
getFilmStripHeight () {
if (this.isFilmStripVisible()) {
return this.filmStrip.outerHeight();
} else {
return 0;
}
},
getFilmStripWidth () {
return this.filmStrip.width();
},
resizeThumbnails (thumbWidth, thumbHeight,
animate = false, forceUpdate = false) {
return new Promise(resolve => {
this.filmStrip.animate({
// adds 2 px because of small video 1px border
height: thumbHeight + 2
}, {
queue: false,
duration: animate ? 500 : 0
});
this.getThumbs(!forceUpdate).animate({
height: thumbHeight,
width: thumbWidth
}, {
queue: false,
duration: animate ? 500 : 0,
complete: resolve
});
if (!animate) {
resolve();
}
});
},
resizeToolbar (thumbWidth, thumbHeight) { resizeToolbar (thumbWidth, thumbHeight) {
let bottom = (thumbHeight - this.toolbar.outerHeight())/2 + 18; let bottom = (thumbHeight - this.toolbar.outerHeight())/2 + 18;
this.toolbar.css({bottom}); this.toolbar.css({bottom});
}, },
getThumbs (only_visible = false) { /**
let selector = 'span'; * Returns true if this toolbar is currently visible, or false otherwise.
if (only_visible) { * @return <tt>true</tt> if currently visible, <tt>false</tt> - otherwise
selector += ':visible'; */
} isVisible() {
return this.toolbar.is(":visible");
},
return this.filmStrip.children(selector); /**
* Hides the bottom toolbar with animation or not depending on the animate
* parameter.
* @param animate <tt>true</tt> to hide the bottom toolbar with animation,
* <tt>false</tt> or nothing to hide it without animation.
*/
hide(animate) {
if (animate)
this.toolbar.hide("slide", {direction: "right", duration: 300});
else
this.toolbar.css("display", "none");
},
/**
* Shows the bottom toolbar with animation or not depending on the animate
* parameter.
* @param animate <tt>true</tt> to show the bottom toolbar with animation,
* <tt>false</tt> or nothing to show it without animation.
*/
show(animate) {
if (animate)
this.toolbar.show("slide", {direction: "right", duration: 300});
else
this.toolbar.css("display", "block");
} }
}; };

View File

@ -2,6 +2,7 @@
import UIUtil from '../util/UIUtil'; import UIUtil from '../util/UIUtil';
import BottomToolbar from './BottomToolbar'; import BottomToolbar from './BottomToolbar';
import FilmStrip from '../videolayout/FilmStrip.js';
let toolbarTimeoutObject; let toolbarTimeoutObject;
let toolbarTimeout = interfaceConfig.INITIAL_TOOLBAR_TIMEOUT; let toolbarTimeout = interfaceConfig.INITIAL_TOOLBAR_TIMEOUT;
@ -28,7 +29,6 @@ function hideToolbar() {
} }
let header = $("#header"); let header = $("#header");
let bottomToolbar = $("#bottomToolbar");
let isToolbarHover = false; let isToolbarHover = false;
header.find('*').each(function () { header.find('*').each(function () {
let id = $(this).attr('id'); let id = $(this).attr('id');
@ -48,10 +48,8 @@ function hideToolbar() {
} else { } else {
header.hide("slide", { direction: "up", duration: 300}); header.hide("slide", { direction: "up", duration: 300});
$('#subject').animate({top: "-=40"}, 300); $('#subject').animate({top: "-=40"}, 300);
if (!BottomToolbar.isFilmStripVisible()) { if (!FilmStrip.isFilmStripVisible()) {
bottomToolbar.hide( BottomToolbar.hide(true);
"slide", {direction: "right", duration: 300}
);
} }
} }
} }
@ -65,14 +63,11 @@ const ToolbarToggler = {
return; return;
} }
let header = $("#header"); let header = $("#header");
let bottomToolbar = $("#bottomToolbar"); if (!header.is(':visible') || !BottomToolbar.isVisible()) {
if (!header.is(':visible') || !bottomToolbar.is(":visible")) {
header.show("slide", { direction: "up", duration: 300}); header.show("slide", { direction: "up", duration: 300});
$('#subject').animate({top: "+=40"}, 300); $('#subject').animate({top: "+=40"}, 300);
if (!bottomToolbar.is(":visible")) { if (!BottomToolbar.isVisible()) {
bottomToolbar.show( BottomToolbar.show(true);
"slide", {direction: "right", duration: 300}
);
} }
if (toolbarTimeoutObject) { if (toolbarTimeoutObject) {

View File

@ -32,6 +32,7 @@
return window.innerWidth - rightPanelWidth; return window.innerWidth - rightPanelWidth;
}, },
/** /**
* Changes the style class of the element given by id. * Changes the style class of the element given by id.
*/ */

View File

@ -0,0 +1,140 @@
/* global $, APP, interfaceConfig, config*/
import UIUtil from "../util/UIUtil";
const thumbAspectRatio = 16.0 / 9.0;
const FilmStrip = {
init () {
this.filmStrip = $('#remoteVideos');
},
toggleFilmStrip () {
this.filmStrip.toggleClass("hidden");
},
isFilmStripVisible () {
return !this.filmStrip.hasClass('hidden');
},
setupFilmStripOnly () {
this.filmStrip.css({
padding: "0px 0px 18px 0px",
right: 0
});
},
getFilmStripHeight () {
if (this.isFilmStripVisible()) {
return this.filmStrip.outerHeight();
} else {
return 0;
}
},
getFilmStripWidth () {
return this.filmStrip.innerWidth()
- parseInt(this.filmStrip.css('paddingLeft'), 10)
- parseInt(this.filmStrip.css('paddingRight'), 10);
},
/**
* Calculates the thumbnail size.
* @param videoAreaAvailableWidth the currently available video area width
* that we want to take into account when calculating the film strip width.
*/
calculateThumbnailSize (videoAreaAvailableWidth) {
// Calculate the available height, which is the inner window height
// minus 39px for the header minus 2px for the delimiter lines on the
// top and bottom of the large video, minus the 36px space inside the
// remoteVideos container used for highlighting shadow.
let availableHeight = 100;
let numvids = this.getThumbs(true).length;
let localVideoContainer = $("#localVideoContainer");
/**
* If the videoAreaAvailableWidth is set we use this one to calculate
* the filmStrip width, because we're probably in a state where the
* film strip size hasn't been updated yet, but it will be.
*/
let filmStripWidth = videoAreaAvailableWidth
? videoAreaAvailableWidth
- parseInt(this.filmStrip.css('right'), 10)
- parseInt(this.filmStrip.css('paddingLeft'), 10)
- parseInt(this.filmStrip.css('paddingRight'), 10)
- parseInt(this.filmStrip.css('borderLeftWidth'), 10)
- parseInt(this.filmStrip.css('borderRightWidth'), 10)
: this.getFilmStripWidth();
let availableWidth = Math.floor(
(filmStripWidth - numvids * (
parseInt(localVideoContainer.css('borderLeftWidth'), 10)
+ parseInt(localVideoContainer.css('borderRightWidth'), 10)
+ parseInt(localVideoContainer.css('paddingLeft'), 10)
+ parseInt(localVideoContainer.css('paddingRight'), 10)
+ parseInt(localVideoContainer.css('marginLeft'), 10)
+ parseInt(localVideoContainer.css('marginRight'), 10)))
/ numvids) - numvids*10;
let maxHeight
// If the MAX_HEIGHT property hasn't been specified
// we have the static value.
= Math.min( interfaceConfig.FILM_STRIP_MAX_HEIGHT || 160,
availableHeight);
availableHeight
= Math.min( maxHeight,
availableWidth / thumbAspectRatio,
window.innerHeight - 18);
if (availableHeight < availableWidth / thumbAspectRatio) {
availableWidth = Math.floor(availableHeight * thumbAspectRatio);
}
return {
thumbWidth: availableWidth,
thumbHeight: availableHeight
};
},
resizeThumbnails (thumbWidth, thumbHeight,
animate = false, forceUpdate = false) {
return new Promise(resolve => {
this.getThumbs(!forceUpdate).animate({
height: thumbHeight,
width: thumbWidth
}, {
queue: false,
duration: animate ? 500 : 0,
complete: resolve
});
this.filmStrip.animate({
// adds 2 px because of small video 1px border
height: thumbHeight + 2
}, {
queue: false,
duration: animate ? 500 : 0
});
if (!animate) {
resolve();
}
});
},
getThumbs (only_visible = false) {
let selector = 'span';
if (only_visible) {
selector += ':visible';
}
return this.filmStrip.children(selector);
}
};
export default FilmStrip;

View File

@ -4,7 +4,7 @@
import UIUtil from "../util/UIUtil"; import UIUtil from "../util/UIUtil";
import UIEvents from "../../../service/UI/UIEvents"; import UIEvents from "../../../service/UI/UIEvents";
import LargeContainer from './LargeContainer'; import LargeContainer from './LargeContainer';
import BottomToolbar from '../toolbars/BottomToolbar'; import FilmStrip from './FilmStrip';
import Avatar from "../avatar/Avatar"; import Avatar from "../avatar/Avatar";
import {createDeferred} from '../../util/helpers'; import {createDeferred} from '../../util/helpers';
@ -43,7 +43,7 @@ function getDesktopVideoSize(videoWidth,
let availableWidth = Math.max(videoWidth, videoSpaceWidth); let availableWidth = Math.max(videoWidth, videoSpaceWidth);
let availableHeight = Math.max(videoHeight, videoSpaceHeight); let availableHeight = Math.max(videoHeight, videoSpaceHeight);
videoSpaceHeight -= BottomToolbar.getFilmStripHeight(); videoSpaceHeight -= FilmStrip.getFilmStripHeight();
if (availableWidth / aspectRatio >= videoSpaceHeight) { if (availableWidth / aspectRatio >= videoSpaceHeight) {
availableHeight = videoSpaceHeight; availableHeight = videoSpaceHeight;

View File

@ -33,7 +33,7 @@ RemoteVideo.prototype.addRemoteVideoContainer = function() {
if (APP.conference.isModerator) { if (APP.conference.isModerator) {
this.addRemoteVideoMenu(); this.addRemoteVideoMenu();
} }
let {thumbWidth, thumbHeight} = this.VideoLayout.calculateThumbnailSize(); let {thumbWidth, thumbHeight} = this.VideoLayout.resizeThumbnails();
AudioLevels.updateAudioLevelCanvas(this.id, thumbWidth, thumbHeight); AudioLevels.updateAudioLevelCanvas(this.id, thumbWidth, thumbHeight);
return this.container; return this.container;

View File

@ -4,7 +4,7 @@
import AudioLevels from "../audio_levels/AudioLevels"; import AudioLevels from "../audio_levels/AudioLevels";
import Avatar from "../avatar/Avatar"; import Avatar from "../avatar/Avatar";
import BottomToolbar from "../toolbars/BottomToolbar"; import BottomToolbar from "../toolbars/BottomToolbar";
import FilmStrip from "./FilmStrip";
import UIEvents from "../../../service/UI/UIEvents"; import UIEvents from "../../../service/UI/UIEvents";
import UIUtil from "../util/UIUtil"; import UIUtil from "../util/UIUtil";
@ -34,8 +34,6 @@ var eventEmitter = null;
*/ */
var focusedVideoResourceJid = null; var focusedVideoResourceJid = null;
const thumbAspectRatio = 16.0 / 9.0;
/** /**
* On contact list item clicked. * On contact list item clicked.
*/ */
@ -149,9 +147,8 @@ var VideoLayout = {
let localId = APP.conference.localId; let localId = APP.conference.localId;
this.onVideoTypeChanged(localId, stream.videoType); this.onVideoTypeChanged(localId, stream.videoType);
let {thumbWidth, thumbHeight} = this.calculateThumbnailSize(); let {thumbWidth, thumbHeight} = this.resizeThumbnails(false, true);
AudioLevels.updateAudioLevelCanvas( AudioLevels.updateAudioLevelCanvas(null, thumbWidth, thumbHeight);
null, thumbWidth, thumbHeight);
if (!stream.isMuted()) { if (!stream.isMuted()) {
localVideoThumbnail.changeVideo(stream); localVideoThumbnail.changeVideo(stream);
@ -221,7 +218,7 @@ var VideoLayout = {
electLastVisibleVideo () { electLastVisibleVideo () {
// pick the last visible video in the row // pick the last visible video in the row
// if nobody else is left, this picks the local video // if nobody else is left, this picks the local video
let thumbs = BottomToolbar.getThumbs(true).filter('[id!="mixedstream"]'); let thumbs = FilmStrip.getThumbs(true).filter('[id!="mixedstream"]');
let lastVisible = thumbs.filter(':visible:last'); let lastVisible = thumbs.filter(':visible:last');
if (lastVisible.length) { if (lastVisible.length) {
@ -235,7 +232,7 @@ var VideoLayout = {
} }
console.info("Last visible video no longer exists"); console.info("Last visible video no longer exists");
thumbs = BottomToolbar.getThumbs(); thumbs = FilmStrip.getThumbs();
if (thumbs.length) { if (thumbs.length) {
let id = getPeerContainerResourceId(thumbs[0]); let id = getPeerContainerResourceId(thumbs[0]);
if (remoteVideos[id]) { if (remoteVideos[id]) {
@ -345,7 +342,7 @@ var VideoLayout = {
// In case this is not currently in the last n we don't show it. // In case this is not currently in the last n we don't show it.
if (localLastNCount && localLastNCount > 0 && if (localLastNCount && localLastNCount > 0 &&
BottomToolbar.getThumbs().length >= localLastNCount + 2) { FilmStrip.getThumbs().length >= localLastNCount + 2) {
remoteVideo.showPeerContainer('hide'); remoteVideo.showPeerContainer('hide');
} else { } else {
VideoLayout.resizeThumbnails(false, true); VideoLayout.resizeThumbnails(false, true);
@ -411,74 +408,26 @@ var VideoLayout = {
localVideoThumbnail.showAudioIndicator(isMuted); localVideoThumbnail.showAudioIndicator(isMuted);
}, },
/**
* Resizes the large video container.
*/
resizeLargeVideoContainer (isSideBarVisible, forceUpdate) {
let animate = false;
if (largeVideo) {
largeVideo.updateContainerSize(isSideBarVisible);
largeVideo.resize(animate);
}
this.resizeVideoSpace(animate, isSideBarVisible);
this.resizeThumbnails(false, forceUpdate);
},
/** /**
* Resizes thumbnails. * Resizes thumbnails.
*/ */
resizeThumbnails (animate = false, forceUpdate = false) { resizeThumbnails ( animate = false,
let {thumbWidth, thumbHeight} = this.calculateThumbnailSize(); forceUpdate = false,
videoAreaAvailableWidth = null) {
let {thumbWidth, thumbHeight}
= FilmStrip.calculateThumbnailSize(videoAreaAvailableWidth);
$('.userAvatar').css('left', (thumbWidth - thumbHeight) / 2); $('.userAvatar').css('left', (thumbWidth - thumbHeight) / 2);
BottomToolbar.resizeThumbnails(thumbWidth, thumbHeight, FilmStrip.resizeThumbnails(thumbWidth, thumbHeight,
animate, forceUpdate) animate, forceUpdate)
.then(function () { .then(function () {
BottomToolbar.resizeToolbar(thumbWidth, thumbHeight); BottomToolbar.resizeToolbar(thumbWidth, thumbHeight);
AudioLevels.updateCanvasSize(thumbWidth, thumbHeight); AudioLevels.updateCanvasSize(thumbWidth, thumbHeight);
}); });
return {thumbWidth, thumbHeight};
}, },
/**
* Calculates the thumbnail size.
*
*/
calculateThumbnailSize () {
let videoSpaceWidth = BottomToolbar.getFilmStripWidth();
// Calculate the available height, which is the inner window height
// minus 39px for the header minus 2px for the delimiter lines on the
// top and bottom of the large video, minus the 36px space inside the
// remoteVideos container used for highlighting shadow.
let availableHeight = 100;
let numvids = BottomToolbar.getThumbs().length;
if (localLastNCount && localLastNCount > 0) {
numvids = Math.min(localLastNCount + 1, numvids);
}
// Remove the 3px borders arround videos and border around the remote
// videos area and the 4 pixels between the local video and the others
//TODO: Find out where the 4 pixels come from and remove them
let availableWinWidth = videoSpaceWidth - 2 * 3 * numvids - 70 - 4;
let availableWidth = availableWinWidth / numvids;
let maxHeight = Math.min(160, availableHeight);
availableHeight
= Math.min( maxHeight,
availableWidth / thumbAspectRatio,
window.innerHeight - 18);
if (availableHeight < availableWidth / thumbAspectRatio) {
availableWidth = Math.floor(availableHeight * thumbAspectRatio);
}
return {
thumbWidth: availableWidth,
thumbHeight: availableHeight
};
},
/** /**
* On audio muted event. * On audio muted event.
*/ */
@ -613,7 +562,7 @@ var VideoLayout = {
var updateLargeVideo = false; var updateLargeVideo = false;
// Handle LastN/local LastN changes. // Handle LastN/local LastN changes.
BottomToolbar.getThumbs().each(( index, element ) => { FilmStrip.getThumbs().each(( index, element ) => {
var resourceJid = getPeerContainerResourceId(element); var resourceJid = getPeerContainerResourceId(element);
var smallVideo = remoteVideos[resourceJid]; var smallVideo = remoteVideos[resourceJid];
@ -667,7 +616,8 @@ var VideoLayout = {
endpointsEnteringLastN.forEach(function (resourceJid) { endpointsEnteringLastN.forEach(function (resourceJid) {
var remoteVideo = remoteVideos[resourceJid]; var remoteVideo = remoteVideos[resourceJid];
remoteVideo.showPeerContainer('show'); if (remoteVideo)
remoteVideo.showPeerContainer('show');
if (!remoteVideo.isVisible()) { if (!remoteVideo.isVisible()) {
console.log("Add to last N", resourceJid); console.log("Add to last N", resourceJid);
@ -840,40 +790,31 @@ var VideoLayout = {
* Resizes the video area. * Resizes the video area.
* *
* @param isSideBarVisible indicates if the side bar is currently visible * @param isSideBarVisible indicates if the side bar is currently visible
* @param callback a function to be called when the video space is * @param forceUpdate indicates that hidden thumbnails will be shown
* @param completeFunction a function to be called when the video area is
* resized. * resized.
*/ */resizeVideoArea (isSideBarVisible,
resizeVideoArea (isSideBarVisible, callback) { forceUpdate = false,
let animate = true; animate = false,
completeFunction = null) {
if (largeVideo) { if (largeVideo) {
largeVideo.updateContainerSize(isSideBarVisible); largeVideo.updateContainerSize(isSideBarVisible);
largeVideo.resize(animate); largeVideo.resize(animate);
this.resizeVideoSpace(animate, isSideBarVisible, callback);
} }
VideoLayout.resizeThumbnails(animate); // Calculate available width and height.
},
/**
* Resizes the #videospace html element
* @param animate boolean property that indicates whether the resize should
* be animated or not.
* @param isChatVisible boolean property that indicates whether the chat
* area is displayed or not.
* If that parameter is null the method will check the chat panel
* visibility.
* @param completeFunction a function to be called when the video space
* is resized.
*/
resizeVideoSpace (animate, isChatVisible, completeFunction) {
let availableHeight = window.innerHeight; let availableHeight = window.innerHeight;
let availableWidth = UIUtil.getAvailableVideoWidth(isChatVisible); let availableWidth = UIUtil.getAvailableVideoWidth(isSideBarVisible);
if (availableWidth < 0 || availableHeight < 0) { if (availableWidth < 0 || availableHeight < 0) {
return; return;
} }
// Resize the thumbnails first.
this.resizeThumbnails(false, forceUpdate, availableWidth);
// Resize the video area element.
$('#videospace').animate({ $('#videospace').animate({
right: window.innerWidth - availableWidth, right: window.innerWidth - availableWidth,
width: availableWidth, width: availableWidth,

View File

@ -47,7 +47,6 @@ function initShortcutHandlers() {
}; };
} }
var KeyboardShortcut = { var KeyboardShortcut = {
init: function () { init: function () {
initShortcutHandlers(); initShortcutHandlers();