ShiftOS-C-/source/ShiftUI/Internal/Splitter.cs
MichaelTheShifter d40fed5ce2 Move ShiftUI source code to ShiftOS
This'll be a lot easier to work on.
2016-07-20 09:40:36 -04:00

707 lines
18 KiB
C#

// Permission is hereby granted, free of charge, to any person obtaining
// a copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to
// permit persons to whom the Software is furnished to do so, subject to
// the following conditions:
//
// The above copyright notice and this permission notice shall be
// included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
//
// Copyright (c) 2005-2008 Novell, Inc. (http://www.novell.com)
//
// Authors:
// Peter Dennis Bartok (pbartok@novell.com)
// Ivan N. Zlatev (contact i-nz.net)
//
//
// COMPLETE
#undef Debug
using System;
using System.ComponentModel;
using System.Drawing;
using System.Reflection;
using System.Runtime.InteropServices;
namespace ShiftUI {
[ComVisible (true)]
[ClassInterface (ClassInterfaceType.AutoDispatch)]
[DefaultEvent("SplitterMoved")]
[Designer("ShiftUI.Design.SplitterDesigner, " + Consts.AssemblySystem_Design, "System.ComponentModel.Design.IDesigner")]
[DefaultProperty("Dock")]
public class Splitter : Widget
{
#region Local Variables
static private Cursor splitter_ns;
static private Cursor splitter_we;
// XXX this "new" shouldn't be here. Widget shouldn't define border_style as internal.
new private BorderStyle border_style;
private int min_extra;
private int min_size;
private int max_size;
private int splitter_size; // Size (width or height) of our splitter control
private bool horizontal; // true if we've got a horizontal splitter
private Widget affected; // The control that the splitter resizes
private int split_requested; // If the user requests a position before we have ever laid out the doc
private int splitter_prev_move;
private Rectangle splitter_rectangle_moving;
private int moving_offset;
#endregion // Local Variables
#region Constructors
static Splitter() {
splitter_ns = Cursors.HSplit;
splitter_we = Cursors.VSplit;
}
public Splitter() {
min_extra = 25;
min_size = 25;
split_requested = -1;
splitter_size = 3;
horizontal = false;
SetStyle(Widgetstyles.Selectable, false);
Anchor = AnchorStyles.None;
Dock = DockStyle.Left;
Layout += new LayoutEventHandler(LayoutSplitter);
this.ParentChanged += new EventHandler(ReparentSplitter);
Cursor = splitter_we;
}
#endregion // Constructors
#region Public Instance Properties
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public override bool AllowDrop {
get {
return base.AllowDrop;
}
set {
base.AllowDrop = value;
}
}
[Browsable(false)]
[DefaultValue(AnchorStyles.None)]
[EditorBrowsable(EditorBrowsableState.Never)]
public override AnchorStyles Anchor {
get {
return AnchorStyles.None;
}
set {
; // MS doesn't set it
}
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public override Image BackgroundImage {
get {
return base.BackgroundImage;
}
set {
base.BackgroundImage = value;
}
}
[Browsable (false)]
[EditorBrowsable (EditorBrowsableState.Never)]
public override ImageLayout BackgroundImageLayout {
get { return base.BackgroundImageLayout; }
set { base.BackgroundImageLayout = value; }
}
[DispId(-504)]
[DefaultValue (BorderStyle.None)]
[MWFDescription("Sets the border style for the splitter")]
[MWFCategory("Appearance")]
public BorderStyle BorderStyle {
get {
return border_style;
}
set {
border_style = value;
switch(value) {
case BorderStyle.FixedSingle:
splitter_size = 4; // We don't get motion events for 1px wide windows on X11. sigh.
break;
case BorderStyle.Fixed3D:
value = BorderStyle.None;
splitter_size = 3;
break;
case BorderStyle.None:
splitter_size = 3;
break;
default:
throw new InvalidEnumArgumentException (string.Format("Enum argument value '{0}' is not valid for BorderStyle", value));
}
base.InternalBorderStyle = value;
}
}
[DefaultValue(DockStyle.Left)]
[Localizable(true)]
public override DockStyle Dock {
get {
return base.Dock;
}
set {
if (!Enum.IsDefined (typeof (DockStyle), value) || (value == DockStyle.None) || (value == DockStyle.Fill)) {
throw new ArgumentException("Splitter must be docked left, top, bottom or right");
}
if ((value == DockStyle.Top) || (value == DockStyle.Bottom)) {
horizontal = true;
Cursor = splitter_ns;
} else {
horizontal = false;
Cursor = splitter_we;
}
base.Dock = value;
}
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public override Font Font {
get {
return base.Font;
}
set {
base.Font = value;
}
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public override Color ForeColor {
get {
return base.ForeColor;
}
set {
base.ForeColor = value;
}
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public new ImeMode ImeMode {
get {
return base.ImeMode;
}
set {
base.ImeMode = value;
}
}
[DefaultValue(25)]
[Localizable(true)]
[MWFDescription("Sets minimum size of undocked window")]
[MWFCategory("Behaviour")]
public int MinExtra {
get {
return min_extra;
}
set {
min_extra = value;
}
}
[DefaultValue(25)]
[Localizable(true)]
[MWFDescription("Sets minimum size of the resized control")]
[MWFCategory("Behaviour")]
public int MinSize {
get {
return min_size;
}
set {
min_size = value;
}
}
internal int MaxSize {
get {
if (this.Parent == null)
return 0;
if (affected == null)
affected = AffectedControl;
int widths = 0;
int heights = 0;
int vert_offset = 0;
int horiz_offset = 0;
foreach (Widget c in this.Parent.Widgets) {
if (c != affected) {
switch (c.Dock) {
case DockStyle.Left:
case DockStyle.Right:
widths += c.Width;
if (c.Location.X < this.Location.X)
vert_offset += c.Width;
break;
case DockStyle.Top:
case DockStyle.Bottom:
heights += c.Height;
if (c.Location.Y < this.Location.Y)
horiz_offset += c.Height;
break;
}
}
}
if (horizontal) {
moving_offset = horiz_offset;
return Parent.ClientSize.Height - heights - MinExtra;
} else {
moving_offset = vert_offset;
return Parent.ClientSize.Width - widths - MinExtra;
}
}
}
[Browsable(false)]
[DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
[MWFDescription("Current splitter position")]
[MWFCategory("Layout")]
public int SplitPosition {
get {
affected = AffectedControl;
if (affected == null) {
return -1;
}
if (Capture) {
return CalculateSplitPosition();
}
if (horizontal) {
return affected.Height;
} else {
return affected.Width;
}
}
set {
if (value > MaxSize)
value = MaxSize;
if (value < MinSize)
value = MinSize;
affected = AffectedControl;
if (affected == null)
split_requested = value;
else {
if (horizontal)
affected.Height = value;
else
affected.Width = value;
OnSplitterMoved (new SplitterEventArgs (Left, Top, value, value));
}
}
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public new bool TabStop {
get { return base.TabStop; }
set { base.TabStop = value; }
}
[Bindable(false)]
[Browsable(false)]
[DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
[EditorBrowsable(EditorBrowsableState.Never)]
public override string Text {
get {
return base.Text;
}
set {
base.Text = value;
}
}
#endregion // Public Instance Properties
#region Protected Instance Properties
protected override CreateParams CreateParams {
get {
return base.CreateParams;
}
}
protected override Cursor DefaultCursor {
get { return base.DefaultCursor; }
}
protected override ImeMode DefaultImeMode {
get {
return ImeMode.Disable;
}
}
protected override Size DefaultSize {
get {
return new Size (3, 3);
}
}
#endregion // Protected Instance Properties
#region Public Instance Methods
public override string ToString() {
return base.ToString () + String.Format(", MinExtra: {0}, MinSize: {1}", min_extra, min_size);
}
#endregion // Public Instance Methods
#region Protected Instance Methods
protected override void OnKeyDown(KeyEventArgs e) {
base.OnKeyDown (e);
if (Capture && (e.KeyCode == Keys.Escape)) {
Capture = false;
SplitterEndMove (Point.Empty, true);
}
}
protected override void OnMouseDown(MouseEventArgs e) {
base.OnMouseDown (e);
// Only allow if we are set up properly
if (affected == null)
affected = AffectedControl;
max_size = MaxSize;
if (affected == null || e.Button != MouseButtons.Left)
return;
Capture = true;
SplitterBeginMove (Parent.PointToClient (PointToScreen (new Point (e.X, e.Y))));
}
protected override void OnMouseMove (MouseEventArgs e)
{
base.OnMouseMove (e);
if (!Capture || e.Button != MouseButtons.Left || affected == null)
return;
// We need our mouse coordinates relative to our parent
SplitterMove (Parent.PointToClient (PointToScreen (e.Location)));
}
protected override void OnMouseUp (MouseEventArgs e)
{
base.OnMouseUp (e);
if (!Capture || e.Button != MouseButtons.Left || affected == null)
return;
SplitterEndMove (Parent.PointToClient (PointToScreen (e.Location)), false);
Capture = false;
}
private void SplitterBeginMove (Point location)
{
splitter_rectangle_moving = new Rectangle (Bounds.X, Bounds.Y,
Width, Height);
splitter_prev_move = horizontal ? location.Y : location.X;
}
private void SplitterMove (Point location)
{
int currentMove = horizontal ? location.Y : location.X;
int delta = currentMove - splitter_prev_move;
Rectangle prev_location = splitter_rectangle_moving;
bool moved = false;
int min = this.MinSize + moving_offset;
int max = max_size + moving_offset;
if (horizontal) {
if (splitter_rectangle_moving.Y + delta > min && splitter_rectangle_moving.Y + delta < max) {
splitter_rectangle_moving.Y += delta;
moved = true;
} else {
// Ensure that the splitter is set to minimum or maximum position,
// even if the mouse "skips".
//
if (splitter_rectangle_moving.Y + delta <= min && splitter_rectangle_moving.Y != min) {
splitter_rectangle_moving.Y = min;
moved = true;
} else if (splitter_rectangle_moving.Y + delta >= max && splitter_rectangle_moving.Y != max) {
splitter_rectangle_moving.Y = max;
moved = true;
}
}
} else {
if (splitter_rectangle_moving.X + delta > min && splitter_rectangle_moving.X + delta < max) {
splitter_rectangle_moving.X += delta;
moved = true;
} else {
// Ensure that the splitter is set to minimum or maximum position,
// even if the mouse "skips".
//
if (splitter_rectangle_moving.X + delta <= min && splitter_rectangle_moving.X != min) {
splitter_rectangle_moving.X = min;
moved = true;
} else if (splitter_rectangle_moving.X + delta >= max && splitter_rectangle_moving.X != max) {
splitter_rectangle_moving.X = max;
moved = true;
}
}
}
if (moved) {
splitter_prev_move = currentMove;
OnSplitterMoving (new SplitterEventArgs (location.X, location.Y,
splitter_rectangle_moving.X,
splitter_rectangle_moving.Y));
XplatUI.DrawReversibleRectangle (this.Parent.Handle, prev_location, 1);
XplatUI.DrawReversibleRectangle (this.Parent.Handle, splitter_rectangle_moving, 1);
}
}
private void SplitterEndMove (Point location, bool cancel)
{
if (!cancel) {
// Resize the affected window
if (horizontal)
affected.Height = CalculateSplitPosition();
else
affected.Width = CalculateSplitPosition();
}
this.Parent.Refresh (); // to clean up the drag handle artifacts from all Widgets
SplitterEventArgs args = new SplitterEventArgs (location.X, location.Y,
splitter_rectangle_moving.X,
splitter_rectangle_moving.Y);
OnSplitterMoved (args);
}
protected virtual void OnSplitterMoved(SplitterEventArgs sevent) {
SplitterEventHandler eh = (SplitterEventHandler)(Events [SplitterMovedEvent]);
if (eh != null)
eh (this, sevent);
}
protected virtual void OnSplitterMoving(SplitterEventArgs sevent) {
SplitterEventHandler eh = (SplitterEventHandler)(Events [SplitterMovingEvent]);
if (eh != null)
eh (this, sevent);
}
protected override void SetBoundsCore(int x, int y, int width, int height, BoundsSpecified specified) {
// enforce our width / height
if (horizontal) {
splitter_size = height;
if (splitter_size < 1) {
splitter_size = 3;
}
base.SetBoundsCore (x, y, width, splitter_size, specified);
} else {
splitter_size = width;
if (splitter_size < 1) {
splitter_size = 3;
}
base.SetBoundsCore (x, y, splitter_size, height, specified);
}
}
#endregion // Protected Instance Methods
#region Private Properties and Methods
private Widget AffectedControl {
get {
if (Parent == null)
return null;
// Doc says the first control preceeding us in the zorder
for (int i = Parent.Widgets.GetChildIndex(this) + 1; i < Parent.Widgets.Count; i++) {
switch (Dock) {
case DockStyle.Top:
if (Top == Parent.Widgets[i].Bottom)
return Parent.Widgets[i];
break;
case DockStyle.Bottom:
if (Bottom == Parent.Widgets[i].Top)
return Parent.Widgets[i];
break;
case DockStyle.Left:
if (Left == Parent.Widgets[i].Right)
return Parent.Widgets[i];
break;
case DockStyle.Right:
if (Right == Parent.Widgets[i].Left)
return Parent.Widgets[i];
break;
}
}
return null;
}
}
private int CalculateSplitPosition() {
if (horizontal) {
if (Dock == DockStyle.Top)
return splitter_rectangle_moving.Y - affected.Top;
else
return affected.Bottom - splitter_rectangle_moving.Y - splitter_size;
} else {
if (Dock == DockStyle.Left)
return splitter_rectangle_moving.X - affected.Left;
else
return affected.Right - splitter_rectangle_moving.X - splitter_size;
}
}
internal override void OnPaintInternal (PaintEventArgs e) {
e.Graphics.FillRectangle(ThemeEngine.Current.ResPool.GetSolidBrush(this.BackColor), e.ClipRectangle);
}
private void LayoutSplitter(object sender, LayoutEventArgs e) {
affected = AffectedControl;
if (split_requested != -1) {
SplitPosition = split_requested;
split_requested = -1;
}
}
private void ReparentSplitter(object sender, EventArgs e) {
affected = null;
}
#endregion // Private Properties and Methods
#region Events
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public new event EventHandler BackgroundImageChanged {
add { base.BackgroundImageChanged += value; }
remove { base.BackgroundImageChanged -= value; }
}
[Browsable (false)]
[EditorBrowsable (EditorBrowsableState.Never)]
public new event EventHandler BackgroundImageLayoutChanged
{
add { base.BackgroundImageLayoutChanged += value; }
remove { base.BackgroundImageLayoutChanged -= value; }
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public new event EventHandler Enter {
add { base.Enter += value; }
remove { base.Enter -= value; }
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public new event EventHandler FontChanged {
add { base.FontChanged += value; }
remove { base.FontChanged -= value; }
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public new event EventHandler ForeColorChanged {
add { base.ForeColorChanged += value; }
remove { base.ForeColorChanged -= value; }
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public new event EventHandler ImeModeChanged {
add { base.ImeModeChanged += value; }
remove { base.ImeModeChanged -= value; }
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public new event KeyEventHandler KeyDown {
add { base.KeyDown += value; }
remove { base.KeyDown -= value; }
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public new event KeyPressEventHandler KeyPress {
add { base.KeyPress += value; }
remove { base.KeyPress -= value; }
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public new event KeyEventHandler KeyUp {
add { base.KeyUp += value; }
remove { base.KeyUp -= value; }
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public new event EventHandler Leave {
add { base.Leave += value; }
remove { base.Leave -= value; }
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public new event EventHandler TabStopChanged {
add { base.TabStopChanged += value; }
remove { base.TabStopChanged -= value; }
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
public new event EventHandler TextChanged {
add { base.TextChanged += value; }
remove { base.TextChanged -= value; }
}
static object SplitterMovedEvent = new object ();
static object SplitterMovingEvent = new object ();
public event SplitterEventHandler SplitterMoved {
add { Events.AddHandler (SplitterMovedEvent, value); }
remove { Events.RemoveHandler (SplitterMovedEvent, value); }
}
public event SplitterEventHandler SplitterMoving {
add { Events.AddHandler (SplitterMovingEvent, value); }
remove { Events.RemoveHandler (SplitterMovingEvent, value); }
}
#endregion // Events
}
}