source: neercs/trunk/src/wm.c @ 3934

Last change on this file since 3934 was 3934, checked in by Jean-Yves Lamoureux, 11 years ago
  • Factorise cube stuff, minor optimizations
  • Property svn:keywords set to Id
File size: 18.8 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 *                All Rights Reserved
6 *
7 *  $Id: wm.c 3934 2009-11-18 13:43:02Z jylam $
8 *
9 *  This program is free software. It comes without any warranty, to
10 *  the extent permitted by applicable law. You can redistribute it
11 *  and/or modify it under the terms of the Do What The Fuck You Want
12 *  To Public License, Version 2, as published by Sam Hocevar. See
13 *  http://sam.zoy.org/wtfpl/COPYING for more details.
14 */
15
16#include "config.h"
17
18#include <stdio.h>
19#include <caca.h>
20#include <caca.h>
21#include <stdlib.h>
22#include <math.h>
23
24#include "neercs.h"
25
26
27void resize_screen(struct screen *s, int w, int h)
28{
29    caca_canvas_t *old, *new;
30
31    if (w == s->w && h == s->h)
32        return;
33    if (w <= 0 || h <= 0)
34        return;
35
36    s->changed = 1;
37
38    s->w = w;
39    s->h = h;
40
41    /*
42     * caca_set_canvas_boundaries() is bugged as hell, so let's resize it by
43     * hands
44     */
45    old = s->cv;
46    new = caca_create_canvas(w, h);
47    caca_blit(new, 0, 0, old, NULL);
48    s->cv = new;
49    caca_gotoxy(new, caca_get_cursor_x(old), caca_get_cursor_y(old));
50    caca_free_canvas(old);
51    set_tty_size(s->fd, w, h);
52
53    s->orig_w = s->w;
54    s->orig_h = s->h;
55    s->orig_x = s->x;
56    s->orig_y = s->y;
57}
58
59void update_windows_props(struct screen_list *screen_list)
60{
61    debug("%s, %d screens, type %d\n", __FUNCTION__, screen_list->count,
62          screen_list->wm_type);
63
64    if (!screen_list->count)
65        return;
66
67    switch (screen_list->wm_type)
68    {
69    case WM_CARD:
70        update_windows_props_cards(screen_list);
71        break;
72    case WM_HSPLIT:
73        update_windows_props_hsplit(screen_list);
74        break;
75    case WM_VSPLIT:
76        update_windows_props_vsplit(screen_list);
77        break;
78    case WM_FULL:
79    case WM_CUBE:
80    default:
81        update_windows_props_full(screen_list);
82        break;
83    }
84}
85
86void update_windows_props_hsplit(struct screen_list *screen_list)
87{
88    int i;
89    int w = (screen_list->width / screen_list->count) - 1;
90    int h = screen_list->height - 2;
91
92    for (i = 0; i < screen_list->count; i++)
93    {
94        screen_list->screen[i]->x = (i * w) + 1;
95        screen_list->screen[i]->y = 1;
96        screen_list->screen[i]->visible = 1;
97        if (i != screen_list->count - 1)
98        {
99            resize_screen(screen_list->screen[i], w - 1, h);
100        }
101        else
102        {
103            resize_screen(screen_list->screen[i],
104                          screen_list->width - i * w - 2, h);
105        }
106    }
107}
108
109void update_windows_props_vsplit(struct screen_list *screen_list)
110{
111    int i;
112    int w = screen_list->width - 2;
113    int h = (screen_list->height) / screen_list->count;
114
115    for (i = 0; i < screen_list->count; i++)
116    {
117        screen_list->screen[i]->x = 1;
118        screen_list->screen[i]->y = (i * h) + 1;
119        screen_list->screen[i]->visible = 1;
120        if (i != screen_list->count - 1)
121        {
122            resize_screen(screen_list->screen[i], w, h - 2);
123        }
124        else
125        {
126            resize_screen(screen_list->screen[i],
127                          w, screen_list->height - i * h - 2);
128        }
129    }
130}
131
132
133void update_windows_props_full(struct screen_list *screen_list)
134{
135    int i;
136    int w = screen_list->width - 2;
137    int h = screen_list->height - 2;
138
139    for (i = 0; i < screen_list->count; i++)
140    {
141        screen_list->screen[i]->visible = 0;
142        screen_list->screen[i]->x = 1;
143        screen_list->screen[i]->y = 1;
144
145        resize_screen(screen_list->screen[i], w, h);
146    }
147    screen_list->screen[screen_list->pty]->visible = 1;
148}
149
150
151void update_windows_props_cards(struct screen_list *screen_list)
152{
153    int i;
154    int w = (screen_list->width - screen_list->count * 3) + 1;
155    int h = (screen_list->height - screen_list->count) - 1;
156    int x = 1;
157    int y = screen_list->count;
158
159    for (i = 0; i < screen_list->count; i++)
160    {
161        screen_list->screen[i]->visible = 1;
162        screen_list->screen[i]->x = x;
163        screen_list->screen[i]->y = y;
164
165        resize_screen(screen_list->screen[i], w, h);
166        x += 3;
167        y--;
168    }
169}
170
171/* Window managers refresh */
172
173void wm_refresh(struct screen_list *screen_list)
174{
175    switch (screen_list->wm_type)
176    {
177    case WM_CARD:
178        wm_refresh_card(screen_list);
179        break;
180    case WM_FULL:
181        wm_refresh_full(screen_list);
182        break;
183    case WM_HSPLIT:
184        wm_refresh_hsplit(screen_list);
185        break;
186    case WM_VSPLIT:
187        wm_refresh_hsplit(screen_list);
188        break;
189    case WM_CUBE:
190        wm_refresh_cube(screen_list);
191        break;
192    default:
193        wm_refresh_full(screen_list);
194        break;
195    }
196}
197
198void wm_refresh_card(struct screen_list *screen_list)
199{
200    int i;
201
202    for (i = screen_list->count - 1; i >= 0; i--)
203    {
204        if (i != screen_list->pty && screen_list->screen[i]->visible &&
205            (screen_list->screen[i]->changed || screen_list->changed))
206        {
207            caca_blit(screen_list->cv,
208                      screen_list->screen[i]->x,
209                      screen_list->screen[i]->y,
210                      screen_list->screen[i]->cv, NULL);
211
212            caca_draw_cp437_box(screen_list->cv,
213                                screen_list->screen[i]->x - 1,
214                                screen_list->screen[i]->y - 1,
215                                screen_list->screen[i]->w + 2,
216                                screen_list->screen[i]->h + 2);
217            if (screen_list->screen[i]->title)
218                caca_printf(screen_list->cv,
219                            screen_list->screen[i]->x,
220                            screen_list->screen[i]->y - 1,
221                            " %.*s ",
222                            screen_list->screen[i]->w - 3,
223                            screen_list->screen[i]->title);
224        }
225    }
226
227    if (screen_list->screen[screen_list->pty]->changed || screen_list->changed)
228        caca_blit(screen_list->cv,
229                  screen_list->screen[screen_list->pty]->x,
230                  screen_list->screen[screen_list->pty]->y,
231                  screen_list->screen[screen_list->pty]->cv, NULL);
232
233    if (screen_list->screen[screen_list->pty]->bell)
234    {
235        caca_set_color_ansi(screen_list->cv, CACA_RED, CACA_BLACK);
236        screen_list->screen[screen_list->pty]->bell = 0;
237        screen_list->in_bell--;
238    }
239    else
240    {
241        caca_set_color_ansi(screen_list->cv, CACA_LIGHTGREEN, CACA_BLACK);
242    }
243
244    caca_draw_cp437_box(screen_list->cv,
245                        screen_list->screen[screen_list->pty]->x - 1,
246                        screen_list->screen[screen_list->pty]->y - 1,
247                        screen_list->screen[screen_list->pty]->w + 2,
248                        screen_list->screen[screen_list->pty]->h + 2);
249
250    if (screen_list->screen[screen_list->pty]->title)
251    {
252        caca_printf(screen_list->cv,
253                    screen_list->screen[screen_list->pty]->x,
254                    screen_list->screen[screen_list->pty]->y - 1,
255                    " %.*s ",
256                    screen_list->screen[screen_list->pty]->w - 3,
257                    screen_list->screen[screen_list->pty]->title);
258    }
259}
260
261
262void wm_refresh_full(struct screen_list *screen_list)
263{
264    if (screen_list->screen[screen_list->pty]->changed || screen_list->changed)
265        caca_blit(screen_list->cv,
266                  screen_list->screen[screen_list->pty]->x,
267                  screen_list->screen[screen_list->pty]->y,
268                  screen_list->screen[screen_list->pty]->cv, NULL);
269
270    if (screen_list->screen[screen_list->pty]->bell)
271    {
272        caca_set_color_ansi(screen_list->cv, CACA_RED, CACA_BLACK);
273        screen_list->screen[screen_list->pty]->bell = 0;
274        screen_list->in_bell--;
275    }
276    else
277    {
278        caca_set_color_ansi(screen_list->cv, CACA_LIGHTGREEN, CACA_BLACK);
279    }
280
281    caca_draw_cp437_box(screen_list->cv,
282                        screen_list->screen[screen_list->pty]->x - 1,
283                        screen_list->screen[screen_list->pty]->y - 1,
284                        screen_list->screen[screen_list->pty]->w + 2,
285                        screen_list->screen[screen_list->pty]->h + 2);
286
287    if (screen_list->screen[screen_list->pty]->title)
288    {
289        caca_printf(screen_list->cv,
290                    screen_list->screen[screen_list->pty]->x,
291                    screen_list->screen[screen_list->pty]->y - 1,
292                    " %.*s ",
293                    screen_list->screen[screen_list->pty]->w - 3,
294                    screen_list->screen[screen_list->pty]->title);
295    }
296}
297
298void wm_refresh_vsplit(struct screen_list *screen_list)
299{
300    int i;
301
302    for (i = screen_list->count - 1; i >= 0; i--)
303    {
304        if (i != screen_list->pty && screen_list->screen[i]->visible &&
305            (screen_list->screen[i]->changed || screen_list->changed))
306        {
307            caca_blit(screen_list->cv,
308                      screen_list->screen[i]->x,
309                      screen_list->screen[i]->y,
310                      screen_list->screen[i]->cv, NULL);
311
312            caca_draw_cp437_box(screen_list->cv,
313                                screen_list->screen[i]->x - 1,
314                                screen_list->screen[i]->y - 1,
315                                screen_list->screen[i]->w + 2,
316                                screen_list->screen[i]->h + 2);
317            if (screen_list->screen[i]->title)
318                caca_printf(screen_list->cv,
319                            screen_list->screen[i]->x,
320                            screen_list->screen[i]->y - 1,
321                            " %.*s ",
322                            screen_list->screen[i]->w - 3,
323                            screen_list->screen[i]->title);
324        }
325    }
326
327    if (screen_list->screen[screen_list->pty]->changed || screen_list->changed)
328        caca_blit(screen_list->cv,
329                  screen_list->screen[screen_list->pty]->x,
330                  screen_list->screen[screen_list->pty]->y,
331                  screen_list->screen[screen_list->pty]->cv, NULL);
332
333    if (screen_list->screen[screen_list->pty]->bell)
334    {
335        caca_set_color_ansi(screen_list->cv, CACA_RED, CACA_BLACK);
336        screen_list->screen[screen_list->pty]->bell = 0;
337        screen_list->in_bell--;
338    }
339    else
340    {
341        caca_set_color_ansi(screen_list->cv, CACA_LIGHTGREEN, CACA_BLACK);
342    }
343
344    caca_draw_cp437_box(screen_list->cv,
345                        screen_list->screen[screen_list->pty]->x - 1,
346                        screen_list->screen[screen_list->pty]->y - 1,
347                        screen_list->screen[screen_list->pty]->w + 2,
348                        screen_list->screen[screen_list->pty]->h + 2);
349
350    if (screen_list->screen[screen_list->pty]->title)
351    {
352        caca_printf(screen_list->cv,
353                    screen_list->screen[screen_list->pty]->x,
354                    screen_list->screen[screen_list->pty]->y - 1,
355                    " %.*s ",
356                    screen_list->screen[screen_list->pty]->w - 3,
357                    screen_list->screen[screen_list->pty]->title);
358    }
359}
360
361void wm_refresh_hsplit(struct screen_list *screen_list)
362{
363    int i;
364
365    for (i = screen_list->count - 1; i >= 0; i--)
366    {
367        if (i != screen_list->pty && screen_list->screen[i]->visible &&
368            (screen_list->screen[i]->changed || screen_list->changed))
369        {
370            caca_blit(screen_list->cv,
371                      screen_list->screen[i]->x,
372                      screen_list->screen[i]->y,
373                      screen_list->screen[i]->cv, NULL);
374
375            caca_draw_cp437_box(screen_list->cv,
376                                screen_list->screen[i]->x - 1,
377                                screen_list->screen[i]->y - 1,
378                                screen_list->screen[i]->w + 2,
379                                screen_list->screen[i]->h + 2);
380            if (screen_list->screen[i]->title)
381                caca_printf(screen_list->cv,
382                            screen_list->screen[i]->x,
383                            screen_list->screen[i]->y - 1,
384                            " %.*s ",
385                            screen_list->screen[i]->w - 3,
386                            screen_list->screen[i]->title);
387        }
388    }
389
390    if (screen_list->screen[screen_list->pty]->changed || screen_list->changed)
391        caca_blit(screen_list->cv,
392                  screen_list->screen[screen_list->pty]->x,
393                  screen_list->screen[screen_list->pty]->y,
394                  screen_list->screen[screen_list->pty]->cv, NULL);
395
396    if (screen_list->screen[screen_list->pty]->bell)
397    {
398        caca_set_color_ansi(screen_list->cv, CACA_RED, CACA_BLACK);
399        screen_list->screen[screen_list->pty]->bell = 0;
400        screen_list->in_bell--;
401    }
402    else
403    {
404        caca_set_color_ansi(screen_list->cv, CACA_LIGHTGREEN, CACA_BLACK);
405    }
406
407    caca_draw_cp437_box(screen_list->cv,
408                        screen_list->screen[screen_list->pty]->x - 1,
409                        screen_list->screen[screen_list->pty]->y - 1,
410                        screen_list->screen[screen_list->pty]->w + 2,
411                        screen_list->screen[screen_list->pty]->h + 2);
412
413    if (screen_list->screen[screen_list->pty]->title)
414    {
415        caca_printf(screen_list->cv,
416                    screen_list->screen[screen_list->pty]->x,
417                    screen_list->screen[screen_list->pty]->y - 1,
418                    " %.*s ",
419                    screen_list->screen[screen_list->pty]->w - 3,
420                    screen_list->screen[screen_list->pty]->title);
421    }
422}
423
424#if defined HAVE_CACA_TRIANGLE_TEXTURING
425static float
426get_direction(float p1x, float p1y, float p2x, float p2y, float p3x, float p3y)
427{
428    float d1x, d1y, d2x, d2y;
429
430    d1x = p3x - p1x;
431    d1y = p3y - p1y;
432    d2x = p3x - p2x;
433    d2y = p3y - p2y;
434    return (d1x * d2y) - (d1y * d2x);
435}
436#endif
437
438/* 3D Cube. Yeah I know, it's a mess. Just look anywhere else. */
439static void draw_face(caca_canvas_t * cv,
440                      int p1x, int p1y,
441                      int p2x, int p2y,
442                      int p3x, int p3y,
443                      int p4x, int p4y, caca_canvas_t * tex, int color)
444{
445    if (get_direction(p1x, p1y, p2x, p2y, p3x, p3y) >= 0)
446    {
447        int coords[6];
448        float uv[6];
449        coords[0] = p1x;
450        coords[1] = p1y;
451        coords[2] = p2x;
452        coords[3] = p2y;
453        coords[4] = p3x;
454        coords[5] = p3y;
455        uv[0] = 1;
456        uv[1] = 1;
457        uv[2] = 0;
458        uv[3] = 1;
459        uv[4] = 0;
460        uv[5] = 0;
461#if defined HAVE_CACA_TRIANGLE_TEXTURING
462        caca_fill_triangle_textured(cv, coords, tex, uv);
463#endif
464
465        coords[0] = p1x;
466        coords[1] = p1y;
467        coords[2] = p3x;
468        coords[3] = p3y, coords[4] = p4x;
469        coords[5] = p4y;
470        uv[0] = 1;
471        uv[1] = 1;
472        uv[2] = 0;
473        uv[3] = 0;
474        uv[4] = 1;
475        uv[5] = 0;
476#if defined HAVE_CACA_TRIANGLE_TEXTURING
477        caca_fill_triangle_textured(cv, coords, tex, uv);
478#endif
479        caca_set_color_ansi(cv, color, CACA_BLACK);
480        caca_draw_thin_line(cv, p1x, p1y, p2x, p2y);
481        caca_draw_thin_line(cv, p2x, p2y, p3x, p3y);
482        caca_draw_thin_line(cv, p3x, p3y, p4x, p4y);
483        caca_draw_thin_line(cv, p4x, p4y, p1x, p1y);
484    }
485}
486
487#define CUBE_TIME 1000000
488void wm_refresh_cube(struct screen_list *screen_list)
489{
490    int i;
491
492    if (!screen_list->cube.in_switch)
493    {
494        wm_refresh_full(screen_list);
495        screen_list->force_refresh = 0;
496    }
497    else
498    {
499
500        long long unsigned int cur_time = get_us() - screen_list->last_switch;
501
502        if (cur_time >= CUBE_TIME || screen_list->count == 1)
503        {
504            screen_list->changed = 1;
505            screen_list->force_refresh = 1;
506            screen_list->cube.in_switch = 0;
507        }
508        else
509        {
510            float cube[12][3] = {
511                {-1, -1, 1},
512                {1, -1, 1},
513                {1, 1, 1},
514                {-1, 1, 1},
515
516                {1, -1, 1},
517                {1, -1, -1},
518                {1, 1, -1},
519                {1, 1, 1},
520
521                {-1, -1, -1},
522                {-1, -1, 1},
523                {-1, 1, 1},
524                {-1, 1, -1},
525            };
526
527            float cube_transformed[12][3];
528            float cube_projected[12][2];
529            float fov = 0.5f;
530            float angle = 90.0f * ((float)cur_time / (float)CUBE_TIME);
531
532            angle *= (M_PI / 180.0f);
533
534            if (screen_list->cube.side == 1)
535                angle = -angle;
536
537            float sina = sin(angle);
538            float cosa = cos(angle);
539
540            for (i = 0; i < 12; i++)
541            {
542                cube_transformed[i][2] = cube[i][2] * cosa - cube[i][0] * sina;
543                cube_transformed[i][0] = cube[i][2] * sina + cube[i][0] * cosa;
544                cube_transformed[i][1] = cube[i][1];
545
546                cube_transformed[i][2] -= 3;
547
548                cube_projected[i][0] =
549                    cube_transformed[i][0] / (cube_transformed[i][2] * fov);
550                cube_projected[i][1] =
551                    cube_transformed[i][1] / (cube_transformed[i][2] * fov);
552
553                cube_projected[i][0] /= 2.0f;
554                cube_projected[i][1] /= 2.0f;
555                cube_projected[i][0] += 0.5f;
556                cube_projected[i][1] += 0.5f;
557
558                cube_projected[i][0] *= screen_list->width;
559                cube_projected[i][1] *= screen_list->height;
560            }
561
562            caca_set_color_ansi(screen_list->cv, CACA_WHITE, CACA_BLACK);
563            caca_clear_canvas(screen_list->cv);
564
565            caca_canvas_t *first =
566                screen_list->screen[screen_list->prevpty]->cv;
567            caca_canvas_t *second = screen_list->screen[screen_list->pty]->cv;
568
569
570            draw_face(screen_list->cv,
571                      cube_projected[0][0], cube_projected[0][1],
572                      cube_projected[1][0], cube_projected[1][1],
573                      cube_projected[2][0], cube_projected[2][1],
574                      cube_projected[3][0], cube_projected[3][1],
575                      first, CACA_LIGHTGREEN);
576
577
578            if (screen_list->cube.side)
579            {
580                draw_face(screen_list->cv,
581                          cube_projected[4][0], cube_projected[4][1],
582                          cube_projected[5][0], cube_projected[5][1],
583                          cube_projected[6][0], cube_projected[6][1],
584                          cube_projected[7][0], cube_projected[7][1],
585                          second, CACA_RED);
586            }
587            else
588            {
589                draw_face(screen_list->cv,
590                          cube_projected[8][0], cube_projected[8][1],
591                          cube_projected[9][0], cube_projected[9][1],
592                          cube_projected[10][0], cube_projected[10][1],
593                          cube_projected[11][0], cube_projected[11][1],
594                          second, CACA_RED);
595            }
596
597            screen_list->changed = 1;
598            screen_list->force_refresh = 1;
599            screen_list->cube.in_switch = 1;
600        }
601
602    }
603}
Note: See TracBrowser for help on using the repository browser.