|  |   | 
| (44 intermediate revisions by the same user not shown) | 
| Line 1: | Line 1: | 
|  | cloud
 |  | '''APASRI TITATARN : Bots n Plots class''' | 
|  | my sheep robot. . . 
 |  | 
|  | [[File:Sheep.png |100px|]]
 |  | 
|  | 
 |  | 
 | 
|  | Homework 2nd week
 |  | First touch with Python . . . .gettin to know how to draw basic shapes | 
|  | 
 |  | 
 | 
|  | CloudBot   |  | Here is my first drawing in class with coding.  | 
|  |  | my sheep robot. . . funny one  | 
|  |  | [[File:Sheep.png |200px|]] | 
|  |  |   | 
|  |  |   | 
|  |  |   | 
|  |  | '''My Aim about this class :''' | 
|  |  |   | 
|  |  | " I would like to make imageBot and furthur step would be to make it move as  | 
|  |  | short animated GIF" | 
|  |  |   | 
|  |  |   | 
|  |  |   | 
|  |  | '''Homework 2nd week''' | 
|  |  | Assignment to make your own moving robot. | 
|  |  |   | 
|  |  | This is my CloudBot. A cloud with falling  snowflakes GIF. | 
|  |  | I coded in processing with python language.  | 
|  |  | Here is the code . . . | 
|  | 
 |  | 
 | 
|  | [[File:Cloud_bot_s.gif]] |  | [[File:Cloud_bot_s.gif]] | 
|  | 
 |  | 
 | 
|  | ----Code---- |  |   | 
|  |  |   | 
|  |  | ----Cloud Code---- | 
|  | <pre style="font-size:smaller" > |  | <pre style="font-size:smaller" > | 
|  |  | <source lang="python"> | 
|  | #cloud |  | #cloud | 
|  | width = 400 |  | width = 400 | 
| Line 65: | Line 84: | 
|  | 
 |  | 
 | 
|  | </pre> |  | </pre> | 
|  |  | </source> | 
|  |  |  | 
|  |  |  | 
|  |  |  | 
|  |  |  | 
|  |  |  | 
|  |  | '''TwitterBot :MID TERM ASSIGNMENT | 
|  |  |  | 
|  |  | ======Kaleidoscope Bot=======''' | 
|  | 
 |  | 
 | 
|  |  | '''STEP 1: Still Kaleido''' | 
|  | 
 |  | 
 | 
|  |  | My midtern project is making a image processing bot which make a kaleidoscope style image out of an image.  | 
|  |  | I want the result to look nice and  can totally show the kaleidoscope style whatever the source image is.  | 
|  | 
 |  | 
 | 
|  |  | This is the structure of how I 'kaleidify' the source image. | 
|  | 
 |  | 
 | 
|  |  | [[File:kaleidochart1.jpg]] | 
|  | 
 |  | 
 | 
|  | TwitterBot :MID TERM ASSIGNMENT
 |  | 
|  | ======Kaleidoscope Bot=======
 |  | 
|  | 
 |  | 
 | 
|  | RESULT PICTURE |  | RESULT PICTURE | 
| Line 77: | Line 108: | 
|  | [[File:Screen Shot 2558-06-16 at 7.43.39 PM.png]] |  | [[File:Screen Shot 2558-06-16 at 7.43.39 PM.png]] | 
|  | [[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 result take only small part of the source image. | 
|  |  |  | 
|  |  | 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 187: | Line 223: | 
|  | 
 |  | 
 | 
|  | 
 |  | 
 | 
|  | </pre> |  | </source> | 
|  | 
 |  | 
 | 
|  | 
 |  | 
 | 
|  | I also made the gif version of Kaleidoscope by make 35 different images each turning 10 degree of the source image.
 |  | '''FINAL BOT''' | 
|  | 
 |  | 
 | 
|  | Source image :illustration from Oliver Jeffers
 |  | '''STEP 2 : moving Kaleido'''  | 
|  | [[File:jeffers.jpg||200px]]
 |  | 
|  | 
 |  | 
 | 
|  | This is thecode ofGif Version
 |  | 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.   | 
|  | 
 |  | 
 | 
|  | [[Media:jeffers.gif |Example result .GIF]]
 |  | 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. | 
|  | 
 |  | 
 | 
|  | ======Kaleidoscope 360 GIF ========
 |  | [[File:kaleidochart2.jpg]] | 
|  | <pre style="font-size:normal" >
 |  | 
|  | 
 |  | 
 | 
|  | from PIL import Image,ImageDraw,ImageFilter
 |  | In the code, I made the variable 'n' and 'a' to be able to adjust the the result GIF file.   | 
|  | import PIL.ImageOps as im
 |  | n is the number of the rotation time | 
|  | import numpy as np
 |  | a is the degree of the rotation | 
|  | import moviepy.editor as mpy
 |  | 
|  | 
 |  | 
 | 
|  |  | so they will affect how smooth the animated GIF going to be. | 
|  | 
 |  | 
 | 
|  | 
 |  | 
 | 
|  | def onepic():
 |  | '''STEP 3 Making the bot works online''' | 
|  |     global output
 |  | 
|  | 	#creat mask base thesame 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))
 |  | 
|  | 
 |  | 
 | 
|  | # one variable tovary thesize of triangle
 |  | The @kaleidogif twiiter account is created to be the page for the bot.  | 
|  |     var = y/28 
 |  | ( Before that I just did trial and errors with my own account) | 
|  | 
 |  | 
 | 
|  | # Define triangle mask position (triangle with 20 degree)
 |  | Link : [KaleidoGIF twitter|https://twitter.com/kaleidogif] | 
|  |     (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
 |  | [[File:Botscreen1.PNG|900px]] | 
|  |     draw = ImageDraw.Draw(mask,'RGBA')
 |  | 
|  |     draw.polygon(polygonpos,(0,0,0,255))
 |  | 
|  |     del draw
 |  | 
|  |     mask.save("mask.png")
 |  | 
|  | 
 |  | 
 | 
|  | # Get theAlpha band from the template
 |  | This is the example result. | 
|  |     tmplt = Image.open('mask.png')
 |  | 
|  |     A = tmplt.split()[3]
 |  | 
|  | 
 |  | 
 | 
|  |  | [[File:Botexample1.png|210 px]] | 
|  |  | [[File:Kaleidogif1.gif]] | 
|  | 
 |  | 
 | 
|  | #make one piece oftriangle on transparent bg
 |  | more examples of the final result !! | 
|  |     [R,G,B]=base.split()
 |  | [[File:Botexamplegif2.gif]] | 
|  |     tri = Image.merge('RGBA', (R, G, B, A))
 |  | [[File:Kaleidoexample3.gif.gif|320 px]] | 
|  |  | [[File:Kaleidoexample2.gif|320 px]] | 
|  | 
 |  | 
 | 
|  | #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....')
 |  | 
|  | 
 |  | 
 | 
|  |  | '''FINISHING STEP''' | 
|  | 
 |  | 
 | 
|  | # add style to pattern
 |  | Last step . . . for the summaery event , I improved a bit the quality of the   | 
|  |     pattern = pattern_plain
 |  | 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. | 
|  |     #pattern= pattern_plain.filter(ImageFilter.EDGE_ENHANCE)
 |  | 
|  |     pattern.save('pattern_tri.png')
 |  | 
|  |     print('....stylized pattern....')
 |  | 
|  |   |  | 
|  |   |  | 
|  | #make square canvas for theoutput (wide = double size of height of primary pattern)
 |  | 
|  |     canvas =Image.new('RGBA',(2*trih,2*trih), (255,255,255,0))
 |  | 
|  |     canvas.save('tmpcanvas.png')
 |  | 
|  |     pcanvas=Image.new('RGBA',(2*trih,2*trih), (255,255,255,0))
 |  | 
|  |   |  | 
|  |   |  | 
|  |   |  | 
|  |   |  | 
|  | #put pattern on the canvas
 |  | 
|  | #make sure to put thetip of thetriangle at the center of the canvas
 |  | 
|  | #because when we rotate thecenter of the object is the pivot point
 |  | 
|  | #note: paste command require the coordinate of top left corner
 |  | 
|  | #so point topaste thepattern is. . .
 |  | 
|  |     ccenterx = int(trih-triw/2) 
 |  | 
|  |     canvas.paste(pattern_plain,(ccenterx,trih))
 |  | 
|  |   |  | 
|  |   |  | 
|  | # start rotate thepattern around every 40 degree
 |  | 
|  |   |  | 
|  |     for i in range (0,360,40):
 |  | 
|  |         tmpcanvas = canvas
 |  | 
|  |         tmppat = canvas.rotate(i)
 |  | 
|  |         canvas= Image.alpha_composite(tmpcanvas,tmppat)
 |  | 
|  | 
 |  | 
 | 
|  | # now we gethalf of thethings
 |  | So I added lines of code to get the size of the final GIF and resize it if it exceed the limit.   | 
|  |     half = canvas
 |  | 
|  |     print('half already')
 |  | 
|  | #mirror the half andput in thespace to create simple kaleidoscpoe effect
 |  | 
|  |     mirror = im.mirror(half)
 |  | 
|  |     half2= mirror.rotate(20)
 |  | 
|  |     print('mirrored')
 |  | 
|  | 
 |  | 
 | 
|  | #merge 2 half
 |  | 
|  |     output=Image.alpha_composite(half,half2)
 |  | 
|  |     output.save("output.png")
 |  | 
|  | 
 |  | 
|  | 
 |  | 
|  | #def makegif():
 |  | 
|  |     #clip = mpy.ImageSequenceClip(outfilename, fps=3)
 |  | 
|  |     #clip.write_gif("%s.gif"%filename)
 |  | 
|  | 
 |  | 
 | 
|  | def makekaleido(image) :
 |  | '''The complete code of the final version can be found here in the GitHub link'''. [https://github.com/AaeApasri/KaleidoGIFBot] | 
|  |     global base, filename, source, imgfile, n, outfilename
 |  | feel free to download and develop it further if you are interested.  ;) | 
|  |     #IN CASE Working on filename in thelaptop
 |  | 
|  |     #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
 |  | 
|  |     n= 10
 |  | 
|  |     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'
 |  | 
|  |         ifi!=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
 |  | 
|  |   |  | 
|  |    
 |  | 
|  |   |  | 
|  | if __name__ == '__main__':
 |  | 
|  |   |  | 
|  |     makekaleido(image)
 |  | 
|  | </pre>
 |  |