556 lines
38 KiB
JavaScript
556 lines
38 KiB
JavaScript
/******/ (() => { // webpackBootstrap
|
|
/******/ "use strict";
|
|
/******/ var __webpack_modules__ = ({
|
|
|
|
/***/ "./src/css/main.css":
|
|
/*!**************************!*\
|
|
!*** ./src/css/main.css ***!
|
|
\**************************/
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
// extracted by mini-css-extract-plugin
|
|
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./src/js/descriptor/FileDescriptor.js":
|
|
/*!*********************************************!*\
|
|
!*** ./src/js/descriptor/FileDescriptor.js ***!
|
|
\*********************************************/
|
|
/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => {
|
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
/* harmony export */ "default": () => (/* binding */ FileDescriptor)
|
|
/* harmony export */ });
|
|
class FileDescriptor {
|
|
/**
|
|
* @type {string}
|
|
*/
|
|
path = "";
|
|
|
|
/**
|
|
* @type {FileSystemFileHandle}
|
|
*/
|
|
handle = "";
|
|
|
|
/**
|
|
* @param {string} path
|
|
* @param {FileSystemFileHandle} handle
|
|
*/
|
|
constructor(path, handle){
|
|
this.path = path;
|
|
this.handle = handle;
|
|
}
|
|
}
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./src/js/profile/MediaProfile.js":
|
|
/*!****************************************!*\
|
|
!*** ./src/js/profile/MediaProfile.js ***!
|
|
\****************************************/
|
|
/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => {
|
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
/* harmony export */ "default": () => (/* binding */ MediaProfile)
|
|
/* harmony export */ });
|
|
/* harmony import */ var _MediaProfileRole_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./MediaProfileRole.js */ "./src/js/profile/MediaProfileRole.js");
|
|
/* harmony import */ var _MediaProfileSettings_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./MediaProfileSettings.js */ "./src/js/profile/MediaProfileSettings.js");
|
|
/* harmony import */ var _MediaProfileViewSettings_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./MediaProfileViewSettings.js */ "./src/js/profile/MediaProfileViewSettings.js");
|
|
|
|
|
|
|
|
|
|
class MediaProfile{
|
|
/**
|
|
* @type {string}
|
|
*/
|
|
name = "";
|
|
|
|
/**
|
|
* @type {string}
|
|
*/
|
|
role = _MediaProfileRole_js__WEBPACK_IMPORTED_MODULE_0__["default"].ADMIN;
|
|
|
|
/**
|
|
* @type {MediaProfileSettings}
|
|
*/
|
|
settings = new _MediaProfileSettings_js__WEBPACK_IMPORTED_MODULE_1__["default"]();
|
|
|
|
/**
|
|
* @type {MediaProfileViewSettings}
|
|
*/
|
|
playSettings = new _MediaProfileViewSettings_js__WEBPACK_IMPORTED_MODULE_2__["default"]();
|
|
|
|
/**
|
|
* @param {object} jsonObj
|
|
* @returns {MediaProfile}
|
|
*/
|
|
static fromJson(jsonObj) {
|
|
const profile = new MediaProfile();
|
|
|
|
if (jsonObj == null) {
|
|
return profile;
|
|
}
|
|
|
|
if (jsonObj.name != null) {
|
|
profile.name = jsonObj.name;
|
|
}
|
|
|
|
if (jsonObj.role != null) {
|
|
profile.role = jsonObj.role;
|
|
}
|
|
|
|
if (jsonObj.settings != null && typeof jsonObj.settings === "object") {
|
|
profile.settings = _MediaProfileSettings_js__WEBPACK_IMPORTED_MODULE_1__["default"].fromSettingsJson(jsonObj.settings);
|
|
}
|
|
|
|
if (jsonObj.playSettings != null && typeof jsonObj.playSettings === "object") {
|
|
profile.playSettings = _MediaProfileViewSettings_js__WEBPACK_IMPORTED_MODULE_2__["default"].fromSettingsJson(jsonObj.playSettings);
|
|
}
|
|
|
|
return profile;
|
|
}
|
|
}
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./src/js/profile/MediaProfileRole.js":
|
|
/*!********************************************!*\
|
|
!*** ./src/js/profile/MediaProfileRole.js ***!
|
|
\********************************************/
|
|
/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => {
|
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
/* harmony export */ "default": () => (/* binding */ MediaProfileRole)
|
|
/* harmony export */ });
|
|
class MediaProfileRole {
|
|
/**
|
|
* @type {string}
|
|
*/
|
|
static ADMIN = "ADMIN";
|
|
|
|
/**
|
|
* @type {string}
|
|
*/
|
|
static USER = "USER";
|
|
}
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./src/js/profile/MediaProfileSettings.js":
|
|
/*!************************************************!*\
|
|
!*** ./src/js/profile/MediaProfileSettings.js ***!
|
|
\************************************************/
|
|
/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => {
|
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
/* harmony export */ "default": () => (/* binding */ MediaProfileSettings)
|
|
/* harmony export */ });
|
|
class MediaProfileSettings {
|
|
/**
|
|
* @type {boolean}
|
|
*/
|
|
skipIntro = true;
|
|
|
|
/**
|
|
* @type {boolean}
|
|
*/
|
|
skipOutro = true;
|
|
|
|
/**
|
|
* @type {boolean}
|
|
*/
|
|
skipRecall = true;
|
|
|
|
/**
|
|
* @type {boolean}
|
|
*/
|
|
skipPreview = true;
|
|
|
|
/**
|
|
* @param {object} jsonObj
|
|
* @returns {MediaProfileSettings}
|
|
*/
|
|
static fromSettingsJson(jsonObj) {
|
|
const settings = new MediaProfileSettings();
|
|
|
|
if (jsonObj == null) {
|
|
return settings;
|
|
}
|
|
|
|
if (jsonObj.skipIntro != null) {
|
|
settings.skipIntro = jsonObj.skipIntro;
|
|
}
|
|
|
|
if (jsonObj.skipOutro != null) {
|
|
settings.skipOutro = jsonObj.skipOutro;
|
|
}
|
|
|
|
if (jsonObj.skipRecall != null) {
|
|
settings.skipRecall = jsonObj.skipRecall;
|
|
}
|
|
|
|
if (jsonObj.skipPreview != null) {
|
|
settings.skipPreview = jsonObj.skipPreview;
|
|
}
|
|
|
|
return settings;
|
|
}
|
|
}
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./src/js/profile/MediaProfileViewSettings.js":
|
|
/*!****************************************************!*\
|
|
!*** ./src/js/profile/MediaProfileViewSettings.js ***!
|
|
\****************************************************/
|
|
/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => {
|
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
/* harmony export */ "default": () => (/* binding */ MediaProfileViewSettings)
|
|
/* harmony export */ });
|
|
class MediaProfileViewSettings {
|
|
/**
|
|
* @type {boolean}
|
|
*/
|
|
hasOpenPlaylist = false;
|
|
|
|
/**
|
|
* @type {string}
|
|
*/
|
|
playlistPath = null;
|
|
|
|
/**
|
|
* @type {number}
|
|
*/
|
|
playlistTrackNumber = 0;
|
|
|
|
/**
|
|
* @type {number}
|
|
*/
|
|
trackPosition = 0;
|
|
|
|
/**
|
|
* @param {object} jsonObj
|
|
* @returns {MediaProfileViewSettings}
|
|
*/
|
|
static fromSettingsJson(jsonObj) {
|
|
const settings = new MediaProfileViewSettings();
|
|
|
|
if (jsonObj == null) {
|
|
return settings;
|
|
}
|
|
|
|
if (jsonObj.hasOpenPlaylist != null) {
|
|
settings.hasOpenPlaylist = jsonObj.hasOpenPlaylist;
|
|
}
|
|
|
|
if (jsonObj.playlistPath != null) {
|
|
settings.playlistPath = jsonObj.playlistPath;
|
|
}
|
|
|
|
if (jsonObj.playlistTrackNumber != null) {
|
|
settings.playlistTrackNumber = jsonObj.playlistTrackNumber;
|
|
}
|
|
|
|
if (jsonObj.trackPosition != null) {
|
|
settings.trackPosition = jsonObj.trackPosition;
|
|
}
|
|
|
|
return settings;
|
|
}
|
|
}
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./src/js/util/FSUtil.js":
|
|
/*!*******************************!*\
|
|
!*** ./src/js/util/FSUtil.js ***!
|
|
\*******************************/
|
|
/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => {
|
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
/* harmony export */ "default": () => (/* binding */ FSUtil)
|
|
/* harmony export */ });
|
|
/* harmony import */ var _descriptor_FileDescriptor_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../descriptor/FileDescriptor.js */ "./src/js/descriptor/FileDescriptor.js");
|
|
|
|
|
|
class FSUtil {
|
|
/**
|
|
* @param {FileSystemDirectoryHandle} rootDir
|
|
* @param {string} path
|
|
* @returns {Promise<string | null>}
|
|
*/
|
|
static async getFileContent(rootDir, path) {
|
|
const fileHandle = await FSUtil.getFileHandle(rootDir, FSUtil.fixPath(path));
|
|
return await FSUtil.getFileHandleFileContent(fileHandle);
|
|
}
|
|
|
|
/**
|
|
* @param {FileSystemFileHandle} handle
|
|
* @returns {Promise<string>}
|
|
*/
|
|
static async getFileHandleFileContent(handle) {
|
|
if (handle == null) {
|
|
return null;
|
|
}
|
|
|
|
const file = await handle.getFile();
|
|
return await file.text();
|
|
}
|
|
|
|
/**
|
|
* @param {FileSystemDirectoryHandle} rootDir
|
|
* @param {string} path
|
|
* @returns {Promise<FileSystemFileHandle | null>}
|
|
*/
|
|
static async getFileHandle(rootDir, path) {
|
|
let resultDir = rootDir;
|
|
let outputFile = null;
|
|
const pathes = FSUtil.fixPath(path).split("/");
|
|
let nextDirName = null;
|
|
|
|
while (nextDirName = pathes.shift()) {
|
|
let nextDir = null;
|
|
|
|
for await (const value of resultDir.values()) {
|
|
if (value.kind == 'directory' && value.name == nextDirName) {
|
|
nextDir = value;
|
|
} else if (value.kind == 'file' && value.name == nextDirName) {
|
|
outputFile = value;
|
|
}
|
|
}
|
|
|
|
if (nextDir != null) {
|
|
resultDir = nextDir;
|
|
}
|
|
}
|
|
|
|
return outputFile;
|
|
}
|
|
|
|
/**
|
|
* @param {FileSystemDirectoryHandle} rootDir
|
|
* @param {string} path
|
|
* @returns {Promise<FileSystemDirectoryHandle | null>}
|
|
*/
|
|
static async getDirectoryHandle(curDir, path) {
|
|
let resultDir = curDir;
|
|
const pathes = FSUtil.fixPath(path).split("/");
|
|
let nextDirName = null;
|
|
|
|
while (nextDirName = pathes.shift()) {
|
|
let nextDir = null;
|
|
|
|
for await (const value of resultDir.values()) {
|
|
if (value.kind == 'directory' && value.name == nextDirName) {
|
|
nextDir = value;
|
|
}
|
|
}
|
|
|
|
if (nextDir != null) {
|
|
resultDir = nextDir;
|
|
}
|
|
}
|
|
|
|
return resultDir;
|
|
}
|
|
|
|
/**
|
|
* @param {string} path
|
|
* @returns {string}
|
|
*/
|
|
static fixPath(path) {
|
|
let fixedPath = path.replaceAll("/./", "/"); //remove same folder subpathes
|
|
fixedPath = fixedPath.replaceAll("//", "/"); //remove double slashes
|
|
|
|
if (fixedPath.startsWith("/")) {
|
|
fixedPath = fixedPath.substring(1);
|
|
}
|
|
|
|
const pathParts = fixedPath.split("/");
|
|
for (let i = pathParts.length - 1; i > 0; i--) {
|
|
if (pathParts[i] == "..") {
|
|
pathParts.splice(i - 1, 2);
|
|
i--;
|
|
}
|
|
}
|
|
|
|
return pathParts.join("/");
|
|
}
|
|
|
|
/**
|
|
* @param {FileSystemDirectoryHandle} dir
|
|
* @param {string} fileName
|
|
* @param {boolean} includeSubDirs
|
|
* @returns {Promise<FileDescriptor>}
|
|
*/
|
|
static async findFile(dir, fileName, includeSubDirs = true) {
|
|
const files = await FSUtil.#findFilesInTree(dir, fileName, includeSubDirs, true);
|
|
return files.length > 0 ? files[0] : null;
|
|
}
|
|
|
|
/**
|
|
* @param {FileSystemDirectoryHandle} dir
|
|
* @param {string} fileName
|
|
* @param {boolean} includeSubDirs
|
|
* @returns {Promise<FileDescriptor[]>}
|
|
*/
|
|
static async findFiles(dir, fileName, includeSubDirs = true) {
|
|
return FSUtil.#findFilesInTree(dir, fileName, includeSubDirs, false);;
|
|
}
|
|
|
|
/**
|
|
* @param {FileSystemDirectoryHandle} dir
|
|
* @param {string} fileName
|
|
* @param {boolean} includeSubDirs
|
|
* @param {boolean} stopOnFirst
|
|
* @returns {Promise<FileDescriptor[]>}
|
|
*/
|
|
static async #findFilesInTree(dir, fileName, includeSubDirs = true, stopOnFirst = false, path = "") {
|
|
const files = [];
|
|
const subdirs = [];
|
|
|
|
for await (const child of dir.values()) {
|
|
if (child.kind == 'directory' && includeSubDirs) {
|
|
subdirs.push(child);
|
|
} else if (child.kind == 'file' && child.name == fileName) {
|
|
files.push(new _descriptor_FileDescriptor_js__WEBPACK_IMPORTED_MODULE_0__["default"](path, child));
|
|
if(stopOnFirst){
|
|
return files;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (let i = 0; i < subdirs.length; i++) {
|
|
files.push(...(await FSUtil.#findFilesInTree(subdirs[i], fileName, includeSubDirs, stopOnFirst, path + "/" + subdirs[i].name)));
|
|
if(stopOnFirst && files.length > 0){
|
|
break;
|
|
}
|
|
}
|
|
|
|
return files;
|
|
}
|
|
}
|
|
|
|
/***/ })
|
|
|
|
/******/ });
|
|
/************************************************************************/
|
|
/******/ // The module cache
|
|
/******/ var __webpack_module_cache__ = {};
|
|
/******/
|
|
/******/ // The require function
|
|
/******/ function __webpack_require__(moduleId) {
|
|
/******/ // Check if module is in cache
|
|
/******/ var cachedModule = __webpack_module_cache__[moduleId];
|
|
/******/ if (cachedModule !== undefined) {
|
|
/******/ return cachedModule.exports;
|
|
/******/ }
|
|
/******/ // Create a new module (and put it into the cache)
|
|
/******/ var module = __webpack_module_cache__[moduleId] = {
|
|
/******/ // no module.id needed
|
|
/******/ // no module.loaded needed
|
|
/******/ exports: {}
|
|
/******/ };
|
|
/******/
|
|
/******/ // Execute the module function
|
|
/******/ __webpack_modules__[moduleId](module, module.exports, __webpack_require__);
|
|
/******/
|
|
/******/ // Return the exports of the module
|
|
/******/ return module.exports;
|
|
/******/ }
|
|
/******/
|
|
/************************************************************************/
|
|
/******/ /* webpack/runtime/define property getters */
|
|
/******/ (() => {
|
|
/******/ // define getter functions for harmony exports
|
|
/******/ __webpack_require__.d = (exports, definition) => {
|
|
/******/ for(var key in definition) {
|
|
/******/ if(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {
|
|
/******/ Object.defineProperty(exports, key, { enumerable: true, get: definition[key] });
|
|
/******/ }
|
|
/******/ }
|
|
/******/ };
|
|
/******/ })();
|
|
/******/
|
|
/******/ /* webpack/runtime/hasOwnProperty shorthand */
|
|
/******/ (() => {
|
|
/******/ __webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))
|
|
/******/ })();
|
|
/******/
|
|
/******/ /* webpack/runtime/make namespace object */
|
|
/******/ (() => {
|
|
/******/ // define __esModule on exports
|
|
/******/ __webpack_require__.r = (exports) => {
|
|
/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
|
|
/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
|
|
/******/ }
|
|
/******/ Object.defineProperty(exports, '__esModule', { value: true });
|
|
/******/ };
|
|
/******/ })();
|
|
/******/
|
|
/************************************************************************/
|
|
var __webpack_exports__ = {};
|
|
// This entry needs to be wrapped in an IIFE because it needs to be isolated against other modules in the chunk.
|
|
(() => {
|
|
/*!*************************!*\
|
|
!*** ./src/js/index.js ***!
|
|
\*************************/
|
|
__webpack_require__.r(__webpack_exports__);
|
|
/* harmony import */ var _css_main_css__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../css/main.css */ "./src/css/main.css");
|
|
/* harmony import */ var _profile_MediaProfile_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./profile/MediaProfile.js */ "./src/js/profile/MediaProfile.js");
|
|
/* harmony import */ var _util_FSUtil_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./util/FSUtil.js */ "./src/js/util/FSUtil.js");
|
|
|
|
|
|
|
|
|
|
async function load(){
|
|
const dir = await showDirectoryPicker({"id": "mediaplayer_amin", "mode": "readwrite"});
|
|
const files = await _util_FSUtil_js__WEBPACK_IMPORTED_MODULE_2__["default"].findFiles(dir, "S01-E0001.mp4");
|
|
files.forEach(f => {console.log(f.path + "/" + f.handle.name);});
|
|
//await walkFileTree(dir);
|
|
}
|
|
|
|
async function walkFileTree(currentDir){
|
|
for await (const value of currentDir.values()) {
|
|
if(value.kind == 'directory'){
|
|
await walkFileTree(value);
|
|
} else if(value.kind == 'file' && value.name == 'playlist.json'){
|
|
await getVideo(currentDir, value);
|
|
}
|
|
}
|
|
}
|
|
|
|
async function getVideo(dir, playlistFile){
|
|
const file = await playlistFile.getFile();
|
|
console.log("Playlist file:", file);
|
|
const content = await file.text();
|
|
const playlist = JSON.parse(content);
|
|
console.log(playlist);
|
|
const videoPath = playlist.tracks[0].relativePath;
|
|
console.log(videoPath);
|
|
|
|
const videoHandle = await _util_FSUtil_js__WEBPACK_IMPORTED_MODULE_2__["default"].getFileHandle(dir, videoPath);
|
|
console.log(videoHandle);
|
|
playVideo(await videoHandle.getFile());
|
|
}
|
|
|
|
function playVideo(file){
|
|
document.getElementById("video").src = URL.createObjectURL(file);
|
|
}
|
|
|
|
document.getElementById("startAppButton").addEventListener("click", load);
|
|
})();
|
|
|
|
/******/ })()
|
|
;
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"main.js","mappings":";;;;;;;;;;;AAAA;;;;;;;;;;;;;;;ACAe;AACf;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,eAAe,sBAAsB;AACrC;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACnBqD;AACQ;AACQ;AACrE;AACe;AACf;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,cAAc;AACd;AACA,WAAW,4DAAgB;AAC3B;AACA;AACA,cAAc;AACd;AACA,mBAAmB,gEAAoB;AACvC;AACA;AACA,cAAc;AACd;AACA,uBAAuB,oEAAwB;AAC/C;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+BAA+B,gEAAoB;AACnD;AACA;AACA;AACA,mCAAmC,oEAAwB;AAC3D;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACtDe;AACf;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;;;;;;;;;;;;;;ACVe;AACf;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AClDe;AACf;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;AClD6D;AAC7D;AACe;AACf;AACA,eAAe,2BAA2B;AAC1C,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,sBAAsB;AACrC,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,2BAA2B;AAC1C,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,2BAA2B;AAC1C,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ;AACvB,iBAAiB;AACjB;AACA;AACA,qDAAqD;AACrD,qDAAqD;AACrD;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C,OAAO;AAClD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,2BAA2B;AAC1C,eAAe,QAAQ;AACvB,eAAe,SAAS;AACxB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,2BAA2B;AAC1C,eAAe,QAAQ;AACvB,eAAe,SAAS;AACxB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,2BAA2B;AAC1C,eAAe,QAAQ;AACvB,eAAe,SAAS;AACxB,eAAe,SAAS;AACxB,iBAAiB;AACjB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd,+BAA+B,qEAAc;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,oBAAoB;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;UC9JA;UACA;;UAEA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;;UAEA;UACA;;UAEA;UACA;UACA;;;;;WCtBA;WACA;WACA;WACA;WACA,yCAAyC,wCAAwC;WACjF;WACA;WACA;;;;;WCPA;;;;;WCAA;WACA;WACA;WACA,uDAAuD,iBAAiB;WACxE;WACA,gDAAgD,aAAa;WAC7D;;;;;;;;;;;;;;ACNkC;AACmB;AACf;AACtC;AACA;AACA,2CAA2C,8CAA8C;AACzF,wBAAwB,uDAAM;AAC9B,wBAAwB,2CAA2C;AACnE;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8BAA8B,uDAAM;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0E","sources":["webpack://dashboard/./src/css/main.css?c48f","webpack://dashboard/./src/js/descriptor/FileDescriptor.js","webpack://dashboard/./src/js/profile/MediaProfile.js","webpack://dashboard/./src/js/profile/MediaProfileRole.js","webpack://dashboard/./src/js/profile/MediaProfileSettings.js","webpack://dashboard/./src/js/profile/MediaProfileViewSettings.js","webpack://dashboard/./src/js/util/FSUtil.js","webpack://dashboard/webpack/bootstrap","webpack://dashboard/webpack/runtime/define property getters","webpack://dashboard/webpack/runtime/hasOwnProperty shorthand","webpack://dashboard/webpack/runtime/make namespace object","webpack://dashboard/./src/js/index.js"],"sourcesContent":["// extracted by mini-css-extract-plugin\nexport {};","export default class FileDescriptor {\r\n    /**\r\n     * @type {string}\r\n     */\r\n    path = \"\";\r\n\r\n    /**\r\n     * @type {FileSystemFileHandle}\r\n     */\r\n    handle = \"\";\r\n\r\n    /**\r\n     * @param {string} path \r\n     * @param {FileSystemFileHandle} handle \r\n     */\r\n    constructor(path, handle){\r\n        this.path = path;\r\n        this.handle = handle;\r\n    }\r\n}","import MediaProfileRole from \"./MediaProfileRole.js\";\r\nimport MediaProfileSettings from \"./MediaProfileSettings.js\";\r\nimport MediaProfileViewSettings from \"./MediaProfileViewSettings.js\";\r\n\r\nexport default class MediaProfile{\r\n    /**\r\n     * @type {string}\r\n     */\r\n    name = \"\";\r\n\r\n    /**\r\n     * @type {string}\r\n     */\r\n    role = MediaProfileRole.ADMIN;\r\n\r\n    /**\r\n     * @type {MediaProfileSettings}\r\n     */\r\n    settings = new MediaProfileSettings();\r\n\r\n    /**\r\n     * @type {MediaProfileViewSettings}\r\n     */\r\n    playSettings = new MediaProfileViewSettings();\r\n\r\n    /**\r\n     * @param {object} jsonObj \r\n     * @returns {MediaProfile}\r\n     */\r\n    static fromJson(jsonObj) {\r\n        const profile = new MediaProfile();\r\n\r\n        if (jsonObj == null) {\r\n            return profile;\r\n        }\r\n\r\n        if (jsonObj.name != null) {\r\n            profile.name = jsonObj.name;\r\n        }\r\n\r\n        if (jsonObj.role != null) {\r\n            profile.role = jsonObj.role;\r\n        }\r\n\r\n        if (jsonObj.settings != null && typeof jsonObj.settings === \"object\") {\r\n            profile.settings = MediaProfileSettings.fromSettingsJson(jsonObj.settings);\r\n        }\r\n\r\n        if (jsonObj.playSettings != null && typeof jsonObj.playSettings === \"object\") {\r\n            profile.playSettings = MediaProfileViewSettings.fromSettingsJson(jsonObj.playSettings);\r\n        }\r\n\r\n        return profile;\r\n    }\r\n}","export default class MediaProfileRole {\r\n    /**\r\n     * @type {string}\r\n     */\r\n    static ADMIN = \"ADMIN\";\r\n\r\n    /**\r\n     * @type {string}\r\n     */\r\n    static USER = \"USER\";\r\n}","export default class MediaProfileSettings {\r\n    /**\r\n     * @type {boolean}\r\n     */\r\n    skipIntro = true;\r\n\r\n    /**\r\n     * @type {boolean}\r\n     */\r\n    skipOutro = true;\r\n\r\n    /**\r\n     * @type {boolean}\r\n     */\r\n    skipRecall = true;\r\n\r\n    /**\r\n     * @type {boolean}\r\n     */\r\n    skipPreview = true;\r\n\r\n    /**\r\n     * @param {object} jsonObj \r\n     * @returns {MediaProfileSettings}\r\n     */\r\n    static fromSettingsJson(jsonObj) {\r\n        const settings = new MediaProfileSettings();\r\n\r\n        if (jsonObj == null) {\r\n            return settings;\r\n        }\r\n\r\n        if (jsonObj.skipIntro != null) {\r\n            settings.skipIntro = jsonObj.skipIntro;\r\n        }\r\n\r\n        if (jsonObj.skipOutro != null) {\r\n            settings.skipOutro = jsonObj.skipOutro;\r\n        }\r\n\r\n        if (jsonObj.skipRecall != null) {\r\n            settings.skipRecall = jsonObj.skipRecall;\r\n        }\r\n\r\n        if (jsonObj.skipPreview != null) {\r\n            settings.skipPreview = jsonObj.skipPreview;\r\n        }\r\n\r\n        return settings;\r\n    }\r\n}","export default class MediaProfileViewSettings {\r\n    /**\r\n     * @type {boolean}\r\n     */\r\n    hasOpenPlaylist = false;\r\n\r\n    /**\r\n     * @type {string}\r\n     */\r\n    playlistPath = null;\r\n\r\n    /**\r\n     * @type {number}\r\n     */\r\n    playlistTrackNumber = 0;\r\n\r\n    /**\r\n     * @type {number}\r\n     */\r\n    trackPosition = 0;\r\n\r\n    /**\r\n     * @param {object} jsonObj \r\n     * @returns {MediaProfileViewSettings}\r\n     */\r\n    static fromSettingsJson(jsonObj) {\r\n        const settings = new MediaProfileViewSettings();\r\n\r\n        if (jsonObj == null) {\r\n            return settings;\r\n        }\r\n\r\n        if (jsonObj.hasOpenPlaylist != null) {\r\n            settings.hasOpenPlaylist = jsonObj.hasOpenPlaylist;\r\n        }\r\n\r\n        if (jsonObj.playlistPath != null) {\r\n            settings.playlistPath = jsonObj.playlistPath;\r\n        }\r\n\r\n        if (jsonObj.playlistTrackNumber != null) {\r\n            settings.playlistTrackNumber = jsonObj.playlistTrackNumber;\r\n        }\r\n\r\n        if (jsonObj.trackPosition != null) {\r\n            settings.trackPosition = jsonObj.trackPosition;\r\n        }\r\n\r\n        return settings;\r\n    }\r\n}","import FileDescriptor from \"../descriptor/FileDescriptor.js\";\r\n\r\nexport default class FSUtil {\r\n    /**\r\n     * @param {FileSystemDirectoryHandle} rootDir \r\n     * @param {string} path \r\n     * @returns {Promise<string | null>}\r\n     */\r\n    static async getFileContent(rootDir, path) {\r\n        const fileHandle = await FSUtil.getFileHandle(rootDir, FSUtil.fixPath(path));\r\n        return await FSUtil.getFileHandleFileContent(fileHandle);\r\n    }\r\n\r\n    /**\r\n     * @param {FileSystemFileHandle} handle \r\n     * @returns {Promise<string>}\r\n     */\r\n    static async getFileHandleFileContent(handle) {\r\n        if (handle == null) {\r\n            return null;\r\n        }\r\n\r\n        const file = await handle.getFile();\r\n        return await file.text();\r\n    }\r\n\r\n    /**\r\n     * @param {FileSystemDirectoryHandle} rootDir \r\n     * @param {string} path \r\n     * @returns {Promise<FileSystemFileHandle | null>}\r\n     */\r\n    static async getFileHandle(rootDir, path) {\r\n        let resultDir = rootDir;\r\n        let outputFile = null;\r\n        const pathes = FSUtil.fixPath(path).split(\"/\");\r\n        let nextDirName = null;\r\n\r\n        while (nextDirName = pathes.shift()) {\r\n            let nextDir = null;\r\n\r\n            for await (const value of resultDir.values()) {\r\n                if (value.kind == 'directory' && value.name == nextDirName) {\r\n                    nextDir = value;\r\n                } else if (value.kind == 'file' && value.name == nextDirName) {\r\n                    outputFile = value;\r\n                }\r\n            }\r\n\r\n            if (nextDir != null) {\r\n                resultDir = nextDir;\r\n            }\r\n        }\r\n\r\n        return outputFile;\r\n    }\r\n\r\n    /**\r\n     * @param {FileSystemDirectoryHandle} rootDir \r\n     * @param {string} path \r\n     * @returns {Promise<FileSystemDirectoryHandle | null>}\r\n     */\r\n    static async getDirectoryHandle(curDir, path) {\r\n        let resultDir = curDir;\r\n        const pathes = FSUtil.fixPath(path).split(\"/\");\r\n        let nextDirName = null;\r\n\r\n        while (nextDirName = pathes.shift()) {\r\n            let nextDir = null;\r\n\r\n            for await (const value of resultDir.values()) {\r\n                if (value.kind == 'directory' && value.name == nextDirName) {\r\n                    nextDir = value;\r\n                }\r\n            }\r\n\r\n            if (nextDir != null) {\r\n                resultDir = nextDir;\r\n            }\r\n        }\r\n\r\n        return resultDir;\r\n    }\r\n\r\n    /**\r\n     * @param {string} path \r\n     * @returns {string}\r\n     */\r\n    static fixPath(path) {\r\n        let fixedPath = path.replaceAll(\"/./\", \"/\"); //remove same folder subpathes\r\n        fixedPath = fixedPath.replaceAll(\"//\", \"/\"); //remove double slashes\r\n\r\n        if (fixedPath.startsWith(\"/\")) {\r\n            fixedPath = fixedPath.substring(1);\r\n        }\r\n\r\n        const pathParts = fixedPath.split(\"/\");\r\n        for (let i = pathParts.length - 1; i > 0; i--) {\r\n            if (pathParts[i] == \"..\") {\r\n                pathParts.splice(i - 1, 2);\r\n                i--;\r\n            }\r\n        }\r\n\r\n        return pathParts.join(\"/\");\r\n    }\r\n\r\n    /**\r\n     * @param {FileSystemDirectoryHandle} dir \r\n     * @param {string} fileName \r\n     * @param {boolean} includeSubDirs\r\n     * @returns {Promise<FileDescriptor>}\r\n     */\r\n    static async findFile(dir, fileName, includeSubDirs = true) {\r\n        const files = await FSUtil.#findFilesInTree(dir, fileName, includeSubDirs, true);\r\n        return files.length > 0 ? files[0] : null;\r\n    }\r\n\r\n    /**\r\n     * @param {FileSystemDirectoryHandle} dir \r\n     * @param {string} fileName \r\n     * @param {boolean} includeSubDirs\r\n     * @returns {Promise<FileDescriptor[]>}\r\n     */\r\n    static async findFiles(dir, fileName, includeSubDirs = true) {\r\n        return FSUtil.#findFilesInTree(dir, fileName, includeSubDirs, false);;\r\n    }\r\n\r\n    /**\r\n     * @param {FileSystemDirectoryHandle} dir \r\n     * @param {string} fileName \r\n     * @param {boolean} includeSubDirs\r\n     * @param {boolean} stopOnFirst\r\n     * @returns {Promise<FileDescriptor[]>}\r\n     */\r\n    static async #findFilesInTree(dir, fileName, includeSubDirs = true, stopOnFirst = false, path = \"\") {\r\n        const files = [];\r\n        const subdirs = [];\r\n\r\n        for await (const child of dir.values()) {\r\n            if (child.kind == 'directory' && includeSubDirs) {\r\n                subdirs.push(child);\r\n            } else if (child.kind == 'file' && child.name == fileName) {\r\n                files.push(new FileDescriptor(path, child));\r\n                if(stopOnFirst){\r\n                    return files;\r\n                }\r\n            }\r\n        }\r\n\r\n        for (let i = 0; i < subdirs.length; i++) {\r\n            files.push(...(await FSUtil.#findFilesInTree(subdirs[i], fileName, includeSubDirs, stopOnFirst, path + \"/\" + subdirs[i].name)));\r\n            if(stopOnFirst && files.length > 0){\r\n                break;\r\n            }\r\n        }\r\n\r\n        return files;\r\n    }\r\n}","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","import css from \"../css/main.css\";\r\nimport MediaProfile from \"./profile/MediaProfile.js\";\r\nimport FSUtil from \"./util/FSUtil.js\";\r\n\r\nasync function load(){\r\n    const dir = await showDirectoryPicker({\"id\": \"mediaplayer_amin\", \"mode\": \"readwrite\"});\r\n    const files = await FSUtil.findFiles(dir, \"S01-E0001.mp4\");\r\n    files.forEach(f => {console.log(f.path + \"/\" + f.handle.name);});\r\n    //await walkFileTree(dir);\r\n}\r\n\r\nasync function walkFileTree(currentDir){\r\n    for await (const value of currentDir.values()) {\r\n        if(value.kind == 'directory'){\r\n            await walkFileTree(value);\r\n        } else if(value.kind == 'file' && value.name == 'playlist.json'){\r\n            await getVideo(currentDir, value);\r\n        }\r\n    }\r\n}\r\n\r\nasync function getVideo(dir, playlistFile){\r\n    const file = await playlistFile.getFile();\r\n    console.log(\"Playlist file:\", file);\r\n    const content = await file.text();\r\n    const playlist = JSON.parse(content);\r\n    console.log(playlist);\r\n    const videoPath = playlist.tracks[0].relativePath;\r\n    console.log(videoPath);\r\n\r\n    const videoHandle = await FSUtil.getFileHandle(dir, videoPath);\r\n    console.log(videoHandle);\r\n    playVideo(await videoHandle.getFile());\r\n}\r\n\r\nfunction playVideo(file){\r\n    document.getElementById(\"video\").src = URL.createObjectURL(file);\r\n}\r\n\r\ndocument.getElementById(\"startAppButton\").addEventListener(\"click\", load);"],"names":[],"sourceRoot":""}
|