Files
docx-js/src/file/media/media.ts

139 lines
3.9 KiB
TypeScript
Raw Normal View History

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";
interface IHackedFile {
// tslint:disable-next-line:readonly-keyword
currentRelationshipId: number;
}
export class Media {
2018-08-22 10:39:13 -03:00
public static addImage(
file: File,
buffer: Buffer | string | Uint8Array | ArrayBuffer,
width?: number,
height?: number,
drawingOptions?: IDrawingOptions,
): Image {
// 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);
file.DocumentRelationships.createRelationship(
mediaData.referenceId,
"http://schemas.openxmlformats.org/officeDocument/2006/relationships/image",
`media/${mediaData.fileName}`,
);
2018-08-14 11:27:08 -03:00
return new Image(new ImageParagraph(mediaData, drawingOptions));
}
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-01-29 01:55:25 +00:00
private readonly map: Map<string, IMediaData>;
constructor() {
this.map = new Map<string, IMediaData>();
}
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-08-15 22:20:43 +01:00
return this.createMedia(
key,
referenceId,
{
width: width,
2018-05-06 03:19:36 +01:00
height: height,
2018-08-15 22:20:43 +01:00
},
buffer,
);
}
2018-05-06 03:19:36 +01:00
private createMedia(
key: string,
relationshipsCount: number,
dimensions: { readonly width: number; readonly 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 {
const newData = typeof data === "string" ? this.convertDataURIToBinary(data) : data;
2018-08-15 22:20:43 +01:00
const imageData: IMediaData = {
2018-10-23 00:31:51 +01:00
referenceId: relationshipsCount,
stream: newData,
2018-05-06 03:19:36 +01:00
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-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[] {
const array = new Array<IMediaData>();
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");
}
}
}