Spaces:
Sleeping
Sleeping
MusaedMusaedSadeqMusaedAl-Fareh225739
commited on
Commit
Β·
52a9a25
1
Parent(s):
1df4a51
README file for frontend
Browse files- avatar-frontend/README.md +1208 -18
avatar-frontend/README.md
CHANGED
|
@@ -1,36 +1,1226 @@
|
|
| 1 |
-
|
| 2 |
|
| 3 |
-
|
| 4 |
|
| 5 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 6 |
|
| 7 |
```bash
|
| 8 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 9 |
# or
|
| 10 |
-
|
|
|
|
|
|
|
|
|
|
| 11 |
# or
|
| 12 |
pnpm dev
|
| 13 |
-
|
| 14 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 15 |
```
|
| 16 |
|
| 17 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 18 |
|
| 19 |
-
|
| 20 |
|
| 21 |
-
|
| 22 |
|
| 23 |
-
|
|
|
|
|
|
|
|
|
|
| 24 |
|
| 25 |
-
|
|
|
|
| 26 |
|
| 27 |
-
|
| 28 |
-
- [Learn Next.js](https://nextjs.org/learn) - an interactive Next.js tutorial.
|
| 29 |
|
| 30 |
-
|
| 31 |
|
| 32 |
-
|
|
|
|
|
|
|
|
|
|
| 33 |
|
| 34 |
-
|
| 35 |
|
| 36 |
-
|
|
|
|
|
|
|
|
|
| 1 |
+
# MrrrMe Avatar Frontend
|
| 2 |
|
| 3 |
+
**Next.js 16 Web Application with 3D Avatar Integration**
|
| 4 |
|
| 5 |
+
Real-time emotion detection interface with customizable 3D avatars, WebSocket communication, and multi-lingual support.
|
| 6 |
+
|
| 7 |
+
---
|
| 8 |
+
|
| 9 |
+
## Technology Stack
|
| 10 |
+
|
| 11 |
+
**Framework**: Next.js 16.0.0 (App Router)
|
| 12 |
+
**UI Library**: React 19.2.0
|
| 13 |
+
**Language**: TypeScript 5.9+
|
| 14 |
+
**3D Engine**: React Three Fiber 9.4.0 + Three.js 0.180.0
|
| 15 |
+
**Avatar SDK**: Avaturn SDK (CDN)
|
| 16 |
+
**Styling**: Tailwind CSS v4 + Custom CSS Variables
|
| 17 |
+
**Build Tool**: Next.js Standalone Output
|
| 18 |
+
|
| 19 |
+
---
|
| 20 |
+
|
| 21 |
+
## Project Structure
|
| 22 |
+
|
| 23 |
+
```
|
| 24 |
+
avatar-frontend/
|
| 25 |
+
β
|
| 26 |
+
βββ app/ # Next.js App Router
|
| 27 |
+
β βββ api/
|
| 28 |
+
β β βββ avaturn-proxy/
|
| 29 |
+
β β βββ route.ts # CORS proxy for avatar assets
|
| 30 |
+
β β
|
| 31 |
+
β βββ app/ # Main application (authenticated)
|
| 32 |
+
β β βββ page.tsx # Avatar UI + WebSocket + Emotion detection
|
| 33 |
+
β β
|
| 34 |
+
β βββ login/
|
| 35 |
+
β β βββ page.tsx # Authentication page (signup/login)
|
| 36 |
+
β β
|
| 37 |
+
β βββ layout.tsx # Root layout (fonts, metadata)
|
| 38 |
+
β βββ page.tsx # Landing page (marketing)
|
| 39 |
+
β βββ globals.css # Design system (light/dark mode)
|
| 40 |
+
β
|
| 41 |
+
βββ public/
|
| 42 |
+
β βββ idle-animation.glb # Avatar idle animation (Git LFS, 199 KB)
|
| 43 |
+
β βββ next.svg # Next.js logo
|
| 44 |
+
β βββ vercel.svg # Vercel logo
|
| 45 |
+
β βββ file.svg # UI icons
|
| 46 |
+
β βββ globe.svg
|
| 47 |
+
β βββ window.svg
|
| 48 |
+
β
|
| 49 |
+
βββ package.json # Node dependencies
|
| 50 |
+
βββ tsconfig.json # TypeScript configuration
|
| 51 |
+
βββ next.config.ts # Next.js config (standalone output)
|
| 52 |
+
βββ postcss.config.mjs # PostCSS config
|
| 53 |
+
βββ eslint.config.mjs # ESLint config (Next.js 16)
|
| 54 |
+
βββ .gitignore
|
| 55 |
+
```
|
| 56 |
+
|
| 57 |
+
---
|
| 58 |
+
|
| 59 |
+
## Key Features
|
| 60 |
+
|
| 61 |
+
### 1. 3D Avatar System
|
| 62 |
+
- **Avaturn SDK Integration**: Create custom avatars via embedded modal
|
| 63 |
+
- **React Three Fiber**: Real-time 3D rendering with WebGL
|
| 64 |
+
- **Lip-Sync Animation**: Viseme-based mouth animation synchronized with TTS
|
| 65 |
+
- **Idle Animations**: Natural breathing and blinking using GLB animations
|
| 66 |
+
- **Customizable Positioning**: Adjustable camera angle, position, and scale
|
| 67 |
+
|
| 68 |
+
**Avatar Pipeline**:
|
| 69 |
+
```
|
| 70 |
+
User clicks "Create Avatar"
|
| 71 |
+
β Avaturn SDK modal opens
|
| 72 |
+
β User customizes avatar
|
| 73 |
+
β Exports GLB URL
|
| 74 |
+
β CORS proxy fetches asset
|
| 75 |
+
β Three.js loads and renders
|
| 76 |
+
β Visemes drive morph targets
|
| 77 |
+
```
|
| 78 |
+
|
| 79 |
+
### 2. Multi-Modal Emotion Detection
|
| 80 |
+
- **Facial Emotion**: Real-time ViT-Face-Expression analysis
|
| 81 |
+
- **Voice Emotion**: HuBERT-Large prosody detection
|
| 82 |
+
- **Text Sentiment**: DistilRoBERTa with rule overrides
|
| 83 |
+
- **Fusion Display**: Combined emotion with confidence scores
|
| 84 |
+
|
| 85 |
+
**Emotion Test Modal**:
|
| 86 |
+
- Live probability distribution (4 emotions)
|
| 87 |
+
- Confidence percentage
|
| 88 |
+
- Quality score
|
| 89 |
+
- Prediction counter
|
| 90 |
+
|
| 91 |
+
### 3. WebSocket Communication
|
| 92 |
+
- **Protocol**: `ws://` (dev) or `wss://` (production)
|
| 93 |
+
- **Authentication**: Token-based session management
|
| 94 |
+
- **Real-Time Streams**: Video (200ms), Audio (500ms), Transcription
|
| 95 |
+
- **Bidirectional**: Client sends frames, server sends emotions + responses
|
| 96 |
+
|
| 97 |
+
**Message Types**:
|
| 98 |
+
```typescript
|
| 99 |
+
// Client β Server
|
| 100 |
+
type: "auth" | "video_frame" | "audio_chunk" | "speech_end" | "preferences"
|
| 101 |
+
|
| 102 |
+
// Server β Client
|
| 103 |
+
type: "authenticated" | "face_emotion" | "voice_emotion" | "llm_response" | "error"
|
| 104 |
+
```
|
| 105 |
+
|
| 106 |
+
### 4. Conversation Interface
|
| 107 |
+
- **Message History**: Persistent chat with timestamps
|
| 108 |
+
- **Speech Recognition**: Web Speech API (continuous, interim results)
|
| 109 |
+
- **Text Input**: Keyboard fallback for typing
|
| 110 |
+
- **Auto-Greeting**: AI initiates conversation on connect
|
| 111 |
+
|
| 112 |
+
### 5. User Preferences
|
| 113 |
+
- **Languages**: English, Dutch (switch mid-conversation)
|
| 114 |
+
- **Voice**: Male (Damien Black), Female (Ana Florence)
|
| 115 |
+
- **Personality**: Therapist (empathetic) or Coach (action-focused)
|
| 116 |
+
- **Theme**: Light/Dark mode with smooth transitions
|
| 117 |
+
|
| 118 |
+
### 6. Privacy & Authentication
|
| 119 |
+
- **Session-Based Auth**: Token stored in localStorage
|
| 120 |
+
- **No Data Upload**: Only video/audio chunks sent for processing
|
| 121 |
+
- **Logout Cleanup**: Properly closes WebSocket and media streams
|
| 122 |
+
- **CORS Proxy**: Secure avatar asset loading
|
| 123 |
+
|
| 124 |
+
---
|
| 125 |
+
|
| 126 |
+
## Installation
|
| 127 |
+
|
| 128 |
+
### Prerequisites
|
| 129 |
+
|
| 130 |
+
- Node.js 20+ (LTS recommended)
|
| 131 |
+
- npm 10+ or pnpm 9+
|
| 132 |
+
- Git LFS (for idle-animation.glb)
|
| 133 |
+
|
| 134 |
+
### Setup
|
| 135 |
|
| 136 |
```bash
|
| 137 |
+
# Navigate to frontend directory
|
| 138 |
+
cd avatar-frontend
|
| 139 |
+
|
| 140 |
+
# Install Git LFS (if not installed)
|
| 141 |
+
git lfs install
|
| 142 |
+
git lfs pull
|
| 143 |
+
|
| 144 |
+
# Install dependencies
|
| 145 |
+
npm install
|
| 146 |
# or
|
| 147 |
+
pnpm install
|
| 148 |
+
|
| 149 |
+
# Run development server
|
| 150 |
+
npm run dev
|
| 151 |
# or
|
| 152 |
pnpm dev
|
| 153 |
+
|
| 154 |
+
# Open browser
|
| 155 |
+
open http://localhost:3000
|
| 156 |
+
```
|
| 157 |
+
|
| 158 |
+
### Environment Variables
|
| 159 |
+
|
| 160 |
+
Create `.env.local`:
|
| 161 |
+
|
| 162 |
+
```bash
|
| 163 |
+
# Backend WebSocket URL (auto-detected if not set)
|
| 164 |
+
NEXT_PUBLIC_BACKEND_URL=http://localhost:8000
|
| 165 |
+
|
| 166 |
+
# Avatar TTS URL (auto-detected if not set)
|
| 167 |
+
NEXT_PUBLIC_AVATAR_URL=http://localhost:8765
|
| 168 |
+
```
|
| 169 |
+
|
| 170 |
+
---
|
| 171 |
+
|
| 172 |
+
## Configuration
|
| 173 |
+
|
| 174 |
+
### WebSocket Connection
|
| 175 |
+
|
| 176 |
+
File: `app/app/page.tsx`
|
| 177 |
+
|
| 178 |
+
```typescript
|
| 179 |
+
const getWebSocketURL = () => {
|
| 180 |
+
if (typeof window === "undefined") return "ws://localhost:8000/ws";
|
| 181 |
+
const protocol = window.location.protocol === "https:" ? "wss:" : "ws:";
|
| 182 |
+
return `${protocol}//${window.location.host}/ws`;
|
| 183 |
+
};
|
| 184 |
+
```
|
| 185 |
+
|
| 186 |
+
Automatically uses:
|
| 187 |
+
- `ws://localhost:3000/ws` (local dev)
|
| 188 |
+
- `wss://your-domain.com/ws` (production)
|
| 189 |
+
|
| 190 |
+
### Avatar Positioning
|
| 191 |
+
|
| 192 |
+
File: `app/app/page.tsx` (lines 493-495)
|
| 193 |
+
|
| 194 |
+
```typescript
|
| 195 |
+
const [avatarPosition] = useState({ x: -0.01, y: -2.12, z: 0.06 });
|
| 196 |
+
const [avatarRotation] = useState({ x: 0.00, y: 0.51, z: 0.00 });
|
| 197 |
+
const [avatarScale] = useState(1.25);
|
| 198 |
+
```
|
| 199 |
+
|
| 200 |
+
Adjust these values to change camera framing.
|
| 201 |
+
|
| 202 |
+
### Theme Customization
|
| 203 |
+
|
| 204 |
+
File: `app/globals.css`
|
| 205 |
+
|
| 206 |
+
**Light Mode**:
|
| 207 |
+
```css
|
| 208 |
+
:root {
|
| 209 |
+
--background: #ffffff;
|
| 210 |
+
--foreground: #1d1d1f;
|
| 211 |
+
--accent-gradient-from: #007aff;
|
| 212 |
+
--accent-gradient-to: #5e5ce6;
|
| 213 |
+
--surface: rgba(255, 255, 255, 0.72);
|
| 214 |
+
--border: rgba(0, 0, 0, 0.06);
|
| 215 |
+
/* ... */
|
| 216 |
+
}
|
| 217 |
+
```
|
| 218 |
+
|
| 219 |
+
**Dark Mode**:
|
| 220 |
+
```css
|
| 221 |
+
:root.dark-mode {
|
| 222 |
+
--background: #000000;
|
| 223 |
+
--foreground: #f5f5f7;
|
| 224 |
+
--accent-gradient-from: #0a84ff;
|
| 225 |
+
--accent-gradient-to: #5e5ce6;
|
| 226 |
+
--surface: rgba(28, 28, 30, 0.72);
|
| 227 |
+
--border: rgba(255, 255, 255, 0.08);
|
| 228 |
+
/* ... */
|
| 229 |
+
}
|
| 230 |
+
```
|
| 231 |
+
|
| 232 |
+
---
|
| 233 |
+
|
| 234 |
+
## Component Architecture
|
| 235 |
+
|
| 236 |
+
### Main Application (`app/app/page.tsx`)
|
| 237 |
+
|
| 238 |
+
**State Management**:
|
| 239 |
+
```typescript
|
| 240 |
+
// Authentication
|
| 241 |
+
const [username, setUsername] = useState("");
|
| 242 |
+
const [userToken, setUserToken] = useState("");
|
| 243 |
+
|
| 244 |
+
// Emotion Detection
|
| 245 |
+
const [faceEmotion, setFaceEmotion] = useState("Neutral");
|
| 246 |
+
const [voiceEmotion, setVoiceEmotion] = useState("Neutral");
|
| 247 |
+
|
| 248 |
+
// Avatar
|
| 249 |
+
const [liveBlend, setLiveBlend] = useState<Blend>({});
|
| 250 |
+
const [avatarUrl, setAvatarUrl] = useState(DEFAULT_AVATAR);
|
| 251 |
+
|
| 252 |
+
// UI State
|
| 253 |
+
const [showHistory, setShowHistory] = useState(false);
|
| 254 |
+
const [showSettings, setShowSettings] = useState(false);
|
| 255 |
+
const [isAvatarSpeaking, setIsAvatarSpeaking] = useState(false);
|
| 256 |
+
|
| 257 |
+
// Preferences
|
| 258 |
+
const [selectedLanguage, setSelectedLanguage] = useState<"en" | "nl">("en");
|
| 259 |
+
const [selectedVoice, setSelectedVoice] = useState<"male" | "female">("female");
|
| 260 |
+
const [selectedPersonality, setSelectedPersonality] = useState<"therapist" | "coach">("therapist");
|
| 261 |
+
```
|
| 262 |
+
|
| 263 |
+
**Key Functions**:
|
| 264 |
+
|
| 265 |
+
1. `connectWebSocket()` - Establishes WebSocket connection with auth
|
| 266 |
+
2. `startCapture()` - Initializes camera/microphone access
|
| 267 |
+
3. `startVideoCapture()` - Sends video frames at 5 FPS (200ms intervals)
|
| 268 |
+
4. `startAudioCapture()` - Sends audio chunks every 500ms
|
| 269 |
+
5. `startSpeechRecognition()` - Web Speech API for transcription
|
| 270 |
+
6. `playAvatarResponse()` - Syncs audio + visemes for lip-sync
|
| 271 |
+
|
| 272 |
+
### Avatar Component (`app/app/page.tsx` lines 171-227)
|
| 273 |
+
|
| 274 |
+
```typescript
|
| 275 |
+
function Avatar({ liveBlend, avatarUrl, position, rotation, scale }) {
|
| 276 |
+
const gltf = useGLTF(avatarUrl);
|
| 277 |
+
const { scene, animations } = gltf;
|
| 278 |
+
const idleAnimGLTF = useGLTF('/idle-animation.glb');
|
| 279 |
+
|
| 280 |
+
// Find all meshes with morph targets (for lip-sync)
|
| 281 |
+
const morphMeshes = useMemo(() => {
|
| 282 |
+
const arr = [];
|
| 283 |
+
scene.traverse((o) => {
|
| 284 |
+
if (o.morphTargetDictionary && o.morphTargetInfluences) {
|
| 285 |
+
arr.push(o);
|
| 286 |
+
}
|
| 287 |
+
});
|
| 288 |
+
return arr;
|
| 289 |
+
}, [scene]);
|
| 290 |
+
|
| 291 |
+
// Animation loop: update morph targets + play idle animation
|
| 292 |
+
useFrame((_, dt) => {
|
| 293 |
+
if (mixerRef.current) mixerRef.current.update(dt);
|
| 294 |
+
|
| 295 |
+
morphMeshes.forEach((m) => {
|
| 296 |
+
Object.entries(liveBlend).forEach(([name, target]) => {
|
| 297 |
+
const i = m.morphTargetDictionary[name];
|
| 298 |
+
if (i !== undefined) {
|
| 299 |
+
m.morphTargetInfluences[i] += (target - m.morphTargetInfluences[i]) * dt * 25;
|
| 300 |
+
}
|
| 301 |
+
});
|
| 302 |
+
});
|
| 303 |
+
});
|
| 304 |
+
}
|
| 305 |
+
```
|
| 306 |
+
|
| 307 |
+
**Blend Shapes** (ARKit standard):
|
| 308 |
+
- `jawOpen`: Mouth opening
|
| 309 |
+
- `mouthSmile`: Smile intensity
|
| 310 |
+
- `mouthFrown`: Frown intensity
|
| 311 |
+
- `mouthPucker`: Lip pucker (for "oo" sounds)
|
| 312 |
+
- And ~50 more ARKit blend shapes
|
| 313 |
+
|
| 314 |
+
### Avaturn Modal (`app/app/page.tsx` lines 91-170)
|
| 315 |
+
|
| 316 |
+
```typescript
|
| 317 |
+
function AvaturnModal({ open, onClose, onExport, subdomain = "mrrrme" }) {
|
| 318 |
+
// Dynamically import Avaturn SDK from CDN
|
| 319 |
+
const AvaturnSDK = await importFromCdn(
|
| 320 |
+
"https://cdn.jsdelivr.net/npm/@avaturn/sdk/dist/index.js"
|
| 321 |
+
);
|
| 322 |
+
|
| 323 |
+
// Initialize SDK in container
|
| 324 |
+
await sdk.init(containerRef.current, {
|
| 325 |
+
url: `https://${subdomain}.avaturn.dev`
|
| 326 |
+
});
|
| 327 |
+
|
| 328 |
+
// Listen for export event
|
| 329 |
+
sdk.on("export", (data) => {
|
| 330 |
+
const glbUrl = data?.links?.glb?.url;
|
| 331 |
+
onExport(glbUrl); // Pass URL to parent
|
| 332 |
+
});
|
| 333 |
+
}
|
| 334 |
+
```
|
| 335 |
+
|
| 336 |
+
### Emotion Test Modal (`app/app/page.tsx` lines 20-89)
|
| 337 |
+
|
| 338 |
+
Real-time emotion dashboard:
|
| 339 |
+
- Current emotion with confidence
|
| 340 |
+
- 4-class probability distribution (Neutral, Happy, Sad, Angry)
|
| 341 |
+
- Face quality score
|
| 342 |
+
- Prediction counter
|
| 343 |
+
|
| 344 |
+
---
|
| 345 |
+
|
| 346 |
+
## WebSocket Protocol
|
| 347 |
+
|
| 348 |
+
### Client Messages
|
| 349 |
+
|
| 350 |
+
**Authentication**:
|
| 351 |
+
```json
|
| 352 |
+
{
|
| 353 |
+
"type": "auth",
|
| 354 |
+
"token": "session_token_from_login"
|
| 355 |
+
}
|
| 356 |
+
```
|
| 357 |
+
|
| 358 |
+
**Video Frame** (every 200ms):
|
| 359 |
+
```json
|
| 360 |
+
{
|
| 361 |
+
"type": "video_frame",
|
| 362 |
+
"frame": "..."
|
| 363 |
+
}
|
| 364 |
+
```
|
| 365 |
+
|
| 366 |
+
**Audio Chunk** (every 500ms):
|
| 367 |
+
```json
|
| 368 |
+
{
|
| 369 |
+
"type": "audio_chunk",
|
| 370 |
+
"audio": "base64_webm_audio_data"
|
| 371 |
+
}
|
| 372 |
+
```
|
| 373 |
+
|
| 374 |
+
**Speech End** (when user stops talking):
|
| 375 |
+
```json
|
| 376 |
+
{
|
| 377 |
+
"type": "speech_end",
|
| 378 |
+
"text": "transcribed speech from Web Speech API"
|
| 379 |
+
}
|
| 380 |
+
```
|
| 381 |
+
|
| 382 |
+
**Update Preferences**:
|
| 383 |
+
```json
|
| 384 |
+
{
|
| 385 |
+
"type": "preferences",
|
| 386 |
+
"voice": "female" | "male",
|
| 387 |
+
"language": "en" | "nl",
|
| 388 |
+
"personality": "therapist" | "coach"
|
| 389 |
+
}
|
| 390 |
+
```
|
| 391 |
+
|
| 392 |
+
**Request Greeting**:
|
| 393 |
+
```json
|
| 394 |
+
{
|
| 395 |
+
"type": "request_greeting"
|
| 396 |
+
}
|
| 397 |
+
```
|
| 398 |
+
|
| 399 |
+
### Server Messages
|
| 400 |
+
|
| 401 |
+
**Authentication Success**:
|
| 402 |
+
```json
|
| 403 |
+
{
|
| 404 |
+
"type": "authenticated",
|
| 405 |
+
"username": "alice",
|
| 406 |
+
"summary": "User summary from previous conversations..."
|
| 407 |
+
}
|
| 408 |
+
```
|
| 409 |
+
|
| 410 |
+
**Face Emotion Update**:
|
| 411 |
+
```json
|
| 412 |
+
{
|
| 413 |
+
"type": "face_emotion",
|
| 414 |
+
"emotion": "Happy",
|
| 415 |
+
"confidence": 0.87,
|
| 416 |
+
"probabilities": [0.05, 0.87, 0.04, 0.04],
|
| 417 |
+
"quality": 0.92
|
| 418 |
+
}
|
| 419 |
+
```
|
| 420 |
+
|
| 421 |
+
**Voice Emotion Update**:
|
| 422 |
+
```json
|
| 423 |
+
{
|
| 424 |
+
"type": "voice_emotion",
|
| 425 |
+
"emotion": "Happy"
|
| 426 |
+
}
|
| 427 |
+
```
|
| 428 |
+
|
| 429 |
+
**LLM Response** (with avatar TTS):
|
| 430 |
+
```json
|
| 431 |
+
{
|
| 432 |
+
"type": "llm_response",
|
| 433 |
+
"text": "That's wonderful to hear!",
|
| 434 |
+
"emotion": "Happy",
|
| 435 |
+
"intensity": 0.75,
|
| 436 |
+
"audio_url": "/static/tts_12345.mp3",
|
| 437 |
+
"visemes": [
|
| 438 |
+
{"t": 0.0, "blend": {"jawOpen": 0.0}},
|
| 439 |
+
{"t": 0.1, "blend": {"jawOpen": 0.3, "mouthSmile": 0.2}},
|
| 440 |
+
{"t": 0.2, "blend": {"jawOpen": 0.5}}
|
| 441 |
+
]
|
| 442 |
+
}
|
| 443 |
+
```
|
| 444 |
+
|
| 445 |
+
**Error**:
|
| 446 |
+
```json
|
| 447 |
+
{
|
| 448 |
+
"type": "error",
|
| 449 |
+
"message": "Invalid session - please login again"
|
| 450 |
+
}
|
| 451 |
```
|
| 452 |
|
| 453 |
+
---
|
| 454 |
+
|
| 455 |
+
## API Routes
|
| 456 |
+
|
| 457 |
+
### CORS Proxy (`app/api/avaturn-proxy/route.ts`)
|
| 458 |
+
|
| 459 |
+
Proxies avatar GLB files to bypass CORS restrictions.
|
| 460 |
+
|
| 461 |
+
**Allowed Domains**:
|
| 462 |
+
- `*.avaturn.dev`
|
| 463 |
+
- `*.avaturn.me`
|
| 464 |
+
- `*.cloudfront.net`
|
| 465 |
+
- `storage.googleapis.com`
|
| 466 |
+
- `*.amazonaws.com`
|
| 467 |
+
- `models.readyplayer.me`
|
| 468 |
+
|
| 469 |
+
**Usage**:
|
| 470 |
+
```
|
| 471 |
+
GET /api/avaturn-proxy?url=https://models.readyplayer.me/avatar.glb
|
| 472 |
+
```
|
| 473 |
+
|
| 474 |
+
**Response**:
|
| 475 |
+
- Success: Binary GLB file with `model/gltf-binary` content-type
|
| 476 |
+
- Error 400: URL not allowed
|
| 477 |
+
- Error 502: Upstream fetch failed
|
| 478 |
+
|
| 479 |
+
---
|
| 480 |
+
|
| 481 |
+
## Design System
|
| 482 |
+
|
| 483 |
+
### Color Palette
|
| 484 |
+
|
| 485 |
+
**Light Mode**:
|
| 486 |
+
- Background: `#ffffff` (pure white)
|
| 487 |
+
- Foreground: `#1d1d1f` (near black)
|
| 488 |
+
- Accent: `#007aff` β `#5e5ce6` (blue gradient)
|
| 489 |
+
- Surface: `rgba(255, 255, 255, 0.72)` (frosted glass)
|
| 490 |
+
|
| 491 |
+
**Dark Mode**:
|
| 492 |
+
- Background: `#000000` (pure black)
|
| 493 |
+
- Foreground: `#f5f5f7` (off-white)
|
| 494 |
+
- Accent: `#0a84ff` β `#5e5ce6` (lighter blue gradient)
|
| 495 |
+
- Surface: `rgba(28, 28, 30, 0.72)` (dark frosted glass)
|
| 496 |
+
|
| 497 |
+
### Glass Morphism
|
| 498 |
+
|
| 499 |
+
```css
|
| 500 |
+
.glass {
|
| 501 |
+
background: var(--surface);
|
| 502 |
+
backdrop-filter: blur(24px) saturate(180%);
|
| 503 |
+
border: 1px solid var(--border);
|
| 504 |
+
}
|
| 505 |
+
|
| 506 |
+
.glass-elevated {
|
| 507 |
+
background: var(--surface-elevated);
|
| 508 |
+
backdrop-filter: blur(32px) saturate(200%);
|
| 509 |
+
border: 1px solid var(--border-strong);
|
| 510 |
+
}
|
| 511 |
+
```
|
| 512 |
+
|
| 513 |
+
### Animations
|
| 514 |
+
|
| 515 |
+
```css
|
| 516 |
+
@keyframes fadeIn {
|
| 517 |
+
from { opacity: 0; }
|
| 518 |
+
to { opacity: 1; }
|
| 519 |
+
}
|
| 520 |
+
|
| 521 |
+
@keyframes scaleIn {
|
| 522 |
+
from { opacity: 0; transform: scale(0.95); }
|
| 523 |
+
to { opacity: 1; transform: scale(1); }
|
| 524 |
+
}
|
| 525 |
+
|
| 526 |
+
@keyframes slideInRight {
|
| 527 |
+
from { transform: translateX(100%); }
|
| 528 |
+
to { transform: translateX(0); }
|
| 529 |
+
}
|
| 530 |
+
```
|
| 531 |
+
|
| 532 |
+
---
|
| 533 |
+
|
| 534 |
+
## Component API
|
| 535 |
+
|
| 536 |
+
### Avatar Component
|
| 537 |
+
|
| 538 |
+
```typescript
|
| 539 |
+
<Avatar
|
| 540 |
+
liveBlend={liveBlend} // Current viseme blend shapes
|
| 541 |
+
avatarUrl={avatarUrl} // GLB URL from Avaturn
|
| 542 |
+
position={[-0.01, -2.12, 0.06]} // [x, y, z] position
|
| 543 |
+
rotation={[0.00, 0.51, 0.00]} // [x, y, z] Euler angles
|
| 544 |
+
scale={1.25} // Uniform scale
|
| 545 |
+
/>
|
| 546 |
+
```
|
| 547 |
+
|
| 548 |
+
### Message Bubble
|
| 549 |
+
|
| 550 |
+
```typescript
|
| 551 |
+
<MessageBubble
|
| 552 |
+
message={{
|
| 553 |
+
id: "unique_id",
|
| 554 |
+
role: "user" | "assistant",
|
| 555 |
+
content: "Message text",
|
| 556 |
+
timestamp: new Date(),
|
| 557 |
+
emotion: "Happy" // optional
|
| 558 |
+
}}
|
| 559 |
+
/>
|
| 560 |
+
```
|
| 561 |
+
|
| 562 |
+
### Control Button
|
| 563 |
+
|
| 564 |
+
```typescript
|
| 565 |
+
<ControlButton
|
| 566 |
+
onClick={() => handleAction()}
|
| 567 |
+
icon={<svg>...</svg>}
|
| 568 |
+
label="Button label"
|
| 569 |
+
variant="default" | "danger" | "primary"
|
| 570 |
+
/>
|
| 571 |
+
```
|
| 572 |
+
|
| 573 |
+
### Emotion Test Modal
|
| 574 |
+
|
| 575 |
+
```typescript
|
| 576 |
+
<EmotionTestModal
|
| 577 |
+
open={showEmotionTest}
|
| 578 |
+
onClose={() => setShowEmotionTest(false)}
|
| 579 |
+
wsRef={wsRef} // WebSocket ref for listening to face_emotion messages
|
| 580 |
+
/>
|
| 581 |
+
```
|
| 582 |
+
|
| 583 |
+
---
|
| 584 |
+
|
| 585 |
+
## Build & Deployment
|
| 586 |
+
|
| 587 |
+
### Development
|
| 588 |
+
|
| 589 |
+
```bash
|
| 590 |
+
npm run dev
|
| 591 |
+
```
|
| 592 |
+
|
| 593 |
+
Runs on `http://localhost:3000`
|
| 594 |
+
|
| 595 |
+
### Production Build
|
| 596 |
+
|
| 597 |
+
```bash
|
| 598 |
+
npm run build
|
| 599 |
+
npm start
|
| 600 |
+
```
|
| 601 |
+
|
| 602 |
+
Creates `.next/standalone` directory for deployment.
|
| 603 |
+
|
| 604 |
+
### Docker Build
|
| 605 |
+
|
| 606 |
+
Handled by root `Dockerfile`:
|
| 607 |
+
|
| 608 |
+
```dockerfile
|
| 609 |
+
# Stage 1: Install frontend dependencies
|
| 610 |
+
FROM node:20-alpine AS frontend-deps
|
| 611 |
+
WORKDIR /app/avatar-frontend
|
| 612 |
+
COPY avatar-frontend/package*.json ./
|
| 613 |
+
RUN npm ci
|
| 614 |
+
|
| 615 |
+
# Stage 2: Build frontend
|
| 616 |
+
FROM node:20-alpine AS frontend-builder
|
| 617 |
+
WORKDIR /app/avatar-frontend
|
| 618 |
+
COPY --from=frontend-deps /app/avatar-frontend/node_modules ./node_modules
|
| 619 |
+
COPY avatar-frontend/ ./
|
| 620 |
+
RUN npm run build
|
| 621 |
+
|
| 622 |
+
# Stage 3: Copy to standalone
|
| 623 |
+
COPY --from=frontend-builder /app/avatar-frontend/.next/standalone ./avatar-frontend/.next/standalone
|
| 624 |
+
```
|
| 625 |
+
|
| 626 |
+
### Nginx Proxy
|
| 627 |
+
|
| 628 |
+
Frontend served through Nginx on port 7860:
|
| 629 |
+
|
| 630 |
+
```nginx
|
| 631 |
+
# Next.js frontend
|
| 632 |
+
location / {
|
| 633 |
+
proxy_pass http://127.0.0.1:3001;
|
| 634 |
+
proxy_http_version 1.1;
|
| 635 |
+
proxy_set_header Upgrade $http_upgrade;
|
| 636 |
+
proxy_set_header Connection 'upgrade';
|
| 637 |
+
proxy_set_header Host $host;
|
| 638 |
+
proxy_cache_bypass $http_upgrade;
|
| 639 |
+
}
|
| 640 |
+
|
| 641 |
+
# WebSocket (backend)
|
| 642 |
+
location /ws {
|
| 643 |
+
proxy_pass http://127.0.0.1:8000/ws;
|
| 644 |
+
proxy_http_version 1.1;
|
| 645 |
+
proxy_set_header Upgrade $http_upgrade;
|
| 646 |
+
proxy_set_header Connection "Upgrade";
|
| 647 |
+
}
|
| 648 |
+
|
| 649 |
+
# Backend API
|
| 650 |
+
location /api {
|
| 651 |
+
proxy_pass http://127.0.0.1:8000/api;
|
| 652 |
+
}
|
| 653 |
+
|
| 654 |
+
# Avatar TTS
|
| 655 |
+
location /static {
|
| 656 |
+
proxy_pass http://127.0.0.1:8765/static;
|
| 657 |
+
}
|
| 658 |
+
```
|
| 659 |
+
|
| 660 |
+
---
|
| 661 |
+
|
| 662 |
+
## User Flow
|
| 663 |
+
|
| 664 |
+
### 1. Landing Page (`/`)
|
| 665 |
+
- Marketing content
|
| 666 |
+
- Features showcase
|
| 667 |
+
- "Get Started" CTA β redirects to `/login`
|
| 668 |
+
|
| 669 |
+
### 2. Authentication (`/login`)
|
| 670 |
+
- Username + password form
|
| 671 |
+
- Signup or login toggle
|
| 672 |
+
- Stores token in `localStorage` on success
|
| 673 |
+
- Redirects to `/app`
|
| 674 |
+
|
| 675 |
+
### 3. Main App (`/app`)
|
| 676 |
+
|
| 677 |
+
**On Mount**:
|
| 678 |
+
1. Check `localStorage` for token
|
| 679 |
+
2. Redirect to `/login` if not authenticated
|
| 680 |
+
3. Apply theme preference (light/dark)
|
| 681 |
+
|
| 682 |
+
**On Start**:
|
| 683 |
+
1. Request camera + microphone permissions
|
| 684 |
+
2. Connect WebSocket with auth token
|
| 685 |
+
3. Start video/audio capture
|
| 686 |
+
4. Start speech recognition
|
| 687 |
+
5. Show "Create Avatar" welcome screen
|
| 688 |
+
|
| 689 |
+
**Avatar Creation**:
|
| 690 |
+
1. Click "Create Your Avatar"
|
| 691 |
+
2. Avaturn SDK modal opens
|
| 692 |
+
3. Customize avatar
|
| 693 |
+
4. Export GLB URL
|
| 694 |
+
5. Proxy fetches GLB via `/api/avaturn-proxy`
|
| 695 |
+
6. Three.js loads and renders avatar
|
| 696 |
+
7. Auto-greeting plays with lip-sync
|
| 697 |
+
|
| 698 |
+
**Conversation Loop**:
|
| 699 |
+
1. User speaks (Web Speech API transcribes)
|
| 700 |
+
2. On silence, send `speech_end` message
|
| 701 |
+
3. Backend processes emotions + generates response
|
| 702 |
+
4. Server sends `llm_response` with audio + visemes
|
| 703 |
+
5. Play audio while animating avatar mouth
|
| 704 |
+
6. On audio end, resume speech recognition
|
| 705 |
+
|
| 706 |
+
---
|
| 707 |
+
|
| 708 |
+
## Performance Optimizations
|
| 709 |
+
|
| 710 |
+
### 1. Frame Rate Control
|
| 711 |
+
- **Video**: 5 FPS (200ms intervals) - emotions change slowly
|
| 712 |
+
- **Audio**: 2 Hz (500ms chunks) - sufficient for real-time
|
| 713 |
+
- **Canvas Rendering**: 60 FPS via `requestAnimationFrame`
|
| 714 |
+
|
| 715 |
+
### 2. Asset Loading
|
| 716 |
+
- **Git LFS**: Large files (idle-animation.glb) not in git history
|
| 717 |
+
- **Image Optimization**: Disabled (`unoptimized: true`) for faster builds
|
| 718 |
+
- **Standalone Build**: Minimal production bundle
|
| 719 |
+
|
| 720 |
+
### 3. State Management
|
| 721 |
+
- **Refs for Non-Reactive State**: `isPausedRef`, `recognitionRef`, `wsRef`
|
| 722 |
+
- **Minimal Re-Renders**: Only update UI when necessary
|
| 723 |
+
- **Memoization**: `useMemo` for morph mesh detection
|
| 724 |
+
|
| 725 |
+
### 4. Network Efficiency
|
| 726 |
+
- **JPEG Compression**: 70% quality for video frames
|
| 727 |
+
- **Base64 Encoding**: Binary data transmission
|
| 728 |
+
- **WebSocket Keep-Alive**: Single persistent connection
|
| 729 |
+
|
| 730 |
+
---
|
| 731 |
+
|
| 732 |
+
## Troubleshooting
|
| 733 |
+
|
| 734 |
+
### Camera/Microphone Access Denied
|
| 735 |
+
|
| 736 |
+
**Problem**: Browser doesn't request permissions
|
| 737 |
+
|
| 738 |
+
**Solution**:
|
| 739 |
+
- Use HTTPS in production (required for getUserMedia)
|
| 740 |
+
- Check browser settings β Site permissions
|
| 741 |
+
- Try different browser (Chrome recommended)
|
| 742 |
+
|
| 743 |
+
### WebSocket Connection Failed
|
| 744 |
+
|
| 745 |
+
**Problem**: `ws://localhost:3000/ws` not connecting
|
| 746 |
+
|
| 747 |
+
**Check**:
|
| 748 |
+
```bash
|
| 749 |
+
# Ensure backend is running
|
| 750 |
+
curl http://localhost:8000/health
|
| 751 |
+
|
| 752 |
+
# Check WebSocket endpoint
|
| 753 |
+
curl http://localhost:8000/ws
|
| 754 |
+
```
|
| 755 |
+
|
| 756 |
+
**Fix**: Update `BACKEND_WS` URL in `app/app/page.tsx`
|
| 757 |
+
|
| 758 |
+
### Avatar Not Loading
|
| 759 |
+
|
| 760 |
+
**Problem**: Avatar shows blank screen
|
| 761 |
+
|
| 762 |
+
**Possible Causes**:
|
| 763 |
+
1. GLB URL blocked by CORS β Use `/api/avaturn-proxy?url=...`
|
| 764 |
+
2. Invalid GLB format β Re-export from Avaturn
|
| 765 |
+
3. Git LFS not installed β Run `git lfs pull`
|
| 766 |
+
|
| 767 |
+
**Debug**:
|
| 768 |
+
```bash
|
| 769 |
+
# Check if idle-animation.glb is real file (not pointer)
|
| 770 |
+
file public/idle-animation.glb
|
| 771 |
+
# Should show: "glTF binary" not "ASCII text"
|
| 772 |
+
|
| 773 |
+
# Check if proxy works
|
| 774 |
+
curl "http://localhost:3000/api/avaturn-proxy?url=https://models.readyplayer.me/some-avatar.glb"
|
| 775 |
+
```
|
| 776 |
+
|
| 777 |
+
### Speech Recognition Not Working
|
| 778 |
+
|
| 779 |
+
**Problem**: Microphone captures but no transcription
|
| 780 |
+
|
| 781 |
+
**Fixes**:
|
| 782 |
+
- **Check browser support**: Chrome/Edge only (Safari doesn't support continuous mode)
|
| 783 |
+
- **Language mismatch**: Ensure `recognition.lang` matches `selectedLanguage`
|
| 784 |
+
- **Restart**: Change language to force recognition restart
|
| 785 |
+
|
| 786 |
+
**Debug**:
|
| 787 |
+
```javascript
|
| 788 |
+
// Add to startSpeechRecognition()
|
| 789 |
+
recognition.onerror = (event) => {
|
| 790 |
+
console.log('[SpeechRec] Error:', event.error, event.message);
|
| 791 |
+
};
|
| 792 |
+
|
| 793 |
+
recognition.onresult = (event) => {
|
| 794 |
+
console.log('[SpeechRec] Result:', event.results[event.resultIndex][0].transcript);
|
| 795 |
+
};
|
| 796 |
+
```
|
| 797 |
+
|
| 798 |
+
### Avatar Lip-Sync Out of Sync
|
| 799 |
+
|
| 800 |
+
**Problem**: Mouth moves too early/late
|
| 801 |
+
|
| 802 |
+
**Fix 1**: Adjust viseme interpolation in `useFrame()`:
|
| 803 |
+
```typescript
|
| 804 |
+
// Faster interpolation (current: dt * 25)
|
| 805 |
+
m.morphTargetInfluences[i] += (target - current) * dt * 40;
|
| 806 |
+
|
| 807 |
+
// Slower interpolation
|
| 808 |
+
m.morphTargetInfluences[i] += (target - current) * dt * 15;
|
| 809 |
+
```
|
| 810 |
+
|
| 811 |
+
**Fix 2**: Add offset to viseme timing:
|
| 812 |
+
```typescript
|
| 813 |
+
const t = audioRef.current.currentTime + 0.05; // 50ms lookahead
|
| 814 |
+
```
|
| 815 |
+
|
| 816 |
+
---
|
| 817 |
+
|
| 818 |
+
## Dependencies
|
| 819 |
+
|
| 820 |
+
### Core
|
| 821 |
+
|
| 822 |
+
```json
|
| 823 |
+
{
|
| 824 |
+
"next": "16.0.0",
|
| 825 |
+
"react": "19.2.0",
|
| 826 |
+
"react-dom": "19.2.0",
|
| 827 |
+
"three": "^0.180.0",
|
| 828 |
+
"@react-three/fiber": "^9.4.0",
|
| 829 |
+
"@react-three/drei": "^10.7.6"
|
| 830 |
+
}
|
| 831 |
+
```
|
| 832 |
+
|
| 833 |
+
### Dev Dependencies
|
| 834 |
+
|
| 835 |
+
```json
|
| 836 |
+
{
|
| 837 |
+
"typescript": "^5",
|
| 838 |
+
"tailwindcss": "^4",
|
| 839 |
+
"@tailwindcss/postcss": "^4",
|
| 840 |
+
"eslint": "^9",
|
| 841 |
+
"eslint-config-next": "16.0.0"
|
| 842 |
+
}
|
| 843 |
+
```
|
| 844 |
+
|
| 845 |
+
### Three.js Ecosystem
|
| 846 |
+
|
| 847 |
+
- **React Three Fiber**: React renderer for Three.js
|
| 848 |
+
- **Drei**: Helper components (Environment, Html, useGLTF)
|
| 849 |
+
- **Three.js 0.180**: Core 3D engine
|
| 850 |
+
|
| 851 |
+
**Why Three.js 0.180?**
|
| 852 |
+
- Compatible with Avaturn SDK exports
|
| 853 |
+
- Supports ARKit blend shapes
|
| 854 |
+
- GLTFLoader with morph targets
|
| 855 |
+
|
| 856 |
+
---
|
| 857 |
+
|
| 858 |
+
## Authentication Flow
|
| 859 |
+
|
| 860 |
+
### Signup
|
| 861 |
+
|
| 862 |
+
```typescript
|
| 863 |
+
POST /api/signup
|
| 864 |
+
{
|
| 865 |
+
"username": "alice",
|
| 866 |
+
"password": "secure123"
|
| 867 |
+
}
|
| 868 |
+
|
| 869 |
+
Response:
|
| 870 |
+
{
|
| 871 |
+
"success": true,
|
| 872 |
+
"message": "Account created!"
|
| 873 |
+
}
|
| 874 |
+
```
|
| 875 |
+
|
| 876 |
+
### Login
|
| 877 |
+
|
| 878 |
+
```typescript
|
| 879 |
+
POST /api/login
|
| 880 |
+
{
|
| 881 |
+
"username": "alice",
|
| 882 |
+
"password": "secure123"
|
| 883 |
+
}
|
| 884 |
+
|
| 885 |
+
Response:
|
| 886 |
+
{
|
| 887 |
+
"success": true,
|
| 888 |
+
"token": "random_session_token_32_chars",
|
| 889 |
+
"username": "alice",
|
| 890 |
+
"user_id": "user_abc123",
|
| 891 |
+
"summary": "Previous conversation summary or null"
|
| 892 |
+
}
|
| 893 |
+
```
|
| 894 |
+
|
| 895 |
+
### Token Storage
|
| 896 |
+
|
| 897 |
+
```typescript
|
| 898 |
+
// Save to localStorage
|
| 899 |
+
localStorage.setItem("mrrrme_token", token);
|
| 900 |
+
localStorage.setItem("mrrrme_username", username);
|
| 901 |
+
|
| 902 |
+
// Retrieve on app load
|
| 903 |
+
const token = localStorage.getItem("mrrrme_token");
|
| 904 |
+
if (!token) router.push("/login");
|
| 905 |
+
```
|
| 906 |
+
|
| 907 |
+
### Logout
|
| 908 |
+
|
| 909 |
+
```typescript
|
| 910 |
+
POST /api/logout
|
| 911 |
+
{
|
| 912 |
+
"token": "session_token"
|
| 913 |
+
}
|
| 914 |
+
|
| 915 |
+
// Frontend cleanup:
|
| 916 |
+
localStorage.removeItem("mrrrme_token");
|
| 917 |
+
localStorage.removeItem("mrrrme_username");
|
| 918 |
+
wsRef.current?.close();
|
| 919 |
+
mediaRecorderRef.current?.stop();
|
| 920 |
+
recognitionRef.current?.stop();
|
| 921 |
+
```
|
| 922 |
+
|
| 923 |
+
---
|
| 924 |
+
|
| 925 |
+
## Responsive Design
|
| 926 |
+
|
| 927 |
+
### Breakpoints (Tailwind)
|
| 928 |
+
|
| 929 |
+
```typescript
|
| 930 |
+
// Mobile
|
| 931 |
+
default (< 640px)
|
| 932 |
+
|
| 933 |
+
// Tablet
|
| 934 |
+
md: (>= 768px)
|
| 935 |
+
|
| 936 |
+
// Desktop
|
| 937 |
+
lg: (>= 1024px)
|
| 938 |
+
```
|
| 939 |
+
|
| 940 |
+
### Mobile Adaptations
|
| 941 |
+
|
| 942 |
+
**History Panel**:
|
| 943 |
+
```typescript
|
| 944 |
+
// Mobile: Full width
|
| 945 |
+
className="w-full"
|
| 946 |
+
|
| 947 |
+
// Desktop: Fixed 420px
|
| 948 |
+
className="md:w-[420px]"
|
| 949 |
+
```
|
| 950 |
+
|
| 951 |
+
**Message Bubbles**:
|
| 952 |
+
```typescript
|
| 953 |
+
// Mobile: 85% width
|
| 954 |
+
maxWidth: "85%"
|
| 955 |
+
|
| 956 |
+
// Desktop: 70% width
|
| 957 |
+
md:maxWidth: "70%"
|
| 958 |
+
```
|
| 959 |
+
|
| 960 |
+
---
|
| 961 |
+
|
| 962 |
+
## Known Issues
|
| 963 |
+
|
| 964 |
+
### Current Limitations
|
| 965 |
+
|
| 966 |
+
1. **Browser Support**: Chrome/Edge only for speech recognition
|
| 967 |
+
2. **Mobile Safari**: No continuous speech recognition
|
| 968 |
+
3. **Avatar Loading**: Requires stable internet for GLB download
|
| 969 |
+
4. **Viseme Coverage**: Not all phonemes have perfect ARKit mappings
|
| 970 |
+
5. **Memory Usage**: Three.js can consume 200-400 MB RAM
|
| 971 |
+
|
| 972 |
+
### Workarounds
|
| 973 |
+
|
| 974 |
+
**Speech Recognition on Safari**:
|
| 975 |
+
- Use text input instead (bottom bar)
|
| 976 |
+
- Fallback to server-side Whisper transcription
|
| 977 |
+
|
| 978 |
+
**Slow Avatar Loading**:
|
| 979 |
+
- Preload idle-animation.glb (already in /public)
|
| 980 |
+
- Cache Avaturn exports in IndexedDB (future work)
|
| 981 |
+
|
| 982 |
+
**High Memory Usage**:
|
| 983 |
+
- Clear previous avatar before loading new one:
|
| 984 |
+
```typescript
|
| 985 |
+
if (oldUrl !== DEFAULT_AVATAR) {
|
| 986 |
+
(useGLTF as any).clear?.(oldUrl);
|
| 987 |
+
}
|
| 988 |
+
if (objectUrlRef.current) {
|
| 989 |
+
URL.revokeObjectURL(objectUrlRef.current);
|
| 990 |
+
}
|
| 991 |
+
```
|
| 992 |
+
|
| 993 |
+
---
|
| 994 |
+
|
| 995 |
+
## Future Enhancements
|
| 996 |
+
|
| 997 |
+
### Planned Features (Weeks 10-15)
|
| 998 |
+
|
| 999 |
+
**Avatar Improvements**:
|
| 1000 |
+
- Emotion-driven facial expressions (smile, frown, concern)
|
| 1001 |
+
- Eye gaze tracking (looks at camera)
|
| 1002 |
+
- Head movement (subtle nodding, tilting)
|
| 1003 |
+
- Blink animation at natural intervals
|
| 1004 |
+
|
| 1005 |
+
**UI/UX**:
|
| 1006 |
+
- Emotion timeline graph (Chart.js or Recharts)
|
| 1007 |
+
- Export conversation to CSV/JSON
|
| 1008 |
+
- Session statistics dashboard
|
| 1009 |
+
- Advanced settings (fusion weights, model selection)
|
| 1010 |
+
|
| 1011 |
+
**Performance**:
|
| 1012 |
+
- WebWorker for audio processing
|
| 1013 |
+
- OffscreenCanvas for video encoding
|
| 1014 |
+
- IndexedDB caching for avatars
|
| 1015 |
+
|
| 1016 |
+
**Accessibility**:
|
| 1017 |
+
- Screen reader support
|
| 1018 |
+
- Keyboard navigation
|
| 1019 |
+
- High contrast mode
|
| 1020 |
+
- Text size controls
|
| 1021 |
+
|
| 1022 |
+
---
|
| 1023 |
+
|
| 1024 |
+
## Development Guidelines
|
| 1025 |
+
|
| 1026 |
+
### Code Style
|
| 1027 |
+
|
| 1028 |
+
**TypeScript**:
|
| 1029 |
+
- Strict mode enabled
|
| 1030 |
+
- Explicit types for function parameters
|
| 1031 |
+
- Avoid `any` types
|
| 1032 |
+
|
| 1033 |
+
**React**:
|
| 1034 |
+
- Functional components only
|
| 1035 |
+
- Hooks for state management
|
| 1036 |
+
- `useCallback` for expensive functions
|
| 1037 |
+
- `useMemo` for computed values
|
| 1038 |
+
|
| 1039 |
+
**Naming Conventions**:
|
| 1040 |
+
- Components: `PascalCase`
|
| 1041 |
+
- Functions: `camelCase`
|
| 1042 |
+
- Constants: `UPPER_SNAKE_CASE`
|
| 1043 |
+
- CSS Variables: `--kebab-case`
|
| 1044 |
+
|
| 1045 |
+
### File Organization
|
| 1046 |
+
|
| 1047 |
+
```
|
| 1048 |
+
app/
|
| 1049 |
+
page.tsx # Default export component
|
| 1050 |
+
layout.tsx # Layout wrapper
|
| 1051 |
+
api/
|
| 1052 |
+
route.ts # API route handler
|
| 1053 |
+
```
|
| 1054 |
+
|
| 1055 |
+
### State Management
|
| 1056 |
+
|
| 1057 |
+
**Local State**: `useState` for UI toggles
|
| 1058 |
+
**Refs**: `useRef` for non-reactive values (WebSocket, MediaRecorder)
|
| 1059 |
+
**Global State**: Props drilling (no Redux/Zustand needed for small app)
|
| 1060 |
+
|
| 1061 |
+
---
|
| 1062 |
+
|
| 1063 |
+
## Testing
|
| 1064 |
+
|
| 1065 |
+
### Manual Testing Checklist
|
| 1066 |
+
|
| 1067 |
+
- [ ] Login with new account
|
| 1068 |
+
- [ ] Login with existing account
|
| 1069 |
+
- [ ] Create avatar via Avaturn
|
| 1070 |
+
- [ ] Avatar loads and displays
|
| 1071 |
+
- [ ] Camera permission granted
|
| 1072 |
+
- [ ] Microphone permission granted
|
| 1073 |
+
- [ ] Face emotion updates in real-time
|
| 1074 |
+
- [ ] Speech recognition transcribes correctly
|
| 1075 |
+
- [ ] LLM response plays with lip-sync
|
| 1076 |
+
- [ ] Switch language (English β Dutch)
|
| 1077 |
+
- [ ] Switch voice (Male β Female)
|
| 1078 |
+
- [ ] Switch personality (Therapist β Coach)
|
| 1079 |
+
- [ ] Toggle light/dark mode
|
| 1080 |
+
- [ ] View conversation history
|
| 1081 |
+
- [ ] Pause/resume listening
|
| 1082 |
+
- [ ] Logout properly closes connections
|
| 1083 |
+
|
| 1084 |
+
### Browser Testing
|
| 1085 |
+
|
| 1086 |
+
**Recommended**:
|
| 1087 |
+
- Chrome 120+ (full support)
|
| 1088 |
+
- Edge 120+ (full support)
|
| 1089 |
+
|
| 1090 |
+
**Limited Support**:
|
| 1091 |
+
- Firefox (no Web Speech API continuous mode)
|
| 1092 |
+
- Safari (no Web Speech API on desktop)
|
| 1093 |
+
|
| 1094 |
+
**Mobile**:
|
| 1095 |
+
- Chrome Android (works)
|
| 1096 |
+
- Safari iOS (limited - no continuous speech)
|
| 1097 |
+
|
| 1098 |
+
---
|
| 1099 |
+
|
| 1100 |
+
## Performance Metrics
|
| 1101 |
+
|
| 1102 |
+
### Bundle Size
|
| 1103 |
+
|
| 1104 |
+
```bash
|
| 1105 |
+
npm run build
|
| 1106 |
+
|
| 1107 |
+
# Output:
|
| 1108 |
+
Route (app) Size First Load JS
|
| 1109 |
+
β β / 15.2 kB 105 kB
|
| 1110 |
+
β β /api/avaturn-proxy 0 B 0 B
|
| 1111 |
+
β β /app 89.3 kB 195 kB
|
| 1112 |
+
β β /login 12.8 kB 102 kB
|
| 1113 |
+
|
| 1114 |
+
Total First Load JS: ~195 kB (with Three.js)
|
| 1115 |
+
```
|
| 1116 |
+
|
| 1117 |
+
### Load Times (Local Dev)
|
| 1118 |
+
|
| 1119 |
+
- Initial page load: 1.2s
|
| 1120 |
+
- Avatar GLB download: 2-4s (depends on size)
|
| 1121 |
+
- WebSocket connection: <100ms
|
| 1122 |
+
- First video frame: 200ms
|
| 1123 |
+
|
| 1124 |
+
### Runtime Performance
|
| 1125 |
+
|
| 1126 |
+
- FPS: 60 (Three.js canvas)
|
| 1127 |
+
- Memory: 200-400 MB
|
| 1128 |
+
- CPU: 15-25% (with webcam)
|
| 1129 |
+
- Network: 50-100 KB/s (video + audio upload)
|
| 1130 |
+
|
| 1131 |
+
---
|
| 1132 |
+
|
| 1133 |
+
## Security Considerations
|
| 1134 |
+
|
| 1135 |
+
### Authentication
|
| 1136 |
+
|
| 1137 |
+
- Tokens are random 32-character strings
|
| 1138 |
+
- Passwords hashed with SHA-256 (server-side)
|
| 1139 |
+
- Session validation on every WebSocket message
|
| 1140 |
+
- Auto-logout on invalid token
|
| 1141 |
+
|
| 1142 |
+
### Data Privacy
|
| 1143 |
+
|
| 1144 |
+
- Video/audio chunks sent to backend, not stored
|
| 1145 |
+
- No face recognition or identification
|
| 1146 |
+
- Conversation history saved per-user in backend SQLite
|
| 1147 |
+
- localStorage tokens cleared on logout
|
| 1148 |
+
|
| 1149 |
+
### CORS
|
| 1150 |
+
|
| 1151 |
+
- Avaturn proxy restricts to whitelisted domains
|
| 1152 |
+
- Backend CORS allows all origins (dev only)
|
| 1153 |
+
- Production should restrict to specific domains
|
| 1154 |
+
|
| 1155 |
+
---
|
| 1156 |
+
|
| 1157 |
+
## Contributing
|
| 1158 |
+
|
| 1159 |
+
### Development Setup
|
| 1160 |
+
|
| 1161 |
+
```bash
|
| 1162 |
+
# Fork repository
|
| 1163 |
+
git clone https://github.com/YourUsername/MrrrMe.git
|
| 1164 |
+
cd MrrrMe/avatar-frontend
|
| 1165 |
+
|
| 1166 |
+
# Install dependencies
|
| 1167 |
+
npm install
|
| 1168 |
+
|
| 1169 |
+
# Create feature branch
|
| 1170 |
+
git checkout -b feature/your-feature
|
| 1171 |
+
|
| 1172 |
+
# Run dev server
|
| 1173 |
+
npm run dev
|
| 1174 |
+
|
| 1175 |
+
# Make changes, test thoroughly
|
| 1176 |
+
|
| 1177 |
+
# Commit and push
|
| 1178 |
+
git add .
|
| 1179 |
+
git commit -m "Add your feature"
|
| 1180 |
+
git push origin feature/your-feature
|
| 1181 |
+
|
| 1182 |
+
# Open Pull Request
|
| 1183 |
+
```
|
| 1184 |
+
|
| 1185 |
+
### Code Review Checklist
|
| 1186 |
+
|
| 1187 |
+
- [ ] TypeScript types are correct
|
| 1188 |
+
- [ ] No console.log in production code
|
| 1189 |
+
- [ ] Components are properly memoized
|
| 1190 |
+
- [ ] CSS variables used (no hardcoded colors)
|
| 1191 |
+
- [ ] Responsive design tested
|
| 1192 |
+
- [ ] WebSocket cleanup on unmount
|
| 1193 |
+
- [ ] Error handling implemented
|
| 1194 |
+
|
| 1195 |
+
---
|
| 1196 |
+
|
| 1197 |
+
## License
|
| 1198 |
+
|
| 1199 |
+
MIT License - See root LICENSE file
|
| 1200 |
|
| 1201 |
+
---
|
| 1202 |
|
| 1203 |
+
## Contact
|
| 1204 |
|
| 1205 |
+
**Project Team**:
|
| 1206 |
+
- Musaed Al-Fareh - [email protected]
|
| 1207 |
+
- Michon Goddijn - [email protected]
|
| 1208 |
+
- Lorena KraljiΔ - [email protected]
|
| 1209 |
|
| 1210 |
+
**Course**: Applied Data Science - Artificial Intelligence
|
| 1211 |
+
**Institution**: Breda University of Applied Sciences
|
| 1212 |
|
| 1213 |
+
---
|
|
|
|
| 1214 |
|
| 1215 |
+
## Acknowledgments
|
| 1216 |
|
| 1217 |
+
- **Avaturn**: 3D avatar creation platform
|
| 1218 |
+
- **Pmndrs**: React Three Fiber ecosystem
|
| 1219 |
+
- **Next.js Team**: Framework development
|
| 1220 |
+
- **Three.js**: WebGL rendering engine
|
| 1221 |
|
| 1222 |
+
---
|
| 1223 |
|
| 1224 |
+
**Last Updated**: December 10, 2024
|
| 1225 |
+
**Version**: 2.0.0 (Next.js 16 + React 19)
|
| 1226 |
+
**Status**: Production Ready
|