|
<!DOCTYPE html> |
|
<html lang="de"> |
|
<head> |
|
<meta charset="UTF-8"> |
|
<meta name="viewport" content="width=device-width, initial-scale=1.0"> |
|
<title>Dr. Franz Schwanz - Psychoanalytischer Dialog</title> |
|
<script src="https://cdn.tailwindcss.com"></script> |
|
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css"> |
|
<style> |
|
@import url('https://fonts.googleapis.com/css2?family=Playfair+Display:wght@400;500;600;700&family=Roboto:wght@300;400;500&display=swap'); |
|
|
|
:root { |
|
--primary-color: #6366f1; |
|
--primary-light: #a5b4fc; |
|
--primary-dark: #4f46e5; |
|
--secondary-color: #10b981; |
|
--text-color: #334155; |
|
--text-light: #64748b; |
|
--bg-color: #f8fafc; |
|
--card-color: #ffffff; |
|
} |
|
|
|
body { |
|
font-family: 'Roboto', sans-serif; |
|
background-color: var(--bg-color); |
|
color: var(--text-color); |
|
line-height: 1.6; |
|
} |
|
|
|
.title-font { |
|
font-family: 'Playfair Display', serif; |
|
} |
|
|
|
.chat-container { |
|
height: calc(100vh - 180px); |
|
scrollbar-width: thin; |
|
scrollbar-color: var(--primary-color) #e5e7eb; |
|
} |
|
|
|
.chat-container::-webkit-scrollbar { |
|
width: 6px; |
|
} |
|
|
|
.chat-container::-webkit-scrollbar-track { |
|
background: #e5e7eb; |
|
} |
|
|
|
.chat-container::-webkit-scrollbar-thumb { |
|
background-color: var(--primary-color); |
|
border-radius: 20px; |
|
} |
|
|
|
.psycho-bubble { |
|
background-color: var(--card-color); |
|
border-left: 4px solid var(--primary-color); |
|
box-shadow: 0 2px 4px rgba(0,0,0,0.05); |
|
} |
|
|
|
.user-bubble { |
|
background-color: var(--primary-color); |
|
color: white; |
|
box-shadow: 0 2px 4px rgba(0,0,0,0.1); |
|
} |
|
|
|
.typing-indicator span { |
|
display: inline-block; |
|
width: 8px; |
|
height: 8px; |
|
border-radius: 50%; |
|
background-color: #9ca3af; |
|
margin-right: 4px; |
|
} |
|
|
|
.typing-indicator span:nth-child(1) { |
|
animation: bounce 1s infinite; |
|
} |
|
|
|
.typing-indicator span:nth-child(2) { |
|
animation: bounce 1s infinite 0.2s; |
|
} |
|
|
|
.typing-indicator span:nth-child(3) { |
|
animation: bounce 1s infinite 0.4s; |
|
} |
|
|
|
@keyframes bounce { |
|
0%, 100% { transform: translateY(0); } |
|
50% { transform: translateY(-5px); } |
|
} |
|
|
|
.highlight { |
|
background-color: #fef08a; |
|
padding: 0 2px; |
|
} |
|
|
|
.fade-in { |
|
animation: fadeIn 0.5s ease-in; |
|
} |
|
|
|
@keyframes fadeIn { |
|
from { opacity: 0; transform: translateY(10px); } |
|
to { opacity: 1; transform: translateY(0); } |
|
} |
|
|
|
.personalized-question { |
|
border-left: 3px solid var(--secondary-color); |
|
padding-left: 12px; |
|
margin: 12px 0; |
|
background-color: rgba(16, 185, 129, 0.05); |
|
} |
|
|
|
.memory-badge { |
|
display: inline-block; |
|
background-color: #e0e7ff; |
|
color: #4338ca; |
|
padding: 2px 8px; |
|
border-radius: 12px; |
|
font-size: 0.75rem; |
|
margin-left: 8px; |
|
vertical-align: middle; |
|
} |
|
|
|
.bias-analysis { |
|
background-color: #f8fafc; |
|
border: 1px solid #e2e8f0; |
|
border-radius: 8px; |
|
padding: 12px; |
|
margin-top: 16px; |
|
font-size: 0.85rem; |
|
color: var(--text-light); |
|
} |
|
|
|
.bias-meter { |
|
height: 6px; |
|
background-color: #e2e8f0; |
|
border-radius: 3px; |
|
margin-top: 4px; |
|
overflow: hidden; |
|
} |
|
|
|
.bias-meter-fill { |
|
height: 100%; |
|
background-color: var(--primary-color); |
|
transition: width 0.5s ease; |
|
} |
|
|
|
.insight-tag { |
|
display: inline-block; |
|
background-color: #e0f2fe; |
|
color: #0369a1; |
|
padding: 2px 8px; |
|
border-radius: 4px; |
|
font-size: 0.75rem; |
|
margin-right: 6px; |
|
margin-bottom: 6px; |
|
} |
|
|
|
@media (max-width: 640px) { |
|
.chat-container { |
|
height: calc(100vh - 160px); |
|
} |
|
|
|
.psycho-bubble, .user-bubble { |
|
max-width: 90% !important; |
|
} |
|
} |
|
</style> |
|
</head> |
|
<body class="bg-gray-50"> |
|
<div class="container mx-auto max-w-4xl px-4 py-6 md:py-8"> |
|
|
|
<header class="mb-6 md:mb-8 text-center"> |
|
<div class="flex items-center justify-center mb-3 md:mb-4"> |
|
<div class="bg-indigo-100 p-3 rounded-full mr-3 md:mr-4"> |
|
<i class="fas fa-brain text-indigo-600 text-xl md:text-2xl"></i> |
|
</div> |
|
<h1 class="title-font text-2xl md:text-3xl font-bold text-gray-800">Dr. Franz Schwanz</h1> |
|
</div> |
|
<p class="text-gray-600 max-w-2xl mx-auto text-sm md:text-base"> |
|
Ihr persönlicher psychoanalytischer Gesprächspartner für tiefere Selbsterkenntnis |
|
</p> |
|
</header> |
|
|
|
|
|
<div class="bg-white rounded-xl shadow-lg overflow-hidden"> |
|
|
|
<div class="bg-indigo-600 text-white p-3 md:p-4 flex items-center"> |
|
<div class="w-8 h-8 md:w-10 md:h-10 rounded-full bg-indigo-500 flex items-center justify-center mr-2 md:mr-3"> |
|
<i class="fas fa-user-tie text-sm md:text-base"></i> |
|
</div> |
|
<div> |
|
<h2 class="font-semibold text-sm md:text-base">Dr. Franz Schwanz</h2> |
|
<p class="text-xs text-indigo-200">Psychoanalytischer Gesprächspartner</p> |
|
</div> |
|
</div> |
|
|
|
|
|
<div class="chat-container overflow-y-auto p-3 md:p-4 space-y-3 md:space-y-4" id="chat-messages"> |
|
|
|
<div class="flex fade-in"> |
|
<div class="flex-shrink-0 mr-2 md:mr-3"> |
|
<div class="w-7 h-7 md:w-8 md:h-8 rounded-full bg-indigo-100 flex items-center justify-center"> |
|
<i class="fas fa-brain text-indigo-600 text-xs md:text-sm"></i> |
|
</div> |
|
</div> |
|
<div class="psycho-bubble rounded-lg p-3 md:p-4 max-w-[85%]"> |
|
<p class="text-xs md:text-sm text-gray-500 mb-1" id="greeting-time"></p> |
|
<p class="font-medium text-indigo-800 mb-1 md:mb-2">Willkommen zu unserem persönlichen Dialog.</p> |
|
<p class="text-gray-700 mb-2 md:mb-3">Ich bin Dr. Franz Schwanz und begleite Sie in einem reflektierenden Gespräch. Bevor wir beginnen: </p> |
|
|
|
<div class="personalized-question"> |
|
<p class="text-gray-700 font-medium">Wie darf ich Sie nennen? Und was hat Sie heute zu mir geführt?</p> |
|
</div> |
|
|
|
<div class="mt-2 md:mt-3 text-xs md:text-sm text-gray-500"> |
|
<p>Sie können mir einfach natürlich schreiben, wie Sie es einem vertrauten Gesprächspartner tun würden.</p> |
|
</div> |
|
</div> |
|
</div> |
|
</div> |
|
|
|
|
|
<div class="border-t border-gray-200 p-3 md:p-4 bg-gray-50"> |
|
<div class="flex items-center"> |
|
<div class="flex-grow relative"> |
|
<textarea id="message-input" rows="1" class="w-full border border-gray-300 rounded-full py-2 md:py-3 px-3 md:px-4 pr-10 md:pr-12 focus:outline-none focus:ring-2 focus:ring-indigo-500 focus:border-indigo-500 resize-none text-sm md:text-base" placeholder="Ihre Antwort..."></textarea> |
|
<button id="send-button" class="absolute right-2 md:right-3 top-1/2 transform -translate-y-1/2 w-7 h-7 md:w-8 md:h-8 rounded-full bg-indigo-600 hover:bg-indigo-700 flex items-center justify-center transition-colors"> |
|
<i class="fas fa-paper-plane text-white text-xs md:text-sm"></i> |
|
</button> |
|
</div> |
|
</div> |
|
</div> |
|
</div> |
|
|
|
|
|
<footer class="mt-6 md:mt-8 text-center text-xs md:text-sm text-gray-500"> |
|
<p>Dies ist ein psychoanalytisches Dialogexperiment und ersetzt keine Therapie.</p> |
|
</footer> |
|
</div> |
|
|
|
<script> |
|
|
|
class ConversationManager { |
|
constructor() { |
|
this.userProfile = this.initializeUserProfile(); |
|
this.conversationHistory = []; |
|
this.conversationThreads = {}; |
|
this.currentThread = null; |
|
this.emotionalToneHistory = []; |
|
this.lastUserMessage = ''; |
|
} |
|
|
|
initializeUserProfile() { |
|
return { |
|
name: null, |
|
preferredName: null, |
|
age: null, |
|
gender: null, |
|
knownIssues: [], |
|
emotionalState: null, |
|
emotionalIntensity: 0, |
|
personalityTraits: [], |
|
conversationStage: 'initial', |
|
lastTopics: [], |
|
keyMemories: [], |
|
conversationDepth: 0, |
|
resistanceLevel: 0, |
|
biasScores: { |
|
projection: 0, |
|
pathologization: 0, |
|
solution: 0, |
|
feasibility: 0 |
|
}, |
|
linguisticPatterns: { |
|
sentenceLength: 0, |
|
questionRatio: 0, |
|
emotionalWords: 0 |
|
} |
|
}; |
|
} |
|
|
|
updateProfileFromMessage(message) { |
|
this.lastUserMessage = message; |
|
const lowerMessage = message.toLowerCase(); |
|
|
|
|
|
this.extractBasicInfo(message); |
|
|
|
|
|
this.analyzeEmotionalState(message); |
|
|
|
|
|
this.detectResistance(message); |
|
|
|
|
|
this.trackTopicsAndMemories(message); |
|
|
|
|
|
this.analyzeLinguisticPatterns(message); |
|
|
|
|
|
this.updateConversationStage(); |
|
} |
|
|
|
extractBasicInfo(message) { |
|
if (!this.userProfile.name && this.userProfile.conversationStage === 'initial') { |
|
const namePatterns = [ |
|
/(?:ich bin|mein name ist|ich heiße|nennen sie mich)\s+([A-ZÄÖÜ][a-zäöüß]+(?:\s+[A-ZÄÖÜ][a-zäöüß]+)*)/i, |
|
/(?:name)\s+(?:ist\s+)?([A-ZÄÖÜ][a-zäöüß]+)/i, |
|
/^([A-ZÄÖÜ][a-zäöüß]+)(?:\s|$)/, |
|
/(?:heiße)\s+([A-ZÄÖÜ][a-zäöüß]+)/i |
|
]; |
|
|
|
for (const pattern of namePatterns) { |
|
const nameMatch = message.match(pattern); |
|
if (nameMatch && nameMatch[1]) { |
|
this.userProfile.name = nameMatch[1].trim(); |
|
this.userProfile.preferredName = nameMatch[1].trim(); |
|
break; |
|
} |
|
} |
|
|
|
if (!this.userProfile.name) { |
|
const words = message.split(' '); |
|
if (words.length > 0 && words[0].length > 2 && /^[A-ZÄÖÜ]/.test(words[0])) { |
|
this.userProfile.name = words[0]; |
|
this.userProfile.preferredName = words[0]; |
|
} |
|
} |
|
|
|
if (this.userProfile.name) return; |
|
} |
|
|
|
if (!this.userProfile.age) { |
|
const ageMatch = message.match(/(?:ich bin|alter|ich\s+habe)\s+(\d+)\s+(?:jahre|jahren)?/i); |
|
if (ageMatch && ageMatch[1]) { |
|
this.userProfile.age = parseInt(ageMatch[1]); |
|
} |
|
} |
|
|
|
if (!this.userProfile.gender) { |
|
if (lowerMessage.match(/\b(frau|weiblich|sie\s+ist)\b/i)) { |
|
this.userProfile.gender = 'weiblich'; |
|
} else if (lowerMessage.match(/\b(mann|männlich|er\s+ist)\b/i)) { |
|
this.userProfile.gender = 'männlich'; |
|
} else if (lowerMessage.match(/\b(nicht-binär|divers|non-binary)\b/i)) { |
|
this.userProfile.gender = 'divers'; |
|
} |
|
} |
|
} |
|
|
|
analyzeEmotionalState(message) { |
|
const lowerMessage = message.toLowerCase(); |
|
const emotionalWords = { |
|
distressed: ['verzweifelt', 'hoffnungslos', 'ängstlich', 'unsicher', 'traurig', 'deprimiert', 'überfordert', 'hilflos'], |
|
angry: ['wütend', 'ärgerlich', 'frustriert', 'genervt', 'aggressiv', 'sauer', 'verärgert'], |
|
confused: ['verwirrt', 'unsicher', 'fragend', 'unschlüssig', 'orientierungslos', 'ratlos'], |
|
positive: ['froh', 'glücklich', 'zufrieden', 'gut', 'freue', 'erfreut', 'begeistert'], |
|
anxious: ['besorgt', 'nervös', 'angespannt', 'unruhig', 'panik', 'sorge'] |
|
}; |
|
|
|
let detectedState = null; |
|
let intensity = 0; |
|
|
|
for (const [state, words] of Object.entries(emotionalWords)) { |
|
const matches = words.filter(word => lowerMessage.includes(word)); |
|
if (matches.length > 0) { |
|
detectedState = state; |
|
intensity = matches.length; |
|
break; |
|
} |
|
} |
|
|
|
|
|
const intensityModifiers = lowerMessage.match(/\b(sehr|extrem|wirklich|total|vollkommen|absolut|etwas|leicht|ein wenig)\b/g); |
|
if (intensityModifiers) { |
|
intensity += intensityModifiers.filter(m => ['sehr', 'extrem', 'wirklich', 'total', 'vollkommen', 'absolut'].includes(m)).length; |
|
intensity -= intensityModifiers.filter(m => ['etwas', 'leicht', 'ein wenig'].includes(m)).length; |
|
} |
|
|
|
|
|
const exclamationCount = (message.match(/!/g) || []).length; |
|
const questionCount = (message.match(/\?/g) || []).length; |
|
intensity += exclamationCount * 0.5; |
|
|
|
if (detectedState) { |
|
this.userProfile.emotionalState = detectedState; |
|
this.userProfile.emotionalIntensity = Math.min(Math.max(intensity, 1), 5); |
|
this.emotionalToneHistory.push({ |
|
state: detectedState, |
|
intensity: this.userProfile.emotionalIntensity, |
|
timestamp: new Date() |
|
}); |
|
} |
|
} |
|
|
|
detectResistance(message) { |
|
const lowerMessage = message.toLowerCase(); |
|
const resistancePatterns = [ |
|
/\b(weiß nicht|keine ahnung|unangenehm|schwierig)\b/i, |
|
/\b(darüber möchte ich nicht sprechen|lieber nicht|mag ich nicht)\b/i, |
|
/\b(warum fragen sie|weshalb wollen sie das wissen)\b/i, |
|
/\b(das ist privat|geht sie nichts an)\b/i |
|
]; |
|
|
|
let resistanceScore = 0; |
|
resistancePatterns.forEach(pattern => { |
|
if (lowerMessage.match(pattern)) { |
|
resistanceScore += 1; |
|
} |
|
}); |
|
|
|
|
|
if (message.split(' ').length < 5 && message.length < 20 && !message.endsWith('?')) { |
|
resistanceScore += 1; |
|
} |
|
|
|
if (resistanceScore > 0) { |
|
this.userProfile.resistanceLevel = Math.min(this.userProfile.resistanceLevel + resistanceScore, 5); |
|
} else if (this.userProfile.resistanceLevel > 0) { |
|
this.userProfile.resistanceLevel = Math.max(this.userProfile.resistanceLevel - 0.5, 0); |
|
} |
|
} |
|
|
|
trackTopicsAndMemories(message) { |
|
const lowerMessage = message.toLowerCase(); |
|
|
|
const memoryTriggers = [ |
|
{ pattern: /\b(kindheit|jugend|als kind|früher)\b/i, topic: 'Kindheit' }, |
|
{ pattern: /\b(eltern|mutter|vater|familie)\b/i, topic: 'Familie' }, |
|
{ pattern: /\b(schule|ausbildung|studium|lehre|universität)\b/i, topic: 'Bildung' }, |
|
{ pattern: /\b(arbeit|job|beruf|karriere|kollegen)\b/i, topic: 'Arbeit' }, |
|
{ pattern: /\b(partner|beziehung|ehe|freund|freundin|liebe)\b/i, topic: 'Beziehungen' }, |
|
{ pattern: /\b(freund|freundin|freundschaft|kamerad)\b/i, topic: 'Freundschaften' }, |
|
{ pattern: /\b(angst|sorge|befürchtung|panik|phobie)\b/i, topic: 'Ängste' }, |
|
{ pattern: /\b(traum|ziel|wunsch|wünsche|hoffnung)\b/i, topic: 'Träume' }, |
|
{ pattern: /\b(krise|verlust|tod|trauer|trennung)\b/i, topic: 'Verluste' }, |
|
{ pattern: /\b(erfolg|leistung|anerkennung|beförderung)\b/i, topic: 'Erfolge' }, |
|
{ pattern: /\b(gesundheit|krankheit|schmerzen|arzt)\b/i, topic: 'Gesundheit' }, |
|
{ pattern: /\b(hobby|interesse|leidenschaft|musik|sport)\b/i, topic: 'Interessen' } |
|
]; |
|
|
|
|
|
const topics = []; |
|
memoryTriggers.forEach(trigger => { |
|
if (lowerMessage.match(trigger.pattern)) { |
|
topics.push(trigger.topic); |
|
|
|
if (!this.userProfile.keyMemories.includes(trigger.topic)) { |
|
this.userProfile.keyMemories.push(trigger.topic); |
|
} |
|
} |
|
}); |
|
|
|
if (topics.length > 0) { |
|
this.userProfile.lastTopics = [...new Set([...this.userProfile.lastTopics, ...topics])].slice(-5); |
|
} |
|
|
|
|
|
this.detectPersonalityTraits(message); |
|
} |
|
|
|
detectPersonalityTraits(message) { |
|
const traits = { |
|
analytical: ['weil', 'daher', 'deshalb', 'folglich', 'sodass', 'da', 'denn'], |
|
intuitive: ['ich fühle', 'mein bauchgefühl', 'instinktiv', 'intuitiv', 'ich spüre'], |
|
detailOriented: ['genau', 'konkret', 'speziell', 'detailliert', 'im einzelnen'], |
|
bigPicture: ['insgesamt', 'grundsätzlich', 'im großen und ganzen', 'allgemein', 'übergeordnet'], |
|
optimistic: ['hoffentlich', 'positiv', 'gut', 'schön', 'freue', 'erfreulich'], |
|
pessimistic: ['leider', 'negativ', 'schlecht', 'problematisch', 'sorgen', 'befürchte'] |
|
}; |
|
|
|
const lowerMessage = message.toLowerCase(); |
|
const detectedTraits = []; |
|
|
|
for (const [trait, markers] of Object.entries(traits)) { |
|
if (markers.some(marker => lowerMessage.includes(marker))) { |
|
detectedTraits.push(trait); |
|
} |
|
} |
|
|
|
|
|
this.userProfile.personalityTraits = [...new Set([...this.userProfile.personalityTraits, ...detectedTraits])]; |
|
} |
|
|
|
analyzeLinguisticPatterns(message) { |
|
const sentences = message.split(/[.!?]+/).filter(s => s.trim().length > 0); |
|
const words = message.split(/\s+/); |
|
const questions = message.split('?').length - 1; |
|
|
|
this.userProfile.linguisticPatterns = { |
|
sentenceLength: sentences.length > 0 ? words.length / sentences.length : 0, |
|
questionRatio: sentences.length > 0 ? questions / sentences.length : 0, |
|
emotionalWords: words.filter(word => |
|
word.match(/\b(glücklich|traurig|wütend|ängstlich|freude|leid|schmerz|liebe|hass)\b/i) |
|
).length / words.length || 0 |
|
}; |
|
} |
|
|
|
updateConversationStage() { |
|
if (!this.userProfile.name && this.userProfile.conversationStage === 'initial') { |
|
return; |
|
} |
|
|
|
if (this.userProfile.name && this.userProfile.conversationStage === 'initial') { |
|
this.userProfile.conversationStage = 'name_established'; |
|
return; |
|
} |
|
|
|
if (this.userProfile.conversationStage === 'name_established' && this.lastUserMessage.length > 10) { |
|
this.userProfile.conversationStage = 'issue_shared'; |
|
return; |
|
} |
|
|
|
if (this.userProfile.conversationStage === 'issue_shared' && this.userProfile.conversationDepth > 2) { |
|
this.userProfile.conversationStage = 'deep_dive'; |
|
return; |
|
} |
|
} |
|
|
|
analyzeBias(responseText, responseMeta) { |
|
|
|
const scores = { |
|
projection: 0, |
|
pathologization: 0, |
|
solution: 0, |
|
feasibility: 0 |
|
}; |
|
|
|
let primaryBias = 'Keine signifikante Verzerrung'; |
|
let correction = 'Keine spezifische Korrektur erforderlich'; |
|
|
|
|
|
const biasPatterns = { |
|
projection: [ |
|
{ pattern: /\b(Über-Ich|Es|Ich|Abwehrmechanismus|Übertragung|Projektion|Verdrängung)\b/gi, weight: 15 }, |
|
{ pattern: /\b(Sublimierung|Regression|Widerstand|Trieb|Objektbeziehung)\b/gi, weight: 10 }, |
|
{ pattern: /\b(nach Freud|psychoanalytisch betrachtet|unbewusste Prozesse)\b/gi, weight: 20 } |
|
], |
|
pathologization: [ |
|
{ pattern: /\b(Störung|Pathologie|krankhaft|abnorm|dysfunktional)\b/gi, weight: 20 }, |
|
{ pattern: /\b(neurotisch|psychotisch|Defekt|Kompensation|pathologisch)\b/gi, weight: 15 }, |
|
{ pattern: /\b(krank|gestört|unangepasst|maladaptiv)\b/gi, weight: 10 } |
|
], |
|
solution: [ |
|
{ pattern: /\b(Sie sollten|Sie müssen|versuchen Sie|empfehle ich)\b/gi, weight: 25 }, |
|
{ pattern: /\b(mein Rat wäre|Lösung wäre|besser wäre es|ich rate Ihnen)\b/gi, weight: 20 }, |
|
{ pattern: /\b(sollten Sie|würde ich|am besten|ideal wäre)\b/gi, weight: 15 } |
|
], |
|
feasibility: [ |
|
{ pattern: /\b(einfach|schnell|leicht|problemlos)\b/gi, weight: 20 }, |
|
{ pattern: /\b(garantiert|sicherlich|zweifellos|unbedingt)\b/gi, weight: 15 }, |
|
{ pattern: /\b(sofort|direkt|ohne Probleme|mühelos)\b/gi, weight: 10 } |
|
] |
|
}; |
|
|
|
|
|
for (const [biasType, patterns] of Object.entries(biasPatterns)) { |
|
let biasScore = 0; |
|
|
|
patterns.forEach(({pattern, weight}) => { |
|
const matches = responseText.match(pattern) || []; |
|
biasScore += matches.length * weight; |
|
}); |
|
|
|
|
|
scores[biasType] = Math.min(Math.max(biasScore, 0), 100); |
|
|
|
|
|
if (biasType === 'solution' && this.userProfile.conversationStage === 'initial') { |
|
scores[biasType] = Math.max(scores[biasType] - 30, 0); |
|
} |
|
|
|
if (biasType === 'projection' && this.userProfile.conversationStage === 'deep_dive') { |
|
scores[biasType] = Math.min(scores[biasType] + 15, 100); |
|
} |
|
} |
|
|
|
|
|
const maxScore = Math.max(...Object.values(scores)); |
|
if (maxScore > 40) { |
|
if (scores.projection === maxScore) { |
|
primaryBias = 'Theoretischer Projektions-Bias'; |
|
correction = 'Mehr Fokus auf die subjektive Erfahrung des Patienten'; |
|
} else if (scores.pathologization === maxScore) { |
|
primaryBias = 'Pathologisierungs-Bias'; |
|
correction = 'Ausgewogenere Betrachtung normaler menschlicher Erfahrungen'; |
|
} else if (scores.solution === maxScore) { |
|
primaryBias = 'Lösungsorientierungs-Bias'; |
|
correction = 'Mehr Exploration vor Lösungsvorschlägen'; |
|
} else { |
|
primaryBias = 'Machbarkeits-Bias'; |
|
correction = 'Realistischere Einschätzung der Veränderungsmöglichkeiten'; |
|
} |
|
} |
|
|
|
|
|
if (this.userProfile.resistanceLevel > 3 && maxScore < 40) { |
|
primaryBias = 'Widerstands-Bias'; |
|
correction = 'Mehr Anpassung an die Widerstandsebene des Patienten'; |
|
scores.projection = Math.min(scores.projection + 20, 100); |
|
} |
|
|
|
return { |
|
scores, |
|
primaryBias, |
|
correction, |
|
analysis: this.generateBiasAnalysisText(scores, primaryBias) |
|
}; |
|
} |
|
|
|
generateBiasAnalysisText(scores, primaryBias) { |
|
const analysisParts = []; |
|
|
|
if (scores.projection > 50) { |
|
analysisParts.push(`Die Antwort zeigt eine starke Tendenz zur psychoanalytischen Theorieanwendung (${scores.projection}%), was die subjektive Erfahrung des Patienten möglicherweise überlagert.`); |
|
} |
|
|
|
if (scores.pathologization > 50) { |
|
analysisParts.push(`Es besteht eine Neigung zur Pathologisierung normaler Erfahrungen (${scores.pathologization}%), die die diagnostische Neutralität beeinträchtigen könnte.`); |
|
} |
|
|
|
if (scores.solution > 50) { |
|
analysisParts.push(`Die Antwort ist stark lösungsorientiert (${scores.solution}%), was in frühen Gesprächsphasen die Exploration einschränken kann.`); |
|
} |
|
|
|
if (scores.feasibility > 50) { |
|
analysisParts.push(`Die vorgeschlagenen Veränderungen werden möglicherweise zu optimistisch dargestellt (${scores.feasibility}%), was unrealistische Erwartungen wecken könnte.`); |
|
} |
|
|
|
if (analysisParts.length === 0) { |
|
return 'Die Antwort zeigt ein ausgewogenes Verhältnis zwischen Exploration, Theorieanwendung und Neutralität.'; |
|
} |
|
|
|
return analysisParts.join(' '); |
|
} |
|
|
|
generatePersonalizedResponse(message) { |
|
const responseGenerator = new ResponseGenerator(this); |
|
return responseGenerator.generateResponse(message); |
|
} |
|
|
|
addToHistory(message, sender, meta = {}) { |
|
this.conversationHistory.push({ |
|
sender, |
|
text: message, |
|
meta, |
|
timestamp: new Date() |
|
}); |
|
|
|
if (meta.threadId) { |
|
this.currentThread = meta.threadId; |
|
|
|
if (!this.conversationThreads[meta.threadId]) { |
|
this.conversationThreads[meta.threadId] = { |
|
topic: meta.threadTopic || 'Allgemein', |
|
depth: 1, |
|
lastActive: new Date(), |
|
messages: [] |
|
}; |
|
} else { |
|
this.conversationThreads[meta.threadId].depth += 1; |
|
this.conversationThreads[meta.threadId].lastActive = new Date(); |
|
} |
|
|
|
this.conversationThreads[meta.threadId].messages.push({ |
|
text: message, |
|
meta, |
|
timestamp: new Date() |
|
}); |
|
} |
|
} |
|
} |
|
|
|
|
|
class ResponseGenerator { |
|
constructor(conversationManager) { |
|
this.cm = conversationManager; |
|
this.userProfile = conversationManager.userProfile; |
|
} |
|
|
|
generateResponse(message) { |
|
const lowerMessage = message.toLowerCase(); |
|
|
|
|
|
if (!this.userProfile.name && this.userProfile.conversationStage === 'initial') { |
|
return this.generateNameEstablishmentResponse(message); |
|
} |
|
|
|
|
|
if (this.userProfile.name && this.userProfile.conversationStage === 'initial') { |
|
this.userProfile.conversationStage = 'name_established'; |
|
return { |
|
text: ` |
|
<p>Vielen Dank, ${this.userProfile.name}. Schön, Sie kennenzulernen.</p> |
|
<p class="mt-2">Was führt Sie heute zu mir? Was beschäftigt Sie im Moment am meisten?</p> |
|
`, |
|
meta: { |
|
personalizedGreeting: `Hallo ${this.userProfile.name},`, |
|
followUpQuestion: "Würden Sie mir etwas mehr darüber erzählen, was Sie bewegt?", |
|
updateStage: 'name_established' |
|
} |
|
}; |
|
} |
|
|
|
|
|
if (this.userProfile.conversationStage === 'name_established') { |
|
this.userProfile.conversationStage = 'issue_shared'; |
|
|
|
|
|
if (lowerMessage.match(/^(hi|hallo|guten (tag|morgen|abend)|moin|servus|hey|grü(ß|ss)e?)/i)) { |
|
return { |
|
text: ` |
|
<p>${this.userProfile.name}, ich freue mich, unser Gespräch fortzusetzen.</p> |
|
${this.userProfile.lastTopics.length > 0 ? |
|
`<p class="mt-2">Bei unserem letzten Austausch haben wir über ${this.formatTopics(this.userProfile.lastTopics)} gesprochen.</p>` : |
|
''} |
|
<p class="mt-2">Wie geht es Ihnen damit heute?</p> |
|
`, |
|
meta: { |
|
personalizedGreeting: `Willkommen zurück, ${this.userProfile.name}`, |
|
followUpQuestion: "Möchten Sie an unser letztes Gespräch anknüpfen oder etwas Neues besprechen?", |
|
isFollowUp: true, |
|
threadId: 'greeting_' + Date.now(), |
|
threadTopic: 'Begrüßung' |
|
} |
|
}; |
|
} |
|
|
|
|
|
return this.generateIssueResponse(message); |
|
} |
|
|
|
|
|
if (this.userProfile.conversationStage === 'issue_shared') { |
|
return this.generateDeepDiveResponse(message); |
|
} |
|
|
|
|
|
return this.generateOngoingResponse(message); |
|
} |
|
|
|
generateNameEstablishmentResponse(message) { |
|
|
|
const namePatterns = [ |
|
/(?:ich bin|mein name ist|ich heiße|nennen sie mich)\s+([A-ZÄÖÜ][a-zäöüß]+(?:\s+[A-ZÄÖÜ][a-zäöüß]+)*)/i, |
|
/(?:name)\s+(?:ist\s+)?([A-ZÄÖÜ][a-zäöüß]+)/i, |
|
/^([A-ZÄÖÜ][a-zäöüß]+)(?:\s|$)/, |
|
/(?:heiße)\s+([A-ZÄÖÜ][a-zäöüß]+)/i |
|
]; |
|
|
|
let extractedName = null; |
|
for (const pattern of namePatterns) { |
|
const nameMatch = message.match(pattern); |
|
if (nameMatch && nameMatch[1]) { |
|
extractedName = nameMatch[1].trim(); |
|
break; |
|
} |
|
} |
|
|
|
if (extractedName) { |
|
this.userProfile.name = extractedName; |
|
this.userProfile.preferredName = extractedName; |
|
this.userProfile.conversationStage = 'name_established'; |
|
|
|
return { |
|
text: ` |
|
<p>Vielen Dank, ${this.userProfile.name}. Schön, Sie kennenzulernen.</p> |
|
<p class="mt-2">Was führt Sie heute zu mir? Was beschäftigt Sie im Moment am meisten?</p> |
|
`, |
|
meta: { |
|
personalizedGreeting: `Hallo ${this.userProfile.name},`, |
|
followUpQuestion: "Würden Sie mir etwas mehr darüber erzählen, was Sie bewegt?", |
|
updateStage: 'name_established' |
|
} |
|
}; |
|
} else { |
|
|
|
return { |
|
text: ` |
|
<p>Vielen Dank für Ihre Nachricht.</p> |
|
<p class="mt-2">Um unser Gespräch persönlicher zu gestalten, könnten Sie mir bitte mitteilen, wie ich Sie ansprechen soll?</p> |
|
<p class="mt-2">Zum Beispiel: "Ich bin Anna" oder "Nennen Sie mich Max".</p> |
|
`, |
|
meta: { |
|
followUpQuestion: "Wie soll ich Sie nennen?", |
|
|