ref: 0a7d581eec1319c643337d302176b3f9ba565ea5
dir: /sys/src/cmd/ktrans/main.c/
#include <u.h> #include <libc.h> #include <ctype.h> #include <bio.h> #include <plumb.h> #include <thread.h> #include <draw.h> #include <mouse.h> #include <keyboard.h> #include "hash.h" char* pushutf(char *dst, char *e, char *u, int nrune) { Rune r; char *p; char *d; if(dst >= e) return dst; d = dst; p = u; while(d < e-1){ if(isascii(*p)){ if((*d = *p) == '\0') return d; p++; d++; } else { p += chartorune(&r, p); if(r == Runeerror){ *d = '\0'; return d; } d += runetochar(d, &r); } if(nrune > 0 && --nrune == 0) break; } if(d > e-1) d = e-1; *d = '\0'; return d; } char* peekstr(char *s, char *b) { while(s > b && (*--s & 0xC0)==Runesync) ; return s; } typedef struct Str Str; struct Str { char b[128]; char *p; }; #define strend(s) ((s)->b + sizeof (s)->b) void resetstr(Str *s, ...) { va_list args; va_start(args, s); do { s->p = s->b; s->p[0] = '\0'; s = va_arg(args, Str*); } while(s != nil); va_end(args); } void popstr(Str *s) { s->p = peekstr(s->p, s->b); s->p[0] = '\0'; } typedef struct Map Map; struct Map { char *roma; char *kana; char leadstomore; }; Hmap* openmap(char *file) { Biobuf *b; char *s; Map map; Hmap *h; char *key, *val; Str partial; Rune r; h = hmapalloc(64, sizeof(Map)); b = Bopen(file, OREAD); if(b == nil) return nil; while(key = Brdstr(b, '\n', 1)){ if(key[0] == '\0'){ Err: free(key); continue; } val = strchr(key, '\t'); if(val == nil || val[1] == '\0') goto Err; *val = '\0'; val++; resetstr(&partial, nil); for(s = key; *s; s += chartorune(&r, s)){ partial.p = pushutf(partial.p, strend(&partial), s, 1); map.leadstomore = 0; if(hmapget(h, partial.b, &map) == 0){ if(map.leadstomore == 1 && s[1] == '\0') map.leadstomore = 1; } if(s[1] == '\0'){ map.roma = key; map.kana = val; hmaprepl(&h, strdup(map.roma), &map, nil, 1); } else { map.roma = strdup(partial.b); map.leadstomore = 1; map.kana = nil; hmaprepl(&h, strdup(partial.b), &map, nil, 1); } } } Bterm(b); return h; } enum{ Maxkouho=32, }; Hmap* opendict(Hmap *h, char *name) { Biobuf *b; char *p; char *dot, *rest; char *kouho[Maxkouho]; int i; b = Bopen(name, OREAD); if(b == nil) return nil; if(h == nil) h = hmapalloc(8192, sizeof(kouho)); while(p = Brdstr(b, '\n', 1)){ if(p[0] == '\0' || p[0] == ';'){ Err: free(p); continue; } dot = strchr(p, '\t'); if(dot == nil) goto Err; *dot = '\0'; rest = dot+1; if(*rest == '\0') goto Err; memset(kouho, 0, sizeof kouho); i = 0; while(i < nelem(kouho)-1 && (dot = utfrune(rest, ' '))){ *dot = '\0'; kouho[i++] = rest; rest = dot+1; } if(i < nelem(kouho)-1) kouho[i] = rest; /* key is the base pointer; overwrites clean up for us */ hmaprepl(&h, p, kouho, nil, 1); } Bterm(b); return h; } enum{ LangEN = '', // ^t LangJP = '', // ^n LangJPK = '', // ^k LangKO = '', // ^s LangZH = '', // ^c LangVN = '', // ^v }; int deflang; Hmap *natural; Hmap *hira, *kata, *jisho; Hmap *hangul; Hmap *judou, *zidian; Hmap *telex; Hmap **langtab[] = { [LangEN] &natural, [LangJP] &hira, [LangJPK] &kata, [LangKO] &hangul, [LangZH] &judou, [LangVN] &telex, }; char *langcodetab[] = { [LangEN] "en", [LangJP] "jp", [LangJPK] "jpk", [LangKO] "ko", [LangZH] "zh", [LangVN] "vn", }; int parselang(char *s) { int i; for(i = 0; i < nelem(langcodetab); i++){ if(langcodetab[i] == nil) continue; if(strcmp(langcodetab[i], s) == 0) return i; } return -1; } int checklang(int *dst, int c) { Hmap **p; if(c >= nelem(langtab)) return 0; p = langtab[c]; if(p == nil) return 0; *dst = c; return c; } int maplkup(int lang, char *s, Map *m) { Hmap **h; if(lang >= nelem(langtab)) return -1; h = langtab[lang]; if(h == nil || *h == nil) return -1; return hmapget(*h, s, m); } enum { Msgsize = 256 }; static Channel *dictch; static Channel *output; static Channel *input; static char backspace[Msgsize]; static Channel *displaych; static Channel *selectch; static void displaythread(void*) { Mousectl *mctl; Mouse m; Keyboardctl *kctl; Rune key; char *kouho[Maxkouho], **s, **e; int i, page, total, height, round; Image *back, *text, *board, *high, *scroll; Font *f; Point p; Rectangle r, exitr, selr, scrlr; int selected; char *mp, move[Msgsize]; enum { Adisp, Aresize, Amouse, Asel, Akbd, Amove, Aend }; Alt a[] = { [Adisp] { nil, kouho, CHANRCV }, [Aresize] { nil, nil, CHANRCV }, [Amouse] { nil, &m, CHANRCV }, [Asel] { nil, &selected, CHANRCV }, [Akbd] { nil, &key, CHANRCV }, [Amove] { nil, move, CHANNOP }, [Aend] { nil, nil, CHANEND }, }; if(initdraw(nil, nil, "ktrans") < 0) sysfatal("failed to initdraw: %r"); mctl = initmouse(nil, screen); if(mctl == nil) sysfatal("failed to get mouse: %r"); /* * For keys coming in to our specific window. * We've already transliterated these, but should * consume keys and exit on del to avoid artifacts. */ kctl = initkeyboard(nil); if(kctl == nil) sysfatal("failed to get keyboard: %r"); memset(kouho, 0, sizeof kouho); selected = -1; f = display->defaultfont; high = allocimagemix(display, DYellowgreen, DWhite); text = display->black; back = allocimagemix(display, DPaleyellow, DWhite); board = allocimagemix(display, DBlack, DWhite); scroll = allocimage(display, Rect(0,0,1,1), screen->chan, 1, DYellowgreen); a[Adisp].c = displaych; a[Aresize].c = mctl->resizec; a[Amouse].c = mctl->c; a[Asel].c = selectch; a[Akbd].c = kctl->c; a[Amove].c = input; threadsetname("display"); goto Redraw; for(;;) switch(alt(a)){ case Amove: a[Amove].op = CHANNOP; break; case Akbd: if(key != Kdel) break; closedisplay(display); threadexitsall(nil); case Amouse: if(!m.buttons) break; if(ptinrect(m.xy, exitr)){ closedisplay(display); threadexitsall(nil); } if(kouho[0] == nil) break; if(m.xy.x > scrlr.min.x && m.xy.x < scrlr.max.x){ if(m.xy.y > scrlr.min.y && m.xy.y < scrlr.max.y) break; if(m.xy.y < scrlr.min.y) goto Up; else goto Down; } if(m.buttons & 7){ m.xy.y -= screen->r.min.y; m.xy.y -= f->height; if(m.xy.y < 0) break; i = round + m.xy.y/f->height + 1; if(selected != -1) i = i - selected - 1; } else if(m.buttons == 8){ Up: i = -1 * (selected % height + height); if(selected + i < 0) i = -(selected + (total % height)); } else if(m.buttons == 16){ Down: i = height - (selected % height); if(selected + i > total) i = total - selected; } else break; memset(move, 0, sizeof move); move[0] = 'c'; if(i == 0) break; else if(i > 0) memset(move+1, '', i); else for(mp = move+1; i < 0; i++) mp = seprint(mp, move + sizeof move, "%C", Kup); a[Amove].op = CHANSND; break; case Aresize: getwindow(display, Refnone); case Adisp: Redraw: for(s = kouho, total = 0; *s != nil; s++, total++) ; r = screen->r; height = Dy(r)/f->height - 2; draw(screen, r, back, nil, ZP); r.max.y = r.min.y + f->height; draw(screen, r, board, nil, ZP); round = selected - (selected % height); if(selected >= 0 && kouho[selected] != nil){ selr = screen->r; selr.min.y += f->height*(selected-round+1); selr.max.y = selr.min.y + f->height; draw(screen, selr, high, nil, ZP); } scrlr = screen->r; scrlr.min.y += f->height; scrlr.max.y -= f->height; scrlr.max.x = scrlr.min.x + 10; draw(screen, scrlr, scroll, nil, ZP); if(kouho[0] != nil){ scrlr.max.x--; page = Dy(scrlr) / (total / height + 1); scrlr.min.y = scrlr.min.y + page*(round / height); scrlr.max.y = scrlr.min.y + page; /* fill to the bottom on last page */ if((screen->r.max.y - f->height) - scrlr.max.y < page) scrlr.max.y = screen->r.max.y - f->height; draw(screen, scrlr, back, nil, ZP); } r.min.x += Dx(r)/2; p.y = r.min.y; p.x = r.min.x - stringwidth(f, "候補")/2; string(screen, p, text, ZP, f, "候補"); p.y += f->height; for(s = kouho+round, e = kouho+round+height; *s != nil && s < e; s++){ p.x = r.min.x - stringwidth(f, *s)/2; string(screen, p, text, ZP, f, *s); p.y += f->height; } p.x = r.min.x - stringwidth(f, "終了")/2; p.y = screen->r.max.y - f->height; exitr = Rpt(Pt(0, p.y), screen->r.max); draw(screen, exitr, board, nil, ZP); string(screen, p, text, ZP, f, "終了"); flushimage(display, 1); break; } } static int emitutf(Channel *out, char *u, int nrune) { char b[Msgsize]; char *e; b[0] = 'c'; e = pushutf(b+1, b + Msgsize - 1, u, nrune); send(out, b); return e - b; } static int compacting = 0; static void dictthread(void*) { char m[Msgsize]; Rune r; int n; char *p; Hmap *dict; char *kouho[Maxkouho]; Str line; Str last; Str okuri; int selected, loop; enum{ Kanji, Okuri, Joshi, }; int mode; dict = jisho; selected = -1; loop = 0; mode = Kanji; memset(kouho, 0, sizeof kouho); resetstr(&last, &line, &okuri, nil); threadsetname("dict"); while(recv(dictch, m) != -1){ compacting = 1; for(p = m+1; *p; p += n){ n = chartorune(&r, p); switch(r){ case LangJP: dict = jisho; break; case LangZH: dict = zidian; break; case '': if(line.b == line.p){ emitutf(output, "", 1); break; } emitutf(output, backspace, utflen(line.b)); /* fallthrough */ case '': case ' ': case '\n': mode = Kanji; resetstr(&line, &okuri, &last, nil); break; case '\b': if(mode != Kanji){ if(okuri.p == okuri.b){ mode = Kanji; popstr(&line); }else popstr(&okuri); break; } popstr(&line); break; case Kup: if(selected == -1){ emitutf(output, p, 1); break; } if(--selected < 0){ //wrap while(kouho[++selected] != nil) ; selected--; } loop = 1; goto Select; case Kdown: if(selected == -1){ emitutf(output, p, 1); break; } case '': selected++; if(selected == 0){ if(hmapget(dict, line.b, kouho) < 0) break; if(dict == jisho && line.p > line.b && isascii(line.p[-1])) line.p[-1] = '\0'; } if(kouho[selected] == nil){ selected = 0; loop = 1; } Select: send(selectch, &selected); send(displaych, kouho); if(okuri.p != okuri.b) emitutf(output, backspace, utflen(okuri.b)); if(selected == 0 && !loop) emitutf(output, backspace, utflen(line.b)); else emitutf(output, backspace, utflen(last.b)); emitutf(output, kouho[selected], 0); last.p = pushutf(last.b, strend(&last), kouho[selected], 0); emitutf(output, okuri.b, 0); mode = Kanji; loop = 0; continue; case ',': case '.': case L'。': case L'、': if(dict == zidian || line.p == line.b){ selected = 0; //hit cleanup below break; } mode = Joshi; okuri.p = pushutf(okuri.p, strend(&okuri), p, 1); break; default: if(dict == zidian) goto Line; if(mode == Joshi) goto Okuri; if(isupper(*p)){ if(mode == Okuri){ popstr(&line); mode = Joshi; goto Okuri; } mode = Okuri; *p = tolower(*p); okuri.p = pushutf(okuri.b, strend(&okuri), p, 1); goto Line; } switch(mode){ case Kanji: Line: line.p = pushutf(line.p, strend(&line), p, 1); break; default: Okuri: okuri.p = pushutf(okuri.p, strend(&okuri), p, 1); break; } } if(selected >= 0){ resetstr(&okuri, &last, &line, nil); if(dict == zidian) line.p = pushutf(line.p, strend(&line), p, 1); selected = -1; send(selectch, &selected); } memset(kouho, 0, sizeof kouho); hmapget(dict, line.b, kouho); send(displaych, kouho); } compacting = 0; } } static void telexlkup(Str *line) { Map lkup; char buf[UTFmax*3], *p, *e, *dot; Str out; int n, ln; for(dot = line->b; (ln = utflen(dot)) >= 2; dot += n){ p = pushutf(buf, buf+sizeof buf, dot, 1); n = p-buf; if(hmapget(telex, buf, &lkup) < 0) continue; e = peekstr(line->p, line->b); pushutf(p, buf+sizeof buf, e, 1); if(hmapget(telex, buf, &lkup) < 0) continue; out.p = pushutf(out.b, strend(&out), lkup.kana, 0); out.p = pushutf(out.p, strend(&out), dot+n, 0); popstr(&out); emitutf(output, backspace, ln); emitutf(output, out.b, 0); line->p = pushutf(line->b, strend(line), out.b, 0); return; } } static void dubeolbksp(Str *line) { Map lkup; popstr(line); /* lookup and emit remaining Jamo to output */ if(hmapget(hangul, line->b, &lkup) < 0) return; emitutf(output, lkup.kana, 0); } static void dubeollkup(Str *line) { Map lkup; char buf[UTFmax*2]; char *e2, *e1; e1 = peekstr(line->p, line->b); if(e1 != line->b){ e2 = peekstr(e1, line->b); pushutf(buf, buf+sizeof buf, e2, utflen(e2)); }else pushutf(buf, buf+sizeof buf, e1, utflen(e1)); if(hmapget(hangul, line->b, &lkup) < 0){ if(hmapget(hangul, buf, &lkup) < 0){ resetstr(line, nil); line->p = pushutf(line->p, strend(line), e1, utflen(e1)); if(hmapget(hangul, line->b, &lkup) < 0) return; }else{ /* treat Jongseong as Choseong when it matches with new Jamo */ popstr(line); popstr(line); hmapget(hangul, line->b, &lkup); emitutf(output, backspace, 2); emitutf(output, lkup.kana, 0); hmapget(hangul, buf, &lkup); emitutf(output, lkup.kana, 0); line->p = pushutf(line->b, strend(line), buf, utflen(buf)); return; } } if(utflen(line->b) == 1) emitutf(output, backspace, 1); else emitutf(output, backspace, 2); emitutf(output, lkup.kana, 0); } static void keythread(void*) { int lang; char m[Msgsize]; char *todict; Map lkup; char *p; int n; Rune r; char peek[UTFmax+1]; Str line; peek[0] = lang = deflang; resetstr(&line, nil); if(lang == LangJP || lang == LangZH) emitutf(dictch, peek, 1); todict = smprint("%C%C", Kup, Kdown); threadsetname("keytrans"); while(recv(input, m) != -1){ if(m[0] == 'z'){ emitutf(dictch, "", 1); resetstr(&line, nil); continue; } if(m[0] != 'c'){ send(output, m); continue; } for(p = m+1; *p; p += n){ while(compacting) yield(); n = chartorune(&r, p); if(checklang(&lang, r)){ emitutf(dictch, "", 1); if(lang == LangJP || lang == LangZH) emitutf(dictch, p, 1); resetstr(&line, nil); continue; } if(lang == LangEN){ emitutf(output, p, 1); continue; } if(utfrune(todict, r) != nil){ resetstr(&line, nil); emitutf(dictch, p, 1); continue; } emitutf(output, p, 1); switch(lang){ case LangZH: emitutf(dictch, p, 1); break; case LangJP: emitutf(dictch, p, 1); if(isupper(*p)) *p = tolower(*p); break; } if(utfrune("\n\t ", r) != nil){ resetstr(&line, nil); continue; } else if(r == '\b'){ if(lang == LangKO){ dubeolbksp(&line); continue; } popstr(&line); continue; } line.p = pushutf(line.p, strend(&line), p, 1); if(lang == LangVN){ telexlkup(&line); continue; }else if(lang == LangKO){ dubeollkup(&line); continue; } if(maplkup(lang, line.b, &lkup) < 0){ resetstr(&line, nil); pushutf(peek, peek + sizeof peek, p, 1); if(maplkup(lang, peek, &lkup) == 0) line.p = pushutf(line.p, strend(&line), p, 1); continue; } if(lkup.kana == nil) continue; if(!lkup.leadstomore) resetstr(&line, nil); if(lang == LangJP){ emitutf(dictch, backspace, utflen(lkup.roma)); emitutf(dictch, lkup.kana, 0); } emitutf(output, backspace, utflen(lkup.roma)); emitutf(output, lkup.kana, 0); } } } static int kbdin; static int kbdout; static void kbdtap(void*) { char m[Msgsize]; char buf[Msgsize]; char *p; int n; threadsetname("kbdtap"); for(;;){ Drop: n = read(kbdin, buf, sizeof buf); if(n < 0) break; for(p = buf; p < buf+n; p += strlen(p) + 1){ switch(*p){ case 'c': case 'k': case 'K': case 'z': break; default: goto Drop; } strcpy(m, p); if(send(input, m) == -1) return; } } threadexitsall(nil); } static void kbdsink(void*) { char in[Msgsize]; char out[Msgsize]; char *p; int n; Rune rn; out[0] = 'c'; threadsetname("kbdsink"); while(recv(output, in) != -1){ if(in[0] != 'c'){ if(write(kbdout, in, strlen(in)+1) < 0) break; continue; } for(p = in+1; *p; p += n){ n = chartorune(&rn, p); if(rn == Runeerror || rn == '\0') break; memmove(out+1, p, n); out[1+n] = '\0'; if(write(kbdout, out, 1+n+1) < 0) break; } } } static int plumbfd; static void plumbproc(void*) { char m[Msgsize]; Plumbmsg *p; threadsetname("plumbproc"); for(; p = plumbrecv(plumbfd); plumbfree(p)){ if(p->ndata > sizeof m - 1) continue; memmove(m, p->data, p->ndata); m[p->ndata] = '\0'; m[1] = parselang(m); if(m[1] == -1) continue; m[0] = 'c'; m[2] = '\0'; if(send(input, m) == -1) break; } plumbfree(p); } void usage(void) { fprint(2, "usage: %s [ -G ] [ -l lang ] [ kbdtap ]\n", argv0); threadexits("usage"); } static char *kdir = "/lib/ktrans"; struct { char *s; Hmap **m; } initmaptab[] = { "judou", &judou, "hira", &hira, "kata", &kata, "hangul", &hangul, "telex", &telex, }; struct { char *env; char *def; Hmap **m; } initdicttab[] = { "jisho", "kanji.dict", &jisho, "zidian", "wubi.dict", &zidian, }; mainstacksize = 8192*2; void threadmain(int argc, char *argv[]) { int nogui, i; char buf[128]; char *e, *home; Hmap *m; deflang = LangEN; nogui = 0; ARGBEGIN{ case 'l': deflang = parselang(EARGF(usage())); if(deflang < 0) usage(); break; case 'G': nogui++; break; default: usage(); }ARGEND; switch(argc){ case 0: kbdin = 0; kbdout = 1; break; case 1: kbdin = kbdout = open(argv[0], ORDWR); if(kbdin < 0) sysfatal("failed to open kbdtap: %r"); break; default: usage(); } dictch = chancreate(Msgsize, 0); input = chancreate(Msgsize, 0); output = chancreate(Msgsize, 0); /* allow gui to warm up while we're busy reading maps */ if(nogui || access("/dev/winid", AEXIST) < 0){ displaych = nil; selectch = nil; } else { selectch = chancreate(sizeof(int), 0); displaych = chancreate(sizeof(char*)*Maxkouho, 0); proccreate(displaythread, nil, mainstacksize); } memset(backspace, '\b', sizeof backspace-1); backspace[sizeof backspace-1] = '\0'; if((home = getenv("home")) == nil) sysfatal("$home undefined"); for(i = 0; i < nelem(initdicttab); i++){ e = getenv(initdicttab[i].env); if(e != nil){ snprint(buf, sizeof buf, "%s", e); free(e); } else snprint(buf, sizeof buf, "%s/%s", kdir, initdicttab[i].def); if((*initdicttab[i].m = opendict(*initdicttab[i].m, buf)) == nil) sysfatal("failed to open dict: %r"); snprint(buf, sizeof buf, "%s/%s/%s", home, kdir, initdicttab[i].def); m = opendict(*initdicttab[i].m, buf); if(m != nil) *initdicttab[i].m = m; } free(home); natural = nil; for(i = 0; i < nelem(initmaptab); i++){ snprint(buf, sizeof buf, "%s/%s.map", kdir, initmaptab[i].s); if((*initmaptab[i].m = openmap(buf)) == nil) sysfatal("failed to open map: %r"); } plumbfd = plumbopen("lang", OREAD); if(plumbfd >= 0) proccreate(plumbproc, nil, mainstacksize); proccreate(kbdtap, nil, mainstacksize); proccreate(kbdsink, nil, mainstacksize); threadcreate(dictthread, nil, mainstacksize); threadcreate(keythread, nil, mainstacksize); threadexits(nil); }