Compare commits

..

No commits in common. "main" and "v1.10.0" have entirely different histories.

142 changed files with 34143 additions and 38128 deletions

View File

@ -1,88 +1,102 @@
name: 'Build main restreamer-ui'
name: 'Build restreamer-ui'
on:
workflow_dispatch:
workflow_dispatch:
jobs:
build-frontend:
runs-on: ubuntu-latest
outputs:
version: ${{ steps.latestversion.outputs.version }}
steps:
- name: Checkout
uses: actions/checkout@v4
build-frontend:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v2
- name: Get latest version from package.json
id: latestversion
run: |
echo "version=$(cat ./package.json | jq '.version' | sed 's/\"//g')" >> "$GITHUB_OUTPUT"
- name: Set up Node.js
uses: actions/setup-node@v2
with:
node-version: '21'
- name: Set up Node.js
uses: actions/setup-node@v4
with:
node-version: '21'
- name: Build React App
run: |
yarn install
yarn build
- name: Build React App
run: |
yarn install
yarn build
env:
PUBLIC_URL: './'
- name: Upload React build as artifact
uses: actions/upload-artifact@v2
with:
name: react-build
path: build/
- name: Upload React build as artifact
uses: actions/upload-artifact@v4
with:
name: restreamerui-main-build
path: build/
build-docker:
needs: build-frontend
runs-on: [self-hosted]
steps:
- name: Checkout
uses: actions/checkout@v2
- name: Load ENV from file
run: |
cat .github_build/Build.restreamer-ui.env >> $GITHUB_ENV
build-docker:
needs: build-frontend
runs-on: [self-hosted]
steps:
- name: Checkout
uses: actions/checkout@v4
- name: Download React build artifact
uses: actions/download-artifact@v2
with:
name: react-build
path: build
- name: Download React build artifact
uses: actions/download-artifact@v4
with:
name: restreamerui-main-build
path: build
- uses: actions-ecosystem/action-get-latest-tag@v1
id: get-latest-tag
with:
semver_only: true
- name: Docker meta
id: meta
uses: docker/metadata-action@v5
with:
images: |
datarhei/restreamer-ui
tags: |
type=raw,value=latest
type=raw,value=${{ needs.build-frontend.outputs.version }}
- name: Docker meta
id: meta
uses: docker/metadata-action@v4
env:
RELEASE: ${{ env.RELEASE }}
CADDY_IMAGE: ${{ env.CADDY_IMAGE }}
with:
images: |
datarhei/restreamer-ui
tags: |
type=raw,value=latest
type=raw,value=${{ env.RELEASE }}
- name: Set up QEMU
uses: docker/setup-qemu-action@master
with:
platforms: all
- name: Set up QEMU
uses: docker/setup-qemu-action@master
with:
platforms: all
- name: Set up Docker Buildx
id: buildx
uses: docker/setup-buildx-action@master
- name: Set up Docker Buildx
id: buildx
uses: docker/setup-buildx-action@master
- name: Cache Docker layers
uses: actions/cache@v2
with:
path: /tmp/.buildx-cache
key: ${{ runner.os }}-buildx-${{ github.sha }}
restore-keys: |
${{ runner.os }}-buildx-
- name: Login to DockerHub
if: github.event_name != 'pull_request'
uses: docker/login-action@v3
with:
username: ${{ secrets.DOCKER_USERNAME }}
password: ${{ secrets.DOCKER_PASSWORD }}
- name: Login to DockerHub
if: github.event_name != 'pull_request'
uses: docker/login-action@v1
with:
username: ${{ secrets.DOCKER_USERNAME }}
password: ${{ secrets.DOCKER_PASSWORD }}
- name: Build and Push Docker Image
uses: docker/build-push-action@v5
with:
builder: ${{ steps.buildx.outputs.name }}
context: .
file: ./Dockerfile.workflow
build-args: |
CADDY_IMAGE=caddy:2.7.6-alpine
platforms: linux/amd64,linux/arm64,linux/arm/v7
push: true
tags: ${{ steps.meta.outputs.tags }}
labels: ${{ steps.meta.outputs.labels }}
- name: Build and Push Docker Image
uses: docker/build-push-action@v2
with:
builder: ${{ steps.buildx.outputs.name }}
context: .
file: ./Dockerfile.workflow
build-args: |
PUBLIC_URL=/ui
CADDY_IMAGE=${{ env.CADDY_IMAGE }}
platforms: linux/amd64,linux/arm64,linux/arm/v7
push: true
tags: ${{ steps.meta.outputs.tags }}
labels: ${{ steps.meta.outputs.labels }}
cache-from: type=local,src=/tmp/.buildx-cache
cache-to: type=local,dest=/tmp/.buildx-cache-new

View File

@ -1,88 +0,0 @@
name: 'Build dev restreamer-ui'
on:
workflow_dispatch:
workflow_call:
schedule:
- cron: '37 4 * * *'
push:
branches:
- dev
jobs:
build-frontend:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v4
with:
ref: dev
- name: Set up Node.js
uses: actions/setup-node@v4
with:
node-version: '21'
- name: Build React App
run: |
yarn install
yarn build
env:
PUBLIC_URL: './'
- name: Upload React build as artifact
uses: actions/upload-artifact@v4
with:
name: restreamerui-dev-build
path: build/
build-docker:
needs: build-frontend
runs-on: [self-hosted]
steps:
- name: Checkout
uses: actions/checkout@v4
- name: Download React build artifact
uses: actions/download-artifact@v4
with:
name: restreamerui-dev-build
path: build
- name: Docker meta
id: meta
uses: docker/metadata-action@v5
with:
images: |
datarhei/restreamer-ui
tags: |
type=raw,value=dev
- name: Set up QEMU
uses: docker/setup-qemu-action@master
with:
platforms: all
- name: Set up Docker Buildx
id: buildx
uses: docker/setup-buildx-action@master
- name: Login to DockerHub
if: github.event_name != 'pull_request'
uses: docker/login-action@v3
with:
username: ${{ secrets.DOCKER_USERNAME }}
password: ${{ secrets.DOCKER_PASSWORD }}
- name: Build and Push Docker Image
uses: docker/build-push-action@v5
with:
builder: ${{ steps.buildx.outputs.name }}
context: .
file: ./Dockerfile.workflow
build-args: |
CADDY_IMAGE=caddy:2.7.6-alpine
platforms: linux/amd64,linux/arm64,linux/arm/v7
push: true
tags: ${{ steps.meta.outputs.tags }}
labels: ${{ steps.meta.outputs.labels }}

View File

@ -0,0 +1,102 @@
name: 'Build datarhei/restreamer-ui:dev'
on:
push:
branches:
- dev
jobs:
build-frontend:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v2
- name: Set up Node.js
uses: actions/setup-node@v2
with:
node-version: '21'
- name: Build React App
run: |
yarn install
yarn build
- name: Upload React build as artifact
uses: actions/upload-artifact@v2
with:
name: react-build
path: build/
build-docker:
needs: build-frontend
runs-on: [self-hosted]
steps:
- name: Checkout
uses: actions/checkout@v2
- name: Load ENV from file
run: |
cat .github_build/Build.restreamer-ui.env >> $GITHUB_ENV
- name: Download React build artifact
uses: actions/download-artifact@v2
with:
name: react-build
path: build
- uses: actions-ecosystem/action-get-latest-tag@v1
id: get-latest-tag
with:
semver_only: true
- name: Docker meta
id: meta
uses: docker/metadata-action@v4
env:
CADDY_IMAGE: ${{ env.CADDY_IMAGE }}
with:
images: |
datarhei/restreamer-ui
tags: |
type=raw,value=dev
- name: Set up QEMU
uses: docker/setup-qemu-action@master
with:
platforms: all
- name: Set up Docker Buildx
id: buildx
uses: docker/setup-buildx-action@master
- name: Cache Docker layers
uses: actions/cache@v2
with:
path: /tmp/.buildx-cache
key: ${{ runner.os }}-buildx-${{ github.sha }}
restore-keys: |
${{ runner.os }}-buildx-
- name: Login to DockerHub
if: github.event_name != 'pull_request'
uses: docker/login-action@v1
with:
username: ${{ secrets.DOCKER_USERNAME }}
password: ${{ secrets.DOCKER_PASSWORD }}
- name: Build and Push Docker Image
uses: docker/build-push-action@v2
with:
builder: ${{ steps.buildx.outputs.name }}
context: .
file: ./Dockerfile.workflow
build-args: |
PUBLIC_URL=/ui
CADDY_IMAGE=${{ env.CADDY_IMAGE }}
platforms: linux/amd64,linux/arm64,linux/arm/v7
push: true
tags: ${{ steps.meta.outputs.tags }}
labels: ${{ steps.meta.outputs.labels }}
cache-from: type=local,src=/tmp/.buildx-cache
cache-to: type=local,dest=/tmp/.buildx-cache-new

View File

@ -0,0 +1,2 @@
RELEASE=1.10.0
CADDY_IMAGE=caddy:2.7.6-alpine

View File

@ -22,7 +22,7 @@
"it",
"ko",
"pl",
"pt-br",
"pt",
"ru",
"sl",
"tr",

View File

@ -1,44 +1,5 @@
# Restreamer-UI
## v1.13.0 > v1.14.0
- Add wettercom service
- Add option to select which channels will be displayed on the playersite ([#392](https://github.com/datarhei/restreamer/issues/392), [#800](https://github.com/datarhei/restreamer/issues/800))
- Mod updates public videojs >v8
- Fix erroneous filter setting
- Fix encoded address
- Fix double -filter parameter when encoder sets filter
- Fix Docker build ([#64](https://github.com/datarhei/restreamer-ui/issues/64))
## v1.12.0 > v1.13.0
- Add to allow stream hints in case probing fails
- Mod enables ff-loglevel and prepares the logging component
- Mod uses official Instagram-RTMP target
- Mod Remove unused imports
- Mod Update translations
- Mod updates dep.
- Fix player position
- Fix missing stream URL, summarize streams in probe log, don't lock type for first stream
## v1.11.0 > v1.12.0
- Add option to select different SRT stream in wizard
- Add option to select different RTMP stream in wizard
- Fix selecting other than first audio stream ([#710](https://github.com/datarhei/restreamer/issues/710))
- Fix reset of previous audio settings when editing profile ([#730](https://github.com/datarhei/restreamer/issues/730))
- Fix RTMP URL for receive mode
## v1.10.0 > v1.11.0
- Add allow to stream HEVC and AV1 to Youtube via RTMP
- Add librav1e AV1 encoder
- Add support for AV1 CUDA decoding ([PR 46](https://github.com/datarhei/restreamer-ui/pull/46))
- Add FFmpeg 6 support
- Add HEVC VideoToolbox encoder
- Fix anonymize error message ([#688](https://github.com/datarhei/restreamer/issues/688))
- Fix chromecast config ([#37](https://github.com/datarhei/restreamer-ui/issues/37))
## v1.9.0 > v1.10.0
- Add resource usage and ffmpeg command to process details

View File

@ -1,17 +1,22 @@
ARG NODE_IMAGE=node:21-alpine3.20
ARG CADDY_IMAGE=caddy:2.8.4-alpine
ARG NODE_IMAGE=node:21-alpine3.17
ARG CADDY_IMAGE=caddy:2.7.5-alpine
FROM $NODE_IMAGE AS builder
FROM $NODE_IMAGE as builder
ENV PUBLIC_URL="./"
ARG NODE_SPACE_SIZE=10240
ENV NODE_OPTIONS="--openssl-legacy-provider --max-old-space-size=$NODE_SPACE_SIZE"
ENV PUBLIC_URL "./"
COPY . /ui
WORKDIR /ui
RUN cd /ui && \
yarn install && \
yarn build
yarn set version berry && \
yarn config set httpTimeout 600000 && \
yarn install && \
yarn run build
FROM $CADDY_IMAGE

View File

@ -4,8 +4,6 @@ FROM $CADDY_IMAGE
COPY build /ui/build
COPY Caddyfile /ui/Caddyfile
ENV PUBLIC_URL="./"
WORKDIR /ui
EXPOSE 3000

View File

@ -1,52 +1,52 @@
{
"name": "restreamer-ui",
"version": "1.14.0",
"bundle": "restreamer-v2.12.0",
"version": "1.10.0",
"bundle": "restreamer-v2.8.0",
"private": false,
"license": "Apache-2.0",
"dependencies": {
"@auth0/auth0-spa-js": "^2.1.3",
"@babel/plugin-syntax-flow": "^7.24.7",
"@babel/plugin-transform-react-jsx": "^7.25.2",
"@emotion/react": "^11.13.3",
"@emotion/styled": "^11.13.0",
"@fontsource/dosis": "^5.0.21",
"@fontsource/roboto": "^5.0.14",
"@fortawesome/fontawesome-svg-core": "^6.6.0",
"@fortawesome/free-brands-svg-icons": "^6.6.0",
"@fortawesome/free-solid-svg-icons": "^6.6.0",
"@fortawesome/react-fontawesome": "^0.2.2",
"@lingui/core": "^4.11.4",
"@lingui/macro": "^4.11.4",
"@lingui/react": "^4.11.4",
"@mui/icons-material": "^6.0.1",
"@mui/lab": "^6.0.0-beta.8",
"@mui/material": "^6.0.1",
"@mui/styles": "^6.0.1",
"@testing-library/dom": "^10.4.0",
"@testing-library/jest-dom": "^6.5.0",
"@testing-library/react": "^16.0.1",
"@babel/plugin-syntax-flow": "^7.23.3",
"@babel/plugin-transform-react-jsx": "^7.23.4",
"@emotion/react": "^11.11.3",
"@emotion/styled": "^11.11.0",
"@fontsource/dosis": "^5.0.18",
"@fontsource/roboto": "^5.0.8",
"@fortawesome/fontawesome-svg-core": "^6.5.1",
"@fortawesome/free-brands-svg-icons": "^6.5.1",
"@fortawesome/free-solid-svg-icons": "^6.5.1",
"@fortawesome/react-fontawesome": "^0.2.0",
"@lingui/core": "^4.7.0",
"@lingui/macro": "^4.7.0",
"@lingui/react": "^4.7.0",
"@mui/icons-material": "^5.15.6",
"@mui/lab": "^5.0.0-alpha.162",
"@mui/material": "^5.15.6",
"@mui/styles": "^5.15.6",
"@testing-library/dom": "^9.3.4",
"@testing-library/jest-dom": "^6.3.0",
"@testing-library/react": "^14.1.2",
"@testing-library/user-event": "^14.5.2",
"@types/react": "^18.3.5",
"@types/react": "^18.2.48",
"babel-plugin-macros": "^3.1.0",
"eslint": "^9.9.1",
"eslint": "^8.56.0",
"handlebars": "^4.7.8",
"jwt-decode": "^4.0.0",
"make-plural": "^7.4.0",
"react": "^18.3.1",
"make-plural": "^7.3.0",
"react": "^18.2.0",
"react-colorful": "^5.6.1",
"react-device-detect": "^2.2.3",
"react-dom": "^18.3.1",
"react-dom": "^18.2.0",
"react-markdown": "^9.0.1",
"react-router-dom": "^6.26.1",
"react-router-dom": "^6.21.3",
"react-scripts": "^5.0.1",
"semver": "^7.6.3",
"serve": "^14.2.3",
"typescript": "^5.5.4",
"semver": "^7.5.4",
"serve": "^14.2.1",
"typescript": "^5.3.3",
"url-parse": "^1.5.10",
"util": "^0.12.5",
"uuid": "^10.0.0",
"video.js": "^8.17.3",
"uuid": "^9.0.1",
"video.js": "^8.10.0",
"videojs-overlay": "^3.1.0"
},
"scripts": {
@ -86,12 +86,12 @@
]
},
"devDependencies": {
"@babel/core": "^7.25.2",
"@babel/core": "^7.23.9",
"@babel/plugin-proposal-private-property-in-object": "^7.21.11",
"@lingui/cli": "^4.11.4",
"@lingui/cli": "^4.7.0",
"babel-core": "^6.26.3",
"eslint-config-react-app": "^7.0.1",
"prettier": "^3.3.3",
"prettier": "^3.2.4",
"react-error-overlay": "^6.0.11"
},
"resolutions": {}

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -1 +0,0 @@
.vjs-airplay-button .vjs-icon-placeholder{background:url("ic_airplay_white_24px.svg") center center no-repeat;background-size:contain;display:inline-block;width:12px;height:12px}.vjs-airplay-button:hover{cursor:pointer}.vjs-airplay-button:hover .vjs-icon-placeholder{background-image:url("ic_airplay_white_24px.svg")}.vjs-airplay-button.vjs-airplay-button-lg:not(.vjs-hidden){display:flex;align-items:center;width:auto;padding:0 4px}.vjs-airplay-button.vjs-airplay-button-lg:not(.vjs-hidden) .vjs-airplay-button-label{flex-grow:1;margin-left:4px}.vjs-airplay-button.vjs-airplay-button-lg:not(.vjs-hidden) .vjs-icon-placeholder{flex-grow:1}

View File

@ -1,308 +0,0 @@
(function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i<t.length;i++)o(t[i]);return o}return r})()({1:[function(require,module,exports){
"use strict";
var hasAirPlayAPISupport = require('../lib/hasAirPlayAPISupport');
/**
* Registers the AirPlayButton Component with Video.js. Calls
* {@link http://docs.videojs.com/Component.html#.registerComponent}, which will add a
* component called `airPlayButton` to the list of globally registered Video.js
* components. The `airPlayButton` is added to the player's control bar UI automatically
* once {@link module:enableAirPlay} has been called. If you would like to specify the
* order of the buttons that appear in the control bar, including this button, you can do
* so in the options that you pass to the `videojs` function when creating a player:
*
* ```
* videojs('playerID', {
* controlBar: {
* children: [
* 'playToggle',
* 'progressControl',
* 'volumePanel',
* 'fullscreenToggle',
* 'airPlayButton',
* ],
* }
* });
* ```
*
* @param videojs {object} A reference to {@link http://docs.videojs.com/module-videojs.html|Video.js}
* @see http://docs.videojs.com/module-videojs.html#~registerPlugin
*/
module.exports = function (videojs) {
/**
* The AirPlayButton module contains both the AirPlayButton class definition and the
* function used to register the button as a Video.js Component.
*
* @module AirPlayButton
*/
const ButtonComponent = videojs.getComponent('Button');
/**
* The Video.js Button class is the base class for UI button components.
*
* @external Button
* @see {@link http://docs.videojs.com/Button.html|Button}
*/
/** @lends AirPlayButton.prototype */
class AirPlayButton extends ButtonComponent {
/**
* This class is a button component designed to be displayed in the
* player UI's control bar. It displays an Apple AirPlay selection
* list when clicked.
*
* @constructs
* @extends external:Button
*/
constructor(player, options) {
super(player, options);
if (!hasAirPlayAPISupport()) {
this.hide();
}
this._reactToAirPlayAvailableEvents();
if (options.addAirPlayLabelToButton) {
this.el().classList.add('vjs-airplay-button-lg');
this._labelEl = document.createElement('span');
this._labelEl.classList.add('vjs-airplay-button-label');
this._labelEl.textContent = this.localize('AirPlay');
this.el().appendChild(this._labelEl);
} else {
this.controlText('Start AirPlay');
}
}
/**
* Overrides Button#buildCSSClass to return the classes used on the button element.
*
* @param {DOMElement} el
* @see {@link http://docs.videojs.com/Button.html#buildCSSClass|Button#buildCSSClass}
*/
buildCSSClass() {
return 'vjs-airplay-button ' + super.buildCSSClass();
}
/**
* Overrides Button#handleClick to handle button click events. AirPlay
* functionality is handled outside of this class, which should be limited
* to UI related logic. This function simply triggers an event on the player.
*
* @fires AirPlayButton#airPlayRequested
* @param {DOMElement} el
* @see {@link http://docs.videojs.com/Button.html#handleClick|Button#handleClick}
*/
handleClick() {
this.player().trigger('airPlayRequested');
}
/**
* Gets the underlying DOMElement used by the player.
*
* @private
* @returns {DOMElement} either an <audio> or <video> tag, depending on the type of
* player
*/
_getMediaEl() {
var playerEl = this.player().el();
return playerEl.querySelector('video, audio');
}
/**
* Binds a listener to the `webkitplaybacktargetavailabilitychanged` event, if it is
* supported, that will show or hide this button Component based on the availability
* of the AirPlay function.
*
* @private
*/
_reactToAirPlayAvailableEvents() {
var mediaEl = this._getMediaEl(),
self = this;
if (!mediaEl || !hasAirPlayAPISupport()) {
return;
}
function onTargetAvailabilityChanged(event) {
if (event.availability === 'available') {
self.show();
} else {
self.hide();
}
}
mediaEl.addEventListener('webkitplaybacktargetavailabilitychanged', onTargetAvailabilityChanged);
this.on('dispose', function () {
mediaEl.removeEventListener('webkitplaybacktargetavailabilitychanged', onTargetAvailabilityChanged);
});
}
}
videojs.registerComponent('airPlayButton', AirPlayButton);
};
},{"../lib/hasAirPlayAPISupport":4}],2:[function(require,module,exports){
"use strict";
/**
* @module enableAirPlay
*/
var hasAirPlayAPISupport = require('./lib/hasAirPlayAPISupport');
/**
* @private
* @param {object} the Video.js Player instance
* @returns {AirPlayButton} or `undefined` if it does not exist
*/
function getExistingAirPlayButton(player) {
return player.controlBar.getChild('airPlayButton');
}
/**
* Adds the AirPlayButton Component to the player's ControlBar component, if the
* AirPlayButton does not already exist in the ControlBar.
* @private
* @param player {object} the Video.js Player instance
* @param options {object}
*/
function ensureAirPlayButtonExists(player, options) {
var existingAirPlayButton = getExistingAirPlayButton(player),
indexOpt;
if (options.addButtonToControlBar && !existingAirPlayButton) {
// Figure out AirPlay button's index
indexOpt = player.controlBar.children().length;
if (typeof options.buttonPositionIndex !== 'undefined') {
indexOpt = options.buttonPositionIndex >= 0 ? options.buttonPositionIndex : player.controlBar.children().length + options.buttonPositionIndex;
}
player.controlBar.addChild('airPlayButton', options, indexOpt);
}
}
/**
* Handles requests for AirPlay triggered by the AirPlayButton Component.
*
* @private
* @param player {object} the Video.js Player instance
*/
function onAirPlayRequested(player) {
var mediaEl = player.el().querySelector('video, audio');
if (mediaEl && mediaEl.webkitShowPlaybackTargetPicker) {
mediaEl.webkitShowPlaybackTargetPicker();
}
}
/**
* Adds an event listener for the `airPlayRequested` event triggered by the AirPlayButton
* Component.
*
* @private
* @param player {object} the Video.js Player instance
*/
function listenForAirPlayEvents(player) {
// Respond to requests for AirPlay. The AirPlayButton component triggers this event
// when the user clicks the AirPlay button.
player.on('airPlayRequested', onAirPlayRequested.bind(null, player));
}
/**
* Sets up the AirPlay plugin.
*
* @private
* @param player {object} the Video.js player
* @param options {object} the plugin options
*/
function enableAirPlay(player, options) {
if (!player.controlBar) {
return;
}
if (hasAirPlayAPISupport()) {
listenForAirPlayEvents(player);
ensureAirPlayButtonExists(player, options);
}
}
/**
* Registers the AirPlay plugin with Video.js. Calls
* {@link http://docs.videojs.com/module-videojs.html#~registerPlugin|videojs#registerPlugin},
* which will add a plugin function called `airPlay` to any instance of a Video.js player
* that is created after calling this function. Call `player.airPlay(options)`, passing in
* configuration options, to enable the AirPlay plugin on your Player instance.
*
* Currently, the only configuration option is:
*
* * **buttonText** - the text to display inside of the button component. By default,
* this text is hidden and is used for accessibility purposes.
*
* @param {object} videojs
* @see http://docs.videojs.com/module-videojs.html#~registerPlugin
*/
module.exports = function (videojs) {
videojs.registerPlugin('airPlay', function (options) {
var pluginOptions = Object.assign({
addButtonToControlBar: true
}, options || {});
// `this` is an instance of a Video.js Player.
// Wait until the player is "ready" so that the player's control bar component has
// been created.
this.ready(enableAirPlay.bind(this, this, pluginOptions));
});
};
},{"./lib/hasAirPlayAPISupport":4}],3:[function(require,module,exports){
"use strict";
var createAirPlayButton = require('./components/AirPlayButton'),
createAirPlayPlugin = require('./enableAirPlay');
/**
* @module index
*/
/**
* Registers the AirPlay plugin and AirPlayButton Component with Video.js. See
* {@link module:AirPlayButton} and {@link module:enableAirPlay} for more details about
* how the plugin and button are registered and configured.
*
* @param {object} videojs
* @see module:enableAirPlay
* @see module:AirPlayButton
*/
module.exports = function (videojs) {
videojs = videojs || window.videojs;
createAirPlayButton(videojs);
createAirPlayPlugin(videojs);
};
},{"./components/AirPlayButton":1,"./enableAirPlay":2}],4:[function(require,module,exports){
"use strict";
/**
* @module hasAirPlayAPISupport
*/
/**
* Returns whether or not the current browser environment supports AirPlay.
*
* @private
* @returns {boolean} true if AirPlay support is available
*/
module.exports = function () {
return !!window.WebKitPlaybackTargetAvailabilityEvent;
};
},{}],5:[function(require,module,exports){
"use strict";
/**
* This module is used as an entry point for the build system to bundle this plugin into a
* single javascript file that can be loaded by a script tag on a web page. The javascript
* file that is built assumes that `videojs` is available globally at `window.videojs`, so
* Video.js must be loaded **before** this plugin is loaded.
*
* Run `npm install` and then `grunt build` to build the plugin's bundled javascript
* file, as well as the CSS and image assets into the project's `./dist/` folder.
*
* @module standalone
*/
require('./index')();
},{"./index":3}]},{},[5]);

View File

@ -1 +1 @@
.vjs-airplay-button .vjs-icon-placeholder{background:url("ic_airplay_white_24px.svg") center center no-repeat;background-size:contain;display:inline-block;width:12px;height:12px}.vjs-airplay-button:hover{cursor:pointer}.vjs-airplay-button:hover .vjs-icon-placeholder{background-image:url("ic_airplay_white_24px.svg")}.vjs-airplay-button.vjs-airplay-button-lg:not(.vjs-hidden){display:flex;align-items:center;width:auto;padding:0 4px}.vjs-airplay-button.vjs-airplay-button-lg:not(.vjs-hidden) .vjs-airplay-button-label{flex-grow:1;margin-left:4px}.vjs-airplay-button.vjs-airplay-button-lg:not(.vjs-hidden) .vjs-icon-placeholder{flex-grow:1}
.vjs-airplay-button .vjs-icon-placeholder{background:url("ic_airplay_white_24px.svg") center center no-repeat;background-size:contain;display:inline-block;width:20px;height:20px}.vjs-airplay-button:hover{cursor:pointer}.vjs-airplay-button:hover .vjs-icon-placeholder{background-image:url("ic_airplay_white_24px.svg")}

File diff suppressed because it is too large Load Diff

View File

@ -1 +1,65 @@
.vjs-chromecast-button .vjs-icon-placeholder{background:url("ic_cast_white_24dp.png") center center no-repeat;background-size:contain;display:inline-block;width:12px;height:12px}.vjs-chromecast-button:hover{cursor:pointer}.vjs-chromecast-button:hover .vjs-icon-placeholder{background-image:url("ic_cast_white_24dp.png")}.vjs-chromecast-button.vjs-chromecast-casting-state .vjs-icon-placeholder{background-image:url("ic_cast_connected_white_24dp.png")}.vjs-chromecast-button.vjs-chromecast-casting-state:hover .vjs-icon-placeholder{background-image:url("ic_cast_connected_white_24dp.png")}.vjs-chromecast-button.vjs-chromecast-button-lg:not(.vjs-hidden){display:flex;align-items:center;width:auto;padding:0 4px}.vjs-chromecast-button.vjs-chromecast-button-lg:not(.vjs-hidden) .vjs-chromecast-button-label{flex-grow:1;margin-left:4px}.vjs-chromecast-button.vjs-chromecast-button-lg:not(.vjs-hidden) .vjs-icon-placeholder{flex-grow:1}.vjs-tech-chromecast{display:flex;flex-direction:column;justify-content:center;align-items:center;overflow:hidden}.vjs-tech-chromecast .vjs-tech-chromecast-poster::after{content:" ";display:block;height:2px;width:100px;background-color:#ccc;position:absolute;left:calc(50% - 50px)}.vjs-tech-chromecast .vjs-tech-chromecast-poster-img{max-height:180px;width:auto;border:2px solid #ccc}.vjs-tech-chromecast .vjs-tech-chromecast-poster-img.vjs-tech-chromecast-poster-img-empty{width:160px;height:90px}.vjs-tech-chromecast .vjs-tech-chromecast-title-container{position:absolute;bottom:50%;margin-bottom:100px;color:#ccc;text-align:center}.vjs-tech-chromecast .vjs-tech-chromecast-title{font-size:22px}.vjs-tech-chromecast .vjs-tech-chromecast-title.vjs-tech-chromecast-title-empty{display:none}.vjs-tech-chromecast .vjs-tech-chromecast-subtitle{font-size:18px;padding-top:.5em}.vjs-tech-chromecast .vjs-tech-chromecast-subtitle.vjs-tech-chromecast-subtitle-empty{display:none}
/** Silvermine Chromecast **/
.vjs-chromecast-button .vjs-icon-placeholder {
background: url('ic_cast_white_24dp.png') center center no-repeat;
background-size: contain;
display: inline-block;
width: 20px;
height: 20px;
}
.vjs-chromecast-button:hover {
cursor: pointer;
}
.vjs-chromecast-button:hover .vjs-icon-placeholder {
background-image: url('ic_cast_white_24dp.png');
}
.vjs-chromecast-button.vjs-chromecast-casting-state .vjs-icon-placeholder {
background-image: url('ic_cast_connected_white_24dp.png');
}
.vjs-chromecast-button.vjs-chromecast-casting-state:hover .vjs-icon-placeholder {
background-image: url('ic_cast_connected_white_24dp.png');
}
.vjs-tech-chromecast {
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
overflow: hidden;
}
.vjs-tech-chromecast .vjs-tech-chromecast-poster::after {
content: ' ';
display: block;
height: 2px;
width: 100px;
background-color: #cccccc;
position: absolute;
left: calc(50% - 50px);
}
.vjs-tech-chromecast .vjs-tech-chromecast-poster-img {
max-height: 180px;
width: auto;
border: 2px solid #cccccc;
}
.vjs-tech-chromecast .vjs-tech-chromecast-poster-img.vjs-tech-chromecast-poster-img-empty {
width: 160px;
height: 90px;
}
.vjs-tech-chromecast .vjs-tech-chromecast-title-container {
position: absolute;
bottom: 50%;
margin-bottom: 100px;
color: #cccccc;
text-align: center;
}
.vjs-tech-chromecast .vjs-tech-chromecast-title {
font-size: 22px;
}
.vjs-tech-chromecast .vjs-tech-chromecast-title.vjs-tech-chromecast-title-empty {
display: none;
}
.vjs-tech-chromecast .vjs-tech-chromecast-subtitle {
font-size: 18px;
padding-top: 0.5em;
}
.vjs-tech-chromecast .vjs-tech-chromecast-subtitle.vjs-tech-chromecast-subtitle-empty {
display: none;
}

File diff suppressed because it is too large Load Diff

View File

@ -1 +1 @@
.vjs-chromecast-button .vjs-icon-placeholder{background:url("ic_cast_white_24dp.png") center center no-repeat;background-size:contain;display:inline-block;width:12px;height:12px}.vjs-chromecast-button:hover{cursor:pointer}.vjs-chromecast-button:hover .vjs-icon-placeholder{background-image:url("ic_cast_white_24dp.png")}.vjs-chromecast-button.vjs-chromecast-casting-state .vjs-icon-placeholder{background-image:url("ic_cast_connected_white_24dp.png")}.vjs-chromecast-button.vjs-chromecast-casting-state:hover .vjs-icon-placeholder{background-image:url("ic_cast_connected_white_24dp.png")}.vjs-chromecast-button.vjs-chromecast-button-lg:not(.vjs-hidden){display:flex;align-items:center;width:auto;padding:0 4px}.vjs-chromecast-button.vjs-chromecast-button-lg:not(.vjs-hidden) .vjs-chromecast-button-label{flex-grow:1;margin-left:4px}.vjs-chromecast-button.vjs-chromecast-button-lg:not(.vjs-hidden) .vjs-icon-placeholder{flex-grow:1}.vjs-tech-chromecast{display:flex;flex-direction:column;justify-content:center;align-items:center;overflow:hidden}.vjs-tech-chromecast .vjs-tech-chromecast-poster::after{content:" ";display:block;height:2px;width:100px;background-color:#ccc;position:absolute;left:calc(50% - 50px)}.vjs-tech-chromecast .vjs-tech-chromecast-poster-img{max-height:180px;width:auto;border:2px solid #ccc}.vjs-tech-chromecast .vjs-tech-chromecast-poster-img.vjs-tech-chromecast-poster-img-empty{width:160px;height:90px}.vjs-tech-chromecast .vjs-tech-chromecast-title-container{position:absolute;bottom:50%;margin-bottom:100px;color:#ccc;text-align:center}.vjs-tech-chromecast .vjs-tech-chromecast-title{font-size:22px}.vjs-tech-chromecast .vjs-tech-chromecast-title.vjs-tech-chromecast-title-empty{display:none}.vjs-tech-chromecast .vjs-tech-chromecast-subtitle{font-size:18px;padding-top:.5em}.vjs-tech-chromecast .vjs-tech-chromecast-subtitle.vjs-tech-chromecast-subtitle-empty{display:none}
.vjs-chromecast-button .vjs-icon-placeholder{background:url(ic_cast_white_24dp.png) center center no-repeat;background-size:contain;display:inline-block;width:20px;height:20px}.vjs-chromecast-button:hover{cursor:pointer}.vjs-chromecast-button:hover .vjs-icon-placeholder{background-image:url(ic_cast_white_24dp.png)}.vjs-chromecast-button.vjs-chromecast-casting-state .vjs-icon-placeholder{background-image:url(ic_cast_connected_white_24dp.png)}.vjs-chromecast-button.vjs-chromecast-casting-state:hover .vjs-icon-placeholder{background-image:url(ic_cast_connected_white_24dp.png)}.vjs-tech-chromecast{display:flex;flex-direction:column;justify-content:center;align-items:center;overflow:hidden}.vjs-tech-chromecast .vjs-tech-chromecast-poster::after{content:' ';display:block;height:2px;width:100px;background-color:#ccc;position:absolute;left:calc(50% - 50px)}.vjs-tech-chromecast .vjs-tech-chromecast-poster-img{max-height:180px;width:auto;border:2px solid #ccc}.vjs-tech-chromecast .vjs-tech-chromecast-poster-img.vjs-tech-chromecast-poster-img-empty{width:160px;height:90px}.vjs-tech-chromecast .vjs-tech-chromecast-title-container{position:absolute;bottom:50%;margin-bottom:100px;color:#ccc;text-align:center}.vjs-tech-chromecast .vjs-tech-chromecast-title{font-size:22px}.vjs-tech-chromecast .vjs-tech-chromecast-title.vjs-tech-chromecast-title-empty{display:none}.vjs-tech-chromecast .vjs-tech-chromecast-subtitle{font-size:18px;padding-top:.5em}.vjs-tech-chromecast .vjs-tech-chromecast-subtitle.vjs-tech-chromecast-subtitle-empty{display:none}

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -1,340 +1,459 @@
/*! @name videojs-license @version 0.1.0 @license MIT */
(function (global, factory) {
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory(require('video.js')) :
typeof define === 'function' && define.amd ? define(['video.js'], factory) :
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.videojsLicense = factory(global.videojs));
})(this, (function (videojs) { 'use strict';
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory(require('video.js'), require('global/document')) :
typeof define === 'function' && define.amd ? define(['video.js', 'global/document'], factory) :
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.videojsLicense = factory(global.videojs, global.document));
}(this, (function (videojs, document) { 'use strict';
function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }
function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }
var videojs__default = /*#__PURE__*/_interopDefaultLegacy(videojs);
var videojs__default = /*#__PURE__*/_interopDefaultLegacy(videojs);
var document__default = /*#__PURE__*/_interopDefaultLegacy(document);
var version = "0.1.0";
function createCommonjsModule(fn, basedir, module) {
return module = {
path: basedir,
exports: {},
require: function (path, base) {
return commonjsRequire(path, (base === undefined || base === null) ? module.path : base);
}
}, fn(module, module.exports), module.exports;
}
const Plugin = videojs__default["default"].getPlugin('plugin');
const Component = videojs__default["default"].getComponent('Component');
const Button = videojs__default["default"].getComponent('MenuButton');
function commonjsRequire () {
throw new Error('Dynamic requires are not currently supported by @rollup/plugin-commonjs');
}
// Default options for the plugin.
const defaults = {
license: 'none',
title: '',
author: '',
languages: {
license: 'License',
loading: 'Loading'
}
};
var assertThisInitialized = createCommonjsModule(function (module) {
function _assertThisInitialized(self) {
if (self === void 0) {
throw new ReferenceError("this hasn't been initialised - super() hasn't been called");
}
/**
* An advanced Video.js plugin. For more information on the API
*
* See: https://blog.videojs.com/feature-spotlight-advanced-plugins/
*/
class License extends Plugin {
/**
* Create a License plugin instance.
*
* @param {Player} player
* A Video.js Player instance.
*
* @param {Object} [options]
* An optional options object.
*
* While not a core part of the Video.js plugin architecture, a
* second argument of options is a convenient way to accept inputs
* from your plugin's caller.
*/
constructor(player, options) {
// the parent class will add player under this.player
super(player);
this.playerId = this.player.id();
this.options = videojs__default["default"].mergeOptions(defaults, options);
if (options.license === 'none') {
return;
}
player.addClass('vjs-license');
this.buildUI();
if (videojs__default["default"].browser.IS_IOS || videojs__default["default"].browser.IS_ANDROID) {
this.mobileBuildUI();
}
return self;
}
// close the menu if open on userinactive
this.player.on('userinactive', () => {
document.getElementById(this.playerId).querySelectorAll('.vjs-menu').forEach(element => {
element.classList.remove('vjs-lock-open');
});
});
module.exports = _assertThisInitialized;
module.exports["default"] = module.exports, module.exports.__esModule = true;
});
// close the menu if anywhere in the player is clicked
this.player.on('click', evt => {
if (evt.target.tagName === 'VIDEO') {
document.getElementById(this.playerId).querySelectorAll('.vjs-menu').forEach(element => {
element.classList.remove('vjs-lock-open');
});
}
});
this.player.on('loadstart', _event => {
this.removeElementsByClass('vjs-license-clear');
if (videojs__default["default"].browser.IS_IOS || videojs__default["default"].browser.IS_ANDROID) {
this.mobileBuildTopLevelMenu();
} else {
this.buildTopLevelMenu();
}
});
}
var setPrototypeOf = createCommonjsModule(function (module) {
function _setPrototypeOf(o, p) {
module.exports = _setPrototypeOf = Object.setPrototypeOf || function _setPrototypeOf(o, p) {
o.__proto__ = p;
return o;
};
/**
* Add the menu ui button to the controlbar
*/
buildUI() {
const playerId = this.playerId;
const that = this;
module.exports["default"] = module.exports, module.exports.__esModule = true;
return _setPrototypeOf(o, p);
}
/**
* LicenseMenuButton
*/
class LicenseMenuButton extends Button {
/**
* Contructor
*
* @param {*} player videojs player instance
* @param {*} options videojs player options
*/
constructor(player, options) {
super(player, options);
this.addClass('vjs-license');
this.controlText(that.options.languages.loading);
player.one('canplaythrough', _event => {
this.controlText(that.options.languages.settings);
});
this.menu.contentEl_.id = playerId + '-vjs-license-default';
}
module.exports = _setPrototypeOf;
module.exports["default"] = module.exports, module.exports.__esModule = true;
});
/**
* Handle click
*/
handleClick() {
if (videojs__default["default"].browser.IS_IOS || videojs__default["default"].browser.IS_ANDROID) {
this.player.getChild('licenseMenuMobileModal').el().style.display = 'block';
} else {
this.el().classList.toggle('vjs-toogle-btn');
this.menu.el().classList.toggle('vjs-lock-open');
}
}
}
videojs__default["default"].registerComponent('licenseMenuButton', LicenseMenuButton);
this.player.getChild('controlBar').addChild('licenseMenuButton');
if (this.player.getChild('controlBar').getChild('fullscreenToggle')) {
this.player.getChild('controlBar').el().insertBefore(this.player.getChild('controlBar').getChild('licenseMenuButton').el(), this.player.getChild('controlBar').getChild('fullscreenToggle').el());
}
}
var inheritsLoose = createCommonjsModule(function (module) {
function _inheritsLoose(subClass, superClass) {
subClass.prototype = Object.create(superClass.prototype);
subClass.prototype.constructor = subClass;
setPrototypeOf(subClass, superClass);
}
/**
*
* This is just build the top level menu no sub menus
*/
buildTopLevelMenu() {
const settingsButton = this.player.getChild('controlBar').getChild('licenseMenuButton');
module.exports = _inheritsLoose;
module.exports["default"] = module.exports, module.exports.__esModule = true;
});
// settingsButton.addClass('vjs-license-is-loaded');
var version = "0.1.0";
const main = settingsButton.menu.contentEl_;
var Plugin = videojs__default['default'].getPlugin('plugin');
var Component = videojs__default['default'].getComponent('Component');
var Button = videojs__default['default'].getComponent('MenuButton'); // Default options for the plugin.
// Empty the main menu div to repopulate
main.innerHTML = '';
main.classList.add('vjs-license-top-level');
var defaults = {
license: 'none',
title: '',
author: '',
languages: {
license: 'License',
loading: 'Loading'
}
};
/**
* An advanced Video.js plugin. For more information on the API
*
* See: https://blog.videojs.com/feature-spotlight-advanced-plugins/
*/
// Start building new list items
const menuTitle = document.createElement('li');
menuTitle.className = 'vjs-license-top-level-header';
const menuTitleInner = document.createElement('span');
menuTitleInner.innerHTML = 'About';
menuTitleInner.className = 'vjs-license-top-level-header-titel';
menuTitle.appendChild(menuTitleInner);
main.appendChild(menuTitle);
const itemTitel = document.createElement('li');
itemTitel.innerHTML = this.buildItemTitel();
itemTitel.className = 'vjs-license-top-level-item';
main.appendChild(itemTitel);
if (this.options.author) {
const itemAuthor = document.createElement('li');
itemAuthor.innerHTML = this.buildItemAuthor();
itemAuthor.className = 'vjs-license-top-level-item';
main.appendChild(itemAuthor);
}
const itemLicense = document.createElement('li');
itemLicense.innerHTML = this.buildItemLicense();
itemLicense.className = 'vjs-license-top-level-item';
main.appendChild(itemLicense);
}
var License = /*#__PURE__*/function (_Plugin) {
inheritsLoose(License, _Plugin);
/**
* Add the menu ui button to the controlbar
*/
mobileBuildUI() {
/**
* bla
*/
class LicenseMenuMobileModal extends Component {
/**
* Contructor
*
* @param {*} player videojs player instance
* @param {*} options videojs player options
*/
constructor(player, options) {
super(player, options);
}
/**
* Create a License plugin instance.
*
* @param {Player} player
* A Video.js Player instance.
*
* @param {Object} [options]
* An optional options object.
*
* While not a core part of the Video.js plugin architecture, a
* second argument of options is a convenient way to accept inputs
* from your plugin's caller.
*/
function License(player, options) {
var _this;
/**
* Creates an HTML element
*
* @return {Object} HTML element
*/
createEl() {
return videojs__default["default"].createEl('div', {
className: 'vjs-license-mobile'
});
}
}
videojs__default["default"].registerComponent('licenseMenuMobileModal', LicenseMenuMobileModal);
videojs__default["default"].dom.prependTo(this.player.addChild('licenseMenuMobileModal').el(), document.body);
}
// the parent class will add player under this.player
_this = _Plugin.call(this, player) || this;
_this.playerId = _this.player.id();
_this.options = videojs__default['default'].mergeOptions(defaults, options);
/**
* Add the menu ui button to the controlbar
*/
mobileBuildTopLevelMenu() {
const settingsButton = this.player.getChild('licenseMenuMobileModal');
const menuTopLevel = document.createElement('ul');
menuTopLevel.className = 'vjs-license-mob-top-level vjs-setting-menu-clear';
settingsButton.el().appendChild(menuTopLevel);
if (options.license === 'none') {
return assertThisInitialized(_this);
}
// Empty the main menu div to repopulate
const menuTitle = document.createElement('li');
menuTitle.innerHTML = 'About';
menuTitle.className = 'vjs-setting-menu-mobile-top-header';
menuTopLevel.appendChild(menuTitle);
const itemTitel = document.createElement('li');
itemTitel.innerHTML = this.buildItemTitel();
itemTitel.className = 'vjs-license-top-level-item';
if (this.options.author) {
const itemAuthor = document.createElement('li');
itemAuthor.innerHTML = this.buildItemAuthor();
itemAuthor.className = 'vjs-license-top-level-item';
}
const itemLicense = document.createElement('li');
itemLicense.innerHTML = this.buildItemLicense();
itemLicense.className = 'vjs-license-top-level-item';
const menuClose = document.createElement('li');
menuClose.innerHTML = 'Close';
menuClose.className = 'setting-menu-footer-default';
menuClose.onclick = e => {
this.player.getChild('settingsMenuMobileModal').el().style.display = 'none';
};
menuTopLevel.appendChild(menuClose);
}
_this.player.ready(function () {
_this.player.addClass('vjs-license');
/**
* Add the menu ui button to the controlbar
*
* @return {string} Returns license text
*/
buildItemTitel() {
let titel = '';
if (this.options.title) {
titel = `${this.options.title}`;
}
return 'Title: ' + titel;
}
_this.buildUI();
/**
* Add the menu ui button to the controlbar
*
* @return {string} Returns license text
*/
buildItemAuthor() {
let author = '';
if (this.options.author) {
author = ` by ${this.options.author}`;
}
return 'Author: ' + author;
}
if (videojs__default['default'].browser.IS_IOS || videojs__default['default'].browser.IS_ANDROID) {
_this.mobileBuildUI();
}
}); // close the menu if open on userinactive
/**
* Add the menu ui button to the controlbar
*
* @return {string} Returns license text
*/
buildItemLicense() {
let license = '';
const reVersion = new RegExp('[0-9]+.[0-9]+$');
let version = '4.0';
const matches = this.options.license.match(reVersion);
if (matches !== null) {
version = matches[0];
}
const which = this.options.license.replace(reVersion, '').trim();
let deed = null;
switch (which) {
case 'CC0':
deed = 'https://creativecommons.org/licenses/zero/1.0/';
break;
case 'CC BY':
deed = `https://creativecommons.org/licenses/by/${version}/`;
break;
case 'CC BY-SA':
deed = `https://creativecommons.org/licenses/by-sa/${version}/`;
break;
case 'CC BY-NC':
deed = `https://creativecommons.org/licenses/by-nc/${version}/`;
break;
case 'CC BY-NC-SA':
deed = `https://creativecommons.org/licenses/by-nc-sa/${version}/`;
break;
case 'CC BY-ND':
deed = `https://creativecommons.org/licenses/by-nd/${version}/`;
break;
case 'CC BY-NC-ND':
deed = `https://creativecommons.org/licenses/by-nc-nd/${version}/`;
break;
}
if (deed) {
license = `<a href='${deed}' onclick='window.open('${deed}')' target='_blank' rel='noopener'>${this.options.license}</a>`;
} else {
license = this.options.license;
}
return 'License: ' + license;
}
/**
*
* Helper class to clear menu items before rebuild
*
* @param {*} className Name of a class
*/
removeElementsByClass(className) {
// Need to prevent the menu from not showing sometimes
document.querySelectorAll('.vjs-sm-top-level').forEach(element => {
element.classList.remove('vjs-hidden');
});
const elements = document.getElementsByClassName(className);
while (elements.length > 0) {
elements[0].parentNode.removeChild(elements[0]);
}
}
}
_this.player.on('userinactive', function () {
document__default['default'].getElementById(_this.playerId).querySelectorAll('.vjs-menu').forEach(function (element) {
element.classList.remove('vjs-lock-open');
});
}); // close the menu if anywhere in the player is clicked
// Define default values for the plugin's `state` object here.
License.defaultState = {};
// Include the version number.
License.VERSION = version;
_this.player.on('click', function (evt) {
if (evt.target.tagName === 'VIDEO') {
document__default['default'].getElementById(_this.playerId).querySelectorAll('.vjs-menu').forEach(function (element) {
element.classList.remove('vjs-lock-open');
});
}
});
// Register the plugin with video.js.
videojs__default["default"].registerPlugin('license', License);
_this.player.on('loadstart', function (_event) {
_this.removeElementsByClass('vjs-license-clear');
return License;
if (videojs__default['default'].browser.IS_IOS || videojs__default['default'].browser.IS_ANDROID) {
_this.mobileBuildTopLevelMenu();
} else {
_this.buildTopLevelMenu();
}
});
}));
return _this;
}
/**
* Add the menu ui button to the controlbar
*/
var _proto = License.prototype;
_proto.buildUI = function buildUI() {
var playerId = this.playerId;
var that = this;
/**
* LicenseMenuButton
*/
var LicenseMenuButton = /*#__PURE__*/function (_Button) {
inheritsLoose(LicenseMenuButton, _Button);
/**
* Contructor
*
* @param {*} player videojs player instance
* @param {*} options videojs player options
*/
function LicenseMenuButton(player, options) {
var _this2;
_this2 = _Button.call(this, player, options) || this;
_this2.addClass('vjs-license');
_this2.controlText(that.options.languages.loading);
player.one('canplaythrough', function (_event) {
_this2.controlText(that.options.languages.settings);
});
_this2.menu.contentEl_.id = playerId + '-vjs-license-default';
return _this2;
}
/**
* Handle click
*/
var _proto2 = LicenseMenuButton.prototype;
_proto2.handleClick = function handleClick() {
if (videojs__default['default'].browser.IS_IOS || videojs__default['default'].browser.IS_ANDROID) {
this.player.getChild('licenseMenuMobileModal').el().style.display = 'block';
} else {
this.el().classList.toggle('vjs-toogle-btn');
this.menu.el().classList.toggle('vjs-lock-open');
}
};
return LicenseMenuButton;
}(Button);
videojs__default['default'].registerComponent('licenseMenuButton', LicenseMenuButton);
this.player.getChild('controlBar').addChild('licenseMenuButton');
if (this.player.getChild('controlBar').getChild('fullscreenToggle')) {
this.player.getChild('controlBar').el().insertBefore(this.player.getChild('controlBar').getChild('licenseMenuButton').el(), this.player.getChild('controlBar').getChild('fullscreenToggle').el());
}
}
/**
*
* This is just build the top level menu no sub menus
*/
;
_proto.buildTopLevelMenu = function buildTopLevelMenu() {
var settingsButton = this.player.getChild('controlBar').getChild('licenseMenuButton'); // settingsButton.addClass('vjs-license-is-loaded');
var main = settingsButton.menu.contentEl_; // Empty the main menu div to repopulate
main.innerHTML = '';
main.classList.add('vjs-license-top-level'); // Start building new list items
var menuTitle = document__default['default'].createElement('li');
menuTitle.className = 'vjs-license-top-level-header';
var menuTitleInner = document__default['default'].createElement('span');
menuTitleInner.innerHTML = 'About';
menuTitleInner.className = 'vjs-license-top-level-header-titel';
menuTitle.appendChild(menuTitleInner);
main.appendChild(menuTitle);
var itemTitel = document__default['default'].createElement('li');
itemTitel.innerHTML = this.buildItemTitel();
itemTitel.className = 'vjs-license-top-level-item';
main.appendChild(itemTitel);
if (this.options.author) {
var itemAuthor = document__default['default'].createElement('li');
itemAuthor.innerHTML = this.buildItemAuthor();
itemAuthor.className = 'vjs-license-top-level-item';
main.appendChild(itemAuthor);
}
var itemLicense = document__default['default'].createElement('li');
itemLicense.innerHTML = this.buildItemLicense();
itemLicense.className = 'vjs-license-top-level-item';
main.appendChild(itemLicense);
}
/**
* Add the menu ui button to the controlbar
*/
;
_proto.mobileBuildUI = function mobileBuildUI() {
/**
* bla
*/
var LicenseMenuMobileModal = /*#__PURE__*/function (_Component) {
inheritsLoose(LicenseMenuMobileModal, _Component);
/**
* Contructor
*
* @param {*} player videojs player instance
* @param {*} options videojs player options
*/
function LicenseMenuMobileModal(player, options) {
return _Component.call(this, player, options) || this;
}
/**
* Creates an HTML element
*
* @return {Object} HTML element
*/
var _proto3 = LicenseMenuMobileModal.prototype;
_proto3.createEl = function createEl() {
return videojs__default['default'].createEl('div', {
className: 'vjs-license-mobile'
});
};
return LicenseMenuMobileModal;
}(Component);
videojs__default['default'].registerComponent('licenseMenuMobileModal', LicenseMenuMobileModal);
videojs__default['default'].dom.prependTo(this.player.addChild('licenseMenuMobileModal').el(), document__default['default'].body);
}
/**
* Add the menu ui button to the controlbar
*/
;
_proto.mobileBuildTopLevelMenu = function mobileBuildTopLevelMenu() {
var _this3 = this;
var settingsButton = this.player.getChild('licenseMenuMobileModal');
var menuTopLevel = document__default['default'].createElement('ul');
menuTopLevel.className = 'vjs-license-mob-top-level vjs-setting-menu-clear';
settingsButton.el().appendChild(menuTopLevel); // Empty the main menu div to repopulate
var menuTitle = document__default['default'].createElement('li');
menuTitle.innerHTML = 'About';
menuTitle.className = 'vjs-setting-menu-mobile-top-header';
menuTopLevel.appendChild(menuTitle);
var itemTitel = document__default['default'].createElement('li');
itemTitel.innerHTML = this.buildItemTitel();
itemTitel.className = 'vjs-license-top-level-item';
if (this.options.author) {
var itemAuthor = document__default['default'].createElement('li');
itemAuthor.innerHTML = this.buildItemAuthor();
itemAuthor.className = 'vjs-license-top-level-item';
}
var itemLicense = document__default['default'].createElement('li');
itemLicense.innerHTML = this.buildItemLicense();
itemLicense.className = 'vjs-license-top-level-item';
var menuClose = document__default['default'].createElement('li');
menuClose.innerHTML = 'Close';
menuClose.className = 'setting-menu-footer-default';
menuClose.onclick = function (e) {
_this3.player.getChild('settingsMenuMobileModal').el().style.display = 'none';
};
menuTopLevel.appendChild(menuClose);
}
/**
* Add the menu ui button to the controlbar
*
* @return {string} Returns license text
*/
;
_proto.buildItemTitel = function buildItemTitel() {
var titel = '';
if (this.options.title) {
titel = "" + this.options.title;
}
return 'Title: ' + titel;
}
/**
* Add the menu ui button to the controlbar
*
* @return {string} Returns license text
*/
;
_proto.buildItemAuthor = function buildItemAuthor() {
var author = '';
if (this.options.author) {
author = " by " + this.options.author;
}
return 'Author: ' + author;
}
/**
* Add the menu ui button to the controlbar
*
* @return {string} Returns license text
*/
;
_proto.buildItemLicense = function buildItemLicense() {
var license = '';
var reVersion = new RegExp('[0-9]+.[0-9]+$');
var version = '4.0';
var matches = this.options.license.match(reVersion);
if (matches !== null) {
version = matches[0];
}
var which = this.options.license.replace(reVersion, '').trim();
var deed = null;
switch (which) {
case 'CC0':
deed = 'https://creativecommons.org/licenses/zero/1.0/';
break;
case 'CC BY':
deed = "https://creativecommons.org/licenses/by/" + version + "/";
break;
case 'CC BY-SA':
deed = "https://creativecommons.org/licenses/by-sa/" + version + "/";
break;
case 'CC BY-NC':
deed = "https://creativecommons.org/licenses/by-nc/" + version + "/";
break;
case 'CC BY-NC-SA':
deed = "https://creativecommons.org/licenses/by-nc-sa/" + version + "/";
break;
case 'CC BY-ND':
deed = "https://creativecommons.org/licenses/by-nd/" + version + "/";
break;
case 'CC BY-NC-ND':
deed = "https://creativecommons.org/licenses/by-nc-nd/" + version + "/";
break;
}
if (deed) {
license = "<a href=\"" + deed + "\" onclick=\"window.open('" + deed + "')\" target=\"_blank\" rel=\"noopener\">" + this.options.license + "</a>";
} else {
license = this.options.license;
}
return 'License: ' + license;
}
/**
*
* Helper class to clear menu items before rebuild
*
* @param {*} className Name of a class
*/
;
_proto.removeElementsByClass = function removeElementsByClass(className) {
// Need to prevent the menu from not showing sometimes
document__default['default'].querySelectorAll('.vjs-sm-top-level').forEach(function (element) {
element.classList.remove('vjs-hidden');
});
var elements = document__default['default'].getElementsByClassName(className);
while (elements.length > 0) {
elements[0].parentNode.removeChild(elements[0]);
}
};
return License;
}(Plugin); // Define default values for the plugin's `state` object here.
License.defaultState = {}; // Include the version number.
License.VERSION = version; // Register the plugin with video.js.
videojs__default['default'].registerPlugin('license', License);
return License;
})));

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -82,14 +82,15 @@
liveui: true,
responsive: true,
fluid: true,
sources: [{ src: window.location.origin + '/' + playerConfig.source, type: 'application/x-mpegURL' }],
sources: [{ src: playerConfig.source, type: 'application/x-mpegURL' }],
plugins: {
license: playerConfig.license
}
};
if (playerConfig.chromecast) {
config.techOrder = ["chromecast", "html5"];
config.plugins.chromecast = {
receiverApplicationId: 'CC1AD845'
};
config.plugins.chromecast = {};
}
if (playerConfig.airplay) {
@ -129,8 +130,6 @@
},
],
});
player.license(playerConfig.license);
}
if (autoplay === true) {

View File

@ -6,32 +6,25 @@ var config = {
liveui: true,
responsive: true,
fluid: true,
// Needed to append the url origin in order for the source to properly pass to the cast device
sources: [{ src: window.location.origin + '/' + playerConfig.source, type: 'application/x-mpegURL' }],
plugins: {},
// Needed to append the url orgin in order for the source to properly pass to the cast device
sources: [{ src: playerConfig.source, type: 'application/x-mpegURL' }],
plugins: {
license: playerConfig.license,
},
};
if (chromecast) {
config.techOrder = ['chromecast', 'html5'];
// Provide a default reciever application ID
config.plugins.chromecast = {
receiverApplicationId: 'CC1AD845',
};
config.plugins.chromecast = {};
}
if (airplay) {
config.plugins.airPlay = {};
}
var player = videojs('player', config);
player.ready(function () {
if (chromecast) {
player.chromecast();
}
if (airplay) {
player.airPlay();
}
player.license(playerConfig.license);
if (playerConfig.logo.image.length != 0) {
var overlay = null;

View File

@ -2,6 +2,7 @@ import React from 'react';
import { I18nProvider } from '@lingui/react';
import { i18n } from '@lingui/core';
import * as plurals from 'make-plural/plurals';
import { messages as EN } from './locales/en/messages.js';
import { messages as DA } from './locales/da/messages.js';
@ -12,7 +13,7 @@ import { messages as FR } from './locales/fr/messages.js';
import { messages as IT } from './locales/it/messages.js';
import { messages as KO } from './locales/ko/messages.js';
import { messages as PL } from './locales/pl/messages.js';
import { messages as PT } from './locales/pt-br/messages.js';
import { messages as PT } from './locales/pt/messages.js';
import { messages as RU } from './locales/ru/messages.js';
import { messages as SL } from './locales/sl/messages.js';
import { messages as TR } from './locales/tr/messages.js';
@ -20,6 +21,21 @@ import { messages as UK } from './locales/uk/messages.js';
import { messages as ZH } from './locales/zh-hans/messages.js';
import * as Storage from './utils/storage';
i18n.loadLocaleData('en', { plurals: plurals.en });
i18n.loadLocaleData('da', { plurals: plurals.da });
i18n.loadLocaleData('de', { plurals: plurals.de });
i18n.loadLocaleData('el', { plurals: plurals.el });
i18n.loadLocaleData('es', { plurals: plurals.es });
i18n.loadLocaleData('fr', { plurals: plurals.fr });
i18n.loadLocaleData('it', { plurals: plurals.it });
i18n.loadLocaleData('ko', { plurals: plurals.ko });
i18n.loadLocaleData('pl', { plurals: plurals.pl });
i18n.loadLocaleData('pt', { plurals: plurals.pt });
i18n.loadLocaleData('ru', { plurals: plurals.ru });
i18n.loadLocaleData('sl', { plurals: plurals.sl });
i18n.loadLocaleData('tr', { plurals: plurals.tr });
i18n.loadLocaleData('uk', { plurals: plurals.tr });
i18n.loadLocaleData('zh-hans', { plurals: plurals.zh });
i18n.load({
en: EN,
da: DA,
@ -30,7 +46,7 @@ i18n.load({
it: IT,
ko: KO,
pl: PL,
'pt-br': PT,
pt: PT,
ru: RU,
sl: SL,
tr: TR,
@ -39,7 +55,7 @@ i18n.load({
});
const aliases = {
pt: 'pt-br',
'pt-br': 'pt',
'zh-cn': 'zh-hans',
};
@ -77,7 +93,7 @@ const getBrowserLanguage = (defaultLanguage) => {
return match[0].toLowerCase();
};
i18n.activate(getLanguage('en', ['en', 'da', 'de', 'el', 'es', 'fr', 'it', 'ko', 'pl', 'pt-br', 'ru', 'sl', 'tr', 'uk', 'zh-hans']));
i18n.activate(getLanguage('en', ['en', 'da', 'de', 'el', 'es', 'fr', 'it', 'ko', 'pl', 'pt', 'ru', 'sl', 'tr', 'uk', 'zh-hans']));
export default function Provider(props) {
return <I18nProvider i18n={i18n}>{props.children}</I18nProvider>;

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

View File

@ -155,14 +155,14 @@ export default function EncodingSelect(props) {
encoderList.push(
<MenuItem value={c.coder} key={c.coder}>
{c.name}
</MenuItem>,
</MenuItem>
);
}
} else {
encoderList.push(
<MenuItem value={c.coder} key={c.coder}>
{c.name}
</MenuItem>,
</MenuItem>
);
}
}
@ -195,7 +195,7 @@ export default function EncodingSelect(props) {
decoderList.push(
<MenuItem value={c.coder} key={c.coder}>
{c.name}
</MenuItem>,
</MenuItem>
);
}
}

View File

@ -91,7 +91,7 @@ export default function FilterSelect(props) {
}
filterSettings.push(
<Settings key={c.filter} settings={profile.filter.settings[c.filter].settings} onChange={handleFilterSettingsChange(c.filter)} />,
<Settings key={c.filter} settings={profile.filter.settings[c.filter].settings} onChange={handleFilterSettingsChange(c.filter)} />
);
}
}

View File

@ -15,9 +15,16 @@ const MenuProps = {
export default function Component(props) {
return (
<FormControl variant={props.variant} disabled={props.disabled} fullWidth>
<FormControl variant="outlined" fullWidth>
<InputLabel>{props.label}</InputLabel>
<Select multiple value={props.value} onChange={props.onChange} input={<OutlinedInput />} renderValue={props.renderValue} MenuProps={MenuProps}>
<Select
multiple
value={props.value}
onChange={props.onChange}
input={<OutlinedInput />}
renderValue={(selected) => selected.join(', ')}
MenuProps={MenuProps}
>
{props.children}
</Select>
</FormControl>
@ -25,10 +32,7 @@ export default function Component(props) {
}
Component.defaultProps = {
variant: 'outlined',
label: '',
value: [],
disabled: false,
renderValue: (selected) => selected.join(', '),
onChange: function (event) {},
};

View File

@ -41,11 +41,7 @@ export default function Component(props) {
isCustom: v === props.customKey ? true : false,
});
props.onChange({
target: {
value: v === props.customKey ? value.custom : value.value,
},
});
props.onChange(event);
};
const handleCustomChange = (event) => {
@ -63,7 +59,7 @@ export default function Component(props) {
options.push(
<MenuItem key={o.value} value={o.value} disabled={o.disabled === true}>
{o.label}
</MenuItem>,
</MenuItem>
);
}

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: [],
filter: [],
};
return mapping;

View File

@ -1,22 +1,21 @@
import * as AudioDefault from './audio/default';
import * as AudioDefault from './audio/Default';
import * as NVDEC from './video/nvdec';
import * as H264MMAL from './video/h264_mmal';
import * as H264CUVID from './video/h264_cuvid';
import * as HEVCCUVID from './video/hevc_cuvid';
import * as MJPEGCUVID from './video/mjpeg_cuvid';
import * as MPEG1CUVID from './video/mpeg1_cuvid';
import * as MPEG2CUVID from './video/mpeg2_cuvid';
import * as MPEG2MMAL from './video/mpeg2_mmal';
import * as MPEG4CUVID from './video/mpeg4_cuvid';
import * as MPEG4MMAL from './video/mpeg4_mmal';
import * as VC1CUVID from './video/vc1_cuvid';
import * as VC1MMAL from './video/vc1_mmal';
import * as VideoDefault from './video/default';
import * as VideoToolbox from './video/videotoolbox';
import * as VP8CUVID from './video/vp8_cuvid';
import * as VP9CUVID from './video/vp9_cuvid';
import * as AV1CUVID from './video/av1_cuvid';
import * as NVDEC from './video/NVDEC';
import * as H264MMAL from './video/H264MMAL';
import * as H264CUVID from './video/H264CUVID';
import * as HEVCCUVID from './video/HEVCCUVID';
import * as MJPEGCUVID from './video/MJPEGCUVID';
import * as MPEG1CUVID from './video/MPEG1CUVID';
import * as MPEG2CUVID from './video/MPEG2CUVID';
import * as MPEG2MMAL from './video/MPEG2MMAL';
import * as MPEG4CUVID from './video/MPEG4CUVID';
import * as MPEG4MMAL from './video/MPEG4MMAL';
import * as VC1CUVID from './video/VC1CUVID';
import * as VC1MMAL from './video/VC1MMAL';
import * as VideoDefault from './video/Default';
import * as VideoToolbox from './video/VideoToolbox';
import * as VP8CUVID from './video/VP8CUVID';
import * as VP9CUVID from './video/VP9CUVID';
class Registry {
constructor(type) {
@ -113,7 +112,6 @@ const videoRegistry = new Registry('video');
videoRegistry.Register(VideoDefault);
videoRegistry.Register(VideoToolbox);
videoRegistry.Register(AV1CUVID);
videoRegistry.Register(NVDEC);
videoRegistry.Register(H264MMAL);
videoRegistry.Register(H264CUVID);

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: [],
filter: [],
};
return mapping;

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: ['-c:v', 'h264_cuvid'],
filter: [],
};
return mapping;

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: ['-c:v', 'h264_mmal'],
filter: [],
};
return mapping;

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: ['-c:v', 'hevc_cuvid'],
filter: [],
};
return mapping;

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: ['-c:v', 'mjpeg_cuvid'],
filter: [],
};
return mapping;

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: ['-c:v', 'mpeg1_cuvid'],
filter: [],
};
return mapping;

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: ['-c:v', 'mpeg2_cuvid'],
filter: [],
};
return mapping;

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: ['-c:v', 'mpeg2_mmal'],
filter: [],
};
return mapping;

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: ['-c:v', 'mpeg4_cuvid'],
filter: [],
};
return mapping;

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: ['-c:v', 'mpeg4_mmal'],
filter: [],
};
return mapping;

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: ['-hwaccel', 'cuda', '-hwaccel_output_format', 'cuda'],
filter: [],
};
return mapping;

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: ['-c:v', 'vc1_cuvid'],
filter: [],
};
return mapping;

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: ['-c:v', 'vc1_mmal'],
filter: [],
};
return mapping;

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: ['-c:v', 'vp8_cuvid'],
filter: [],
};
return mapping;

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: ['-c:v', 'vp9_cuvid'],
filter: [],
};
return mapping;

View File

@ -17,7 +17,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: ['-hwaccel', 'videotoolbox'],
filter: [],
};
return mapping;
@ -54,8 +53,8 @@ Coder.defaultProps = {
};
const coder = 'videotoolbox';
const name = 'VideoToolbox';
const codecs = ['h264', 'hevc', 'vp9', 'mpeg1', 'mpeg2', 'mpeg4'];
const name = 'H264 (VideoToolbox)';
const codecs = ['h264'];
const type = 'video';
const hwaccel = true;

View File

@ -1,73 +0,0 @@
import React from 'react';
import Helper from '../../helper';
function init(initialState) {
const state = {
...initialState,
};
return state;
}
function createMapping(settings, stream, skills) {
stream = Helper.InitStream(stream);
skills = Helper.InitSkills(skills);
const mapping = {
global: [],
local: ['-hwaccel', 'cuda', '-c:v', 'av1_cuvid'],
filter: [],
};
return mapping;
}
function Coder(props) {
const settings = init(props.settings);
const stream = Helper.InitStream(props.stream);
const skills = Helper.InitSkills(props.skills);
const handleChange = (newSettings) => {
let automatic = false;
if (!newSettings) {
newSettings = settings;
automatic = true;
}
props.onChange(newSettings, createMapping(newSettings, stream, skills), automatic);
};
React.useEffect(() => {
handleChange(null);
// eslint-disable-next-line react-hooks/exhaustive-deps
}, []);
return null;
}
Coder.defaultProps = {
stream: {},
settings: {},
skills: {},
onChange: function (settings, mapping) {},
};
// -hwaccel nvdec
const coder = 'av1_cuvid';
const name = 'AV1 (CUVID)';
const codecs = ['av1'];
const type = 'video';
const hwaccel = true;
function defaults(stream, skills) {
const settings = init({});
return {
settings: settings,
mapping: createMapping(settings, stream, skills),
};
}
export { coder, name, codecs, type, hwaccel, defaults, Coder as component };

View File

@ -27,7 +27,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: local,
filter: [],
};
return mapping;

View File

@ -27,7 +27,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [['-vsync', 'drop']],
local: local,
filter: [],
};
return mapping;

View File

@ -15,7 +15,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: local,
filter: [],
};
return mapping;

View File

@ -23,7 +23,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: local,
filter: [],
};
return mapping;

View File

@ -23,7 +23,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [['-vsync', 'drop']],
local: local,
filter: [],
};
return mapping;

View File

@ -24,7 +24,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [['-vsync', 'drop']],
local: local,
filter: [],
};
return mapping;

View File

@ -11,7 +11,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: local,
filter: [],
};
return mapping;

View File

@ -7,7 +7,7 @@ import { useLingui } from '@lingui/react';
import { Trans, t } from '@lingui/macro';
import Audio from '../../settings/Audio';
import SelectCustom from '../../../SelectCustom';
import SelectCustom from '../../../../misc/SelectCustom';
import Helper from '../../helper';
function init(initialState) {
@ -44,7 +44,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [['-vsync', 'drop']],
local: local,
filter: [],
};
return mapping;

View File

@ -23,7 +23,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [['-vsync', 'drop']],
local: local,
filter: [],
};
return mapping;

View File

@ -1,28 +1,26 @@
import * as AudioCopy from './audio/copy';
import * as AudioNone from './audio/none';
import * as AAC from './audio/aac';
import * as AACAudioToolbox from './audio/aac_audiotoolbox';
import * as Libopus from './audio/opus_libopus';
import * as Libvorbis from './audio/vorbis_libvorbis';
import * as MP3 from './audio/mp3_libmp3lame';
import * as Opus from './audio/opus';
import * as Vorbis from './audio/vorbis';
import * as AudioCopy from './audio/Copy';
import * as AudioNone from './audio/None';
import * as AAC from './audio/AAC';
import * as AACAudioToolbox from './audio/AACAudioToolbox';
import * as Libopus from './audio/Libopus';
import * as Libvorbis from './audio/Libvorbis';
import * as MP3 from './audio/MP3';
import * as Opus from './audio/Opus';
import * as Vorbis from './audio/Vorbis';
import * as X264 from './video/h264_libx264';
import * as X265 from './video/hevc_libx265';
import * as H264VideoToolbox from './video/h264_videotoolbox';
import * as H264NVENC from './video/h264_nvenc';
import * as H264OMX from './video/h264_omx';
import * as H264V4L2M2M from './video/h264_v4l2m2m';
import * as H264VAAPI from './video/h264_vaapi';
import * as HEVCVAAPI from './video/hevc_vaapi';
import * as HEVCVideoToolbox from './video/hevc_videotoolbox';
import * as VP9VAAPI from './video/vp9_vaapi';
import * as VideoCopy from './video/copy';
import * as VideoNone from './video/none';
import * as VideoRaw from './video/rawvideo';
import * as VP9 from './video/vp9_libvpx';
import * as AV1Rav1e from './video/av1_librav1e';
import * as X264 from './video/X264';
import * as X265 from './video/X265';
import * as H264VideoToolbox from './video/H264VideoToolbox';
import * as H264NVENC from './video/H264NVENC';
import * as H264OMX from './video/H264OMX';
import * as H264V4L2M2M from './video/H264V4L2M2M';
import * as H264VAAPI from './video/H264VAAPI';
import * as HEVCVAAPI from './video/HEVCVAAPI';
import * as VP9VAAPI from './video/VP9VAAPI';
import * as VideoCopy from './video/Copy';
import * as VideoNone from './video/None';
import * as VideoRaw from './video/Raw';
import * as VP9 from './video/VP9';
class Registry {
constructor(type) {
@ -125,17 +123,15 @@ const videoRegistry = new Registry('video');
videoRegistry.Register(VideoCopy);
videoRegistry.Register(VideoNone);
videoRegistry.Register(X264);
videoRegistry.Register(X265);
videoRegistry.Register(H264VideoToolbox);
videoRegistry.Register(H264NVENC);
videoRegistry.Register(H264OMX);
videoRegistry.Register(H264V4L2M2M);
videoRegistry.Register(H264VAAPI);
videoRegistry.Register(X265);
videoRegistry.Register(HEVCVAAPI);
videoRegistry.Register(HEVCVideoToolbox);
videoRegistry.Register(VP9VAAPI);
videoRegistry.Register(VP9);
videoRegistry.Register(AV1Rav1e);
videoRegistry.Register(VideoRaw);
export { audioRegistry as Audio, videoRegistry as Video };

View File

@ -11,7 +11,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: local,
filter: [],
};
return mapping;

View File

@ -64,7 +64,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: local,
filter: [],
};
return mapping;

View File

@ -47,7 +47,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: local,
filter: [],
};
return mapping;

View File

@ -1,4 +1,5 @@
import React from 'react';
import SemverSatisfies from 'semver/functions/satisfies';
import Grid from '@mui/material/Grid';
import Typography from '@mui/material/Typography';
@ -79,6 +80,10 @@ function createMapping(settings, stream, skills) {
stream = Helper.InitStream(stream);
skills = Helper.InitSkills(skills);
let ffversion = 4;
if (SemverSatisfies(skills.ffmpeg.version, '^5.0.0')) {
ffversion = 5;
}
const local = [
'-codec:v',
'h264_v4l2m2m',
@ -114,7 +119,7 @@ function createMapping(settings, stream, skills) {
local.push('-keyint_min', `${parseInt(settings.fps)}`);
}
if (skills.ffmpeg.version_major >= 5) {
if (ffversion === 5) {
local.push('-fps_mode', `${settings.fps_mode}`);
}
@ -125,7 +130,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: local,
filter: [],
};
return mapping;
@ -136,6 +140,11 @@ function Coder(props) {
const stream = Helper.InitStream(props.stream);
const skills = Helper.InitSkills(props.skills);
let ffversion = 4;
if (SemverSatisfies(skills.ffmpeg.version, '^5.0.0')) {
ffversion = 5;
}
const handleChange = (newSettings) => {
let automatic = false;
if (!newSettings) {
@ -181,7 +190,7 @@ function Coder(props) {
<Trans>To stabilize the system, increase the HLS segment length for the keyframe interval by 2-3 * (Processing and Control).</Trans>
</Typography>
</Grid>
{skills.ffmpeg.version_major >= 5 && (
{ffversion === 5 && (
<Grid item xs={12}>
<Video.FpsMode value={settings.fps_mode} onChange={update('fps_mode')} />
</Grid>

View File

@ -30,12 +30,13 @@ function createMapping(settings, stream, skills) {
const global = [];
const local = [];
const filter = [];
// https://trac.ffmpeg.org/wiki/Hardware/VAAPI
global.push(['-vaapi_device', '/dev/dri/renderD128']);
local.push(
'-vf',
'format=nv12,hwupload',
'-codec:v',
'h264_vaapi',
'-profile:v',
@ -50,18 +51,17 @@ function createMapping(settings, stream, skills) {
`${settings.bitrate}k`,
'-r',
`${settings.fps}`,
'-g',
`${settings.gop}`
);
if (settings.gop !== 'auto') {
local.push('-g', `${Math.round(parseInt(settings.fps) * parseInt(settings.gop)).toFixed(0)}`);
}
filter.push('format=nv12', 'hwupload');
return {
global: global,
local: local,
filter: filter,
};
}

View File

@ -58,7 +58,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: local,
filter: [],
};
return mapping;

View File

@ -30,12 +30,13 @@ function createMapping(settings, stream, skills) {
const global = [];
const local = [];
const filter = [];
// https://trac.ffmpeg.org/wiki/Hardware/VAAPI
global.push(['-vaapi_device', '/dev/dri/renderD128']);
local.push(
'-vf',
'format=nv12,hwupload',
'-codec:v',
'hevc_vaapi',
'-profile:v',
@ -51,19 +52,16 @@ function createMapping(settings, stream, skills) {
'-r',
`${settings.fps}`,
'-g',
`${settings.gop}`,
`${settings.gop}`
);
if (settings.gop !== 'auto') {
local.push('-g', `${Math.round(parseInt(settings.fps) * parseInt(settings.gop)).toFixed(0)}`);
}
filter.push('format=nv12', 'hwupload');
return {
global: global,
local: local,
filter: filter,
};
}

View File

@ -11,7 +11,6 @@ function createMapping(settings, stream, skills) {
const mapping = {
global: [],
local: local,
filter: [],
};
return mapping;

Some files were not shown because too many files have changed in this diff Show More