b3a7fd2924
* sample/GtkDemo/DemoApplicationWindow.cs : New sample, Unfinished, 5% Done * sample/GtkDemo/DemoButtonBox.cs : New sample, * sample/GtkDemo/DemoColorSelection.cs : New sample, * sample/GtkDemo/DemoDialog.cs : New sample, * sample/GtkDemo/DemoDrawingArea.cs : New sample, * sample/GtkDemo/DemoEditableCells.cs : New sample, Needs clean up * sample/GtkDemo/DemoImages.cs : New sample, Unfinished, 75% Done * sample/GtkDemo/DemoItemFactory.cs : New sample, Unfinished, 5% Done * sample/GtkDemo/DemoListStore.cs : New sample, * sample/GtkDemo/DemoMain.cs : New sample, Unifnished, 60% Done * sample/GtkDemo/DemoMenus.cs : New sample, * sample/GtkDemo/DemoPanes.cs : New sample, * sample/GtkDemo/DemoPixbuf.cs : New sample, Needs clean up * sample/GtkDemo/DemoSizeGroup.cs : New sample, * sample/GtkDemo/DemoStockBrowser.cs : New sample, Unfinished, 20% Done * sample/GtkDemo/DemoTextView.cs : New sample, Unfinished, 99% Done * sample/GtkDemo/DemoTreeStore.cs : New sample, Unfinished, 95% Done * sample/GtkDemo/images : Several Images for the samples * sample/GtkDemo/Makefile : Very simple Makefile * sample/GtkDemo/README : Mentions explicitely unfinished state of port svn path=/trunk/gtk-sharp/; revision=19489
256 lines
7.4 KiB
C#
256 lines
7.4 KiB
C#
//
|
|
// DemoDrawingArea.cs, port of drawingarea.c from gtk-demo
|
|
//
|
|
// Author: Daniel Kornhauser <dkor@alum.mit.edu>
|
|
// Rachel Hestilow <hestilow@ximian.com>
|
|
//
|
|
// Copyright (C) 2003, Ximian Inc.
|
|
//
|
|
|
|
/* 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 ExposeEvent, 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.
|
|
*/
|
|
|
|
|
|
using System;
|
|
|
|
using Gtk;
|
|
using Gdk;
|
|
using GtkSharp;
|
|
|
|
namespace GtkDemo {
|
|
public class DemoDrawingArea
|
|
{
|
|
private Gtk.Window window;
|
|
private static Pixmap pixmap = null;
|
|
private static DrawingArea drawingArea;
|
|
private static DrawingArea drawingArea1;
|
|
|
|
public DemoDrawingArea ()
|
|
{
|
|
window = new Gtk.Window ("Drawing Area");
|
|
window.DeleteEvent += new DeleteEventHandler (WindowDelete);
|
|
window.BorderWidth = 8;
|
|
|
|
VBox vbox = new VBox (false, 8);
|
|
vbox.BorderWidth = 8;
|
|
window.Add (vbox);
|
|
|
|
// Create the checkerboard area
|
|
Label label = new Label(null);
|
|
label.Markup = "<u>Checkerboard pattern</u>";
|
|
vbox.PackStart (label, false, false, 0);
|
|
|
|
Frame frame = new Frame ();
|
|
frame.ShadowType = ShadowType.In;
|
|
vbox.PackStart (frame, true, true, 0);
|
|
|
|
drawingArea = new DrawingArea();
|
|
// set a minimum size
|
|
drawingArea.SetSizeRequest (100,100);
|
|
frame.Add(drawingArea);
|
|
drawingArea.ExposeEvent += new ExposeEventHandler (CheckerboardExpose);
|
|
|
|
// Create the scribble area
|
|
Label label1 = new Label ("<u>Scribble area</u>");
|
|
label1.UseMarkup = true;
|
|
vbox.PackStart (label1, false, false, 0);
|
|
|
|
Frame frame1 = new Frame ();
|
|
frame1.ShadowType = ShadowType.In;
|
|
vbox.PackStart (frame1, true, true, 0);
|
|
|
|
drawingArea1 = new DrawingArea ();
|
|
// set a minimun size
|
|
drawingArea1.SetSizeRequest (100, 100);
|
|
frame1.Add (drawingArea1);
|
|
// Signals used to handle backing pixmap
|
|
drawingArea1.ExposeEvent += new ExposeEventHandler (ScribbleExpose);
|
|
drawingArea1.ConfigureEvent += new ConfigureEventHandler (ScribbleConfigure);
|
|
// Event signals
|
|
drawingArea1.MotionNotifyEvent += new MotionNotifyEventHandler (ScribbleMotionNotify);
|
|
drawingArea1.ButtonPressEvent += new ButtonPressEventHandler (ScribbleButtonPress);
|
|
|
|
// Ask to receive events the drawing area doesn't normally
|
|
// subscribe to
|
|
|
|
drawingArea1.Events = (int)EventMask.LeaveNotifyMask |
|
|
(int)EventMask.ButtonPressMask |
|
|
(int)EventMask.PointerMotionMask |
|
|
(int)EventMask.PointerMotionHintMask;
|
|
|
|
window.ShowAll ();
|
|
}
|
|
|
|
private void WindowDelete (object o, DeleteEventArgs args)
|
|
{
|
|
window.Hide ();
|
|
window.Destroy ();
|
|
}
|
|
|
|
private void CheckerboardExpose (object o, ExposeEventArgs args)
|
|
{
|
|
|
|
// Defining the size of the Checks
|
|
const int CheckSize = 10;
|
|
const int Spacing = 2;
|
|
|
|
// Defining the color of the Checks
|
|
int i, j, xcount, ycount;
|
|
Gdk.GC gc, gc1, gc2;
|
|
Gdk.Color color = new Gdk.Color ();
|
|
EventExpose eventExpose = args.Event;
|
|
Gdk.Window window = eventExpose.window;
|
|
gc1 = new Gdk.GC (window);
|
|
|
|
color.red = 30000;
|
|
color.green = 0;
|
|
color.blue = 30000;
|
|
gc1.RgbFgColor = color;
|
|
|
|
gc2 = new Gdk.GC (window);
|
|
color.red = 65535;
|
|
color.green = 65535;
|
|
color.blue = 65535;
|
|
gc2.RgbFgColor = color;
|
|
|
|
|
|
// Start redrawing the Checkerboard
|
|
xcount = 0;
|
|
i = Spacing;
|
|
while (i < drawingArea.Allocation.width){
|
|
j = Spacing;
|
|
ycount = xcount % 2; //start with even/odd depending on row
|
|
while (j < drawingArea.Allocation.height){
|
|
gc = new Gdk.GC (window);
|
|
if (ycount % 2 != 0){
|
|
gc = gc1;}
|
|
else{
|
|
gc = gc2;}
|
|
window.DrawRectangle(gc, true, i, j, CheckSize, CheckSize);
|
|
|
|
j += CheckSize + Spacing;
|
|
++ycount;
|
|
}
|
|
i += CheckSize + Spacing;
|
|
++xcount;
|
|
}
|
|
// return true because we've handled this event, so no
|
|
// further processing is required.
|
|
SignalArgs sa = (SignalArgs) args;
|
|
sa.RetVal = true;
|
|
|
|
}
|
|
|
|
private void ScribbleExpose (object o, ExposeEventArgs args)
|
|
{
|
|
// We use the "ForegroundGC" for the widget since it already exists,
|
|
// but honestly any GC would work. The only thing to worry about
|
|
// is whether the GC has an inappropriate clip region set.
|
|
|
|
EventExpose eventExpose = args.Event;
|
|
Gdk.Window window = eventExpose.window;
|
|
Rectangle area = eventExpose.area;
|
|
|
|
window.DrawDrawable (drawingArea1.Style.ForegroundGC(StateType.Normal),
|
|
pixmap,
|
|
area.x, area.y,
|
|
area.x, area.y,
|
|
area.width, area.height);
|
|
SignalArgs sa = (SignalArgs) args;
|
|
sa.RetVal = false;
|
|
}
|
|
|
|
// Create a new pixmap of the appropriate size to store our scribbles
|
|
private void ScribbleConfigure (object o, ConfigureEventArgs args)
|
|
{
|
|
EventConfigure eventConfigure = args.Event;
|
|
Gdk.Window window = eventConfigure.window;
|
|
Rectangle allocation = drawingArea1.Allocation;
|
|
pixmap = new Pixmap (window,
|
|
allocation.width,
|
|
allocation.height,
|
|
-1);
|
|
// Initialize the pixmap to white
|
|
pixmap.DrawRectangle (drawingArea1.Style.WhiteGC, true, 0, 0,
|
|
allocation.width, allocation.height);
|
|
SignalArgs sa = (SignalArgs) args;
|
|
// We've handled the configure event, no need for further processing.
|
|
sa.RetVal = true;
|
|
}
|
|
|
|
private void ScribbleMotionNotify (object o, MotionNotifyEventArgs args)
|
|
{
|
|
|
|
// This call is very important; it requests the next motion event.
|
|
// If you don't call Window.GetPointer() you'll only get
|
|
// a single motion event. The reason is that we specified
|
|
// PointerMotionHintMask in drawingArea1.Events.
|
|
// If we hadn't specified that, we could just use ExposeEvent.x, ExposeEvent.y
|
|
// as the pointer location. But we'd also get deluged in events.
|
|
// By requesting the next event as we handle the current one,
|
|
// we avoid getting a huge number of events faster than we
|
|
// can cope.
|
|
|
|
int x, y;
|
|
ModifierType state;
|
|
EventMotion ev = args.Event;
|
|
Gdk.Window window = ev.window;
|
|
|
|
if (ev.is_hint != 0) {
|
|
ModifierType s;
|
|
window.GetPointer (out x, out y, out s);
|
|
state = s;
|
|
} else {
|
|
x = (int) ev.x;
|
|
y = (int) ev.y;
|
|
state = (ModifierType) ev.state;
|
|
}
|
|
|
|
if ((state & ModifierType.Button1Mask) != 0 && pixmap != null)
|
|
DrawBrush (x, y);
|
|
/* We've handled it, stop processing */
|
|
SignalArgs sa = (SignalArgs) args;
|
|
sa.RetVal = true;
|
|
}
|
|
|
|
|
|
// Draw a rectangle on the screen
|
|
static void DrawBrush (double x, double y)
|
|
{
|
|
Rectangle update_rect = new Rectangle ();
|
|
update_rect.x = (int) x - 3;
|
|
update_rect.y = (int) y - 3;
|
|
update_rect.width = 6;
|
|
update_rect.height = 6;
|
|
|
|
//Paint to the pixmap, where we store our state
|
|
pixmap.DrawRectangle (drawingArea1.Style.BlackGC, true,
|
|
update_rect.x, update_rect.y,
|
|
update_rect.width, update_rect.height);
|
|
drawingArea1.QueueDrawArea (update_rect.x, update_rect.y,
|
|
update_rect.width, update_rect.height);
|
|
}
|
|
|
|
private void ScribbleButtonPress (object o, ButtonPressEventArgs args)
|
|
{
|
|
EventButton ev = args.Event;
|
|
if (ev.button == 1 && pixmap != null)
|
|
DrawBrush (ev.x, ev.y);
|
|
//We've handled the event, stop processing
|
|
SignalArgs sa = (SignalArgs) args;
|
|
sa.RetVal = true;
|
|
}
|
|
}
|
|
}
|