| | |
| | const flowchartData = { |
| | q1: { |
| | text: "Did you develop the GPAI model for the sole purpose of scientific research and development?", |
| | type: "question", |
| | context: "You develop a GPAI model", |
| | choices: [ |
| | { text: "Yes", next: "outcome_research" }, |
| | { text: "No", next: "q2" } |
| | ] |
| | }, |
| | q2: { |
| | text: "Have you made the GPAI model available on the EU market including via a commercial activity or via API or open repository?", |
| | type: "question", |
| | choices: [ |
| | { text: "Yes", next: "q3" }, |
| | { text: "No", next: "outcome_research" } |
| | ] |
| | }, |
| | q3: { |
| | text: "Does the GPAI model you've published qualify as posing a potential systemic risk?", |
| | type: "question", |
| | choices: [ |
| | { text: "Yes", next: "outcome_systemic_risk" }, |
| | { text: "No", next: "q4" } |
| | ] |
| | }, |
| | q4: { |
| | text: "Have you published the GPAI model under a free and open-source license along with documentation about model architecture and usage?", |
| | type: "question", |
| | choices: [ |
| | { text: "Yes", next: "q5" }, |
| | { text: "No", next: "outcome_gpai_provider_with_obligations" } |
| | ] |
| | }, |
| | q5: { |
| | text: "Are you monetising the GPAI model by making its availability contingent on payment, procuring other products/services, viewing ads, or receiving/processing personal data?", |
| | type: "question", |
| | choices: [ |
| | { text: "Yes", next: "outcome_gpai_provider_with_obligations" }, |
| | { text: "No", next: "outcome_open_source" } |
| | ] |
| | }, |
| | outcome_research: { |
| | text: "You're not a GPAI provider. GPAI provisions do not apply.", |
| | type: "outcome" |
| | }, |
| | outcome_systemic_risk: { |
| | text: "You're a GPAISR provider. Open source exemptions from GPAI provisions do not apply.", |
| | type: "outcome", |
| | articles: ["Article 53(1)(a)", "Article 53(1)(b)", "Article 53(1)(c)", "Article 53(1)(d)", "Article 54", "Article 55"], |
| | additional: "Additional obligations for GPAI with Systemic Risk: Article 55" |
| | }, |
| | outcome_gpai_provider_with_obligations: { |
| | text: "You're a GPAI provider. Open source exemptions from GPAI provisions do not apply.", |
| | type: "outcome", |
| | class: "gpai", |
| | articles: ["Article 53(1)(a)", "Article 53(1)(b)", "Article 53(1)(c)", "Article 53(1)(d)", "Article 54"], |
| | }, |
| | outcome_open_source: { |
| | text: "You're an open-source GPAI provider. Open source exemptions apply.", |
| | type: "outcome", |
| | class: "gpai", |
| | articles: ["Article 53(1)(a)", "Article 53(1)(b)"], |
| | } |
| | }; |
| |
|
| | |
| | let currentNode = 'q1'; |
| | let navigationHistory = []; |
| |
|
| | |
| | let scale = 1; |
| | let translateX = 0; |
| | let translateY = 0; |
| | let isDragging = false; |
| | let lastMousePos = { x: 0, y: 0 }; |
| |
|
| | |
| | document.addEventListener('DOMContentLoaded', function() { |
| | updateGuidedView(); |
| | initializeMermaid(); |
| | |
| | |
| | document.getElementById('guided-btn').addEventListener('click', () => switchMode('guided')); |
| | document.getElementById('flowchart-btn').addEventListener('click', () => switchMode('flowchart')); |
| | }); |
| |
|
| | function switchMode(mode) { |
| | const guidedMode = document.getElementById('guided-mode'); |
| | const flowchartMode = document.getElementById('flowchart-mode'); |
| | const guidedBtn = document.getElementById('guided-btn'); |
| | const flowchartBtn = document.getElementById('flowchart-btn'); |
| |
|
| | if (mode === 'guided') { |
| | guidedMode.classList.remove('hidden'); |
| | flowchartMode.classList.add('hidden'); |
| | guidedBtn.classList.add('active'); |
| | flowchartBtn.classList.remove('active'); |
| | } else { |
| | guidedMode.classList.add('hidden'); |
| | flowchartMode.classList.remove('hidden'); |
| | guidedBtn.classList.remove('active'); |
| | flowchartBtn.classList.add('active'); |
| | } |
| | } |
| |
|
| | |
| | function updateGuidedView() { |
| | updateContext(); |
| | updateHistory(); |
| | updateCurrentQuestion(); |
| | updateArticles(); |
| | } |
| |
|
| | function updateContext() { |
| | const contextElement = document.querySelector('.context-text'); |
| | const currentData = flowchartData[currentNode]; |
| | contextElement.textContent = currentData.context || "You develop a GPAI model"; |
| | } |
| |
|
| | function updateHistory() { |
| | const historyContainer = document.querySelector('.history-section'); |
| | const historyHTML = navigationHistory.map((item, index) => ` |
| | <div class="history-item" onclick="goToHistoryStep(${index})"> |
| | <div class="history-question">${truncateText(item.question, 50)}</div> |
| | <div class="history-answer">→ ${item.answer}</div> |
| | <div class="tooltip">${item.question}</div> |
| | </div> |
| | `).join(''); |
| | |
| | historyContainer.innerHTML = ` |
| | <h3>Decision History</h3> |
| | ${historyHTML} |
| | `; |
| | } |
| |
|
| | function updateCurrentQuestion() { |
| | const container = document.getElementById('question-container'); |
| | const currentData = flowchartData[currentNode]; |
| | |
| | if (currentData.type === 'question') { |
| | container.innerHTML = ` |
| | <div class="question">${currentData.text}</div> |
| | <div class="choices"> |
| | ${currentData.choices.map(choice => |
| | `<button class="choice-btn" onclick="nextQuestion('${choice.next}', '${choice.text}')">${choice.text}</button>` |
| | ).join('')} |
| | </div> |
| | `; |
| | } else { |
| | |
| | let articlesHTML = ''; |
| | if (currentData.articles) { |
| | const articlesList = currentData.articles.map(article => |
| | `<li>${article}</li>` |
| | ).join(''); |
| | |
| | articlesHTML = ` |
| | <div class="articles-section" style="margin-top: 30px;"> |
| | <h3>Applicable Articles</h3> |
| | <ul class="articles-list"> |
| | ${articlesList} |
| | </ul> |
| | </div> |
| | `; |
| | } |
| | |
| | container.innerHTML = ` |
| | <div class="outcome ${currentData.class || ''}">${currentData.text}</div> |
| | ${articlesHTML} |
| | <button class="restart-btn" onclick="restart()">Start Over</button> |
| | `; |
| | } |
| | } |
| |
|
| | function updateArticles() { |
| | |
| | |
| | } |
| |
|
| | function nextQuestion(nodeId, selectedChoice = null) { |
| | if (selectedChoice) { |
| | navigationHistory.push({ |
| | node: currentNode, |
| | question: flowchartData[currentNode].text, |
| | answer: selectedChoice, |
| | nextNode: nodeId |
| | }); |
| | } |
| | currentNode = nodeId; |
| | updateGuidedView(); |
| | } |
| |
|
| | function goToHistoryStep(stepIndex) { |
| | navigationHistory = navigationHistory.slice(0, stepIndex); |
| | currentNode = stepIndex === 0 ? 'q1' : navigationHistory[stepIndex - 1].nextNode; |
| | updateGuidedView(); |
| | } |
| |
|
| | function restart() { |
| | currentNode = 'q1'; |
| | navigationHistory = []; |
| | updateGuidedView(); |
| | } |
| |
|
| | function truncateText(text, maxLength) { |
| | return text.length > maxLength ? text.substring(0, maxLength) + '...' : text; |
| | } |
| |
|
| | |
| | function initializeMermaid() { |
| | const svg = generateFlowchartSVG(); |
| | document.getElementById('mermaid-container').innerHTML = svg; |
| | setupFlowchartInteraction(); |
| | } |
| |
|
| | function generateFlowchartSVG() { |
| | |
| | const config = { |
| | startX: 200, |
| | startY: 60, |
| | questionSpacing: 220, |
| | outcomeY: 1450, |
| | outcomeSpacing: 350, |
| | articleOffset: 70, |
| | rightOutcomeX: 600, |
| | diamondSize: 100 |
| | }; |
| |
|
| | let svgContent = `<svg viewBox="0 0 1600 1600" style="width: 100%; height: auto;">`; |
| | |
| | |
| | svgContent += ` |
| | <defs> |
| | <marker id="arrowhead" markerWidth="10" markerHeight="7" refX="9" refY="3.5" orient="auto"> |
| | <polygon points="0 0, 10 3.5, 0 7" fill="#333"/> |
| | </marker> |
| | </defs> |
| | `; |
| |
|
| | |
| | const startY = config.startY; |
| | svgContent += generateStartNode(config.startX, startY); |
| |
|
| | |
| | const questions = ['q1', 'q2', 'q3', 'q4', 'q5']; |
| | questions.forEach((qId, index) => { |
| | const y = startY + 60 + (index + 1) * config.questionSpacing; |
| | svgContent += generateQuestion(qId, config.startX, y); |
| | |
| | |
| | const prevY = index === 0 ? startY + 30 : startY + 60 + index * config.questionSpacing + config.diamondSize; |
| | svgContent += `<line x1="${config.startX}" y1="${prevY}" x2="${config.startX}" y2="${y - config.diamondSize}" stroke="#333" stroke-width="2" marker-end="url(#arrowhead)"/>`; |
| | |
| | |
| | if (index < questions.length - 1) { |
| | |
| | const label = (index === 1 || index === 3) ? "Yes" : "No"; |
| | svgContent += `<text x="${config.startX + 20}" y="${y + config.diamondSize + 20}" text-anchor="start" font-size="12" font-weight="bold" fill="#333">${label}</text>`; |
| | } |
| | }); |
| |
|
| | |
| | svgContent += generateOutcome('outcome_research', config.rightOutcomeX, startY + 60 + 2 * config.questionSpacing); |
| | svgContent += generateArticles('outcome_research', config.rightOutcomeX + config.articleOffset, startY + 60 + 2 * config.questionSpacing - 40); |
| | |
| | |
| | const q1Y = startY + 60 + config.questionSpacing; |
| | const q2Y = startY + 60 + 2 * config.questionSpacing; |
| | svgContent += `<line x1="${config.startX + config.diamondSize}" y1="${q1Y}" x2="${config.rightOutcomeX - 150}" y2="${q1Y}" stroke="#333" stroke-width="2"/>`; |
| | svgContent += `<line x1="${config.rightOutcomeX - 150}" y1="${q1Y}" x2="${config.rightOutcomeX - 150}" y2="${q2Y}" stroke="#333" stroke-width="2"/>`; |
| | svgContent += `<line x1="${config.rightOutcomeX - 150}" y1="${q2Y}" x2="${config.rightOutcomeX - 60}" y2="${q2Y}" stroke="#333" stroke-width="2" marker-end="url(#arrowhead)"/>`; |
| | svgContent += `<text x="${config.startX + 140}" y="${q1Y - 5}" text-anchor="middle" font-size="12" font-weight="bold" fill="#333">Yes</text>`; |
| | |
| | |
| | svgContent += `<line x1="${config.startX + config.diamondSize}" y1="${q2Y}" x2="${config.rightOutcomeX - 60}" y2="${q2Y}" stroke="#333" stroke-width="2" marker-end="url(#arrowhead)"/>`; |
| | svgContent += `<text x="${config.startX + 140}" y="${q2Y - 5}" text-anchor="middle" font-size="12" font-weight="bold" fill="#333">No</text>`; |
| |
|
| | |
| | const bottomOutcomes = ['outcome_open_source', 'outcome_gpai_provider_with_obligations', 'outcome_systemic_risk']; |
| | bottomOutcomes.forEach((outcomeId, index) => { |
| | const x = config.startX + index * config.outcomeSpacing; |
| | |
| | |
| | svgContent += generateOutcome(outcomeId, x, config.outcomeY); |
| | |
| | |
| | svgContent += generateArticles(outcomeId, x + config.articleOffset, config.outcomeY - 40); |
| | |
| | |
| | svgContent += generateOutcomeArrow(outcomeId, x, config); |
| | }); |
| |
|
| | svgContent += `</svg>`; |
| | return svgContent; |
| | } |
| |
|
| | function generateStartNode(x, y) { |
| | return ` |
| | <ellipse cx="${x}" cy="${y}" rx="100" ry="30" fill="#FFA726" stroke="#FF8F00" stroke-width="3"/> |
| | <text x="${x}" y="${y + 5}" text-anchor="middle" font-size="14" font-weight="bold" fill="#333">You develop a GPAI model</text> |
| | `; |
| | } |
| |
|
| | function generateQuestion(questionId, x, y) { |
| | const question = flowchartData[questionId]; |
| | if (!question) return ''; |
| |
|
| | const lines = wrapText(question.text, 25); |
| | const lineHeight = 15; |
| | const totalHeight = lines.length * lineHeight; |
| | const startY = y - totalHeight / 2 + lineHeight / 2; |
| | const size = 100; |
| |
|
| | let content = `<polygon points="${x},${y-size} ${x-size},${y} ${x},${y+size} ${x+size},${y}" fill="#A5D6A7" stroke="#4CAF50" stroke-width="3"/>`; |
| | |
| | lines.forEach((line, index) => { |
| | content += `<text x="${x}" y="${startY + index * lineHeight}" text-anchor="middle" font-size="11" font-weight="bold" fill="#333">${line}</text>`; |
| | }); |
| |
|
| | return content; |
| | } |
| |
|
| | function generateOutcome(outcomeId, x, y) { |
| | const outcome = flowchartData[outcomeId]; |
| | if (!outcome) return ''; |
| |
|
| | const lines = wrapText(outcome.text, 18); |
| | const lineHeight = 15; |
| | const totalHeight = lines.length * lineHeight; |
| | const startY = y - totalHeight / 2 + lineHeight / 2; |
| |
|
| | let content = `<circle cx="${x}" cy="${y}" r="60" fill="#333"/>`; |
| | |
| | lines.forEach((line, index) => { |
| | content += `<text x="${x}" y="${startY + index * lineHeight}" text-anchor="middle" font-size="10" fill="white" font-weight="bold">${line}</text>`; |
| | }); |
| |
|
| | return content; |
| | } |
| |
|
| | function generateArticles(outcomeId, x, y) { |
| | const outcome = flowchartData[outcomeId]; |
| | if (!outcome || !outcome.articles) return ''; |
| |
|
| | |
| | const articleCount = outcome.articles.length; |
| | const width = 180; |
| | const height = Math.max(100, articleCount * 18 + 60); |
| | |
| | |
| | const adjustedY = y; |
| | |
| | let content = `<rect x="${x}" y="${adjustedY}" width="${width}" height="${height}" fill="#E8F5E8" stroke="#4CAF50" stroke-width="3" rx="10"/>`; |
| | |
| | |
| | const titleLines = wrapText(`Applicable obligations for ${getOutcomeTitle(outcomeId)}`, 22); |
| | let titleY = adjustedY + 20; |
| | titleLines.forEach((line, index) => { |
| | content += `<text x="${x + width/2}" y="${titleY + index * 15}" text-anchor="middle" font-size="11" font-weight="bold" fill="#333">${line}</text>`; |
| | }); |
| |
|
| | |
| | let articleStartY = adjustedY + 20 + titleLines.length * 15 + 15; |
| | outcome.articles.forEach((article, index) => { |
| | const articleY = articleStartY + index * 18; |
| | content += `<text x="${x + 15}" y="${articleY}" text-anchor="start" font-size="10" fill="#667eea">${article}</text>`; |
| | }); |
| |
|
| | return content; |
| | } |
| |
|
| | function generateOutcomeArrow(outcomeId, x, config) { |
| | const questionY = config.startY + 60; |
| | let content = ''; |
| |
|
| | switch(outcomeId) { |
| | case 'outcome_open_source': |
| | |
| | const q5Y = questionY + 5 * config.questionSpacing; |
| | content = ` |
| | <line x1="${config.startX}" y1="${q5Y + config.diamondSize}" x2="${config.startX}" y2="${config.outcomeY - 60}" stroke="#333" stroke-width="2" marker-end="url(#arrowhead)"/> |
| | <text x="${config.startX - 30}" y="${q5Y + config.diamondSize + 40}" text-anchor="middle" font-size="12" font-weight="bold" fill="#333">No</text> |
| | `; |
| | break; |
| | |
| | case 'outcome_gpai_provider_with_obligations': |
| | |
| | const q4Y = questionY + 4 * config.questionSpacing; |
| | const q5Y_yes = questionY + 5 * config.questionSpacing; |
| | const midPointX = x - 50; |
| | |
| | content = ` |
| | <!-- Q4 No path --> |
| | <line x1="${config.startX + config.diamondSize}" y1="${q4Y}" x2="${midPointX}" y2="${q4Y}" stroke="#333" stroke-width="2"/> |
| | <line x1="${midPointX}" y1="${q4Y}" x2="${midPointX}" y2="${config.outcomeY - 100}" stroke="#333" stroke-width="2"/> |
| | <text x="${config.startX + 150}" y="${q4Y - 5}" text-anchor="middle" font-size="12" font-weight="bold" fill="#333">No</text> |
| | |
| | <!-- Q5 Yes path --> |
| | <line x1="${config.startX + config.diamondSize}" y1="${q5Y_yes}" x2="${midPointX}" y2="${q5Y_yes}" stroke="#333" stroke-width="2"/> |
| | <line x1="${midPointX}" y1="${q5Y_yes}" x2="${midPointX}" y2="${config.outcomeY - 100}" stroke="#333" stroke-width="2"/> |
| | <text x="${config.startX + 150}" y="${q5Y_yes - 5}" text-anchor="middle" font-size="12" font-weight="bold" fill="#333">Yes</text> |
| | |
| | <!-- Combined path to circle --> |
| | <line x1="${midPointX}" y1="${config.outcomeY - 100}" x2="${x}" y2="${config.outcomeY - 60}" stroke="#333" stroke-width="2" marker-end="url(#arrowhead)"/> |
| | `; |
| | break; |
| | |
| | case 'outcome_systemic_risk': |
| | |
| | const q3Y = questionY + 3 * config.questionSpacing; |
| | content = ` |
| | <line x1="${config.startX + config.diamondSize}" y1="${q3Y}" x2="${x}" y2="${q3Y}" stroke="#333" stroke-width="2"/> |
| | <line x1="${x}" y1="${q3Y}" x2="${x}" y2="${config.outcomeY - 60}" stroke="#333" stroke-width="2" marker-end="url(#arrowhead)"/> |
| | <text x="${config.startX + 150}" y="${q3Y - 5}" text-anchor="middle" font-size="12" font-weight="bold" fill="#333">Yes</text> |
| | `; |
| | break; |
| | } |
| |
|
| | return content; |
| | } |
| |
|
| | function getOutcomeTitle(outcomeId) { |
| | switch(outcomeId) { |
| | case 'outcome_research': return 'research providers'; |
| | case 'outcome_open_source': return 'open-source GPAI providers'; |
| | case 'outcome_gpai_provider_with_obligations': return 'GPAI providers'; |
| | case 'outcome_systemic_risk': return 'GPAISR providers'; |
| | default: return 'providers'; |
| | } |
| | } |
| |
|
| | function wrapText(text, maxLength) { |
| | const words = text.split(' '); |
| | const lines = []; |
| | let currentLine = ''; |
| |
|
| | words.forEach(word => { |
| | if ((currentLine + word).length <= maxLength) { |
| | currentLine += (currentLine ? ' ' : '') + word; |
| | } else { |
| | if (currentLine) lines.push(currentLine); |
| | currentLine = word; |
| | } |
| | }); |
| | |
| | if (currentLine) lines.push(currentLine); |
| | return lines; |
| | } |
| |
|
| | |
| | function updateTransform() { |
| | const mermaidContainer = document.getElementById('mermaid-container'); |
| | if (mermaidContainer) { |
| | mermaidContainer.style.transform = `translate(${translateX}px, ${translateY}px) scale(${scale})`; |
| | } |
| | } |
| |
|
| | function zoomIn() { |
| | scale = Math.min(scale * 1.2, 3); |
| | updateTransform(); |
| | } |
| |
|
| | function zoomOut() { |
| | scale = Math.max(scale / 1.2, 0.3); |
| | updateTransform(); |
| | } |
| |
|
| | function resetView() { |
| | scale = 1; |
| | translateX = 0; |
| | translateY = 0; |
| | updateTransform(); |
| | } |
| |
|
| | function setupFlowchartInteraction() { |
| | const container = document.getElementById('flowchart-container'); |
| | if (!container) return; |
| |
|
| | |
| | container.addEventListener('mousedown', function(e) { |
| | isDragging = true; |
| | lastMousePos = { x: e.clientX, y: e.clientY }; |
| | e.preventDefault(); |
| | }); |
| |
|
| | document.addEventListener('mousemove', function(e) { |
| | if (!isDragging) return; |
| | |
| | const deltaX = e.clientX - lastMousePos.x; |
| | const deltaY = e.clientY - lastMousePos.y; |
| | |
| | translateX += deltaX; |
| | translateY += deltaY; |
| | |
| | updateTransform(); |
| | |
| | lastMousePos = { x: e.clientX, y: e.clientY }; |
| | }); |
| |
|
| | document.addEventListener('mouseup', function() { |
| | isDragging = false; |
| | }); |
| |
|
| | |
| | container.addEventListener('wheel', function(e) { |
| | e.preventDefault(); |
| | |
| | const rect = container.getBoundingClientRect(); |
| | const x = e.clientX - rect.left; |
| | const y = e.clientY - rect.top; |
| | |
| | const zoom = e.deltaY > 0 ? 0.9 : 1.1; |
| | const newScale = Math.min(Math.max(scale * zoom, 0.3), 3); |
| | |
| | const factor = newScale / scale; |
| | translateX = x - (x - translateX) * factor; |
| | translateY = y - (y - translateY) * factor; |
| | scale = newScale; |
| | |
| | updateTransform(); |
| | }); |
| |
|
| | |
| | let touchStartDistance = 0; |
| | let touchStartScale = 1; |
| |
|
| | container.addEventListener('touchstart', function(e) { |
| | if (e.touches.length === 1) { |
| | isDragging = true; |
| | lastMousePos = { x: e.touches[0].clientX, y: e.touches[0].clientY }; |
| | } else if (e.touches.length === 2) { |
| | isDragging = false; |
| | const touch1 = e.touches[0]; |
| | const touch2 = e.touches[1]; |
| | touchStartDistance = Math.sqrt( |
| | Math.pow(touch2.clientX - touch1.clientX, 2) + |
| | Math.pow(touch2.clientY - touch1.clientY, 2) |
| | ); |
| | touchStartScale = scale; |
| | } |
| | e.preventDefault(); |
| | }); |
| |
|
| | container.addEventListener('touchmove', function(e) { |
| | if (e.touches.length === 1 && isDragging) { |
| | const deltaX = e.touches[0].clientX - lastMousePos.x; |
| | const deltaY = e.touches[0].clientY - lastMousePos.y; |
| | |
| | translateX += deltaX; |
| | translateY += deltaY; |
| | |
| | updateTransform(); |
| | |
| | lastMousePos = { x: e.touches[0].clientX, y: e.touches[0].clientY }; |
| | } else if (e.touches.length === 2) { |
| | const touch1 = e.touches[0]; |
| | const touch2 = e.touches[1]; |
| | const currentDistance = Math.sqrt( |
| | Math.pow(touch2.clientX - touch1.clientX, 2) + |
| | Math.pow(touch2.clientY - touch1.clientY, 2) |
| | ); |
| | |
| | scale = Math.min(Math.max(touchStartScale * (currentDistance / touchStartDistance), 0.3), 3); |
| | updateTransform(); |
| | } |
| | e.preventDefault(); |
| | }); |
| |
|
| | container.addEventListener('touchend', function(e) { |
| | isDragging = false; |
| | e.preventDefault(); |
| | }); |
| | } |