2013-05-11 07:25:30 +00:00
|
|
|
from kivy.app import App
|
|
|
|
from kivy.uix.boxlayout import BoxLayout
|
|
|
|
from kivy.uix.scatter import Scatter
|
|
|
|
from kivy.uix.popup import Popup
|
|
|
|
from kivy.properties import ObjectProperty, StringProperty
|
2016-03-15 10:17:09 +00:00
|
|
|
from kivy.graphics import Color, Point
|
2013-05-11 07:25:30 +00:00
|
|
|
|
|
|
|
from math import sqrt
|
|
|
|
from os import walk
|
2013-06-12 00:40:30 +00:00
|
|
|
from os.path import dirname, join
|
2013-05-11 07:25:30 +00:00
|
|
|
|
|
|
|
from kivy.lang import Builder
|
|
|
|
|
|
|
|
Builder.load_string('''
|
|
|
|
#:import os os
|
|
|
|
<Picture>:
|
|
|
|
# each time a picture is created, the image can delay the loading
|
|
|
|
# as soon as the image is loaded, ensure that the center is changed
|
|
|
|
# to the center of the screen.
|
|
|
|
on_size: self.center = app.main_root_widget.center
|
2015-03-21 15:58:01 +00:00
|
|
|
size: img.size
|
2013-05-11 07:25:30 +00:00
|
|
|
size_hint: None, None
|
2015-03-21 15:58:01 +00:00
|
|
|
on_touch_down: if self.collide_point(*args[1].pos): app.current_image = img
|
2013-05-11 07:25:30 +00:00
|
|
|
|
|
|
|
Image:
|
2015-03-21 15:58:01 +00:00
|
|
|
id: img
|
2013-05-11 07:25:30 +00:00
|
|
|
source: root.source
|
|
|
|
|
|
|
|
# create initial image to be 400 pixels width
|
|
|
|
size: 400, 400 / self.image_ratio
|
|
|
|
|
|
|
|
# add shadow background
|
|
|
|
canvas.before:
|
|
|
|
Color:
|
2015-03-21 15:58:01 +00:00
|
|
|
rgba: 1, 1, 1, 1
|
2013-05-11 07:25:30 +00:00
|
|
|
BorderImage:
|
|
|
|
source: '../demo/pictures/shadow32.png'
|
2015-03-21 15:58:01 +00:00
|
|
|
border: (36, 36, 36, 36)
|
|
|
|
size:(self.width + 72, self.height + 72)
|
|
|
|
pos: (-36, -36)
|
2013-05-11 07:25:30 +00:00
|
|
|
|
|
|
|
<ColorSelector>:
|
|
|
|
color: 1, 1, 1, 1
|
|
|
|
title: 'Color Slector'
|
|
|
|
content:content
|
|
|
|
BoxLayout:
|
|
|
|
id: content
|
|
|
|
orientation: 'vertical'
|
|
|
|
ColorPicker:
|
|
|
|
id: clr_picker
|
|
|
|
color: root.color
|
|
|
|
BoxLayout:
|
|
|
|
size_hint_y: None
|
|
|
|
height: '27sp'
|
|
|
|
Button:
|
|
|
|
text: 'ok'
|
|
|
|
on_release:
|
|
|
|
root.color = clr_picker.color
|
|
|
|
root.dismiss()
|
|
|
|
Button:
|
|
|
|
text: 'cancel'
|
|
|
|
on_release: root.dismiss()
|
|
|
|
|
|
|
|
<LeftPanel@BoxLayout>
|
|
|
|
orientation: 'vertical'
|
|
|
|
padding: '2pt'
|
|
|
|
canvas.before:
|
|
|
|
Color:
|
|
|
|
rgba: .5, .4, .9, .2
|
|
|
|
Rectangle:
|
|
|
|
pos: self.pos
|
|
|
|
size: self.size
|
|
|
|
Label:
|
|
|
|
size_hint_y: None
|
|
|
|
font_size: '18sp'
|
|
|
|
text_size: self.width, None
|
|
|
|
valign: 'middle'
|
|
|
|
halign: 'center'
|
|
|
|
height: self.texture.size[1] if self.texture else 10
|
2015-03-21 15:58:01 +00:00
|
|
|
text:
|
2016-03-18 01:50:13 +00:00
|
|
|
("Selected Image:\\n" + app.current_image.source.split(os.sep)[-1]
|
|
|
|
if app.current_image else 'None')
|
2013-05-11 07:25:30 +00:00
|
|
|
Button:
|
|
|
|
text: 'Brush'
|
|
|
|
size_hint_y: None
|
|
|
|
height: self.parent.width
|
|
|
|
on_release:
|
|
|
|
app.color_selector.open()
|
|
|
|
app.color_mode = 'brush'
|
|
|
|
Image:
|
|
|
|
color: app.color_selector.color
|
|
|
|
source: '../demo/touchtracer/particle.png'
|
|
|
|
allow_stretch: True
|
|
|
|
size: self.parent.size
|
|
|
|
pos: self.parent.pos
|
|
|
|
Button:
|
|
|
|
text: 'cursor'
|
|
|
|
on_release: app.color_mode = 'cursor'
|
|
|
|
Button:
|
|
|
|
text: 'clear'
|
|
|
|
on_release:
|
2016-03-18 02:05:12 +00:00
|
|
|
app.handle_clear()
|
2013-05-11 07:25:30 +00:00
|
|
|
|
|
|
|
<MainRootWidget>
|
|
|
|
current_image: None
|
|
|
|
client_area: client_area
|
|
|
|
RelativeLayout:
|
|
|
|
id: client_area
|
|
|
|
Splitter:
|
|
|
|
sizable_from: 'left'
|
|
|
|
size_hint: None, 1
|
|
|
|
width: '99dp'
|
|
|
|
LeftPanel:
|
|
|
|
|
|
|
|
''')
|
2015-02-02 01:09:20 +00:00
|
|
|
|
|
|
|
|
2013-05-11 07:25:30 +00:00
|
|
|
def calculate_points(x1, y1, x2, y2, steps=5):
|
|
|
|
dx = x2 - x1
|
|
|
|
dy = y2 - y1
|
|
|
|
dist = sqrt(dx * dx + dy * dy)
|
|
|
|
if dist < steps:
|
2017-06-29 15:12:21 +00:00
|
|
|
return
|
2013-05-11 07:25:30 +00:00
|
|
|
o = []
|
|
|
|
m = dist / steps
|
2013-06-12 00:40:30 +00:00
|
|
|
for i in range(1, int(m)):
|
2013-05-11 07:25:30 +00:00
|
|
|
mi = i / m
|
|
|
|
lastx = x1 + dx * mi
|
|
|
|
lasty = y1 + dy * mi
|
|
|
|
o.extend([lastx, lasty])
|
|
|
|
return o
|
|
|
|
|
2015-02-02 01:09:20 +00:00
|
|
|
|
2013-05-11 07:25:30 +00:00
|
|
|
class ColorSelector(Popup):
|
|
|
|
pass
|
|
|
|
|
2015-02-02 01:09:20 +00:00
|
|
|
|
2013-05-11 07:25:30 +00:00
|
|
|
class Picture(Scatter):
|
|
|
|
|
|
|
|
source = StringProperty(None)
|
|
|
|
'''path to the Image to be loaded
|
|
|
|
'''
|
|
|
|
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
super(Picture, self).__init__(**kwargs)
|
|
|
|
self._app = App.get_running_app()
|
|
|
|
|
|
|
|
def on_touch_down(self, touch):
|
|
|
|
_app = self._app
|
|
|
|
if (_app.color_mode[0] == 'c' or
|
2016-12-17 07:50:52 +00:00
|
|
|
not self.collide_point(*touch.pos)):
|
2013-05-11 07:25:30 +00:00
|
|
|
return super(Picture, self).on_touch_down(touch)
|
|
|
|
ud = touch.ud
|
|
|
|
ud['group'] = g = str(touch.uid)
|
2015-03-21 15:58:01 +00:00
|
|
|
_pos = list(self.ids.img.to_widget(*touch.pos))
|
2013-05-11 07:25:30 +00:00
|
|
|
_pos[0] += self.parent.x
|
2015-03-21 15:58:01 +00:00
|
|
|
with self.ids.img.canvas.after:
|
2013-05-11 07:25:30 +00:00
|
|
|
ud['color'] = Color(*_app.color_selector.color, group=g)
|
|
|
|
ud['lines'] = Point(points=(_pos),
|
2015-10-08 09:28:47 +00:00
|
|
|
source='../demo/touchtracer/particle.png',
|
2015-02-02 01:09:20 +00:00
|
|
|
pointsize=5, group=g)
|
2013-05-11 07:25:30 +00:00
|
|
|
touch.grab(self)
|
|
|
|
return True
|
|
|
|
|
|
|
|
def on_touch_move(self, touch):
|
|
|
|
if touch.grab_current is not self:
|
|
|
|
return
|
|
|
|
_app = self._app
|
|
|
|
if _app.color_mode[0] == 'c' or not self.collide_point(*touch.pos):
|
|
|
|
return super(Picture, self).on_touch_move(touch)
|
|
|
|
ud = touch.ud
|
2015-03-21 15:58:01 +00:00
|
|
|
_pos = list(self.ids.img.to_widget(*touch.pos))
|
2013-05-11 07:25:30 +00:00
|
|
|
_pos[0] += self.parent.x
|
|
|
|
points = ud['lines'].points
|
|
|
|
oldx, oldy = points[-2], points[-1]
|
|
|
|
points = calculate_points(oldx, oldy, _pos[0], _pos[1])
|
|
|
|
if points:
|
|
|
|
try:
|
|
|
|
lp = ud['lines'].add_point
|
2013-06-12 00:40:30 +00:00
|
|
|
for idx in range(0, len(points), 2):
|
2015-02-02 01:09:20 +00:00
|
|
|
lp(points[idx], points[idx + 1])
|
2013-05-11 07:25:30 +00:00
|
|
|
except GraphicException:
|
|
|
|
pass
|
|
|
|
|
|
|
|
def on_touch_up(self, touch):
|
|
|
|
if touch.grab_current is not self:
|
|
|
|
return
|
|
|
|
_app = self._app
|
|
|
|
if _app.color_mode[0] == 'c':
|
|
|
|
return super(Picture, self).on_touch_up(touch)
|
|
|
|
touch.ungrab(self)
|
|
|
|
ud = touch.ud
|
|
|
|
self.canvas.remove_group(ud['group'])
|
|
|
|
|
|
|
|
|
|
|
|
class MainRootWidget(BoxLayout):
|
|
|
|
|
|
|
|
clent_area = ObjectProperty(None)
|
2015-02-02 01:09:20 +00:00
|
|
|
# The Client Area in which all editing is Done
|
2013-05-11 07:25:30 +00:00
|
|
|
|
|
|
|
def on_parent(self, instance, parent):
|
|
|
|
if parent:
|
2013-06-12 00:40:30 +00:00
|
|
|
_dir = join(dirname(__file__), 'lists/fruit_images/')
|
2013-05-11 07:25:30 +00:00
|
|
|
for image in list(walk(_dir))[0][2]:
|
|
|
|
if image.find('512') > -1:
|
2015-02-02 01:09:20 +00:00
|
|
|
self.client_area.add_widget(Picture(source=_dir + image))
|
|
|
|
|
2013-05-11 07:25:30 +00:00
|
|
|
|
|
|
|
class MainApp(App):
|
|
|
|
|
|
|
|
main_root_widget = ObjectProperty(None)
|
|
|
|
# we will be accessing this later as App.main_root_widget
|
|
|
|
|
|
|
|
current_image = ObjectProperty(None)
|
|
|
|
'''This is a handle to the currently selected image on which the effects
|
|
|
|
would be applied.'''
|
|
|
|
|
|
|
|
color_mode = StringProperty('cursor')
|
|
|
|
'''This defines the current mode `brush` or `cursor`. `brush` mode allows
|
|
|
|
adding brush strokes to the currently selected Image.
|
|
|
|
'''
|
|
|
|
|
|
|
|
def build(self):
|
|
|
|
self.color_selector = ColorSelector()
|
|
|
|
self.main_root_widget = MainRootWidget()
|
|
|
|
return self.main_root_widget
|
|
|
|
|
2016-03-18 02:05:12 +00:00
|
|
|
def handle_clear(self):
|
|
|
|
if self.current_image:
|
|
|
|
self.current_image.canvas.after.clear()
|
2013-05-11 07:25:30 +00:00
|
|
|
|
2016-12-17 09:41:12 +00:00
|
|
|
|
2013-05-11 07:25:30 +00:00
|
|
|
if __name__ == '__main__':
|
|
|
|
MainApp().run()
|