.PS # Escher.m4 # https://tex.stackexchange.com/questions/129274/showcase-of-optical-illusions-made-with-tex-latex-luatex-context threeD_init [] ifelse(1,1,` [ u=1/2.54 # unit size; could use scale for this define(`treadcolor',`0.8,0.75,0.7') define(`darkgrn',`.1,.1,0') define(`lightgrn',`.7,.7,.6') define(`northsteps',3) define(`eaststeps',8) define(`southsteps',7) define(`weststeps',4) define(`baselayers',10) define(`eb',eval(baselayers+eaststeps)) define(`ebs',eval(eb+southsteps)) define(`under',7) azimuth = -45 # view angles elevation = 25 setview(azimuth,elevation,0) # projected unit vectors UX: Project(u,0,0) UY: Project(0,-u,0) UZ: Project(0,0,u) # layer thickness in units of u f= -(eaststeps*UX.y + southsteps*UY.y - weststeps*UX.y - northsteps*UY.y)/dnl ((eaststeps+southsteps+weststeps+northsteps)*UZ.y) # XY projection of 3D coords in units of u define proj {UX*($1)+UY*($2)+UZ*($3)} define tread {line from UY/2 \ to UY then to UY+UX then to UX then to (0,0) \ then to UY/2 shaded rgbstring(treadcolor) \ with .start at proj($1,$2,$3)+UY/2 } define wrect {line invis from (0,0) to UX*($4) \ then to UX*($4)+UZ*($5) \ then to UZ*($5) then to (0,0) with .start at proj($1,$2,$3) \ shaded rgbstring(r,g,b)} define srect {line invis from (0,0) to UY*($4) then to UY*($4)+UZ*($5) \ then to UZ*($5) then to (0,0) with .start at proj($1,$2,$3) \ shaded rgbstring(r,g,b) } # color blending define(`blendwht',`between3D(`$1',`$2',`$3',`$4',1,1,1)') define makergb { r = $1; g = $2; b = $3 } define dorgb { gg = $2 if $1==0 then { makergb(blendwht(gg,lightgrn)) } \ else { makergb(blendwht(gg,darkgrn)) } } # East for i=-under to baselayers do { dorgb((i%2),0.2) wrect(1,1,i*f,eaststeps-2,-f) } for i=0 to eaststeps do { dorgb((i%2),0.2) wrect(i,1,(baselayers+i)*f,(eaststeps+1-i),-f) tread(i,0,(baselayers+i)*f) line from proj(i,1,(baselayers+i)*f) \ to proj(i,1,(baselayers-1+i)*f) } # South for i=0 to eb do { dorgb((i%2),0.4) srect(eaststeps+1,0,i*f,southsteps+1,-f) } for i=1 to southsteps do { dorgb((i%2),0.4) srect(eaststeps+1,i,(eb+i)*f,(southsteps+1-i),-f) tread(eaststeps,i,(eb+i)*f) line from proj(eaststeps+1,i,(eb+i)*f) \ to proj(eaststeps+1,i,(eb+i-1)*f) } # North for i=-under to baselayers-northsteps-1 do { dorgb((i%2),0.1) srect(1,1,i*f,northsteps-1,-f) } for i=1 to northsteps do { dorgb((i%2),0.1) srect(1,1,(baselayers-i)*f,i,-f) wrect(0,1+i,(baselayers-i)*f,1,-f) line from proj(0,1+i,(baselayers-i)*f) \ to proj(0,1+i,(baselayers-i-1)*f) line from proj(1,1+i,(baselayers-i)*f) \ to proj(1,1+i,(baselayers-i-1)*f) tread(0,i,(baselayers-i)*f) } # West for i=0 to ebs do { dorgb((i%2),0.4) wrect(eaststeps+1,southsteps+1,i*f,-(weststeps+1),-f) } for i=0 to weststeps-1 do { dorgb((i%2),0.4) wrect(eaststeps-i,southsteps+1,(ebs+i+1)*f,-(weststeps-i),-f,) srect(eaststeps-i,southsteps,(ebs+i+1)*f,1,-f) if i!=weststeps-1 then { line from proj(eaststeps-i,southsteps,(ebs+i+1)*f) \ to proj(eaststeps-i,southsteps,(ebs+i)*f) } line from proj(eaststeps-i,southsteps+1,(ebs+i+1)*f) \ to proj(eaststeps-i,southsteps+1,(ebs+i)*f) tread(eaststeps-(i+1),southsteps,(ebs+i+1)*f) } # Corner lines define(`dgreen',`outlined rgbstring(blendwht(0.2,darkgrn))') line dgreen from proj(1,1,(baselayers-1)*f) to proj(1,1,-under*f) line dgreen from proj(eaststeps+1,southsteps+1,ebs*f) \ to proj(eaststeps+1,southsteps+1,-f) line dgreen from proj(0,northsteps+1,(baselayers-northsteps)*f) \ to last line.end-UX*(weststeps+1) then to last line.end \ then to proj(eaststeps+1,0,-f) then to proj(eaststeps+1,0,eb*f) ] ') ifelse(1,1,` define(`DPL',`(project(`$1',`$2',`$3'))') define(`ISL',` S1: `$2'; E1: `$2' + `$3' S2: `$4'; E2: `$4' + `$5' `$1': line from S2 to intersect_(S1,E1,S2,E2) ') [ bdp = 1.8 bwd = 2.8 bht = 1.5 bth = bwd/10 setview(-45,20,0) L1: line from DPL(0,0,0) to DPL(0,0,bht) \ then to DPL(0,bwd,bht) \ then to DPL(bdp,bwd,bht) \ then to DPL(bdp,bwd,0) \ then to DPL(bdp,0,0) \ then to DPL(0,0,0) L2: line from DPL(bth,bth,bht) to DPL(bth,bwd-bth,bht) \ then to DPL(bdp-bth,bwd-bth,bht) \ then to DPL(bdp-bth,bth,bht) \ then to DPL(bth,bth,bht) L3: line from DPL(bth,bth,0) to DPL(bth,bwd-bth,0) \ then to DPL(bdp-bth,bwd-bth,0) \ then to DPL(bdp-bth,bth,0) \ then to DPL(bth,bth,0) L4: line from DPL(bdp,bth,bht-bth) to DPL(bdp,bwd-bth,bht-bth) \ then to DPL(bdp,bwd-bth,bth) \ then to DPL(bth+bth,bwd-bth,bth) L5: line from DPL(bdp-bth,0,bht-bth) to DPL(bth,0,bht-bth) \ then to DPL(bth,0,bth) \ then to DPL(bth,bwd-bth-bth,bth) L6: line from DPL(0,0,bht) to DPL(bdp,0,bht) \ then to DPL(bdp,bwd,bht) L7: line from DPL(0,0,0) to DPL(0,bwd,0) \ then to DPL(bdp,bwd,0) ISL(L8,L4.start,UPy_,L4.end,UPz_) ISL(L9,L4.start,UPy_,DPL(0,bwd,0),UPz_) ISL(L10,L4.start,UPy_,L5.end,UPz_) ISL(L11,L5.end,UPy_,L4.start,UPz_) ISL(L12,L5.end,UPy_,DPL(bdp,0,bht),UPz_) ISL(L13,L5.end,UPy_,L5.start,UPz_) ISL(L14,DPL(bdp-bth,bth,bht),UPy_,DPL(bth,bwd-bth-bth,bht-bth),UPz_) ISL(L15,L2.start,UPx_,DPL(bth,bwd-bth-bth,bht-bth),UPy_) ISL(L16,DPL(bdp-bth,bth,bht),UPy_,DPL(bth,bwd-bth,bht),UPz_) ISL(L17,DPL(bdp-bth,bth,bht),UPy_,DPL(bth+bth,bwd-bth,bht-bth),UPz_) ISL(L18,DPL(bdp-bth,bth,bht),UPy_,DPL(bth+bth,bwd-bth,bht-bth),UPx_) ISL(L19,L3.start,UPy_,L5.start+DPL(0,0,-bht+bth+bth),UPx_) ISL(L20,L3.start,UPy_,L5.start+DPL(0,0,-bht+bth+bth),UPz_) ISL(L21,L3.start,UPy_,DPL(bdp-bth,bth,0),UPz_) ISL(L22,L3.start,UPy_,L4.start+DPL(0,0,-bht+bth+bth),UPz_) ISL(L23,DPL(bth,bwd-bth,0),UPx_,L4.start+DPL(0,0,-bht+bth+bth),UPy_) ISL(L24,L5.start,UPx_,DPL(bth,bth,bth),UPz_) ISL(L25,L4.start,UPy_,DPL(bdp-bth,bwd-bth,bth),UPz_) ifelse(0,1,`for i=1 to 7 do { exec sprintf("\"L%g.s\" at L%g.start",i,i) } for i=8 to 10 do { exec sprintf("\"L%g\" at L%g",i,i) }') ] with .sw at last [].se+(0.2,0) ') .PE