244 lines
8.8 KiB
Plaintext
244 lines
8.8 KiB
Plaintext
---
|
||
---
|
||
|
||
<!DOCTYPE html>
|
||
<html lang="en">
|
||
<head>
|
||
<meta charset="UTF-8">
|
||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||
<title>Cagire - Forth-based live coding sequencer</title>
|
||
<meta name="description" content="Forth-based live coding music sequencer">
|
||
<meta name="author" content="Raphaël Forment">
|
||
<meta name="keywords" content="live coding, forth, sequencer, music, audio, ableton link, synthesizer">
|
||
<link rel="canonical" href="https://cagire.raphaelforment.fr">
|
||
<meta property="og:title" content="Cagire - Forth-based live coding sequencer">
|
||
<meta property="og:description" content="Forth-based live coding music sequencer">
|
||
<meta property="og:type" content="website">
|
||
<meta property="og:url" content="https://cagire.raphaelforment.fr">
|
||
<meta name="twitter:card" content="summary">
|
||
<meta name="twitter:title" content="Cagire - Forth-based live coding sequencer">
|
||
<meta name="twitter:description" content="Forth-based live coding music sequencer">
|
||
<style>
|
||
@font-face {
|
||
font-family: 'CozetteVector';
|
||
src: url('/CozetteVector.ttf') format('truetype');
|
||
}
|
||
|
||
:root {
|
||
--bg: #000;
|
||
--surface: #121212;
|
||
--text: #fff;
|
||
--text-dim: #b4b4b4;
|
||
--text-muted: #787878;
|
||
}
|
||
|
||
:root.light {
|
||
--bg: #fff;
|
||
--surface: #f0f0f0;
|
||
--text: #000;
|
||
--text-dim: #505050;
|
||
--text-muted: #8c8c8c;
|
||
}
|
||
|
||
body {
|
||
font-family: 'CozetteVector', monospace;
|
||
background: var(--bg);
|
||
color: var(--text);
|
||
max-width: 800px;
|
||
margin: 0 auto;
|
||
padding: 1rem;
|
||
line-height: 1.3;
|
||
font-size: 0.9rem;
|
||
}
|
||
|
||
.header {
|
||
display: flex;
|
||
justify-content: space-between;
|
||
align-items: center;
|
||
margin-bottom: 0.25rem;
|
||
}
|
||
|
||
h1 {
|
||
color: var(--text);
|
||
margin: 0;
|
||
}
|
||
|
||
h2 {
|
||
color: var(--text);
|
||
margin-top: 1rem;
|
||
margin-bottom: 0.25rem;
|
||
font-size: 1rem;
|
||
}
|
||
|
||
p { margin: 0.25rem 0; }
|
||
|
||
video {
|
||
max-width: 100%;
|
||
margin: 0.5rem 0;
|
||
display: block;
|
||
}
|
||
|
||
a { color: var(--text-dim); }
|
||
|
||
ul {
|
||
padding-left: 1.5rem;
|
||
margin: 0.25rem 0;
|
||
}
|
||
|
||
li { margin: 0.1rem 0; }
|
||
|
||
pre {
|
||
background: var(--surface);
|
||
padding: 0.5rem;
|
||
overflow-x: auto;
|
||
margin: 0.25rem 0;
|
||
}
|
||
|
||
.downloads-table {
|
||
border-collapse: collapse;
|
||
margin: 0.5rem 0;
|
||
width: 100%;
|
||
}
|
||
|
||
.downloads-table th,
|
||
.downloads-table td {
|
||
padding: 0.25rem 0.75rem;
|
||
text-align: left;
|
||
}
|
||
|
||
.downloads-table th {
|
||
color: var(--text);
|
||
}
|
||
|
||
.downloads-table td:first-child {
|
||
color: var(--text-muted);
|
||
}
|
||
|
||
.downloads-table tr:nth-child(even) {
|
||
background: var(--surface);
|
||
}
|
||
|
||
.note {
|
||
color: var(--text-muted);
|
||
font-size: 0.8rem;
|
||
font-style: italic;
|
||
}
|
||
|
||
.note a {
|
||
color: var(--text-muted);
|
||
}
|
||
|
||
#theme-toggle {
|
||
font-family: 'CozetteVector', monospace;
|
||
background: var(--surface);
|
||
color: var(--text);
|
||
border: 1px solid var(--text-muted);
|
||
padding: 0.25rem 0.5rem;
|
||
cursor: pointer;
|
||
font-size: 0.8rem;
|
||
}
|
||
|
||
#theme-toggle:hover {
|
||
background: var(--text-muted);
|
||
color: var(--bg);
|
||
}
|
||
</style>
|
||
</head>
|
||
<body>
|
||
<div class="header">
|
||
<h1>CAGIRE: LIVE CODING IN FORTH</h1>
|
||
<button id="theme-toggle" aria-label="Toggle theme">LIGHT</button>
|
||
</div>
|
||
|
||
<table class="downloads-table">
|
||
<tr>
|
||
<th>Platform</th>
|
||
<th>Desktop</th>
|
||
<th>Terminal</th>
|
||
</tr>
|
||
<tr>
|
||
<td>macOS (ARM)</td>
|
||
<td><a href="https://github.com/Bubobubobubobubo/cagire/releases/latest/download/cagire-macos-aarch64-desktop.app.zip">.app</a></td>
|
||
<td><a href="https://github.com/Bubobubobubobubo/cagire/releases/latest/download/cagire-macos-aarch64.tar.gz">.tar.gz</a></td>
|
||
</tr>
|
||
<tr>
|
||
<td>macOS (Intel)</td>
|
||
<td><a href="https://github.com/Bubobubobubobubo/cagire/releases/latest/download/cagire-macos-x86_64-desktop.app.zip">.app</a></td>
|
||
<td><a href="https://github.com/Bubobubobubobubo/cagire/releases/latest/download/cagire-macos-x86_64.tar.gz">.tar.gz</a></td>
|
||
</tr>
|
||
<tr>
|
||
<td>Windows</td>
|
||
<td><a href="https://github.com/Bubobubobubobubo/cagire/releases/latest/download/cagire-windows-x86_64-desktop.exe">.exe</a></td>
|
||
<td><a href="https://github.com/Bubobubobubobubo/cagire/releases/latest/download/cagire-windows-x86_64.zip">.zip</a></td>
|
||
</tr>
|
||
<tr>
|
||
<td>Linux</td>
|
||
<td><a href="https://github.com/Bubobubobubobubo/cagire/releases/latest/download/cagire-linux-x86_64-desktop.deb">.deb</a></td>
|
||
<td><a href="https://github.com/Bubobubobubobubo/cagire/releases/latest/download/cagire-linux-x86_64.tar.gz">.tar.gz</a></td>
|
||
</tr>
|
||
</table>
|
||
<p class="note">All releases on <a href="https://github.com/Bubobubobubobubo/cagire/releases/latest">GitHub</a>. You can also compile the software yourself or get it from Cargo!</p>
|
||
|
||
<video src="/mono_cagire.mp4" autoplay muted loop playsinline></video>
|
||
|
||
<h2>About</h2>
|
||
<p>Cagire is a step sequencer where each step contains a Forth script instead of typical note data. When the sequencer reaches a step, it runs the script. Scripts can produce sound, trigger samples, apply effects, or do nothing at all. You are free to define what your scripts will do. Cagire includes a built-in audio engine called <a href="https://doux.livecoding.fr">Doux</a>. No external software is needed to make sound. It comes with oscillators, sample players, filters, reverb, delay, distortion, and more.</p>
|
||
|
||
<h2>Code Examples</h2>
|
||
<p>A minimal script that plays a middle C note using a sine wave:</p>
|
||
<pre>c4 note sine sound .</pre>
|
||
|
||
<p>Sawtooth wave with lowpass filter, chorus and reverb:</p>
|
||
<pre>saw sound 1200 lpf 0.2 chorus 0.8 verb .</pre>
|
||
|
||
<p>Pitched-down kick drum sample with distortion:</p>
|
||
<pre>kkick sound 1.5 distort 0.8 speed .</pre>
|
||
|
||
<h2>Features</h2>
|
||
<ul>
|
||
<li>32 banks × 32 patterns × 128 steps per project</li>
|
||
<li>Ableton Link synchronization</li>
|
||
<li>Built-in synthesis engine (oscillators, samples, wavetables)</li>
|
||
<li>Effects: filters, reverb, delay, distortion, chorus</li>
|
||
<li>User-defined words and shared variables</li>
|
||
<li>Embedded dictionary and documentation</li>
|
||
</ul>
|
||
|
||
<h2>Live Coding</h2>
|
||
<p>Live coding is a technique where a programmer writes code in real-time, often in front of an audience. It can be used to create music, visual art, and other forms of media. Learn more at <a href="https://toplap.org">TOPLAP</a> or <a href="https://livecoding.fr">livecoding.fr</a>.</p>
|
||
|
||
<h2>Credits</h2>
|
||
<ul>
|
||
<li><a href="https://raphaelforment.fr">BuboBubo</a> (Raphaël Forment)</li>
|
||
<li><a href="https://doux.livecoding.fr">Doux</a> audio engine, Rust port of Dough by <a href="https://eddyflux.cc/">Felix Roos</a></li>
|
||
<li><a href="https://github.com/sourcebox/mi-plaits-dsp-rs">mi-plaits-dsp-rs</a> by Oliver Rockstedt, based on <a href="https://mutable-instruments.net/">Mutable Instruments</a> Plaits by Emilie Gillet</li>
|
||
<li>Related: <a href="https://strudel.cc">Strudel</a>, <a href="https://tidalcycles.org">TidalCycles</a>, <a href="https://sova.livecoding.fr">Sova</a></li>
|
||
</ul>
|
||
|
||
<h2>Links</h2>
|
||
<ul>
|
||
<li><a href="https://github.com/Bubobubobubobubo/cagire">GitHub</a></li>
|
||
<li><a href="https://ko-fi.com/raphaelbubo">Ko-fi</a></li>
|
||
</ul>
|
||
|
||
<p style="margin-top: 2rem; color: var(--text-muted);">AGPL-3.0 License</p>
|
||
|
||
<script>
|
||
const toggle = document.getElementById('theme-toggle');
|
||
const root = document.documentElement;
|
||
|
||
if (localStorage.getItem('theme') === 'light') {
|
||
root.classList.add('light');
|
||
toggle.textContent = 'DARK';
|
||
}
|
||
|
||
toggle.addEventListener('click', () => {
|
||
root.classList.toggle('light');
|
||
const isLight = root.classList.contains('light');
|
||
toggle.textContent = isLight ? 'DARK' : 'LIGHT';
|
||
localStorage.setItem('theme', isLight ? 'light' : 'dark');
|
||
});
|
||
</script>
|
||
</body>
|
||
</html>
|