Merge pull request #858 from tidalcycles/pattern-organization

Pattern organization
This commit is contained in:
Felix Roos 2023-12-11 22:48:35 +01:00 committed by GitHub
commit 783c856fd7
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 73 additions and 51 deletions

View File

@ -17,7 +17,16 @@ 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, setActivePattern } from '../settings.mjs'; import {
settingsMap,
useSettings,
setLatestCode,
updateUserCode,
setActivePattern,
getActivePattern,
getUserPattern,
initUserCode,
} 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';
@ -131,7 +140,6 @@ export function Repl({ embedded = false }) {
isLineWrappingEnabled, isLineWrappingEnabled,
panelPosition, panelPosition,
isZen, isZen,
activePattern,
} = useSettings(); } = useSettings();
const paintOptions = useMemo(() => ({ fontFamily }), [fontFamily]); const paintOptions = useMemo(() => ({ fontFamily }), [fontFamily]);
@ -177,7 +185,7 @@ export function Repl({ embedded = false }) {
let msg; let msg;
if (decoded) { if (decoded) {
setCode(decoded); setCode(decoded);
setActivePattern(''); initUserCode(decoded);
msg = `I have loaded the code from the URL.`; msg = `I have loaded the code from the URL.`;
} else if (latestCode) { } else if (latestCode) {
setCode(latestCode); setCode(latestCode);

View File

@ -1,27 +1,27 @@
import { DocumentDuplicateIcon, PencilSquareIcon, TrashIcon } from '@heroicons/react/20/solid';
import { useMemo } from 'react'; import { useMemo } from 'react';
import * as tunes from '../tunes.mjs';
import { import {
useSettings,
clearUserPatterns, clearUserPatterns,
newUserPattern,
setActivePattern,
deleteActivePattern, deleteActivePattern,
duplicateActivePattern, duplicateActivePattern,
exportPatterns,
getUserPattern, getUserPattern,
getUserPatterns, importPatterns,
newUserPattern,
renameActivePattern, renameActivePattern,
addUserPattern, setActivePattern,
setUserPatterns, useActivePattern,
useSettings,
} from '../../settings.mjs'; } from '../../settings.mjs';
import { logger } from '@strudel.cycles/core'; import * as tunes from '../tunes.mjs';
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(' ');
} }
export function PatternsTab({ context }) { export function PatternsTab({ context }) {
const { userPatterns, activePattern } = useSettings(); const { userPatterns } = useSettings();
const activePattern = useActivePattern();
const isExample = useMemo(() => activePattern && !!tunes[activePattern], [activePattern]); const isExample = useMemo(() => activePattern && !!tunes[activePattern], [activePattern]);
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">
@ -85,38 +85,11 @@ export function PatternsTab({ context }) {
type="file" type="file"
multiple multiple
accept="text/plain,application/json" accept="text/plain,application/json"
onChange={async (e) => { onChange={(e) => importPatterns(e.target.files)}
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 import
</label> </label>
<button <button className="hover:opacity-50" onClick={() => exportPatterns()}>
className="hover:opacity-50"
onClick={() => {
const blob = new Blob([JSON.stringify(userPatterns)], { type: 'application/json' });
const downloadLink = document.createElement('a');
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);
}}
>
export export
</button> </button>
</div> </div>

View File

@ -1,7 +1,8 @@
import { persistentMap } from '@nanostores/persistent'; import { persistentMap, persistentAtom } from '@nanostores/persistent';
import { useStore } from '@nanostores/react'; import { useStore } from '@nanostores/react';
import { register } from '@strudel.cycles/core'; import { register } from '@strudel.cycles/core';
import * as tunes from './repl/tunes.mjs'; import * as tunes from './repl/tunes.mjs';
import { logger } from '@strudel.cycles/core';
export const defaultSettings = { export const defaultSettings = {
activeFooter: 'intro', activeFooter: 'intro',
@ -19,11 +20,28 @@ export const defaultSettings = {
soundsFilter: 'all', soundsFilter: 'all',
panelPosition: 'bottom', panelPosition: 'bottom',
userPatterns: '{}', userPatterns: '{}',
activePattern: '',
}; };
export const settingsMap = persistentMap('strudel-settings', defaultSettings); export const settingsMap = persistentMap('strudel-settings', defaultSettings);
// active pattern is separate, because it shouldn't sync state across tabs
// reason: https://github.com/tidalcycles/strudel/issues/857
const $activePattern = persistentAtom('activePattern', '', { listen: false });
export function setActivePattern(key) {
$activePattern.set(key);
}
export function getActivePattern() {
return $activePattern.get();
}
export function useActivePattern() {
return useStore($activePattern);
}
export function initUserCode(code) {
const userPatterns = getUserPatterns();
const match = Object.entries(userPatterns).find(([_, pat]) => pat.code === code);
setActivePattern(match?.[0] || '');
}
export function useSettings() { export function useSettings() {
const state = useStore(settingsMap); const state = useStore(settingsMap);
return { return {
@ -116,7 +134,7 @@ export function getUserPattern(key) {
} }
export function renameActivePattern() { export function renameActivePattern() {
let activePattern = getSetting('activePattern'); let activePattern = getActivePattern();
let userPatterns = getUserPatterns(); let userPatterns = getUserPatterns();
if (!userPatterns[activePattern]) { if (!userPatterns[activePattern]) {
alert('Cannot rename examples'); alert('Cannot rename examples');
@ -139,7 +157,7 @@ export function renameActivePattern() {
export function updateUserCode(code) { export function updateUserCode(code) {
const userPatterns = getUserPatterns(); const userPatterns = getUserPatterns();
let activePattern = getSetting('activePattern'); let activePattern = getActivePattern();
// check if code is that of an example tune // check if code is that of an example tune
const [example] = Object.entries(tunes).find(([_, tune]) => tune === code) || []; const [example] = Object.entries(tunes).find(([_, tune]) => tune === code) || [];
if (example && (!activePattern || activePattern === example)) { if (example && (!activePattern || activePattern === example)) {
@ -160,7 +178,7 @@ export function updateUserCode(code) {
} }
export function deleteActivePattern() { export function deleteActivePattern() {
let activePattern = getSetting('activePattern'); let activePattern = getActivePattern();
if (!activePattern) { if (!activePattern) {
console.warn('cannot delete: no pattern selected'); console.warn('cannot delete: no pattern selected');
return; return;
@ -178,7 +196,7 @@ export function deleteActivePattern() {
} }
export function duplicateActivePattern() { export function duplicateActivePattern() {
let activePattern = getSetting('activePattern'); let activePattern = getActivePattern();
let latestCode = getSetting('latestCode'); let latestCode = getSetting('latestCode');
if (!activePattern) { if (!activePattern) {
console.warn('cannot duplicate: no pattern selected'); console.warn('cannot duplicate: no pattern selected');
@ -190,8 +208,31 @@ export function duplicateActivePattern() {
setActivePattern(activePattern); setActivePattern(activePattern);
} }
export function setActivePattern(key) { export async function importPatterns(fileList) {
settingsMap.setKey('activePattern', key); const files = Array.from(fileList);
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!`);
} }
export function importUserPatternJSON(jsonString) {} export async function exportPatterns() {
const userPatterns = getUserPatterns() || {};
const blob = new Blob([JSON.stringify(userPatterns)], { type: 'application/json' });
const downloadLink = document.createElement('a');
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);
}