Finished changes required to resolve issue #1 and issue #2. Added super16 and 35mm (4 perf) image sizes
This commit is contained in:
parent
7cb27e7bea
commit
73775da9d1
|
@ -1,3 +1,4 @@
|
||||||
|
node_modules
|
||||||
tmp/*
|
tmp/*
|
||||||
temp/*
|
temp/*
|
||||||
.nexe
|
.nexe
|
258
v2f.js
258
v2f.js
|
@ -1,53 +1,103 @@
|
||||||
/*jshint strict: true, esversion:6, node: true, asi: true*/
|
/*jshint strict: true, esversion:6, node: true, asi: true*/
|
||||||
|
|
||||||
'use strict'
|
'use strict'
|
||||||
|
const cmd = require('commander')
|
||||||
|
const async = require('async')
|
||||||
const exec = require('child_process').exec
|
const exec = require('child_process').exec
|
||||||
const fs = require('fs')
|
const fs = require('fs')
|
||||||
const _tmp = '/tmp'
|
const path = require('path')
|
||||||
|
|
||||||
//var frame_height = 7.61
|
const TMP = '/tmp/v2f/'
|
||||||
//var frame_height = 7.49
|
|
||||||
let frame_height = 7.62
|
|
||||||
let frame_padding = 0
|
|
||||||
|
|
||||||
class Dimensions{
|
class Dimensions{
|
||||||
constructor (film, dpi) {
|
constructor (filmStr, dpi) {
|
||||||
const IN = dpi / 25.4
|
const IN = dpi / 25.4
|
||||||
this.h = Math.round(film.frame_height * IN)
|
const film = this._gauge(filmStr)
|
||||||
this.w = Math.round(film.a * IN)
|
|
||||||
this.o = Math.round(film.b * IN)
|
this.h = Math.round(film.h * IN) //frame height
|
||||||
|
this.w = Math.round(film.w * IN) //frame width
|
||||||
|
this.o = Math.round(film.o * IN) //space between columns
|
||||||
this.dpi = dpi
|
this.dpi = dpi
|
||||||
this.film = film
|
this.film = film
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_gauge (film) {
|
||||||
|
if (film === '16mm') {
|
||||||
|
return {
|
||||||
|
h: 7.62,
|
||||||
|
w : 10.5,
|
||||||
|
o : 16
|
||||||
|
}
|
||||||
|
} else if (film === 'super16'){
|
||||||
|
return {
|
||||||
|
h: 7.62,
|
||||||
|
w : 12.75,
|
||||||
|
o : 16
|
||||||
|
}
|
||||||
|
} else if (film === '35mm') {
|
||||||
|
return {
|
||||||
|
h : 19.05,
|
||||||
|
w : 22,
|
||||||
|
o : 35
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
error('Film type not found, see --help for more info')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* @function
|
||||||
|
* @param {Object} Commander object
|
||||||
|
*/
|
||||||
|
function initialize (command) {
|
||||||
|
const dpi = command.dpi || 300
|
||||||
|
const film = command.film || '16mm'
|
||||||
|
const input = command.input || command.args[0] || error('No input file, see --help for more info')
|
||||||
|
const output = command.output || command.args[1] || error('No ouput directory, see --help for more info')
|
||||||
|
const dim = new Dimensions(film, dpi)
|
||||||
|
|
||||||
|
if (!fs.existsSync(input)) error(`Video "${input}" cannot be found`)
|
||||||
|
|
||||||
|
async.series([
|
||||||
|
next => {
|
||||||
|
convert(input, dim, next)
|
||||||
|
},
|
||||||
|
next => {
|
||||||
|
stitch(output, dim, next)
|
||||||
|
},
|
||||||
|
cleanup
|
||||||
|
], () => {
|
||||||
|
console.log(`Finished creating pages`)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
/** *
|
/** *
|
||||||
* Turn video into sheet of images
|
* Create image sequence from source video, using
|
||||||
|
*
|
||||||
* @function
|
* @function
|
||||||
* @param {String} path file path (absolute)
|
* @param {String} input file path (absolute)
|
||||||
* @param {Integer} dpi target printing dpi
|
* @param {Integer} dpi target printing dpi
|
||||||
* @param {Integer} length strip length in frames
|
* @param {Integer} length strip length in frames
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
function convert (path, dpi) {
|
function convert (input, dim, next) {
|
||||||
const film = { frame_height: frame_height, a : 10.5, b : 16}
|
const file = input.split('/').pop()
|
||||||
const dim = new Dimensions(film, dpi)
|
const execStr = `avconv -i "${input}" -s ${dim.w}x${dim.h} -qscale 1 "${TMP}v2f_sequence_%04d.jpg"`
|
||||||
const file = path.split('/').pop()
|
|
||||||
const loc = _tmp + '/'
|
|
||||||
const execStr = `avconv -i "${path}" -s ${dim.w}x${dim.h} -qscale 1 "${loc}sequence_%04d.jpg"`
|
|
||||||
|
|
||||||
console.log(`Converting ${file}...`)
|
console.log(`Converting ${file}...`)
|
||||||
console.log(`Exporting all frames with aspect ratio: ${dim.w / dim.h}...`)
|
console.log(`Exporting all frames with aspect ratio: ${dim.w / dim.h}...`)
|
||||||
|
|
||||||
if (!fs.existsSync(_tmp)) fs.mkdirSync(_tmp)
|
if (!fs.existsSync(TMP)) fs.mkdirSync(TMP)
|
||||||
|
|
||||||
exec(execStr, (ste, std) => {
|
exec(execStr, (ste, std) => {
|
||||||
if (ste) {
|
if (ste) {
|
||||||
return errorHandle(ste)
|
return error(ste)
|
||||||
}
|
}
|
||||||
console.log('Frames exported successfully!')
|
console.log('Frames exported successfully!')
|
||||||
stitch(loc.substring(0, loc.length - 1), dim)
|
next()
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -55,114 +105,110 @@ function convert (path, dpi) {
|
||||||
* Stitch rendered frames into strips
|
* Stitch rendered frames into strips
|
||||||
|
|
||||||
* @function
|
* @function
|
||||||
* @param {String} loc Path of folder containing frames
|
* @param {String} output Path of folder containing frames
|
||||||
* @param {Object} dim Dimensions object
|
* @param {Object} dim Dimensions object
|
||||||
|
* @param {Function} next Async lib callback function
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
function stitch (output, dim, next) {
|
||||||
function stitch (loc, dim) {
|
|
||||||
const length = Math.floor((11 * dim.dpi) / dim.h) - 1
|
const length = Math.floor((11 * dim.dpi) / dim.h) - 1
|
||||||
const width = Math.floor((8.5 * dim.dpi / dim.o)) - 1
|
const width = Math.floor((8.5 * dim.dpi / dim.o)) - 1
|
||||||
|
const loc = TMP.substring(0, TMP.length - 1)
|
||||||
|
const diff = Math.round((dim.o - dim.w) / 2)
|
||||||
let page = 0
|
let page = 0
|
||||||
let pageCount = 0
|
let pageCount = 0
|
||||||
let cmd = `find "${loc}" -type f -name "sequence_*.jpg"`
|
let cmd = `find "${loc}" -type f -name "v2f_sequence_*.jpg"`
|
||||||
function find_cb (ste, std) {
|
|
||||||
if (ste) {
|
|
||||||
return errorHandle(ste)
|
|
||||||
}
|
|
||||||
const frames = std.split('\n')
|
|
||||||
let execStr = 'montage '
|
|
||||||
function montage_cb (stee, stdd) {
|
|
||||||
if (stee) {
|
|
||||||
return errorHandle(ste)
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log(`Created page_${pageCount}.jpg!`)
|
|
||||||
pageCount++
|
|
||||||
if (pageCount === page) {
|
|
||||||
console.log('Cleaning up...');
|
|
||||||
setTimeout(() => {
|
|
||||||
exec(`find "${loc}" -type f -name "sequence_*.jpg" -delete`, () => {
|
|
||||||
fs.rmdirSync(_tmp)
|
|
||||||
console.log('Done!')
|
|
||||||
})
|
|
||||||
}, 1000)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for (let i = 0; i < frames.length; i++) {
|
|
||||||
execStr += frames[i] + ' '
|
|
||||||
if ((i + 1) % (width * length) === 0 || i === frames.length - 1) {
|
|
||||||
execStr += '\ -tile 1x' + length + ' -geometry ' + dim.w + 'x' + dim.h + '+' + Math.round((dim.o - dim.w) / 2) + '+0 miff:- |\ \nmontage - -geometry +0+0 -tile ' + width + 'x1 -density ' + dim.dpi + ' "./page_' + page + '.jpg"'
|
|
||||||
exec(execStr, montage_cb)
|
|
||||||
execStr = 'montage '
|
|
||||||
page++
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
loc = _tmp
|
|
||||||
|
|
||||||
console.log('Stitching frames into sheets...')
|
console.log('Stitching frames into sheets...')
|
||||||
console.log(`Sheets will contain ${width}x${length} frames...`)
|
console.log(`Sheets will contain ${width}x${length} frames...`)
|
||||||
exec(cmd, find_cb)
|
|
||||||
|
exec(cmd, (ste, std) => {
|
||||||
|
if (ste) {
|
||||||
|
return error(ste)
|
||||||
|
}
|
||||||
|
let jobs = []
|
||||||
|
let cmds = []
|
||||||
|
let frames = std.split('\n')
|
||||||
|
let execStr = 'montage '
|
||||||
|
let pagePath = ``
|
||||||
|
let i = 0
|
||||||
|
|
||||||
|
frames = frames.filter(elem => {
|
||||||
|
if (elem.indexOf('find: ') === -1) {
|
||||||
|
return elem
|
||||||
|
}
|
||||||
|
})
|
||||||
|
frames.sort()
|
||||||
|
for (let frame of frames) {
|
||||||
|
execStr += `${frame} `
|
||||||
|
if ((i + 1) % (width * length) === 0 || i === frames.length - 1) {
|
||||||
|
pagePath = path.join(output, `./page_${pad(page)}.jpg`)
|
||||||
|
execStr += `\ -tile 1x${length} -geometry ${dim.w}x${dim.h}+${diff}+0 miff:- |\ \nmontage - -geometry +0+0 -tile ${width}x1 -density ${dim.dpi} "${pagePath}"`
|
||||||
|
console.log(execStr)
|
||||||
|
process.exit()
|
||||||
|
cmds.push(execStr)
|
||||||
|
execStr = 'montage '
|
||||||
|
page++
|
||||||
|
}
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
jobs = cmds.map(cmd => {
|
||||||
|
return cb => {
|
||||||
|
exec(cmd, (err, std, ste) => {
|
||||||
|
if (err) {
|
||||||
|
return error(err)
|
||||||
|
}
|
||||||
|
console.log(`Created page of ${width}x${length} frames!`)
|
||||||
|
cb()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
async.series(jobs, next)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
var errorHandle = function (err) {
|
function cleanup (next) {
|
||||||
|
console.log('Cleaning up...');
|
||||||
|
exec(`rm -r "${TMP}"`, (err) => {
|
||||||
|
if (err) console.error(err)
|
||||||
|
if (next) next()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function pad (n) {
|
||||||
|
return ('00000' + n).slice(-5)
|
||||||
|
}
|
||||||
|
|
||||||
|
var error = function (err) {
|
||||||
if (process.argv.indexOf('-v') !== -1 || process.argv.indexOf('--verbose') !== -1){
|
if (process.argv.indexOf('-v') !== -1 || process.argv.indexOf('--verbose') !== -1){
|
||||||
console.error(err)
|
console.error(err)
|
||||||
} else {
|
} else {
|
||||||
console.error('Error running program. Run in verbose mode for more info (-v,--verbose)')
|
console.error('Error running program. Run in verbose mode for more info (-v,--verbose)')
|
||||||
}
|
}
|
||||||
process.exit(1)
|
process.exit(1)
|
||||||
};
|
}
|
||||||
|
|
||||||
process.on('uncaughtException', err => {
|
process.on('uncaughtException', err => {
|
||||||
errorHandle(err)
|
error(err)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
//convert(process.argv[2], process.argv[3])
|
||||||
|
|
||||||
if (typeof process.argv[2] === 'undefined') {
|
//fix for nexe
|
||||||
console.error('No path to video defined')
|
let args = [].concat(process.argv)
|
||||||
process.exit(1)
|
if (args[1].indexOf('v2f.js') === -1) {
|
||||||
|
args.reverse()
|
||||||
|
args.push('node')
|
||||||
|
args.reverse()
|
||||||
}
|
}
|
||||||
|
|
||||||
if (typeof process.argv[3] === 'undefined') {
|
cmd.arguments('<input> <output>')
|
||||||
process.argv[3] = 300
|
.version('1.1.0')
|
||||||
console.log('Using default 300dpi')
|
.option('-i, --input <path>', 'Video source to print to film strip, anything that avconv can read')
|
||||||
}
|
.option('-o, --output <path>', 'Output directory, will print images on A4 standard paper file')
|
||||||
|
.option('-d, --dpi <dpi>', 'DPI output pages')
|
||||||
|
.option('-f, --film <gauge>', 'Choose film gauge: 16mm, super16, 35mm')
|
||||||
|
.option('-v, --verbose', 'Run in verbose mode')
|
||||||
|
.parse(args)
|
||||||
|
|
||||||
convert(process.argv[2], process.argv[3])
|
initialize(cmd)
|
||||||
|
|
||||||
/*
|
|
||||||
|
|
||||||
INSTALLATION AND RUNNING
|
|
||||||
|
|
||||||
*/
|
|
||||||
|
|
||||||
//Install in this order to satisfy requirements: GCC required by MacPorts, etc.
|
|
||||||
|
|
||||||
//Install Node http://nodejs.org/dist/v0.10.22/node-v0.10.22.pkg
|
|
||||||
//Install GCC https://github.com/kennethreitz/osx-gcc-installer#option-1-downloading-pre-built-binaries
|
|
||||||
//Install MacPorts http://www.macports.org/install.php
|
|
||||||
//Install ImageMagick in terminal type "sudo port install ImageMagick"
|
|
||||||
//Install avconv -see below
|
|
||||||
|
|
||||||
/*
|
|
||||||
Download and unzip http://libav.org/releases/libav-9.6.tar.gz
|
|
||||||
in terminal type "cd " and drag in unzipped folder and hit enter
|
|
||||||
in terminal copy "sudo port install yasm zlib bzip2 faac lame speex libogg libvorbis libtheora libvpx x264 XviD openjpeg15 opencore-amr freetype" (without quotes) hit enter
|
|
||||||
in terminal copy "./configure \ --enable-gpl --enable-libx264 --enable-libxvid \ --enable-version3 --enable-libopencore-amrnb --enable-libopencore-amrwb \ --enable-nonfree --enable-libfaac \ --enable-libmp3lame --enable-libspeex --enable-libvorbis --enable-libtheora --enable-libvpx \ --enable-libopenjpeg --enable-libfreetype --enable-doc --enable-gnutls --enable-shared" hit enter
|
|
||||||
(if that gives errors just use "./configure" and hit enter)
|
|
||||||
in terminal copy "make && sudo make install" hit enter
|
|
||||||
|
|
||||||
that will install it
|
|
||||||
*/
|
|
||||||
|
|
||||||
//run by going to terminal typing "node " dragging this script into the terminal, dragging the video into the terminal and adding a DPI value and hitting enter
|
|
||||||
//the command should look something like this:
|
|
||||||
//node /Users/stenzel/Desktop/video_to_page.js /Users/stenzel/Desktop/PaulRobeson/Paul\ Robeson\ discusses\ Othello.mp4 600
|
|
||||||
|
|
||||||
|
|
||||||
//Get the absolute path of any file by dragging it into terminal and copying the results
|
|
||||||
//Must be enclosed by ''
|
|
||||||
//Will generate pages and frames in same folder as source video
|
|
Loading…
Reference in New Issue