import useEvent from '@src/useEvent.mjs'; import { useStore } from '@nanostores/react'; import { getAudioContext, soundMap, connectToDestination } from '@strudel/webaudio'; import { useMemo, useRef, useState } from 'react'; import { settingsMap, useSettings } from '../../../settings.mjs'; import { ButtonGroup } from './Forms.jsx'; import ImportSoundsButton from './ImportSoundsButton.jsx'; import { Textbox } from '../textbox/Textbox.jsx'; const getSamples = (samples) => Array.isArray(samples) ? samples.length : typeof samples === 'object' ? Object.values(samples).length : 1; export function SoundsTab() { const sounds = useStore(soundMap); const { soundsFilter } = useSettings(); const [search, setSearch] = useState(''); const { BASE_URL } = import.meta.env; const baseNoTrailing = BASE_URL.endsWith('/') ? BASE_URL.slice(0, -1) : BASE_URL; const soundEntries = useMemo(() => { if (!sounds) { return []; } let filtered = Object.entries(sounds) .filter(([key]) => !key.startsWith('_')) .sort((a, b) => a[0].localeCompare(b[0])) .filter(([name]) => name.toLowerCase().includes(search.toLowerCase())); if (soundsFilter === 'user') { return filtered.filter(([_, { data }]) => !data.prebake); } if (soundsFilter === 'drums') { return filtered.filter(([_, { data }]) => data.type === 'sample' && data.tag === 'drum-machines'); } if (soundsFilter === 'samples') { return filtered.filter(([_, { data }]) => data.type === 'sample' && data.tag !== 'drum-machines'); } if (soundsFilter === 'synths') { return filtered.filter(([_, { data }]) => ['synth', 'soundfont'].includes(data.type)); } if (soundsFilter === 'importSounds') { return []; } return filtered; }, [sounds, soundsFilter, search]); // holds mutable ref to current triggered sound const trigRef = useRef(); // stop current sound on mouseup useEvent('mouseup', () => { const t = trigRef.current; trigRef.current = undefined; t?.then((ref) => { ref?.stop(getAudioContext().currentTime + 0.01); }); }); return (
To import sounds into strudel, they must be contained{' '} within a folder or subfolder . The best way to do this is to upload a “samples” folder containing subfolders of individual sounds or soundbanks (see diagram below).{' '}
{`└─ samples <-- import this folder
├─ swoop
│ ├─ swoopshort.wav
│ ├─ swooplong.wav
│ └─ swooptight.wav
└─ smash
├─ smashhigh.wav
├─ smashlow.wav
└─ smashmiddle.wav`}
The name of a subfolder corresponds to the sound name under the “user” tab. Multiple samples within a subfolder are all labelled with the same name, but can be accessed using “.n( )” - remember sounds are zero-indexed and in alphabetical order!
For more information, and other ways to use your own sounds in strudel,{' '} check out the docs !
n("0 1 2 3 4 5").s("sample-name")
Paste the line above into the main editor to hear the uploaded folder. Remember to use the name of your sample as it appears under the "user" tab.