sonificator / index.html
DmitryYarov's picture
Add 2 files
9998059 verified
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Shamanic Data Sonification</title>
<script src="https://cdn.tailwindcss.com"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/xlsx/0.18.5/xlsx.full.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/tone/14.8.49/Tone.js"></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=Almendra+Display&family=IM+Fell+DW+Pica&display=swap');
body {
background-color: #0a0a0a;
color: #e2d5b6;
font-family: 'IM Fell DW Pica', serif;
}
.title-font {
font-family: 'Almendra Display', cursive;
text-shadow: 0 0 10px #8a5a44;
}
.drum-circle {
background: radial-gradient(circle, #1a0f0b 0%, #0a0603 100%);
border: 2px solid #5a3921;
box-shadow: 0 0 20px #5a3921;
}
.pulse {
animation: pulse 2s infinite;
}
@keyframes pulse {
0% { box-shadow: 0 0 5px #8a5a44; }
50% { box-shadow: 0 0 20px #c77d3e; }
100% { box-shadow: 0 0 5px #8a5a44; }
}
.data-node {
position: absolute;
width: 12px;
height: 12px;
background-color: #c77d3e;
border-radius: 50%;
transform: translate(-50%, -50%);
opacity: 0.7;
}
canvas {
background-color: rgba(26, 15, 11, 0.7);
border-radius: 8px;
}
.upload-area {
border: 2px dashed #5a3921;
transition: all 0.3s;
}
.upload-area:hover {
border-color: #c77d3e;
background-color: rgba(90, 57, 33, 0.2);
}
.control-btn {
transition: all 0.3s;
background-color: #1a0f0b;
border: 1px solid #5a3921;
}
.control-btn:hover {
background-color: #5a3921;
transform: scale(1.05);
}
.tooltip {
position: relative;
}
.tooltip-text {
visibility: hidden;
width: 200px;
background-color: #1a0f0b;
color: #e2d5b6;
text-align: center;
border-radius: 6px;
padding: 5px;
position: absolute;
z-index: 1;
bottom: 125%;
left: 50%;
transform: translateX(-50%);
opacity: 0;
transition: opacity 0.3s;
border: 1px solid #5a3921;
font-size: 0.9rem;
}
.tooltip:hover .tooltip-text {
visibility: visible;
opacity: 1;
}
</style>
</head>
<body class="min-h-screen flex flex-col">
<div class="container mx-auto px-4 py-8 flex-grow">
<!-- Header -->
<header class="text-center mb-12">
<h1 class="title-font text-5xl md:text-6xl mb-4">Shamanic Data Sonification</h1>
<p class="text-xl max-w-2xl mx-auto">Transform your spreadsheet data into sacred drum rhythms from the spirit world</p>
</header>
<div class="grid grid-cols-1 lg:grid-cols-2 gap-8">
<!-- Left Column -->
<div class="space-y-8">
<!-- File Upload -->
<div class="upload-area rounded-lg p-8 text-center cursor-pointer" id="dropArea">
<div class="flex flex-col items-center justify-center">
<i class="fas fa-file-excel text-5xl mb-4 text-amber-700"></i>
<h3 class="text-2xl mb-2">Upload Your Data</h3>
<p class="mb-4">Drag & drop an Excel file or click to browse</p>
<input type="file" id="fileInput" accept=".xlsx, .xls, .csv" class="hidden" />
<button id="uploadBtn" class="px-6 py-2 rounded-full bg-amber-800 hover:bg-amber-700 transition">
Select File
</button>
</div>
</div>
<!-- Data Visualization -->
<div class="bg-black bg-opacity-30 rounded-lg p-6">
<h3 class="text-2xl mb-4 flex items-center">
<i class="fas fa-chart-line mr-2"></i> Data Visualization
</h3>
<div class="relative h-64 w-full" id="dataVizContainer">
<canvas id="dataCanvas"></canvas>
</div>
</div>
<!-- Parameters -->
<div class="bg-black bg-opacity-30 rounded-lg p-6">
<h3 class="text-2xl mb-4 flex items-center">
<i class="fas fa-sliders-h mr-2"></i> Sonification Parameters
</h3>
<div class="space-y-4">
<div>
<label class="block mb-2">Rhythm Intensity</label>
<input type="range" min="0" max="100" value="50" class="w-full accent-amber-700" id="intensitySlider">
</div>
<div>
<label class="block mb-2">Tempo (BPM)</label>
<input type="range" min="40" max="200" value="120" class="w-full accent-amber-700" id="tempoSlider">
</div>
<div>
<label class="block mb-2">Spiritual Resonance</label>
<input type="range" min="0" max="100" value="30" class="w-full accent-amber-700" id="resonanceSlider">
</div>
</div>
</div>
</div>
<!-- Right Column -->
<div class="space-y-8">
<!-- Drum Visualization -->
<div class="drum-circle rounded-full mx-auto p-8 flex items-center justify-center relative" id="drumCircle">
<div class="w-64 h-64 rounded-full bg-gradient-to-br from-amber-900 to-black flex items-center justify-center pulse">
<div class="w-48 h-48 rounded-full bg-gradient-to-br from-black to-amber-900 flex items-center justify-center">
<div class="w-32 h-32 rounded-full bg-amber-900 flex items-center justify-center">
<div class="w-16 h-16 rounded-full bg-black"></div>
</div>
</div>
</div>
<div id="dataNodes"></div>
</div>
<!-- Controls -->
<div class="bg-black bg-opacity-30 rounded-lg p-6">
<h3 class="text-2xl mb-4 flex items-center">
<i class="fas fa-compact-disc mr-2 animate-spin"></i> Rhythm Controls
</h3>
<div class="grid grid-cols-3 gap-4">
<button id="playBtn" class="control-btn py-3 rounded-lg flex flex-col items-center justify-center tooltip">
<i class="fas fa-play text-2xl mb-1"></i>
<span>Play</span>
<span class="tooltip-text">Begin the sacred rhythm journey</span>
</button>
<button id="stopBtn" class="control-btn py-3 rounded-lg flex flex-col items-center justify-center tooltip">
<i class="fas fa-stop text-2xl mb-1"></i>
<span>Stop</span>
<span class="tooltip-text">Silence the spirits</span>
</button>
<button id="exportBtn" class="control-btn py-3 rounded-lg flex flex-col items-center justify-center tooltip">
<i class="fas fa-download text-2xl mb-1"></i>
<span>Export</span>
<span class="tooltip-text">Save the rhythm as audio</span>
</button>
<button id="randomBtn" class="control-btn py-3 rounded-lg flex flex-col items-center justify-center tooltip">
<i class="fas fa-random text-2xl mb-1"></i>
<span>Randomize</span>
<span class="tooltip-text">Let the spirits choose</span>
</button>
<button id="loopBtn" class="control-btn py-3 rounded-lg flex flex-col items-center justify-center tooltip">
<i class="fas fa-redo text-2xl mb-1"></i>
<span>Loop</span>
<span class="tooltip-text">Create eternal rhythm</span>
</button>
<button id="helpBtn" class="control-btn py-3 rounded-lg flex flex-col items-center justify-center tooltip">
<i class="fas fa-question text-2xl mb-1"></i>
<span>Guide</span>
<span class="tooltip-text">Learn the ways of data sonification</span>
</button>
</div>
</div>
<!-- Rhythm Pattern -->
<div class="bg-black bg-opacity-30 rounded-lg p-6">
<h3 class="text-2xl mb-4 flex items-center">
<i class="fas fa-drum mr-2"></i> Rhythm Pattern
</h3>
<div class="overflow-x-auto">
<table class="w-full text-center" id="patternTable">
<thead>
<tr>
<th class="px-4 py-2">Beat</th>
<th class="px-4 py-2">Type</th>
<th class="px-4 py-2">Pitch</th>
<th class="px-4 py-2">Volume</th>
</tr>
</thead>
<tbody id="patternBody">
<!-- Will be populated by JavaScript -->
</tbody>
</table>
</div>
</div>
</div>
</div>
</div>
<!-- Footer -->
<footer class="bg-black bg-opacity-50 py-6 mt-12">
<div class="container mx-auto px-4 text-center">
<p class="mb-2">Shamanic Data Sonification - Connect with the spirit of your data</p>
<p class="text-sm opacity-70">Created with Tone.js and the wisdom of ancient rhythms</p>
</div>
</footer>
<script>
// Initialize Tone.js
document.addEventListener('DOMContentLoaded', function() {
// Audio context initialization
let isPlaying = false;
let currentData = null;
let tempo = 120;
let loop = false;
let sequence = null;
// Drum sounds
const drumSounds = {
bass: new Tone.MembraneSynth().toDestination(),
tom: new Tone.MembraneSynth({
pitchDecay: 0.05,
octaves: 2,
oscillator: {
type: "sine"
},
envelope: {
attack: 0.001,
decay: 0.5,
sustain: 0.01,
release: 0.5,
attackCurve: "exponential"
}
}).toDestination(),
snare: new Tone.NoiseSynth({
noise: {
type: "white"
},
envelope: {
attack: 0.001,
decay: 0.2,
sustain: 0.01,
release: 0.2
}
}).toDestination(),
shaker: new Tone.NoiseSynth({
noise: {
type: "pink"
},
envelope: {
attack: 0.001,
decay: 0.5,
sustain: 0.01,
release: 0.5
}
}).toDestination()
};
// Set up canvas
const canvas = document.getElementById('dataCanvas');
const ctx = canvas.getContext('2d');
function resizeCanvas() {
const container = document.getElementById('dataVizContainer');
canvas.width = container.clientWidth;
canvas.height = container.clientHeight;
}
window.addEventListener('resize', resizeCanvas);
resizeCanvas();
// File upload handling
const dropArea = document.getElementById('dropArea');
const fileInput = document.getElementById('fileInput');
const uploadBtn = document.getElementById('uploadBtn');
uploadBtn.addEventListener('click', () => fileInput.click());
fileInput.addEventListener('change', handleFiles);
['dragenter', 'dragover', 'dragleave', 'drop'].forEach(eventName => {
dropArea.addEventListener(eventName, preventDefaults, false);
});
function preventDefaults(e) {
e.preventDefault();
e.stopPropagation();
}
['dragenter', 'dragover'].forEach(eventName => {
dropArea.addEventListener(eventName, highlight, false);
});
['dragleave', 'drop'].forEach(eventName => {
dropArea.addEventListener(eventName, unhighlight, false);
});
function highlight() {
dropArea.classList.add('border-amber-600');
}
function unhighlight() {
dropArea.classList.remove('border-amber-600');
}
dropArea.addEventListener('drop', handleDrop, false);
function handleDrop(e) {
const dt = e.dataTransfer;
const files = dt.files;
handleFiles({ target: { files } });
}
function handleFiles(e) {
const files = e.target.files;
if (files.length === 0) return;
const file = files[0];
const reader = new FileReader();
reader.onload = function(e) {
const data = new Uint8Array(e.target.result);
processExcel(data);
};
reader.readAsArrayBuffer(file);
}
function processExcel(data) {
const workbook = XLSX.read(data, { type: 'array' });
const firstSheet = workbook.Sheets[workbook.SheetNames[0]];
const jsonData = XLSX.utils.sheet_to_json(firstSheet, { header: 1 });
currentData = jsonData;
visualizeData(jsonData);
generatePattern(jsonData);
}
function visualizeData(data) {
// Clear previous visualization
ctx.clearRect(0, 0, canvas.width, canvas.height);
// Draw some visualization based on data
if (data.length === 0) return;
const rowsToShow = Math.min(20, data.length);
const colWidth = canvas.width / data[0].length;
const rowHeight = canvas.height / rowsToShow;
// Draw grid
ctx.strokeStyle = '#5a3921';
ctx.lineWidth = 0.5;
// Vertical lines
for (let i = 0; i <= data[0].length; i++) {
ctx.beginPath();
ctx.moveTo(i * colWidth, 0);
ctx.lineTo(i * colWidth, canvas.height);
ctx.stroke();
}
// Horizontal lines
for (let i = 0; i <= rowsToShow; i++) {
ctx.beginPath();
ctx.moveTo(0, i * rowHeight);
ctx.lineTo(canvas.width, i * rowHeight);
ctx.stroke();
}
// Draw data points
for (let row = 0; row < rowsToShow; row++) {
for (let col = 0; col < data[row].length; col++) {
const value = data[row][col];
if (value === undefined || value === null) continue;
// Convert value to a number if possible
let numValue = typeof value === 'string' ? parseFloat(value) : Number(value);
if (isNaN(numValue)) {
// For non-numeric values, use a hash of the string
numValue = hashString(value) % 100;
}
const x = col * colWidth + colWidth / 2;
const y = row * rowHeight + rowHeight / 2;
const radius = Math.max(2, (numValue % 10) + 2);
// Color based on value
const hue = (numValue * 3.6) % 360;
ctx.fillStyle = `hsla(${hue}, 70%, 50%, 0.7)`;
ctx.beginPath();
ctx.arc(x, y, radius, 0, Math.PI * 2);
ctx.fill();
}
}
// Also create data nodes in the drum circle
const drumCircle = document.getElementById('dataNodes');
drumCircle.innerHTML = '';
for (let row = 0; row < rowsToShow; row++) {
for (let col = 0; col < data[row].length; col++) {
const value = data[row][col];
if (value === undefined || value === null) continue;
let numValue = typeof value === 'string' ? parseFloat(value) : Number(value);
if (isNaN(numValue)) {
numValue = hashString(value) % 100;
}
// Position nodes in a spiral pattern within the drum circle
const angle = (col / data[row].length) * Math.PI * 2;
const distance = 0.3 + (row / rowsToShow) * 0.5;
const x = 50 + Math.cos(angle) * distance * 50;
const y = 50 + Math.sin(angle) * distance * 50;
const node = document.createElement('div');
node.className = 'data-node';
node.style.left = `${x}%`;
node.style.top = `${y}%`;
node.style.width = `${Math.max(5, (numValue % 10) + 5)}px`;
node.style.height = `${Math.max(5, (numValue % 10) + 5)}px`;
node.style.backgroundColor = `hsla(${(numValue * 3.6) % 360}, 70%, 50%, 0.7)`;
drumCircle.appendChild(node);
}
}
}
function hashString(str) {
let hash = 0;
for (let i = 0; i < str.length; i++) {
const char = str.charCodeAt(i);
hash = ((hash << 5) - hash) + char;
hash = hash & hash; // Convert to 32bit integer
}
return Math.abs(hash);
}
function generatePattern(data) {
if (!data || data.length === 0) return;
// Clear previous pattern
const patternBody = document.getElementById('patternBody');
patternBody.innerHTML = '';
// Generate rhythm pattern based on data
const pattern = [];
const rowsToUse = Math.min(16, data.length);
for (let i = 0; i < rowsToUse; i++) {
const row = data[i];
if (!row || row.length === 0) continue;
// Determine drum type based on column
const drumTypes = ['bass', 'tom', 'snare', 'shaker'];
const drumType = drumTypes[i % drumTypes.length];
// Get a numeric value from the row
let value = 0;
for (let j = 0; j < row.length; j++) {
const cell = row[j];
if (typeof cell === 'number') {
value += cell;
} else if (typeof cell === 'string') {
const num = parseFloat(cell);
if (!isNaN(num)) {
value += num;
} else {
value += hashString(cell) % 100;
}
}
}
// Normalize value
value = Math.abs(value) % 100;
// Create pattern item
const patternItem = {
beat: i + 1,
type: drumType,
pitch: 30 + (value % 40),
volume: 0.3 + (value % 70) / 100
};
pattern.push(patternItem);
// Add to table
const rowElement = document.createElement('tr');
rowElement.className = i % 2 === 0 ? 'bg-amber-900 bg-opacity-20' : '';
rowElement.innerHTML = `
<td class="px-4 py-2">${patternItem.beat}</td>
<td class="px-4 py-2 capitalize">${patternItem.type}</td>
<td class="px-4 py-2">${patternItem.pitch.toFixed(1)}</td>
<td class="px-4 py-2">${Math.round(patternItem.volume * 100)}%</td>
`;
patternBody.appendChild(rowElement);
}
// Create Tone.js sequence
if (sequence) {
sequence.dispose();
}
sequence = new Tone.Sequence((time, item) => {
if (!item) return;
// Play the drum sound
if (item.type === 'bass' || item.type === 'tom') {
drumSounds[item.type].triggerAttackRelease(item.pitch, "8n", time, item.volume);
} else {
drumSounds[item.type].triggerAttackRelease("8n", time, item.volume);
}
// Visual feedback
animateDrumHit(item.type);
}, pattern, "8n");
// Set tempo
Tone.Transport.bpm.value = tempo;
}
function animateDrumHit(type) {
const drumCircle = document.getElementById('drumCircle');
let color;
switch(type) {
case 'bass': color = '#8a5a44'; break;
case 'tom': color = '#c77d3e'; break;
case 'snare': color = '#e2d5b6'; break;
case 'shaker': color = '#5a3921'; break;
default: color = '#8a5a44';
}
drumCircle.style.boxShadow = `0 0 30px ${color}`;
setTimeout(() => {
drumCircle.style.boxShadow = '0 0 20px #5a3921';
}, 200);
}
// Control buttons
document.getElementById('playBtn').addEventListener('click', () => {
if (!currentData) {
alert('Please upload an Excel file first');
return;
}
if (!isPlaying) {
Tone.start();
Tone.Transport.start();
sequence.start();
isPlaying = true;
document.getElementById('playBtn').querySelector('i').className = 'fas fa-pause text-2xl mb-1';
} else {
Tone.Transport.pause();
isPlaying = false;
document.getElementById('playBtn').querySelector('i').className = 'fas fa-play text-2xl mb-1';
}
});
document.getElementById('stopBtn').addEventListener('click', () => {
Tone.Transport.stop();
sequence.stop();
isPlaying = false;
document.getElementById('playBtn').querySelector('i').className = 'fas fa-play text-2xl mb-1';
});
document.getElementById('exportBtn').addEventListener('click', () => {
if (!currentData) {
alert('Please upload an Excel file and generate a rhythm first');
return;
}
alert('Audio export would be implemented here with Tone.js offline rendering');
// In a real implementation, you would use:
// Tone.Offline(() => {
// // Play your sequence
// }, duration).then(buffer => {
// // Save buffer as WAV file
// });
});
document.getElementById('randomBtn').addEventListener('click', () => {
if (!currentData) {
alert('Please upload an Excel file first');
return;
}
// Randomize some parameters
document.getElementById('intensitySlider').value = Math.floor(Math.random() * 100);
document.getElementById('tempoSlider').value = 40 + Math.floor(Math.random() * 160);
document.getElementById('resonanceSlider').value = Math.floor(Math.random() * 100);
// Regenerate pattern
generatePattern(currentData);
});
document.getElementById('loopBtn').addEventListener('click', function() {
loop = !loop;
sequence.loop = loop;
this.classList.toggle('bg-amber-800');
this.querySelector('i').classList.toggle('text-amber-500');
});
document.getElementById('helpBtn').addEventListener('click', () => {
alert('Shamanic Data Sonification Guide:\n\n1. Upload your Excel file\n2. Adjust parameters to taste\n3. Play the rhythm\n4. Connect with the spirit world\n\nColumns become drum types, rows become beats, and values affect pitch and volume.');
});
// Slider events
document.getElementById('tempoSlider').addEventListener('input', function() {
tempo = this.value;
Tone.Transport.bpm.value = tempo;
});
document.getElementById('intensitySlider').addEventListener('input', function() {
if (currentData) {
generatePattern(currentData);
}
});
document.getElementById('resonanceSlider').addEventListener('input', function() {
// This would affect reverb/delay in a more complete implementation
// For now just regenerate pattern
if (currentData) {
generatePattern(currentData);
}
});
});
</script>
<p style="border-radius: 8px; text-align: center; font-size: 12px; color: #fff; margin-top: 16px;position: fixed; left: 8px; bottom: 8px; z-index: 10; background: rgba(0, 0, 0, 0.8); padding: 4px 8px;">Made with <img src="https://enzostvs-deepsite.hf.space/logo.svg" alt="DeepSite Logo" style="width: 16px; height: 16px; vertical-align: middle;display:inline-block;margin-right:3px;filter:brightness(0) invert(1);"><a href="https://enzostvs-deepsite.hf.space" style="color: #fff;text-decoration: underline;" target="_blank" >DeepSite</a> - 🧬 <a href="https://enzostvs-deepsite.hf.space?remix=DmitryYarov/sonificator" style="color: #fff;text-decoration: underline;" target="_blank" >Remix</a></p></body>
</html>