diff --git a/openvidu-server/src/angular/frontend/.angular-cli.json b/openvidu-server/src/angular/frontend/.angular-cli.json index 0e144dd187..db44ba5f1c 100644 --- a/openvidu-server/src/angular/frontend/.angular-cli.json +++ b/openvidu-server/src/angular/frontend/.angular-cli.json @@ -3,39 +3,36 @@ "project": { "name": "openvidu-server-frontend" }, - "apps": [ - { - "root": "src", - "outDir": "dist", - "assets": [ - "assets", - "favicon.ico" - ], - "index": "index.html", - "main": "main.ts", - "polyfills": "polyfills.ts", - "test": "test.ts", - "tsconfig": "tsconfig.app.json", - "testTsconfig": "tsconfig.spec.json", - "prefix": "app", - "styles": [ - "styles.css" - ], - "scripts": [], - "environmentSource": "environments/environment.ts", - "environments": { - "dev": "environments/environment.ts", - "prod": "environments/environment.prod.ts" - } + "apps": [{ + "root": "src", + "outDir": "dist", + "assets": [ + "assets", + "favicon.ico" + ], + "index": "index.html", + "main": "main.ts", + "polyfills": "polyfills.ts", + "test": "test.ts", + "tsconfig": "tsconfig.app.json", + "testTsconfig": "tsconfig.spec.json", + "prefix": "app", + "styles": [ + "styles.css" + ], + "scripts": [], + "environmentSource": "environments/environment.ts", + "environments": { + "dev": "environments/environment.ts", + "prod": "environments/environment.prod.ts" } - ], + }], "e2e": { "protractor": { "config": "./protractor.conf.js" } }, - "lint": [ - { + "lint": [{ "project": "src/tsconfig.app.json" }, { diff --git a/openvidu-server/src/angular/frontend/package.json b/openvidu-server/src/angular/frontend/package.json index f2a4861760..6329eccc9f 100644 --- a/openvidu-server/src/angular/frontend/package.json +++ b/openvidu-server/src/angular/frontend/package.json @@ -1,52 +1,52 @@ { - "dependencies": { - "@angular/animations": "5.0.5", - "@angular/cdk": "5.0.0-rc.2", - "@angular/common": "5.0.5", - "@angular/compiler": "5.0.5", - "@angular/core": "5.0.5", - "@angular/flex-layout": "2.0.0-beta.10-4905443", - "@angular/forms": "5.0.5", - "@angular/http": "5.0.5", - "@angular/material": "5.0.0-rc.1", - "@angular/platform-browser": "5.0.5", - "@angular/platform-browser-dynamic": "5.0.5", - "@angular/router": "5.0.5", - "core-js": "2.5.1", - "hammerjs": "2.0.8", - "openvidu-browser": "1.7.0", - "rxjs": "5.5.3", - "zone.js": "0.8.18" - }, - "devDependencies": { - "@angular/cli": "^1.6.0", - "@angular/compiler-cli": "5.0.5", - "@types/jasmine": "2.5.38", - "@types/node": "~6.0.60", - "codelyzer": "4.0.1", - "jasmine-core": "~2.5.2", - "jasmine-spec-reporter": "~3.2.0", - "karma": "~1.4.1", - "karma-chrome-launcher": "~2.0.0", - "karma-cli": "~1.0.1", - "karma-coverage-istanbul-reporter": "^0.2.0", - "karma-jasmine": "~1.1.0", - "karma-jasmine-html-reporter": "^0.2.2", - "protractor": "~5.1.0", - "ts-node": "3.3.0", - "tslint": "5.8.0", - "typescript": "2.4.2" - }, - "license": "MIT", - "name": "frontend", - "private": true, - "scripts": { - "build": "ng build", - "e2e": "ng e2e", - "lint": "ng lint", - "ng": "ng", - "start": "ng serve", - "test": "ng test" - }, - "version": "0.0.0" -} \ No newline at end of file + "dependencies": { + "@angular/animations": "5.0.5", + "@angular/cdk": "5.0.0-rc.2", + "@angular/common": "5.0.5", + "@angular/compiler": "5.0.5", + "@angular/core": "5.0.5", + "@angular/flex-layout": "2.0.0-beta.10-4905443", + "@angular/forms": "5.0.5", + "@angular/http": "5.0.5", + "@angular/material": "5.0.0-rc.1", + "@angular/platform-browser": "5.0.5", + "@angular/platform-browser-dynamic": "5.0.5", + "@angular/router": "5.0.5", + "core-js": "2.5.1", + "hammerjs": "2.0.8", + "openvidu-browser": "1.7.0", + "rxjs": "5.5.3", + "zone.js": "0.8.18" + }, + "devDependencies": { + "@angular/cli": "^1.6.0", + "@angular/compiler-cli": "5.0.5", + "@types/jasmine": "2.5.38", + "@types/node": "~6.0.60", + "codelyzer": "4.0.1", + "jasmine-core": "~2.5.2", + "jasmine-spec-reporter": "~3.2.0", + "karma": "~1.4.1", + "karma-chrome-launcher": "~2.0.0", + "karma-cli": "~1.0.1", + "karma-coverage-istanbul-reporter": "^0.2.0", + "karma-jasmine": "~1.1.0", + "karma-jasmine-html-reporter": "^0.2.2", + "protractor": "~5.1.0", + "ts-node": "3.3.0", + "tslint": "5.8.0", + "typescript": "2.4.2" + }, + "license": "MIT", + "name": "frontend", + "private": true, + "scripts": { + "build": "ng build", + "e2e": "ng e2e", + "lint": "ng lint", + "ng": "ng", + "start": "ng serve", + "test": "ng test" + }, + "version": "0.0.0" +} diff --git a/openvidu-server/src/angular/frontend/src/app/components/layouts/layout-best-fit/layout-best-fit.component.css b/openvidu-server/src/angular/frontend/src/app/components/layouts/layout-best-fit/layout-best-fit.component.css index 9aba615a1e..e1eee28a20 100644 --- a/openvidu-server/src/angular/frontend/src/app/components/layouts/layout-best-fit/layout-best-fit.component.css +++ b/openvidu-server/src/angular/frontend/src/app/components/layouts/layout-best-fit/layout-best-fit.component.css @@ -1,10 +1,869 @@ .bounds { background-color: black; - height: 100%; overflow: hidden; cursor: none !important; + position: absolute; + left: 0; + right: 0; + top: 0; + bottom: 0; } video { + -o-object-fit: cover; + object-fit: cover; + display: block; + position: absolute; + width: 100%; + height: 100%; + color: #ffffff; + margin: 0; + padding: 0; + border: 0; + font-size: 100%; + font-family: Arial, Helvetica, sans-serif; +} + +/*! + * Copyright (c) 2017 TokBox, Inc. + * Released under the MIT license + * http://opensource.org/licenses/MIT + */ + +.custom-class { + min-height: 0px !important; +} + +/** + * OT Base styles + */ + +/* Root OT object, this is where our CSS reset happens */ +.OT_root, +.OT_root * { + color: #ffffff; + margin: 0; + padding: 0; + border: 0; + font-size: 100%; + font-family: Arial, Helvetica, sans-serif; + vertical-align: baseline; +} + +.OT_dialog-centering { + display: table; + width: 100%; + height: 100%; +} + +.OT_dialog-centering-child { + display: table-cell; + vertical-align: middle; +} + +.OT_dialog { + position: relative; + + -webkit-box-sizing: border-box; + + box-sizing: border-box; + max-width: 576px; + margin-right: auto; + margin-left: auto; + padding: 36px; + text-align: center; /* centers all the inline content */ + + background-color: #363636; + color: #fff; + -webkit-box-shadow: 2px 4px 6px #999; + box-shadow: 2px 4px 6px #999; + font-family: 'Didact Gothic', sans-serif; + font-size: 13px; + line-height: 1.4; +} + +.OT_dialog * { + font-family: inherit; + -webkit-box-sizing: inherit; + box-sizing: inherit; +} + +.OT_closeButton { + color: #999999; + cursor: pointer; + font-size: 32px; + line-height: 36px; + position: absolute; + right: 18px; + top: 0; +} + +.OT_dialog-messages { + text-align: center; +} + +.OT_dialog-messages-main { + margin-bottom: 36px; + line-height: 36px; + + font-weight: 300; + font-size: 24px; +} + +.OT_dialog-messages-minor { + margin-bottom: 18px; + + font-size: 13px; + line-height: 18px; + color: #A4A4A4; +} + +.OT_dialog-messages-minor strong { + color: #ffffff; +} + +.OT_dialog-actions-card { + display: inline-block; +} + +.OT_dialog-button-title { + margin-bottom: 18px; + line-height: 18px; + + font-weight: 300; + text-align: center; + font-size: 14px; + color: #999999; +} +.OT_dialog-button-title label { + color: #999999; +} + +.OT_dialog-button-title a, +.OT_dialog-button-title a:link, +.OT_dialog-button-title a:active { + color: #02A1DE; +} + +.OT_dialog-button-title strong { + color: #ffffff; + font-weight: 100; + display: block; +} + +.OT_dialog-button { + display: inline-block; + + margin-bottom: 18px; + padding: 0 1em; + + background-color: #1CA3DC; + text-align: center; + cursor: pointer; +} + +.OT_dialog-button:disabled { + cursor: not-allowed; + opacity: 0.5; +} + +.OT_dialog-button-large { + line-height: 36px; + padding-top: 9px; + padding-bottom: 9px; + + font-weight: 100; + font-size: 24px; +} + +.OT_dialog-button-small { + line-height: 18px; + padding-top: 9px; + padding-bottom: 9px; + + background-color: #444444; + color: #999999; + font-size: 16px; +} + +.OT_dialog-progress-bar { + display: inline-block; /* prevents margin collapse */ + width: 100%; + margin-top: 5px; + margin-bottom: 41px; + + border: 1px solid #4E4E4E; + height: 8px; +} + +.OT_dialog-progress-bar-fill { + height: 100%; + + background-color: #29A4DA; +} + +.OT_dialog-plugin-upgrading .OT_dialog-plugin-upgrade-percentage { + line-height: 54px; + + font-size: 48px; + font-weight: 100; +} + +/* Helpers */ + +.OT_centered { + position: fixed; + left: 50%; + top: 50%; + margin: 0; +} + +.OT_dialog-hidden { + display: none; +} + +.OT_dialog-button-block { + display: block; +} + +.OT_dialog-no-natural-margin { + margin-bottom: 0; +} + +/* Publisher and Subscriber styles */ + +.OT_publisher, .OT_subscriber { + position: relative; + min-width: 48px; + min-height: 48px; +} + +.OT_publisher .OT_video-element, +.OT_subscriber .OT_video-element { + display: block; + position: absolute; + width: 100%; + height: 100%; + + -webkit-transform-origin: 0 0; + + transform-origin: 0 0; +} + +/* Styles that are applied when the video element should be mirrored */ +.OT_publisher.OT_mirrored .OT_video-element { + -webkit-transform: scale(-1, 1); + transform: scale(-1, 1); + -webkit-transform-origin: 50% 50%; + transform-origin: 50% 50%; +} + +.OT_subscriber_error { + background-color: #000; + color: #fff; + text-align: center; +} + +.OT_subscriber_error > p { + padding: 20px; +} + +/* The publisher/subscriber name/mute background */ +.OT_publisher .OT_bar, +.OT_subscriber .OT_bar, +.OT_publisher .OT_name, +.OT_subscriber .OT_name, +.OT_publisher .OT_archiving, +.OT_subscriber .OT_archiving, +.OT_publisher .OT_archiving-status, +.OT_subscriber .OT_archiving-status, +.OT_publisher .OT_archiving-light-box, +.OT_subscriber .OT_archiving-light-box { + -webkit-box-sizing: border-box; + -ms-box-sizing: border-box; + box-sizing: border-box; + top: 0; + left: 0; + right: 0; + display: block; + height: 34px; + position: absolute; +} + +.OT_publisher .OT_bar, +.OT_subscriber .OT_bar { + background: rgba(0, 0, 0, 0.4); +} + +.OT_publisher .OT_edge-bar-item, +.OT_subscriber .OT_edge-bar-item { + z-index: 1; /* required to get audio level meter underneath */ +} + +/* The publisher/subscriber name panel/archiving status bar */ +.OT_publisher .OT_name, +.OT_subscriber .OT_name { + background-color: transparent; + color: #ffffff; + font-size: 15px; + line-height: 34px; + font-weight: normal; + padding: 0 4px 0 36px; +} + +.OT_publisher .OT_archiving-status, +.OT_subscriber .OT_archiving-status { + background: rgba(0, 0, 0, 0.4); + top: auto; + bottom: 0; + left: 34px; + padding: 0 4px; + color: rgba(255, 255, 255, 0.8); + font-size: 15px; + line-height: 34px; + font-weight: normal; +} + +.OT_micro .OT_archiving-status, +.OT_micro:hover .OT_archiving-status, +.OT_mini .OT_archiving-status, +.OT_mini:hover .OT_archiving-status { + display: none; +} + +.OT_publisher .OT_archiving-light-box, +.OT_subscriber .OT_archiving-light-box { + background: rgba(0, 0, 0, 0.4); + top: auto; + bottom: 0; + right: auto; + width: 34px; + height: 34px; +} + +.OT_archiving-light { + width: 7px; + height: 7px; + border-radius: 30px; + position: absolute; + top: 14px; + left: 14px; + background-color: #575757; + -webkit-box-shadow: 0 0 5px 1px #575757; + box-shadow: 0 0 5px 1px #575757; +} + +.OT_archiving-light.OT_active { + background-color: #970d13; + animation: OT_pulse 1.3s ease-in; + -webkit-animation: OT_pulse 1.3s ease-in; + -moz-animation: OT_pulse 1.3s ease-in; + -webkit-animation: OT_pulse 1.3s ease-in; + animation-iteration-count: infinite; + -webkit-animation-iteration-count: infinite; + -moz-animation-iteration-count: infinite; + -webkit-animation-iteration-count: infinite; +} + +@-webkit-keyframes OT_pulse { + 0% { + -webkit-box-shadow: 0 0 0px 0px #c70019; + box-shadow: 0 0 0px 0px #c70019; + } + + 30% { + -webkit-box-shadow: 0 0 5px 1px #c70019; + box-shadow: 0 0 5px 1px #c70019; + } + + 50% { + -webkit-box-shadow: 0 0 5px 1px #c70019; + box-shadow: 0 0 5px 1px #c70019; + } + + 80% { + -webkit-box-shadow: 0 0 0px 0px #c70019; + box-shadow: 0 0 0px 0px #c70019; + } + + 100% { + -webkit-box-shadow: 0 0 0px 0px #c70019; + box-shadow: 0 0 0px 0px #c70019; + } +} + +@-webkit-keyframes OT_pulse { + 0% { + -webkit-box-shadow: 0 0 0px 0px #c70019; + box-shadow: 0 0 0px 0px #c70019; + } + + 30% { + -webkit-box-shadow: 0 0 5px 1px #c70019; + box-shadow: 0 0 5px 1px #c70019; + } + + 50% { + -webkit-box-shadow: 0 0 5px 1px #c70019; + box-shadow: 0 0 5px 1px #c70019; + } + + 80% { + -webkit-box-shadow: 0 0 0px 0px #c70019; + box-shadow: 0 0 0px 0px #c70019; + } + + 100% { + -webkit-box-shadow: 0 0 0px 0px #c70019; + box-shadow: 0 0 0px 0px #c70019; + } +} + +.OT_mini .OT_bar, +.OT_bar.OT_mode-mini, +.OT_bar.OT_mode-mini-auto { + bottom: 0; + height: auto; +} + +.OT_mini .OT_name.OT_mode-off, +.OT_mini .OT_name.OT_mode-on, +.OT_mini .OT_name.OT_mode-auto, +.OT_mini:hover .OT_name.OT_mode-auto { + display: none; +} + +.OT_publisher .OT_name, +.OT_subscriber .OT_name { + left: 10px; + right: 37px; + height: 34px; + padding-left: 0; +} + +.OT_publisher .OT_mute, +.OT_subscriber .OT_mute { + border: none; + cursor: pointer; + display: block; + position: absolute; + text-align: center; + text-indent: -9999em; + background-color: transparent; + background-repeat: no-repeat; +} + +.OT_publisher .OT_mute, +.OT_subscriber .OT_mute { + right: 0; + top: 0; + border-left: 1px solid rgba(255, 255, 255, 0.2); + height: 36px; + width: 37px; +} + +.OT_mini .OT_mute, +.OT_publisher.OT_mini .OT_mute.OT_mode-auto.OT_mode-on-hold, +.OT_subscriber.OT_mini .OT_mute.OT_mode-auto.OT_mode-on-hold { + top: 50%; + left: 50%; + right: auto; + margin-top: -18px; + margin-left: -18.5px; + border-left: none; +} + +.OT_publisher .OT_mute { + background-image: url(); + background-position: 9px 5px; +} + +.OT_publisher .OT_mute.OT_active { + background-image: url(); + background-position: 9px 4px; +} + +.OT_subscriber .OT_mute { + background-image: url(); + background-position: 8px 7px; +} + +.OT_subscriber .OT_mute.OT_active { + background-image: url(); + background-position: 7px 7px; +} + +/** + * Styles for display modes + * + * Note: It's important that these completely control the display and opacity + * attributes, no other selectors should atempt to change them. + */ + +/* Default display mode transitions for various chrome elements */ +.OT_publisher .OT_edge-bar-item, +.OT_subscriber .OT_edge-bar-item { + -webkit-transition-property: top, bottom, opacity; + transition-property: top, bottom, opacity; + -webkit-transition-duration: 0.5s; + transition-duration: 0.5s; + -webkit-transition-timing-function: ease-in; + transition-timing-function: ease-in; +} + +.OT_publisher .OT_edge-bar-item.OT_mode-off, +.OT_subscriber .OT_edge-bar-item.OT_mode-off, +.OT_publisher .OT_edge-bar-item.OT_mode-auto, +.OT_subscriber .OT_edge-bar-item.OT_mode-auto, +.OT_publisher .OT_edge-bar-item.OT_mode-mini-auto, +.OT_subscriber .OT_edge-bar-item.OT_mode-mini-auto { + top: -25px; + opacity: 0; +} + +.OT_publisher .OT_edge-bar-item.OT_mode-off, +.OT_subscriber .OT_edge-bar-item.OT_mode-off { + display: none; +} + +.OT_mini .OT_mute.OT_mode-auto, +.OT_publisher .OT_mute.OT_mode-mini-auto, +.OT_subscriber .OT_mute.OT_mode-mini-auto { + top: 50%; +} + +.OT_publisher .OT_edge-bar-item.OT_edge-bottom.OT_mode-off, +.OT_subscriber .OT_edge-bar-item.OT_edge-bottom.OT_mode-off, +.OT_publisher .OT_edge-bar-item.OT_edge-bottom.OT_mode-auto, +.OT_subscriber .OT_edge-bar-item.OT_edge-bottom.OT_mode-auto, +.OT_publisher .OT_edge-bar-item.OT_edge-bottom.OT_mode-mini-auto, +.OT_subscriber .OT_edge-bar-item.OT_edge-bottom.OT_mode-mini-auto { + top: auto; + bottom: -25px; +} + +.OT_publisher .OT_edge-bar-item.OT_mode-on, +.OT_subscriber .OT_edge-bar-item.OT_mode-on, +.OT_publisher .OT_edge-bar-item.OT_mode-auto.OT_mode-on-hold, +.OT_subscriber .OT_edge-bar-item.OT_mode-auto.OT_mode-on-hold, +.OT_publisher:hover .OT_edge-bar-item.OT_mode-auto, +.OT_subscriber:hover .OT_edge-bar-item.OT_mode-auto, +.OT_publisher:hover .OT_edge-bar-item.OT_mode-mini-auto, +.OT_subscriber:hover .OT_edge-bar-item.OT_mode-mini-auto { + top: 0; + opacity: 1; +} + +.OT_mini .OT_mute.OT_mode-on, +.OT_mini:hover .OT_mute.OT_mode-auto, +.OT_mute.OT_mode-mini, +.OT_root:hover .OT_mute.OT_mode-mini-auto { + top: 50%; +} + +.OT_publisher .OT_edge-bar-item.OT_edge-bottom.OT_mode-on, +.OT_subscriber .OT_edge-bar-item.OT_edge-bottom.OT_mode-on, +.OT_publisher:hover .OT_edge-bar-item.OT_edge-bottom.OT_mode-auto, +.OT_subscriber:hover .OT_edge-bar-item.OT_edge-bottom.OT_mode-auto { + top: auto; + bottom: 0; + opacity: 1; +} + + +/* Contains the video element, used to fix video letter-boxing */ +.OT_widget-container { + position: absolute; + background-color: #000000; + overflow: hidden; +} + +/* Load animation */ +.OT_root .OT_video-loading { + position: absolute; + z-index: 1; + width: 100%; + height: 100%; + display: none; + + background-color: rgba(0, 0, 0, .75); +} + +.OT_root .OT_video-loading .OT_video-loading-spinner { + background: url() no-repeat; + position: absolute; + width: 32px; + height: 32px; + left: 50%; + top: 50%; + margin-left: -16px; + margin-top: -16px; + -webkit-animation: OT_spin 2s linear infinite; + animation: OT_spin 2s linear infinite; +} +@-webkit-keyframes OT_spin { + 100% { + -webkit-transform: rotate(360deg); + } +} +@keyframes OT_spin { + 100% { + -webkit-transform: rotate(360deg); + transform: rotate(360deg); + } +} + +.OT_publisher.OT_loading .OT_video-loading, +.OT_subscriber.OT_loading .OT_video-loading { + display: block; +} + +.OT_video-centering { + display: table; + width: 100%; + height: 100%; +} + +.OT_video-container { + display: table-cell; + vertical-align: middle; +} + +.OT_video-poster { + position: absolute; + z-index: 1; + width: 100%; + height: 100%; + display: none; + + opacity: .25; + + background-repeat: no-repeat; + background-image: url(); + background-size: auto 76%; +} + +.OT_fit-mode-cover .OT_video-element { + -o-object-fit: cover; + object-fit: cover; +} + +/* Workaround for iOS freezing issue when cropping videos */ +/* https://bugs.webkit.org/show_bug.cgi?id=176439 */ +@media only screen + and (orientation: portrait) { + .OT_subscriber.OT_ForceContain.OT_fit-mode-cover .OT_video-element { + -o-object-fit: contain !important; + object-fit: contain !important; + } +} + +.OT_fit-mode-contain .OT_video-element { + -o-object-fit: contain; + object-fit: contain; +} + +.OT_fit-mode-cover .OT_video-poster { + background-position: center bottom; +} + +.OT_fit-mode-contain .OT_video-poster { + background-position: center; +} + +.OT_audio-level-meter { + position: absolute; + width: 25%; + max-width: 224px; + min-width: 21px; + top: 0; + right: 0; + overflow: hidden; +} + +.OT_audio-level-meter:before { + /* makes the height of the container equals its width */ + content: ''; + display: block; + padding-top: 100%; +} + +.OT_audio-level-meter__bar { + position: absolute; + width: 192%; /* meter value can overflow of 8% */ + height: 192%; + top: -96% /* half of the size */; + right: -96%; + border-radius: 50%; + + background-color: rgba(0, 0, 0, .8); +} + +.OT_audio-level-meter__audio-only-img { + position: absolute; + top: 22%; + right: 15%; + width: 40%; + + opacity: .7; + + background: url() no-repeat center; +} + +.OT_audio-level-meter__audio-only-img:before { + /* makes the height of the container equals its width */ + content: ''; + display: block; + padding-top: 100%; +} + +.OT_audio-level-meter__value { + position: absolute; + border-radius: 50%; + background-image: radial-gradient(circle, rgba(151, 206, 0, 1) 0%, rgba(151, 206, 0, 0) 100%); +} + +.OT_audio-level-meter.OT_mode-off { + display: none; +} + +.OT_audio-level-meter.OT_mode-on, +.OT_audio-only .OT_audio-level-meter.OT_mode-auto { + display: block; +} + +.OT_audio-only.OT_publisher .OT_video-element, +.OT_audio-only.OT_subscriber .OT_video-element { + display: none; +} + + +.OT_video-disabled-indicator { + opacity: 1; + border: none; + display: none; + position: absolute; + background-color: transparent; + background-repeat: no-repeat; + background-position: bottom right; + pointer-events: none; + top: 0; + left: 0; + bottom: 3px; + right: 3px; +} + +.OT_video-disabled { + background-image: url(); + background-size: 33px auto; +} + +.OT_video-disabled-warning { + background-image: url(); + background-size: 33px auto; +} + +.OT_video-disabled-indicator.OT_active { + display: block; +} + +.OT_audio-blocked-indicator { + opacity: 1; + border: none; + display: none; + position: absolute; + background-color: transparent; + background-repeat: no-repeat; + background-position: center; + pointer-events: none; + top: 0; + left: 0; + bottom: 0; + right: 0; +} + +.OT_audio-blocked { + background-image: url(); + background-size: 90px auto; +} + +.OT_container-audio-blocked { + cursor: pointer; +} + +.OT_container-audio-blocked.OT_mini .OT_edge-bar-item { + display: none; +} + +.OT_container-audio-blocked .OT_mute { + display: none; +} + +.OT_audio-blocked-indicator.OT_active { + display: block; +} + +.OT_video-unsupported { + opacity: 1; + border: none; + display: none; + position: absolute; + background-color: transparent; + background-repeat: no-repeat; + background-position: center; + background-image: url(); + background-size: 58px auto; + pointer-events: none; + top: 0; + left: 0; + bottom: 0; + right: 0; + margin-top: -30px; +} + +.OT_video-unsupported-bar { + display: none; + position: absolute; + width: 192%; /* copy the size of the audio meter bar for symmetry */ + height: 192%; + top: -96% /* half of the size */; + left: -96%; + border-radius: 50%; + + background-color: rgba(0, 0, 0, .8); +} + +.OT_video-unsupported-img { + display: none; + position: absolute; + top: 11%; + left: 15%; + width: 70%; + opacity: .7; + background-image: url(); + background-repeat: no-repeat; + background-position: center; + background-size: 100% auto; +} + +.OT_video-unsupported-img:before { + /* makes the height of the container 93% of its width (90/97 px) */ + content: ''; + display: block; + padding-top: 93%; +} + +.OT_video-unsupported-text { + display: -webkit-box; + display: -ms-flexbox; + display: flex; + -webkit-box-pack: center; + -ms-flex-pack: center; + justify-content: center; + -webkit-box-align: center; + -ms-flex-align: center; + align-items: center; + text-align: center; height: 100%; + margin-top: 40px; } diff --git a/openvidu-server/src/angular/frontend/src/app/components/layouts/layout-best-fit/layout-best-fit.component.html b/openvidu-server/src/angular/frontend/src/app/components/layouts/layout-best-fit/layout-best-fit.component.html index 0ebbb35821..a70028b75e 100644 --- a/openvidu-server/src/angular/frontend/src/app/components/layouts/layout-best-fit/layout-best-fit.component.html +++ b/openvidu-server/src/angular/frontend/src/app/components/layouts/layout-best-fit/layout-best-fit.component.html @@ -1,8 +1,7 @@ -
-
-
- +
+
+
+
diff --git a/openvidu-server/src/angular/frontend/src/app/components/layouts/layout-best-fit/layout-best-fit.component.ts b/openvidu-server/src/angular/frontend/src/app/components/layouts/layout-best-fit/layout-best-fit.component.ts index 8b5f2a03b1..7b9f3e5f8c 100644 --- a/openvidu-server/src/angular/frontend/src/app/components/layouts/layout-best-fit/layout-best-fit.component.ts +++ b/openvidu-server/src/angular/frontend/src/app/components/layouts/layout-best-fit/layout-best-fit.component.ts @@ -1,22 +1,28 @@ -import { Component, OnInit, OnDestroy, HostListener } from '@angular/core'; +import { Component, OnInit, OnDestroy, HostListener, ViewEncapsulation, ApplicationRef } from '@angular/core'; import { ActivatedRoute } from '@angular/router'; -import { OpenVidu, Session, Stream } from 'openvidu-browser'; +import { OpenVidu, Session, Stream, Subscriber } from 'openvidu-browser'; + +import { OpenViduLayout } from '../../../../assets/openvidu-layout'; @Component({ selector: 'app-layout-best-fit', templateUrl: './layout-best-fit.component.html', - styleUrls: ['./layout-best-fit.component.css'] + styleUrls: ['./layout-best-fit.component.css'], + encapsulation: ViewEncapsulation.None }) export class LayoutBestFitComponent implements OnInit, OnDestroy { + openviduLayout: OpenViduLayout; sessionId: string; secret: string; session: Session; - numberOfVideos = 0; - remoteStreams = []; + streams = []; + + layout: any; + resizeTimeout; - constructor(private route: ActivatedRoute) { + constructor(private route: ActivatedRoute, private appRef: ApplicationRef) { this.route.params.subscribe(params => { this.sessionId = params.sessionId; this.secret = params.secret; @@ -28,6 +34,14 @@ export class LayoutBestFitComponent implements OnInit, OnDestroy { this.leaveSession(); } + @HostListener('window:resize', ['$event']) + sizeChange(event) { + clearTimeout(this.resizeTimeout); + this.resizeTimeout = setTimeout(() => { + this.openviduLayout.updateLayout(); + }, 20); + } + ngOnDestroy() { this.leaveSession(); } @@ -39,15 +53,14 @@ export class LayoutBestFitComponent implements OnInit, OnDestroy { this.session = OV.initSession(fullSessionId); this.session.on('streamCreated', (event) => { - this.numberOfVideos++; + const subscriber: Subscriber = this.session.subscribe(event.stream, ''); this.addRemoteStream(event.stream); - this.session.subscribe(event.stream, ''); }); this.session.on('streamDestroyed', (event) => { - this.numberOfVideos--; event.preventDefault(); this.deleteRemoteStream(event.stream); + this.openviduLayout.updateLayout(); }); this.session.connect(null, (error) => { @@ -55,79 +68,46 @@ export class LayoutBestFitComponent implements OnInit, OnDestroy { console.error(error); } }); + + this.openviduLayout = new OpenViduLayout(); + this.openviduLayout.initLayoutContainer(document.getElementById('layout'), { + maxRatio: 3 / 2, // The narrowest ratio that will be used (default 2x3) + minRatio: 9 / 16, // The widest ratio that will be used (default 16x9) + fixedRatio: false, /* If this is true then the aspect ratio of the video is maintained + and minRatio and maxRatio are ignored (default false) */ + bigClass: 'OV_big', // The class to add to elements that should be sized bigger + bigPercentage: 0.8, // The maximum percentage of space the big ones should take up + bigFixedRatio: false, // fixedRatio for the big ones + bigMaxRatio: 3 / 2, // The narrowest ratio to use for the big elements (default 2x3) + bigMinRatio: 9 / 16, // The widest ratio to use for the big elements (default 16x9) + bigFirst: true, // Whether to place the big one in the top left (true) or bottom right + animate: true // Whether you want to animate the transitions + }); } private addRemoteStream(stream: Stream): void { - switch (true) { - case (this.numberOfVideos <= 2): - if (this.remoteStreams[0] == null) { this.remoteStreams[0] = []; } - this.remoteStreams[0].push(stream); - break; - case (this.numberOfVideos <= 4): - if (this.remoteStreams[1] == null) { this.remoteStreams[1] = []; } - this.remoteStreams[1].push(stream); - break; - case (this.numberOfVideos <= 5): - this.remoteStreams[0].push(stream); - break; - case (this.numberOfVideos <= 6): - this.remoteStreams[1].push(stream); - break; - default: - if (this.remoteStreams[2] == null) { this.remoteStreams[2] = []; } - this.remoteStreams[2].push(stream); - break; - } + this.streams.push(stream); + this.appRef.tick(); } private deleteRemoteStream(stream: Stream): void { - for (let i = 0; i < this.remoteStreams.length; i++) { - const index = this.remoteStreams[i].indexOf(stream, 0); - if (index > -1) { - this.remoteStreams[i].splice(index, 1); - this.reArrangeVideos(); - break; - } + const index = this.streams.indexOf(stream, 0); + if (index > -1) { + this.streams.splice(index, 1); } - } - - private reArrangeVideos(): void { - switch (true) { - case (this.numberOfVideos === 1): - if (this.remoteStreams[0].length === 0) { - this.remoteStreams[0].push(this.remoteStreams[1].pop()); - } - break; - case (this.numberOfVideos === 2): - if (this.remoteStreams[0].length === 1) { - this.remoteStreams[0].push(this.remoteStreams[1].pop()); - } - break; - case (this.numberOfVideos === 3): - if (this.remoteStreams[0].length === 1) { - this.remoteStreams[0].push(this.remoteStreams[1].pop()); - } - break; - case (this.numberOfVideos === 4): - if (this.remoteStreams[0].length === 3) { - this.remoteStreams[1].unshift(this.remoteStreams[0].pop()); - } - break; - case (this.numberOfVideos === 5): - if (this.remoteStreams[0].length === 2) { - this.remoteStreams[0].push(this.remoteStreams[1].shift()); - } - break; - } - this.remoteStreams = this.remoteStreams.filter((array) => { return array.length > 0 }); - + this.appRef.tick(); } leaveSession() { if (this.session) { this.session.disconnect(); }; - this.remoteStreams = []; - this.numberOfVideos = 0; + this.streams = []; this.session = null; } + onVideoPlaying(event) { + const video: HTMLVideoElement = event.target; + video.parentElement.parentElement.classList.remove('custom-class'); + this.openviduLayout.updateLayout(); + } + } diff --git a/openvidu-server/src/angular/frontend/src/assets/openvidu-layout.ts b/openvidu-server/src/angular/frontend/src/assets/openvidu-layout.ts new file mode 100644 index 0000000000..58d42daead --- /dev/null +++ b/openvidu-server/src/angular/frontend/src/assets/openvidu-layout.ts @@ -0,0 +1,357 @@ +declare var $: any; + +export interface OpenViduLayoutOptions { + maxRatio: number; + minRatio: number; + fixedRatio: boolean; + animate: any; + bigClass: string; + bigPercentage: any; + bigFixedRatio: any; + bigMaxRatio: any; + bigMinRatio: any; + bigFirst: any; +} + +export class OpenViduLayout { + + private layoutContainer: HTMLElement; + private opts: OpenViduLayoutOptions; + + private fixAspectRatio(elem: HTMLVideoElement, width: number) { + const sub: HTMLVideoElement = elem.querySelector('.OT_root'); + if (sub) { + // If this is the parent of a subscriber or publisher then we need + // to force the mutation observer on the publisher or subscriber to + // trigger to get it to fix it's layout + const oldWidth = sub.style.width; + sub.style.width = width + 'px'; + // sub.style.height = height + 'px'; + sub.style.width = oldWidth || ''; + } + } + + private positionElement(elem: HTMLVideoElement, x: number, y: number, width: number, height: number, animate: any) { + const targetPosition = { + left: x + 'px', + top: y + 'px', + width: width + 'px', + height: height + 'px' + }; + + this.fixAspectRatio(elem, width); + + if (animate && $) { + $(elem).stop(); + $(elem).animate(targetPosition, animate.duration || 200, animate.easing || 'swing', + () => { + this.fixAspectRatio(elem, width); + if (animate.complete) { animate.complete.call(this); } + }); + } else { + $(elem).css(targetPosition); + } + this.fixAspectRatio(elem, width); + } + + private getVideoRatio(elem: HTMLVideoElement) { + if (!elem) { + return 3 / 4; + } + const video: HTMLVideoElement = elem.querySelector('video'); + if (video && video.videoHeight && video.videoWidth) { + return video.videoHeight / video.videoWidth; + } else if (elem.videoHeight && elem.videoWidth) { + return elem.videoHeight / elem.videoWidth; + } + return 3 / 4; + } + + private getCSSNumber(elem: HTMLElement, prop: string) { + const cssStr = $(elem).css(prop); + return cssStr ? parseInt(cssStr, 10) : 0; + } + + // Really cheap UUID function + private cheapUUID() { + return (Math.random() * 100000000).toFixed(0); + } + + private getHeight(elem: HTMLElement) { + const heightStr = $(elem).css('height'); + return heightStr ? parseInt(heightStr, 10) : 0; + } + + private getWidth(elem: HTMLElement) { + const widthStr = $(elem).css('width'); + return widthStr ? parseInt(widthStr, 10) : 0; + } + + private getBestDimensions(minR: number, maxR: number, count: number, WIDTH: number, HEIGHT: number, targetHeight: number) { + let maxArea, + targetCols, + targetRows, + targetWidth, + tWidth, + tHeight, + tRatio; + + // Iterate through every possible combination of rows and columns + // and see which one has the least amount of whitespace + for (let i = 1; i <= count; i++) { + const colsAux = i; + const rowsAux = Math.ceil(count / colsAux); + + // Try taking up the whole height and width + tHeight = Math.floor(HEIGHT / rowsAux); + tWidth = Math.floor(WIDTH / colsAux); + + tRatio = tHeight / tWidth; + if (tRatio > maxR) { + // We went over decrease the height + tRatio = maxR; + tHeight = tWidth * tRatio; + } else if (tRatio < minR) { + // We went under decrease the width + tRatio = minR; + tWidth = tHeight / tRatio; + } + + const area = (tWidth * tHeight) * count; + + // If this width and height takes up the most space then we're going with that + if (maxArea === undefined || (area > maxArea)) { + maxArea = area; + targetHeight = tHeight; + targetWidth = tWidth; + targetCols = colsAux; + targetRows = rowsAux; + } + } + return { + maxArea: maxArea, + targetCols: targetCols, + targetRows: targetRows, + targetHeight: targetHeight, + targetWidth: targetWidth, + ratio: targetHeight / targetWidth + }; + }; + + private arrange(children: HTMLVideoElement[], WIDTH: number, HEIGHT: number, offsetLeft: number, offsetTop: number, fixedRatio: boolean, + minRatio: number, maxRatio: number, animate: any) { + + let targetHeight; + + const count = children.length; + let dimensions; + + if (!fixedRatio) { + dimensions = this.getBestDimensions(minRatio, maxRatio, count, WIDTH, HEIGHT, targetHeight); + } else { + // Use the ratio of the first video element we find to approximate + const ratio = this.getVideoRatio(children.length > 0 ? children[0] : null); + dimensions = this.getBestDimensions(ratio, ratio, count, WIDTH, HEIGHT, targetHeight); + } + + // Loop through each stream in the container and place it inside + let x = 0, + y = 0; + const rows = []; + let row; + // Iterate through the children and create an array with a new item for each row + // and calculate the width of each row so that we know if we go over the size and need + // to adjust + for (let i = 0; i < children.length; i++) { + if (i % dimensions.targetCols === 0) { + // This is a new row + row = { + children: [], + width: 0, + height: 0 + }; + rows.push(row); + } + const elem: HTMLVideoElement = children[i]; + row.children.push(elem); + let targetWidth = dimensions.targetWidth; + targetHeight = dimensions.targetHeight; + // If we're using a fixedRatio then we need to set the correct ratio for this element + if (fixedRatio) { + targetWidth = targetHeight / this.getVideoRatio(elem); + } + row.width += targetWidth; + row.height = targetHeight; + } + // Calculate total row height adjusting if we go too wide + let totalRowHeight = 0; + let remainingShortRows = 0; + for (let i = 0; i < rows.length; i++) { + row = rows[i]; + if (row.width > WIDTH) { + // Went over on the width, need to adjust the height proportionally + row.height = Math.floor(row.height * (WIDTH / row.width)); + row.width = WIDTH; + } else if (row.width < WIDTH) { + remainingShortRows += 1; + } + totalRowHeight += row.height; + } + if (totalRowHeight < HEIGHT && remainingShortRows > 0) { + // We can grow some of the rows, we're not taking up the whole height + let remainingHeightDiff = HEIGHT - totalRowHeight; + totalRowHeight = 0; + for (let i = 0; i < rows.length; i++) { + row = rows[i]; + if (row.width < WIDTH) { + // Evenly distribute the extra height between the short rows + let extraHeight = remainingHeightDiff / remainingShortRows; + if ((extraHeight / row.height) > ((WIDTH - row.width) / row.width)) { + // We can't go that big or we'll go too wide + extraHeight = Math.floor(((WIDTH - row.width) / row.width) * row.height); + } + row.width += Math.floor((extraHeight / row.height) * row.width); + row.height += extraHeight; + remainingHeightDiff -= extraHeight; + remainingShortRows -= 1; + } + totalRowHeight += row.height; + } + } + // vertical centering + y = ((HEIGHT - (totalRowHeight)) / 2); + // Iterate through each row and place each child + for (let i = 0; i < rows.length; i++) { + row = rows[i]; + // center the row + const rowMarginLeft = ((WIDTH - row.width) / 2); + x = rowMarginLeft; + for (let j = 0; j < row.children.length; j++) { + const elem: HTMLVideoElement = row.children[j]; + + let targetWidth = dimensions.targetWidth; + targetHeight = row.height; + // If we're using a fixedRatio then we need to set the correct ratio for this element + if (fixedRatio) { + targetWidth = Math.floor(targetHeight / this.getVideoRatio(elem)); + } + elem.style.position = 'absolute'; + // $(elem).css('position', 'absolute'); + const actualWidth = targetWidth - this.getCSSNumber(elem, 'paddingLeft') - + this.getCSSNumber(elem, 'paddingRight') - + this.getCSSNumber(elem, 'marginLeft') - + this.getCSSNumber(elem, 'marginRight') - + this.getCSSNumber(elem, 'borderLeft') - + this.getCSSNumber(elem, 'borderRight'); + + const actualHeight = targetHeight - this.getCSSNumber(elem, 'paddingTop') - + this.getCSSNumber(elem, 'paddingBottom') - + this.getCSSNumber(elem, 'marginTop') - + this.getCSSNumber(elem, 'marginBottom') - + this.getCSSNumber(elem, 'borderTop') - + this.getCSSNumber(elem, 'borderBottom'); + + this.positionElement(elem, x + offsetLeft, y + offsetTop, actualWidth, actualHeight, animate); + x += targetWidth; + } + y += targetHeight; + } + } + + private filterDisplayNone(element: HTMLElement) { + return element.style.display !== 'none'; + } + + updateLayout() { + if (this.layoutContainer.style.display === 'none') { + return; + } + let id = this.layoutContainer.id; + if (!id) { + id = 'OT_' + this.cheapUUID(); + this.layoutContainer.id = id; + } + + const HEIGHT = this.getHeight(this.layoutContainer) - + this.getCSSNumber(this.layoutContainer, 'borderTop') - + this.getCSSNumber(this.layoutContainer, 'borderBottom'); + const WIDTH = this.getWidth(this.layoutContainer) - + this.getCSSNumber(this.layoutContainer, 'borderLeft') - + this.getCSSNumber(this.layoutContainer, 'borderRight'); + + const availableRatio = HEIGHT / WIDTH; + + let offsetLeft = 0; + let offsetTop = 0; + let bigOffsetTop = 0; + let bigOffsetLeft = 0; + + const bigOnes = Array.prototype.filter.call( + this.layoutContainer.querySelectorAll('#' + id + '>.' + this.opts.bigClass), + this.filterDisplayNone); + const smallOnes = Array.prototype.filter.call( + this.layoutContainer.querySelectorAll('#' + id + '>*:not(.' + this.opts.bigClass + ')'), + this.filterDisplayNone); + + if (bigOnes.length > 0 && smallOnes.length > 0) { + let bigWidth, bigHeight; + + if (availableRatio > this.getVideoRatio(bigOnes[0])) { + // We are tall, going to take up the whole width and arrange small + // guys at the bottom + bigWidth = WIDTH; + bigHeight = Math.floor(HEIGHT * this.opts.bigPercentage); + offsetTop = bigHeight; + bigOffsetTop = HEIGHT - offsetTop; + } else { + // We are wide, going to take up the whole height and arrange the small + // guys on the right + bigHeight = HEIGHT; + bigWidth = Math.floor(WIDTH * this.opts.bigPercentage); + offsetLeft = bigWidth; + bigOffsetLeft = WIDTH - offsetLeft; + } + if (this.opts.bigFirst) { + this.arrange(bigOnes, bigWidth, bigHeight, 0, 0, this.opts.bigFixedRatio, this.opts.bigMinRatio, + this.opts.bigMaxRatio, this.opts.animate); + this.arrange(smallOnes, WIDTH - offsetLeft, HEIGHT - offsetTop, offsetLeft, offsetTop, + this.opts.fixedRatio, this.opts.minRatio, this.opts.maxRatio, this.opts.animate); + } else { + this.arrange(smallOnes, WIDTH - offsetLeft, HEIGHT - offsetTop, 0, 0, this.opts.fixedRatio, + this.opts.minRatio, this.opts.maxRatio, this.opts.animate); + this.arrange(bigOnes, bigWidth, bigHeight, bigOffsetLeft, bigOffsetTop, + this.opts.bigFixedRatio, this.opts.bigMinRatio, this.opts.bigMaxRatio, this.opts.animate); + } + } else if (bigOnes.length > 0 && smallOnes.length === 0) { + this. + // We only have one bigOne just center it + arrange(bigOnes, WIDTH, HEIGHT, 0, 0, this.opts.bigFixedRatio, this.opts.bigMinRatio, + this.opts.bigMaxRatio, this.opts.animate); + } else { + this.arrange(smallOnes, WIDTH - offsetLeft, HEIGHT - offsetTop, offsetLeft, offsetTop, + this.opts.fixedRatio, this.opts.minRatio, this.opts.maxRatio, this.opts.animate); + } + } + + initLayoutContainer(container, opts) { + this.opts = { + maxRatio: (opts.maxRatio != null) ? opts.maxRatio : 3 / 2, + minRatio: (opts.minRatio != null) ? opts.minRatio : 9 / 16, + fixedRatio: (opts.fixedRatio != null) ? opts.fixedRatio : false, + animate: (opts.animate != null) ? opts.animate : false, + bigClass: (opts.bigClass != null) ? opts.bigClass : 'OT_big', + bigPercentage: (opts.bigPercentage != null) ? opts.bigPercentage : 0.8, + bigFixedRatio: (opts.bigFixedRatio != null) ? opts.bigFixedRatio : false, + bigMaxRatio: (opts.bigMaxRatio != null) ? opts.bigMaxRatio : 3 / 2, + bigMinRatio: (opts.bigMinRatio != null) ? opts.bigMinRatio : 9 / 16, + bigFirst: (opts.bigFirst != null) ? opts.bigFirst : true + }; + this.layoutContainer = typeof (container) === 'string' ? $(container) : container; + } + + setLayoutOptions(options: OpenViduLayoutOptions) { + this.opts = options; + } + +} + diff --git a/openvidu-server/src/angular/frontend/src/index.html b/openvidu-server/src/angular/frontend/src/index.html index 253efe2d3c..38df65e237 100644 --- a/openvidu-server/src/angular/frontend/src/index.html +++ b/openvidu-server/src/angular/frontend/src/index.html @@ -9,7 +9,7 @@ - diff --git a/openvidu-server/src/angular/frontend/src/styles.css b/openvidu-server/src/angular/frontend/src/styles.css index c709b01983..c487131c62 100644 --- a/openvidu-server/src/angular/frontend/src/styles.css +++ b/openvidu-server/src/angular/frontend/src/styles.css @@ -16,7 +16,7 @@ li { list-style: none; } -video { +#mirrored-video video { width: 100%; }