source: neercs/trunk/src/client.c @ 4075

Last change on this file since 4075 was 4075, checked in by Jean-Yves Lamoureux, 11 years ago
  • Removed double includes of caca.h on most of the files
File size: 8.3 KB
Line 
1/*
2 *  neercs        console-based window manager
3 *  Copyright (c) 2006 Sam Hocevar <sam@zoy.org>
4 *                2008 Jean-Yves Lamoureux <jylam@lnxscene.org>
5 *                2008 Pascal Terjan <pterjan@linuxfr.org>
6 *                All Rights Reserved
7 *
8 *  $Id: client.c 3939 2009-11-18 14:40:39Z jylam $
9 *
10 *  This program is free software. It comes without any warranty, to
11 *  the extent permitted by applicable law. You can redistribute it
12 *  and/or modify it under the terms of the Do What The Fuck You Want
13 *  To Public License, Version 2, as published by Sam Hocevar. See
14 *  http://sam.zoy.org/wtfpl/COPYING for more details.
15 */
16
17#include "config.h"
18
19#include <stdio.h>
20#include <string.h>
21#include <stdlib.h>
22#include <unistd.h>
23#include <fcntl.h>
24#include <signal.h>
25#include <sys/ioctl.h>
26#include <sys/types.h>
27#include <sys/wait.h>
28#include <sys/time.h>
29#include <time.h>
30#include <pwd.h>
31
32#include <errno.h>
33#include <caca.h>
34
35#include "neercs.h"
36
37#define NEERCS_RECV_BUFSIZE 128*1024
38
39
40int start_client(struct screen_list * screen_list)
41{
42    char *sess = NULL;
43    create_socket(screen_list, SOCK_CLIENT);
44    while ((sess = connect_socket(screen_list, SOCK_SERVER)) == NULL)
45        usleep(100);
46    free(sess);
47
48    /* Create main canvas and associated caca window */
49    screen_list->cv = caca_create_canvas(0, 0);
50    screen_list->dp = caca_create_display(screen_list->cv);
51
52    if (!screen_list->dp)
53        return -3;
54
55    caca_set_display_time(screen_list->dp, screen_list->delay * 1000);
56    caca_set_cursor(screen_list->dp, 1);
57
58    request_attach(screen_list);
59   
60    return 0;
61}
62
63int send_event(caca_event_t ev, struct screen_list *screen_list)
64{
65   
66    enum caca_event_type t;
67   
68    t = caca_get_event_type(&ev);
69   
70    if (t & CACA_EVENT_KEY_PRESS)
71    {
72        char buf[16];
73        int bytes;
74        bytes =
75        snprintf(buf, sizeof(buf) - 1, "KEY %d",
76                 caca_get_event_key_ch(&ev));
77        buf[bytes] = '\0';
78        debug("Sending key press to server: %s", buf);
79        debug("Sending '%s', %d bytes\n", buf, strlen(buf));
80        return write(screen_list->comm.socket[SOCK_SERVER], buf, strlen(buf)+1) <= 0;
81    }
82    else if (t & CACA_EVENT_RESIZE)
83    {
84       
85        char buf[32];
86        int bytes;
87        bytes = snprintf(buf, sizeof(buf) - 1, "RESIZE %10d %10d",
88                         caca_get_event_resize_width(&ev),
89                         caca_get_event_resize_height(&ev));
90        buf[bytes] = '\0';
91        debug("Sending '%s', %d bytes\n", buf, strlen(buf));
92        return write(screen_list->comm.socket[SOCK_SERVER], buf, strlen(buf)+1) <= 0;
93    }
94    else if (t & CACA_EVENT_MOUSE_PRESS)
95    {
96        char buf[52];
97        int bytes;
98        bytes = snprintf(buf, sizeof(buf) - 1, "MOUSEP %10d %10d %10d",
99                                caca_get_mouse_x (screen_list->dp),
100                            caca_get_mouse_y (screen_list->dp),
101                                caca_get_event_mouse_button(&ev));
102        buf[bytes] = '\0';
103        debug("Sending '%s', %d bytes\n", buf, strlen(buf));
104        return write(screen_list->comm.socket[SOCK_SERVER], buf, strlen(buf)+1) <= 0;
105    }
106    else if (t & CACA_EVENT_QUIT)
107        return write(screen_list->comm.socket[SOCK_SERVER], "QUIT",
108                     strlen("QUIT")) <= 0;
109   
110    return 0;
111}
112
113int send_delay(struct screen_list *screen_list)
114{
115    debug("Sending DELAY\n");
116    char buf[18];
117    int bytes;
118    bytes = snprintf(buf, sizeof(buf) - 1, "DELAY %10d", screen_list->delay);
119    buf[bytes] = '\0';
120    return write(screen_list->comm.socket[SOCK_SERVER], buf, strlen(buf)) <= 0;
121}
122
123
124void mainloop(struct screen_list *screen_list)
125{
126    char *buf = NULL;
127
128    screen_list->last_key_time = get_us();
129
130    for (;;)
131    {
132        caca_event_t ev;
133        int ret = 0;
134        ssize_t n;
135        if (!screen_list)
136            goto end;
137        if (!buf)
138            buf = malloc(NEERCS_RECV_BUFSIZE);
139        if (!buf)
140        {
141            debug("Failed to allocate memory");
142            goto end;
143        }
144        if (screen_list->comm.socket[SOCK_CLIENT]
145            && (n =
146                read(screen_list->comm.socket[SOCK_CLIENT], buf,
147                     NEERCS_RECV_BUFSIZE - 1)) > 0)
148        {
149            buf[n] = 0;
150            debug("Received from server: '%s' (%d bytes)", buf, n);
151            if (!strncmp("DETACH", buf, 6))
152            {
153                /* ret = 1; Not used */
154                break;
155            }
156            else if (!strncmp("UPDATE ", buf, 7))
157            {
158                int x, y;
159                ssize_t l2 = 0, lb = 0;
160                char *buf2;
161                size_t l = 0;
162                /* FIXME check the length before calling atoi */
163                x = atoi(buf + 8);
164                y = atoi(buf + 19);
165
166                /* 0 means we have valid data but incomplete, so read the rest
167                 */
168                while (l == 0)
169                {
170                    buf2 = realloc(buf, l2 + NEERCS_RECV_BUFSIZE);
171                    if (!buf2)
172                    {
173                        debug("Failed to allocate memory");
174                        goto end;
175                    }
176                    buf = buf2;
177                    fcntl(screen_list->comm.socket[SOCK_CLIENT], F_SETFL, 0);
178                    lb = read(screen_list->comm.socket[SOCK_CLIENT], buf + l2,
179                              NEERCS_RECV_BUFSIZE - 1);
180                    if (lb < 0)
181                    {
182                        debug
183                            ("Failed to read the end of the refresh message (%s)",
184                             strerror(errno));
185                        l = -1;
186                    }
187                    else
188                    {
189                        l2 += lb;
190#if defined HAVE_CACA_DIRTY_RECTANGLES
191                        l = caca_import_area_from_memory(screen_list->cv, x, y,
192                                                         buf, l2, "caca");
193#else
194                        l = caca_import_from_memory(screen_list->cv, buf, l2,
195                                               "caca");
196#endif
197                    }
198                }
199                fcntl(screen_list->comm.socket[SOCK_CLIENT], F_SETFL, O_NONBLOCK);
200            }
201            else if (!strncmp("REFRESH ", buf, 8))
202            {
203                int dt, x, y;
204                /* FIXME check the length before calling atoi */
205                x = atoi(buf + 8);
206                y = atoi(buf + 19);
207                caca_gotoxy(screen_list->cv, x, y);
208                caca_refresh_display(screen_list->dp);
209                dt = caca_get_display_time(screen_list->dp);
210
211                /* Adjust refresh delay so that the server do not compute
212                   useless things */
213                if (dt > 2 * 1000 * screen_list->delay
214                    && screen_list->delay <= 100)
215                {
216                    screen_list->delay *= 2;
217                    send_delay(screen_list);
218                }
219                else if (dt < screen_list->delay * 1000 * 1.2 &&
220                         screen_list->delay >=
221                         3 * screen_list->requested_delay / 2)
222                {
223                    screen_list->delay = 2 * screen_list->delay / 3;
224                    send_delay(screen_list);
225                }
226            }
227            else if (!strncmp("CURSOR ", buf, 7))
228            {
229                caca_set_cursor(screen_list->dp, atoi(buf + 7));
230            }
231            else if (!strncmp("TITLE ", buf, 6))
232            {
233                caca_set_display_title(screen_list->dp, buf + 6);
234                caca_refresh_display(screen_list->dp);
235            }
236            else
237            {
238                debug("Unknown message received from server: %s", buf);
239            }
240        }
241
242        if (ret)
243        {
244            debug("ret1\n");
245            break;
246        }
247        ret = caca_get_event(screen_list->dp,
248                             CACA_EVENT_KEY_PRESS
249                             | CACA_EVENT_MOUSE_PRESS
250                             | CACA_EVENT_MOUSE_RELEASE
251                             | CACA_EVENT_MOUSE_MOTION
252                             | CACA_EVENT_RESIZE
253                             | CACA_EVENT_QUIT, &ev, 10000);
254        if (ret)
255            ret = send_event(ev, screen_list);
256
257        if (ret)
258        {
259            debug("ret2\n");
260            break;
261        }
262    }
263
264  end:
265    if (buf)
266        free(buf);
267
268}
Note: See TracBrowser for help on using the repository browser.