2023-07-27 14:31:52 +09:00
|
|
|
/*
|
|
|
|
* SPDX-FileCopyrightText: syuilo and other misskey contributors
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
2023-08-05 10:33:00 +09:00
|
|
|
import { Injectable } from '@nestjs/common';
|
2022-09-18 03:27:08 +09:00
|
|
|
import sharp from 'sharp';
|
|
|
|
|
|
|
|
export type IImage = {
|
|
|
|
data: Buffer;
|
|
|
|
ext: string | null;
|
|
|
|
type: string;
|
|
|
|
};
|
2022-12-30 12:00:50 +09:00
|
|
|
|
2023-01-26 16:06:29 +09:00
|
|
|
export type IImageStream = {
|
|
|
|
data: Readable;
|
|
|
|
ext: string | null;
|
|
|
|
type: string;
|
|
|
|
};
|
|
|
|
|
2023-03-11 14:11:40 +09:00
|
|
|
export type IImageSharp = {
|
|
|
|
data: sharp.Sharp;
|
|
|
|
ext: string | null;
|
|
|
|
type: string;
|
|
|
|
};
|
|
|
|
|
|
|
|
export type IImageStreamable = IImage | IImageStream | IImageSharp;
|
2023-01-26 16:06:29 +09:00
|
|
|
|
2022-12-30 12:00:50 +09:00
|
|
|
export const webpDefault: sharp.WebpOptions = {
|
2023-03-11 14:11:40 +09:00
|
|
|
quality: 77,
|
2022-12-30 12:00:50 +09:00
|
|
|
alphaQuality: 95,
|
|
|
|
lossless: false,
|
|
|
|
nearLossless: false,
|
|
|
|
smartSubsample: true,
|
|
|
|
mixed: true,
|
2023-03-11 14:11:40 +09:00
|
|
|
effort: 2,
|
|
|
|
};
|
|
|
|
|
|
|
|
export const avifDefault: sharp.AvifOptions = {
|
|
|
|
quality: 60,
|
|
|
|
lossless: false,
|
|
|
|
effort: 2,
|
2022-12-30 12:00:50 +09:00
|
|
|
};
|
|
|
|
|
2022-12-04 15:03:09 +09:00
|
|
|
import { bindThis } from '@/decorators.js';
|
2023-01-26 16:06:29 +09:00
|
|
|
import { Readable } from 'node:stream';
|
2022-09-18 03:27:08 +09:00
|
|
|
|
|
|
|
@Injectable()
|
|
|
|
export class ImageProcessingService {
|
|
|
|
constructor(
|
|
|
|
) {
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-03-11 14:11:40 +09:00
|
|
|
* Convert to WebP
|
2022-09-18 03:27:08 +09:00
|
|
|
* with resize, remove metadata, resolve orientation, stop animation
|
|
|
|
*/
|
2022-12-04 15:03:09 +09:00
|
|
|
@bindThis
|
2023-03-11 14:11:40 +09:00
|
|
|
public async convertToWebp(path: string, width: number, height: number, options: sharp.WebpOptions = webpDefault): Promise<IImage> {
|
|
|
|
return this.convertSharpToWebp(sharp(path), width, height, options);
|
2022-09-18 03:27:08 +09:00
|
|
|
}
|
|
|
|
|
2022-12-04 15:03:09 +09:00
|
|
|
@bindThis
|
2023-03-11 14:11:40 +09:00
|
|
|
public async convertSharpToWebp(sharp: sharp.Sharp, width: number, height: number, options: sharp.WebpOptions = webpDefault): Promise<IImage> {
|
|
|
|
const result = this.convertSharpToWebpStream(sharp, width, height, options);
|
|
|
|
|
|
|
|
return {
|
|
|
|
data: await result.data.toBuffer(),
|
|
|
|
ext: result.ext,
|
|
|
|
type: result.type,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
@bindThis
|
|
|
|
public convertToWebpStream(path: string, width: number, height: number, options: sharp.WebpOptions = webpDefault): IImageSharp {
|
|
|
|
return this.convertSharpToWebpStream(sharp(path), width, height, options);
|
|
|
|
}
|
|
|
|
|
|
|
|
@bindThis
|
|
|
|
public convertSharpToWebpStream(sharp: sharp.Sharp, width: number, height: number, options: sharp.WebpOptions = webpDefault): IImageSharp {
|
|
|
|
const data = sharp
|
2022-09-18 03:27:08 +09:00
|
|
|
.resize(width, height, {
|
|
|
|
fit: 'inside',
|
|
|
|
withoutEnlargement: true,
|
|
|
|
})
|
|
|
|
.rotate()
|
2023-03-11 14:11:40 +09:00
|
|
|
.webp(options);
|
2022-09-18 03:27:08 +09:00
|
|
|
|
|
|
|
return {
|
|
|
|
data,
|
2023-03-11 14:11:40 +09:00
|
|
|
ext: 'webp',
|
|
|
|
type: 'image/webp',
|
2022-09-18 03:27:08 +09:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-03-11 14:11:40 +09:00
|
|
|
* Convert to Avif
|
2022-09-18 03:27:08 +09:00
|
|
|
* with resize, remove metadata, resolve orientation, stop animation
|
|
|
|
*/
|
2022-12-04 15:03:09 +09:00
|
|
|
@bindThis
|
2023-03-11 14:11:40 +09:00
|
|
|
public async convertToAvif(path: string, width: number, height: number, options: sharp.AvifOptions = avifDefault): Promise<IImage> {
|
|
|
|
return this.convertSharpToAvif(sharp(path), width, height, options);
|
2022-09-18 03:27:08 +09:00
|
|
|
}
|
|
|
|
|
2022-12-04 15:03:09 +09:00
|
|
|
@bindThis
|
2023-03-11 14:11:40 +09:00
|
|
|
public async convertSharpToAvif(sharp: sharp.Sharp, width: number, height: number, options: sharp.AvifOptions = avifDefault): Promise<IImage> {
|
|
|
|
const result = this.convertSharpToAvifStream(sharp, width, height, options);
|
2022-09-18 03:27:08 +09:00
|
|
|
|
|
|
|
return {
|
2023-03-11 14:11:40 +09:00
|
|
|
data: await result.data.toBuffer(),
|
|
|
|
ext: result.ext,
|
|
|
|
type: result.type,
|
2022-09-18 03:27:08 +09:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-01-26 16:06:29 +09:00
|
|
|
@bindThis
|
2023-03-11 14:11:40 +09:00
|
|
|
public convertToAvifStream(path: string, width: number, height: number, options: sharp.AvifOptions = avifDefault): IImageSharp {
|
|
|
|
return this.convertSharpToAvifStream(sharp(path), width, height, options);
|
2023-01-26 16:06:29 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
@bindThis
|
2023-03-11 14:11:40 +09:00
|
|
|
public convertSharpToAvifStream(sharp: sharp.Sharp, width: number, height: number, options: sharp.AvifOptions = avifDefault): IImageSharp {
|
2023-01-26 16:06:29 +09:00
|
|
|
const data = sharp
|
|
|
|
.resize(width, height, {
|
|
|
|
fit: 'inside',
|
|
|
|
withoutEnlargement: true,
|
|
|
|
})
|
|
|
|
.rotate()
|
2023-03-11 14:11:40 +09:00
|
|
|
.avif(options);
|
2023-01-26 16:06:29 +09:00
|
|
|
|
|
|
|
return {
|
|
|
|
data,
|
2023-03-11 14:11:40 +09:00
|
|
|
ext: 'avif',
|
|
|
|
type: 'image/avif',
|
2023-01-26 16:06:29 +09:00
|
|
|
};
|
|
|
|
}
|
2023-03-11 14:11:40 +09:00
|
|
|
|
2022-09-18 03:27:08 +09:00
|
|
|
/**
|
|
|
|
* Convert to PNG
|
|
|
|
* with resize, remove metadata, resolve orientation, stop animation
|
|
|
|
*/
|
2022-12-04 15:03:09 +09:00
|
|
|
@bindThis
|
2022-09-18 03:27:08 +09:00
|
|
|
public async convertToPng(path: string, width: number, height: number): Promise<IImage> {
|
2023-03-11 14:11:40 +09:00
|
|
|
return this.convertSharpToPng(sharp(path), width, height);
|
2022-09-18 03:27:08 +09:00
|
|
|
}
|
|
|
|
|
2022-12-04 15:03:09 +09:00
|
|
|
@bindThis
|
2022-09-18 03:27:08 +09:00
|
|
|
public async convertSharpToPng(sharp: sharp.Sharp, width: number, height: number): Promise<IImage> {
|
|
|
|
const data = await sharp
|
|
|
|
.resize(width, height, {
|
|
|
|
fit: 'inside',
|
|
|
|
withoutEnlargement: true,
|
|
|
|
})
|
|
|
|
.rotate()
|
|
|
|
.png()
|
|
|
|
.toBuffer();
|
|
|
|
|
|
|
|
return {
|
|
|
|
data,
|
|
|
|
ext: 'png',
|
|
|
|
type: 'image/png',
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|