|
@@ -0,0 +1,105 @@
|
|
|
+/**
|
|
|
+ * @author Matt C [matt@artemisbot.uk]
|
|
|
+ * @copyright Crown Copyright 2018
|
|
|
+ * @license Apache-2.0
|
|
|
+ */
|
|
|
+
|
|
|
+import Operation from "../Operation";
|
|
|
+import OperationError from "../errors/OperationError";
|
|
|
+import Utils from "../Utils";
|
|
|
+import Magic from "../lib/Magic";
|
|
|
+
|
|
|
+import jimp from "jimp";
|
|
|
+
|
|
|
+/**
|
|
|
+ * Split Colour Channels operation
|
|
|
+ */
|
|
|
+class SplitColourChannels extends Operation {
|
|
|
+
|
|
|
+ /**
|
|
|
+ * SplitColourChannels constructor
|
|
|
+ */
|
|
|
+ constructor() {
|
|
|
+ super();
|
|
|
+
|
|
|
+ this.name = "Split Colour Channels";
|
|
|
+ this.module = "Image";
|
|
|
+ this.description = "Splits the given image into its red, green and blue colour channels.";
|
|
|
+ this.infoURL = "https://wikipedia.org/wiki/Channel_(digital_image)";
|
|
|
+ this.inputType = "byteArray";
|
|
|
+ this.outputType = "List<File>";
|
|
|
+ this.presentType = "html";
|
|
|
+ this.args = [];
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * @param {byteArray} input
|
|
|
+ * @param {Object[]} args
|
|
|
+ * @returns {List<File>}
|
|
|
+ */
|
|
|
+ async run(input, args) {
|
|
|
+ const type = Magic.magicFileType(input);
|
|
|
+ // Make sure that the input is an image
|
|
|
+ if (type && type.mime.indexOf("image") === 0) {
|
|
|
+ const parsedImage = await jimp.read(Buffer.from(input));
|
|
|
+
|
|
|
+ const red = new Promise(async (resolve, reject) => {
|
|
|
+ try {
|
|
|
+ const split = parsedImage
|
|
|
+ .clone()
|
|
|
+ .color([
|
|
|
+ {apply: "blue", params: [-255]},
|
|
|
+ {apply: "green", params: [-255]}
|
|
|
+ ])
|
|
|
+ .getBufferAsync(jimp.MIME_PNG);
|
|
|
+ resolve(new File([new Uint8Array((await split).values())], "red.png", {type: "image/png"}));
|
|
|
+ } catch (err) {
|
|
|
+ reject(new OperationError(`Could not split red channel: ${err}`));
|
|
|
+ }
|
|
|
+ });
|
|
|
+
|
|
|
+ const green = new Promise(async (resolve, reject) => {
|
|
|
+ try {
|
|
|
+ const split = parsedImage.clone()
|
|
|
+ .color([
|
|
|
+ {apply: "red", params: [-255]},
|
|
|
+ {apply: "blue", params: [-255]},
|
|
|
+ ]).getBufferAsync(jimp.MIME_PNG);
|
|
|
+ resolve(new File([new Uint8Array((await split).values())], "green.png", {type: "image/png"}));
|
|
|
+ } catch (err) {
|
|
|
+ reject(new OperationError(`Could not split green channel: ${err}`));
|
|
|
+ }
|
|
|
+ });
|
|
|
+
|
|
|
+ const blue = new Promise(async (resolve, reject) => {
|
|
|
+ try {
|
|
|
+ const split = parsedImage
|
|
|
+ .color([
|
|
|
+ {apply: "red", params: [-255]},
|
|
|
+ {apply: "green", params: [-255]},
|
|
|
+ ]).getBufferAsync(jimp.MIME_PNG);
|
|
|
+ resolve(new File([new Uint8Array((await split).values())], "blue.png", {type: "image/png"}));
|
|
|
+ } catch (err) {
|
|
|
+ reject(new OperationError(`Could not split blue channel: ${err}`));
|
|
|
+ }
|
|
|
+ });
|
|
|
+
|
|
|
+ return await Promise.all([red, green, blue]);
|
|
|
+ } else {
|
|
|
+ throw new OperationError("Invalid file type.");
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * Displays the files in HTML for web apps.
|
|
|
+ *
|
|
|
+ * @param {File[]} files
|
|
|
+ * @returns {html}
|
|
|
+ */
|
|
|
+ async present(files) {
|
|
|
+ return await Utils.displayFilesAsHTML(files);
|
|
|
+ }
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+export default SplitColourChannels;
|