<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>MCP UI Viewer</title>
<style>
body {
margin: 0;
padding: 0;
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif;
}
#debug-panel {
position: fixed;
top: 0;
right: 0;
width: 350px;
height: 100vh;
background: #1e1e1e;
color: #d4d4d4;
padding: 20px;
overflow-y: auto;
font-size: 13px;
border-left: 1px solid #333;
z-index: 10000;
box-shadow: -2px 0 8px rgba(0,0,0,0.3);
}
#debug-panel h3 {
margin: 0 0 16px 0;
font-size: 16px;
color: #4fc3f7;
font-weight: 600;
}
#status {
display: inline-block;
padding: 4px 12px;
border-radius: 12px;
background: #4caf50;
color: white;
font-size: 11px;
font-weight: 600;
}
#debug-log {
background: #252526;
padding: 12px;
border-radius: 6px;
max-height: 500px;
overflow-y: auto;
font-family: 'Monaco', 'Menlo', 'Courier New', monospace;
font-size: 12px;
line-height: 1.5;
margin-top: 16px;
}
.log-entry {
margin-bottom: 10px;
padding: 6px;
border-left: 3px solid #4fc3f7;
padding-left: 10px;
background: rgba(79, 195, 247, 0.05);
border-radius: 0 4px 4px 0;
}
.log-entry.error {
border-left-color: #f44336;
background: rgba(244, 67, 54, 0.05);
color: #ff8a80;
}
.log-entry.success {
border-left-color: #4caf50;
background: rgba(76, 175, 80, 0.05);
color: #69f0ae;
}
.log-entry.warning {
border-left-color: #ff9800;
background: rgba(255, 152, 0, 0.05);
color: #ffab40;
}
#ui-iframe {
position: fixed;
top: 0;
left: 0;
width: calc(100% - 350px);
height: 100vh;
border: none;
}
#toggle-debug {
position: fixed;
top: 12px;
right: 366px;
background: #4fc3f7;
color: white;
border: none;
padding: 10px 20px;
border-radius: 6px;
cursor: pointer;
z-index: 10001;
font-size: 13px;
font-weight: 600;
box-shadow: 0 2px 8px rgba(0,0,0,0.2);
transition: all 0.2s;
}
#toggle-debug:hover {
background: #039be5;
transform: translateY(-1px);
box-shadow: 0 4px 12px rgba(0,0,0,0.3);
}
.hidden {
display: none !important;
}
</style>
</head>
<body>
<button id="toggle-debug" onclick="toggleDebug()">📊 Toggle Debug</button>
<div id="debug-panel">
<h3>🔍 MCP UI Debug Panel</h3>
<div>
<strong>Status:</strong> <span id="status">Ready</span>
</div>
<div style="margin-top: 12px; font-size: 11px; color: #888;">
<strong>Mode:</strong> Static Viewer<br>
<strong>Note:</strong> Tool calls are logged but not executed
</div>
<div>
<h4 style="margin: 16px 0 8px 0; font-size: 14px; color: #4fc3f7;">Tool Calls Log:</h4>
<div id="debug-log"></div>
</div>
</div>
<iframe id="ui-iframe" sandbox="allow-scripts allow-same-origin"></iframe>
<script>
const debugLog = document.getElementById('debug-log');
const status = document.getElementById('status');
const iframe = document.getElementById('ui-iframe');
function log(message, type = 'info') {
const entry = document.createElement('div');
entry.className = `log-entry ${type}`;
const timestamp = new Date().toLocaleTimeString();
entry.innerHTML = `<strong>[${timestamp}]</strong> ${message}`;
debugLog.appendChild(entry);
debugLog.scrollTop = debugLog.scrollHeight;
}
function toggleDebug() {
const panel = document.getElementById('debug-panel');
const button = document.getElementById('toggle-debug');
const iframe = document.getElementById('ui-iframe');
if (panel.classList.contains('hidden')) {
panel.classList.remove('hidden');
iframe.style.width = 'calc(100% - 350px)';
button.style.right = '366px';
button.textContent = '📊 Toggle Debug';
} else {
panel.classList.add('hidden');
iframe.style.width = '100%';
button.style.right = '12px';
button.textContent = '📊 Show Debug';
}
}
// Load the UI HTML into the iframe
const uiHtml = atob('<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>AWS re:Invent Venue Map</title>
    <link rel="stylesheet" href="https://unpkg.com/leaflet@1.9.4/dist/leaflet.css" />
    <style>
        * { margin: 0; padding: 0; box-sizing: border-box; }
        body {
            font-family: system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
            margin: 0;
            padding: 0;
            height: 100vh;
            display: flex;
            flex-direction: column;
        }

        /* Header with filter information */
        .header {
            background: linear-gradient(135deg, #232f3e 0%, #ff9900 100%);
            color: white;
            padding: 12px 20px;
            box-shadow: 0 2px 8px rgba(0,0,0,0.15);
            z-index: 1000;
        }

        .header h1 {
            font-size: 20px;
            margin-bottom: 6px;
            font-weight: 600;
        }

        .filters {
            display: flex;
            gap: 16px;
            flex-wrap: wrap;
            font-size: 13px;
            opacity: 0.95;
        }

        .filter-badge {
            background: rgba(255, 255, 255, 0.2);
            padding: 4px 12px;
            border-radius: 12px;
            backdrop-filter: blur(10px);
        }

        .filter-label {
            opacity: 0.8;
            margin-right: 4px;
        }

        .filter-value {
            font-weight: 600;
        }

        /* Map container */
        #map {
            flex: 1;
            width: 100%;
        }

        /* Custom marker styling */
        .venue-marker {
            background: #ff9900;
            border: 3px solid white;
            border-radius: 50%;
            box-shadow: 0 2px 8px rgba(0,0,0,0.3);
            display: flex;
            align-items: center;
            justify-content: center;
            color: white;
            font-weight: bold;
            font-size: 12px;
        }

        /* Venue popup styling */
        .venue-popup {
            font-family: system-ui, sans-serif;
            min-width: 300px;
            max-width: 400px;
        }

        .venue-popup h3 {
            margin: 0 0 8px 0;
            color: #232f3e;
            font-size: 18px;
            border-bottom: 2px solid #ff9900;
            padding-bottom: 6px;
        }

        .venue-stats {
            background: #f0f8ff;
            padding: 8px 12px;
            border-radius: 6px;
            margin-bottom: 12px;
            border-left: 3px solid #ff9900;
        }

        .venue-stats p {
            margin: 4px 0;
            color: #232f3e;
            font-size: 14px;
        }

        .venue-stats strong {
            color: #ff9900;
        }

        .session-list {
            max-height: 300px;
            overflow-y: auto;
        }

        .session-item {
            padding: 10px;
            margin: 8px 0;
            background: white;
            border: 1px solid #e0e0e0;
            border-radius: 6px;
            transition: all 0.2s;
        }

        .session-item:hover {
            border-color: #ff9900;
            box-shadow: 0 2px 8px rgba(255, 153, 0, 0.2);
        }

        .session-id {
            font-weight: 600;
            color: #ff9900;
            font-size: 12px;
        }

        .session-title {
            color: #232f3e;
            font-size: 14px;
            margin: 4px 0;
            font-weight: 500;
        }

        .session-meta {
            display: flex;
            gap: 12px;
            font-size: 12px;
            color: #666;
            margin-top: 6px;
        }

        .session-meta span {
            display: flex;
            align-items: center;
            gap: 4px;
        }

        .level-badge {
            background: #232f3e;
            color: white;
            padding: 2px 8px;
            border-radius: 10px;
            font-size: 11px;
            font-weight: 600;
        }

        .level-100 { background: #4CAF50; }
        .level-200 { background: #2196F3; }
        .level-300 { background: #FF9800; }
        .level-400 { background: #F44336; }
        .level-500 { background: #9C27B0; }

        .more-sessions {
            text-align: center;
            padding: 12px;
            background: #f5f5f5;
            border-radius: 6px;
            color: #666;
            font-size: 13px;
            margin-top: 8px;
            font-style: italic;
        }

        /* Loading state */
        .loading {
            position: absolute;
            top: 50%;
            left: 50%;
            transform: translate(-50%, -50%);
            text-align: center;
            z-index: 2000;
            background: white;
            padding: 24px 32px;
            border-radius: 8px;
            box-shadow: 0 4px 16px rgba(0,0,0,0.2);
        }

        .spinner {
            border: 3px solid #f3f3f3;
            border-top: 3px solid #ff9900;
            border-radius: 50%;
            width: 40px;
            height: 40px;
            animation: spin 1s linear infinite;
            margin: 0 auto 12px;
        }

        @keyframes spin {
            0% { transform: rotate(0deg); }
            100% { transform: rotate(360deg); }
        }

        /* Scrollbar styling */
        .session-list::-webkit-scrollbar {
            width: 8px;
        }

        .session-list::-webkit-scrollbar-track {
            background: #f1f1f1;
            border-radius: 4px;
        }

        .session-list::-webkit-scrollbar-thumb {
            background: #ff9900;
            border-radius: 4px;
        }

        .session-list::-webkit-scrollbar-thumb:hover {
            background: #e68a00;
        }
    </style>
</head>
<body>
    <div class="header">
        <h1>🗺️ AWS re:Invent 2025 Venue Map</h1>
        <div class="filters" id="filters">
            <div class="filter-badge">
                <span class="filter-label">Total Sessions:</span>
                <span class="filter-value" id="total-sessions">Loading...</span>
            </div>
        </div>
    </div>

    <div id="map"></div>

    <div class="loading" id="loading">
        <div class="spinner"></div>
        <div>Loading venue data...</div>
    </div>

    <script src="https://unpkg.com/leaflet@1.9.4/dist/leaflet.js"></script>
    <script>
        // Initialize the map centered on Las Vegas Strip
        const map = L.map('map').setView([36.1147, -115.1728], 13);

        // Add OpenStreetMap tiles with AWS color scheme
        L.tileLayer('https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', {
            attribution: '© OpenStreetMap contributors',
            maxZoom: 19
        }).addTo(map);

        // Hide loading indicator
        function hideLoading() {
            document.getElementById('loading').style.display = 'none';
        }

        // Show filters in header
        function displayFilters(filters, totalSessions) {
            const filtersDiv = document.getElementById('filters');
            let html = `
                <div class="filter-badge">
                    <span class="filter-label">Total Sessions:</span>
                    <span class="filter-value">${totalSessions.toLocaleString()}</span>
                </div>
            `;

            if (filters.query) {
                html += `
                    <div class="filter-badge">
                        <span class="filter-label">🔍 Search:</span>
                        <span class="filter-value">"${filters.query}"</span>
                    </div>
                `;
            }

            if (filters.day) {
                html += `
                    <div class="filter-badge">
                        <span class="filter-label">📅 Day:</span>
                        <span class="filter-value">${filters.day}</span>
                    </div>
                `;
            }

            if (filters.level) {
                html += `
                    <div class="filter-badge">
                        <span class="filter-label">📊 Level:</span>
                        <span class="filter-value">${filters.level}</span>
                    </div>
                `;
            }

            filtersDiv.innerHTML = html;
        }

        // Create custom div icon with session count
        function createMarkerIcon(sessionCount) {
            // Size based on session count (min 30px, max 60px)
            const size = Math.min(60, Math.max(30, 30 + sessionCount / 10));

            return L.divIcon({
                className: 'venue-marker',
                html: `<div style="width: ${size}px; height: ${size}px; line-height: ${size}px; font-size: ${Math.max(12, size/3)}px;">${sessionCount}</div>`,
                iconSize: [size, size],
                iconAnchor: [size/2, size/2],
                popupAnchor: [0, -size/2]
            });
        }

        // Format session for display
        function formatSession(session) {
            const levelBadge = session.level
                ? `<span class="level-badge level-${session.level}">Level ${session.level}</span>`
                : '';

            return `
                <div class="session-item">
                    <div class="session-id">${session.short_id}</div>
                    <div class="session-title">${session.title}</div>
                    <div class="session-meta">
                        <span>📅 ${session.day} ${session.start_time}-${session.end_time}</span>
                        ${levelBadge}
                    </div>
                    ${session.room ? `<div class="session-meta"><span>📍 ${session.room}</span></div>` : ''}
                </div>
            `;
        }

        // Create popup content for venue
        function createVenuePopup(venue) {
            let html = `
                <div class="venue-popup">
                    <h3>${venue.name}</h3>
                    <div class="venue-stats">
                        <p><strong>${venue.session_count}</strong> matching session${venue.session_count !== 1 ? 's' : ''}</p>
                    </div>
                    <div class="session-list">
            `;

            venue.sessions.forEach(session => {
                html += formatSession(session);
            });

            if (venue.session_count > venue.sessions.length) {
                const remaining = venue.session_count - venue.sessions.length;
                html += `
                    <div class="more-sessions">
                        + ${remaining} more session${remaining !== 1 ? 's' : ''} at this venue
                    </div>
                `;
            }

            html += `
                    </div>
                </div>
            `;

            return html;
        }

        // Handle incoming venue data
        window.addEventListener('message', (event) => {
            if (event.data.type === 'mcp-tool-result') {
                const data = event.data.result;

                hideLoading();

                if (!data.venues || data.venues.length === 0) {
                    alert('No venues found matching your filters. Try adjusting your search criteria.');
                    return;
                }

                // Display filter info
                displayFilters(data.filters_applied, data.total_sessions);

                const bounds = [];

                // Add markers for each venue
                data.venues.forEach(venue => {
                    const marker = L.marker([venue.lat, venue.lon], {
                        icon: createMarkerIcon(venue.session_count)
                    }).addTo(map);

                    bounds.push([venue.lat, venue.lon]);

                    // Create and bind popup
                    const popupContent = createVenuePopup(venue);
                    marker.bindPopup(popupContent, {
                        maxWidth: 420,
                        maxHeight: 500
                    });
                });

                // Fit map to show all venues with padding
                if (bounds.length > 0) {
                    map.fitBounds(bounds, {
                        padding: [50, 50],
                        maxZoom: 14
                    });
                }
            }
        });

        // Request venue data from MCP server
        // This will be called automatically when the UI loads
        window.parent.postMessage({
            jsonrpc: '2.0',
            method: 'tools/call',
            params: {
                name: 'get_venue_map',
                arguments: {}
            },
            id: 1
        }, '*');
    </script>
</body>
</html>
');
const blob = new Blob([uiHtml], { type: 'text/html' });
const blobUrl = URL.createObjectURL(blob);
iframe.src = blobUrl;
log('✅ UI loaded successfully into sandboxed iframe', 'success');
// Listen for postMessage from iframe (tool calls)
window.addEventListener('message', async (event) => {
const data = event.data;
if (data.jsonrpc === '2.0' && data.method === 'tools/call') {
const toolName = data.params?.name || 'unknown';
const args = data.params?.arguments || {};
log(`🔧 Tool call: <strong>${toolName}</strong>`, 'info');
log(`📝 Arguments: ${JSON.stringify(args, null, 2)}`, 'info');
// NOTE: In this static viewer, we can't actually call the MCP server
log('⚠️ Static UI mode - tool calls are logged but not executed', 'warning');
log('💡 For interactive testing, use <code>cargo pmcp test</code> with <code>--serve-ui</code> flag', 'info');
// Send mock error response to UI
iframe.contentWindow.postMessage({
type: 'mcp-tool-result',
id: data.id,
error: {
code: -1,
message: 'Static UI mode - tool call not executed. Use --serve-ui for interactive testing.'
}
}, '*');
} else if (data.type === 'mcp-ui-ready') {
log('✅ UI framework initialized and ready', 'success');
}
});
log('🌉 PostMessage bridge initialized', 'success');
log('👁️ Monitoring tool calls from UI...', 'info');
// Optional: Send initial ready message to UI
setTimeout(() => {
iframe.contentWindow.postMessage({
type: 'mcp-host-ready',
timestamp: Date.now()
}, '*');
}, 100);
</script>
</body>
</html>