15 static char **events = NULL;
16 static int nevents = 0, negsize = 0;
19 static int sesslist_has_focus;
21 static struct prefslist cipherlist;
23 void force_normal(HWND hwnd)
25 static int recurse = 0;
33 wp.length = sizeof(wp);
34 if (GetWindowPlacement(hwnd, &wp) && wp.showCmd == SW_SHOWMAXIMIZED) {
35 wp.showCmd = SW_SHOWNORMAL;
36 SetWindowPlacement(hwnd, &wp);
41 static void MyGetDlgItemInt(HWND hwnd, int id, int *result)
45 n = GetDlgItemInt(hwnd, id, &ok, FALSE);
50 static void MyGetDlgItemFlt(HWND hwnd, int id, int *result, int scale)
54 ok = GetDlgItemText(hwnd, id, text, sizeof(text) - 1);
56 *result = (int) (scale * atof(text));
59 static void MySetDlgItemFlt(HWND hwnd, int id, double value)
62 sprintf(text, "%g", value);
63 SetDlgItemText(hwnd, id, text);
66 static int CALLBACK LogProc(HWND hwnd, UINT msg,
67 WPARAM wParam, LPARAM lParam)
74 static int tabs[4] = { 78, 108 };
75 SendDlgItemMessage(hwnd, IDN_LIST, LB_SETTABSTOPS, 2,
78 for (i = 0; i < nevents; i++)
79 SendDlgItemMessage(hwnd, IDN_LIST, LB_ADDSTRING,
80 0, (LPARAM) events[i]);
83 switch (LOWORD(wParam)) {
87 SetActiveWindow(GetParent(hwnd));
91 if (HIWORD(wParam) == BN_CLICKED ||
92 HIWORD(wParam) == BN_DOUBLECLICKED) {
95 selcount = SendDlgItemMessage(hwnd, IDN_LIST,
96 LB_GETSELCOUNT, 0, 0);
97 if (selcount == 0) { /* don't even try to copy zero items */
102 selitems = smalloc(selcount * sizeof(int));
104 int count = SendDlgItemMessage(hwnd, IDN_LIST,
111 static unsigned char sel_nl[] = SEL_NL;
113 if (count == 0) { /* can't copy zero stuff */
119 for (i = 0; i < count; i++)
121 strlen(events[selitems[i]]) + sizeof(sel_nl);
123 clipdata = smalloc(size);
126 for (i = 0; i < count; i++) {
127 char *q = events[selitems[i]];
128 int qlen = strlen(q);
131 memcpy(p, sel_nl, sizeof(sel_nl));
134 write_aclip(clipdata, size, TRUE);
139 for (i = 0; i < nevents; i++)
140 SendDlgItemMessage(hwnd, IDN_LIST, LB_SETSEL,
149 SetActiveWindow(GetParent(hwnd));
156 static int CALLBACK LicenceProc(HWND hwnd, UINT msg,
157 WPARAM wParam, LPARAM lParam)
163 switch (LOWORD(wParam)) {
176 static int CALLBACK AboutProc(HWND hwnd, UINT msg,
177 WPARAM wParam, LPARAM lParam)
181 SetDlgItemText(hwnd, IDA_VERSION, ver);
184 switch (LOWORD(wParam)) {
187 EndDialog(hwnd, TRUE);
190 EnableWindow(hwnd, 0);
191 DialogBox(hinst, MAKEINTRESOURCE(IDD_LICENCEBOX),
193 EnableWindow(hwnd, 1);
194 SetActiveWindow(hwnd);
198 /* Load web browser */
199 ShellExecute(hwnd, "open",
200 "http://www.chiark.greenend.org.uk/~sgtatham/putty/",
201 0, 0, SW_SHOWDEFAULT);
206 EndDialog(hwnd, TRUE);
213 * Null dialog procedure.
215 static int CALLBACK NullDlgProc(HWND hwnd, UINT msg,
216 WPARAM wParam, LPARAM lParam)
221 static char savedsession[2048];
224 IDC_ABOUT, IDCX_TVSTATIC, IDCX_TREEVIEW, controlstartvalue,
369 appearancepanelstart,
370 IDC_TITLE_APPEARANCE,
390 connectionpanelstart,
391 IDC_TITLE_CONNECTION,
498 translationpanelstart,
499 IDC_TITLE_TRANSLATION,
500 IDC_BOX_TRANSLATION1,
501 IDC_BOX_TRANSLATION2,
537 static const char *const colours[] = {
538 "Default Foreground", "Default Bold Foreground",
539 "Default Background", "Default Bold Background",
540 "Cursor Text", "Cursor Colour",
541 "ANSI Black", "ANSI Black Bold",
542 "ANSI Red", "ANSI Red Bold",
543 "ANSI Green", "ANSI Green Bold",
544 "ANSI Yellow", "ANSI Yellow Bold",
545 "ANSI Blue", "ANSI Blue Bold",
546 "ANSI Magenta", "ANSI Magenta Bold",
547 "ANSI Cyan", "ANSI Cyan Bold",
548 "ANSI White", "ANSI White Bold"
550 static const int permcolour[] = {
551 TRUE, FALSE, TRUE, FALSE, TRUE, TRUE,
552 TRUE, FALSE, TRUE, FALSE, TRUE, FALSE, TRUE, FALSE,
553 TRUE, FALSE, TRUE, FALSE, TRUE, FALSE, TRUE, FALSE
556 static void fmtfont(char *buf)
558 sprintf(buf, "Font: %s, ", cfg.font);
560 strcat(buf, "bold, ");
561 if (cfg.fontheight == 0)
562 strcat(buf, "default height");
564 sprintf(buf + strlen(buf), "%d-point",
565 (cfg.fontheight < 0 ? -cfg.fontheight : cfg.fontheight));
568 /* 2nd arg: NZ => don't redraw session list (use when loading
570 static void init_dlg_ctrls(HWND hwnd, int keepsess)
573 char fontstatic[256];
575 SetDlgItemText(hwnd, IDC_HOST, cfg.host);
576 SetDlgItemText(hwnd, IDC_SESSEDIT, savedsession);
579 n = SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_GETCOUNT, 0, 0);
580 for (i = n; i-- > 0;)
581 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_DELETESTRING, i, 0);
582 for (i = 0; i < nsessions; i++)
583 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_ADDSTRING,
584 0, (LPARAM) (sessions[i]));
586 SetDlgItemInt(hwnd, IDC_PORT, cfg.port, FALSE);
587 CheckRadioButton(hwnd, IDC_PROTRAW, IDC_PROTSSH,
588 cfg.protocol == PROT_SSH ? IDC_PROTSSH :
589 cfg.protocol == PROT_TELNET ? IDC_PROTTELNET :
591 PROT_RLOGIN ? IDC_PROTRLOGIN : IDC_PROTRAW);
592 SetDlgItemInt(hwnd, IDC_PINGEDIT, cfg.ping_interval, FALSE);
594 CheckRadioButton(hwnd, IDC_DEL008, IDC_DEL127,
595 cfg.bksp_is_delete ? IDC_DEL127 : IDC_DEL008);
596 CheckRadioButton(hwnd, IDC_HOMETILDE, IDC_HOMERXVT,
597 cfg.rxvt_homeend ? IDC_HOMERXVT : IDC_HOMETILDE);
598 CheckRadioButton(hwnd, IDC_FUNCTILDE, IDC_FUNCSCO,
599 cfg.funky_type == 0 ? IDC_FUNCTILDE :
600 cfg.funky_type == 1 ? IDC_FUNCLINUX :
601 cfg.funky_type == 2 ? IDC_FUNCXTERM :
602 cfg.funky_type == 3 ? IDC_FUNCVT400 :
603 cfg.funky_type == 4 ? IDC_FUNCVT100P :
604 cfg.funky_type == 5 ? IDC_FUNCSCO : IDC_FUNCTILDE);
605 CheckDlgButton(hwnd, IDC_NOAPPLICC, cfg.no_applic_c);
606 CheckDlgButton(hwnd, IDC_NOAPPLICK, cfg.no_applic_k);
607 CheckRadioButton(hwnd, IDC_CURNORMAL, IDC_CURAPPLIC,
608 cfg.app_cursor ? IDC_CURAPPLIC : IDC_CURNORMAL);
609 CheckRadioButton(hwnd, IDC_KPNORMAL, IDC_KPNH,
610 cfg.nethack_keypad ? IDC_KPNH :
611 cfg.app_keypad ? IDC_KPAPPLIC : IDC_KPNORMAL);
612 CheckDlgButton(hwnd, IDC_ALTF4, cfg.alt_f4);
613 CheckDlgButton(hwnd, IDC_ALTSPACE, cfg.alt_space);
614 CheckDlgButton(hwnd, IDC_ALTONLY, cfg.alt_only);
615 CheckDlgButton(hwnd, IDC_COMPOSEKEY, cfg.compose_key);
616 CheckDlgButton(hwnd, IDC_CTRLALTKEYS, cfg.ctrlaltkeys);
617 CheckDlgButton(hwnd, IDC_TELNETKEY, cfg.telnet_keyboard);
618 CheckRadioButton(hwnd, IDC_ECHOBACKEND, IDC_ECHONO,
619 cfg.localecho == LD_BACKEND ? IDC_ECHOBACKEND :
620 cfg.localecho == LD_YES ? IDC_ECHOYES : IDC_ECHONO);
621 CheckRadioButton(hwnd, IDC_EDITBACKEND, IDC_EDITNO,
622 cfg.localedit == LD_BACKEND ? IDC_EDITBACKEND :
623 cfg.localedit == LD_YES ? IDC_EDITYES : IDC_EDITNO);
624 SetDlgItemText(hwnd, IDC_ANSWEREDIT, cfg.answerback);
625 CheckDlgButton(hwnd, IDC_ALWAYSONTOP, cfg.alwaysontop);
626 CheckDlgButton(hwnd, IDC_SCROLLKEY, cfg.scroll_on_key);
627 CheckDlgButton(hwnd, IDC_SCROLLDISP, cfg.scroll_on_disp);
629 CheckDlgButton(hwnd, IDC_WRAPMODE, cfg.wrap_mode);
630 CheckDlgButton(hwnd, IDC_DECOM, cfg.dec_om);
631 CheckDlgButton(hwnd, IDC_LFHASCR, cfg.lfhascr);
632 SetDlgItemInt(hwnd, IDC_ROWSEDIT, cfg.height, FALSE);
633 SetDlgItemInt(hwnd, IDC_COLSEDIT, cfg.width, FALSE);
634 SetDlgItemInt(hwnd, IDC_SAVEEDIT, cfg.savelines, FALSE);
636 SetDlgItemText(hwnd, IDC_FONTSTATIC, fontstatic);
637 CheckRadioButton(hwnd, IDC_BELL_DISABLED, IDC_BELL_VISUAL,
638 cfg.beep == BELL_DISABLED ? IDC_BELL_DISABLED :
639 cfg.beep == BELL_DEFAULT ? IDC_BELL_DEFAULT :
640 cfg.beep == BELL_WAVEFILE ? IDC_BELL_WAVEFILE :
642 BELL_VISUAL ? IDC_BELL_VISUAL : IDC_BELL_DEFAULT);
643 CheckRadioButton(hwnd, IDC_B_IND_DISABLED, IDC_B_IND_STEADY,
645 B_IND_DISABLED ? IDC_B_IND_DISABLED : cfg.beep_ind ==
646 B_IND_FLASH ? IDC_B_IND_FLASH : cfg.beep_ind ==
647 B_IND_STEADY ? IDC_B_IND_STEADY : IDC_B_IND_DISABLED);
648 SetDlgItemText(hwnd, IDC_BELL_WAVEEDIT, cfg.bell_wavefile);
649 CheckDlgButton(hwnd, IDC_BELLOVL, cfg.bellovl);
650 SetDlgItemInt(hwnd, IDC_BELLOVLN, cfg.bellovl_n, FALSE);
651 MySetDlgItemFlt(hwnd, IDC_BELLOVLT, cfg.bellovl_t / 1000.0);
652 MySetDlgItemFlt(hwnd, IDC_BELLOVLS, cfg.bellovl_s / 1000.0);
654 CheckDlgButton(hwnd, IDC_BCE, cfg.bce);
655 CheckDlgButton(hwnd, IDC_BLINKTEXT, cfg.blinktext);
657 SetDlgItemText(hwnd, IDC_WINEDIT, cfg.wintitle);
658 CheckDlgButton(hwnd, IDC_WINNAME, cfg.win_name_always);
659 CheckDlgButton(hwnd, IDC_HIDEMOUSE, cfg.hide_mouseptr);
660 CheckDlgButton(hwnd, IDC_SUNKENEDGE, cfg.sunken_edge);
661 CheckRadioButton(hwnd, IDC_CURBLOCK, IDC_CURVERT,
662 cfg.cursor_type == 0 ? IDC_CURBLOCK :
663 cfg.cursor_type == 1 ? IDC_CURUNDER : IDC_CURVERT);
664 CheckDlgButton(hwnd, IDC_BLINKCUR, cfg.blink_cur);
665 CheckDlgButton(hwnd, IDC_SCROLLBAR, cfg.scrollbar);
666 CheckDlgButton(hwnd, IDC_LOCKSIZE, cfg.locksize);
667 CheckRadioButton(hwnd, IDC_COEALWAYS, IDC_COENORMAL,
668 cfg.close_on_exit == COE_NORMAL ? IDC_COENORMAL :
670 COE_NEVER ? IDC_COENEVER : IDC_COEALWAYS);
671 CheckDlgButton(hwnd, IDC_CLOSEWARN, cfg.warn_on_close);
673 SetDlgItemText(hwnd, IDC_TTEDIT, cfg.termtype);
674 SetDlgItemText(hwnd, IDC_TSEDIT, cfg.termspeed);
675 SetDlgItemText(hwnd, IDC_R_TSEDIT, cfg.termspeed);
676 SetDlgItemText(hwnd, IDC_RLLUSEREDIT, cfg.localusername);
677 SetDlgItemText(hwnd, IDC_LOGEDIT, cfg.username);
678 SetDlgItemText(hwnd, IDC_LGFEDIT, cfg.logfilename);
679 CheckRadioButton(hwnd, IDC_LSTATOFF, IDC_LSTATRAW,
680 cfg.logtype == 0 ? IDC_LSTATOFF :
681 cfg.logtype == 1 ? IDC_LSTATASCII : IDC_LSTATRAW);
682 CheckRadioButton(hwnd, IDC_LSTATXOVR, IDC_LSTATXASK,
683 cfg.logxfovr == LGXF_OVR ? IDC_LSTATXOVR :
684 cfg.logxfovr == LGXF_ASK ? IDC_LSTATXASK :
687 char *p = cfg.environmt;
688 SendDlgItemMessage(hwnd, IDC_ENVLIST, LB_RESETCONTENT, 0, 0);
690 SendDlgItemMessage(hwnd, IDC_ENVLIST, LB_ADDSTRING, 0,
696 SendDlgItemMessage(hwnd, IDC_PFWDLIST, LB_ADDSTRING, 0,
701 CheckRadioButton(hwnd, IDC_EMBSD, IDC_EMRFC,
702 cfg.rfc_environ ? IDC_EMRFC : IDC_EMBSD);
703 CheckRadioButton(hwnd, IDC_TPASSIVE, IDC_TACTIVE,
704 cfg.passive_telnet ? IDC_TPASSIVE : IDC_TACTIVE);
706 SetDlgItemText(hwnd, IDC_TTEDIT, cfg.termtype);
707 SetDlgItemText(hwnd, IDC_LOGEDIT, cfg.username);
708 CheckDlgButton(hwnd, IDC_NOPTY, cfg.nopty);
709 CheckDlgButton(hwnd, IDC_COMPRESS, cfg.compression);
710 CheckDlgButton(hwnd, IDC_BUGGYMAC, cfg.buggymac);
711 CheckDlgButton(hwnd, IDC_AGENTFWD, cfg.agentfwd);
712 CheckRadioButton(hwnd, IDC_SSHPROT1, IDC_SSHPROT2,
713 cfg.sshprot == 1 ? IDC_SSHPROT1 : IDC_SSHPROT2);
714 CheckDlgButton(hwnd, IDC_AUTHTIS, cfg.try_tis_auth);
715 SetDlgItemText(hwnd, IDC_PKEDIT, cfg.keyfile);
716 SetDlgItemText(hwnd, IDC_CMDEDIT, cfg.remote_cmd);
720 static const struct { char *s; int c; } ciphers[] = {
721 { "3DES", CIPHER_3DES },
722 { "Blowfish", CIPHER_BLOWFISH },
723 { "DES (SSH 1 only)", CIPHER_DES },
724 { "AES (SSH 2 only)", CIPHER_AES },
725 { "-- warn below here --", CIPHER_WARN }
728 /* Set up the "selected ciphers" box. */
729 /* (cipherlist assumed to contain all ciphers) */
730 SendDlgItemMessage(hwnd, IDC_CIPHERLIST, LB_RESETCONTENT, 0, 0);
731 for (i = 0; i < CIPHER_MAX; i++) {
732 int c = cfg.ssh_cipherlist[i];
735 for (j = 0; j < (sizeof ciphers) / (sizeof ciphers[0]); j++) {
736 if (ciphers[j].c == c) {
741 pos = SendDlgItemMessage(hwnd, IDC_CIPHERLIST, LB_ADDSTRING,
743 SendDlgItemMessage(hwnd, IDC_CIPHERLIST, LB_SETITEMDATA,
750 CheckRadioButton(hwnd, IDC_MBWINDOWS, IDC_MBXTERM,
751 cfg.mouse_is_xterm ? IDC_MBXTERM : IDC_MBWINDOWS);
752 CheckDlgButton(hwnd, IDC_RAWCNP, cfg.rawcnp);
754 static int tabs[4] = { 25, 61, 96, 128 };
755 SendDlgItemMessage(hwnd, IDC_CCLIST, LB_SETTABSTOPS, 4,
758 for (i = 0; i < 128; i++) {
760 sprintf(str, "%d\t(0x%02X)\t%c\t%d", i, i,
761 (i >= 0x21 && i != 0x7F) ? i : ' ', cfg.wordness[i]);
762 SendDlgItemMessage(hwnd, IDC_CCLIST, LB_ADDSTRING, 0,
766 CheckDlgButton(hwnd, IDC_BOLDCOLOUR, cfg.bold_colour);
767 CheckDlgButton(hwnd, IDC_PALETTE, cfg.try_palette);
770 n = SendDlgItemMessage(hwnd, IDC_COLOURLIST, LB_GETCOUNT, 0, 0);
771 for (i = n; i-- > 0;)
772 SendDlgItemMessage(hwnd, IDC_COLOURLIST,
773 LB_DELETESTRING, i, 0);
774 for (i = 0; i < 22; i++)
775 if (cfg.bold_colour || permcolour[i])
776 SendDlgItemMessage(hwnd, IDC_COLOURLIST, LB_ADDSTRING, 0,
777 (LPARAM) colours[i]);
779 SendDlgItemMessage(hwnd, IDC_COLOURLIST, LB_SETCURSEL, 0, 0);
780 SetDlgItemInt(hwnd, IDC_RVALUE, cfg.colours[0][0], FALSE);
781 SetDlgItemInt(hwnd, IDC_GVALUE, cfg.colours[0][1], FALSE);
782 SetDlgItemInt(hwnd, IDC_BVALUE, cfg.colours[0][2], FALSE);
788 SendDlgItemMessage(hwnd, IDC_CODEPAGE, CB_RESETCONTENT, 0, 0);
789 for (i = 0; (cp = cp_enumerate(i)) != NULL; i++) {
790 SendDlgItemMessage(hwnd, IDC_CODEPAGE, CB_ADDSTRING,
792 if (!strcmp(cp, cfg.line_codepage))
795 SendDlgItemMessage(hwnd, IDC_CODEPAGE, CB_SETCURSEL, index, 0);
798 CheckRadioButton(hwnd, IDC_VTXWINDOWS, IDC_VTUNICODE,
799 cfg.vtmode == VT_XWINDOWS ? IDC_VTXWINDOWS :
800 cfg.vtmode == VT_OEMANSI ? IDC_VTOEMANSI :
801 cfg.vtmode == VT_OEMONLY ? IDC_VTOEMONLY :
802 cfg.vtmode == VT_UNICODE ? IDC_VTUNICODE :
805 CheckDlgButton(hwnd, IDC_X11_FORWARD, cfg.x11_forward);
806 SetDlgItemText(hwnd, IDC_X11_DISPLAY, cfg.x11_display);
808 CheckDlgButton(hwnd, IDC_LPORT_ALL, cfg.lport_acceptall);
809 CheckRadioButton(hwnd, IDC_PFWDLOCAL, IDC_PFWDREMOTE, IDC_PFWDLOCAL);
812 struct treeview_faff {
817 static HTREEITEM treeview_insert(struct treeview_faff *faff,
818 int level, char *text)
823 ins.hParent = (level > 0 ? faff->lastat[level - 1] : TVI_ROOT);
824 ins.hInsertAfter = faff->lastat[level];
825 #if _WIN32_IE >= 0x0400 && defined NONAMELESSUNION
826 #define INSITEM DUMMYUNIONNAME.item
830 ins.INSITEM.mask = TVIF_TEXT;
831 ins.INSITEM.pszText = text;
832 newitem = TreeView_InsertItem(faff->treeview, &ins);
834 TreeView_Expand(faff->treeview, faff->lastat[level - 1],
836 faff->lastat[level] = newitem;
837 for (i = level + 1; i < 4; i++)
838 faff->lastat[i] = NULL;
843 * Create the panelfuls of controls in the configuration box.
845 static void create_controls(HWND hwnd, int dlgtype, int panel)
847 if (panel == sessionpanelstart) {
848 /* The Session panel. Accelerators used: [acgo] nprtih elsd w */
850 ctlposinit(&cp, hwnd, 80, 3, 13);
851 bartitle(&cp, "Basic options for your PuTTY session",
854 beginbox(&cp, "Specify your connection by host name or IP address",
857 "Host &Name (or IP address)",
858 IDC_HOSTSTATIC, IDC_HOST, 75,
859 "&Port", IDC_PORTSTATIC, IDC_PORT, 25, NULL);
860 if (backends[3].backend == NULL) {
861 /* this is PuTTYtel, so only three protocols available */
862 radioline(&cp, "Protocol:", IDC_PROTSTATIC, 3,
864 "&Telnet", IDC_PROTTELNET,
865 "Rlog&in", IDC_PROTRLOGIN, NULL);
867 radioline(&cp, "Protocol:", IDC_PROTSTATIC, 4,
869 "&Telnet", IDC_PROTTELNET,
870 "Rlog&in", IDC_PROTRLOGIN,
879 beginbox(&cp, "Load, save or delete a stored session",
881 sesssaver(&cp, "Sav&ed Sessions",
882 IDC_SESSSTATIC, IDC_SESSEDIT, IDC_SESSLIST,
883 "&Load", IDC_SESSLOAD,
884 "&Save", IDC_SESSSAVE, "&Delete", IDC_SESSDEL, NULL);
887 beginbox(&cp, NULL, IDC_BOX_SESSION3);
888 radioline(&cp, "Close &window on exit:", IDC_CLOSEEXIT, 4,
889 "Always", IDC_COEALWAYS,
890 "Never", IDC_COENEVER,
891 "Only on clean exit", IDC_COENORMAL, NULL);
895 if (panel == loggingpanelstart) {
896 /* The Logging panel. Accelerators used: [acgo] tplfwe */
898 ctlposinit(&cp, hwnd, 80, 3, 13);
899 bartitle(&cp, "Options controlling session logging",
901 beginbox(&cp, NULL, IDC_BOX_LOGGING1);
903 "Session logging:", IDC_LSTATSTATIC,
904 "Logging &turned off completely", IDC_LSTATOFF,
905 "Log &printable output only", IDC_LSTATASCII,
906 "&Log all session output", IDC_LSTATRAW, NULL);
907 editbutton(&cp, "Log &file name:",
908 IDC_LGFSTATIC, IDC_LGFEDIT, "Bro&wse...",
911 "What to do if the log file already &exists:",
912 IDC_LSTATXIST, "Always overwrite it", IDC_LSTATXOVR,
913 "Always append to the end of it", IDC_LSTATXAPN,
914 "Ask the user every time", IDC_LSTATXASK, NULL);
918 if (panel == terminalpanelstart) {
919 /* The Terminal panel. Accelerators used: [acgo] wdlen hts */
921 ctlposinit(&cp, hwnd, 80, 3, 13);
922 bartitle(&cp, "Options controlling the terminal emulation",
924 beginbox(&cp, "Set various terminal options", IDC_BOX_TERMINAL1);
925 checkbox(&cp, "Auto &wrap mode initially on", IDC_WRAPMODE);
926 checkbox(&cp, "&DEC Origin Mode initially on", IDC_DECOM);
927 checkbox(&cp, "Implicit CR in every &LF", IDC_LFHASCR);
928 checkbox(&cp, "Use background colour to &erase screen", IDC_BCE);
929 checkbox(&cp, "Enable bli&nking text", IDC_BLINKTEXT);
931 "An&swerback to ^E:", IDC_ANSWERBACK,
932 IDC_ANSWEREDIT, 100, NULL);
935 beginbox(&cp, "Line discipline options", IDC_BOX_TERMINAL2);
936 radioline(&cp, "Local ec&ho:", IDC_ECHOSTATIC, 3,
937 "Auto", IDC_ECHOBACKEND,
938 "Force on", IDC_ECHOYES, "Force off", IDC_ECHONO, NULL);
939 radioline(&cp, "Local line edi&ting:", IDC_EDITSTATIC, 3,
940 "Auto", IDC_EDITBACKEND,
941 "Force on", IDC_EDITYES, "Force off", IDC_EDITNO, NULL);
945 if (panel == bellpanelstart) {
946 /* The Bell panel. Accelerators used: [acgo] bdsm wit */
948 ctlposinit(&cp, hwnd, 80, 3, 13);
949 bartitle(&cp, "Options controlling the terminal bell",
951 beginbox(&cp, "Set the style of bell", IDC_BOX_BELL1);
953 "Action to happen when a &bell occurs:", IDC_BELLSTATIC,
954 "None (bell disabled)", IDC_BELL_DISABLED,
955 "Play Windows Default Sound", IDC_BELL_DEFAULT,
956 "Play a custom sound file", IDC_BELL_WAVEFILE,
957 "Visual bell (flash window)", IDC_BELL_VISUAL, NULL);
958 editbutton(&cp, "Custom sound file to play as a bell:",
959 IDC_BELL_WAVESTATIC, IDC_BELL_WAVEEDIT,
960 "Bro&wse...", IDC_BELL_WAVEBROWSE);
961 radioline(&cp, "Taskbar/caption &indication on bell:",
962 IDC_B_IND_STATIC, 3, "Disabled", IDC_B_IND_DISABLED,
963 "Flashing", IDC_B_IND_FLASH, "Steady", IDC_B_IND_STEADY,
966 beginbox(&cp, "Control the bell overload behaviour",
968 checkbox(&cp, "Bell is temporarily &disabled when over-used",
970 staticedit(&cp, "Over-use means this &many bells...",
971 IDC_BELLOVLNSTATIC, IDC_BELLOVLN, 20);
972 staticedit(&cp, "... in &this many seconds",
973 IDC_BELLOVLTSTATIC, IDC_BELLOVLT, 20);
975 "The bell is re-enabled after a few seconds of silence.",
977 staticedit(&cp, "Seconds of &silence required", IDC_BELLOVLSSTATIC,
982 if (panel == keyboardpanelstart) {
983 /* The Keyboard panel. Accelerators used: [acgo] bhf ruyntd */
985 ctlposinit(&cp, hwnd, 80, 3, 13);
987 bartitle(&cp, "Options controlling the effects of keys",
990 beginbox(&cp, "Change the sequences sent by:", IDC_BOX_KEYBOARD1);
991 radioline(&cp, "The &Backspace key", IDC_DELSTATIC, 2,
992 "Control-H", IDC_DEL008,
993 "Control-? (127)", IDC_DEL127, NULL);
994 radioline(&cp, "The &Home and End keys", IDC_HOMESTATIC, 2,
995 "Standard", IDC_HOMETILDE, "rxvt", IDC_HOMERXVT, NULL);
996 radioline(&cp, "The &Function keys and keypad", IDC_FUNCSTATIC, 3,
997 "ESC[n~", IDC_FUNCTILDE,
998 "Linux", IDC_FUNCLINUX,
999 "Xterm R6", IDC_FUNCXTERM,
1000 "VT400", IDC_FUNCVT400,
1001 "VT100+", IDC_FUNCVT100P, "SCO", IDC_FUNCSCO, NULL);
1003 beginbox(&cp, "Application keypad settings:", IDC_BOX_KEYBOARD2);
1005 "Application c&ursor keys totally disabled",
1007 radioline(&cp, "Initial state of cu&rsor keys:", IDC_CURSTATIC, 2,
1008 "Normal", IDC_CURNORMAL,
1009 "Application", IDC_CURAPPLIC, NULL);
1011 "Application ke&ypad keys totally disabled",
1013 radioline(&cp, "Initial state of &numeric keypad:", IDC_KPSTATIC,
1014 3, "Normal", IDC_KPNORMAL, "Application", IDC_KPAPPLIC,
1015 "NetHack", IDC_KPNH, NULL);
1017 beginbox(&cp, "Enable extra keyboard features:",
1019 checkbox(&cp, "AltGr ac&ts as Compose key", IDC_COMPOSEKEY);
1020 checkbox(&cp, "Control-Alt is &different from AltGr",
1025 if (panel == windowpanelstart) {
1026 /* The Window panel. Accelerators used: [acgo] rmz sdkp w4ylt */
1028 ctlposinit(&cp, hwnd, 80, 3, 13);
1029 bartitle(&cp, "Options controlling PuTTY's window",
1031 beginbox(&cp, "Set the size of the window", IDC_BOX_WINDOW1);
1033 "&Rows", IDC_ROWSSTATIC, IDC_ROWSEDIT, 50,
1034 "Colu&mns", IDC_COLSSTATIC, IDC_COLSEDIT, 50, NULL);
1035 checkbox(&cp, "Lock window size against resi&zing", IDC_LOCKSIZE);
1037 beginbox(&cp, "Control the scrollback in the window",
1039 staticedit(&cp, "Lines of &scrollback",
1040 IDC_SAVESTATIC, IDC_SAVEEDIT, 50);
1041 checkbox(&cp, "&Display scrollbar", IDC_SCROLLBAR);
1042 checkbox(&cp, "Reset scrollback on &keypress", IDC_SCROLLKEY);
1043 checkbox(&cp, "Reset scrollback on dis&play activity",
1046 beginbox(&cp, NULL, IDC_BOX_WINDOW3);
1047 checkbox(&cp, "&Warn before closing window", IDC_CLOSEWARN);
1048 checkbox(&cp, "Window closes on ALT-F&4", IDC_ALTF4);
1049 checkbox(&cp, "S&ystem menu appears on ALT-Space", IDC_ALTSPACE);
1050 checkbox(&cp, "System menu appears on A< alone", IDC_ALTONLY);
1051 checkbox(&cp, "Ensure window is always on &top", IDC_ALWAYSONTOP);
1055 if (panel == appearancepanelstart) {
1056 /* The Appearance panel. Accelerators used: [acgo] luvb h ti p s */
1058 ctlposinit(&cp, hwnd, 80, 3, 13);
1059 bartitle(&cp, "Options controlling PuTTY's appearance",
1060 IDC_TITLE_APPEARANCE);
1061 beginbox(&cp, "Adjust the use of the cursor", IDC_BOX_APPEARANCE1);
1062 radioline(&cp, "Cursor appearance:", IDC_CURSORSTATIC, 3,
1063 "B&lock", IDC_CURBLOCK,
1064 "&Underline", IDC_CURUNDER,
1065 "&Vertical line", IDC_CURVERT, NULL);
1066 checkbox(&cp, "Cursor &blinks", IDC_BLINKCUR);
1068 beginbox(&cp, "Set the font used in the terminal window",
1069 IDC_BOX_APPEARANCE2);
1070 staticbtn(&cp, "", IDC_FONTSTATIC, "C&hange...", IDC_CHOOSEFONT);
1072 beginbox(&cp, "Adjust the use of the window title",
1073 IDC_BOX_APPEARANCE3);
1075 "Window &title:", IDC_WINTITLE, IDC_WINEDIT, 100, NULL);
1076 checkbox(&cp, "Avoid ever using &icon title", IDC_WINNAME);
1078 beginbox(&cp, "Adjust the use of the mouse pointer",
1079 IDC_BOX_APPEARANCE4);
1080 checkbox(&cp, "Hide mouse &pointer when typing in window",
1083 beginbox(&cp, "Adjust the window border", IDC_BOX_APPEARANCE5);
1084 checkbox(&cp, "&Sunken-edge border (slightly thicker)",
1089 if (panel == translationpanelstart) {
1090 /* The Translation panel. Accelerators used: [acgo] xbep t s */
1092 ctlposinit(&cp, hwnd, 80, 3, 13);
1093 bartitle(&cp, "Options controlling character set translation",
1094 IDC_TITLE_TRANSLATION);
1095 beginbox(&cp, "Adjust how PuTTY displays line drawing characters",
1096 IDC_BOX_TRANSLATION1);
1098 "Handling of line drawing characters:", IDC_VTSTATIC,
1099 "Font has &XWindows encoding", IDC_VTXWINDOWS,
1100 "Use font in &both ANSI and OEM modes", IDC_VTOEMANSI,
1101 "Use font in O&EM mode only", IDC_VTOEMONLY,
1102 "&Poor man's line drawing (" "+" ", " "-" " and " "|" ")",
1103 IDC_VTPOORMAN, "&Unicode mode", IDC_VTUNICODE, NULL);
1105 beginbox(&cp, "Character set translation on received data",
1106 IDC_BOX_TRANSLATION2);
1108 "Received data assumed to be in which character set:",
1109 IDC_CODEPAGESTATIC, IDC_CODEPAGE);
1113 if (panel == selectionpanelstart) {
1114 /* The Selection panel. Accelerators used: [acgo] d wx hst */
1116 ctlposinit(&cp, hwnd, 80, 3, 13);
1117 bartitle(&cp, "Options controlling copy and paste",
1118 IDC_TITLE_SELECTION);
1119 beginbox(&cp, "Translation of pasted characters",
1120 IDC_BOX_SELECTION1);
1122 "&Don't translate line drawing chars into +, - and |",
1125 beginbox(&cp, "Control which mouse button does which thing",
1126 IDC_BOX_SELECTION2);
1127 radiobig(&cp, "Action of mouse buttons:", IDC_MBSTATIC,
1128 "&Windows (Right pastes, Middle extends)", IDC_MBWINDOWS,
1129 "&xterm (Right extends, Middle pastes)", IDC_MBXTERM,
1132 beginbox(&cp, "Control the select-one-word-at-a-time mode",
1133 IDC_BOX_SELECTION3);
1134 charclass(&cp, "C&haracter classes:", IDC_CCSTATIC, IDC_CCLIST,
1135 "&Set", IDC_CCSET, IDC_CCEDIT,
1136 "&to class", IDC_CCSTATIC2);
1140 if (panel == colourspanelstart) {
1141 /* The Colours panel. Accelerators used: [acgo] blum */
1143 ctlposinit(&cp, hwnd, 80, 3, 13);
1144 bartitle(&cp, "Options controlling use of colours",
1146 beginbox(&cp, "General options for colour usage",
1148 checkbox(&cp, "&Bolded text is a different colour",
1150 checkbox(&cp, "Attempt to use &logical palettes", IDC_PALETTE);
1152 beginbox(&cp, "Adjust the precise colours PuTTY displays",
1154 colouredit(&cp, "Select a colo&ur and then click to modify it:",
1155 IDC_COLOURSTATIC, IDC_COLOURLIST,
1156 "&Modify...", IDC_CHANGE,
1157 "Red:", IDC_RSTATIC, IDC_RVALUE,
1158 "Green:", IDC_GSTATIC, IDC_GVALUE,
1159 "Blue:", IDC_BSTATIC, IDC_BVALUE, NULL);
1163 if (panel == connectionpanelstart) {
1164 /* The Connection panel. Accelerators used: [acgo] tuk */
1166 ctlposinit(&cp, hwnd, 80, 3, 13);
1167 bartitle(&cp, "Options controlling the connection",
1168 IDC_TITLE_CONNECTION);
1170 beginbox(&cp, "Data to send to the server",
1171 IDC_BOX_CONNECTION1);
1172 staticedit(&cp, "Terminal-&type string", IDC_TTSTATIC,
1174 staticedit(&cp, "Auto-login &username", IDC_LOGSTATIC,
1178 beginbox(&cp, "Adjust telnet session.", IDC_BOX_CONNECTION1);
1179 checkbox(&cp, "Keyboard sends telnet Backspace and Interrupt",
1183 beginbox(&cp, "Sending of null packets to keep session active",
1184 IDC_BOX_CONNECTION2);
1185 staticedit(&cp, "Seconds between &keepalives (0 to turn off)",
1186 IDC_PINGSTATIC, IDC_PINGEDIT, 20);
1190 if (panel == telnetpanelstart) {
1191 /* The Telnet panel. Accelerators used: [acgo] svldr bftk */
1193 ctlposinit(&cp, hwnd, 80, 3, 13);
1195 bartitle(&cp, "Options controlling Telnet connections",
1197 beginbox(&cp, "Data to send to the server", IDC_BOX_TELNET1);
1198 staticedit(&cp, "Terminal-&speed string", IDC_TSSTATIC,
1200 envsetter(&cp, "Environment variables:", IDC_ENVSTATIC,
1201 "&Variable", IDC_VARSTATIC, IDC_VAREDIT, "Va&lue",
1202 IDC_VALSTATIC, IDC_VALEDIT, IDC_ENVLIST, "A&dd",
1203 IDC_ENVADD, "&Remove", IDC_ENVREMOVE);
1205 beginbox(&cp, "Telnet protocol adjustments", IDC_BOX_TELNET2);
1206 radioline(&cp, "Handling of OLD_ENVIRON ambiguity:",
1207 IDC_EMSTATIC, 2, "&BSD (commonplace)", IDC_EMBSD,
1208 "R&FC 1408 (unusual)", IDC_EMRFC, NULL);
1209 radioline(&cp, "&Telnet negotiation mode:", IDC_ACTSTATIC, 2,
1210 "Passive", IDC_TPASSIVE, "Active",
1212 checkbox(&cp, "&Keyboard sends telnet Backspace and Interrupt",
1218 if (panel == rloginpanelstart) {
1219 /* The Rlogin panel. Accelerators used: [acgo] sl */
1221 ctlposinit(&cp, hwnd, 80, 3, 13);
1223 bartitle(&cp, "Options controlling Rlogin connections",
1225 beginbox(&cp, "Data to send to the server", IDC_BOX_RLOGIN1);
1226 staticedit(&cp, "Terminal-&speed string", IDC_R_TSSTATIC,
1228 staticedit(&cp, "&Local username:", IDC_RLLUSERSTATIC,
1229 IDC_RLLUSEREDIT, 50);
1234 if (panel == sshpanelstart) {
1235 /* The SSH panel. Accelerators used: [acgo] r pe12i sud */
1237 ctlposinit(&cp, hwnd, 80, 3, 13);
1239 bartitle(&cp, "Options controlling SSH connections",
1241 beginbox(&cp, "Data to send to the server", IDC_BOX_SSH1);
1243 "&Remote command:", IDC_CMDSTATIC, IDC_CMDEDIT, 100,
1246 beginbox(&cp, "Protocol options", IDC_BOX_SSH2);
1247 checkbox(&cp, "Don't allocate a &pseudo-terminal", IDC_NOPTY);
1248 checkbox(&cp, "Enable compr&ession", IDC_COMPRESS);
1249 radioline(&cp, "Preferred SSH protocol version:",
1250 IDC_SSHPROTSTATIC, 2,
1251 "&1", IDC_SSHPROT1, "&2", IDC_SSHPROT2, NULL);
1252 checkbox(&cp, "&Imitate SSH 2 MAC bug in commercial <= v2.3.x",
1255 beginbox(&cp, "Encryption options", IDC_BOX_SSH3);
1256 prefslist(&cipherlist, &cp, "Encryption cipher &selection policy:",
1257 IDC_CIPHERSTATIC2, IDC_CIPHERLIST, IDC_CIPHERUP,
1263 if (panel == sshauthpanelstart) {
1264 /* The SSH authentication panel. Accelerators used: [acgo] m fkw */
1266 ctlposinit(&cp, hwnd, 80, 3, 13);
1268 bartitle(&cp, "Options controlling SSH authentication",
1270 beginbox(&cp, "Authentication methods",
1272 checkbox(&cp, "Atte&mpt TIS or CryptoCard authentication",
1275 beginbox(&cp, "Authentication parameters",
1277 checkbox(&cp, "Allow agent &forwarding", IDC_AGENTFWD);
1278 editbutton(&cp, "Private &key file for authentication:",
1279 IDC_PKSTATIC, IDC_PKEDIT, "Bro&wse...",
1285 if (panel == tunnelspanelstart) {
1286 /* The Tunnels panel. Accelerators used: [acgo] deilmrstx */
1288 ctlposinit(&cp, hwnd, 80, 3, 13);
1290 bartitle(&cp, "Options controlling SSH tunnelling",
1292 beginbox(&cp, "X11 forwarding", IDC_BOX_TUNNELS1);
1293 checkbox(&cp, "&Enable X11 forwarding", IDC_X11_FORWARD);
1294 multiedit(&cp, "&X display location", IDC_X11_DISPSTATIC,
1295 IDC_X11_DISPLAY, 50, NULL);
1297 beginbox(&cp, "Port forwarding", IDC_BOX_TUNNELS2);
1298 checkbox(&cp, "Local ports accept connections from o&ther hosts", IDC_LPORT_ALL);
1299 staticbtn(&cp, "Forwarded ports:", IDC_PFWDSTATIC,
1300 "&Remove", IDC_PFWDREMOVE);
1301 fwdsetter(&cp, IDC_PFWDLIST,
1302 "Add new forwarded port:", IDC_PFWDSTATIC2,
1303 "&Source port", IDC_SPORTSTATIC, IDC_SPORTEDIT,
1304 "Dest&ination", IDC_DPORTSTATIC, IDC_DPORTEDIT,
1305 "A&dd", IDC_PFWDADD);
1306 bareradioline(&cp, 2,
1307 "&Local", IDC_PFWDLOCAL, "Re&mote", IDC_PFWDREMOTE, NULL);
1315 * Helper function to load the session selected in SESSLIST
1316 * if any, as this is done in more than one place in
1317 * GenericMainDlgProc(). 0 => failure.
1319 static int load_selected_session(HWND hwnd)
1321 int n = SendDlgItemMessage(hwnd, IDC_SESSLIST,
1322 LB_GETCURSEL, 0, 0);
1328 isdef = !strcmp(sessions[n], "Default Settings");
1329 load_settings(sessions[n], !isdef, &cfg);
1330 init_dlg_ctrls(hwnd, TRUE);
1332 SetDlgItemText(hwnd, IDC_SESSEDIT, sessions[n]);
1334 SetDlgItemText(hwnd, IDC_SESSEDIT, "");
1335 /* Restore the selection, which will have been clobbered by
1336 * SESSEDIT handling. */
1337 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_SETCURSEL, n, 0);
1342 * This function is the configuration box.
1344 static int GenericMainDlgProc(HWND hwnd, UINT msg,
1345 WPARAM wParam, LPARAM lParam, int dlgtype)
1348 struct treeview_faff tvfaff;
1351 char filename[sizeof(cfg.keyfile)];
1354 char fontstatic[256];
1356 struct servent *service;
1358 static UINT draglistmsg = WM_NULL;
1363 SetWindowLong(hwnd, GWL_USERDATA, 0);
1364 SendMessage(hwnd, WM_SETICON, (WPARAM) ICON_BIG,
1365 (LPARAM) LoadIcon(hinst, MAKEINTRESOURCE(IDI_CFGICON)));
1367 * Centre the window.
1369 { /* centre the window */
1372 hw = GetDesktopWindow();
1373 if (GetWindowRect(hw, &rs) && GetWindowRect(hwnd, &rd))
1375 (rs.right + rs.left + rd.left - rd.right) / 2,
1376 (rs.bottom + rs.top + rd.top - rd.bottom) / 2,
1377 rd.right - rd.left, rd.bottom - rd.top, TRUE);
1381 * Create the tree view.
1389 r.right = r.left + 75;
1391 r.bottom = r.top + 10;
1392 MapDialogRect(hwnd, &r);
1393 tvstatic = CreateWindowEx(0, "STATIC", "Cate&gory:",
1394 WS_CHILD | WS_VISIBLE,
1396 r.right - r.left, r.bottom - r.top,
1397 hwnd, (HMENU) IDCX_TVSTATIC, hinst,
1399 font = SendMessage(hwnd, WM_GETFONT, 0, 0);
1400 SendMessage(tvstatic, WM_SETFONT, font, MAKELPARAM(TRUE, 0));
1403 r.right = r.left + 75;
1405 r.bottom = r.top + 206;
1406 MapDialogRect(hwnd, &r);
1407 treeview = CreateWindowEx(WS_EX_CLIENTEDGE, WC_TREEVIEW, "",
1408 WS_CHILD | WS_VISIBLE |
1409 WS_TABSTOP | TVS_HASLINES |
1410 TVS_DISABLEDRAGDROP | TVS_HASBUTTONS
1412 TVS_SHOWSELALWAYS, r.left, r.top,
1413 r.right - r.left, r.bottom - r.top,
1414 hwnd, (HMENU) IDCX_TREEVIEW, hinst,
1416 font = SendMessage(hwnd, WM_GETFONT, 0, 0);
1417 SendMessage(treeview, WM_SETFONT, font, MAKELPARAM(TRUE, 0));
1418 tvfaff.treeview = treeview;
1419 memset(tvfaff.lastat, 0, sizeof(tvfaff.lastat));
1423 * Set up the tree view contents.
1425 hsession = treeview_insert(&tvfaff, 0, "Session");
1426 treeview_insert(&tvfaff, 1, "Logging");
1427 treeview_insert(&tvfaff, 0, "Terminal");
1428 treeview_insert(&tvfaff, 1, "Keyboard");
1429 treeview_insert(&tvfaff, 1, "Bell");
1430 treeview_insert(&tvfaff, 0, "Window");
1431 treeview_insert(&tvfaff, 1, "Appearance");
1432 treeview_insert(&tvfaff, 1, "Translation");
1433 treeview_insert(&tvfaff, 1, "Selection");
1434 treeview_insert(&tvfaff, 1, "Colours");
1435 treeview_insert(&tvfaff, 0, "Connection");
1437 treeview_insert(&tvfaff, 1, "Telnet");
1438 treeview_insert(&tvfaff, 1, "Rlogin");
1439 if (backends[3].backend != NULL) {
1440 treeview_insert(&tvfaff, 1, "SSH");
1441 /* XXX long name is ugly */
1442 /* XXX make it closed by default? */
1443 treeview_insert(&tvfaff, 2, "Auth");
1444 treeview_insert(&tvfaff, 2, "Tunnels");
1449 * Put the treeview selection on to the Session panel. This
1450 * should also cause creation of the relevant controls.
1452 TreeView_SelectItem(treeview, hsession);
1455 * Set focus into the first available control.
1459 ctl = GetDlgItem(hwnd, IDC_HOST);
1461 ctl = GetDlgItem(hwnd, IDC_CLOSEEXIT);
1465 SetWindowLong(hwnd, GWL_USERDATA, 1);
1466 sesslist_has_focus = 0;
1470 * Button release should trigger WM_OK if there was a
1471 * previous double click on the session list.
1475 SendMessage(hwnd, WM_COMMAND, IDOK, 0);
1478 if (LOWORD(wParam) == IDCX_TREEVIEW &&
1479 ((LPNMHDR) lParam)->code == TVN_SELCHANGED) {
1481 TreeView_GetSelection(((LPNMHDR) lParam)->hwndFrom);
1486 item.pszText = buffer;
1487 item.cchTextMax = sizeof(buffer);
1488 item.mask = TVIF_TEXT;
1489 TreeView_GetItem(((LPNMHDR) lParam)->hwndFrom, &item);
1490 for (j = controlstartvalue; j < controlendvalue; j++) {
1491 HWND item = GetDlgItem(hwnd, j);
1493 DestroyWindow(item);
1495 if (!strcmp(buffer, "Session"))
1496 create_controls(hwnd, dlgtype, sessionpanelstart);
1497 if (!strcmp(buffer, "Logging"))
1498 create_controls(hwnd, dlgtype, loggingpanelstart);
1499 if (!strcmp(buffer, "Keyboard"))
1500 create_controls(hwnd, dlgtype, keyboardpanelstart);
1501 if (!strcmp(buffer, "Terminal"))
1502 create_controls(hwnd, dlgtype, terminalpanelstart);
1503 if (!strcmp(buffer, "Bell"))
1504 create_controls(hwnd, dlgtype, bellpanelstart);
1505 if (!strcmp(buffer, "Window"))
1506 create_controls(hwnd, dlgtype, windowpanelstart);
1507 if (!strcmp(buffer, "Appearance"))
1508 create_controls(hwnd, dlgtype, appearancepanelstart);
1509 if (!strcmp(buffer, "Tunnels"))
1510 create_controls(hwnd, dlgtype, tunnelspanelstart);
1511 if (!strcmp(buffer, "Connection"))
1512 create_controls(hwnd, dlgtype, connectionpanelstart);
1513 if (!strcmp(buffer, "Telnet"))
1514 create_controls(hwnd, dlgtype, telnetpanelstart);
1515 if (!strcmp(buffer, "Rlogin"))
1516 create_controls(hwnd, dlgtype, rloginpanelstart);
1517 if (!strcmp(buffer, "SSH"))
1518 create_controls(hwnd, dlgtype, sshpanelstart);
1519 if (!strcmp(buffer, "Auth"))
1520 create_controls(hwnd, dlgtype, sshauthpanelstart);
1521 if (!strcmp(buffer, "Selection"))
1522 create_controls(hwnd, dlgtype, selectionpanelstart);
1523 if (!strcmp(buffer, "Colours"))
1524 create_controls(hwnd, dlgtype, colourspanelstart);
1525 if (!strcmp(buffer, "Translation"))
1526 create_controls(hwnd, dlgtype, translationpanelstart);
1528 init_dlg_ctrls(hwnd, FALSE);
1530 SetFocus(((LPNMHDR) lParam)->hwndFrom); /* ensure focus stays */
1536 * Only process WM_COMMAND once the dialog is fully formed.
1538 if (GetWindowLong(hwnd, GWL_USERDATA) == 1)
1539 switch (LOWORD(wParam)) {
1541 /* Behaviour of the "Open" button is different if the
1542 * session list has focus, *unless* the user just
1543 * double-clicked... */
1544 if (sesslist_has_focus && !readytogo) {
1545 if (!load_selected_session(hwnd)) {
1550 /* If at this point we have a valid session, go! */
1559 case IDC_PROTTELNET:
1560 case IDC_PROTRLOGIN:
1563 if (HIWORD(wParam) == BN_CLICKED ||
1564 HIWORD(wParam) == BN_DOUBLECLICKED) {
1565 int i = IsDlgButtonChecked(hwnd, IDC_PROTSSH);
1566 int j = IsDlgButtonChecked(hwnd, IDC_PROTTELNET);
1567 int k = IsDlgButtonChecked(hwnd, IDC_PROTRLOGIN);
1569 i ? PROT_SSH : j ? PROT_TELNET : k ? PROT_RLOGIN :
1571 if ((cfg.protocol == PROT_SSH && cfg.port != 22)
1572 || (cfg.protocol == PROT_TELNET && cfg.port != 23)
1573 || (cfg.protocol == PROT_RLOGIN
1574 && cfg.port != 513)) {
1575 cfg.port = i ? 22 : j ? 23 : 513;
1576 SetDlgItemInt(hwnd, IDC_PORT, cfg.port, FALSE);
1581 if (HIWORD(wParam) == EN_CHANGE)
1582 GetDlgItemText(hwnd, IDC_HOST, cfg.host,
1583 sizeof(cfg.host) - 1);
1586 if (HIWORD(wParam) == EN_CHANGE) {
1587 GetDlgItemText(hwnd, IDC_PORT, portname, 31);
1588 if (isdigit(portname[0]))
1589 MyGetDlgItemInt(hwnd, IDC_PORT, &cfg.port);
1591 service = getservbyname(portname, NULL);
1593 cfg.port = ntohs(service->s_port);
1600 if (HIWORD(wParam) == EN_CHANGE) {
1601 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_SETCURSEL,
1603 GetDlgItemText(hwnd, IDC_SESSEDIT,
1604 savedsession, sizeof(savedsession) - 1);
1605 savedsession[sizeof(savedsession) - 1] = '\0';
1609 if (HIWORD(wParam) == BN_CLICKED ||
1610 HIWORD(wParam) == BN_DOUBLECLICKED) {
1615 GetDlgItemText(hwnd, IDC_SESSEDIT, str,
1618 int n = SendDlgItemMessage(hwnd, IDC_SESSLIST,
1619 LB_GETCURSEL, 0, 0);
1624 strcpy(str, sessions[n]);
1626 save_settings(str, !!strcmp(str, "Default Settings"),
1628 get_sesslist(FALSE);
1630 SendDlgItemMessage(hwnd, IDC_SESSLIST, WM_SETREDRAW,
1632 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_RESETCONTENT,
1634 for (i = 0; i < nsessions; i++)
1635 SendDlgItemMessage(hwnd, IDC_SESSLIST,
1637 (LPARAM) (sessions[i]));
1638 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_SETCURSEL,
1640 SendDlgItemMessage(hwnd, IDC_SESSLIST, WM_SETREDRAW,
1642 InvalidateRect(GetDlgItem(hwnd, IDC_SESSLIST), NULL,
1648 if (LOWORD(wParam) == IDC_SESSLIST) {
1649 if (HIWORD(wParam) == LBN_SETFOCUS)
1650 sesslist_has_focus = 1;
1651 else if (HIWORD(wParam) == LBN_KILLFOCUS)
1652 sesslist_has_focus = 0;
1654 if (LOWORD(wParam) == IDC_SESSLOAD &&
1655 HIWORD(wParam) != BN_CLICKED &&
1656 HIWORD(wParam) != BN_DOUBLECLICKED) break;
1657 if (LOWORD(wParam) == IDC_SESSLIST &&
1658 HIWORD(wParam) != LBN_DBLCLK) break;
1659 /* Load the session selected in SESSLIST. */
1660 if (load_selected_session(hwnd) &&
1661 LOWORD(wParam) == IDC_SESSLIST) {
1663 * A double-click on a saved session should
1664 * actually start the session, not just load it.
1665 * Unless it's Default Settings or some other
1666 * host-less set of saved settings.
1675 if (HIWORD(wParam) == BN_CLICKED ||
1676 HIWORD(wParam) == BN_DOUBLECLICKED) {
1677 int n = SendDlgItemMessage(hwnd, IDC_SESSLIST,
1678 LB_GETCURSEL, 0, 0);
1679 if (n == LB_ERR || n == 0) {
1683 del_settings(sessions[n]);
1684 get_sesslist(FALSE);
1686 SendDlgItemMessage(hwnd, IDC_SESSLIST, WM_SETREDRAW,
1688 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_RESETCONTENT,
1690 for (i = 0; i < nsessions; i++)
1691 SendDlgItemMessage(hwnd, IDC_SESSLIST,
1693 (LPARAM) (sessions[i]));
1694 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_SETCURSEL,
1696 SendDlgItemMessage(hwnd, IDC_SESSLIST, WM_SETREDRAW,
1698 InvalidateRect(GetDlgItem(hwnd, IDC_SESSLIST), NULL,
1702 if (HIWORD(wParam) == EN_CHANGE)
1703 MyGetDlgItemInt(hwnd, IDC_PINGEDIT,
1704 &cfg.ping_interval);
1708 if (HIWORD(wParam) == BN_CLICKED ||
1709 HIWORD(wParam) == BN_DOUBLECLICKED)
1710 cfg.bksp_is_delete =
1711 IsDlgButtonChecked(hwnd, IDC_DEL127);
1715 if (HIWORD(wParam) == BN_CLICKED ||
1716 HIWORD(wParam) == BN_DOUBLECLICKED)
1718 IsDlgButtonChecked(hwnd, IDC_HOMERXVT);
1724 case IDC_FUNCVT100P:
1726 if (HIWORD(wParam) == BN_CLICKED ||
1727 HIWORD(wParam) == BN_DOUBLECLICKED)
1728 switch (LOWORD(wParam)) {
1741 case IDC_FUNCVT100P:
1751 if (HIWORD(wParam) == BN_CLICKED ||
1752 HIWORD(wParam) == BN_DOUBLECLICKED) {
1754 IsDlgButtonChecked(hwnd, IDC_KPAPPLIC);
1755 cfg.nethack_keypad = FALSE;
1759 if (HIWORD(wParam) == BN_CLICKED ||
1760 HIWORD(wParam) == BN_DOUBLECLICKED) {
1761 cfg.app_keypad = FALSE;
1762 cfg.nethack_keypad = TRUE;
1767 if (HIWORD(wParam) == BN_CLICKED ||
1768 HIWORD(wParam) == BN_DOUBLECLICKED)
1770 IsDlgButtonChecked(hwnd, IDC_CURAPPLIC);
1773 if (HIWORD(wParam) == BN_CLICKED ||
1774 HIWORD(wParam) == BN_DOUBLECLICKED)
1776 IsDlgButtonChecked(hwnd, IDC_NOAPPLICC);
1779 if (HIWORD(wParam) == BN_CLICKED ||
1780 HIWORD(wParam) == BN_DOUBLECLICKED)
1782 IsDlgButtonChecked(hwnd, IDC_NOAPPLICK);
1785 if (HIWORD(wParam) == BN_CLICKED ||
1786 HIWORD(wParam) == BN_DOUBLECLICKED)
1787 cfg.alt_f4 = IsDlgButtonChecked(hwnd, IDC_ALTF4);
1790 if (HIWORD(wParam) == BN_CLICKED ||
1791 HIWORD(wParam) == BN_DOUBLECLICKED)
1793 IsDlgButtonChecked(hwnd, IDC_ALTSPACE);
1796 if (HIWORD(wParam) == BN_CLICKED ||
1797 HIWORD(wParam) == BN_DOUBLECLICKED)
1799 IsDlgButtonChecked(hwnd, IDC_ALTONLY);
1801 case IDC_ECHOBACKEND:
1804 if (HIWORD(wParam) == BN_CLICKED ||
1805 HIWORD(wParam) == BN_DOUBLECLICKED) {
1806 if (LOWORD(wParam) == IDC_ECHOBACKEND)
1807 cfg.localecho = LD_BACKEND;
1808 if (LOWORD(wParam) == IDC_ECHOYES)
1809 cfg.localecho = LD_YES;
1810 if (LOWORD(wParam) == IDC_ECHONO)
1811 cfg.localecho = LD_NO;
1814 case IDC_EDITBACKEND:
1817 if (HIWORD(wParam) == BN_CLICKED ||
1818 HIWORD(wParam) == BN_DOUBLECLICKED) {
1819 if (LOWORD(wParam) == IDC_EDITBACKEND)
1820 cfg.localedit = LD_BACKEND;
1821 if (LOWORD(wParam) == IDC_EDITYES)
1822 cfg.localedit = LD_YES;
1823 if (LOWORD(wParam) == IDC_EDITNO)
1824 cfg.localedit = LD_NO;
1827 case IDC_ANSWEREDIT:
1828 if (HIWORD(wParam) == EN_CHANGE)
1829 GetDlgItemText(hwnd, IDC_ANSWEREDIT, cfg.answerback,
1830 sizeof(cfg.answerback) - 1);
1832 case IDC_ALWAYSONTOP:
1833 if (HIWORD(wParam) == BN_CLICKED ||
1834 HIWORD(wParam) == BN_DOUBLECLICKED)
1836 IsDlgButtonChecked(hwnd, IDC_ALWAYSONTOP);
1839 if (HIWORD(wParam) == BN_CLICKED ||
1840 HIWORD(wParam) == BN_DOUBLECLICKED)
1842 IsDlgButtonChecked(hwnd, IDC_SCROLLKEY);
1844 case IDC_SCROLLDISP:
1845 if (HIWORD(wParam) == BN_CLICKED ||
1846 HIWORD(wParam) == BN_DOUBLECLICKED)
1847 cfg.scroll_on_disp =
1848 IsDlgButtonChecked(hwnd, IDC_SCROLLDISP);
1850 case IDC_COMPOSEKEY:
1851 if (HIWORD(wParam) == BN_CLICKED ||
1852 HIWORD(wParam) == BN_DOUBLECLICKED)
1854 IsDlgButtonChecked(hwnd, IDC_COMPOSEKEY);
1856 case IDC_CTRLALTKEYS:
1857 if (HIWORD(wParam) == BN_CLICKED ||
1858 HIWORD(wParam) == BN_DOUBLECLICKED)
1860 IsDlgButtonChecked(hwnd, IDC_CTRLALTKEYS);
1863 if (HIWORD(wParam) == BN_CLICKED ||
1864 HIWORD(wParam) == BN_DOUBLECLICKED)
1865 cfg.telnet_keyboard =
1866 IsDlgButtonChecked(hwnd, IDC_TELNETKEY);
1869 if (HIWORD(wParam) == BN_CLICKED ||
1870 HIWORD(wParam) == BN_DOUBLECLICKED)
1872 IsDlgButtonChecked(hwnd, IDC_WRAPMODE);
1875 if (HIWORD(wParam) == BN_CLICKED ||
1876 HIWORD(wParam) == BN_DOUBLECLICKED)
1877 cfg.dec_om = IsDlgButtonChecked(hwnd, IDC_DECOM);
1880 if (HIWORD(wParam) == BN_CLICKED ||
1881 HIWORD(wParam) == BN_DOUBLECLICKED)
1883 IsDlgButtonChecked(hwnd, IDC_LFHASCR);
1886 if (HIWORD(wParam) == EN_CHANGE)
1887 MyGetDlgItemInt(hwnd, IDC_ROWSEDIT, &cfg.height);
1890 if (HIWORD(wParam) == EN_CHANGE)
1891 MyGetDlgItemInt(hwnd, IDC_COLSEDIT, &cfg.width);
1894 if (HIWORD(wParam) == EN_CHANGE)
1895 MyGetDlgItemInt(hwnd, IDC_SAVEEDIT, &cfg.savelines);
1897 case IDC_CHOOSEFONT:
1900 lf.lfHeight = -MulDiv(cfg.fontheight,
1901 GetDeviceCaps(hdc, LOGPIXELSY),
1905 lf.lfWidth = lf.lfEscapement = lf.lfOrientation = 0;
1906 lf.lfItalic = lf.lfUnderline = lf.lfStrikeOut = 0;
1907 lf.lfWeight = (cfg.fontisbold ? FW_BOLD : 0);
1908 lf.lfCharSet = cfg.fontcharset;
1909 lf.lfOutPrecision = OUT_DEFAULT_PRECIS;
1910 lf.lfClipPrecision = CLIP_DEFAULT_PRECIS;
1911 lf.lfQuality = DEFAULT_QUALITY;
1912 lf.lfPitchAndFamily = FIXED_PITCH | FF_DONTCARE;
1913 strncpy(lf.lfFaceName, cfg.font,
1914 sizeof(lf.lfFaceName) - 1);
1915 lf.lfFaceName[sizeof(lf.lfFaceName) - 1] = '\0';
1917 cf.lStructSize = sizeof(cf);
1918 cf.hwndOwner = hwnd;
1920 cf.Flags = CF_FIXEDPITCHONLY | CF_FORCEFONTEXIST |
1921 CF_INITTOLOGFONTSTRUCT | CF_SCREENFONTS;
1923 if (ChooseFont(&cf)) {
1924 strncpy(cfg.font, lf.lfFaceName, sizeof(cfg.font) - 1);
1925 cfg.font[sizeof(cfg.font) - 1] = '\0';
1926 cfg.fontisbold = (lf.lfWeight == FW_BOLD);
1927 cfg.fontcharset = lf.lfCharSet;
1928 cfg.fontheight = cf.iPointSize / 10;
1929 fmtfont(fontstatic);
1930 SetDlgItemText(hwnd, IDC_FONTSTATIC, fontstatic);
1933 case IDC_BELL_DISABLED:
1934 case IDC_BELL_DEFAULT:
1935 case IDC_BELL_WAVEFILE:
1936 case IDC_BELL_VISUAL:
1937 if (HIWORD(wParam) == BN_CLICKED ||
1938 HIWORD(wParam) == BN_DOUBLECLICKED) {
1939 if (LOWORD(wParam) == IDC_BELL_DISABLED)
1940 cfg.beep = BELL_DISABLED;
1941 if (LOWORD(wParam) == IDC_BELL_DEFAULT)
1942 cfg.beep = BELL_DEFAULT;
1943 if (LOWORD(wParam) == IDC_BELL_WAVEFILE)
1944 cfg.beep = BELL_WAVEFILE;
1945 if (LOWORD(wParam) == IDC_BELL_VISUAL)
1946 cfg.beep = BELL_VISUAL;
1949 case IDC_B_IND_DISABLED:
1950 case IDC_B_IND_FLASH:
1951 case IDC_B_IND_STEADY:
1952 if (HIWORD(wParam) == BN_CLICKED ||
1953 HIWORD(wParam) == BN_DOUBLECLICKED) {
1954 if (LOWORD(wParam) == IDC_B_IND_DISABLED)
1955 cfg.beep_ind = B_IND_DISABLED;
1956 if (LOWORD(wParam) == IDC_B_IND_FLASH)
1957 cfg.beep_ind = B_IND_FLASH;
1958 if (LOWORD(wParam) == IDC_B_IND_STEADY)
1959 cfg.beep_ind = B_IND_STEADY;
1962 case IDC_BELL_WAVEBROWSE:
1963 memset(&of, 0, sizeof(of));
1964 #ifdef OPENFILENAME_SIZE_VERSION_400
1965 of.lStructSize = OPENFILENAME_SIZE_VERSION_400;
1967 of.lStructSize = sizeof(of);
1969 of.hwndOwner = hwnd;
1970 of.lpstrFilter = "Wave Files\0*.WAV\0AllFiles\0*\0\0\0";
1971 of.lpstrCustomFilter = NULL;
1972 of.nFilterIndex = 1;
1973 of.lpstrFile = filename;
1974 strcpy(filename, cfg.bell_wavefile);
1975 of.nMaxFile = sizeof(filename);
1976 of.lpstrFileTitle = NULL;
1977 of.lpstrInitialDir = NULL;
1978 of.lpstrTitle = "Select Bell Sound File";
1980 if (GetOpenFileName(&of)) {
1981 strcpy(cfg.bell_wavefile, filename);
1982 SetDlgItemText(hwnd, IDC_BELL_WAVEEDIT,
1986 case IDC_BELL_WAVEEDIT:
1987 if (HIWORD(wParam) == EN_CHANGE)
1988 GetDlgItemText(hwnd, IDC_BELL_WAVEEDIT,
1990 sizeof(cfg.bell_wavefile) - 1);
1993 if (HIWORD(wParam) == BN_CLICKED ||
1994 HIWORD(wParam) == BN_DOUBLECLICKED)
1996 IsDlgButtonChecked(hwnd, IDC_BELLOVL);
1999 if (HIWORD(wParam) == EN_CHANGE)
2000 MyGetDlgItemInt(hwnd, IDC_BELLOVLN, &cfg.bellovl_n);
2003 if (HIWORD(wParam) == EN_CHANGE)
2004 MyGetDlgItemFlt(hwnd, IDC_BELLOVLT, &cfg.bellovl_t,
2008 if (HIWORD(wParam) == EN_CHANGE)
2009 MyGetDlgItemFlt(hwnd, IDC_BELLOVLS, &cfg.bellovl_s,
2013 if (HIWORD(wParam) == BN_CLICKED ||
2014 HIWORD(wParam) == BN_DOUBLECLICKED)
2016 IsDlgButtonChecked(hwnd, IDC_BLINKTEXT);
2019 if (HIWORD(wParam) == BN_CLICKED ||
2020 HIWORD(wParam) == BN_DOUBLECLICKED)
2021 cfg.bce = IsDlgButtonChecked(hwnd, IDC_BCE);
2024 if (HIWORD(wParam) == BN_CLICKED ||
2025 HIWORD(wParam) == BN_DOUBLECLICKED)
2026 cfg.win_name_always =
2027 IsDlgButtonChecked(hwnd, IDC_WINNAME);
2030 if (HIWORD(wParam) == BN_CLICKED ||
2031 HIWORD(wParam) == BN_DOUBLECLICKED)
2033 IsDlgButtonChecked(hwnd, IDC_HIDEMOUSE);
2035 case IDC_SUNKENEDGE:
2036 if (HIWORD(wParam) == BN_CLICKED ||
2037 HIWORD(wParam) == BN_DOUBLECLICKED)
2039 IsDlgButtonChecked(hwnd, IDC_SUNKENEDGE);
2042 if (HIWORD(wParam) == BN_CLICKED ||
2043 HIWORD(wParam) == BN_DOUBLECLICKED)
2044 cfg.cursor_type = 0;
2047 if (HIWORD(wParam) == BN_CLICKED ||
2048 HIWORD(wParam) == BN_DOUBLECLICKED)
2049 cfg.cursor_type = 1;
2052 if (HIWORD(wParam) == BN_CLICKED ||
2053 HIWORD(wParam) == BN_DOUBLECLICKED)
2054 cfg.cursor_type = 2;
2057 if (HIWORD(wParam) == BN_CLICKED ||
2058 HIWORD(wParam) == BN_DOUBLECLICKED)
2060 IsDlgButtonChecked(hwnd, IDC_BLINKCUR);
2063 if (HIWORD(wParam) == BN_CLICKED ||
2064 HIWORD(wParam) == BN_DOUBLECLICKED)
2066 IsDlgButtonChecked(hwnd, IDC_SCROLLBAR);
2069 if (HIWORD(wParam) == BN_CLICKED ||
2070 HIWORD(wParam) == BN_DOUBLECLICKED)
2072 IsDlgButtonChecked(hwnd, IDC_LOCKSIZE);
2075 if (HIWORD(wParam) == EN_CHANGE)
2076 GetDlgItemText(hwnd, IDC_WINEDIT, cfg.wintitle,
2077 sizeof(cfg.wintitle) - 1);
2082 if (HIWORD(wParam) == BN_CLICKED ||
2083 HIWORD(wParam) == BN_DOUBLECLICKED) {
2085 IsDlgButtonChecked(hwnd,
2086 IDC_COEALWAYS) ? COE_ALWAYS :
2087 IsDlgButtonChecked(hwnd,
2088 IDC_COENEVER) ? COE_NEVER :
2093 if (HIWORD(wParam) == BN_CLICKED ||
2094 HIWORD(wParam) == BN_DOUBLECLICKED)
2096 IsDlgButtonChecked(hwnd, IDC_CLOSEWARN);
2099 if (HIWORD(wParam) == EN_CHANGE)
2100 GetDlgItemText(hwnd, IDC_TTEDIT, cfg.termtype,
2101 sizeof(cfg.termtype) - 1);
2104 if (HIWORD(wParam) == EN_CHANGE)
2105 GetDlgItemText(hwnd, IDC_LGFEDIT, cfg.logfilename,
2106 sizeof(cfg.logfilename) - 1);
2109 memset(&of, 0, sizeof(of));
2110 #ifdef OPENFILENAME_SIZE_VERSION_400
2111 of.lStructSize = OPENFILENAME_SIZE_VERSION_400;
2113 of.lStructSize = sizeof(of);
2115 of.hwndOwner = hwnd;
2116 of.lpstrFilter = "All Files\0*\0\0\0";
2117 of.lpstrCustomFilter = NULL;
2118 of.nFilterIndex = 1;
2119 of.lpstrFile = filename;
2120 strcpy(filename, cfg.logfilename);
2121 of.nMaxFile = sizeof(filename);
2122 of.lpstrFileTitle = NULL;
2123 of.lpstrInitialDir = NULL;
2124 of.lpstrTitle = "Select session log file";
2126 if (GetSaveFileName(&of)) {
2127 strcpy(cfg.logfilename, filename);
2128 SetDlgItemText(hwnd, IDC_LGFEDIT, cfg.logfilename);
2132 case IDC_LSTATASCII:
2134 if (HIWORD(wParam) == BN_CLICKED ||
2135 HIWORD(wParam) == BN_DOUBLECLICKED) {
2136 if (IsDlgButtonChecked(hwnd, IDC_LSTATOFF))
2138 if (IsDlgButtonChecked(hwnd, IDC_LSTATASCII))
2140 if (IsDlgButtonChecked(hwnd, IDC_LSTATRAW))
2147 if (HIWORD(wParam) == BN_CLICKED ||
2148 HIWORD(wParam) == BN_DOUBLECLICKED) {
2149 if (IsDlgButtonChecked(hwnd, IDC_LSTATXASK))
2150 cfg.logxfovr = LGXF_ASK;
2151 if (IsDlgButtonChecked(hwnd, IDC_LSTATXAPN))
2152 cfg.logxfovr = LGXF_APN;
2153 if (IsDlgButtonChecked(hwnd, IDC_LSTATXOVR))
2154 cfg.logxfovr = LGXF_OVR;
2159 if (HIWORD(wParam) == EN_CHANGE)
2160 GetDlgItemText(hwnd, LOWORD(wParam), cfg.termspeed,
2161 sizeof(cfg.termspeed) - 1);
2164 if (HIWORD(wParam) == EN_CHANGE)
2165 GetDlgItemText(hwnd, IDC_LOGEDIT, cfg.username,
2166 sizeof(cfg.username) - 1);
2168 case IDC_RLLUSEREDIT:
2169 if (HIWORD(wParam) == EN_CHANGE)
2170 GetDlgItemText(hwnd, IDC_RLLUSEREDIT,
2172 sizeof(cfg.localusername) - 1);
2176 cfg.rfc_environ = IsDlgButtonChecked(hwnd, IDC_EMRFC);
2180 cfg.passive_telnet =
2181 IsDlgButtonChecked(hwnd, IDC_TPASSIVE);
2184 if (HIWORD(wParam) == BN_CLICKED ||
2185 HIWORD(wParam) == BN_DOUBLECLICKED) {
2186 char str[sizeof(cfg.environmt)];
2188 GetDlgItemText(hwnd, IDC_VAREDIT, str,
2194 p = str + strlen(str);
2196 GetDlgItemText(hwnd, IDC_VALEDIT, p,
2197 sizeof(str) - 1 - (p - str));
2208 if ((p - cfg.environmt) + strlen(str) + 2 <
2209 sizeof(cfg.environmt)) {
2211 p[strlen(str) + 1] = '\0';
2212 SendDlgItemMessage(hwnd, IDC_ENVLIST, LB_ADDSTRING,
2214 SetDlgItemText(hwnd, IDC_VAREDIT, "");
2215 SetDlgItemText(hwnd, IDC_VALEDIT, "");
2217 MessageBox(hwnd, "Environment too big",
2218 "PuTTY Error", MB_OK | MB_ICONERROR);
2223 if (HIWORD(wParam) != BN_CLICKED &&
2224 HIWORD(wParam) != BN_DOUBLECLICKED) break;
2226 SendDlgItemMessage(hwnd, IDC_ENVLIST, LB_GETCURSEL, 0,
2233 SendDlgItemMessage(hwnd, IDC_ENVLIST, LB_DELETESTRING,
2260 if (HIWORD(wParam) == BN_CLICKED ||
2261 HIWORD(wParam) == BN_DOUBLECLICKED)
2262 cfg.nopty = IsDlgButtonChecked(hwnd, IDC_NOPTY);
2265 if (HIWORD(wParam) == BN_CLICKED ||
2266 HIWORD(wParam) == BN_DOUBLECLICKED)
2268 IsDlgButtonChecked(hwnd, IDC_COMPRESS);
2271 if (HIWORD(wParam) == BN_CLICKED ||
2272 HIWORD(wParam) == BN_DOUBLECLICKED)
2274 IsDlgButtonChecked(hwnd, IDC_BUGGYMAC);
2277 if (HIWORD(wParam) == BN_CLICKED ||
2278 HIWORD(wParam) == BN_DOUBLECLICKED)
2280 IsDlgButtonChecked(hwnd, IDC_AGENTFWD);
2282 case IDC_CIPHERLIST:
2285 handle_prefslist(&cipherlist,
2286 cfg.ssh_cipherlist, CIPHER_MAX,
2287 0, hwnd, wParam, lParam);
2291 if (HIWORD(wParam) == BN_CLICKED ||
2292 HIWORD(wParam) == BN_DOUBLECLICKED) {
2293 if (IsDlgButtonChecked(hwnd, IDC_SSHPROT1))
2295 else if (IsDlgButtonChecked(hwnd, IDC_SSHPROT2))
2300 if (HIWORD(wParam) == BN_CLICKED ||
2301 HIWORD(wParam) == BN_DOUBLECLICKED)
2303 IsDlgButtonChecked(hwnd, IDC_AUTHTIS);
2306 if (HIWORD(wParam) == EN_CHANGE)
2307 GetDlgItemText(hwnd, IDC_PKEDIT, cfg.keyfile,
2308 sizeof(cfg.keyfile) - 1);
2311 if (HIWORD(wParam) == EN_CHANGE)
2312 GetDlgItemText(hwnd, IDC_CMDEDIT, cfg.remote_cmd,
2313 sizeof(cfg.remote_cmd) - 1);
2316 memset(&of, 0, sizeof(of));
2317 #ifdef OPENFILENAME_SIZE_VERSION_400
2318 of.lStructSize = OPENFILENAME_SIZE_VERSION_400;
2320 of.lStructSize = sizeof(of);
2322 of.hwndOwner = hwnd;
2323 of.lpstrFilter = "All Files\0*\0\0\0";
2324 of.lpstrCustomFilter = NULL;
2325 of.nFilterIndex = 1;
2326 of.lpstrFile = filename;
2327 strcpy(filename, cfg.keyfile);
2328 of.nMaxFile = sizeof(filename);
2329 of.lpstrFileTitle = NULL;
2330 of.lpstrInitialDir = NULL;
2331 of.lpstrTitle = "Select Public Key File";
2333 if (GetOpenFileName(&of)) {
2334 strcpy(cfg.keyfile, filename);
2335 SetDlgItemText(hwnd, IDC_PKEDIT, cfg.keyfile);
2339 cfg.rawcnp = IsDlgButtonChecked(hwnd, IDC_RAWCNP);
2342 cfg.mouse_is_xterm = IsDlgButtonChecked(hwnd, IDC_MBXTERM);
2348 int n = GetDlgItemInt(hwnd, IDC_CCEDIT, &ok, FALSE);
2353 for (i = 0; i < 128; i++)
2354 if (SendDlgItemMessage
2355 (hwnd, IDC_CCLIST, LB_GETSEL, i, 0)) {
2357 cfg.wordness[i] = n;
2358 SendDlgItemMessage(hwnd, IDC_CCLIST,
2359 LB_DELETESTRING, i, 0);
2360 sprintf(str, "%d\t(0x%02X)\t%c\t%d", i, i,
2361 (i >= 0x21 && i != 0x7F) ? i : ' ',
2363 SendDlgItemMessage(hwnd, IDC_CCLIST,
2370 case IDC_BOLDCOLOUR:
2371 if (HIWORD(wParam) == BN_CLICKED ||
2372 HIWORD(wParam) == BN_DOUBLECLICKED) {
2375 IsDlgButtonChecked(hwnd, IDC_BOLDCOLOUR);
2376 SendDlgItemMessage(hwnd, IDC_COLOURLIST, WM_SETREDRAW,
2379 SendDlgItemMessage(hwnd, IDC_COLOURLIST,
2381 if (n != 12 + 10 * cfg.bold_colour) {
2382 for (i = n; i-- > 0;)
2383 SendDlgItemMessage(hwnd, IDC_COLOURLIST,
2384 LB_DELETESTRING, i, 0);
2385 for (i = 0; i < 22; i++)
2386 if (cfg.bold_colour || permcolour[i])
2387 SendDlgItemMessage(hwnd, IDC_COLOURLIST,
2389 (LPARAM) colours[i]);
2391 SendDlgItemMessage(hwnd, IDC_COLOURLIST, WM_SETREDRAW,
2393 InvalidateRect(GetDlgItem(hwnd, IDC_COLOURLIST), NULL,
2398 if (HIWORD(wParam) == BN_CLICKED ||
2399 HIWORD(wParam) == BN_DOUBLECLICKED)
2401 IsDlgButtonChecked(hwnd, IDC_PALETTE);
2403 case IDC_COLOURLIST:
2404 if (HIWORD(wParam) == LBN_DBLCLK ||
2405 HIWORD(wParam) == LBN_SELCHANGE) {
2407 SendDlgItemMessage(hwnd, IDC_COLOURLIST,
2410 if (!cfg.bold_colour)
2411 i = (i < 3 ? i * 2 : i == 3 ? 5 : i * 2 - 2);
2412 SetDlgItemInt(hwnd, IDC_RVALUE, cfg.colours[i][0],
2414 SetDlgItemInt(hwnd, IDC_GVALUE, cfg.colours[i][1],
2416 SetDlgItemInt(hwnd, IDC_BVALUE, cfg.colours[i][2],
2421 if (HIWORD(wParam) == BN_CLICKED ||
2422 HIWORD(wParam) == BN_DOUBLECLICKED) {
2423 static CHOOSECOLOR cc;
2424 static DWORD custom[16] = { 0 }; /* zero initialisers */
2426 SendDlgItemMessage(hwnd, IDC_COLOURLIST,
2429 if (!cfg.bold_colour)
2430 i = (i < 3 ? i * 2 : i == 3 ? 5 : i * 2 - 2);
2431 cc.lStructSize = sizeof(cc);
2432 cc.hwndOwner = hwnd;
2433 cc.hInstance = (HWND) hinst;
2434 cc.lpCustColors = custom;
2436 RGB(cfg.colours[i][0], cfg.colours[i][1],
2438 cc.Flags = CC_FULLOPEN | CC_RGBINIT;
2439 if (ChooseColor(&cc)) {
2441 (unsigned char) (cc.rgbResult & 0xFF);
2443 (unsigned char) (cc.rgbResult >> 8) & 0xFF;
2445 (unsigned char) (cc.rgbResult >> 16) & 0xFF;
2446 SetDlgItemInt(hwnd, IDC_RVALUE, cfg.colours[i][0],
2448 SetDlgItemInt(hwnd, IDC_GVALUE, cfg.colours[i][1],
2450 SetDlgItemInt(hwnd, IDC_BVALUE, cfg.colours[i][2],
2456 if (HIWORD(wParam) == CBN_SELCHANGE) {
2457 int index = SendDlgItemMessage(hwnd, IDC_CODEPAGE,
2458 CB_GETCURSEL, 0, 0);
2459 SendDlgItemMessage(hwnd, IDC_CODEPAGE, CB_GETLBTEXT,
2460 index, (LPARAM)cfg.line_codepage);
2463 case IDC_VTXWINDOWS:
2469 (IsDlgButtonChecked(hwnd, IDC_VTXWINDOWS) ? VT_XWINDOWS
2470 : IsDlgButtonChecked(hwnd,
2471 IDC_VTOEMANSI) ? VT_OEMANSI :
2472 IsDlgButtonChecked(hwnd,
2473 IDC_VTOEMONLY) ? VT_OEMONLY :
2474 IsDlgButtonChecked(hwnd,
2475 IDC_VTUNICODE) ? VT_UNICODE :
2478 case IDC_X11_FORWARD:
2479 if (HIWORD(wParam) == BN_CLICKED ||
2480 HIWORD(wParam) == BN_DOUBLECLICKED)
2482 IsDlgButtonChecked(hwnd, IDC_X11_FORWARD);
2485 if (HIWORD(wParam) == BN_CLICKED ||
2486 HIWORD(wParam) == BN_DOUBLECLICKED)
2487 cfg.lport_acceptall =
2488 IsDlgButtonChecked(hwnd, IDC_LPORT_ALL);
2490 case IDC_X11_DISPLAY:
2491 if (HIWORD(wParam) == EN_CHANGE)
2492 GetDlgItemText(hwnd, IDC_X11_DISPLAY, cfg.x11_display,
2493 sizeof(cfg.x11_display) - 1);
2496 if (HIWORD(wParam) == BN_CLICKED ||
2497 HIWORD(wParam) == BN_DOUBLECLICKED) {
2498 char str[sizeof(cfg.portfwd)];
2500 if (IsDlgButtonChecked(hwnd, IDC_PFWDLOCAL))
2504 GetDlgItemText(hwnd, IDC_SPORTEDIT, str+1,
2508 "You need to specify a source port number",
2509 "PuTTY Error", MB_OK | MB_ICONERROR);
2512 p = str + strlen(str);
2514 GetDlgItemText(hwnd, IDC_DPORTEDIT, p,
2515 sizeof(str) - 1 - (p - str));
2516 if (!*p || !strchr(p, ':')) {
2518 "You need to specify a destination address\n"
2519 "in the form \"host.name:port\"",
2520 "PuTTY Error", MB_OK | MB_ICONERROR);
2529 if ((p - cfg.portfwd) + strlen(str) + 2 <
2530 sizeof(cfg.portfwd)) {
2532 p[strlen(str) + 1] = '\0';
2533 SendDlgItemMessage(hwnd, IDC_PFWDLIST, LB_ADDSTRING,
2535 SetDlgItemText(hwnd, IDC_SPORTEDIT, "");
2536 SetDlgItemText(hwnd, IDC_DPORTEDIT, "");
2538 MessageBox(hwnd, "Too many forwardings",
2539 "PuTTY Error", MB_OK | MB_ICONERROR);
2543 case IDC_PFWDREMOVE:
2544 if (HIWORD(wParam) != BN_CLICKED &&
2545 HIWORD(wParam) != BN_DOUBLECLICKED) break;
2546 i = SendDlgItemMessage(hwnd, IDC_PFWDLIST,
2547 LB_GETCURSEL, 0, 0);
2553 SendDlgItemMessage(hwnd, IDC_PFWDLIST, LB_DELETESTRING,
2585 /* Grrr Explorer will maximize Dialogs! */
2587 if (wParam == SIZE_MAXIMIZED)
2593 * Handle application-defined messages eg. DragListBox
2595 /* First find out what the number is (once). */
2596 if (draglistmsg == WM_NULL)
2597 draglistmsg = RegisterWindowMessage (DRAGLISTMSGSTRING);
2599 if (msg == draglistmsg) {
2600 /* Only process once dialog is fully formed. */
2601 if (GetWindowLong(hwnd, GWL_USERDATA) == 1) switch (LOWORD(wParam)) {
2602 case IDC_CIPHERLIST:
2603 return handle_prefslist(&cipherlist,
2604 cfg.ssh_cipherlist, CIPHER_MAX,
2605 1, hwnd, wParam, lParam);
2614 static int CALLBACK MainDlgProc(HWND hwnd, UINT msg,
2615 WPARAM wParam, LPARAM lParam)
2617 if (msg == WM_COMMAND && LOWORD(wParam) == IDOK) {
2619 if (msg == WM_COMMAND && LOWORD(wParam) == IDCX_ABOUT) {
2620 EnableWindow(hwnd, 0);
2621 DialogBox(hinst, MAKEINTRESOURCE(IDD_ABOUTBOX), hwnd, AboutProc);
2622 EnableWindow(hwnd, 1);
2623 SetActiveWindow(hwnd);
2625 return GenericMainDlgProc(hwnd, msg, wParam, lParam, 0);
2628 static int CALLBACK ReconfDlgProc(HWND hwnd, UINT msg,
2629 WPARAM wParam, LPARAM lParam)
2631 return GenericMainDlgProc(hwnd, msg, wParam, lParam, 1);
2634 void defuse_showwindow(void)
2637 * Work around the fact that the app's first call to ShowWindow
2638 * will ignore the default in favour of the shell-provided
2643 hwnd = CreateDialog(hinst, MAKEINTRESOURCE(IDD_ABOUTBOX),
2645 ShowWindow(hwnd, SW_HIDE);
2646 SetActiveWindow(hwnd);
2647 DestroyWindow(hwnd);
2656 savedsession[0] = '\0';
2658 DialogBox(hinst, MAKEINTRESOURCE(IDD_MAINBOX), NULL, MainDlgProc);
2659 get_sesslist(FALSE);
2664 int do_reconfig(HWND hwnd)
2669 backup_cfg = cfg; /* structure copy */
2671 DialogBox(hinst, MAKEINTRESOURCE(IDD_RECONF), hwnd, ReconfDlgProc);
2673 cfg = backup_cfg; /* structure copy */
2678 void logevent(char *string)
2683 if (nevents >= negsize) {
2685 events = srealloc(events, negsize * sizeof(*events));
2689 strftime(timebuf, sizeof(timebuf), "%Y-%m-%d %H:%M:%S\t",
2692 events[nevents] = smalloc(strlen(timebuf) + strlen(string) + 1);
2693 strcpy(events[nevents], timebuf);
2694 strcat(events[nevents], string);
2697 SendDlgItemMessage(logbox, IDN_LIST, LB_ADDSTRING,
2698 0, (LPARAM) events[nevents]);
2699 count = SendDlgItemMessage(logbox, IDN_LIST, LB_GETCOUNT, 0, 0);
2700 SendDlgItemMessage(logbox, IDN_LIST, LB_SETTOPINDEX, count - 1, 0);
2705 void showeventlog(HWND hwnd)
2708 logbox = CreateDialog(hinst, MAKEINTRESOURCE(IDD_LOGBOX),
2710 ShowWindow(logbox, SW_SHOWNORMAL);
2712 SetActiveWindow(logbox);
2715 void showabout(HWND hwnd)
2717 DialogBox(hinst, MAKEINTRESOURCE(IDD_ABOUTBOX), hwnd, AboutProc);
2720 void verify_ssh_host_key(char *host, int port, char *keytype,
2721 char *keystr, char *fingerprint)
2725 static const char absentmsg[] =
2726 "The server's host key is not cached in the registry. You\n"
2727 "have no guarantee that the server is the computer you\n"
2729 "The server's key fingerprint is:\n"
2731 "If you trust this host, hit Yes to add the key to\n"
2732 "PuTTY's cache and carry on connecting.\n"
2733 "If you want to carry on connecting just once, without\n"
2734 "adding the key to the cache, hit No.\n"
2735 "If you do not trust this host, hit Cancel to abandon the\n"
2738 static const char wrongmsg[] =
2739 "WARNING - POTENTIAL SECURITY BREACH!\n"
2741 "The server's host key does not match the one PuTTY has\n"
2742 "cached in the registry. This means that either the\n"
2743 "server administrator has changed the host key, or you\n"
2744 "have actually connected to another computer pretending\n"
2745 "to be the server.\n"
2746 "The new key fingerprint is:\n"
2748 "If you were expecting this change and trust the new key,\n"
2749 "hit Yes to update PuTTY's cache and continue connecting.\n"
2750 "If you want to carry on connecting but without updating\n"
2751 "the cache, hit No.\n"
2752 "If you want to abandon the connection completely, hit\n"
2753 "Cancel. Hitting Cancel is the ONLY guaranteed safe\n" "choice.\n";
2755 static const char mbtitle[] = "PuTTY Security Alert";
2758 /* sensible fingerprint max size */
2759 (sizeof(absentmsg) > sizeof(wrongmsg) ?
2760 sizeof(absentmsg) : sizeof(wrongmsg))];
2763 * Verify the key against the registry.
2765 ret = verify_host_key(host, port, keytype, keystr);
2767 if (ret == 0) /* success - key matched OK */
2769 if (ret == 2) { /* key was different */
2771 sprintf(message, wrongmsg, fingerprint);
2772 mbret = MessageBox(NULL, message, mbtitle,
2773 MB_ICONWARNING | MB_YESNOCANCEL);
2775 store_host_key(host, port, keytype, keystr);
2776 if (mbret == IDCANCEL)
2779 if (ret == 1) { /* key was absent */
2781 sprintf(message, absentmsg, fingerprint);
2782 mbret = MessageBox(NULL, message, mbtitle,
2783 MB_ICONWARNING | MB_YESNOCANCEL);
2785 store_host_key(host, port, keytype, keystr);
2786 if (mbret == IDCANCEL)
2792 * Ask whether the selected cipher is acceptable (since it was
2793 * below the configured 'warn' threshold).
2794 * cs: 0 = both ways, 1 = client->server, 2 = server->client
2796 void askcipher(char *ciphername, int cs)
2798 static const char mbtitle[] = "PuTTY Security Alert";
2799 static const char msg[] =
2800 "The first %.35scipher supported by the server\n"
2801 "is %.64s, which is below the configured\n"
2802 "warning threshold.\n"
2803 "Do you want to continue with this connection?\n";
2804 /* guessed cipher name + type max length */
2805 char message[100 + sizeof(msg)];
2808 sprintf(message, msg,
2810 (cs == 1) ? "client-to-server " :
2811 "server-to-client ",
2813 mbret = MessageBox(NULL, message, mbtitle,
2814 MB_ICONWARNING | MB_YESNO);
2822 * Ask whether to wipe a session log file before writing to it.
2823 * Returns 2 for wipe, 1 for append, 0 for cancel (don't log).
2825 int askappend(char *filename)
2827 static const char mbtitle[] = "PuTTY Log to File";
2828 static const char msgtemplate[] =
2829 "The session log file \"%.*s\" already exists.\n"
2830 "You can overwrite it with a new session log,\n"
2831 "append your session log to the end of it,\n"
2832 "or disable session logging for this session.\n"
2833 "Hit Yes to wipe the file, No to append to it,\n"
2834 "or Cancel to disable logging.";
2835 char message[sizeof(msgtemplate) + FILENAME_MAX];
2837 if (cfg.logxfovr != LGXF_ASK) {
2838 return ((cfg.logxfovr == LGXF_OVR) ? 2 : 1);
2840 sprintf(message, msgtemplate, FILENAME_MAX, filename);
2842 mbret = MessageBox(NULL, message, mbtitle,
2843 MB_ICONQUESTION | MB_YESNOCANCEL);
2846 else if (mbret == IDNO)