Add own font for chyron

Alters Current Speakery Chyron, optionally shows the structure level in
a new line (slightly smaler, faint)
Font for user name in chyron can be configured
Support way more font mime types
Adds an own font for chyron (server, client)
Extends loads-font service
Cleanup countdown-time component from font loading
This commit is contained in:
Sean 2021-05-11 17:09:51 +02:00 committed by Emanuel Schütze
parent 83efb19562
commit 74b32af293
9 changed files with 120 additions and 51 deletions

View File

@ -1,6 +1,7 @@
import { Injectable } from '@angular/core'; import { Injectable } from '@angular/core';
import { ConfigService } from './config.service'; import { ConfigService } from './config.service';
import { FontConfigObject } from './media-manage.service';
/** /**
* Enables the usage of the FontFace constructor * Enables the usage of the FontFace constructor
@ -10,7 +11,7 @@ declare let FontFace: any;
/** /**
* The linter refuses to allow Document['fonts']. * The linter refuses to allow Document['fonts'].
* Since Document.fonts is working draft since 2016, typescript * Since Document.fonts is working draft since 2016, typescript
* dies not yet support it natively (even though it exists in normal browsers) * does not yet support it natively (even though it exists in normal browsers)
*/ */
interface FontDocument extends Document { interface FontDocument extends Document {
fonts: any; fonts: any;
@ -40,23 +41,33 @@ export class LoadFontService {
} }
/** /**
* Observes and loads custom fonts for the projector. * Observes and loads custom fonts.
* Currently, normal and regular fonts can be considered, since
* italic fonts can easily be calculated by the browser.
* Falls back to the normal OSFont when no custom font was set. * Falls back to the normal OSFont when no custom font was set.
*/ */
private loadCustomFont(): void { private loadCustomFont(): void {
this.configService.get<any>('font_regular').subscribe(regular => { this.configService.get<FontConfigObject>('font_regular').subscribe(regular => {
if (regular) { if (regular) {
this.setCustomProjectorFont(regular, 400); this.setCustomProjectorFont(regular, 400);
} }
}); });
this.configService.get<any>('font_bold').subscribe(bold => { this.configService.get<FontConfigObject>('font_bold').subscribe(bold => {
if (bold) { if (bold) {
this.setCustomProjectorFont(bold, 500); this.setCustomProjectorFont(bold, 500);
} }
}); });
this.configService.get<FontConfigObject>('font_monospace').subscribe(mono => {
if (mono) {
this.setNewFontFace('OSFont Monospace', mono.path || mono.default);
}
});
this.configService.get<FontConfigObject>('font_chyron_speaker_name').subscribe(chyronFont => {
if (chyronFont) {
this.setNewFontFace('OSFont ChyronName', chyronFont.path || chyronFont.default);
}
});
} }
/** /**
@ -66,20 +77,24 @@ export class LoadFontService {
* @param font the font object from the config service * @param font the font object from the config service
* @param weight the desired weight of the font * @param weight the desired weight of the font
*/ */
private setCustomProjectorFont(font: any, weight: number): void { private setCustomProjectorFont(font: FontConfigObject, weight: number): void {
const path = font.path ? font.path : font.default; const path = font.path || font.default;
if (!path) { if (!path) {
return; return;
} }
const url = font.path ? `${this.urlPrefix}${path}` : path; const url: string = font.path ? `${this.urlPrefix}${path}` : path;
const fontFace = new FontFace('customProjectorFont', `url(${url})`, { weight: weight }); this.setNewFontFace('customProjectorFont', url, weight);
fontFace }
private setNewFontFace(fontName: string, fontPath: string, weight: number = 400): void {
const customFont = new FontFace(fontName, `url(${fontPath})`, { weight: weight });
customFont
.load() .load()
.then(res => { .then(res => {
(document as FontDocument).fonts.add(res); (document as FontDocument).fonts.add(res);
}) })
.catch(error => { .catch(error => {
console.error(error); console.log(error);
}); });
} }
} }

View File

@ -1,10 +1,6 @@
import { Component, Input, OnDestroy, OnInit } from '@angular/core'; import { Component, Input, OnDestroy } from '@angular/core';
import { ServertimeService } from 'app/core/core-services/servertime.service'; import { ServertimeService } from 'app/core/core-services/servertime.service';
import { ConfigService } from 'app/core/ui-services/config.service';
import { FontConfigObject } from 'app/core/ui-services/media-manage.service';
declare let FontFace: any;
export interface CountdownData { export interface CountdownData {
running: boolean; running: boolean;
@ -19,7 +15,7 @@ export interface CountdownData {
templateUrl: './countdown-time.component.html', templateUrl: './countdown-time.component.html',
styleUrls: ['./countdown-time.component.scss'] styleUrls: ['./countdown-time.component.scss']
}) })
export class CountdownTimeComponent implements OnInit, OnDestroy { export class CountdownTimeComponent implements OnDestroy {
/** /**
* The time in seconds to make the countdown orange, is the countdown is below this value. * The time in seconds to make the countdown orange, is the countdown is below this value.
*/ */
@ -96,23 +92,7 @@ export class CountdownTimeComponent implements OnInit, OnDestroy {
return this._countdown; return this._countdown;
} }
public constructor(private servertimeService: ServertimeService, private configService: ConfigService) {} public constructor(private servertimeService: ServertimeService) {}
public ngOnInit(): void {
this.configService.get<FontConfigObject>('font_monospace').subscribe(font => {
if (font) {
const customFont = new FontFace('OSFont Monospace', `url(${font.path || font.default})`);
customFont
.load()
.then(res => {
(document as any).fonts.add(res);
})
.catch(error => {
console.log(error);
});
}
});
}
/** /**
* Updates the countdown time and string format it. * Updates the countdown time and string format it.

View File

@ -6,7 +6,33 @@ import { Searchable } from 'app/site/base/searchable';
import { ViewGroup } from 'app/site/users/models/view-group'; import { ViewGroup } from 'app/site/users/models/view-group';
export const IMAGE_MIMETYPES = ['image/png', 'image/jpeg', 'image/gif']; export const IMAGE_MIMETYPES = ['image/png', 'image/jpeg', 'image/gif'];
export const FONT_MIMETYPES = ['font/ttf', 'font/woff', 'application/font-woff', 'application/font-sfnt'];
export const FONT_MIMETYPES = [
/**
* Standard fonts by iana May 2021. See:
* https://www.iana.org/assignments/media-types/media-types.xhtml#font
*/
'font/ttf',
'font/sfnt',
'font/otf',
'font/woff',
'font/woff2',
/**
* Non standard types
*/
/** (IANA: March 2013) (special non standard OTF fonts) */
'font/opentype',
'application/x-font-opentype',
'application/vnd.oasis.opendocument.formula-template',
/** (IANA: January 2013) */
'application/font-woff',
/** (W3C W./E.Draft: May 2014/March 2016) */
'application/font-woff2',
/** (IANA: March 2013) */
'application/font-sfnt',
'application/x-font-ttf',
'application/x-font-truetype'
];
export const PDF_MIMETYPES = ['application/pdf']; export const PDF_MIMETYPES = ['application/pdf'];
export const VIDEO_MIMETYPES = [ export const VIDEO_MIMETYPES = [
'video/quicktime', 'video/quicktime',

View File

@ -1,5 +1,6 @@
export interface CurrentSpeakerChyronSlideData { export interface CurrentSpeakerChyronSlideData {
current_speaker?: string; current_speaker_name?: string;
current_speaker_level?: string;
background_color: string; background_color: string;
font_color: string; font_color: string;
} }

View File

@ -1,8 +1,13 @@
<div id="chyron" *ngIf="data" [ngStyle]="{ <div
'background-color': data.data.background_color, id="chyron"
color: data.data.font_color}"> *ngIf="data"
[ngStyle]="{
'background-color': data.data.background_color,
color: data.data.font_color
}"
>
<span id="inner"> <span id="inner">
<b>{{ data.data.current_speaker }}</b> <div id="inner-name">{{ data.data.current_speaker_name }}</div>
<div id="inner-level">{{ data.data.current_speaker_level }}</div>
</span> </span>
</div> </div>

View File

@ -1,19 +1,33 @@
@import '~assets/styles/fonts.scss';
#chyron { #chyron {
position: absolute; position: absolute;
left: 0; left: 50px;
bottom: 0; bottom: 20px;
right: 0;
z-index: 10; z-index: 10;
width: 100%; height: 80px;
height: 100px;
font-size: 32px; font-size: 32px;
text-align: center; text-align: left;
line-height: 1.1; line-height: 1;
display: table; opacity: 0.8;
#inner { #inner {
display: table-cell;
vertical-align: middle; vertical-align: middle;
padding-left: 20px; padding-left: 18px;
padding-right: 20px; padding-right: 20px;
padding-top: 10px;
padding-bottom: 10px;
height: 60px;
display: table-cell;
#inner-name {
font-family: $font-chyronname;
}
#inner-level {
margin-top: 5px;
font-size: 70%;
}
} }
} }

View File

@ -25,3 +25,8 @@ $font-weight-condensed-regular: 400;
$font-monospace: 'OSFont Monospace'; $font-monospace: 'OSFont Monospace';
$font-monospace-src: url('../fonts/roboto-condensed-bold.woff') format('woff'); $font-monospace-src: url('../fonts/roboto-condensed-bold.woff') format('woff');
$font-weight-monospace: 400; $font-weight-monospace: 400;
/** Special Chyron Name Font */
$font-chyronname: 'OSFont ChyronName';
$font-chyronname-src: url('../fonts/fira-sans-latin-400.woff') format('woff');
$font-weight-chyronname: 400;

View File

@ -53,3 +53,12 @@
font-weight: $font-weight-monospace; font-weight: $font-weight-monospace;
src: $font-monospace-src; src: $font-monospace-src;
} }
/** Chyron Name */
@font-face {
font-family: $font-chyronname;
font-style: normal;
font-display: swap;
font-weight: $font-weight-chyronname;
src: $font-chyronname-src;
}

View File

@ -478,6 +478,7 @@ def get_config_variables():
"font_bold", "font_bold",
"font_bold_italic", "font_bold_italic",
"font_monospace", "font_monospace",
"font_chyron_speaker_name",
], ],
weight=320, weight=320,
group="Font", group="Font",
@ -549,6 +550,19 @@ def get_config_variables():
hidden=True, hidden=True,
) )
yield ConfigVariable(
name="font_chyron_speaker_name",
default_value={
"display_name": "Font for speaker name (chyron)",
"default": "assets/fonts/fira-sans-latin-400.woff",
"path": "",
},
input_type="static",
weight=321,
group="Font",
hidden=True,
)
# Custom translations # Custom translations
yield ConfigVariable( yield ConfigVariable(
name="translations", name="translations",