refactoring

This commit is contained in:
Jade (Rose) Rowland 2024-01-07 22:48:07 -05:00
parent f7fe44063a
commit 1ff7548b52
3 changed files with 189 additions and 170 deletions

View File

@ -16,18 +16,19 @@ import {
setActivePattern, setActivePattern,
setLatestCode, setLatestCode,
settingsMap, settingsMap,
updateUserCode,
useSettings, useSettings,
getViewingPattern, getViewingPattern,
setViewingPattern, setViewingPattern,
getNextCloneName, createPatternID,
userPattern,
examplePattern,
getNextCloneID,
} from '../settings.mjs'; } from '../settings.mjs';
import { Header } from './Header'; import { Header } from './Header';
import Loader from './Loader'; import Loader from './Loader';
import { Panel } from './panel/Panel'; import { Panel } from './panel/Panel';
import { useStore } from '@nanostores/react'; import { useStore } from '@nanostores/react';
import { prebake } from './prebake.mjs'; import { prebake } from './prebake.mjs';
import * as tunes from './tunes.mjs';
import { getRandomTune, initCode, loadModules, shareCode } from './util.mjs'; import { getRandomTune, initCode, loadModules, shareCode } from './util.mjs';
import PlayCircleIcon from '@heroicons/react/20/solid/PlayCircleIcon'; import PlayCircleIcon from '@heroicons/react/20/solid/PlayCircleIcon';
import './Repl.css'; import './Repl.css';
@ -76,22 +77,26 @@ export function Repl({ embedded = false }) {
}, },
afterEval: ({ code }) => { afterEval: ({ code }) => {
setLatestCode(code); setLatestCode(code);
let pattern = getViewingPattern(); const data = { code };
let id = getViewingPattern();
window.location.hash = '#' + code2hash(code); window.location.hash = '#' + code2hash(code);
const isExamplePattern = !!tunes[pattern]; const examplePatternData = examplePattern.getPatternData(id);
const isExamplePattern = examplePatternData != null;
if (isExamplePattern) { if (isExamplePattern) {
const codeHasChanged = code !== tunes[pattern]; const codeHasChanged = code !== examplePatternData.code;
if (codeHasChanged) { if (codeHasChanged) {
// fork example // fork example
pattern = getNextCloneName(pattern); id = getNextCloneID(id);
setViewingPattern(pattern); setViewingPattern(id);
updateUserCode(pattern, code); userPattern.update(id, data);
} }
setActivePattern(pattern); setActivePattern(id);
} else { } else {
setActivePattern(pattern); id = id == null ? createPatternID() : id;
updateUserCode(pattern, code); setActivePattern(id);
setViewingPattern(id);
userPattern.update(id, data);
} }
}, },
bgFill: false, bgFill: false,
@ -159,21 +164,25 @@ export function Repl({ embedded = false }) {
// payload = {reset?: boolean, code?: string, evaluate?: boolean, pattern?: string } // payload = {reset?: boolean, code?: string, evaluate?: boolean, pattern?: string }
const handleUpdate = async (payload) => { const handleUpdate = async (payload) => {
const { reset = false, code = null, evaluate = true, pattern = null } = payload; const { id, code } = payload;
setViewingPattern(id);
editorRef.current.setCode(code);
if (reset) { // const { reset = false, code = null, evaluate = true, pattern = null } = payload;
clearCanvas();
resetLoadedSounds(); // if (reset) {
editorRef.current.repl.setCps(1); // clearCanvas();
await prebake(); // declare default samples // resetLoadedSounds();
} // editorRef.current.repl.setCps(1);
if (code != null && pattern != null) { // await prebake(); // declare default samples
editorRef.current.setCode(code); // }
setViewingPattern(pattern); // if (code != null && pattern != null) {
} // setViewingPattern(pattern);
if (evaluate) { // editorRef.current.setCode(code);
editorRef.current.evaluate(); // }
} // if (evaluate) {
// editorRef.current.evaluate();
// }
}; };
const handleShuffle = async () => { const handleShuffle = async () => {
// window.postMessage('strudel-shuffle'); // window.postMessage('strudel-shuffle');

View File

@ -1,19 +1,15 @@
import { DocumentDuplicateIcon, PencilSquareIcon, TrashIcon } from '@heroicons/react/20/solid'; import { DocumentDuplicateIcon, PencilSquareIcon, TrashIcon } from '@heroicons/react/20/solid';
import { import {
clearUserPatterns,
deletePattern,
createDuplicatePattern,
exportPatterns, exportPatterns,
addUserPattern,
importPatterns, importPatterns,
createNewUserPattern,
renamePattern,
useActivePattern, useActivePattern,
useViewingPattern, useViewingPattern,
useSettings, useSettings,
userPattern,
examplePattern,
} from '../../settings.mjs'; } from '../../settings.mjs';
import * as tunes from '../tunes.mjs'; import { useMemo } from 'react';
function classNames(...classes) { function classNames(...classes) {
return classes.filter(Boolean).join(' '); return classes.filter(Boolean).join(' ');
@ -34,16 +30,16 @@ function PatternButton({ showOutline, onClick, label, showHiglight }) {
); );
} }
function PatternButtons({ patterns, activePattern, onClick, viewingPattern }) { function PatternButtons({ patterns, activePattern, onClick, viewingPattern, isExample = false }) {
return ( return (
<div className="font-mono text-sm"> <div className="font-mono text-sm">
{Object.entries(patterns).map(([key, data]) => ( {Object.entries(patterns).map(([id]) => (
<PatternButton <PatternButton
key={key} key={id}
label={key} label={id}
showHiglight={key === viewingPattern} showHiglight={id === viewingPattern}
showOutline={key === activePattern} showOutline={id === activePattern}
onClick={() => onClick(key, data)} onClick={() => onClick(id, isExample)}
/> />
))} ))}
</div> </div>
@ -52,20 +48,20 @@ function PatternButtons({ patterns, activePattern, onClick, viewingPattern }) {
export function PatternsTab({ context }) { export function PatternsTab({ context }) {
const { userPatterns } = useSettings(); const { userPatterns } = useSettings();
const examplePatterns = useMemo(() => examplePattern.getAll(), []);
const activePattern = useActivePattern(); const activePattern = useActivePattern();
const viewingPattern = useViewingPattern(); const viewingPattern = useViewingPattern();
const onPatternClick = (pattern, data) => {
const updateCodeWindow = (id, code) => {
context.handleUpdate({ code, id, evaluate: false });
};
const onPatternBtnClick = (id, isExample) => {
const code = isExample ? examplePatterns[id].code : userPatterns[id].code;
// display selected pattern code in the window // display selected pattern code in the window
context.handleUpdate({ pattern, code: data.code, evaluate: false }); updateCodeWindow(id, code);
}; };
const addPattern = ({ pattern, code }) => {
addUserPattern(pattern, { code });
context.handleUpdate({ code, pattern, evaluate: false });
};
const examplePatterns = {};
Object.entries(tunes).forEach(([key, code]) => (examplePatterns[key] = { code }));
const isExample = examplePatterns[viewingPattern] != null; const isExample = examplePatterns[viewingPattern] != null;
return ( return (
<div className="px-4 w-full dark:text-white text-stone-900 space-y-4 pb-4"> <div className="px-4 w-full dark:text-white text-stone-900 space-y-4 pb-4">
@ -75,14 +71,27 @@ export function PatternsTab({ context }) {
<h1 className="text-xl">{viewingPattern}</h1> <h1 className="text-xl">{viewingPattern}</h1>
<div className="space-x-4 flex w-min"> <div className="space-x-4 flex w-min">
{!isExample && ( {!isExample && (
<button className="hover:opacity-50" onClick={() => renamePattern(viewingPattern)} title="Rename"> <button
className="hover:opacity-50"
onClick={() => {
const { id, data } = userPattern.rename(viewingPattern);
updateCodeWindow(id, data.code);
}}
title="Rename"
>
<PencilSquareIcon className="w-5 h-5" /> <PencilSquareIcon className="w-5 h-5" />
{/* <PencilIcon className="w-5 h-5" /> */} {/* <PencilIcon className="w-5 h-5" /> */}
</button> </button>
)} )}
<button <button
className="hover:opacity-50" className="hover:opacity-50"
onClick={() => addPattern(createDuplicatePattern(viewingPattern))} onClick={() => {
const { id, data } = userPattern.duplicate(
viewingPattern,
userPatterns[viewingPattern]?.code ?? examplePatterns[viewingPattern]?.code,
);
updateCodeWindow(id, data.code);
}}
title="Duplicate" title="Duplicate"
> >
<DocumentDuplicateIcon className="w-5 h-5" /> <DocumentDuplicateIcon className="w-5 h-5" />
@ -91,8 +100,9 @@ export function PatternsTab({ context }) {
<button <button
className="hover:opacity-50" className="hover:opacity-50"
onClick={() => { onClick={() => {
const { code, pattern } = deletePattern(viewingPattern); const { id, data } = userPattern.delete(viewingPattern);
context.handleUpdate({ code, pattern, evaluate: false }); console.log({ id, data });
updateCodeWindow(id, data.code);
}} }}
title="Delete" title="Delete"
> >
@ -103,7 +113,7 @@ export function PatternsTab({ context }) {
</div> </div>
)} )}
<PatternButtons <PatternButtons
onClick={onPatternClick} onClick={onPatternBtnClick}
patterns={userPatterns} patterns={userPatterns}
activePattern={activePattern} activePattern={activePattern}
viewingPattern={viewingPattern} viewingPattern={viewingPattern}
@ -112,12 +122,19 @@ export function PatternsTab({ context }) {
<button <button
className="hover:opacity-50" className="hover:opacity-50"
onClick={() => { onClick={() => {
addPattern(createNewUserPattern()); const { id, data } = userPattern.create();
updateCodeWindow(id, data.code);
}} }}
> >
new new
</button> </button>
<button className="hover:opacity-50" onClick={() => clearUserPatterns()}> <button
className="hover:opacity-50"
onClick={() => {
const { id, data } = userPattern.clearAll();
updateCodeWindow(id, data.code);
}}
>
clear clear
</button> </button>
<label className="hover:opacity-50 cursor-pointer"> <label className="hover:opacity-50 cursor-pointer">
@ -138,7 +155,8 @@ export function PatternsTab({ context }) {
<section> <section>
<h2 className="text-xl mb-2">Examples</h2> <h2 className="text-xl mb-2">Examples</h2>
<PatternButtons <PatternButtons
onClick={onPatternClick} isExample={true}
onClick={onPatternBtnClick}
patterns={examplePatterns} patterns={examplePatterns}
activePattern={activePattern} activePattern={activePattern}
viewingPattern={viewingPattern} viewingPattern={viewingPattern}

View File

@ -3,6 +3,7 @@ import { useStore } from '@nanostores/react';
import { register } from '@strudel.cycles/core'; import { register } from '@strudel.cycles/core';
import { defaultAudioDeviceName } from './repl/panel/AudioDeviceSelector'; import { defaultAudioDeviceName } from './repl/panel/AudioDeviceSelector';
import { logger } from '@strudel.cycles/core'; import { logger } from '@strudel.cycles/core';
import * as tunes from './repl/tunes.mjs';
export const defaultSettings = { export const defaultSettings = {
activeFooter: 'intro', activeFooter: 'intro',
@ -27,6 +28,7 @@ export const defaultSettings = {
export const settingsMap = persistentMap('strudel-settings', defaultSettings); export const settingsMap = persistentMap('strudel-settings', defaultSettings);
const defaultCode = '';
//pattern that the use is currently viewing in the window //pattern that the use is currently viewing in the window
const $viewingPattern = persistentAtom('viewingPattern', '', { listen: false }); const $viewingPattern = persistentAtom('viewingPattern', '', { listen: false });
export function setViewingPattern(key) { export function setViewingPattern(key) {
@ -54,7 +56,9 @@ export function useActivePattern() {
export function initUserCode(code) { export function initUserCode(code) {
const userPatterns = getUserPatterns(); const userPatterns = getUserPatterns();
const match = Object.entries(userPatterns).find(([_, pat]) => pat.code === code); const match = Object.entries(userPatterns).find(([_, pat]) => pat.code === code);
setActivePattern(match?.[0] || ''); const id = match?.[0] || '';
setActivePattern(id);
setViewingPattern(id);
} }
export function useSettings() { export function useSettings() {
@ -105,128 +109,116 @@ function getSetting(key) {
} }
export function setUserPatterns(obj) { export function setUserPatterns(obj) {
settingsMap.setKey('userPatterns', JSON.stringify(obj)); return settingsMap.setKey('userPatterns', JSON.stringify(obj));
} }
export function addUserPattern(name, config) { export const createPatternID = () => {
if (typeof config !== 'object') {
throw new Error('addUserPattern expected object as second param');
}
if (!config.code) {
throw new Error('addUserPattern expected code as property of second param');
}
const userPatterns = getUserPatterns();
setUserPatterns({ ...userPatterns, [name]: config });
}
export function createNewUserPattern() {
const userPatterns = getUserPatterns(); const userPatterns = getUserPatterns();
const date = new Date().toISOString().split('T')[0]; const date = new Date().toISOString().split('T')[0];
const todays = Object.entries(userPatterns).filter(([name]) => name.startsWith(date)); const todays = Object.entries(userPatterns).filter(([name]) => name.startsWith(date));
const num = String(todays.length + 1).padStart(3, '0'); const num = String(todays.length + 1).padStart(3, '0');
const pattern = date + '_' + num; const id = date + '_' + num;
const code = 's("hh")'; return id;
return { pattern, code }; };
}
export function clearUserPatterns() { export const getNextCloneID = (id) => {
if (!confirm(`This will delete all your patterns. Are you really sure?`)) { const userPatterns = this.getAll();
return; const clones = Object.entries(userPatterns).filter(([patID]) => patID.startsWith(id));
}
setUserPatterns({});
}
export function getNextCloneName(key) {
const userPatterns = getUserPatterns();
const clones = Object.entries(userPatterns).filter(([name]) => name.startsWith(key));
const num = String(clones.length + 1).padStart(3, '0'); const num = String(clones.length + 1).padStart(3, '0');
return key + '_' + num; const newID = id + '_' + num;
} return newID;
};
export function getUserPattern(key) { export const examplePattern = {
const userPatterns = getUserPatterns(); getAll() {
return userPatterns[key]; const examplePatterns = {};
} Object.entries(tunes).forEach(([key, code]) => (examplePatterns[key] = { code }));
return examplePatterns;
},
getPatternData(id) {
const userPatterns = this.getAll();
return userPatterns[id];
},
};
export function renamePattern(pattern) { export const userPattern = {
let userPatterns = getUserPatterns(); getAll() {
if (!userPatterns[pattern]) { return JSON.parse(settingsMap.get().userPatterns);
alert('Cannot rename examples'); },
return; getPatternData(id) {
} const userPatterns = this.getAll();
const newName = prompt('Enter new name', pattern); return userPatterns[id];
if (newName === null) { },
// canceled exists(id) {
return; const userPatterns = this.getAll();
} return userPatterns[id] != null;
if (userPatterns[newName]) { },
alert('Name already taken!'); create() {
return; const newID = createPatternID();
} const code = defaultCode;
userPatterns[newName] = userPatterns[pattern]; // copy code const data = { code };
delete userPatterns[pattern]; this.update(newID, data);
setUserPatterns({ ...userPatterns }); return { id: newID, data };
setViewingPattern(newName); },
} update(id, data) {
const userPatterns = this.getAll();
setUserPatterns({ ...userPatterns, [id]: data });
},
duplicate(id, data) {
const newID = getNextCloneID(id);
this.update(newID, data);
return { id: newID, data };
},
clearAll() {
if (!confirm(`This will delete all your patterns. Are you really sure?`)) {
return;
}
const viewingPattern = getViewingPattern();
const examplePatternData = examplePattern.getPatternData(viewingPattern);
setUserPatterns({});
if (examplePatternData != null) {
return { id: viewingPattern, data: examplePatternData };
}
setViewingPattern(null);
setActivePattern(null);
return { id: null, data: { code: '' } };
},
delete(id) {
const userPatterns = this.getAll();
const updatedPatterns = Object.fromEntries(Object.entries(userPatterns).filter(([key]) => key !== id));
if (getActivePattern() === id) {
setActivePattern(null);
}
setUserPatterns(updatedPatterns);
const viewingPattern = getViewingPattern();
if (viewingPattern === id) {
return this.create();
}
return { id: viewingPattern, data: updatedPatterns[id] };
},
export function updateUserCode(pattern, code) { rename(id) {
const userPatterns = getUserPatterns(); const userPatterns = this.getAll();
setUserPatterns({ ...userPatterns, [pattern]: { code } }); const newID = prompt('Enter new name', id);
} if (newID === null) {
// canceled
return;
}
if (userPatterns[newID]) {
alert('Name already taken!');
return;
}
const data = userPatterns[id];
userPatterns[newID] = data; // copy code
delete userPatterns[id];
export function deletePattern(pattern) { setUserPatterns({ ...userPatterns });
if (!pattern) { if (id === getActivePattern()) {
console.warn('cannot delete: no pattern selected'); setActivePattern(newID);
return; }
} return { id: newID, data };
const userPatterns = getUserPatterns(); },
if (!userPatterns[pattern]) { };
alert('Cannot delete examples');
return;
}
if (!confirm(`Really delete the selected pattern "${pattern}"?`)) {
return;
}
// const updatedPatterns = Object.fromEntries(Object.entries(userPatterns).filter(([key]) => key !== pattern));
let patternsArray = Object.entries(userPatterns).sort((a, b) => a[0].localeCompare(b[0]));
const deleteIndex = patternsArray.findIndex(([key]) => key === pattern);
patternsArray.splice(deleteIndex, 1);
const updatedPatterns = Object.fromEntries(patternsArray);
setUserPatterns(updatedPatterns);
//create new pattern if no other patterns
if (!patternsArray.length) {
return createNewUserPattern();
}
// // or default to active pattern
// const activePatternID = getActivePattern();
// const activePatternData = updatedPatterns[activePatternID];
// if (activePatternData?.code != null) {
// return { pattern: activePatternID, code: activePatternData.code };
// }
// or find pattern at next index
const next = patternsArray[deleteIndex];
if (next != null) {
const [pat, data] = next;
return { pattern: pat, code: data.code };
}
// or find pattern at previous index
const previous = patternsArray[deleteIndex - 1];
const [pat, data] = previous;
return { patttern: pat, code: data.code };
}
export function createDuplicatePattern(pattern) {
let latestCode = getSetting('latestCode');
if (!pattern) {
console.warn('cannot duplicate: no pattern selected');
return;
}
const newPattern = getNextCloneName(pattern);
return { pattern: newPattern, code: latestCode };
}
export async function importPatterns(fileList) { export async function importPatterns(fileList) {
const files = Array.from(fileList); const files = Array.from(fileList);
@ -237,8 +229,8 @@ export async function importPatterns(fileList) {
const userPatterns = getUserPatterns() || {}; const userPatterns = getUserPatterns() || {};
setUserPatterns({ ...userPatterns, ...JSON.parse(content) }); setUserPatterns({ ...userPatterns, ...JSON.parse(content) });
} else if (file.type === 'text/plain') { } else if (file.type === 'text/plain') {
const name = file.name.replace(/\.[^/.]+$/, ''); const id = file.name.replace(/\.[^/.]+$/, '');
addUserPattern(name, { code: content }); userPattern.update(id, { code: content });
} }
}), }),
); );