design-angular-kit
Version:
Un toolkit Angular conforme alle linee guida di design per i servizi web della PA
301 lines (299 loc) • 31.9 kB
JavaScript
import { NgTemplateOutlet } from '@angular/common';
import { afterNextRender, ChangeDetectionStrategy, Component, DestroyRef, inject, Injector, Input, NgZone, signal, ViewChild, ViewEncapsulation, } from '@angular/core';
import { toObservable } from '@angular/core/rxjs-interop';
import { delay } from 'rxjs';
import videojs from 'video.js';
import { ItAbstractComponent } from '../../../abstracts/abstract.component';
import { VideoPlayerI18nService } from './video-player-i18n.service';
import { cookies } from './video-player.cookie';
import * as i0 from "@angular/core";
import * as i1 from "./video-player.config";
var ViewType;
(function (ViewType) {
ViewType["Default"] = "DEFAULT";
ViewType["Overlay"] = "OVERLAY";
})(ViewType || (ViewType = {}));
/**
* Video Player
* @description Component that allows playing a video.
*/
export class ItVideoPlayerComponent extends ItAbstractComponent {
#destroyRef;
constructor(config) {
super();
this.config = config;
this.player = null;
this.viewTypes = ViewType;
this.viewType = signal(undefined);
this.cookieAccepted = signal(false);
this.i18nService = inject(VideoPlayerI18nService);
this.#destroyRef = inject(DestroyRef);
this.ngZone = inject(NgZone);
this.injector = inject(Injector);
afterNextRender(() => {
if (this.viewType() === ViewType.Overlay && cookies.isChoiceRemembered('youtube.com')) {
this.hideOverlay();
}
});
this.#destroyRef.onDestroy(() => this.player?.dispose());
}
async ngOnInit() {
const config = this.config.mergeConfig(this.options);
this.setViewType(config);
// Avoid running change detections while the script is being loaded.
await this.ngZone.runOutsideAngular(() => this.config.configureTech(config));
if (!this.videoPlayerRef) {
// Note: No need to pipe with `takeUntilDestroyed`; `toObservable` is
// completed by Angular when the `DestroyRef` from the injector is destroyed.
toObservable(this.cookieAccepted, { injector: this.injector })
.pipe(delay(0))
.subscribe(value => {
if (value && !this.player) {
this.initVideoPlayer();
}
});
return;
}
this.initVideoPlayer();
}
acceptCookieHandler() {
this.rememberHandler();
this.hideOverlay();
this.cookieAccepted.set(true);
}
initVideoPlayer() {
const config = this.config.mergeConfig(this.options);
this.setVideoAttributes(config);
this.setVideoPlayer();
}
setVideoPlayer() {
const config = this.config.mergeConfig(this.options);
const onPlayerReadyCb = () => {
if (!this.player) {
return;
}
this.i18nService.init(this.player, this.#destroyRef);
};
const element = this.videoPlayerRef?.nativeElement;
if (!element) {
throw Error('videoPlayerRef is undefined');
}
this.player = this.ngZone.runOutsideAngular(() => videojs(element, config, onPlayerReadyCb));
}
setViewType(config) {
this.viewType.set(config.tech === 'youtube' ? ViewType.Overlay : ViewType.Default);
this.cookieAccepted.set(this.viewType() === ViewType.Overlay && cookies.isChoiceRemembered('youtube.com'));
}
hideOverlay() {
if (!this.acceptOverlayableRef) {
return;
}
const classes = ['show'];
this.acceptOverlayableRef.nativeElement.classList.remove(...classes);
if (!this.acceptOveralyRef) {
return;
}
this.acceptOveralyRef.nativeElement.classList.remove(...classes);
this.acceptOveralyRef.nativeElement.setAttribute('aria-hidden', 'true');
}
rememberHandler() {
if (!this.chrRememberRef) {
return;
}
const remember = this.chrRememberRef.nativeElement.checked;
cookies.rememberChoice('youtube.com', remember);
}
setVideoAttributes(options) {
if (!this.videoPlayerRef) {
return;
}
const v = this.videoPlayerRef.nativeElement;
const { autoplay, controls, loop, muted, poster, fluid } = options;
if (autoplay) {
v.setAttribute('autoplay', autoplay.toString());
}
if (controls) {
v.setAttribute('controls', '');
}
if (loop) {
v.setAttribute('loop', '');
}
if (muted) {
v.setAttribute('muted', '');
}
if (poster) {
v.setAttribute('poster', poster);
}
if (fluid) {
v.setAttribute('fluid', '');
}
v.setAttribute('preload', 'none');
v.setAttribute('playsinline', '');
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.6", ngImport: i0, type: ItVideoPlayerComponent, deps: [{ token: i1.VideoPlayerConfigService }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.6", type: ItVideoPlayerComponent, isStandalone: true, selector: "it-video-player", inputs: { options: "options" }, viewQueries: [{ propertyName: "videoPlayerRef", first: true, predicate: ["videoPlayer"], descendants: true }, { propertyName: "acceptOveralyRef", first: true, predicate: ["acceptOveraly"], descendants: true }, { propertyName: "acceptOverlayableRef", first: true, predicate: ["acceptOverlayable"], descendants: true }, { propertyName: "chrRememberRef", first: true, predicate: ["chkRemember"], descendants: true }], usesInheritance: true, ngImport: i0, template: ` (viewType()) {
(viewTypes.Default) {
<div class="row">
<ng-container *ngTemplateOutlet="videoTemplate"></ng-container>
<ng-container *ngTemplateOutlet="transcriptionTemplate"></ng-container>
</div>
}
(viewTypes.Overlay) {
<div #acceptOverlayable class="acceptoverlayable show">
<div #acceptOveraly class="acceptoverlay acceptoverlay-primary fade show">
<div class="acceptoverlay-inner">
<div class="acceptoverlay-icon">
<svg class="icon icon-xl"><use href="/bootstrap-italia/dist/svg/sprites.svg#it-video"></use></svg>
</div>
<p>
Accetta i cookie di YouTube per vedere il video. Puoi gestire le preferenze nella
<a href="#" class="text-white">cookie policy</a>.
</p>
<div class="acceptoverlay-buttons bg-dark">
<button type="button" class="btn btn-primary" (click)="acceptCookieHandler()">Accetta</button>
<div class="form-check">
<input id="chk-remember{{ id }}" type="checkbox" #chkRemember />
<label for="chk-remember{{ id }}">Ricorda per tutti i video</label>
</div>
</div>
</div>
</div>
(cookieAccepted()) {
<div>
<ng-container *ngTemplateOutlet="videoTemplate"></ng-container>
<ng-container *ngTemplateOutlet="transcriptionTemplate"></ng-container>
</div>
}
</div>
}
{
<h1>No video provider</h1>
}
}
<ng-template #videoTemplate>
<div>
<video #videoPlayer class="video-js vjs-theme-bootstrap-italia vjs-fluid vjs-big-play-centered"></video>
</div>
</ng-template>
<ng-template #transcriptionTemplate>
<div class="vjs-transcription accordion">
<div class="accordion-item">
<h2 class="accordion-header " id="transcription-{{ id }}-head">
<button
class="accordion-button collapsed"
type="button"
data-bs-toggle="collapse"
[attr.data-bs-target]="'#transcription-' + id"
[attr.aria-controls]="'transcription-' + id"
aria-expanded="true">
<ng-content select="[transcriptionTitle]">Trascrizione</ng-content>
</button>
</h2>
<div
id="transcription-{{ id }}"
class="accordion-collapse collapse"
role="region"
[attr.aria-labelledby]="'transcription-' + id + '-head'">
<div class="accordion-body">
<ng-content select="[transcriptionText]">-</ng-content>
</div>
</div>
</div>
</div>
</ng-template> `, isInline: true, dependencies: [{ kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.6", ngImport: i0, type: ItVideoPlayerComponent, decorators: [{
type: Component,
args: [{
standalone: true,
selector: 'it-video-player',
template: ` (viewType()) {
(viewTypes.Default) {
<div class="row">
<ng-container *ngTemplateOutlet="videoTemplate"></ng-container>
<ng-container *ngTemplateOutlet="transcriptionTemplate"></ng-container>
</div>
}
(viewTypes.Overlay) {
<div #acceptOverlayable class="acceptoverlayable show">
<div #acceptOveraly class="acceptoverlay acceptoverlay-primary fade show">
<div class="acceptoverlay-inner">
<div class="acceptoverlay-icon">
<svg class="icon icon-xl"><use href="/bootstrap-italia/dist/svg/sprites.svg#it-video"></use></svg>
</div>
<p>
Accetta i cookie di YouTube per vedere il video. Puoi gestire le preferenze nella
<a href="#" class="text-white">cookie policy</a>.
</p>
<div class="acceptoverlay-buttons bg-dark">
<button type="button" class="btn btn-primary" (click)="acceptCookieHandler()">Accetta</button>
<div class="form-check">
<input id="chk-remember{{ id }}" type="checkbox" #chkRemember />
<label for="chk-remember{{ id }}">Ricorda per tutti i video</label>
</div>
</div>
</div>
</div>
(cookieAccepted()) {
<div>
<ng-container *ngTemplateOutlet="videoTemplate"></ng-container>
<ng-container *ngTemplateOutlet="transcriptionTemplate"></ng-container>
</div>
}
</div>
}
{
<h1>No video provider</h1>
}
}
<ng-template #videoTemplate>
<div>
<video #videoPlayer class="video-js vjs-theme-bootstrap-italia vjs-fluid vjs-big-play-centered"></video>
</div>
</ng-template>
<ng-template #transcriptionTemplate>
<div class="vjs-transcription accordion">
<div class="accordion-item">
<h2 class="accordion-header " id="transcription-{{ id }}-head">
<button
class="accordion-button collapsed"
type="button"
data-bs-toggle="collapse"
[attr.data-bs-target]="'#transcription-' + id"
[attr.aria-controls]="'transcription-' + id"
aria-expanded="true">
<ng-content select="[transcriptionTitle]">Trascrizione</ng-content>
</button>
</h2>
<div
id="transcription-{{ id }}"
class="accordion-collapse collapse"
role="region"
[attr.aria-labelledby]="'transcription-' + id + '-head'">
<div class="accordion-body">
<ng-content select="[transcriptionText]">-</ng-content>
</div>
</div>
</div>
</div>
</ng-template> `,
imports: [NgTemplateOutlet],
encapsulation: ViewEncapsulation.None,
changeDetection: ChangeDetectionStrategy.OnPush,
}]
}], ctorParameters: () => [{ type: i1.VideoPlayerConfigService }], propDecorators: { options: [{
type: Input
}], videoPlayerRef: [{
type: ViewChild,
args: ['videoPlayer', { static: false }]
}], acceptOveralyRef: [{
type: ViewChild,
args: ['acceptOveraly', { static: false }]
}], acceptOverlayableRef: [{
type: ViewChild,
args: ['acceptOverlayable', { static: false }]
}], chrRememberRef: [{
type: ViewChild,
args: ['chkRemember', { static: false }]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"video-player.component.js","sourceRoot":"","sources":["../../../../../../../projects/design-angular-kit/src/lib/components/core/video-player/video-player.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { NgTemplateOutlet } from '@angular/common';\nimport {\n  afterNextRender,\n  ChangeDetectionStrategy,\n  Component,\n  DestroyRef,\n  ElementRef,\n  inject,\n  Injector,\n  Input,\n  NgZone,\n  OnInit,\n  signal,\n  ViewChild,\n  ViewEncapsulation,\n} from '@angular/core';\nimport { toObservable } from '@angular/core/rxjs-interop';\nimport { delay } from 'rxjs';\nimport videojs from 'video.js';\nimport type Player from 'video.js/dist/types/player';\n\nimport { ItAbstractComponent } from '../../../abstracts/abstract.component';\nimport { VideoPlayerI18nService } from './video-player-i18n.service';\nimport { Tech, VideoPlayerConfigService } from './video-player.config';\nimport { cookies } from './video-player.cookie';\nimport { ItVideoPlayerConfig, ItVideoPlayerOptions } from './video-player.model';\n\nenum ViewType {\n  Default = 'DEFAULT',\n  Overlay = 'OVERLAY',\n}\n\n/**\n * Video Player\n * @description Component that allows playing a video.\n */\n@Component({\n  standalone: true,\n  selector: 'it-video-player',\n  template: `@switch (viewType()) {\n      @case (viewTypes.Default) {\n        <div class=\"row\">\n          <ng-container *ngTemplateOutlet=\"videoTemplate\"></ng-container>\n          <ng-container *ngTemplateOutlet=\"transcriptionTemplate\"></ng-container>\n        </div>\n      }\n      @case (viewTypes.Overlay) {\n        <div #acceptOverlayable class=\"acceptoverlayable show\">\n          <div #acceptOveraly class=\"acceptoverlay acceptoverlay-primary fade show\">\n            <div class=\"acceptoverlay-inner\">\n              <div class=\"acceptoverlay-icon\">\n                <svg class=\"icon icon-xl\"><use href=\"/bootstrap-italia/dist/svg/sprites.svg#it-video\"></use></svg>\n              </div>\n              <p>\n                Accetta i cookie di YouTube per vedere il video. Puoi gestire le preferenze nella\n                <a href=\"#\" class=\"text-white\">cookie policy</a>.\n              </p>\n              <div class=\"acceptoverlay-buttons bg-dark\">\n                <button type=\"button\" class=\"btn btn-primary\" (click)=\"acceptCookieHandler()\">Accetta</button>\n                <div class=\"form-check\">\n                  <input id=\"chk-remember{{ id }}\" type=\"checkbox\" #chkRemember />\n                  <label for=\"chk-remember{{ id }}\">Ricorda per tutti i video</label>\n                </div>\n              </div>\n            </div>\n          </div>\n          @if (cookieAccepted()) {\n            <div>\n              <ng-container *ngTemplateOutlet=\"videoTemplate\"></ng-container>\n              <ng-container *ngTemplateOutlet=\"transcriptionTemplate\"></ng-container>\n            </div>\n          }\n        </div>\n      }\n      @default {\n        <h1>No video provider</h1>\n      }\n    }\n    <ng-template #videoTemplate>\n      <div>\n        <video #videoPlayer class=\"video-js vjs-theme-bootstrap-italia vjs-fluid vjs-big-play-centered\"></video>\n      </div>\n    </ng-template>\n\n    <ng-template #transcriptionTemplate>\n      <div class=\"vjs-transcription accordion\">\n        <div class=\"accordion-item\">\n          <h2 class=\"accordion-header \" id=\"transcription-{{ id }}-head\">\n            <button\n              class=\"accordion-button collapsed\"\n              type=\"button\"\n              data-bs-toggle=\"collapse\"\n              [attr.data-bs-target]=\"'#transcription-' + id\"\n              [attr.aria-controls]=\"'transcription-' + id\"\n              aria-expanded=\"true\">\n              <ng-content select=\"[transcriptionTitle]\">Trascrizione</ng-content>\n            </button>\n          </h2>\n          <div\n            id=\"transcription-{{ id }}\"\n            class=\"accordion-collapse collapse\"\n            role=\"region\"\n            [attr.aria-labelledby]=\"'transcription-' + id + '-head'\">\n            <div class=\"accordion-body\">\n              <ng-content select=\"[transcriptionText]\">-</ng-content>\n            </div>\n          </div>\n        </div>\n      </div>\n    </ng-template> `,\n  imports: [NgTemplateOutlet],\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class ItVideoPlayerComponent extends ItAbstractComponent implements OnInit {\n  /**\n   * Options for video player configuration\n   */\n  @Input() options!: ItVideoPlayerOptions;\n\n  @ViewChild('videoPlayer', { static: false }) videoPlayerRef?: ElementRef<HTMLVideoElement>;\n\n  @ViewChild('acceptOveraly', { static: false }) acceptOveralyRef?: ElementRef<HTMLDivElement>;\n\n  @ViewChild('acceptOverlayable', { static: false }) acceptOverlayableRef?: ElementRef<HTMLDivElement>;\n\n  @ViewChild('chkRemember', { static: false }) chrRememberRef?: ElementRef<HTMLInputElement>;\n\n  player: Player | null = null;\n\n  readonly viewTypes = ViewType;\n\n  readonly viewType = signal<ViewType | undefined>(undefined);\n\n  readonly cookieAccepted = signal(false);\n\n  protected readonly i18nService = inject(VideoPlayerI18nService);\n\n  readonly #destroyRef = inject(DestroyRef);\n\n  private ngZone = inject(NgZone);\n  private injector = inject(Injector);\n\n  constructor(private config: VideoPlayerConfigService) {\n    super();\n\n    afterNextRender(() => {\n      if (this.viewType() === ViewType.Overlay && cookies.isChoiceRemembered('youtube.com')) {\n        this.hideOverlay();\n      }\n    });\n\n    this.#destroyRef.onDestroy(() => this.player?.dispose());\n  }\n\n  async ngOnInit() {\n    const config = this.config.mergeConfig(this.options);\n    this.setViewType(config);\n    // Avoid running change detections while the script is being loaded.\n    await this.ngZone.runOutsideAngular(() => this.config.configureTech(config as { tech: Tech }));\n\n    if (!this.videoPlayerRef) {\n      // Note: No need to pipe with `takeUntilDestroyed`; `toObservable` is\n      // completed by Angular when the `DestroyRef` from the injector is destroyed.\n      toObservable(this.cookieAccepted, { injector: this.injector })\n        .pipe(delay(0))\n        .subscribe(value => {\n          if (value && !this.player) {\n            this.initVideoPlayer();\n          }\n        });\n\n      return;\n    }\n\n    this.initVideoPlayer();\n  }\n\n  acceptCookieHandler() {\n    this.rememberHandler();\n    this.hideOverlay();\n    this.cookieAccepted.set(true);\n  }\n\n  private initVideoPlayer() {\n    const config = this.config.mergeConfig(this.options);\n    this.setVideoAttributes(config);\n    this.setVideoPlayer();\n  }\n\n  private setVideoPlayer() {\n    const config = this.config.mergeConfig(this.options);\n    const onPlayerReadyCb = () => {\n      if (!this.player) {\n        return;\n      }\n      this.i18nService.init(this.player, this.#destroyRef);\n    };\n\n    const element = this.videoPlayerRef?.nativeElement;\n    if (!element) {\n      throw Error('videoPlayerRef is undefined');\n    }\n\n    this.player = this.ngZone.runOutsideAngular(() => videojs(element, config, onPlayerReadyCb));\n  }\n\n  private setViewType(config: ItVideoPlayerConfig) {\n    this.viewType.set(config.tech === 'youtube' ? ViewType.Overlay : ViewType.Default);\n\n    this.cookieAccepted.set(this.viewType() === ViewType.Overlay && cookies.isChoiceRemembered('youtube.com'));\n  }\n\n  private hideOverlay() {\n    if (!this.acceptOverlayableRef) {\n      return;\n    }\n    const classes = ['show'];\n    this.acceptOverlayableRef.nativeElement.classList.remove(...classes);\n    if (!this.acceptOveralyRef) {\n      return;\n    }\n    this.acceptOveralyRef.nativeElement.classList.remove(...classes);\n    this.acceptOveralyRef.nativeElement.setAttribute('aria-hidden', 'true');\n  }\n\n  private rememberHandler() {\n    if (!this.chrRememberRef) {\n      return;\n    }\n    const remember = this.chrRememberRef.nativeElement.checked;\n    cookies.rememberChoice('youtube.com', remember);\n  }\n\n  private setVideoAttributes(options: ItVideoPlayerOptions) {\n    if (!this.videoPlayerRef) {\n      return;\n    }\n    const v = this.videoPlayerRef.nativeElement;\n\n    const { autoplay, controls, loop, muted, poster, fluid } = options;\n\n    if (autoplay) {\n      v.setAttribute('autoplay', autoplay.toString());\n    }\n\n    if (controls) {\n      v.setAttribute('controls', '');\n    }\n\n    if (loop) {\n      v.setAttribute('loop', '');\n    }\n\n    if (muted) {\n      v.setAttribute('muted', '');\n    }\n\n    if (poster) {\n      v.setAttribute('poster', poster);\n    }\n\n    if (fluid) {\n      v.setAttribute('fluid', '');\n    }\n\n    v.setAttribute('preload', 'none');\n    v.setAttribute('playsinline', '');\n  }\n}\n"]}