wam-community
Version:
A collection of prebuilt Web Audio Modules ready for use
921 lines (809 loc) • 42.8 kB
JavaScript
/*
Code generated with Faust version 2.40.7
Compilation options: -lang wasm-ib -cn stonephaser -es 1 -mcd 16 -single -ftz 2
*/
function getJSONstonephaser() {
return '{"name": "stonephaser","filename": "stonephaser.dsp","version": "2.40.7","compile_options": "-lang wasm-ib -cn stonephaser -es 1 -mcd 16 -single -ftz 2","include_pathnames": ["/usr/local/share/faust","/usr/local/share/faust","/usr/share/faust",".","/tmp/sessions/C5D0714DA2175AAC910865B152AF68896F1CD9E8/web/wap"],"size": 720,"inputs": 1,"outputs": 1,"meta": [ { "author": "Jean Pierre Cimalando" },{ "basics_lib_name": "Faust Basic Element Library" },{ "basics_lib_version": "0.6" },{ "compilation_options": "-single -scal -I libraries/ -I project/ -lang wasm" },{ "compile_options": "-lang wasm-ib -cn stonephaser -es 1 -mcd 16 -single -ftz 2" },{ "filename": "stonephaser.dsp" },{ "filters_lib_fir_author": "Julius O. Smith III" },{ "filters_lib_fir_copyright": "Copyright (C) 2003-2019 by Julius O. Smith III <jos@ccrma.stanford.edu>" },{ "filters_lib_fir_license": "MIT-style STK-4.3 license" },{ "filters_lib_iir_author": "Julius O. Smith III" },{ "filters_lib_iir_copyright": "Copyright (C) 2003-2019 by Julius O. Smith III <jos@ccrma.stanford.edu>" },{ "filters_lib_iir_license": "MIT-style STK-4.3 license" },{ "filters_lib_lowpass0_highpass1": "MIT-style STK-4.3 license" },{ "filters_lib_name": "Faust Filters Library" },{ "filters_lib_version": "0.3" },{ "library_path0": "/libraries/stdfaust.lib" },{ "library_path1": "/libraries/oscillators.lib" },{ "library_path2": "/libraries/signals.lib" },{ "library_path3": "/libraries/basics.lib" },{ "library_path4": "/libraries/maths.lib" },{ "library_path5": "/libraries/platform.lib" },{ "library_path6": "/libraries/filters.lib" },{ "license": "CC0-1.0 or BSL-1.0" },{ "maths_lib_author": "GRAME" },{ "maths_lib_copyright": "GRAME" },{ "maths_lib_license": "LGPL with exception" },{ "maths_lib_name": "Faust Math Library" },{ "maths_lib_version": "2.5" },{ "name": "stonephaser" },{ "oscillators_lib_name": "Faust Oscillator Library" },{ "oscillators_lib_version": "0.3" },{ "platform_lib_name": "Generic Platform Library" },{ "platform_lib_version": "0.2" },{ "signals_lib_name": "Faust Signal Routing Library" },{ "signals_lib_version": "0.1" },{ "version": "1.2.2" }],"ui": [ {"type": "vgroup","label": "stonephaser","items": [ {"type": "checkbox","label": "Bypass","address": "/stonephaser/Bypass","index": 524,"meta": [{ "0": "" },{ "symbol": "bypass" }]},{"type": "hslider","label": "Color","address": "/stonephaser/Color","index": 572,"meta": [{ "1": "" },{ "boolean": "" },{ "symbol": "color" }],"init": 1,"min": 0,"max": 1,"step": 1},{"type": "hslider","label": "LFO frequency","address": "/stonephaser/LFO frequency","index": 644,"meta": [{ "2": "" },{ "scale": "log" },{ "symbol": "lfo_frequency" },{ "unit": "Hz" }],"init": 0.2,"min": 0.01,"max": 5,"step": 0.01},{"type": "hslider","label": "Feedback depth","address": "/stonephaser/Feedback depth","index": 580,"meta": [{ "3": "" },{ "integer": "" },{ "symbol": "feedback_depth" },{ "unit": "%" }],"init": 75,"min": 0,"max": 99,"step": 1},{"type": "hslider","label": "Feedback bass cut","address": "/stonephaser/Feedback bass cut","index": 604,"meta": [{ "4": "" },{ "abbrev": "Fb bass cut" },{ "scale": "log" },{ "symbol": "feedback_hpf_cutoff" },{ "unit": "Hz" }],"init": 500,"min": 10,"max": 5000,"step": 1},{"type": "hslider","label": "Dry/wet mix","address": "/stonephaser/Dry/wet mix","index": 540,"meta": [{ "5": "" },{ "integer": "" },{ "symbol": "mix" },{ "unit": "%" }],"init": 50,"min": 0,"max": 100,"step": 1},{"type": "hbargraph","label": "Bypass meter","address": "/stonephaser/Bypass meter","index": 536,"meta": [{ "7": "" },{ "symbol": "bypass_meter" }],"min": 0,"max": 1}]}]}';
}
function getBase64Codestonephaser() {
return "AGFzbQEAAAAB4ICAgAASYAJ/fwBgBH9/f38AYAF9AX1gAX0BfWABfwF/YAF/AX9gAn9/AX1gAX8Bf2ACf38AYAF/AGACf38AYAJ/fwBgAX8AYAJ/fwF/YAJ/fwF/YAJ9fQF9YAN/f30AYAF9AX0CsYCAgAAEA2VudgVfY29zZgACA2VudgVfZXhwZgADA2VudgVfcG93ZgAPA2VudgVfc2luZgARA4+AgIAADgABBAUGBwgJCgsMDQ4QBYyAgIAAAQGCgICAAOqHgIAAB7qBgIAADAdjb21wdXRlAAUMZ2V0TnVtSW5wdXRzAAYNZ2V0TnVtT3V0cHV0cwAHDWdldFBhcmFtVmFsdWUACA1nZXRTYW1wbGVSYXRlAAkEaW5pdAAKDWluc3RhbmNlQ2xlYXIACxFpbnN0YW5jZUNvbnN0YW50cwAMDGluc3RhbmNlSW5pdAANGmluc3RhbmNlUmVzZXRVc2VySW50ZXJmYWNlAA4Nc2V0UGFyYW1WYWx1ZQARBm1lbW9yeQIACqCagIAADvaBgIAAAgJ/BH1BACEDQwAAADxBACgCyAVBf2qylCEEIASosiEFIAQgBZMhBiAFQwAAgD8gBJOSIQdBACECQQAhAgNAAkBByAUgAkECdGpBADYCACACQQFqIQIgAkECSARADAIMAQsLC0EAIQMDQAJAQQBBACgCzAVBAWo2AsgFQwAAADxBACgCyAVBf2qylCEEIASosiEFIAQgBZMhBiAFQwAAgD8gBJOSIQcgA0ECdEMAAIA/QzVIgz9DW2EsQCAGQwAAAD9dBH0gBgUgBwuUEAOUkzgCAEEAQQAoAsgFNgLMBSADQQFqIQMgA0GAAUgEQAwCDAELCwsLlY6AgAACBX8ifUEAIQRBACEFQwAAAAAhCUMAAAAAIQpDAAAAACELQQAhBkMAAAAAIQxDAAAAACENQwAAAAAhDkMAAAAAIQ9DAAAAACEQQwAAAAAhEUEAIQdDAAAAACESQwAAAAAhE0MAAAAAIRRDAAAAACEVQwAAAAAhFkMAAAAAIRdDAAAAACEYQwAAAAAhGUMAAAAAIRpDAAAAACEbQwAAAAAhHEMAAAAAIR1DAAAAACEeQwAAAAAhH0MAAAAAISBDAAAAACEhQwAAAAAhIkEAIQhDAAAAACEjQwAAAAAhJEMAAAAAISVDAAAAACEmQwAAAAAhJ0MAAAAAIShDAAAAACEpQwAAAAAhKiACQQBqKAIAIQQgA0EAaigCACEFQQAqAogEQQFBACoCjARDAAAAP15rspQhCUP9rYA8QQAqApwElCEKQQAqAogEIAoQAJQhC0EAKgK8BKghBkEAKgLABEEAKgLEBJQhDEEAKgKIBEEAKgLcBJQhDUEAKgKIBCAGBH1DgfIdQgVDYHp5QguUIQ5BACoCiAQgBgR9Q+25wUIFQx135EILlCEPQQAqAogEQQAqAoQFlCEQQQAqAogEIAoQA5QhEUEAIQcDQAJAIAlBACoChARBACoClASUkiESQQAgErxBgICA/AdxBH0gEgVDAAAAAAs4ApAEQQBBACoCkAQ4ApgEIAQgB2oqAgAhEyALQQAqAoQEQQAqAqQElJIhFEEAIBS8QYCAgPwHcQR9IBQFQwAAAAALOAKgBCATQQAqAqgEQQAqArgElJIhFUEAIBW8QYCAgPwHcQR9IBUFQwAAAAALOAK0BCAMQQAqAoQEQQAqAswElJIhFkEAIBa8QYCAgPwHcQR9IBYFQwAAAAALOALIBEPNzMw9QQAqAsgElCEXQQAqAogEIAYEfUEAKgLIBAUgFwuUQQAqAoQEQQAqAtQElJIhGEEAIBi8QYCAgPwHcQR9IBgFQwAAAAALOALQBCANQQAqAoQEQQAqAuQElJIhGUEAIBm8QYCAgPwHcQR9IBkFQwAAAAALOALgBEEAKgLYBEMAAAAAQ9sPyUBBACoC4ASUk5QQASEaQQAqArwFQQAqAuwEIBqUkiEbQQAgG7xBgICA/AdxBH0gGwVDAAAAAAs4AugEIBpDAACAP5IhHCAOQQAqAoQEQQAqAvgElJIhHUEAIB28QYCAgPwHcQR9IB0FQwAAAAALOAL0BCAPQQAqAoQEQQAqAoAFlJIhHkEAIB68QYCAgPwHcQR9IB4FQwAAAAALOAL8BCAQQQAqAoQEQQAqAowFlJIhH0EAIB+8QYCAgPwHcQR9IB8FQwAAAAALOAKIBUEAKgKUBUEAKgLYBEEAKgKIBZSSISAgICAgjpMhIUEAICG8QYCAgPwHcQR9ICEFQwAAAAALOAKQBUMAAABDQQAqApAFlCEiICKoIQggCEECdCoCACEjQQAqAvAEQwAAAEBDq6qqPUEAKgL0BEEAKgL8BEEAKgL0BJMgIyAiIAiykyAIQQFqQYABb0ECdCoCACAjk5SSlJJDAACKwpKUEAKUQwAAgL+SISRBACoCsARBACoCuASUQQAqAtAEQwAAAD9BACoC6AQgHJSUQQAqAuwEQwAAAABDAAAAPyAclJOUkpRBACoCrARBACoCtASUkpJBACoCnAUgJJSTISVBACAlvEGAgID8B3EEfSAlBUMAAAAACzgCmAVBACoCnAUgJEEAKgKYBUEAKgKkBZOUkiEmQQAgJrxBgICA/AdxBH0gJgVDAAAAAAs4AqAFQQAqAqQFICRBACoCoAVBACoCrAWTlJIhJ0EAICe8QYCAgPwHcQR9ICcFQwAAAAALOAKoBUEAKgKsBSAkQQAqAqgFQQAqArQFk5SSIShBACAovEGAgID8B3EEfSAoBUMAAAAACzgCsAVBACoCtAVBACoCsAUgJJSSISlBACApvEGAgID8B3EEfSApBUMAAAAACzgCuAUgEUEAKgKEBEEAKgLEBZSSISpBACAqvEGAgID8B3EEfSAqBUMAAAAACzgCwAUgBSAHaiATQQAqAqAElEEAKgK4BUEAKgLABZSSQQAqApAElCATQwAAgD9BACoCkASTlJI4AgBBAEEAKgKQBDgClARBAEEAKgKgBDgCpARBAEEAKgK0BDgCuARBAEEAKgLIBDgCzARBAEEAKgLQBDgC1ARBAEEAKgLgBDgC5ARBAEEAKgLoBDgC7ARBAEEAKgL0BDgC+ARBAEEAKgL8BDgCgAVBAEEAKgKIBTgCjAVBAEEAKgKQBTgClAVBAEEAKgKYBTgCnAVBAEEAKgKgBTgCpAVBAEEAKgKoBTgCrAVBAEEAKgKwBTgCtAVBAEEAKgK4BTgCvAVBAEEAKgLABTgCxAUgB0EEaiEHIAdBBCABbEgEQAwCDAELCwsLhYCAgAAAQQEPC4WAgIAAAEEBDwuLgICAAAAgACABaioCAA8LiYCAgAAAQQAoAoAEDwuOgICAAAAgACABEAQgACABEA0L1oaAgAABEX9BACEBQQAhAkEAIQNBACEEQQAhBUEAIQZBACEHQQAhCEEAIQlBACEKQQAhC0EAIQxBACENQQAhDkEAIQ9BACEQQQAhEUEAIQEDQAJAQZAEIAFBAnRqQwAAAAA4AgAgAUEBaiEBIAFBAkgEQAwCDAELCwtBACECA0ACQEGgBCACQQJ0akMAAAAAOAIAIAJBAWohAiACQQJIBEAMAgwBCwsLQQAhAwNAAkBBtAQgA0ECdGpDAAAAADgCACADQQFqIQMgA0ECSARADAIMAQsLC0EAIQQDQAJAQcgEIARBAnRqQwAAAAA4AgAgBEEBaiEEIARBAkgEQAwCDAELCwtBACEFA0ACQEHQBCAFQQJ0akMAAAAAOAIAIAVBAWohBSAFQQJIBEAMAgwBCwsLQQAhBgNAAkBB4AQgBkECdGpDAAAAADgCACAGQQFqIQYgBkECSARADAIMAQsLC0EAIQcDQAJAQegEIAdBAnRqQwAAAAA4AgAgB0EBaiEHIAdBAkgEQAwCDAELCwtBACEIA0ACQEH0BCAIQQJ0akMAAAAAOAIAIAhBAWohCCAIQQJIBEAMAgwBCwsLQQAhCQNAAkBB/AQgCUECdGpDAAAAADgCACAJQQFqIQkgCUECSARADAIMAQsLC0EAIQoDQAJAQYgFIApBAnRqQwAAAAA4AgAgCkEBaiEKIApBAkgEQAwCDAELCwtBACELA0ACQEGQBSALQQJ0akMAAAAAOAIAIAtBAWohCyALQQJIBEAMAgwBCwsLQQAhDANAAkBBmAUgDEECdGpDAAAAADgCACAMQQFqIQwgDEECSARADAIMAQsLC0EAIQ0DQAJAQaAFIA1BAnRqQwAAAAA4AgAgDUEBaiENIA1BAkgEQAwCDAELCwtBACEOA0ACQEGoBSAOQQJ0akMAAAAAOAIAIA5BAWohDiAOQQJIBEAMAgwBCwsLQQAhDwNAAkBBsAUgD0ECdGpDAAAAADgCACAPQQFqIQ8gD0ECSARADAIMAQsLC0EAIRADQAJAQbgFIBBBAnRqQwAAAAA4AgAgEEEBaiEQIBBBAkgEQAwCDAELCwtBACERA0ACQEHABSARQQJ0akMAAAAAOAIAIBFBAWohESARQQJIBEAMAgwBCwsLC8yBgIAAAQF9QwCAO0hDAACAP0EAKAKABLKXliECQQAgATYCgARDAIA7SEMAAIA/QQAoAoAEspeWIQJBAEMAAAAAQwAAIEEgApWTEAE4AoQEQQBDAACAP0EAKgKEBJM4AogEQQBDAAAAAENZWE9DIAKVkxABOAKoBEEAQwAAAD9BACoCqARDAACAP5KUOAKsBEEAQwAAAABBACoCrASTOAKwBEEAQwrXIzxBACoCiASUOALABEEAQwAAgD8gApU4AtgEQQBDoMksRSAClTgC8AQLkICAgAAAIAAgARAMIAAQDiAAEAsLxICAgAAAQQBDAAAAADgCjARBAEMAAEhCOAKcBEEAQwAAgD84ArwEQQBDAACWQjgCxARBAEMAAPpDOALcBEEAQ83MTD44AoQFC5CAgIAAACAAIAFIBH8gAQUgAAsPC5CAgIAAACAAIAFIBH8gAAUgAQsPC4yAgIAAACAAIAFqIAI4AgALC/ubgIAAAQBBAAv0G3sibmFtZSI6ICJ1bnRpdGxlZCIsImZpbGVuYW1lIjogInVudGl0bGVkLmRzcCIsInZlcnNpb24iOiAiMi40MC43IiwiY29tcGlsZV9vcHRpb25zIjogIi1sYW5nIHdhc20taWIgLWNuIHVudGl0bGVkIC1lcyAxIC1tY2QgMTYgLXNpbmdsZSAtZnR6IDIiLCJpbmNsdWRlX3BhdGhuYW1lcyI6IFsiL3Vzci9sb2NhbC9zaGFyZS9mYXVzdCIsIi91c3IvbG9jYWwvc2hhcmUvZmF1c3QiLCIvdXNyL3NoYXJlL2ZhdXN0IiwiLiIsIi90bXAvc2Vzc2lvbnMvQzVEMDcxNERBMjE3NUFBQzkxMDg2NUIxNTJBRjY4ODk2RjFDRDlFOC93ZWIvd2FwIl0sInNpemUiOiA3MjAsImlucHV0cyI6IDEsIm91dHB1dHMiOiAxLCJtZXRhIjogWyB7ICJhdXRob3IiOiAiSmVhbiBQaWVycmUgQ2ltYWxhbmRvIiB9LHsgImJhc2ljc19saWJfbmFtZSI6ICJGYXVzdCBCYXNpYyBFbGVtZW50IExpYnJhcnkiIH0seyAiYmFzaWNzX2xpYl92ZXJzaW9uIjogIjAuNiIgfSx7ICJjb21waWxhdGlvbl9vcHRpb25zIjogIi1zaW5nbGUgLXNjYWwgLUkgbGlicmFyaWVzLyAtSSBwcm9qZWN0LyAtbGFuZyB3YXNtIiB9LHsgImNvbXBpbGVfb3B0aW9ucyI6ICItbGFuZyB3YXNtLWliIC1jbiB1bnRpdGxlZCAtZXMgMSAtbWNkIDE2IC1zaW5nbGUgLWZ0eiAyIiB9LHsgImZpbGVuYW1lIjogInVudGl0bGVkLmRzcCIgfSx7ICJmaWx0ZXJzX2xpYl9maXJfYXV0aG9yIjogIkp1bGl1cyBPLiBTbWl0aCBJSUkiIH0seyAiZmlsdGVyc19saWJfZmlyX2NvcHlyaWdodCI6ICJDb3B5cmlnaHQgKEMpIDIwMDMtMjAxOSBieSBKdWxpdXMgTy4gU21pdGggSUlJIDxqb3NAY2NybWEuc3RhbmZvcmQuZWR1PiIgfSx7ICJmaWx0ZXJzX2xpYl9maXJfbGljZW5zZSI6ICJNSVQtc3R5bGUgU1RLLTQuMyBsaWNlbnNlIiB9LHsgImZpbHRlcnNfbGliX2lpcl9hdXRob3IiOiAiSnVsaXVzIE8uIFNtaXRoIElJSSIgfSx7ICJmaWx0ZXJzX2xpYl9paXJfY29weXJpZ2h0IjogIkNvcHlyaWdodCAoQykgMjAwMy0yMDE5IGJ5IEp1bGl1cyBPLiBTbWl0aCBJSUkgPGpvc0BjY3JtYS5zdGFuZm9yZC5lZHU+IiB9LHsgImZpbHRlcnNfbGliX2lpcl9saWNlbnNlIjogIk1JVC1zdHlsZSBTVEstNC4zIGxpY2Vuc2UiIH0seyAiZmlsdGVyc19saWJfbG93cGFzczBfaGlnaHBhc3MxIjogIk1JVC1zdHlsZSBTVEstNC4zIGxpY2Vuc2UiIH0seyAiZmlsdGVyc19saWJfbmFtZSI6ICJGYXVzdCBGaWx0ZXJzIExpYnJhcnkiIH0seyAiZmlsdGVyc19saWJfdmVyc2lvbiI6ICIwLjMiIH0seyAibGlicmFyeV9wYXRoMCI6ICIvbGlicmFyaWVzL3N0ZGZhdXN0LmxpYiIgfSx7ICJsaWJyYXJ5X3BhdGgxIjogIi9saWJyYXJpZXMvb3NjaWxsYXRvcnMubGliIiB9LHsgImxpYnJhcnlfcGF0aDIiOiAiL2xpYnJhcmllcy9zaWduYWxzLmxpYiIgfSx7ICJsaWJyYXJ5X3BhdGgzIjogIi9saWJyYXJpZXMvYmFzaWNzLmxpYiIgfSx7ICJsaWJyYXJ5X3BhdGg0IjogIi9saWJyYXJpZXMvbWF0aHMubGliIiB9LHsgImxpYnJhcnlfcGF0aDUiOiAiL2xpYnJhcmllcy9wbGF0Zm9ybS5saWIiIH0seyAibGlicmFyeV9wYXRoNiI6ICIvbGlicmFyaWVzL2ZpbHRlcnMubGliIiB9LHsgImxpY2Vuc2UiOiAiQ0MwLTEuMCBvciBCU0wtMS4wIiB9LHsgIm1hdGhzX2xpYl9hdXRob3IiOiAiR1JBTUUiIH0seyAibWF0aHNfbGliX2NvcHlyaWdodCI6ICJHUkFNRSIgfSx7ICJtYXRoc19saWJfbGljZW5zZSI6ICJMR1BMIHdpdGggZXhjZXB0aW9uIiB9LHsgIm1hdGhzX2xpYl9uYW1lIjogIkZhdXN0IE1hdGggTGlicmFyeSIgfSx7ICJtYXRoc19saWJfdmVyc2lvbiI6ICIyLjUiIH0seyAibmFtZSI6ICJ1bnRpdGxlZCIgfSx7ICJvc2NpbGxhdG9yc19saWJfbmFtZSI6ICJGYXVzdCBPc2NpbGxhdG9yIExpYnJhcnkiIH0seyAib3NjaWxsYXRvcnNfbGliX3ZlcnNpb24iOiAiMC4zIiB9LHsgInBsYXRmb3JtX2xpYl9uYW1lIjogIkdlbmVyaWMgUGxhdGZvcm0gTGlicmFyeSIgfSx7ICJwbGF0Zm9ybV9saWJfdmVyc2lvbiI6ICIwLjIiIH0seyAic2lnbmFsc19saWJfbmFtZSI6ICJGYXVzdCBTaWduYWwgUm91dGluZyBMaWJyYXJ5IiB9LHsgInNpZ25hbHNfbGliX3ZlcnNpb24iOiAiMC4xIiB9LHsgInZlcnNpb24iOiAiMS4yLjIiIH1dLCJ1aSI6IFsgeyJ0eXBlIjogInZncm91cCIsImxhYmVsIjogInVudGl0bGVkIiwiaXRlbXMiOiBbIHsidHlwZSI6ICJjaGVja2JveCIsImxhYmVsIjogIkJ5cGFzcyIsImFkZHJlc3MiOiAiL3VudGl0bGVkL0J5cGFzcyIsImluZGV4IjogNTI0LCJtZXRhIjogW3sgIjAiOiAiIiB9LHsgInN5bWJvbCI6ICJieXBhc3MiIH1dfSx7InR5cGUiOiAiaHNsaWRlciIsImxhYmVsIjogIkNvbG9yIiwiYWRkcmVzcyI6ICIvdW50aXRsZWQvQ29sb3IiLCJpbmRleCI6IDU3MiwibWV0YSI6IFt7ICIxIjogIiIgfSx7ICJib29sZWFuIjogIiIgfSx7ICJzeW1ib2wiOiAiY29sb3IiIH1dLCJpbml0IjogMSwibWluIjogMCwibWF4IjogMSwic3RlcCI6IDF9LHsidHlwZSI6ICJoc2xpZGVyIiwibGFiZWwiOiAiTEZPIGZyZXF1ZW5jeSIsImFkZHJlc3MiOiAiL3VudGl0bGVkL0xGTyBmcmVxdWVuY3kiLCJpbmRleCI6IDY0NCwibWV0YSI6IFt7ICIyIjogIiIgfSx7ICJzY2FsZSI6ICJsb2ciIH0seyAic3ltYm9sIjogImxmb19mcmVxdWVuY3kiIH0seyAidW5pdCI6ICJIeiIgfV0sImluaXQiOiAwLjIsIm1pbiI6IDAuMDEsIm1heCI6IDUsInN0ZXAiOiAwLjAxfSx7InR5cGUiOiAiaHNsaWRlciIsImxhYmVsIjogIkZlZWRiYWNrIGRlcHRoIiwiYWRkcmVzcyI6ICIvdW50aXRsZWQvRmVlZGJhY2sgZGVwdGgiLCJpbmRleCI6IDU4MCwibWV0YSI6IFt7ICIzIjogIiIgfSx7ICJpbnRlZ2VyIjogIiIgfSx7ICJzeW1ib2wiOiAiZmVlZGJhY2tfZGVwdGgiIH0seyAidW5pdCI6ICIlIiB9XSwiaW5pdCI6IDc1LCJtaW4iOiAwLCJtYXgiOiA5OSwic3RlcCI6IDF9LHsidHlwZSI6ICJoc2xpZGVyIiwibGFiZWwiOiAiRmVlZGJhY2sgYmFzcyBjdXQiLCJhZGRyZXNzIjogIi91bnRpdGxlZC9GZWVkYmFjayBiYXNzIGN1dCIsImluZGV4IjogNjA0LCJtZXRhIjogW3sgIjQiOiAiIiB9LHsgImFiYnJldiI6ICJGYiBiYXNzIGN1dCIgfSx7ICJzY2FsZSI6ICJsb2ciIH0seyAic3ltYm9sIjogImZlZWRiYWNrX2hwZl9jdXRvZmYiIH0seyAidW5pdCI6ICJIeiIgfV0sImluaXQiOiA1MDAsIm1pbiI6IDEwLCJtYXgiOiA1MDAwLCJzdGVwIjogMX0seyJ0eXBlIjogImhzbGlkZXIiLCJsYWJlbCI6ICJEcnkvd2V0IG1peCIsImFkZHJlc3MiOiAiL3VudGl0bGVkL0RyeS93ZXQgbWl4IiwiaW5kZXgiOiA1NDAsIm1ldGEiOiBbeyAiNSI6ICIiIH0seyAiaW50ZWdlciI6ICIiIH0seyAic3ltYm9sIjogIm1peCIgfSx7ICJ1bml0IjogIiUiIH1dLCJpbml0IjogNTAsIm1pbiI6IDAsIm1heCI6IDEwMCwic3RlcCI6IDF9LHsidHlwZSI6ICJoYmFyZ3JhcGgiLCJsYWJlbCI6ICJCeXBhc3MgbWV0ZXIiLCJhZGRyZXNzIjogIi91bnRpdGxlZC9CeXBhc3MgbWV0ZXIiLCJpbmRleCI6IDUzNiwibWV0YSI6IFt7ICI3IjogIiIgfSx7ICJzeW1ib2wiOiAiYnlwYXNzX21ldGVyIiB9XSwibWluIjogMCwibWF4IjogMX1dfV19";
}
/*
faust2wasm: GRAME 2017-2019
*/
("use strict");
if (typeof AudioWorkletNode === "undefined") {
alert("AudioWorklet is not supported in this browser !");
}
class stonephaserNode extends AudioWorkletNode {
constructor(context, baseURL, options) {
super(context, "stonephaser", options);
this.baseURL = baseURL;
this.json = options.processorOptions.json;
this.json_object = JSON.parse(this.json);
// JSON parsing functions
this.parse_ui = function (ui, obj) {
for (var i = 0; i < ui.length; i++) {
this.parse_group(ui[i], obj);
}
};
this.parse_group = function (group, obj) {
if (group.items) {
this.parse_items(group.items, obj);
}
};
this.parse_items = function (items, obj) {
for (var i = 0; i < items.length; i++) {
this.parse_item(items[i], obj);
}
};
this.parse_item = function (item, obj) {
if (item.type === "vgroup" || item.type === "hgroup" || item.type === "tgroup") {
this.parse_items(item.items, obj);
} else if (item.type === "hbargraph" || item.type === "vbargraph") {
// Keep bargraph adresses
obj.outputs_items.push(item.address);
} else if (
item.type === "vslider" ||
item.type === "hslider" ||
item.type === "button" ||
item.type === "checkbox" ||
item.type === "nentry"
) {
// Keep inputs adresses
obj.inputs_items.push(item.address);
obj.descriptor.push(item);
// Decode MIDI
if (item.meta !== undefined) {
for (var i = 0; i < item.meta.length; i++) {
if (item.meta[i].midi !== undefined) {
if (item.meta[i].midi.trim() === "pitchwheel") {
obj.fPitchwheelLabel.push({
path: item.address,
min: parseFloat(item.min),
max: parseFloat(item.max),
});
} else if (item.meta[i].midi.trim().split(" ")[0] === "ctrl") {
obj.fCtrlLabel[parseInt(item.meta[i].midi.trim().split(" ")[1])].push({
path: item.address,
min: parseFloat(item.min),
max: parseFloat(item.max),
});
}
}
}
}
// Define setXXX/getXXX, replacing '/c' with 'C' everywhere in the string
var set_name = "set" + item.address;
var get_name = "get" + item.address;
set_name = set_name.replace(/\/./g, (x) => {
return x.substr(1, 1).toUpperCase();
});
get_name = get_name.replace(/\/./g, (x) => {
return x.substr(1, 1).toUpperCase();
});
obj[set_name] = (val) => {
obj.setParamValue(item.address, val);
};
obj[get_name] = () => {
return obj.getParamValue(item.address);
};
//console.log(set_name);
//console.log(get_name);
}
};
this.output_handler = null;
// input/output items
this.inputs_items = [];
this.outputs_items = [];
this.descriptor = [];
// MIDI
this.fPitchwheelLabel = [];
this.fCtrlLabel = new Array(128);
for (var i = 0; i < this.fCtrlLabel.length; i++) {
this.fCtrlLabel[i] = [];
}
// Parse UI
this.parse_ui(this.json_object.ui, this);
// Set message handler
this.port.onmessage = this.handleMessage.bind(this);
try {
if (this.parameters) this.parameters.forEach((p) => (p.automationRate = "k-rate"));
} catch (e) {}
}
// To be called by the message port with messages coming from the processor
handleMessage(event) {
var msg = event.data;
if (this.output_handler) {
this.output_handler(msg.path, msg.value);
}
}
// Public API
/**
* Destroy the node, deallocate resources.
*/
destroy() {
this.port.postMessage({ type: "destroy" });
this.port.close();
}
/**
* Returns a full JSON description of the DSP.
*/
getJSON() {
return this.json;
}
// For WAP
async getMetadata() {
return new Promise((resolve) => {
let real_url = this.baseURL === "" ? "main.json" : this.baseURL + "/main.json";
fetch(real_url)
.then((responseJSON) => {
return responseJSON.json();
})
.then((json) => {
resolve(json);
});
});
}
/**
* Set the control value at a given path.
*
* @param path - a path to the control
* @param val - the value to be set
*/
setParamValue(path, val) {
// Needed for sample accurate control
this.parameters.get(path).setValueAtTime(val, 0);
}
// For WAP
setParam(path, val) {
// Needed for sample accurate control
this.parameters.get(path).setValueAtTime(val, 0);
}
/**
* Get the control value at a given path.
*
* @return the current control value
*/
getParamValue(path) {
return this.parameters.get(path).value;
}
// For WAP
getParam(path) {
return this.parameters.get(path).value;
}
/**
* Setup a control output handler with a function of type (path, value)
* to be used on each generated output value. This handler will be called
* each audio cycle at the end of the 'compute' method.
*
* @param handler - a function of type function(path, value)
*/
setOutputParamHandler(handler) {
this.output_handler = handler;
}
/**
* Get the current output handler.
*/
getOutputParamHandler() {
return this.output_handler;
}
getNumInputs() {
return parseInt(this.json_object.inputs);
}
getNumOutputs() {
return parseInt(this.json_object.outputs);
}
// For WAP
inputChannelCount() {
return parseInt(this.json_object.inputs);
}
outputChannelCount() {
return parseInt(this.json_object.outputs);
}
/**
* Returns an array of all input paths (to be used with setParamValue/getParamValue)
*/
getParams() {
return this.inputs_items;
}
// For WAP
getDescriptor() {
var desc = {};
for (const item in this.descriptor) {
if (this.descriptor.hasOwnProperty(item)) {
if (this.descriptor[item].label != "bypass") {
desc = Object.assign(
{
[this.descriptor[item].label]: {
minValue: this.descriptor[item].min,
maxValue: this.descriptor[item].max,
defaultValue: this.descriptor[item].init,
},
},
desc
);
}
}
}
return desc;
}
/**
* Control change
*
* @param channel - the MIDI channel (0..15, not used for now)
* @param ctrl - the MIDI controller number (0..127)
* @param value - the MIDI controller value (0..127)
*/
ctrlChange(channel, ctrl, value) {
if (this.fCtrlLabel[ctrl] !== []) {
for (var i = 0; i < this.fCtrlLabel[ctrl].length; i++) {
var path = this.fCtrlLabel[ctrl][i].path;
this.setParamValue(
path,
stonephaserNode.remap(value, 0, 127, this.fCtrlLabel[ctrl][i].min, this.fCtrlLabel[ctrl][i].max)
);
if (this.output_handler) {
this.output_handler(path, this.getParamValue(path));
}
}
}
}
/**
* PitchWeel
*
* @param channel - the MIDI channel (0..15, not used for now)
* @param value - the MIDI controller value (0..16383)
*/
pitchWheel(channel, wheel) {
for (var i = 0; i < this.fPitchwheelLabel.length; i++) {
var pw = this.fPitchwheelLabel[i];
this.setParamValue(pw.path, stonephaserNode.remap(wheel, 0, 16383, pw.min, pw.max));
if (this.output_handler) {
this.output_handler(pw.path, this.getParamValue(pw.path));
}
}
}
/**
* Generic MIDI message handler.
*/
midiMessage(data) {
var cmd = data[0] >> 4;
var channel = data[0] & 0xf;
var data1 = data[1];
var data2 = data[2];
if (channel === 9) {
return;
} else if (cmd === 11) {
this.ctrlChange(channel, data1, data2);
} else if (cmd === 14) {
this.pitchWheel(channel, data2 * 128.0 + data1);
}
}
// For WAP
onMidi(data) {
midiMessage(data);
}
/**
* @returns {Object} describes the path for each available param and its current value
*/
async getState() {
var params = new Object();
for (let i = 0; i < this.getParams().length; i++) {
Object.assign(params, { [this.getParams()[i]]: `${this.getParam(this.getParams()[i])}` });
}
return new Promise((resolve) => {
resolve(params);
});
}
/**
* Sets each params with the value indicated in the state object
* @param {Object} state
*/
async setState(state) {
return new Promise((resolve) => {
for (const param in state) {
if (state.hasOwnProperty(param)) this.setParam(param, state[param]);
}
try {
this.gui.setAttribute("state", JSON.stringify(state));
} catch (error) {
console.warn("Plugin without gui or GUI not defined", error);
}
resolve(state);
});
}
/**
* A different call closer to the preset management
* @param {Object} patch to assign as a preset to the node
*/
setPatch(patch) {
this.setState(this.presets[patch]);
}
static remap(v, mn0, mx0, mn1, mx1) {
return ((1.0 * (v - mn0)) / (mx0 - mn0)) * (mx1 - mn1) + mn1;
}
}
// Factory class
export default class stonephaser {
static fWorkletProcessors;
/**
* Factory constructor.
*
* @param context - the audio context
* @param baseURL - the baseURL of the plugin folder
*/
constructor(context, baseURL = "") {
console.log("baseLatency " + context.baseLatency);
console.log("outputLatency " + context.outputLatency);
console.log("sampleRate " + context.sampleRate);
this.context = context;
this.baseURL = baseURL;
this.pathTable = [];
this.fWorkletProcessors = this.fWorkletProcessors || [];
}
heap2Str(buf) {
let str = "";
let i = 0;
while (buf[i] !== 0) {
str += String.fromCharCode(buf[i++]);
}
return str;
}
/**
* Load additionnal resources to prepare the custom AudioWorkletNode. Returns a promise to be used with the created node.
*/
async load() {
try {
const importObject = {
env: {
memoryBase: 0,
tableBase: 0,
_abs: Math.abs,
// Float version
_acosf: Math.acos,
_asinf: Math.asin,
_atanf: Math.atan,
_atan2f: Math.atan2,
_ceilf: Math.ceil,
_cosf: Math.cos,
_expf: Math.exp,
_floorf: Math.floor,
_fmodf: (x, y) => x % y,
_logf: Math.log,
_log10f: Math.log10,
_max_f: Math.max,
_min_f: Math.min,
_remainderf: (x, y) => x - Math.round(x / y) * y,
_powf: Math.pow,
_roundf: Math.fround,
_sinf: Math.sin,
_sqrtf: Math.sqrt,
_tanf: Math.tan,
_acoshf: Math.acosh,
_asinhf: Math.asinh,
_atanhf: Math.atanh,
_coshf: Math.cosh,
_sinhf: Math.sinh,
_tanhf: Math.tanh,
_isnanf: Number.isNaN,
_isinff: function (x) {
return !isFinite(x);
},
_copysignf: function (x, y) {
return Math.sign(x) === Math.sign(y) ? x : -x;
},
// Double version
_acos: Math.acos,
_asin: Math.asin,
_atan: Math.atan,
_atan2: Math.atan2,
_ceil: Math.ceil,
_cos: Math.cos,
_exp: Math.exp,
_floor: Math.floor,
_fmod: (x, y) => x % y,
_log: Math.log,
_log10: Math.log10,
_max_: Math.max,
_min_: Math.min,
_remainder: (x, y) => x - Math.round(x / y) * y,
_pow: Math.pow,
_round: Math.fround,
_sin: Math.sin,
_sqrt: Math.sqrt,
_tan: Math.tan,
_acosh: Math.acosh,
_asinh: Math.asinh,
_atanh: Math.atanh,
_cosh: Math.cosh,
_sinh: Math.sinh,
_tanh: Math.tanh,
_isnan: Number.isNaN,
_isinf: function (x) {
return !isFinite(x);
},
_copysign: function (x, y) {
return Math.sign(x) === Math.sign(y) ? x : -x;
},
table: new WebAssembly.Table({ initial: 0, element: "anyfunc" }),
},
};
let real_url = this.baseURL === "" ? "stonephaser.wasm" : this.baseURL + "/stonephaser.wasm";
const dspFile = await fetch(real_url);
const dspBuffer = await dspFile.arrayBuffer();
const dspModule = await WebAssembly.compile(dspBuffer);
const dspInstance = await WebAssembly.instantiate(dspModule, importObject);
let HEAPU8 = new Uint8Array(dspInstance.exports.memory.buffer);
let json = this.heap2Str(HEAPU8);
let json_object = JSON.parse(json);
let options = { wasm_module: dspModule, json: json };
if (this.fWorkletProcessors.indexOf(name) === -1) {
try {
let re = /JSON_STR/g;
let stonephaserProcessorString1 = stonephaserProcessorString.replace(re, json);
let real_url = window.URL.createObjectURL(
new Blob([stonephaserProcessorString1], { type: "text/javascript" })
);
await this.context.audioWorklet.addModule(real_url);
// Keep the DSP name
console.log("Keep the DSP name");
this.fWorkletProcessors.push(name);
} catch (e) {
console.error(e);
console.error("Faust " + this.name + " cannot be loaded or compiled");
return null;
}
}
this.node = new stonephaserNode(this.context, this.baseURL, {
numberOfInputs: parseInt(json_object.inputs) > 0 ? 1 : 0,
numberOfOutputs: parseInt(json_object.outputs) > 0 ? 1 : 0,
channelCount: Math.max(1, parseInt(json_object.inputs)),
outputChannelCount: [parseInt(json_object.outputs)],
channelCountMode: "explicit",
channelInterpretation: "speakers",
processorOptions: options,
});
this.node.onprocessorerror = () => {
console.log("An error from stonephaser-processor was detected.");
};
return this.node;
} catch (e) {
console.error(e);
console.error("Faust " + this.name + " cannot be loaded or compiled");
return null;
}
}
async loadGui() {
return new Promise((resolve, reject) => {
try {
// DO THIS ONLY ONCE. If another instance has already been added, do not add the html file again
let real_url = this.baseURL === "" ? "main.html" : this.baseURL + "/main.html";
if (!this.linkExists(real_url)) {
// LINK DOES NOT EXIST, let's add it to the document
var link = document.createElement("link");
link.rel = "import";
link.href = real_url;
document.head.appendChild(link);
link.onload = (e) => {
// the file has been loaded, instanciate GUI
// and get back the HTML elem
// HERE WE COULD REMOVE THE HARD CODED NAME
var element = createstonephaserGUI(this.node);
resolve(element);
};
} else {
// LINK EXIST, WE AT LEAST CREATED ONE INSTANCE PREVIOUSLY
// so we can create another instance
var element = createstonephaserGUI(this.node);
resolve(element);
}
} catch (e) {
console.log(e);
reject(e);
}
});
}
linkExists(url) {
return document.querySelectorAll(`link[href="${url}"]`).length > 0;
}
}
// Template string for AudioWorkletProcessor
let stonephaserProcessorString = `
'use strict';
// Monophonic Faust DSP
class stonephaserProcessor extends AudioWorkletProcessor {
// JSON parsing functions
static parse_ui(ui, obj, callback)
{
for (var i = 0; i < ui.length; i++) {
stonephaserProcessor.parse_group(ui[i], obj, callback);
}
}
static parse_group(group, obj, callback)
{
if (group.items) {
stonephaserProcessor.parse_items(group.items, obj, callback);
}
}
static parse_items(items, obj, callback)
{
for (var i = 0; i < items.length; i++) {
callback(items[i], obj, callback);
}
}
static parse_item1(item, obj, callback)
{
if (item.type === "vgroup"
|| item.type === "hgroup"
|| item.type === "tgroup") {
stonephaserProcessor.parse_items(item.items, obj, callback);
} else if (item.type === "hbargraph"
|| item.type === "vbargraph") {
// Nothing
} else if (item.type === "vslider"
|| item.type === "hslider"
|| item.type === "button"
|| item.type === "checkbox"
|| item.type === "nentry") {
obj.push({ name: item.address,
defaultValue: item.init,
minValue: item.min,
maxValue: item.max });
}
}
static parse_item2(item, obj, callback)
{
if (item.type === "vgroup"
|| item.type === "hgroup"
|| item.type === "tgroup") {
stonephaserProcessor.parse_items(item.items, obj, callback);
} else if (item.type === "hbargraph"
|| item.type === "vbargraph") {
// Keep bargraph adresses
obj.outputs_items.push(item.address);
obj.pathTable[item.address] = parseInt(item.index);
} else if (item.type === "vslider"
|| item.type === "hslider"
|| item.type === "button"
|| item.type === "checkbox"
|| item.type === "nentry") {
// Keep inputs adresses
obj.inputs_items.push(item.address);
obj.pathTable[item.address] = parseInt(item.index);
}
}
static get parameterDescriptors()
{
// Analyse JSON to generate AudioParam parameters
var params = [];
stonephaserProcessor.parse_ui(JSON.parse(\`JSON_STR\`).ui, params, stonephaserProcessor.parse_item1);
return params;
}
constructor(options)
{
super(options);
this.running = true;
const importObject = {
env: {
memoryBase: 0,
tableBase: 0,
// Integer version
_abs: Math.abs,
// Float version
_acosf: Math.acos,
_asinf: Math.asin,
_atanf: Math.atan,
_atan2f: Math.atan2,
_ceilf: Math.ceil,
_cosf: Math.cos,
_expf: Math.exp,
_floorf: Math.floor,
_fmodf: function(x, y) { return x % y; },
_logf: Math.log,
_log10f: Math.log10,
_max_f: Math.max,
_min_f: Math.min,
_remainderf: function(x, y) { return x - Math.round(x/y) * y; },
_powf: Math.pow,
_roundf: Math.fround,
_sinf: Math.sin,
_sqrtf: Math.sqrt,
_tanf: Math.tan,
_acoshf: Math.acosh,
_asinhf: Math.asinh,
_atanhf: Math.atanh,
_coshf: Math.cosh,
_sinhf: Math.sinh,
_tanhf: Math.tanh,
// Double version
_acos: Math.acos,
_asin: Math.asin,
_atan: Math.atan,
_atan2: Math.atan2,
_ceil: Math.ceil,
_cos: Math.cos,
_exp: Math.exp,
_floor: Math.floor,
_fmod: function(x, y) { return x % y; },
_log: Math.log,
_log10: Math.log10,
_max_: Math.max,
_min_: Math.min,
_remainder:function(x, y) { return x - Math.round(x/y) * y; },
_pow: Math.pow,
_round: Math.fround,
_sin: Math.sin,
_sqrt: Math.sqrt,
_tan: Math.tan,
_acosh: Math.acosh,
_asinh: Math.asinh,
_atanh: Math.atanh,
_cosh: Math.cosh,
_sinh: Math.sinh,
_tanh: Math.tanh,
table: new WebAssembly.Table({ initial: 0, element: 'anyfunc' })
}
};
this.stonephaser_instance = new WebAssembly.Instance(options.processorOptions.wasm_module, importObject);
this.json_object = JSON.parse(options.processorOptions.json);
this.output_handler = function(path, value) { this.port.postMessage({ path: path, value: value }); };
this.ins = null;
this.outs = null;
this.dspInChannnels = [];
this.dspOutChannnels = [];
this.numIn = parseInt(this.json_object.inputs);
this.numOut = parseInt(this.json_object.outputs);
// Memory allocator
this.ptr_size = 4;
this.sample_size = 4;
this.integer_size = 4;
this.factory = this.stonephaser_instance.exports;
this.HEAP = this.stonephaser_instance.exports.memory.buffer;
this.HEAP32 = new Int32Array(this.HEAP);
this.HEAPF32 = new Float32Array(this.HEAP);
// Warning: keeps a ref on HEAP in Chrome and prevent proper GC
//console.log(this.HEAP);
//console.log(this.HEAP32);
//console.log(this.HEAPF32);
// bargraph
this.outputs_timer = 5;
this.outputs_items = [];
// input items
this.inputs_items = [];
// Start of HEAP index
// DSP is placed first with index 0. Audio buffer start at the end of DSP.
this.audio_heap_ptr = parseInt(this.json_object.size);
// Setup pointers offset
this.audio_heap_ptr_inputs = this.audio_heap_ptr;
this.audio_heap_ptr_outputs = this.audio_heap_ptr_inputs + (this.numIn * this.ptr_size);
// Setup buffer offset
this.audio_heap_inputs = this.audio_heap_ptr_outputs + (this.numOut * this.ptr_size);
this.audio_heap_outputs = this.audio_heap_inputs + (this.numIn * NUM_FRAMES * this.sample_size);
// Start of DSP memory : DSP is placed first with index 0
this.dsp = 0;
this.pathTable = [];
// Send output values to the AudioNode
this.update_outputs = function ()
{
if (this.outputs_items.length > 0 && this.output_handler && this.outputs_timer-- === 0) {
this.outputs_timer = 5;
for (var i = 0; i < this.outputs_items.length; i++) {
this.output_handler(this.outputs_items[i], this.HEAPF32[this.pathTable[this.outputs_items[i]] >> 2]);
}
}
}
this.initAux = function ()
{
var i;
if (this.numIn > 0) {
this.ins = this.audio_heap_ptr_inputs;
for (i = 0; i < this.numIn; i++) {
this.HEAP32[(this.ins >> 2) + i] = this.audio_heap_inputs + ((NUM_FRAMES * this.sample_size) * i);
}
// Prepare Ins buffer tables
var dspInChans = this.HEAP32.subarray(this.ins >> 2, (this.ins + this.numIn * this.ptr_size) >> 2);
for (i = 0; i < this.numIn; i++) {
this.dspInChannnels[i] = this.HEAPF32.subarray(dspInChans[i] >> 2, (dspInChans[i] + NUM_FRAMES * this.sample_size) >> 2);
}
}
if (this.numOut > 0) {
this.outs = this.audio_heap_ptr_outputs;
for (i = 0; i < this.numOut; i++) {
this.HEAP32[(this.outs >> 2) + i] = this.audio_heap_outputs + ((NUM_FRAMES * this.sample_size) * i);
}
// Prepare Out buffer tables
var dspOutChans = this.HEAP32.subarray(this.outs >> 2, (this.outs + this.numOut * this.ptr_size) >> 2);
for (i = 0; i < this.numOut; i++) {
this.dspOutChannnels[i] = this.HEAPF32.subarray(dspOutChans[i] >> 2, (dspOutChans[i] + NUM_FRAMES * this.sample_size) >> 2);
}
}
// Parse UI
stonephaserProcessor.parse_ui(this.json_object.ui, this, stonephaserProcessor.parse_item2);
// Init DSP
this.factory.init(this.dsp, sampleRate); // 'sampleRate' is defined in AudioWorkletGlobalScope
}
this.setParamValue = function (path, val)
{
this.HEAPF32[this.pathTable[path] >> 2] = val;
}
this.getParamValue = function (path)
{
return this.HEAPF32[this.pathTable[path] >> 2];
}
// Init resulting DSP
this.initAux();
}
process(inputs, outputs, parameters)
{
var input = inputs[0];
var output = outputs[0];
// Check inputs
if (this.numIn > 0 && (!input || !input[0] || input[0].length === 0)) {
//console.log("Process input error");
return true;
}
// Check outputs
if (this.numOut > 0 && (!output || !output[0] || output[0].length === 0)) {
//console.log("Process output error");
return true;
}
// Copy inputs
if (input !== undefined) {
for (var chan = 0; chan < Math.min(this.numIn, input.length); ++chan) {
var dspInput = this.dspInChannnels[chan];
dspInput.set(input[chan]);
}
}
/*
TODO: sample accurate control change is not yet handled
When no automation occurs, params[i][1] has a length of 1,
otherwise params[i][1] has a length of NUM_FRAMES with possible control change each sample
*/
// Update controls
for (const path in parameters) {
const paramArray = parameters[path];
this.setParamValue(path, paramArray[0]);
}
// Compute
try {
this.factory.compute(this.dsp, NUM_FRAMES, this.ins, this.outs);
} catch(e) {
console.log("ERROR in compute (" + e + ")");
}
// Update bargraph
this.update_outputs();
// Copy outputs
if (output !== undefined) {
for (var chan = 0; chan < Math.min(this.numOut, output.length); ++chan) {
var dspOutput = this.dspOutChannnels[chan];
output[chan].set(dspOutput);
}
}
return this.running;
}
handleMessage(event)
{
var msg = event.data;
switch (msg.type) {
case "destroy": this.running = false; break;
}
}
}
// Globals
const NUM_FRAMES = 128;
try {
registerProcessor('stonephaser', stonephaserProcessor);
} catch (error) {
console.warn(error);
}
`;
const dspName = "stonephaser";
// WAP factory or npm package module
if (typeof module === "undefined") {
window.stonephaser = stonephaser;
window.Fauststonephaser = stonephaser;
window[dspName] = stonephaser;
} else {
module.exports = { stonephaser };
}