diff --git a/include/keyboard.h b/include/keyboard.h index ec9bff46..a3d02d0e 100644 --- a/include/keyboard.h +++ b/include/keyboard.h @@ -32,7 +32,9 @@ enum { Kalt= KF|0x15, Kshift= KF|0x16, - Kctl= KF|0x17 + Kctl= KF|0x17, + + Kcmd= 0xF100 /* Rune: beginning of Cmd+'a', Cmd+'A', etc on Mac */ }; #if defined(__cplusplus) diff --git a/src/cmd/acme/text.c b/src/cmd/acme/text.c index 729e4303..30f5804d 100644 --- a/src/cmd/acme/text.c +++ b/src/cmd/acme/text.c @@ -734,6 +734,11 @@ texttype(Text *t, Rune r) q0++; textshow(t, q0, q0, TRUE); return; + case Kcmd+'c': /* %C: copy */ + typecommit(t); + cut(t, t, nil, TRUE, FALSE, nil, 0); + return; + Tagdown: /* expand tag to show all text */ if(!t->w->tagexpand){ @@ -755,6 +760,27 @@ texttype(Text *t, Rune r) seq++; filemark(t->file); } + /* cut/paste must be done after the seq++/filemark */ + switch(r){ + case Kcmd+'x': /* %X: cut */ + typecommit(t); + if(t->what == Body){ + seq++; + filemark(t->file); + } + cut(t, t, nil, TRUE, TRUE, nil, 0); + textshow(t, t->q0, t->q0, 1); + return; + case Kcmd+'v': /* %V: paste */ + typecommit(t); + if(t->what == Body){ + seq++; + filemark(t->file); + } + paste(t, t, nil, TRUE, FALSE, nil, 0); + textshow(t, t->q0, t->q1, 1); + return; + } if(t->q1 > t->q0){ if(t->ncache != 0) error("text.type"); diff --git a/src/cmd/devdraw/osx-screen.c b/src/cmd/devdraw/osx-screen.c index 4b1f6ede..76416cba 100644 --- a/src/cmd/devdraw/osx-screen.c +++ b/src/cmd/devdraw/osx-screen.c @@ -2,7 +2,6 @@ #define Rect OSXRect #define Cursor OSXCursor #include -#include // for full screen #undef Rect #undef Point #undef Cursor @@ -23,7 +22,6 @@ #include "glendapng.h" AUTOFRAMEWORK(Carbon) -AUTOFRAMEWORK(QuickTime) #define panic sysfatal @@ -52,6 +50,8 @@ struct { PasteboardRef snarf; int needflush; QLock flushlock; + int active; + int infullscreen; } osx; enum @@ -66,8 +66,9 @@ enum static void screenproc(void*); static void eresized(int); -static void fullscreen(void); +static void fullscreen(int); static void seticon(void); +static void activated(int); static OSStatus quithandler(EventHandlerCallRef, EventRef, void*); static OSStatus eventhandler(EventHandlerCallRef, EventRef, void*); @@ -159,7 +160,9 @@ _screeninit(void) const EventTypeSpec cmds[] = { { kEventClassWindow, kEventWindowClosed }, { kEventClassWindow, kEventWindowBoundsChanged }, - { kEventClassCommand, kEventCommandProcess } + { kEventClassCommand, kEventCommandProcess }, + { kEventClassWindow, kEventWindowActivated }, + { kEventClassWindow, kEventWindowDeactivated }, }; const EventTypeSpec events[] = { { kEventClassKeyboard, kEventRawKeyDown }, @@ -256,7 +259,7 @@ eventhandler(EventHandlerCallRef next, EventRef event, void *arg) exit(0); case CmdFullScreen: - fullscreen(); + fullscreen(1); break; default: @@ -273,6 +276,14 @@ eventhandler(EventHandlerCallRef next, EventRef event, void *arg) eresized(1); break; + case kEventWindowActivated: + activated(1); + return eventNotHandledErr; + + case kEventWindowDeactivated: + activated(0); + return eventNotHandledErr; + default: return eventNotHandledErr; } @@ -419,7 +430,22 @@ kbdevent(EventRef event) if(mod == cmdKey){ if(ch == 'F' || ch == 'f'){ if(osx.isfullscreen && msec() - osx.fullscreentime > 500) - fullscreen(); + fullscreen(0); + return noErr; + } + + // Pass most Cmd keys through as Kcmd + ch. + // OS X interprets a few no matter what we do, + // so it is useless to pass them through as keystrokes too. + switch(ch) { + case 'm': // minimize window + case 'h': // hide window + case 'H': // hide others + case 'q': // quit + return eventNotHandledErr; + } + if(' ' <= ch && ch <= '~') { + keystroke(Kcmd + ch); return noErr; } return eventNotHandledErr; @@ -472,7 +498,7 @@ eresized(int new) CGDataProviderRef provider; CGImageRef image; CGColorSpaceRef cspace; - + GetWindowBounds(osx.window, kWindowContentRgn, &or); r = Rect(or.left, or.top, or.right, or.bottom); if(Dx(r) == Dx(osx.screenr) && Dy(r) == Dy(osx.screenr)){ @@ -561,38 +587,51 @@ _flushmemscreen(Rectangle r) } void -fullscreen(void) +activated(int active) { - static Ptr restore; - static WindowRef oldwindow; - GDHandle device; - - qlock(&osx.flushlock); - if(osx.isfullscreen){ - if(osx.windowctx){ - QDEndCGContext(GetWindowPort(osx.window), &osx.windowctx); - osx.windowctx = nil; - } - EndFullScreen(restore, 0); - osx.window = oldwindow; - ShowWindow(osx.window); - osx.isfullscreen = 0; - }else{ - if(osx.windowctx){ - QDEndCGContext(GetWindowPort(osx.window), &osx.windowctx); - osx.windowctx = nil; - } - HideWindow(osx.window); - oldwindow = osx.window; - GetWindowGreatestAreaDevice(osx.window, kWindowTitleBarRgn, &device, nil); - BeginFullScreen(&restore, device, 0, 0, &osx.window, 0, 0); - osx.isfullscreen = 1; - osx.fullscreentime = msec(); - } - qunlock(&osx.flushlock); - eresized(1); + osx.active = active; } +void +fullscreen(int wascmd) +{ + static OSXRect oldrect; + GDHandle device; + OSXRect dr; + + if(!wascmd) + return; + + if(!osx.isfullscreen){ + GetWindowGreatestAreaDevice(osx.window, + kWindowTitleBarRgn, &device, nil); + dr = (*device)->gdRect; + if(dr.top == 0 && dr.left == 0) + HideMenuBar(); + GetWindowBounds(osx.window, kWindowContentRgn, &oldrect); + ChangeWindowAttributes(osx.window, + kWindowNoTitleBarAttribute, + kWindowResizableAttribute); + MoveWindow(osx.window, 0, 0, 1); + MoveWindow(osx.window, dr.left, dr.top, 0); + SizeWindow(osx.window, + dr.right - dr.left, + dr.bottom - dr.top, 0); + osx.isfullscreen = 1; + }else{ + ShowMenuBar(); + ChangeWindowAttributes(osx.window, + kWindowResizableAttribute, + kWindowNoTitleBarAttribute); + SizeWindow(osx.window, + oldrect.right - oldrect.left, + oldrect.bottom - oldrect.top, 0); + MoveWindow(osx.window, oldrect.left, oldrect.top, 0); + osx.isfullscreen = 0; + } + eresized(1); +} + void setmouse(Point p) { diff --git a/src/cmd/samterm/main.c b/src/cmd/samterm/main.c index bf0d9a55..5b645e84 100644 --- a/src/cmd/samterm/main.c +++ b/src/cmd/samterm/main.c @@ -494,6 +494,9 @@ flushtyping(int clearesc) #define PAGEUP Kpgup #define RIGHTARROW Kright #define SCROLLKEY Kdown +#define CUT (Kcmd+'x') +#define COPY (Kcmd+'c') +#define PASTE (Kcmd+'v') int nontypingkey(int c) @@ -511,6 +514,8 @@ nontypingkey(int c) case SCROLLKEY: return 1; } + if(c >= Kcmd) + return 1; return 0; } @@ -673,6 +678,20 @@ type(Flayer *l, int res) /* what a bloody mess this is */ for(l=t->l; l<&t->l[NL]; l++) if(l->textfn) flsetselect(l, l->p0, l->p1); + switch(c) { + case CUT: + flushtyping(0); + cut(t, t->front, 1, 1); + break; + case COPY: + flushtyping(0); + snarf(t, t->front); + break; + case PASTE: + flushtyping(0); + paste(t, t->front); + break; + } } } diff --git a/src/libdraw/drawclient.c b/src/libdraw/drawclient.c index f3c42b80..361cb0ac 100644 --- a/src/libdraw/drawclient.c +++ b/src/libdraw/drawclient.c @@ -45,8 +45,17 @@ _displayconnect(Display *d) * The NOLIBTHREADDAEMONIZE keeps devdraw from * forking before threadmain. OS X hates it when * guis fork. + * + * If client didn't use ARGBEGIN, argv0 == nil. + * Can't send nil through because OS X expects + * argv[0] to be non-nil. Also, OS X apparently + * expects argv[0] to be a valid executable name, + * so "(argv0)" is not okay. Use "devdraw" + * instead. */ putenv("NOLIBTHREADDAEMONIZE", "1"); + if(argv0 == nil) + argv0 = "devdraw"; execl("devdraw", argv0, argv0, "(devdraw)", nil); sysfatal("exec devdraw: %r"); } diff --git a/src/libmemdraw/draw.c b/src/libmemdraw/draw.c index 2d8681e5..7616997b 100644 --- a/src/libmemdraw/draw.c +++ b/src/libmemdraw/draw.c @@ -10,22 +10,34 @@ static int tablesbuilt; #define RGB2K(r,g,b) ((156763*(r)+307758*(g)+59769*(b))>>19) /* - * for 0 ≤ x ≤ 255*255, (x*0x0101+0x100)>>16 is a perfect approximation. - * for 0 ≤ x < (1<<16), x/255 = ((x+1)*0x0101)>>16 is a perfect approximation. - * the last one is perfect for all up to 1<<16, avoids a multiply, but requires a rathole. + * For 16-bit values, x / 255 == (t = x+1, (t+(t>>8)) >> 8). + * We add another 127 to round to the nearest value rather + * than truncate. + * + * CALCxy does x bytewise calculations on y input images (x=1,4; y=1,2). + * CALC2x does two parallel 16-bit calculations on y input images (y=1,2). */ -/* #define DIV255(x) (((x)*257+256)>>16) */ -#define DIV255(x) ((((x)+1)*257)>>16) -/* #define DIV255(x) (tmp=(x)+1, (tmp+(tmp>>8))>>8) */ +#define CALC11(a, v, tmp) \ + (tmp=(a)*(v)+128, (tmp+(tmp>>8))>>8) -#define MUL(x, y, t) (t = (x)*(y)+128, (t+(t>>8))>>8) -#define MASK13 0xFF00FF00 -#define MASK02 0x00FF00FF -#define MUL13(a, x, t) (t = (a)*(((x)&MASK13)>>8)+128, ((t+((t>>8)&MASK02))>>8)&MASK02) -#define MUL02(a, x, t) (t = (a)*(((x)&MASK02)>>0)+128, ((t+((t>>8)&MASK02))>>8)&MASK02) -#define MUL0123(a, x, s, t) ((MUL13(a, x, s)<<8)|MUL02(a, x, t)) +#define CALC12(a1, v1, a2, v2, tmp) \ + (tmp=(a1)*(v1)+(a2)*(v2)+128, (tmp+(tmp>>8))>>8) -#define MUL2(u, v, x, y) (t = (u)*(v)+(x)*(y)+256, (t+(t>>8))>>8) +#define MASK 0xFF00FF + +#define CALC21(a, vvuu, tmp) \ + (tmp=(a)*(vvuu)+0x00800080, ((tmp+((tmp>>8)&MASK))>>8)&MASK) + +#define CALC41(a, rgba, tmp1, tmp2) \ + (CALC21(a, rgba & MASK, tmp1) | \ + (CALC21(a, (rgba>>8)&MASK, tmp2)<<8)) + +#define CALC22(a1, vvuu1, a2, vvuu2, tmp) \ + (tmp=(a1)*(vvuu1)+(a2)*(vvuu2)+0x00800080, ((tmp+((tmp>>8)&MASK))>>8)&MASK) + +#define CALC42(a1, rgba1, a2, rgba2, tmp1, tmp2) \ + (CALC22(a1, rgba1 & MASK, a2, rgba2 & MASK, tmp1) | \ + (CALC22(a1, (rgba1>>8) & MASK, a2, (rgba2>>8) & MASK, tmp2)<<8)) static void mktables(void); typedef int Subdraw(Memdrawparam*); @@ -786,41 +798,85 @@ alphacalc0(Buffer bdst, Buffer b1, Buffer b2, int dx, int grey, int op) return bdst; } +/* + * Do the channels in the buffers match enough + * that we can do word-at-a-time operations + * on the pixels? + */ +static int +chanmatch(Buffer *bdst, Buffer *bsrc) +{ + uchar *drgb, *srgb; + + /* + * first, r, g, b must be in the same place + * in the rgba word. + */ + drgb = (uchar*)bdst->rgba; + srgb = (uchar*)bsrc->rgba; + if(bdst->red - drgb != bsrc->red - srgb + || bdst->blu - drgb != bsrc->blu - srgb + || bdst->grn - drgb != bsrc->grn - srgb) + return 0; + + /* + * that implies alpha is in the same place, + * if it is there at all (it might be == &ones). + * if the destination is &ones, we can scribble + * over the rgba slot just fine. + */ + if(bdst->alpha == &ones) + return 1; + + /* + * if the destination is not ones but the src is, + * then the simultaneous calculation will use + * bogus bytes from the src's rgba. no good. + */ + if(bsrc->alpha == &ones) + return 0; + + /* + * otherwise, alphas are in the same place. + */ + return 1; +} + static Buffer alphacalc14(Buffer bdst, Buffer bsrc, Buffer bmask, int dx, int grey, int op) { Buffer obdst; int fd, sadelta; int i, sa, ma, q; - u32int s, t; + u32int t, t1; obdst = bdst; sadelta = bsrc.alpha == &ones ? 0 : bsrc.delta; - q = bsrc.delta == 4 && bdst.delta == 4; + q = bsrc.delta == 4 && bdst.delta == 4 && chanmatch(&bdst, &bsrc); for(i=0; i>11)&31; @@ -171,7 +169,7 @@ Bprintr5g6b5(Biobuf *bio, char*, u32int v) } static void -Bprintr5g5b5a1(Biobuf *bio, char*, u32int v) +Bprintr5g5b5a1(Biobuf *bio, char* _, u32int v) { int r,g,b,a; r = (v>>11)&31; @@ -974,7 +972,7 @@ drawonepixel(Memimage *dst, Point dp, Memimage *src, Point sp, Memimage *mask, P pixtorgba(getpixel(dst, dp), &dr, &dg, &db, &da); pixtorgba(getpixel(src, sp), &sr, &sg, &sb, &sa); m = getmask(mask, mp); - M = 255-(sa*m)/255; + M = 255-(sa*m + 127)/255; DBG print("dst %x %x %x %x src %x %x %x %x m %x = ", dr,dg,db,da, sr,sg,sb,sa, m); if(dst->flags&Fgrey){ @@ -985,18 +983,18 @@ DBG print("dst %x %x %x %x src %x %x %x %x m %x = ", dr,dg,db,da, sr,sg,sb,sa, m */ sk = RGB2K(sr, sg, sb); dk = RGB2K(dr, dg, db); - dk = (sk*m + dk*M)/255; + dk = (sk*m + dk*M + 127)/255; dr = dg = db = dk; - da = (sa*m + da*M)/255; + da = (sa*m + da*M + 127)/255; }else{ /* * True color alpha calculation treats all channels (including alpha) * the same. It might have been nice to use an array, but oh well. */ - dr = (sr*m + dr*M)/255; - dg = (sg*m + dg*M)/255; - db = (sb*m + db*M)/255; - da = (sa*m + da*M)/255; + dr = (sr*m + dr*M + 127)/255; + dg = (sg*m + dg*M + 127)/255; + db = (sb*m + db*M + 127)/255; + da = (sa*m + da*M + 127)/255; } DBG print("%x %x %x %x\n", dr,dg,db,da);