2 // valac -g --pkg gee-1.0 --pkg libxml-2.0 --pkg gobject-introspection-1.0 --pkg json-glib-1.0 Palete/Gir.vala -o /tmp/Gir
4 public static int main (string[] args) {
6 var g = Palete.Gir.factory("Gtk");
7 var test = g.classes.get("ToolButton");
10 var generator = new Json.Generator ();
11 var n = new Json.Node(Json.NodeType.OBJECT);
12 n.set_object(test.toJSON());
13 generator.set_root(n);
15 generator.pretty = true;
17 print(generator.to_data(null));
22 public errordomain GirError {
35 public class Gir : GirObject {
37 public static Gee.HashMap<string,Gir> cache = null;
38 //Gee.Hashmap<string,what> nodes;
42 public static Gir? factory(string ns) {
44 cache = new Gee.HashMap<string,Gir>();
46 var ret = cache.get(ns);
49 var add = new Gir(ns);
52 var iter = add.classes.map_iterator();
54 iter.get_value().overlayParent();
56 // loop again and add the ctor properties.
57 iter = add.classes.map_iterator();
59 iter.get_value().overlayCtorProperties();
70 public static GirObject? factoryFqn(string fqn)
72 var bits = fqn.split(".");
73 if (bits.length < 1) {
77 var f = (GirObject)factory(bits[0]);
79 if (bits.length == 1 || f ==null) {
82 return f.fetchByFqn(fqn.substring(bits[0].length+1)); // since classes are stored in fqn format...?
89 * guess the fqn of a type == eg. gboolean or Widget etc...
91 public static string fqtypeLookup(string type, string ns) {
93 if (g.classes.has_key(type)) {
94 return ns + "." + type;
97 if (g.consts.has_key(type)) {
98 return ns + "." + type;
102 // look at includes..
103 var iter = g.includes.map_iterator();
105 // skip empty namespaces on include..?
106 if ( iter.get_key() == "") {
109 var ret = fqtypeLookup(type, iter.get_key());
121 public static string guessDefaultValueForType(string type) {
122 //print("guessDefaultValueForType: %s\n", type);
123 if (type.length < 1 || type.contains(".")) {
134 return "?"+ type + "?";
140 public Gir.new_empty (string ns)
144 public Gir (string ns)
146 var xns = ns == "Glade" ? "Gladeui" : ns;
147 var gi = GI.Repository.get_default();
148 gi.require(xns, null, 0);
150 var ver = gi.get_version(xns);
151 unowned GLib.SList<string> pth = GI.Repository.get_search_path ();
152 var gir_path = pth.nth_data(0).replace("/lib/girepository-1.0", "/share/gir-1.0");
154 gir_path = gir_path.replace("/lib/x86_64-linux-gnu/girepository-1.0", "/share/gir-1.0");
160 var file = gir_path + "/" + xns + "-" + ver + ".gir";
161 // print("ns: " + ns + "\n");
162 // print("ver: " + ver + "\n");
168 //this.nodes = new Gee.Hashmap<string,what>();
170 var doc = Xml.Parser.parse_file (file);
171 var root = doc->get_root_element();
172 this.walk( root, (GirObject) this );
177 public void walk(Xml.Node* element, GirObject? parent)
179 var n = element->get_prop("name");
180 // ignore null or c:include...
181 if (n == null || (element->ns->prefix != null && element->ns->prefix == "c")) {
184 //print("%s:%s (%s ==> %s\n", element->ns->prefix , element->name , parent.name , n);
185 switch (element->name) {
190 parent.includes.set(n, element->get_prop("version"));
206 //break; // not handled..
209 var c = new GirObject("Class", parent.name + "." + n);
210 parent.classes.set(n, c);
212 c.parent = element->get_prop("parent");
214 if (c.parent == null) {
221 var c = new GirObject("Interface", parent.name + "." + n);
223 parent.classes.set(n, c);
226 c.parent = element->get_prop("parent");
227 if (c.parent == null) {
235 parent.doctxt = element->get_content();
239 parent.implements.add(n);
244 var c = new GirObject("Ctor",n);
247 parent.ctors.set(n,c);
252 var c = new GirObject("Return", "return-value");
255 parent.return_value = c;
259 case "virtual-method": // not sure...
262 var c = new GirObject("Signal",n);
263 parent.signals.set(n,c);
267 case "signal": // Glib:signal
268 var c = new GirObject("Signal",n.replace("-", "_"));
271 parent.signals.set(n.replace("-", "_"),c);
277 case "callback": // not sure...
284 return; // no children?
288 var c = new GirObject("Method",n);
291 c.propertyof = parent.name;
292 parent.methods.set(n,c);
297 var c = new GirObject("Paramset",n);
304 case "instance-parameter":
306 // looks like this is the C first arg, that is ignored (as it is
307 // treated as 'this' )
308 var c = new GirObject("Param",n);
311 c.is_instance = true;
312 parent.params.add(c);
317 var c = new GirObject("Param",n);
320 parent.params.add(c);
322 this.checkParamOverride(c);
327 var c = new GirObject("Prop",n.replace("-", "_"));
330 c.propertyof = parent.name;
331 parent.props.set(n.replace("-", "_"),c);
336 var c = new GirObject("Function",n);
339 parent.methods.set(n,c);
344 parent.is_array = true;
345 break; // type is added soon..
348 parent.is_varargs= true;
352 var c = new GirObject("Const",n);
354 c.value = element->get_prop("value");
356 parent.consts.set(n,c);
362 var c = new GirObject("Enum",n);
365 parent.consts.set(n,c);
371 var c = new GirObject("EnumMember",n);
374 c.value = element->get_prop("value");
375 parent.consts.set(n,c);
380 case "doc-deprecated":
383 case "record": // struct?
388 case "prerequisite": // ignore?
390 case "union": // ignore?
393 print("UNHANDLED Gir file element: " + element->name +"\n");
397 if (element->name == "signal") {
402 if (element->name == "return-value") {
403 path += ".return-value";
405 print(path + ":" + element->name + "\n");
407 //var d = getAttribute(element,'doc');
409 // Seed.print(path + ':' + d);
412 for (Xml.Node* iter = element->children; iter != null; iter = iter->next) {
413 if (iter->type == Xml.ElementType.TEXT_NODE) {
416 this.walk(iter, parent);
420 public string doc(string what)
422 var ar = what.split(".");
423 var cls = this.classes.get(ar[1]);
424 if (ar.length == 2) {
425 return cls.doctxt != null ? cls.doctxt : "";
427 // return the property.. by default..
428 var pr = cls.props.get(ar[2]);
429 return pr.doctxt != null ? pr.doctxt : "";
435 public void checkParamOverride(GirObject c)
437 var parset = c.gparent;
438 if (parset == null || parset.nodetype != "Paramset") {
441 var method = parset.gparent;
442 if (method == null || method.nodetype != "Ctor") {
445 var cls = method.gparent;
446 if (cls == null || cls.nodetype != "Class") {
452 c.name = this.fetchOverride( cls.name, method.name, c.name);
454 public static bool overrides_loaded = false;
455 public static Gee.HashMap<string,string> overrides;
457 public string fetchOverride( string cls, string method, string param)
459 // overrides should be in a file Gir.overides
460 // in that "Gtk.Label.new.str" : "label"
461 this.loadOverrides();
462 var key = "%s.%s.%s".printf(cls,method,param);
463 //print("Chekcing for key %s\n", key);
464 if (!overrides.has_key(key)) {
467 return overrides.get(key);
471 public void loadOverrides(bool force = false)
473 if (overrides_loaded && ! force) {
477 var pa = new Json.Parser();
478 pa.load_from_file(BuilderApplication.configDirectory() + "/resources/Gir.overides");
479 var node = pa.get_root();
481 if (node.get_node_type () != Json.NodeType.OBJECT) {
482 throw new GirError.INVALID_FORMAT ("Error loading gir.overides : Unexpected element type %s", node.type_name ());
484 overrides = new Gee.HashMap<string,string>();
487 var obj = node.get_object ();
490 obj.foreach_member((o , key, value) => {
493 var v = obj.get_string_member(key);
496 overrides.set(key, v);
500 overrides_loaded = true;