client/src/sass/include/_mixins.scss
@use 'sass:math';
@use 'sass:color';
@use '_variables' as *;
@import '_bootstrap-mixins';
@mixin disable-default-a-behaviour {
&:hover,
&:focus,
&:active {
text-decoration: none;
}
&:focus:not(:focus-visible) {
outline: none;
}
}
@mixin disable-outline {
&:focus:not(:focus-visible) {
outline: none !important;
}
}
@mixin ellipsis {
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
@mixin muted {
color: pvar(--greyForegroundColor) !important;
}
@mixin fade-text ($fade-after, $background-color) {
position: relative;
overflow: hidden;
&::after {
content: '';
pointer-events: none;
width: 100%;
height: 100%;
position: absolute;
left: 0;
top: 0;
background: linear-gradient(transparent $fade-after, $background-color);
}
}
@mixin peertube-word-wrap ($with-hyphen: true) {
word-break: break-word;
word-wrap: break-word;
overflow-wrap: break-word;
@if $with-hyphen {
hyphens: auto;
}
}
@mixin apply-svg-color ($color) {
::ng-deep .feather,
::ng-deep .material,
::ng-deep .misc {
color: $color;
}
}
@mixin fill-svg-color ($color) {
::ng-deep svg {
path {
fill: $color;
}
}
}
@mixin button-focus($color) {
&:focus,
&:focus-visible {
box-shadow: #{$focus-box-shadow-form} $color;
}
}
@mixin rounded-line-height-1-5 ($font-size) {
line-height: $font-size + math.round(math.div($font-size, 2));
}
@mixin peertube-input-text($width, $font-size: $form-input-font-size) {
font-size: $font-size;
padding: 3px 15px;
width: $width;
max-width: $width;
color: pvar(--inputForegroundColor);
background-color: pvar(--inputBackgroundColor);
border: 1px solid pvar(--inputBorderColor);
border-radius: 3px;
@include rounded-line-height-1-5($font-size);
&::placeholder {
color: pvar(--inputPlaceholderColor);
}
&[readonly] {
color: pvar(--greyForegroundColor);
}
@media screen and (max-width: calc(#{$width} + 40px)) {
width: 100%;
}
}
@mixin peertube-textarea ($width, $height) {
color: pvar(--textareaForegroundColor) !important;
background-color: pvar(--textareaBackgroundColor) !important;
height: $height;
padding: 5px 15px;
@include peertube-input-text($width);
}
@mixin orange-button {
@include button-focus(pvar(--mainColorLightest));
&,
&:active,
&:focus {
color: #fff;
background-color: pvar(--mainColor);
border-color: inherit;
}
// Override bootstrap
&.btn:active,
&.btn:focus-visible,
&.btn.show {
color: #fff !important;
background-color: pvar(--mainColor) !important;
border-color: inherit !important;
}
&:hover {
color: #fff;
background-color: pvar(--mainHoverColor);
}
&[disabled] {
cursor: default;
color: #fff;
background-color: pvar(--inputBorderColor);
}
my-global-icon {
@include apply-svg-color(#fff);
}
}
@mixin orange-button-inverted {
padding: 2px 13px;
border: 2px solid pvar(--mainColor);
font-weight: $font-semibold;
@include button-focus(pvar(--mainColorLightest));
&,
&:active,
&:focus {
color: pvar(--mainColor);
background-color: pvar(--mainBackgroundColor);
}
// Override bootstrap
&.btn:active,
&.btn:focus-visible,
&.btn.show {
color: pvar(--mainColor);
background-color: pvar(--mainBackgroundColor);
border-color: inherit !important;
}
&:hover {
color: pvar(--mainColor);
background-color: pvar(--mainColorLightest);
}
&[disabled] {
cursor: default;
color: pvar(--mainColor);
background-color: pvar(--inputBorderColor);
}
my-global-icon {
@include apply-svg-color(pvar(--mainColor));
}
}
@mixin tertiary-button {
color: pvar(--greyForegroundColor);
background-color: transparent;
@include button-focus($grey-button-outline-color);
&[disabled] {
cursor: default;
}
my-global-icon {
@include apply-svg-color(transparent);
}
}
@mixin grey-button {
color: pvar(--greyForegroundColor);
background-color: pvar(--greyBackgroundColor);
@include button-focus($grey-button-outline-color);
&:hover,
&:active,
&:focus,
&[disabled] {
color: pvar(--greyForegroundColor);
background-color: pvar(--greySecondaryBackgroundColor);
}
// Override bootstrap
&.btn:active,
&.btn:focus-visible,
&.btn.show {
color: pvar(--greyForegroundColor);
background-color: pvar(--greySecondaryBackgroundColor);
border-color: inherit !important;
}
&[disabled] {
cursor: default;
}
my-global-icon {
@include apply-svg-color(pvar(--greyForegroundColor));
}
}
@mixin danger-button {
background-color: pvar(--red);
color: pvar(--white);
@include button-focus(pvar(--red));
&:hover,
&:active,
&:focus:not(:focus-visible) {
opacity: 0.8;
}
&[disabled] {
cursor: default;
opacity: 0.7;
}
my-global-icon {
@include apply-svg-color(pvar(--white));
}
}
@mixin peertube-button {
padding: 4px 13px;
border: 0;
font-weight: $font-semibold;
border-radius: 3px;
text-align: center;
cursor: pointer;
font-size: $button-font-size;
@include rounded-line-height-1-5($button-font-size);
my-global-icon + * {
@include margin-right(4px);
@include margin-left(4px);
}
}
@mixin peertube-button-big {
height: auto;
padding: 10px 25px;
font-size: 18px;
line-height: 1.2;
border: 0;
font-weight: $font-semibold;
// Because of primeng that redefines border-radius of all input[type="..."]
border-radius: 3px !important;
}
@mixin peertube-button-link {
display: inline-block;
@include disable-default-a-behaviour;
@include peertube-button;
}
@mixin peertube-button-big-link {
display: inline-block;
@include disable-default-a-behaviour;
@include peertube-button-big;
}
@mixin peertube-button-outline {
display: inline-block;
border: 1px solid;
@include disable-default-a-behaviour;
@include peertube-button;
}
@mixin button-with-icon($width: 20px, $margin-right: 3px, $top: -1px) {
my-global-icon {
position: relative;
display: inline-block;
width: $width;
top: $top;
@include margin-right($margin-right);
}
}
@mixin peertube-file {
position: relative;
overflow: hidden;
display: inline-block;
input[type=file] {
position: absolute;
top: 0;
right: 0;
width: 100%;
height: 100%;
font-size: 100px;
text-align: end;
filter: alpha(opacity=0);
opacity: 0;
outline: none;
background: pvar(--mainBackgroundColor);
cursor: inherit;
display: block;
}
}
@mixin peertube-button-file ($width) {
width: $width;
@include peertube-file;
@include peertube-button;
}
@mixin icon ($size) {
display: inline-block;
background-repeat: no-repeat;
background-size: contain;
width: $size;
height: $size;
vertical-align: middle;
cursor: pointer;
}
@mixin responsive-width ($width) {
width: $width;
@media screen and (max-width: $width) {
width: 100%;
}
}
@mixin peertube-select-container ($width) {
padding: 0;
margin: 0;
width: $width;
border-radius: 3px;
color: pvar(--inputForegroundColor);
background: pvar(--inputBackgroundColor);
position: relative;
height: min-content;
select[disabled] {
background-color: #f9f9f9;
}
@media screen and (max-width: $width) {
width: 100%;
}
&::after {
top: 50%;
right: calc(0% + 15px);
content: ' ';
height: 0;
width: 0;
position: absolute;
pointer-events: none;
border: 5px solid rgba(0, 0, 0, 0);
border-top-color: pvar(--mainForegroundColor);
margin-top: -2px;
z-index: 100;
}
select {
font-size: $form-input-font-size;
padding: 3px 35px 3px 12px;
position: relative;
border: 1px solid pvar(--inputBorderColor);
background: transparent none;
appearance: none;
text-overflow: ellipsis;
color: pvar(--mainForegroundColor);
@include rounded-line-height-1-5($form-input-font-size);
option {
color: #000;
}
}
&.peertube-select-button {
@include grey-button;
select {
font-weight: $font-semibold;
color: pvar(--greyForegroundColor);
border: 0;
// No border, add +1 to vertical padding
padding: 4px 35px 4px 12px;
}
}
}
// Thanks: https://codepen.io/manabox/pen/raQmpL
@mixin peertube-radio-container {
label {
font-size: $form-input-font-size;
}
[type=radio]:focus-visible + label::before {
outline: 2px solid;
}
[type=radio]:checked,
[type=radio]:not(:checked) {
position: absolute;
opacity: 0;
cursor: pointer;
height: 0;
width: 0;
}
[type=radio]:checked + label,
[type=radio]:not(:checked) + label {
position: relative;
padding-left: 28px;
cursor: pointer;
line-height: 20px;
display: inline-block;
font-weight: $font-regular;
}
[type=radio]:checked + label::before,
[type=radio]:not(:checked) + label::before {
content: '';
position: absolute;
left: 0;
top: 0;
width: 18px;
height: 18px;
border: 1px solid pvar(--inputBorderColor);
border-radius: 100%;
background: #fff;
}
[type=radio]:checked + label::after,
[type=radio]:not(:checked) + label::after {
content: '';
width: 10px;
height: 10px;
background: pvar(--mainColor);
position: absolute;
top: 4px;
left: 4px;
border-radius: 100%;
transition: all 0.2s ease;
}
[type=radio]:not(:checked) + label::after {
opacity: 0;
transform: scale(0);
}
[type=radio]:checked + label::after {
opacity: 1;
transform: scale(1);
}
.form-group-description {
display: block;
margin-top: -7px;
margin-bottom: 10px;
margin-left: 29px;
}
}
@mixin peertube-checkbox ($border-width) {
opacity: 0;
position: absolute;
&:focus + span {
box-shadow: #{$focus-box-shadow-form} pvar(--mainColorLightest);
}
+ span {
position: relative;
width: 18px;
min-width: 18px;
height: 18px;
border: $border-width solid pvar(--inputBorderColor);
border-radius: 3px;
vertical-align: middle;
cursor: pointer;
&::after {
content: '';
position: absolute;
top: calc(2px - #{$border-width});
left: 5px;
width: 5px;
height: 12px;
opacity: 0;
transform: rotate(45deg) scale(0);
border-right: 2px solid pvar(--mainBackgroundColor);
border-bottom: 2px solid pvar(--mainBackgroundColor);
}
}
&:checked + span {
border-color: transparent;
background: pvar(--mainColor);
animation: jelly 0.6s ease;
&::after {
opacity: 1;
transform: rotate(45deg) scale(1);
}
}
+ span + span {
font-weight: $font-regular;
cursor: pointer;
display: inline;
@include margin-left(5px);
}
&[disabled] + span,
&[disabled] + span + span {
opacity: 0.5;
cursor: default;
}
}
@mixin actor-counters ($separator-margin: 10px) {
color: pvar(--greyForegroundColor);
display: flex;
align-items: center;
> *:not(:last-child)::after {
content: '•';
margin: 0 $separator-margin;
color: pvar(--mainColor);
}
}
@mixin in-content-small-title {
text-transform: uppercase;
color: pvar(--mainColor);
font-weight: $font-bold;
font-size: 13px;
}
@mixin row-blocks ($column-responsive: true, $min-height: 130px, $separator: true) {
display: flex;
min-height: $min-height;
padding-bottom: 20px;
margin-bottom: 20px;
@if $separator {
border-bottom: 1px solid pvar(--inputBorderColor);
}
@media screen and (max-width: $small-view) {
@if $column-responsive {
flex-direction: column;
height: auto;
align-items: center;
} @else {
min-height: initial;
padding-bottom: 10px;
margin-bottom: 10px;
}
}
}
@mixin dropdown-with-icon-item {
padding: 6px 15px;
my-global-icon {
width: 22px;
opacity: .7;
position: relative;
top: -2px;
@include margin-right(10px);
}
}
@mixin divider($color: pvar(--submenuBackgroundColor), $background: pvar(--mainBackgroundColor)) {
width: 95%;
border-top: .05rem solid $color;
height: .05rem;
text-align: center;
display: block;
position: relative;
&[data-content] {
margin: .8rem 0;
&::after {
background: $background;
color: $color;
content: attr(data-content);
display: inline-block;
font-size: .7rem;
padding: 0 .4rem;
transform: translateY(-.65rem);
}
}
}
// applies ratio (default to 16:9) to a child element (using $selector) only using
// an immediate's parent size. This allows to set a ratio without explicit
// dimensions, as width/height cannot be computed from each other.
@mixin block-ratio ($selector: 'div', $inverted-ratio: math.div(9, 16)) {
$padding-percent: math.percentage($inverted-ratio);
position: relative;
height: 0;
width: 100%;
padding-top: $padding-percent;
#{$selector} {
position: absolute;
width: 100%;
height: 100%;
top: 0;
@content;
}
}
@mixin sub-menu-h1 {
::ng-deep .sub-title-container {
display: flex;
align-items: center;
justify-content: space-between;
flex-wrap: wrap;
gap: 0.5rem;
border-bottom: 2px solid $grey-background-color;
padding-bottom: 15px;
margin-bottom: $sub-menu-margin-bottom;
> * {
gap: 0.5rem;
display: flex;
align-items: center;
}
.sub-title {
&,
h1 {
font-size: 1.3rem;
}
h1 {
margin: 0;
}
> my-global-icon {
width: 24px;
height: 24px;
svg {
vertical-align: initial;
}
}
my-feed {
display: inline-block;
width: 15px;
my-global-icon svg {
vertical-align: initial;
position: relative;
top: 1px;
}
}
}
}
}
@mixin play-icon ($width, $height) {
width: 0;
height: 0;
position: absolute;
left: 50%;
top: 50%;
transform: translate(-50%, -50%) scale(0.5);
border-top: #{math.div($height, 2)} solid transparent;
border-bottom: #{math.div($height, 2)} solid transparent;
border-left: $width solid rgba(255, 255, 255, 0.95);
}
@mixin on-small-main-col () {
:host-context(.main-col:not(.expanded)) {
@media screen and (max-width: #{$small-view + $menu-width}) {
@content;
}
}
:host-context(.main-col.expanded) {
@media screen and (max-width: $small-view) {
@content;
}
}
}
@mixin on-mobile-main-col () {
:host-context(.main-col:not(.expanded)) {
@media screen and (max-width: #{$mobile-view + $menu-width}) {
@content;
}
}
:host-context(.main-col.expanded) {
@media screen and (max-width: $mobile-view) {
@content;
}
}
}
// ---------------------------------------------------------------------------
@mixin margin ($arg1: null, $arg2: null, $arg3: null, $arg4: null) {
@if $arg2 == null and $arg3 == null and $arg4 == null {
@include margin-original($arg1, $arg1, $arg1, $arg1);
} @else if $arg3 == null and $arg4 == null {
@include margin-original($arg1, $arg2, $arg1, $arg2);
} @else if $arg4 == null {
@include margin-original($arg1, $arg2, $arg3, $arg2);
} @else {
@include margin-original($arg1, $arg2, $arg3, $arg4);
}
}
@mixin margin-original ($block-start, $inline-end, $block-end, $inline-start) {
@include margin-left($inline-start);
@include margin-right($inline-end);
@include margin-top($block-start);
@include margin-bottom($block-end);
}
@mixin margin-left ($value) {
@include rfs($value, margin-inline-start);
}
@mixin margin-right ($value) {
@include rfs($value, margin-inline-end);
}
// ---------------------------------------------------------------------------
@mixin padding-original ($block-start, $inline-end, $block-end, $inline-start) {
@include padding-left($inline-start);
@include padding-right($inline-end);
@include padding-top($block-start);
@include padding-bottom($block-end);
}
@mixin padding ($arg1: null, $arg2: null, $arg3: null, $arg4: null) {
@if $arg2 == null and $arg3 == null and $arg4 == null {
@include padding-original($arg1, $arg1, $arg1, $arg1);
} @else if $arg3 == null and $arg4 == null {
@include padding-original($arg1, $arg2, $arg1, $arg2);
} @else if $arg4 == null {
@include padding-original($arg1, $arg2, $arg3, $arg2);
} @else {
@include padding-original($arg1, $arg2, $arg3, $arg4);
}
}
@mixin padding-left ($value) {
@include rfs($value, padding-inline-start);
}
@mixin padding-right ($value) {
@include rfs($value, padding-inline-end);
}
// ---------------------------------------------------------------------------
/**
*
* inset-inline properties are not supported by iOS < 14.5
*
*/
@mixin right ($value) {
@supports (inset-inline-end: $value) {
inset-inline-end: $value;
}
@supports not (inset-inline-end: $value) {
right: $value;
}
}
@mixin left ($value) {
@supports (inset-inline-start: $value) {
inset-inline-start: $value;
}
@supports not (inset-inline-start: $value) {
left: $value;
}
}