mirror of
https://github.com/eliasstepanik/strudel.git
synced 2026-01-11 05:38:35 +00:00
125 lines
5.3 KiB
JavaScript
125 lines
5.3 KiB
JavaScript
import React, { useState } from 'react';
|
|
import { Tone } from '@strudel.cycles/tone';
|
|
import useRepl from '../useRepl.mjs';
|
|
import cx from '../cx';
|
|
import useHighlighting from '../useHighlighting';
|
|
import { useInView } from 'react-hook-inview';
|
|
|
|
// eval stuff start
|
|
import { extend } from '@strudel.cycles/eval';
|
|
import * as strudel from '@strudel.cycles/core';
|
|
import gist from '@strudel.cycles/core/gist.js';
|
|
import { mini } from '@strudel.cycles/mini/mini.mjs';
|
|
import { Tone } from '@strudel.cycles/tone';
|
|
import * as toneHelpers from '@strudel.cycles/tone/tone.mjs';
|
|
import * as voicingHelpers from '@strudel.cycles/tonal/voicings.mjs';
|
|
import * as uiHelpers from '@strudel.cycles/tone/ui.mjs';
|
|
import * as drawHelpers from '@strudel.cycles/tone/draw.mjs';
|
|
import euclid from '@strudel.cycles/core/euclid.mjs';
|
|
import '@strudel.cycles/tone/tone.mjs';
|
|
import '@strudel.cycles/midi/midi.mjs';
|
|
import '@strudel.cycles/tonal/voicings.mjs';
|
|
import '@strudel.cycles/tonal/tonal.mjs';
|
|
import '@strudel.cycles/xen/xen.mjs';
|
|
import '@strudel.cycles/xen/tune.mjs';
|
|
import '@strudel.cycles/core/euclid.mjs';
|
|
import '@strudel.cycles/tone/pianoroll.mjs';
|
|
import '@strudel.cycles/tone/draw.mjs';
|
|
import CodeMirror6 from '../CodeMirror6';
|
|
|
|
extend(Tone, strudel, strudel.Pattern.prototype.bootstrap(), toneHelpers, voicingHelpers, drawHelpers, uiHelpers, {
|
|
gist,
|
|
euclid,
|
|
mini,
|
|
Tone,
|
|
});
|
|
// eval stuff end
|
|
|
|
const defaultSynth = new Tone.PolySynth().chain(new Tone.Gain(0.5), Tone.Destination).set({
|
|
oscillator: { type: 'triangle' },
|
|
envelope: {
|
|
release: 0.01,
|
|
},
|
|
});
|
|
|
|
// "balanced" | "interactive" | "playback";
|
|
// Tone.setContext(new Tone.Context({ latencyHint: 'playback', lookAhead: 1 }));
|
|
function MiniRepl({ tune, maxHeight = 500 }) {
|
|
const { code, setCode, pattern, activateCode, error, cycle, dirty, togglePlay } = useRepl({
|
|
tune,
|
|
defaultSynth,
|
|
autolink: false,
|
|
});
|
|
const lines = code.split('\n').length;
|
|
const [view, setView] = useState();
|
|
const [ref, isVisible] = useInView({
|
|
threshold: 0.01,
|
|
});
|
|
useHighlighting({ view, pattern, active: cycle.started });
|
|
return (
|
|
<div className="rounded-md overflow-hidden bg-[#444C57]" ref={ref}>
|
|
<div className="flex justify-between bg-slate-700 border-t border-slate-500">
|
|
<div className="flex">
|
|
<button
|
|
className={cx(
|
|
'w-16 flex items-center justify-center p-1 bg-slate-700 border-r border-slate-500 text-white hover:bg-slate-600',
|
|
cycle.started ? 'animate-pulse' : '',
|
|
)}
|
|
onClick={() => togglePlay()}
|
|
>
|
|
{!cycle.started ? (
|
|
<svg xmlns="http://www.w3.org/2000/svg" className="h-5 w-5" viewBox="0 0 20 20" fill="currentColor">
|
|
<path
|
|
fillRule="evenodd"
|
|
d="M10 18a8 8 0 100-16 8 8 0 000 16zM9.555 7.168A1 1 0 008 8v4a1 1 0 001.555.832l3-2a1 1 0 000-1.664l-3-2z"
|
|
clipRule="evenodd"
|
|
/>
|
|
</svg>
|
|
) : (
|
|
<svg xmlns="http://www.w3.org/2000/svg" className="h-5 w-5" viewBox="0 0 20 20" fill="currentColor">
|
|
<path
|
|
fillRule="evenodd"
|
|
d="M18 10a8 8 0 11-16 0 8 8 0 0116 0zM7 8a1 1 0 012 0v4a1 1 0 11-2 0V8zm5-1a1 1 0 00-1 1v4a1 1 0 102 0V8a1 1 0 00-1-1z"
|
|
clipRule="evenodd"
|
|
/>
|
|
</svg>
|
|
)}
|
|
</button>
|
|
<button
|
|
className={cx(
|
|
'w-16 flex items-center justify-center p-1 border-slate-500 hover:bg-slate-600',
|
|
dirty
|
|
? 'bg-slate-700 border-r border-slate-500 text-white'
|
|
: 'bg-slate-600 text-slate-400 cursor-not-allowed',
|
|
)}
|
|
onClick={() => activateCode()}
|
|
>
|
|
<svg xmlns="http://www.w3.org/2000/svg" className="h-5 w-5" viewBox="0 0 20 20" fill="currentColor">
|
|
<path
|
|
fillRule="evenodd"
|
|
d="M4 2a1 1 0 011 1v2.101a7.002 7.002 0 0111.601 2.566 1 1 0 11-1.885.666A5.002 5.002 0 005.999 7H9a1 1 0 010 2H4a1 1 0 01-1-1V3a1 1 0 011-1zm.008 9.057a1 1 0 011.276.61A5.002 5.002 0 0014.001 13H11a1 1 0 110-2h5a1 1 0 011 1v5a1 1 0 11-2 0v-2.101a7.002 7.002 0 01-11.601-2.566 1 1 0 01.61-1.276z"
|
|
clipRule="evenodd"
|
|
/>
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
<div className="text-right p-1 text-sm">{error && <span className="text-red-200">{error.message}</span>}</div>{' '}
|
|
</div>
|
|
<div className="flex space-y-0 overflow-auto relative">
|
|
{isVisible && <CodeMirror6 value={code} onChange={setCode} onViewChanged={setView} />}
|
|
</div>
|
|
{/* <div className="bg-slate-700 border-t border-slate-500 content-right pr-2 text-right">
|
|
<a href={`https://strudel.tidalcycles.org/#${hash}`} className="text-white items-center inline-flex">
|
|
<span>open in REPL</span>
|
|
<svg xmlns="http://www.w3.org/2000/svg" className="h-5 w-5" viewBox="0 0 20 20" fill="currentColor">
|
|
<path d="M11 3a1 1 0 100 2h2.586l-6.293 6.293a1 1 0 101.414 1.414L15 6.414V9a1 1 0 102 0V4a1 1 0 00-1-1h-5z" />
|
|
<path d="M5 5a2 2 0 00-2 2v8a2 2 0 002 2h8a2 2 0 002-2v-3a1 1 0 10-2 0v3H5V7h3a1 1 0 000-2H5z" />
|
|
</svg>
|
|
</a>
|
|
</div> */}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default MiniRepl;
|