2001-09-19 02:04:57 +00:00
// Object.cs - GObject class wrapper implementation
//
2003-07-23 17:19:21 +00:00
// Authors: Mike Kestner <mkestner@speakeasy.net>
2001-09-19 02:04:57 +00:00
//
2004-05-27 16:35:21 +00:00
// Copyright (c) 2001-2003 Mike Kestner
2005-08-15 15:56:16 +00:00
// Copyright (c) 2004-2005 Novell, Inc.
2004-06-25 18:42:19 +00:00
//
// 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.
2004-03-16 19:43:04 +00:00
2001-09-19 11:37:15 +00:00
namespace GLib {
2001-09-19 02:04:57 +00:00
using System ;
2011-03-24 18:11:10 -05:00
using System.Collections.Generic ;
2001-09-19 11:37:15 +00:00
using System.ComponentModel ;
2002-12-25 00:36:00 +00:00
using System.Reflection ;
2001-09-19 02:04:57 +00:00
using System.Runtime.InteropServices ;
2005-08-24 00:36:57 +00:00
using System.Text ;
2001-09-19 02:04:57 +00:00
2002-09-04 05:25:58 +00:00
public class Object : IWrapper , IDisposable {
2001-09-27 18:39:53 +00:00
2007-11-16 18:35:38 +00:00
IntPtr handle ;
2008-01-22 16:54:44 +00:00
ToggleRef tref ;
2002-09-04 05:25:58 +00:00
bool disposed = false ;
2011-11-20 19:41:56 +01:00
static uint idx = 1 ;
2011-03-24 18:11:10 -05:00
static Dictionary < IntPtr , ToggleRef > Objects = new Dictionary < IntPtr , ToggleRef > ( ) ;
2001-09-27 18:39:53 +00:00
2002-09-04 05:25:58 +00:00
~ Object ( )
{
2011-03-25 12:22:04 -05:00
if ( WarnOnFinalize )
Console . Error . WriteLine ( "Unexpected finalization of " + GetType ( ) + " instance. Consider calling Dispose." ) ;
2003-03-15 20:49:37 +00:00
2011-03-25 12:22:04 -05:00
Dispose ( false ) ;
2003-03-15 20:49:37 +00:00
}
2011-03-25 12:22:04 -05:00
public void Dispose ( )
2002-09-04 05:25:58 +00:00
{
if ( disposed )
return ;
2011-03-25 12:22:04 -05:00
Dispose ( true ) ;
2002-09-04 05:25:58 +00:00
disposed = true ;
2011-03-25 12:22:04 -05:00
GC . SuppressFinalize ( this ) ;
}
protected virtual void Dispose ( bool disposing )
{
ToggleRef tref ;
lock ( Objects ) {
if ( Objects . TryGetValue ( Handle , out tref ) ) {
tref . QueueUnref ( ) ;
Objects . Remove ( Handle ) ;
}
2003-03-15 20:49:37 +00:00
}
2011-03-25 12:22:04 -05:00
2007-11-16 18:35:38 +00:00
handle = IntPtr . Zero ;
2011-03-25 12:22:04 -05:00
if ( tref = = null )
return ;
if ( disposing )
2011-07-29 13:33:02 -05:00
tref . Dispose ( ) ;
2011-03-25 12:22:04 -05:00
else
tref . QueueUnref ( ) ;
2002-09-04 05:25:58 +00:00
}
2011-03-25 12:22:04 -05:00
public static bool WarnOnFinalize { get ; set ; }
2009-09-03 19:50:53 +00:00
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
2008-04-04 16:10:08 +00:00
static extern IntPtr g_object_ref ( IntPtr raw ) ;
2002-09-12 05:21:16 +00:00
2011-03-25 12:22:04 -05:00
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
static extern void g_object_unref ( IntPtr raw ) ;
2003-07-23 17:19:21 +00:00
public static Object GetObject ( IntPtr o , bool owned_ref )
2001-09-19 04:47:48 +00:00
{
2005-03-03 20:50:46 +00:00
if ( o = = IntPtr . Zero )
return null ;
2005-08-15 15:56:16 +00:00
Object obj = null ;
2007-11-16 18:35:38 +00:00
2011-03-24 18:11:10 -05:00
ToggleRef toggle_ref ;
if ( Objects . TryGetValue ( o , out toggle_ref ) ) {
2011-03-23 17:35:39 -05:00
if ( toggle_ref ! = null )
2007-02-16 16:18:59 +00:00
obj = toggle_ref . Target ;
}
2005-05-31 19:22:58 +00:00
2007-11-16 18:35:38 +00:00
if ( obj ! = null & & obj . Handle = = o ) {
2003-07-23 17:19:21 +00:00
if ( owned_ref )
2007-11-16 18:35:38 +00:00
g_object_unref ( obj . Handle ) ;
2003-07-23 17:19:21 +00:00
return obj ;
2003-04-09 17:50:51 +00:00
}
2007-11-16 18:35:38 +00:00
if ( ! owned_ref )
2007-11-29 02:02:24 +00:00
g_object_ref ( o ) ;
2007-11-16 18:35:38 +00:00
2004-05-28 16:59:21 +00:00
obj = GLib . ObjectManager . CreateObject ( o ) ;
2007-11-16 18:35:38 +00:00
if ( obj = = null ) {
g_object_unref ( o ) ;
2003-07-23 17:19:21 +00:00
return null ;
2007-11-16 18:35:38 +00:00
}
2003-07-23 17:19:21 +00:00
return obj ;
2001-09-19 02:04:57 +00:00
}
2001-09-20 04:03:27 +00:00
2003-12-10 22:56:49 +00:00
public static Object GetObject ( IntPtr o )
{
return GetObject ( o , false ) ;
}
2003-12-30 22:09:42 +00:00
private static void ConnectDefaultHandlers ( GType gtype , System . Type t )
{
2004-06-05 01:01:07 +00:00
foreach ( MethodInfo minfo in t . GetMethods ( BindingFlags . Instance | BindingFlags . NonPublic | BindingFlags . Public | BindingFlags . DeclaredOnly ) ) {
2003-12-30 22:09:42 +00:00
MethodInfo baseinfo = minfo . GetBaseDefinition ( ) ;
if ( baseinfo = = minfo )
continue ;
2005-01-14 19:54:07 +00:00
foreach ( object attr in baseinfo . GetCustomAttributes ( typeof ( DefaultSignalHandlerAttribute ) , false ) ) {
2003-12-30 22:09:42 +00:00
DefaultSignalHandlerAttribute sigattr = attr as DefaultSignalHandlerAttribute ;
2009-04-13 17:44:48 +00:00
MethodInfo connector = sigattr . Type . GetMethod ( sigattr . ConnectionMethod , BindingFlags . Static | BindingFlags . NonPublic , null , new Type [ ] { typeof ( GType ) } , new ParameterModifier [ 0 ] ) ;
2003-12-30 22:09:42 +00:00
object [ ] parms = new object [ 1 ] ;
parms [ 0 ] = gtype ;
connector . Invoke ( null , parms ) ;
break ;
}
}
}
2011-02-09 10:05:40 -06:00
private static void InvokeTypeInitializers ( GType gtype , System . Type t )
2004-12-23 22:59:59 +00:00
{
object [ ] parms = { gtype , t } ;
2007-09-07 14:40:46 +00:00
BindingFlags flags = BindingFlags . Static | BindingFlags . NonPublic ;
foreach ( TypeInitializerAttribute tia in t . GetCustomAttributes ( typeof ( TypeInitializerAttribute ) , true ) ) {
MethodInfo m = tia . Type . GetMethod ( tia . MethodName , flags ) ;
if ( m ! = null )
m . Invoke ( null , parms ) ;
}
}
2008-06-06 16:55:00 +00:00
2011-12-01 16:49:50 +00:00
// Key: The Type for the set of properties
// Value->SubKey: The pointer to the ParamSpec of the property
// Value->SubValue: The corresponding PropertyInfo object
static Dictionary < Type , Dictionary < IntPtr , PropertyInfo > > properties ;
static Dictionary < Type , Dictionary < IntPtr , PropertyInfo > > Properties {
2008-06-06 16:55:00 +00:00
get {
if ( properties = = null )
2011-12-01 16:49:50 +00:00
properties = new Dictionary < Type , Dictionary < IntPtr , PropertyInfo > > ( ) ;
2008-06-06 16:55:00 +00:00
return properties ;
}
}
2011-11-20 19:41:56 +01:00
static Dictionary < IntPtr , Dictionary < Type , PropertyInfo > > interface_properties ;
static Dictionary < IntPtr , Dictionary < Type , PropertyInfo > > IProperties {
get {
if ( interface_properties = = null )
interface_properties = new Dictionary < IntPtr , Dictionary < Type , PropertyInfo > > ( ) ;
return interface_properties ;
}
}
2008-12-19 18:57:42 +00:00
struct GTypeClass {
2009-05-01 18:40:44 +00:00
public IntPtr gtype ;
2008-12-19 18:57:42 +00:00
}
struct GObjectClass {
2011-03-23 14:05:01 -05:00
public GTypeClass type_class ;
public IntPtr construct_props ;
2009-09-12 01:01:12 +00:00
public ConstructorDelegate constructor_cb ;
2008-12-19 18:57:42 +00:00
public SetPropertyDelegate set_prop_cb ;
public GetPropertyDelegate get_prop_cb ;
2011-03-23 14:05:01 -05:00
public IntPtr dispose ;
public IntPtr finalize ;
public IntPtr dispatch_properties_changed ;
public IntPtr notify ;
public IntPtr constructed ;
public IntPtr dummy1 ;
public IntPtr dummy2 ;
public IntPtr dummy3 ;
public IntPtr dummy4 ;
public IntPtr dummy5 ;
public IntPtr dummy6 ;
public IntPtr dummy7 ;
2008-12-19 18:57:42 +00:00
}
2009-09-12 01:01:12 +00:00
[UnmanagedFunctionPointer (CallingConvention.Cdecl)]
delegate IntPtr ConstructorDelegate ( IntPtr gtype , uint n_construct_properties , IntPtr construct_properties ) ;
static ConstructorDelegate constructor_handler ;
static ConstructorDelegate ConstructorHandler {
get {
if ( constructor_handler = = null )
constructor_handler = new ConstructorDelegate ( ConstructorCallback ) ;
return constructor_handler ;
}
}
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
static extern IntPtr g_param_spec_get_name ( IntPtr pspec ) ;
static IntPtr ConstructorCallback ( IntPtr gtypeval , uint n_construct_properties , IntPtr construct_properties )
2008-12-19 18:57:42 +00:00
{
2009-09-12 01:01:12 +00:00
GType gtype = new GLib . GType ( gtypeval ) ;
GObjectClass threshold_class = ( GObjectClass ) Marshal . PtrToStructure ( gtype . GetThresholdType ( ) . GetClassPtr ( ) , typeof ( GObjectClass ) ) ;
IntPtr raw = threshold_class . constructor_cb ( gtypeval , n_construct_properties , construct_properties ) ;
bool construct_needed = true ;
for ( int i = 0 ; i < n_construct_properties ; i + + ) {
IntPtr p = new IntPtr ( construct_properties . ToInt64 ( ) + i * 2 * IntPtr . Size ) ;
string prop_name = Marshaller . Utf8PtrToString ( g_param_spec_get_name ( Marshal . ReadIntPtr ( p ) ) ) ;
if ( prop_name ! = "gtk-sharp-managed-instance" )
continue ;
Value val = ( Value ) Marshal . PtrToStructure ( Marshal . ReadIntPtr ( p , IntPtr . Size ) , typeof ( Value ) ) ;
if ( ( IntPtr ) val . Val ! = IntPtr . Zero ) {
GCHandle gch = ( GCHandle ) ( IntPtr ) val . Val ;
Object o = ( GLib . Object ) gch . Target ;
o . Raw = raw ;
construct_needed = false ;
break ;
}
}
if ( construct_needed )
GetObject ( raw , false ) ;
return raw ;
2008-12-19 18:57:42 +00:00
}
2011-11-20 19:41:56 +01:00
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
static extern void g_object_class_override_property ( IntPtr klass , uint prop_id , IntPtr name ) ;
public static void OverrideProperty ( IntPtr declaring_class , string name )
{
IntPtr native_name = GLib . Marshaller . StringToPtrGStrdup ( name ) ;
g_object_class_override_property ( declaring_class , idx , native_name ) ;
idx + + ;
}
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
static extern IntPtr g_object_class_find_property ( IntPtr klass , IntPtr name ) ;
static IntPtr FindClassProperty ( GType type , string name )
{
IntPtr g_iface = type . GetDefaultInterfacePtr ( ) ;
IntPtr native_name = GLib . Marshaller . StringToPtrGStrdup ( name ) ;
return g_object_class_find_property ( g_iface , native_name ) ;
}
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
static extern IntPtr g_object_interface_find_property ( IntPtr klass , IntPtr name ) ;
static IntPtr FindInterfaceProperty ( GType type , string name )
{
IntPtr g_iface = type . GetDefaultInterfacePtr ( ) ;
IntPtr native_name = GLib . Marshaller . StringToPtrGStrdup ( name ) ;
return g_object_interface_find_property ( g_iface , native_name ) ;
}
2009-09-03 19:50:53 +00:00
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
2008-12-19 18:57:42 +00:00
static extern void g_object_class_install_property ( IntPtr klass , uint prop_id , IntPtr param_spec ) ;
static IntPtr RegisterProperty ( GType type , string name , string nick , string blurb , uint property_id , GType property_type , bool can_read , bool can_write )
{
2009-08-08 23:42:15 +00:00
IntPtr declaring_class = type . GetClassPtr ( ) ;
2008-12-19 18:57:42 +00:00
ParamSpec pspec = new ParamSpec ( name , nick , blurb , property_type , can_read , can_write ) ;
g_object_class_install_property ( declaring_class , property_id , pspec . Handle ) ;
return pspec . Handle ;
}
2008-06-06 16:55:00 +00:00
2011-11-20 19:41:56 +01:00
static void AddProperties ( GType gtype , System . Type t , bool register_instance_prop , ref bool handlers_overridden )
2008-06-06 16:55:00 +00:00
{
2009-09-12 01:01:12 +00:00
if ( register_instance_prop ) {
IntPtr declaring_class = gtype . GetClassPtr ( ) ;
ParamSpec pspec = new ParamSpec ( "gtk-sharp-managed-instance" , "" , "" , GType . Pointer , ParamFlags . Writable | ParamFlags . ConstructOnly ) ;
g_object_class_install_property ( declaring_class , idx , pspec . Handle ) ;
2011-12-03 16:24:23 +01:00
idx + + ;
2009-09-12 01:01:12 +00:00
}
2008-06-06 16:55:00 +00:00
foreach ( PropertyInfo pinfo in t . GetProperties ( BindingFlags . Instance | BindingFlags . Public | BindingFlags . DeclaredOnly ) ) {
foreach ( object attr in pinfo . GetCustomAttributes ( typeof ( PropertyAttribute ) , false ) ) {
if ( pinfo . GetIndexParameters ( ) . Length > 0 )
throw ( new InvalidOperationException ( String . Format ( "GLib.RegisterPropertyAttribute cannot be applied to property {0} of type {1} because the property expects one or more indexed parameters" , pinfo . Name , t . FullName ) ) ) ;
if ( ! handlers_overridden ) {
2009-09-12 01:01:12 +00:00
IntPtr class_ptr = gtype . GetClassPtr ( ) ;
GObjectClass gobject_class = ( GObjectClass ) Marshal . PtrToStructure ( class_ptr , typeof ( GObjectClass ) ) ;
gobject_class . get_prop_cb = GetPropertyHandler ;
gobject_class . set_prop_cb = SetPropertyHandler ;
Marshal . StructureToPtr ( gobject_class , class_ptr , false ) ;
2008-06-06 16:55:00 +00:00
handlers_overridden = true ;
}
2009-09-12 01:01:12 +00:00
PropertyAttribute property_attr = attr as PropertyAttribute ;
2008-12-19 18:57:42 +00:00
try {
IntPtr param_spec = RegisterProperty ( gtype , property_attr . Name , property_attr . Nickname , property_attr . Blurb , idx , ( GType ) pinfo . PropertyType , pinfo . CanRead , pinfo . CanWrite ) ;
2011-12-01 16:49:50 +00:00
Type type = ( Type ) gtype ;
Dictionary < IntPtr , PropertyInfo > gtype_properties ;
if ( ! Properties . TryGetValue ( type , out gtype_properties ) ) {
gtype_properties = new Dictionary < IntPtr , PropertyInfo > ( ) ;
Properties [ type ] = gtype_properties ;
}
gtype_properties . Add ( param_spec , pinfo ) ;
2008-12-19 18:57:42 +00:00
idx + + ;
} catch ( ArgumentException ) {
2008-06-06 16:55:00 +00:00
throw new InvalidOperationException ( String . Format ( "GLib.PropertyAttribute cannot be applied to property {0} of type {1} because the return type of the property is not supported" , pinfo . Name , t . FullName ) ) ;
2008-12-19 18:57:42 +00:00
}
2008-06-06 16:55:00 +00:00
}
}
}
2009-09-03 19:50:53 +00:00
[UnmanagedFunctionPointer (CallingConvention.Cdecl)]
2008-06-06 16:55:00 +00:00
delegate void GetPropertyDelegate ( IntPtr GObject , uint property_id , ref GLib . Value value , IntPtr pspec ) ;
static void GetPropertyCallback ( IntPtr handle , uint property_id , ref GLib . Value value , IntPtr param_spec )
{
2011-12-01 16:49:50 +00:00
GLib . Object obj = GLib . Object . GetObject ( handle , false ) ;
var type = ( Type ) obj . LookupGType ( ) ;
Dictionary < IntPtr , PropertyInfo > props ;
if ( ! Properties . TryGetValue ( type , out props ) )
2009-09-12 01:01:12 +00:00
return ;
2011-12-03 16:24:23 +01:00
PropertyInfo prop ;
2011-12-01 16:49:50 +00:00
if ( ! props . TryGetValue ( param_spec , out prop ) )
return ;
value . Val = prop . GetValue ( obj , new object [ 0 ] ) ;
2008-06-06 16:55:00 +00:00
}
static GetPropertyDelegate get_property_handler ;
static GetPropertyDelegate GetPropertyHandler {
get {
if ( get_property_handler = = null )
get_property_handler = new GetPropertyDelegate ( GetPropertyCallback ) ;
return get_property_handler ;
}
}
2009-09-03 19:50:53 +00:00
[UnmanagedFunctionPointer (CallingConvention.Cdecl)]
2008-06-06 16:55:00 +00:00
delegate void SetPropertyDelegate ( IntPtr GObject , uint property_id , ref GLib . Value value , IntPtr pspec ) ;
static void SetPropertyCallback ( IntPtr handle , uint property_id , ref GLib . Value value , IntPtr param_spec )
{
2011-11-20 19:41:56 +01:00
// FIXME: Here is a big quick hack to avoid race condition when trying to set up adjustment with contructor
// Because Raw is set too late
if ( param_spec ! = IntPtr . Zero ) {
ParamSpec foo = new ParamSpec ( param_spec ) ;
if ( foo . Name = = "gtk-sharp-managed-instance" ) {
GCHandle gch = ( GCHandle ) ( IntPtr ) value . Val ;
Object o = ( GLib . Object ) gch . Target ;
o . Raw = handle ;
}
}
2011-12-01 16:49:50 +00:00
GLib . Object obj = GLib . Object . GetObject ( handle , false ) ;
var type = ( Type ) obj . LookupGType ( ) ;
Dictionary < IntPtr , PropertyInfo > props ;
if ( ! Properties . TryGetValue ( type , out props ) )
2009-09-12 01:01:12 +00:00
return ;
2011-12-03 16:24:23 +01:00
PropertyInfo prop ;
2011-12-01 16:49:50 +00:00
if ( ! props . TryGetValue ( param_spec , out prop ) )
return ;
prop . SetValue ( obj , value . Val , new object [ 0 ] ) ;
2008-06-06 16:55:00 +00:00
}
static SetPropertyDelegate set_property_handler ;
static SetPropertyDelegate SetPropertyHandler {
get {
if ( set_property_handler = = null )
set_property_handler = new SetPropertyDelegate ( SetPropertyCallback ) ;
return set_property_handler ;
}
}
2004-12-23 22:59:59 +00:00
2009-09-03 19:50:53 +00:00
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
2007-09-11 20:34:24 +00:00
static extern void g_type_add_interface_static ( IntPtr gtype , IntPtr iface_type , ref GInterfaceInfo info ) ;
2011-11-20 19:41:56 +01:00
static void AddInterfaces ( GType gtype , Type t , ref bool handlers_overridden )
2007-09-11 20:34:24 +00:00
{
foreach ( Type iface in t . GetInterfaces ( ) ) {
2011-11-20 19:41:56 +01:00
if ( ! iface . IsDefined ( typeof ( GInterfaceAttribute ) , true ) )
2007-09-11 20:34:24 +00:00
continue ;
GInterfaceAttribute attr = iface . GetCustomAttributes ( typeof ( GInterfaceAttribute ) , false ) [ 0 ] as GInterfaceAttribute ;
GInterfaceAdapter adapter = Activator . CreateInstance ( attr . AdapterType , null ) as GInterfaceAdapter ;
2011-11-20 19:41:56 +01:00
if ( ! handlers_overridden ) {
IntPtr class_ptr = gtype . GetClassPtr ( ) ;
GObjectClass gobject_class = ( GObjectClass ) Marshal . PtrToStructure ( class_ptr , typeof ( GObjectClass ) ) ;
gobject_class . get_prop_cb = GetPropertyHandler ;
gobject_class . set_prop_cb = SetPropertyHandler ;
Marshal . StructureToPtr ( gobject_class , class_ptr , false ) ;
handlers_overridden = true ;
}
if ( ! iface . IsAssignableFrom ( t . BaseType ) ) {
GInterfaceInfo info = adapter . Info ;
info . Data = gtype . GetClassPtr ( ) ;
//FIXME: overiding prop is done inside the init of interface adapter
// not sure that it is the good solution but
// it is the only one I found without exception or loop
g_type_add_interface_static ( gtype . Val , adapter . GType . Val , ref info ) ;
}
foreach ( PropertyInfo p in iface . GetProperties ( ) ) {
PropertyAttribute [ ] attrs = p . GetCustomAttributes ( typeof ( PropertyAttribute ) , true ) as PropertyAttribute [ ] ;
if ( attrs . Length = = 0 )
continue ;
PropertyAttribute property_attr = attrs [ 0 ] ;
2011-12-03 16:18:24 +01:00
PropertyInfo declared_prop = t . GetProperty ( p . Name , BindingFlags . Public | BindingFlags . Instance ) ;
2011-11-20 19:41:56 +01:00
if ( declared_prop = = null )
continue ;
IntPtr param_spec = FindInterfaceProperty ( adapter . GType , property_attr . Name ) ;
2011-12-01 16:49:50 +00:00
Dictionary < IntPtr , PropertyInfo > props ;
2011-12-03 16:18:24 +01:00
if ( ! Properties . TryGetValue ( t , out props ) ) {
2011-12-01 16:49:50 +00:00
props = new Dictionary < IntPtr , PropertyInfo > ( ) ;
2011-12-03 16:18:24 +01:00
Properties [ t ] = props ;
2011-12-01 16:49:50 +00:00
}
props [ param_spec ] = declared_prop ;
2011-11-20 19:41:56 +01:00
}
2007-09-11 20:34:24 +00:00
}
}
2008-12-19 18:57:42 +00:00
protected internal static GType RegisterGType ( System . Type t )
2002-12-25 00:36:00 +00:00
{
2008-12-19 18:57:42 +00:00
GType gtype = GType . RegisterGObjectType ( t ) ;
2009-09-12 01:01:12 +00:00
bool is_first_subclass = gtype . GetBaseType ( ) = = gtype . GetThresholdType ( ) ;
2011-11-20 19:41:56 +01:00
2009-09-12 01:01:12 +00:00
if ( is_first_subclass ) {
IntPtr class_ptr = gtype . GetClassPtr ( ) ;
GObjectClass gobject_class = ( GObjectClass ) Marshal . PtrToStructure ( class_ptr , typeof ( GObjectClass ) ) ;
gobject_class . constructor_cb = ConstructorHandler ;
gobject_class . get_prop_cb = GetPropertyHandler ;
gobject_class . set_prop_cb = SetPropertyHandler ;
Marshal . StructureToPtr ( gobject_class , class_ptr , false ) ;
}
2011-11-20 19:41:56 +01:00
idx = 1 ;
bool handlers_overridden = is_first_subclass ;
AddProperties ( gtype , t , is_first_subclass , ref handlers_overridden ) ;
2003-12-30 22:09:42 +00:00
ConnectDefaultHandlers ( gtype , t ) ;
2011-02-09 10:05:40 -06:00
InvokeTypeInitializers ( gtype , t ) ;
2011-11-20 19:41:56 +01:00
AddInterfaces ( gtype , t , ref handlers_overridden ) ;
2003-12-30 22:09:42 +00:00
return gtype ;
2002-12-25 00:36:00 +00:00
}
2004-05-17 17:52:00 +00:00
protected GType LookupGType ( )
2004-05-07 13:42:59 +00:00
{
2009-05-03 22:49:12 +00:00
if ( Handle ! = IntPtr . Zero ) {
GTypeInstance obj = ( GTypeInstance ) Marshal . PtrToStructure ( Handle , typeof ( GTypeInstance ) ) ;
GTypeClass klass = ( GTypeClass ) Marshal . PtrToStructure ( obj . g_class , typeof ( GTypeClass ) ) ;
return new GLib . GType ( klass . gtype ) ;
} else {
return LookupGType ( GetType ( ) ) ;
}
2004-05-07 13:42:59 +00:00
}
2005-05-04 16:54:24 +00:00
protected internal static GType LookupGType ( System . Type t )
2004-04-07 19:15:01 +00:00
{
2008-12-19 18:57:42 +00:00
return GType . LookupGObjectType ( t ) ;
2004-04-07 19:15:01 +00:00
}
2002-02-03 03:44:10 +00:00
2004-05-17 17:52:00 +00:00
protected Object ( IntPtr raw )
2002-02-03 03:44:10 +00:00
{
2002-02-19 19:46:44 +00:00
Raw = raw ;
2002-02-03 03:44:10 +00:00
}
2001-09-19 02:04:57 +00:00
2007-10-02 03:02:43 +00:00
protected Object ( )
{
CreateNativeObject ( new string [ 0 ] , new GLib . Value [ 0 ] ) ;
}
2009-09-03 19:50:53 +00:00
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
2003-12-15 16:59:25 +00:00
static extern IntPtr g_object_new ( IntPtr gtype , IntPtr dummy ) ;
2002-12-25 00:36:00 +00:00
2008-12-19 18:57:42 +00:00
struct GParameter {
public IntPtr name ;
public GLib . Value val ;
}
2009-09-03 19:50:53 +00:00
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
2008-12-19 18:57:42 +00:00
static extern IntPtr g_object_newv ( IntPtr gtype , int n_params , GParameter [ ] parms ) ;
2004-05-07 13:42:59 +00:00
2004-11-09 14:22:39 +00:00
protected virtual void CreateNativeObject ( string [ ] names , GLib . Value [ ] vals )
2004-05-07 13:42:59 +00:00
{
2009-09-12 01:01:12 +00:00
GType gtype = LookupGType ( ) ;
bool is_managed_subclass = gtype . ToString ( ) . StartsWith ( "__gtksharp" ) ;
GParameter [ ] parms = new GParameter [ is_managed_subclass ? names . Length + 1 : names . Length ] ;
2008-12-19 18:57:42 +00:00
for ( int i = 0 ; i < names . Length ; i + + ) {
parms [ i ] . name = GLib . Marshaller . StringToPtrGStrdup ( names [ i ] ) ;
parms [ i ] . val = vals [ i ] ;
}
2009-09-12 01:01:12 +00:00
if ( is_managed_subclass ) {
GCHandle gch = GCHandle . Alloc ( this ) ;
parms [ names . Length ] . name = GLib . Marshaller . StringToPtrGStrdup ( "gtk-sharp-managed-instance" ) ;
parms [ names . Length ] . val = new GLib . Value ( ( IntPtr ) gch ) ;
Raw = g_object_newv ( gtype . Val , parms . Length , parms ) ;
gch . Free ( ) ;
} else {
Raw = g_object_newv ( gtype . Val , parms . Length , parms ) ;
}
2008-12-19 18:57:42 +00:00
foreach ( GParameter p in parms )
GLib . Marshaller . Free ( p . name ) ;
2004-05-07 13:42:59 +00:00
}
2002-11-10 10:09:05 +00:00
protected virtual IntPtr Raw {
2001-09-19 11:37:15 +00:00
get {
2007-11-16 18:35:38 +00:00
return handle ;
2001-09-19 02:04:57 +00:00
}
2001-09-19 11:37:15 +00:00
set {
2007-11-16 18:35:38 +00:00
if ( handle = = value )
2004-06-09 17:53:05 +00:00
return ;
2007-11-16 18:35:38 +00:00
if ( handle ! = IntPtr . Zero ) {
2008-01-17 21:10:25 +00:00
Objects . Remove ( handle ) ;
2008-01-22 16:54:44 +00:00
if ( tref ! = null ) {
2011-07-29 13:33:02 -05:00
tref . Dispose ( ) ;
2008-01-22 16:54:44 +00:00
tref = null ;
}
2007-11-16 18:35:38 +00:00
}
handle = value ;
2008-01-22 16:54:44 +00:00
if ( value ! = IntPtr . Zero ) {
tref = new ToggleRef ( this ) ;
Objects [ value ] = tref ;
}
2001-09-19 02:04:57 +00:00
}
2002-11-10 10:09:05 +00:00
}
2001-09-19 02:04:57 +00:00
2004-05-27 16:35:21 +00:00
public static GLib . GType GType {
2011-03-24 18:11:10 -05:00
get { return GType . Object ; }
2002-08-09 03:56:27 +00:00
}
2004-05-17 17:52:00 +00:00
protected string TypeName {
2011-03-24 18:11:10 -05:00
get { return NativeType . ToString ( ) ; }
2003-12-03 20:23:25 +00:00
}
2004-05-17 17:52:00 +00:00
internal GLib . GType NativeType {
2011-03-24 18:11:10 -05:00
get { return LookupGType ( ) ; }
2002-11-18 18:55:39 +00:00
}
2008-01-22 16:54:44 +00:00
internal ToggleRef ToggleRef {
2011-03-24 18:11:10 -05:00
get { return tref ; }
2007-11-16 18:35:38 +00:00
}
2008-01-22 16:54:44 +00:00
public IntPtr Handle {
2011-03-24 18:11:10 -05:00
get { return handle ; }
2001-10-07 00:41:52 +00:00
}
2008-04-04 16:10:08 +00:00
public IntPtr OwnedHandle {
2011-03-24 18:11:10 -05:00
get { return g_object_ref ( handle ) ; }
2005-01-28 16:44:30 +00:00
}
public void AddNotification ( string property , NotifyHandler handler )
{
2011-03-24 18:11:10 -05:00
AddSignalHandler ( "notify::" + property , handler , typeof ( NotifyArgs ) ) ;
2005-01-28 16:44:30 +00:00
}
public void AddNotification ( NotifyHandler handler )
{
2011-03-24 18:11:10 -05:00
AddSignalHandler ( "notify" , handler , typeof ( NotifyArgs ) ) ;
2005-01-28 16:44:30 +00:00
}
public void RemoveNotification ( string property , NotifyHandler handler )
{
2011-03-24 18:11:10 -05:00
RemoveSignalHandler ( "notify::" + property , handler ) ;
2005-01-28 16:44:30 +00:00
}
public void RemoveNotification ( NotifyHandler handler )
{
2011-03-24 18:11:10 -05:00
RemoveSignalHandler ( "notify" , handler ) ;
2005-01-28 16:44:30 +00:00
}
2001-10-31 01:31:05 +00:00
public override int GetHashCode ( )
{
return Handle . GetHashCode ( ) ;
}
2011-03-24 18:11:10 -05:00
System . Collections . Hashtable data ;
public System . Collections . Hashtable Data {
2003-07-04 07:13:19 +00:00
get {
if ( data = = null )
2011-03-24 18:11:10 -05:00
data = new System . Collections . Hashtable ( ) ;
2003-07-04 07:13:19 +00:00
return data ;
}
2001-09-19 02:04:57 +00:00
}
2001-09-20 04:03:27 +00:00
2009-09-03 19:50:53 +00:00
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
2005-03-08 21:28:08 +00:00
static extern void g_object_get_property ( IntPtr obj , IntPtr name , ref GLib . Value val ) ;
2004-03-16 19:43:04 +00:00
2004-04-12 15:54:57 +00:00
protected GLib . Value GetProperty ( string name )
2001-09-28 18:23:14 +00:00
{
2004-04-12 15:54:57 +00:00
Value val = new Value ( this , name ) ;
2005-03-08 21:28:08 +00:00
IntPtr native_name = GLib . Marshaller . StringToPtrGStrdup ( name ) ;
g_object_get_property ( Raw , native_name , ref val ) ;
GLib . Marshaller . Free ( native_name ) ;
2004-04-12 15:54:57 +00:00
return val ;
2001-09-28 18:23:14 +00:00
}
2009-09-03 19:50:53 +00:00
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
2005-03-08 21:28:08 +00:00
static extern void g_object_set_property ( IntPtr obj , IntPtr name , ref GLib . Value val ) ;
2004-03-16 19:43:04 +00:00
2004-04-12 15:54:57 +00:00
protected void SetProperty ( string name , GLib . Value val )
2002-01-12 02:08:16 +00:00
{
2005-03-08 21:28:08 +00:00
IntPtr native_name = GLib . Marshaller . StringToPtrGStrdup ( name ) ;
g_object_set_property ( Raw , native_name , ref val ) ;
GLib . Marshaller . Free ( native_name ) ;
2002-01-12 02:08:16 +00:00
}
2009-09-03 19:50:53 +00:00
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
2008-11-14 16:21:16 +00:00
static extern void g_object_notify ( IntPtr obj , IntPtr property_name ) ;
protected void Notify ( string property_name )
{
IntPtr native_name = GLib . Marshaller . StringToPtrGStrdup ( property_name ) ;
g_object_notify ( Handle , native_name ) ;
GLib . Marshaller . Free ( native_name ) ;
}
2011-03-24 18:11:10 -05:00
Dictionary < string , Signal > signals ;
Dictionary < string , Signal > Signals {
get {
if ( signals = = null )
signals = new Dictionary < string , Signal > ( ) ;
return signals ;
}
}
public void AddSignalHandler ( string name , Delegate handler )
{
AddSignalHandler ( name , handler , typeof ( EventArgs ) ) ;
}
public void AddSignalHandler ( string name , Delegate handler , Delegate marshaler )
{
Signal sig ;
if ( ! Signals . TryGetValue ( name , out sig ) ) {
sig = new Signal ( this , name , marshaler ) ;
Signals [ name ] = sig ;
}
sig . AddDelegate ( handler ) ;
}
public void AddSignalHandler ( string name , Delegate handler , Type args_type )
{
if ( args_type = = null )
args_type = handler . Method . GetParameters ( ) [ 1 ] . ParameterType ;
Signal sig ;
if ( ! Signals . TryGetValue ( name , out sig ) ) {
sig = new Signal ( this , name , args_type ) ;
Signals [ name ] = sig ;
}
sig . AddDelegate ( handler ) ;
}
public void RemoveSignalHandler ( string name , Delegate handler )
{
Signal sig ;
if ( Signals . TryGetValue ( name , out sig ) )
sig . RemoveDelegate ( handler ) ;
}
2003-12-15 16:59:25 +00:00
protected static void OverrideVirtualMethod ( GType gtype , string name , Delegate cb )
{
2008-12-19 18:57:42 +00:00
Signal . OverrideDefaultHandler ( gtype , name , cb ) ;
2003-12-15 16:59:25 +00:00
}
2003-12-10 22:56:49 +00:00
2009-09-03 19:50:53 +00:00
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
2004-04-12 15:54:57 +00:00
protected static extern void g_signal_chain_from_overridden ( IntPtr args , ref GLib . Value retval ) ;
2003-12-09 05:01:22 +00:00
2009-09-03 19:50:53 +00:00
[DllImport ("libgobject-2.0-0.dll", CallingConvention = CallingConvention.Cdecl)]
2008-12-19 18:57:42 +00:00
static extern bool g_type_check_instance_is_a ( IntPtr obj , IntPtr gtype ) ;
2002-09-01 04:46:38 +00:00
internal static bool IsObject ( IntPtr obj )
{
2008-12-19 18:57:42 +00:00
return g_type_check_instance_is_a ( obj , GType . Object . Val ) ;
}
struct GTypeInstance {
public IntPtr g_class ;
2002-09-01 04:46:38 +00:00
}
2002-11-10 10:09:05 +00:00
2008-12-19 18:57:42 +00:00
struct GObject {
public GTypeInstance type_instance ;
public uint ref_count ;
public IntPtr qdata ;
}
2002-11-10 10:09:05 +00:00
2004-05-17 17:52:00 +00:00
protected int RefCount {
2002-11-10 10:09:05 +00:00
get {
2008-12-19 18:57:42 +00:00
GObject native = ( GObject ) Marshal . PtrToStructure ( Handle , typeof ( GObject ) ) ;
return ( int ) native . ref_count ;
2002-11-10 10:09:05 +00:00
}
}
2008-04-17 16:32:44 +00:00
internal void Harden ( )
{
tref . Harden ( ) ;
}
2008-04-30 20:15:45 +00:00
static Object ( )
{
2008-05-02 16:58:46 +00:00
if ( Environment . GetEnvironmentVariable ( "GTK_SHARP_DEBUG" ) ! = null )
2008-08-25 20:43:09 +00:00
GLib . Log . SetLogHandler ( "GLib-GObject" , GLib . LogLevelFlags . All , new GLib . LogFunc ( GLib . Log . PrintTraceLogFunction ) ) ;
2008-04-30 20:15:45 +00:00
}
2001-09-19 02:04:57 +00:00
}
}