342
|
1 /*****************************************************************************/
|
|
2 /* gtkui.c - GTK+ UI related functions for gFTP */
|
885
|
3 /* Copyright (C) 1998-2007 Brian Masney <masneyb@gftp.org> */
|
342
|
4 /* */
|
|
5 /* This program is free software; you can redistribute it and/or modify */
|
|
6 /* it under the terms of the GNU General Public License as published by */
|
|
7 /* the Free Software Foundation; either version 2 of the License, or */
|
|
8 /* (at your option) any later version. */
|
|
9 /* */
|
|
10 /* This program 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 */
|
|
13 /* GNU General Public License for more details. */
|
|
14 /* */
|
|
15 /* You should have received a copy of the GNU General Public License */
|
|
16 /* along with this program; if not, write to the Free Software */
|
|
17 /* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111 USA */
|
|
18 /*****************************************************************************/
|
|
19
|
|
20 #include "gftp-gtk.h"
|
|
21 static const char cvsid[] = "$Id$";
|
|
22
|
|
23
|
|
24 void
|
|
25 gftpui_lookup_file_colors (gftp_file * fle, char **start_color,
|
|
26 char ** end_color)
|
|
27 {
|
|
28 *start_color = GFTPUI_COMMON_COLOR_NONE;
|
|
29 *end_color = GFTPUI_COMMON_COLOR_NONE;
|
|
30 }
|
|
31
|
|
32
|
|
33 void
|
|
34 gftpui_run_command (GtkWidget * widget, gpointer data)
|
|
35 {
|
|
36 const char *txt;
|
|
37
|
|
38 txt = gtk_entry_get_text (GTK_ENTRY (gftpui_command_widget));
|
374
|
39 gftpui_common_process_command (&window1, window1.request,
|
|
40 &window2, window2.request, txt);
|
342
|
41 gtk_entry_set_text (GTK_ENTRY (gftpui_command_widget), "");
|
|
42 }
|
|
43
|
|
44
|
|
45 void
|
521
|
46 gftpui_refresh (void *uidata, int clear_cache_entry)
|
342
|
47 {
|
|
48 gftp_window_data * wdata;
|
|
49
|
|
50 wdata = uidata;
|
879
|
51 wdata->request->refreshing = 1;
|
|
52
|
342
|
53 if (!check_status (_("Refresh"), wdata, 0, 0, 0, 1))
|
879
|
54 {
|
|
55 wdata->request->refreshing = 0;
|
|
56 return;
|
|
57 }
|
342
|
58
|
807
|
59 if (clear_cache_entry)
|
|
60 gftp_delete_cache_entry (wdata->request, NULL, 0);
|
|
61
|
342
|
62 if (check_reconnect (wdata) < 0)
|
879
|
63 {
|
|
64 wdata->request->refreshing = 0;
|
|
65 return;
|
|
66 }
|
342
|
67
|
|
68 gtk_clist_freeze (GTK_CLIST (wdata->listbox));
|
|
69 remove_files_window (wdata);
|
514
|
70
|
496
|
71 ftp_list_files (wdata);
|
342
|
72 gtk_clist_thaw (GTK_CLIST (wdata->listbox));
|
879
|
73
|
|
74 wdata->request->refreshing = 0;
|
342
|
75 }
|
|
76
|
|
77
|
356
|
78 #define _GFTPUI_GTK_USER_PW_SIZE 256
|
|
79
|
|
80 static void
|
380
|
81 _gftpui_gtk_set_username (gftp_request * request, gftp_dialog_data * ddata)
|
356
|
82 {
|
380
|
83 gftp_set_username (request, gtk_entry_get_text (GTK_ENTRY (ddata->edit)));
|
|
84 request->stopable = 0;
|
|
85 }
|
|
86
|
|
87
|
|
88 static void
|
|
89 _gftpui_gtk_set_password (gftp_request * request, gftp_dialog_data * ddata)
|
|
90 {
|
|
91 gftp_set_password (request, gtk_entry_get_text (GTK_ENTRY (ddata->edit)));
|
|
92 request->stopable = 0;
|
356
|
93 }
|
|
94
|
|
95
|
|
96 static void
|
380
|
97 _gftpui_gtk_abort (gftp_request * request, gftp_dialog_data * ddata)
|
356
|
98 {
|
380
|
99 request->stopable = 0;
|
356
|
100 }
|
|
101
|
775
|
102 void
|
|
103 gftpui_show_busy (gboolean busy)
|
|
104 {
|
|
105 GtkWidget * toplevel = gtk_widget_get_toplevel (openurl_btn);
|
|
106 GdkDisplay * display = gtk_widget_get_display (toplevel);
|
|
107
|
|
108 GdkCursor * busyCursor =
|
|
109 (busy) ? (gdk_cursor_new_for_display (display, GDK_WATCH)) : NULL;
|
|
110
|
|
111 gdk_window_set_cursor (toplevel->window, busyCursor);
|
|
112
|
|
113 if (busy)
|
|
114 gdk_cursor_unref (busyCursor);
|
|
115 }
|
356
|
116
|
380
|
117 void
|
356
|
118 gftpui_prompt_username (void *uidata, gftp_request * request)
|
|
119 {
|
|
120 MakeEditDialog (_("Enter Username"),
|
|
121 _("Please enter your username for this site"), NULL,
|
755
|
122 1, NULL, gftp_dialog_button_connect,
|
380
|
123 _gftpui_gtk_set_username, request,
|
|
124 _gftpui_gtk_abort, request);
|
356
|
125
|
380
|
126 request->stopable = 1;
|
|
127 while (request->stopable)
|
356
|
128 {
|
|
129 GDK_THREADS_LEAVE ();
|
|
130 #if GTK_MAJOR_VERSION == 1
|
|
131 g_main_iteration (TRUE);
|
|
132 #else
|
|
133 g_main_context_iteration (NULL, TRUE);
|
|
134 #endif
|
|
135 }
|
|
136 }
|
|
137
|
|
138
|
380
|
139 void
|
356
|
140 gftpui_prompt_password (void *uidata, gftp_request * request)
|
|
141 {
|
|
142 MakeEditDialog (_("Enter Password"),
|
|
143 _("Please enter your password for this site"), NULL,
|
|
144 0, NULL, gftp_dialog_button_connect,
|
380
|
145 _gftpui_gtk_set_password, request,
|
|
146 _gftpui_gtk_abort, request);
|
356
|
147
|
380
|
148 request->stopable = 1;
|
|
149 while (request->stopable)
|
356
|
150 {
|
|
151 GDK_THREADS_LEAVE ();
|
|
152 #if GTK_MAJOR_VERSION == 1
|
|
153 g_main_iteration (TRUE);
|
|
154 #else
|
|
155 g_main_context_iteration (NULL, TRUE);
|
|
156 #endif
|
|
157 }
|
|
158 }
|
|
159
|
|
160
|
355
|
161 /* The wakeup main thread functions are so that after the thread terminates
|
|
162 there won't be a delay in updating the GUI */
|
|
163 static void
|
|
164 _gftpui_wakeup_main_thread (gpointer data, gint source,
|
|
165 GdkInputCondition condition)
|
|
166 {
|
|
167 gftp_request * request;
|
|
168 char c;
|
|
169
|
|
170 request = data;
|
|
171 if (request->wakeup_main_thread[0] > 0)
|
|
172 read (request->wakeup_main_thread[0], &c, 1);
|
|
173 }
|
|
174
|
|
175
|
|
176 static gint
|
|
177 _gftpui_setup_wakeup_main_thread (gftp_request * request)
|
|
178 {
|
|
179 gint handler;
|
|
180
|
|
181 if (socketpair (AF_UNIX, SOCK_STREAM, 0, request->wakeup_main_thread) == 0)
|
|
182 {
|
|
183 handler = gdk_input_add (request->wakeup_main_thread[0],
|
|
184 GDK_INPUT_READ, _gftpui_wakeup_main_thread,
|
|
185 request);
|
|
186 }
|
|
187 else
|
|
188 {
|
|
189 request->wakeup_main_thread[0] = 0;
|
|
190 request->wakeup_main_thread[1] = 0;
|
|
191 handler = 0;
|
|
192 }
|
|
193
|
|
194 return (handler);
|
|
195 }
|
|
196
|
|
197
|
|
198 static void
|
|
199 _gftpui_teardown_wakeup_main_thread (gftp_request * request, gint handler)
|
|
200 {
|
|
201 if (request->wakeup_main_thread[0] > 0 && request->wakeup_main_thread[1] > 0)
|
|
202 {
|
|
203 gdk_input_remove (handler);
|
|
204 close (request->wakeup_main_thread[0]);
|
|
205 close (request->wakeup_main_thread[1]);
|
|
206 request->wakeup_main_thread[0] = 0;
|
|
207 request->wakeup_main_thread[1] = 0;
|
|
208 }
|
|
209 }
|
|
210
|
|
211 static void *
|
|
212 _gftpui_gtk_thread_func (void *data)
|
|
213 {
|
|
214 gftpui_gtk_thread_data * thread_data;
|
|
215 void *ret;
|
|
216
|
|
217 thread_data = data;
|
|
218 ret = thread_data->func (thread_data->cdata);
|
|
219
|
|
220 if (thread_data->cdata->request->wakeup_main_thread[1] > 0)
|
|
221 write (thread_data->cdata->request->wakeup_main_thread[1], " ", 1);
|
|
222
|
|
223 return (ret);
|
|
224 }
|
|
225
|
|
226
|
342
|
227 void *
|
|
228 gftpui_generic_thread (void * (*func) (void *), void *data)
|
|
229 {
|
355
|
230 gftpui_gtk_thread_data * thread_data;
|
342
|
231 gftpui_callback_data * cdata;
|
|
232 gftp_window_data * wdata;
|
518
|
233 gint handler;
|
342
|
234 void * ret;
|
|
235
|
|
236 cdata = data;
|
|
237 wdata = cdata->uidata;
|
|
238
|
|
239 wdata->request->stopable = 1;
|
|
240 gtk_widget_set_sensitive (stop_btn, 1);
|
355
|
241
|
|
242 thread_data = g_malloc0 (sizeof (*thread_data));
|
|
243 thread_data->func = func;
|
|
244 thread_data->cdata = cdata;
|
|
245
|
|
246 handler = _gftpui_setup_wakeup_main_thread (cdata->request);
|
|
247 pthread_create (&wdata->tid, NULL, _gftpui_gtk_thread_func, thread_data);
|
342
|
248
|
|
249 while (wdata->request->stopable)
|
|
250 {
|
|
251 GDK_THREADS_LEAVE ();
|
|
252 #if GTK_MAJOR_VERSION == 1
|
|
253 g_main_iteration (TRUE);
|
|
254 #else
|
|
255 g_main_context_iteration (NULL, TRUE);
|
|
256 #endif
|
|
257 }
|
|
258
|
355
|
259 _gftpui_teardown_wakeup_main_thread (cdata->request, handler);
|
|
260 g_free (thread_data);
|
|
261
|
342
|
262 pthread_join (wdata->tid, &ret);
|
|
263 gtk_widget_set_sensitive (stop_btn, 0);
|
|
264
|
|
265 if (!GFTP_IS_CONNECTED (wdata->request))
|
380
|
266 gftpui_disconnect (wdata);
|
342
|
267
|
|
268 return (ret);
|
|
269 }
|
|
270
|
|
271
|
|
272 int
|
|
273 gftpui_check_reconnect (gftpui_callback_data * cdata)
|
|
274 {
|
|
275 gftp_window_data * wdata;
|
|
276
|
|
277 wdata = cdata->uidata;
|
|
278 return (wdata->request->cached && wdata->request->datafd < 0 &&
|
|
279 !wdata->request->always_connected &&
|
767
|
280 !ftp_connect (wdata, wdata->request) ? -1 : 0);
|
342
|
281 }
|
|
282
|
|
283
|
|
284 void
|
|
285 gftpui_run_function_callback (gftp_window_data * wdata,
|
|
286 gftp_dialog_data * ddata)
|
|
287 {
|
|
288 gftpui_callback_data * cdata;
|
|
289 const char *edttext;
|
|
290
|
|
291 cdata = ddata->yespointer;
|
|
292 if (ddata->edit != NULL)
|
|
293 {
|
|
294 edttext = gtk_entry_get_text (GTK_ENTRY (ddata->edit));
|
|
295 if (*edttext == '\0')
|
|
296 {
|
677
|
297 ftp_log (gftp_logging_error, NULL,
|
342
|
298 _("Operation canceled...you must enter a string\n"));
|
|
299 return;
|
|
300 }
|
|
301
|
|
302 cdata->input_string = g_strdup (edttext);
|
|
303 }
|
|
304
|
633
|
305 if (ddata->checkbox != NULL)
|
|
306 cdata->toggled = GTK_TOGGLE_BUTTON (ddata->checkbox)->active;
|
|
307 else
|
|
308 cdata->toggled = 0;
|
|
309
|
679
|
310 gtk_widget_destroy (ddata->dialog);
|
|
311 ddata->dialog = NULL;
|
|
312
|
517
|
313 gftpui_common_run_callback_function (cdata);
|
342
|
314 }
|
|
315
|
|
316
|
|
317 void
|
|
318 gftpui_run_function_cancel_callback (gftp_window_data * wdata,
|
|
319 gftp_dialog_data * ddata)
|
|
320 {
|
|
321 gftpui_callback_data * cdata;
|
|
322
|
|
323 cdata = ddata->yespointer;
|
|
324 if (cdata->input_string != NULL)
|
|
325 g_free (cdata->input_string);
|
|
326 if (cdata->source_string != NULL)
|
|
327 g_free (cdata->source_string);
|
|
328 g_free (cdata);
|
|
329 }
|
|
330
|
|
331
|
|
332 void
|
|
333 gftpui_mkdir_dialog (gpointer data)
|
|
334 {
|
|
335 gftpui_callback_data * cdata;
|
|
336 gftp_window_data * wdata;
|
|
337
|
|
338 wdata = data;
|
|
339 cdata = g_malloc0 (sizeof (*cdata));
|
|
340 cdata->request = wdata->request;
|
|
341 cdata->uidata = wdata;
|
|
342 cdata->run_function = gftpui_common_run_mkdir;
|
|
343
|
|
344 if (!check_status (_("Mkdir"), wdata, gftpui_common_use_threads (wdata->request), 0, 0, wdata->request->mkdir != NULL))
|
|
345 return;
|
|
346
|
|
347 MakeEditDialog (_("Make Directory"), _("Enter name of directory to create"),
|
|
348 NULL, 1, NULL, gftp_dialog_button_create,
|
|
349 gftpui_run_function_callback, cdata,
|
|
350 gftpui_run_function_cancel_callback, cdata);
|
|
351 }
|
|
352
|
|
353
|
|
354 void
|
|
355 gftpui_rename_dialog (gpointer data)
|
|
356 {
|
|
357 gftpui_callback_data * cdata;
|
|
358 GList *templist, *filelist;
|
|
359 gftp_window_data * wdata;
|
|
360 gftp_file * curfle;
|
|
361 char *tempstr;
|
|
362 int num;
|
|
363
|
|
364 wdata = data;
|
|
365 cdata = g_malloc0 (sizeof (*cdata));
|
|
366 cdata->request = wdata->request;
|
|
367 cdata->uidata = wdata;
|
|
368 cdata->run_function = gftpui_common_run_rename;
|
|
369
|
|
370 if (!check_status (_("Rename"), wdata, gftpui_common_use_threads (wdata->request), 1, 1, wdata->request->rename != NULL))
|
|
371 return;
|
|
372
|
525
|
373 templist = gftp_gtk_get_list_selection (wdata);
|
342
|
374 num = 0;
|
|
375 filelist = wdata->files;
|
|
376 templist = get_next_selection (templist, &filelist, &num);
|
|
377 curfle = filelist->data;
|
|
378 cdata->source_string = g_strdup (curfle->file);
|
|
379
|
|
380 tempstr = g_strdup_printf (_("What would you like to rename %s to?"),
|
|
381 cdata->source_string);
|
|
382 MakeEditDialog (_("Rename"), tempstr, cdata->source_string, 1, NULL,
|
|
383 gftp_dialog_button_rename,
|
|
384 gftpui_run_function_callback, cdata,
|
|
385 gftpui_run_function_cancel_callback, cdata);
|
|
386 g_free (tempstr);
|
|
387 }
|
|
388
|
|
389
|
|
390 void
|
|
391 gftpui_site_dialog (gpointer data)
|
|
392 {
|
|
393 gftpui_callback_data * cdata;
|
|
394 gftp_window_data * wdata;
|
|
395
|
|
396 wdata = data;
|
|
397 cdata = g_malloc0 (sizeof (*cdata));
|
|
398 cdata->request = wdata->request;
|
|
399 cdata->uidata = wdata;
|
|
400 cdata->run_function = gftpui_common_run_site;
|
|
401
|
|
402 if (!check_status (_("Site"), wdata, 0, 0, 0, wdata->request->site != NULL))
|
|
403 return;
|
|
404
|
|
405 MakeEditDialog (_("Site"), _("Enter site-specific command"), NULL, 1,
|
633
|
406 _("Prepend with SITE"), gftp_dialog_button_ok,
|
342
|
407 gftpui_run_function_callback, cdata,
|
|
408 gftpui_run_function_cancel_callback, cdata);
|
|
409 }
|
|
410
|
|
411
|
|
412 int
|
|
413 gftpui_run_chdir (gpointer uidata, char *directory)
|
|
414 {
|
|
415 gftpui_callback_data * cdata;
|
|
416 gftp_window_data * wdata;
|
470
|
417 char *tempstr;
|
342
|
418 int ret;
|
|
419
|
556
|
420 wdata = uidata;
|
555
|
421 if ((tempstr = gftp_expand_path (wdata->request, directory)) == NULL)
|
470
|
422 return (FALSE);
|
|
423
|
342
|
424 cdata = g_malloc0 (sizeof (*cdata));
|
|
425 cdata->request = wdata->request;
|
|
426 cdata->uidata = wdata;
|
|
427 cdata->run_function = gftpui_common_run_chdir;
|
470
|
428 cdata->input_string = tempstr;
|
514
|
429 cdata->dont_clear_cache = 1;
|
342
|
430
|
|
431 ret = gftpui_common_run_callback_function (cdata);
|
|
432
|
470
|
433 g_free(tempstr);
|
342
|
434 g_free (cdata);
|
|
435 return (ret);
|
|
436 }
|
|
437
|
|
438
|
|
439 void
|
|
440 gftpui_chdir_dialog (gpointer data)
|
|
441 {
|
|
442 GList *templist, *filelist;
|
|
443 gftp_window_data * wdata;
|
|
444 gftp_file * curfle;
|
|
445 char *tempstr;
|
|
446 int num;
|
|
447
|
|
448 wdata = data;
|
|
449 if (!check_status (_("Chdir"), wdata, gftpui_common_use_threads (wdata->request), 1, 0,
|
|
450 wdata->request->chdir != NULL))
|
|
451 return;
|
|
452
|
525
|
453 templist = gftp_gtk_get_list_selection (wdata);
|
342
|
454 num = 0;
|
|
455 filelist = wdata->files;
|
|
456 templist = get_next_selection (templist, &filelist, &num);
|
|
457 curfle = filelist->data;
|
|
458
|
555
|
459 tempstr = gftp_build_path (wdata->request, wdata->request->directory,
|
|
460 curfle->file, NULL);
|
342
|
461 gftpui_run_chdir (wdata, tempstr);
|
|
462 g_free (tempstr);
|
|
463 }
|
|
464
|
380
|
465
|
|
466 void
|
|
467 gftpui_disconnect (void *uidata)
|
|
468 {
|
|
469 gftp_window_data * wdata;
|
|
470
|
|
471 wdata = uidata;
|
|
472 gftp_delete_cache_entry (wdata->request, NULL, 1);
|
|
473 gftp_disconnect (wdata->request);
|
|
474 remove_files_window (wdata);
|
912
|
475
|
|
476 /* Free the request structure so that all old settings are purged. */
|
924
|
477 gftp_request_destroy (wdata->request, 0);
|
912
|
478 gftp_gtk_init_request (wdata);
|
|
479
|
604
|
480 update_window_info ();
|
380
|
481 }
|
|
482
|
397
|
483
|
|
484 char *
|
|
485 gftpui_gtk_get_utf8_file_pos (gftp_file * fle)
|
|
486 {
|
830
|
487 char *pos;
|
397
|
488
|
830
|
489 if ((pos = strrchr (fle->file, '/')) != NULL)
|
397
|
490 pos++;
|
|
491 else
|
830
|
492 pos = fle->file;
|
397
|
493
|
|
494 return (pos);
|
|
495 }
|
|
496
|
460
|
497
|
|
498 static void
|
|
499 _protocol_yes_answer (gpointer answer, gftp_dialog_data * ddata)
|
|
500 {
|
|
501 *(int *) answer = 1;
|
|
502 }
|
|
503
|
|
504
|
|
505 static void
|
|
506 _protocol_no_answer (gpointer answer, gftp_dialog_data * ddata)
|
|
507 {
|
|
508 *(int *) answer = 0;
|
|
509 }
|
|
510
|
|
511
|
|
512 int
|
|
513 gftpui_protocol_ask_yes_no (gftp_request * request, char *title,
|
|
514 char *question)
|
|
515 {
|
|
516 int answer = -1;
|
|
517
|
|
518 GDK_THREADS_ENTER ();
|
|
519
|
|
520 MakeYesNoDialog (title, question, _protocol_yes_answer, &answer,
|
|
521 _protocol_no_answer, &answer);
|
|
522
|
|
523 if (gftp_protocols[request->protonum].use_threads)
|
|
524 {
|
484
|
525 /* Let the main loop in the main thread run the events */
|
460
|
526 GDK_THREADS_LEAVE ();
|
|
527
|
|
528 while (answer == -1)
|
|
529 {
|
|
530 sleep (1);
|
|
531 }
|
|
532 }
|
|
533 else
|
|
534 {
|
|
535 while (answer == -1)
|
|
536 {
|
|
537 GDK_THREADS_LEAVE ();
|
|
538 #if GTK_MAJOR_VERSION == 1
|
|
539 g_main_iteration (TRUE);
|
|
540 #else
|
|
541 g_main_context_iteration (NULL, TRUE);
|
|
542 #endif
|
|
543 }
|
|
544 }
|
|
545
|
|
546 return (answer);
|
|
547 }
|
|
548
|
484
|
549
|
|
550 static void
|
|
551 _protocol_ok_answer (char *buf, gftp_dialog_data * ddata)
|
|
552 {
|
|
553 buf[1] = ' '; /* In case this is an empty string entered */
|
|
554 strncpy (buf, gtk_entry_get_text (GTK_ENTRY (ddata->edit)), BUFSIZ);
|
|
555 }
|
|
556
|
|
557
|
|
558 static void
|
|
559 _protocol_cancel_answer (char *buf, gftp_dialog_data * ddata)
|
|
560 {
|
|
561 buf[0] = '\0';
|
|
562 buf[1] = '\0';
|
|
563 }
|
|
564
|
|
565
|
|
566 char *
|
|
567 gftpui_protocol_ask_user_input (gftp_request * request, char *title,
|
|
568 char *question, int shown)
|
|
569 {
|
|
570 char buf[BUFSIZ];
|
|
571
|
|
572 GDK_THREADS_ENTER ();
|
|
573
|
|
574 *buf = '\0';
|
|
575 *(buf + 1) = ' ';
|
|
576 MakeEditDialog (title, question, NULL, shown, NULL, gftp_dialog_button_ok,
|
|
577 _protocol_ok_answer, &buf, _protocol_cancel_answer, &buf);
|
|
578
|
|
579 if (gftp_protocols[request->protonum].use_threads)
|
|
580 {
|
|
581 /* Let the main loop in the main thread run the events */
|
|
582 GDK_THREADS_LEAVE ();
|
|
583
|
|
584 while (*buf == '\0' && *(buf + 1) == ' ')
|
|
585 {
|
|
586 sleep (1);
|
|
587 }
|
|
588 }
|
|
589 else
|
|
590 {
|
|
591 while (*buf == '\0' && *(buf + 1) == ' ')
|
|
592 {
|
|
593 GDK_THREADS_LEAVE ();
|
|
594 #if GTK_MAJOR_VERSION == 1
|
|
595 g_main_iteration (TRUE);
|
|
596 #else
|
|
597 g_main_context_iteration (NULL, TRUE);
|
|
598 #endif
|
|
599 }
|
|
600 }
|
|
601
|
|
602 if (*buf != '\0')
|
|
603 return (g_strdup (buf));
|
|
604 else
|
|
605 return (NULL);
|
|
606 }
|
|
607
|