<!DOCTYPE html>
<html>

<head>
    <title>CarDreamer Visualization</title>
    <style>
        body {
            font-family: 'Arial', sans-serif;
            background-color: #f4f4f8;
            margin: 0;
            padding: 0;
        }

        .container {
            display: flex;
            flex-direction: column;
            align-items: center;
            padding: 20px;
            height: 100vh;
            box-sizing: border-box;
        }

        .section {
            width: 100%;
            margin-bottom: 20px;
            background-color: #ffffff;
            border-radius: 5px;
            box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1);
            padding: 20px;
            box-sizing: border-box;
        }

        .section-title {
            font-size: 18px;
            font-weight: bold;
            color: #333;
            margin-bottom: 10px;
        }

        .image-reward-row {
            display: flex;
            flex-wrap: wrap;
            justify-content: space-between;
        }

        .render-grid {
            display: grid;
            grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
            grid-gap: 20px;
            width: 60%;
        }

        .render-item {
            display: flex;
            flex-direction: column;
            align-items: center;
        }

        .render-image {
            width: 100%;
            height: auto;
            max-height: 512px;
            object-fit: contain;
            border-radius: 5px;
            box-shadow: 0 2px 5px rgba(0, 0, 0, 0.15);
        }

        .save-button {
            margin-top: 10px;
            padding: 5px 10px;
            background-color: #4CAF50;
            color: white;
            border: none;
            border-radius: 4px;
            cursor: pointer;
        }

        .save-button:hover {
            background-color: #45a049;
        }

        .render-key {
            margin-top: 10px;
            font-weight: bold;
            color: #555;
        }

        .reward-plot-container {
            width: 40%;
            height: auto;
            max-height: 400px;
        }

        .plot-grid {
            display: grid;
            grid-template-columns: repeat(auto-fit, minmax(400px, 1fr));
            grid-gap: 20px;
            width: 100%;
            max-width: 100%;
            overflow-x: hidden;
        }

        .plot-item {
            background-color: #ffffff;
            border-radius: 5px;
            box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1);
            padding: 10px;
            box-sizing: border-box;
            width: 100%;
            max-width: 100%;
        }

        .plot-title {
            font-size: 16px;
            font-weight: bold;
            color: #333;
            margin-bottom: 10px;
        }

        .plot-container {
            width: calc(100% - 20px);
            height: 200px;
            margin-right: 20px;
        }

        .info-grid {
            display: grid;
            grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
            grid-gap: 20px;
        }

        .info-item {
            background-color: #ffffff;
            border-radius: 5px;
            box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1);
            padding: 10px;
            box-sizing: border-box;
            display: flex;
            flex-direction: column;
            overflow-y: auto;
            max-height: 150px;
        }

        .info-key {
            font-size: 16px;
            font-weight: bold;
            color: #333;
            margin-bottom: 10px;
            word-wrap: break-word;
        }

        .info-value {
            font-size: 24px;
            font-weight: bold;
            color: #555;
            word-wrap: break-word;
        }

        .hide-button {
            background-color: transparent;
            border: none;
            color: #999;
            font-size: 18px;
            cursor: pointer;
            align-self: flex-end;
            margin-left: auto;
        }

        .hide-button:hover {
            color: #333;
        }
    </style>
</head>

<body>
    <div class="container">
        <div class="section">
            <div class="section-title">Observations</div>
            <div class="image-reward-row">
                <div class="render-grid"></div>
                <div class="reward-plot-container">
                    <canvas></canvas>
                </div>
            </div>
        </div>
        <div class="section">
            <div class="section-title">Info</div>
            <div class="info-grid"></div>
        </div>
        <div class="section">
            <div class="section-title">Plots</div>
            <div class="plot-grid"></div>
        </div>
    </div>

    <script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
    <script src="https://cdn.jsdelivr.net/npm/sortablejs@1.14.0/Sortable.min.js"></script>
    <script>
        const renderGrid = document.querySelector('.render-grid');
        const rewardPlotContainer = document.querySelector('.reward-plot-container');
        const plotGrid = document.querySelector('.plot-grid');
        const infoGrid = document.querySelector('.info-grid');

        const valueHistory = {};

        const eventSource = new EventSource('/stream');
        eventSource.onmessage = function (event) {
            const data = JSON.parse(event.data);
            updateRenderGrid(data.images);
            updateRewardPlot(data.info);
            updatePlotGrid(data.info);
            updateInfoGrid(data.info);
        };

        function updateRenderGrid(images) {
            renderGrid.innerHTML = '';
            images.forEach(item => {
                const renderItem = document.createElement('div');
                renderItem.className = 'render-item';
                const img = document.createElement('img');
                img.src = `data:image/webp;base64,${item.image}`;
                img.className = 'render-image';
                const key = document.createElement('div');
                key.className = 'render-key';
                key.textContent = item.key;
                const saveButton = document.createElement('button');
                saveButton.className = 'save-button';
                saveButton.textContent = 'Save';
                saveButton.addEventListener('click', () => {
                    saveImage(item.key, item.image);
                });
                renderItem.appendChild(img);
                renderItem.appendChild(key);
                renderItem.appendChild(saveButton);
                renderGrid.appendChild(renderItem);
            });
        }

        function saveImage(key, base64Image) {
            const link = document.createElement('a');
            link.href = `data:image/webp;base64,${base64Image}`;
            link.download = `${key}_${Date.now()}.webp`;
            link.click();
        }

        function updateRewardPlot(info) {
            const rewardData = {};
            Object.entries(info).forEach(([key, value]) => {
                if (key.startsWith('r_')) {
                    rewardData[key] = value;
                }
            });

            const colorPalette = generateColorPalette(Object.keys(rewardData).length);

            if (!rewardPlotContainer.chart) {
                const ctx = rewardPlotContainer.querySelector('canvas').getContext('2d');
                rewardPlotContainer.chart = new Chart(ctx, {
                    type: 'line',
                    data: {
                        labels: [],
                        datasets: Object.keys(rewardData).map((key, index) => ({
                            label: key,
                            data: [],
                            borderColor: colorPalette[index],
                            backgroundColor: colorPalette[index] + '80',
                            pointRadius: 0,
                            lineTension: 0.4,
                            borderWidth: 1.5
                        }))
                    },
                    options: {
                        scales: {
                            x: { display: false },
                            y: {
                                display: true,
                                ticks: {
                                    callback: function (value) {
                                        return value.toFixed(2);
                                    }
                                }
                            }
                        },
                        plugins: {
                            legend: {
                                display: true,
                                position: 'bottom',
                                labels: {
                                    boxWidth: 12,
                                    boxHeight: 12,
                                    font: {
                                        size: 12
                                    },
                                    padding: 20,
                                    usePointStyle: true,
                                    generateLabels: function (chart) {
                                        const datasets = chart.data.datasets;
                                        return datasets.map((dataset, i) => ({
                                            text: dataset.label,
                                            fillStyle: dataset.backgroundColor,
                                            strokeStyle: dataset.borderColor,
                                            lineWidth: dataset.borderWidth,
                                            hidden: !chart.isDatasetVisible(i),
                                            index: i
                                        }));
                                    }
                                },
                                onClick: function (e, legendItem, legend) {
                                    const index = legendItem.index;
                                    const ci = legend.chart;
                                    if (ci.isDatasetVisible(index)) {
                                        ci.hide(index);
                                        legendItem.hidden = true;
                                    } else {
                                        ci.show(index);
                                        legendItem.hidden = false;
                                    }
                                    ci.update();
                                }
                            },
                        },
                        aspectRatio: 3,
                        animation: false,
                        responsive: true,
                        maintainAspectRatio: false
                    }
                });
            } else {
                rewardPlotContainer.chart.data.labels.push(rewardPlotContainer.chart.data.labels.length);
                Object.entries(rewardData).forEach(([key, value], index) => {
                    rewardPlotContainer.chart.data.datasets[index].data.push(value);
                });
                if (rewardPlotContainer.chart.data.labels.length > 100) {
                    rewardPlotContainer.chart.data.labels.shift();
                    rewardPlotContainer.chart.data.datasets.forEach(dataset => {
                        dataset.data.shift();
                    });
                }
                rewardPlotContainer.chart.update();
            }
        }

        function updatePlotGrid(info) {
            Object.entries(info).forEach(([key, value]) => {
                if (typeof value === 'number') {
                    let plotItem = plotGrid.querySelector(`[data-key="${key}"]`);
                    if (!plotItem) {
                        plotItem = document.createElement('div');
                        plotItem.className = 'plot-item';
                        plotItem.setAttribute('data-key', key);
                        plotItem.innerHTML = `
                            <div class="plot-title">${key}</div>
                            <div class="plot-container">
                                <canvas></canvas>
                            </div>
                        `;
                        plotGrid.appendChild(plotItem);
                    }
                    updatePlot(key, value, plotItem.querySelector('.plot-container canvas').getContext('2d'));
                }
            });
        }

        function updatePlot(key, value, ctx) {
            if (!valueHistory[key]) {
                valueHistory[key] = [];
            }
            valueHistory[key].push(value);
            if (valueHistory[key].length > 100) {
                valueHistory[key].shift();
            }

            if (!ctx.chart) {
                ctx.chart = new Chart(ctx, {
                    type: 'line',
                    data: {
                        labels: valueHistory[key].map((_, i) => i),
                        datasets: [{
                            label: key,
                            data: valueHistory[key],
                            borderColor: 'rgba(75, 192, 192, 1)',
                            backgroundColor: 'rgba(75, 192, 192, 0.2)',
                            pointRadius: 0,
                            lineTension: 0.4,
                            borderWidth: 1.5
                        }]
                    },
                    options: {
                        scales: {
                            x: { display: false },
                            y: {
                                display: true,
                                ticks: {
                                    callback: function (value) {
                                        return value.toFixed(2);
                                    }
                                }
                            }
                        },
                        plugins: {
                            legend: { display: false },
                            tooltip: { enabled: false }
                        },
                        aspectRatio: 2,
                        animation: false,
                        responsive: true,
                        maintainAspectRatio: false
                    }
                });
            } else {
                ctx.chart.data.labels.push(ctx.chart.data.labels.length);
                ctx.chart.data.datasets[0].data.push(value);
                if (ctx.chart.data.labels.length > 100) {
                    ctx.chart.data.labels.shift();
                    ctx.chart.data.datasets[0].data.shift();
                }
                ctx.chart.update();
            }
        }

        function updateInfoGrid(info) {
            Object.entries(info).forEach(([key, value]) => {
                if (typeof value !== 'number') {
                    let infoItem = infoGrid.querySelector(`[data-key="${key}"]`);
                    if (!infoItem) {
                        infoItem = document.createElement('div');
                        infoItem.className = 'info-item';
                        infoItem.setAttribute('data-key', key);
                        infoItem.innerHTML = `
                            <div class="info-key">${key}</div>
                            <div class="info-value"></div>
                            <button class="hide-button" data-key="${key}">&times;</button>
                        `;
                        infoGrid.appendChild(infoItem);
                        infoItem.querySelector('.hide-button').addEventListener('click', () => {
                            infoItem.style.display = infoItem.style.display === 'none' ? 'flex' : 'none';
                        });
                    }
                    infoItem.querySelector('.info-value').textContent = value;
                }
            });
        }

        function updatePlotGrid(info) {
            Object.entries(info).forEach(([key, value]) => {
                if (typeof value === 'number') {
                    let plotItem = plotGrid.querySelector(`[data-key="${key}"]`);
                    if (!plotItem) {
                        plotItem = document.createElement('div');
                        plotItem.className = 'plot-item';
                        plotItem.setAttribute('data-key', key);
                        plotItem.innerHTML = `
                            <div class="plot-title">${key}</div>
                            <div class="plot-container">
                                <canvas></canvas>
                            </div>
                            <button class="hide-button" data-key="${key}">&times;</button>
                        `;
                        plotGrid.appendChild(plotItem);
                        plotItem.querySelector('.hide-button').addEventListener('click', () => {
                            plotItem.style.display = plotItem.style.display === 'none' ? 'block' : 'none';
                        });
                    }
                    updatePlot(key, value, plotItem.querySelector('.plot-container canvas').getContext('2d'));
                }
            });
        }

        Sortable.create(plotGrid, {
            animation: 150,
            ghostClass: 'plot-item-ghost',
            group: 'grid',
            onEnd: function () {
                adjustGridColumns();
            }
        });

        Sortable.create(infoGrid, {
            animation: 150,
            ghostClass: 'info-item-ghost',
            group: 'grid',
            onEnd: function () {
                adjustGridColumns();
            }
        });

        function adjustGridColumns() {
            const renderItems = renderGrid.querySelectorAll('.render-item');
            const numRenderColumns = Math.min(Math.max(Math.floor(renderGrid.clientWidth / 200), 1), 2);
            renderGrid.style.gridTemplateColumns = `repeat(${numRenderColumns}, 1fr)`;

            const plotItems = plotGrid.querySelectorAll('.plot-item');
            const numPlotColumns = Math.min(Math.max(Math.floor(plotGrid.clientWidth / 400), 1), plotItems.length);
            plotGrid.style.gridTemplateColumns = `repeat(${numPlotColumns}, 1fr)`;

            const infoItems = infoGrid.querySelectorAll('.info-item');
            const numInfoColumns = Math.min(Math.max(Math.floor(infoGrid.clientWidth / 200), 1), infoItems.length);
            infoGrid.style.gridTemplateColumns = `repeat(${numInfoColumns}, 1fr)`;
        }

        function generateColorPalette(numColors) {
            const colors = [];
            for (let i = 0; i < numColors; i++) {
                const hue = (i * 137.5) % 360;
                const color = `hsl(${hue}, 100%, 50%)`;
                colors.push(color);
            }
            return colors;
        }

        window.addEventListener('resize', adjustGridColumns);
    </script>
</body>

</html>
