|  | @@ -33,12 +33,14 @@ laBaseNodeType LA_IDN_VCA;
 | 
	
		
			
				|  |  |  laBaseNodeType LA_IDN_NOISE;
 | 
	
		
			
				|  |  |  laBaseNodeType LA_IDN_OUTPUT;
 | 
	
		
			
				|  |  |  laBaseNodeType LA_IDN_SCOPE;
 | 
	
		
			
				|  |  | +laBaseNodeType LA_IDN_ENVELOPE;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  laPropContainer* LA_PC_IDN_FM;
 | 
	
		
			
				|  |  |  laPropContainer* LA_PC_IDN_VCA;
 | 
	
		
			
				|  |  |  laPropContainer* LA_PC_IDN_NOISE;
 | 
	
		
			
				|  |  |  laPropContainer* LA_PC_IDN_OUTPUT;
 | 
	
		
			
				|  |  |  laPropContainer* LA_PC_IDN_SCOPE;
 | 
	
		
			
				|  |  | +laPropContainer* LA_PC_IDN_ENVELOPE;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  #define _2PI 6.283185307
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -55,6 +57,9 @@ laPropContainer* LA_PC_IDN_SCOPE;
 | 
	
		
			
				|  |  |      {if(socket->Source && socket->Source->DataType==(LA_PROP_FLOAT|LA_PROP_ARRAY) && socket->Source->ArrLen>=LA_SYNTH_PLEN){ \
 | 
	
		
			
				|  |  |          ptr=((real*)socket->Source->Data); }else{ ptr=0; } }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +#define FRAME_INTERVAL (MAIN.Audio->AudioFrameInterval)
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +#define SAMPLE_RATE (MAIN.Audio->AudioSampleRate)
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  void IDN_FMInit(laSynthNodeFM* n, int NoCreate){
 | 
	
		
			
				|  |  |      INITPACKET(n->OutSamples);
 | 
	
	
		
			
				|  | @@ -76,7 +81,7 @@ int IDN_FMEval(laSynthNodeFM* n){
 | 
	
		
			
				|  |  |      INPUTPACKET(inputf,n->InFrequency); LA_GET_SRC_AS_VALUE(constf,n->InFrequency); 
 | 
	
		
			
				|  |  |      for(int i=0;i<LA_SYNTH_PLEN;i++){
 | 
	
		
			
				|  |  |          real useF=inputf?(inputf[i]+f):(f+constf); if(n->Slow) useF-=10;
 | 
	
		
			
				|  |  | -        n->Phase+=MAIN.Audio->AudioFrameInterval*VAL2FREQ(useF)*_2PI;
 | 
	
		
			
				|  |  | +        n->Phase+=FRAME_INTERVAL*VAL2FREQ(useF)*_2PI;
 | 
	
		
			
				|  |  |          WRAPPHASE(n->Phase);
 | 
	
		
			
				|  |  |          n->OutSamples[i]=sin(n->Phase)*10;
 | 
	
		
			
				|  |  |      }
 | 
	
	
		
			
				|  | @@ -85,7 +90,7 @@ int IDN_FMEval(laSynthNodeFM* n){
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  void IDN_FMCopy(laSynthNodeFM* new, laSynthNodeFM* old, int DoRematch){
 | 
	
		
			
				|  |  |      if(DoRematch){ LA_IDN_NEW_LINK(InFrequency) return; }
 | 
	
		
			
				|  |  | -    LA_IDN_OLD_DUPL(Out); new->Frequency=old->Frequency;
 | 
	
		
			
				|  |  | +    LA_IDN_OLD_DUPL(Out); new->Frequency=old->Frequency; new->Slow=old->Slow;
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  void laui_FMNode(laUiList *uil, laPropPack *This, laPropPack *Extra, laColumn *UNUSED, int context){
 | 
	
		
			
				|  |  |      laColumn* c=laFirstColumn(uil); laSynthNodeFM*n=This->EndInstance;
 | 
	
	
		
			
				|  | @@ -232,7 +237,8 @@ int IDN_ScopeEval(laSynthNodeScope* n){
 | 
	
		
			
				|  |  |              if(ch2){ smin2=TNS_MIN2(smin2,ch2[sp]); smax2=TNS_MAX2(smax2,ch2[sp]); }
 | 
	
		
			
				|  |  |              sp++;
 | 
	
		
			
				|  |  |          }
 | 
	
		
			
				|  |  | -        if(!ch1){smin1=smax1=0;} if(!ch2){smin2=smax2=0;}
 | 
	
		
			
				|  |  | +        if(!ch1){smin1=smax1=0; LA_GET_SRC_AS_VALUE(smin1,n->In1); smax1=smin1; }
 | 
	
		
			
				|  |  | +        if(!ch2){smin2=smax2=0; LA_GET_SRC_AS_VALUE(smin2,n->In2); smax2=smin2; }
 | 
	
		
			
				|  |  |          int ns=n->NextSample;
 | 
	
		
			
				|  |  |          n->Display1[ns]=smin1; n->Display1[ns+LA_SYNTH_PLEN]=smax1;
 | 
	
		
			
				|  |  |          n->Display2[ns]=smin2; n->Display2[ns+LA_SYNTH_PLEN]=smax2;
 | 
	
	
		
			
				|  | @@ -306,6 +312,95 @@ void laui_OutputNode(laUiList *uil, laPropPack *This, laPropPack *Extra, laColum
 | 
	
		
			
				|  |  |      laEndRow(uil,b);
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +void IDN_EnvelopeInit(laSynthNodeEnvelope* n, int NoCreate){
 | 
	
		
			
				|  |  | +    if(!NoCreate){ strSafeSet(&n->Base.Name,"Envelope");
 | 
	
		
			
				|  |  | +        n->Attack=laCreateInSocket("ATTACK",0); n->Delay=laCreateInSocket("DELAY",0);
 | 
	
		
			
				|  |  | +        n->Sustain=laCreateInSocket("SUSTAIN",0); n->Release=laCreateInSocket("RELEASE",0);
 | 
	
		
			
				|  |  | +        n->Trigger=laCreateInSocket("TRIGGER",0); n->Out=laCreateOutSocket(n, "OUT",0);
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +    if(!n->OutSamples){ n->OutSamples=memAcquireSimple(sizeof(real)*LA_SYNTH_PLEN); }
 | 
	
		
			
				|  |  | +    n->Out->Data = n->OutSamples; n->Out->DataType = LA_PROP_FLOAT|LA_PROP_ARRAY; n->Out->ArrLen=LA_SYNTH_PLEN;
 | 
	
		
			
				|  |  | +    n->Time=20; n->rAttack=1; n->rDelay=5; n->rSustain=5; n->rRelease=7;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +void IDN_EnvelopeDestroy(laSynthNodeEnvelope* n){
 | 
	
		
			
				|  |  | +    laDestroyInSocket(n->Attack); laDestroyInSocket(n->Delay);
 | 
	
		
			
				|  |  | +    laDestroyInSocket(n->Sustain); laDestroyInSocket(n->Release);
 | 
	
		
			
				|  |  | +    laDestroyInSocket(n->Trigger); laDestroyOutSocket(n->Out);
 | 
	
		
			
				|  |  | +    memFree(n->OutSamples);
 | 
	
		
			
				|  |  | +    strSafeDestroy(&n->Base.Name);
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +int IDN_EnvelopeVisit(laSynthNodeEnvelope* n, laNodeVisitInfo* vi){
 | 
	
		
			
				|  |  | +    LA_GUARD_THIS_NODE(n,vi);
 | 
	
		
			
				|  |  | +    if(LA_SRC_AND_PARENT(n->Attack)){ laBaseNode*bn=n->Attack->Source->Parent; LA_VISIT_NODE(bn,vi); }
 | 
	
		
			
				|  |  | +    if(LA_SRC_AND_PARENT(n->Delay)){ laBaseNode*bn=n->Delay->Source->Parent; LA_VISIT_NODE(bn,vi); }
 | 
	
		
			
				|  |  | +    if(LA_SRC_AND_PARENT(n->Sustain)){ laBaseNode*bn=n->Sustain->Source->Parent; LA_VISIT_NODE(bn,vi); }
 | 
	
		
			
				|  |  | +    if(LA_SRC_AND_PARENT(n->Release)){ laBaseNode*bn=n->Release->Source->Parent; LA_VISIT_NODE(bn,vi); }
 | 
	
		
			
				|  |  | +    if(LA_SRC_AND_PARENT(n->Trigger)){ laBaseNode*bn=n->Trigger->Source->Parent; LA_VISIT_NODE(bn,vi); }
 | 
	
		
			
				|  |  | +    LA_ADD_THIS_NODE(n,vi);
 | 
	
		
			
				|  |  | +    return LA_DAG_FLAG_PERM;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +int IDN_EnvelopeEval(laSynthNodeEnvelope* n){
 | 
	
		
			
				|  |  | +    int trigger_at=-1; real trig=n->bTrigger*100;
 | 
	
		
			
				|  |  | +    real* trigger; INPUTPACKET(trigger,n->Trigger); if(!trigger){ LA_GET_SRC_AS_VALUE(trig,n->Trigger); }
 | 
	
		
			
				|  |  | +    real attack=n->rAttack,delay=n->rDelay,sustain=n->rSustain,release=n->rRelease;
 | 
	
		
			
				|  |  | +    LA_GET_SRC_AS_VALUE(attack,n->Attack); LA_GET_SRC_AS_VALUE(delay,n->Delay);
 | 
	
		
			
				|  |  | +    LA_GET_SRC_AS_VALUE(sustain,n->Sustain); LA_GET_SRC_AS_VALUE(release,n->Release);
 | 
	
		
			
				|  |  | +    attack=attack?pow(attack/10,2):0; delay=delay?pow(delay/10,2):0; release=release?pow(release/10,2):0;
 | 
	
		
			
				|  |  | +    for(int i=0;i<LA_SYNTH_PLEN;i++){
 | 
	
		
			
				|  |  | +        if(trigger){ trig=trigger[i]; }
 | 
	
		
			
				|  |  | +        if(!n->Triggered){ if(trig > 5.0f){ n->Triggered = 1; n->Time = 0; n->ReleaseTime=10000; n->AtLevel=0; } }
 | 
	
		
			
				|  |  | +        else{ if(trig < 2.0f){ n->Triggered = 0; n->ReleaseTime = n->Time; } }
 | 
	
		
			
				|  |  | +        if(n->Time < n->ReleaseTime){
 | 
	
		
			
				|  |  | +            if(n->Time<=attack){
 | 
	
		
			
				|  |  | +                n->OutSamples[i]=10.0f*pow(n->Time/attack,0.3);
 | 
	
		
			
				|  |  | +            }elif(n->Time<=delay + attack){
 | 
	
		
			
				|  |  | +                real fac=pow((n->Time-attack)/delay,0.3);
 | 
	
		
			
				|  |  | +                n->OutSamples[i]= tnsLinearItp(10.0f, sustain, fac);
 | 
	
		
			
				|  |  | +            }else{
 | 
	
		
			
				|  |  | +                n->OutSamples[i]=sustain;
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            n->AtLevel=n->OutSamples[i];
 | 
	
		
			
				|  |  | +        }else{
 | 
	
		
			
				|  |  | +            if(n->Time - n->ReleaseTime < release){
 | 
	
		
			
				|  |  | +                real fac=pow((n->Time - n->ReleaseTime)/release,0.3);
 | 
	
		
			
				|  |  | +                n->OutSamples[i]= tnsLinearItp(n->AtLevel,0.0f, fac);
 | 
	
		
			
				|  |  | +            }else{
 | 
	
		
			
				|  |  | +                n->OutSamples[i]=0;
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +        n->Time+=FRAME_INTERVAL;
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +    return 1;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +void IDN_EnvelopeCopy(laSynthNodeEnvelope* new, laSynthNodeEnvelope* old, int DoRematch){
 | 
	
		
			
				|  |  | +    if(DoRematch){
 | 
	
		
			
				|  |  | +        LA_IDN_NEW_LINK(Attack) LA_IDN_NEW_LINK(Delay)
 | 
	
		
			
				|  |  | +        LA_IDN_NEW_LINK(Sustain) LA_IDN_NEW_LINK(Release) LA_IDN_NEW_LINK(Trigger) return;
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +    LA_IDN_OLD_DUPL(Out);
 | 
	
		
			
				|  |  | +    new->rAttack=old->rAttack;new->rSustain=old->rSustain;new->rDelay=old->rDelay;new->rRelease=old->rRelease;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +void laui_EnvelopeNode(laUiList *uil, laPropPack *This, laPropPack *Extra, laColumn *UNUSED, int context){
 | 
	
		
			
				|  |  | +    laColumn* c=laFirstColumn(uil); laSynthNodeEnvelope*n=This->EndInstance;
 | 
	
		
			
				|  |  | +    LA_BASE_NODE_HEADER(uil,c,This);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    laUiItem* b,*b1;
 | 
	
		
			
				|  |  | +#define ADSR_ROW(what) \
 | 
	
		
			
				|  |  | +    b=laBeginRow(uil,c,0,0); \
 | 
	
		
			
				|  |  | +    laShowNodeSocket(uil,c,This,"in_" what,0)->Flags|=LA_UI_SOCKET_LABEL_E; \
 | 
	
		
			
				|  |  | +    b1=laOnConditionThat(uil,c,laNot(laPropExpression(This,"in_" what ".source")));{ \
 | 
	
		
			
				|  |  | +    laShowItem(uil,c,This,what)->Expand=1; \
 | 
	
		
			
				|  |  | +    } laEndCondition(uil,b1); \
 | 
	
		
			
				|  |  | +    laEndRow(uil,b);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    ADSR_ROW("attack") ADSR_ROW("delay") ADSR_ROW("sustain") ADSR_ROW("release")
 | 
	
		
			
				|  |  | +    
 | 
	
		
			
				|  |  | +    b=laBeginRow(uil,c,0,0);
 | 
	
		
			
				|  |  | +    laShowItem(uil,c,This,"in_trigger"); laShowItem(uil,c,This,"trigger")->Flags|=LA_UI_FLAGS_CYCLE;
 | 
	
		
			
				|  |  | +    laShowSeparator(uil,c); laShowItem(uil,c,This,"out");
 | 
	
		
			
				|  |  | +    laEndRow(uil,b);
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  void laRebuildSynthGraphs(){
 | 
	
		
			
				|  |  |      if(MAIN.Audio->CurrentSynth){
 | 
	
		
			
				|  |  |          laSpinLock(&MAIN.Audio->CurrentSynth->Lock);
 | 
	
	
		
			
				|  | @@ -636,16 +731,34 @@ void laInitAudio(){
 | 
	
		
			
				|  |  |      laAddSubGroup(pc,"base","Base","Base node","la_base_node",0,0,0,0,0,0,0,0,0,0,0,LA_UDF_LOCAL);
 | 
	
		
			
				|  |  |      laAddSubGroup(pc,"in", "Input","Input sound","la_in_socket",0,0,0,offsetof(laSynthNodeOutput,In),0,0,0,0,0,0,0,LA_UDF_SINGLE);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +    pc=laAddPropertyContainer("la_node_synth_envelope", "Envelope Node", "Sound envelope",0,laui_EnvelopeNode,sizeof(laSynthNodeEnvelope),lapost_Node,0,1);
 | 
	
		
			
				|  |  | +    LA_PC_IDN_ENVELOPE=pc; laPropContainerExtraFunctions(pc,0,0,0,0,laui_DefaultNodeOperationsPropUiDefine);
 | 
	
		
			
				|  |  | +    laAddSubGroup(pc,"base","Base","Base node","la_base_node",0,0,0,0,0,0,0,0,0,0,0,LA_UDF_LOCAL);
 | 
	
		
			
				|  |  | +    laAddSubGroup(pc,"in_attack", "Attack","Attack input","la_in_socket",0,0,0,offsetof(laSynthNodeEnvelope,Attack),0,0,0,0,0,0,0,LA_UDF_SINGLE);
 | 
	
		
			
				|  |  | +    laAddSubGroup(pc,"in_delay", "Delay","Delay input","la_in_socket",0,0,0,offsetof(laSynthNodeEnvelope,Delay),0,0,0,0,0,0,0,LA_UDF_SINGLE);
 | 
	
		
			
				|  |  | +    laAddSubGroup(pc,"in_sustain", "Sustain","Sustain input","la_in_socket",0,0,0,offsetof(laSynthNodeEnvelope,Sustain),0,0,0,0,0,0,0,LA_UDF_SINGLE);
 | 
	
		
			
				|  |  | +    laAddSubGroup(pc,"in_release", "Release","Release input","la_in_socket",0,0,0,offsetof(laSynthNodeEnvelope,Release),0,0,0,0,0,0,0,LA_UDF_SINGLE);
 | 
	
		
			
				|  |  | +    laAddSubGroup(pc,"in_trigger", "Trigger","Trigger input","la_in_socket",0,0,0,offsetof(laSynthNodeEnvelope,Trigger),0,0,0,0,0,0,0,LA_UDF_SINGLE);
 | 
	
		
			
				|  |  | +    laAddSubGroup(pc,"out", "Output","Envelope output","la_out_socket",0,0,0,offsetof(laSynthNodeEnvelope,Out),0,0,0,0,0,0,0,LA_UDF_SINGLE);
 | 
	
		
			
				|  |  | +    laAddFloatProperty(pc,"attack","Attack","Attack value",0,0,0,10,0,0.1,0,0,offsetof(laSynthNodeEnvelope,rAttack),0,0,0,0,0,0,0,0,0,0,0);
 | 
	
		
			
				|  |  | +    laAddFloatProperty(pc,"delay","Delay","Delay value",0,0,0,10,0,0.1,0,0,offsetof(laSynthNodeEnvelope,rDelay),0,0,0,0,0,0,0,0,0,0,0);
 | 
	
		
			
				|  |  | +    laAddFloatProperty(pc,"sustain","Sustain","Sustain value",0,0,0,10,0,0.1,0,0,offsetof(laSynthNodeEnvelope,rSustain),0,0,0,0,0,0,0,0,0,0,0);
 | 
	
		
			
				|  |  | +    laAddFloatProperty(pc,"release","Release","Release value",0,0,0,10,0,0.1,0,0,offsetof(laSynthNodeEnvelope,rRelease),0,0,0,0,0,0,0,0,0,0,0);
 | 
	
		
			
				|  |  | +    p=laAddEnumProperty(pc,"trigger","Trigger","Trigger the envelope",LA_WIDGET_ENUM_CYCLE,0,0,0,0,offsetof(laSynthNodeEnvelope,bTrigger),0,0,0,0,0,0,0,0,0,0);
 | 
	
		
			
				|  |  | +    laAddEnumItemAs(p,"IDLE","Idle","Trigger idle",0,0);
 | 
	
		
			
				|  |  | +    laAddEnumItemAs(p,"TRIG","Trigger","Trigger active",1,0);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |      LA_IDN_REGISTER("VCO",'f',LA_IDN_FM, LA_PC_IDN_FM, IDN_FM, laSynthNodeFM);
 | 
	
		
			
				|  |  |      LA_IDN_REGISTER("VCA",'a',LA_IDN_VCA, LA_PC_IDN_VCA, IDN_VCA, laSynthNodeVCA);
 | 
	
		
			
				|  |  |      LA_IDN_REGISTER("Noise",'~',LA_IDN_NOISE, LA_PC_IDN_NOISE, IDN_Noise, laSynthNodeNoise);
 | 
	
		
			
				|  |  |      LA_IDN_REGISTER("Scope",'s',LA_IDN_SCOPE, LA_PC_IDN_SCOPE, IDN_Scope, laSynthNodeScope);
 | 
	
		
			
				|  |  |      LA_IDN_REGISTER("Sound Output",U'🕪',LA_IDN_OUTPUT, LA_PC_IDN_OUTPUT, IDN_Output, laSynthNodeOutput);
 | 
	
		
			
				|  |  | +    LA_IDN_REGISTER("Envelope",U'^',LA_IDN_ENVELOPE, LA_PC_IDN_ENVELOPE, IDN_Envelope, laSynthNodeEnvelope);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |      LA_NODE_CATEGORY_SYNTHESIZER=laAddNodeCategory("OSC",0,LA_RACK_TYPE_AUDIO);
 | 
	
		
			
				|  |  |      LA_NODE_CATEGORY_SYSTEM_SOUND=laAddNodeCategory("System",0,LA_RACK_TYPE_AUDIO);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -    laNodeCategoryAddNodeTypes(LA_NODE_CATEGORY_SYNTHESIZER, &LA_IDN_FM,&LA_IDN_VCA,&LA_IDN_NOISE,&LA_IDN_SCOPE,0);
 | 
	
		
			
				|  |  | +    laNodeCategoryAddNodeTypes(LA_NODE_CATEGORY_SYNTHESIZER, &LA_IDN_FM,&LA_IDN_NOISE,&LA_IDN_VCA,&LA_IDN_ENVELOPE,&LA_IDN_SCOPE,0);
 | 
	
		
			
				|  |  |      laNodeCategoryAddNodeTypes(LA_NODE_CATEGORY_SYSTEM_SOUND, &LA_IDN_OUTPUT,0);
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 |