10
10
just be for the background.
13
__copyright__ = "Copyright © 2005-2011 Canonical Ltd."
13
__copyright__ = "Copyright © 2005 Canonical Ltd."
14
14
__author__ = "Scott James Remnant <scott@ubuntu.com>"
19
from gi.repository import Gtk
20
from gi.repository import GObject
21
from gi.repository import Pango
22
from gi.repository import PangoCairo
25
# Cairo constants are not exported yet. These are taken from documentation.
26
CAIRO_LINE_CAP_BUTT = 0
27
CAIRO_LINE_CAP_ROUND = 1
28
CAIRO_LINE_CAP_SQUARE = 2
31
CAIRO_FILL_RULE_WINDING = 0
32
CAIRO_FILL_RULE_EVEN_ODD = 1
35
# Macro from Pango header.
37
return (d + 512) / 1000
40
class CellRendererGraph(Gtk.CellRendererPixbuf):
25
class CellRendererGraph(gtk.GenericCellRenderer):
41
26
"""Cell renderer for directed graph.
45
30
in_lines (start, end, colour) tuple list to draw inward lines,
46
31
out_lines (start, end, colour) tuple list to draw outward lines.
51
36
__gproperties__ = {
52
"node": ( GObject.TYPE_PYOBJECT, "node",
37
"node": ( gobject.TYPE_PYOBJECT, "node",
53
38
"revision node instruction",
54
GObject.PARAM_WRITABLE
39
gobject.PARAM_WRITABLE
56
"tags": ( GObject.TYPE_PYOBJECT, "tags",
41
"tags": ( gobject.TYPE_PYOBJECT, "tags",
57
42
"list of tags associated with the node",
58
GObject.PARAM_WRITABLE
43
gobject.PARAM_WRITABLE
60
"in-lines": ( GObject.TYPE_PYOBJECT, "in-lines",
45
"in-lines": ( gobject.TYPE_PYOBJECT, "in-lines",
61
46
"instructions to draw lines into the cell",
62
GObject.PARAM_WRITABLE
47
gobject.PARAM_WRITABLE
64
"out-lines": ( GObject.TYPE_PYOBJECT, "out-lines",
49
"out-lines": ( gobject.TYPE_PYOBJECT, "out-lines",
65
50
"instructions to draw lines out of the cell",
66
GObject.PARAM_WRITABLE
51
gobject.PARAM_WRITABLE
70
55
def do_set_property(self, property, value):
71
56
"""Set properties from GObject properties."""
72
57
if property.name == "node":
91
76
except AttributeError:
92
77
pango_ctx = widget.get_pango_context()
93
78
font_desc = widget.get_style().font_desc
94
metrics = pango_ctx.get_metrics(font_desc, None)
95
ascent = PANGO_PIXELS(metrics.get_ascent())
96
descent = PANGO_PIXELS(metrics.get_descent())
79
metrics = pango_ctx.get_metrics(font_desc)
81
ascent = pango.PIXELS(metrics.get_ascent())
82
descent = pango.PIXELS(metrics.get_descent())
97
84
self._box_size = ascent + descent + 6
98
85
return self._box_size
127
114
ctx.set_source_rgb(red, green, blue)
129
def do_activate(event, widget, path, bg_area, cell_area, flags):
130
"""Renderers cannot be activated; always return True."""
133
def do_editing_started(event, widget, path, fb_area, cell_area, flags):
134
"""Renderers cannot be edited; always return None."""
137
def do_get_size(self, widget, cell_area):
116
def on_get_size(self, widget, cell_area):
138
117
"""Return the size we need for this cell.
140
119
Each cell is drawn individually and is only as wide as it needs
149
128
# FIXME I have no idea how to use cell_area properly
150
129
return (0, 0, width, height)
152
def do_render(self, ctx, widget, bg_area, cell_area, flags):
131
def on_render(self, window, widget, bg_area, cell_area, exp_area, flags):
153
132
"""Render an individual cell.
155
134
Draws the cell contents using cairo, taking care to clip what we
163
142
instead of a pure diagonal ... this reduces confusion by an
164
143
incredible amount.
145
ctx = window.cairo_create()
166
146
ctx.rectangle(bg_area.x, bg_area.y, bg_area.width, bg_area.height)
169
149
box_size = self.box_size(widget)
171
151
ctx.set_line_width(box_size / 8)
152
ctx.set_line_cap(cairo.LINE_CAP_ROUND)
173
154
# Draw lines into the cell
174
155
for start, end, colour in self.in_lines:
175
156
self.render_line (ctx, cell_area, box_size,
176
157
bg_area.y, bg_area.height,
177
start, end, colour, flags)
179
160
# Draw lines out of the cell
180
161
for start, end, colour in self.out_lines:
181
162
self.render_line (ctx, cell_area, box_size,
182
163
bg_area.y + bg_area.height, bg_area.height,
183
start, end, colour, flags)
185
166
# Draw the revision node in the right column
186
167
(column, colour) = self.node
188
169
cell_area.y + cell_area.height / 2,
189
170
box_size / 4, 0, 2 * math.pi)
191
if flags & Gtk.CellRendererState.SELECTED:
192
ctx.set_source_rgb(1.0, 1.0, 1.0)
193
ctx.set_line_width(box_size / 4)
194
ctx.stroke_preserve()
195
ctx.set_line_width(box_size / 8)
197
172
self.set_colour(ctx, colour, 0.0, 0.5)
198
173
ctx.stroke_preserve()
203
178
self.render_tags(ctx, widget.create_pango_context(), cell_area, box_size)
205
def render_line(self, ctx, cell_area, box_size,
206
mid, height, start, end, colour, flags):
180
def render_line(self, ctx, cell_area, box_size, mid, height, start, end, colour):
207
181
if start is None:
208
ctx.set_line_cap(CAIRO_LINE_CAP_ROUND)
209
182
x = cell_area.x + box_size * end + box_size / 2
210
183
ctx.move_to(x, mid + height / 3)
211
184
ctx.line_to(x, mid + height / 3)
212
185
ctx.move_to(x, mid + height / 6)
213
186
ctx.line_to(x, mid + height / 6)
215
188
elif end is None:
216
ctx.set_line_cap(CAIRO_LINE_CAP_ROUND)
217
189
x = cell_area.x + box_size * start + box_size / 2
218
190
ctx.move_to(x, mid - height / 3)
219
191
ctx.line_to(x, mid - height / 3)
220
192
ctx.move_to(x, mid - height / 6)
221
193
ctx.line_to(x, mid - height / 6)
224
ctx.set_line_cap(CAIRO_LINE_CAP_BUTT)
225
195
startx = cell_area.x + box_size * start + box_size / 2
226
196
endx = cell_area.x + box_size * end + box_size / 2
228
198
ctx.move_to(startx, mid - height / 2)
230
200
if start - end == 0 :
231
ctx.line_to(endx, mid + height / 2 + 1)
201
ctx.line_to(endx, mid + height / 2)
233
203
ctx.curve_to(startx, mid - height / 5,
234
204
startx, mid - height / 5,
235
205
startx + (endx - startx) / 2, mid)
237
207
ctx.curve_to(endx, mid + height / 5,
238
208
endx, mid + height / 5 ,
239
endx, mid + height / 2 + 1)
241
if flags & Gtk.CellRendererState.SELECTED:
242
ctx.set_source_rgb(1.0, 1.0, 1.0)
243
ctx.set_line_width(box_size / 5)
244
ctx.stroke_preserve()
245
ctx.set_line_width(box_size / 8)
209
endx, mid + height / 2)
247
211
self.set_colour(ctx, colour, 0.0, 0.65)
251
214
def render_tags(self, ctx, pango_ctx, cell_area, box_size):
255
218
(column, colour) = self.node
257
font_desc = Pango.FontDescription()
258
font_desc.set_size(Pango.SCALE * 7)
220
font_desc = pango.FontDescription()
221
font_desc.set_size(pango.SCALE * 7)
260
tag_layout = Pango.Layout(pango_ctx)
223
tag_layout = pango.Layout(pango_ctx)
261
224
tag_layout.set_font_description(font_desc)
263
226
# The width of the tag label stack
266
229
for tag_idx, tag in enumerate(self.tags):
267
tag_layout.set_text(" " + tag + " ", -1)
230
tag_layout.set_text(" " + tag + " ")
268
231
text_width, text_height = tag_layout.get_pixel_size()
270
233
x0 = cell_area.x + \
293
256
self.set_colour(ctx, TAG_COLOUR_ID, 0.0, 0.5)
294
257
ctx.stroke_preserve()
296
ctx.set_fill_rule (CAIRO_FILL_RULE_EVEN_ODD)
259
ctx.set_fill_rule (cairo.FILL_RULE_EVEN_ODD)
297
260
self.set_colour(ctx, TAG_COLOUR_ID, 0.5, 1.0)
300
263
# Draw the tag text
301
264
self.set_colour(ctx, 0, 0.0, 0.0)
302
265
ctx.move_to(x0, y0)
303
PangoCairo.show_layout(ctx, tag_layout)
266
ctx.show_layout(tag_layout)