218 lines
6.0 KiB
TypeScript
218 lines
6.0 KiB
TypeScript
/**
|
|
* Domain Entity: AvatarGenerationRequest
|
|
*
|
|
* Represents a request to generate a racing avatar from a face photo.
|
|
*/
|
|
|
|
export type RacingSuitColor =
|
|
| 'red'
|
|
| 'blue'
|
|
| 'green'
|
|
| 'yellow'
|
|
| 'orange'
|
|
| 'purple'
|
|
| 'black'
|
|
| 'white'
|
|
| 'pink'
|
|
| 'cyan';
|
|
|
|
export type AvatarStyle =
|
|
| 'realistic'
|
|
| 'cartoon'
|
|
| 'pixel-art';
|
|
|
|
export type AvatarGenerationStatus =
|
|
| 'pending'
|
|
| 'validating'
|
|
| 'generating'
|
|
| 'completed'
|
|
| 'failed';
|
|
|
|
export interface AvatarGenerationRequestProps {
|
|
id: string;
|
|
userId: string;
|
|
facePhotoUrl: string;
|
|
suitColor: RacingSuitColor;
|
|
style: AvatarStyle;
|
|
status: AvatarGenerationStatus;
|
|
generatedAvatarUrls: string[];
|
|
selectedAvatarIndex?: number;
|
|
errorMessage?: string;
|
|
createdAt: Date;
|
|
updatedAt: Date;
|
|
}
|
|
|
|
export class AvatarGenerationRequest {
|
|
readonly id: string;
|
|
readonly userId: string;
|
|
readonly facePhotoUrl: string;
|
|
readonly suitColor: RacingSuitColor;
|
|
readonly style: AvatarStyle;
|
|
private _status: AvatarGenerationStatus;
|
|
private _generatedAvatarUrls: string[];
|
|
private _selectedAvatarIndex?: number;
|
|
private _errorMessage?: string;
|
|
readonly createdAt: Date;
|
|
private _updatedAt: Date;
|
|
|
|
private constructor(props: AvatarGenerationRequestProps) {
|
|
this.id = props.id;
|
|
this.userId = props.userId;
|
|
this.facePhotoUrl = props.facePhotoUrl;
|
|
this.suitColor = props.suitColor;
|
|
this.style = props.style;
|
|
this._status = props.status;
|
|
this._generatedAvatarUrls = [...props.generatedAvatarUrls];
|
|
this._selectedAvatarIndex = props.selectedAvatarIndex;
|
|
this._errorMessage = props.errorMessage;
|
|
this.createdAt = props.createdAt;
|
|
this._updatedAt = props.updatedAt;
|
|
}
|
|
|
|
static create(props: {
|
|
id: string;
|
|
userId: string;
|
|
facePhotoUrl: string;
|
|
suitColor: RacingSuitColor;
|
|
style?: AvatarStyle;
|
|
}): AvatarGenerationRequest {
|
|
if (!props.userId) {
|
|
throw new Error('User ID is required');
|
|
}
|
|
if (!props.facePhotoUrl) {
|
|
throw new Error('Face photo URL is required');
|
|
}
|
|
|
|
const now = new Date();
|
|
return new AvatarGenerationRequest({
|
|
id: props.id,
|
|
userId: props.userId,
|
|
facePhotoUrl: props.facePhotoUrl,
|
|
suitColor: props.suitColor,
|
|
style: props.style ?? 'realistic',
|
|
status: 'pending',
|
|
generatedAvatarUrls: [],
|
|
createdAt: now,
|
|
updatedAt: now,
|
|
});
|
|
}
|
|
|
|
static reconstitute(props: AvatarGenerationRequestProps): AvatarGenerationRequest {
|
|
return new AvatarGenerationRequest(props);
|
|
}
|
|
|
|
get status(): AvatarGenerationStatus {
|
|
return this._status;
|
|
}
|
|
|
|
get generatedAvatarUrls(): string[] {
|
|
return [...this._generatedAvatarUrls];
|
|
}
|
|
|
|
get selectedAvatarIndex(): number | undefined {
|
|
return this._selectedAvatarIndex;
|
|
}
|
|
|
|
get selectedAvatarUrl(): string | undefined {
|
|
if (this._selectedAvatarIndex !== undefined && this._generatedAvatarUrls[this._selectedAvatarIndex]) {
|
|
return this._generatedAvatarUrls[this._selectedAvatarIndex];
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
get errorMessage(): string | undefined {
|
|
return this._errorMessage;
|
|
}
|
|
|
|
get updatedAt(): Date {
|
|
return this._updatedAt;
|
|
}
|
|
|
|
markAsValidating(): void {
|
|
if (this._status !== 'pending') {
|
|
throw new Error('Can only start validation from pending status');
|
|
}
|
|
this._status = 'validating';
|
|
this._updatedAt = new Date();
|
|
}
|
|
|
|
markAsGenerating(): void {
|
|
if (this._status !== 'validating') {
|
|
throw new Error('Can only start generation from validating status');
|
|
}
|
|
this._status = 'generating';
|
|
this._updatedAt = new Date();
|
|
}
|
|
|
|
completeWithAvatars(avatarUrls: string[]): void {
|
|
if (avatarUrls.length === 0) {
|
|
throw new Error('At least one avatar URL is required');
|
|
}
|
|
this._status = 'completed';
|
|
this._generatedAvatarUrls = [...avatarUrls];
|
|
this._updatedAt = new Date();
|
|
}
|
|
|
|
fail(errorMessage: string): void {
|
|
this._status = 'failed';
|
|
this._errorMessage = errorMessage;
|
|
this._updatedAt = new Date();
|
|
}
|
|
|
|
selectAvatar(index: number): void {
|
|
if (this._status !== 'completed') {
|
|
throw new Error('Can only select avatar when generation is completed');
|
|
}
|
|
if (index < 0 || index >= this._generatedAvatarUrls.length) {
|
|
throw new Error('Invalid avatar index');
|
|
}
|
|
this._selectedAvatarIndex = index;
|
|
this._updatedAt = new Date();
|
|
}
|
|
|
|
/**
|
|
* Build the AI prompt for avatar generation.
|
|
* We control the prompt completely - users cannot enter free text.
|
|
*/
|
|
buildPrompt(): string {
|
|
const colorDescriptions: Record<RacingSuitColor, string> = {
|
|
red: 'vibrant racing red',
|
|
blue: 'deep motorsport blue',
|
|
green: 'racing green',
|
|
yellow: 'bright championship yellow',
|
|
orange: 'McLaren-style papaya orange',
|
|
purple: 'royal purple',
|
|
black: 'stealth black',
|
|
white: 'clean white',
|
|
pink: 'hot pink',
|
|
cyan: 'electric cyan',
|
|
};
|
|
|
|
const styleDescriptions: Record<AvatarStyle, string> = {
|
|
realistic: 'photorealistic, professional motorsport portrait',
|
|
cartoon: 'stylized cartoon racing character',
|
|
'pixel-art': '8-bit pixel art retro racing avatar',
|
|
};
|
|
|
|
const suitColorDesc = colorDescriptions[this.suitColor];
|
|
const styleDesc = styleDescriptions[this.style];
|
|
|
|
return `Create a ${styleDesc} of a racing driver wearing a ${suitColorDesc} racing suit with matching helmet. The driver should look professional and confident, as if posing for a team photo. Background should be a subtle racing paddock or garage setting. High quality, well-lit, professional motorsport photography style.`;
|
|
}
|
|
|
|
toProps(): AvatarGenerationRequestProps {
|
|
return {
|
|
id: this.id,
|
|
userId: this.userId,
|
|
facePhotoUrl: this.facePhotoUrl,
|
|
suitColor: this.suitColor,
|
|
style: this.style,
|
|
status: this._status,
|
|
generatedAvatarUrls: [...this._generatedAvatarUrls],
|
|
selectedAvatarIndex: this._selectedAvatarIndex,
|
|
errorMessage: this._errorMessage,
|
|
createdAt: this.createdAt,
|
|
updatedAt: this._updatedAt,
|
|
};
|
|
}
|
|
} |