diff --git a/README.md b/README.md
index ca04b02..634cf23 100644
--- a/README.md
+++ b/README.md
@@ -41,7 +41,7 @@ Alternatively, you can link to a `p5.brush.js` file hosted online. All versions
```html
-
+
```
### Install with NPM and other modular-based apps
diff --git a/dist/p5.brush.js b/dist/p5.brush.js
index d2d076c..644bd0f 100644
--- a/dist/p5.brush.js
+++ b/dist/p5.brush.js
@@ -1 +1 @@
-!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?e(exports):"function"==typeof define&&define.amd?define(["exports"],e):e((t="undefined"!=typeof globalThis?globalThis:t||self).brush={})}(this,(function(t){"use strict";let e,i=!1,s=!1,n=!1;function r(t=!1){let r=!(!s||!t)&&n;i&&o(!1),!t&&s&&(t=n),e=t||window.self,i||(i=!0,w.create(),_(e.width/250)),v.load(r)}function o(t=!0){i&&(v.masks[0].remove(),v.masks[0]=null,v.masks[1].remove(),v.masks[1]=null,v.masks[2].remove(),v.masks[2]=null,t&&brush.load())}function a(){i||r()}let h=new Math.seedrandom(Math.random());const l={random(t=0,e=1){return 1===arguments.length?this.map(h(),0,1,0,t):this.map(h(),0,1,t,e)},randInt(t,e){return Math.floor(this.random(t,e))},gaussian(t=0,e=1){const i=1-l.random(),s=l.random();return Math.sqrt(-2*Math.log(i))*Math.cos(2*Math.PI*s)*e+t},weightedRand(t){let e,i,s=[];for(e in t)for(i=0;i<10*t[e];i++)s.push(e);return s[Math.floor(h()*s.length)]},map(t,e,i,s,n,r=!1){let o=s+(t-e)/(i-e)*(n-s);return r?sMath.max(Math.min(t,i),e),cos(t){return this.c[Math.floor((t%360+360)%360*4)]},sin(t){return this.s[Math.floor((t%360+360)%360*4)]},isPrecalculationDone:!1,preCalculation(){if(this.isPrecalculationDone)return;const t=1440,e=2*Math.PI/t;this.c=new Float64Array(t),this.s=new Float64Array(t);for(let i=0;i!isNaN(t),toDegrees:t=>(("radians"===e.angleMode()?180*t/Math.PI:t)%360+360)%360,dist:(t,e,i,s)=>Math.hypot(i-t,s-e)};function c(t,e,i,s,n=!1){let r=t.x,o=t.y,a=e.x,h=e.y,l=i.x,c=i.y,m=s.x,d=s.y;if(r===a&&o===h||l===m&&c===d)return!1;let u=a-r,p=h-o,f=m-l,v=d-c,g=v*u-f*p;if(0===g)return!1;let x=(f*(o-c)-v*(r-l))/g,y=(u*(o-c)-p*(r-l))/g;return!(!n&&(y<0||y>1))&&{x:r+x*u,y:o+x*p}}function m(t,e,i,s){return(Math.atan2(-(s-e),i-t)*(180/Math.PI)%360+360)%360}l.preCalculation();const d={field:{},stroke:{},hatch:{},fill:{},others:{}};const u={translation:[0,0],rotation:0,trans(){return this.translation=[e._renderer.uMVMatrix.mat4[12],e._renderer.uMVMatrix.mat4[13]],this.translation}};let p=1;function f(t){p*=t}const v={loaded:!1,isBlending:!1,isCaching:!0,currentColor:new Float32Array(3),load(t){this.type=s&&!t?0:t?2:1,this.masks=[];for(let i=0;i<3;i++)switch(this.type){case 0:this.masks[i]=e.createGraphics(e.width,e.height,1==i?e.WEBGL:e.P2D);break;case 1:this.masks[i]=createGraphics(e.width,e.height,1==i?WEBGL:P2D);break;case 2:this.masks[i]=t.createGraphics(t.width,t.height,1==i?t.WEBGL:t.P2D)}for(let t of this.masks)t.pixelDensity(e.pixelDensity()),t.clear(),t.angleMode(e.DEGREES),t.noSmooth();this.shader=e.createShader(this.vert,this.frag),v.loaded=!0},getPigment(t){let e=t.levels,i=new Float32Array(3);return i[0]=e[0]/255,i[1]=e[1]/255,i[2]=e[2]/255,i},color1:new Float32Array(3),color2:new Float32Array(3),blending1:!1,blending2:!1,blend(t=!1,i=!1,s=!1){if(this.isBlending=s?this.blending1:this.blending2,this.currentColor=s?this.color1:this.color2,!this.isBlending)if(t)this.currentColor=this.getPigment(t),s?(this.blending1=!0,this.color1=this.currentColor):(this.blending2=!0,this.color2=this.currentColor);else if(i)return void(s||g());if((t?this.getPigment(t):this.currentColor).toString()!==this.currentColor.toString()||i||!this.isCaching){if(g(),this.isBlending){e.push(),e.translate(-u.trans()[0],-u.trans()[1]),e.shader(this.shader),this.shader.setUniform("addColor",this.currentColor),this.shader.setUniform("source",e._renderer),this.shader.setUniform("active",v.watercolor),this.shader.setUniform("random",[l.random(),l.random(),l.random()]);let t=s?this.masks[1]:this.masks[0];this.shader.setUniform("mask",t),e.fill(0,0,0,0),e.noStroke(),e.rect(-e.width/2,-e.height/2,e.width,e.height),e.pop(),t.clear()}i||(this.currentColor=this.getPigment(t),s?this.color1=this.currentColor:this.color2=this.currentColor)}i&&(this.isBlending=!1,s?this.blending1=this.isBlending:this.blending2=this.isBlending)},vert:"precision highp float;attribute vec3 aPosition;attribute vec2 aTexCoord;uniform mat4 uModelViewMatrix,uProjectionMatrix;varying vec2 vVertTexCoord;void main(){gl_Position=uProjectionMatrix*uModelViewMatrix*vec4(aPosition,1);vVertTexCoord=aTexCoord;}",frag:"precision highp float;varying vec2 vVertTexCoord;uniform sampler2D source,mask;uniform vec4 addColor;uniform vec3 random;uniform bool active;\n #ifndef SPECTRAL\n #define SPECTRAL\n float x(float v){return v<.04045?v/12.92:pow((v+.055)/1.055,2.4);}float v(float v){return v<.0031308?v*12.92:1.055*pow(v,1./2.4)-.055;}vec3 m(vec3 v){return vec3(x(v[0]),x(v[1]),x(v[2]));}vec3 f(vec3 f){return clamp(vec3(v(f[0]),v(f[1]),v(f[2])),0.,1.);}void f(vec3 v,out float m,out float f,out float x,out float y,out float z,out float i,out float r){m=min(v.x,min(v.y,v.z));v-=m;f=min(v.y,v.z);x=min(v.x,v.z);y=min(v.x,v.y);z=min(max(0.,v.x-v.z),max(0.,v.x-v.y));i=min(max(0.,v.y-v.z),max(0.,v.y-v.x));r=min(max(0.,v.z-v.y),max(0.,v.z-v.x));}void f(vec3 v,inout float i[38]){float x,y,d,z,o,m,e;f(v,x,y,d,z,o,m,e);i[0]=max(1e-4,x+y*.96853629+d*.51567122+z*.02055257+o*.03147571+m*.49108579+e*.97901834);i[1]=max(1e-4,x+y*.96855103+d*.5401552+z*.02059936+o*.03146636+m*.46944057+e*.97901649);i[2]=max(1e-4,x+y*.96859338+d*.62645502+z*.02062723+o*.03140624+m*.4016578+e*.97901118);i[3]=max(1e-4,x+y*.96877345+d*.75595012+z*.02073387+o*.03119611+m*.2449042+e*.97892146);i[4]=max(1e-4,x+y*.96942204+d*.92826996+z*.02114202+o*.03053888+m*.0682688+e*.97858555);i[5]=max(1e-4,x+y*.97143709+d*.97223624+z*.02233154+o*.02856855+m*.02732883+e*.97743705);i[6]=max(1e-4,x+y*.97541862+d*.98616174+z*.02556857+o*.02459485+m*.013606+e*.97428075);i[7]=max(1e-4,x+y*.98074186+d*.98955255+z*.03330189+o*.0192952+m*.01000187+e*.96663223);i[8]=max(1e-4,x+y*.98580992+d*.98676237+z*.05185294+o*.01423112+m*.01284127+e*.94822893);i[9]=max(1e-4,x+y*.98971194+d*.97312575+z*.10087639+o*.01033111+m*.02636635+e*.89937713);i[10]=max(1e-4,x+y*.99238027+d*.91944277+z*.24000413+o*.00765876+m*.07058713+e*.76070164);i[11]=max(1e-4,x+y*.99409844+d*.32564851+z*.53589066+o*.00593693+m*.70421692+e*.4642044);i[12]=max(1e-4,x+y*.995172+d*.13820628+z*.79874659+o*.00485616+m*.85473994+e*.20123039);i[13]=max(1e-4,x+y*.99576545+d*.05015143+z*.91186529+o*.00426186+m*.95081565+e*.08808402);i[14]=max(1e-4,x+y*.99593552+d*.02912336+z*.95399623+o*.00409039+m*.9717037+e*.04592894);i[15]=max(1e-4,x+y*.99564041+d*.02421691+z*.97137099+o*.00438375+m*.97651888+e*.02860373);i[16]=max(1e-4,x+y*.99464769+d*.02660696+z*.97939505+o*.00537525+m*.97429245+e*.02060067);i[17]=max(1e-4,x+y*.99229579+d*.03407586+z*.98345207+o*.00772962+m*.97012917+e*.01656701);i[18]=max(1e-4,x+y*.98638762+d*.04835936+z*.98553736+o*.0136612+m*.9425863+e*.01451549);i[19]=max(1e-4,x+y*.96829712+d*.0001172+z*.98648905+o*.03181352+m*.99989207+e*.01357964);i[20]=max(1e-4,x+y*.89228016+d*8.554e-5+z*.98674535+o*.10791525+m*.99989891+e*.01331243);i[21]=max(1e-4,x+y*.53740239+d*.85267882+z*.98657555+o*.46249516+m*.13823139+e*.01347661);i[22]=max(1e-4,x+y*.15360445+d*.93188793+z*.98611877+o*.84604333+m*.06968113+e*.01387181);i[23]=max(1e-4,x+y*.05705719+d*.94810268+z*.98559942+o*.94275572+m*.05628787+e*.01435472);i[24]=max(1e-4,x+y*.03126539+d*.94200977+z*.98507063+o*.96860996+m*.06111561+e*.01479836);i[25]=max(1e-4,x+y*.02205445+d*.91478045+z*.98460039+o*.97783966+m*.08987709+e*.0151525);i[26]=max(1e-4,x+y*.01802271+d*.87065445+z*.98425301+o*.98187757+m*.13656016+e*.01540513);i[27]=max(1e-4,x+y*.0161346+d*.78827548+z*.98403909+o*.98377315+m*.22169624+e*.01557233);i[28]=max(1e-4,x+y*.01520947+d*.65738359+z*.98388535+o*.98470202+m*.32176956+e*.0156571);i[29]=max(1e-4,x+y*.01475977+d*.59909403+z*.98376116+o*.98515481+m*.36157329+e*.01571025);i[30]=max(1e-4,x+y*.01454263+d*.56817268+z*.98368246+o*.98537114+m*.4836192+e*.01571916);i[31]=max(1e-4,x+y*.01444459+d*.54031997+z*.98365023+o*.98546685+m*.46488579+e*.01572133);i[32]=max(1e-4,x+y*.01439897+d*.52110241+z*.98361309+o*.98550011+m*.47440306+e*.01572502);i[33]=max(1e-4,x+y*.0143762+d*.51041094+z*.98357259+o*.98551031+m*.4857699+e*.01571717);i[34]=max(1e-4,x+y*.01436343+d*.50526577+z*.98353856+o*.98550741+m*.49267971+e*.01571905);i[35]=max(1e-4,x+y*.01435687+d*.5025508+z*.98351247+o*.98551323+m*.49625685+e*.01571059);i[36]=max(1e-4,x+y*.0143537+d*.50126452+z*.98350101+o*.98551563+m*.49807754+e*.01569728);i[37]=max(1e-4,x+y*.01435408+d*.50083021+z*.98350852+o*.98551547+m*.49889859+e*.0157002);}vec3 t(vec3 x){mat3 i;i[0]=vec3(3.24306333,-1.53837619,-.49893282);i[1]=vec3(-.96896309,1.87542451,.04154303);i[2]=vec3(.05568392,-.20417438,1.05799454);float v=dot(i[0],x),y=dot(i[1],x),o=dot(i[2],x);return f(vec3(v,y,o));}vec3 d(float m[38]){vec3 i=vec3(0);i+=m[0]*vec3(6.469e-5,1.84e-6,.00030502);i+=m[1]*vec3(.00021941,6.21e-6,.00103681);i+=m[2]*vec3(.00112057,3.101e-5,.00531314);i+=m[3]*vec3(.00376661,.00010475,.01795439);i+=m[4]*vec3(.01188055,.00035364,.05707758);i+=m[5]*vec3(.02328644,.00095147,.11365162);i+=m[6]*vec3(.03455942,.00228226,.17335873);i+=m[7]*vec3(.03722379,.00420733,.19620658);i+=m[8]*vec3(.03241838,.0066888,.18608237);i+=m[9]*vec3(.02123321,.0098884,.13995048);i+=m[10]*vec3(.01049099,.01524945,.08917453);i+=m[11]*vec3(.00329584,.02141831,.04789621);i+=m[12]*vec3(.00050704,.03342293,.02814563);i+=m[13]*vec3(.00094867,.05131001,.01613766);i+=m[14]*vec3(.00627372,.07040208,.0077591);i+=m[15]*vec3(.01686462,.08783871,.00429615);i+=m[16]*vec3(.02868965,.09424905,.00200551);i+=m[17]*vec3(.04267481,.09795667,.00086147);i+=m[18]*vec3(.05625475,.09415219,.00036904);i+=m[19]*vec3(.0694704,.08678102,.00019143);i+=m[20]*vec3(.08305315,.07885653,.00014956);i+=m[21]*vec3(.0861261,.0635267,9.231e-5);i+=m[22]*vec3(.09046614,.05374142,6.813e-5);i+=m[23]*vec3(.08500387,.04264606,2.883e-5);i+=m[24]*vec3(.07090667,.03161735,1.577e-5);i+=m[25]*vec3(.05062889,.02088521,3.94e-6);i+=m[26]*vec3(.03547396,.01386011,1.58e-6);i+=m[27]*vec3(.02146821,.00810264,0);i+=m[28]*vec3(.01251646,.0046301,0);i+=m[29]*vec3(.00680458,.00249138,0);i+=m[30]*vec3(.00346457,.0012593,0);i+=m[31]*vec3(.00149761,.00054165,0);i+=m[32]*vec3(.0007697,.00027795,0);i+=m[33]*vec3(.00040737,.00014711,0);i+=m[34]*vec3(.00016901,6.103e-5,0);i+=m[35]*vec3(9.522e-5,3.439e-5,0);i+=m[36]*vec3(4.903e-5,1.771e-5,0);i+=m[37]*vec3(2e-5,7.22e-6,0);return i;}float d(float y,float m,float v){float z=m*pow(v,2.);return z/(y*pow(1.-v,2.)+z);}vec3 f(vec3 v,vec3 y,float z){vec3 x=m(v),o=m(y);float i[38],a[38];f(x,i);f(o,a);float r=d(i)[1],e=d(a)[1];z=d(r,e,z);float s[38];for(int u=0;u<38;u++){float p=(1.-z)*(pow(1.-i[u],2.)/(2.*i[u]))+z*(pow(1.-a[u],2.)/(2.*a[u]));s[u]=1.+p-sqrt(pow(p,2.)+2.*p);}return t(d(s));}vec4 f(vec4 v,vec4 x,float y){return vec4(f(v.xyz,x.xyz,y),mix(v.w,x.w,y));}\n #endif\n float d(vec2 m,vec2 v,float y,out vec2 i){vec2 f=vec2(m.x+m.y*.5,m.y),x=floor(f),o=fract(f);float z=step(o.y,o.x);vec2 d=vec2(z,1.-z),r=x+d,e=x+1.,a=vec2(x.x-x.y*.5,x.y),p=vec2(a.x+d.x-d.y*.5,a.y+d.y),s=vec2(a.x+.5,a.y+1.),w=m-a,g=m-p,k=m-s;vec3 u,c,t,A;if(any(greaterThan(v,vec2(0)))){t=vec3(a.x,p.x,s);A=vec3(a.y,p.y,s.y);if(v.x>0.)t=mod(vec3(a.x,p.x,s),v.x);if(v.y>0.)A=mod(vec3(a.y,p.y,s.y),v.y);u=floor(t+.5*A+.5);c=floor(A+.5);}else u=vec3(x.x,r.x,e),c=vec3(x.y,r.y,e.y);vec3 S=mod(u,289.);S=mod((S*51.+2.)*S+c,289.);S=mod((S*34.+10.)*S,289.);vec3 b=S*.07482+y,C=cos(b),D=sin(b);vec2 h=vec2(C.x,D),B=vec2(C.y,D.y),E=vec2(C.z,D.z);vec3 F=.8-vec3(dot(w,w),dot(g,g),dot(k,k));F=max(F,0.);vec3 G=F*F,H=G*G,I=vec3(dot(h,w),dot(B,g),dot(E,k)),J=G*F,K=-8.*J*I;i=10.9*(H.x*h+K.x*w+(H.y*B+K.y*g)+(H.z*E+K.z*k));return 10.9*dot(H,I);}vec4 d(vec3 v,float x){return vec4(mix(v,vec3(dot(vec3(.299,.587,.114),v)),x),1);}float f(vec2 v,float x,float y,float f){return fract(sin(dot(v,vec2(x,y)))*f);}void main(){vec4 v=texture2D(mask,vVertTexCoord);if(v.x>0.){vec2 x=vec2(12.9898,78.233),o=vec2(7.9898,58.233),m=vec2(17.9898,3.233);float y=f(vVertTexCoord,x.x,x.y,43358.5453)*2.-1.,z=f(vVertTexCoord,o.x,o.y,43213.5453)*2.-1.,e=f(vVertTexCoord,m.x,m.y,33358.5453)*2.-1.;const vec2 i=vec2(0);vec2 s;vec4 r;if(active){float a=d(vVertTexCoord*5.,i,10.*random.x,s),p=d(vVertTexCoord*5.,i,10.*random.y,s),g=d(vVertTexCoord*5.,i,10.*random.z,s),k=.25+.25*d(vVertTexCoord*4.,i,3.*random.x,s);r=vec4(d(addColor.xyz,k).xyz+vec3(a,p,g)*.03*abs(addColor.x-addColor.y-addColor.z),1);}else r=vec4(addColor.xyz,1);if(v.w>.7){float a=.5*(v.w-.7);r=r*(1.-a)-vec4(.5)*a;}vec3 a=f(texture2D(source,vVertTexCoord).xyz,r.xyz,.9*v.w);gl_FragColor=vec4(a+.01*vec3(y,z,e),1);}}"};function g(){e.push(),e.translate(-u.trans()[0],-u.trans()[1]),e.image(v.masks[2],-e.width/2,-e.height/2),v.masks[2].clear(),e.pop()}function x(t){t.registerMethod("afterSetup",(()=>v.blend(!1,!0))),t.registerMethod("afterSetup",(()=>v.blend(!1,!0,!0))),t.registerMethod("post",(()=>v.blend(!1,!0))),t.registerMethod("post",(()=>v.blend(!1,!0,!0)))}function y(t,e){a(),w.list.set(t,{gen:e}),w.current=t,w.refresh()}"undefined"!=typeof p5&&x(p5.prototype);const w={isActive:!1,list:new Map,current:"",step_length:()=>Math.min(e.width,e.height)/1e3,create(){this.R=.01*e.width,this.left_x=-1*e.width,this.top_y=-1*e.height,this.num_columns=Math.round(2*e.width/this.R),this.num_rows=Math.round(2*e.height/this.R),this.addStandard()},flow_field(){return this.list.get(this.current).field},refresh(t=0){this.list.get(this.current).field=this.list.get(this.current).gen(t,this.genField())},genField(){let t=new Array(this.num_columns);for(let e=0;e=0&&this.row_index>=0&&this.column_index=-t-u.trans()[0]&&this.x<=t-u.trans()[0]&&this.y>=-i-u.trans()[1]&&this.y<=i-u.trans()[1]}angle(){return this.isIn()&&w.isActive?w.flow_field()[this.column_index][this.row_index]:0}moveTo(t,e,i=A.spacing(),s=!0){if(this.isIn()){let n,r;s||(n=l.cos(-e),r=l.sin(-e));for(let o=0;o=A.cr[0]&&this.position.x<=A.cr[2]&&this.position.y>=A.cr[1]&&this.position.y<=A.cr[3];{let t=.55*e.width,i=.55*e.height;return this.position.x>=-t-u.trans()[0]&&this.position.x<=t-u.trans()[0]&&this.position.y>=-i-u.trans()[1]&&this.position.y<=i-u.trans()[1]}},drawSpray(t){let e=this.w*this.p.vibration*t+this.w*l.gaussian()*this.p.vibration/3,i=this.p.weight*l.random(.9,1.1);const s=this.p.quality/t;for(let t=0;t.4&&this.mask.circle(this.position.x+.7*e*l.random(-1,1),this.position.y+e*l.random(-1,1),t*this.p.weight*l.random(.85,1.15))},adjustSizeAndRotation(t,e){if(this.mask.scale(t),"image"===this.p.type&&(this.p.blend?this.mask.tint(255,0,0,e/2):this.mask.tint(this.mask.red(this.c),this.mask.green(this.c),this.mask.blue(this.c),e)),"random"===this.p.rotate)this.mask.rotate(l.randInt(0,360));else if("natural"===this.p.rotate){let t=(this.plot?-this.plot.angle(this.position.plotted):-this.dir)+(this.flow?this.position.angle():0);this.mask.rotate(t)}},markerTip(){if(this.isInsideClippingArea()){let t=this.calculatePressure(),e=this.calculateAlpha(t);if(this.mask.fill(255,0,0,e/1.5),"marker"===A.p.type)for(let e=1;e<5;e++)this.drawMarker(t*e/5,!1);else if("custom"===A.p.type||"image"===A.p.type)for(let i=1;i<5;i++)this.drawCustomOrImage(t*i/5,e,!1)}}};function b(t,e){const i="marker"===e.type||"custom"===e.type||"image"===e.type;i||"spray"===e.type||(e.type="default"),"image"===e.type&&(I.add(e.image.src),e.tip=()=>A.mask.image(I.tips.get(A.p.image.src),-A.p.weight/2,-A.p.weight/2,A.p.weight,A.p.weight)),e.blend=!!(i&&!1!==e.blend||e.blend),A.list.set(t,{param:e,colors:[],buffers:[]})}function M(t,e,i=1){C(t),A.c=e,A.w=i,A.isActive=!0}function C(t){A.name=t}function S(t,e,i,s){a();let n=l.dist(t,e,i,s);if(0==n)return;A.initializeDrawingState(t,e,n,!1,!1);let r=m(t,e,i,s);A.draw(r,!1)}function P(t,e,i,s){a(),A.initializeDrawingState(e,i,t.length,!0,t),A.draw(s,!0)}const I={tips:new Map,add(t){this.tips.set(t,!1)},imageToWhite(t){t.loadPixels();for(let e=0;e<4*t.width*t.height;e+=4){let i=(t.pixels[e]+t.pixels[e+1]+t.pixels[e+2])/3;t.pixels[e]=t.pixels[e+1]=t.pixels[e+2]=255,t.pixels[e+3]=255-i}t.updatePixels()},load(){for(let t of this.tips.keys()){let e=(s?n:window.self).loadImage(t,(()=>I.imageToWhite(e)));this.tips.set(t,e)}}};function T(t=5,e=45,i={rand:!1,continuous:!1,gradient:!1}){D.isActive=!0,D.hatchingParams=[t,e,i]}const D={isActive:!1,hatchingParams:[5,45,{}],hatchingBrush:!1,hatch(t){let e=D.hatchingParams[0],i=D.hatchingParams[1],s=D.hatchingParams[2],n=A.c,r=A.name,o=A.w,a=A.isActive;D.hatchingBrush&&M(D.hatchingBrush[0],D.hatchingBrush[1],D.hatchingBrush[2]),i=l.toDegrees(i)%180;let h=1/0,c=-1/0,m=1/0,d=-1/0,u=t=>{for(let e of t.a)h=e[0]c?e[0]:c,m=e[1]d?e[1]:d};Array.isArray(t)||(t=[t]);for(let e of t)u(e);let p=new E([[h,m],[c,m],[c,d],[h,d]]),f=i<=90&&i>=0?m:d,v=s.gradient?l.map(s.gradient,0,1,1,1.1,!0):1,g=[],x=0,y=e,w=t=>({point1:{x:h+y*t*l.cos(90-i),y:f+y*t*l.sin(90-i)},point2:{x:h+y*t*l.cos(90-i)+l.cos(-i),y:f+y*t*l.sin(90-i)+l.sin(-i)}});for(;p.intersect(w(x)).length>0;){let e=[];for(let i of t)e.push(i.intersect(w(x)));g[x]=e.flat().sort(((t,e)=>t.x===e.x?t.y-e.y:t.x-e.x)),y*=v,x++}let k=[];for(let t of g)void 0!==t[0]&&k.push(t);let _=s.rand?s.rand:0;for(let t=0;t0&&s.continuous;for(let s=0;s({x:t[0],y:t[1]}))),e&&(this.vertices=t),this.sides=this.vertices.map(((t,e,i)=>[t,i[(e+1)%i.length]]))}intersect(t){let e=`${t.point1.x},${t.point1.y}-${t.point2.x},${t.point2.y}`;if(this._intersectionCache&&this._intersectionCache[e])return this._intersectionCache[e];let i=[];for(let e of this.sides){let s=c(t.point1,t.point2,e[0],e[1]);!1!==s&&i.push(s)}return this._intersectionCache||(this._intersectionCache={}),this._intersectionCache[e]=i,i}draw(t=!1,e,i){let s=A.isActive;if(t&&M(t,e,i),A.isActive){a();for(let t of this.sides)S(t[0].x,t[0].y,t[1].x,t[1].y)}A.isActive=s}fill(t=!1,e,i,s,n,r){let o=U.isActive;t&&(W(t,e),q(i,r),j(s,n)),U.isActive&&(a(),U.fill(this)),U.isActive=o}hatch(t=!1,e,i){let s=D.isActive;t&&T(t,e,i),D.isActive&&(a(),D.hatch(this)),D.isActive=s}erase(t=!1,i=N.a){if(N.isActive||t){v.masks[2].push(),v.masks[2].noStroke();let s=e.color(t||N.c);s.setAlpha(i),v.masks[2].fill(s),v.masks[2].beginShape();for(let t of this.vertices)v.masks[2].vertex(t.x,t.y);v.masks[2].endShape(e.CLOSE),v.masks[2].pop()}}show(){this.fill(),this.hatch(),this.draw(),this.erase()}}class F{constructor(t){this.segments=[],this.angles=[],this.pres=[],this.type=t,this.dir=0,this.calcIndex(0),this.pol=!1}addSegment(t=0,e=0,i=1,s=!1){this.angles.length>0&&this.angles.splice(-1),t=s?(t%360+360)%360:l.toDegrees(t),this.angles.push(t),this.pres.push(i),this.segments.push(e),this.length=this.segments.reduce(((t,e)=>t+e),0),this.angles.push(t)}endPlot(t=0,e=1,i=!1){t=i?(t%360+360)%360:l.toDegrees(t),this.angles.splice(-1),this.angles.push(t),this.pres.push(e)}rotate(t){this.dir=l.toDegrees(t)}pressure(t){return t>this.length?this.pres[this.pres.length-1]:this.curving(this.pres,t)}angle(t){return t>this.length?this.angles[this.angles.length-1]:(this.calcIndex(t),"curve"===this.type?this.curving(this.angles,t)+this.dir:this.angles[this.index]+this.dir)}curving(t,e){let i=t[this.index],s=t[this.index+1];return void 0===s&&(s=i),Math.abs(s-i)>180&&(s>i?s=-(360-s):i=-(360-i)),l.map(e-this.suma,0,this.segments[this.index],i,s,!0)}calcIndex(t){this.index=-1,this.suma=0;let e=0;for(;e<=t;)this.suma=e,e+=this.segments[this.index+1],this.index++;return this.index}genPol(t,e,i=1,s=!1){a();const n=.5,r=[],o=Math.round(this.length/n),h=new k(t,e);let c=s?.15:3*U.bleed_strength,m=0,d=0;for(let t=0;t=this.segments[t]*c*l.random(.7,1.3)||t>=d)&&h.x&&(r.push([h.x,h.y]),m=0,t>=d&&d++)}return new E(r)}draw(t,e,i){A.isActive&&(a(),this.origin&&(t=this.origin[0],e=this.origin[1],i=1),P(this,t,e,i))}fill(t,e,i){U.isActive&&(a(),this.origin&&(t=this.origin[0],e=this.origin[1],i=1),this.pol=this.genPol(t,e,i),this.pol.fill())}hatch(t,e,i){D.isActive&&(a(),this.origin&&(t=this.origin[0],e=this.origin[1],i=1),this.pol=this.genPol(t,e,i,!0),this.pol.hatch())}erase(t,i,s){if(N.isActive){this.origin&&(t=this.origin[0],i=this.origin[1],s=1),this.pol=this.genPol(t,i,s,!0),v.masks[2].push(),v.masks[2].noStroke();let n=e.color(N.c);n.setAlpha(N.a),v.masks[2].fill(n),v.masks[2].beginShape();for(let t of this.pol.vertices)v.masks[2].vertex(t.x,t.y);v.masks[2].endShape(e.CLOSE),v.masks[2].pop()}}show(t,e,i=1){this.draw(t,e,i),this.fill(t,e,i),this.hatch(t,e,i),this.erase(t,e,i)}}let V,R=!1;function G(t=0){V=l.constrain(t,0,1),R=[]}function L(t,e,i){R.push([t,e,i])}function H(t){t===e.CLOSE&&(R.push(R[0]),R.push(R[1])),(0!=V||w.isActive?O(R,V,t===e.CLOSE):new E(R)).show(),R=!1}function O(t,e=.5,i=!1){let s=new F(0===e?"segments":"curve");if(t&&t.length>0){let n,r,o,a=0;for(let h=0;h0&&h{let s=l.random(.8,1.2)*this.bleed_strength;return ithis.size&&(this.size=e)}if(n)for(let t=0;t(e.x-t.x)*(i.y-t.y)-(e.y-t.y)*(i.x-t.x)>.01;let a=0;for(let t of U.polygon.intersect(r))o(e,i,t)&&a++;this.dir[t]=a%2==0}}trim(t){let e=[...this.v],i=[...this.m],s=[...this.dir];if(this.v.length>10&&t>=.2){let n=~~((1-t)*this.v.length),r=~~this.v.length/2-~~n/2;e.splice(r,n),i.splice(r,n),s.splice(r,n)}return{v:e,m:i,dir:s}}grow(t,e=!1){const i=[],s=[],n=[];let r=this.trim(t);const o=e?-.5:1,a=t=>t+.1*(l.gaussian(.5,.1)-.5);for(let e=0;es?l.random(-1,1):0;n.addSegment(0+o+a(),r+a(),1,!0),n.addSegment(-90+o+a(),r+a(),1,!0),n.addSegment(-180+o+a(),r+a(),1,!0),n.addSegment(-270+o+a(),r+a(),1,!0);let h=s?l.randInt(-5,5):0;s&&n.addSegment(0+o,h*(Math.PI/180)*i,!0),n.endPlot(h+o,1,!0);let c=[t-i*l.sin(o),e-i*l.cos(-o)];n.show(c[0],c[1],1)},t.clip=function(t){A.cr=t},t.colorCache=function(t=!0){v.isCaching=t},t.endShape=H,t.endStroke=function(t,e){R.endPlot(t,e),R.draw(V[0],V[1],1),R=!1},t.erase=function(t="white",e=255){N.isActive=!0,N.c=t,N.a=e},t.field=function(t){a(),w.isActive=!0,w.current=t},t.fill=W,t.fillAnimatedMode=function(t){U.isAnimated=t},t.fillTexture=j,t.flowLine=function(t,e,i,s){a(),A.initializeDrawingState(t,e,i,!0,!1),A.draw(l.toDegrees(s),!1)},t.gravity=function(t,e){a(),U.light_source={x:t,y:e}},t.hatch=T,t.hatchArray=B,t.instance=function(t){s=!0,n=t,x(t)},t.line=S,t.listFields=function(){return Array.from(w.list.keys())},t.load=r,t.noClip=function(){A.cr=null},t.noErase=function(){N.isActive=!1},t.noField=function(){w.isActive=!1},t.noFill=function(){U.isActive=!1},t.noGravity=function(){U.light_source=!1},t.noHatch=function(){D.isActive=!1,D.hatchingBrush=!1},t.noStroke=function(){A.isActive=!1},t.pick=C,t.plot=P,t.polygon=function(t){new E(t).show()},t.pop=function(){w.isActive=d.field.isActive,w.current=d.field.current,A.isActive=d.stroke.isActive,A.name=d.stroke.name,A.c=d.stroke.color,A.w=d.stroke.weight,A.cr=d.stroke.clip,D.isActive=d.hatch.isActive,D.hatchingParams=d.hatch.hatchingParams,D.hatchingBrush=d.hatch.hatchingBrush,U.isActive=d.fill.isActive,U.color=d.fill.color,U.opacity=d.fill.opacity,U.bleed_strength=d.fill.bleed_strength,U.texture_strength=d.fill.texture_strength,U.border_strength=d.fill.border_strength,u.rotation=d.others.rotate},t.preload=function(){I.load()},t.push=function(){d.field.isActive=w.isActive,d.field.current=w.current,d.stroke.isActive=A.isActive,d.stroke.name=A.name,d.stroke.color=A.c,d.stroke.weight=A.w,d.stroke.clip=A.cr,d.hatch.isActive=D.isActive,d.hatch.hatchingParams=D.hatchingParams,d.hatch.hatchingBrush=D.hatchingBrush,d.fill.isActive=U.isActive,d.fill.color=U.color,d.fill.opacity=U.opacity,d.fill.bleed_strength=U.bleed_strength,d.fill.texture_strength=U.texture_strength,d.fill.border_strength=U.border_strength,d.others.rotate=u.rotation},t.reBlend=function(){e.push(),e.set("marker","white",1),e.line(-10,-10,-5,-5),e.pop()},t.reDraw=g,t.rect=function(t,i,s,n,r=e.CORNER){if(r==e.CENTER&&(t-=s/2,i-=n/2),w.isActive)G(0),L(t,i),L(t+s,i),L(t+s,i+n),L(t,i+n),H(e.CLOSE);else{new E([[t,i],[t+s,i],[t+s,i+n],[t,i+n]]).show()}},t.refreshField=function(t){w.refresh(t)},t.remove=o,t.rotate=function(t=0){u.rotation=l.toDegrees(t)},t.scale=f,t.scaleBrushes=_,t.seed=function(t){h=new Math.seedrandom(t)},t.segment=function(t,e,i){R.addSegment(t,e,i)},t.set=M,t.setHatch=function(t,e="black",i=1){D.hatchingBrush=[t,e,i]},t.spline=function(t,e=.5){O(t,e).draw()},t.stroke=function(t,e,i){arguments.length>0&&(A.c=arguments.length<2?t:[t,e,i]),A.isActive=!0},t.strokeWeight=function(t){A.w=t},t.vertex=L}));
+!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?e(exports):"function"==typeof define&&define.amd?define(["exports"],e):e((t="undefined"!=typeof globalThis?globalThis:t||self).brush={})}(this,(function(t){"use strict";function e(t,e){let s=new i(t),n=()=>s.next();return n.double=()=>n()+11102230246251565e-32*(2097152*n()|0),n.int32=()=>4294967296*s.next()|0,n.quick=n,function(t,e,i){let s=i&&i.state;s&&("object"==typeof s&&e.copy(s,e),t.state=()=>e.copy(e,{}))}(n,s,e),n}class i{constructor(t){null==t&&(t=+new Date);let e=4022871197;function i(t){t=String(t);for(let i=0;i>>0,s-=e,s*=e,e=s>>>0,s-=e,e+=4294967296*s}return 2.3283064365386963e-10*(e>>>0)}this.c=1,this.s0=i(" "),this.s1=i(" "),this.s2=i(" "),this.s0-=i(t),this.s0<0&&(this.s0+=1),this.s1-=i(t),this.s1<0&&(this.s1+=1),this.s2-=i(t),this.s2<0&&(this.s2+=1)}next(){let{c:t,s0:e,s1:i,s2:s}=this,n=2091639*e+2.3283064365386963e-10*t;return this.s0=i,this.s1=s,this.s2=n-(this.c=0|n)}copy(t,e){return e.c=t.c,e.s0=t.s0,e.s1=t.s1,e.s2=t.s2,e}}let s,n=!1,r=!1,o=!1;function a(t=!1){let e=!(!r||!t)&&o;n&&h(!1),!t&&r&&(t=o),s=t||window.self,n||(n=!0,_.create(),A(s.width/250)),x.load(e)}function h(t=!0){n&&(x.masks[0].remove(),x.masks[0]=null,x.masks[1].remove(),x.masks[1]=null,x.masks[2].remove(),x.masks[2]=null,t&&brush.load())}function l(){n||a()}let c=new e(Math.random());const m={random(t=0,e=1){return 1===arguments.length?this.map(c(),0,1,0,t):this.map(c(),0,1,t,e)},randInt(t,e){return Math.floor(this.random(t,e))},gaussian(t=0,e=1){const i=1-m.random(),s=m.random();return Math.sqrt(-2*Math.log(i))*Math.cos(2*Math.PI*s)*e+t},weightedRand(t){let e,i,s=[];for(e in t)for(i=0;i<10*t[e];i++)s.push(e);return s[Math.floor(c()*s.length)]},map(t,e,i,s,n,r=!1){let o=s+(t-e)/(i-e)*(n-s);return r?sMath.max(Math.min(t,i),e),cos(t){return this.c[Math.floor((t%360+360)%360*4)]},sin(t){return this.s[Math.floor((t%360+360)%360*4)]},isPrecalculationDone:!1,preCalculation(){if(this.isPrecalculationDone)return;const t=1440,e=2*Math.PI/t;this.c=new Float64Array(t),this.s=new Float64Array(t);for(let i=0;i!isNaN(t),toDegrees:t=>(("radians"===s.angleMode()?180*t/Math.PI:t)%360+360)%360,dist:(t,e,i,s)=>Math.hypot(i-t,s-e)};function d(t,e,i,s,n=!1){let r=t.x,o=t.y,a=e.x,h=e.y,l=i.x,c=i.y,m=s.x,d=s.y;if(r===a&&o===h||l===m&&c===d)return!1;let u=a-r,p=h-o,f=m-l,v=d-c,g=v*u-f*p;if(0===g)return!1;let x=(f*(o-c)-v*(r-l))/g,y=(u*(o-c)-p*(r-l))/g;return!(!n&&(y<0||y>1))&&{x:r+x*u,y:o+x*p}}function u(t,e,i,s){return(Math.atan2(-(s-e),i-t)*(180/Math.PI)%360+360)%360}m.preCalculation();const p={field:{},stroke:{},hatch:{},fill:{},others:{}};const f={translation:[0,0],rotation:0,trans(){return this.translation=[s._renderer.uMVMatrix.mat4[12],s._renderer.uMVMatrix.mat4[13]],this.translation}};let v=1;function g(t){v*=t}const x={loaded:!1,isBlending:!1,isCaching:!0,currentColor:new Float32Array(3),load(t){this.type=r&&!t?0:t?2:1,this.masks=[];for(let e=0;e<3;e++)switch(this.type){case 0:this.masks[e]=s.createGraphics(s.width,s.height,1==e?s.WEBGL:s.P2D);break;case 1:this.masks[e]=createGraphics(s.width,s.height,1==e?WEBGL:P2D);break;case 2:this.masks[e]=t.createGraphics(t.width,t.height,1==e?t.WEBGL:t.P2D)}for(let t of this.masks)t.pixelDensity(s.pixelDensity()),t.clear(),t.angleMode(s.DEGREES),t.noSmooth();this.shader=s.createShader(this.vert,this.frag),x.loaded=!0},getPigment(t){let e=t.levels,i=new Float32Array(3);return i[0]=e[0]/255,i[1]=e[1]/255,i[2]=e[2]/255,i},color1:new Float32Array(3),color2:new Float32Array(3),blending1:!1,blending2:!1,blend(t=!1,e=!1,i=!1){if(this.isBlending=i?this.blending1:this.blending2,this.currentColor=i?this.color1:this.color2,!this.isBlending)if(t)this.currentColor=this.getPigment(t),i?(this.blending1=!0,this.color1=this.currentColor):(this.blending2=!0,this.color2=this.currentColor);else if(e)return void(i||y());if((t?this.getPigment(t):this.currentColor).toString()!==this.currentColor.toString()||e||!this.isCaching){if(y(),this.isBlending){s.push(),s.translate(-f.trans()[0],-f.trans()[1]),s.shader(this.shader),this.shader.setUniform("addColor",this.currentColor),this.shader.setUniform("source",s._renderer),this.shader.setUniform("active",x.watercolor),this.shader.setUniform("random",[m.random(),m.random(),m.random()]);let t=i?this.masks[1]:this.masks[0];this.shader.setUniform("mask",t),s.fill(0,0,0,0),s.noStroke(),s.rect(-s.width/2,-s.height/2,s.width,s.height),s.pop(),t.clear()}e||(this.currentColor=this.getPigment(t),i?this.color1=this.currentColor:this.color2=this.currentColor)}e&&(this.isBlending=!1,i?this.blending1=this.isBlending:this.blending2=this.isBlending)},vert:"precision highp float;attribute vec3 aPosition;attribute vec2 aTexCoord;uniform mat4 uModelViewMatrix,uProjectionMatrix;varying vec2 vVertTexCoord;void main(){gl_Position=uProjectionMatrix*uModelViewMatrix*vec4(aPosition,1);vVertTexCoord=aTexCoord;}",frag:"precision highp float;varying vec2 vVertTexCoord;uniform sampler2D source,mask;uniform vec4 addColor;uniform vec3 random;uniform bool active;\n #ifndef SPECTRAL\n #define SPECTRAL\n float x(float v){return v<.04045?v/12.92:pow((v+.055)/1.055,2.4);}float v(float v){return v<.0031308?v*12.92:1.055*pow(v,1./2.4)-.055;}vec3 m(vec3 v){return vec3(x(v[0]),x(v[1]),x(v[2]));}vec3 f(vec3 f){return clamp(vec3(v(f[0]),v(f[1]),v(f[2])),0.,1.);}void f(vec3 v,out float m,out float f,out float x,out float y,out float z,out float i,out float r){m=min(v.x,min(v.y,v.z));v-=m;f=min(v.y,v.z);x=min(v.x,v.z);y=min(v.x,v.y);z=min(max(0.,v.x-v.z),max(0.,v.x-v.y));i=min(max(0.,v.y-v.z),max(0.,v.y-v.x));r=min(max(0.,v.z-v.y),max(0.,v.z-v.x));}void f(vec3 v,inout float i[38]){float x,y,d,z,o,m,e;f(v,x,y,d,z,o,m,e);i[0]=max(1e-4,x+y*.96853629+d*.51567122+z*.02055257+o*.03147571+m*.49108579+e*.97901834);i[1]=max(1e-4,x+y*.96855103+d*.5401552+z*.02059936+o*.03146636+m*.46944057+e*.97901649);i[2]=max(1e-4,x+y*.96859338+d*.62645502+z*.02062723+o*.03140624+m*.4016578+e*.97901118);i[3]=max(1e-4,x+y*.96877345+d*.75595012+z*.02073387+o*.03119611+m*.2449042+e*.97892146);i[4]=max(1e-4,x+y*.96942204+d*.92826996+z*.02114202+o*.03053888+m*.0682688+e*.97858555);i[5]=max(1e-4,x+y*.97143709+d*.97223624+z*.02233154+o*.02856855+m*.02732883+e*.97743705);i[6]=max(1e-4,x+y*.97541862+d*.98616174+z*.02556857+o*.02459485+m*.013606+e*.97428075);i[7]=max(1e-4,x+y*.98074186+d*.98955255+z*.03330189+o*.0192952+m*.01000187+e*.96663223);i[8]=max(1e-4,x+y*.98580992+d*.98676237+z*.05185294+o*.01423112+m*.01284127+e*.94822893);i[9]=max(1e-4,x+y*.98971194+d*.97312575+z*.10087639+o*.01033111+m*.02636635+e*.89937713);i[10]=max(1e-4,x+y*.99238027+d*.91944277+z*.24000413+o*.00765876+m*.07058713+e*.76070164);i[11]=max(1e-4,x+y*.99409844+d*.32564851+z*.53589066+o*.00593693+m*.70421692+e*.4642044);i[12]=max(1e-4,x+y*.995172+d*.13820628+z*.79874659+o*.00485616+m*.85473994+e*.20123039);i[13]=max(1e-4,x+y*.99576545+d*.05015143+z*.91186529+o*.00426186+m*.95081565+e*.08808402);i[14]=max(1e-4,x+y*.99593552+d*.02912336+z*.95399623+o*.00409039+m*.9717037+e*.04592894);i[15]=max(1e-4,x+y*.99564041+d*.02421691+z*.97137099+o*.00438375+m*.97651888+e*.02860373);i[16]=max(1e-4,x+y*.99464769+d*.02660696+z*.97939505+o*.00537525+m*.97429245+e*.02060067);i[17]=max(1e-4,x+y*.99229579+d*.03407586+z*.98345207+o*.00772962+m*.97012917+e*.01656701);i[18]=max(1e-4,x+y*.98638762+d*.04835936+z*.98553736+o*.0136612+m*.9425863+e*.01451549);i[19]=max(1e-4,x+y*.96829712+d*.0001172+z*.98648905+o*.03181352+m*.99989207+e*.01357964);i[20]=max(1e-4,x+y*.89228016+d*8.554e-5+z*.98674535+o*.10791525+m*.99989891+e*.01331243);i[21]=max(1e-4,x+y*.53740239+d*.85267882+z*.98657555+o*.46249516+m*.13823139+e*.01347661);i[22]=max(1e-4,x+y*.15360445+d*.93188793+z*.98611877+o*.84604333+m*.06968113+e*.01387181);i[23]=max(1e-4,x+y*.05705719+d*.94810268+z*.98559942+o*.94275572+m*.05628787+e*.01435472);i[24]=max(1e-4,x+y*.03126539+d*.94200977+z*.98507063+o*.96860996+m*.06111561+e*.01479836);i[25]=max(1e-4,x+y*.02205445+d*.91478045+z*.98460039+o*.97783966+m*.08987709+e*.0151525);i[26]=max(1e-4,x+y*.01802271+d*.87065445+z*.98425301+o*.98187757+m*.13656016+e*.01540513);i[27]=max(1e-4,x+y*.0161346+d*.78827548+z*.98403909+o*.98377315+m*.22169624+e*.01557233);i[28]=max(1e-4,x+y*.01520947+d*.65738359+z*.98388535+o*.98470202+m*.32176956+e*.0156571);i[29]=max(1e-4,x+y*.01475977+d*.59909403+z*.98376116+o*.98515481+m*.36157329+e*.01571025);i[30]=max(1e-4,x+y*.01454263+d*.56817268+z*.98368246+o*.98537114+m*.4836192+e*.01571916);i[31]=max(1e-4,x+y*.01444459+d*.54031997+z*.98365023+o*.98546685+m*.46488579+e*.01572133);i[32]=max(1e-4,x+y*.01439897+d*.52110241+z*.98361309+o*.98550011+m*.47440306+e*.01572502);i[33]=max(1e-4,x+y*.0143762+d*.51041094+z*.98357259+o*.98551031+m*.4857699+e*.01571717);i[34]=max(1e-4,x+y*.01436343+d*.50526577+z*.98353856+o*.98550741+m*.49267971+e*.01571905);i[35]=max(1e-4,x+y*.01435687+d*.5025508+z*.98351247+o*.98551323+m*.49625685+e*.01571059);i[36]=max(1e-4,x+y*.0143537+d*.50126452+z*.98350101+o*.98551563+m*.49807754+e*.01569728);i[37]=max(1e-4,x+y*.01435408+d*.50083021+z*.98350852+o*.98551547+m*.49889859+e*.0157002);}vec3 t(vec3 x){mat3 i;i[0]=vec3(3.24306333,-1.53837619,-.49893282);i[1]=vec3(-.96896309,1.87542451,.04154303);i[2]=vec3(.05568392,-.20417438,1.05799454);float v=dot(i[0],x),y=dot(i[1],x),o=dot(i[2],x);return f(vec3(v,y,o));}vec3 d(float m[38]){vec3 i=vec3(0);i+=m[0]*vec3(6.469e-5,1.84e-6,.00030502);i+=m[1]*vec3(.00021941,6.21e-6,.00103681);i+=m[2]*vec3(.00112057,3.101e-5,.00531314);i+=m[3]*vec3(.00376661,.00010475,.01795439);i+=m[4]*vec3(.01188055,.00035364,.05707758);i+=m[5]*vec3(.02328644,.00095147,.11365162);i+=m[6]*vec3(.03455942,.00228226,.17335873);i+=m[7]*vec3(.03722379,.00420733,.19620658);i+=m[8]*vec3(.03241838,.0066888,.18608237);i+=m[9]*vec3(.02123321,.0098884,.13995048);i+=m[10]*vec3(.01049099,.01524945,.08917453);i+=m[11]*vec3(.00329584,.02141831,.04789621);i+=m[12]*vec3(.00050704,.03342293,.02814563);i+=m[13]*vec3(.00094867,.05131001,.01613766);i+=m[14]*vec3(.00627372,.07040208,.0077591);i+=m[15]*vec3(.01686462,.08783871,.00429615);i+=m[16]*vec3(.02868965,.09424905,.00200551);i+=m[17]*vec3(.04267481,.09795667,.00086147);i+=m[18]*vec3(.05625475,.09415219,.00036904);i+=m[19]*vec3(.0694704,.08678102,.00019143);i+=m[20]*vec3(.08305315,.07885653,.00014956);i+=m[21]*vec3(.0861261,.0635267,9.231e-5);i+=m[22]*vec3(.09046614,.05374142,6.813e-5);i+=m[23]*vec3(.08500387,.04264606,2.883e-5);i+=m[24]*vec3(.07090667,.03161735,1.577e-5);i+=m[25]*vec3(.05062889,.02088521,3.94e-6);i+=m[26]*vec3(.03547396,.01386011,1.58e-6);i+=m[27]*vec3(.02146821,.00810264,0);i+=m[28]*vec3(.01251646,.0046301,0);i+=m[29]*vec3(.00680458,.00249138,0);i+=m[30]*vec3(.00346457,.0012593,0);i+=m[31]*vec3(.00149761,.00054165,0);i+=m[32]*vec3(.0007697,.00027795,0);i+=m[33]*vec3(.00040737,.00014711,0);i+=m[34]*vec3(.00016901,6.103e-5,0);i+=m[35]*vec3(9.522e-5,3.439e-5,0);i+=m[36]*vec3(4.903e-5,1.771e-5,0);i+=m[37]*vec3(2e-5,7.22e-6,0);return i;}float d(float y,float m,float v){float z=m*pow(v,2.);return z/(y*pow(1.-v,2.)+z);}vec3 f(vec3 v,vec3 y,float z){vec3 x=m(v),o=m(y);float i[38],a[38];f(x,i);f(o,a);float r=d(i)[1],e=d(a)[1];z=d(r,e,z);float s[38];for(int u=0;u<38;u++){float p=(1.-z)*(pow(1.-i[u],2.)/(2.*i[u]))+z*(pow(1.-a[u],2.)/(2.*a[u]));s[u]=1.+p-sqrt(pow(p,2.)+2.*p);}return t(d(s));}vec4 f(vec4 v,vec4 x,float y){return vec4(f(v.xyz,x.xyz,y),mix(v.w,x.w,y));}\n #endif\n float d(vec2 m,vec2 v,float y,out vec2 i){vec2 f=vec2(m.x+m.y*.5,m.y),x=floor(f),o=fract(f);float z=step(o.y,o.x);vec2 d=vec2(z,1.-z),r=x+d,e=x+1.,a=vec2(x.x-x.y*.5,x.y),p=vec2(a.x+d.x-d.y*.5,a.y+d.y),s=vec2(a.x+.5,a.y+1.),w=m-a,g=m-p,k=m-s;vec3 u,c,t,A;if(any(greaterThan(v,vec2(0)))){t=vec3(a.x,p.x,s);A=vec3(a.y,p.y,s.y);if(v.x>0.)t=mod(vec3(a.x,p.x,s),v.x);if(v.y>0.)A=mod(vec3(a.y,p.y,s.y),v.y);u=floor(t+.5*A+.5);c=floor(A+.5);}else u=vec3(x.x,r.x,e),c=vec3(x.y,r.y,e.y);vec3 S=mod(u,289.);S=mod((S*51.+2.)*S+c,289.);S=mod((S*34.+10.)*S,289.);vec3 b=S*.07482+y,C=cos(b),D=sin(b);vec2 h=vec2(C.x,D),B=vec2(C.y,D.y),E=vec2(C.z,D.z);vec3 F=.8-vec3(dot(w,w),dot(g,g),dot(k,k));F=max(F,0.);vec3 G=F*F,H=G*G,I=vec3(dot(h,w),dot(B,g),dot(E,k)),J=G*F,K=-8.*J*I;i=10.9*(H.x*h+K.x*w+(H.y*B+K.y*g)+(H.z*E+K.z*k));return 10.9*dot(H,I);}vec4 d(vec3 v,float x){return vec4(mix(v,vec3(dot(vec3(.299,.587,.114),v)),x),1);}float f(vec2 v,float x,float y,float f){return fract(sin(dot(v,vec2(x,y)))*f);}void main(){vec4 v=texture2D(mask,vVertTexCoord);if(v.x>0.){vec2 x=vec2(12.9898,78.233),o=vec2(7.9898,58.233),m=vec2(17.9898,3.233);float y=f(vVertTexCoord,x.x,x.y,43358.5453)*2.-1.,z=f(vVertTexCoord,o.x,o.y,43213.5453)*2.-1.,e=f(vVertTexCoord,m.x,m.y,33358.5453)*2.-1.;const vec2 i=vec2(0);vec2 s;vec4 r;if(active){float a=d(vVertTexCoord*5.,i,10.*random.x,s),p=d(vVertTexCoord*5.,i,10.*random.y,s),g=d(vVertTexCoord*5.,i,10.*random.z,s),k=.25+.25*d(vVertTexCoord*4.,i,3.*random.x,s);r=vec4(d(addColor.xyz,k).xyz+vec3(a,p,g)*.03*abs(addColor.x-addColor.y-addColor.z),1);}else r=vec4(addColor.xyz,1);if(v.w>.7){float a=.5*(v.w-.7);r=r*(1.-a)-vec4(.5)*a;}vec3 a=f(texture2D(source,vVertTexCoord).xyz,r.xyz,.9*v.w);gl_FragColor=vec4(a+.01*vec3(y,z,e),1);}}"};function y(){s.push(),s.translate(-f.trans()[0],-f.trans()[1]),s.image(x.masks[2],-s.width/2,-s.height/2),x.masks[2].clear(),s.pop()}function w(t){t.registerMethod("afterSetup",(()=>x.blend(!1,!0))),t.registerMethod("afterSetup",(()=>x.blend(!1,!0,!0))),t.registerMethod("post",(()=>x.blend(!1,!0))),t.registerMethod("post",(()=>x.blend(!1,!0,!0)))}function k(t,e){l(),_.list.set(t,{gen:e}),_.current=t,_.refresh()}"undefined"!=typeof p5&&w(p5.prototype);const _={isActive:!1,list:new Map,current:"",step_length:()=>Math.min(s.width,s.height)/1e3,create(){this.R=.01*s.width,this.left_x=-1*s.width,this.top_y=-1*s.height,this.num_columns=Math.round(2*s.width/this.R),this.num_rows=Math.round(2*s.height/this.R),this.addStandard()},flow_field(){return this.list.get(this.current).field},refresh(t=0){this.list.get(this.current).field=this.list.get(this.current).gen(t,this.genField())},genField(){let t=new Array(this.num_columns);for(let e=0;e=0&&this.row_index>=0&&this.column_index<_.num_columns&&this.row_index<_.num_rows:this.isInCanvas()}isInCanvas(){let t=s.width,e=s.height;return this.x>=-t-f.trans()[0]&&this.x<=t-f.trans()[0]&&this.y>=-e-f.trans()[1]&&this.y<=e-f.trans()[1]}angle(){return this.isIn()&&_.isActive?_.flow_field()[this.column_index][this.row_index]:0}moveTo(t,e,i=C.spacing(),s=!0){if(this.isIn()){let n,r;s||(n=m.cos(-e),r=m.sin(-e));for(let o=0;o=C.cr[0]&&this.position.x<=C.cr[2]&&this.position.y>=C.cr[1]&&this.position.y<=C.cr[3];{let t=.55*s.width,e=.55*s.height;return this.position.x>=-t-f.trans()[0]&&this.position.x<=t-f.trans()[0]&&this.position.y>=-e-f.trans()[1]&&this.position.y<=e-f.trans()[1]}},drawSpray(t){let e=this.w*this.p.vibration*t+this.w*m.gaussian()*this.p.vibration/3,i=this.p.weight*m.random(.9,1.1);const s=this.p.quality/t;for(let t=0;t.4&&this.mask.circle(this.position.x+.7*e*m.random(-1,1),this.position.y+e*m.random(-1,1),t*this.p.weight*m.random(.85,1.15))},adjustSizeAndRotation(t,e){if(this.mask.scale(t),"image"===this.p.type&&(this.p.blend?this.mask.tint(255,0,0,e/2):this.mask.tint(this.mask.red(this.c),this.mask.green(this.c),this.mask.blue(this.c),e)),"random"===this.p.rotate)this.mask.rotate(m.randInt(0,360));else if("natural"===this.p.rotate){let t=(this.plot?-this.plot.angle(this.position.plotted):-this.dir)+(this.flow?this.position.angle():0);this.mask.rotate(t)}},markerTip(){if(this.isInsideClippingArea()){let t=this.calculatePressure(),e=this.calculateAlpha(t);if(this.mask.fill(255,0,0,e/1.5),"marker"===C.p.type)for(let e=1;e<5;e++)this.drawMarker(t*e/5,!1);else if("custom"===C.p.type||"image"===C.p.type)for(let i=1;i<5;i++)this.drawCustomOrImage(t*i/5,e,!1)}}};function M(t,e){const i="marker"===e.type||"custom"===e.type||"image"===e.type;i||"spray"===e.type||(e.type="default"),"image"===e.type&&(D.add(e.image.src),e.tip=()=>C.mask.image(D.tips.get(C.p.image.src),-C.p.weight/2,-C.p.weight/2,C.p.weight,C.p.weight)),e.blend=!!(i&&!1!==e.blend||e.blend),C.list.set(t,{param:e,colors:[],buffers:[]})}function S(t,e,i=1){P(t),C.c=e,C.w=i,C.isActive=!0}function P(t){C.name=t}function I(t,e,i,s){l();let n=m.dist(t,e,i,s);if(0==n)return;C.initializeDrawingState(t,e,n,!1,!1);let r=u(t,e,i,s);C.draw(r,!1)}function T(t,e,i,s){l(),C.initializeDrawingState(e,i,t.length,!0,t),C.draw(s,!0)}const D={tips:new Map,add(t){this.tips.set(t,!1)},imageToWhite(t){t.loadPixels();for(let e=0;e<4*t.width*t.height;e+=4){let i=(t.pixels[e]+t.pixels[e+1]+t.pixels[e+2])/3;t.pixels[e]=t.pixels[e+1]=t.pixels[e+2]=255,t.pixels[e+3]=255-i}t.updatePixels()},load(){for(let t of this.tips.keys()){let e=(r?o:window.self).loadImage(t,(()=>D.imageToWhite(e)));this.tips.set(t,e)}}};function B(t=5,e=45,i={rand:!1,continuous:!1,gradient:!1}){E.isActive=!0,E.hatchingParams=[t,e,i]}const E={isActive:!1,hatchingParams:[5,45,{}],hatchingBrush:!1,hatch(t){let e=E.hatchingParams[0],i=E.hatchingParams[1],s=E.hatchingParams[2],n=C.c,r=C.name,o=C.w,a=C.isActive;E.hatchingBrush&&S(E.hatchingBrush[0],E.hatchingBrush[1],E.hatchingBrush[2]),i=m.toDegrees(i)%180;let h=1/0,l=-1/0,c=1/0,d=-1/0,u=t=>{for(let e of t.a)h=e[0]l?e[0]:l,c=e[1]d?e[1]:d};Array.isArray(t)||(t=[t]);for(let e of t)u(e);let p=new V([[h,c],[l,c],[l,d],[h,d]]),f=i<=90&&i>=0?c:d,v=s.gradient?m.map(s.gradient,0,1,1,1.1,!0):1,g=[],x=0,y=e,w=t=>({point1:{x:h+y*t*m.cos(90-i),y:f+y*t*m.sin(90-i)},point2:{x:h+y*t*m.cos(90-i)+m.cos(-i),y:f+y*t*m.sin(90-i)+m.sin(-i)}});for(;p.intersect(w(x)).length>0;){let e=[];for(let i of t)e.push(i.intersect(w(x)));g[x]=e.flat().sort(((t,e)=>t.x===e.x?t.y-e.y:t.x-e.x)),y*=v,x++}let k=[];for(let t of g)void 0!==t[0]&&k.push(t);let _=s.rand?s.rand:0;for(let t=0;t0&&s.continuous;for(let s=0;s({x:t[0],y:t[1]}))),e&&(this.vertices=t),this.sides=this.vertices.map(((t,e,i)=>[t,i[(e+1)%i.length]]))}intersect(t){let e=`${t.point1.x},${t.point1.y}-${t.point2.x},${t.point2.y}`;if(this._intersectionCache&&this._intersectionCache[e])return this._intersectionCache[e];let i=[];for(let e of this.sides){let s=d(t.point1,t.point2,e[0],e[1]);!1!==s&&i.push(s)}return this._intersectionCache||(this._intersectionCache={}),this._intersectionCache[e]=i,i}draw(t=!1,e,i){let s=C.isActive;if(t&&S(t,e,i),C.isActive){l();for(let t of this.sides)I(t[0].x,t[0].y,t[1].x,t[1].y)}C.isActive=s}fill(t=!1,e,i,s,n,r){let o=$.isActive;t&&(j(t,e),U(i,r),K(s,n)),$.isActive&&(l(),$.fill(this)),$.isActive=o}hatch(t=!1,e,i){let s=E.isActive;t&&B(t,e,i),E.isActive&&(l(),E.hatch(this)),E.isActive=s}erase(t=!1,e=W.a){if(W.isActive||t){x.masks[2].push(),x.masks[2].noStroke();let i=s.color(t||W.c);i.setAlpha(e),x.masks[2].fill(i),x.masks[2].beginShape();for(let t of this.vertices)x.masks[2].vertex(t.x,t.y);x.masks[2].endShape(s.CLOSE),x.masks[2].pop()}}show(){this.fill(),this.hatch(),this.draw(),this.erase()}}class R{constructor(t){this.segments=[],this.angles=[],this.pres=[],this.type=t,this.dir=0,this.calcIndex(0),this.pol=!1}addSegment(t=0,e=0,i=1,s=!1){this.angles.length>0&&this.angles.splice(-1),t=s?(t%360+360)%360:m.toDegrees(t),this.angles.push(t),this.pres.push(i),this.segments.push(e),this.length=this.segments.reduce(((t,e)=>t+e),0),this.angles.push(t)}endPlot(t=0,e=1,i=!1){t=i?(t%360+360)%360:m.toDegrees(t),this.angles.splice(-1),this.angles.push(t),this.pres.push(e)}rotate(t){this.dir=m.toDegrees(t)}pressure(t){return t>this.length?this.pres[this.pres.length-1]:this.curving(this.pres,t)}angle(t){return t>this.length?this.angles[this.angles.length-1]:(this.calcIndex(t),"curve"===this.type?this.curving(this.angles,t)+this.dir:this.angles[this.index]+this.dir)}curving(t,e){let i=t[this.index],s=t[this.index+1];return void 0===s&&(s=i),Math.abs(s-i)>180&&(s>i?s=-(360-s):i=-(360-i)),m.map(e-this.suma,0,this.segments[this.index],i,s,!0)}calcIndex(t){this.index=-1,this.suma=0;let e=0;for(;e<=t;)this.suma=e,e+=this.segments[this.index+1],this.index++;return this.index}genPol(t,e,i=1,s=!1){l();const n=.5,r=[],o=Math.round(this.length/n),a=new z(t,e);let h=s?.15:3*$.bleed_strength,c=0,d=0;for(let t=0;t=this.segments[t]*h*m.random(.7,1.3)||t>=d)&&a.x&&(r.push([a.x,a.y]),c=0,t>=d&&d++)}return new V(r)}draw(t,e,i){C.isActive&&(l(),this.origin&&(t=this.origin[0],e=this.origin[1],i=1),T(this,t,e,i))}fill(t,e,i){$.isActive&&(l(),this.origin&&(t=this.origin[0],e=this.origin[1],i=1),this.pol=this.genPol(t,e,i),this.pol.fill())}hatch(t,e,i){E.isActive&&(l(),this.origin&&(t=this.origin[0],e=this.origin[1],i=1),this.pol=this.genPol(t,e,i,!0),this.pol.hatch())}erase(t,e,i){if(W.isActive){this.origin&&(t=this.origin[0],e=this.origin[1],i=1),this.pol=this.genPol(t,e,i,!0),x.masks[2].push(),x.masks[2].noStroke();let n=s.color(W.c);n.setAlpha(W.a),x.masks[2].fill(n),x.masks[2].beginShape();for(let t of this.pol.vertices)x.masks[2].vertex(t.x,t.y);x.masks[2].endShape(s.CLOSE),x.masks[2].pop()}}show(t,e,i=1){this.draw(t,e,i),this.fill(t,e,i),this.hatch(t,e,i),this.erase(t,e,i)}}let G,L=!1;function H(t=0){G=m.constrain(t,0,1),L=[]}function O(t,e,i){L.push([t,e,i])}function q(t){t===s.CLOSE&&(L.push(L[0]),L.push(L[1])),(0!=G||_.isActive?N(L,G,t===s.CLOSE):new V(L)).show(),L=!1}function N(t,e=.5,i=!1){let s=new R(0===e?"segments":"curve");if(t&&t.length>0){let n,r,o,a=0;for(let h=0;h0&&h{let s=m.random(.8,1.2)*this.bleed_strength;return ithis.size&&(this.size=e)}if(n)for(let t=0;t(e.x-t.x)*(i.y-t.y)-(e.y-t.y)*(i.x-t.x)>.01;let a=0;for(let t of $.polygon.intersect(r))o(e,i,t)&&a++;this.dir[t]=a%2==0}}trim(t){let e=[...this.v],i=[...this.m],s=[...this.dir];if(this.v.length>10&&t>=.2){let n=~~((1-t)*this.v.length),r=~~this.v.length/2-~~n/2;e.splice(r,n),i.splice(r,n),s.splice(r,n)}return{v:e,m:i,dir:s}}grow(t,e=!1){const i=[],s=[],n=[];let r=this.trim(t);const o=e?-.5:1,a=t=>t+.1*(m.gaussian(.5,.1)-.5);for(let e=0;es?m.random(-1,1):0;n.addSegment(0+o+a(),r+a(),1,!0),n.addSegment(-90+o+a(),r+a(),1,!0),n.addSegment(-180+o+a(),r+a(),1,!0),n.addSegment(-270+o+a(),r+a(),1,!0);let h=s?m.randInt(-5,5):0;s&&n.addSegment(0+o,h*(Math.PI/180)*i,!0),n.endPlot(h+o,1,!0);let l=[t-i*m.sin(o),e-i*m.cos(-o)];n.show(l[0],l[1],1)},t.clip=function(t){C.cr=t},t.colorCache=function(t=!0){x.isCaching=t},t.endShape=q,t.endStroke=function(t,e){L.endPlot(t,e),L.draw(G[0],G[1],1),L=!1},t.erase=function(t="white",e=255){W.isActive=!0,W.c=t,W.a=e},t.field=function(t){l(),_.isActive=!0,_.current=t},t.fill=j,t.fillAnimatedMode=function(t){$.isAnimated=t},t.fillTexture=K,t.flowLine=function(t,e,i,s){l(),C.initializeDrawingState(t,e,i,!0,!1),C.draw(m.toDegrees(s),!1)},t.gravity=function(t,e){l(),$.light_source={x:t,y:e}},t.hatch=B,t.hatchArray=F,t.instance=function(t){r=!0,o=t,w(t)},t.line=I,t.listFields=function(){return Array.from(_.list.keys())},t.load=a,t.noClip=function(){C.cr=null},t.noErase=function(){W.isActive=!1},t.noField=function(){_.isActive=!1},t.noFill=function(){$.isActive=!1},t.noGravity=function(){$.light_source=!1},t.noHatch=function(){E.isActive=!1,E.hatchingBrush=!1},t.noStroke=function(){C.isActive=!1},t.pick=P,t.plot=T,t.polygon=function(t){new V(t).show()},t.pop=function(){_.isActive=p.field.isActive,_.current=p.field.current,C.isActive=p.stroke.isActive,C.name=p.stroke.name,C.c=p.stroke.color,C.w=p.stroke.weight,C.cr=p.stroke.clip,E.isActive=p.hatch.isActive,E.hatchingParams=p.hatch.hatchingParams,E.hatchingBrush=p.hatch.hatchingBrush,$.isActive=p.fill.isActive,$.color=p.fill.color,$.opacity=p.fill.opacity,$.bleed_strength=p.fill.bleed_strength,$.texture_strength=p.fill.texture_strength,$.border_strength=p.fill.border_strength,f.rotation=p.others.rotate},t.preload=function(){D.load()},t.push=function(){p.field.isActive=_.isActive,p.field.current=_.current,p.stroke.isActive=C.isActive,p.stroke.name=C.name,p.stroke.color=C.c,p.stroke.weight=C.w,p.stroke.clip=C.cr,p.hatch.isActive=E.isActive,p.hatch.hatchingParams=E.hatchingParams,p.hatch.hatchingBrush=E.hatchingBrush,p.fill.isActive=$.isActive,p.fill.color=$.color,p.fill.opacity=$.opacity,p.fill.bleed_strength=$.bleed_strength,p.fill.texture_strength=$.texture_strength,p.fill.border_strength=$.border_strength,p.others.rotate=f.rotation},t.reBlend=function(){s.push(),s.set("marker","white",1),s.line(-10,-10,-5,-5),s.pop()},t.reDraw=y,t.rect=function(t,e,i,n,r=s.CORNER){if(r==s.CENTER&&(t-=i/2,e-=n/2),_.isActive)H(0),O(t,e),O(t+i,e),O(t+i,e+n),O(t,e+n),q(s.CLOSE);else{new V([[t,e],[t+i,e],[t+i,e+n],[t,e+n]]).show()}},t.refreshField=function(t){_.refresh(t)},t.remove=h,t.rotate=function(t=0){f.rotation=m.toDegrees(t)},t.scale=g,t.scaleBrushes=A,t.seed=function(t){c=new e(t)},t.segment=function(t,e,i){L.addSegment(t,e,i)},t.set=S,t.setHatch=function(t,e="black",i=1){E.hatchingBrush=[t,e,i]},t.spline=function(t,e=.5){N(t,e).draw()},t.stroke=function(t,e,i){arguments.length>0&&(C.c=arguments.length<2?t:[t,e,i]),C.isActive=!0},t.strokeWeight=function(t){C.w=t},t.vertex=O}));
diff --git a/example/index.html b/example/index.html
index f6c87b6..df6eb2a 100644
--- a/example/index.html
+++ b/example/index.html
@@ -3,7 +3,7 @@
p5.brush.js Example
-
+
diff --git a/example/sketch.js b/example/sketch.js
index 2decebd..4c4c204 100644
--- a/example/sketch.js
+++ b/example/sketch.js
@@ -97,11 +97,14 @@ function setup () {
}
*/
+
+
brush.noStroke()
brush.gravity(0,height)
brush.fill("black", 60)
+
brush.bleed(0.3)
brush.beginShape(0.5)
brush.vertex(50,50)
@@ -125,6 +128,7 @@ function setup () {
function draw() {
+ /*
//background("#e2e7dc")
translate(-width/2,-height/2)
strokeWeight(2)
@@ -140,6 +144,7 @@ function draw() {
brush.endShape(CLOSE)
noLoop()
+ */
}
function mouseDragged() {
diff --git a/package.json b/package.json
index f0de29b..64b57b6 100644
--- a/package.json
+++ b/package.json
@@ -1,6 +1,6 @@
{
"name": "p5.brush",
- "version": "1.1.0",
+ "version": "1.1.1",
"description": "Unlock custom brushes, natural fill effects and intuitive hatching in p5.js",
"main": "src/index.js",
"module": "src/index.js",
@@ -37,5 +37,8 @@
},
"peerDependencies": {
"p5": "^1.9.0"
+ },
+ "dependencies": {
+ "esm-seedrandom": "^3.0.5"
}
}
diff --git a/src/index.js b/src/index.js
index 92cecbe..f8be424 100644
--- a/src/index.js
+++ b/src/index.js
@@ -36,7 +36,7 @@
* @license
* MIT License
*
- * Copyright (c) 2023 Alejandro Campos Uribe
+ * Copyright (c) 2023-2024 Alejandro Campos Uribe
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
@@ -149,13 +149,15 @@
* and angle calculation.
*/
+ import { prng_alea } from 'esm-seedrandom';
+
/**
* The basic source of randomness, can be seeded for determinism.
* @returns {number} A random number between 0 and 1.
*/
- let rng = new Math.seedrandom(Math.random())
+ let rng = new prng_alea(Math.random())
export function seed (s) {
- rng = new Math.seedrandom(s)
+ rng = new prng_alea(s)
}
/**
@@ -2387,6 +2389,13 @@
}
}
+ function _rotate(cx, cy, x, y, angle) {
+ let cos = R.cos(angle), sin = R.sin(angle),
+ nx = (cos * (x - cx)) + (sin * (y - cy)) + cx,
+ ny = (cos * (y - cy)) - (sin * (x - cx)) + cy;
+ return { x: nx, y: ny };
+ }
+
/**
* The FillPolygon class is used to create and manage the properties of the polygons that produces
* the watercolor effect. It includes methods to grow (expand) the polygon and apply layers