forked from elizaOS/eliza
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathavatar-panel.tsx
103 lines (92 loc) · 3.49 KB
/
avatar-panel.tsx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
import { Button } from '@/components/ui/button';
import type { Agent } from '@elizaos/core';
import { Image as ImageIcon, Upload, X } from 'lucide-react';
import { useRef, useState, useEffect } from 'react';
import { compressImage } from '@/lib/utils';
interface AvatarPanelProps {
characterValue: Agent;
setCharacterValue: {
updateAvatar?: (avatarUrl: string) => void;
updateSetting?: <T>(path: string, value: T) => void;
updateField?: <T>(path: string, value: T) => void;
[key: string]: any;
};
}
export default function AvatarPanel({ characterValue, setCharacterValue }: AvatarPanelProps) {
const [avatar, setAvatar] = useState<string | null>(characterValue?.settings?.avatar || null);
const [hasChanged, setHasChanged] = useState(false);
const fileInputRef = useRef<HTMLInputElement>(null);
// Reset the change flag when component initializes or character changes
useEffect(() => {
setAvatar(characterValue?.settings?.avatar || null);
setHasChanged(false);
}, [characterValue.id]);
const handleFileUpload = async (event: React.ChangeEvent<HTMLInputElement>) => {
const file = event.target.files?.[0];
if (file) {
try {
const compressedImage = await compressImage(file);
setAvatar(compressedImage);
setHasChanged(true);
// Only update when there's a real change
updateCharacterAvatar(compressedImage);
} catch (error) {
console.error('Error compressing image:', error);
}
}
};
const handleRemoveAvatar = () => {
if (avatar) {
setAvatar(null);
setHasChanged(true);
updateCharacterAvatar('');
}
};
// Centralized update function to avoid code duplication
const updateCharacterAvatar = (avatarUrl: string) => {
if (setCharacterValue.updateAvatar) {
// Use the specialized method for avatar updates when available
setCharacterValue.updateAvatar(avatarUrl);
} else if (setCharacterValue.updateSetting) {
// Use updateSetting as fallback
setCharacterValue.updateSetting('avatar', avatarUrl);
} else if (setCharacterValue.updateField) {
// Last resort - use the generic field update
setCharacterValue.updateField('settings.avatar', avatarUrl);
}
};
return (
<div className="rounded-lg w-full">
<h2 className="text-xl font-bold mb-4 pb-5 ml-1">Avatar Settings</h2>
<div className="flex flex-col items-center gap-4 pb-4">
{avatar ? (
<div className="relative w-64 h-64">
<img src={avatar} alt="Character Avatar" className="object-cover rounded-lg border" />
<button
className="absolute -top-2 -right-2 bg-white p-1 rounded-full shadow-md"
onClick={handleRemoveAvatar}
type="button"
>
<X className="w-5 h-5 text-card" />
</button>
</div>
) : (
<div className="w-64 h-64 flex items-center justify-center border border-dashed rounded-lg text-gray-500">
<ImageIcon className="w-10 h-10" />
</div>
)}
<input
type="file"
accept="image/*"
className="hidden"
ref={fileInputRef}
onChange={handleFileUpload}
/>
<Button className="flex items-center gap-2" onClick={() => fileInputRef.current?.click()}>
<Upload className="w-5 h-5" /> Upload Avatar
</Button>
{hasChanged && <p className="text-xs text-blue-500">Avatar has been updated</p>}
</div>
</div>
);
}