wxruby3 0.9.1 → 0.9.2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (153) hide show
  1. checksums.yaml +4 -4
  2. data/.yardopts +1 -0
  3. data/ext/wxruby3/include/wxRubyApp.h +338 -0
  4. data/ext/wxruby3/include/wxruby-Config.h +6 -6
  5. data/ext/wxruby3/swig/common.i +1 -1
  6. data/lib/wx/aui/aui_tab_ctrl.rb +18 -0
  7. data/lib/wx/aui/auinotebook.rb +6 -2
  8. data/lib/wx/aui/require.rb +1 -0
  9. data/lib/wx/core/app.rb +2 -2
  10. data/lib/wx/core/collapsible_pane.rb +1 -1
  11. data/lib/wx/core/controlwithitems.rb +8 -6
  12. data/lib/wx/core/menu.rb +10 -0
  13. data/lib/wx/core/notebook.rb +6 -2
  14. data/lib/wx/core/sizer.rb +11 -0
  15. data/lib/wx/core/task_bar_button.rb +19 -0
  16. data/lib/wx/core/textctrl.rb +11 -1
  17. data/lib/wx/core/{treectrl.rb → tree_ctrl.rb} +31 -12
  18. data/lib/wx/core/window.rb +10 -0
  19. data/lib/wx/doc/app.rb +48 -38
  20. data/lib/wx/doc/art_locator.rb +47 -43
  21. data/lib/wx/doc/aui/auimanager.rb +16 -8
  22. data/lib/wx/doc/aui/auinotebook.rb +20 -5
  23. data/lib/wx/doc/clipboard.rb +11 -7
  24. data/lib/wx/doc/colour_dialog.rb +14 -10
  25. data/lib/wx/doc/controlwithitems.rb +17 -7
  26. data/lib/wx/doc/data_object.rb +2 -2
  27. data/lib/wx/doc/evthandler.rb +113 -108
  28. data/lib/wx/doc/font.rb +1 -0
  29. data/lib/wx/doc/gc_dc.rb +6 -1
  30. data/lib/wx/doc/graphics_context.rb +1 -0
  31. data/lib/wx/doc/grid/grid.rb +22 -1
  32. data/lib/wx/doc/help_controller.rb +11 -7
  33. data/lib/wx/doc/html/html_help_controller.rb +12 -4
  34. data/lib/wx/doc/list_ctrl.rb +33 -29
  35. data/lib/wx/doc/menu.rb +20 -0
  36. data/lib/wx/doc/notebook.rb +21 -0
  37. data/lib/wx/doc/pg/events.rb +13 -9
  38. data/lib/wx/doc/pg/pg_property.rb +18 -0
  39. data/lib/wx/doc/progress_dialog.rb +36 -32
  40. data/lib/wx/doc/prt/page_setup_dialog.rb +20 -12
  41. data/lib/wx/doc/prt/print_data.rb +13 -5
  42. data/lib/wx/doc/prt/print_dialog.rb +31 -23
  43. data/lib/wx/doc/prt/printer.rb +20 -12
  44. data/lib/wx/doc/radio_box.rb +19 -15
  45. data/lib/wx/doc/rbn/ribbon_bar.rb +13 -5
  46. data/lib/wx/doc/rbn/ribbon_button_bar.rb +13 -5
  47. data/lib/wx/doc/rbn/ribbon_gallery.rb +13 -5
  48. data/lib/wx/doc/rbn/ribbon_tool_bar.rb +13 -5
  49. data/lib/wx/doc/region_iterator.rb +32 -28
  50. data/lib/wx/doc/rtc/rich_text_composite_object.rb +24 -0
  51. data/lib/wx/doc/rtc/rich_text_ctrl.rb +24 -0
  52. data/lib/wx/doc/rtc/rich_text_paragraph.rb +24 -0
  53. data/lib/wx/doc/rtc/richtext_buffer.rb +27 -19
  54. data/lib/wx/doc/rtc/richtext_printing.rb +17 -9
  55. data/lib/wx/doc/rtc/richtext_style_sheet.rb +17 -9
  56. data/lib/wx/doc/sizer.rb +20 -0
  57. data/lib/wx/doc/stc/styled_text_ctrl.rb +24 -0
  58. data/lib/wx/doc/stream.rb +39 -35
  59. data/lib/wx/doc/system_settings.rb +30 -26
  60. data/lib/wx/doc/text_validator.rb +12 -8
  61. data/lib/wx/doc/textctrl.rb +16 -0
  62. data/lib/wx/doc/tree_ctrl.rb +95 -0
  63. data/lib/wx/doc/treebook.rb +9 -5
  64. data/lib/wx/doc/v_list_box.rb +9 -5
  65. data/lib/wx/doc/variant.rb +164 -160
  66. data/lib/wx/doc/window.rb +57 -47
  67. data/lib/wx/doc/window_disabler.rb +10 -6
  68. data/lib/wx/grid/grid.rb +27 -4
  69. data/lib/wx/pg/pg_property.rb +22 -0
  70. data/lib/wx/rtc/require.rb +3 -0
  71. data/lib/wx/rtc/rich_text_composite_object.rb +25 -0
  72. data/lib/wx/rtc/rich_text_ctrl.rb +25 -0
  73. data/lib/wx/rtc/rich_text_paragraph.rb +25 -0
  74. data/lib/wx/stc/require.rb +1 -0
  75. data/lib/wx/stc/styled_text_ctrl.rb +25 -0
  76. data/lib/wx/version.rb +1 -1
  77. data/rakelib/lib/config/linux.rb +0 -3
  78. data/rakelib/lib/config/macosx.rb +2 -2
  79. data/rakelib/lib/config/mingw.rb +1 -1
  80. data/rakelib/lib/config/unixish.rb +1 -1
  81. data/rakelib/lib/config.rb +14 -4
  82. data/rakelib/lib/core/package.rb +14 -7
  83. data/rakelib/lib/core/spec.rb +6 -1
  84. data/rakelib/lib/director/accelerator.rb +2 -3
  85. data/rakelib/lib/director/accessible.rb +47 -0
  86. data/rakelib/lib/director/app.rb +12 -319
  87. data/rakelib/lib/director/app_traits.rb +10 -12
  88. data/rakelib/lib/director/ctrl_with_items.rb +17 -5
  89. data/rakelib/lib/director/data_format.rb +1 -1
  90. data/rakelib/lib/director/derived_dc.rb +1 -1
  91. data/rakelib/lib/director/dialog.rb +1 -0
  92. data/rakelib/lib/director/drag_image.rb +2 -3
  93. data/rakelib/lib/director/event.rb +2 -2
  94. data/rakelib/lib/director/frame.rb +1 -3
  95. data/rakelib/lib/director/gdicommon.rb +8 -10
  96. data/rakelib/lib/director/graphics_context.rb +2 -4
  97. data/rakelib/lib/director/grid_ctrl.rb +34 -3
  98. data/rakelib/lib/director/icon.rb +5 -2
  99. data/rakelib/lib/director/list_ctrl.rb +5 -6
  100. data/rakelib/lib/director/locale.rb +1 -3
  101. data/rakelib/lib/director/log.rb +1 -4
  102. data/rakelib/lib/director/media_ctrl.rb +54 -0
  103. data/rakelib/lib/director/menu.rb +16 -1
  104. data/rakelib/lib/director/menu_item.rb +2 -2
  105. data/rakelib/lib/director/pgproperties.rb +1 -1
  106. data/rakelib/lib/director/pgproperty.rb +24 -1
  107. data/rakelib/lib/director/property_grid_interface.rb +5 -10
  108. data/rakelib/lib/director/richtext_composite_object.rb +25 -0
  109. data/rakelib/lib/director/richtext_ctrl.rb +14 -4
  110. data/rakelib/lib/director/richtext_formatting_dialog.rb +7 -5
  111. data/rakelib/lib/director/richtext_paragraph_layout_box.rb +9 -7
  112. data/rakelib/lib/director/sizer.rb +15 -0
  113. data/rakelib/lib/director/static_box.rb +4 -5
  114. data/rakelib/lib/director/styled_text_ctrl.rb +12 -0
  115. data/rakelib/lib/director/task_bar_button.rb +30 -0
  116. data/rakelib/lib/director/task_bar_icon.rb +5 -13
  117. data/rakelib/lib/director/textctrl.rb +12 -1
  118. data/rakelib/lib/director/tool_tip.rb +1 -1
  119. data/rakelib/lib/director/top_level_window.rb +4 -5
  120. data/rakelib/lib/director/tree_ctrl.rb +5 -6
  121. data/rakelib/lib/director/variant.rb +1 -1
  122. data/rakelib/lib/director/window.rb +24 -5
  123. data/rakelib/lib/director.rb +4 -4
  124. data/rakelib/lib/extractor/function.rb +6 -6
  125. data/rakelib/lib/extractor.rb +34 -5
  126. data/rakelib/lib/generate/analyzer.rb +8 -3
  127. data/rakelib/lib/generate/doc/busy_info.yaml +0 -2
  128. data/rakelib/lib/generate/doc/clipboard.yaml +0 -2
  129. data/rakelib/lib/generate/doc/cursor.yaml +0 -2
  130. data/rakelib/lib/generate/doc/panel.yaml +7 -0
  131. data/rakelib/lib/generate/doc.rb +76 -14
  132. data/rakelib/lib/specs/interfaces.rb +161 -160
  133. data/rakelib/lib/typemap/common.rb +30 -1
  134. data/rakelib/yard/templates/default/fulldoc/html/css/wxruby3.css +74 -2
  135. data/rakelib/yard/templates/default/fulldoc/html/full_list.erb +38 -0
  136. data/rakelib/yard/templates/default/fulldoc/html/setup.rb +39 -0
  137. data/rakelib/yard/templates/default/tags/html/wxrb_require.erb +10 -0
  138. data/rakelib/yard/templates/default/tags/setup.rb +16 -0
  139. data/rakelib/yard/yard-custom-templates.rb +3 -0
  140. data/samples/treectrl/treectrl.rb +1 -1
  141. data/tests/media/beep_lo.wav +0 -0
  142. data/tests/test_list_ctrl.rb +1 -1
  143. data/tests/test_media_ctrl.rb +38 -0
  144. data/tests/test_menu.rb +69 -0
  145. data/tests/test_pg.rb +27 -0
  146. data/tests/test_richtext.rb +45 -0
  147. data/tests/test_sizer.rb +16 -0
  148. data/tests/test_std_controls.rb +42 -1
  149. data/tests/test_styled_text_ctrl.rb +46 -0
  150. data/tests/test_tree_ctrl.rb +138 -0
  151. data/tests/test_window.rb +12 -0
  152. data/tests/testapp_noframe.rb +1 -1
  153. metadata +31 -3
@@ -0,0 +1,25 @@
1
+ # Copyright (c) 2023 M.J.N. Corino, The Netherlands
2
+ #
3
+ # This software is released under the MIT license.
4
+
5
+ module Wx
6
+
7
+ module RTC
8
+
9
+ class RichTextParagraph
10
+
11
+ # Overload to provide Enumerator without block
12
+ wx_each_line = instance_method :each_line
13
+ define_method :each_line do |&block|
14
+ if block
15
+ wx_each_line.bind(self).call(&block)
16
+ else
17
+ ::Enumerator.new { |y| wx_each_line.bind(self).call { |ln| y << ln } }
18
+ end
19
+ end
20
+
21
+ end
22
+
23
+ end
24
+
25
+ end
@@ -6,5 +6,6 @@
6
6
 
7
7
  require_relative './events/evt_list'
8
8
  require_relative './keyword_defs'
9
+ require_relative './styled_text_ctrl'
9
10
 
10
11
  Wx::Dialog.setup_dialog_functors(Wx::STC)
@@ -0,0 +1,25 @@
1
+ # Copyright (c) 2023 M.J.N. Corino, The Netherlands
2
+ #
3
+ # This software is released under the MIT license.
4
+
5
+ module Wx
6
+
7
+ module STC
8
+
9
+ class StyledTextCtrl
10
+
11
+ # Overload to provide Enumerator without block
12
+ wx_each_line = instance_method :each_line
13
+ define_method :each_line do |&block|
14
+ if block
15
+ wx_each_line.bind(self).call(&block)
16
+ else
17
+ ::Enumerator.new { |y| wx_each_line.bind(self).call { |ln| y << ln } }
18
+ end
19
+ end
20
+
21
+ end
22
+
23
+ end
24
+
25
+ end
data/lib/wx/version.rb CHANGED
@@ -3,5 +3,5 @@
3
3
  # This software is released under the MIT license.
4
4
 
5
5
  module Wx
6
- WXRUBY_VERSION = '0.9.1'
6
+ WXRUBY_VERSION = '0.9.2'
7
7
  end
@@ -65,9 +65,6 @@ module WXRuby3
65
65
  # create a .so binary
66
66
  @extra_ldflags << '-shared'
67
67
 
68
- # This class is not available on linux ports (wxGTK/wxQT/wxX11/wxMotif)
69
- exclude_module('wxPrinterDC')
70
-
71
68
  unless @wx_path.empty?
72
69
  libdirs = @wx_libs.select {|s| s.start_with?('-L')}.collect {|s| s.sub(/^-L/,'')}
73
70
  @exec_env['LD_LIBRARY_PATH'] = "#{ENV['LD_LIBRARY_PATH']}:#{dest_dir}:#{libdirs.join(':')}"
@@ -65,7 +65,7 @@ module WXRuby3
65
65
  end
66
66
  end
67
67
  # some weird thing with this; at least sometimes '--libs all' will not output media library even if feature active
68
- if features_set?('wxUSE_MEDIACTRL')
68
+ if features_set?('USE_MEDIACTRL')
69
69
  lib_list = wx_config("--libs media").split(' ')
70
70
  until lib_list.empty?
71
71
  s = lib_list.shift
@@ -107,7 +107,7 @@ module WXRuby3
107
107
  @ld.sub!(/-o\s*\Z/, '')
108
108
 
109
109
  @extra_cflags.concat %w[-Wno-unused-function -Wno-conversion-null -Wno-sometimes-uninitialized
110
- -Wno-overloaded-virtual -Wno-deprecated-copy -Wno-format-security]
110
+ -Wno-overloaded-virtual -Wno-deprecated-copy]
111
111
  @extra_cflags << ' -Wno-deprecated-declarations' unless @no_deprecated
112
112
 
113
113
  # create a .dylib binary
@@ -35,7 +35,7 @@ module WXRuby3
35
35
  end
36
36
 
37
37
  def debug_command(*args)
38
- args.unshift(FileUtils::RUBY)
38
+ args.unshift(nix_path(FileUtils::RUBY))
39
39
  args.unshift('--args')
40
40
  args.unshift('gdb')
41
41
  args.join(' ')
@@ -136,7 +136,7 @@ module WXRuby3
136
136
  wx_libset = ::Set.new
137
137
  wx_libset.merge wx_config("--libs all").split(' ')
138
138
  # some weird thing with this; at least sometimes '--libs all' will not output media library even if feature active
139
- if features_set?('wxUSE_MEDIACTRL')
139
+ if features_set?('USE_MEDIACTRL')
140
140
  wx_libset.merge wx_config("--libs media").split(' ')
141
141
  end
142
142
  wx_libset.collect { |s| s.dup }
@@ -589,7 +589,7 @@ module WXRuby3
589
589
  private :any_feature_set?
590
590
 
591
591
  def features_set?(*featureset)
592
- featureset.all? do |feature|
592
+ featureset.flatten.all? do |feature|
593
593
  if AnyOf === feature
594
594
  any_feature_set?(*feature.features)
595
595
  else
@@ -615,9 +615,19 @@ module WXRuby3
615
615
  def _retrieve_features(wxwidgets_setup_h)
616
616
  features = {}
617
617
 
618
- File.read(wxwidgets_setup_h).scan(/^\s*#define\s+(wx\w+|__\w+__)\s+([01])/) do | define |
619
- features[$1] = $2.to_i.zero? ? false : true
620
- end if is_configured? && wxwidgets_setup_h
618
+ if is_configured? && wxwidgets_setup_h
619
+ File.read(wxwidgets_setup_h).scan(/^\s*#define\s+(wx\w+|__\w+__)\s+([01])/) do | define |
620
+ feat_val = $2.to_i.zero? ? false : true
621
+ feat_id = $1.sub(/\Awx/i, '').gsub(/\A__|__\Z/, '')
622
+ features[feat_id] = feat_val
623
+ end
624
+ # make sure correct platform defines are included as well which will not be the case always
625
+ # (for example __WXMSW__ and __WXOSX__ are not in setup.h)
626
+ features['WXMSW'] = true if features['GNUWIN32']
627
+ features['WXOSX'] = true if features['DARWIN']
628
+ # prior to wxWidgets 3.3 this feature was not set for wxGTK builds
629
+ features['WXGTK'] = true if features['LINUX'] && !features['WXGTK'] && wx_port == :wxgtk
630
+ end
621
631
 
622
632
  features
623
633
  end
@@ -105,7 +105,7 @@ module WXRuby3
105
105
  end
106
106
 
107
107
  def included_directors
108
- directors.select { |dir| !Config.instance.excluded_module?(dir.spec) }
108
+ directors.select { |dir| Package.full_docs? || !Config.instance.excluded_module?(dir.spec) }
109
109
  end
110
110
 
111
111
  def director_for_class(class_name)
@@ -248,7 +248,7 @@ module WXRuby3
248
248
  end
249
249
 
250
250
  # next initialize all modules with classes depending (bases AND mixins) on classes in any modules already
251
- # selected until there are no more modules left or none that are left depend on any selected ones
251
+ # selecteduntil there are no more modules left or none that are left depend on any selected ones
252
252
  while dir_inx = inc_dirs.find_index { |dir| !dir.spec.initialize_at_end && is_dir_with_fulfilled_deps?(dir, cls_set) }
253
253
  dir = inc_dirs[dir_inx]
254
254
  modreg = Spec.module_registry[dir.spec.module_name]
@@ -359,8 +359,7 @@ module WXRuby3
359
359
  # generate constant definitions for feature defines from setup.h
360
360
  fsrc.puts %Q{VALUE mWxSetup = rb_define_module_under(#{module_variable}, "Setup");}
361
361
  Config.instance.features.each do |feature, val|
362
- const_name = rb_wx_name(feature).gsub(/\A__|__\Z/, '')
363
- fsrc.puts %Q{rb_define_const(mWxSetup, "#{const_name}", Q#{val});}
362
+ fsrc.puts %Q{rb_define_const(mWxSetup, "#{feature}", Q#{val});}
364
363
  end
365
364
  else
366
365
  fsrc.puts %Q{#{module_variable} = rb_define_module_under(wxRuby_Core(), "#{name}");}
@@ -523,10 +522,12 @@ module WXRuby3
523
522
  # ----------------------------------------------------------------------------
524
523
 
525
524
 
526
- class Wx::EvtHandler
525
+ module Wx
527
526
 
528
- __HEREDOC
529
- fdoc.indent do
527
+ class EvtHandler
528
+
529
+ __HEREDOC
530
+ fdoc.indent(2) do
530
531
  fdoc.doc.puts "@!group #{name} Event handler methods"
531
532
  fdoc.puts
532
533
  evts_handled = ::Set.new
@@ -557,6 +558,8 @@ module WXRuby3
557
558
  fdoc.doc.puts '@!endgroup'
558
559
  end
559
560
  fdoc.puts
561
+ fdoc.puts ' end'
562
+ fdoc.puts
560
563
  fdoc.puts 'end'
561
564
  end
562
565
  end
@@ -634,6 +637,10 @@ module WXRuby3
634
637
  end
635
638
  private :generate_core_doc
636
639
 
640
+ def self.full_docs?
641
+ !!ENV['WXRUBY_FULLDOCS']
642
+ end
643
+
637
644
  def generate_docs
638
645
  # make sure all modules have been extracted from xml
639
646
  included_directors.each {|dir| dir.extract_interface(false, gendoc: true) }
@@ -327,7 +327,12 @@ module WXRuby3
327
327
  end
328
328
 
329
329
  def ignore(*names, ignore_doc: true)
330
- names.flatten.each {|n| @ignores[n] = ignore_doc}
330
+ names.flatten.each { |n| @ignores[n] = {ignore: true, ignore_doc: ignore_doc} }
331
+ self
332
+ end
333
+
334
+ def ignore_unless(feature_set, *names)
335
+ names.flatten.each { |n| @ignores[n] = {ignore: feature_set} }
331
336
  self
332
337
  end
333
338
 
@@ -47,9 +47,8 @@ module WXRuby3
47
47
  (TYPE($input) == T_STRING && RSTRING_LEN($input) == 1) );
48
48
  __CODE
49
49
  end
50
- unless Config.instance.features_set?('__WXMSW__')
51
- spec.ignore('wxAcceleratorTable::wxAcceleratorTable(const wxString &)')
52
- end
50
+ spec.ignore_unless('WXMSW',
51
+ 'wxAcceleratorTable::wxAcceleratorTable(const wxString &)')
53
52
  spec.add_swig_code <<~__HEREDOC
54
53
  %warnfilter(509) wxAcceleratorTable::wxAcceleratorTable;
55
54
  __HEREDOC
@@ -0,0 +1,47 @@
1
+ # Copyright (c) 2023 M.J.N. Corino, The Netherlands
2
+ #
3
+ # This software is released under the MIT license.
4
+
5
+ ###
6
+ # wxRuby3 wxWidgets interface director
7
+ ###
8
+
9
+ module WXRuby3
10
+
11
+ class Director
12
+
13
+ class Accessible < Director
14
+
15
+ def setup
16
+ super
17
+ spec.make_abstract 'wxAccessible'
18
+ spec.map 'wxAccessible**' => 'Wx::Accessible' do
19
+ map_in ignore: true, temp: 'wxAccessible* tmp', code: '$1 = &tmp;'
20
+ map_argout code: <<~__HEREDOC
21
+ if (tmp$argnum)
22
+ {
23
+ $result = SWIG_Ruby_AppendOutput($result, SWIG_NewPointerObj(SWIG_as_voidptr(tmp$argnum), SWIGTYPE_p_wxAccessible, 0));
24
+ }
25
+ __HEREDOC
26
+
27
+ end
28
+ spec.map 'wxString *' => 'String' do
29
+ map_in ignore: true, temp: 'wxString tmp', code: '$1 = &tmp;'
30
+ map_argout code: '$result = SWIG_Ruby_AppendOutput($result, WXSTR_TO_RSTR(tmp$argnum));'
31
+ end
32
+ spec.map 'wxAccRole *' => 'Wx::AccRole' do
33
+ map_in ignore: true, temp: 'wxAccRole tmp', code: '$1 = &tmp;'
34
+ map_argout code: <<~__HEREDOC
35
+ $result = SWIG_Ruby_AppendOutput($result, wxRuby_GetEnumValueObject("AccRole", static_cast<int>(tmp$argnum)));
36
+ __HEREDOC
37
+ end
38
+ spec.map_apply 'int * OUTPUT' => ['int * childCount', 'int * childId', 'int * toId']
39
+ spec.map_apply 'long * OUTPUT' => 'long * state'
40
+ end
41
+
42
+ end
43
+
44
+ end
45
+
46
+ end
47
+
@@ -16,6 +16,15 @@ module WXRuby3
16
16
 
17
17
  def setup
18
18
  spec.items << 'wxAppConsole' << 'wxEventFilter'
19
+ if Config.instance.wx_version >= '3.3.0'
20
+ spec.items << 'wxDarkModeSettings'
21
+ spec.ignore_unless('WXMSW', 'wxDarkModeSettings', 'wxMenuColour')
22
+ if Config.instance.features_set?('WXMSW')
23
+ spec.disown 'wxDarkModeSettings *settings'
24
+ # wxDarkModeSettings does has have virt dtor; it's just not documented
25
+ spec.suppress_warning(514, 'wxDarkModeSettings')
26
+ end
27
+ end
19
28
  spec.fold_bases('wxApp' => 'wxAppConsole', 'wxAppConsole' => 'wxEventFilter')
20
29
  spec.override_inheritance_chain('wxApp', %w[wxEvtHandler wxObject])
21
30
  spec.ignore %w{
@@ -134,339 +143,23 @@ module WXRuby3
134
143
  extern "C" void Init_wxRubyStockObjects();
135
144
  extern void wxRuby_MarkProtectedEvtHandlerProcs();
136
145
 
137
- #ifdef __WXMSW__
138
- extern "C"
139
- {
140
- WXDLLIMPEXP_BASE HINSTANCE wxGetInstance();
141
- }
142
- #endif
143
-
144
146
  static wxVector<WXRBMarkFunction> WXRuby_Mark_List;
145
147
 
146
148
  WXRUBY_EXPORT void wxRuby_AppendMarker(WXRBMarkFunction marker)
147
149
  {
148
150
  WXRuby_Mark_List.push_back(marker);
149
151
  }
150
-
151
- class wxRubyApp : public wxApp
152
- {
153
- private:
154
- static wxRubyApp* instance_;
155
152
 
156
- bool is_running_ = false;
157
- public:
158
- static wxRubyApp* GetInstance () { return instance_; }
159
-
160
- virtual ~wxRubyApp()
161
- {
162
- #ifdef __WXTRACE__
163
- std::wcout << "~wxRubyApp" << std::endl;
164
- #endif
165
- // unlink
166
- VALUE the_app = rb_const_get(#{spec.package.module_variable}, rb_intern("THE_APP"));
167
- if (the_app != Qnil)
168
- {
169
- DATA_PTR(the_app) = 0;
170
- }
171
- }
172
-
173
- // special event handler for destruction of windows which is done
174
- // automatically by wxWidgets. Tag the object as having been destroyed
175
- // by WxWidgets.
176
- void OnWindowDestroy(wxWindowDestroyEvent &event)
177
- {
178
- wxObject* wx_obj = event.GetEventObject();
179
- #ifdef __WXRB_DEBUG__
180
- if (wxRuby_TraceLevel()>0)
181
- std::wcout << "<= OnWindowDestroy [" << wx_obj << "]" << std::endl;
182
- #endif
183
- GC_SetWindowDeleted((void *)wx_obj);
184
- event.Skip();
185
- #ifdef __WXRB_DEBUG__
186
- if (wxRuby_TraceLevel()>0)
187
- std::wcout << "=> OnWindowDestroy [" << wx_obj << "]" << std::endl;
188
- #endif
189
- }
190
-
191
- bool IsRunning() const { return this->is_running_; }
192
-
193
- // When ruby's garbage collection runs, if the app is still active, it
194
- // cycles through all currently known SWIG objects and calls this
195
- // function on each to preserve still active Wx::Windows, and also
196
- // pending Wx::Events which have been queued from the Ruby side (the
197
- // only sort of events that will be in the tracking hash.
198
- static void markIterate(void* ptr, VALUE rb_obj)
199
- {
200
- // Check if it's a valid object (sometimes SWIG doesn't return what we're
201
- // expecting), a descendant of Wx::Window (but not a Dialog), and if it has not yet been
202
- // deleted by WxWidgets; if so, mark it.
203
- if ( TYPE(rb_obj) == T_DATA )
204
- {
205
- if ( rb_obj_is_kind_of(rb_obj, wxRuby_GetWindowClass()) )
206
- {
207
- rb_gc_mark(rb_obj);
208
- }
209
- else if (rb_obj_is_kind_of(rb_obj, wxRuby_GetDefaultEventClass()) )
210
- rb_gc_mark(rb_obj);
211
- }
212
- else if (TYPE(rb_obj) == T_ARRAY )
213
- {
214
- VALUE proc = rb_ary_entry(rb_obj, 0);
215
- if (rb_obj_is_kind_of(proc, rb_cProc) || rb_obj_is_kind_of(proc, rb_cMethod))
216
- {
217
- // keep the async call alive
218
- rb_gc_mark(rb_obj);
219
- }
220
- }
221
- }
222
-
223
- // Implements GC protection across wxRuby. Always called because
224
- // Wx::THE_APP is a constant so always checked in GC mark phase.
225
- static void mark_wxRubyApp(void *ptr)
226
- {
227
-
228
- #ifdef __WXRB_DEBUG__
229
- if (wxRuby_TraceLevel()>0)
230
- std::wcout << "=== Starting App GC mark phase" << std::endl;
231
- #endif
232
-
233
- // If the App has ended, the ruby object will have been unlinked from
234
- // the C++ one; this implies that all Windows have already been destroyed
235
- // so there is no point trying to mark them, and doing so may cause
236
- // errors.
237
- if ( !wxRubyApp::GetInstance() || !wxRubyApp::GetInstance()->IsRunning() )
238
- {
239
- #ifdef __WXRB_DEBUG__
240
- if (wxRuby_TraceLevel()>0)
241
- std::wcout << "=== App has ended, skipping mark phase" << std::endl;
242
- #endif
243
- return;
244
- }
245
-
246
- // Mark any active (tracked) log target
247
- wxLog* curLog = wxLog::GetActiveTarget();
248
- VALUE rb_cur_log = wxRuby_FindTracking(curLog);
249
- if (!NIL_P(rb_cur_log))
250
- {
251
- rb_gc_mark(rb_cur_log);
252
- }
253
-
254
- // Mark evt handler procs associated with live windows - see
255
- // classes/EvtHandler.i
256
- wxRuby_MarkProtectedEvtHandlerProcs();
257
-
258
- // run registered markers
259
- for (wxVector<WXRBMarkFunction>::iterator it = WXRuby_Mark_List.begin();
260
- it != WXRuby_Mark_List.end() ;++it)
261
- {
262
- (*it) ();
263
- }
264
-
265
- // To do the main marking, primarily of Windows, iterate over SWIG's
266
- // list of tracked objects
267
- wxRuby_IterateTracking(&wxRubyApp::markIterate);
268
-
269
- #ifdef __WXRB_DEBUG__
270
- if (wxRuby_TraceLevel()>0)
271
- std::wcout << "=== App GC mark phase completed" << std::endl;
272
- #endif
273
- }
274
-
275
- // This is the method run when main_loop is called in Ruby
276
- // wxEntry calls the C++ App::OnInit method
277
- int main_loop()
278
- {
279
- VALUE rb_app = SWIG_RubyInstanceFor(this);
280
- rb_define_const(#{spec.package.module_variable}, "THE_APP", rb_app);
281
- this->Connect(wxEVT_DESTROY,
282
- wxWindowDestroyEventHandler(wxRubyApp::OnWindowDestroy));
283
-
284
- #ifdef __WXRB_DEBUG__
285
- if (wxRuby_TraceLevel()>0)
286
- std::wcout << "Calling wxEntry, this=" << this << std::endl;
287
- #endif
288
-
289
- #ifdef __WXMSW__
290
- extern int wxEntry(HINSTANCE hInstance,
291
- HINSTANCE WXUNUSED(hPrevInstance),
292
- wxCmdLineArgType WXUNUSED(pCmdLine),
293
- int nCmdShow);
294
- wxEntry(wxGetInstance(),
295
- (HINSTANCE)0,
296
- (wxCmdLineArgType)"",
297
- (int)true);
298
- #else
299
- int argc = 1;
300
- const char* argv[2] = { "ruby", 0 };
301
- wxEntry(argc, const_cast<char**>(&argv[0]));
302
- #endif
303
-
304
- rb_const_remove(#{spec.package.module_variable}, rb_intern("THE_APP"));
305
-
306
- #ifdef __WXRB_DEBUG__
307
- if (wxRuby_TraceLevel()>0)
308
- std::wcout << "returned from wxEntry..." << std::endl;
309
- #endif
310
- rb_gc_start();
311
- #ifdef __WXRB_DEBUG__
312
- if (wxRuby_TraceLevel()>0)
313
- std::wcout << "survived gc" << std::endl;
314
- #endif
315
-
316
- VALUE exc = rb_iv_get(rb_app, "@exception");
317
- if (!NIL_P(exc))
318
- {
319
- rb_exc_raise(exc);
320
- }
321
- return 0;
322
- }
323
-
324
- // This method initializes the stock objects (Pens, Brushes, Fonts)
325
- // before yielding to ruby by calling the App's on_init method.
326
- // Note that as of wxWidget 2.8, the stock fonts in particular cannot
327
- // be initialized any earlier than this without crashing
328
- bool OnInit() override
329
- {
330
- #ifdef __WXRB_DEBUG__
331
- if (wxRuby_TraceLevel()>0)
332
- std::wcout << "OnInit..." << std::endl;
333
- #endif
334
- // set standard App name
335
- this->SetAppName(wxString("wxruby"));
336
- // Signal that we've started
337
- wxRubyApp::instance_ = this; // there should ALWAYS EVER be only 1 app instance running/created
338
- this->is_running_ = true;
339
- // Set up the GDI objects
340
- Init_wxRubyStockObjects();
341
- // Get the ruby representation of the App object, and call the
342
- // ruby on_init method to set up the initial window state
343
- VALUE the_app = rb_const_get(#{spec.package.module_variable}, rb_intern("THE_APP"));
344
- bool ex_caught = false;
345
- VALUE result = wxRuby_Funcall(ex_caught, the_app, rb_intern("on_ruby_init"), 0, 0);
346
-
347
- if (ex_caught)
348
- {
349
- #ifdef __WXRB_DEBUG__
350
- wxRuby_PrintException(result);
351
- #endif
352
- rb_iv_set(the_app, "@exception", result);
353
- result = Qfalse; // exit app
354
- }
355
-
356
- // If on_init return any (ruby) true value, signal to wxWidgets to
357
- // enter the main event loop by returning true, else return false
358
- // which will make wxWidgets exit.
359
- if ( result == Qfalse || result == Qnil )
360
- {
361
- wxRubyApp::instance_ = 0;
362
- this->is_running_ = false;
363
- return false;
364
- }
365
- else
366
- {
367
- return true;
368
- }
369
- }
370
-
371
- int OnExit() override
372
- {
373
- #ifdef __WXRB_DEBUG__
374
- if (wxRuby_TraceLevel()>0)
375
- std::wcout << "OnExit..." << std::endl;
376
- #endif
377
-
378
- // Get the ruby representation of the App object, and call the
379
- // ruby on_exit method (if any) for application level cleanup
380
- VALUE the_app = rb_const_get(#{spec.package.module_variable}, rb_intern("THE_APP"));
381
- ID on_exit_id = rb_intern("on_exit");
382
- if (rb_funcall(the_app, rb_intern("respond_to?"), 1, ID2SYM(on_exit_id)) == Qtrue)
383
- {
384
- bool ex_caught = false;
385
- VALUE rc = wxRuby_Funcall(ex_caught, the_app, on_exit_id, 0, 0);
386
- if (ex_caught)
387
- {
388
- #ifdef __WXRB_DEBUG__
389
- wxRuby_PrintException(rc);
390
- #endif
391
- rb_iv_set(the_app, "@exception", rc);
392
- }
393
- }
394
-
395
- // perform wxRuby cleanup
396
- _wxRuby_Cleanup();
397
-
398
- // execute base wxWidgets functionality
399
- return this->wxApp::OnExit();
400
- }
401
-
402
- // actually implemented in ruby in classes/app.rb
403
- virtual void OnAssertFailure(const wxChar *file, int line, const wxChar *func, const wxChar *cond, const wxChar *msg) override
404
- {
405
- VALUE rb_app = SWIG_RubyInstanceFor(this);
406
- if (rb_during_gc() || NIL_P(rb_app))
407
- {
408
- std::wcout << file << "(" << line << "): ASSERT " << cond
409
- << (NIL_P(rb_app) ? " fired without THE_APP in " : " fired during GC phase in ")
410
- << func << "() with message [" << msg << "]" << std::endl;
411
- }
412
- else
413
- {
414
- VALUE obj0 = Qnil ;
415
- VALUE obj1 = Qnil ;
416
- VALUE obj2 = Qnil ;
417
- VALUE obj3 = Qnil ;
418
- VALUE obj4 = Qnil ;
419
-
420
- obj0 = rb_str_new2((const char *)wxString(file).utf8_str());
421
- obj1 = INT2NUM(line);
422
- obj2 = rb_str_new2((const char *)wxString(func).utf8_str());
423
- obj3 = rb_str_new2((const char *)wxString(cond).utf8_str());
424
- obj4 = rb_str_new2((const char *)wxString(msg).utf8_str());
425
- (void)wxRuby_Funcall(rb_app, rb_intern("on_assert_failure"), 5,obj0,obj1,obj2,obj3,obj4);
426
- }
427
- }
428
-
429
- void _wxRuby_Cleanup()
430
- {
431
- #ifdef __WXRB_DEBUG__
432
- if (wxRuby_TraceLevel()>0)
433
- std::wcout << "wxRuby_Cleanup..." << std::endl;
434
- #endif
435
- // Note in a global variable that the App has ended, so that we
436
- // can skip any GC marking later
437
- wxRubyApp::instance_ = 0;
438
- this->is_running_ = false;
439
-
440
- // if a Ruby implemented logger has been installed clean that up
441
- // before we exit, otherwise let wxWidgets handle things
442
- wxLog *oldlog = wxLog::GetActiveTarget();
443
- if (wxRuby_FindTracking(oldlog) != Qnil)
444
- {
445
- oldlog = wxLog::SetActiveTarget(new wxLogStderr);
446
- }
447
- else
448
- {
449
- oldlog = 0;
450
- }
451
- SetTopWindow(0);
452
- if ( oldlog )
453
- {
454
- SWIG_RubyUnlinkObjects(oldlog);
455
- SWIG_RubyRemoveTracking(oldlog);
456
- delete oldlog;
457
- }
458
- }
459
- };
460
- wxRubyApp* wxRubyApp::instance_ = 0;
153
+ #include "wxRubyApp.h"
461
154
 
462
155
  WXRUBY_EXPORT bool wxRuby_IsAppRunning()
463
156
  {
464
- return wxRubyApp::GetInstance() != 0 && wxRubyApp::GetInstance()->IsRunning();
157
+ return wxRubyApp::GetInstance() && wxRubyApp::GetInstance()->IsRunning();
465
158
  }
466
159
 
467
160
  WXRUBY_EXPORT void wxRuby_ExitMainLoop(VALUE exception)
468
161
  {
469
- if (wxRubyApp::GetInstance() != 0 && wxRubyApp::GetInstance()->IsRunning())
162
+ if (wxRubyApp::GetInstance() && wxRubyApp::GetInstance()->IsRunning())
470
163
  {
471
164
  if (!NIL_P(exception))
472
165
  {