Merge branch 'main' into audio_device_selection

This commit is contained in:
Felix Roos 2023-12-10 19:32:29 +01:00
commit 6440621d5d
5 changed files with 126 additions and 61 deletions

View File

@ -87,7 +87,7 @@ export function Header({ context }) {
)} )}
</button> </button>
<button <button
onClick={handleUpdate} onClick={() => handleUpdate()}
title="update" title="update"
className={cx( className={cx(
'flex items-center space-x-1', 'flex items-center space-x-1',

View File

@ -17,7 +17,7 @@ import { prebake } from './prebake.mjs';
import * as tunes from './tunes.mjs'; import * as tunes from './tunes.mjs';
import PlayCircleIcon from '@heroicons/react/20/solid/PlayCircleIcon'; import PlayCircleIcon from '@heroicons/react/20/solid/PlayCircleIcon';
import { themes } from './themes.mjs'; import { themes } from './themes.mjs';
import { settingsMap, useSettings, setLatestCode, updateUserCode } from '../settings.mjs'; import { settingsMap, useSettings, setLatestCode, updateUserCode, setActivePattern } from '../settings.mjs';
import Loader from './Loader'; import Loader from './Loader';
import { settingPatterns } from '../settings.mjs'; import { settingPatterns } from '../settings.mjs';
import { code2hash, hash2code } from './helpers.mjs'; import { code2hash, hash2code } from './helpers.mjs';
@ -247,14 +247,21 @@ export function Repl({ embedded = false }) {
stop(); stop();
} }
}; };
const handleUpdate = (newCode) => { const handleUpdate = async (newCode, reset = false) => {
if (reset) {
clearCanvas();
resetLoadedSounds();
scheduler.setCps(1);
await prebake(); // declare default samples
}
(newCode || isDirty) && activateCode(newCode); (newCode || isDirty) && activateCode(newCode);
logger('[repl] code updated! tip: you can also update the code by pressing ctrl+enter', 'highlight'); logger('[repl] code updated!');
}; };
const handleShuffle = async () => { const handleShuffle = async () => {
const { code, name } = getRandomTune(); const { code, name } = getRandomTune();
logger(`[repl] ✨ loading random tune "${name}"`); logger(`[repl] ✨ loading random tune "${name}"`);
setActivePattern(name);
clearCanvas(); clearCanvas();
resetLoadedSounds(); resetLoadedSounds();
scheduler.setCps(1); scheduler.setCps(1);

View File

@ -1,4 +1,4 @@
import React from 'react'; import { useMemo } from 'react';
import * as tunes from '../tunes.mjs'; import * as tunes from '../tunes.mjs';
import { import {
useSettings, useSettings,
@ -8,8 +8,13 @@ import {
deleteActivePattern, deleteActivePattern,
duplicateActivePattern, duplicateActivePattern,
getUserPattern, getUserPattern,
getUserPatterns,
renameActivePattern, renameActivePattern,
addUserPattern,
setUserPatterns,
} from '../../settings.mjs'; } from '../../settings.mjs';
import { logger } from '@strudel.cycles/core';
import { DocumentDuplicateIcon, PencilSquareIcon, TrashIcon } from '@heroicons/react/20/solid';
function classNames(...classes) { function classNames(...classes) {
return classes.filter(Boolean).join(' '); return classes.filter(Boolean).join(' ');
@ -17,77 +22,125 @@ function classNames(...classes) {
export function PatternsTab({ context }) { export function PatternsTab({ context }) {
const { userPatterns, activePattern } = useSettings(); const { userPatterns, activePattern } = useSettings();
const isExample = useMemo(() => activePattern && !!tunes[activePattern], [activePattern]);
return ( return (
<div className="px-4 w-full text-foreground space-y-4"> <div className="px-4 w-full dark:text-white text-stone-900 space-y-4 pb-4">
<section> <section>
<h2 className="text-xl mb-2">Pattern Collection</h2> {activePattern && (
<div className="space-x-4 border-b border-foreground mb-1"> <div className="flex items-center mb-2 space-x-2 overflow-auto">
<h1 className="text-xl">{activePattern}</h1>
<div className="space-x-4 flex w-min">
{!isExample && (
<button className="hover:opacity-50" onClick={() => renameActivePattern()} title="Rename">
<PencilSquareIcon className="w-5 h-5" />
{/* <PencilIcon className="w-5 h-5" /> */}
</button>
)}
<button className="hover:opacity-50" onClick={() => duplicateActivePattern()} title="Duplicate">
<DocumentDuplicateIcon className="w-5 h-5" />
</button>
{!isExample && (
<button className="hover:opacity-50" onClick={() => deleteActivePattern()} title="Delete">
<TrashIcon className="w-5 h-5" />
</button>
)}
</div>
</div>
)}
<div className="font-mono text-sm">
{Object.entries(userPatterns).map(([key, up]) => (
<a
key={key}
className={classNames(
'mr-4 hover:opacity-50 cursor-pointer inline-block',
key === activePattern ? 'outline outline-1' : '',
)}
onClick={() => {
const { code } = up;
setActivePattern(key);
context.handleUpdate(code, true);
}}
>
{key}
</a>
))}
</div>
<div className="pr-4 space-x-4 border-b border-foreground mb-2 h-8 flex overflow-auto max-w-full items-center">
<button <button
className="hover:opacity-50" className="hover:opacity-50"
onClick={() => { onClick={() => {
const name = newUserPattern(); const name = newUserPattern();
const { code } = getUserPattern(name); const { code } = getUserPattern(name);
context.handleUpdate(code); context.handleUpdate(code, true);
}} }}
> >
new new
</button> </button>
<button className="hover:opacity-50" onClick={() => duplicateActivePattern()}>
duplicate
</button>
<button className="hover:opacity-50" onClick={() => renameActivePattern()}>
rename
</button>
<button className="hover:opacity-50" onClick={() => deleteActivePattern()}>
delete
</button>
<button className="hover:opacity-50" onClick={() => clearUserPatterns()}> <button className="hover:opacity-50" onClick={() => clearUserPatterns()}>
clear clear
</button> </button>
</div> <label className="hover:opacity-50 cursor-pointer">
{Object.entries(userPatterns).map(([key, up]) => ( <input
<a style={{ display: 'none' }}
key={key} type="file"
className={classNames( multiple
'mr-4 hover:opacity-50 cursor-pointer inline-block', accept="text/plain,application/json"
key === activePattern ? 'underline' : '', onChange={async (e) => {
)} const files = Array.from(e.target.files);
await Promise.all(
files.map(async (file, i) => {
const content = await file.text();
if (file.type === 'application/json') {
const userPatterns = getUserPatterns() || {};
setUserPatterns({ ...userPatterns, ...JSON.parse(content) });
} else if (file.type === 'text/plain') {
const name = file.name.replace(/\.[^/.]+$/, '');
addUserPattern(name, { code: content });
}
}),
);
logger(`import done!`);
}}
/>
import
</label>
<button
className="hover:opacity-50"
onClick={() => { onClick={() => {
const { code } = up; const blob = new Blob([JSON.stringify(userPatterns)], { type: 'application/json' });
setActivePattern(key); const downloadLink = document.createElement('a');
context.handleUpdate(code); downloadLink.href = window.URL.createObjectURL(blob);
const date = new Date().toISOString().split('T')[0];
downloadLink.download = `strudel_patterns_${date}.json`;
document.body.appendChild(downloadLink);
downloadLink.click();
document.body.removeChild(downloadLink);
}} }}
> >
{key} export
</a> </button>
))} </div>
</section> </section>
<section> <section>
<h2 className="text-xl mb-2">Examples</h2> <h2 className="text-xl mb-2">Examples</h2>
{Object.entries(tunes).map(([key, tune]) => ( <div className="font-mono text-sm">
<a {Object.entries(tunes).map(([key, tune]) => (
key={key} <a
className={classNames( key={key}
'mr-4 hover:opacity-50 cursor-pointer inline-block', className={classNames(
key === activePattern ? 'underline' : '', 'mr-4 hover:opacity-50 cursor-pointer inline-block',
)} key === activePattern ? 'outline outline-1' : '',
onClick={() => { )}
setActivePattern(key); onClick={() => {
context.handleUpdate(tune); setActivePattern(key);
}} context.handleUpdate(tune, true);
> }}
{key} >
</a> {key}
))} </a>
))}
</div>
</section> </section>
</div> </div>
); );
} }
/*
selectable examples
if example selected
type character -> create new user pattern with exampleName_n
even if
clicking (+) opens the "new" example with same behavior as above
*/

View File

@ -42,8 +42,8 @@ export function SoundsTab() {
}); });
}); });
return ( return (
<div id="sounds-tab" className="flex flex-col w-full h-full dark:text-white text-stone-900"> <div id="sounds-tab" className="px-4 flex flex-col w-full h-full dark:text-white text-stone-900">
<div className="px-2 pb-2 flex-none"> <div className="pb-2 flex-none">
<ButtonGroup <ButtonGroup
value={soundsFilter} value={soundsFilter}
onChange={(value) => settingsMap.setKey('soundsFilter', value)} onChange={(value) => settingsMap.setKey('soundsFilter', value)}
@ -55,7 +55,7 @@ export function SoundsTab() {
}} }}
></ButtonGroup> ></ButtonGroup>
</div> </div>
<div className="p-2 min-h-0 max-h-full grow overflow-auto font-mono text-sm break-normal"> <div className="min-h-0 max-h-full grow overflow-auto font-mono text-sm break-normal">
{soundEntries.map(([name, { data, onTrigger }]) => ( {soundEntries.map(([name, { data, onTrigger }]) => (
<span <span
key={name} key={name}

View File

@ -63,14 +63,14 @@ export const fontSize = patternSetting('fontSize');
export const settingPatterns = { theme, fontFamily, fontSize }; export const settingPatterns = { theme, fontFamily, fontSize };
function getUserPatterns() { export function getUserPatterns() {
return JSON.parse(settingsMap.get().userPatterns); return JSON.parse(settingsMap.get().userPatterns);
} }
function getSetting(key) { function getSetting(key) {
return settingsMap.get()[key]; return settingsMap.get()[key];
} }
function setUserPatterns(obj) { export function setUserPatterns(obj) {
settingsMap.setKey('userPatterns', JSON.stringify(obj)); settingsMap.setKey('userPatterns', JSON.stringify(obj));
} }
@ -124,6 +124,10 @@ export function renameActivePattern() {
return; return;
} }
const newName = prompt('Enter new name', activePattern); const newName = prompt('Enter new name', activePattern);
if (newName === null) {
// canceled
return;
}
if (userPatterns[newName]) { if (userPatterns[newName]) {
alert('Name already taken!'); alert('Name already taken!');
return; return;
@ -188,6 +192,7 @@ export function duplicateActivePattern() {
} }
export function setActivePattern(key) { export function setActivePattern(key) {
console.log('set', key);
settingsMap.setKey('activePattern', key); settingsMap.setKey('activePattern', key);
} }
export function importUserPatternJSON(jsonString) {}