|  |   | 
| (29 intermediate revisions by the same user not shown) | 
| Line 5: | Line 5: | 
|  | Here is my first drawing in class with coding.   |  | Here is my first drawing in class with coding.   | 
|  | my sheep robot. . . funny one   |  | my sheep robot. . . funny one   | 
|  | [[File:Sheep.png |100px|]] |  | [[File:Sheep.png |200px|]] | 
|  | 
 |  | 
 | 
|  | 
 |  | 
 | 
| Line 29: | Line 29: | 
|  | ----Cloud Code---- |  | ----Cloud Code---- | 
|  | <pre style="font-size:smaller" > |  | <pre style="font-size:smaller" > | 
|  |  | <source lang="python"> | 
|  | #cloud |  | #cloud | 
|  | width = 400 |  | width = 400 | 
| Line 83: | Line 84: | 
|  | 
 |  | 
 | 
|  | </pre> |  | </pre> | 
|  |  | </source> | 
|  | 
 |  | 
 | 
|  | 
 |  | 
 | 
| Line 107: | Line 109: | 
|  | [[File:Screen Shot 2558-06-17 at 12.03.18 AM.png]] |  | [[File:Screen Shot 2558-06-17 at 12.03.18 AM.png]] | 
|  | 
 |  | 
 | 
|  | As can be seen here,in this step, the final resault take only small part of the source image. |  | As can be seen here,in this step, the final result take only small part of the source image. | 
|  | 
 |  | 
 | 
|  | It is quite difficult to trance back what thesorce image look like but it is the preparation for the next step. |  | It is quite difficult to trace back what the source image look like but it is the preparation for the next step. | 
|  | 
 |  | 
 | 
|  | 
 |  | 
 | 
|  | code for the image processing |  | code for the image processing | 
|  | ============================= |  | ============================= | 
|  | <pre style="font-size:normal" > |  |   | 
|  |  | <source lang="python"> | 
|  | 
 |  | 
 | 
|  | from PIL import Image, ImageDraw,ImageFilter |  | from PIL import Image, ImageDraw,ImageFilter | 
| Line 220: | Line 223: | 
|  | 
 |  | 
 | 
|  | 
 |  | 
 | 
|  | </pre> |  | </source> | 
|  |   |  | 
|  | 
 |  | 
 | 
|  | 
 |  | 
 | 
|  |  | '''FINAL BOT''' | 
|  | 
 |  | 
 | 
|  | '''STEP 2 : moving Kaleido'''   |  | '''STEP 2 : moving Kaleido'''   | 
|  | 
 |  | 
 | 
|  | I also made thegif version of Kaleidoscopeby make 35 different images each turning 10 degree of the source image.
 |  | The next step is to make the bot gives more feeling of Kaleidoscope. The nice part of Kaleidoscope is whhen one rotate it around and see the abtract patterns moving.    | 
|  |   |  | 
|  | Source image : illustration from Oliver Jeffers
 |  | 
|  | [[File:JEFFERS>JPG]]
 |  | 
|  |   |  | 
|  | This is the code ofGif Version
 |  | 
|  |   |  | 
|  | [[Media:jeffers.gif |Example result .GIF]]
 |  | 
|  |   |  | 
|  | ======Kaleidoscope360 GIF ========
 |  | 
|  | <pre style="font-size:normal" >
 |  | 
|  |   |  | 
|  | from PIL import Image, ImageDraw,ImageFilter
 |  | 
|  | import PIL.ImageOps as im
 |  | 
|  | import numpy as np
 |  | 
|  | import moviepy.editor as mpy
 |  | 
|  |   |  | 
|  |   |  | 
|  |   |  | 
|  | def onepic():
 |  | 
|  |     global output
 |  | 
|  | 	#creat mask base the same size with source image 
 |  | 
|  |     mask = Image.new('RGBA', base.size, (255,255,255,0))
 |  | 
|  |     x, y = base.size
 |  | 
|  |     print("base size: %sx%s"%(x,y))
 |  | 
|  |   |  | 
|  | # onevariable to vary thesize of triangle
 |  | 
|  |     var = y/28 
 |  | 
|  |   |  | 
|  | # Define triangle mask position (triangle with 20 degree)
 |  | 
|  |     (originx,originy) = (int(0.4*x),int(y))
 |  | 
|  |     trih = int(12*var) #fix formular for triangle height
 |  | 
|  |     triw = int(4.2*var) #fix formular for triangle width
 |  | 
|  |     polygonpos = [(originx,originy),
 |  | 
|  |     (originx+triw,originy), 
 |  | 
|  |     (originx+triw/2,originy-trih)]
 |  | 
|  |     print(trih,triw)
 |  | 
|  |   |  | 
|  |   |  | 
|  | # Create mask
 |  | 
|  |     draw = ImageDraw.Draw(mask,'RGBA')
 |  | 
|  |     draw.polygon(polygonpos,(0,0,0,255))
 |  | 
|  |     del draw
 |  | 
|  |     mask.save("mask.png")
 |  | 
|  |   |  | 
|  | # Get the Alpha band from the template
 |  | 
|  |     tmplt = Image.open('mask.png')
 |  | 
|  |     A = tmplt.split()[3]
 |  | 
|  |   |  | 
|  |   |  | 
|  | #make one piece of triangle on transparent bg
 |  | 
|  |     [R,G,B]=base.split()
 |  | 
|  |     tri = Image.merge('RGBA', (R, G, B, A))
 |  | 
|  |   |  | 
|  | #crop it to the exact size of triangle!! to create primary pattern
 |  | 
|  | #box (left, top , right, buttom)
 |  | 
|  |     box =(originx,(originy-trih),(originx+triw),originy)
 |  | 
|  |     pattern_plain=tri.crop(box)
 |  | 
|  |     pattern_plain.save('pattern_plain.png')
 |  | 
|  |     print('....pattern created....')
 |  | 
|  |   |  | 
|  | 
 |  | 
 | 
|  | # add style topattern
 |  | I try to make it the similar feeling here with the bot that each still image is so abstact but when one finish watching the whole loop of rotation then you will have clue about what is the source image. | 
|  |     pattern = pattern_plain
 |  | 
|  |     #pattern= pattern_plain.filter(ImageFilter.EDGE_ENHANCE)
 |  | 
|  |     pattern.save('pattern_tri.png')
 |  | 
|  |     print('....stylized pattern....')
 |  | 
|  | 
 |  | 
 | 
|  |  | [[File:kaleidochart2.jpg]] | 
|  | 
 |  | 
 | 
|  | #make square canvas for theoutput (wide = double size of height of primary pattern)
 |  | In the code, I made the variable 'n' and 'a' to be able to adjust the the result GIF file.   | 
|  |     canvas =Image.new('RGBA',(2*trih,2*trih), (255,255,255,0))
 |  | n is the number of the rotation time | 
|  |     canvas.save('tmpcanvas.png')
 |  | a is the degree of the rotation | 
|  |     pcanvas=Image.new('RGBA',(2*trih,2*trih), (255,255,255,0))
 |  | 
|  | 
 |  | 
 | 
|  |  | so they will affect how smooth the animated GIF going to be. | 
|  | 
 |  | 
 | 
|  | 
 |  | 
 | 
|  |  | '''STEP 3 Making the bot works online''' | 
|  | 
 |  | 
 | 
|  | #put pattern on the canvas
 |  | The @kaleidogif twiiter account is created to be the page for the bot.   | 
|  | #make sure toput the tip of the triangle at the center of the canvas
 |  | ( Before that I just did trial and errors with my own account) | 
|  | #because when we rotate the center of the object is the pivot point
 |  | 
|  | #note: paste command require thecoordinate of top left corner
 |  | 
|  | #so point to paste thepattern is . . .
 |  | 
|  |     ccenterx = int(trih-triw/2) 
 |  | 
|  |     canvas.paste(pattern_plain,(ccenterx,trih))
 |  | 
|  | 
 |  | 
 | 
|  |  | Link : [KaleidoGIF twitter|https://twitter.com/kaleidogif] | 
|  | 
 |  | 
 | 
|  | # start rotate the pattern around every 40 degree
 |  | 
|  | 
 |  | 
 | 
|  |     for i in range (0,360,40):
 |  | [[File:Botscreen1.PNG|900px]] | 
|  |         tmpcanvas = canvas
 |  | 
|  |         tmppat = canvas.rotate(i)
 |  | 
|  |         canvas= Image.alpha_composite(tmpcanvas,tmppat)
 |  | 
|  | 
 |  | 
 | 
|  | # now we get half of thethings
 |  | This is the example result. | 
|  |     half = canvas
 |  | 
|  |     print('half already')
 |  | 
|  | #mirror the half and put in the space to create simple kaleidoscpoe effect
 |  | 
|  |     mirror = im.mirror(half)
 |  | 
|  |     half2= mirror.rotate(20)
 |  | 
|  |     print('mirrored')
 |  | 
|  | 
 |  | 
 | 
|  | #merge 2 half
 |  | [[File:Botexample1.png|210 px]] | 
|  |     output=Image.alpha_composite(half,half2)
 |  | [[File:Kaleidogif1.gif]] | 
|  |     output.save("output.png")
 |  | 
|  | 
 |  | 
 | 
|  |  | more examples of the final result !! | 
|  |  | [[File:Botexamplegif2.gif]] | 
|  |  | [[File:Kaleidoexample3.gif.gif|320 px]] | 
|  |  | [[File:Kaleidoexample2.gif|320 px]] | 
|  | 
 |  | 
 | 
|  | #def makegif():
 |  | 
|  |     #clip = mpy.ImageSequenceClip(outfilename, fps=3)
 |  | 
|  |     #clip.write_gif("%s.gif"%filename)
 |  | 
|  | 
 |  | 
 | 
|  | def makekaleido(image) :
 |  | '''FINISHING STEP''' | 
|  |     global base, filename, source, imgfile, n, outfilename
 |  | 
|  |     #IN CASE Working on filename in the laptop
 |  | 
|  |     #put image path
 |  | 
|  |     imgfile = "tweetimg.jpg"
 |  | 
|  |     filename = imgfile[:-4]
 |  | 
|  |     print(filename)
 |  | 
|  |     
 |  | 
|  |     #load Image
 |  | 
|  |     source = Image.open(imgfile)
 |  | 
|  |     
 |  | 
|  |     #source = image.copy()
 |  | 
|  |     x, y = source.size
 |  | 
|  |   |  | 
|  |     if x > 1000:
 |  | 
|  |         newy = int(1000*y/x)
 |  | 
|  |         source =source.resize((1000,newy), Image.ANTIALIAS)
 |  | 
|  |         print('resize to %sx%s'%source.size)
 |  | 
|  |   |  | 
|  |     #creat mask base the same size with source image 
 |  | 
|  |     mask = Image.new('RGBA', source.size, (255,255,255,0))
 |  | 
|  |     print("mask size:%sx%s"%source.size)
 |  | 
|  |     
 |  | 
|  | 
 |  | 
 | 
|  |     #rotate source n times for a degree
 |  | Last step . . . for the summaery event , I improved a bit the quality of the  | 
|  |     n= 10
 |  | GIF by editing the code to be able to make best quality it can and also the final GIF is not larger that the twitter status update limit. | 
|  |     outfilename=[]
 |  | 
|  |     for a in range(n+n-2):
 |  | 
|  |         outfilename.insert(a,'0')
 |  | 
|  |     a= 10
 |  | 
|  |     base = source
 |  | 
|  |     for i in range(n):
 |  | 
|  |         base = source.rotate(31+i*a)
 |  | 
|  |         onepic()
 |  | 
|  |         output.save('%s%s.png'%(filename,i))
 |  | 
|  |         outfilename[i] = filename+str(i)+'.png'
 |  | 
|  |         if i!=0:
 |  | 
|  |             outfilename[2*n-2-i] = filename+str(i)+'.png'
 |  | 
|  |     
 |  | 
|  |     clip = mpy.ImageSequenceClip(outfilename, fps=3)
 |  | 
|  |     clip.write_gif("%s.gif"%filename)
 |  | 
|  |     image2 = Image.open('tweetimg3.png')
 |  | 
|  | 
 |  | 
 | 
|  |     return image2,clip
 |  | So I added lines of code to get the size of the final GIF and resize it if it exceed the limit.   | 
|  | 
 |  | 
 | 
|  |    
 |  | 
|  | 
 |  | 
 | 
|  | if __name__ == '__main__':
 |  | '''The complete code of the final version can be found here in the GitHub link'''. [https://github.com/AaeApasri/KaleidoGIFBot] | 
|  |   |  | feel free to download and develop it further if you are interested.  ;) | 
|  |     makekaleido(image)
 |  | 
|  | </pre>
 |  |