2018-08-14 11:06:17 -03:00
|
|
|
import { IDrawingOptions } from "../drawing";
|
2018-08-14 11:28:01 -03:00
|
|
|
import { File } from "../file";
|
2018-08-09 01:55:50 +01:00
|
|
|
import { ImageParagraph } from "../paragraph";
|
2018-01-23 01:33:12 +00:00
|
|
|
import { IMediaData } from "./data";
|
2018-08-09 01:55:50 +01:00
|
|
|
import { Image } from "./image";
|
2017-03-25 21:50:33 +00:00
|
|
|
|
2018-08-02 02:09:00 +01:00
|
|
|
interface IHackedFile {
|
|
|
|
currentRelationshipId: number;
|
|
|
|
}
|
|
|
|
|
2017-03-25 21:50:33 +00:00
|
|
|
export class Media {
|
2018-08-02 02:09:00 +01:00
|
|
|
|
2018-08-22 08:04:45 -03:00
|
|
|
public static addImage(file: File, buffer: Buffer | string | Uint8Array | ArrayBuffer, width?: number, height?: number, drawingOptions?: IDrawingOptions): Image {
|
2018-08-02 02:09:00 +01:00
|
|
|
// Workaround to expose id without exposing to API
|
|
|
|
const exposedFile = (file as {}) as IHackedFile;
|
2018-08-12 23:07:31 +01:00
|
|
|
const mediaData = file.Media.addMedia(buffer, exposedFile.currentRelationshipId++, width, height);
|
2018-08-02 02:09:00 +01:00
|
|
|
file.DocumentRelationships.createRelationship(
|
|
|
|
mediaData.referenceId,
|
|
|
|
"http://schemas.openxmlformats.org/officeDocument/2006/relationships/image",
|
|
|
|
`media/${mediaData.fileName}`,
|
|
|
|
);
|
2018-08-03 00:01:42 +01:00
|
|
|
|
2018-08-14 11:27:08 -03:00
|
|
|
return new Image(new ImageParagraph(mediaData, drawingOptions));
|
2018-08-02 02:09:00 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private static generateId(): string {
|
|
|
|
// https://gist.github.com/6174/6062387
|
2018-08-02 02:21:15 +01:00
|
|
|
return (
|
|
|
|
Math.random()
|
|
|
|
.toString(36)
|
|
|
|
.substring(2, 15) +
|
|
|
|
Math.random()
|
|
|
|
.toString(36)
|
|
|
|
.substring(2, 15)
|
|
|
|
);
|
2018-08-02 02:09:00 +01:00
|
|
|
}
|
|
|
|
|
2018-01-29 01:55:25 +00:00
|
|
|
private readonly map: Map<string, IMediaData>;
|
2017-03-25 21:50:33 +00:00
|
|
|
|
|
|
|
constructor() {
|
2017-12-30 21:18:55 +00:00
|
|
|
this.map = new Map<string, IMediaData>();
|
2017-03-25 21:50:33 +00:00
|
|
|
}
|
|
|
|
|
2018-04-20 15:59:06 +02:00
|
|
|
public getMedia(key: string): IMediaData {
|
|
|
|
const data = this.map.get(key);
|
|
|
|
|
|
|
|
if (data === undefined) {
|
|
|
|
throw new Error(`Cannot find image with the key ${key}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2018-08-15 22:20:43 +01:00
|
|
|
public addMedia(
|
|
|
|
buffer: Buffer | string | Uint8Array | ArrayBuffer,
|
|
|
|
referenceId: number,
|
|
|
|
width: number = 100,
|
|
|
|
height: number = 100,
|
|
|
|
): IMediaData {
|
2018-08-12 23:07:31 +01:00
|
|
|
const key = `${Media.generateId()}.png`;
|
2018-04-20 15:59:06 +02:00
|
|
|
|
2018-08-15 22:20:43 +01:00
|
|
|
return this.createMedia(
|
|
|
|
key,
|
|
|
|
referenceId,
|
|
|
|
{
|
2018-04-20 15:59:06 +02:00
|
|
|
width: width,
|
2018-05-06 03:19:36 +01:00
|
|
|
height: height,
|
2018-08-15 22:20:43 +01:00
|
|
|
},
|
|
|
|
buffer,
|
|
|
|
);
|
2018-04-20 15:59:06 +02:00
|
|
|
}
|
2017-03-25 21:50:33 +00:00
|
|
|
|
2018-05-06 03:19:36 +01:00
|
|
|
private createMedia(
|
|
|
|
key: string,
|
|
|
|
relationshipsCount: number,
|
|
|
|
dimensions: { width: number; height: number },
|
2018-08-15 22:20:43 +01:00
|
|
|
data: Buffer | string | Uint8Array | ArrayBuffer,
|
2018-05-06 03:19:36 +01:00
|
|
|
filePath?: string,
|
|
|
|
): IMediaData {
|
2018-08-15 22:20:43 +01:00
|
|
|
if (typeof data === "string") {
|
|
|
|
data = this.convertDataURIToBinary(data);
|
|
|
|
}
|
|
|
|
|
2018-05-06 03:19:36 +01:00
|
|
|
const imageData = {
|
|
|
|
referenceId: this.map.size + relationshipsCount + 1,
|
|
|
|
stream: data,
|
|
|
|
path: filePath,
|
|
|
|
fileName: key,
|
|
|
|
dimensions: {
|
|
|
|
pixels: {
|
2018-08-09 23:22:03 +01:00
|
|
|
x: Math.round(dimensions.width),
|
|
|
|
y: Math.round(dimensions.height),
|
2018-05-06 03:19:36 +01:00
|
|
|
},
|
|
|
|
emus: {
|
2018-08-09 23:22:03 +01:00
|
|
|
x: Math.round(dimensions.width * 9525),
|
|
|
|
y: Math.round(dimensions.height * 9525),
|
2018-05-06 03:19:36 +01:00
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
2018-08-02 02:09:00 +01:00
|
|
|
|
2018-05-06 03:19:36 +01:00
|
|
|
this.map.set(key, imageData);
|
|
|
|
|
|
|
|
return imageData;
|
|
|
|
}
|
|
|
|
|
2018-08-07 01:25:28 +01:00
|
|
|
public get Array(): IMediaData[] {
|
2017-12-30 21:18:55 +00:00
|
|
|
const array = new Array<IMediaData>();
|
2017-03-25 21:50:33 +00:00
|
|
|
|
|
|
|
this.map.forEach((data) => {
|
|
|
|
array.push(data);
|
|
|
|
});
|
|
|
|
|
|
|
|
return array;
|
|
|
|
}
|
2018-08-15 22:20:43 +01:00
|
|
|
|
|
|
|
private convertDataURIToBinary(dataURI: string): Uint8Array {
|
|
|
|
// https://gist.github.com/borismus/1032746
|
|
|
|
// https://github.com/mafintosh/base64-to-uint8array
|
|
|
|
const BASE64_MARKER = ";base64,";
|
|
|
|
|
|
|
|
const base64Index = dataURI.indexOf(BASE64_MARKER) + BASE64_MARKER.length;
|
|
|
|
|
|
|
|
if (typeof atob === "function") {
|
|
|
|
return new Uint8Array(
|
|
|
|
atob(dataURI.substring(base64Index))
|
|
|
|
.split("")
|
|
|
|
.map((c) => c.charCodeAt(0)),
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
const b = require("buf" + "fer");
|
|
|
|
return new b.Buffer(dataURI, "base64");
|
|
|
|
}
|
|
|
|
}
|
2017-03-25 21:50:33 +00:00
|
|
|
}
|