-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathimgdisplay.py
executable file
·249 lines (224 loc) · 9.12 KB
/
imgdisplay.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
#!/usr/bin/python
import gtk
import gtk.gdk
import cairo
from sys import argv
from PIL import Image
class ImgDisplay(gtk.DrawingArea):
def __init__(self, pic_name):
gtk.DrawingArea.__init__(self)
#some constants
self.std_height = 500 # standard height to scale to
#temporary, initial variables
self.div = 4
self.portrait = True
self.vertical = True
self.paper_ratio = 8.5/11 # letter paper
self.pic_name = pic_name
#load in the picture and scale it
imgbuf = gtk.gdk.pixbuf_new_from_file(pic_name)
scale_r = float(self.std_height)/imgbuf.get_height()
self.pixbuf = imgbuf.scale_simple(int(imgbuf.get_width()*scale_r), self.std_height, gtk.gdk.INTERP_HYPER)
self.set_size_request( self.pixbuf.get_width(),self.pixbuf.get_height())
self.connect("expose_event", self.expose)
def expose(self,widget,event):
self.cairo_cxt = widget.window.cairo_create()
self.cairo_cxt.set_line_width(1)
self.cairo_cxt.set_dash([5,2.5])
#create gdk context for widget.window
gdk_cxt = widget.window.new_gc()
color = gdk_cxt.get_colormap().alloc_color("white")
gdk_cxt.set_foreground(color)
#draw a white rectangle
widget.window.draw_rectangle(gdk_cxt, filled=True, x = 0, y = 0, width = self.pixbuf.get_width(),
height = self.pixbuf.get_height())
#draw the pixbuf on to the window
widget.window.draw_pixbuf(gdk_cxt, self.pixbuf, src_x = 0, src_y = 0, dest_x = 0, dest_y = 0,
width = -1, height = -1)
#draw initial lines
self.recalculate_lines()
return False
#draws dotted lines to represent where the splits will be
def draw_lines(self):
for x in self.x_break:
self.cairo_cxt.move_to(x, 0)
self.cairo_cxt.line_to(x, self.allocation.height)
self.cairo_cxt.stroke()
for y in self.y_break:
self.cairo_cxt.move_to(0, y)
self.cairo_cxt.line_to(self.allocation.width, y)
self.cairo_cxt.stroke()
def set_portrait(self, widget):
if widget.get_active():
self.portrait = True
self.redraw()
def set_landscape(self, widget):
if widget.get_active():
self.portrait = False
self.redraw()
def set_vertical(self, widget):
if widget.get_active():
self.vertical = True
self.redraw()
def set_horizontal(self, widget):
if widget.get_active():
self.vertical = False
self.redraw()
def set_letter(self, widget):
if widget.get_active():
self.paper_ratio = 8.5/11
self.redraw()
def set_A4(self, widget):
if widget.get_active():
self.paper_ratio = float(210)/297 # A4
self.redraw()
def set_div(self, widget):
self.div = int(widget.get_text())
self.redraw()
def recalculate_lines(self):
# if we're in portrait, set the height side accordingly. Otherwise use landscape
if self.portrait and self.vertical or not self.portrait and not self.vertical:
paper_ratio = self.paper_ratio
else:
paper_ratio = 1/self.paper_ratio
#now assuming that the height side will be fitted to the denominator in paper_ratio
if self.vertical:
height_div = self.allocation.height/self.div
width_div = height_div*paper_ratio
width_n = int(self.allocation.width/width_div)
height_n = self.div
else:
width_div = self.allocation.width/self.div
height_div = width_div*paper_ratio
height_n = int(self.allocation.height/height_div)
width_n = self.div
self.y_break = [y*height_div for y in range(1,height_n + 1)]
self.x_break = [x*width_div for x in range(1,width_n + 1)]
#each time we recalculate, we need to redraw
self.draw_lines()
#dump the x,y break data to a processing function to output images
def dump_data(self, func, filename):
return func(self.div, self.portrait, self.paper_ratio, self.vertical, self.pic_name, filename)
def redraw(self):
if self.window:
rect = gtk.gdk.Rectangle(0,0,self.allocation.width,self.allocation.height)
self.window.invalidate_rect(rect, True)
self.window.process_updates(True)
def runDialog(widget, imgD, func):
fileD = gtk.FileChooserDialog(title="Open..", action=gtk.FILE_CHOOSER_ACTION_SELECT_FOLDER,
buttons=(gtk.STOCK_CANCEL,gtk.RESPONSE_CANCEL,gtk.STOCK_OPEN,gtk.RESPONSE_OK))
fileD.set_current_folder(".")
response = fileD.run()
if response == gtk.RESPONSE_OK:
f_name = fileD.get_filename()
imgD.dump_data(func, f_name)
fileD.destroy()
def save_to_pdf(div, portrait, paper_ratio, vertical, img_filename, save_filename):
print "saving..."
im = Image.open(img_filename)
#I have no idea how printers work, so let's set the long side of each page to be 1000px
long_side = 1000
short_side = int(long_side*paper_ratio)
#set the target resolution of the rescale, as well as some dimensions that depend on orientation
if portrait:
h_len = long_side
w_len = short_side
if vertical:
rescale_r = long_side*div/float(im.size[1])
else:
rescale_r = short_side*div/float(im.size[0])
else:
h_len = short_side
w_len = long_side
if vertical:
rescale_r = short_side*div/float(im.size[1])
else:
rescale_r = long_side*div/float(im.size[0])
im_rescaled = im.resize(tuple(map((lambda x:int(x*rescale_r)),im.size)), Image.ANTIALIAS)
#start chopping up the image
result_images = []
if vertical:
height_pages = div
width_pages = int(im_rescaled.size[0]/w_len) + 1
else:
width_pages = div
height_pages = int(im_rescaled.size[1]/w_len) + 1
for y in range(height_pages):
h_end = (y+1)*h_len
if h_end > im_rescaled.size[1]:
h_end = im_rescaled.size[1]
for x in range(width_pages):
w_end = (x+1)*w_len
if w_end > im_rescaled.size[0]:
w_end = im_rescaled.size[0]
im_part = im_rescaled.crop((x*w_len, y*h_len, w_end, h_end))
#if im.mode == "RGBA":
#im_part = Image.composite(Image.new("RGB",(w_end-x*w_len,h_end-y*h_len),(255,255,255)),im_part,im_part.split()[3])
#blank_im = Image.new("RGB",(w_len,h_len),(255,255,255))
#blank_im.paste(im_part,(0,0))
#im_part = blank_im
result_images.append(im_part)
#create a ~1cm border around each image
if portrait:
w_pad = 34
h_pad = 44
else:
w_pad = 44
h_pad = 34
bordered_im = [Image.new("RGBA",(w_len+w_pad,h_len+h_pad),(255,255,255,0)) for i in range(len(result_images))]
if(im.mode=="RGB"):
bordered_im=bordered_im.convert("RGB")
#change to a directory and save everything into it
for i in range(len(bordered_im)):
bordered_im[i].paste(result_images[i],(w_pad/2,h_pad/2))
bordered_im[i].save(save_filename+"/"+str(i)+".png")
print "done"
#intialize the gtk interface
window = gtk.Window()
img = ImgDisplay(argv[1])
topVBox = gtk.VBox()
upperHBox = gtk.HBox()
#make radio buttons to specify portrait/landscape
port_r_button = gtk.RadioButton(group=None, label="Portrait")
port_r_button.connect("toggled",img.set_portrait)
land_r_button = gtk.RadioButton(group=port_r_button, label="Landscape")
land_r_button.connect("toggled",img.set_landscape)
#make radio buttons to specify horizontal/vertical fit
vert_r_button = gtk.RadioButton(group=None, label="Vertical")
vert_r_button.connect("toggled",img.set_vertical)
horiz_r_button = gtk.RadioButton(group=vert_r_button, label="Horizontal")
horiz_r_button.connect("toggled",img.set_horizontal)
#make radio buttons to specify paper type
lett_r_button = gtk.RadioButton(group=None, label="Letter")
lett_r_button.connect("toggled",img.set_letter)
a4_r_button = gtk.RadioButton(group=lett_r_button, label="A4")
a4_r_button.connect("toggled",img.set_A4)
#make selector for division
div_num = gtk.Entry()
div_num.set_text("")
div_num.set_max_length(2)
div_num.set_width_chars(3)
div_num.connect("activate",img.set_div)
#make a label for height
d_label = gtk.Label()
d_label.set_text("Division:")
#make a button to make the pdf
make_pdf_btn = gtk.Button("Save to directory...")
make_pdf_btn.connect("clicked",runDialog, img, save_to_pdf)
#pack into HBox
upperHBox.pack_start(port_r_button, expand=False)
upperHBox.pack_start(land_r_button)
upperHBox.pack_start(vert_r_button, expand=False)
upperHBox.pack_start(horiz_r_button)
upperHBox.pack_start(lett_r_button, expand=False)
upperHBox.pack_start(a4_r_button)
upperHBox.pack_start(make_pdf_btn)
upperHBox.pack_end(div_num, expand=False, fill=False)
upperHBox.pack_end(d_label, expand=False, fill=False)
#pack into VBox
topVBox.pack_start(upperHBox)
topVBox.pack_start(img)
window.add(topVBox)
window.connect("destroy",gtk.main_quit)
window.show_all()
gtk.main()