# Copyright (c) 2016 Ruby-GNOME2 Project Team # This program is licenced under the same licence as Ruby-GNOME2. # =begin = Drawing Area GtkDrawingArea is a blank area where you can draw custom displays of various kinds. This demo has two drawing areas. The checkerboard area shows how you can just draw something; all you have to do is write a signal handler for expose_event, as shown here. The "scribble" area is a bit more advanced, and shows how to handle events such as button presses and mouse motion. Click the mouse and drag in the scribble area to draw squiggles. Resize the window to clear the area. =end class DrawingareaDemo def initialize(main_window) @window = Gtk::Window.new(:toplevel) @window.screen = main_window.screen @window.title = "Drawing Area" @window.border_width = 8 initialize_vbox create_the_checkerboard_area create_the_scribble_area end def run if !@window.visible? @window.show_all else @window.destroy end @window end private def initialize_vbox @vbox = Gtk::Box.new(:vertical, 8) @vbox.border_width = 8 @window.add(@vbox) end def create_da_in_frame(text) label = Gtk::Label.new label.set_markup(text, :use_underline => true) @vbox.pack_start(label, :expand => false, :fill => false, :padding => 0) frame = Gtk::Frame.new frame.shadow_type = :in @vbox.pack_start(frame, :expand => true, :fill => true, :padding => 0) da = Gtk::DrawingArea.new da.set_size_request(100, 100) frame.add(da) da end def checkerboard_draw(da, cr) check_size = 10 spacing = 2 # At the start of a draw handler, a clip region has been set on # the Cairo context, and the contents have been cleared to the # widget's background color. The docs for # gdk_window_begin_paint_region() give more details on how this # works. ycount = 0 xcount = 0 i = spacing while i < da.allocated_width j = spacing ycount = xcount % 2 # start with even/odd depending on row while j < da.allocated_height if ycount % 2 cr.set_source_rgb(0.45777, 0, 0.45777) else cr.set_source_rgb(1, 1, 1) end # if we are outside the clip, this will do nothing cr.rectangle(i, j, check_size, check_size) cr.fill j += check_size + spacing ++ycount end i += check_size + spacing ++xcount end end def create_the_checkerboard_area da = create_da_in_frame("Checkerboard pattern") da.signal_connect "draw" do |widget, cr| checkerboard_draw(widget, cr) end end def scribble_configure_event(da) @surface.destroy if @surface @surface = da.window.create_similar_surface(Cairo::CONTENT_COLOR, da.allocated_width, da.allocated_height) # initialize the surface to white cr = Cairo::Context.new(@surface) cr.set_source_rgb(1, 1, 1) cr.paint cr.destroy # we have handled the configure event, no need to further processing true end def scribble_draw(cr) # Redraw the screen from the surface cr.set_source(@surface, 0, 0) cr.paint false end def draw_brush(widget, x, y) cr = Cairo::Context.new(@surface) cr.rectangle(x - 3, y - 3, 6, 6) cr.fill cr.destroy widget.queue_draw_area(x - 3, y - 3, 6, 6) end def scribble_button_press_event(da, event) return false unless @surface draw_brush(da, event.x, event.y) if event.button == Gdk::BUTTON_PRIMARY true end def scribble_motion_notify_event(da, event) return false unless @surface _window, x, y, state = event.window.get_device_position(event.device) if (state & Gdk::EventMask::BUTTON_PRESS_MASK.to_i).nonzero? draw_brush(da, x, y) end true end def create_the_scribble_area da = create_da_in_frame("Scribble area") da.signal_connect "draw" do |_widget, cr| scribble_draw(cr) end da.signal_connect "configure-event" do |widget| scribble_configure_event(widget) end da.signal_connect "motion-notify-event" do |widget, event| scribble_motion_notify_event(widget, event) end da.signal_connect "button-press-event" do |widget, event| scribble_button_press_event(widget, event) end # da.events |= (Gdk::EventMask::LEAVE_NOTIFY_MASK | # Gdk::EventMask::BUTTON_PRESS_MASK | # Gdk::EventMask::POINTER_MOTION_MASK | # Gdk::EventMask::POINTER_MOTION_HINT_MASK) da.add_events([:leave_notify_mask, :button_press_mask, :pointer_motion_mask, :pointer_motion_hint_mask]) end end