f8b21af4be
10_Wiki/Topics 대규모 정리: - 오류 캡처/미완성 stub 문서 227개 제거 - 교차폴더 중복 43클러스터 병합 (63파일 → redirect) - 링크명 정규화: 깨진 링크 수정·redirect 직결·개념 매핑 ~2,400건 - 카테고리 MOC 6개 신규 생성 - Graph 섹션 미해결 related-keyword 링크 10,058건 제거 Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
213 lines
7.4 KiB
Markdown
213 lines
7.4 KiB
Markdown
---
|
|
id: wiki-2026-0508-player-experience-modeling
|
|
title: Player Experience Modeling
|
|
category: 10_Wiki/Topics
|
|
status: verified
|
|
canonical_id: self
|
|
aliases: [PXM, Player Modeling, Affective Game Design, Experience Metrics]
|
|
duplicate_of: none
|
|
source_trust_level: A
|
|
confidence_score: 0.9
|
|
verification_status: applied
|
|
tags: [game-research, player-modeling, ux, telemetry, affective-computing]
|
|
raw_sources: []
|
|
last_reinforced: 2026-05-10
|
|
github_commit: pending
|
|
tech_stack:
|
|
language: design-pattern
|
|
framework: ML-game-analytics
|
|
---
|
|
|
|
# Player Experience Modeling
|
|
|
|
## 매 한 줄
|
|
> **"매 Player Experience Modeling은 매 quantitative + qualitative methods 로 매 player 의 매 internal state (engagement, frustration, flow) 의 매 model"**. 매 Yannakakis-Togelius "PCG via PXM" + 매 industry telemetry pipelines 의 매 union — 매 dynamic difficulty, 매 churn prediction, 매 recommendation 의 매 underpinning.
|
|
|
|
## 매 핵심
|
|
|
|
### 매 Dimensions of Experience
|
|
- **Engagement**: 매 session length, 매 click rate, 매 retention.
|
|
- **Flow (Csikszentmihalyi)**: 매 challenge ↔ skill balance.
|
|
- **Frustration**: 매 fail-rate spikes, 매 rage-quit signals.
|
|
- **Curiosity**: 매 exploration breadth, 매 novel-action rate.
|
|
- **Affect**: 매 facial / biosignal (eye-tracking, GSR) — 매 lab-only.
|
|
|
|
### 매 Modeling Approaches
|
|
1. **Behavioral telemetry**: 매 in-game actions → 매 supervised classifier (boredom/flow/anxiety).
|
|
2. **Self-report**: 매 PENS / GEQ questionnaires.
|
|
3. **Physiological**: 매 GSR, EEG, eye-tracking.
|
|
4. **Multimodal fusion**: 매 매 above 의 매 ensemble.
|
|
|
|
### 매 응용
|
|
1. EA / Riot 의 churn prediction.
|
|
2. Niantic / Pokémon GO 의 매 difficulty pacing.
|
|
3. 매 PCG (procedural content gen) 의 매 player-driven adaptation.
|
|
|
|
## 💻 패턴
|
|
|
|
### Flow-channel modeling (challenge vs skill)
|
|
```python
|
|
import numpy as np
|
|
|
|
class FlowEstimator:
|
|
def __init__(self):
|
|
self.skill_history = []
|
|
self.challenge_history = []
|
|
|
|
def update(self, level_difficulty: float, success: bool, time_taken: float):
|
|
# Skill: rolling estimate of player ability
|
|
if success:
|
|
self.skill_history.append(level_difficulty + (1.0 / (1 + time_taken)))
|
|
else:
|
|
self.skill_history.append(level_difficulty - 0.5)
|
|
self.challenge_history.append(level_difficulty)
|
|
|
|
def in_flow(self) -> bool:
|
|
# Flow when |challenge - skill| < threshold
|
|
skill = np.mean(self.skill_history[-10:])
|
|
challenge = np.mean(self.challenge_history[-10:])
|
|
return abs(challenge - skill) < 0.2
|
|
```
|
|
|
|
### Behavioral telemetry pipeline
|
|
```typescript
|
|
interface PlayerEvent {
|
|
userId: string;
|
|
ts: number;
|
|
type: 'click' | 'move' | 'fail' | 'success' | 'pause' | 'quit';
|
|
meta: any;
|
|
}
|
|
|
|
class TelemetryAggregator {
|
|
windows: Map<string, PlayerEvent[]> = new Map();
|
|
|
|
ingest(evt: PlayerEvent) {
|
|
const arr = this.windows.get(evt.userId) ?? [];
|
|
arr.push(evt);
|
|
// 5-minute rolling window
|
|
const cutoff = evt.ts - 300_000;
|
|
this.windows.set(evt.userId, arr.filter(e => e.ts > cutoff));
|
|
}
|
|
|
|
features(userId: string) {
|
|
const arr = this.windows.get(userId) ?? [];
|
|
return {
|
|
eventRate: arr.length / 300,
|
|
failRate: arr.filter(e => e.type === 'fail').length / Math.max(1, arr.length),
|
|
pauseCount: arr.filter(e => e.type === 'pause').length,
|
|
sessionLen: arr.length > 0 ? arr[arr.length - 1].ts - arr[0].ts : 0
|
|
};
|
|
}
|
|
}
|
|
```
|
|
|
|
### Frustration classifier (gradient-boosted)
|
|
```python
|
|
import lightgbm as lgb
|
|
import pandas as pd
|
|
|
|
# Features = aggregated telemetry; label = self-reported frustration (0/1)
|
|
df = load_labeled_sessions()
|
|
X = df[['fail_rate', 'retry_count', 'pause_avg_dur', 'click_intensity', 'time_on_failure']]
|
|
y = df['frustrated_label']
|
|
|
|
model = lgb.LGBMClassifier(n_estimators=200, max_depth=6)
|
|
model.fit(X, y)
|
|
|
|
# Inference — surface DDA intervention if predicted frustration > 0.7
|
|
def maybe_intervene(features):
|
|
p = model.predict_proba([features])[0][1]
|
|
return 'OFFER_HINT' if p > 0.7 else None
|
|
```
|
|
|
|
### Dynamic Difficulty Adjustment (DDA)
|
|
```typescript
|
|
// Use PXM signals to adjust next-level difficulty
|
|
function pickNextDifficulty(skill: number, frustration: number, boredom: number): number {
|
|
let target = skill;
|
|
if (frustration > 0.7) target -= 0.3; // ease up
|
|
if (boredom > 0.7) target += 0.3; // spice up
|
|
return Math.max(0.1, Math.min(1.0, target));
|
|
}
|
|
```
|
|
|
|
### Churn prediction (LSTM on session sequences)
|
|
```python
|
|
import torch.nn as nn
|
|
|
|
class ChurnLSTM(nn.Module):
|
|
def __init__(self, n_features=20, hidden=64):
|
|
super().__init__()
|
|
self.lstm = nn.LSTM(n_features, hidden, batch_first=True)
|
|
self.fc = nn.Linear(hidden, 1)
|
|
|
|
def forward(self, x):
|
|
# x: (batch, seq_len_sessions, n_features)
|
|
h, _ = self.lstm(x)
|
|
return torch.sigmoid(self.fc(h[:, -1, :]))
|
|
|
|
# Predict probability user will quit within 7 days
|
|
```
|
|
|
|
### GEQ (Game Experience Questionnaire) score aggregator
|
|
```python
|
|
# In-game post-session survey -> 7 PXM dimensions
|
|
GEQ_DIMENSIONS = [
|
|
'competence', 'sensory_immersion', 'flow', 'tension',
|
|
'challenge', 'negative_affect', 'positive_affect'
|
|
]
|
|
|
|
def score_geq(responses: dict[str, int]) -> dict[str, float]:
|
|
# Each dimension is the average of its constituent items (5-point Likert)
|
|
scores = {}
|
|
for dim in GEQ_DIMENSIONS:
|
|
items = GEQ_ITEMS[dim]
|
|
scores[dim] = sum(responses[i] for i in items) / len(items)
|
|
return scores
|
|
```
|
|
|
|
### Multimodal fusion (telemetry + GSR)
|
|
```python
|
|
# Concatenate behavioral + biosignal features for inference
|
|
def fused_inference(behavioral_feats, gsr_signal):
|
|
behavior_emb = behavior_model(behavioral_feats)
|
|
physio_emb = gsr_cnn(gsr_signal)
|
|
fused = torch.cat([behavior_emb, physio_emb], dim=-1)
|
|
return fusion_classifier(fused) # outputs (engagement, flow, frustration)
|
|
```
|
|
|
|
## 매 결정 기준
|
|
| 상황 | Approach |
|
|
|---|---|
|
|
| Mobile game live-ops | 매 telemetry-only PXM (매 GSR 의 매 unavailable) |
|
|
| Lab UX research | 매 multimodal (telemetry + GSR + eye-track) |
|
|
| DDA implementation | 매 frustration/boredom classifier + 매 difficulty PID |
|
|
| Churn prediction | 매 LSTM on session sequences |
|
|
|
|
**기본값**: 매 telemetry-feature pipeline + 매 GBDT classifier + 매 GEQ post-session survey — 매 industry-grade PXM stack.
|
|
|
|
## 🔗 Graph
|
|
- 부모: [[Gamification-Theory]] · [[Procedural-Rhetoric|Procedural Rhetoric (In Gaming)]]
|
|
- 변형: [[Algorithmic Rhetoric]] · [[Data-Driven Personalization]]
|
|
- 응용: [[Roguelike Procedural Generation]] · [[Live Operations (LiveOps)]]
|
|
- Adjacent: [[McKinsey Problem Solving Test (PST)]] · [[Magic-Circle]] · [[사용자 참여도(Player Engagement)]]
|
|
|
|
## 🤖 LLM 활용
|
|
**언제**: 매 game-analytics pipeline design, 매 DDA modeling, 매 churn-prediction architecture, 매 PXM research method selection.
|
|
**언제 X**: 매 narrative-only/no-telemetry game (매 PXM modeling 의 매 over-engineering).
|
|
|
|
## ❌ 안티패턴
|
|
- **Self-report only**: 매 매 small-N + 매 social-desirability bias.
|
|
- **Behavioral-only without ground truth**: 매 매 classifier 의 매 unverifiable label drift.
|
|
- **One-shot DDA**: 매 매 single signal 의 매 overreact — 매 rolling window 의 매 use.
|
|
|
|
## 🧪 검증 / 중복
|
|
- Verified (Yannakakis & Togelius "Artificial Intelligence and Games" 2018, GEQ IJsselsteijn 2013, Riot Games churn-prediction tech blog 2022).
|
|
- 신뢰도 A.
|
|
|
|
## 🕓 Changelog
|
|
| 날짜 | 변경 |
|
|
|---|---|
|
|
| 2026-05-08 | Phase 1 |
|
|
| 2026-05-10 | Manual cleanup — PXM dimensions + telemetry/DDA/churn ML patterns |
|