1
0
Fork 0
peertube/server/lib/activitypub/actors/image.ts
kontrollanten d0800f7661
Implement avatar miniatures (#4639)
* client: remove unused file

* refactor(client/my-actor-avatar): size from input

Read size from component input instead of scss, to make it possible to
use smaller avatar images when implemented.

* implement avatar miniatures

close #4560

* fix(test): max file size

* fix(search-index): normalize res acc to avatarMini

* refactor avatars to an array

* client/search: resize channel avatar to 120

* refactor(client/videos): remove unused function

* client(actor-avatar): set default size

* fix tests and avatars full result

When findOne is used only an array containting one avatar is returned.

* update migration version and version notations

* server/search: harmonize normalizing

* Cleanup avatar miniature PR

Co-authored-by: Chocobozzz <me@florianbigard.com>
2022-02-28 08:34:43 +01:00

101 lines
2.8 KiB
TypeScript

import { Transaction } from 'sequelize/types'
import { logger } from '@server/helpers/logger'
import { ActorImageModel } from '@server/models/actor/actor-image'
import { MActorImage, MActorImages } from '@server/types/models'
import { ActorImageType } from '@shared/models'
type ImageInfo = {
name: string
fileUrl: string
height: number
width: number
onDisk?: boolean
}
async function updateActorImages (actor: MActorImages, type: ActorImageType, imagesInfo: ImageInfo[], t: Transaction) {
const avatarsOrBanners = type === ActorImageType.AVATAR
? actor.Avatars
: actor.Banners
if (imagesInfo.length === 0) {
await deleteActorImages(actor, type, t)
}
for (const imageInfo of imagesInfo) {
const oldImageModel = (avatarsOrBanners || []).find(i => i.width === imageInfo.width)
if (oldImageModel) {
// Don't update the avatar if the file URL did not change
if (imageInfo?.fileUrl && oldImageModel.fileUrl === imageInfo.fileUrl) {
continue
}
await safeDeleteActorImage(actor, oldImageModel, type, t)
}
const imageModel = await ActorImageModel.create({
filename: imageInfo.name,
onDisk: imageInfo.onDisk ?? false,
fileUrl: imageInfo.fileUrl,
height: imageInfo.height,
width: imageInfo.width,
type,
actorId: actor.id
}, { transaction: t })
addActorImage(actor, type, imageModel)
}
return actor
}
async function deleteActorImages (actor: MActorImages, type: ActorImageType, t: Transaction) {
try {
const association = buildAssociationName(type)
for (const image of actor[association]) {
await image.destroy({ transaction: t })
}
actor[association] = []
} catch (err) {
logger.error('Cannot remove old image of actor %s.', actor.url, { err })
}
return actor
}
async function safeDeleteActorImage (actor: MActorImages, toDelete: MActorImage, type: ActorImageType, t: Transaction) {
try {
await toDelete.destroy({ transaction: t })
const association = buildAssociationName(type)
actor[association] = actor[association].filter(image => image.id !== toDelete.id)
} catch (err) {
logger.error('Cannot remove old actor image of actor %s.', actor.url, { err })
}
}
// ---------------------------------------------------------------------------
export {
ImageInfo,
updateActorImages,
deleteActorImages
}
// ---------------------------------------------------------------------------
function addActorImage (actor: MActorImages, type: ActorImageType, imageModel: MActorImage) {
const association = buildAssociationName(type)
if (!actor[association]) actor[association] = []
actor[association].push(imageModel)
}
function buildAssociationName (type: ActorImageType) {
return type === ActorImageType.AVATAR
? 'Avatars'
: 'Banners'
}