gecko-dev/browser/components/loop/content/shared/css/conversation.css

1079 lines
23 KiB
CSS

/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
/* Shared conversation window styles */
.conversation {
position: relative;
}
.conversation-toolbar {
z-index: 999; /* required to have it superimposed to the video element */
border: 1px solid #5a5a5a;
border-left: 0;
border-right: 0;
background: rgba(0,0,0,.70);
}
/* desktop version */
.fx-embedded .conversation-toolbar {
/* required to have dropdowns float atop the .room-invitation-overlay: */
z-index: 1020;
position: absolute;
top: 0;
left: 0;
right: 0;
/* note that .room-invitation-overlay top matches this */
height: 26px;
}
/* standalone version */
.standalone .conversation-toolbar {
padding: 20px;
height: 64px;
}
.standalone .focus-stream {
/* Set at maximum height, minus height of conversation toolbar */
height: calc(100% - 64px);
}
.standalone .in-call .focus-stream {
height: 100%;
}
.conversation-toolbar > li {
float: left;
font-size: 0; /* prevents vertical bottom padding added to buttons in google
chrome */
}
.standalone .conversation-toolbar > li {
/* XXX Might make sense to use relative units here.
*/
margin-right: 16px;
}
.btn-screen-share-entry {
float: right !important;
border-left: 1px solid #5a5a5a;
}
.conversation-toolbar-btn-box {
border-right: 1px solid #5a5a5a;
}
.standalone .conversation-toolbar-btn-box {
/* overwrite the style for standalone
* because we reuse the same component */
border: none;
}
.conversation-toolbar .btn {
/* dimensions according to spec
* https://people.mozilla.org/~dhenein/labs/loop-link-spec/#video-call */
width: 32px;
height: 24px;
background-position: center;
background-size: 40%;
background-repeat: no-repeat;
}
.standalone .media-control {
width: 36px;
background-position: center;
border-radius: 28px;
}
.standalone-conversation-toolbar-media-btn:hover {
background-color: rgba(255,255,255,.35);
}
.fx-embedded-answer-btn-text {
vertical-align: bottom;
/* always leave space for the icon (width and margin) */
max-width: calc(100% - .8rem - .2rem);
}
.fx-embedded-btn-icon-video,
.fx-embedded-btn-icon-audio {
display: inline-block;
vertical-align: top;
width: .8rem;
height: .8rem;
background-repeat: no-repeat;
cursor: pointer;
-moz-margin-start: .2rem;
}
.fx-embedded-btn-icon-video,
.fx-embedded-btn-video-small,
.fx-embedded-tiny-video-icon {
background-image: url("../img/video-inverse-14x14.png");
}
.fx-embedded-btn-icon-audio,
.fx-embedded-btn-audio-small,
.fx-embedded-tiny-audio-icon {
background-image: url("../img/audio-inverse-14x14.png");
}
.fx-embedded-btn-audio-small,
.fx-embedded-btn-video-small {
width: 26px;
height: 26px;
border-left: 1px solid rgba(255,255,255,.4);
border-top-right-radius: 2px;
border-bottom-right-radius: 2px;
background-color: #5bc0a4;
background-position: center;
background-size: 1rem;
background-repeat: no-repeat;
cursor: pointer;
}
.fx-embedded-btn-video-small:hover,
.fx-embedded-btn-audio-small:hover {
background-color: #6cb23e;
}
@media (min-resolution: 2dppx) {
.fx-embedded-btn-audio-small {
background-image: url("../img/audio-inverse-14x14@2x.png");
}
.fx-embedded-btn-video-small {
background-image: url("../img/video-inverse-14x14@2x.png");
}
}
.standalone .btn-hangup {
width: auto;
font-size: 12px;
border-radius: 2px;
padding: 0 20px;
}
.fx-embedded .conversation-toolbar .btn-hangup {
background-image: url(../img/hangup-inverse-14x14.png);
}
@media (min-resolution: 2dppx) {
.fx-embedded .conversation-toolbar .btn-hangup {
background-image: url(../img/hangup-inverse-14x14@2x.png);
}
}
/* Common media control buttons behavior */
.conversation-toolbar .transparent-button {
background-color: transparent;
opacity: 1;
}
.conversation-toolbar .transparent-button:hover,
.conversation-toolbar .transparent-button.menu-showing {
background-color: rgba(255,255,255,.35);
opacity: 1;
}
.conversation-toolbar .media-control.muted {
background-color: #0096DD;
opacity: 1;
}
/* Audio mute button */
.btn-mute-audio {
background-image: url(../img/audio-inverse-14x14.png);
}
.btn-mute-audio.muted {
background-image: url(../img/mute-inverse-14x14.png);
}
@media (min-resolution: 2dppx) {
.btn-mute-audio {
background-image: url(../img/audio-inverse-14x14@2x.png);
}
.btn-mute-audio.muted {
background-image: url(../img/mute-inverse-14x14@2x.png);
}
}
/* Video mute button */
.btn-mute-video {
background-image: url(../img/video-inverse-14x14.png);
}
.btn-mute-video.muted {
background-image: url(../img/facemute-14x14.png);
}
@media (min-resolution: 2dppx) {
.btn-mute-video {
background-image: url(../img/video-inverse-14x14@2x.png);
}
.btn-mute-video.muted {
background-image: url(../img/facemute-14x14@2x.png);
}
}
/* Screen share button */
.btn-screen-share {
position: relative;
background-image: url(../img/icons-16x16.svg#screen-white);
background-size: 16px 16px;
width: 42px;
}
/* Make room for the chevron. */
.conversation-toolbar .btn-screen-share:not(.active) {
background-position: 5px center;
}
.btn-screen-share.active {
background-image: url(../img/icons-16x16.svg#screenmute-white);
background-color: #6CB23E;
opacity: 1;
}
.btn-screen-share.disabled {
background-image: url(../img/icons-16x16.svg#screen-disabled);
}
.btn-screen-share .chevron {
background-image: url(../img/icons-10x10.svg#dropdown-white);
background-size: 10px 10px;
position: absolute;
right: 2px;
top: 8px;
width: 10px;
height: 10px;
}
.btn-screen-share.disabled .chevron {
background-image: url(../img/icons-10x10.svg#dropdown-disabled);
}
.fx-embedded .remote_wrapper {
position: absolute;
top: 0px;
right: 0px;
bottom: 0px;
left: 0px;
}
.standalone .local-stream,
.standalone .remote-inset-stream {
/* required to have it superimposed to the control toolbar */
z-index: 1001;
}
.standalone .room-conversation .local-stream,
.standalone .room-conversation .remote-inset-stream {
box-shadow: none;
}
/* Side by side video elements */
.conversation .media.side-by-side .focus-stream {
width: 50%;
float: left;
}
.conversation .media.side-by-side .local-stream,
.conversation .media.side-by-side .remote-inset-stream {
width: 50%;
}
/* Call ended view */
.call-ended p {
text-align: center;
}
/* General Call (incoming or outgoing). */
/*
* Height matches the height of the docked window
* but the UI breaks when you pop out
* Bug 1040985
*/
.call-window {
display: flex;
flex-direction: column;
align-items: center;
justify-content: space-between;
min-height: 230px;
}
.call-window > .btn-label {
text-align: center;
}
.call-window > .error {
text-align: center;
color: #f00;
font-size: 90%;
}
.call-action-group {
display: flex;
padding: 2.5em 4px 0 4px;
width: 100%;
}
.call-action-group > .btn {
min-height: 26px;
border-radius: 2px;
margin: 0 4px;
min-width: 64px;
}
.call-action-group .btn-group-chevron,
.call-action-group .btn-group {
width: 100%;
}
/* XXX Once we get the incoming call avatar, bug 1047435, the H2 should
* disappear from our markup, and we should remove this rule entirely.
*/
.call-window h2 {
font-size: 1.5em;
font-weight: normal;
text-align: center;
/* compensate for reset.css overriding this; values borrowed from
Firefox Mac html.css */
margin: 0.83em 0;
}
.fx-embedded-call-button-spacer {
display: flex;
flex: 1;
}
/*
* Dropdown menu hidden behind a chevron
*
* .native-dropdown-menu[-large-parent] Generic class, contains common styles
* .standalone-dropdown-menu Initiate call dropdown menu
* .conversation-window-dropdown Dropdown menu for answer/decline/block options
*/
.native-dropdown-menu,
.native-dropdown-large-parent {
/* Should match a native select menu */
padding: 0;
position: absolute; /* element can be wider than the parent */
background: #fff;
margin: 0;
box-shadow: 0 4px 5px rgba(30,30,30,.3);
border-style: solid;
border-width: 1px 1px 1px 2px;
border-color: #aaa #111 #111 #aaa;
}
/*
* If the component is smaller than the parent
* we need it to display block to occupy full width
* Same as above but overrides apropriate styles
*/
.native-dropdown-large-parent {
position: relative;
display: block;
}
.native-dropdown-menu li,
.native-dropdown-large-parent li {
list-style: none;
cursor: pointer;
color: #000;
}
.native-dropdown-menu li:hover,
.native-dropdown-large-parent li:hover,
.native-dropdown-large-parent li:hover button {
color: #fff;
background-color: #111;
}
.conversation-window-dropdown > li {
padding: 2px;
font-size: .7rem;
white-space: nowrap;
}
/* Expired call url page */
.expired-url-info {
width: 400px;
margin: 0 auto;
}
.promote-firefox {
text-align: center;
font-size: 18px;
line-height: 24px;
margin: 2em 0;
}
.promote-firefox h3 {
font-weight: 300;
}
/* Feedback form */
.feedback {
padding: 14px;
}
.feedback p {
margin: 0px;
}
.feedback h3 {
color: #666;
font-size: 12px;
font-weight: 700;
text-align: center;
margin: 0 0 1em 0;
}
.feedback .faces {
display: flex;
flex-direction: row;
align-items: center;
justify-content: center;
padding: 20px 0;
}
.feedback .face {
border: 1px solid transparent;
box-shadow: 0 1px 2px #CCC;
cursor: pointer;
border-radius: 4px;
margin: 0 10px;
width: 80px;
height: 80px;
background-color: #fbfbfb;
background-size: 60px auto;
background-position: center center;
background-repeat: no-repeat;
}
.feedback .face:hover {
border: 1px solid #DDD;
background-color: #FEFEFE;
}
.feedback .face.face-happy {
background-image: url("../img/happy.png");
}
.feedback .face.face-sad {
background-image: url("../img/sad.png");
}
.fx-embedded-btn-back {
margin-bottom: 1rem;
padding: .2rem .8rem;
border: 1px solid #aaa;
border-radius: 2px;
background: transparent;
color: #777;
cursor: pointer;
}
.feedback-category-label {
display: block;
line-height: 1.5em;
}
.feedback-category-radio {
margin-right: .5em;
}
.feedback > form > .btn-success,
.feedback-description {
width: 100%;
margin-top: 14px;
}
.feedback > form > .btn-success {
padding-top: .5em;
padding-bottom: .5em;
border-radius: 2px;
}
.feedback .info {
display: block;
font-size: 10px;
color: #CCC;
text-align: center;
}
.fx-embedded .local-stream {
position: absolute;
right: 3px;
bottom: 5px;
/* next two lines are workaround for lack of object-fit; see bug 1020445 */
max-width: 140px;
width: 30%;
height: 28%;
max-height: 105px;
box-shadow: 0px 2px 4px rgba(0,0,0,.5);
}
.fx-embedded .room-conversation .local-stream {
box-shadow: none;
}
.fx-embedded .local-stream.room-preview {
top: 0px;
left: 0px;
right: 0px;
bottom: 0px;
height: 100%;
width: 100%;
max-width: none;
max-height: none;
}
.conversation .media.nested .focus-stream {
display: inline-block;
position: absolute; /* workaround for lack of object-fit; see bug 1020445 */
width: 100%;
top: 0;
bottom: 0;
left: 0;
right: 0;
}
/*
* XXX this approach is fragile because it makes assumptions
* about the generated OT markup, any change will break it
*/
/*
* For any audio-only streams, we want to display our own background
*/
.OT_audio-only .OT_widget-container .OT_video-poster {
background-image: url("../img/audio-call-avatar.svg");
background-repeat: no-repeat;
background-color: #4BA6E7;
background-size: contain;
background-position: center;
}
/*
* Audio-only. For local streams, cancel out the SDK's opacity of 0.25.
* For remote streams we leave them shaded, as otherwise its too bright.
*/
.local-stream-audio .OT_publisher .OT_video-poster {
opacity: 1
}
/*
* In audio-only mode, don't display the video element, doing so interferes
* with the background opacity of the video-poster element.
*/
.OT_audio-only .OT_widget-container .OT_video-element {
display: none;
}
/*
* Ensure that the publisher (i.e. local) video is never cropped, so that it's
* not possible for someone to be presented with a picture that displays
* (for example) a person from the neck up, even though the camera is capturing
* and transmitting a picture of that person from the waist up.
*
* The !importants are necessary to override the SDK attempts to avoid
* letterboxing entirely.
*
* If we could easily use test video streams with the SDK (eg if initPublisher
* supported something like a "testMediaToStreamURI" parameter that it would
* use to source the stream rather than the output of gUM, it wouldn't be too
* hard to generate a video with a 1 pixel border at the edges that one could
* at least visually see wasn't being cropped.
*
* Another less ugly possibility would be to work with Ted Mielczarek to use
* the fake camera drivers he has for Linux.
*/
.room-conversation .OT_publisher .OT_widget-container {
height: 100% !important;
width: 100% !important;
top: 0 !important;
left: 0 !important;
background-color: transparent; /* avoid visually obvious letterboxing */
}
.room-conversation .OT_publisher .OT_widget-container video {
background-color: transparent; /* avoid visually obvious letterboxing */
}
.fx-embedded .room-conversation .room-preview .OT_publisher .OT_widget-container,
.fx-embedded .room-conversation .room-preview .OT_publisher .OT_widget-container video {
/* Desktop conversation window room preview local stream actually wants
a black background */
background-color: #000;
}
.fx-embedded .media.nested {
min-height: 200px;
}
.fx-embedded-call-identifier {
display: inline;
width: 100%;
padding: 1.2em;
}
.fx-embedded-call-identifier-item {
height: 50px;
}
.fx-embedded-call-identifier-avatar {
max-width: 50px;
min-width: 50px;
background: #ccc;
border-radius: 50%;
background-image: url("../img/audio-call-avatar.svg");
background-repeat: no-repeat;
background-color: #4ba6e7;
background-size: contain;
overflow: hidden;
box-shadow: inset 0 0 0 1px rgba(255,255,255,.3);
float: left;
-moz-margin-end: 1em;
}
.fx-embedded-call-identifier-text {
font-weight: bold;
}
.fx-embedded-call-identifier-info {
flex: 1;
display: flex;
flex-direction: column;
justify-content: center;
-moz-margin-start: 1em;
}
.fx-embedded-conversation-timestamp {
font-size: .6rem;
line-height: 17px;
display: inline-block;
vertical-align: top;
}
.fx-embedded-call-detail {
padding-top: 1.2em;
}
.fx-embedded-tiny-video-icon {
margin: 0 0.8em;
}
.fx-embedded-tiny-audio-icon,
.fx-embedded-tiny-video-icon {
width: 18px;
height: 18px;
background-size: 12px 12px;
background-color: #4ba6e7;
display: inline-block;
background-repeat: no-repeat;
background-position: center;
border-radius: 50%;
}
.fx-embedded-tiny-video-icon.muted {
background-color: rgba(0,0,0,.2)
}
/* Force full height on all parents up to the video elements
* this way we can ensure the aspect ratio and use height 100%
* on the video element
* */
html, .fx-embedded, #main,
.video-layout-wrapper,
.conversation {
height: 100%;
}
@media screen and (min-width:640px) {
.standalone .conversation-toolbar {
position: absolute;
bottom: 0;
left: 0;
right: 0;
}
.fx-embedded .local-stream {
position: fixed;
}
.standalone .local-stream,
.standalone .remote-inset-stream {
position: absolute;
right: 15px;
bottom: 15px;
width: 20%;
height: 20%;
max-width: 400px;
max-height: 300px;
}
/* Nested video elements */
.conversation .media.nested {
position: relative;
height: 100%;
}
.standalone .remote_wrapper {
position: relative;
width: calc(100% - 10px);
height: 100%;
margin-left: 10px;
}
.standalone {
margin: 0 auto;
}
}
@media screen and (max-width:640px) {
.standalone .video-layout-wrapper,
.standalone .conversation {
height: 100%;
}
.standalone .media {
height: 90%;
}
.standalone .OT_subscriber {
height: 100%;
width: auto;
}
.standalone .media.nested {
min-height: 500px;
}
.standalone .remote-inset-stream {
display: none;
}
.standalone .local-stream {
flex: 1;
min-width: 120px;
min-height: 150px;
width: 100%;
box-shadow: none;
}
/* Nested video elements */
.conversation .media.nested {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
flex: 1 1 0%;
}
.standalone .video_wrapper.remote_wrapper {
/* Because of OT markup we need to set a high flex value
* Flex rule assures remote and local streams stack on top of eachother
* Computed width is not 100% unless the `width` rule */
flex: 2;
width: 100%;
position: relative;
}
}
@media screen and (max-width:420px) {
/* Restore video height so that we get
* vertical centering for free on a small screen
**/
.standalone .conversation .media video {
height: 100%;
}
}
/**
* Rooms
*/
.room-conversation-wrapper {
position: relative;
height: 100%;
}
.room-conversation-wrapper header {
background: #000;
height: 50px;
text-align: left;
width: 75%;
}
.room-conversation-wrapper header h1 {
font-size: 1.5em;
color: #fff;
line-height: 50px;
text-indent: 60px;
background-image: url("../img/firefox-logo.png");
background-size: 30px;
background-position: 20px;
background-repeat: no-repeat;
display: inline-block;
}
.room-conversation-wrapper header a {
float: right;
}
.room-conversation-wrapper header .icon-help {
display: inline-block;
background-size: contain;
margin-top: 20px;
width: 20px;
height: 20px;
background: transparent url("../img/svg/glyph-help-16x16.svg") no-repeat;
}
.room-conversation-wrapper footer {
background: #000;
height: 60px;
margin-top: -12px;
}
.room-conversation-wrapper footer a {
color: #555;
}
.fx-embedded .room-conversation .conversation-toolbar .btn-hangup {
background-image: url("../img/icons-16x16.svg#leave");
}
.room-invitation-overlay {
position: absolute;
background: rgba(0,0,0,.6);
/* This matches .fx-embedded .conversation toolbar height */
top: 26px;
right: 0;
bottom: 0;
left: 0;
text-align: center;
color: #fff;
z-index: 1010;
}
.room-invitation-overlay .error-display-area.error,
.room-invitation-overlay input[type="text"] {
display: block;
background-color: rgba(0,0,0,.5);
border-radius: 3px;
padding: .5em;
}
.room-invitation-overlay .error-display-area {
display: none;
}
.room-invitation-overlay .error-display-area.error {
position: absolute;
top: 2em;
left: 1em;
right: 1em;
text-align: start;
width: calc(258px - 2em);
color: #d74345;
}
.room-invitation-overlay form {
padding: 6em 0 2em 0;
}
.room-invitation-overlay textarea {
display: block;
background: rgba(0, 0, 0, .5);
color: #fff;
font-family: "Helvetica Neue", Arial, sans;
font-size: 1.2em;
border: none;
width: 200px;
margin: 0 auto;
padding: .2em .4em;
border-radius: .5em;
}
.room-invitation-overlay .btn-group {
position: absolute;
bottom: 10px;
}
/* Standalone rooms */
.standalone .room-conversation-wrapper {
position: relative;
height: 100%;
background: #000;
}
.standalone .room-conversation-wrapper .video-layout-wrapper {
height: calc(100% - 50px - 60px);
}
.standalone .room-conversation-wrapper .room-inner-info-area {
position: absolute;
/* `top` is chosen to vertically position the area near the center
of the media element. */
top: calc(50% - 140px);
left: calc(25% + 62.5px + 10px);
z-index: 1000;
/* `width` here is specified by the design spec. */
width: 250px;
color: #fff;
box-sizing: content-box;
}
.standalone .prompt-media-message {
padding-top: 136px; /* Fallback for browsers that don't support calc() */
/* 122px is 2x the intrinsic height of the background-image, and
1rem puts one line of margin between the background-image and
supporting text. */
padding-top: calc(122px + 1rem);
color: #000;
background-color: #fff;
background-image: url("../../img/gum-others.svg");
background-position: top center;
/* The background-image is scaled up at 2x the instrinsic size
(witdh & height) to make it easier to see. */
background-size: 202px 122px;
background-repeat: no-repeat;
border: 1rem #fff solid;
box-shadow: 0 0 5px #000;
margin: 0;
}
.standalone .prompt-media-message.chrome {
background-image: url("../../img/gum-chrome.svg");
}
.standalone .prompt-media-message.firefox {
background-image: url("../../img/gum-firefox.svg");
}
.standalone .prompt-media-message.opera {
background-image: url("../../img/gum-opera.svg");
}
.standalone .room-conversation-wrapper .room-inner-info-area button {
border-radius: 3px;
font-size: 1.2em;
padding: .2em 1.2em;
cursor: pointer;
}
.standalone .room-conversation-wrapper .room-inner-info-area a.btn {
padding: .5em 3em .3em 3em;
border-radius: 3px;
font-weight: normal;
max-width: 400px;
}
.standalone .room-conversation h2.room-name {
position: absolute;
display: inline-block;
top: 0;
right: 0;
color: #fff;
z-index: 2000000;
font-size: 1.2em;
padding: .4em;
}
.standalone .room-conversation .media {
background: #000;
}
.standalone .room-conversation .video_wrapper.remote_wrapper {
background-color: #4e4e4e;
width: calc(75% - 10px); /* Take the left margin into account. */
}
.standalone .room-conversation .conversation-toolbar {
background: #000;
border: none;
}
.standalone .room-conversation .conversation-toolbar .btn-hangup-entry {
display: block;
}
.standalone .room-conversation-wrapper .ended-conversation {
position: relative;
height: auto;
}
@media screen and (max-width:640px) {
/* Rooms specific responsive styling */
.standalone .room-conversation {
background: #000;
}
.room-conversation-wrapper header {
width: 100%;
}
.standalone .room-conversation-wrapper .room-inner-info-area {
right: 0;
margin: auto;
width: 100%;
}
.standalone .room-conversation-wrapper .video-layout-wrapper {
/* 50px: header's height; 25px: footer's height */
height: calc(100% - 50px - 25px);
}
.standalone .room-conversation .video_wrapper.remote_wrapper {
width: 100%;
}
.standalone .conversation-toolbar {
height: 38px;
padding: 8px;
}
.standalone .focus-stream {
/* Set at maximum height, minus height of conversation toolbar */
height: 100%;
}
.standalone .media.nested {
/* This forces the remote video stream to fit within wrapper's height */
min-height: 0px;
}
.standalone .room-conversation-wrapper footer {
font-size: 80%;
height: 25px;
}
.standalone .room-conversation-wrapper footer p {
padding-top: 4px;
}
.standalone .room-conversation-wrapper footer .footer-logo {
display: none;
}
}
.self-view-hidden-message {
/* Not displayed by default; display is turned on elsewhere when the
* self-view is actually hidden.
*/
display: none;
}
/* Avoid the privacy problem where a user can size the window so small that
* part of the self view is not shown. If the self view isn't completely
* displayable...
*/
@media screen and (max-height:160px) {
/* disable the self view */
.standalone .OT_publisher {
display: none;
}
/* and enable a message telling the user how to get it back */
.standalone .self-view-hidden-message {
display: inline;
position: relative;
top: 90px;
}
}