15 static char **events = NULL;
16 static int nevents = 0, negsize = 0;
19 static int sesslist_has_focus;
21 void force_normal(HWND hwnd)
23 static int recurse = 0;
31 wp.length = sizeof(wp);
32 if (GetWindowPlacement(hwnd, &wp) && wp.showCmd == SW_SHOWMAXIMIZED) {
33 wp.showCmd = SW_SHOWNORMAL;
34 SetWindowPlacement(hwnd, &wp);
39 static void MyGetDlgItemInt(HWND hwnd, int id, int *result)
43 n = GetDlgItemInt(hwnd, id, &ok, FALSE);
48 static void MyGetDlgItemFlt(HWND hwnd, int id, int *result, int scale)
52 ok = GetDlgItemText(hwnd, id, text, sizeof(text) - 1);
54 *result = (int) (scale * atof(text));
57 static void MySetDlgItemFlt(HWND hwnd, int id, double value)
60 sprintf(text, "%g", value);
61 SetDlgItemText(hwnd, id, text);
64 static int CALLBACK LogProc(HWND hwnd, UINT msg,
65 WPARAM wParam, LPARAM lParam)
72 static int tabs[4] = { 78, 108 };
73 SendDlgItemMessage(hwnd, IDN_LIST, LB_SETTABSTOPS, 2,
76 for (i = 0; i < nevents; i++)
77 SendDlgItemMessage(hwnd, IDN_LIST, LB_ADDSTRING,
78 0, (LPARAM) events[i]);
81 switch (LOWORD(wParam)) {
85 SetActiveWindow(GetParent(hwnd));
89 if (HIWORD(wParam) == BN_CLICKED ||
90 HIWORD(wParam) == BN_DOUBLECLICKED) {
93 selcount = SendDlgItemMessage(hwnd, IDN_LIST,
94 LB_GETSELCOUNT, 0, 0);
95 if (selcount == 0) { /* don't even try to copy zero items */
100 selitems = smalloc(selcount * sizeof(int));
102 int count = SendDlgItemMessage(hwnd, IDN_LIST,
109 static unsigned char sel_nl[] = SEL_NL;
111 if (count == 0) { /* can't copy zero stuff */
117 for (i = 0; i < count; i++)
119 strlen(events[selitems[i]]) + sizeof(sel_nl);
121 clipdata = smalloc(size);
124 for (i = 0; i < count; i++) {
125 char *q = events[selitems[i]];
126 int qlen = strlen(q);
129 memcpy(p, sel_nl, sizeof(sel_nl));
132 write_aclip(clipdata, size, TRUE);
137 for (i = 0; i < nevents; i++)
138 SendDlgItemMessage(hwnd, IDN_LIST, LB_SETSEL,
147 SetActiveWindow(GetParent(hwnd));
154 static int CALLBACK LicenceProc(HWND hwnd, UINT msg,
155 WPARAM wParam, LPARAM lParam)
161 switch (LOWORD(wParam)) {
174 static int CALLBACK AboutProc(HWND hwnd, UINT msg,
175 WPARAM wParam, LPARAM lParam)
179 SetDlgItemText(hwnd, IDA_VERSION, ver);
182 switch (LOWORD(wParam)) {
185 EndDialog(hwnd, TRUE);
188 EnableWindow(hwnd, 0);
189 DialogBox(hinst, MAKEINTRESOURCE(IDD_LICENCEBOX),
191 EnableWindow(hwnd, 1);
192 SetActiveWindow(hwnd);
196 /* Load web browser */
197 ShellExecute(hwnd, "open",
198 "http://www.chiark.greenend.org.uk/~sgtatham/putty/",
199 0, 0, SW_SHOWDEFAULT);
204 EndDialog(hwnd, TRUE);
211 * Null dialog procedure.
213 static int CALLBACK NullDlgProc(HWND hwnd, UINT msg,
214 WPARAM wParam, LPARAM lParam)
219 static char savedsession[2048];
222 IDC_ABOUT, IDCX_TVSTATIC, IDCX_TREEVIEW, controlstartvalue,
363 appearancepanelstart,
364 IDC_TITLE_APPEARANCE,
384 connectionpanelstart,
385 IDC_TITLE_CONNECTION,
486 translationpanelstart,
487 IDC_TITLE_TRANSLATION,
488 IDC_BOX_TRANSLATION1,
489 IDC_BOX_TRANSLATION2,
511 static const char *const colours[] = {
512 "Default Foreground", "Default Bold Foreground",
513 "Default Background", "Default Bold Background",
514 "Cursor Text", "Cursor Colour",
515 "ANSI Black", "ANSI Black Bold",
516 "ANSI Red", "ANSI Red Bold",
517 "ANSI Green", "ANSI Green Bold",
518 "ANSI Yellow", "ANSI Yellow Bold",
519 "ANSI Blue", "ANSI Blue Bold",
520 "ANSI Magenta", "ANSI Magenta Bold",
521 "ANSI Cyan", "ANSI Cyan Bold",
522 "ANSI White", "ANSI White Bold"
524 static const int permcolour[] = {
525 TRUE, FALSE, TRUE, FALSE, TRUE, TRUE,
526 TRUE, FALSE, TRUE, FALSE, TRUE, FALSE, TRUE, FALSE,
527 TRUE, FALSE, TRUE, FALSE, TRUE, FALSE, TRUE, FALSE
530 static void fmtfont(char *buf)
532 sprintf(buf, "Font: %s, ", cfg.font);
534 strcat(buf, "bold, ");
535 if (cfg.fontheight == 0)
536 strcat(buf, "default height");
538 sprintf(buf + strlen(buf), "%d-point",
539 (cfg.fontheight < 0 ? -cfg.fontheight : cfg.fontheight));
542 /* 2nd arg: NZ => don't redraw session list (use when loading
544 static void init_dlg_ctrls(HWND hwnd, int keepsess)
547 char fontstatic[256];
549 SetDlgItemText(hwnd, IDC_HOST, cfg.host);
550 SetDlgItemText(hwnd, IDC_SESSEDIT, savedsession);
553 n = SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_GETCOUNT, 0, 0);
554 for (i = n; i-- > 0;)
555 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_DELETESTRING, i, 0);
556 for (i = 0; i < nsessions; i++)
557 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_ADDSTRING,
558 0, (LPARAM) (sessions[i]));
560 SetDlgItemInt(hwnd, IDC_PORT, cfg.port, FALSE);
561 CheckRadioButton(hwnd, IDC_PROTRAW, IDC_PROTSSH,
562 cfg.protocol == PROT_SSH ? IDC_PROTSSH :
563 cfg.protocol == PROT_TELNET ? IDC_PROTTELNET :
565 PROT_RLOGIN ? IDC_PROTRLOGIN : IDC_PROTRAW);
566 SetDlgItemInt(hwnd, IDC_PINGEDIT, cfg.ping_interval, FALSE);
568 CheckRadioButton(hwnd, IDC_DEL008, IDC_DEL127,
569 cfg.bksp_is_delete ? IDC_DEL127 : IDC_DEL008);
570 CheckRadioButton(hwnd, IDC_HOMETILDE, IDC_HOMERXVT,
571 cfg.rxvt_homeend ? IDC_HOMERXVT : IDC_HOMETILDE);
572 CheckRadioButton(hwnd, IDC_FUNCTILDE, IDC_FUNCSCO,
573 cfg.funky_type == 0 ? IDC_FUNCTILDE :
574 cfg.funky_type == 1 ? IDC_FUNCLINUX :
575 cfg.funky_type == 2 ? IDC_FUNCXTERM :
576 cfg.funky_type == 3 ? IDC_FUNCVT400 :
577 cfg.funky_type == 4 ? IDC_FUNCVT100P :
578 cfg.funky_type == 5 ? IDC_FUNCSCO : IDC_FUNCTILDE);
579 CheckDlgButton(hwnd, IDC_NOAPPLICC, cfg.no_applic_c);
580 CheckDlgButton(hwnd, IDC_NOAPPLICK, cfg.no_applic_k);
581 CheckRadioButton(hwnd, IDC_CURNORMAL, IDC_CURAPPLIC,
582 cfg.app_cursor ? IDC_CURAPPLIC : IDC_CURNORMAL);
583 CheckRadioButton(hwnd, IDC_KPNORMAL, IDC_KPNH,
584 cfg.nethack_keypad ? IDC_KPNH :
585 cfg.app_keypad ? IDC_KPAPPLIC : IDC_KPNORMAL);
586 CheckDlgButton(hwnd, IDC_ALTF4, cfg.alt_f4);
587 CheckDlgButton(hwnd, IDC_ALTSPACE, cfg.alt_space);
588 CheckDlgButton(hwnd, IDC_ALTONLY, cfg.alt_only);
589 CheckDlgButton(hwnd, IDC_COMPOSEKEY, cfg.compose_key);
590 CheckDlgButton(hwnd, IDC_CTRLALTKEYS, cfg.ctrlaltkeys);
591 CheckDlgButton(hwnd, IDC_TELNETKEY, cfg.telnet_keyboard);
592 CheckRadioButton(hwnd, IDC_ECHOBACKEND, IDC_ECHONO,
593 cfg.localecho == LD_BACKEND ? IDC_ECHOBACKEND :
594 cfg.localecho == LD_YES ? IDC_ECHOYES : IDC_ECHONO);
595 CheckRadioButton(hwnd, IDC_EDITBACKEND, IDC_EDITNO,
596 cfg.localedit == LD_BACKEND ? IDC_EDITBACKEND :
597 cfg.localedit == LD_YES ? IDC_EDITYES : IDC_EDITNO);
598 SetDlgItemText(hwnd, IDC_ANSWEREDIT, cfg.answerback);
599 CheckDlgButton(hwnd, IDC_ALWAYSONTOP, cfg.alwaysontop);
600 CheckDlgButton(hwnd, IDC_SCROLLKEY, cfg.scroll_on_key);
601 CheckDlgButton(hwnd, IDC_SCROLLDISP, cfg.scroll_on_disp);
603 CheckDlgButton(hwnd, IDC_WRAPMODE, cfg.wrap_mode);
604 CheckDlgButton(hwnd, IDC_DECOM, cfg.dec_om);
605 CheckDlgButton(hwnd, IDC_LFHASCR, cfg.lfhascr);
606 SetDlgItemInt(hwnd, IDC_ROWSEDIT, cfg.height, FALSE);
607 SetDlgItemInt(hwnd, IDC_COLSEDIT, cfg.width, FALSE);
608 SetDlgItemInt(hwnd, IDC_SAVEEDIT, cfg.savelines, FALSE);
610 SetDlgItemText(hwnd, IDC_FONTSTATIC, fontstatic);
611 CheckRadioButton(hwnd, IDC_BELL_DISABLED, IDC_BELL_VISUAL,
612 cfg.beep == BELL_DISABLED ? IDC_BELL_DISABLED :
613 cfg.beep == BELL_DEFAULT ? IDC_BELL_DEFAULT :
614 cfg.beep == BELL_WAVEFILE ? IDC_BELL_WAVEFILE :
616 BELL_VISUAL ? IDC_BELL_VISUAL : IDC_BELL_DEFAULT);
617 SetDlgItemText(hwnd, IDC_BELL_WAVEEDIT, cfg.bell_wavefile);
618 CheckDlgButton(hwnd, IDC_BELLOVL, cfg.bellovl);
619 SetDlgItemInt(hwnd, IDC_BELLOVLN, cfg.bellovl_n, FALSE);
620 MySetDlgItemFlt(hwnd, IDC_BELLOVLT, cfg.bellovl_t / 1000.0);
621 MySetDlgItemFlt(hwnd, IDC_BELLOVLS, cfg.bellovl_s / 1000.0);
623 CheckDlgButton(hwnd, IDC_BCE, cfg.bce);
624 CheckDlgButton(hwnd, IDC_BLINKTEXT, cfg.blinktext);
626 SetDlgItemText(hwnd, IDC_WINEDIT, cfg.wintitle);
627 CheckDlgButton(hwnd, IDC_WINNAME, cfg.win_name_always);
628 CheckDlgButton(hwnd, IDC_HIDEMOUSE, cfg.hide_mouseptr);
629 CheckDlgButton(hwnd, IDC_SUNKENEDGE, cfg.sunken_edge);
630 CheckRadioButton(hwnd, IDC_CURBLOCK, IDC_CURVERT,
631 cfg.cursor_type == 0 ? IDC_CURBLOCK :
632 cfg.cursor_type == 1 ? IDC_CURUNDER : IDC_CURVERT);
633 CheckDlgButton(hwnd, IDC_BLINKCUR, cfg.blink_cur);
634 CheckDlgButton(hwnd, IDC_SCROLLBAR, cfg.scrollbar);
635 CheckDlgButton(hwnd, IDC_LOCKSIZE, cfg.locksize);
636 CheckRadioButton(hwnd, IDC_COEALWAYS, IDC_COENORMAL,
637 cfg.close_on_exit == COE_NORMAL ? IDC_COENORMAL :
639 COE_NEVER ? IDC_COENEVER : IDC_COEALWAYS);
640 CheckDlgButton(hwnd, IDC_CLOSEWARN, cfg.warn_on_close);
642 SetDlgItemText(hwnd, IDC_TTEDIT, cfg.termtype);
643 SetDlgItemText(hwnd, IDC_TSEDIT, cfg.termspeed);
644 SetDlgItemText(hwnd, IDC_R_TSEDIT, cfg.termspeed);
645 SetDlgItemText(hwnd, IDC_RLLUSEREDIT, cfg.localusername);
646 SetDlgItemText(hwnd, IDC_LOGEDIT, cfg.username);
647 SetDlgItemText(hwnd, IDC_LGFEDIT, cfg.logfilename);
648 CheckRadioButton(hwnd, IDC_LSTATOFF, IDC_LSTATRAW,
649 cfg.logtype == 0 ? IDC_LSTATOFF :
650 cfg.logtype == 1 ? IDC_LSTATASCII : IDC_LSTATRAW);
651 CheckRadioButton(hwnd, IDC_LSTATXOVR, IDC_LSTATXASK,
652 cfg.logxfovr == LGXF_OVR ? IDC_LSTATXOVR :
653 cfg.logxfovr == LGXF_ASK ? IDC_LSTATXASK :
656 char *p = cfg.environmt;
658 SendDlgItemMessage(hwnd, IDC_ENVLIST, LB_ADDSTRING, 0,
663 CheckRadioButton(hwnd, IDC_EMBSD, IDC_EMRFC,
664 cfg.rfc_environ ? IDC_EMRFC : IDC_EMBSD);
665 CheckRadioButton(hwnd, IDC_TPASSIVE, IDC_TACTIVE,
666 cfg.passive_telnet ? IDC_TPASSIVE : IDC_TACTIVE);
668 SetDlgItemText(hwnd, IDC_TTEDIT, cfg.termtype);
669 SetDlgItemText(hwnd, IDC_LOGEDIT, cfg.username);
670 CheckDlgButton(hwnd, IDC_NOPTY, cfg.nopty);
671 CheckDlgButton(hwnd, IDC_COMPRESS, cfg.compression);
672 CheckDlgButton(hwnd, IDC_BUGGYMAC, cfg.buggymac);
673 CheckDlgButton(hwnd, IDC_AGENTFWD, cfg.agentfwd);
674 CheckRadioButton(hwnd, IDC_CIPHER3DES, IDC_CIPHERAES,
675 cfg.cipher == CIPHER_BLOWFISH ? IDC_CIPHERBLOWF :
676 cfg.cipher == CIPHER_DES ? IDC_CIPHERDES :
677 cfg.cipher == CIPHER_AES ? IDC_CIPHERAES :
679 CheckRadioButton(hwnd, IDC_SSHPROT1, IDC_SSHPROT2,
680 cfg.sshprot == 1 ? IDC_SSHPROT1 : IDC_SSHPROT2);
681 CheckDlgButton(hwnd, IDC_AUTHTIS, cfg.try_tis_auth);
682 SetDlgItemText(hwnd, IDC_PKEDIT, cfg.keyfile);
683 SetDlgItemText(hwnd, IDC_CMDEDIT, cfg.remote_cmd);
685 CheckRadioButton(hwnd, IDC_MBWINDOWS, IDC_MBXTERM,
686 cfg.mouse_is_xterm ? IDC_MBXTERM : IDC_MBWINDOWS);
687 CheckDlgButton(hwnd, IDC_RAWCNP, cfg.rawcnp);
689 static int tabs[4] = { 25, 61, 96, 128 };
690 SendDlgItemMessage(hwnd, IDC_CCLIST, LB_SETTABSTOPS, 4,
693 for (i = 0; i < 128; i++) {
695 sprintf(str, "%d\t(0x%02X)\t%c\t%d", i, i,
696 (i >= 0x21 && i != 0x7F) ? i : ' ', cfg.wordness[i]);
697 SendDlgItemMessage(hwnd, IDC_CCLIST, LB_ADDSTRING, 0,
701 CheckDlgButton(hwnd, IDC_BOLDCOLOUR, cfg.bold_colour);
702 CheckDlgButton(hwnd, IDC_PALETTE, cfg.try_palette);
705 n = SendDlgItemMessage(hwnd, IDC_COLOURLIST, LB_GETCOUNT, 0, 0);
706 for (i = n; i-- > 0;)
707 SendDlgItemMessage(hwnd, IDC_COLOURLIST,
708 LB_DELETESTRING, i, 0);
709 for (i = 0; i < 22; i++)
710 if (cfg.bold_colour || permcolour[i])
711 SendDlgItemMessage(hwnd, IDC_COLOURLIST, LB_ADDSTRING, 0,
712 (LPARAM) colours[i]);
714 SendDlgItemMessage(hwnd, IDC_COLOURLIST, LB_SETCURSEL, 0, 0);
715 SetDlgItemInt(hwnd, IDC_RVALUE, cfg.colours[0][0], FALSE);
716 SetDlgItemInt(hwnd, IDC_GVALUE, cfg.colours[0][1], FALSE);
717 SetDlgItemInt(hwnd, IDC_BVALUE, cfg.colours[0][2], FALSE);
719 SetDlgItemText(hwnd, IDC_CODEPAGE, cfg.line_codepage);
720 CheckRadioButton(hwnd, IDC_VTXWINDOWS, IDC_VTUNICODE,
721 cfg.vtmode == VT_XWINDOWS ? IDC_VTXWINDOWS :
722 cfg.vtmode == VT_OEMANSI ? IDC_VTOEMANSI :
723 cfg.vtmode == VT_OEMONLY ? IDC_VTOEMONLY :
724 cfg.vtmode == VT_UNICODE ? IDC_VTUNICODE :
727 CheckDlgButton(hwnd, IDC_X11_FORWARD, cfg.x11_forward);
728 SetDlgItemText(hwnd, IDC_X11_DISPLAY, cfg.x11_display);
731 struct treeview_faff {
736 static HTREEITEM treeview_insert(struct treeview_faff *faff,
737 int level, char *text)
742 ins.hParent = (level > 0 ? faff->lastat[level - 1] : TVI_ROOT);
743 ins.hInsertAfter = faff->lastat[level];
744 #if _WIN32_IE >= 0x0400 && defined NONAMELESSUNION
745 #define INSITEM DUMMYUNIONNAME.item
749 ins.INSITEM.mask = TVIF_TEXT;
750 ins.INSITEM.pszText = text;
751 newitem = TreeView_InsertItem(faff->treeview, &ins);
753 TreeView_Expand(faff->treeview, faff->lastat[level - 1],
755 faff->lastat[level] = newitem;
756 for (i = level + 1; i < 4; i++)
757 faff->lastat[i] = NULL;
762 * Create the panelfuls of controls in the configuration box.
764 static void create_controls(HWND hwnd, int dlgtype, int panel)
766 if (panel == sessionpanelstart) {
767 /* The Session panel. Accelerators used: [acgo] nprtih elsd w */
769 ctlposinit(&cp, hwnd, 80, 3, 13);
770 bartitle(&cp, "Basic options for your PuTTY session",
773 beginbox(&cp, "Specify your connection by host name",
776 "Host &Name", IDC_HOSTSTATIC, IDC_HOST, 75,
777 "&Port", IDC_PORTSTATIC, IDC_PORT, 25, NULL);
778 if (backends[3].backend == NULL) {
779 /* this is PuTTYtel, so only three protocols available */
780 radioline(&cp, "Protocol:", IDC_PROTSTATIC, 3,
782 "&Telnet", IDC_PROTTELNET,
783 "Rlog&in", IDC_PROTRLOGIN, NULL);
785 radioline(&cp, "Protocol:", IDC_PROTSTATIC, 4,
787 "&Telnet", IDC_PROTTELNET,
788 "Rlog&in", IDC_PROTRLOGIN,
797 beginbox(&cp, "Load, save or delete a stored session",
799 sesssaver(&cp, "Sav&ed Sessions",
800 IDC_SESSSTATIC, IDC_SESSEDIT, IDC_SESSLIST,
801 "&Load", IDC_SESSLOAD,
802 "&Save", IDC_SESSSAVE, "&Delete", IDC_SESSDEL, NULL);
805 beginbox(&cp, NULL, IDC_BOX_SESSION3);
806 radioline(&cp, "Close &window on exit:", IDC_CLOSEEXIT, 4,
807 "Always", IDC_COEALWAYS,
808 "Never", IDC_COENEVER,
809 "Only on clean exit", IDC_COENORMAL, NULL);
813 if (panel == loggingpanelstart) {
814 /* The Logging panel. Accelerators used: [acgo] tplfwe */
816 ctlposinit(&cp, hwnd, 80, 3, 13);
817 bartitle(&cp, "Options controlling session logging",
819 beginbox(&cp, NULL, IDC_BOX_LOGGING1);
821 "Session logging:", IDC_LSTATSTATIC,
822 "Logging &turned off completely", IDC_LSTATOFF,
823 "Log &printable output only", IDC_LSTATASCII,
824 "&Log all session output", IDC_LSTATRAW, NULL);
825 editbutton(&cp, "Log &file name:",
826 IDC_LGFSTATIC, IDC_LGFEDIT, "Bro&wse...",
829 "What to do if the log file already &exists:",
830 IDC_LSTATXIST, "Always overwrite it", IDC_LSTATXOVR,
831 "Always append to the end of it", IDC_LSTATXAPN,
832 "Ask the user every time", IDC_LSTATXASK, NULL);
836 if (panel == terminalpanelstart) {
837 /* The Terminal panel. Accelerators used: [acgo] wdlen hts */
839 ctlposinit(&cp, hwnd, 80, 3, 13);
840 bartitle(&cp, "Options controlling the terminal emulation",
842 beginbox(&cp, "Set various terminal options", IDC_BOX_TERMINAL1);
843 checkbox(&cp, "Auto &wrap mode initially on", IDC_WRAPMODE);
844 checkbox(&cp, "&DEC Origin Mode initially on", IDC_DECOM);
845 checkbox(&cp, "Implicit CR in every &LF", IDC_LFHASCR);
846 checkbox(&cp, "Use background colour to &erase screen", IDC_BCE);
847 checkbox(&cp, "Enable bli&nking text", IDC_BLINKTEXT);
849 "An&swerback to ^E:", IDC_ANSWERBACK,
850 IDC_ANSWEREDIT, 100, NULL);
853 beginbox(&cp, "Line discipline options", IDC_BOX_TERMINAL2);
854 radioline(&cp, "Local ec&ho:", IDC_ECHOSTATIC, 3,
855 "Auto", IDC_ECHOBACKEND,
856 "Force on", IDC_ECHOYES, "Force off", IDC_ECHONO, NULL);
857 radioline(&cp, "Local line edi&ting:", IDC_EDITSTATIC, 3,
858 "Auto", IDC_EDITBACKEND,
859 "Force on", IDC_EDITYES, "Force off", IDC_EDITNO, NULL);
863 if (panel == bellpanelstart) {
864 /* The Bell panel. Accelerators used: [acgo] bdsm wt */
866 ctlposinit(&cp, hwnd, 80, 3, 13);
867 bartitle(&cp, "Options controlling the terminal bell",
869 beginbox(&cp, "Set the style of bell", IDC_BOX_BELL1);
871 "Action to happen when a &bell occurs:", IDC_BELLSTATIC,
872 "None (bell disabled)", IDC_BELL_DISABLED,
873 "Play Windows Default Sound", IDC_BELL_DEFAULT,
874 "Play a custom sound file", IDC_BELL_WAVEFILE,
875 "Visual bell (flash window)", IDC_BELL_VISUAL, NULL);
876 editbutton(&cp, "Custom sound file to play as a bell:",
877 IDC_BELL_WAVESTATIC, IDC_BELL_WAVEEDIT,
878 "Bro&wse...", IDC_BELL_WAVEBROWSE);
880 beginbox(&cp, "Control the bell overload behaviour",
882 checkbox(&cp, "Bell is temporarily &disabled when over-used",
884 staticedit(&cp, "Over-use means this &many bells...",
885 IDC_BELLOVLNSTATIC, IDC_BELLOVLN, 20);
886 staticedit(&cp, "... in &this many seconds",
887 IDC_BELLOVLTSTATIC, IDC_BELLOVLT, 20);
889 "The bell is re-enabled after a few seconds of silence.",
891 staticedit(&cp, "Seconds of &silence required", IDC_BELLOVLSSTATIC,
896 if (panel == keyboardpanelstart) {
897 /* The Keyboard panel. Accelerators used: [acgo] bhf ruyntd */
899 ctlposinit(&cp, hwnd, 80, 3, 13);
901 bartitle(&cp, "Options controlling the effects of keys",
904 beginbox(&cp, "Change the sequences sent by:", IDC_BOX_KEYBOARD1);
905 radioline(&cp, "The &Backspace key", IDC_DELSTATIC, 2,
906 "Control-H", IDC_DEL008,
907 "Control-? (127)", IDC_DEL127, NULL);
908 radioline(&cp, "The &Home and End keys", IDC_HOMESTATIC, 2,
909 "Standard", IDC_HOMETILDE, "rxvt", IDC_HOMERXVT, NULL);
910 radioline(&cp, "The &Function keys and keypad", IDC_FUNCSTATIC, 3,
911 "ESC[n~", IDC_FUNCTILDE,
912 "Linux", IDC_FUNCLINUX,
913 "Xterm R6", IDC_FUNCXTERM,
914 "VT400", IDC_FUNCVT400,
915 "VT100+", IDC_FUNCVT100P, "SCO", IDC_FUNCSCO, NULL);
917 beginbox(&cp, "Application keypad settings:", IDC_BOX_KEYBOARD2);
919 "Application c&ursor keys totally disabled",
921 radioline(&cp, "Initial state of cu&rsor keys:", IDC_CURSTATIC, 2,
922 "Normal", IDC_CURNORMAL,
923 "Application", IDC_CURAPPLIC, NULL);
925 "Application ke&ypad keys totally disabled",
927 radioline(&cp, "Initial state of &numeric keypad:", IDC_KPSTATIC,
928 3, "Normal", IDC_KPNORMAL, "Application", IDC_KPAPPLIC,
929 "NetHack", IDC_KPNH, NULL);
931 beginbox(&cp, "Enable extra keyboard features:",
933 checkbox(&cp, "AltGr ac&ts as Compose key", IDC_COMPOSEKEY);
934 checkbox(&cp, "Control-Alt is &different from AltGr",
939 if (panel == windowpanelstart) {
940 /* The Window panel. Accelerators used: [acgo] rmz sdkp w4ylt */
942 ctlposinit(&cp, hwnd, 80, 3, 13);
943 bartitle(&cp, "Options controlling PuTTY's window",
945 beginbox(&cp, "Set the size of the window", IDC_BOX_WINDOW1);
947 "&Rows", IDC_ROWSSTATIC, IDC_ROWSEDIT, 50,
948 "Colu&mns", IDC_COLSSTATIC, IDC_COLSEDIT, 50, NULL);
949 checkbox(&cp, "Lock window size against resi&zing", IDC_LOCKSIZE);
951 beginbox(&cp, "Control the scrollback in the window",
953 staticedit(&cp, "Lines of &scrollback",
954 IDC_SAVESTATIC, IDC_SAVEEDIT, 50);
955 checkbox(&cp, "&Display scrollbar", IDC_SCROLLBAR);
956 checkbox(&cp, "Reset scrollback on &keypress", IDC_SCROLLKEY);
957 checkbox(&cp, "Reset scrollback on dis&play activity",
960 beginbox(&cp, NULL, IDC_BOX_WINDOW3);
961 checkbox(&cp, "&Warn before closing window", IDC_CLOSEWARN);
962 checkbox(&cp, "Window closes on ALT-F&4", IDC_ALTF4);
963 checkbox(&cp, "S&ystem menu appears on ALT-Space", IDC_ALTSPACE);
964 checkbox(&cp, "System menu appears on A< alone", IDC_ALTONLY);
965 checkbox(&cp, "Ensure window is always on &top", IDC_ALWAYSONTOP);
969 if (panel == appearancepanelstart) {
970 /* The Appearance panel. Accelerators used: [acgo] luvb h ti p s */
972 ctlposinit(&cp, hwnd, 80, 3, 13);
973 bartitle(&cp, "Options controlling PuTTY's appearance",
974 IDC_TITLE_APPEARANCE);
975 beginbox(&cp, "Adjust the use of the cursor", IDC_BOX_APPEARANCE1);
976 radioline(&cp, "Cursor appearance:", IDC_CURSORSTATIC, 3,
977 "B&lock", IDC_CURBLOCK,
978 "&Underline", IDC_CURUNDER,
979 "&Vertical line", IDC_CURVERT, NULL);
980 checkbox(&cp, "Cursor &blinks", IDC_BLINKCUR);
982 beginbox(&cp, "Set the font used in the terminal window",
983 IDC_BOX_APPEARANCE2);
984 staticbtn(&cp, "", IDC_FONTSTATIC, "C&hange...", IDC_CHOOSEFONT);
986 beginbox(&cp, "Adjust the use of the window title",
987 IDC_BOX_APPEARANCE3);
989 "Window &title:", IDC_WINTITLE, IDC_WINEDIT, 100, NULL);
990 checkbox(&cp, "Avoid ever using &icon title", IDC_WINNAME);
992 beginbox(&cp, "Adjust the use of the mouse pointer",
993 IDC_BOX_APPEARANCE4);
994 checkbox(&cp, "Hide mouse &pointer when typing in window",
997 beginbox(&cp, "Adjust the window border", IDC_BOX_APPEARANCE5);
998 checkbox(&cp, "&Sunken-edge border (slightly thicker)",
1003 if (panel == translationpanelstart) {
1004 /* The Translation panel. Accelerators used: [acgo] xbep t s */
1006 ctlposinit(&cp, hwnd, 80, 3, 13);
1007 bartitle(&cp, "Options controlling character set translation",
1008 IDC_TITLE_TRANSLATION);
1009 beginbox(&cp, "Adjust how PuTTY displays line drawing characters",
1010 IDC_BOX_TRANSLATION1);
1012 "Handling of line drawing characters:", IDC_VTSTATIC,
1013 "Font has &XWindows encoding", IDC_VTXWINDOWS,
1014 "Use font in &both ANSI and OEM modes", IDC_VTOEMANSI,
1015 "Use font in O&EM mode only", IDC_VTOEMONLY,
1016 "&Poor man's line drawing (" "+" ", " "-" " and " "|" ")",
1017 IDC_VTPOORMAN, "&Unicode mode", IDC_VTUNICODE, NULL);
1019 beginbox(&cp, "Enable character set translation on received data",
1020 IDC_BOX_TRANSLATION2);
1021 multiedit(&cp, "Line codepage:", IDC_CODEPAGESTATIC,
1022 IDC_CODEPAGE, 100, NULL);
1026 if (panel == selectionpanelstart) {
1027 /* The Selection panel. Accelerators used: [acgo] d wx hst */
1029 ctlposinit(&cp, hwnd, 80, 3, 13);
1030 bartitle(&cp, "Options controlling copy and paste",
1031 IDC_TITLE_SELECTION);
1032 beginbox(&cp, "Translation of pasted characters",
1033 IDC_BOX_SELECTION1);
1035 "&Don't translate line drawing chars into +, - and |",
1038 beginbox(&cp, "Control which mouse button does which thing",
1039 IDC_BOX_SELECTION2);
1040 radiobig(&cp, "Action of mouse buttons:", IDC_MBSTATIC,
1041 "&Windows (Right pastes, Middle extends)", IDC_MBWINDOWS,
1042 "&xterm (Right extends, Middle pastes)", IDC_MBXTERM,
1045 beginbox(&cp, "Control the select-one-word-at-a-time mode",
1046 IDC_BOX_SELECTION3);
1047 charclass(&cp, "C&haracter classes:", IDC_CCSTATIC, IDC_CCLIST,
1048 "&Set", IDC_CCSET, IDC_CCEDIT,
1049 "&to class", IDC_CCSTATIC2);
1053 if (panel == colourspanelstart) {
1054 /* The Colours panel. Accelerators used: [acgo] blum */
1056 ctlposinit(&cp, hwnd, 80, 3, 13);
1057 bartitle(&cp, "Options controlling use of colours",
1059 beginbox(&cp, "General options for colour usage",
1061 checkbox(&cp, "&Bolded text is a different colour",
1063 checkbox(&cp, "Attempt to use &logical palettes", IDC_PALETTE);
1065 beginbox(&cp, "Adjust the precise colours PuTTY displays",
1067 colouredit(&cp, "Select a colo&ur and then click to modify it:",
1068 IDC_COLOURSTATIC, IDC_COLOURLIST,
1069 "&Modify...", IDC_CHANGE,
1070 "Red:", IDC_RSTATIC, IDC_RVALUE,
1071 "Green:", IDC_GSTATIC, IDC_GVALUE,
1072 "Blue:", IDC_BSTATIC, IDC_BVALUE, NULL);
1076 if (panel == connectionpanelstart) {
1077 /* The Connection panel. Accelerators used: [acgo] tuk */
1079 ctlposinit(&cp, hwnd, 80, 3, 13);
1080 bartitle(&cp, "Options controlling the connection",
1081 IDC_TITLE_CONNECTION);
1083 beginbox(&cp, "Data to send to the server",
1084 IDC_BOX_CONNECTION1);
1085 staticedit(&cp, "Terminal-&type string", IDC_TTSTATIC,
1087 staticedit(&cp, "Auto-login &username", IDC_LOGSTATIC,
1091 beginbox(&cp, "Adjust telnet session.", IDC_BOX_CONNECTION1);
1092 checkbox(&cp, "Keyboard sends telnet Backspace and Interrupt",
1096 beginbox(&cp, "Sending of null packets to keep session active",
1097 IDC_BOX_CONNECTION2);
1098 staticedit(&cp, "Seconds between &keepalives (0 to turn off)",
1099 IDC_PINGSTATIC, IDC_PINGEDIT, 20);
1103 if (panel == telnetpanelstart) {
1104 /* The Telnet panel. Accelerators used: [acgo] svldr bftk */
1106 ctlposinit(&cp, hwnd, 80, 3, 13);
1108 bartitle(&cp, "Options controlling Telnet connections",
1110 beginbox(&cp, "Data to send to the server", IDC_BOX_TELNET1);
1111 staticedit(&cp, "Terminal-&speed string", IDC_TSSTATIC,
1113 envsetter(&cp, "Environment variables:", IDC_ENVSTATIC,
1114 "&Variable", IDC_VARSTATIC, IDC_VAREDIT, "Va&lue",
1115 IDC_VALSTATIC, IDC_VALEDIT, IDC_ENVLIST, "A&dd",
1116 IDC_ENVADD, "&Remove", IDC_ENVREMOVE);
1118 beginbox(&cp, "Telnet protocol adjustments", IDC_BOX_TELNET2);
1119 radioline(&cp, "Handling of OLD_ENVIRON ambiguity:",
1120 IDC_EMSTATIC, 2, "&BSD (commonplace)", IDC_EMBSD,
1121 "R&FC 1408 (unusual)", IDC_EMRFC, NULL);
1122 radioline(&cp, "&Telnet negotiation mode:", IDC_ACTSTATIC, 2,
1123 "Passive", IDC_TPASSIVE, "Active",
1125 checkbox(&cp, "&Keyboard sends telnet Backspace and Interrupt",
1131 if (panel == rloginpanelstart) {
1132 /* The Rlogin panel. Accelerators used: [acgo] sl */
1134 ctlposinit(&cp, hwnd, 80, 3, 13);
1136 bartitle(&cp, "Options controlling Rlogin connections",
1138 beginbox(&cp, "Data to send to the server", IDC_BOX_RLOGIN1);
1139 staticedit(&cp, "Terminal-&speed string", IDC_R_TSSTATIC,
1141 staticedit(&cp, "&Local username:", IDC_RLLUSERSTATIC,
1142 IDC_RLLUSEREDIT, 50);
1147 if (panel == sshpanelstart) {
1148 /* The SSH panel. Accelerators used: [acgo] rmfkw pe123bds i */
1150 ctlposinit(&cp, hwnd, 80, 3, 13);
1152 bartitle(&cp, "Options controlling SSH connections",
1154 beginbox(&cp, "Data to send to the server", IDC_BOX_SSH1);
1156 "&Remote command:", IDC_CMDSTATIC, IDC_CMDEDIT, 100,
1159 beginbox(&cp, "Authentication options", IDC_BOX_SSH2);
1160 checkbox(&cp, "Atte&mpt TIS or CryptoCard authentication",
1162 checkbox(&cp, "Allow agent &forwarding", IDC_AGENTFWD);
1163 editbutton(&cp, "Private &key file for authentication:",
1164 IDC_PKSTATIC, IDC_PKEDIT, "Bro&wse...",
1167 beginbox(&cp, "Protocol options", IDC_BOX_SSH3);
1168 checkbox(&cp, "Don't allocate a &pseudo-terminal", IDC_NOPTY);
1169 checkbox(&cp, "Enable compr&ession", IDC_COMPRESS);
1170 radioline(&cp, "Preferred SSH protocol version:",
1171 IDC_SSHPROTSTATIC, 2,
1172 "&1", IDC_SSHPROT1, "&2", IDC_SSHPROT2, NULL);
1173 radioline(&cp, "Preferred encryption algorithm:",
1174 IDC_CIPHERSTATIC, 4, "&3DES", IDC_CIPHER3DES,
1175 "&Blowfish", IDC_CIPHERBLOWF, "&DES", IDC_CIPHERDES,
1176 "AE&S", IDC_CIPHERAES, NULL);
1177 checkbox(&cp, "&Imitate SSH 2 MAC bug in commercial <= v2.3.x",
1183 if (panel == tunnelspanelstart) {
1184 /* The Tunnels panel. Accelerators used: [acgo] ex */
1186 ctlposinit(&cp, hwnd, 80, 3, 13);
1188 bartitle(&cp, "Options controlling SSH tunnelling",
1190 beginbox(&cp, "X11 forwarding options", IDC_BOX_TUNNELS);
1191 checkbox(&cp, "&Enable X11 forwarding", IDC_X11_FORWARD);
1192 multiedit(&cp, "&X display location", IDC_X11_DISPSTATIC,
1193 IDC_X11_DISPLAY, 50, NULL);
1200 * Helper function to load the session selected in SESSLIST
1201 * if any, as this is done in more than one place in
1202 * GenericMainDlgProc(). 0 => failure.
1204 static int load_selected_session(HWND hwnd)
1206 int n = SendDlgItemMessage(hwnd, IDC_SESSLIST,
1207 LB_GETCURSEL, 0, 0);
1213 isdef = !strcmp(sessions[n], "Default Settings");
1214 load_settings(sessions[n], !isdef, &cfg);
1215 init_dlg_ctrls(hwnd, TRUE);
1217 SetDlgItemText(hwnd, IDC_SESSEDIT, sessions[n]);
1219 SetDlgItemText(hwnd, IDC_SESSEDIT, "");
1220 /* Restore the selection, which will have been clobbered by
1221 * SESSEDIT handling. */
1222 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_SETCURSEL, n, 0);
1227 * This function is the configuration box.
1229 static int GenericMainDlgProc(HWND hwnd, UINT msg,
1230 WPARAM wParam, LPARAM lParam, int dlgtype)
1233 struct treeview_faff tvfaff;
1236 char filename[sizeof(cfg.keyfile)];
1239 char fontstatic[256];
1241 struct servent *service;
1247 SetWindowLong(hwnd, GWL_USERDATA, 0);
1248 SendMessage(hwnd, WM_SETICON, (WPARAM) ICON_BIG,
1249 (LPARAM) LoadIcon(hinst, MAKEINTRESOURCE(IDI_CFGICON)));
1251 * Centre the window.
1253 { /* centre the window */
1256 hw = GetDesktopWindow();
1257 if (GetWindowRect(hw, &rs) && GetWindowRect(hwnd, &rd))
1259 (rs.right + rs.left + rd.left - rd.right) / 2,
1260 (rs.bottom + rs.top + rd.top - rd.bottom) / 2,
1261 rd.right - rd.left, rd.bottom - rd.top, TRUE);
1265 * Create the tree view.
1273 r.right = r.left + 75;
1275 r.bottom = r.top + 10;
1276 MapDialogRect(hwnd, &r);
1277 tvstatic = CreateWindowEx(0, "STATIC", "Cate&gory:",
1278 WS_CHILD | WS_VISIBLE,
1280 r.right - r.left, r.bottom - r.top,
1281 hwnd, (HMENU) IDCX_TVSTATIC, hinst,
1283 font = SendMessage(hwnd, WM_GETFONT, 0, 0);
1284 SendMessage(tvstatic, WM_SETFONT, font, MAKELPARAM(TRUE, 0));
1287 r.right = r.left + 75;
1289 r.bottom = r.top + 206;
1290 MapDialogRect(hwnd, &r);
1291 treeview = CreateWindowEx(WS_EX_CLIENTEDGE, WC_TREEVIEW, "",
1292 WS_CHILD | WS_VISIBLE |
1293 WS_TABSTOP | TVS_HASLINES |
1294 TVS_DISABLEDRAGDROP | TVS_HASBUTTONS
1296 TVS_SHOWSELALWAYS, r.left, r.top,
1297 r.right - r.left, r.bottom - r.top,
1298 hwnd, (HMENU) IDCX_TREEVIEW, hinst,
1300 font = SendMessage(hwnd, WM_GETFONT, 0, 0);
1301 SendMessage(treeview, WM_SETFONT, font, MAKELPARAM(TRUE, 0));
1302 tvfaff.treeview = treeview;
1303 memset(tvfaff.lastat, 0, sizeof(tvfaff.lastat));
1307 * Set up the tree view contents.
1309 hsession = treeview_insert(&tvfaff, 0, "Session");
1310 treeview_insert(&tvfaff, 1, "Logging");
1311 treeview_insert(&tvfaff, 0, "Terminal");
1312 treeview_insert(&tvfaff, 1, "Keyboard");
1313 treeview_insert(&tvfaff, 1, "Bell");
1314 treeview_insert(&tvfaff, 0, "Window");
1315 treeview_insert(&tvfaff, 1, "Appearance");
1316 treeview_insert(&tvfaff, 1, "Translation");
1317 treeview_insert(&tvfaff, 1, "Selection");
1318 treeview_insert(&tvfaff, 1, "Colours");
1319 treeview_insert(&tvfaff, 0, "Connection");
1321 treeview_insert(&tvfaff, 1, "Telnet");
1322 treeview_insert(&tvfaff, 1, "Rlogin");
1323 if (backends[3].backend != NULL) {
1324 treeview_insert(&tvfaff, 1, "SSH");
1325 treeview_insert(&tvfaff, 2, "Tunnels");
1330 * Put the treeview selection on to the Session panel. This
1331 * should also cause creation of the relevant controls.
1333 TreeView_SelectItem(treeview, hsession);
1336 * Set focus into the first available control.
1340 ctl = GetDlgItem(hwnd, IDC_HOST);
1342 ctl = GetDlgItem(hwnd, IDC_CLOSEEXIT);
1346 SetWindowLong(hwnd, GWL_USERDATA, 1);
1350 * Button release should trigger WM_OK if there was a
1351 * previous double click on the session list.
1355 SendMessage(hwnd, WM_COMMAND, IDOK, 0);
1358 if (LOWORD(wParam) == IDCX_TREEVIEW &&
1359 ((LPNMHDR) lParam)->code == TVN_SELCHANGED) {
1361 TreeView_GetSelection(((LPNMHDR) lParam)->hwndFrom);
1366 item.pszText = buffer;
1367 item.cchTextMax = sizeof(buffer);
1368 item.mask = TVIF_TEXT;
1369 TreeView_GetItem(((LPNMHDR) lParam)->hwndFrom, &item);
1370 for (j = controlstartvalue; j < controlendvalue; j++) {
1371 HWND item = GetDlgItem(hwnd, j);
1373 DestroyWindow(item);
1375 if (!strcmp(buffer, "Session"))
1376 create_controls(hwnd, dlgtype, sessionpanelstart);
1377 if (!strcmp(buffer, "Logging"))
1378 create_controls(hwnd, dlgtype, loggingpanelstart);
1379 if (!strcmp(buffer, "Keyboard"))
1380 create_controls(hwnd, dlgtype, keyboardpanelstart);
1381 if (!strcmp(buffer, "Terminal"))
1382 create_controls(hwnd, dlgtype, terminalpanelstart);
1383 if (!strcmp(buffer, "Bell"))
1384 create_controls(hwnd, dlgtype, bellpanelstart);
1385 if (!strcmp(buffer, "Window"))
1386 create_controls(hwnd, dlgtype, windowpanelstart);
1387 if (!strcmp(buffer, "Appearance"))
1388 create_controls(hwnd, dlgtype, appearancepanelstart);
1389 if (!strcmp(buffer, "Tunnels"))
1390 create_controls(hwnd, dlgtype, tunnelspanelstart);
1391 if (!strcmp(buffer, "Connection"))
1392 create_controls(hwnd, dlgtype, connectionpanelstart);
1393 if (!strcmp(buffer, "Telnet"))
1394 create_controls(hwnd, dlgtype, telnetpanelstart);
1395 if (!strcmp(buffer, "Rlogin"))
1396 create_controls(hwnd, dlgtype, rloginpanelstart);
1397 if (!strcmp(buffer, "SSH"))
1398 create_controls(hwnd, dlgtype, sshpanelstart);
1399 if (!strcmp(buffer, "Selection"))
1400 create_controls(hwnd, dlgtype, selectionpanelstart);
1401 if (!strcmp(buffer, "Colours"))
1402 create_controls(hwnd, dlgtype, colourspanelstart);
1403 if (!strcmp(buffer, "Translation"))
1404 create_controls(hwnd, dlgtype, translationpanelstart);
1406 init_dlg_ctrls(hwnd, FALSE);
1408 SetFocus(((LPNMHDR) lParam)->hwndFrom); /* ensure focus stays */
1414 * Only process WM_COMMAND once the dialog is fully formed.
1416 if (GetWindowLong(hwnd, GWL_USERDATA) == 1)
1417 switch (LOWORD(wParam)) {
1419 /* Behaviour of the "Open" button is different if the
1420 * session list has focus, *unless* the user just
1421 * double-clicked... */
1422 if (sesslist_has_focus && !readytogo) {
1423 if (!load_selected_session(hwnd)) {
1428 /* If at this point we have a valid session, go! */
1437 case IDC_PROTTELNET:
1438 case IDC_PROTRLOGIN:
1441 if (HIWORD(wParam) == BN_CLICKED ||
1442 HIWORD(wParam) == BN_DOUBLECLICKED) {
1443 int i = IsDlgButtonChecked(hwnd, IDC_PROTSSH);
1444 int j = IsDlgButtonChecked(hwnd, IDC_PROTTELNET);
1445 int k = IsDlgButtonChecked(hwnd, IDC_PROTRLOGIN);
1447 i ? PROT_SSH : j ? PROT_TELNET : k ? PROT_RLOGIN :
1449 if ((cfg.protocol == PROT_SSH && cfg.port != 22)
1450 || (cfg.protocol == PROT_TELNET && cfg.port != 23)
1451 || (cfg.protocol == PROT_RLOGIN
1452 && cfg.port != 513)) {
1453 cfg.port = i ? 22 : j ? 23 : 513;
1454 SetDlgItemInt(hwnd, IDC_PORT, cfg.port, FALSE);
1459 if (HIWORD(wParam) == EN_CHANGE)
1460 GetDlgItemText(hwnd, IDC_HOST, cfg.host,
1461 sizeof(cfg.host) - 1);
1464 if (HIWORD(wParam) == EN_CHANGE) {
1465 GetDlgItemText(hwnd, IDC_PORT, portname, 31);
1466 if (isdigit(portname[0]))
1467 MyGetDlgItemInt(hwnd, IDC_PORT, &cfg.port);
1469 service = getservbyname(portname, NULL);
1471 cfg.port = ntohs(service->s_port);
1478 if (HIWORD(wParam) == EN_CHANGE) {
1479 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_SETCURSEL,
1481 GetDlgItemText(hwnd, IDC_SESSEDIT,
1482 savedsession, sizeof(savedsession) - 1);
1483 savedsession[sizeof(savedsession) - 1] = '\0';
1487 if (HIWORD(wParam) == BN_CLICKED ||
1488 HIWORD(wParam) == BN_DOUBLECLICKED) {
1493 GetDlgItemText(hwnd, IDC_SESSEDIT, str,
1496 int n = SendDlgItemMessage(hwnd, IDC_SESSLIST,
1497 LB_GETCURSEL, 0, 0);
1502 strcpy(str, sessions[n]);
1504 save_settings(str, !!strcmp(str, "Default Settings"),
1506 get_sesslist(FALSE);
1508 SendDlgItemMessage(hwnd, IDC_SESSLIST, WM_SETREDRAW,
1510 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_RESETCONTENT,
1512 for (i = 0; i < nsessions; i++)
1513 SendDlgItemMessage(hwnd, IDC_SESSLIST,
1515 (LPARAM) (sessions[i]));
1516 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_SETCURSEL,
1518 SendDlgItemMessage(hwnd, IDC_SESSLIST, WM_SETREDRAW,
1520 InvalidateRect(GetDlgItem(hwnd, IDC_SESSLIST), NULL,
1526 if (LOWORD(wParam) == IDC_SESSLIST) {
1527 if (HIWORD(wParam) == LBN_SETFOCUS)
1528 sesslist_has_focus = 1;
1529 else if (HIWORD(wParam) == LBN_KILLFOCUS)
1530 sesslist_has_focus = 0;
1532 if (LOWORD(wParam) == IDC_SESSLOAD &&
1533 HIWORD(wParam) != BN_CLICKED &&
1534 HIWORD(wParam) != BN_DOUBLECLICKED) break;
1535 if (LOWORD(wParam) == IDC_SESSLIST &&
1536 HIWORD(wParam) != LBN_DBLCLK) break;
1537 /* Load the session selected in SESSLIST. */
1538 if (load_selected_session(hwnd) &&
1539 LOWORD(wParam) == IDC_SESSLIST) {
1541 * A double-click on a saved session should
1542 * actually start the session, not just load it.
1543 * Unless it's Default Settings or some other
1544 * host-less set of saved settings.
1553 if (HIWORD(wParam) == BN_CLICKED ||
1554 HIWORD(wParam) == BN_DOUBLECLICKED) {
1555 int n = SendDlgItemMessage(hwnd, IDC_SESSLIST,
1556 LB_GETCURSEL, 0, 0);
1557 if (n == LB_ERR || n == 0) {
1561 del_settings(sessions[n]);
1562 get_sesslist(FALSE);
1564 SendDlgItemMessage(hwnd, IDC_SESSLIST, WM_SETREDRAW,
1566 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_RESETCONTENT,
1568 for (i = 0; i < nsessions; i++)
1569 SendDlgItemMessage(hwnd, IDC_SESSLIST,
1571 (LPARAM) (sessions[i]));
1572 SendDlgItemMessage(hwnd, IDC_SESSLIST, LB_SETCURSEL,
1574 SendDlgItemMessage(hwnd, IDC_SESSLIST, WM_SETREDRAW,
1576 InvalidateRect(GetDlgItem(hwnd, IDC_SESSLIST), NULL,
1580 if (HIWORD(wParam) == EN_CHANGE)
1581 MyGetDlgItemInt(hwnd, IDC_PINGEDIT,
1582 &cfg.ping_interval);
1586 if (HIWORD(wParam) == BN_CLICKED ||
1587 HIWORD(wParam) == BN_DOUBLECLICKED)
1588 cfg.bksp_is_delete =
1589 IsDlgButtonChecked(hwnd, IDC_DEL127);
1593 if (HIWORD(wParam) == BN_CLICKED ||
1594 HIWORD(wParam) == BN_DOUBLECLICKED)
1596 IsDlgButtonChecked(hwnd, IDC_HOMERXVT);
1602 case IDC_FUNCVT100P:
1604 if (HIWORD(wParam) == BN_CLICKED ||
1605 HIWORD(wParam) == BN_DOUBLECLICKED)
1606 switch (LOWORD(wParam)) {
1619 case IDC_FUNCVT100P:
1629 if (HIWORD(wParam) == BN_CLICKED ||
1630 HIWORD(wParam) == BN_DOUBLECLICKED) {
1632 IsDlgButtonChecked(hwnd, IDC_KPAPPLIC);
1633 cfg.nethack_keypad = FALSE;
1637 if (HIWORD(wParam) == BN_CLICKED ||
1638 HIWORD(wParam) == BN_DOUBLECLICKED) {
1639 cfg.app_keypad = FALSE;
1640 cfg.nethack_keypad = TRUE;
1645 if (HIWORD(wParam) == BN_CLICKED ||
1646 HIWORD(wParam) == BN_DOUBLECLICKED)
1648 IsDlgButtonChecked(hwnd, IDC_CURAPPLIC);
1651 if (HIWORD(wParam) == BN_CLICKED ||
1652 HIWORD(wParam) == BN_DOUBLECLICKED)
1654 IsDlgButtonChecked(hwnd, IDC_NOAPPLICC);
1657 if (HIWORD(wParam) == BN_CLICKED ||
1658 HIWORD(wParam) == BN_DOUBLECLICKED)
1660 IsDlgButtonChecked(hwnd, IDC_NOAPPLICK);
1663 if (HIWORD(wParam) == BN_CLICKED ||
1664 HIWORD(wParam) == BN_DOUBLECLICKED)
1665 cfg.alt_f4 = IsDlgButtonChecked(hwnd, IDC_ALTF4);
1668 if (HIWORD(wParam) == BN_CLICKED ||
1669 HIWORD(wParam) == BN_DOUBLECLICKED)
1671 IsDlgButtonChecked(hwnd, IDC_ALTSPACE);
1674 if (HIWORD(wParam) == BN_CLICKED ||
1675 HIWORD(wParam) == BN_DOUBLECLICKED)
1677 IsDlgButtonChecked(hwnd, IDC_ALTONLY);
1679 case IDC_ECHOBACKEND:
1682 if (HIWORD(wParam) == BN_CLICKED ||
1683 HIWORD(wParam) == BN_DOUBLECLICKED) {
1684 if (LOWORD(wParam) == IDC_ECHOBACKEND)
1685 cfg.localecho = LD_BACKEND;
1686 if (LOWORD(wParam) == IDC_ECHOYES)
1687 cfg.localecho = LD_YES;
1688 if (LOWORD(wParam) == IDC_ECHONO)
1689 cfg.localecho = LD_NO;
1692 case IDC_EDITBACKEND:
1695 if (HIWORD(wParam) == BN_CLICKED ||
1696 HIWORD(wParam) == BN_DOUBLECLICKED) {
1697 if (LOWORD(wParam) == IDC_EDITBACKEND)
1698 cfg.localedit = LD_BACKEND;
1699 if (LOWORD(wParam) == IDC_EDITYES)
1700 cfg.localedit = LD_YES;
1701 if (LOWORD(wParam) == IDC_EDITNO)
1702 cfg.localedit = LD_NO;
1705 case IDC_ANSWEREDIT:
1706 if (HIWORD(wParam) == EN_CHANGE)
1707 GetDlgItemText(hwnd, IDC_ANSWEREDIT, cfg.answerback,
1708 sizeof(cfg.answerback) - 1);
1710 case IDC_ALWAYSONTOP:
1711 if (HIWORD(wParam) == BN_CLICKED ||
1712 HIWORD(wParam) == BN_DOUBLECLICKED)
1714 IsDlgButtonChecked(hwnd, IDC_ALWAYSONTOP);
1717 if (HIWORD(wParam) == BN_CLICKED ||
1718 HIWORD(wParam) == BN_DOUBLECLICKED)
1720 IsDlgButtonChecked(hwnd, IDC_SCROLLKEY);
1722 case IDC_SCROLLDISP:
1723 if (HIWORD(wParam) == BN_CLICKED ||
1724 HIWORD(wParam) == BN_DOUBLECLICKED)
1725 cfg.scroll_on_disp =
1726 IsDlgButtonChecked(hwnd, IDC_SCROLLDISP);
1728 case IDC_COMPOSEKEY:
1729 if (HIWORD(wParam) == BN_CLICKED ||
1730 HIWORD(wParam) == BN_DOUBLECLICKED)
1732 IsDlgButtonChecked(hwnd, IDC_COMPOSEKEY);
1734 case IDC_CTRLALTKEYS:
1735 if (HIWORD(wParam) == BN_CLICKED ||
1736 HIWORD(wParam) == BN_DOUBLECLICKED)
1738 IsDlgButtonChecked(hwnd, IDC_CTRLALTKEYS);
1741 if (HIWORD(wParam) == BN_CLICKED ||
1742 HIWORD(wParam) == BN_DOUBLECLICKED)
1743 cfg.telnet_keyboard =
1744 IsDlgButtonChecked(hwnd, IDC_TELNETKEY);
1747 if (HIWORD(wParam) == BN_CLICKED ||
1748 HIWORD(wParam) == BN_DOUBLECLICKED)
1750 IsDlgButtonChecked(hwnd, IDC_WRAPMODE);
1753 if (HIWORD(wParam) == BN_CLICKED ||
1754 HIWORD(wParam) == BN_DOUBLECLICKED)
1755 cfg.dec_om = IsDlgButtonChecked(hwnd, IDC_DECOM);
1758 if (HIWORD(wParam) == BN_CLICKED ||
1759 HIWORD(wParam) == BN_DOUBLECLICKED)
1761 IsDlgButtonChecked(hwnd, IDC_LFHASCR);
1764 if (HIWORD(wParam) == EN_CHANGE)
1765 MyGetDlgItemInt(hwnd, IDC_ROWSEDIT, &cfg.height);
1768 if (HIWORD(wParam) == EN_CHANGE)
1769 MyGetDlgItemInt(hwnd, IDC_COLSEDIT, &cfg.width);
1772 if (HIWORD(wParam) == EN_CHANGE)
1773 MyGetDlgItemInt(hwnd, IDC_SAVEEDIT, &cfg.savelines);
1775 case IDC_CHOOSEFONT:
1778 lf.lfHeight = -MulDiv(cfg.fontheight,
1779 GetDeviceCaps(hdc, LOGPIXELSY),
1783 lf.lfWidth = lf.lfEscapement = lf.lfOrientation = 0;
1784 lf.lfItalic = lf.lfUnderline = lf.lfStrikeOut = 0;
1785 lf.lfWeight = (cfg.fontisbold ? FW_BOLD : 0);
1786 lf.lfCharSet = cfg.fontcharset;
1787 lf.lfOutPrecision = OUT_DEFAULT_PRECIS;
1788 lf.lfClipPrecision = CLIP_DEFAULT_PRECIS;
1789 lf.lfQuality = DEFAULT_QUALITY;
1790 lf.lfPitchAndFamily = FIXED_PITCH | FF_DONTCARE;
1791 strncpy(lf.lfFaceName, cfg.font,
1792 sizeof(lf.lfFaceName) - 1);
1793 lf.lfFaceName[sizeof(lf.lfFaceName) - 1] = '\0';
1795 cf.lStructSize = sizeof(cf);
1796 cf.hwndOwner = hwnd;
1798 cf.Flags = CF_FIXEDPITCHONLY | CF_FORCEFONTEXIST |
1799 CF_INITTOLOGFONTSTRUCT | CF_SCREENFONTS;
1801 if (ChooseFont(&cf)) {
1802 strncpy(cfg.font, lf.lfFaceName, sizeof(cfg.font) - 1);
1803 cfg.font[sizeof(cfg.font) - 1] = '\0';
1804 cfg.fontisbold = (lf.lfWeight == FW_BOLD);
1805 cfg.fontcharset = lf.lfCharSet;
1806 cfg.fontheight = cf.iPointSize / 10;
1807 fmtfont(fontstatic);
1808 SetDlgItemText(hwnd, IDC_FONTSTATIC, fontstatic);
1811 case IDC_BELL_DISABLED:
1812 case IDC_BELL_DEFAULT:
1813 case IDC_BELL_WAVEFILE:
1814 case IDC_BELL_VISUAL:
1815 if (HIWORD(wParam) == BN_CLICKED ||
1816 HIWORD(wParam) == BN_DOUBLECLICKED) {
1817 if (LOWORD(wParam) == IDC_BELL_DISABLED)
1818 cfg.beep = BELL_DISABLED;
1819 if (LOWORD(wParam) == IDC_BELL_DEFAULT)
1820 cfg.beep = BELL_DEFAULT;
1821 if (LOWORD(wParam) == IDC_BELL_WAVEFILE)
1822 cfg.beep = BELL_WAVEFILE;
1823 if (LOWORD(wParam) == IDC_BELL_VISUAL)
1824 cfg.beep = BELL_VISUAL;
1827 case IDC_BELL_WAVEBROWSE:
1828 memset(&of, 0, sizeof(of));
1829 #ifdef OPENFILENAME_SIZE_VERSION_400
1830 of.lStructSize = OPENFILENAME_SIZE_VERSION_400;
1832 of.lStructSize = sizeof(of);
1834 of.hwndOwner = hwnd;
1835 of.lpstrFilter = "Wave Files\0*.WAV\0AllFiles\0*\0\0\0";
1836 of.lpstrCustomFilter = NULL;
1837 of.nFilterIndex = 1;
1838 of.lpstrFile = filename;
1839 strcpy(filename, cfg.bell_wavefile);
1840 of.nMaxFile = sizeof(filename);
1841 of.lpstrFileTitle = NULL;
1842 of.lpstrInitialDir = NULL;
1843 of.lpstrTitle = "Select Bell Sound File";
1845 if (GetOpenFileName(&of)) {
1846 strcpy(cfg.bell_wavefile, filename);
1847 SetDlgItemText(hwnd, IDC_BELL_WAVEEDIT,
1851 case IDC_BELL_WAVEEDIT:
1852 if (HIWORD(wParam) == EN_CHANGE)
1853 GetDlgItemText(hwnd, IDC_BELL_WAVEEDIT,
1855 sizeof(cfg.bell_wavefile) - 1);
1858 if (HIWORD(wParam) == BN_CLICKED ||
1859 HIWORD(wParam) == BN_DOUBLECLICKED)
1861 IsDlgButtonChecked(hwnd, IDC_BELLOVL);
1864 if (HIWORD(wParam) == EN_CHANGE)
1865 MyGetDlgItemInt(hwnd, IDC_BELLOVLN, &cfg.bellovl_n);
1868 if (HIWORD(wParam) == EN_CHANGE)
1869 MyGetDlgItemFlt(hwnd, IDC_BELLOVLT, &cfg.bellovl_t,
1873 if (HIWORD(wParam) == EN_CHANGE)
1874 MyGetDlgItemFlt(hwnd, IDC_BELLOVLS, &cfg.bellovl_s,
1878 if (HIWORD(wParam) == BN_CLICKED ||
1879 HIWORD(wParam) == BN_DOUBLECLICKED)
1881 IsDlgButtonChecked(hwnd, IDC_BLINKTEXT);
1884 if (HIWORD(wParam) == BN_CLICKED ||
1885 HIWORD(wParam) == BN_DOUBLECLICKED)
1886 cfg.bce = IsDlgButtonChecked(hwnd, IDC_BCE);
1889 if (HIWORD(wParam) == BN_CLICKED ||
1890 HIWORD(wParam) == BN_DOUBLECLICKED)
1891 cfg.win_name_always =
1892 IsDlgButtonChecked(hwnd, IDC_WINNAME);
1895 if (HIWORD(wParam) == BN_CLICKED ||
1896 HIWORD(wParam) == BN_DOUBLECLICKED)
1898 IsDlgButtonChecked(hwnd, IDC_HIDEMOUSE);
1900 case IDC_SUNKENEDGE:
1901 if (HIWORD(wParam) == BN_CLICKED ||
1902 HIWORD(wParam) == BN_DOUBLECLICKED)
1904 IsDlgButtonChecked(hwnd, IDC_SUNKENEDGE);
1907 if (HIWORD(wParam) == BN_CLICKED ||
1908 HIWORD(wParam) == BN_DOUBLECLICKED)
1909 cfg.cursor_type = 0;
1912 if (HIWORD(wParam) == BN_CLICKED ||
1913 HIWORD(wParam) == BN_DOUBLECLICKED)
1914 cfg.cursor_type = 1;
1917 if (HIWORD(wParam) == BN_CLICKED ||
1918 HIWORD(wParam) == BN_DOUBLECLICKED)
1919 cfg.cursor_type = 2;
1922 if (HIWORD(wParam) == BN_CLICKED ||
1923 HIWORD(wParam) == BN_DOUBLECLICKED)
1925 IsDlgButtonChecked(hwnd, IDC_BLINKCUR);
1928 if (HIWORD(wParam) == BN_CLICKED ||
1929 HIWORD(wParam) == BN_DOUBLECLICKED)
1931 IsDlgButtonChecked(hwnd, IDC_SCROLLBAR);
1934 if (HIWORD(wParam) == BN_CLICKED ||
1935 HIWORD(wParam) == BN_DOUBLECLICKED)
1937 IsDlgButtonChecked(hwnd, IDC_LOCKSIZE);
1940 if (HIWORD(wParam) == EN_CHANGE)
1941 GetDlgItemText(hwnd, IDC_WINEDIT, cfg.wintitle,
1942 sizeof(cfg.wintitle) - 1);
1947 if (HIWORD(wParam) == BN_CLICKED ||
1948 HIWORD(wParam) == BN_DOUBLECLICKED) {
1950 IsDlgButtonChecked(hwnd,
1951 IDC_COEALWAYS) ? COE_ALWAYS :
1952 IsDlgButtonChecked(hwnd,
1953 IDC_COENEVER) ? COE_NEVER :
1958 if (HIWORD(wParam) == BN_CLICKED ||
1959 HIWORD(wParam) == BN_DOUBLECLICKED)
1961 IsDlgButtonChecked(hwnd, IDC_CLOSEWARN);
1964 if (HIWORD(wParam) == EN_CHANGE)
1965 GetDlgItemText(hwnd, IDC_TTEDIT, cfg.termtype,
1966 sizeof(cfg.termtype) - 1);
1969 if (HIWORD(wParam) == EN_CHANGE)
1970 GetDlgItemText(hwnd, IDC_LGFEDIT, cfg.logfilename,
1971 sizeof(cfg.logfilename) - 1);
1974 memset(&of, 0, sizeof(of));
1975 #ifdef OPENFILENAME_SIZE_VERSION_400
1976 of.lStructSize = OPENFILENAME_SIZE_VERSION_400;
1978 of.lStructSize = sizeof(of);
1980 of.hwndOwner = hwnd;
1981 of.lpstrFilter = "All Files\0*\0\0\0";
1982 of.lpstrCustomFilter = NULL;
1983 of.nFilterIndex = 1;
1984 of.lpstrFile = filename;
1985 strcpy(filename, cfg.logfilename);
1986 of.nMaxFile = sizeof(filename);
1987 of.lpstrFileTitle = NULL;
1988 of.lpstrInitialDir = NULL;
1989 of.lpstrTitle = "Select session log file";
1991 if (GetSaveFileName(&of)) {
1992 strcpy(cfg.logfilename, filename);
1993 SetDlgItemText(hwnd, IDC_LGFEDIT, cfg.logfilename);
1997 case IDC_LSTATASCII:
1999 if (HIWORD(wParam) == BN_CLICKED ||
2000 HIWORD(wParam) == BN_DOUBLECLICKED) {
2001 if (IsDlgButtonChecked(hwnd, IDC_LSTATOFF))
2003 if (IsDlgButtonChecked(hwnd, IDC_LSTATASCII))
2005 if (IsDlgButtonChecked(hwnd, IDC_LSTATRAW))
2012 if (HIWORD(wParam) == BN_CLICKED ||
2013 HIWORD(wParam) == BN_DOUBLECLICKED) {
2014 if (IsDlgButtonChecked(hwnd, IDC_LSTATXASK))
2015 cfg.logxfovr = LGXF_ASK;
2016 if (IsDlgButtonChecked(hwnd, IDC_LSTATXAPN))
2017 cfg.logxfovr = LGXF_APN;
2018 if (IsDlgButtonChecked(hwnd, IDC_LSTATXOVR))
2019 cfg.logxfovr = LGXF_OVR;
2024 if (HIWORD(wParam) == EN_CHANGE)
2025 GetDlgItemText(hwnd, LOWORD(wParam), cfg.termspeed,
2026 sizeof(cfg.termspeed) - 1);
2029 if (HIWORD(wParam) == EN_CHANGE)
2030 GetDlgItemText(hwnd, IDC_LOGEDIT, cfg.username,
2031 sizeof(cfg.username) - 1);
2033 case IDC_RLLUSEREDIT:
2034 if (HIWORD(wParam) == EN_CHANGE)
2035 GetDlgItemText(hwnd, IDC_RLLUSEREDIT,
2037 sizeof(cfg.localusername) - 1);
2041 cfg.rfc_environ = IsDlgButtonChecked(hwnd, IDC_EMRFC);
2045 cfg.passive_telnet =
2046 IsDlgButtonChecked(hwnd, IDC_TPASSIVE);
2049 if (HIWORD(wParam) == BN_CLICKED ||
2050 HIWORD(wParam) == BN_DOUBLECLICKED) {
2051 char str[sizeof(cfg.environmt)];
2053 GetDlgItemText(hwnd, IDC_VAREDIT, str,
2059 p = str + strlen(str);
2061 GetDlgItemText(hwnd, IDC_VALEDIT, p,
2062 sizeof(str) - 1 - (p - str));
2073 if ((p - cfg.environmt) + strlen(str) + 2 <
2074 sizeof(cfg.environmt)) {
2076 p[strlen(str) + 1] = '\0';
2077 SendDlgItemMessage(hwnd, IDC_ENVLIST, LB_ADDSTRING,
2079 SetDlgItemText(hwnd, IDC_VAREDIT, "");
2080 SetDlgItemText(hwnd, IDC_VALEDIT, "");
2082 MessageBox(hwnd, "Environment too big",
2083 "PuTTY Error", MB_OK | MB_ICONERROR);
2088 if (HIWORD(wParam) != BN_CLICKED &&
2089 HIWORD(wParam) != BN_DOUBLECLICKED) break;
2091 SendDlgItemMessage(hwnd, IDC_ENVLIST, LB_GETCURSEL, 0,
2098 SendDlgItemMessage(hwnd, IDC_ENVLIST, LB_DELETESTRING,
2125 if (HIWORD(wParam) == BN_CLICKED ||
2126 HIWORD(wParam) == BN_DOUBLECLICKED)
2127 cfg.nopty = IsDlgButtonChecked(hwnd, IDC_NOPTY);
2130 if (HIWORD(wParam) == BN_CLICKED ||
2131 HIWORD(wParam) == BN_DOUBLECLICKED)
2133 IsDlgButtonChecked(hwnd, IDC_COMPRESS);
2136 if (HIWORD(wParam) == BN_CLICKED ||
2137 HIWORD(wParam) == BN_DOUBLECLICKED)
2139 IsDlgButtonChecked(hwnd, IDC_BUGGYMAC);
2142 if (HIWORD(wParam) == BN_CLICKED ||
2143 HIWORD(wParam) == BN_DOUBLECLICKED)
2145 IsDlgButtonChecked(hwnd, IDC_AGENTFWD);
2147 case IDC_CIPHER3DES:
2148 case IDC_CIPHERBLOWF:
2151 if (HIWORD(wParam) == BN_CLICKED ||
2152 HIWORD(wParam) == BN_DOUBLECLICKED) {
2153 if (IsDlgButtonChecked(hwnd, IDC_CIPHER3DES))
2154 cfg.cipher = CIPHER_3DES;
2155 else if (IsDlgButtonChecked(hwnd, IDC_CIPHERBLOWF))
2156 cfg.cipher = CIPHER_BLOWFISH;
2157 else if (IsDlgButtonChecked(hwnd, IDC_CIPHERDES))
2158 cfg.cipher = CIPHER_DES;
2159 else if (IsDlgButtonChecked(hwnd, IDC_CIPHERAES))
2160 cfg.cipher = CIPHER_AES;
2165 if (HIWORD(wParam) == BN_CLICKED ||
2166 HIWORD(wParam) == BN_DOUBLECLICKED) {
2167 if (IsDlgButtonChecked(hwnd, IDC_SSHPROT1))
2169 else if (IsDlgButtonChecked(hwnd, IDC_SSHPROT2))
2174 if (HIWORD(wParam) == BN_CLICKED ||
2175 HIWORD(wParam) == BN_DOUBLECLICKED)
2177 IsDlgButtonChecked(hwnd, IDC_AUTHTIS);
2180 if (HIWORD(wParam) == EN_CHANGE)
2181 GetDlgItemText(hwnd, IDC_PKEDIT, cfg.keyfile,
2182 sizeof(cfg.keyfile) - 1);
2185 if (HIWORD(wParam) == EN_CHANGE)
2186 GetDlgItemText(hwnd, IDC_CMDEDIT, cfg.remote_cmd,
2187 sizeof(cfg.remote_cmd) - 1);
2190 memset(&of, 0, sizeof(of));
2191 #ifdef OPENFILENAME_SIZE_VERSION_400
2192 of.lStructSize = OPENFILENAME_SIZE_VERSION_400;
2194 of.lStructSize = sizeof(of);
2196 of.hwndOwner = hwnd;
2197 of.lpstrFilter = "All Files\0*\0\0\0";
2198 of.lpstrCustomFilter = NULL;
2199 of.nFilterIndex = 1;
2200 of.lpstrFile = filename;
2201 strcpy(filename, cfg.keyfile);
2202 of.nMaxFile = sizeof(filename);
2203 of.lpstrFileTitle = NULL;
2204 of.lpstrInitialDir = NULL;
2205 of.lpstrTitle = "Select Public Key File";
2207 if (GetOpenFileName(&of)) {
2208 strcpy(cfg.keyfile, filename);
2209 SetDlgItemText(hwnd, IDC_PKEDIT, cfg.keyfile);
2213 cfg.rawcnp = IsDlgButtonChecked(hwnd, IDC_RAWCNP);
2216 cfg.mouse_is_xterm = IsDlgButtonChecked(hwnd, IDC_MBXTERM);
2222 int n = GetDlgItemInt(hwnd, IDC_CCEDIT, &ok, FALSE);
2227 for (i = 0; i < 128; i++)
2228 if (SendDlgItemMessage
2229 (hwnd, IDC_CCLIST, LB_GETSEL, i, 0)) {
2231 cfg.wordness[i] = n;
2232 SendDlgItemMessage(hwnd, IDC_CCLIST,
2233 LB_DELETESTRING, i, 0);
2234 sprintf(str, "%d\t(0x%02X)\t%c\t%d", i, i,
2235 (i >= 0x21 && i != 0x7F) ? i : ' ',
2237 SendDlgItemMessage(hwnd, IDC_CCLIST,
2244 case IDC_BOLDCOLOUR:
2245 if (HIWORD(wParam) == BN_CLICKED ||
2246 HIWORD(wParam) == BN_DOUBLECLICKED) {
2249 IsDlgButtonChecked(hwnd, IDC_BOLDCOLOUR);
2250 SendDlgItemMessage(hwnd, IDC_COLOURLIST, WM_SETREDRAW,
2253 SendDlgItemMessage(hwnd, IDC_COLOURLIST,
2255 if (n != 12 + 10 * cfg.bold_colour) {
2256 for (i = n; i-- > 0;)
2257 SendDlgItemMessage(hwnd, IDC_COLOURLIST,
2258 LB_DELETESTRING, i, 0);
2259 for (i = 0; i < 22; i++)
2260 if (cfg.bold_colour || permcolour[i])
2261 SendDlgItemMessage(hwnd, IDC_COLOURLIST,
2263 (LPARAM) colours[i]);
2265 SendDlgItemMessage(hwnd, IDC_COLOURLIST, WM_SETREDRAW,
2267 InvalidateRect(GetDlgItem(hwnd, IDC_COLOURLIST), NULL,
2272 if (HIWORD(wParam) == BN_CLICKED ||
2273 HIWORD(wParam) == BN_DOUBLECLICKED)
2275 IsDlgButtonChecked(hwnd, IDC_PALETTE);
2277 case IDC_COLOURLIST:
2278 if (HIWORD(wParam) == LBN_DBLCLK ||
2279 HIWORD(wParam) == LBN_SELCHANGE) {
2281 SendDlgItemMessage(hwnd, IDC_COLOURLIST,
2284 if (!cfg.bold_colour)
2285 i = (i < 3 ? i * 2 : i == 3 ? 5 : i * 2 - 2);
2286 SetDlgItemInt(hwnd, IDC_RVALUE, cfg.colours[i][0],
2288 SetDlgItemInt(hwnd, IDC_GVALUE, cfg.colours[i][1],
2290 SetDlgItemInt(hwnd, IDC_BVALUE, cfg.colours[i][2],
2295 if (HIWORD(wParam) == BN_CLICKED ||
2296 HIWORD(wParam) == BN_DOUBLECLICKED) {
2297 static CHOOSECOLOR cc;
2298 static DWORD custom[16] = { 0 }; /* zero initialisers */
2300 SendDlgItemMessage(hwnd, IDC_COLOURLIST,
2303 if (!cfg.bold_colour)
2304 i = (i < 3 ? i * 2 : i == 3 ? 5 : i * 2 - 2);
2305 cc.lStructSize = sizeof(cc);
2306 cc.hwndOwner = hwnd;
2307 cc.hInstance = (HWND) hinst;
2308 cc.lpCustColors = custom;
2310 RGB(cfg.colours[i][0], cfg.colours[i][1],
2312 cc.Flags = CC_FULLOPEN | CC_RGBINIT;
2313 if (ChooseColor(&cc)) {
2315 (unsigned char) (cc.rgbResult & 0xFF);
2317 (unsigned char) (cc.rgbResult >> 8) & 0xFF;
2319 (unsigned char) (cc.rgbResult >> 16) & 0xFF;
2320 SetDlgItemInt(hwnd, IDC_RVALUE, cfg.colours[i][0],
2322 SetDlgItemInt(hwnd, IDC_GVALUE, cfg.colours[i][1],
2324 SetDlgItemInt(hwnd, IDC_BVALUE, cfg.colours[i][2],
2330 if (HIWORD(wParam) == EN_CHANGE)
2331 GetDlgItemText(hwnd, IDC_CODEPAGE, cfg.line_codepage,
2332 sizeof(cfg.line_codepage) - 1);
2333 if (HIWORD(wParam) == EN_KILLFOCUS) {
2334 int cp = decode_codepage(cfg.line_codepage);
2339 "Unable to identify character set '%s', "
2340 "translation disabled.",
2344 "Character set '%s' is a DBCS, "
2345 "translation is not available.",
2347 MessageBox(hwnd, buf, "PuTTY Error",
2348 MB_ICONERROR | MB_OK);
2350 strcpy(cfg.line_codepage, cp_name(cp));
2351 SetDlgItemText(hwnd, IDC_CODEPAGE, cfg.line_codepage);
2354 case IDC_VTXWINDOWS:
2360 (IsDlgButtonChecked(hwnd, IDC_VTXWINDOWS) ? VT_XWINDOWS
2361 : IsDlgButtonChecked(hwnd,
2362 IDC_VTOEMANSI) ? VT_OEMANSI :
2363 IsDlgButtonChecked(hwnd,
2364 IDC_VTOEMONLY) ? VT_OEMONLY :
2365 IsDlgButtonChecked(hwnd,
2366 IDC_VTUNICODE) ? VT_UNICODE :
2369 case IDC_X11_FORWARD:
2370 if (HIWORD(wParam) == BN_CLICKED ||
2371 HIWORD(wParam) == BN_DOUBLECLICKED)
2373 IsDlgButtonChecked(hwnd, IDC_X11_FORWARD);
2375 case IDC_X11_DISPLAY:
2376 if (HIWORD(wParam) == EN_CHANGE)
2377 GetDlgItemText(hwnd, IDC_X11_DISPLAY, cfg.x11_display,
2378 sizeof(cfg.x11_display) - 1);
2386 /* Grrr Explorer will maximize Dialogs! */
2388 if (wParam == SIZE_MAXIMIZED)
2395 static int CALLBACK MainDlgProc(HWND hwnd, UINT msg,
2396 WPARAM wParam, LPARAM lParam)
2398 if (msg == WM_COMMAND && LOWORD(wParam) == IDOK) {
2400 if (msg == WM_COMMAND && LOWORD(wParam) == IDCX_ABOUT) {
2401 EnableWindow(hwnd, 0);
2402 DialogBox(hinst, MAKEINTRESOURCE(IDD_ABOUTBOX),
2403 GetParent(hwnd), AboutProc);
2404 EnableWindow(hwnd, 1);
2405 SetActiveWindow(hwnd);
2407 return GenericMainDlgProc(hwnd, msg, wParam, lParam, 0);
2410 static int CALLBACK ReconfDlgProc(HWND hwnd, UINT msg,
2411 WPARAM wParam, LPARAM lParam)
2413 return GenericMainDlgProc(hwnd, msg, wParam, lParam, 1);
2416 void defuse_showwindow(void)
2419 * Work around the fact that the app's first call to ShowWindow
2420 * will ignore the default in favour of the shell-provided
2425 hwnd = CreateDialog(hinst, MAKEINTRESOURCE(IDD_ABOUTBOX),
2427 ShowWindow(hwnd, SW_HIDE);
2428 DestroyWindow(hwnd);
2437 savedsession[0] = '\0';
2439 DialogBox(hinst, MAKEINTRESOURCE(IDD_MAINBOX), NULL, MainDlgProc);
2440 get_sesslist(FALSE);
2445 int do_reconfig(HWND hwnd)
2450 backup_cfg = cfg; /* structure copy */
2452 DialogBox(hinst, MAKEINTRESOURCE(IDD_RECONF), hwnd, ReconfDlgProc);
2454 cfg = backup_cfg; /* structure copy */
2459 void logevent(char *string)
2464 if (nevents >= negsize) {
2466 events = srealloc(events, negsize * sizeof(*events));
2470 strftime(timebuf, sizeof(timebuf), "%Y-%m-%d %H:%M:%S\t",
2473 events[nevents] = smalloc(strlen(timebuf) + strlen(string) + 1);
2474 strcpy(events[nevents], timebuf);
2475 strcat(events[nevents], string);
2478 SendDlgItemMessage(logbox, IDN_LIST, LB_ADDSTRING,
2479 0, (LPARAM) events[nevents]);
2480 count = SendDlgItemMessage(logbox, IDN_LIST, LB_GETCOUNT, 0, 0);
2481 SendDlgItemMessage(logbox, IDN_LIST, LB_SETTOPINDEX, count - 1, 0);
2486 void showeventlog(HWND hwnd)
2489 logbox = CreateDialog(hinst, MAKEINTRESOURCE(IDD_LOGBOX),
2491 ShowWindow(logbox, SW_SHOWNORMAL);
2493 SetActiveWindow(logbox);
2496 void showabout(HWND hwnd)
2498 DialogBox(hinst, MAKEINTRESOURCE(IDD_ABOUTBOX), hwnd, AboutProc);
2501 void verify_ssh_host_key(char *host, int port, char *keytype,
2502 char *keystr, char *fingerprint)
2506 static const char absentmsg[] =
2507 "The server's host key is not cached in the registry. You\n"
2508 "have no guarantee that the server is the computer you\n"
2510 "The server's key fingerprint is:\n"
2512 "If you trust this host, hit Yes to add the key to\n"
2513 "PuTTY's cache and carry on connecting.\n"
2514 "If you do not trust this host, hit No to abandon the\n"
2517 static const char wrongmsg[] =
2518 "WARNING - POTENTIAL SECURITY BREACH!\n"
2520 "The server's host key does not match the one PuTTY has\n"
2521 "cached in the registry. This means that either the\n"
2522 "server administrator has changed the host key, or you\n"
2523 "have actually connected to another computer pretending\n"
2524 "to be the server.\n"
2525 "The new key fingerprint is:\n"
2527 "If you were expecting this change and trust the new key,\n"
2528 "hit Yes to update PuTTY's cache and continue connecting.\n"
2529 "If you want to carry on connecting but without updating\n"
2530 "the cache, hit No.\n"
2531 "If you want to abandon the connection completely, hit\n"
2532 "Cancel. Hitting Cancel is the ONLY guaranteed safe\n" "choice.\n";
2534 static const char mbtitle[] = "PuTTY Security Alert";
2538 /* sensible fingerprint max size */
2539 (sizeof(absentmsg) > sizeof(wrongmsg) ?
2540 sizeof(absentmsg) : sizeof(wrongmsg))];
2543 * Verify the key against the registry.
2545 ret = verify_host_key(host, port, keytype, keystr);
2547 if (ret == 0) /* success - key matched OK */
2549 if (ret == 2) { /* key was different */
2551 sprintf(message, wrongmsg, fingerprint);
2552 mbret = MessageBox(NULL, message, mbtitle,
2553 MB_ICONWARNING | MB_YESNOCANCEL);
2555 store_host_key(host, port, keytype, keystr);
2556 if (mbret == IDCANCEL)
2559 if (ret == 1) { /* key was absent */
2561 sprintf(message, absentmsg, fingerprint);
2562 mbret = MessageBox(NULL, message, mbtitle,
2563 MB_ICONWARNING | MB_YESNO);
2566 store_host_key(host, port, keytype, keystr);
2571 * Ask whether to wipe a session log file before writing to it.
2572 * Returns 2 for wipe, 1 for append, 0 for cancel (don't log).
2574 int askappend(char *filename)
2576 static const char mbtitle[] = "PuTTY Log to File";
2577 static const char msgtemplate[] =
2578 "The session log file \"%.*s\" already exists.\n"
2579 "You can overwrite it with a new session log,\n"
2580 "append your session log to the end of it,\n"
2581 "or disable session logging for this session.\n"
2582 "Hit Yes to wipe the file, No to append to it,\n"
2583 "or Cancel to disable logging.";
2584 char message[sizeof(msgtemplate) + FILENAME_MAX];
2586 if (cfg.logxfovr != LGXF_ASK) {
2587 return ((cfg.logxfovr == LGXF_OVR) ? 2 : 1);
2589 sprintf(message, msgtemplate, FILENAME_MAX, filename);
2591 mbret = MessageBox(NULL, message, mbtitle,
2592 MB_ICONQUESTION | MB_YESNOCANCEL);
2595 else if (mbret == IDNO)