#include "vnc.h"
#include <keyboard.h>
#include "utf2ksym.h"
enum {
Xshift = 0xFFE1,
Xctl = 0xFFE3,
Xmeta = 0xFFE7,
Xalt = 0xFFE9
};
static struct {
Rune kbdc;
ulong keysym;
} ktab[] = {
{'\b', 0xff08},
{'\t', 0xff09},
{'\n', 0xff0d},
/* {0x0b, 0xff0b}, */
{'\r', 0xff0d},
{0x1b, 0xff1b}, /* escape */
{Kins, 0xff63},
{0x7F, 0xffff},
{Khome, 0xff50},
{Kend, 0xff57},
{Kpgup, 0xff55},
{Kpgdown, 0xff56},
{Kleft, 0xff51},
{Kup, 0xff52},
{Kright, 0xff53},
{Kdown, 0xff54},
{KF|1, 0xffbe},
{KF|2, 0xffbf},
{KF|3, 0xffc0},
{KF|4, 0xffc1},
{KF|5, 0xffc2},
{KF|6, 0xffc3},
{KF|7, 0xffc4},
{KF|8, 0xffc5},
{KF|9, 0xffc6},
{KF|10, 0xffc7},
{KF|11, 0xffc8},
{KF|12, 0xffc9},
};
static char shiftkey[128] = {
0, 0, 0, 0, 0, 0, 0, 0, /* nul soh stx etx eot enq ack bel */
0, 0, 0, 0, 0, 0, 0, 0, /* bs ht nl vt np cr so si */
0, 0, 0, 0, 0, 0, 0, 0, /* dle dc1 dc2 dc3 dc4 nak syn etb */
0, 0, 0, 0, 0, 0, 0, 0, /* can em sub esc fs gs rs us */
0, 1, 1, 1, 1, 1, 1, 0, /* sp ! " # $ % & ' */
1, 1, 1, 1, 0, 0, 0, 0, /* ( ) * + , - . / */
0, 0, 0, 0, 0, 0, 0, 0, /* 0 1 2 3 4 5 6 7 */
0, 0, 1, 0, 1, 0, 1, 1, /* 8 9 : ; < = > ? */
1, 1, 1, 1, 1, 1, 1, 1, /* @ A B C D E F G */
1, 1, 1, 1, 1, 1, 1, 1, /* H I J K L M N O */
1, 1, 1, 1, 1, 1, 1, 1, /* P Q R S T U V W */
1, 1, 1, 0, 0, 0, 1, 1, /* X Y Z [ \ ] ^ _ */
0, 0, 0, 0, 0, 0, 0, 0, /* ` a b c d e f g */
0, 0, 0, 0, 0, 0, 0, 0, /* h i j k l m n o */
0, 0, 0, 0, 0, 0, 0, 0, /* p q r s t u v w */
0, 0, 0, 1, 1, 1, 1, 0, /* x y z { | } ~ del */
};
ulong
runetoksym(Rune r)
{
int i;
for(i=0; i<nelem(ktab); i++)
if(ktab[i].kbdc == r)
return ktab[i].keysym;
return r;
}
static void
keyevent(Vnc *v, ulong ksym, int down)
{
vnclock(v);
vncwrchar(v, MKey);
vncwrchar(v, down);
vncwrshort(v, 0);
vncwrlong(v, ksym);
vncflush(v);
vncunlock(v);
}
void
readkbd(Vnc *v)
{
char buf[256], k[10];
ulong ks;
int ctlfd, fd, kr, kn, w, shift, ctl, alt;
Rune r;
snprint(buf, sizeof buf, "%s/cons", display->devdir);
if((fd = open(buf, OREAD)) < 0)
sysfatal("open %s: %r", buf);
snprint(buf, sizeof buf, "%s/consctl", display->devdir);
if((ctlfd = open(buf, OWRITE)) < 0)
sysfatal("open %s: %r", buf);
write(ctlfd, "rawon", 5);
kn = 0;
shift = alt = ctl = 0;
for(;;){
while(!fullrune(k, kn)){
kr = read(fd, k+kn, sizeof k - kn);
if(kr <= 0)
sysfatal("bad read from kbd");
kn += kr;
}
w = chartorune(&r, k);
kn -= w;
memmove(k, &k[w], kn);
ks = runetoksym(r);
switch(r){
case Kalt:
alt = !alt;
keyevent(v, Xalt, alt);
break;
case Kctl:
ctl = !ctl;
keyevent(v, Xctl, ctl);
break;
case Kshift:
shift = !shift;
keyevent(v, Xshift, shift);
break;
default:
if(r == ks && r < 0x1A){ /* control key */
keyevent(v, Xctl, 1);
keyevent(v, r+0x60, 1); /* 0x60: make capital letter */
keyevent(v, r+0x60, 0);
keyevent(v, Xctl, 0);
}else{
/*
* to send an upper case letter or shifted
* punctuation, mac os x vnc server,
* at least, needs a `shift' sent first.
*/
if(!shift && r == ks && r < sizeof shiftkey && shiftkey[r]){
shift = 1;
keyevent(v, Xshift, 1);
}
/*
* map an xkeysym onto a utf-8 char.
* allows Xvnc to read us, see utf2ksym.h
*/
if((ks & 0xff00) && ks < nelem(utf2ksym) && utf2ksym[ks] != 0)
ks = utf2ksym[ks];
keyevent(v, ks, 1);
/*
* up event needed by vmware inside linux vnc server,
* perhaps others.
*/
keyevent(v, ks, 0);
}
if(alt){
keyevent(v, Xalt, 0);
alt = 0;
}
if(ctl){
keyevent(v, Xctl, 0);
ctl = 0;
}
if(shift){
keyevent(v, Xshift, 0);
shift = 0;
}
break;
}
}
}
|