Discord-Media-Compressor-su.../lib/ui.js

126 lines
5.2 KiB
JavaScript
Raw Normal View History

2022-05-04 22:45:37 +00:00
const termkit = require("terminal-kit")
const cliProgress = require("cli-progress")
const fs = require("fs")
const path = require("path")
class UI {
term
bars = []
multibar
settings
currentSetting
2022-05-04 22:45:37 +00:00
settingsFile
constructor(settings, currentSetting, settingsFile) {
this.term = termkit.terminal
this.multibar = new cliProgress.MultiBar({
format: '[{bar}] {percentage}% | output: "{filename}" | ETA: {eta_formatted}s | {duration_formatted} | {value}s/{total}s ',
align: "left",
hideCursor: true,
autopadding: true,
}, cliProgress.Presets.shades_grey)
this.settings = settings
this.currentSetting = currentSetting
2022-05-04 22:45:37 +00:00
this.settingsFile = settingsFile
}
/**
*
* @param {Number} duration Duration of the encoded media
* @param {String} filename name of the encoding file
* @param {Boolean} isTwoPass is the encoded media two pass
* @returns
*/
async newBar(encoderOutput) {
let duration = encoderOutput[0]
let filename = encoderOutput[1]
let isTwoPass = encoderOutput[2]
this.bars.push({
"bar": this.multibar.create(duration, 0, { speed: "N/A" }),
"isTwoPass": isTwoPass,
"isPastHalf": false,
"filename": filename,
"duration": duration,
2022-05-05 00:57:42 +00:00
"finished": false,
"isVideo": false
}
)
const barIndex = this.bars.length - 1
return barIndex
}
async updateBar(chunk, barIndex = 0, isVideo = true, isImage = false) {
if (!chunk) return
if (isImage) {
this.bars[barIndex]?.bar.update(1, { filename: `${this.bars[barIndex].filename}.webp` })
return
}
2022-05-05 00:57:42 +00:00
this.bars[barIndex].isVideo = isVideo
if (isVideo) {
const currentTime = chunk.split("time=")[1]?.split(" ")[0]
if (!currentTime) return
const arr = currentTime.split(":")
let seconds = Number.parseFloat(arr[0] * 3600 + arr[1] * 60 + (+arr[2])) // converting to s
//If 2 pass divide bar into two parts to show both in progress in one progress
if (this.bars[barIndex].isTwoPass) {
if (seconds / 2 >= (this.bars[barIndex].duration - 0.2) / 2) this.bars[barIndex].isPastHalf = true
if (this.bars[barIndex].isPastHalf) this.bars[barIndex].bar.update(Math.round(seconds * 50) / 100 + (this.bars[barIndex].duration / 2), { filename: `${this.bars[barIndex].filename}.webm` })
else this.bars[barIndex].bar.update(Math.round(seconds * 50) / 100, { filename: `${this.bars[barIndex].filename}.webm` })
}
else {
this.bars[barIndex].bar.update(Math.round(seconds * 100) / 100, { filename: `${this.bars[barIndex].filename}.webm` })
}
}
else {
const currentTime = chunk.split("time=")[1]?.split(" ")[0]
if (!currentTime) return
const arr = currentTime.split(":")
let seconds = Number.parseFloat(arr[0] * 3600 + arr[1] * 60 + (+arr[2])) // converting to s
this.bars[barIndex].bar.update(Math.round(seconds * 100) / 100, { filename: `${this.bars[barIndex].filename}.ogg` })
}
}
2022-05-05 00:57:42 +00:00
async encodeFinished(barIndex) {
this.bars[barIndex].finished = true
//sets bar to 100%
const chunk = new Date(this.bars[barIndex].duration * 1000).toISOString().substr(11, 8)
this.updateBar(chunk, barIndex, this.bars[barIndex].isVideo)
// if all are finished stop multibars and exit
for (let i = 0; i < this.bars.length; i++) {
2022-05-05 00:57:42 +00:00
if (!this.bars[i].finished) return
}
this.multibar.stop()
fs.rm("ffmpeg2pass-0.log", (error) => { error })
this.term.bold.green("Finished!\n")
this.term.grey("press enter to exit...\n")
this.term.inputField(() => { process.exit() })
}
2022-05-04 22:45:37 +00:00
async startMenu() {
await this.#menu()
}
2022-05-04 22:45:37 +00:00
async #menu(settingsFile) {
let menu = []
2022-05-04 11:09:33 +00:00
for (let i = 0; i < this.settings.presets.length; i++) {
menu.push(`${i}. ${this.settings.presets[i].name}`)
}
2022-05-07 21:13:44 +00:00
this.term("How to convert: 8mb [optional: -preset {Index}] [filename.extension(s)]\n")
this.term("examples: \n")
2022-05-07 21:13:44 +00:00
this.term.italic(" 8mb -preset 0 file.mp3 file4.mov img.jpg\n")
this.term.italic(" 8mb.exe file34.wav file2.mp3\n\n")
this.term.yellow("Hello! This menu is for selecting performance/speed preset.\n")
2022-05-04 11:09:33 +00:00
this.term.yellow("Currently using ").bgMagenta(`"${this.settings.presets[this.settings.currentSetting].name}"`).yellow(" preset")
this.term.singleColumnMenu(menu, (error, response) => {
2022-05-04 11:09:33 +00:00
this.settings.currentSetting = response.selectedIndex
this.term.green("\n Using").green.bold(` ${this.settings.presets[this.settings.currentSetting].name} `).green("setting\n")
2022-05-04 22:45:37 +00:00
fs.writeFileSync(path.resolve(this.settingsFile, "settings.json"), JSON.stringify(this.settings))
this.term.grey("Press enter to exit...")
this.term.inputField(() => { process.exit() })
})
}
2022-05-04 22:45:37 +00:00
}
module.exports = { UI }