__init__.py 36 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012
  1. import ast
  2. import time
  3. from gi.repository import GLib, Gtk, Gdk, GdkPixbuf
  4. import Activity
  5. import Export
  6. import threading
  7. import cv2
  8. import numpy
  9. import getpass
  10. import os
  11. from PF2 import Histogram, Layer
  12. from PF2.Tools import BlackWhite
  13. from PF2.Tools import Colours
  14. from PF2.Tools import Contrast
  15. from PF2.Tools import Details
  16. from PF2.Tools import Tonemap
  17. from PF2.Tools import HueEqualiser
  18. from PF2.Tools import Blur
  19. class PF2(Activity.Activity):
  20. def on_init(self):
  21. self.id = "PF2"
  22. self.name = "Edit a Photo"
  23. self.subtitle = "Edit a Raster Image with PhotoFiddle"
  24. UI_FILE = "ui/PF2_Activity.glade"
  25. self.builder.add_from_file(UI_FILE)
  26. self.widget = self.builder.get_object("PF2_main")
  27. self.stack.add_titled(self.widget, self.id, self.name)
  28. self.header_widget = self.builder.get_object("PF2_header")
  29. self.header_stack.add_titled(self.header_widget, self.id, self.name)
  30. # Get all UI Components
  31. self.ui = {}
  32. components = [
  33. "main",
  34. "control_reveal",
  35. "histogram_reveal",
  36. "layers_reveal",
  37. "upper_peak_toggle",
  38. "lower_peak_toggle",
  39. "histogram",
  40. "layer_stack",
  41. "preview",
  42. "preview_eventbox",
  43. "scroll_window",
  44. "open_button",
  45. "original_toggle",
  46. "tool_box_stack",
  47. "tool_stack",
  48. "open_window",
  49. "open_header",
  50. "open_cancel_button",
  51. "open_open_button",
  52. "open_chooser",
  53. "popovermenu",
  54. "show_hist",
  55. "export_image",
  56. "undo",
  57. "redo",
  58. "zoom_toggle",
  59. "zoom_reveal",
  60. "zoom",
  61. "reset",
  62. "mask_draw_toggle",
  63. "mask_erase_toggle",
  64. "mask_brush_size",
  65. "layers_list",
  66. "new_layer",
  67. "layer_mask_reveal",
  68. "add_layer_button",
  69. "remove_layer_button",
  70. "mask_brush_feather",
  71. "mask_brush_feather_scale",
  72. "edit_layer_mask_button",
  73. "layer_opacity",
  74. "layer_opacity_scale",
  75. "layer_blend_mode",
  76. "viewport"
  77. ]
  78. for component in components:
  79. self.ui[component] = self.builder.get_object("%s_%s" % (self.id, component))
  80. self.menu_popover = self.ui["popovermenu"]
  81. # Set up tools
  82. self.tools = [
  83. Contrast.Contrast,
  84. Tonemap.Tonemap,
  85. Details.Details,
  86. Colours.Colours,
  87. HueEqualiser.HueEqualiser,
  88. BlackWhite.BlackWhite,
  89. Blur.Blur
  90. ]
  91. # Setup layers
  92. self.layers = []
  93. self.base_layer = self.create_layer("base", True)
  94. # Set the first tool to active
  95. # self.tools[0].tool_button.set_active(True)
  96. # self.ui["tools"].show_all()
  97. # Disable ui components by default
  98. self.ui["original_toggle"].set_sensitive(False)
  99. self.ui["export_image"].set_sensitive(False)
  100. # Setup editor variables
  101. self.is_editing = False
  102. self.has_loaded = False
  103. self.image_path = ""
  104. self.bit_depth = 8
  105. self.image = None
  106. self.image_is_dirty = True
  107. self.original_image = None
  108. self.pwidth = 0
  109. self.pheight = 0
  110. self.awidth = 0
  111. self.aheight = 0
  112. self.pimage = None
  113. self.poriginal = None
  114. self.change_occurred = False
  115. self.additional_change_occurred = False
  116. self.running_stack = False
  117. self.upper_peak_on = False
  118. self.lower_peak_on = False
  119. self.is_exporting = False
  120. self.is_zooming = False
  121. self.undo_stack = []
  122. self.undo_position = 0
  123. self.undoing = False
  124. self.current_layer_path = None
  125. self.mousex = 0
  126. self.mousey = 0
  127. self.mousedown = False
  128. self.layer_order = []
  129. self.pre_undo_layer_name = "base"
  130. self.pre_undo_editing = False
  131. self.pre_undo_erasing = False
  132. self.last_selected_layer = None
  133. self.was_editing_mask = False
  134. self.source_image = None
  135. self.is_scaling = False
  136. self.current_processing_layer_name = "base"
  137. self.current_processing_layer_index = 0
  138. # Setup Open Dialog
  139. self.ui["open_window"].set_transient_for(self.root)
  140. self.ui["open_window"].set_titlebar(self.ui["open_header"])
  141. # Connect UI signals
  142. self.ui["open_button"].connect("clicked", self.on_open_clicked)
  143. self.ui["preview"].connect("draw", self.on_preview_draw)
  144. self.ui["original_toggle"].connect("toggled", self.on_preview_toggled)
  145. self.ui["upper_peak_toggle"].connect("toggled", self.on_upper_peak_toggled)
  146. self.ui["lower_peak_toggle"].connect("toggled", self.on_lower_peak_toggled)
  147. self.ui["show_hist"].connect("toggled", self.toggle_hist)
  148. self.ui["export_image"].connect("clicked", self.on_export_clicked)
  149. self.ui["open_open_button"].connect("clicked", self.on_file_opened)
  150. self.ui["open_cancel_button"].connect("clicked", self.on_file_canceled)
  151. self.ui["zoom_toggle"].connect("toggled", self.on_zoom_toggled)
  152. self.ui["zoom"].connect("value-changed", self.on_zoom_changed)
  153. self.ui["undo"].connect("clicked", self.on_undo)
  154. self.ui["redo"].connect("clicked", self.on_redo)
  155. self.ui["reset"].connect("clicked", self.on_reset)
  156. self.ui["preview_eventbox"].connect('motion-notify-event', self.preview_dragged)
  157. self.ui["preview_eventbox"].connect('button-press-event', self.new_path)
  158. self.ui["mask_draw_toggle"].connect("toggled", self.mask_draw_toggled)
  159. self.ui["mask_erase_toggle"].connect("toggled", self.mask_erase_toggled)
  160. self.ui["new_layer"].connect("clicked", self.new_layer_button_clicked)
  161. self.ui["layers_list"].connect("row-activated", self.layer_ui_activated)
  162. self.ui["add_layer_button"].connect("clicked", self.new_layer_button_clicked)
  163. self.ui["remove_layer_button"].connect("clicked", self.remove_layer_button_clicked)
  164. self.ui["edit_layer_mask_button"].connect("toggled", self.edit_mask_toggled)
  165. self.ui["layer_opacity"].connect("value-changed", self.update_layer_opacity)
  166. self.ui["scroll_window"].connect_after("draw", self.draw_ui_brush_circle)
  167. self.ui["scroll_window"].connect_after('motion-notify-event', self.mouse_coords_changed)
  168. self.ui["mask_brush_size"].connect("value-changed", self.brush_size_changed)
  169. self.ui["layer_blend_mode"].connect("changed", self.layer_blend_mode_changed)
  170. def on_open_clicked(self, sender):
  171. self.ui["open_window"].show_all()
  172. def on_file_opened(self, sender, path=None):
  173. self.has_loaded = False
  174. self.image = None
  175. self.undo_position = 0
  176. self.undo_stack = []
  177. self.update_undo_state()
  178. self.ui["open_window"].hide()
  179. self.ui["control_reveal"].set_reveal_child(True)
  180. self.ui["control_reveal"].set_sensitive(True)
  181. self.ui["original_toggle"].set_sensitive(True)
  182. self.ui["export_image"].set_sensitive(True)
  183. self.ui["reset"].set_sensitive(True)
  184. self.ui["new_layer"].set_sensitive(True)
  185. self.ui["layers_reveal"].set_reveal_child(False)
  186. self.is_editing = True
  187. if(path == None):
  188. self.image_path = self.ui["open_chooser"].get_filename()
  189. else:
  190. self.image_path = path
  191. w = (self.ui["scroll_window"].get_allocated_width() - 12) * self.ui["zoom"].get_value()
  192. h = (self.ui["scroll_window"].get_allocated_height() - 12) * self.ui["zoom"].get_value()
  193. self.show_message("Loading Photo…", "Please wait while PhotoFiddle loads your photo", True)
  194. self.root.get_titlebar().set_subtitle("%s…" % (self.image_path))
  195. # Delete all layers, except the base
  196. for layer in self.layers:
  197. if(layer.name != "base"):
  198. self.ui["layers_list"].remove(layer.selector_row)
  199. self.layers = [self.base_layer,]
  200. self.select_layer(self.base_layer)
  201. thread = threading.Thread(target=self.open_image,
  202. args=(w, h))
  203. thread.start()
  204. def on_zoom_toggled(self, sender):
  205. state = sender.get_active()
  206. self.ui["zoom_reveal"].set_reveal_child(state)
  207. if(not state):
  208. self.ui["zoom"].set_value(1)
  209. def on_zoom_changed(self, sender):
  210. threading.Thread(target=self.zoom_delay).start()
  211. def zoom_delay(self):
  212. if(not self.is_zooming):
  213. self.is_zooming = True
  214. time.sleep(0.5)
  215. GLib.idle_add(self.on_preview_draw, None, None)
  216. self.is_zooming = False
  217. def on_file_canceled(self, sender):
  218. self.ui["open_window"].hide()
  219. def on_preview_draw(self, sender, arg):
  220. if(self.is_editing):
  221. w = (self.ui["scroll_window"].get_allocated_width() - 12) * self.ui["zoom"].get_value()
  222. h = (self.ui["scroll_window"].get_allocated_height() - 12) * self.ui["zoom"].get_value()
  223. if(self.pheight != h) or (self.pwidth != w):
  224. thread = threading.Thread(target=self.resize_preview,
  225. args=(w, h))
  226. thread.start()
  227. def on_preview_toggled(self, sender):
  228. if(sender.get_active()):
  229. threading.Thread(target=self.draw_hist, args=(self.original_image,)).start()
  230. self.show_original()
  231. else:
  232. threading.Thread(target=self.draw_hist, args=(self.image,)).start()
  233. self.show_current()
  234. def toggle_hist(self, sender):
  235. show = sender.get_active()
  236. self.ui["histogram_reveal"].set_reveal_child(show)
  237. def on_open(self, path):
  238. self.root.get_titlebar().set_subtitle("Raster Editor")
  239. if(path != None):
  240. # We have been sent here from another
  241. # activity, clear any existing profiles
  242. self.image_path = path
  243. dataPath = self.get_data_path()
  244. if(os.path.exists(dataPath)):
  245. os.unlink(dataPath)
  246. self.on_file_opened(None, path)
  247. def on_exit(self):
  248. if(self.is_exporting):
  249. return False
  250. else:
  251. fname = "/tmp/phf2-preview-%s.png" % getpass.getuser()
  252. if (os.path.exists(fname)):
  253. os.unlink(fname)
  254. self.root.get_titlebar().set_subtitle("")
  255. self.on_init()
  256. return True
  257. def show_original(self):
  258. self.ui["preview"].set_from_pixbuf(self.poriginal)
  259. if (not self.ui["original_toggle"].get_active()):
  260. self.ui["original_toggle"].set_active(True)
  261. def show_current(self):
  262. self.ui["preview"].set_from_pixbuf(self.pimage)
  263. if (self.ui["original_toggle"].get_active()):
  264. self.ui["original_toggle"].set_active(False)
  265. def on_layer_change(self, layer):
  266. print(layer, "changed")
  267. if(self.has_loaded):
  268. if(not self.change_occurred):
  269. self.change_occurred = True
  270. thread = threading.Thread(target=self.update_image, args=(False, layer))
  271. thread.start()
  272. else:
  273. self.additional_change_occurred = True
  274. def on_layer_mask_change(self, layer):
  275. if(self.has_loaded):
  276. if(not self.change_occurred):
  277. self.change_occurred = True
  278. threading.Thread(target=self.__on_layer_mask_change).start()
  279. self.save_image_data()
  280. def __on_layer_mask_change(self):
  281. image = self.get_selected_layer().layer_copy.copy()
  282. image = self.get_selected_layer().get_layer_mask_preview(image)
  283. self.image_is_dirty = True
  284. self.image = image
  285. self.update_preview()
  286. self.change_occurred = False
  287. self.image_is_dirty = False
  288. def on_lower_peak_toggled(self, sender):
  289. self.lower_peak_on = sender.get_active()
  290. if (self.lower_peak_on):
  291. thread = threading.Thread(target=self.process_peaks, args=(True,))
  292. thread.start()
  293. else:
  294. thread = threading.Thread(target=self.update_image, args=(True,))
  295. thread.start()
  296. def on_upper_peak_toggled(self, sender):
  297. self.upper_peak_on = sender.get_active()
  298. if (self.lower_peak_on):
  299. thread = threading.Thread(target=self.process_peaks, args=(True,))
  300. thread.start()
  301. else:
  302. thread = threading.Thread(target=self.update_image, args=(True,))
  303. thread.start()
  304. def image_opened(self):
  305. self.root.get_titlebar().set_subtitle("%s (%s Bit)" % (self.image_path, self.bit_depth))
  306. self.hide_message()
  307. def on_export_clicked(self, sender):
  308. Export.ExportDialog(self.root, self.builder, self.awidth, self.aheight, self.get_export_image, self.on_export_complete, self.image_path)
  309. def on_export_complete(self, filename):
  310. self.is_exporting = False
  311. self.on_export_state_change()
  312. self.show_message("Export Complete!", "Your photo has been exported to '%s'" % filename)
  313. def on_export_state_change(self):
  314. self.ui["control_reveal"].set_sensitive(not self.is_exporting)
  315. self.ui["export_image"].set_sensitive(not self.is_exporting)
  316. self.ui["open_button"].set_sensitive(not self.is_exporting)
  317. def on_export_started(self):
  318. self.is_exporting = True
  319. self.on_export_state_change()
  320. def update_undo_state(self):
  321. self.ui["undo"].set_sensitive(self.undo_position > 0)
  322. self.ui["redo"].set_sensitive(len(self.undo_stack)-1 > self.undo_position)
  323. def on_undo(self, sender):
  324. self.pre_undo_layer_name = self.get_selected_layer().name
  325. self.pre_undo_erasing = self.ui["mask_erase_toggle"].get_active()
  326. self.pre_undo_editing = self.ui["edit_layer_mask_button"].get_active()
  327. self.undo_position -= 1
  328. self.update_undo_state()
  329. self.update_from_undo_stack(self.undo_stack[self.undo_position])
  330. def on_redo(self, sender):
  331. self.pre_undo_layer_name = self.get_selected_layer().name
  332. self.pre_undo_erasing = self.ui["mask_erase_toggle"].get_active()
  333. self.pre_undo_editing = self.ui["edit_layer_mask_button"].get_active()
  334. self.undo_position += 1
  335. self.update_undo_state()
  336. self.update_from_undo_stack(self.undo_stack[self.undo_position])
  337. def on_reset(self, sender):
  338. for layer in self.layers:
  339. layer.reset_tools()
  340. ## Background Tasks ##
  341. def open_image(self, w, h):
  342. self.load_image_data()
  343. try:
  344. # Get Bit Depth
  345. imdepth = cv2.imread(self.image_path, 2 | 1)
  346. self.bit_depth = str(imdepth.dtype).replace("uint", "").replace("float", "")
  347. # Read the 8Bit Preview
  348. self.source_image = cv2.imread(self.image_path).astype(numpy.uint8)
  349. # Load the image
  350. self.resize_preview(w, h)
  351. except:
  352. pass
  353. while(self.image == None):
  354. time.sleep(1)
  355. GLib.idle_add(self.image_opened)
  356. time.sleep(1)
  357. self.has_loaded = True
  358. def resize_preview(self, w, h):
  359. if(self.source_image != None):
  360. # Inhibit undo stack to prevent
  361. # Adding an action on resize
  362. self.undoing = True
  363. self.original_image = self.source_image.copy()
  364. height, width = self.original_image.shape[:2]
  365. self.aheight = height
  366. self.awidth = width
  367. self.pheight = h
  368. self.pwidth = w
  369. # Get fitting size
  370. ratio = float(w)/width
  371. if(height*ratio > h):
  372. ratio = float(h)/height
  373. nw = width * ratio
  374. nh = height * ratio
  375. # Do quick ui resize
  376. # if(self.image != None):
  377. # # If we have an edited version, show that
  378. # image = cv2.resize(self.image, (int(nw), int(nh)), interpolation=cv2.INTER_NEAREST)
  379. # self.pimage = self.pimage = self.create_pixbuf(image)
  380. # GLib.idle_add(self.show_current)
  381. if(self.pimage != None and not self.is_scaling):
  382. self.is_scaling = True
  383. # If we have an edited version, show that
  384. image = self.pimage.scale_simple(int(nw), int(nh), GdkPixbuf.InterpType.NEAREST)
  385. self.pimage = image
  386. GLib.idle_add(self.show_current)
  387. self.is_scaling = False
  388. self.poriginal = GdkPixbuf.Pixbuf.new_from_file_at_scale(self.image_path,
  389. int(nw), int(nh), True)
  390. if(self.pimage == None):
  391. # Otherwise show the original
  392. GLib.idle_add(self.show_original)
  393. # Resize OPENCV Copy
  394. self.original_image = cv2.resize(self.original_image, (int(nw), int(nh)), interpolation = cv2.INTER_AREA)
  395. self.image_is_dirty = True
  396. self.image = self.original_image.copy()
  397. # Update image
  398. if (not self.change_occurred):
  399. self.change_occurred = True
  400. self.update_image()
  401. else:
  402. self.additional_change_occurred = True
  403. def update_image(self, immediate=False, changed_layer=None):
  404. if(not immediate):
  405. time.sleep(0.5)
  406. self.additional_change_occurred = False
  407. GLib.idle_add(self.start_work)
  408. image = numpy.copy(self.original_image)
  409. rst = time.time()
  410. self.image = self.run_stack(image, changed_layer=changed_layer)
  411. self.process_mask()
  412. self.image_is_dirty = False
  413. if(self.additional_change_occurred):
  414. self.update_image()
  415. else:
  416. self.save_image_data()
  417. threading.Thread(target=self.draw_hist, args=(self.image,)).start()
  418. self.process_peaks()
  419. self.update_preview()
  420. GLib.idle_add(self.stop_work)
  421. self.change_occurred = False
  422. self.undoing = False
  423. def run_stack(self, image, callback=None, changed_layer=None):
  424. if(not self.running_stack):
  425. self.running_stack = True
  426. baseImage = image.copy()
  427. image = None
  428. carry = True
  429. if(baseImage.shape == self.image.shape) and (changed_layer != None) and (changed_layer in self.layers):
  430. changed_layer_index = self.layers.index(changed_layer)
  431. if(changed_layer_index > 0):
  432. carry = False
  433. layer_under = self.layers[changed_layer_index -1]
  434. image = layer_under.layer_copy.copy()
  435. for layer in self.layers[changed_layer_index: ]:
  436. self.current_processing_layer_name = layer.name
  437. self.current_processing_layer_index = self.layers.index(layer)
  438. if(self.additional_change_occurred):
  439. break
  440. print(layer)
  441. image = layer.render_layer(baseImage, image, callback)
  442. if(carry):
  443. for layer in self.layers:
  444. self.current_processing_layer_name = layer.name
  445. self.current_processing_layer_index = self.layers.index(layer)
  446. if (self.additional_change_occurred):
  447. break
  448. print(layer)
  449. image = layer.render_layer(baseImage, image, callback)
  450. self.running_stack = False
  451. return image
  452. else:
  453. while(self.running_stack):
  454. time.sleep(1)
  455. return self.run_stack(image, callback)
  456. def update_preview(self):
  457. if(not self.is_scaling):
  458. image = self.create_pixbuf(self.image)
  459. self.pimage = image
  460. GLib.idle_add(self.show_current)
  461. def create_pixbuf(self, im):
  462. image = cv2.cvtColor(im.copy(), cv2.COLOR_BGR2RGB).astype(numpy.uint8)
  463. pb = GdkPixbuf.Pixbuf.new_from_data(image.tostring(),
  464. GdkPixbuf.Colorspace.RGB,
  465. False,
  466. 8,
  467. self.image.shape[1],
  468. self.image.shape[0],
  469. self.image.shape[2] * self.image.shape[1])
  470. return pb
  471. def draw_hist(self, image):
  472. path = "/tmp/phf2-hist-%s.png" % getpass.getuser()
  473. Histogram.Histogram.draw_hist(image, path)
  474. GLib.idle_add(self.update_hist_ui, path)
  475. def update_hist_ui(self, path):
  476. try:
  477. self.ui["histogram"].set_from_pixbuf(GdkPixbuf.Pixbuf.new_from_file(path))
  478. except:
  479. pass
  480. def process_peaks(self, do_update=False):
  481. bpp = float(str(self.image.dtype).replace("uint", "").replace("float", ""))
  482. if(self.upper_peak_on):
  483. self.image[(self.image == 2 ** bpp - 1).all(axis=2)] = 1
  484. if(self.lower_peak_on):
  485. self.image[(self.image == 0).all(axis=2)] = 2 ** bpp - 2
  486. if(do_update):
  487. self.update_preview()
  488. def process_mask(self, do_update=False):
  489. if(self.ui["edit_layer_mask_button"].get_active()):
  490. self.image = self.get_selected_layer().get_layer_mask_preview(self.image)
  491. if (do_update):
  492. self.update_preview()
  493. ## FILE STUFF ##
  494. def get_data_path(self):
  495. return "%s/.%s.pf2" % ("/".join(self.image_path.split("/")[:-1]), self.image_path.split("/")[-1:][0])
  496. def save_image_data(self):
  497. path = self.get_data_path()
  498. print(path)
  499. f = open(path, "w")
  500. layerDict = {}
  501. layerOrder = []
  502. for layer in self.layers:
  503. layerDict[layer.name] = layer.get_layer_dict()
  504. layerOrder.append(layer.name)
  505. if(not self.undoing) and (self.has_loaded):
  506. if(len(self.undo_stack)-1 != self.undo_position):
  507. self.undo_stack = self.undo_stack[:self.undo_position+1]
  508. if(self.undo_stack[self.undo_position] != {"layers": layerDict, "layer-order": layerOrder}):
  509. self.undo_stack += [{"layers": layerDict, "layer-order": layerOrder},]
  510. self.undo_position = len(self.undo_stack)-1
  511. GLib.idle_add(self.update_undo_state)
  512. data = {
  513. "path":self.image_path,
  514. "format-revision":1,
  515. "layers": layerDict,
  516. "layer-order": layerOrder
  517. }
  518. f.write(str(data))
  519. f.close()
  520. def load_image_data(self):
  521. path = self.get_data_path()
  522. loadDefaults = True
  523. if(os.path.exists(path)):
  524. f = open(path, 'r')
  525. sdata = f.read()
  526. try:
  527. data = ast.literal_eval(sdata)
  528. if(data["format-revision"] == 1):
  529. if("layer-order" not in data):
  530. # Backwards compatability
  531. data["layer-order"] = ["base",]
  532. for layer_name in data["layer-order"]:
  533. GLib.idle_add(self.create_layer_with_data, layer_name, data["layers"][layer_name])
  534. self.undo_stack = [data,]
  535. self.undo_position = 0
  536. loadDefaults = False
  537. except:
  538. GLib.idle_add(self.show_message,"Unable to load previous edits…",
  539. "The edit file for this photo is corrupted and could not be loaded.")
  540. if(loadDefaults):
  541. for layer in self.layers:
  542. for tool in layer.tools:
  543. GLib.idle_add(tool.reset)
  544. layerDict = {}
  545. layerOrder = []
  546. for layer in self.layers:
  547. layerDict[layer.name] = layer.get_layer_dict()
  548. layerOrder.append(layer.name)
  549. self.undo_stack = [{"layers": layerDict, "layer-order": layerOrder}, ]
  550. self.undo_position = 0
  551. GLib.idle_add(self.update_undo_state)
  552. #time.sleep(2)
  553. self.undoing = False
  554. def create_layer_with_data(self, layer, data):
  555. if (layer == "base"):
  556. self.base_layer.set_from_layer_dict(data)
  557. else:
  558. GLib.idle_add(self.show_layers)
  559. ilayer = self.create_layer(layer, False)
  560. ilayer.set_from_layer_dict(data)
  561. def update_from_undo_stack(self, data):
  562. self.undoing = True
  563. self.delete_all_editable_layers()
  564. print(data["layer-order"])
  565. for layer_name in data["layer-order"]:
  566. if (layer_name == "base"):
  567. self.base_layer.set_from_layer_dict(data["layers"][layer_name])
  568. else:
  569. ilayer = self.create_layer(layer_name, False, layer_name == self.pre_undo_layer_name, True)
  570. ilayer.set_from_layer_dict(data["layers"][layer_name])
  571. self.show_layers()
  572. def get_export_image(self, w, h):
  573. GLib.idle_add(self.on_export_started)
  574. GLib.idle_add(self.show_message, "Exporting Photo", "Please wait…", True, True)
  575. img = cv2.imread(self.image_path, 2 | 1)
  576. img = cv2.resize(img, (int(w), int(h)), interpolation=cv2.INTER_AREA)
  577. img = self.run_stack(img, self.export_progress_callback)
  578. GLib.idle_add(self.show_message, "Exporting Photo", "Saving to filesystem…", True, True)
  579. GLib.idle_add(self.update_message_progress, 1, 1)
  580. return img
  581. def export_progress_callback(self, name, count, current):
  582. layer_name = self.current_processing_layer_name
  583. if(layer_name == "base"):
  584. layer_name = "Base Layer"
  585. GLib.idle_add(self.show_message, "Exporting Photo", "%s: %s" % (layer_name,
  586. name), True, True)
  587. GLib.idle_add(self.update_message_progress, current+(self.current_processing_layer_index*count),
  588. count*len(self.layers))
  589. ## Layers Stuff ##
  590. def preview_dragged(self, widget, event):
  591. x, y = widget.translate_coordinates(self.ui["preview"], event.x, event.y)
  592. draw = self.ui["mask_draw_toggle"].get_active()
  593. erase = self.ui["mask_erase_toggle"].get_active()
  594. layer = self.get_selected_layer()
  595. if((draw or erase) and layer.editable and self.current_layer_path):
  596. if (x < 0.0):
  597. x = 0
  598. if (y < 0.0):
  599. y = 0
  600. pwidth = self.pimage.get_width()
  601. pheight = self.pimage.get_height()
  602. if (x > pwidth):
  603. x = pwidth
  604. if (y > pheight):
  605. y = pheight
  606. print(x, y)
  607. fill = (0, 0, 255)
  608. if(erase):
  609. fill = (255, 0, 0)
  610. self.draw_path(x, y, pheight, pwidth, fill)
  611. self.on_layer_mask_change(layer)
  612. self.mouse_down_coords_changed(widget, event)
  613. return True
  614. def draw_path(self, x, y, pheight, pwidth, fill):
  615. preview = self.current_layer_path.add_point(int(x), int(y), (pheight, pwidth, 3), fill)
  616. # Bits per pixel
  617. bpp = float(str(self.image.dtype).replace("uint", "").replace("float", ""))
  618. # Pixel value range
  619. np = float(2 ** bpp - 1)
  620. self.image[preview == 255] = np
  621. cv2.imwrite("/tmp/phf2-preview-%s-drag.png" % getpass.getuser(), self.image)
  622. temppbuf = GdkPixbuf.Pixbuf.new_from_file("/tmp/phf2-preview-%s-drag.png" % getpass.getuser())
  623. self.ui["preview"].set_from_pixbuf(temppbuf)
  624. def new_path(self, widget, event):
  625. draw = self.ui["mask_draw_toggle"].get_active()
  626. erase = self.ui["mask_erase_toggle"].get_active()
  627. layer = self.get_selected_layer()
  628. if((draw or erase) and layer.editable):
  629. print(self.pimage.get_width(), self.pimage.get_height())
  630. width = self.pimage.get_width()
  631. size = self.ui["mask_brush_size"].get_value()
  632. feather = self.ui["mask_brush_feather"].get_value()
  633. self.current_layer_path = layer.mask.get_new_path(size, feather, float(self.awidth)/float(width), draw)
  634. def mask_draw_toggled(self, widget):
  635. self.ui["mask_erase_toggle"].set_active(not widget.get_active())
  636. def mask_erase_toggled(self, widget):
  637. self.ui["mask_draw_toggle"].set_active(not widget.get_active())
  638. self.ui["mask_brush_feather_scale"].set_sensitive(not widget.get_active())
  639. def edit_mask_toggled(self, widget):
  640. self.ui["layer_mask_reveal"].set_reveal_child(widget.get_active())
  641. self.ui["mask_draw_toggle"].set_active(widget.get_active())
  642. self.ui["mask_erase_toggle"].set_active(False)
  643. self.ui["scroll_window"].queue_draw()
  644. if (widget.get_active()):
  645. thread = threading.Thread(target=self.process_mask, args=(True,))
  646. thread.start()
  647. elif(self.was_editing_mask):
  648. self.on_layer_change(self.get_selected_layer())
  649. self.was_editing_mask = widget.get_active()
  650. def update_layer_opacity(self, sender):
  651. layer = self.get_selected_layer()
  652. if(layer.opacity != sender.get_value()):
  653. layer.set_opacity(sender.get_value())
  654. def create_layer(self, layer_name, is_base, select = False, set_pre_undo_draw_state = False):
  655. layer = Layer.Layer(is_base, layer_name, self.on_layer_change)
  656. for tool in self.tools:
  657. tool_instance = tool()
  658. layer.add_tool(tool_instance)
  659. self.layers += [layer,]
  660. GLib.idle_add(self.create_layer_ui, layer, select, set_pre_undo_draw_state)
  661. return layer
  662. def create_layer_ui(self, layer, select, set_pre_undo_draw_state = False):
  663. layer_box = Gtk.HBox()
  664. layer_box.set_hexpand(False)
  665. layer_box.set_halign(Gtk.Align.START)
  666. layer_toggle = Gtk.CheckButton()
  667. layer_toggle.set_sensitive(layer.editable)
  668. layer_toggle.set_active(layer.enabled)
  669. layer_toggle.set_hexpand(False)
  670. layer_toggle.set_halign(Gtk.Align.START)
  671. layer_toggle.set_margin_right(8)
  672. layer_toggle.set_margin_left(8)
  673. layer_toggle.set_margin_top(4)
  674. layer_toggle.set_margin_bottom(4)
  675. layer_toggle.connect("toggled", self.toggle_layer, layer)
  676. layer_label = Gtk.Label()
  677. layer_label.set_label(layer.name)
  678. if(layer.name == "base"):
  679. layer_label.set_label("Base Layer")
  680. layer_label.set_hexpand(True)
  681. layer_label.set_halign(Gtk.Align.FILL)
  682. layer_label.set_margin_top(4)
  683. layer_label.set_margin_bottom(4)
  684. layer_box.add(layer_toggle)
  685. layer_box.add(layer_label)
  686. layer.show_all()
  687. layer_box.show_all()
  688. self.ui["layers_list"].add(layer_box)
  689. layer.selector_row = layer_box.get_parent()
  690. self.ui["tool_box_stack"].add(layer.tool_box)
  691. self.ui["tool_stack"].add(layer.tool_stack)
  692. if(select):
  693. self.select_layer(layer)
  694. if(set_pre_undo_draw_state) and self.get_selected_layer().editable:
  695. print(self.pre_undo_editing, self.pre_undo_erasing)
  696. self.ui["mask_erase_toggle"].set_active(self.pre_undo_erasing)
  697. self.ui["edit_layer_mask_button"].set_active(self.pre_undo_editing)
  698. def layer_ui_activated(self, widget, row):
  699. layer_index = row.get_index()
  700. self.ui["tool_stack"].set_visible_child(self.layers[layer_index].tool_stack)
  701. self.ui["tool_box_stack"].set_visible_child(self.layers[layer_index].tool_box)
  702. self.ui["remove_layer_button"].set_sensitive(self.layers[layer_index].editable)
  703. self.ui["edit_layer_mask_button"].set_sensitive(self.layers[layer_index].editable)
  704. self.ui["layer_opacity_scale"].set_sensitive(self.layers[layer_index].editable)
  705. self.ui["layer_blend_mode"].set_sensitive(self.layers[layer_index].editable)
  706. self.ui["layer_blend_mode"].set_active(["additive", "overlay"].index(self.layers[layer_index].blend_mode))
  707. self.ui["layer_opacity"].set_value(self.layers[layer_index].opacity)
  708. if(self.ui["edit_layer_mask_button"].get_active()):
  709. self.ui["edit_layer_mask_button"].set_active(self.layers[layer_index].editable)
  710. self.on_layer_change(self.last_selected_layer)
  711. self.last_selected_layer = self.get_selected_layer()
  712. def toggle_layer(self, sender, layer):
  713. layer.set_enabled(sender.get_active())
  714. def new_layer_button_clicked(self, widget):
  715. self.show_layers()
  716. # Allocate an un-used layer name
  717. layer_number = len(self.layers)
  718. layer_name = "Layer %i" % layer_number
  719. while(self.layer_exists(layer_name)):
  720. layer_number += 1
  721. layer_name = "Layer %i" % layer_number
  722. # Create the layer
  723. layer = self.create_layer(layer_name, False)
  724. layer.set_size(self.awidth, self.aheight)
  725. # Save changes
  726. threading.Thread(target=self.save_image_data).start()
  727. self.on_layer_change(layer)
  728. def remove_layer_button_clicked(self, widget):
  729. layer_row = self.ui["layers_list"].get_selected_row()
  730. selected_index = 1
  731. new_layers = []
  732. for layer in self.layers:
  733. if(layer.selector_row != layer_row):
  734. new_layers += [layer]
  735. else:
  736. selected_index = self.layers.index(layer)
  737. self.ui["layers_list"].remove(layer_row)
  738. self.layers = new_layers
  739. # Select next layer
  740. self.select_layer(self.layers[selected_index -1])
  741. if (len(self.layers) == 1):
  742. self.ui["layers_reveal"].set_reveal_child(False)
  743. if(widget != None):
  744. # Only do this if the layer was actualy deleted by the user
  745. # and not by the undo-redo system for example
  746. # Save changes
  747. threading.Thread(target=self.save_image_data).start()
  748. self.on_layer_change(self.get_selected_layer())
  749. def get_selected_layer(self):
  750. layer_row = self.ui["layers_list"].get_selected_row()
  751. for layer in self.layers:
  752. if (layer.selector_row == layer_row):
  753. return layer
  754. def layer_exists(self, layer_name):
  755. for layer in self.layers:
  756. if(layer.name == layer_name):
  757. return True
  758. return False
  759. def show_layers(self):
  760. self.ui["layers_reveal"].set_reveal_child(True)
  761. def select_layer(self, layer):
  762. self.ui["layers_list"].select_row(layer.selector_row)
  763. self.layer_ui_activated(self.ui["layers_list"], layer.selector_row)
  764. def delete_all_editable_layers(self):
  765. count = len(self.layers) -1
  766. while(len(self.layers) != 1):
  767. self.select_layer(self.layers[1])
  768. self.remove_layer_button_clicked(None)
  769. def draw_ui_brush_circle(self, widget, context):
  770. drawing = self.ui["edit_layer_mask_button"].get_active()
  771. if(drawing):
  772. size = self.ui["mask_brush_size"].get_value()
  773. if(self.mousedown):
  774. context.set_source_rgb(255, 0, 0)
  775. else:
  776. context.set_source_rgb(255, 255, 255)
  777. context.arc(self.mousex, self.mousey, size/2.0, 0.0, 2 * numpy.pi)
  778. context.stroke()
  779. def mouse_coords_changed(self, widget, event):
  780. self.mousedown = False
  781. self.mousex, self.mousey = event.x, event.y
  782. widget.queue_draw()
  783. def mouse_down_coords_changed(self, widget, event):
  784. self.mousedown = True
  785. self.mousex, self.mousey = widget.translate_coordinates(self.ui["scroll_window"], event.x, event.y)
  786. widget.queue_draw()
  787. def brush_size_changed(self, sender):
  788. self.ui["scroll_window"].queue_draw()
  789. def layer_blend_mode_changed(self, sender):
  790. layer = self.get_selected_layer()
  791. if(layer.blend_mode != sender.get_active_text().lower()):
  792. layer.set_blending_mode(sender.get_active_text().lower())