From de541742412c66ddcea653f8ee42d33a8c413393 Mon Sep 17 00:00:00 2001 From: Rolando Islas Date: Wed, 30 Dec 2015 04:35:33 -0700 Subject: [PATCH] Initial commit --- .gitignore | 2 + LightHost.jucer | 86 +++++++++ Resources/icon.png | Bin 0 -> 18740 bytes Resources/menu_icon.png | Bin 0 -> 386 bytes Resources/menu_icon_white.png | Bin 0 -> 731 bytes Source/HostStartup.cpp | 67 +++++++ Source/IconMenu.cpp | 284 ++++++++++++++++++++++++++++ Source/IconMenu.hpp | 48 +++++ Source/PluginWindow.cpp | 190 +++++++++++++++++++ Source/PluginWindow.h | 56 ++++++ gpl.txt | 339 ++++++++++++++++++++++++++++++++++ license | 16 ++ readme.md | 4 + third_party | 14 ++ 14 files changed, 1106 insertions(+) create mode 100644 .gitignore create mode 100644 LightHost.jucer create mode 100644 Resources/icon.png create mode 100644 Resources/menu_icon.png create mode 100644 Resources/menu_icon_white.png create mode 100644 Source/HostStartup.cpp create mode 100644 Source/IconMenu.cpp create mode 100644 Source/IconMenu.hpp create mode 100644 Source/PluginWindow.cpp create mode 100644 Source/PluginWindow.h create mode 100644 gpl.txt create mode 100644 license create mode 100644 readme.md create mode 100644 third_party diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..7a6185d --- /dev/null +++ b/.gitignore @@ -0,0 +1,2 @@ +Builds/ +JuceLibraryCode/ \ No newline at end of file diff --git a/LightHost.jucer b/LightHost.jucer new file mode 100644 index 0000000..d7396ee --- /dev/null +++ b/LightHost.jucer @@ -0,0 +1,86 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Resources/icon.png b/Resources/icon.png new file mode 100644 index 0000000000000000000000000000000000000000..7397021f3f7b29913b910aba5905a13e3c2140c3 GIT binary patch literal 18740 zcmY&=2{=^Y_wc>fVC*x@B1?5yZZh9|L^;r=ROaQ_r2$w_nh;d{fe?QH)3JrV*~(z#l%?uEC4|8uMmKw zgMY01ckKcI1~AdrIq!dYA>+&&52MX)>#)eIRF4%J+B2b&k95^lQ3w-ys@{mWI{Pml z{Vw7?4o;(+>a(OM@f8|hOh)9iaT0l5eugKxhd6@T{)s>J`a1@I{_}vK6N&&J z4<8Z%i38})t2@yBXETTdTmS^_-$SPlK*zH>XI3qdUr^}wqWP5`^h)-VrByy80q2Fl z0F6h&{q;ofmHr`9$j6>WKUCPL1yk$iK1TAGjguwNE8nw!hTzvdjrGqIbZ{YMZ~#;c z$Q-35enWoodgE!=YZ1cjQ5fLMp`c2k1b8>10L1^G^ioEY>V-&QUk~oC@A*qOo=Iip zV^>JcXZXD&&qghxSwW)_ySf0>kx~p zjT6|Nrac>qQ#>S>GgNg3o!T~$XK;lLb8_Q&@9GFX6>7X(8cbqEex-?KR(8+ub=lT$OWgmca@Kn40vigO>E9T_l1 zgZfS-tY-DJDvo3j)er+9Fdy&6FK|IJyJe`+rTfAFZ!eSWY{5eSk;T`gbRG$EQFL)< z&nO@;G<8`yuhL*dH(#B(xzQVIZk3eG;lf9{8SC30|fd$ zFK{!h+hcXx>+1^DY(&LhAP?#t8@>8UIF=tm0Vm$AO zmH_c2l6<_E&hPm}Vc_Ju{C@Pa-P$;#kP9t0m=SWq80@n6%tKzxX*2^l&%`h@+F~`4$1;c}xLRkt>@2Bqa3SgXwgho0R0@aWqOt7m2u&pXnY z59DSLNHe5l9BGeU2($!=8t|E!!x@M@tX08^TMFVh5;v%ZY~ovv^5mZBvxir`6IO%F z%hnnPimqzG=833)vYgI}1j^WAUoL&D4hLf12WG5UOdi~^%9_F)KMDQ*0BkWctFmGM znqw5aRh$PO2SXNYwPGFQ3zAv ztUAXLlF*ZP@%L7R0k9!2=`8ylXA3FOxAcgg288H`F}`fz^7+NFuybrx!4{NGY1Er2 zz&up>S79#~A`9DlB<-wOZQJg~bLi0CilHsrfE~JcjV#f3EOViN5~3FW`_YWGB-=k) zHuZ7Cybc;>GXfwIH^rzqzo{)=S`=c?NrRe3f)AK$VwJN*9%rDBKpkZVy&GEv+Bx*Z zAvB4v;{%;A5Oj=G(c~7R*kJ$dtvn4;I?x)yDFq+bgLmHe zJBkpTY;dwR*ylkG=J+w_Dtg{{4MosO&V7Hcab!56FrFP@XhhJURp$j!g#Aa~e#zIy z-aNhSt3NIB9{YRlj{WARX#4eng8ju3C5L7aVMX;Lr&I(D8s#T;`f(kXZzAw0vw{~$ zB3r?Ln zfY&kmDjC@W)!m7%_KRsEC#AdH`x0@2l8kJa6=f$W ze{bxf0gYeBN0P=fEhH|9?yg=>=%L5=|M7hLdt)$)Clat2CyVv8vG&TxA`+zpp?D)< z|DCB{b#^9KYkIIn8T;$^Z##V1Vg?h4QJm8%^+-xA>njBN3@a3`$7?^Dq{ntTw`1`# z6Q#7*yRCxrt8cqSJsp;2_p50!YyCyfJ8vze$1VCF0HxMH{6t1iwe9_tzVZ-ZxBMyW z85f1OXxUf{A+DIEZ;J{w&GNi%7!1NbGJ=1W{O2)z%}zq)E55S(p11SEdbad%ZTEhM z77dr_PH!fKr=?)B#+1-t0z~;)foHhqn;AsKA!a|Dy8FQ|_A?`bvc@Mu1kMc_vfG_9 za5{7mLUq!^L7dZ(x3P$#a=*{t4;o*+mIRS^T_<#aV>H3b9tX@xAwlG#((M`NymmS5 zCw6zy*GgC;QJVd^w8tF62wYTa;X(JTOH@{b)2X}eL!vbyEI0SQe1BETR(R?UBX2{l z>?-Z~ZhA!4QL@+OIjs;=4TR15U+2tf*9P2CmYIt>=`_;QG>|1Y9+slr7VKNBb$ep% zdp%+;n1BENEnh4Rc43>j^cSX$cmy8-%(Io;rR>IA&I)6ef)+6AXyKE!M;BvdrKSiC zRg=emA$*nd)nwcOR|)unYk>Fht4SY)Qg-GeR&ocmCcocHjq6G@1}+jS`K(1_(`lkm zy`UeG#F}-KD`;Wgc7N;ZR{6!ddwVWsOk3#GeCUxVMt0>OJFJDM;miz5vgLCHecNvu zYw`#QF}vZc2K$buE}th^z9D+D?v4o}akiE#>+;c475(FlMIVC-xdw&Y?NZ|ix`E2n z;Z!Q4g`7TN{VNk#wV{e2bLp;8+eWp8-YZ)KbD}HFIrD0ir&r8n5k8t(cT5>aDJTiE zV$^38K@nH7o$__1lq+M&y{t$mov`^e)ox1%_cHv^(GIp>G!qIB5u7AWs8~>+i@Lz` zqI-G6nfrlCNWiVPAGDxd!-ivo?@qS!!XW=jIB<(5zAl76cjzkWF0O#X5JG9xf4FRvaVvnL>C6-sFbBX5ZBp z$zVg2Ksc#gmpN!wa2Xkn`_ei?RIBuzxJv9Ug(<78D267*xM8K+DX`4+aA@J(*dk$yas_ zo(3yvvI79Lfc7836S*pACTB){za?e@w4i@zSj&w}gYYl6rBcuxg_(bMd4w_X9N_IM zoVyQ|$(6y$57sXvCh7o6W#18ugXv%f^TD+&ZU+B=^`*XP zE?{B%(^?8G!HXBv+~*a%0x7NxJe!vje=)#LwXXYYn@Co9@WSrOvRgO- z7lOR*lFd1g!%>s3F(HFy;3v;1Xw{+^xuz<5VKK0?)DQ7<_TWNbV}On z;bWrfaX5+?aMga|2;2x?Vxx0@mHf5a6#)_Iso#b7zY6CTZIEAk!`k~1-I04a&Q1H& z&#=t|a1nuO?Rv8=YpugYAZ+)Fw8@cU>Ub&#ZU*+4n(oKs9*3;kr8M*xjy>hxR zYyD&7b}f3v5j#4pxM&?G@%X+igaIuurDojo#(--5y^~WWiys&Fww_~0*$jRxowDbx ze9a|Tu^*_vpA@-W?|zJ&;m?V#mB$7z;z-@NS3ex7GS~xW{r!67Bh34(TgUD0w_&BoGvHWt+>;YbXzanz`_5%&c{fRzG zCR5McWkEF5u5{~(R7VsLS#M2JTlq=>Sv53}NYUiYf~|=5&EHA=M-J*HiVE)O ziEeCpOVrWKmwxQL6QxzC+~oKu((k6y)h;?wa+KojL_`IO03$n<9rgLMG;HY-_L|Wv zQz^BdXIRu24;CZ0Ri58m3D_w!3RgUL@TkdQR~o&-eXOkSzBd+BXXiO(w!A>Y>M-}4 z2;dtn5DEu#ubrC(TWez@hd-F#dS4c7rN96F;m~CK;I5<+`5-gc5qLB{~1{lubO8`wRcmia6W@7L! z*`Awz_qS66QWrw$YF4}(d|H2hZ)0$Jw^hbU11lU-=`Q*g$;J>lt=|964LW}B zMF-As*)bhIt3Jfkedye@R>o%2A7 zkpwvSkZ#iyFl60vSs6ZfNx2UT@U!)~{D{4@69e4+_bt(2(im~9eiwu2E7u;9#VmAzr&*uc zKcWah_|}VaZ$z`A-w=)>N$zN2=(yd?&zP>4S1`cB6FL}Wm>qkJDby)6cufMx7T^En zFp;K|90aHJ^I>y|cEZ5vj#+9Sn}}}XQ^R93aZm{dL1w4loweUWME4?FQ!pGUZhpV( z(TXBe81HCGeB2hvTx^*&nXQ5pIS6QPN=N|Go6|dIJ_0~EY2TXaW(HHEliR&ZrjMB| zie<{R&YJXBK_%<}3#Y@H^b*zT(G|)cM=mNwnHB668oMNm13ZMscl?6>`aWJ2VRu|wVgeBJbTP$-y89c1EQQv7 za|r_rz8!XYC@`7)?&f)So+xs8Uz_SCRJNw_q zY6qw2p$^u)x3J>h7rdH)Gbd*2(@i6BMxd0bICMO5EHnLJ9GNt$DOnepFM>d1v9|Tt z5rNVk9WI8_84d=2xoDh5bAouYI~OdipS9<&Wo6Lc|2Yfe`ve_I%#1x|oHJEi&k0`d zRI_Zt$D28ZGM3>Dx`N8&`t zacOWS(xY>c!LW(5cfE<+jc`gXu2s66Y3ml~T7dzcfZD$W{=qnX{E;^9cQNTH!ikeM zB)EJ;e(fs%nUgRXkM$18s6u3K(ve-MU)BYioXOCg$vl(Qxd=A8v zz#HVZ@@V1LwP9-1r%gi9>_0D*L0#g2$o#?kkL53}-C?(dg>eKt6vmLK6GbxrLmI*< ze6ZGxdgWd6z(4%34qT=`Wl9eA#b1!doq>)+_rI?X*Gn7%w)h9wDij?Hy|YblM-$L@ zss5C$HMK)(j{G~=6H?F#+g3))XrMgePGHb+%IQ1F`~LvL+6N!ZG>auS7J9MtiRQ$3 z;T769I)Gs>rIrs}*%#GG4;N!L?}Z+UshvGg0_&5M@Ba4Ro`F_nA%h=&4n{9;)WQbf z2-Ck4h~dj*mOKt&1Ph=`AXP3e%GTQpu$-#@T_-<=fhERX%4F9jt#BxKrC zgN|{-lz1K<1s_kM6cQXf83&#Nfk)8tsxc)eg2-vZ!T%6w9BO2_F^2ly+FsqeGtQv^ zxX%vwT{CP6sXf68!|?kDBY|{Eh2&vA39kiQZ#wt;19`ir?C_Fp(8E;HNHA!{M5?5# z`*d;Js_H|JQmT;z??}4Xkh%!x4Mt($k8nr_=8$#iqIf^#K;7sUZV-RXr7Gf@YHL$Y z5X@O?=ELOyQGaRdkoR2IOYJI~j4ce{wVSk->0x*~{&~N2)YWu|Qz}wU2T?7zfTcLzT@oCf~E_xaLM;N05H)#0}PN<$>_u zYY$QKm_YT9XW_L11nJL?=f@kaInVw$JHk@?a#h-dNG|pW91b}PTaV>`Es*n>8 zUd*tZ}^@#8V#32t36-?EnG5jzxSw> zX!!$z`)9ieM8_S{Xo)LHBaK?I;<^zJB0%+faeu*oXeGI+%`jqNrc@lLhf7s8l8F)jx+NNBG$m!Q@p;#Z*?z z@;I@8cqM8rKYI8CAG6QDIv$R!2<;Bly3loux)79MMh5DR3VnQr^E>RNkQt%aC?c1T zOOY_(F-g7fn&UOszxE5R2gzLfR8|ZqeqC6v@h$8hA^{YhnZe$Nxi}g4as<7^%OaFc zjY&4KK7`~X40#9>qhrCR?0`yAC}sxhT^9Q0cIWBYR9Mw=Ek&28B~jmgAFh!j34bZT$8-Ji!oZTz4;PUId39-p zSrrHHA5i~?giF`==Cc>X5M=3W`yExP0!8Ovdq^CSu1M4L!K(hRO0zvgzH7c}r60Lr zbx7smk-ta!e@8yFJ-F@7K)AF$Jb!{c8%%g9J$#Z$q?I70=r6o{SAW0b{b`5wNe%Rh zEq7uYhgUax3x>pb5U~?71oV$dm6?2|H|$b72^ATJr*}$jpz%!%$E5}Eav+|O`hDLq z}(_#?ewqk$S>A7WWU&ItMU#iNI$q63ScZ`g3~80}bSDQ!fSG)}G%N!>8FC^uLOb zg~=<$QJkNf`_b6j?-j~EG8dY&wN3Df(Quba|C#*h)ii89F+gipZBOyJ&A-tuvwYz8 zW&JNndfd-*ErV}o%vtthVIi?@yrrM(5%EI zro#wUr|YE9L(kDGBT}U)NW#%Tk+`ELO?0=1#kU{Tz_mR_i6^`;?G3r-Q!KaRu%lJ5 z{x#F$tdlU~d~SNiG$4;|tB)5A%wQJs!t%T56Z?J9M7f1n2b=b~ee z@`8sSF3HPLG#-4{wpnV2`5dQ|B-L}(i|khZ|~`;cfhfNNqeq(+IHF6{=CHY>*sIwvj+1oCH0- zD6_)U1XgzM?3~{DVJGt|Ok~9yL8_Y}Zd2>VT(-F3j+34lO$IFXhCE!#nH8G6ZFpUb zhTKb z2s7()EHXt^zDfZs6>MlrcEx?rSM-+lb7 z4)D-QSRQ54UQx6y>nPQk>b#vQSUg*RaGh8z~_D^l3ElYu~I zcJ+GECN-EuwT;m80Vwj{Cq~@$iRFP+iw00SJCG7iFA7{?pU?;1;>hYSAJRp}pQB$Q z)go4X2DQb3;aH?Ca8F+QF;XfzkqzYL%3!k<_0oaqv`&(XC0K}z$HMZhf5s22nPXBE zo^m^j&I!VDyUbE4+IV06C%hV+M56*sX5dhSFO@na0*6XEJ(A8TXXRy)G%m;iRJBbA z8htmV19>iP_yis{%%0(t{aYqpeaU-1?gA~F=xg2tZqA55!{wr7t%VR-0#Ccfqk!R} zFZX;`lJ4^{`kMjvZ7OYpp7DOhkHvTqPBKBi|4zVop1NFIV~H@XpssSC0scD+kQeM# zAD#PM3}+A8dmoeWWP*y}Vj~ZHhj1Q28ul#t*XazP9||De>0}Pjb}9e^RBVL#EWm9( z*!e3n*vB?4bst0l<>|LkVEp}6_%L(yGyVTeLYO;yxoY$91eA&-G)L?#MEYxQZS&fx zr`$PN%09DS2vcwCH3Y|&xsb|B4X=i%V(rxFnP zO1Hlnc05O?F^bSU-WVl!Rt>hX4Fv}8$5!E59F%&$qrw;ty?%*O{tMyh`<|c9Ls0;p z!LkCZu|9($RLAFDK5UroDDNai`pea(T8n*YrX}kGP1gJ%B>chft#sA>fo zHZ4nNfIsP{z`}~zf6PL1n4^X1Ol@l-{_OD(*Ww=>~jsV=j7?eNkRiKJWtmym?hHZ&X()E~tE;w3EET-u*1 zU0fC1dkQbQrK);A^!V#$|I_Xg=Z!&Lu*PNQ)7+}y{UcC(NU@s9wBns~A6-YU!=7Mn zoOfrhO{)QLIXW_K%_aEUx3T9eo?0z5@Sy=EVuqo{;c!AkiGaY=k_>S2=;NT6=P%)w zD|11xx;72WZ{~P*%~2dM^Z*d$F`h?X8tbny?yqm4_6Kve!0MJ_$U(QB)2;kXB>B4^xcn{M8 zjE)B^xQdMvCR375!oZg1toSm=dh*$8j|z$RA9g%;Wpo^_QH2CChQ<#*to?hMn7IaqDq#OuP2-g^2OPS~-)+}V>sWXT?szBqKf z9u%|AJ``a~Wb}?us$(*otHeRV0A-rd!87xz=(q@o`JH?+325w>Sw z0yAH71-4XO5pRqcpbO}2C9F@gz~NyK!GOs`zR$fM9GEIci z_O9vgm$O+FneM#f05L>EA(7Q8^a@2w-@p|0<`JxVE6#toZroylln~doZ(Vc3B^|U;uY(P@PAr6U;>5J$d&s(`8^y>1H zRsN*0^e{b|DK|=tal}eDTfcEX4WxbQ*3WzKzCOGT`>FXzxDB;_LY9Gc-pJb6jvmwn z=E(N1j%YxbrJX5L%-*lOoA{a;Ojb&~yhOLnn6R8HPaP zr2WWe*|`!9am-50UImLM$efhl5Rpp^C3zqNdC|i|1$U3hJtokA4630+)C23H%=tj|FqYR6uEH`vF+jdr$w&< zzRiQF&ta3D*Ff9RuK$fKOUOZZrUr@ksgSP`b1y4-YJnh4xWq^mj9U)Z%IyQ2^{HH5 zBBY%V4@$b}l$TF{k$PP9q@Vyl{5~AeRXnPc?@Bny0eKpQa~&}iLaK%Xj5AfAsHwod zB=*IU9zA#Nd0i0&es~F4sGWCSTg*vwfEnx zo`&HyRmoZ?{GF#Ev(kQWVDW`N3Nh`Xc;;+n@obHyQ9m=1Ov%(?T64v(*xsu&Jjwy2 zWb_)LU)EUe>JVHAhqE27DAK+oU47c8a+}7?$qZGPs7V$qc+(v2f+TRhVUF$UWP?Ph zGe*`)IdZ{X7IBSn41^@9r>BX#&+S>QBq=Xrf5@WBs~&wa_+DDU07h5_fXnOCGd?fR z^9*`F$26@dCID*d#rljXB5%ZkCi@i7q@5z|>Z))V>bh#o3LIFbsva-eQ28qQfeBpT z2+AoW0##H#$rR7;6RE31=p8sTjY43=@H-5a*@h`A)-=-n(;VQdj*J z8Y7|=Oz4G6Ova87s7XDY)mzA#;?hX(KB@+hOZGaUSAjsX78=h`H3J`CE(&mN6#JF$ zm{KU?XFRY@V)Ts#RD=9Ykmo~6l5hR0bTWSy6IA90y3e}_ZmUwZccc$X1kLy$TK3ROk6w`rf2aTZyoteth69Gme zA)PKu6Z2MGVi0*0DVggWIu|sp^^72Qr<~#*cCEUxZl2?#jQr9|3k7;tTK+yG>~MDMeYP8AL=Fu|v>UC00|HZnnq)7fKP+Tq zB@g)JfLrhB<^$NbS=r4zW!Nn6uD9x0ov0GM@{2cJ-C{-vd^SE_*W{{0nd|&OpR>2o zIU6UU`+Q`mxlaFl4}AMKMWmL7XpzxDFtU}Rh_jAu>N+;ee-t!yf(Z&6FA91b(|$y`egBlJX>Pv! zZDmwFBpR+KWT}YgRvA8GgVNrVENU+O7Q5GphV1rh^-3~hl}=imm(b3BZzWZRB>YxG zju6)9LOd;I-k9n?X&?+K^^|>C-R|iL8CFj4kc^)hIS>*4 zv&!!97|+~a|D`KW^1oqwu!w;cs8aACD$`d6A=x_$W-C$Yb+t^&x7j;yR98KF;ssYbn&`6hdP5U|u(*hpT#e!U|QF1~L5k7A}m+F4p&bx5N z>T9~(SB~Qu=JBuC|HjA1`=Ri86Z`Y{#>_xuomxJlw}r z?ydrrs-O#}=^V!eXl|v6I8@6dy)Y(8JD5B!wncUDzT*mUkThH|B2BCc-MD%2Zc_z< z)OsS(1)w$t4(O>s8c#R-*Pjj?+{SzUdG~xLq?=DA1t?cDlDrc$Dk3$8Asb$)|7Ab; zV)lz;Z2MJ~g6R8m-RWPbZkH;i*ntaga=7I`b)WsZ={Rm*gh>lJpF1v_Hhq<4UeEDU zZdb+IyP=}nZyyK#-MkzbS=9f?vUHl{22V}679;G{KzThM#(ytw<*yMA$%2|POP z7|fKDZItR#5%F%}$pPI#x1!spk%UWjP5~QDj5uBnaA92^QA`)}mio3waR;7Bbd6LL zhx#eajO4S0uc@aN$JwE*tsk)&TB$`RRvdorzYD(J8I3@R6*JZ>I$r4+u4Yv6Bi(k9 z)|Hl?$}fRUZNRy9#wV!{MA%SsW}(VzP^p6$9U}xw(8e#ebM73JKyUZ zx<$8=MX9&>AQtqT??d{k7~w{OI#0M!S}**v2g;^Tf|U*Ls*?fLqcj68DFcD2Slk&z zvd(f)EbUF*%ek^*8^3I|c`5Hmg9aiVFTX~&P&+&LAo4XutA;o)@pCvb517I72&wv@ zCiac9?0)}>^cFoWT>st1V)eHjnnpzy;L7L5j*(o@JR_1CS(Hv2#bcbE_%XD2d8*rjYi2;EFS?Zq z7k>HuzlF9CY+qmVP{VVzjV;{xcKXMQ{@E_P?3WQkYtF#!+hFjOp zPsa*G=|9>C`%SJPNViv%Y|ccaQ}(Vb*i=PUDNB#wSz35pjfwCKckkh1NAPz#>|;E&Z&0s7xpnE1!0s9 z&oj<+xlOy}z7hGOXTKch0PD}p-CTo{N%FMW%Z!Ls$P$+&4q@rAh!W1}vaqDio(PA1 zpNmDqs~>)`O>HR~ao=VjwzGDtN+_LFD=IQU_%OfFy4du3{Gu`Q`0HN>yK5g0^n)=9 zTFP{$N;9D@qe5>DZEsm8%@iPT(3?9dZW8%`_{W+t4;-L=@#w2OJ<^>md8X&-vDPVY zXJJA20fSjS@IK;*JQrMMvIO+s0mQIAM+B+hR!u`80>pngYehpVP5A3G^B!*S)vP}# z4M5`uC`=2j)_o}$fc=Mp1Y!5V5zJD~-4tesT09>zi9j*r6`u7{jii1w9I%k?AsC*d&Fr^H0R-eiZeg?G58q5+)tm6YmPUV z2(!O4W|nwjwex|8rVCl8Q3TfH@W{7^K)9KTWN*L(Ub(|JXak;Y_dY#>0iNE>J@*6! zuQJtKvF=F`1=!81T+qhALbA(B%J zXJ%Af8%{HtDNNPA4Y`j3+%_MWoS11j&e==^IHN9{?2>ipX9KO>e1t!%tqn`f!&?-N z+k!9tTGXF$mzyN~D&>tG>k-sX_f4?psC54F%KvjvGa9Xz#JX9f1s6 zo+`eww@*I8Vo1NI^X?0>{94YDQ6c>wmvUbY!%i*&VM-rP`}Am3m)|3p*9suChQ{B; zaWhugyx775?7hz(CkR|W_%Yt-i#!ZUdaBbVWn+QPH{UL!dWyBBJXGGVy+DxvMy&Dl zBv>m6d_N+(`-u;Xa8X)y%fF~CFNY(4irn`9x>kLv@_Ai=1%!zgY)MU_V{#@|pUJKn z_r(CZflHodj&=t9oydlfCFyhZ+RmTeAGy8MIMdd0_e5**3$d=A!=lLI`G+JH6khdM z72V&$r;~7}RDCjy=la=xkI83rNm1f!hefSyCQ-P`RW;_g@qzBDbAQoZc`YSdo&Gjy2Z`U3&&=aqr>YZZQ-u+~Ldkc`k(sBR(0oQUK{97g=F9KdP>nkhM2Bs;; z`;Q5gTuy$JZh*qa?r`HS+@*eEfOil^QiIv*+_`Q`}zbgnrTTKC5HePelWl#^mvd2B3*v(+4}k0TaJIBtWB6)JSTa* z@y1_nbkmUMp8@3rlS>;N_goPJXQ-hc`xk3zLCfch7l94*N4E+2up8d2&y<9L*I^&M z$_$TFtOqG&SU|)E441V6pVXorwyL3k382G1i$JnN*b7|o+^C&htUUtq)G=nUWnRg0 z{h3Q3@gcOLB8}WHn{ArkcFevyXq5gKSuU}U*8Jl~m1Q7FUL?Td0k5UuFCcVCVttKe zm~ZY~?HcHj7jq$Jy*36zPD5FrKj{VEAPB<)Q`aSoKw>Vk90insa(^`2nZT5*Hf5!K zSGR`KfS6bIg)BA7`#XDL2>VH-XsSXDC%EMv9Jg})q_$mVtD34>D{pTJVBE-wvI!+H zftX|0qPhbvA_E2B!QG*1rN@ak&H#oDGny<;TA_>QDgvqYlk_#mC+Z)1KNbn!KeOx3 zC)=F{Aj7TFHM^|abThOs?X9;8^E>@^Xw$m+vO5EX|8TFBIF+sE^+m0Han+0t%KCUz zjDGgiSf2oG!ZCkVQ%&y}g5F8ohE4*&3HJs_##0Ayaoct~?O~J~2H-=9JPS0A`DTP? zfaHP|ZEKWI7fxuDw*sbXDm0Q_%zaz{8jjpA5c8Uxmg-5KK`xtnUeuJU_S95Xn=HW&1PxA>oNl!Y7kxaZN%Koe?*(=6dy< zF=qsI3{!mtw|>=0tN)288e1WYR&oVp>^1XxBzG#!%U99jRv-It z6J)K)&_4*CB%-HQ>r>rnAtURQ?iuXKFYl$s6w$PuE}q7)xGtN692thdljA>aYbq#HJeX- zYF41|!naQ$SA#~-WjkkP;=1_|boq72#wD%eGp5(z9;#szN|l8b9yvObGznGcLu6!c zprkT~Qz^cdVH*C1WucW&+l{{qVfq}}v{RUYZ!{@c6V&|X0zxNeEa6~mqclR~;T}Vn zJX;mDly8X$#9TkwNf6dNzEFAtn|fLlXX^GwoKG;@O7_5XJ$q^Uy4Zyls@n2>43PMD zg4t&VK|shDv}^@%pXP}r)VQng%{$uHCC|6#ksI6x$>DuIKqLc8Y+= z-!r1to*Z|&vH=!3V_bEd_0Om2$m78PuaS+~GsDaKRi@M3{9p&_iOGJhm&>R3X&BJq z?fdd#xE13NBpU7htLj?va%n4Gy1qeK3MjJ zi~o3?Bj1^&jS%(ud!oQ>*zbVpGz(DFiRC7xb_~uw&&0E3QCIv2!hY2@zU4Tu=et?W z4s6tm$=qCU1K5kXDuRc-o3h)gxhb*V&ovqi@A&ceb)EouHVO%2jr0u5JCSWOQRmLe zEYhZCuMD(L^Y0Hi_3Xa5De7BDz4SfL4a%x{?0>vP6Nanf#rkTi? z^pPbS7YC!#b#ovGndklMQzqn`b>fZ37mJ>|$y@ocHI)bcteRX#)xA^aCk8KlfG0dt z-+XzLDdeyFZZ3!As0*Mv@ooBmM*U}eRnO_2mkskpON$M;&u@kFB#+~OF73RJzhsjb z*^6nbwT666kv|l0LM|SOs~O*`Wr9qq<4Omw?(q|WH*wh!FU{^fM9iNUIq_$9lpbTI zM%SQJ($@THutt+&)G`;snjj~)`|L!%y$o>9afRML6d~%=HodgPn}@QA1-;_*Y^^_S z$N6V0skH{q>W2@&guc^?&3?_Ob+lNqsL4!B5x1gCYGlKu2}@bFt-KYi<)Z#--;-Fd zN#7GPbW-NK-l){jlc`mHs+FVtW=H9qC z+N}SaOU_zGM3&}(t1$0RK6O5@<2Xi@-x5Z9Z3MQWS}lJn^mktLM?!)opGo zzGIB8`cT#gu97^L6e>at5PB?QQ6WfUR_|LW6<@*%-leSF`DnJLk+%mA^E;1KdZ%S? zRkY(pk8{8M+^%+%25A+o(ZkQoa+lJK6WTPLO<*8AM2-;4Bp8jLGKHIz@V^r{ju$2R zQe5{_PxL*eVPK<90CtnxnM~iO5jfR~#TS#YM78z=)$6RLw=;iT*$KRazI!8M99QQ2 z#4$;8|I^EtEk;iMZs(Asq_6PTA}TkE=~}+OtvlyoLX1@ymn5%7-W$WO@N#d&e51&Hr2+jPGK)I( ziGJfNW-j8hs2XI!$2{8=4=W?w1J9hNzenAQP~vPu|}R;3;)JE>i}|3yJJ@jbfw^N|u6Z|$AE zp7yjNwjIYv$*&OJoAdFqa5clXZ}dtZN;MLCPUZ?iBC4gax{yyzQEG78PFZW7GY4fL zl!b}p8s5O=@4lW+-GgHymC6~FTrH?yEcilg->j^95-?+_viV`zQ|O}1f%Wllz+T8H zp0L4TsbPgCzVvc0ig-W$9P3!bS)kzN>LH3_s=d4aV=aZ(Q|3Sg9}Wy0y&g~a%j0lQ zw;pK*brS*&C8+JlB>wL~_8sIcso+TMc zVKvp+r@wApwOdyk8+P+aRPz!qJnXIu9vxG4;=P>mV$rw$;_EHno4m@2uIgS**I44U zIO~fBwMGK>T0xcCcuG9>$nvpU!r*(S`^%lc8aETqP;FapHH*!?b0)?ZUc>{StV8lK%PF1yS6LFFuzzb=~u9|9D=8UDrtD*O&pz5`X}b>GYx8oP!iQkbos&n zKPe1xi-rwb%oLpWB-VpgmA)k3+>5I#E|PdR-dph8gQ?RI4=&de?e;gGAAVh~wsRgI zUe10WNirqJUj8oOr13DNl9gBv0LZb!{|lhi={5b$$VsEd2Ap($;}Uw>h&F9%etv4i zW=u+O4y9FrzJVke0&GRzL`z-)c{KRU!vJyf#n@s`_O7xE?VC6?*~JLp#leH(2LLCu zTC5RoVl}DLq=l4Bh>ygveHhxq6xUqL2j)JEwF)4o-xvmSwNQ0;35S~-*s~07hHK8d zCS1DsTn4o3@S$ti84jE`Y+_XvI`U#31l9H$$F+N`vm>;WQ_xUx9HU`*qOZ{S(?9R( zGRKHJsW-OPSN7`z*b!$jgpb4KH{r>2#QX@mfa}4lE*-)cF+erW>l|5A$^V!JR5@+Z zt<}T|C%kuojW=MPCR=IilvUws% z0%K^Y7s6TZzqw|-IuG~mEg7cuPGCVka;0!g=TUHn$;=8NaugkjoEha~M3(B4=kx-p z4dGuH4p0Q~;^BZ#HNRiqu&XAJ^bv-;$UB@QpX-kS;b0bJc#4W%tBtr&T-#kDqr`ZM zuMV5_cTTJMEn|+N5s(z(LJN;vAH%f~6~Ai@Uu3q^*1<*Bu0D=q57j>poP+B%sr&A) zf^ucyp1eT=^XXmk<~>yjBA4O9*@5VvS+QD}SBq&L!`ff=|IU%$BH%HpCXzfRPA*m~ z!IO{=E*f;dksr`7p{5CE{m^1u`W^U6^l)}YA5E{|b5z!4Q+%Kwe)iKN1ik<-HyCq;fsB1`KUU~>qhTV`CJY{^Y5J-l zo#|n?VmJ9ME6}A#CCYf}FYzZlIPgUOjguDT3m@7Jlfh(8Z0jGa^%N3l+uQPt;yMX8 zXL}KjPf~nEF(2opFW6-#TtZF@tkqp&Du*eajUlRF%_NIdmw%~p^BA9eatUCb3)R8u_Rvx}_zoJcD%U%gk98{xzec*gOk zPO;R!UlkP{2#P01ncmPCQv6g{9xS%OJ2RSZw#}VH3t2KZi63!hxeo39Hgpu4t^s`0 z1i;(%m*%>kv|3E>+BI`)Uv0!2w+8`etbhv5%Nn3Fymu+VJF6xc^)pB|zx_Nl$ET1i*km%g;{OJ}1V zVfCf09i)W->;P)=D&-_7{{W5!$|y=^QysNXo*;k|YW|TR2S0rSwD?m7MgRgZrt&!! zD5rM_z(XuK5-2ymO4c+j{963Ew5KGpoz?{b=u~lq#O7FFJCOJQT*gqoC)#C_plbU# z+foAy{}%L&--Q4eD|bQHWWv_2;H3rc@aPi+AOP1#0wlBDOkv?~hkX|DA;PAoDGz+uKOK z%&lm*K{`C4u_c1}c=j7SxC8;b`j?=^vjG8M5L`llY=XDPhD@x&iLwj<9SevEkg0MR zZkL``D5`e+W=cUz6>2X09FsHyBF~fSdX8`r%e1>_(!~pVm)%a z`4j;N0E1yJ0uX_RNR50Cz}#^&=Q+mK9F(3sqp(vg{3P$0j~P#1KmY>3&{&Fqa=&ZW z@ygN{&7Ri|ObZuIR^=2s5dk?2;(ziCSDo#D!fwCZK%=H72tWWB32PA$5s;(csOPx4 zF-=hul=2A3uREp%UW?T*1DgQddkq+!+lsvKp-(Y< z5CQ~1pacRCfD$ND#})*%`w|(*nomHBzbqD*bGK-jcR>II(jfo=NT-rv1)8h^A_G~H zzhL$sb~w^;5b#F;0^qOX9($OE< literal 0 HcmV?d00001 diff --git a/Resources/menu_icon.png b/Resources/menu_icon.png new file mode 100644 index 0000000000000000000000000000000000000000..934c9178b6097c65822421c709560a9930c7aa11 GIT binary patch literal 386 zcmV-|0e$|7P)VpaW6?9Y`vm15^+yAQhkjbRZR=0#N}v&;j$^+{^5on6WdNxqt8W?R#%G z6Mn}Ia0DC!o{cZy5m@!}rfM?Op97b@@Dupx_j|x@FTMcoQ&#~@o&Ybv74STO&wBo_ z1Dtsu06Sj;55Q?J6!5RJjbw5ofb?E_XLTpOOM`m$UO!|q6+pV`o)*nMGVgCZ`)$*A zU@3rFb^D+{R_oK2>6n@qZ`PVBfb?zweK7tD&`b}&vd|5^Z6vD9CS3q$?a$5$>Fo~) zZ$+k8O}YS5aq~oX;|%?@pQ6H_syZ(79HZGjFBH zE|G?A>?{CM62!++bt;CNcn7$)F3+M%(+`2#7!!hI!_;b#prfJsA#n3+{5P6&oqeFW g0t)PR8VIz9|8>y;bpKhp88yV>WRp=I1=9MH?=;jqGLkxkLMfuL^+7WFhI$72aI=A0Z9t+{{zaLoK$}74+Zoz`RicPN?Xl4Z zS&rlwh)=F0yXAD&F&c&tBn@|h<(uIzhf<=;ur z=!&{+%sn?#w@PD|gvLU&jEUDaXR4+%4KZ_Fr}G&r^8! zdeZ*lrQbJw3p;C?wmwLG<=%U8w~fyxt@C&1DfqiMRs0q67k~CEu^VOc78*CZHZAIR zSo6hAvUE+_LXVy81{&AgJcFt~t(`Klm-+Zw{}bB20ejjT?Vg|8HS4Fw)|+NUm0KAf ztC^QvO6cvKC>g%5|8Qmdr-L?*&dWDBcJDhNC--X2FZPh-Cr^s@=$C_%p{J{#%Q~lo FCIEAA5hDNq literal 0 HcmV?d00001 diff --git a/Source/HostStartup.cpp b/Source/HostStartup.cpp new file mode 100644 index 0000000..977ba24 --- /dev/null +++ b/Source/HostStartup.cpp @@ -0,0 +1,67 @@ +#include "../JuceLibraryCode/JuceHeader.h" +#include "IconMenu.hpp" + +#if ! (JUCE_PLUGINHOST_VST || JUCE_PLUGINHOST_VST3 || JUCE_PLUGINHOST_AU) + #error "If you're building the audio plugin host, you probably want to enable VST and/or AU support" +#endif + +class PluginHostApp : public JUCEApplication +{ +public: + PluginHostApp() {} + + void initialise (const String&) override + { + PropertiesFile::Options options; + options.applicationName = getApplicationName(); + options.filenameSuffix = "settings"; + options.osxLibrarySubFolder = "Preferences"; + + appProperties = new ApplicationProperties(); + appProperties->setStorageParameters (options); + + LookAndFeel::setDefaultLookAndFeel (&lookAndFeel); + + mainWindow = new IconMenu(); + Process::setDockIconVisible(false); + + File fileToOpen; + + for (int i = 0; i < getCommandLineParameterArray().size(); ++i) + { + fileToOpen = File::getCurrentWorkingDirectory().getChildFile (getCommandLineParameterArray()[i]); + + if (fileToOpen.existsAsFile()) + break; + } + } + + void shutdown() override + { + mainWindow = nullptr; + appProperties = nullptr; + LookAndFeel::setDefaultLookAndFeel (nullptr); + } + + void systemRequestedQuit() override + { + JUCEApplicationBase::quit(); + } + + const String getApplicationName() override { return "Light Host"; } + const String getApplicationVersion() override { return ProjectInfo::versionString; } + bool moreThanOneInstanceAllowed() override { return false; } + + ApplicationCommandManager commandManager; + ScopedPointer appProperties; + LookAndFeel_V3 lookAndFeel; + +private: + ScopedPointer mainWindow; +}; + +static PluginHostApp& getApp() { return *dynamic_cast(JUCEApplication::getInstance()); } +ApplicationCommandManager& getCommandManager() { return getApp().commandManager; } +ApplicationProperties& getAppProperties() { return *getApp().appProperties; } + +START_JUCE_APPLICATION (PluginHostApp) diff --git a/Source/IconMenu.cpp b/Source/IconMenu.cpp new file mode 100644 index 0000000..384d918 --- /dev/null +++ b/Source/IconMenu.cpp @@ -0,0 +1,284 @@ +// +// IconMenu.cpp +// Light Host +// +// Created by Rolando Islas on 12/26/15. +// +// + +#include "../JuceLibraryCode/JuceHeader.h" +#include "IconMenu.hpp" +#include "PluginWindow.h" + +IconMenu::IconMenu() +{ + // Initiialization + formatManager.addDefaultFormats(); + // Audio device + ScopedPointer savedAudioState (getAppProperties().getUserSettings()->getXmlValue("audioDeviceState")); + deviceManager.initialise(256, 256, savedAudioState, true); + player.setProcessor(&graph); + deviceManager.addAudioCallback(&player); + // Plugins - all + ScopedPointer savedPluginList(getAppProperties().getUserSettings()->getXmlValue("pluginList")); + if (savedPluginList != nullptr) + knownPluginList.recreateFromXml(*savedPluginList); + pluginSortMethod = KnownPluginList::sortByManufacturer; + knownPluginList.addChangeListener(this); + // Plugins - active + ScopedPointer savedPluginListActive(getAppProperties().getUserSettings()->getXmlValue("pluginListActive")); + if (savedPluginListActive != nullptr) + activePluginList.recreateFromXml(*savedPluginListActive); + loadActivePlugins(); + activePluginList.addChangeListener(this); + // Set menu icon + if (exec("defaults read -g AppleInterfaceStyle").compare("Dark") == 1) + setIconImage(ImageFileFormat::loadFrom(BinaryData::menu_icon_white_png, BinaryData::menu_icon_white_pngSize)); + else + setIconImage(ImageFileFormat::loadFrom(BinaryData::menu_icon_png, BinaryData::menu_icon_pngSize)); +}; + +IconMenu::~IconMenu() +{ + +} + +void IconMenu::loadActivePlugins() +{ + graph.clear(); + inputNode = graph.addNode(new AudioProcessorGraph::AudioGraphIOProcessor(AudioProcessorGraph::AudioGraphIOProcessor::audioInputNode), 1); + outputNode = graph.addNode(new AudioProcessorGraph::AudioGraphIOProcessor(AudioProcessorGraph::AudioGraphIOProcessor::audioOutputNode), 2); + if (activePluginList.getNumTypes() == 0) + { + graph.addConnection(1, 0, 2, 0); + graph.addConnection(1, 1, 2, 1); + } + for (int i = 0; i < activePluginList.getNumTypes(); i++) + { + PluginDescription plugin = *activePluginList.getType(i); + String errorMessage; + AudioPluginInstance* instance = formatManager.createPluginInstance(plugin, graph.getSampleRate(), graph.getBlockSize(), errorMessage); + String pluginUid = "pluginState-" + std::to_string(i); + String savedPluginState = getAppProperties().getUserSettings()->getValue(pluginUid); + MemoryBlock savedPluginBinary; + savedPluginBinary.fromBase64Encoding(savedPluginState); + instance->setStateInformation(savedPluginBinary.getData(), savedPluginBinary.getSize()); + graph.addNode(instance, i+3); + // Input to plugin + if (i == 0) + { + graph.addConnection(1, 0, i+3, 0); + graph.addConnection(1, 1, i+3, 1); + } + // Plugin to output + if (i == activePluginList.getNumTypes() - 1) + { + graph.addConnection(i+3, 0, 2, 0); + graph.addConnection(i+3, 1, 2, 1); + } + // Connect previous plugin to current + if (i > 0) + { + graph.addConnection(i+2, 0, i+3, 0); + graph.addConnection(i+2, 1, i+3, 1); + } + } +} + +void IconMenu::changeListenerCallback(ChangeBroadcaster* changed) +{ + if (changed == &knownPluginList) + { + ScopedPointer savedPluginList (knownPluginList.createXml()); + if (savedPluginList != nullptr) + { + getAppProperties().getUserSettings()->setValue ("pluginList", savedPluginList); + getAppProperties().saveIfNeeded(); + } + } + else if (changed == &activePluginList) + { + ScopedPointer savedPluginList (activePluginList.createXml()); + if (savedPluginList != nullptr) + { + getAppProperties().getUserSettings()->setValue ("pluginListActive", savedPluginList); + getAppProperties().saveIfNeeded(); + loadActivePlugins(); + } + } +} + +std::string IconMenu::exec(const char* cmd) +{ + std::shared_ptr pipe(popen(cmd, "r"), pclose); + if (!pipe) return "ERROR"; + char buffer[128]; + std::string result = ""; + while (!feof(pipe.get())) + { + if (fgets(buffer, 128, pipe.get()) != NULL) + result += buffer; + } + return result; +} + +void IconMenu::timerCallback() +{ + stopTimer(); + menu.clear(); + menu.addSectionHeader(JUCEApplication::getInstance()->getApplicationName()); + if (menuIconLeftClicked) { + menu.addItem(1, "Preferences"); + menu.addItem(2, "Reload Plugins"); + menu.addSeparator(); + // Active plugins + for (int i = 0; i < activePluginList.getNumTypes(); i++) + { + PopupMenu options; + options.addItem(i+3, "Edit"); + options.addItem(activePluginList.getNumTypes()+i+3, "Delete"); + // TODO bypass + menu.addSubMenu(activePluginList.getType(i)->name, options); + } + menu.addSeparator(); + // All plugins + knownPluginList.addToMenu(menu, pluginSortMethod); + } + else + { + menu.addItem(1, "Quit"); + } + menu.showMenuAsync(PopupMenu::Options().withTargetComponent(this), ModalCallbackFunction::forComponent(menuInvocationCallback, this)); +} + +void IconMenu::mouseDown(const MouseEvent& e) +{ + Process::setDockIconVisible(true); + Process::makeForegroundProcess(); + menuIconLeftClicked = e.mods.isLeftButtonDown(); + startTimer(50); +} + +void IconMenu::menuInvocationCallback(int id, IconMenu* im) +{ + // Right click + if ((!im->menuIconLeftClicked) && id == 1) + { + im->savePluginStates(); + return JUCEApplication::getInstance()->quit(); + } + // Click elsewhere + if (id == 0 && !PluginWindow::containsActiveWindows()) + Process::setDockIconVisible(false); + // Audio settings + if (id == 1) + im->showAudioSettings(); + // Reload + if (id == 2) + im->reloadPlugins(); + // Plugins + if (id > 2) + { + // Delete plugin + if (id > im->activePluginList.getNumTypes() + 2 && id <= im->activePluginList.getNumTypes() * 2 + 2) + { + im->deletePluginStates(); + im->activePluginList.removeType(id - im->activePluginList.getNumTypes() - 3); + + } + // Add plugin + else if (id > im->activePluginList.getNumTypes() + 2) + { + im->deletePluginStates(); + im->activePluginList.addType(*im->knownPluginList.getType(im->knownPluginList.getIndexChosenByMenu(id))); + } + // Show active plugin GUI + else + { + if (const AudioProcessorGraph::Node::Ptr f = im->graph.getNodeForId(id)) + if (PluginWindow* const w = PluginWindow::getWindowFor(f, PluginWindow::Normal)) + w->toFront(true); + } + // Update menu + im->startTimer(50); + } +} + +void IconMenu::deletePluginStates() +{ + for (int i = 0; i < activePluginList.getNumTypes(); i++) + { + String pluginUid = "pluginState-" + std::to_string(i); + getAppProperties().getUserSettings()->removeValue(pluginUid); + getAppProperties().saveIfNeeded(); + } +} + +void IconMenu::savePluginStates() +{ + for (int i = 0; i < activePluginList.getNumTypes(); i++) + { + AudioProcessor& processor = *graph.getNodeForId(i+3)->getProcessor(); + String pluginUid = "pluginState-" + std::to_string(i); + MemoryBlock savedStateBinary; + processor.getStateInformation(savedStateBinary); + ScopedPointer savedStateXml(XmlElement::createTextElement(savedStateBinary.toBase64Encoding())); + getAppProperties().getUserSettings()->setValue(pluginUid, savedStateBinary.toBase64Encoding()); + getAppProperties().saveIfNeeded(); + } +} + +void IconMenu::showAudioSettings() +{ + AudioDeviceSelectorComponent audioSettingsComp (deviceManager, 0, 256, 0, 256, false, false, true, true); + audioSettingsComp.setSize(500, 450); + + DialogWindow::LaunchOptions o; + o.content.setNonOwned(&audioSettingsComp); + o.dialogTitle = "Audio Settings"; + o.componentToCentreAround = this; + o.dialogBackgroundColour = Colour::fromRGB(236, 236, 236); + o.escapeKeyTriggersCloseButton = true; + o.useNativeTitleBar = true; + o.resizable = false; + + o.runModal(); + + ScopedPointer audioState(deviceManager.createStateXml()); + + getAppProperties().getUserSettings()->setValue("audioDeviceState", audioState); + getAppProperties().getUserSettings()->saveIfNeeded(); +} + +void IconMenu::reloadPlugins() +{ + NativeMessageBox::showOkCancelBox(AlertWindow::AlertIconType::InfoIcon, "Reload Plugins?", "Confirm scan and load of any new or updated plugins.", this, ModalCallbackFunction::forComponent(doReload, this)); +} + +void IconMenu::doReload(int id, IconMenu* im) +{ + // Canceled + if (id == 0) + return Process::setDockIconVisible(false); + // Scan + const File deadMansPedalFile (getAppProperties().getUserSettings()->getFile().getSiblingFile("RecentlyCrashedPluginsList")); + String pluginName; + for (int i = 0; i < im->formatManager.getNumFormats(); i++) + { + im->scanner = new PluginDirectoryScanner(im->knownPluginList, *im->formatManager.getFormat(i), im->formatManager.getFormat(i)->getDefaultLocationsToSearch(), true, deadMansPedalFile); + while (im->scanner->scanNextFile(true, pluginName)) { } + } + // Remove plugins without inputs and/or outputs + std::vector removeIndex; + for (int i = 0; i < im->knownPluginList.getNumTypes(); i++) + { + PluginDescription* plugin = im->knownPluginList.getType(i); + if (plugin->numInputChannels < 2 || plugin->numOutputChannels < 2) + removeIndex.push_back(i); + } + for (int i = 0; i < removeIndex.size(); i++) + im->knownPluginList.removeType(removeIndex[i] - i); + // Finish + NativeMessageBox::showMessageBox(AlertWindow::AlertIconType::InfoIcon, "Completed", "Plugins have been refreshed."); + Process::setDockIconVisible(false); +} \ No newline at end of file diff --git a/Source/IconMenu.hpp b/Source/IconMenu.hpp new file mode 100644 index 0000000..f1dcb39 --- /dev/null +++ b/Source/IconMenu.hpp @@ -0,0 +1,48 @@ +// +// IconMenu.hpp +// Light Host +// +// Created by Rolando Islas on 12/26/15. +// +// + +#ifndef IconMenu_hpp +#define IconMenu_hpp + +ApplicationProperties& getAppProperties(); + +class IconMenu : public SystemTrayIconComponent, private Timer, public ChangeListener +{ +public: + IconMenu(); + ~IconMenu(); + void mouseDown(const MouseEvent&); + static void menuInvocationCallback(int id, IconMenu*); + static void doReload(int id, IconMenu*); + void changeListenerCallback(ChangeBroadcaster* changed); +private: + std::string exec(const char* cmd); + void timerCallback(); + void reloadPlugins(); + void showAudioSettings(); + void loadActivePlugins(); + void savePluginStates(); + void deletePluginStates(); + + AudioDeviceManager deviceManager; + AudioPluginFormatManager formatManager; + KnownPluginList knownPluginList; + KnownPluginList activePluginList; + KnownPluginList::SortMethod pluginSortMethod; + PopupMenu menu; + ScopedPointer scanner; + bool menuIconLeftClicked; + AudioProcessorGraph graph; + AudioProcessorPlayer player; + AudioProcessorGraph::Node *inputNode; + AudioProcessorGraph::Node *outputNode; + + class ScanThread; +}; + +#endif /* IconMenu_hpp */ diff --git a/Source/PluginWindow.cpp b/Source/PluginWindow.cpp new file mode 100644 index 0000000..79f8974 --- /dev/null +++ b/Source/PluginWindow.cpp @@ -0,0 +1,190 @@ +#include "../JuceLibraryCode/JuceHeader.h" +#include "PluginWindow.h" + +class PluginWindow; +static Array activePluginWindows; + +PluginWindow::PluginWindow (Component* const pluginEditor, + AudioProcessorGraph::Node* const o, + WindowFormatType t) + : DocumentWindow (pluginEditor->getName(), Colours::lightgrey, + DocumentWindow::minimiseButton | DocumentWindow::closeButton), + owner (o), + type (t) +{ + setSize (400, 300); + setUsingNativeTitleBar(true); + setContentOwned (pluginEditor, true); + + setTopLeftPosition (owner->properties.getWithDefault (getLastXProp (type), Random::getSystemRandom().nextInt (500)), + owner->properties.getWithDefault (getLastYProp (type), Random::getSystemRandom().nextInt (500))); + + owner->properties.set (getOpenProp (type), true); + + setVisible (true); + + activePluginWindows.add (this); + +} + +void PluginWindow::closeCurrentlyOpenWindowsFor (const uint32 nodeId) +{ + for (int i = activePluginWindows.size(); --i >= 0;) + if (activePluginWindows.getUnchecked(i)->owner->nodeId == nodeId) + delete activePluginWindows.getUnchecked (i); +} + +void PluginWindow::closeAllCurrentlyOpenWindows() +{ + if (activePluginWindows.size() > 0) + { + for (int i = activePluginWindows.size(); --i >= 0;) + delete activePluginWindows.getUnchecked (i); + + Component dummyModalComp; + dummyModalComp.enterModalState(); + MessageManager::getInstance()->runDispatchLoopUntil (50); + } +} + +bool PluginWindow::containsActiveWindows() +{ + return activePluginWindows.size() > 0; +} + +//============================================================================== +class ProcessorProgramPropertyComp : public PropertyComponent, + private AudioProcessorListener +{ +public: + ProcessorProgramPropertyComp (const String& name, AudioProcessor& p, int index_) + : PropertyComponent (name), + owner (p), + index (index_) + { + owner.addListener (this); + } + + ~ProcessorProgramPropertyComp() + { + owner.removeListener (this); + } + + void refresh() { } + virtual void audioProcessorChanged (AudioProcessor*) { } + virtual void audioProcessorParameterChanged(AudioProcessor* processor, int, float) { } + +private: + AudioProcessor& owner; + const int index; + + JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ProcessorProgramPropertyComp) +}; + +class ProgramAudioProcessorEditor : public AudioProcessorEditor +{ +public: + ProgramAudioProcessorEditor (AudioProcessor* const p) + : AudioProcessorEditor (p) + { + jassert (p != nullptr); + setOpaque (true); + + addAndMakeVisible (panel); + + Array programs; + + const int numPrograms = p->getNumPrograms(); + int totalHeight = 0; + + for (int i = 0; i < numPrograms; ++i) + { + String name (p->getProgramName (i).trim()); + + if (name.isEmpty()) + name = "Unnamed"; + + ProcessorProgramPropertyComp* const pc = new ProcessorProgramPropertyComp (name, *p, i); + programs.add (pc); + totalHeight += pc->getPreferredHeight(); + } + + panel.addProperties (programs); + + setSize (400, jlimit (25, 400, totalHeight)); + } + + void paint (Graphics& g) + { + g.fillAll (Colours::grey); + } + + void resized() + { + panel.setBounds (getLocalBounds()); + } + +private: + PropertyPanel panel; + + JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ProgramAudioProcessorEditor) +}; + +//============================================================================== +PluginWindow* PluginWindow::getWindowFor (AudioProcessorGraph::Node* const node, + WindowFormatType type) +{ + jassert (node != nullptr); + + for (int i = activePluginWindows.size(); --i >= 0;) + if (activePluginWindows.getUnchecked(i)->owner == node + && activePluginWindows.getUnchecked(i)->type == type) + return activePluginWindows.getUnchecked(i); + + AudioProcessor* processor = node->getProcessor(); + AudioProcessorEditor* ui = nullptr; + + if (type == Normal) + { + ui = processor->createEditorIfNeeded(); + + if (ui == nullptr) + type = Generic; + } + + if (ui == nullptr) + { + if (type == Generic || type == Parameters) + ui = new GenericAudioProcessorEditor (processor); + else if (type == Programs) + ui = new ProgramAudioProcessorEditor (processor); + } + + if (ui != nullptr) + { + if (AudioPluginInstance* const plugin = dynamic_cast (processor)) + ui->setName (plugin->getName()); + + return new PluginWindow (ui, node, type); + } + + return nullptr; +} + +PluginWindow::~PluginWindow() +{ + activePluginWindows.removeFirstMatchingValue (this); + clearContentComponent(); +} + +void PluginWindow::moved() +{ + owner->properties.set (getLastXProp (type), getX()); + owner->properties.set (getLastYProp (type), getY()); +} + +void PluginWindow::closeButtonPressed() +{ + owner->properties.set (getOpenProp (type), false); + delete this; +} diff --git a/Source/PluginWindow.h b/Source/PluginWindow.h new file mode 100644 index 0000000..fb7ea84 --- /dev/null +++ b/Source/PluginWindow.h @@ -0,0 +1,56 @@ +#ifndef PluginWindow_h +#define PluginWindow_h + +ApplicationProperties& getAppProperties(); + +class PluginWindow : public DocumentWindow +{ +public: + enum WindowFormatType + { + Normal = 0, + Generic, + Programs, + Parameters, + NumTypes + }; + + PluginWindow (Component* pluginEditor, AudioProcessorGraph::Node*, WindowFormatType); + ~PluginWindow(); + + static PluginWindow* getWindowFor (AudioProcessorGraph::Node*, WindowFormatType); + + static void closeCurrentlyOpenWindowsFor (const uint32 nodeId); + static void closeAllCurrentlyOpenWindows(); + static bool containsActiveWindows(); + + void moved() override; + void closeButtonPressed() override; + +private: + AudioProcessorGraph::Node* owner; + WindowFormatType type; + + float getDesktopScaleFactor() const override { return 1.0f; } + + JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (PluginWindow) +}; + +inline String toString (PluginWindow::WindowFormatType type) +{ + switch (type) + { + case PluginWindow::Normal: return "Normal"; + case PluginWindow::Generic: return "Generic"; + case PluginWindow::Programs: return "Programs"; + case PluginWindow::Parameters: return "Parameters"; + default: return String(); + } +} + +inline String getLastXProp (PluginWindow::WindowFormatType type) { return "uiLastX_" + toString (type); } +inline String getLastYProp (PluginWindow::WindowFormatType type) { return "uiLastY_" + toString (type); } +inline String getOpenProp (PluginWindow::WindowFormatType type) { return "uiopen_" + toString (type); } + + +#endif /* PluginWindow_hpp */ diff --git a/gpl.txt b/gpl.txt new file mode 100644 index 0000000..23cb790 --- /dev/null +++ b/gpl.txt @@ -0,0 +1,339 @@ + GNU GENERAL PUBLIC LICENSE + Version 2, June 1991 + + Copyright (C) 1989, 1991 Free Software Foundation, Inc., + 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The licenses for most software are designed to take away your +freedom to share and change it. By contrast, the GNU General Public +License is intended to guarantee your freedom to share and change free +software--to make sure the software is free for all its users. This +General Public License applies to most of the Free Software +Foundation's software and to any other program whose authors commit to +using it. (Some other Free Software Foundation software is covered by +the GNU Lesser General Public License instead.) You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +this service if you wish), that you receive source code or can get it +if you want it, that you can change the software or use pieces of it +in new free programs; and that you know you can do these things. + + To protect your rights, we need to make restrictions that forbid +anyone to deny you these rights or to ask you to surrender the rights. +These restrictions translate to certain responsibilities for you if you +distribute copies of the software, or if you modify it. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must give the recipients all the rights that +you have. You must make sure that they, too, receive or can get the +source code. And you must show them these terms so they know their +rights. + + We protect your rights with two steps: (1) copyright the software, and +(2) offer you this license which gives you legal permission to copy, +distribute and/or modify the software. + + Also, for each author's protection and ours, we want to make certain +that everyone understands that there is no warranty for this free +software. If the software is modified by someone else and passed on, we +want its recipients to know that what they have is not the original, so +that any problems introduced by others will not reflect on the original +authors' reputations. + + Finally, any free program is threatened constantly by software +patents. We wish to avoid the danger that redistributors of a free +program will individually obtain patent licenses, in effect making the +program proprietary. To prevent this, we have made it clear that any +patent must be licensed for everyone's free use or not licensed at all. + + The precise terms and conditions for copying, distribution and +modification follow. + + GNU GENERAL PUBLIC LICENSE + TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION + + 0. This License applies to any program or other work which contains +a notice placed by the copyright holder saying it may be distributed +under the terms of this General Public License. The "Program", below, +refers to any such program or work, and a "work based on the Program" +means either the Program or any derivative work under copyright law: +that is to say, a work containing the Program or a portion of it, +either verbatim or with modifications and/or translated into another +language. (Hereinafter, translation is included without limitation in +the term "modification".) Each licensee is addressed as "you". + +Activities other than copying, distribution and modification are not +covered by this License; they are outside its scope. The act of +running the Program is not restricted, and the output from the Program +is covered only if its contents constitute a work based on the +Program (independent of having been made by running the Program). +Whether that is true depends on what the Program does. + + 1. You may copy and distribute verbatim copies of the Program's +source code as you receive it, in any medium, provided that you +conspicuously and appropriately publish on each copy an appropriate +copyright notice and disclaimer of warranty; keep intact all the +notices that refer to this License and to the absence of any warranty; +and give any other recipients of the Program a copy of this License +along with the Program. + +You may charge a fee for the physical act of transferring a copy, and +you may at your option offer warranty protection in exchange for a fee. + + 2. You may modify your copy or copies of the Program or any portion +of it, thus forming a work based on the Program, and copy and +distribute such modifications or work under the terms of Section 1 +above, provided that you also meet all of these conditions: + + a) You must cause the modified files to carry prominent notices + stating that you changed the files and the date of any change. + + b) You must cause any work that you distribute or publish, that in + whole or in part contains or is derived from the Program or any + part thereof, to be licensed as a whole at no charge to all third + parties under the terms of this License. + + c) If the modified program normally reads commands interactively + when run, you must cause it, when started running for such + interactive use in the most ordinary way, to print or display an + announcement including an appropriate copyright notice and a + notice that there is no warranty (or else, saying that you provide + a warranty) and that users may redistribute the program under + these conditions, and telling the user how to view a copy of this + License. (Exception: if the Program itself is interactive but + does not normally print such an announcement, your work based on + the Program is not required to print an announcement.) + +These requirements apply to the modified work as a whole. If +identifiable sections of that work are not derived from the Program, +and can be reasonably considered independent and separate works in +themselves, then this License, and its terms, do not apply to those +sections when you distribute them as separate works. But when you +distribute the same sections as part of a whole which is a work based +on the Program, the distribution of the whole must be on the terms of +this License, whose permissions for other licensees extend to the +entire whole, and thus to each and every part regardless of who wrote it. + +Thus, it is not the intent of this section to claim rights or contest +your rights to work written entirely by you; rather, the intent is to +exercise the right to control the distribution of derivative or +collective works based on the Program. + +In addition, mere aggregation of another work not based on the Program +with the Program (or with a work based on the Program) on a volume of +a storage or distribution medium does not bring the other work under +the scope of this License. + + 3. You may copy and distribute the Program (or a work based on it, +under Section 2) in object code or executable form under the terms of +Sections 1 and 2 above provided that you also do one of the following: + + a) Accompany it with the complete corresponding machine-readable + source code, which must be distributed under the terms of Sections + 1 and 2 above on a medium customarily used for software interchange; or, + + b) Accompany it with a written offer, valid for at least three + years, to give any third party, for a charge no more than your + cost of physically performing source distribution, a complete + machine-readable copy of the corresponding source code, to be + distributed under the terms of Sections 1 and 2 above on a medium + customarily used for software interchange; or, + + c) Accompany it with the information you received as to the offer + to distribute corresponding source code. (This alternative is + allowed only for noncommercial distribution and only if you + received the program in object code or executable form with such + an offer, in accord with Subsection b above.) + +The source code for a work means the preferred form of the work for +making modifications to it. For an executable work, complete source +code means all the source code for all modules it contains, plus any +associated interface definition files, plus the scripts used to +control compilation and installation of the executable. However, as a +special exception, the source code distributed need not include +anything that is normally distributed (in either source or binary +form) with the major components (compiler, kernel, and so on) of the +operating system on which the executable runs, unless that component +itself accompanies the executable. + +If distribution of executable or object code is made by offering +access to copy from a designated place, then offering equivalent +access to copy the source code from the same place counts as +distribution of the source code, even though third parties are not +compelled to copy the source along with the object code. + + 4. You may not copy, modify, sublicense, or distribute the Program +except as expressly provided under this License. Any attempt +otherwise to copy, modify, sublicense or distribute the Program is +void, and will automatically terminate your rights under this License. +However, parties who have received copies, or rights, from you under +this License will not have their licenses terminated so long as such +parties remain in full compliance. + + 5. You are not required to accept this License, since you have not +signed it. However, nothing else grants you permission to modify or +distribute the Program or its derivative works. These actions are +prohibited by law if you do not accept this License. Therefore, by +modifying or distributing the Program (or any work based on the +Program), you indicate your acceptance of this License to do so, and +all its terms and conditions for copying, distributing or modifying +the Program or works based on it. + + 6. Each time you redistribute the Program (or any work based on the +Program), the recipient automatically receives a license from the +original licensor to copy, distribute or modify the Program subject to +these terms and conditions. You may not impose any further +restrictions on the recipients' exercise of the rights granted herein. +You are not responsible for enforcing compliance by third parties to +this License. + + 7. If, as a consequence of a court judgment or allegation of patent +infringement or for any other reason (not limited to patent issues), +conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot +distribute so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you +may not distribute the Program at all. For example, if a patent +license would not permit royalty-free redistribution of the Program by +all those who receive copies directly or indirectly through you, then +the only way you could satisfy both it and this License would be to +refrain entirely from distribution of the Program. + +If any portion of this section is held invalid or unenforceable under +any particular circumstance, the balance of the section is intended to +apply and the section as a whole is intended to apply in other +circumstances. + +It is not the purpose of this section to induce you to infringe any +patents or other property right claims or to contest validity of any +such claims; this section has the sole purpose of protecting the +integrity of the free software distribution system, which is +implemented by public license practices. Many people have made +generous contributions to the wide range of software distributed +through that system in reliance on consistent application of that +system; it is up to the author/donor to decide if he or she is willing +to distribute software through any other system and a licensee cannot +impose that choice. + +This section is intended to make thoroughly clear what is believed to +be a consequence of the rest of this License. + + 8. If the distribution and/or use of the Program is restricted in +certain countries either by patents or by copyrighted interfaces, the +original copyright holder who places the Program under this License +may add an explicit geographical distribution limitation excluding +those countries, so that distribution is permitted only in or among +countries not thus excluded. In such case, this License incorporates +the limitation as if written in the body of this License. + + 9. The Free Software Foundation may publish revised and/or new versions +of the General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + +Each version is given a distinguishing version number. If the Program +specifies a version number of this License which applies to it and "any +later version", you have the option of following the terms and conditions +either of that version or of any later version published by the Free +Software Foundation. If the Program does not specify a version number of +this License, you may choose any version ever published by the Free Software +Foundation. + + 10. If you wish to incorporate parts of the Program into other free +programs whose distribution conditions are different, write to the author +to ask for permission. For software which is copyrighted by the Free +Software Foundation, write to the Free Software Foundation; we sometimes +make exceptions for this. Our decision will be guided by the two goals +of preserving the free status of all derivatives of our free software and +of promoting the sharing and reuse of software generally. + + NO WARRANTY + + 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY +FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN +OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES +PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED +OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF +MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS +TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE +PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, +REPAIR OR CORRECTION. + + 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR +REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, +INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING +OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED +TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY +YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER +PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE +POSSIBILITY OF SUCH DAMAGES. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +convey the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + {description} + Copyright (C) {year} {fullname} + + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 2 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License along + with this program; if not, write to the Free Software Foundation, Inc., + 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + +Also add information on how to contact you by electronic and paper mail. + +If the program is interactive, make it output a short notice like this +when it starts in an interactive mode: + + Gnomovision version 69, Copyright (C) year name of author + Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'. + This is free software, and you are welcome to redistribute it + under certain conditions; type `show c' for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License. Of course, the commands you use may +be called something other than `show w' and `show c'; they could even be +mouse-clicks or menu items--whatever suits your program. + +You should also get your employer (if you work as a programmer) or your +school, if any, to sign a "copyright disclaimer" for the program, if +necessary. Here is a sample; alter the names: + + Yoyodyne, Inc., hereby disclaims all copyright interest in the program + `Gnomovision' (which makes passes at compilers) written by James Hacker. + + {signature of Ty Coon}, 1 April 1989 + Ty Coon, President of Vice + +This General Public License does not permit incorporating your program into +proprietary programs. If your program is a subroutine library, you may +consider it more useful to permit linking proprietary applications with the +library. If this is what you want to do, use the GNU Lesser General +Public License instead of this License. diff --git a/license b/license new file mode 100644 index 0000000..6b5ed54 --- /dev/null +++ b/license @@ -0,0 +1,16 @@ +Copyright (C) 2015 Rolando Islas + +This file is part of Light Host. + +Light Host is free software: you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation, either version 2 of the License, or +(at your option) any later version. + +Light Host is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +You should have received a copy of the GNU General Public License +along with Light Host. If not, see . \ No newline at end of file diff --git a/readme.md b/readme.md new file mode 100644 index 0000000..578038f --- /dev/null +++ b/readme.md @@ -0,0 +1,4 @@ +Light Host +--- + +A simple VST/AU host for OS X that sits in the menubar. \ No newline at end of file diff --git a/third_party b/third_party new file mode 100644 index 0000000..27a7abe --- /dev/null +++ b/third_party @@ -0,0 +1,14 @@ +The following are third-party tools used in Light Host. + +Plug Icon +Website: Bogdan Rosu (http://www.bogdanrosu.com) + www.flaticon.com +License: CC BY 3.0 + +JUCE +Website: http://www.juce.com +License: GPLv3 + +Steinberg VST Plug-In SDK +Website: http://www.steinberg.net/en/company/developers.html +License: Steinberg VST Plug-In SDK Licensing Agreement \ No newline at end of file