strudel-docker/website/src/settings.mjs
Jade (Rose) Rowland 1ff7548b52 refactoring
2024-01-07 22:48:07 -05:00

251 lines
8.0 KiB
JavaScript

import { persistentMap, persistentAtom } from '@nanostores/persistent';
import { useStore } from '@nanostores/react';
import { register } from '@strudel.cycles/core';
import { defaultAudioDeviceName } from './repl/panel/AudioDeviceSelector';
import { logger } from '@strudel.cycles/core';
import * as tunes from './repl/tunes.mjs';
export const defaultSettings = {
activeFooter: 'intro',
keybindings: 'codemirror',
isLineNumbersDisplayed: true,
isActiveLineHighlighted: true,
isAutoCompletionEnabled: false,
isTooltipEnabled: false,
isFlashEnabled: true,
isLineWrappingEnabled: false,
isPatternHighlightingEnabled: true,
theme: 'strudelTheme',
fontFamily: 'monospace',
fontSize: 18,
latestCode: '',
isZen: false,
soundsFilter: 'all',
panelPosition: 'right',
userPatterns: '{}',
audioDeviceName: defaultAudioDeviceName,
};
export const settingsMap = persistentMap('strudel-settings', defaultSettings);
const defaultCode = '';
//pattern that the use is currently viewing in the window
const $viewingPattern = persistentAtom('viewingPattern', '', { listen: false });
export function setViewingPattern(key) {
$viewingPattern.set(key);
}
export function getViewingPattern() {
return $viewingPattern.get();
}
export function useViewingPattern() {
return useStore($viewingPattern);
}
// 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);
const id = match?.[0] || '';
setActivePattern(id);
setViewingPattern(id);
}
export function useSettings() {
const state = useStore(settingsMap);
return {
...state,
isZen: [true, 'true'].includes(state.isZen) ? true : false,
isLineNumbersDisplayed: [true, 'true'].includes(state.isLineNumbersDisplayed) ? true : false,
isActiveLineHighlighted: [true, 'true'].includes(state.isActiveLineHighlighted) ? true : false,
isAutoCompletionEnabled: [true, 'true'].includes(state.isAutoCompletionEnabled) ? true : false,
isPatternHighlightingEnabled: [true, 'true'].includes(state.isPatternHighlightingEnabled) ? true : false,
isTooltipEnabled: [true, 'true'].includes(state.isTooltipEnabled) ? true : false,
isLineWrappingEnabled: [true, 'true'].includes(state.isLineWrappingEnabled) ? true : false,
isFlashEnabled: [true, 'true'].includes(state.isFlashEnabled) ? true : false,
fontSize: Number(state.fontSize),
panelPosition: state.activeFooter !== '' ? state.panelPosition : 'bottom', // <-- keep this 'bottom' where it is!
userPatterns: JSON.parse(state.userPatterns),
};
}
export const setActiveFooter = (tab) => settingsMap.setKey('activeFooter', tab);
export const setLatestCode = (code) => settingsMap.setKey('latestCode', code);
export const setIsZen = (active) => settingsMap.setKey('isZen', !!active);
const patternSetting = (key) =>
register(key, (value, pat) =>
pat.onTrigger(() => {
value = Array.isArray(value) ? value.join(' ') : value;
if (value !== settingsMap.get()[key]) {
settingsMap.setKey(key, value);
}
return pat;
}, false),
);
export const theme = patternSetting('theme');
export const fontFamily = patternSetting('fontFamily');
export const fontSize = patternSetting('fontSize');
export const settingPatterns = { theme, fontFamily, fontSize };
export function getUserPatterns() {
return JSON.parse(settingsMap.get().userPatterns);
}
function getSetting(key) {
return settingsMap.get()[key];
}
export function setUserPatterns(obj) {
return settingsMap.setKey('userPatterns', JSON.stringify(obj));
}
export const createPatternID = () => {
const userPatterns = getUserPatterns();
const date = new Date().toISOString().split('T')[0];
const todays = Object.entries(userPatterns).filter(([name]) => name.startsWith(date));
const num = String(todays.length + 1).padStart(3, '0');
const id = date + '_' + num;
return id;
};
export const getNextCloneID = (id) => {
const userPatterns = this.getAll();
const clones = Object.entries(userPatterns).filter(([patID]) => patID.startsWith(id));
const num = String(clones.length + 1).padStart(3, '0');
const newID = id + '_' + num;
return newID;
};
export const examplePattern = {
getAll() {
const examplePatterns = {};
Object.entries(tunes).forEach(([key, code]) => (examplePatterns[key] = { code }));
return examplePatterns;
},
getPatternData(id) {
const userPatterns = this.getAll();
return userPatterns[id];
},
};
export const userPattern = {
getAll() {
return JSON.parse(settingsMap.get().userPatterns);
},
getPatternData(id) {
const userPatterns = this.getAll();
return userPatterns[id];
},
exists(id) {
const userPatterns = this.getAll();
return userPatterns[id] != null;
},
create() {
const newID = createPatternID();
const code = defaultCode;
const data = { code };
this.update(newID, data);
return { id: newID, data };
},
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] };
},
rename(id) {
const userPatterns = this.getAll();
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];
setUserPatterns({ ...userPatterns });
if (id === getActivePattern()) {
setActivePattern(newID);
}
return { id: newID, data };
},
};
export async function importPatterns(fileList) {
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 id = file.name.replace(/\.[^/.]+$/, '');
userPattern.update(id, { code: content });
}
}),
);
logger(`import done!`);
}
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);
}