/* $Id: macterm.c,v 1.1.2.14 1999/03/03 22:03:54 ben Exp $ */ /* * Copyright (c) 1999 Ben Harris * All rights reserved. * * Permission is hereby granted, free of charge, to any person * obtaining a copy of this software and associated documentation * files (the "Software"), to deal in the Software without * restriction, including without limitation the rights to use, * copy, modify, merge, publish, distribute, sublicense, and/or * sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following * conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF * CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ /* * macterm.c -- Macintosh terminal front-end */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include "macresid.h" #include "putty.h" #include "mac.h" #define DEFAULT_FG 16 #define DEFAULT_FG_BOLD 17 #define DEFAULT_BG 18 #define DEFAULT_BG_BOLD 19 struct mac_session { short fontnum; int font_ascent; WindowPtr window; PaletteHandle palette; ControlHandle scrollbar; }; static void mac_initfont(struct mac_session *); static void mac_initpalette(struct mac_session *); static void mac_adjustsize(struct mac_session *, int, int); static pascal void mac_scrolltracker(ControlHandle, short); static pascal void do_text_for_device(short, short, GDHandle, long); /* * Temporary hack till I get the terminal emulator supporting multiple * sessions */ static struct mac_session *onlysession; static void inbuf_putc(int c) { inbuf[inbuf_head] = c; inbuf_head = (inbuf_head+1) & INBUF_MASK; } static void inbuf_putstr(const char *c) { while (*c) inbuf_putc(*c++); } static void display_resource(unsigned long type, short id) { Handle h; int len, i; char *t; h = GetResource(type, id); if (h == NULL) fatalbox("Can't get test resource"); SetResAttrs(h, GetResAttrs(h) | resLocked); t = *h; len = GetResourceSizeOnDisk(h); for (i = 0; i < len; i++) { inbuf_putc(t[i]); term_out(); } SetResAttrs(h, GetResAttrs(h) & ~resLocked); ReleaseResource(h); } void mac_newsession(void) { struct mac_session *s; int i; /* This should obviously be initialised by other means */ mac_loadconfig(&cfg); s = smalloc(sizeof(*s)); onlysession = s; /* XXX: Own storage management? */ if (mac_gestalts.qdvers == gestaltOriginalQD) s->window = GetNewWindow(wTerminal, NULL, (WindowPtr)-1); else s->window = GetNewCWindow(wTerminal, NULL, (WindowPtr)-1); SetWRefCon(s->window, (long)s); s->scrollbar = GetNewControl(cVScroll, s->window); term_init(); term_size(cfg.height, cfg.width, cfg.savelines); mac_initfont(s); mac_initpalette(s); /* Set to FALSE to not get palette updates in the background. */ SetPalette(s->window, s->palette, TRUE); ActivatePalette(s->window); ShowWindow(s->window); display_resource('pTST', 128); } static void mac_initfont(struct mac_session *s) { Str255 macfont; FontInfo fi; SetPort(s->window); macfont[0] = sprintf((char *)&macfont[1], "%s", cfg.font); GetFNum(macfont, &s->fontnum); TextFont(s->fontnum); TextFace(cfg.fontisbold ? bold : 0); TextSize(cfg.fontheight); GetFontInfo(&fi); font_width = fi.widMax; font_height = fi.ascent + fi.descent + fi.leading; s->font_ascent = fi.ascent; mac_adjustsize(s, rows, cols); } /* * To be called whenever the window size changes. * rows and cols should be desired values. * It's assumed the terminal emulator will be informed, and will set rows * and cols for us. */ static void mac_adjustsize(struct mac_session *s, int newrows, int newcols) { int winwidth, winheight; winwidth = newcols * font_width + 15; winheight = newrows * font_height; SizeWindow(s->window, winwidth, winheight, true); HideControl(s->scrollbar); MoveControl(s->scrollbar, winwidth - 15, -1); SizeControl(s->scrollbar, 16, winheight - 13); ShowControl(s->scrollbar); } static void mac_initpalette(struct mac_session *s) { WinCTab ct; if (mac_gestalts.qdvers == gestaltOriginalQD) return; s->palette = NewPalette((*cfg.colours)->pmEntries, NULL, pmCourteous, 0); if (s->palette == NULL) fatalbox("Unable to create palette"); CopyPalette(cfg.colours, s->palette, 0, 0, (*cfg.colours)->pmEntries); } /* * I don't think this is (a) safe or (b) a good way to do this. */ static void mac_updatewinbg(struct mac_session *s) { WinCTab ct; WCTabPtr ctp = &ct; WCTabHandle cth = &ctp; ct.wCSeed = 0; ct.wCReserved = 0; ct.ctSize = 1; ct.ctTable[0].value = wContentColor; ct.ctTable[0].rgb = (*s->palette)->pmInfo[16].ciRGB; SetWinColor(s->window, cth); } void mac_clickterm(WindowPtr window, EventRecord *event) { struct mac_session *s; Point mouse; ControlHandle control; int part; s = (struct mac_session *)GetWRefCon(window); SetPort(window); mouse = event->where; GlobalToLocal(&mouse); part = FindControl(mouse, window, &control); if (control == s->scrollbar) { switch (part) { case kControlIndicatorPart: if (TrackControl(control, mouse, NULL) == kControlIndicatorPart) term_scroll(+1, GetControlValue(control)); break; case kControlUpButtonPart: case kControlDownButtonPart: case kControlPageUpPart: case kControlPageDownPart: TrackControl(control, mouse, mac_scrolltracker); break; } } } static pascal void mac_scrolltracker(ControlHandle control, short part) { struct mac_session *s; s = (struct mac_session *)GetWRefCon((*control)->contrlOwner); switch (part) { case kControlUpButtonPart: term_scroll(0, -1); break; case kControlDownButtonPart: term_scroll(0, +1); break; case kControlPageUpPart: term_scroll(0, -(rows - 1)); break; case kControlPageDownPart: term_scroll(0, +(rows - 1)); break; } } void mac_growterm(WindowPtr window, EventRecord *event) { Rect limits; long grow_result; int newrows, newcols; struct mac_session *s; s = (struct mac_session *)GetWRefCon(window); SetRect(&limits, font_width + 15, font_height, SHRT_MAX, SHRT_MAX); grow_result = GrowWindow(window, event->where, &limits); if (grow_result != 0) { newrows = HiWord(grow_result) / font_height; newcols = (LoWord(grow_result) - 15) / font_width; mac_adjustsize(s, newrows, newcols); term_size(newrows, newcols, cfg.savelines); } } void mac_activateterm(WindowPtr window, Boolean active) { struct mac_session *s; s = (struct mac_session *)GetWRefCon(window); if (active) ShowControl(s->scrollbar); else HideControl(s->scrollbar); } void mac_updateterm(WindowPtr window) { struct mac_session *s; Rect clip; s = (struct mac_session *)GetWRefCon(window); BeginUpdate(window); term_paint(s, (*window->visRgn)->rgnBBox.left, (*window->visRgn)->rgnBBox.top, (*window->visRgn)->rgnBBox.right, (*window->visRgn)->rgnBBox.bottom); /* Restore default colours in case the Window Manager uses them */ PmForeColor(16); PmBackColor(18); if (FrontWindow() != window) EraseRect(&(*s->scrollbar)->contrlRect); UpdateControls(window, window->visRgn); /* Stop DrawGrowIcon giving us space for a horizontal scrollbar */ SetRect(&clip, window->portRect.right - 15, SHRT_MIN, SHRT_MAX, SHRT_MAX); ClipRect(&clip); DrawGrowIcon(window); clip.left = SHRT_MIN; ClipRect(&clip); EndUpdate(window); } struct do_text_args { struct mac_session *s; Rect textrect; char *text; int len; unsigned long attr; }; /* * Call from the terminal emulator to draw a bit of text * * x and y are text row and column (zero-based) */ void do_text(struct mac_session *s, int x, int y, char *text, int len, unsigned long attr) { int style = 0; int bgcolour, fgcolour; RGBColor rgbfore, rgbback; struct do_text_args a; RgnHandle textrgn; SetPort(s->window); /* First check this text is relevant */ a.textrect.top = y * font_height; a.textrect.bottom = (y + 1) * font_height; a.textrect.left = x * font_width; a.textrect.right = (x + len) * font_width; if (!RectInRgn(&a.textrect, s->window->visRgn)) return; a.s = s; a.text = text; a.len = len; a.attr = attr; SetPort(s->window); TextFont(s->fontnum); if (cfg.fontisbold || (attr & ATTR_BOLD) && !cfg.bold_colour) style |= bold; if (attr & ATTR_UNDER) style |= underline; TextFace(style); TextSize(cfg.fontheight); if (attr & ATTR_REVERSE) TextMode(notSrcCopy); else TextMode(srcCopy); SetFractEnable(FALSE); /* We want characters on pixel boundaries */ textrgn = NewRgn(); RectRgn(textrgn, &a.textrect); DeviceLoop(textrgn, do_text_for_device, (long)&a, 0); /* Tell the window manager about it in case this isn't an update */ DisposeRgn(textrgn); ValidRect(&a.textrect); } static pascal void do_text_for_device(short depth, short devflags, GDHandle device, long cookie) { struct do_text_args *a; int bgcolour, fgcolour; a = (struct do_text_args *)cookie; switch (depth) { case 1: /* XXX This should be done with a _little_ more configurability */ ForeColor(whiteColor); BackColor(blackColor); break; case 2: if ((a->attr & ATTR_BOLD) && cfg.bold_colour) PmForeColor(DEFAULT_FG_BOLD); else PmForeColor(DEFAULT_FG); PmBackColor(DEFAULT_BG); break; default: fgcolour = ((a->attr & ATTR_FGMASK) >> ATTR_FGSHIFT) * 2; bgcolour = ((a->attr & ATTR_BGMASK) >> ATTR_BGSHIFT) * 2; if ((a->attr & ATTR_BOLD) && cfg.bold_colour) if (a->attr & ATTR_REVERSE) bgcolour++; else fgcolour++; PmForeColor(fgcolour); PmBackColor(bgcolour); break; } MoveTo(a->textrect.left, a->textrect.top + a->s->font_ascent); DrawText(a->text, 0, a->len); } /* * Call from the terminal emulator to get its graphics context. */ struct mac_session *get_ctx(void) { return onlysession; } /* * Presumably this does something in Windows */ void free_ctx(struct mac_session *ctx) { } /* * Set the scroll bar position * * total is the line number of the bottom of the working screen * start is the line number of the top of the display * page is the length of the displayed page */ void set_sbar(int total, int start, int page) { struct mac_session *s = onlysession; /* We don't redraw until we've set everything up, to avoid glitches */ (*s->scrollbar)->contrlMin = 0; (*s->scrollbar)->contrlMax = total - page; SetControlValue(s->scrollbar, start); #if 0 /* XXX: This doesn't link for me. */ if (mac_gestalts.cntlattr & gestaltControlMgrPresent) SetControlViewSize(s->scrollbar, page); #endif } /* * Beep */ void beep(void) { SysBeep(30); /* * XXX We should indicate the relevant window and/or use the * Notification Manager */ } /* * Set icon string -- a no-op here (Windowshade?) */ void set_icon(char *icon) { } /* * Set the window title */ void set_title(char *title) { Str255 mactitle; struct mac_session *s = onlysession; mactitle[0] = sprintf((char *)&mactitle[1], "%s", title); SetWTitle(s->window, mactitle); } /* * Resize the window at the emulator's request */ void request_resize(int w, int h) { cols = w; rows = h; mac_initfont(onlysession); } /* * Set the logical palette */ void palette_set(int n, int r, int g, int b) { RGBColor col; struct mac_session *s = onlysession; static const int first[21] = { 0, 2, 4, 6, 8, 10, 12, 14, 1, 3, 5, 7, 9, 11, 13, 15, 16, 17, 18, 20, 22 }; if (mac_gestalts.qdvers == gestaltOriginalQD) return; col.red = r * 0x0101; col.green = g * 0x0101; col.blue = b * 0x0101; SetEntryColor(s->palette, first[n], &col); if (first[n] >= 18) SetEntryColor(s->palette, first[n]+1, &col); ActivatePalette(s->window); } /* * Reset to the default palette */ void palette_reset(void) { struct mac_session *s = onlysession; if (mac_gestalts.qdvers == gestaltOriginalQD) return; CopyPalette(cfg.colours, s->palette, 0, 0, (*cfg.colours)->pmEntries); ActivatePalette(s->window); /* Palette Manager will generate update events as required. */ } /* * Scroll the screen. (`lines' is +ve for scrolling forward, -ve * for backward.) */ void do_scroll(int topline, int botline, int lines) { struct mac_session *s = onlysession; Rect r; RgnHandle update; SetPort(s->window); update = NewRgn(); SetRect(&r, 0, topline * font_height, cols * font_width, (botline + 1) * font_height); ScrollRect(&r, 0, - lines * font_height, update); /* XXX: move update region? */ InvalRgn(update); DisposeRgn(update); }