2017-04-11 13:58:35 -05:00
|
|
|
'use strict';
|
|
|
|
|
2017-04-21 18:07:36 -05:00
|
|
|
const electron = require('electron');
|
|
|
|
const app = electron.app;
|
2017-04-11 13:58:35 -05:00
|
|
|
const childProcess = require('child_process');
|
|
|
|
const fs = require('fs');
|
|
|
|
const os = require('os');
|
|
|
|
const path = require('path');
|
|
|
|
|
2017-04-21 18:07:36 -05:00
|
|
|
const { isMac, isDevEnv } = require('../utils/misc.js');
|
2017-05-31 23:39:08 -05:00
|
|
|
const log = require('../log.js');
|
|
|
|
const logLevels = require('../enums/logLevels.js');
|
2017-04-11 13:58:35 -05:00
|
|
|
|
|
|
|
// static ref to child process, only allow one screen snippet at time, so
|
|
|
|
// hold ref to prev, so can kill before starting next snippet.
|
|
|
|
let child;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Captures a user selected portion of the monitor and returns jpeg image
|
|
|
|
* encoded in base64 format.
|
|
|
|
*/
|
|
|
|
class ScreenSnippet {
|
|
|
|
/**
|
|
|
|
* Returns promise.
|
|
|
|
*
|
|
|
|
* If successful will resolves with jpeg image encoded in base64 format:
|
|
|
|
* {
|
|
|
|
* type: 'image/jpg;base64',
|
|
|
|
* data: base64-data
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* Otherwise if not successful will reject with object
|
|
|
|
* containing: { type: ['WARN','ERROR'], message: String }
|
|
|
|
*/
|
|
|
|
capture() {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
let captureUtil, captureUtilArgs;
|
|
|
|
|
2017-06-02 11:08:55 -05:00
|
|
|
log.send(logLevels.INFO, 'ScreenSnippet: starting screen capture');
|
2017-05-31 23:39:08 -05:00
|
|
|
|
2017-04-11 13:58:35 -05:00
|
|
|
let tmpFilename = 'symphonyImage-' + Date.now() + '.jpg';
|
|
|
|
let tmpDir = os.tmpdir();
|
|
|
|
|
|
|
|
let outputFileName = path.join(tmpDir, tmpFilename);
|
|
|
|
|
|
|
|
if (isMac) {
|
|
|
|
// utilize Mac OSX built-in screencapture tool which has been
|
|
|
|
// available since OSX ver 10.2.
|
|
|
|
captureUtil = '/usr/sbin/screencapture';
|
2017-08-23 11:38:33 -05:00
|
|
|
captureUtilArgs = ['-i', '-s', '-t', 'jpg', outputFileName];
|
2017-04-11 13:58:35 -05:00
|
|
|
} else {
|
2017-04-21 18:07:36 -05:00
|
|
|
// use custom built windows screen capture tool
|
|
|
|
if (isDevEnv) {
|
|
|
|
// for dev env pick up tool from node nodules
|
|
|
|
captureUtil =
|
|
|
|
path.join(__dirname,
|
2017-08-23 11:38:33 -05:00
|
|
|
'../../node_modules/screen-snippet/bin/Release/ScreenSnippet.exe');
|
2017-04-21 18:07:36 -05:00
|
|
|
} else {
|
|
|
|
// for production gets installed next to exec.
|
|
|
|
let execPath = path.dirname(app.getPath('exe'));
|
|
|
|
captureUtil = path.join(execPath, 'ScreenSnippet.exe');
|
|
|
|
}
|
|
|
|
|
2017-08-23 11:38:33 -05:00
|
|
|
captureUtilArgs = [outputFileName];
|
2017-04-11 13:58:35 -05:00
|
|
|
}
|
|
|
|
|
2017-06-02 11:08:55 -05:00
|
|
|
log.send(logLevels.INFO, 'ScreenSnippet: starting screen capture util: ' + captureUtil + ' with args=' + captureUtilArgs);
|
2017-05-31 23:39:08 -05:00
|
|
|
|
2017-04-11 13:58:35 -05:00
|
|
|
// only allow one screen capture at a time.
|
|
|
|
if (child) {
|
|
|
|
child.kill();
|
|
|
|
}
|
|
|
|
|
|
|
|
child = childProcess.execFile(captureUtil, captureUtilArgs, (error) => {
|
|
|
|
// will be called when child process exits.
|
|
|
|
if (error && error.killed) {
|
|
|
|
// processs was killed, just resolve with no data.
|
|
|
|
resolve();
|
|
|
|
} else {
|
2017-06-07 11:05:25 -05:00
|
|
|
readResult.call(this, outputFileName, resolve, reject, error);
|
2017-04-11 13:58:35 -05:00
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
2017-06-07 11:05:25 -05:00
|
|
|
}
|
2017-04-11 13:58:35 -05:00
|
|
|
|
2017-06-07 11:05:25 -05:00
|
|
|
// this function was moved outside of class since class is exposed to web
|
|
|
|
// client via preload API, we do NOT want web client to be able to call this
|
|
|
|
// method - then they could read any file on the disk!
|
|
|
|
function readResult(outputFileName, resolve, reject, childProcessErr) {
|
|
|
|
fs.readFile(outputFileName, (readErr, data) => {
|
|
|
|
if (readErr) {
|
|
|
|
let returnErr;
|
|
|
|
if (readErr.code === 'ENOENT') {
|
|
|
|
// no such file exists, user likely aborted
|
|
|
|
// creating snippet. also include any error when
|
|
|
|
// creating child process.
|
|
|
|
returnErr = createWarn('file does not exist ' +
|
|
|
|
childProcessErr);
|
|
|
|
} else {
|
|
|
|
returnErr = createError(readErr + ',' +
|
|
|
|
childProcessErr);
|
2017-04-11 13:58:35 -05:00
|
|
|
}
|
|
|
|
|
2017-06-07 11:05:25 -05:00
|
|
|
reject(returnErr);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!data) {
|
|
|
|
reject(createWarn('no file data provided'));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
// convert binary data to base64 encoded string
|
|
|
|
let output = Buffer(data).toString('base64');
|
|
|
|
resolve({
|
|
|
|
type: 'image/jpg;base64',
|
|
|
|
data: output
|
|
|
|
});
|
|
|
|
} catch (error) {
|
|
|
|
reject(createError(error));
|
2017-08-23 11:38:33 -05:00
|
|
|
} finally {
|
2017-06-07 11:05:25 -05:00
|
|
|
// remove tmp file (async)
|
|
|
|
fs.unlink(outputFileName, function(removeErr) {
|
|
|
|
// note: node complains if calling async
|
|
|
|
// func without callback.
|
|
|
|
if (removeErr) {
|
|
|
|
log.send(logLevels.ERROR, 'ScreenSnippet: error removing temp snippet file: ' +
|
|
|
|
outputFileName + ', err:' + removeErr);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2017-04-11 13:58:35 -05:00
|
|
|
|
2017-06-07 11:05:25 -05:00
|
|
|
/* eslint-disable class-methods-use-this */
|
|
|
|
function createError(msg) {
|
2017-08-24 02:51:02 -05:00
|
|
|
let err = new Error(msg);
|
2017-06-07 11:05:25 -05:00
|
|
|
err.type = 'ERROR';
|
|
|
|
return err;
|
|
|
|
}
|
2017-04-11 13:58:35 -05:00
|
|
|
|
2017-06-07 11:05:25 -05:00
|
|
|
function createWarn(msg) {
|
2017-08-24 02:51:02 -05:00
|
|
|
let err = new Error(msg);
|
2017-06-07 11:05:25 -05:00
|
|
|
err.type = 'WARN';
|
|
|
|
return err;
|
2017-04-11 13:58:35 -05:00
|
|
|
}
|
2017-06-07 11:05:25 -05:00
|
|
|
/* eslint-enable class-methods-use-this */
|
2017-04-11 13:58:35 -05:00
|
|
|
|
2017-06-07 11:05:25 -05:00
|
|
|
module.exports = {
|
|
|
|
ScreenSnippet: ScreenSnippet,
|
|
|
|
// note: readResult only exposed for testing purposes
|
|
|
|
readResult: readResult
|
2017-08-23 11:38:33 -05:00
|
|
|
};
|