blob: d1495ef213e997b7800ff4e2b7b6908d5ccc9721 [file] [log] [blame]
Pierre Ossmand463b572011-05-16 12:04:43 +00001/* Copyright 2011 Pierre Ossman <ossman@cendio.se> for Cendio AB
2 *
3 * This is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License as published by
5 * the Free Software Foundation; either version 2 of the License, or
6 * (at your option) any later version.
7 *
8 * This software 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
11 * GNU General Public License for more details.
12 *
13 * You should have received a copy of the GNU General Public License
14 * along with this software; if not, write to the Free Software
15 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
16 * USA.
17 */
18
Peter Åstrandc359f362011-08-23 12:04:46 +000019#ifdef HAVE_CONFIG_H
20#include <config.h>
21#endif
22
Pierre Ossman61fd4862011-05-16 12:46:51 +000023#include <stdlib.h>
24
Pierre Ossmand463b572011-05-16 12:04:43 +000025#include <list>
26
Pierre Ossmand463b572011-05-16 12:04:43 +000027#include <rdr/types.h>
28#include <rfb/encodings.h>
29
30#ifdef HAVE_GNUTLS
31#include <rfb/Security.h>
32#include <rfb/SecurityClient.h>
33#include <rfb/CSecurityTLS.h>
34#endif
35
36#include "OptionsDialog.h"
37#include "fltk_layout.h"
38#include "i18n.h"
Martin Koegler498ef462011-09-04 07:04:43 +000039#include "menukey.h"
Pierre Ossmand463b572011-05-16 12:04:43 +000040#include "parameters.h"
41
DRCb65bb932011-06-24 03:17:00 +000042#include <FL/Fl_Tabs.H>
43#include <FL/Fl_Button.H>
Pierre Ossman769963f2014-01-20 14:43:52 +010044#include <FL/Fl_Check_Button.H>
DRCb65bb932011-06-24 03:17:00 +000045#include <FL/Fl_Return_Button.H>
Pierre Ossman769963f2014-01-20 14:43:52 +010046#include <FL/Fl_Round_Button.H>
47#include <FL/Fl_Int_Input.H>
48#include <FL/Fl_Choice.H>
DRCb65bb932011-06-24 03:17:00 +000049
Pierre Ossmand463b572011-05-16 12:04:43 +000050using namespace std;
51using namespace rdr;
52using namespace rfb;
53
Pierre Ossman0c41e1d2011-05-17 09:36:04 +000054std::map<OptionsCallback*, void*> OptionsDialog::callbacks;
55
Pierre Ossmand463b572011-05-16 12:04:43 +000056OptionsDialog::OptionsDialog()
57 : Fl_Window(450, 450, _("VNC Viewer: Connection Options"))
58{
59 int x, y;
60 Fl_Button *button;
61
62 Fl_Tabs *tabs = new Fl_Tabs(OUTER_MARGIN, OUTER_MARGIN,
63 w() - OUTER_MARGIN*2,
64 h() - OUTER_MARGIN*2 - INNER_MARGIN - BUTTON_HEIGHT);
65
66 {
67 int tx, ty, tw, th;
68
69 tabs->client_area(tx, ty, tw, th, TABS_HEIGHT);
70
71 createCompressionPage(tx, ty, tw, th);
72 createSecurityPage(tx, ty, tw, th);
73 createInputPage(tx, ty, tw, th);
Pierre Ossman1c2189b2012-07-05 09:23:03 +000074 createScreenPage(tx, ty, tw, th);
Pierre Ossmand463b572011-05-16 12:04:43 +000075 createMiscPage(tx, ty, tw, th);
76 }
77
78 tabs->end();
79
80 x = w() - BUTTON_WIDTH * 2 - INNER_MARGIN - OUTER_MARGIN;
81 y = h() - BUTTON_HEIGHT - OUTER_MARGIN;
82
83 button = new Fl_Button(x, y, BUTTON_WIDTH, BUTTON_HEIGHT, _("Cancel"));
84 button->callback(this->handleCancel, this);
85
86 x += BUTTON_WIDTH + INNER_MARGIN;
87
88 button = new Fl_Return_Button(x, y, BUTTON_WIDTH, BUTTON_HEIGHT, _("OK"));
89 button->callback(this->handleOK, this);
90
91 callback(this->handleCancel, this);
92
93 set_modal();
94}
95
96
97OptionsDialog::~OptionsDialog()
98{
99}
100
101
102void OptionsDialog::showDialog(void)
103{
104 static OptionsDialog *dialog = NULL;
105
106 if (!dialog)
107 dialog = new OptionsDialog();
108
109 if (dialog->shown())
110 return;
111
112 dialog->show();
113}
114
115
Pierre Ossman0c41e1d2011-05-17 09:36:04 +0000116void OptionsDialog::addCallback(OptionsCallback *cb, void *data)
117{
118 callbacks[cb] = data;
119}
120
121
122void OptionsDialog::removeCallback(OptionsCallback *cb)
123{
124 callbacks.erase(cb);
125}
126
127
Pierre Ossmand463b572011-05-16 12:04:43 +0000128void OptionsDialog::show(void)
129{
Pierre Ossman33a233b2011-09-30 12:21:58 +0000130 /* show() gets called for raise events as well */
131 if (!shown())
132 loadOptions();
Pierre Ossmand463b572011-05-16 12:04:43 +0000133
134 Fl_Window::show();
135}
136
137
138void OptionsDialog::loadOptions(void)
139{
140 /* Compression */
141 autoselectCheckbox->value(autoSelect);
142
143 int encNum = encodingNum(preferredEncoding);
144
145 switch (encNum) {
146 case encodingTight:
147 tightButton->setonly();
148 break;
149 case encodingZRLE:
150 zrleButton->setonly();
151 break;
152 case encodingHextile:
153 hextileButton->setonly();
154 break;
155 case encodingRaw:
156 rawButton->setonly();
157 break;
158 }
159
160 if (fullColour)
161 fullcolorCheckbox->setonly();
162 else {
163 switch (lowColourLevel) {
164 case 0:
Pierre Ossmancf836f22011-07-14 14:34:09 +0000165 verylowcolorCheckbox->setonly();
Pierre Ossmand463b572011-05-16 12:04:43 +0000166 break;
167 case 1:
168 lowcolorCheckbox->setonly();
169 break;
170 case 2:
Pierre Ossmancf836f22011-07-14 14:34:09 +0000171 mediumcolorCheckbox->setonly();
Pierre Ossmand463b572011-05-16 12:04:43 +0000172 break;
173 }
174 }
175
176 char digit[2] = "0";
177
178 compressionCheckbox->value(customCompressLevel);
179 jpegCheckbox->value(!noJpeg);
180 digit[0] = '0' + compressLevel;
181 compressionInput->value(digit);
182 digit[0] = '0' + qualityLevel;
183 jpegInput->value(digit);
184
185 handleAutoselect(autoselectCheckbox, this);
186 handleCompression(compressionCheckbox, this);
187 handleJpeg(jpegCheckbox, this);
188
189#ifdef HAVE_GNUTLS
190 /* Security */
191 Security security(SecurityClient::secTypes);
192
193 list<U8> secTypes;
194 list<U8>::iterator iter;
195
196 list<U32> secTypesExt;
197 list<U32>::iterator iterExt;
198
Pierre Ossmand463b572011-05-16 12:04:43 +0000199 encNoneCheckbox->value(false);
200 encTLSCheckbox->value(false);
201 encX509Checkbox->value(false);
202
203 authNoneCheckbox->value(false);
204 authVncCheckbox->value(false);
205 authPlainCheckbox->value(false);
206
207 secTypes = security.GetEnabledSecTypes();
208 for (iter = secTypes.begin(); iter != secTypes.end(); ++iter) {
209 switch (*iter) {
Pierre Ossmand463b572011-05-16 12:04:43 +0000210 case secTypeNone:
211 encNoneCheckbox->value(true);
212 authNoneCheckbox->value(true);
213 break;
214 case secTypeVncAuth:
215 encNoneCheckbox->value(true);
216 authVncCheckbox->value(true);
217 break;
218 }
219 }
220
221 secTypesExt = security.GetEnabledExtSecTypes();
222 for (iterExt = secTypesExt.begin(); iterExt != secTypesExt.end(); ++iterExt) {
223 switch (*iterExt) {
224 case secTypePlain:
225 encNoneCheckbox->value(true);
226 authPlainCheckbox->value(true);
227 break;
228 case secTypeTLSNone:
229 encTLSCheckbox->value(true);
230 authNoneCheckbox->value(true);
231 break;
232 case secTypeTLSVnc:
233 encTLSCheckbox->value(true);
234 authVncCheckbox->value(true);
235 break;
236 case secTypeTLSPlain:
237 encTLSCheckbox->value(true);
238 authPlainCheckbox->value(true);
239 break;
240 case secTypeX509None:
241 encX509Checkbox->value(true);
242 authNoneCheckbox->value(true);
243 break;
244 case secTypeX509Vnc:
245 encX509Checkbox->value(true);
246 authVncCheckbox->value(true);
247 break;
248 case secTypeX509Plain:
249 encX509Checkbox->value(true);
250 authPlainCheckbox->value(true);
251 break;
252 }
253 }
254
Pierre Ossman3d2a84b2014-09-17 16:45:35 +0200255 caInput->value(CSecurityTLS::X509CA);
256 crlInput->value(CSecurityTLS::X509CRL);
Pierre Ossmand463b572011-05-16 12:04:43 +0000257
Pierre Ossmand463b572011-05-16 12:04:43 +0000258 handleX509(encX509Checkbox, this);
259#endif
260
261 /* Input */
Pierre Ossman4e7271e2011-05-24 12:47:12 +0000262 const char *menuKeyBuf;
263
Pierre Ossmand463b572011-05-16 12:04:43 +0000264 viewOnlyCheckbox->value(viewOnly);
265 acceptClipboardCheckbox->value(acceptClipboard);
266 sendClipboardCheckbox->value(sendClipboard);
267 sendPrimaryCheckbox->value(sendPrimary);
Pierre Ossman407a5c32011-05-26 14:48:29 +0000268 systemKeysCheckbox->value(fullscreenSystemKeys);
Pierre Ossmand463b572011-05-16 12:04:43 +0000269
Pierre Ossman4e7271e2011-05-24 12:47:12 +0000270 menuKeyChoice->value(0);
271
272 menuKeyBuf = menuKey;
Martin Koegler498ef462011-09-04 07:04:43 +0000273 for (int i = 0; i < getMenuKeySymbolCount(); i++)
274 if (!strcmp(getMenuKeySymbols()[i].name, menuKeyBuf))
275 menuKeyChoice->value(i + 1);
Pierre Ossman4e7271e2011-05-24 12:47:12 +0000276
Pierre Ossman1c2189b2012-07-05 09:23:03 +0000277 /* Screen */
Pierre Ossman9ff733a2012-07-05 11:01:23 +0000278 int width, height;
279
280 if (sscanf(desktopSize.getValueStr(), "%dx%d", &width, &height) != 2) {
281 desktopSizeCheckbox->value(false);
282 desktopWidthInput->value("1024");
283 desktopHeightInput->value("768");
284 } else {
285 char buf[32];
286 desktopSizeCheckbox->value(true);
287 snprintf(buf, sizeof(buf), "%d", width);
288 desktopWidthInput->value(buf);
289 snprintf(buf, sizeof(buf), "%d", height);
290 desktopHeightInput->value(buf);
291 }
Pierre Ossman99197012012-07-05 11:06:18 +0000292 remoteResizeCheckbox->value(remoteResize);
Pierre Ossman2441e822012-07-10 11:11:23 +0000293#ifdef HAVE_FLTK_FULLSCREEN
Pierre Ossman1c2189b2012-07-05 09:23:03 +0000294 fullScreenCheckbox->value(fullScreen);
Pierre Ossmanaae38912012-07-13 11:22:55 +0000295#ifdef HAVE_FLTK_FULLSCREEN_SCREENS
296 fullScreenAllMonitorsCheckbox->value(fullScreenAllMonitors);
297#endif // HAVE_FLTK_FULLSCREEN_SCREENS
Pierre Ossman2441e822012-07-10 11:11:23 +0000298#endif // HAVE_FLTK_FULLSCREEN
Pierre Ossman1c2189b2012-07-05 09:23:03 +0000299
Pierre Ossman9ff733a2012-07-05 11:01:23 +0000300 handleDesktopSize(desktopSizeCheckbox, this);
301
Pierre Ossmand463b572011-05-16 12:04:43 +0000302 /* Misc. */
303 sharedCheckbox->value(shared);
Pierre Ossmand463b572011-05-16 12:04:43 +0000304 dotCursorCheckbox->value(dotWhenNoCursor);
305}
306
307
308void OptionsDialog::storeOptions(void)
309{
Pierre Ossman61fd4862011-05-16 12:46:51 +0000310 /* Compression */
311 autoSelect.setParam(autoselectCheckbox->value());
312
313 if (tightButton->value())
314 preferredEncoding.setParam(encodingName(encodingTight));
315 else if (zrleButton->value())
316 preferredEncoding.setParam(encodingName(encodingZRLE));
317 else if (hextileButton->value())
318 preferredEncoding.setParam(encodingName(encodingHextile));
319 else if (rawButton->value())
320 preferredEncoding.setParam(encodingName(encodingRaw));
321
322 fullColour.setParam(fullcolorCheckbox->value());
Pierre Ossmancf836f22011-07-14 14:34:09 +0000323 if (verylowcolorCheckbox->value())
Pierre Ossman61fd4862011-05-16 12:46:51 +0000324 lowColourLevel.setParam(0);
325 else if (lowcolorCheckbox->value())
326 lowColourLevel.setParam(1);
Pierre Ossmancf836f22011-07-14 14:34:09 +0000327 else if (mediumcolorCheckbox->value())
Pierre Ossman61fd4862011-05-16 12:46:51 +0000328 lowColourLevel.setParam(2);
329
330 customCompressLevel.setParam(compressionCheckbox->value());
331 noJpeg.setParam(!jpegCheckbox->value());
332 compressLevel.setParam(atoi(compressionInput->value()));
333 qualityLevel.setParam(atoi(jpegInput->value()));
334
335#ifdef HAVE_GNUTLS
336 /* Security */
337 Security security;
338
339 /* Process security types which don't use encryption */
340 if (encNoneCheckbox->value()) {
341 if (authNoneCheckbox->value())
342 security.EnableSecType(secTypeNone);
343 if (authVncCheckbox->value())
344 security.EnableSecType(secTypeVncAuth);
Pierre Ossmanc1764bd2011-09-30 12:26:25 +0000345 if (authPlainCheckbox->value())
346 security.EnableSecType(secTypePlain);
Pierre Ossman61fd4862011-05-16 12:46:51 +0000347 }
348
Pierre Ossmanc1764bd2011-09-30 12:26:25 +0000349 /* Process security types which use TLS encryption */
350 if (encTLSCheckbox->value()) {
351 if (authNoneCheckbox->value())
352 security.EnableSecType(secTypeTLSNone);
353 if (authVncCheckbox->value())
354 security.EnableSecType(secTypeTLSVnc);
355 if (authPlainCheckbox->value())
356 security.EnableSecType(secTypeTLSPlain);
357 }
Pierre Ossman61fd4862011-05-16 12:46:51 +0000358
Pierre Ossmanc1764bd2011-09-30 12:26:25 +0000359 /* Process security types which use X509 encryption */
360 if (encX509Checkbox->value()) {
361 if (authNoneCheckbox->value())
362 security.EnableSecType(secTypeX509None);
363 if (authVncCheckbox->value())
364 security.EnableSecType(secTypeX509Vnc);
365 if (authPlainCheckbox->value())
366 security.EnableSecType(secTypeX509Plain);
Pierre Ossman61fd4862011-05-16 12:46:51 +0000367 }
368
Pierre Ossman5535fe62011-09-30 12:11:52 +0000369 SecurityClient::secTypes.setParam(security.ToString());
370
Pierre Ossman3d2a84b2014-09-17 16:45:35 +0200371 CSecurityTLS::X509CA.setParam(caInput->value());
372 CSecurityTLS::X509CRL.setParam(crlInput->value());
Pierre Ossman61fd4862011-05-16 12:46:51 +0000373#endif
374
375 /* Input */
376 viewOnly.setParam(viewOnlyCheckbox->value());
377 acceptClipboard.setParam(acceptClipboardCheckbox->value());
378 sendClipboard.setParam(sendClipboardCheckbox->value());
379 sendPrimary.setParam(sendPrimaryCheckbox->value());
Pierre Ossman407a5c32011-05-26 14:48:29 +0000380 fullscreenSystemKeys.setParam(systemKeysCheckbox->value());
Pierre Ossman61fd4862011-05-16 12:46:51 +0000381
Pierre Ossman4e7271e2011-05-24 12:47:12 +0000382 if (menuKeyChoice->value() == 0)
383 menuKey.setParam("");
384 else {
Martin Koegler498ef462011-09-04 07:04:43 +0000385 menuKey.setParam(menuKeyChoice->text());
Pierre Ossman4e7271e2011-05-24 12:47:12 +0000386 }
387
Pierre Ossman1c2189b2012-07-05 09:23:03 +0000388 /* Screen */
Pierre Ossman9ff733a2012-07-05 11:01:23 +0000389 int width, height;
390
391 if (desktopSizeCheckbox->value() &&
392 (sscanf(desktopWidthInput->value(), "%d", &width) == 1) &&
393 (sscanf(desktopHeightInput->value(), "%d", &height) == 1)) {
394 char buf[64];
395 snprintf(buf, sizeof(buf), "%dx%d", width, height);
396 desktopSize.setParam(buf);
397 } else {
398 desktopSize.setParam("");
399 }
Pierre Ossman99197012012-07-05 11:06:18 +0000400 remoteResize.setParam(remoteResizeCheckbox->value());
Pierre Ossman2441e822012-07-10 11:11:23 +0000401#ifdef HAVE_FLTK_FULLSCREEN
Pierre Ossman1c2189b2012-07-05 09:23:03 +0000402 fullScreen.setParam(fullScreenCheckbox->value());
Pierre Ossmanaae38912012-07-13 11:22:55 +0000403#ifdef HAVE_FLTK_FULLSCREEN_SCREENS
404 fullScreenAllMonitors.setParam(fullScreenAllMonitorsCheckbox->value());
405#endif // HAVE_FLTK_FULLSCREEN_SCREENS
Pierre Ossman2441e822012-07-10 11:11:23 +0000406#endif // HAVE_FLTK_FULLSCREEN
Pierre Ossman1c2189b2012-07-05 09:23:03 +0000407
Pierre Ossman61fd4862011-05-16 12:46:51 +0000408 /* Misc. */
409 shared.setParam(sharedCheckbox->value());
Pierre Ossman61fd4862011-05-16 12:46:51 +0000410 dotWhenNoCursor.setParam(dotCursorCheckbox->value());
Pierre Ossman0c41e1d2011-05-17 09:36:04 +0000411
412 std::map<OptionsCallback*, void*>::const_iterator iter;
413
414 for (iter = callbacks.begin();iter != callbacks.end();++iter)
415 iter->first(iter->second);
Pierre Ossmand463b572011-05-16 12:04:43 +0000416}
417
418
419void OptionsDialog::createCompressionPage(int tx, int ty, int tw, int th)
420{
421 Fl_Group *group = new Fl_Group(tx, ty, tw, th, _("Compression"));
422
423 int orig_tx, orig_ty;
424 int half_width, full_width;
Pierre Ossmaneb955322015-03-03 16:10:53 +0100425 int height;
Pierre Ossmand463b572011-05-16 12:04:43 +0000426
427 tx += OUTER_MARGIN;
428 ty += OUTER_MARGIN;
429
430 full_width = tw - OUTER_MARGIN * 2;
431 half_width = (full_width - INNER_MARGIN) / 2;
432
433 /* AutoSelect checkbox */
434 autoselectCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
435 CHECK_MIN_WIDTH,
436 CHECK_HEIGHT,
437 _("Auto select")));
438 autoselectCheckbox->callback(handleAutoselect, this);
439 ty += CHECK_HEIGHT + INNER_MARGIN;
440
441 /* Two columns */
442 orig_tx = tx;
443 orig_ty = ty;
444
445 /* VNC encoding box */
446 ty += GROUP_LABEL_OFFSET;
447 height = GROUP_MARGIN * 2 + TIGHT_MARGIN * 3 + RADIO_HEIGHT * 4;
448 encodingGroup = new Fl_Group(tx, ty, half_width, height,
449 _("Preferred encoding"));
450 encodingGroup->box(FL_ENGRAVED_BOX);
451 encodingGroup->align(FL_ALIGN_LEFT | FL_ALIGN_TOP);
452
453 {
454 tx += GROUP_MARGIN;
455 ty += GROUP_MARGIN;
456
Pierre Ossmand463b572011-05-16 12:04:43 +0000457 tightButton = new Fl_Round_Button(LBLRIGHT(tx, ty,
458 RADIO_MIN_WIDTH,
459 RADIO_HEIGHT,
Peter Åstrand4c446002011-08-15 12:33:06 +0000460 "Tight"));
Pierre Ossmand463b572011-05-16 12:04:43 +0000461 tightButton->type(FL_RADIO_BUTTON);
462 ty += RADIO_HEIGHT + TIGHT_MARGIN;
463
464 zrleButton = new Fl_Round_Button(LBLRIGHT(tx, ty,
465 RADIO_MIN_WIDTH,
466 RADIO_HEIGHT,
Peter Åstrand4c446002011-08-15 12:33:06 +0000467 "ZRLE"));
Pierre Ossmand463b572011-05-16 12:04:43 +0000468 zrleButton->type(FL_RADIO_BUTTON);
469 ty += RADIO_HEIGHT + TIGHT_MARGIN;
470
471 hextileButton = new Fl_Round_Button(LBLRIGHT(tx, ty,
472 RADIO_MIN_WIDTH,
473 RADIO_HEIGHT,
Peter Åstrand4c446002011-08-15 12:33:06 +0000474 "Hextile"));
Pierre Ossmand463b572011-05-16 12:04:43 +0000475 hextileButton->type(FL_RADIO_BUTTON);
476 ty += RADIO_HEIGHT + TIGHT_MARGIN;
477
478 rawButton = new Fl_Round_Button(LBLRIGHT(tx, ty,
479 RADIO_MIN_WIDTH,
480 RADIO_HEIGHT,
Peter Åstrand4c446002011-08-15 12:33:06 +0000481 "Raw"));
Pierre Ossmand463b572011-05-16 12:04:43 +0000482 rawButton->type(FL_RADIO_BUTTON);
483 ty += RADIO_HEIGHT + TIGHT_MARGIN;
484 }
485
486 ty += GROUP_MARGIN - TIGHT_MARGIN;
487
488 encodingGroup->end();
489
490 /* Second column */
491 tx = orig_tx + half_width + INNER_MARGIN;
492 ty = orig_ty;
493
494 /* Color box */
495 ty += GROUP_LABEL_OFFSET;
496 height = GROUP_MARGIN * 2 + TIGHT_MARGIN * 3 + RADIO_HEIGHT * 4;
497 colorlevelGroup = new Fl_Group(tx, ty, half_width, height, _("Color level"));
498 colorlevelGroup->box(FL_ENGRAVED_BOX);
499 colorlevelGroup->align(FL_ALIGN_LEFT | FL_ALIGN_TOP);
500
501 {
502 tx += GROUP_MARGIN;
503 ty += GROUP_MARGIN;
504
Pierre Ossmand463b572011-05-16 12:04:43 +0000505 fullcolorCheckbox = new Fl_Round_Button(LBLRIGHT(tx, ty,
506 RADIO_MIN_WIDTH,
507 RADIO_HEIGHT,
508 _("Full (all available colors)")));
509 fullcolorCheckbox->type(FL_RADIO_BUTTON);
510 ty += RADIO_HEIGHT + TIGHT_MARGIN;
511
512 mediumcolorCheckbox = new Fl_Round_Button(LBLRIGHT(tx, ty,
513 RADIO_MIN_WIDTH,
514 RADIO_HEIGHT,
515 _("Medium (256 colors)")));
516 mediumcolorCheckbox->type(FL_RADIO_BUTTON);
517 ty += RADIO_HEIGHT + TIGHT_MARGIN;
518
519 lowcolorCheckbox = new Fl_Round_Button(LBLRIGHT(tx, ty,
520 RADIO_MIN_WIDTH,
521 RADIO_HEIGHT,
522 _("Low (64 colors)")));
523 lowcolorCheckbox->type(FL_RADIO_BUTTON);
524 ty += RADIO_HEIGHT + TIGHT_MARGIN;
525
526 verylowcolorCheckbox = new Fl_Round_Button(LBLRIGHT(tx, ty,
527 RADIO_MIN_WIDTH,
528 RADIO_HEIGHT,
529 _("Very low (8 colors)")));
530 verylowcolorCheckbox->type(FL_RADIO_BUTTON);
531 ty += RADIO_HEIGHT + TIGHT_MARGIN;
532 }
533
534 ty += GROUP_MARGIN - TIGHT_MARGIN;
535
536 colorlevelGroup->end();
537
538 /* Back to normal */
539 tx = orig_tx;
540 ty += INNER_MARGIN;
541
542 /* Checkboxes */
543 compressionCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
544 CHECK_MIN_WIDTH,
545 CHECK_HEIGHT,
546 _("Custom compression level:")));
547 compressionCheckbox->callback(handleCompression, this);
548 ty += CHECK_HEIGHT + TIGHT_MARGIN;
549
550 compressionInput = new Fl_Int_Input(tx + INDENT, ty,
551 INPUT_HEIGHT, INPUT_HEIGHT,
DRCba7bc512011-08-17 02:30:34 +0000552 _("level (1=fast, 6=best [4-6 are rarely useful])"));
Pierre Ossmand463b572011-05-16 12:04:43 +0000553 compressionInput->align(FL_ALIGN_RIGHT);
554 ty += INPUT_HEIGHT + INNER_MARGIN;
555
556 jpegCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
557 CHECK_MIN_WIDTH,
558 CHECK_HEIGHT,
559 _("Allow JPEG compression:")));
560 jpegCheckbox->callback(handleJpeg, this);
561 ty += CHECK_HEIGHT + TIGHT_MARGIN;
562
563 jpegInput = new Fl_Int_Input(tx + INDENT, ty,
564 INPUT_HEIGHT, INPUT_HEIGHT,
DRC41036ed2011-08-23 20:36:50 +0000565 _("quality (0=poor, 9=best)"));
Pierre Ossmand463b572011-05-16 12:04:43 +0000566 jpegInput->align(FL_ALIGN_RIGHT);
567 ty += INPUT_HEIGHT + INNER_MARGIN;
568
569 group->end();
570}
571
572
573void OptionsDialog::createSecurityPage(int tx, int ty, int tw, int th)
574{
575#ifdef HAVE_GNUTLS
576 Fl_Group *group = new Fl_Group(tx, ty, tw, th, _("Security"));
577
578 int orig_tx;
579 int width, height;
580
581 tx += OUTER_MARGIN;
582 ty += OUTER_MARGIN;
583
584 width = tw - OUTER_MARGIN * 2;
585
586 orig_tx = tx;
587
Pierre Ossmand463b572011-05-16 12:04:43 +0000588 /* Encryption */
589 ty += GROUP_LABEL_OFFSET;
590 height = GROUP_MARGIN * 2 + TIGHT_MARGIN * 4 + CHECK_HEIGHT * 3 + (INPUT_LABEL_OFFSET + INPUT_HEIGHT) * 2;
591 encryptionGroup = new Fl_Group(tx, ty, width, height, _("Encryption"));
592 encryptionGroup->box(FL_ENGRAVED_BOX);
593 encryptionGroup->align(FL_ALIGN_LEFT | FL_ALIGN_TOP);
594
595 {
596 tx += GROUP_MARGIN;
597 ty += GROUP_MARGIN;
598
599 encNoneCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
600 CHECK_MIN_WIDTH,
601 CHECK_HEIGHT,
602 _("None")));
603 ty += CHECK_HEIGHT + TIGHT_MARGIN;
604
605 encTLSCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
606 CHECK_MIN_WIDTH,
607 CHECK_HEIGHT,
608 _("TLS with anonymous certificates")));
609 ty += CHECK_HEIGHT + TIGHT_MARGIN;
610
611 encX509Checkbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
612 CHECK_MIN_WIDTH,
613 CHECK_HEIGHT,
614 _("TLS with X509 certificates")));
615 encX509Checkbox->callback(handleX509, this);
616 ty += CHECK_HEIGHT + TIGHT_MARGIN;
617
618 ty += INPUT_LABEL_OFFSET;
619 caInput = new Fl_Input(tx + INDENT, ty,
620 width - GROUP_MARGIN*2 - INDENT, INPUT_HEIGHT,
621 _("Path to X509 CA certificate"));
622 caInput->align(FL_ALIGN_LEFT | FL_ALIGN_TOP);
623 ty += INPUT_HEIGHT + TIGHT_MARGIN;
624
625 ty += INPUT_LABEL_OFFSET;
626 crlInput = new Fl_Input(tx + INDENT, ty,
627 width - GROUP_MARGIN*2 - INDENT, INPUT_HEIGHT,
628 _("Path to X509 CRL file"));
629 crlInput->align(FL_ALIGN_LEFT | FL_ALIGN_TOP);
630 ty += INPUT_HEIGHT + TIGHT_MARGIN;
631 }
632
633 ty += GROUP_MARGIN - TIGHT_MARGIN;
634
635 encryptionGroup->end();
636
637 /* Back to normal */
638 tx = orig_tx;
639 ty += INNER_MARGIN;
640
641 /* Authentication */
Pierre Ossmand463b572011-05-16 12:04:43 +0000642 ty += GROUP_LABEL_OFFSET;
643 height = GROUP_MARGIN * 2 + TIGHT_MARGIN * 2 + CHECK_HEIGHT * 3;
644 authenticationGroup = new Fl_Group(tx, ty, width, height, _("Authentication"));
645 authenticationGroup->box(FL_ENGRAVED_BOX);
646 authenticationGroup->align(FL_ALIGN_LEFT | FL_ALIGN_TOP);
647
648 {
649 tx += GROUP_MARGIN;
650 ty += GROUP_MARGIN;
651
652 authNoneCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
653 CHECK_MIN_WIDTH,
654 CHECK_HEIGHT,
655 _("None")));
656 ty += CHECK_HEIGHT + TIGHT_MARGIN;
657
658 authVncCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
659 CHECK_MIN_WIDTH,
660 CHECK_HEIGHT,
661 _("Standard VNC (insecure without encryption)")));
662 ty += CHECK_HEIGHT + TIGHT_MARGIN;
663
664 authPlainCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
665 CHECK_MIN_WIDTH,
666 CHECK_HEIGHT,
667 _("Username and password (insecure without encryption)")));
668 ty += CHECK_HEIGHT + TIGHT_MARGIN;
669 }
670
671 ty += GROUP_MARGIN - TIGHT_MARGIN;
672
673 authenticationGroup->end();
674
675 /* Back to normal */
676 tx = orig_tx;
677 ty += INNER_MARGIN;
678
679 group->end();
680#endif
681}
682
683
684void OptionsDialog::createInputPage(int tx, int ty, int tw, int th)
685{
686 Fl_Group *group = new Fl_Group(tx, ty, tw, th, _("Input"));
687
688 tx += OUTER_MARGIN;
689 ty += OUTER_MARGIN;
690
691 viewOnlyCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
692 CHECK_MIN_WIDTH,
693 CHECK_HEIGHT,
694 _("View only (ignore mouse and keyboard)")));
695 ty += CHECK_HEIGHT + TIGHT_MARGIN;
696
697 acceptClipboardCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
698 CHECK_MIN_WIDTH,
699 CHECK_HEIGHT,
700 _("Accept clipboard from server")));
701 ty += CHECK_HEIGHT + TIGHT_MARGIN;
702
703 sendClipboardCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
704 CHECK_MIN_WIDTH,
705 CHECK_HEIGHT,
706 _("Send clipboard to server")));
707 ty += CHECK_HEIGHT + TIGHT_MARGIN;
708
709 sendPrimaryCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
710 CHECK_MIN_WIDTH,
711 CHECK_HEIGHT,
712 _("Send primary selection and cut buffer as clipboard")));
713 ty += CHECK_HEIGHT + TIGHT_MARGIN;
714
Pierre Ossman407a5c32011-05-26 14:48:29 +0000715 systemKeysCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
716 CHECK_MIN_WIDTH,
717 CHECK_HEIGHT,
718 _("Pass system keys directly to server (full screen)")));
719 ty += CHECK_HEIGHT + TIGHT_MARGIN;
720
Pierre Ossman4e7271e2011-05-24 12:47:12 +0000721 menuKeyChoice = new Fl_Choice(LBLLEFT(tx, ty, 150, CHOICE_HEIGHT, _("Menu key")));
722
Pierre Ossman245c8022015-02-25 11:27:49 +0100723 fltk_menu_add(menuKeyChoice, _("None"), 0, NULL, (void*)0, FL_MENU_DIVIDER);
Martin Koegler498ef462011-09-04 07:04:43 +0000724 for (int i = 0; i < getMenuKeySymbolCount(); i++)
Pierre Ossman245c8022015-02-25 11:27:49 +0100725 fltk_menu_add(menuKeyChoice, getMenuKeySymbols()[i].name, 0, NULL, 0, 0);
Pierre Ossman4e7271e2011-05-24 12:47:12 +0000726
727 ty += CHOICE_HEIGHT + TIGHT_MARGIN;
728
Pierre Ossmand463b572011-05-16 12:04:43 +0000729 group->end();
730}
731
732
Pierre Ossman1c2189b2012-07-05 09:23:03 +0000733void OptionsDialog::createScreenPage(int tx, int ty, int tw, int th)
734{
Pierre Ossman9ff733a2012-07-05 11:01:23 +0000735 int x;
736
Pierre Ossman1c2189b2012-07-05 09:23:03 +0000737 Fl_Group *group = new Fl_Group(tx, ty, tw, th, _("Screen"));
738
739 tx += OUTER_MARGIN;
740 ty += OUTER_MARGIN;
741
Pierre Ossman9ff733a2012-07-05 11:01:23 +0000742 desktopSizeCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
743 CHECK_MIN_WIDTH,
744 CHECK_HEIGHT,
745 _("Resize remote session on connect")));
746 desktopSizeCheckbox->callback(handleDesktopSize, this);
747 ty += CHECK_HEIGHT + TIGHT_MARGIN;
748
749 desktopWidthInput = new Fl_Int_Input(tx + INDENT, ty, 50, INPUT_HEIGHT);
750 x = desktopWidthInput->x() + desktopWidthInput->w() + \
751 gui_str_len("x") + 3 * 2;
752 desktopHeightInput = new Fl_Int_Input(x, ty, 50, INPUT_HEIGHT, "x");
753 ty += INPUT_HEIGHT + TIGHT_MARGIN;
754
Pierre Ossman99197012012-07-05 11:06:18 +0000755 remoteResizeCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
756 CHECK_MIN_WIDTH,
757 CHECK_HEIGHT,
758 _("Resize remote session to the local window")));
759 ty += CHECK_HEIGHT + TIGHT_MARGIN;
760
Pierre Ossman2441e822012-07-10 11:11:23 +0000761#ifdef HAVE_FLTK_FULLSCREEN
762
Pierre Ossman1c2189b2012-07-05 09:23:03 +0000763 fullScreenCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
764 CHECK_MIN_WIDTH,
765 CHECK_HEIGHT,
766 _("Full-screen mode")));
767 ty += CHECK_HEIGHT + TIGHT_MARGIN;
768
Pierre Ossmanaae38912012-07-13 11:22:55 +0000769#ifdef HAVE_FLTK_FULLSCREEN_SCREENS
770
771 fullScreenAllMonitorsCheckbox = new Fl_Check_Button(LBLRIGHT(tx + INDENT, ty,
772 CHECK_MIN_WIDTH,
773 CHECK_HEIGHT,
774 _("Enable full-screen mode over all monitors")));
775 ty += CHECK_HEIGHT + TIGHT_MARGIN;
776
777#endif // HAVE_FLTK_FULLSCREEN_SCREENS
778
Pierre Ossman2441e822012-07-10 11:11:23 +0000779#endif // HAVE_FLTK_FULLSCREEN
780
Pierre Ossman1c2189b2012-07-05 09:23:03 +0000781 group->end();
782}
783
784
Pierre Ossmand463b572011-05-16 12:04:43 +0000785void OptionsDialog::createMiscPage(int tx, int ty, int tw, int th)
786{
787 Fl_Group *group = new Fl_Group(tx, ty, tw, th, _("Misc."));
788
789 tx += OUTER_MARGIN;
790 ty += OUTER_MARGIN;
791
792 sharedCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
793 CHECK_MIN_WIDTH,
794 CHECK_HEIGHT,
795 _("Shared (don't disconnect other viewers)")));
796 ty += CHECK_HEIGHT + TIGHT_MARGIN;
797
Pierre Ossmand463b572011-05-16 12:04:43 +0000798 dotCursorCheckbox = new Fl_Check_Button(LBLRIGHT(tx, ty,
799 CHECK_MIN_WIDTH,
800 CHECK_HEIGHT,
801 _("Show dot when no cursor")));
802 ty += CHECK_HEIGHT + TIGHT_MARGIN;
803
804 group->end();
805}
806
807
808void OptionsDialog::handleAutoselect(Fl_Widget *widget, void *data)
809{
810 OptionsDialog *dialog = (OptionsDialog*)data;
811
812 if (dialog->autoselectCheckbox->value()) {
813 dialog->encodingGroup->deactivate();
814 dialog->colorlevelGroup->deactivate();
815 } else {
816 dialog->encodingGroup->activate();
817 dialog->colorlevelGroup->activate();
818 }
819
820 // JPEG setting is also affected by autoselection
821 dialog->handleJpeg(dialog->jpegCheckbox, dialog);
822}
823
824
825void OptionsDialog::handleCompression(Fl_Widget *widget, void *data)
826{
827 OptionsDialog *dialog = (OptionsDialog*)data;
828
829 if (dialog->compressionCheckbox->value())
830 dialog->compressionInput->activate();
831 else
832 dialog->compressionInput->deactivate();
833}
834
835
836void OptionsDialog::handleJpeg(Fl_Widget *widget, void *data)
837{
838 OptionsDialog *dialog = (OptionsDialog*)data;
839
840 if (dialog->jpegCheckbox->value() &&
841 !dialog->autoselectCheckbox->value())
842 dialog->jpegInput->activate();
843 else
844 dialog->jpegInput->deactivate();
845}
846
847
Pierre Ossmand463b572011-05-16 12:04:43 +0000848void OptionsDialog::handleX509(Fl_Widget *widget, void *data)
849{
850 OptionsDialog *dialog = (OptionsDialog*)data;
851
852 if (dialog->encX509Checkbox->value()) {
853 dialog->caInput->activate();
854 dialog->crlInput->activate();
855 } else {
856 dialog->caInput->deactivate();
857 dialog->crlInput->deactivate();
858 }
859}
860
861
Pierre Ossman9ff733a2012-07-05 11:01:23 +0000862void OptionsDialog::handleDesktopSize(Fl_Widget *widget, void *data)
863{
864 OptionsDialog *dialog = (OptionsDialog*)data;
865
866 if (dialog->desktopSizeCheckbox->value()) {
867 dialog->desktopWidthInput->activate();
868 dialog->desktopHeightInput->activate();
869 } else {
870 dialog->desktopWidthInput->deactivate();
871 dialog->desktopHeightInput->deactivate();
872 }
873}
874
Pierre Ossmand463b572011-05-16 12:04:43 +0000875void OptionsDialog::handleCancel(Fl_Widget *widget, void *data)
876{
877 OptionsDialog *dialog = (OptionsDialog*)data;
878
879 dialog->hide();
880}
881
882
883void OptionsDialog::handleOK(Fl_Widget *widget, void *data)
884{
885 OptionsDialog *dialog = (OptionsDialog*)data;
886
887 dialog->hide();
888
889 dialog->storeOptions();
890}