mirror of
https://github.com/Ryujinx/GtkSharp.git
synced 2024-12-29 17:45:28 +00:00
4afdaf6602
* glib/Value.cs : Add GParam support and beef up the Boxed type marshaling to support types with New methods via reflection. svn path=/trunk/gtk-sharp/; revision=102064
520 lines
16 KiB
C#
Executable file
520 lines
16 KiB
C#
Executable file
// GLib.Value.cs - GLib Value class implementation
|
|
//
|
|
// Author: Mike Kestner <mkestner@speakeasy.net>
|
|
//
|
|
// Copyright (c) 2001 Mike Kestner
|
|
// Copyright (c) 2003-2004 Novell, Inc.
|
|
//
|
|
// This program is free software; you can redistribute it and/or
|
|
// modify it under the terms of version 2 of the Lesser GNU General
|
|
// Public License as published by the Free Software Foundation.
|
|
//
|
|
// 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
|
|
// Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
// License along with this program; if not, write to the
|
|
// Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
// Boston, MA 02111-1307, USA.
|
|
|
|
|
|
namespace GLib {
|
|
|
|
using System;
|
|
using System.Collections;
|
|
using System.Reflection;
|
|
using System.Runtime.InteropServices;
|
|
|
|
[StructLayout (LayoutKind.Sequential)]
|
|
public struct Value : IDisposable {
|
|
|
|
IntPtr type;
|
|
long pad_1;
|
|
long pad_2;
|
|
|
|
public static Value Empty;
|
|
|
|
public Value (GLib.GType gtype)
|
|
{
|
|
type = IntPtr.Zero;
|
|
pad_1 = pad_2 = 0;
|
|
g_value_init (ref this, gtype.Val);
|
|
}
|
|
|
|
public Value (object obj)
|
|
{
|
|
type = IntPtr.Zero;
|
|
pad_1 = pad_2 = 0;
|
|
|
|
GType gtype = (GType) obj.GetType ();
|
|
g_value_init (ref this, gtype.Val);
|
|
Val = obj;
|
|
}
|
|
|
|
public Value (bool val) : this (GType.Boolean)
|
|
{
|
|
g_value_set_boolean (ref this, val);
|
|
}
|
|
|
|
public Value (int val) : this (GType.Int)
|
|
{
|
|
g_value_set_int (ref this, val);
|
|
}
|
|
|
|
public Value (uint val) : this (GType.UInt)
|
|
{
|
|
g_value_set_uint (ref this, val);
|
|
}
|
|
|
|
public Value (ushort val) : this (GType.UInt)
|
|
{
|
|
g_value_set_uint (ref this, val);
|
|
}
|
|
|
|
public Value (long val) : this (GType.Int64)
|
|
{
|
|
g_value_set_int64 (ref this, val);
|
|
}
|
|
|
|
public Value (ulong val) : this (GType.UInt64)
|
|
{
|
|
g_value_set_uint64 (ref this, val);
|
|
}
|
|
|
|
[Obsolete ("Replaced by Value(object) constructor")]
|
|
public Value (EnumWrapper wrap, string type_name)
|
|
{
|
|
type = IntPtr.Zero;
|
|
pad_1 = pad_2 = 0;
|
|
IntPtr native = GLib.Marshaller.StringToPtrGStrdup (type_name);
|
|
gtksharp_value_create_from_type_name (ref this, native);
|
|
GLib.Marshaller.Free (native);
|
|
if (wrap.flags)
|
|
g_value_set_flags (ref this, (uint) (int) wrap);
|
|
else
|
|
g_value_set_enum (ref this, (int) wrap);
|
|
}
|
|
|
|
public Value (float val) : this (GType.Float)
|
|
{
|
|
g_value_set_float (ref this, val);
|
|
}
|
|
|
|
public Value (double val) : this (GType.Double)
|
|
{
|
|
g_value_set_double (ref this, val);
|
|
}
|
|
|
|
public Value (string val) : this (GType.String)
|
|
{
|
|
IntPtr native_val = GLib.Marshaller.StringToPtrGStrdup (val);
|
|
g_value_set_string (ref this, native_val);
|
|
GLib.Marshaller.Free (native_val);
|
|
}
|
|
|
|
public Value (IntPtr val) : this (GType.Pointer)
|
|
{
|
|
g_value_set_pointer (ref this, val);
|
|
}
|
|
|
|
public Value (Opaque val, string type_name)
|
|
{
|
|
type = IntPtr.Zero;
|
|
pad_1 = pad_2 = 0;
|
|
IntPtr native = GLib.Marshaller.StringToPtrGStrdup (type_name);
|
|
gtksharp_value_create_from_type_name (ref this, native);
|
|
GLib.Marshaller.Free (native);
|
|
g_value_set_boxed (ref this, val.Handle);
|
|
}
|
|
|
|
public Value (GLib.Object val) : this (val == null ? GType.Object : val.NativeType)
|
|
{
|
|
g_value_set_object (ref this, val == null ? IntPtr.Zero : val.Handle);
|
|
}
|
|
|
|
public Value (GLib.GInterfaceAdapter val) : this (val == null ? GType.Object : val.GType)
|
|
{
|
|
g_value_set_object (ref this, val == null ? IntPtr.Zero : val.Handle);
|
|
}
|
|
|
|
public Value (GLib.Object obj, string prop_name)
|
|
{
|
|
type = IntPtr.Zero;
|
|
pad_1 = pad_2 = 0;
|
|
IntPtr prop = GLib.Marshaller.StringToPtrGStrdup (prop_name);
|
|
gtksharp_value_create_from_property (ref this, obj.Handle, prop);
|
|
GLib.Marshaller.Free (prop);
|
|
}
|
|
|
|
[Obsolete]
|
|
public Value (GLib.Object obj, string prop_name, EnumWrapper wrap)
|
|
{
|
|
type = IntPtr.Zero;
|
|
pad_1 = pad_2 = 0;
|
|
IntPtr native = GLib.Marshaller.StringToPtrGStrdup (prop_name);
|
|
gtksharp_value_create_from_type_and_property (ref this, obj.NativeType.Val, native);
|
|
GLib.Marshaller.Free (native);
|
|
if (wrap.flags)
|
|
g_value_set_flags (ref this, (uint) (int) wrap);
|
|
else
|
|
g_value_set_enum (ref this, (int) wrap);
|
|
}
|
|
|
|
[Obsolete]
|
|
public Value (IntPtr obj, string prop_name, Opaque val)
|
|
{
|
|
type = IntPtr.Zero;
|
|
pad_1 = pad_2 = 0;
|
|
IntPtr native = GLib.Marshaller.StringToPtrGStrdup (prop_name);
|
|
gtksharp_value_create_from_property (ref this, obj, native);
|
|
GLib.Marshaller.Free (native);
|
|
g_value_set_boxed (ref this, val.Handle);
|
|
}
|
|
|
|
public Value (string[] val) : this (new GLib.GType (g_strv_get_type ()))
|
|
{
|
|
if (val == null) {
|
|
g_value_set_boxed (ref this, IntPtr.Zero);
|
|
return;
|
|
}
|
|
|
|
IntPtr native_array = Marshal.AllocHGlobal ((val.Length + 1) * IntPtr.Size);
|
|
for (int i = 0; i < val.Length; i++)
|
|
Marshal.WriteIntPtr (native_array, i * IntPtr.Size, GLib.Marshaller.StringToPtrGStrdup (val[i]));
|
|
Marshal.WriteIntPtr (native_array, val.Length * IntPtr.Size, IntPtr.Zero);
|
|
|
|
g_value_set_boxed (ref this, native_array);
|
|
|
|
for (int i = 0; i < val.Length; i++)
|
|
GLib.Marshaller.Free (Marshal.ReadIntPtr (native_array, i * IntPtr.Size));
|
|
Marshal.FreeHGlobal (native_array);
|
|
}
|
|
|
|
|
|
public void Dispose ()
|
|
{
|
|
g_value_unset (ref this);
|
|
}
|
|
|
|
public void Init (GLib.GType gtype)
|
|
{
|
|
g_value_init (ref this, gtype.Val);
|
|
}
|
|
|
|
|
|
public static explicit operator bool (Value val)
|
|
{
|
|
return g_value_get_boolean (ref val);
|
|
}
|
|
|
|
public static explicit operator int (Value val)
|
|
{
|
|
return g_value_get_int (ref val);
|
|
}
|
|
|
|
public static explicit operator uint (Value val)
|
|
{
|
|
return g_value_get_uint (ref val);
|
|
}
|
|
|
|
public static explicit operator ushort (Value val)
|
|
{
|
|
return (ushort) g_value_get_uint (ref val);
|
|
}
|
|
|
|
public static explicit operator long (Value val)
|
|
{
|
|
return g_value_get_int64 (ref val);
|
|
}
|
|
|
|
public static explicit operator ulong (Value val)
|
|
{
|
|
return g_value_get_uint64 (ref val);
|
|
}
|
|
|
|
[Obsolete ("Replaced by Enum cast")]
|
|
public static explicit operator EnumWrapper (Value val)
|
|
{
|
|
if (glibsharp_value_holds_flags (ref val))
|
|
return new EnumWrapper ((int)g_value_get_flags (ref val), true);
|
|
else
|
|
return new EnumWrapper (g_value_get_enum (ref val), false);
|
|
}
|
|
|
|
public static explicit operator Enum (Value val)
|
|
{
|
|
if (glibsharp_value_holds_flags (ref val))
|
|
return (Enum)Enum.ToObject (GType.LookupType (val.type), g_value_get_flags (ref val));
|
|
else
|
|
return (Enum)Enum.ToObject (GType.LookupType (val.type), g_value_get_enum (ref val));
|
|
}
|
|
|
|
public static explicit operator float (Value val)
|
|
{
|
|
return g_value_get_float (ref val);
|
|
}
|
|
|
|
public static explicit operator double (Value val)
|
|
{
|
|
return g_value_get_double (ref val);
|
|
}
|
|
|
|
public static explicit operator string (Value val)
|
|
{
|
|
IntPtr str = g_value_get_string (ref val);
|
|
return str == IntPtr.Zero ? null : GLib.Marshaller.Utf8PtrToString (str);
|
|
}
|
|
|
|
public static explicit operator IntPtr (Value val)
|
|
{
|
|
return g_value_get_pointer (ref val);
|
|
}
|
|
|
|
public static explicit operator GLib.Opaque (Value val)
|
|
{
|
|
return GLib.Opaque.GetOpaque (g_value_get_boxed (ref val), (Type) new GType (val.type), false);
|
|
}
|
|
|
|
public static explicit operator GLib.Boxed (Value val)
|
|
{
|
|
return new GLib.Boxed (g_value_get_boxed (ref val));
|
|
}
|
|
|
|
public static explicit operator GLib.Object (Value val)
|
|
{
|
|
return GLib.Object.GetObject (g_value_get_object (ref val), false);
|
|
}
|
|
|
|
[Obsolete ("Replaced by GLib.Object cast")]
|
|
public static explicit operator GLib.UnwrappedObject (Value val)
|
|
{
|
|
return new UnwrappedObject (g_value_get_object (ref val));
|
|
}
|
|
|
|
public static explicit operator string[] (Value val)
|
|
{
|
|
IntPtr native_array = g_value_get_boxed (ref val);
|
|
if (native_array == IntPtr.Zero)
|
|
return null;
|
|
|
|
int count = 0;
|
|
while (Marshal.ReadIntPtr (native_array, count * IntPtr.Size) != IntPtr.Zero)
|
|
count++;
|
|
string[] strings = new string[count];
|
|
for (int i = 0; i < count; i++)
|
|
strings[i] = GLib.Marshaller.Utf8PtrToString (Marshal.ReadIntPtr (native_array, i * IntPtr.Size));
|
|
return strings;
|
|
}
|
|
|
|
object ToBoxed ()
|
|
{
|
|
IntPtr boxed_ptr = g_value_get_boxed (ref this);
|
|
Type t = GType.LookupType (type);
|
|
if (t == null)
|
|
throw new Exception ("Unknown type " + new GType (type).ToString ());
|
|
else if (t.IsSubclassOf (typeof (GLib.Opaque)))
|
|
return (GLib.Opaque) this;
|
|
|
|
MethodInfo mi = t.GetMethod ("New", BindingFlags.Static | BindingFlags.Public | BindingFlags.FlattenHierarchy);
|
|
if (mi == null)
|
|
return Marshal.PtrToStructure (boxed_ptr, t);
|
|
else
|
|
return mi.Invoke (null, new object[] {boxed_ptr});
|
|
}
|
|
|
|
public object Val
|
|
{
|
|
get {
|
|
if (type == GType.Boolean.Val)
|
|
return (bool) this;
|
|
else if (type == GType.Int.Val)
|
|
return (int) this;
|
|
else if (type == GType.UInt.Val)
|
|
return (uint) this;
|
|
else if (type == GType.Int64.Val)
|
|
return (long) this;
|
|
else if (type == GType.UInt64.Val)
|
|
return (ulong) this;
|
|
else if (g_type_is_a (type, GType.Enum.Val) ||
|
|
g_type_is_a (type, GType.Flags.Val))
|
|
return (Enum) this;
|
|
else if (type == GType.Float.Val)
|
|
return (float) this;
|
|
else if (type == GType.Double.Val)
|
|
return (double) this;
|
|
else if (type == GType.String.Val)
|
|
return (string) this;
|
|
else if (type == GType.Pointer.Val)
|
|
return (IntPtr) this;
|
|
else if (type == GType.Param.Val)
|
|
return g_value_get_param (ref this);
|
|
else if (type == ManagedValue.GType.Val)
|
|
return ManagedValue.ObjectForWrapper (g_value_get_boxed (ref this));
|
|
else if (g_type_is_a (type, GType.Object.Val))
|
|
return (GLib.Object) this;
|
|
else if (g_type_is_a (type, GType.Boxed.Val))
|
|
return ToBoxed ();
|
|
else
|
|
throw new Exception ("Unknown type " + new GType (type).ToString ());
|
|
}
|
|
set {
|
|
if (type == GType.Boolean.Val)
|
|
g_value_set_boolean (ref this, (bool) value);
|
|
else if (type == GType.Int.Val)
|
|
g_value_set_int (ref this, (int) value);
|
|
else if (type == GType.UInt.Val)
|
|
g_value_set_uint (ref this, (uint) value);
|
|
else if (type == GType.Int64.Val)
|
|
g_value_set_int64 (ref this, (long) value);
|
|
else if (type == GType.UInt64.Val)
|
|
g_value_set_uint64 (ref this, (ulong) value);
|
|
else if (g_type_is_a (type, GType.Enum.Val))
|
|
g_value_set_enum (ref this, (int)value);
|
|
else if (g_type_is_a (type, GType.Flags.Val))
|
|
g_value_set_flags (ref this, (uint)(int)value);
|
|
else if (type == GType.Float.Val)
|
|
g_value_set_float (ref this, (float) value);
|
|
else if (type == GType.Double.Val)
|
|
g_value_set_double (ref this, (double) value);
|
|
else if (type == GType.String.Val) {
|
|
IntPtr native = GLib.Marshaller.StringToPtrGStrdup ((string)value);
|
|
g_value_set_string (ref this, native);
|
|
GLib.Marshaller.Free (native);
|
|
} else if (type == GType.Pointer.Val) {
|
|
if (value is IWrapper) {
|
|
g_value_set_pointer (ref this, ((IWrapper)value).Handle);
|
|
return;
|
|
}
|
|
IntPtr buf = Marshal.AllocHGlobal (Marshal.SizeOf (value.GetType()));
|
|
Marshal.StructureToPtr (value, buf, false);
|
|
g_value_set_pointer (ref this, buf);
|
|
} else if (type == GType.Param.Val) {
|
|
g_value_set_param (ref this, (IntPtr) value);
|
|
} else if (type == ManagedValue.GType.Val) {
|
|
IntPtr wrapper = ManagedValue.WrapObject (value);
|
|
g_value_set_boxed (ref this, wrapper);
|
|
ManagedValue.ReleaseWrapper (wrapper);
|
|
} else if (g_type_is_a (type, GType.Object.Val))
|
|
g_value_set_object (ref this, ((GLib.Object) value).Handle);
|
|
else if (g_type_is_a (type, GType.Boxed.Val)) {
|
|
if (value is IWrapper) {
|
|
g_value_set_boxed (ref this, ((IWrapper)value).Handle);
|
|
return;
|
|
}
|
|
IntPtr buf = Marshaller.StructureToPtrAlloc (value);
|
|
g_value_set_boxed (ref this, buf);
|
|
Marshal.FreeHGlobal (buf);
|
|
} else
|
|
throw new Exception ("Unknown type " + new GType (type).ToString ());
|
|
}
|
|
}
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_init (ref GLib.Value val, IntPtr gtype);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_unset (ref GLib.Value val);
|
|
|
|
[DllImport("glibsharpglue-2")]
|
|
static extern IntPtr gtksharp_value_create_from_property(ref GLib.Value val, IntPtr obj, IntPtr name);
|
|
|
|
[DllImport("glibsharpglue-2")]
|
|
static extern IntPtr gtksharp_value_create_from_type_and_property(ref GLib.Value val, IntPtr gtype, IntPtr name);
|
|
|
|
[DllImport("glibsharpglue-2")]
|
|
static extern IntPtr gtksharp_value_create_from_type_name(ref GLib.Value val, IntPtr type_name);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_set_boolean (ref Value val, bool data);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_set_boxed (ref Value val, IntPtr data);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_set_double (ref Value val, double data);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_set_float (ref Value val, float data);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_set_int (ref Value val, int data);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_set_int64 (ref Value val, long data);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_set_uint64 (ref Value val, ulong data);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_set_object (ref Value val, IntPtr data);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_set_param (ref Value val, IntPtr data);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_set_pointer (ref Value val, IntPtr data);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_set_string (ref Value val, IntPtr data);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_set_uint (ref Value val, uint data);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_set_enum (ref Value val, int data);
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern void g_value_set_flags (ref Value val, uint data);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern bool g_value_get_boolean (ref Value val);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern IntPtr g_value_get_boxed (ref Value val);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern double g_value_get_double (ref Value val);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern float g_value_get_float (ref Value val);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern int g_value_get_int (ref Value val);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern long g_value_get_int64 (ref Value val);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern ulong g_value_get_uint64 (ref Value val);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern IntPtr g_value_get_object (ref Value val);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern IntPtr g_value_get_param (ref Value val);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern IntPtr g_value_get_pointer (ref Value val);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern IntPtr g_value_get_string (ref Value val);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern uint g_value_get_uint (ref Value val);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern int g_value_get_enum (ref Value val);
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern uint g_value_get_flags (ref Value val);
|
|
[DllImport("glibsharpglue-2")]
|
|
static extern bool glibsharp_value_holds_flags (ref Value val);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern bool g_type_is_a (IntPtr type, IntPtr is_a_type);
|
|
|
|
[DllImport("libgobject-2.0-0.dll")]
|
|
static extern IntPtr g_strv_get_type ();
|
|
}
|
|
}
|