nobody at rubyforge.org
2006-Dec-23 03:04 UTC
[Wxruby-development] [794] trunk/wxruby2: Fix get_ruby_object so it works with SWIG tracking, move find_window_xx
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN"
"http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head><meta http-equiv="content-type"
content="text/html; charset=utf-8" /><style
type="text/css"><!--
#msg dl { border: 1px #006 solid; background: #369; padding: 6px; color: #fff; }
#msg dt { float: left; width: 6em; font-weight: bold; }
#msg dt:after { content:'':'';}
#msg dl, #msg dt, #msg ul, #msg li, #header, #footer { font-family:
verdana,arial,helvetica,sans-serif; font-size: 10pt; }
#msg dl a { font-weight: bold}
#msg dl a:link { color:#fc3; }
#msg dl a:active { color:#ff0; }
#msg dl a:visited { color:#cc6; }
h3 { font-family: verdana,arial,helvetica,sans-serif; font-size: 10pt;
font-weight: bold; }
#msg pre { overflow: auto; background: #ffc; border: 1px #fc0 solid; padding:
6px; }
#msg ul, pre { overflow: auto; }
#header, #footer { color: #fff; background: #636; border: 1px #300 solid;
padding: 6px; }
#patch { width: 100%; }
#patch h4 {font-family:
verdana,arial,helvetica,sans-serif;font-size:10pt;padding:8px;background:#369;color:#fff;margin:0;}
#patch .propset h4, #patch .binary h4 {margin:0;}
#patch pre {padding:0;line-height:1.2em;margin:0;}
#patch .diff {width:100%;background:#eee;padding: 0 0 10px 0;overflow:auto;}
#patch .propset .diff, #patch .binary .diff {padding:10px 0;}
#patch span {display:block;padding:0 10px;}
#patch .modfile, #patch .addfile, #patch .delfile, #patch .propset, #patch
.binary, #patch .copfile {border:1px solid #ccc;margin:10px 0;}
#patch ins {background:#dfd;text-decoration:none;display:block;padding:0 10px;}
#patch del {background:#fdd;text-decoration:none;display:block;padding:0 10px;}
#patch .lines, .info {color:#888;background:#fff;}
--></style>
<title>[794] trunk/wxruby2: Fix get_ruby_object so it works with SWIG
tracking, move find_window_xx</title>
</head>
<body>
<div id="msg">
<dl>
<dt>Revision</dt> <dd>794</dd>
<dt>Author</dt> <dd>brokentoy</dd>
<dt>Date</dt> <dd>2006-12-22 22:04:53 -0500 (Fri, 22 Dec
2006)</dd>
</dl>
<h3>Log Message</h3>
<pre>Fix get_ruby_object so it works with SWIG tracking, move
find_window_xx
instance methods into ruby</pre>
<h3>Modified Paths</h3>
<ul>
<li><a
href="#trunkwxruby2libwxclasseswindowrb">trunk/wxruby2/lib/wx/classes/window.rb</a></li>
<li><a
href="#trunkwxruby2swigclassesWindowi">trunk/wxruby2/swig/classes/Window.i</a></li>
</ul>
</div>
<div id="patch">
<h3>Diff</h3>
<a id="trunkwxruby2libwxclasseswindowrb"></a>
<div class="modfile"><h4>Modified:
trunk/wxruby2/lib/wx/classes/window.rb (793 => 794)</h4>
<pre class="diff"><span>
<span class="info">---
trunk/wxruby2/lib/wx/classes/window.rb        2006-12-21
00:09:12 UTC (rev 793)
+++
trunk/wxruby2/lib/wx/classes/window.rb        2006-12-23
03:04:53 UTC (rev 794)
</span><span class="lines">@@ -2,8 +2,30 @@
</span><span class="cx"> # released under the MIT-style
wxruby2 license
</span><span class="cx">
</span><span class="cx"> class Wx::Window
</span><del>- # create a DC for drawing on the window, run the
block passing in the DC
</del><ins>+ # Creates a device context which can be used for
drawing on the
+ # window, and passes it into a block. Note that this method should
+ # only be used within an on_paint handler.
</ins><span class="cx"> def paint
</span><span class="cx"> yield Wx::PaintDC.new(self)
</span><span class="cx"> end
</span><ins>+
+ # Recursively searches all windows below +self+ and returns the first
+ # window which has the id +an_id+. This corresponds to the find_window
+ # method method in WxWidgets when called with an integer.
+ def find_window_by_id(an_id)
+ Wx::Window.find_window_by_id(an_id, self)
+ end
+
+ # Searches all windows below +self+ and returns the first window which
+ # has the name +a_name+ This corresponds to the find_window method method
+ # in WxWidgets when called with an string.
+ def find_window_by_name(a_name)
+ Wx::Window.find_window_by_name(a_name, self)
+ end
+
+ # Searches all windows below +self+ and returns the first window which
+ # has the label +a_label+.
+ def find_window_by_label(a_label)
+ Wx:Window.find_window_by_label(a_label, self)
+ end
</ins><span class="cx"> end
</span></span></pre></div>
<a id="trunkwxruby2swigclassesWindowi"></a>
<div class="modfile"><h4>Modified:
trunk/wxruby2/swig/classes/Window.i (793 => 794)</h4>
<pre class="diff"><span>
<span class="info">---
trunk/wxruby2/swig/classes/Window.i        2006-12-21
00:09:12 UTC (rev 793)
+++
trunk/wxruby2/swig/classes/Window.i        2006-12-23
03:04:53 UTC (rev 794)
</span><span class="lines">@@ -8,7 +8,6 @@
</span><span class="cx"> %ignore wxWindow::Clear;
</span><span class="cx"> %ignore wxWindow::GetAccessible;
</span><span class="cx"> %ignore
wxWindow::PopEventHandler;
</span><del>-# %ignore wxWindow::SetCaret;
</del><span class="cx">
</span><span class="cx"> // LayoutConstraints are
deprecated and not supported in WxRuby
</span><span class="cx"> %ignore wxWindow::SetConstraints;
</span><span class="lines">@@ -24,7 +23,7 @@
</span><span class="cx">
</span><span class="cx"> %apply int * INOUT { int *
x_INOUT, int * y_INOUT }
</span><span class="cx">
</span><del>-// Typemap for GetChildren
</del><ins>+// Typemap for GetChildren - casts wxObjects to correct
ruby wrappers
</ins><span class="cx"> %typemap(out) wxList& {
</span><span class="cx"> $result = rb_ary_new();
</span><span class="cx">
</span><span class="lines">@@ -37,38 +36,57 @@
</span><span class="cx"> }
</span><span class="cx"> }
</span><span class="cx">
</span><del>-// general-purpose function used to return ruby
wrappers around windows
-// whose type is not known in advance - as in find_window, get_children etc
-
-// TODO - doesn''t work correctly with object tracking - it should
return
-// the same Ruby object if the object has been seen before
-// use something like:
-//        
returnVal = SWIG_NewPointerObj(obj, SWIGTYPE_p_wxWindow, 0);
-// but this causes problems for when a new Ruby object is needed - eg with XRC
</del><ins>+// Returns a ruby wrapper around a wxObject whose wx
class is not known
+// in advance - needed for find_window, get_children etc This is a bit
+// complicated because it must check if the ruby object already exists,
+// and if not, wrap it in the correct class, and ensure that future
+// calls return the same object.
</ins><span class="cx"> %{
</span><del>-static VALUE get_ruby_object(wxObject *obj)
</del><ins>+static VALUE get_ruby_object(wxObject *wx_obj)
</ins><span class="cx"> {
</span><del>-        if
( ! obj )
</del><ins>+ if ( ! wx_obj ) return Qnil;
+
+ // Get the wx class and the ruby class we are converting into
+ wxString class_name( wx_obj->GetClassInfo()->GetClassName() );
+ VALUE r_class;
+ if ( class_name.Len() > 2 )
</ins><span class="cx">
        {
</span><del>-        
return Qnil;
</del><ins>+        
r_class = rb_iv_get(mWxruby2, class_name.mb_str() + 2);
</ins><span class="cx">
        }
</span><del>-        
-        VALUE
returnVal = Qnil;
-        wxString
classNameString(obj->GetClassInfo()->GetClassName());
-        if(classNameString.Len()
> 2)
</del><ins>+ else return Qnil;
+
+ // First, Try fetching a tracked (previously seen) object
+ VALUE r_obj = SWIG_RubyInstanceFor(wx_obj);
+ if ( r_obj != Qnil ) // Found a tracked object
</ins><span class="cx">
        {
</span><del>-        
VALUE ruby_class = rb_iv_get(mWxruby2, classNameString.mb_str()+2);
-        
returnVal = Data_Wrap_Struct(ruby_class,0,0,obj);
</del><ins>+        
// Check the types match (see SWIG_NewPointerObj)
+         VALUE
r_swigtype = rb_iv_get(r_obj, "__swigtype__");
+         if (
r_swigtype != Qnil && rb_obj_is_kind_of(r_obj, r_class) )
+                return
r_obj;
</ins><span class="cx">
        }
</span><del>-        
-        return
returnVal;
</del><ins>+
+ // Otherwise - no previously tracked object found - so wrap a new one.
+ // Get a string with the internal SWIG name tag of the class (eg _p_wxButton)
+ char *swigtype = (char *) malloc(3 + class_name.Len() + 1);
+ sprintf(swigtype, "_p_%s", (const char *)(class_name.mb_str()));
+
+ // Wrap the object, tag its swigtype (core Wx type), and track it
+ // Imitates latter part of SWIG_NewPointerObj
+ r_obj = Data_Wrap_Struct(r_class, 0, 0, wx_obj);
+ rb_iv_set(r_obj, "__swigtype__", rb_str_new2(swigtype));
+ SWIG_RubyAddTracking(wx_obj, r_obj);
+
+ free((void *) swigtype);
+ return r_obj;
</ins><span class="cx"> }
</span><span class="cx">
</span><span class="cx"> %}
</span><span class="cx">
</span><del>-# These are implemented below by hand so casting of
types are correct which is
-# very important when using XRC.
</del><ins>+// This is the instance method - implemented instead in
Ruby - see window.rb
</ins><span class="cx"> %ignore wxWindow::FindWindow;
</span><ins>+
+// These are class methods implemented below by hand so casting of types
+// is correct, which is very important when using XRC.
</ins><span class="cx"> %ignore wxWindow::FindWindowById;
</span><span class="cx"> %ignore
wxWindow::FindWindowByName;
</span><span class="cx"> %ignore
wxWindow::FindWindowByLabel;
</span><span class="lines">@@ -79,29 +97,10 @@
</span><span class="cx"> %include
"include/wxWindow.h"
</span><span class="cx">
</span><span class="cx"> %extend wxWindow {
</span><del>- VALUE find_window(long id)
- {
- VALUE returnVal = Qnil;
-
- wxObject* obj = self->FindWindow(id);
- returnVal = get_ruby_object(obj);
-
- return returnVal;
- }
-
- VALUE find_window(const wxString& name)
- {
- VALUE returnVal = Qnil;
-
- wxObject* obj = self->FindWindow(name);
- returnVal = get_ruby_object(obj);
-
- return returnVal;
- }
-
</del><span class="cx"> static VALUE
find_window_by_id(long id , wxWindow* parent = NULL)
</span><span class="cx"> {
</span><span class="cx"> VALUE returnVal = Qnil;
</span><ins>+
</ins><span class="cx"> wxObject* obj =
wxWindow::FindWindowById(id,parent);
</span><span class="cx"> returnVal =
get_ruby_object(obj);
</span><span class="cx">
</span><span class="lines">@@ -127,5 +126,4 @@
</span><span class="cx">
</span><span class="cx"> return returnVal;
</span><span class="cx"> }
</span><del>-
</del><span class="cx"> }
</span></span></pre>
</div>
</div>
</body>
</html>
Reasonably Related Threads
- [789] trunk/wxruby2/swig/classes/Window.i: Make GetChildren work (tho general fix for get_ruby_object still needed)
- [910] branches/wxruby2/wxwidgets_282: Moved paint() method back into C++, so DC object is destroyed in timely
- [707] trunk/wxruby2/swig/classes/Window.i: Window-paint method now implemented in Ruby (Alex Fenton)
- [744] trunk/wxruby2: get_data support & mem mgmt to all ControlWithItems types, using more
- [1109] trunk/wxruby2/swig: Move marking of MenuBar from direct %markfunc to via mark_wxFrame, so
