next-video
Version:
A React component for adding video to your Next.js application. It extends both the video element and your Next app with features for automatic video optimization.
318 lines (316 loc) • 15.6 kB
JavaScript
var __create = Object.create;
var __defProp = Object.defineProperty;
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
var __getOwnPropNames = Object.getOwnPropertyNames;
var __getProtoOf = Object.getPrototypeOf;
var __hasOwnProp = Object.prototype.hasOwnProperty;
var __export = (target, all) => {
for (var name in all)
__defProp(target, name, { get: all[name], enumerable: true });
};
var __copyProps = (to, from, except, desc2) => {
if (from && typeof from === "object" || typeof from === "function") {
for (let key of __getOwnPropNames(from))
if (!__hasOwnProp.call(to, key) && key !== except)
__defProp(to, key, { get: () => from[key], enumerable: !(desc2 = __getOwnPropDesc(from, key)) || desc2.enumerable });
}
return to;
};
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
// If the importer is in node compatibility mode or this is not an ESM
// file that has been converted to a CommonJS file using a Babel-
// compatible transform (i.e. "__esModule" has not been set), then set
// "default" to the CommonJS "module.exports" for node compatibility.
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
mod
));
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
var init_exports = {};
__export(init_exports, {
builder: () => builder,
command: () => command,
desc: () => desc,
handler: () => handler
});
module.exports = __toCommonJS(init_exports);
var import_prompts = require("@inquirer/prompts");
var import_chalk = __toESM(require("chalk"), 1);
var import_node_os = __toESM(require("node:os"), 1);
var import_node_child_process = require("node:child_process");
var import_promises = require("node:fs/promises");
var import_node_path = __toESM(require("node:path"), 1);
var import_logger = __toESM(require("../utils/logger.js"), 1);
var import_json_configs = require("./lib/json-configs.js");
var import_next_config = __toESM(require("./lib/next-config.js"), 1);
const GET_STARTED_CONTENTS = `{
"status": "ready",
"originalFilePath": "videos/get-started.mp4",
"provider": "mux",
"providerMetadata": {
"mux": {
"uploadId": "LJeQjU9A1n029JC8Nx2Z71U7wOsdVA02n9mPOL02iYspkY",
"assetId": "3fMT5ZqYShCbiI00Um3K00eGAqf7d6xmylwDKVEYAl4Ck",
"playbackId": "sxY31L6Opl02RWPpm3Gro9XTe7fRHBjs92x93kiB1vpc"
}
},
"createdAt": 1701286882695,
"updatedAt": 1701287023961,
"size": 431539343,
"sources": [{
"src": "https://stream.mux.com/sxY31L6Opl02RWPpm3Gro9XTe7fRHBjs92x93kiB1vpc.m3u8",
"type": "application/x-mpegURL"
}],
"poster": "https://image.mux.com/sxY31L6Opl02RWPpm3Gro9XTe7fRHBjs92x93kiB1vpc/thumbnail.webp",
"blurDataURL": ""
}
`;
const TYPES_FILE_CONTENTS = `/// <reference types="next-video/video-types/global" />
`;
const DEFAULT_DIR = "videos";
const DEV_SCRIPT = "& npx next-video sync -w";
const gitIgnoreContents = (videosDir) => `
# next-video
${videosDir}/*
!${videosDir}/*.json
!${videosDir}/*.js
!${videosDir}/*.ts
public/_next-video
`;
async function preInitCheck(dir) {
try {
await (0, import_promises.stat)(dir);
return false;
} catch (err) {
if (err.code === "ENOENT") {
return true;
}
throw err;
}
}
async function checkVersionManager() {
try {
await (0, import_promises.access)("package-lock.json");
return "npm";
} catch {
}
try {
await (0, import_promises.access)("yarn.lock");
return "yarn";
} catch {
}
try {
await (0, import_promises.access)("pnpm-lock.yaml");
return "pnpm";
} catch {
}
return void 0;
}
function execPromise(command2) {
return new Promise((resolve, reject) => {
(0, import_node_child_process.exec)(command2, (err, stdout, stderr) => {
if (err) {
return reject(err);
}
return resolve(stdout);
});
});
}
async function createVideoDir(dir) {
const fullPath = import_node_path.default.join(process.cwd(), dir);
await (0, import_promises.mkdir)(fullPath, { recursive: true });
await (0, import_promises.writeFile)(import_node_path.default.join(dir, "get-started.mp4.json"), GET_STARTED_CONTENTS);
await (0, import_promises.appendFile)(".gitignore", gitIgnoreContents(dir));
return;
}
async function createTSFile(filePath) {
await (0, import_promises.writeFile)(filePath, TYPES_FILE_CONTENTS);
return;
}
async function updateTSConfigFile(tsConfigPath) {
const configContents = await (0, import_promises.readFile)(tsConfigPath, "utf-8");
const updatedContents = (0, import_json_configs.updateTSConfigFileContent)(configContents);
return (0, import_promises.writeFile)(tsConfigPath, updatedContents);
}
const command = "init [dir]";
const desc = "Initializes next-video in a project.";
function builder(yargs) {
return yargs.options({
dir: {
alias: "d",
describe: "The directory you want to initialize next-video with.",
type: "string"
},
force: {
alias: "f",
describe: "Continue with initialization even if the chosen directory already exists.",
type: "boolean",
default: false
},
typescript: {
alias: "ts",
describe: "Initialize next-video for use with TypeScript.",
type: "boolean",
default: false
},
tsconfig: {
describe: "Automatically update your tsconfig.json file to include the next-video types.",
type: "boolean",
default: false
},
devscript: {
describe: `Automatically update your package.json to add the watch command to your dev script.`,
type: "boolean",
default: false
}
});
}
async function handler(argv) {
let baseDir = argv.dir;
let packageInstalled = false;
let ts = argv.typescript;
let updateTsConfig = argv.tsconfig;
let updateDevScript = argv.devscript;
let changes = [];
try {
packageInstalled = await (0, import_json_configs.checkPackageJsonForNextVideo)("./package.json");
} catch (err) {
if (err.code === "ENOENT") {
import_logger.default.error(
`Failed to find/read a local package.json. Double check that you're running this from the root of your project.`
);
return;
}
console.log(err);
}
if (!packageInstalled) {
const install = await (0, import_prompts.confirm)({
message: `It doesn't look like ${import_chalk.default.magenta.bold(
"next-video"
)} is installed in this project. Would you like to install it now?`,
default: true
});
if (install) {
const manager = await checkVersionManager();
if (!manager) {
import_logger.default.error("Failed to detect a package manager. Please install next-video manually and re-run this command.");
import_logger.default.info("For example, in NPM: npm install --save-dev next-video");
return;
}
import_logger.default.info("Detected package manager:", manager);
import_logger.default.info("Installing next-video...");
try {
if (manager === "npm") {
await execPromise("npm install next-video");
} else if (manager === "yarn") {
await execPromise("yarn add next-video");
} else if (manager === "pnpm") {
await execPromise("pnpm add next-video");
}
import_logger.default.info("Successfully installed next-video!");
} catch (err) {
import_logger.default.error("Failed to install next-video:", err);
}
} else {
import_logger.default.info("Make sure to add next-video to your package.json manually");
}
}
if (!baseDir) {
baseDir = await (0, import_prompts.input)({ message: "What directory should next-video use for video files?", default: DEFAULT_DIR });
}
const shouldContinue = await preInitCheck(baseDir);
if (!argv.force && !shouldContinue) {
import_logger.default.warning("Directory already exists:", baseDir);
import_logger.default.info("If you'd like to proceed anyway, re-run with --force");
return;
}
await createVideoDir(baseDir);
changes.push([import_logger.default.add, `Created ${baseDir} directory.`]);
if (!ts) {
ts = await (0, import_prompts.confirm)({ message: "Is this a TypeScript project?", default: true });
}
if (ts) {
await createTSFile(import_node_path.default.join(process.cwd(), "video.d.ts"));
changes.push([import_logger.default.add, `Created video.d.ts.`]);
}
if (ts && !updateTsConfig) {
updateTsConfig = await (0, import_prompts.confirm)({ message: "Update tsconfig.json to include next-video types?", default: true });
}
if (updateTsConfig) {
try {
await updateTSConfigFile(import_node_path.default.join(process.cwd(), "tsconfig.json"));
changes.push([import_logger.default.add, `Updated tsconfig.json to include next-video types.`]);
} catch (err) {
changes.push([import_logger.default.error, 'Failed to update tsconfig.json, please add "video.d.ts" to the include array']);
}
} else if (ts) {
changes.push([import_logger.default.info, `Add ${import_chalk.default.underline("video.d.ts")} to the includes array in tsconfig.json.`]);
}
const cmd = await isCmd();
if (!cmd && !updateDevScript) {
updateDevScript = await (0, import_prompts.confirm)({
message: `Update package.json to add the watch command to your dev script?`,
default: true
});
}
if (!cmd && updateDevScript) {
try {
const devScript = (await execPromise(`npm pkg get scripts.dev`))?.trim().slice(1, -1);
if (devScript && !devScript.includes(DEV_SCRIPT)) {
await execPromise(`npm pkg set scripts.dev='${devScript} ${DEV_SCRIPT}'`);
}
changes.push([import_logger.default.add, `Updated package.json to add the watch command to your dev script.`]);
} catch (err) {
changes.push([import_logger.default.error, `Failed to update package.json, please add "${DEV_SCRIPT}" to your dev script.`]);
}
}
try {
const update = await (0, import_next_config.default)("./", { folder: baseDir });
if (update) {
changes.push([import_logger.default.add, `Updated ${update.configPath} to include next-video.`]);
}
} catch (e) {
if (e.error === "not_found") {
changes.push([
import_logger.default.error,
"No next.config.(js|mjs|ts) file found. Please add next-video to your config manually."
]);
} else if (e.error === "already_added") {
changes.push([import_logger.default.info, "It seems like next-video is already added to your Next Config"]);
} else {
changes.push([import_logger.default.error, "Failed to update next.config.(js|mjs|ts), please add next-video to your config manually."]);
}
}
import_logger.default.success(`${import_chalk.default.magenta.bold("next-video")} initialized!`);
changes.forEach(([loggerFn, change]) => loggerFn(change));
import_logger.default.space();
import_logger.default.info(`Why don't you try adding the component to a page?`);
import_logger.default.space();
import_logger.default.space(`${import_chalk.default.magenta("import")} Video ${import_chalk.default.magenta("from")} ${import_chalk.default.cyan("'next-video'")};
${import_chalk.default.magenta("import")} getStarted ${import_chalk.default.magenta("from")} ${import_chalk.default.cyan("'/videos/get-started.mp4'")};
${import_chalk.default.magenta("export default function")} Page() {
${import_chalk.default.magenta("return")} ${import_chalk.default.cyan("<")}Video ${import_chalk.default.cyan("src=")}{getStarted} ${import_chalk.default.cyan("/>")};
}
`);
import_logger.default.space();
import_logger.default.info(`NEXT STEP: Set up remote storage`);
import_logger.default.info(import_chalk.default.magenta.bold("https://next-video.dev/docs#remote-storage-and-optimization"));
import_logger.default.space();
}
async function isCmd() {
if (import_node_os.default.platform() === "win32") {
try {
await execPromise(`ls`);
} catch (err) {
return true;
}
}
return false;
}
// Annotate the CommonJS export names for ESM import in node:
0 && (module.exports = {
builder,
command,
desc,
handler
});
;