summaryrefslogtreecommitdiff
path: root/convert_track.py
diff options
context:
space:
mode:
authorskal <pascal.massimino@gmail.com>2026-02-07 19:22:43 +0100
committerskal <pascal.massimino@gmail.com>2026-02-07 19:22:43 +0100
commit726ae79dd3ba8f368d3a671f371e747c33195edd (patch)
tree9fbd250b47853a4b81312ff6baddd307341cb15c /convert_track.py
parent0eef80ccb12ced607b953bf680459028485b9c67 (diff)
refactor(audio): Convert tracker to unit-less timing system
Changes tracker timing from beat-based to unit-less system to separate musical structure from BPM-dependent playback speed. TIMING CONVENTION: - 1 unit = 4 beats (by convention) - Conversion: seconds = units * (4 / BPM) * 60 - At 120 BPM: 1 unit = 2 seconds BENEFITS: - Pattern structure independent of BPM - BPM changes only affect playback speed, not structure - Easier pattern composition (0.00-1.00 for typical 4-beat pattern) - Fixes issue where patterns played for 2s instead of expected duration DATA STRUCTURES (tracker.h): - TrackerEvent.beat → TrackerEvent.unit_time - TrackerPattern.num_beats → TrackerPattern.unit_length - TrackerPatternTrigger.time_sec → TrackerPatternTrigger.unit_time RUNTIME (tracker.cc): - Added BEATS_PER_UNIT constant (4.0) - Convert units to seconds at playback time using BPM - Pattern remains active for full unit_length duration - Fixed premature pattern deactivation bug COMPILER (tracker_compiler.cc): - Parse LENGTH parameter from PATTERN lines (defaults to 1.0) - Parse unit_time instead of beat values - Generate code with unit-less timing ASSETS: - test_demo.track: converted to unit-less (8 score triggers) - music.track: converted to unit-less (all patterns) - Events: beat/4 conversion (e.g., beat 2.0 → unit 0.50) - Score: seconds/unit_duration (e.g., 4s → 2.0 units at 120 BPM) VISUALIZER (track_visualizer/index.html): - Parse LENGTH parameter and BPM directive - Convert unit-less time to seconds for rendering - Update tick positioning to use unit_time - Display correct pattern durations DOCUMENTATION (doc/TRACKER.md): - Added complete .track format specification - Timing conversion reference table - Examples with unit-less timing - Pattern LENGTH parameter documentation FILES MODIFIED: - src/audio/tracker.{h,cc} (data structures + runtime conversion) - tools/tracker_compiler.cc (parser + code generation) - assets/{test_demo,music}.track (converted to unit-less) - tools/track_visualizer/index.html (BPM-aware rendering) - doc/TRACKER.md (format documentation) - convert_track.py (conversion utility script) TEST RESULTS: - test_demo builds and runs correctly - demo64k builds successfully - Generated code verified (unit-less values in music_data.cc) Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
Diffstat (limited to 'convert_track.py')
-rw-r--r--convert_track.py77
1 files changed, 77 insertions, 0 deletions
diff --git a/convert_track.py b/convert_track.py
new file mode 100644
index 0000000..ec9d62c
--- /dev/null
+++ b/convert_track.py
@@ -0,0 +1,77 @@
+#!/usr/bin/env python3
+"""Convert .track files from beat-based to unit-less timing."""
+
+import re
+import sys
+
+def convert_beat_to_unit(beat_str):
+ """Convert beat value to unit-less (beat / 4)."""
+ beat = float(beat_str)
+ unit = beat / 4.0
+ return f"{unit:.2f}"
+
+def process_line(line):
+ """Process a single line, converting beat values."""
+ line = line.rstrip('\n')
+
+ # Skip comments and empty lines
+ if not line.strip() or line.strip().startswith('#'):
+ return line
+
+ # PATTERN line - add LENGTH 1.0
+ if line.strip().startswith('PATTERN '):
+ # Check if LENGTH already exists
+ if ' LENGTH ' in line:
+ return line
+ parts = line.split()
+ if len(parts) >= 2:
+ return f"PATTERN {parts[1]} LENGTH 1.0"
+ return line
+
+ # Event line (starts with a number)
+ match = re.match(r'^(\s*)([0-9.]+),\s*(.+)$', line)
+ if match:
+ indent, beat_str, rest = match.groups()
+ unit_str = convert_beat_to_unit(beat_str)
+ return f"{indent}{unit_str}, {rest}"
+
+ return line
+
+def main():
+ if len(sys.argv) != 3:
+ print(f"Usage: {sys.argv[0]} <input.track> <output.track>")
+ sys.exit(1)
+
+ input_file = sys.argv[1]
+ output_file = sys.argv[2]
+
+ with open(input_file, 'r') as f:
+ lines = f.readlines()
+
+ # Add header comment about timing
+ output_lines = []
+ output_lines.append("# Enhanced Demo Track - Progressive buildup with varied percussion\n")
+ output_lines.append("# Features acceleration/deceleration with diverse samples and melodic progression\n")
+ output_lines.append("#\n")
+ output_lines.append("# TIMING: Unit-less (1 unit = 4 beats at 120 BPM = 2 seconds)\n")
+ output_lines.append("# Pattern events use unit-less time (0.0-1.0 for 4-beat pattern)\n")
+ output_lines.append("# Score triggers use unit-less time\n")
+
+ # Skip old header lines
+ start_idx = 0
+ for i, line in enumerate(lines):
+ if line.strip() and not line.strip().startswith('#'):
+ start_idx = i
+ break
+
+ # Process rest of file
+ for line in lines[start_idx:]:
+ output_lines.append(process_line(line) + '\n')
+
+ with open(output_file, 'w') as f:
+ f.writelines(output_lines)
+
+ print(f"Converted {input_file} -> {output_file}")
+
+if __name__ == '__main__':
+ main()