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
|
# Copyright (C) 2009, Tutorius.org
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
from sugar.tutorius.actions import Action, DragWrapper
from sugar.tutorius.properties import TScreenClipProperty, \
TArrayProperty
import gtk
class ScreenClip(Action):
# Create the position as an array of fixed-size 2
position = TArrayProperty((0,0), 2, 2)
# Do the same for the tail position
clip_image = TScreenClipProperty("")
def __init__(self, position=None,
clip_image=None,
**kwargs):
"""
Shows a dialog with a given text, at the given position on the screen.
@param position A list of the form [x, y]
@param clip_image the screen clip image path
@param clip_rect rectangle of the clip image
"""
Action.__init__(self, **kwargs)
if position:
self.position = position
if clip_image:
self.clip_image = clip_image
self.overlay = None
self._bubble = None
def do(self, overlayer=None, **kwargs):
"""
Show the dialog
"""
if overlayer is None:
raise TypeError("Missing overlayer argument")
self.overlay = overlayer
if not self._bubble:
# Normal gtk widgets use the parent window to draw themselves.
# For normal layouts, this is good, but as we are using the layout
# for stacking widgets, the rendering order is sometimes wrong.
# Thus, by adding the Image widget in a visible EventBox, we ensure
# the Image is drawn in its own window, and stacking is correct.
x, y = self.position
self._bubble = gtk.EventBox()
self._bubble.set_visible_window(True)
image = gtk.Image()
image.set_from_file(self.clip_image)
self._bubble.add(image)
self._bubble.show_all()
self.overlay.put(self._bubble, x, y)
self.overlay.queue_draw()
def undo(self):
"""
Destroy the dialog
"""
if self._bubble:
self.overlay.remove(self._bubble)
self._bubble.destroy()
self._bubble = None
def enter_editmode(self, overlayer=None, *args, **kwargs):
"""
Enters edit mode. The action should display itself in some way,
without affecting the currently running application.
"""
if overlayer is None:
raise TypeError("Missing overlayer argument")
self.overlay = overlayer
assert not self._drag, "bubble action set to editmode twice"
x, y = self.position
if self.clip_image:
self._bubble = gtk.EventBox()
self._bubble.set_visible_window(True)
image = gtk.Image()
image.set_from_file(self.clip_image)
self._bubble.add(image)
self._bubble.show_all()
self.overlay.put(self._bubble, x, y)
self._drag = DragWrapper(self._bubble, self.position,
update_action_cb=self.update_property,
draggable=True)
def exit_editmode(self, *args):
if self._drag:
x, y = self._drag.position
self.position = (int(x), int(y))
self._drag.draggable = False
self._drag = None
if self._bubble:
self.overlay.remove(self._bubble)
self._bubble = None
self.overlay = None
__action__ = {
"name" : ScreenClip.__name__,
"display_name" : "Screen Capture Clip",
"icon" : "screenclip",
"class" : ScreenClip,
"mandatory_props" : []
}
|