Merge pull request #855 from tidalcycles/export-patterns

Export patterns + ui tweaks
This commit is contained in:
Felix Roos 2023-12-09 17:25:11 +01:00 committed by GitHub
commit 8b1dfab077
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 93 additions and 52 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

@ -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,10 +8,13 @@ import {
deleteActivePattern, deleteActivePattern,
duplicateActivePattern, duplicateActivePattern,
getUserPattern, getUserPattern,
getUserPatterns,
renameActivePattern, renameActivePattern,
addUserPattern, addUserPattern,
setUserPatterns,
} from '../../settings.mjs'; } from '../../settings.mjs';
import { logger } from '@strudel.cycles/core'; 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(' ');
@ -19,54 +22,31 @@ 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 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">
<section> <section>
<div className="px-4 space-x-4 border-b border-foreground mb-2 h-8"> {activePattern && (
<button <div className="flex items-center mb-2 space-x-2 overflow-auto">
className="hover:opacity-50" <h1 className="text-xl">{activePattern}</h1>
onClick={() => { <div className="space-x-4 flex w-min">
const name = newUserPattern(); {!isExample && (
const { code } = getUserPattern(name); <button className="hover:opacity-50" onClick={() => renameActivePattern()} title="Rename">
context.handleUpdate(code, true); <PencilSquareIcon className="w-5 h-5" />
}} {/* <PencilIcon className="w-5 h-5" /> */}
> </button>
new )}
</button> <button className="hover:opacity-50" onClick={() => duplicateActivePattern()} title="Duplicate">
<button className="hover:opacity-50" onClick={() => duplicateActivePattern()}> <DocumentDuplicateIcon className="w-5 h-5" />
duplicate </button>
</button> {!isExample && (
<button className="hover:opacity-50" onClick={() => renameActivePattern()}> <button className="hover:opacity-50" onClick={() => deleteActivePattern()} title="Delete">
rename <TrashIcon className="w-5 h-5" />
</button> </button>
<button className="hover:opacity-50" onClick={() => deleteActivePattern()}> )}
delete </div>
</button> </div>
<button className="hover:opacity-50" onClick={() => clearUserPatterns()}> )}
clear
</button>
<label className="hover:opacity-50 cursor-pointer">
<input
style={{ display: 'none' }}
type="file"
multiple
accept="text/plain"
onChange={async (e) => {
const files = Array.from(e.target.files);
await Promise.all(
files.map(async (file, i) => {
const code = await file.text();
const name = file.name.replace(/\.[^/.]+$/, '');
console.log(i, name, code);
addUserPattern(name, { code });
}),
);
logger(`import done!`);
}}
/>
import
</label>
</div>
<div className="font-mono text-sm"> <div className="font-mono text-sm">
{Object.entries(userPatterns).map(([key, up]) => ( {Object.entries(userPatterns).map(([key, up]) => (
<a <a
@ -85,6 +65,61 @@ export function PatternsTab({ context }) {
</a> </a>
))} ))}
</div> </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
className="hover:opacity-50"
onClick={() => {
const name = newUserPattern();
const { code } = getUserPattern(name);
context.handleUpdate(code, true);
}}
>
new
</button>
<button className="hover:opacity-50" onClick={() => clearUserPatterns()}>
clear
</button>
<label className="hover:opacity-50 cursor-pointer">
<input
style={{ display: 'none' }}
type="file"
multiple
accept="text/plain,application/json"
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={() => {
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
</button>
</div>
</section> </section>
<section> <section>
<h2 className="text-xl mb-2">Examples</h2> <h2 className="text-xl mb-2">Examples</h2>

View File

@ -62,14 +62,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));
} }
@ -81,7 +81,7 @@ export function addUserPattern(name, config) {
throw new Error('addUserPattern expected code as property of second param'); throw new Error('addUserPattern expected code as property of second param');
} }
const userPatterns = getUserPatterns(); const userPatterns = getUserPatterns();
setUserPatterns({ ...userPatterns, [name]: config }); setUserPatterns({ [name]: config, ...userPatterns });
} }
export function newUserPattern() { export function newUserPattern() {
@ -123,6 +123,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;
@ -152,7 +156,7 @@ export function updateUserCode(code) {
activePattern = getNextCloneName(activePattern); activePattern = getNextCloneName(activePattern);
setActivePattern(activePattern); setActivePattern(activePattern);
} }
setUserPatterns({ ...userPatterns, [activePattern]: { code } }); setUserPatterns({ [activePattern]: { code }, ...userPatterns });
} }
export function deleteActivePattern() { export function deleteActivePattern() {
@ -182,10 +186,12 @@ export function duplicateActivePattern() {
} }
const userPatterns = getUserPatterns(); const userPatterns = getUserPatterns();
activePattern = getNextCloneName(activePattern); activePattern = getNextCloneName(activePattern);
setUserPatterns({ ...userPatterns, [activePattern]: { code: latestCode } }); setUserPatterns({ [activePattern]: { code: latestCode }, ...userPatterns });
setActivePattern(activePattern); setActivePattern(activePattern);
} }
export function setActivePattern(key) { export function setActivePattern(key) {
settingsMap.setKey('activePattern', key); settingsMap.setKey('activePattern', key);
} }
export function importUserPatternJSON(jsonString) {}