mirror of
https://github.com/Ryujinx/GtkSharp.git
synced 2025-01-12 07:15:41 +00:00
6857128f07
* generator/ClassBase.cs: New base class for classes and interfaces. * generator/InterfaceGen.cs: Inherit from ClassBase, generate declarations. * generator/ObjectGen.cs: Move half of this into ClassBase. * generator/Method.cs: Turn all applicable Get/Set functions into .NET accessors. Remove redundant == overload and move into Equals, as it was confusing "!= null". * generator/Parameters.cs: Alter signature creation to accept "is_set" option, add support for variable arguments. Add properties "Count", "IsVarArgs", "VAType". * generator/Ctor.cs: Fixup for changes in Parameters (indenting, signature creation). * generator/Signal.cs: Support generating declarations. * generator/SymbolTable: Change GetObjectGen to GetClassGen. * glib/IWrapper.cs: Move "Handle" declaration to here, so both classes and interfaces can benefit from it. * glib/Object.cs: Inherit from IWrapper.cs * parser/Metadata.pm: Support attribute changes on constructors, methods, signals, and paramater lists. * parser/gapi2xml.pl: Parse init funcs for interfaces. Ignore "_" functions here. * parser/gapi_pp.pl: Remove boxed_type_register check, as it will be caught in the init funcs. * parser/Atk.metadata: Added. * parser/Gtk.metadata: Add all needed signal/method collision renames. Rename GtkEditable.Editable accessors to IsEditable, as .NET does not like accessors with the same name as their declaring type. Tag TreeStore constructor as varargs. * samples/ButtonApp.cs: s/EmitAdd/Add. * samples/Menu.cs: s/EmitAdd/Add, s/Activate/Activated. svn path=/trunk/gtk-sharp/; revision=5394
318 lines
7.4 KiB
C#
318 lines
7.4 KiB
C#
// GtkSharp.Generation.Method.cs - The Method Generatable.
|
|
//
|
|
// Author: Mike Kestner <mkestner@speakeasy.net>
|
|
//
|
|
// (c) 2001-2002 Mike Kestner
|
|
|
|
namespace GtkSharp.Generation {
|
|
|
|
using System;
|
|
using System.Collections;
|
|
using System.IO;
|
|
using System.Xml;
|
|
|
|
public class Method {
|
|
|
|
private string ns;
|
|
private XmlElement elem;
|
|
private Parameters parms;
|
|
private ClassBase container_type;
|
|
|
|
private bool initialized = false;
|
|
private string sig, isig, call;
|
|
private string rettype, m_ret, s_ret;
|
|
private string name, cname, safety;
|
|
private bool is_get, is_set;
|
|
|
|
public Method (string ns, XmlElement elem, ClassBase container_type)
|
|
{
|
|
this.ns = ns;
|
|
this.elem = elem;
|
|
if (elem["parameters"] != null)
|
|
parms = new Parameters (elem["parameters"]);
|
|
this.container_type = container_type;
|
|
this.name = elem.GetAttribute("name");
|
|
}
|
|
|
|
public string Name {
|
|
get {
|
|
return name;
|
|
}
|
|
set {
|
|
name = value;
|
|
}
|
|
}
|
|
|
|
public Parameters Params {
|
|
get {
|
|
return parms;
|
|
}
|
|
}
|
|
|
|
public override bool Equals (object o)
|
|
{
|
|
if (!(o is Method))
|
|
return false;
|
|
/*
|
|
return (this == (Method) o);
|
|
}
|
|
|
|
public static bool operator == (Method a, Method b)
|
|
{
|
|
if (a == null)
|
|
return (b
|
|
*/
|
|
Method a = this;
|
|
Method b = (Method) o;
|
|
if (a.Name != b.Name)
|
|
return false;
|
|
|
|
if (a.Params == null)
|
|
return b.Params == null;
|
|
|
|
if (b.Params == null)
|
|
return false;
|
|
|
|
return (a.Params.SignatureTypes == b.Params.SignatureTypes);
|
|
}
|
|
/*
|
|
public static bool operator != (Method a, Method b)
|
|
{
|
|
return !(
|
|
if (a.Name == b.Name)
|
|
return false;
|
|
|
|
if (a.Params == null)
|
|
return b.Params != null;
|
|
|
|
if (b.Params == null)
|
|
return true;
|
|
|
|
return (a.Params.SignatureTypes != b.Params.SignatureTypes);
|
|
}
|
|
*/
|
|
private bool Initialize ()
|
|
{
|
|
if (initialized)
|
|
return true;
|
|
|
|
if (parms != null && !parms.Validate ())
|
|
return false;
|
|
|
|
XmlElement ret_elem = elem["return-type"];
|
|
if (ret_elem == null) {
|
|
Console.Write("Missing return type in method ");
|
|
Statistics.ThrottledCount++;
|
|
return false;
|
|
}
|
|
|
|
rettype = ret_elem.GetAttribute("type");
|
|
m_ret = SymbolTable.GetMarshalType(rettype);
|
|
s_ret = SymbolTable.GetCSType(rettype);
|
|
cname = elem.GetAttribute("cname");
|
|
|
|
if (parms != null && parms.ThrowsException)
|
|
safety = "unsafe ";
|
|
else
|
|
safety = "";
|
|
|
|
is_get = ((parms != null && (parms.IsAccessor || (parms.Count == 0 && s_ret != "void")) || (parms == null && s_ret != "void")) && Name.Substring(0, 3) == "Get");
|
|
is_set = ((parms != null && (parms.IsAccessor || (parms.Count == 1 && s_ret == "void"))) && (Name.Substring(0, 3) == "Set"));
|
|
|
|
if (parms != null) {
|
|
parms.CreateSignature (is_set);
|
|
sig = "(" + parms.Signature + ")";
|
|
isig = "(IntPtr raw, " + parms.ImportSig + ");";
|
|
call = "(Handle, " + parms.CallString + ")";
|
|
} else {
|
|
sig = "()";
|
|
isig = "(IntPtr raw);";
|
|
call = "(Handle)";
|
|
}
|
|
|
|
initialized = true;
|
|
return true;
|
|
}
|
|
|
|
public bool Validate ()
|
|
{
|
|
if (!Initialize ())
|
|
return false;
|
|
|
|
if (m_ret == "" || s_ret == "") {
|
|
Console.Write("rettype: " + rettype + " method ");
|
|
Statistics.ThrottledCount++;
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
private Method GetComplement ()
|
|
{
|
|
char complement;
|
|
if (is_get)
|
|
complement = 'S';
|
|
else
|
|
complement = 'G';
|
|
|
|
return container_type.GetMethod (complement + elem.GetAttribute("name").Substring (1));
|
|
}
|
|
|
|
private void GenerateDeclCommon (StreamWriter sw)
|
|
{
|
|
sw.Write(safety);
|
|
if (is_get || is_set) {
|
|
if (s_ret == "void")
|
|
s_ret = parms.AccessorReturnType;
|
|
sw.Write(s_ret);
|
|
sw.Write(" ");
|
|
sw.Write(Name.Substring (3));
|
|
sw.WriteLine(" { ");
|
|
} else {
|
|
if (elem.HasAttribute("new_flag"))
|
|
sw.Write("new ");
|
|
sw.Write(s_ret + " " + Name + sig);
|
|
}
|
|
}
|
|
|
|
public void GenerateDecl (StreamWriter sw)
|
|
{
|
|
if (!Initialize ())
|
|
return;
|
|
|
|
if (is_get || is_set)
|
|
{
|
|
Method comp = GetComplement ();
|
|
if (comp != null && comp.Validate () && is_set)
|
|
return;
|
|
|
|
sw.Write("\t\t");
|
|
GenerateDeclCommon (sw);
|
|
|
|
sw.Write("\t\t\t");
|
|
sw.Write ((is_get) ? "get;" : "set;");
|
|
|
|
if (comp != null && comp.is_set)
|
|
sw.WriteLine (" set;");
|
|
else
|
|
sw.WriteLine ();
|
|
|
|
sw.WriteLine ("\t\t}");
|
|
}
|
|
else
|
|
{
|
|
sw.Write("\t\t");
|
|
GenerateDeclCommon (sw);
|
|
sw.WriteLine (";");
|
|
}
|
|
|
|
Statistics.MethodCount++;
|
|
}
|
|
|
|
protected void GenerateImport (StreamWriter sw)
|
|
{
|
|
sw.WriteLine("\t\t[DllImport(\"" + SymbolTable.GetDllName(ns) +
|
|
"\", CallingConvention=CallingConvention.Cdecl)]");
|
|
sw.Write("\t\tstatic extern " + safety + m_ret + " " + cname + isig);
|
|
sw.WriteLine();
|
|
}
|
|
|
|
public void Generate (StreamWriter sw)
|
|
{
|
|
Method comp = null;
|
|
|
|
if (!Initialize ())
|
|
return;
|
|
|
|
/* we are generated by the get Method, if there is one */
|
|
if (is_set || is_get)
|
|
{
|
|
comp = GetComplement ();
|
|
if (comp != null && comp.Validate () && is_set && parms.AccessorReturnType == comp.s_ret)
|
|
return;
|
|
if (comp != null && is_set && parms.AccessorReturnType != comp.s_ret)
|
|
{
|
|
is_set = false;
|
|
parms.CreateSignature (false);
|
|
call = "(Handle, " + parms.CallString + ")";
|
|
comp = null;
|
|
}
|
|
/* some setters take more than one arg */
|
|
if (comp != null && !comp.is_set)
|
|
comp = null;
|
|
}
|
|
else {
|
|
sw.WriteLine("\t\t/// <summary> " + Name + " Method </summary>");
|
|
sw.WriteLine("\t\t/// <remarks> To be completed </remarks>");
|
|
sw.WriteLine();
|
|
}
|
|
|
|
GenerateImport (sw);
|
|
if (comp != null && s_ret == comp.parms.AccessorReturnType)
|
|
comp.GenerateImport (sw);
|
|
|
|
sw.Write("\t\tpublic ");
|
|
GenerateDeclCommon (sw);
|
|
|
|
if (is_get || is_set)
|
|
{
|
|
sw.Write ("\t\t\t");
|
|
sw.Write ((is_get) ? "get" : "set");
|
|
GenerateBody (sw, "\t");
|
|
}
|
|
else
|
|
GenerateBody (sw, "");
|
|
|
|
if (is_get || is_set)
|
|
{
|
|
if (comp != null && s_ret == comp.parms.AccessorReturnType)
|
|
{
|
|
sw.WriteLine ();
|
|
sw.Write ("\t\t\tset");
|
|
comp.GenerateBody (sw, "\t");
|
|
}
|
|
sw.WriteLine ();
|
|
sw.WriteLine ("\t\t}");
|
|
}
|
|
else
|
|
sw.WriteLine();
|
|
|
|
sw.WriteLine();
|
|
|
|
Statistics.MethodCount++;
|
|
}
|
|
|
|
protected void GenerateBody (StreamWriter sw, string indent)
|
|
{
|
|
sw.WriteLine(" {");
|
|
if (parms != null)
|
|
parms.Initialize(sw, is_get, indent);
|
|
|
|
sw.Write(indent + "\t\t\t");
|
|
if (m_ret == "void") {
|
|
sw.WriteLine(cname + call + ";");
|
|
} else {
|
|
if (SymbolTable.IsObject (rettype))
|
|
{
|
|
sw.WriteLine(m_ret + " raw_ret = " + cname + call + ";");
|
|
sw.WriteLine(indent +"\t\t\t" + s_ret + " ret = " + SymbolTable.FromNative(rettype, "raw_ret") + ";");
|
|
sw.WriteLine(indent + "\t\t\tif (ret == null) ret = new " + s_ret + "(raw_ret);");
|
|
}
|
|
else
|
|
sw.WriteLine(s_ret + " ret = " + SymbolTable.FromNative(rettype, cname + call) + ";");
|
|
}
|
|
|
|
if (parms != null)
|
|
parms.HandleException (sw, indent);
|
|
|
|
if (is_get && parms != null)
|
|
sw.WriteLine (indent + "\t\t\treturn " + parms.AccessorName + ";");
|
|
else if (m_ret != "void")
|
|
sw.WriteLine (indent + "\t\t\treturn ret;");
|
|
|
|
sw.Write(indent + "\t\t}");
|
|
}
|
|
}
|
|
}
|
|
|