gtk4 4.2.1 → 4.2.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,323 @@
1
+ # Copyright (C) 2024 Ruby-GNOME Project Team
2
+ #
3
+ # This library is free software; you can redistribute it and/or
4
+ # modify it under the terms of the GNU Lesser General Public
5
+ # License as published by the Free Software Foundation; either
6
+ # version 2.1 of the License, or (at your option) any later version.
7
+ #
8
+ # This library is distributed in the hope that it will be useful,
9
+ # but WITHOUT ANY WARRANTY; without even the implied warranty of
10
+ # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
11
+ # Lesser General Public License for more details.
12
+ #
13
+ # You should have received a copy of the GNU Lesser General Public
14
+ # License along with this library; if not, write to the Free Software
15
+ # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
16
+
17
+ # Original:
18
+ # https://gitlab.gnome.org/GNOME/gtk/-/blob/main/demos/gtk-demo/glarea.c
19
+ #
20
+ # See the following for license information:
21
+ # * https://gitlab.gnome.org/GNOME/gtk/-/blob/main/AUTHORS
22
+ # * https://gitlab.gnome.org/GNOME/gtk/-/blob/main/COPYING
23
+
24
+ require "opengl"
25
+
26
+ class GtkGLAreaDemo < GtkDemo
27
+ X_AXIS = 0
28
+ Y_AXIS = 1
29
+ Z_AXIS = 2
30
+ N_AXISES = 3
31
+
32
+ def initialize
33
+ super(__FILE__,
34
+ "OpenGL/OpenGL Area",
35
+ "Gtk::GLArea is a widget that allows custom drawing using OpenGL calls.",
36
+ [])
37
+ end
38
+
39
+ def run(parent)
40
+ window = Gtk::Window.new
41
+ window.display = parent.display
42
+ window.title = "OpenGL Area"
43
+ window.set_default_size(400, 600)
44
+
45
+ box = Gtk::Box.new(:vertical)
46
+ box.margin_start = 12
47
+ box.margin_end = 12
48
+ box.margin_top = 12
49
+ box.margin_bottom = 12
50
+ box.spacing = 6
51
+ window.child = box
52
+
53
+ gl_area = Gtk::GLArea.new
54
+ gl_area.hexpand = true
55
+ gl_area.vexpand = true
56
+ gl_area.set_size_request(100, 200)
57
+ box.append(gl_area)
58
+
59
+ # We need to initialize and free GL resources, so we use the
60
+ # realize and unrealize signals on the widget
61
+ gl_area.signal_connect("realize") do
62
+ realize(gl_area)
63
+ end
64
+ gl_area.signal_connect("unrealize") do
65
+ unrealize(gl_area)
66
+ end
67
+
68
+ # The main "draw" call for GtkGLArea
69
+ gl_area.signal_connect("render") do
70
+ render(gl_area)
71
+ end
72
+
73
+ controls = Gtk::Box.new(:vertical)
74
+ controls.hexpand = true
75
+ box.append(controls)
76
+
77
+ N_AXISES.times do |i|
78
+ controls.append(create_axis_slider(gl_area, i))
79
+ end
80
+
81
+ button = Gtk::Button.new(label: "Quit")
82
+ button.hexpand = true
83
+ box.append(button)
84
+
85
+ button.signal_connect("clicked") do
86
+ window.destroy
87
+ end
88
+
89
+ window.present
90
+
91
+ window
92
+ end
93
+
94
+ private
95
+ # We need to set up our state when we realize the GtkGLArea widget
96
+ def realize(gl_area)
97
+ @rotation_angles = [0.0] * N_AXISES
98
+
99
+ GL.load_lib
100
+
101
+ gl_area.make_current
102
+ api_name = gl_area.api.nick
103
+ vertex_path = "/glarea/glarea-#{api_name}.vs.glsl"
104
+ fragment_path = "/glarea/glarea-#{api_name}.fs.glsl"
105
+ init_buffers
106
+ init_shaders(vertex_path, fragment_path)
107
+ end
108
+
109
+ # The object we are drawing
110
+ VERTEX_DATA = [
111
+ 0.0, 0.5, 0.0, 1.0,
112
+ 0.5, -0.366, 0.0, 1.0,
113
+ -0.5, -0.366, 0.0, 1.0,
114
+ ]
115
+
116
+ def init_buffers
117
+ # We only use one VAO, so we always keep it bound
118
+ vao_buffer = " " * Fiddle::SIZEOF_UINT32_T
119
+ GL.GenVertexArrays(1, vao_buffer)
120
+ @vao = vao_buffer.unpack1("L")
121
+ GL.BindVertexArray(@vao)
122
+
123
+ # This is the buffer that holds the vertices
124
+ position_buffer = " " * Fiddle::SIZEOF_UINT32_T
125
+ GL.GenBuffers(1, position_buffer)
126
+ GL.GenBuffers(1, position_buffer)
127
+ @position = position_buffer.unpack1("L")
128
+ GL.BindBuffer(GL::ARRAY_BUFFER, @position)
129
+ GL.BufferData(GL::ARRAY_BUFFER,
130
+ Fiddle::SIZEOF_FLOAT * VERTEX_DATA.size,
131
+ VERTEX_DATA.pack("f*"),
132
+ GL::STATIC_DRAW)
133
+ GL.BindBuffer(GL::ARRAY_BUFFER, 0)
134
+ end
135
+
136
+ # Initialize the shaders and link them into a program
137
+ def init_shaders(vertex_path, fragment_path)
138
+ @program = nil
139
+ vertex = 0
140
+ fragment = 0
141
+
142
+ begin
143
+ source = Gio::Resources.lookup_data(vertex_path)
144
+ vertex = create_shader(GL::VERTEX_SHADER, source.to_s)
145
+ return if vertex.zero?
146
+
147
+ source = Gio::Resources.lookup_data(fragment_path)
148
+ fragment = create_shader(GL::FRAGMENT_SHADER, source.to_s)
149
+ return if fragment.zero?
150
+
151
+ @program = GL.CreateProgram
152
+ return if @program.zero?
153
+
154
+ GL.AttachShader(@program, vertex)
155
+ GL.AttachShader(@program, fragment)
156
+
157
+ GL.LinkProgram(@program)
158
+
159
+ program_ok_buffer = " " * Fiddle::SIZEOF_UINT32_T
160
+ GL.GetProgramiv(@program, GL::LINK_STATUS, program_ok_buffer)
161
+ program_ok = (program_ok_buffer.unpack1("L") == GL::TRUE)
162
+ unless program_ok
163
+ log_length_buffer = " " * Fiddle::SIZEOF_UINT32_T
164
+ GL.GetProgramiv(@program, GL::INFO_LOG_LENGTH, log_length_buffer)
165
+ log_length = log_length_buffer.unpack1("L")
166
+ log = " " * log_length
167
+ GL.GetProgramInfoLog(@program, log_length, nil, log)
168
+ $stderr.puts("Linking failure:")
169
+ $stderr.puts(log)
170
+ GL.DeleteProgram(@program)
171
+ @program = nil
172
+ return
173
+ end
174
+
175
+ # Get the location of the "mvp" uniform
176
+ @mvp_location = GL.GetUniformLocation(@program, "mvp")
177
+
178
+ GL.DetachShader(@program, vertex)
179
+ GL.DetachShader(@program, fragment)
180
+ ensure
181
+ GL.DeleteShader(vertex) if vertex.zero?
182
+ GL.DeleteShader(fragment) if fragment.zero?
183
+ end
184
+ end
185
+
186
+ # Create and compile a shader
187
+ def create_shader(type, source)
188
+ shader = GL.CreateShader(type)
189
+ GL.ShaderSource(shader, 1, [source].pack("p"), nil)
190
+ GL.CompileShader(shader)
191
+
192
+ shader_ok_buffer = " " * Fiddle::SIZEOF_UINT32_T
193
+ GL.GetShaderiv(shader, GL::COMPILE_STATUS, shader_ok_buffer)
194
+ shader_ok = (shader_ok_buffer.unpack1("L") == GL::TRUE)
195
+ unless shader_ok
196
+ log_length_buffer = " " * Fiddle::SIZEOF_UINT32_t
197
+ GL.GetShaderiv(shader, GL::INFO_LOG_LENGTH, log_length_buffer)
198
+ log_length = log_length_buffer.unpack1("L")
199
+ log = " " * log_length
200
+ GL.GetShaderInfoLog(shader, log_length, nil, buffer)
201
+ type_name = (type == GL::VERTEX_SHADER ? "vertex" : "fragment")
202
+ $stderr.puts("Compile failure in #{type_name} shader:")
203
+ $stderr.puts(log)
204
+ GL.DeleteShader(shader)
205
+ return 0
206
+ end
207
+
208
+ shader
209
+ end
210
+
211
+ # We should tear down the state when unrealizing
212
+ def unrealize(gl_area)
213
+ gl_area.make_current
214
+ GL.DeleteBuffers(1, [[@position].pack("L")].pack("p"))
215
+ GL.DeleteProgram(@program)
216
+ end
217
+
218
+ def compute_mvp(phi, theta, psi)
219
+ x = phi * (Math::PI / 180.0)
220
+ y = theta * (Math::PI / 180.0)
221
+ z = psi * (Math::PI / 180.0)
222
+ c1 = Math.cos(x)
223
+ s1 = Math.sin(x)
224
+ c2 = Math.cos(y)
225
+ s2 = Math.sin(y)
226
+ c3 = Math.cos(z)
227
+ s3 = Math.sin(z)
228
+ c3c2 = c3 * c2
229
+ s3c1 = s3 * c1
230
+ c3s2s1 = c3 * s2 * s1
231
+ s3s1 = s3 * s1
232
+ c3s2c1 = c3 * s2 * c1
233
+ s3c2 = s3 * c2
234
+ c3c1 = c3 * c1
235
+ s3s2s1 = s3 * s2 * s1
236
+ c3s1 = c3 * s1
237
+ s3s2c1 = s3 * s2 * c1
238
+ c2s1 = c2 * s1
239
+ c2c1 = c2 * c1
240
+
241
+ # apply all three rotations using the three matrices:
242
+ #
243
+ # ⎡ c3 s3 0 ⎤ ⎡ c2 0 -s2 ⎤ ⎡ 1 0 0 ⎤
244
+ # ⎢ -s3 c3 0 ⎥ ⎢ 0 1 0 ⎥ ⎢ 0 c1 s1 ⎥
245
+ # ⎣ 0 0 1 ⎦ ⎣ s2 0 c2 ⎦ ⎣ 0 -s1 c1 ⎦
246
+ mvp = [0.0] * 16
247
+ mvp[0] = c3c2; mvp[4] = s3c1 + c3s2s1; mvp[8] = s3s1 - c3s2c1; mvp[12] = 0.0;
248
+ mvp[1] = -s3c2; mvp[5] = c3c1 - s3s2s1; mvp[9] = c3s1 + s3s2c1; mvp[13] = 0.0;
249
+ mvp[2] = s2; mvp[6] = -c2s1; mvp[10] = c2c1; mvp[14] = 0.0;
250
+ mvp[3] = 0.0; mvp[7] = 0.0; mvp[11] = 0.0; mvp[15] = 1.0;
251
+ mvp
252
+ end
253
+
254
+ def draw_triangle
255
+ # Compute the model view projection matrix using the
256
+ # rotation angles specified through the GtkRange widgets
257
+ mvp = compute_mvp(@rotation_angles[X_AXIS],
258
+ @rotation_angles[Y_AXIS],
259
+ @rotation_angles[Z_AXIS])
260
+
261
+ # Use our shaders
262
+ GL.UseProgram(@program)
263
+
264
+ # Update the "mvp" matrix we use in the shader
265
+ GL.UniformMatrix4fv(@mvp_location, 1, GL::FALSE, mvp.pack("f*"))
266
+
267
+ # Use the vertices in our buffer
268
+ GL.BindBuffer(GL::ARRAY_BUFFER, @position)
269
+ GL.EnableVertexAttribArray(0)
270
+ GL.VertexAttribPointer(0, 4, GL::FLOAT, GL::FALSE, 0, 0)
271
+
272
+ # Draw the three vertices as a triangle
273
+ GL.DrawArrays(GL::TRIANGLES, 0, 3)
274
+
275
+ # We finished using the buffers and program
276
+ GL.DisableVertexAttribArray(0)
277
+ GL.BindBuffer(GL::ARRAY_BUFFER, 0)
278
+ GL.UseProgram(0)
279
+ end
280
+
281
+ def render(gl_area)
282
+ # Clear the viewport
283
+ GL.ClearColor(0.5, 0.5, 0.5, 1.0)
284
+ GL.Clear(GL::COLOR_BUFFER_BIT)
285
+
286
+ # Draw our object
287
+ draw_triangle
288
+
289
+ # Flush the contents of the pipeline
290
+ GL.Flush
291
+
292
+ true
293
+ end
294
+
295
+ def create_axis_slider(gl_area, axis)
296
+ box = Gtk::Box.new(:horizontal)
297
+ case axis
298
+ when X_AXIS
299
+ text = "X axis"
300
+ when Y_AXIS
301
+ text = "Y axis"
302
+ when Z_AXIS
303
+ text = "Z axis"
304
+ end
305
+ label = Gtk::Label.new(text)
306
+ box.append(label)
307
+
308
+ adjustment = Gtk::Adjustment.new(0.0, 0.0, 360.0, 1.0, 12.0, 0.0)
309
+ adjustment.signal_connect("value-changed") do
310
+ @rotation_angles[axis] = adjustment.value
311
+ # Update the contents of the GL drawing area
312
+ gl_area.queue_draw
313
+ end
314
+
315
+ slider = Gtk::Scale.new(:horizontal, adjustment)
316
+ box.append(slider)
317
+ slider.hexpand = true
318
+
319
+ box
320
+ end
321
+
322
+ GtkDemo.register(new)
323
+ end
@@ -0,0 +1,25 @@
1
+ <?xml version="1.0" encoding="UTF-8"?>
2
+ <interface>
3
+ <template class="GtkListItem">
4
+ <property name="focusable">0</property>
5
+ <property name="child">
6
+ <object class="GtkTreeExpander" id="expander">
7
+ <binding name="list-row">
8
+ <lookup name="item">GtkListItem</lookup>
9
+ </binding>
10
+ <property name="child">
11
+ <object class="GtkInscription">
12
+ <property name="hexpand">1</property>
13
+ <property name="nat-chars">25</property>
14
+ <property name="text-overflow">ellipsize-end</property>
15
+ <binding name="text">
16
+ <lookup name="title" type="GtkDemo">
17
+ <lookup name="item">expander</lookup>
18
+ </lookup>
19
+ </binding>
20
+ </object>
21
+ </property>
22
+ </object>
23
+ </property>
24
+ </template>
25
+ </interface>
@@ -0,0 +1,264 @@
1
+ #!/usr/bin/env ruby
2
+ #
3
+ # Copyright (C) 2024 Ruby-GNOME Project Team
4
+ #
5
+ # This library is free software; you can redistribute it and/or
6
+ # modify it under the terms of the GNU Lesser General Public
7
+ # License as published by the Free Software Foundation; either
8
+ # version 2.1 of the License, or (at your option) any later version.
9
+ #
10
+ # This library is distributed in the hope that it will be useful,
11
+ # but WITHOUT ANY WARRANTY; without even the implied warranty of
12
+ # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13
+ # Lesser General Public License for more details.
14
+ #
15
+ # You should have received a copy of the GNU Lesser General Public
16
+ # License along with this library; if not, write to the Free Software
17
+ # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
18
+
19
+ # Original:
20
+ # https://gitlab.gnome.org/GNOME/gtk/-/blob/main/demos/gtk-demo/main.c
21
+ #
22
+ # See the following for license information:
23
+ # * https://gitlab.gnome.org/GNOME/gtk/-/blob/main/AUTHORS
24
+ # * https://gitlab.gnome.org/GNOME/gtk/-/blob/main/COPYING
25
+
26
+ require "gtk4"
27
+
28
+ gresource = File.join(__dir__, "demo.gresource")
29
+ gresource_xml = "#{gresource}.xml"
30
+ need_generate = false
31
+ if File.exist?(gresource)
32
+ need_generate = File.mtime(gresource_xml) > File.mtime(gresource)
33
+ else
34
+ need_generate = true
35
+ end
36
+ if need_generate
37
+ unless system("glib-compile-resources",
38
+ "--target", gresource,
39
+ "--sourcedir", __dir__,
40
+ gresource_xml)
41
+ $stderr.puts("Failed to compile resource")
42
+ exit(false)
43
+ end
44
+ end
45
+ resource = Gio::Resource.load(gresource)
46
+ Gio::Resources.register(resource)
47
+
48
+
49
+ require_relative "demo"
50
+ Dir.glob("#{__dir__}/*.rb") do |rb|
51
+ base_name = File.basename(rb, ".rb")
52
+ next if base_name == "main"
53
+ require_relative base_name
54
+ end
55
+
56
+
57
+ def run_demo(demo, window)
58
+ demo_window = demo.run(window)
59
+ demo_window.transient_for = window
60
+ demo_window.modal = true
61
+ demo_window
62
+ end
63
+
64
+ def create_demo_model(window)
65
+ store = Gio::ListStore.new(GtkDemo)
66
+ GtkDemo.all.each do |demo|
67
+ store.append(demo)
68
+ end
69
+ store
70
+ end
71
+
72
+ def load_file(demo, notebook, info_view)
73
+ notebook.n_pages.downto(1) do |i|
74
+ notebook.remove_page(i)
75
+ end
76
+
77
+ info_buffer = Gtk::TextBuffer.new
78
+ info_buffer.create_tag("title",
79
+ size: 18 * Pango::SCALE,
80
+ pixels_below_lines: 10)
81
+ info_buffer.begin_irreversible_action do
82
+ info_buffer_iter = info_buffer.get_iter_at(offset: 0)
83
+ info_buffer.insert(info_buffer_iter, demo.title, tags: ["title"])
84
+ info_buffer.insert(info_buffer_iter, "\n")
85
+ info_buffer.insert(info_buffer_iter, demo.description)
86
+ end
87
+ info_view.buffer = info_buffer
88
+ end
89
+
90
+ def select_item(selection, window, notebook, info_view)
91
+ row = selection.selected_item
92
+ notebook.sensitive = !!row
93
+ if row
94
+ demo = row.item
95
+ load_file(demo, notebook, info_view)
96
+ run_action = window.lookup_action("run")
97
+ run_action.enabled = true
98
+ window.title = demo.title
99
+ else
100
+ window.title = "No match"
101
+ end
102
+ end
103
+
104
+
105
+ app = Gtk::Application.new("com.github.ruby-gnome.gtk4.Demo",
106
+ [:non_unique, :handles_command_line])
107
+
108
+ about_action = Gio::SimpleAction.new("about")
109
+ about_action.signal_connect("activate") do
110
+ dialog = Gtk::AboutDialog.new
111
+ dialog.modal = true
112
+ dialog.transient_for = app.active_window
113
+ dialog.destroy_with_parent = true
114
+ dialog.program_name = "Ruby/GTK4 Demo"
115
+ dialog.version = GLib::BINDING_VERSION.join(".")
116
+ dialog.copyright = "(C) 2024-#{Time.now.year} Ruby-GNOME Project Team"
117
+ dialog.license_type = Gtk::License::LGPL_2_1
118
+ dialog.website = "https://ruby-gnome.github.io/"
119
+ dialog.comments = "Program to demonstrate Ruby/GTK4 widgets"
120
+ dialog.authors = ["Ruby-GNOME Project Team"]
121
+ dialog.logo_icon_name = "org.gtk.Demo4"
122
+ dialog.title = "About Ruby/GTK4 Demo"
123
+ dialog.system_information = <<-INFO.chomp
124
+ OS\t#{GLib.get_os_info(:name)} #{GLib.get_os_info(:version)}
125
+
126
+ System Libraries
127
+ \tGLib\t#{GLib::Version::STRING}
128
+ \tPango\t#{Pango::Version::STRING}
129
+ \tGTK\t#{Gtk::Version::STRING}
130
+
131
+ A link can appear here: <https://ruby-gnome.github.io/>
132
+ INFO
133
+ dialog.present
134
+ end
135
+ app.add_action(about_action)
136
+
137
+ quit_action = Gio::SimpleAction.new("quit")
138
+ quit_action.signal_connect("activate") do
139
+ app.windows.each do |window|
140
+ window.destroy
141
+ end
142
+ end
143
+ app.add_action(quit_action)
144
+
145
+ inspector_action = Gio::SimpleAction.new("inspector")
146
+ inspector_action.signal_connect("activate") do
147
+ Gtk::Window.interactive_debugging = true
148
+ end
149
+ app.add_action(inspector_action)
150
+ app.set_accels_for_action("app.about", ["F1"])
151
+ app.set_accels_for_action("app.quit", ["<Control>q"])
152
+
153
+ app.add_main_option("version",
154
+ 0,
155
+ GLib::OptionFlags::NONE,
156
+ GLib::OptionArg::NONE,
157
+ "Show program version")
158
+ app.add_main_option("run",
159
+ 0,
160
+ GLib::OptionFlags::NONE,
161
+ GLib::OptionArg::STRING,
162
+ "Run an example",
163
+ "EXAMPLE")
164
+ app.add_main_option("list",
165
+ 0,
166
+ GLib::OptionFlags::NONE,
167
+ GLib::OptionArg::NONE,
168
+ "List examples")
169
+ app.add_main_option("auto-quit",
170
+ 0,
171
+ GLib::OptionFlags::NONE,
172
+ GLib::OptionArg::NONE,
173
+ "Quit after a delay")
174
+ run = nil
175
+ list = false
176
+ auto_quit = false
177
+ app.signal_connect("command-line") do |_app, command_line|
178
+ run = command_line.options_dict.lookup("run", "s")
179
+ list = command_line.options_dict.lookup("list", "b")
180
+ auto_quit = command_line.options_dict.lookup("auto-quit", "b")
181
+ app.activate
182
+ 0
183
+ end
184
+
185
+ app.signal_connect("handle-local-options") do |_app, options|
186
+ if options.lookup("version", "b")
187
+ puts("ruby-gtk4-demo: #{GLib::BINDING_VERSION.join(".")}")
188
+ 0
189
+ else
190
+ -1
191
+ end
192
+ end
193
+
194
+ app.signal_connect("activate") do |_app|
195
+ builder = Gtk::Builder.new(resource: "/ui/main.ui")
196
+ window = builder["window"]
197
+ app.add_window(window)
198
+
199
+ run_action = Gio::SimpleAction.new("run")
200
+ run_action.signal_connect("activate") do
201
+ row = selection.selected_item
202
+ demo = row.item
203
+ run_demo(demo, window)
204
+ end
205
+ window.add_action(run_action)
206
+
207
+ notebook = builder["notebook"]
208
+
209
+ info_view = builder["info-textview"]
210
+ source_view = builder["source-textview"]
211
+ list_view = builder["listview"]
212
+ list_view.signal_connect("activate") do |_, position|
213
+ model = list_view.model
214
+ row = model.get_item(position)
215
+ run_demo(row.item, window)
216
+ end
217
+ search_bar = builder["searchbar"]
218
+ search_bar.signal_connect("notify::search-mode-enabled") do
219
+ unless search_bar.search_mode?
220
+ bar.child.text = ""
221
+ end
222
+ end
223
+
224
+ list_model = create_demo_model(window)
225
+ tree_model = Gtk::TreeListModel.new(list_model, false, true) do |item|
226
+ item.children_model
227
+ end
228
+ filter_model = Gtk::FilterListModel.new(tree_model, nil)
229
+ filter = Gtk::CustomFilter.new do
230
+ # TODO
231
+ true
232
+ end
233
+ filter_model.filter = filter
234
+
235
+ search_entry = builder["search-entry"]
236
+
237
+ selection = Gtk::SingleSelection.new(filter_model)
238
+ selection.signal_connect("notify::selected-item") do
239
+ select_item(selection, window, notebook, info_view)
240
+ end
241
+ list_view.model = selection
242
+ select_item(selection, window, notebook, info_view)
243
+
244
+ target_demo = nil
245
+ if run
246
+ GtkDemo.each do |demo|
247
+ if demo.name == run
248
+ target_demo = demo
249
+ break
250
+ end
251
+ end
252
+ end
253
+ if target_demo
254
+ demo_window = run_demo(target_demo, window)
255
+ # TODO: This doesn't work
256
+ demo_window.signal_connect("destroy") do
257
+ app.quit
258
+ end
259
+ else
260
+ window.present
261
+ end
262
+ end
263
+
264
+ app.run([$0] + ARGV)