3 * This code renders the Gtk tree into a set of Gtk elements.
4 * principle = one NodeToGtk wraps around the original 'node'
6 * it's called by the view element with
7 * var x = new JsRender.NodeToGtk(file.tree);
8 var obj = x.munge() as Gtk.Widget;
12 * The idea behind the Javascript tools stuff is that we can
13 * transform what is actually being requested to be rendered
14 * -- eg. an AboutBox, and turn that into load of real widgets..
15 * that could be displayed..
17 * we could go on the theory that we send the whole tree to the 'plugin'
18 * and that would do all the transformations before rendering..
19 * -- this would make more sense...
20 * -- otherwise we would call it on each element, and might get really confused
26 public class JsRender.NodeToGtk : Object {
29 Object wrapped_object;
32 Gee.ArrayList<NodeToGtk> children;
34 Gee.ArrayList<string> els;
35 //Gee.ArrayList<string> skip;
36 Gee.HashMap<string,string> ar_props;
37 public static int vcnt = 0;
39 public NodeToGtk( Node node , NodeToGtk? parent_obj = null)
42 this.els = new Gee.ArrayList<string>();
43 this.children = new Gee.ArrayList<NodeToGtk>();
44 //this.skip = new Gee.ArrayList<string>();
45 this.ar_props = new Gee.HashMap<string,string>();
46 this.parentObj = parent_obj;
48 if (parent_obj == null) {
49 // then serialize up the node,
50 // send it to javascript for processsing,
51 // then rebuild node from return value..
53 var ret = Palete.Javascript.singleton().executeFile(
54 BuilderApplication.configDirectory() + "/resources/node_to_gtk.js",
58 var new_node = new Node();
59 var pa = new Json.Parser();
60 pa.load_from_data(ret);
61 var rnode = pa.get_root();
64 new_node.loadFromJson(rnode.get_object(), 2);
67 } catch (Palete.JavascriptError e) {
68 print("Error: %s\n", e.message);
76 public Object? munge ( )
78 var ret = this.mungeNode();
83 return ret.wrapped_object;
86 public NodeToGtk? mungeChild( Node cnode)
88 var x = new NodeToGtk(cnode, this);
94 public NodeToGtk? mungeNode()
97 var parent = this.parentObj != null ? this.parentObj.wrapped_object : null;
98 var cls = this.node.fqn().replace(".", "");
99 var ns = this.node.fqn().split(".")[0];
100 var gtkbuilder = new global::Gtk.Builder();
102 var cls_gtype = gtkbuilder.get_type_from_name(cls);
103 print("Type: %s ?= %s\n", this.node.fqn(), cls_gtype.name());
105 if (cls_gtype == GLib.Type.INVALID) {
106 print("SKIP - gtype is invalid\n");
109 // if it's a window... -- things we can not render....
111 if (cls_gtype.is_a(typeof(global::Gtk.Window))) {
112 // what if it has none...
113 if (this.node.items.size < 1) {
116 return this.mungeChild(this.node.items.get(0));
118 if (cls_gtype.is_a(typeof(global::Gtk.Popover))) {
119 // what if it has none...
120 if (this.node.items.size < 1) {
123 return this.mungeChild(this.node.items.get(0));
126 var ret = Object.new(cls_gtype);
127 ret.ref(); //??? problematic?
128 this.wrapped_object = ret;
133 //case "GtkTreeStore": // top level.. - named and referenced
134 case "GtkListStore": // top level.. - named and referenced
135 //case "GtkTreeViewColumn": // part of liststore?!?!
136 //case "GtkMenu": // top level..
137 //case "GtkCellRendererText":
138 case "GtkSourceBuffer":
139 case "GtkClutterActor"://fixme..
140 case "GtkClutterEmbed"://fixme.. -- we can not nest embedded.. need to solve..
151 if (parent != null && parent.get_type().is_a(typeof(global::Gtk.Container))) {
152 this.packContainerParams();
155 var cls_gir =Palete.Gir.factoryFqn(this.node.fqn());
156 if (cls_gir == null) {
159 //var id = this.node.uid();
160 //var ret = @"$pad<object class=\"$cls\" id=\"$id\">\n";
162 var props = cls_gir.props;
165 var pviter = props.map_iterator();
166 while (pviter.next()) {
168 // print("Check: " +cls + "::(" + pviter.get_value().propertyof + ")" + pviter.get_key() + " " );
169 var k = pviter.get_key();
170 // skip items we have already handled..
171 if (!this.node.has(k)) {
174 // find out the type of the property...
175 var type = pviter.get_value().type;
176 type = Palete.Gir.fqtypeLookup(type, ns);
178 var ocl = (ObjectClass) cls_gtype.class_ref ();
179 var ps = ocl.find_property(k);
181 // attempt to read property type and enum...
184 var vt = ps.value_type;
187 var raw_val = this.node.get(k).strip();
188 var rv_s = raw_val.split(".");
189 if (rv_s.length > 0) {
190 raw_val = rv_s[rv_s.length-1];
191 EnumClass ec = (EnumClass) vt.class_ref ();
192 for (var i =0;i< ec.n_values; i++) {
193 var ev = ec.values[i].value_ame;
194 var ev_s= ev.split("_");
195 if (raw_val == ev_s[ev_s.length-1]) {
196 var sval = GLib.Value(typeof(int));
197 sval.set_int(ec.values[i].value);
198 ret.set_property(k, sval);
209 var val = this.toValue(this.node.get(k).strip(), type);
211 print("skip (failed to transform value %s type = %s from %s\n",
212 cls + "." + k, type, this.node.get(k).strip());
215 print ("set_property ( %s , %s / %s)\n", k, this.node.get(k).strip(), val.strdup_contents());
218 ret.set_property(k, val);
223 // for now... - just try the builder style packing
227 if (this.node.items.size < 1) {
231 for (var i = 0; i < this.node.items.size; i++ ) {
233 var ch = this.mungeChild(this.node.items.get(i));
235 this.children.add(ch);
240 this.afterChildren();
249 public void afterChildren()
251 // things like GtkNotebook - we have to pack children after they have been created..
252 var cls = this.node.fqn().replace(".", "");
254 if (cls == "GtkNotebook") {
255 this.afterChildrenGtkNotebook();
262 public void afterChildrenGtkNotebook()
264 // we have a number of children..
265 // some are labels - this might need to be more complex...
266 // perhaps labels should be a special property labels[] of the notebook..
267 var labels = new Gee.ArrayList<NodeToGtk>();
268 var bodies = new Gee.ArrayList<NodeToGtk>();
269 for (var i = 0; i < this.children.size; i++) {
270 var cn = this.children.get(i).node.fqn().replace(".", "");
271 if (cn != "GtkLabel") {
272 bodies.add(this.children.get(i));
275 labels.add(this.children.get(i));
277 for (var i = 0; i < bodies.size; i++) {
278 ((global::Gtk.Notebook)this.wrapped_object).append_page(
279 (global::Gtk.Notebook) bodies.get(i).wrapped_object,
280 (labels.size > i - 1) ?
281 (global::Gtk.Notebook) labels.get(i).wrapped_object :
294 * called after the this.object has been created
295 * and it needs to be packed onto parent.
297 public void packParent()
299 var cls = this.node.fqn().replace(".", "");
301 var gtkbuilder = new global::Gtk.Builder();
302 var cls_gtype = gtkbuilder.get_type_from_name(cls);
304 if (this.parentObj == null) {
309 var parent = this.parentObj.wrapped_object;
313 if (parent == null) { // no parent.. can not pack.
316 // ------------- handle various special parents .. -----------
318 var par_type = this.parentObj.node.fqn().replace(".", "");
320 if (par_type == "GtkNotebook") {
321 // do not pack - it's done afterwards...
325 // ------------- handle various child types.. -----------
327 if (cls == "GtkMenu") {
332 if (cls == "GtkTreeStore") { // other stores?
333 // tree store is buildable??? ---
334 this.packTreeStore();
337 if (cls =="GtkTreeViewColumn") { // other stores?
338 //?? treeview column is actually buildable -- but we do not use the builder???
339 this.packTreeViewColumn();
342 if (cls_gtype.is_a(typeof(global::Gtk.CellRenderer))) { // other stores?
343 this.packCellRenderer();
349 // -- handle buildable add_child..
350 if ( cls_gtype.is_a(typeof(global::Gtk.Buildable))
352 parent.get_type().is_a(typeof(global::Gtk.Buildable))
355 ((global::Gtk.Buildable)parent).add_child(gtkbuilder,
356 this.wrapped_object, null);
365 public void packMenu()
369 var parent = this.parentObj.wrapped_object;
370 if (!parent.get_type().is_a(typeof(global::Gtk.Widget))) {
371 print("skip menu pack - parent is not a widget");
375 var p = (global::Gtk.Menu)this.wrapped_object;
376 ((global::Gtk.Widget)parent).button_press_event.connect((s, ev) => {
377 p.set_screen(Gdk.Screen.get_default());
379 p.popup(null, null, null, ev.button, ev.time);
384 public void packTreeStore()
386 var parent = this.parentObj.wrapped_object;
387 if (!parent.get_type().is_a(typeof(global::Gtk.TreeView))) {
388 print("skip treestore pack - parent is not a treeview");
391 ((global::Gtk.TreeView)parent).set_model((global::Gtk.TreeModel)this.wrapped_object);
394 public void packTreeViewColumn()
396 var parent = this.parentObj.wrapped_object;
397 if (!parent.get_type().is_a(typeof(global::Gtk.TreeView))) {
398 print("skip packGtkViewColumn pack - parent is not a treeview");
401 ((global::Gtk.TreeView)parent).append_column((global::Gtk.TreeViewColumn)this.wrapped_object);
402 // init contains the add_attribute for what to render...
407 public void packCellRenderer()
409 var parent = this.parentObj.wrapped_object;
410 if (!parent.get_type().is_a(typeof(global::Gtk.TreeViewColumn))) {
411 print("skip packGtkViewColumn pack - parent is not a treeview");
414 ((global::Gtk.TreeViewColumn)parent).pack_start((global::Gtk.CellRenderer)this.wrapped_object, false);
415 // init contains the add_attribute for what to render...
420 public void packContainerParams()
423 if (this.parentObj == null) {
426 // child must be a widget..
427 if (!this.wrapped_object.get_type().is_a(typeof(global::Gtk.Widget))) {
431 var parent_gir = Palete.Gir.factoryFqn(this.parentObj.node.fqn());
433 var parent = this.parentObj.wrapped_object;
435 if (parent_gir == null) {
439 // let's test just setting expand to false...
440 var cls_methods = parent_gir.methods;
441 if (cls_methods == null) {
445 if (!this.node.props.has_key("* pack") ||
446 this.node.props.get("* pack").length < 1) {
450 var ns = this.parentObj.node.fqn().split(".")[0];
452 var pack = this.node.props.get("* pack").split(",");
455 if (cls_methods.has_key(pack[0])) {
456 var mparams = cls_methods.get(pack[0]).paramset.params;
457 for (var i = 1; i < mparams.size; i++ ) {
458 if (i > (pack.length -1)) {
462 var k = mparams.get(i).name;
466 var type = mparams.get(i).type;
467 type = Palete.Gir.fqtypeLookup(type, ns);
469 var val = this.toValue(pack[i].strip(), type);
471 print("skip (failed to transform value %s type = %s from %s\n",
472 this.parentObj.node.fqn() + "." + k, type, pack[i].strip());
475 print ("pack:set_property ( %s , %s / %s)\n", k, pack[i].strip(), val.strdup_contents());
477 ((global::Gtk.Container)parent).child_set_property(
478 (global::Gtk.Widget)this.wrapped_object , k, val);
490 public GLib.Value? toValue(string val, string type) {
494 if (type == "string") {
495 var qret = GLib.Value(typeof(string));
496 qret.set_string(val);
502 * var gtkbuilder = new global::Gtk.Builder();
503 var prop_gtype = gtkbuilder.get_type_from_name(type);
506 if (prop_gtype == GLib.Type.INVALID) {
517 var ret = GLib.Value(typeof(bool));
518 ret.set_boolean(val.down() == "false" ? false : true);
522 var ret = GLib.Value(typeof(uint));
523 ret.set_uint(int.parse(val));
527 var ret = GLib.Value(typeof(int));
528 ret.set_int(int.parse(val));
534 var ret = GLib.Value(typeof(long));
535 ret.set_long((long)int64.parse(val));
539 var ret = GLib.Value(typeof(ulong));
540 ret.set_ulong((ulong) uint64.parse(val));
544 var ret = GLib.Value(typeof(float));
545 ret.set_float((float)double.parse(val));
549 var ret = GLib.Value(typeof(string));
556 var sval = GLib.Value(typeof(string));
557 sval.set_string(val);
559 if (!sval.transform(ref ret)) {
566 // should not get here..