ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/deliantra/Deliantra-Client/Client.xs
(Generate patch)

Comparing deliantra/Deliantra-Client/Client.xs (file contents):
Revision 1.103 by root, Sun Jun 4 23:05:05 2006 UTC vs.
Revision 1.153 by root, Tue Sep 19 22:02:38 2006 UTC

1#ifdef _WIN32 1#ifdef _WIN32
2# define WIN32_LEAN_AND_MEAN
2# define _WIN32_WINNT 0x0500 // needed to get win2000 api calls 3# define _WIN32_WINNT 0x0500 // needed to get win2000 api calls
3# include <malloc.h> 4# include <malloc.h>
4# include <windows.h> 5# include <windows.h>
5# pragma warning(disable:4244) 6# pragma warning(disable:4244)
6#endif 7#endif
7 8
8#include "EXTERN.h" 9#include "EXTERN.h"
9#include "perl.h" 10#include "perl.h"
10#include "XSUB.h" 11#include "XSUB.h"
11 12
13#ifdef _WIN32
14# undef pipe
15#endif
16
12#include <math.h> 17#include <math.h>
13#include <string.h> 18#include <string.h>
14#include <stdio.h> 19#include <stdio.h>
20#include <stdlib.h>
15 21
16#include <SDL.h> 22#include <SDL.h>
17#include <SDL_endian.h> 23#include <SDL_endian.h>
18#include <SDL_image.h> 24#include <SDL_image.h>
19#include <SDL_mixer.h> 25#include <SDL_mixer.h>
20#include <SDL_opengl.h> 26#include <SDL_opengl.h>
21 27
28#define PANGO_ENABLE_BACKEND
29#define G_DISABLE_CAST_CHECKS
30
22#include <glib/gmacros.h> 31#include <glib/gmacros.h>
23 32
24#include <pango/pango.h> 33#include <pango/pango.h>
25#include <pango/pangofc-fontmap.h>
26#include <pango/pangoft2.h>
27#include <pango/pangocairo.h>
28 34
29#ifndef _WIN32 35#ifndef _WIN32
30# include <sys/types.h> 36# include <sys/types.h>
31# include <sys/socket.h> 37# include <sys/socket.h>
32# include <netinet/in.h> 38# include <netinet/in.h>
39 typedef signed char int8_t; 45 typedef signed char int8_t;
40 typedef signed short int16_t; 46 typedef signed short int16_t;
41 typedef signed int int32_t; 47 typedef signed int int32_t;
42#endif 48#endif
43 49
44#include "glext.h" 50#define OBJ_STR "\xef\xbf\xbc" /* U+FFFC, objetc replacement character */
45 51
46#define FOW_DARKNESS 32 52#define FOW_DARKNESS 32
47 53
48#define MAP_EXTEND_X 32 54#define MAP_EXTEND_X 32
49#define MAP_EXTEND_Y 512 55#define MAP_EXTEND_Y 512
50 56
51#define MIN_FONT_HEIGHT 10 57#define MIN_FONT_HEIGHT 10
58
59#if 0
60# define PARACHUTE SDL_INIT_NOPARACHUTE
61#else
62# define PARACHUTE 0
63#endif
52 64
53static struct 65static struct
54{ 66{
55#define GL_FUNC(ptr,name) ptr name; 67#define GL_FUNC(ptr,name) ptr name;
56#include "glfunc.h" 68#include "glfunc.h"
65 gl.BlendFuncSeparateEXT (sa, da, saa, daa); 77 gl.BlendFuncSeparateEXT (sa, da, saa, daa);
66 else 78 else
67 glBlendFunc (sa, da); 79 glBlendFunc (sa, da);
68} 80}
69 81
82#include "texcache.c"
83
84#include "pango-font.c"
85#include "pango-fontmap.c"
86#include "pango-render.c"
87
70typedef Mix_Chunk *CFClient__MixChunk; 88typedef Mix_Chunk *CFPlus__MixChunk;
71typedef Mix_Music *CFClient__MixMusic; 89typedef Mix_Music *CFPlus__MixMusic;
72 90
73typedef PangoFontDescription *CFClient__Font; 91typedef PangoFontDescription *CFPlus__Font;
92
93static int
94shape_attr_p (PangoLayoutRun *run)
95{
96 GSList *attrs = run->item->analysis.extra_attrs;
97
98 while (attrs)
99 {
100 PangoAttribute *attr = attrs->data;
101
102 if (attr->klass->type == PANGO_ATTR_SHAPE)
103 return 1;
104
105 attrs = attrs->next;
106 }
107
108 return 0;
109}
74 110
75typedef struct cf_layout { 111typedef struct cf_layout {
76 PangoLayout *pl; // either derived from a cairo or ft2 context 112 PangoLayout *pl;
77 int rgba; // wether we use rgba (cairo) or grayscale (ft2)
78 float r, g, b, a; // default color for rgba mode 113 float r, g, b, a; // default color for rgba mode
79 int base_height; 114 int base_height;
80 CFClient__Font font; 115 CFPlus__Font font;
81} *CFClient__Layout; 116} *CFPlus__Layout;
82 117
83static CFClient__Font default_font; 118static CFPlus__Font default_font;
84static PangoContext *ft2_context, *cairo_context; 119static PangoContext *opengl_context;
85static PangoFontMap *ft2_fontmap, *cairo_fontmap; 120static PangoFontMap *opengl_fontmap;
86 121
87static void 122static void
88substitute_func (FcPattern *pattern, gpointer data) 123substitute_func (FcPattern *pattern, gpointer data)
89{ 124{
90 FcPatternAddBool (pattern, FC_HINTING , 1); 125 FcPatternAddBool (pattern, FC_HINTING, 1);
91#ifdef _WIN32 126#ifdef FC_HINT_STYLE
92 FcPatternAddBool (pattern, FC_AUTOHINT, 1); 127 FcPatternAddBool (pattern, FC_HINT_STYLE, FC_HINT_FULL);
93#else 128#endif
94 FcPatternAddBool (pattern, FC_AUTOHINT, 0); 129 FcPatternAddBool (pattern, FC_AUTOHINT, 0);
95#endif
96} 130}
97 131
98static void 132static void
99layout_update_font (CFClient__Layout self) 133layout_update_font (CFPlus__Layout self)
100{ 134{
101 /* use a random scale factor to account for unknown descenders, 0.8 works 135 /* use a random scale factor to account for unknown descenders, 0.8 works
102 * reasonably well with bitstream vera 136 * reasonably well with bitstream vera
103 */ 137 */
104 PangoFontDescription *font = self->font ? self->font : default_font; 138 PangoFontDescription *font = self->font ? self->font : default_font;
108 142
109 pango_layout_set_font_description (self->pl, font); 143 pango_layout_set_font_description (self->pl, font);
110} 144}
111 145
112static void 146static void
113layout_get_pixel_size (CFClient__Layout self, int *w, int *h) 147layout_get_pixel_size (CFPlus__Layout self, int *w, int *h)
114{ 148{
149 PangoRectangle rect;
150
151 // get_pixel_* wrongly rounds down
115 pango_layout_get_pixel_size (self->pl, w, h); 152 pango_layout_get_extents (self->pl, 0, &rect);
116 153
117 if (!*w) *w = 1; 154 rect.width = (rect.width + PANGO_SCALE - 1) / PANGO_SCALE;
118 if (!*h) *h = 1; 155 rect.height = (rect.height + PANGO_SCALE - 1) / PANGO_SCALE;
119 156
120 *w = (*w + 3) & ~3; 157 if (!rect.width) rect.width = 1;
158 if (!rect.height) rect.height = 1;
159
160 *w = rect.width;
161 *h = rect.height;
121} 162}
122 163
123typedef uint16_t mapface; 164typedef uint16_t mapface;
124 165
125typedef struct { 166typedef struct {
128 float s, t; 169 float s, t;
129 uint8_t r, g, b, a; 170 uint8_t r, g, b, a;
130} maptex; 171} maptex;
131 172
132typedef struct { 173typedef struct {
133 int16_t darkness; 174 uint32_t player;
134 mapface face[3]; 175 mapface face[3];
176 uint16_t darkness;
177 uint8_t stat_width, stat_hp, flags;
135} mapcell; 178} mapcell;
136 179
137typedef struct { 180typedef struct {
138 int32_t c0, c1; 181 int32_t c0, c1;
139 mapcell *col; 182 mapcell *col;
148 int texs; 191 int texs;
149 maptex *tex; 192 maptex *tex;
150 193
151 int32_t rows; 194 int32_t rows;
152 maprow *row; 195 maprow *row;
153} *CFClient__Map; 196} *CFPlus__Map;
154 197
155static char * 198static char *
156prepend (char *ptr, int sze, int inc) 199prepend (char *ptr, int sze, int inc)
157{ 200{
158 char *p; 201 char *p;
176 219
177#define Append(type,ptr,sze,inc) (ptr) = (type *)append ((char *)ptr, (sze) * sizeof (type), (inc) * sizeof (type)) 220#define Append(type,ptr,sze,inc) (ptr) = (type *)append ((char *)ptr, (sze) * sizeof (type), (inc) * sizeof (type))
178#define Prepend(type,ptr,sze,inc) (ptr) = (type *)prepend ((char *)ptr, (sze) * sizeof (type), (inc) * sizeof (type)) 221#define Prepend(type,ptr,sze,inc) (ptr) = (type *)prepend ((char *)ptr, (sze) * sizeof (type), (inc) * sizeof (type))
179 222
180static maprow * 223static maprow *
181map_get_row (CFClient__Map self, int y) 224map_get_row (CFPlus__Map self, int y)
182{ 225{
183 if (0 > y) 226 if (0 > y)
184 { 227 {
185 int extend = - y + MAP_EXTEND_Y; 228 int extend = - y + MAP_EXTEND_Y;
186 Prepend (maprow, self->row, self->rows, extend); 229 Prepend (maprow, self->row, self->rows, extend);
224 267
225 return row->col + (x - row->c0); 268 return row->col + (x - row->c0);
226} 269}
227 270
228static mapcell * 271static mapcell *
229map_get_cell (CFClient__Map self, int x, int y) 272map_get_cell (CFPlus__Map self, int x, int y)
230{ 273{
231 return row_get_cell (map_get_row (self, y), x); 274 return row_get_cell (map_get_row (self, y), x);
232} 275}
233 276
234static void 277static void
235map_clear (CFClient__Map self) 278map_clear (CFPlus__Map self)
236{ 279{
237 int r; 280 int r;
238 281
239 for (r = 0; r < self->rows; r++) 282 for (r = 0; r < self->rows; r++)
240 Safefree (self->row[r].col); 283 Safefree (self->row[r].col);
248 self->row = 0; 291 self->row = 0;
249 self->rows = 0; 292 self->rows = 0;
250} 293}
251 294
252static void 295static void
253map_blank (CFClient__Map self, int x0, int y0, int w, int h) 296map_blank (CFPlus__Map self, int x0, int y0, int w, int h)
254{ 297{
255 int x, y; 298 int x, y;
256 maprow *row; 299 maprow *row;
300 mapcell *cell;
257 301
258 for (y = y0; y < y0 + h; y++) 302 for (y = y0; y < y0 + h; y++)
259 if (y >= 0) 303 if (y >= 0)
260 { 304 {
261 if (y >= self->rows) 305 if (y >= self->rows)
267 if (x >= row->c0) 311 if (x >= row->c0)
268 { 312 {
269 if (x >= row->c1) 313 if (x >= row->c1)
270 break; 314 break;
271 315
272 row->col[x - row->c0].darkness = -1; 316 cell = row->col + x - row->c0;
317
318 cell->darkness = 0;
319 cell->stat_hp = 0;
273 } 320 }
274 } 321 }
275} 322}
276 323
277static void 324static void
298 ev.data2 = 0; 345 ev.data2 = 0;
299 346
300 SDL_PushEvent ((SDL_Event *)&ev); 347 SDL_PushEvent ((SDL_Event *)&ev);
301} 348}
302 349
350static unsigned int
351minpot (unsigned int n)
352{
353 if (!n)
354 return 0;
355
356 --n;
357
358 n |= n >> 1;
359 n |= n >> 2;
360 n |= n >> 4;
361 n |= n >> 8;
362 n |= n >> 16;
363
364 return n + 1;
365}
366
303MODULE = CFClient PACKAGE = CFClient 367MODULE = CFPlus PACKAGE = CFPlus
304 368
305PROTOTYPES: ENABLE 369PROTOTYPES: ENABLE
306 370
307BOOT: 371BOOT:
308{ 372{
309 HV *stash = gv_stashpv ("CFClient", 1); 373 HV *stash = gv_stashpv ("CFPlus", 1);
310 static const struct { 374 static const struct {
311 const char *name; 375 const char *name;
312 IV iv; 376 IV iv;
313 } *civ, const_iv[] = { 377 } *civ, const_iv[] = {
314# define const_iv(name) { # name, (IV)name } 378# define const_iv(name) { # name, (IV)name }
415 479
416 for (civ = const_iv + sizeof (const_iv) / sizeof (const_iv [0]); civ-- > const_iv; ) 480 for (civ = const_iv + sizeof (const_iv) / sizeof (const_iv [0]); civ-- > const_iv; )
417 newCONSTSUB (stash, (char *)civ->name, newSViv (civ->iv)); 481 newCONSTSUB (stash, (char *)civ->name, newSViv (civ->iv));
418} 482}
419 483
484int
485in_destruct ()
486 CODE:
487 RETVAL = PL_main_cv == Nullcv;
488 OUTPUT:
489 RETVAL
490
491NV floor (NV x)
492
493NV ceil (NV x)
494
420void 495void
421pango_init () 496pango_init ()
422 CODE: 497 CODE:
423{ 498{
424 // delayed, so it can pick up new fonts added by AddFontResourceEx
425 ft2_fontmap = pango_ft2_font_map_new (); 499 opengl_fontmap = pango_opengl_font_map_new ();
426 pango_ft2_font_map_set_default_substitute ((PangoFT2FontMap *)ft2_fontmap, substitute_func, 0, 0); 500 pango_opengl_font_map_set_default_substitute ((PangoOpenGLFontMap *)opengl_fontmap, substitute_func, 0, 0);
427 ft2_context = pango_ft2_font_map_create_context ((PangoFT2FontMap *)ft2_fontmap); 501 opengl_context = pango_opengl_font_map_create_context ((PangoOpenGLFontMap *)opengl_fontmap);
428
429 cairo_fontmap = pango_cairo_font_map_get_default ();
430 cairo_context = pango_cairo_font_map_create_context ((PangoCairoFontMap *)cairo_fontmap);
431} 502}
432 503
433int 504int
434SDL_Init (U32 flags = SDL_INIT_VIDEO | SDL_INIT_AUDIO) 505SDL_Init (U32 flags = SDL_INIT_VIDEO | SDL_INIT_AUDIO | PARACHUTE)
435 506
436void 507void
437SDL_Quit () 508SDL_Quit ()
438 509
439void 510void
446 SDL_GL_SetAttribute (SDL_GL_GREEN_SIZE, 5); 517 SDL_GL_SetAttribute (SDL_GL_GREEN_SIZE, 5);
447 SDL_GL_SetAttribute (SDL_GL_BLUE_SIZE, 5); 518 SDL_GL_SetAttribute (SDL_GL_BLUE_SIZE, 5);
448 SDL_GL_SetAttribute (SDL_GL_ALPHA_SIZE, 1); 519 SDL_GL_SetAttribute (SDL_GL_ALPHA_SIZE, 1);
449 520
450 SDL_GL_SetAttribute (SDL_GL_BUFFER_SIZE, 15); 521 SDL_GL_SetAttribute (SDL_GL_BUFFER_SIZE, 15);
451 SDL_GL_SetAttribute (SDL_GL_DEPTH_SIZE, 16); 522 SDL_GL_SetAttribute (SDL_GL_DEPTH_SIZE, 0);
452 523
453 SDL_GL_SetAttribute (SDL_GL_ACCUM_RED_SIZE, 0); 524 SDL_GL_SetAttribute (SDL_GL_ACCUM_RED_SIZE, 0);
454 SDL_GL_SetAttribute (SDL_GL_ACCUM_GREEN_SIZE, 0); 525 SDL_GL_SetAttribute (SDL_GL_ACCUM_GREEN_SIZE, 0);
455 SDL_GL_SetAttribute (SDL_GL_ACCUM_BLUE_SIZE, 0); 526 SDL_GL_SetAttribute (SDL_GL_ACCUM_BLUE_SIZE, 0);
456 SDL_GL_SetAttribute (SDL_GL_ACCUM_ALPHA_SIZE, 0); 527 SDL_GL_SetAttribute (SDL_GL_ACCUM_ALPHA_SIZE, 0);
457 528
458 SDL_GL_SetAttribute (SDL_GL_DOUBLEBUFFER, 1); 529 SDL_GL_SetAttribute (SDL_GL_DOUBLEBUFFER, 1);
530#if SDL_VERSION_ATLEAST(1,2,10)
531 SDL_GL_SetAttribute (SDL_GL_ACCELERATED_VISUAL, 1);
532 SDL_GL_SetAttribute (SDL_GL_SWAP_CONTROL, 1);
533#endif
459 534
460 SDL_EnableUNICODE (1); 535 SDL_EnableUNICODE (1);
461 SDL_EnableKeyRepeat (SDL_DEFAULT_REPEAT_DELAY, SDL_DEFAULT_REPEAT_INTERVAL); 536 SDL_EnableKeyRepeat (SDL_DEFAULT_REPEAT_DELAY, SDL_DEFAULT_REPEAT_INTERVAL);
462 537
463 m = SDL_ListModes (0, SDL_FULLSCREEN | SDL_OPENGL); 538 m = SDL_ListModes (0, SDL_FULLSCREEN | SDL_OPENGL);
550 hv_store (hv, "data1", 5, newSViv ((IV)ev.user.data1), 0); 625 hv_store (hv, "data1", 5, newSViv ((IV)ev.user.data1), 0);
551 hv_store (hv, "data2", 5, newSViv ((IV)ev.user.data2), 0); 626 hv_store (hv, "data2", 5, newSViv ((IV)ev.user.data2), 0);
552 break; 627 break;
553 } 628 }
554 629
555 XPUSHs (sv_2mortal (newRV_noinc ((SV *)hv))); 630 XPUSHs (sv_2mortal (sv_bless (newRV_noinc ((SV *)hv), gv_stashpv ("CFPlus::UI::Event", 1))));
556 } 631 }
557} 632}
558 633
559int 634int
560Mix_OpenAudio (int frequency = 48000, int format = MIX_DEFAULT_FORMAT, int channels = 1, int chunksize = 2048) 635Mix_OpenAudio (int frequency = 48000, int format = MIX_DEFAULT_FORMAT, int channels = 1, int chunksize = 2048)
576#endif 651#endif
577 652
578void 653void
579add_font (char *file) 654add_font (char *file)
580 CODE: 655 CODE:
581 FcConfigAppFontAddFile (0, (const FcChar8 *)file); /* no idea wether this is required */ 656 FcConfigAppFontAddFile (0, (const FcChar8 *)file);
582#ifdef _WIN32
583 // cairo... sigh... requires win2000
584 AddFontResourceEx (file, FR_PRIVATE, 0);
585#endif
586 657
587void 658void
588load_image_inline (SV *image_) 659load_image_inline (SV *image_)
589 ALIAS: 660 ALIAS:
590 load_image_file = 1 661 load_image_file = 1
632 703
633 surface2 = SDL_ConvertSurface (surface, &fmt, SDL_SWSURFACE); 704 surface2 = SDL_ConvertSurface (surface, &fmt, SDL_SWSURFACE);
634 705
635 assert (surface2->pitch == surface2->w * 4); 706 assert (surface2->pitch == surface2->w * 4);
636 707
708 SDL_LockSurface (surface2);
637 EXTEND (SP, 5); 709 EXTEND (SP, 6);
638 PUSHs (sv_2mortal (newSViv (surface2->w))); 710 PUSHs (sv_2mortal (newSViv (surface2->w)));
639 PUSHs (sv_2mortal (newSViv (surface2->h))); 711 PUSHs (sv_2mortal (newSViv (surface2->h)));
640 SDL_LockSurface (surface2);
641 PUSHs (sv_2mortal (newSVpvn (surface2->pixels, surface2->h * surface2->pitch))); 712 PUSHs (sv_2mortal (newSVpvn (surface2->pixels, surface2->h * surface2->pitch)));
642 SDL_UnlockSurface (surface2);
643 PUSHs (sv_2mortal (newSViv (surface->flags & (SDL_SRCCOLORKEY | SDL_SRCALPHA) ? GL_RGBA : GL_RGB))); 713 PUSHs (sv_2mortal (newSViv (surface->flags & (SDL_SRCCOLORKEY | SDL_SRCALPHA) ? GL_RGBA : GL_RGB)));
644 PUSHs (sv_2mortal (newSViv (GL_RGBA))); 714 PUSHs (sv_2mortal (newSViv (GL_RGBA)));
645 PUSHs (sv_2mortal (newSViv (GL_UNSIGNED_BYTE))); 715 PUSHs (sv_2mortal (newSViv (GL_UNSIGNED_BYTE)));
716 SDL_UnlockSurface (surface2);
646 717
647 SDL_FreeSurface (surface); 718 SDL_FreeSurface (surface);
648 SDL_FreeSurface (surface2); 719 SDL_FreeSurface (surface2);
649} 720}
650 721
686 CODE: 757 CODE:
687 fprintf (stderr, "FATAL: %s\n", message); 758 fprintf (stderr, "FATAL: %s\n", message);
688#ifdef _WIN32 759#ifdef _WIN32
689 MessageBox (0, message, "Crossfire+ Fatal Error", MB_OK | MB_ICONERROR); 760 MessageBox (0, message, "Crossfire+ Fatal Error", MB_OK | MB_ICONERROR);
690#endif 761#endif
691 exit (1); 762 _exit (1);
692 763
764void
765_exit (int retval)
766 CODE:
767 _exit (retval);
768
693MODULE = CFClient PACKAGE = CFClient::Font 769MODULE = CFPlus PACKAGE = CFPlus::Font
694 770
695CFClient::Font 771CFPlus::Font
696new_from_file (SV *class, char *path, int id = 0) 772new_from_file (SV *class, char *path, int id = 0)
697 CODE: 773 CODE:
698{ 774{
699 int count; 775 int count;
700 FcPattern *pattern = FcFreeTypeQuery ((const FcChar8 *)path, id, 0, &count); 776 FcPattern *pattern = FcFreeTypeQuery ((const FcChar8 *)path, id, 0, &count);
703} 779}
704 OUTPUT: 780 OUTPUT:
705 RETVAL 781 RETVAL
706 782
707void 783void
708DESTROY (CFClient::Font self) 784DESTROY (CFPlus::Font self)
709 CODE: 785 CODE:
710 pango_font_description_free (self); 786 pango_font_description_free (self);
711 787
712void 788void
713make_default (CFClient::Font self) 789make_default (CFPlus::Font self)
714 CODE: 790 CODE:
715 default_font = self; 791 default_font = self;
716 792
717MODULE = CFClient PACKAGE = CFClient::Layout 793MODULE = CFPlus PACKAGE = CFPlus::Layout
718 794
719CFClient::Layout 795void
720new (SV *class, int rgba = 0) 796reset_glyph_cache ()
721 CODE: 797 CODE:
722#if _WIN32 798 tc_clear ();
723 //rgba = 0;//D makes text nicer, breaks TextView 799
724#endif 800CFPlus::Layout
801new (SV *class)
802 CODE:
725 New (0, RETVAL, 1, struct cf_layout); 803 New (0, RETVAL, 1, struct cf_layout);
726 804
727 RETVAL->pl = pango_layout_new (rgba ? cairo_context : ft2_context); 805 RETVAL->pl = pango_layout_new (opengl_context);
728 RETVAL->rgba = rgba;
729 RETVAL->r = 1.; 806 RETVAL->r = 1.;
730 RETVAL->g = 1.; 807 RETVAL->g = 1.;
731 RETVAL->b = 1.; 808 RETVAL->b = 1.;
732 RETVAL->a = 1.; 809 RETVAL->a = 1.;
733 RETVAL->base_height = MIN_FONT_HEIGHT; 810 RETVAL->base_height = MIN_FONT_HEIGHT;
737 layout_update_font (RETVAL); 814 layout_update_font (RETVAL);
738 OUTPUT: 815 OUTPUT:
739 RETVAL 816 RETVAL
740 817
741void 818void
742DESTROY (CFClient::Layout self) 819DESTROY (CFPlus::Layout self)
743 CODE: 820 CODE:
744 g_object_unref (self->pl); 821 g_object_unref (self->pl);
745 Safefree (self); 822 Safefree (self);
746 823
747int
748is_rgba (CFClient::Layout self)
749 CODE:
750 RETVAL = self->rgba;
751 OUTPUT:
752 RETVAL
753
754void 824void
755set_text (CFClient::Layout self, SV *text_) 825set_text (CFPlus::Layout self, SV *text_)
756 CODE: 826 CODE:
757{ 827{
758 STRLEN textlen; 828 STRLEN textlen;
759 char *text = SvPVutf8 (text_, textlen); 829 char *text = SvPVutf8 (text_, textlen);
760 830
761 pango_layout_set_text (self->pl, text, textlen); 831 pango_layout_set_text (self->pl, text, textlen);
762} 832}
763 833
764void 834void
765set_markup (CFClient::Layout self, SV *text_) 835set_markup (CFPlus::Layout self, SV *text_)
766 CODE: 836 CODE:
767{ 837{
768 STRLEN textlen; 838 STRLEN textlen;
769 char *text = SvPVutf8 (text_, textlen); 839 char *text = SvPVutf8 (text_, textlen);
770 840
771 pango_layout_set_markup (self->pl, text, textlen); 841 pango_layout_set_markup (self->pl, text, textlen);
772} 842}
773 843
844void
845set_shapes (CFPlus::Layout self, ...)
846 CODE:
847{
848 PangoAttrList *attrs = 0;
849 const char *text = pango_layout_get_text (self->pl);
850 const char *pos = text;
851 int arg = 4;
852
853 while (arg < items && (pos = strstr (pos, OBJ_STR)))
854 {
855 PangoRectangle inkrect, rect;
856 PangoAttribute *attr;
857
858 int x = SvIV (ST (arg - 3));
859 int y = SvIV (ST (arg - 2));
860 int w = SvIV (ST (arg - 1));
861 int h = SvIV (ST (arg ));
862
863 inkrect.x = 0;
864 inkrect.y = 0;
865 inkrect.width = 0;
866 inkrect.height = 0;
867
868 rect.x = x * PANGO_SCALE;
869 rect.y = y * PANGO_SCALE;
870 rect.width = w * PANGO_SCALE;
871 rect.height = h * PANGO_SCALE;
872
873 if (!attrs)
874 attrs = pango_layout_get_attributes (self->pl);
875
876 attr = pango_attr_shape_new (&inkrect, &rect);
877 attr->start_index = pos - text;
878 attr->end_index = attr->start_index + sizeof (OBJ_STR) - 1;
879 pango_attr_list_insert (attrs, attr);
880
881 arg += 4;
882 pos += sizeof (OBJ_STR) - 1;
883 }
884
885 if (attrs)
886 pango_layout_set_attributes (self->pl, attrs);
887}
888
889void
890get_shapes (CFPlus::Layout self)
891 PPCODE:
892{
893 PangoLayoutIter *iter = pango_layout_get_iter (self->pl);
894
895 do
896 {
897 PangoLayoutRun *run = pango_layout_iter_get_run (iter);
898
899 if (run && shape_attr_p (run))
900 {
901 PangoRectangle extents;
902 pango_layout_iter_get_run_extents (iter, 0, &extents);
903
904 EXTEND (SP, 2);
905 PUSHs (sv_2mortal (newSViv (PANGO_PIXELS (extents.x))));
906 PUSHs (sv_2mortal (newSViv (PANGO_PIXELS (extents.y))));
907 }
908 }
909 while (pango_layout_iter_next_run (iter));
910
911 pango_layout_iter_free (iter);
912}
913
914int
915has_wrapped (CFPlus::Layout self)
916 CODE:
917{
918 int lines = 1;
919 const char *text = pango_layout_get_text (self->pl);
920
921 while (*text)
922 lines += *text++ == '\n';
923
924 RETVAL = lines < pango_layout_get_line_count (self->pl);
925}
926 OUTPUT:
927 RETVAL
928
774SV * 929SV *
775get_text (CFClient::Layout self) 930get_text (CFPlus::Layout self)
776 CODE: 931 CODE:
777 RETVAL = newSVpv (pango_layout_get_text (self->pl), 0); 932 RETVAL = newSVpv (pango_layout_get_text (self->pl), 0);
778 SvUTF8_on (RETVAL); 933 sv_utf8_decode (RETVAL);
779 OUTPUT: 934 OUTPUT:
780 RETVAL 935 RETVAL
781 936
782void 937void
783set_foreground (CFClient::Layout self, float r, float g, float b, float a = 1.) 938set_foreground (CFPlus::Layout self, float r, float g, float b, float a = 1.)
784 CODE: 939 CODE:
785 self->r = r; 940 self->r = r;
786 self->g = g; 941 self->g = g;
787 self->b = b; 942 self->b = b;
788 self->a = a; 943 self->a = a;
789 944
790void 945void
791set_font (CFClient::Layout self, CFClient::Font font = 0) 946set_font (CFPlus::Layout self, CFPlus::Font font = 0)
792 CODE: 947 CODE:
793 if (self->font != font) 948 if (self->font != font)
794 { 949 {
795 self->font = font; 950 self->font = font;
796 layout_update_font (self); 951 layout_update_font (self);
797 } 952 }
798 953
799void 954void
800set_height (CFClient::Layout self, int base_height) 955set_height (CFPlus::Layout self, int base_height)
801 CODE: 956 CODE:
802 if (self->base_height != base_height) 957 if (self->base_height != base_height)
803 { 958 {
804 self->base_height = base_height; 959 self->base_height = base_height;
805 layout_update_font (self); 960 layout_update_font (self);
806 } 961 }
807 962
808void 963void
809set_width (CFClient::Layout self, int max_width = -1) 964set_width (CFPlus::Layout self, int max_width = -1)
810 CODE: 965 CODE:
811 pango_layout_set_width (self->pl, max_width < 0 ? max_width : max_width * PANGO_SCALE); 966 pango_layout_set_width (self->pl, max_width < 0 ? max_width : max_width * PANGO_SCALE);
812 967
813void 968void
814set_indent (CFClient::Layout self, int indent) 969set_indent (CFPlus::Layout self, int indent)
815 CODE: 970 CODE:
816 pango_layout_set_indent (self->pl, indent * PANGO_SCALE); 971 pango_layout_set_indent (self->pl, indent * PANGO_SCALE);
817 972
818void 973void
819set_spacing (CFClient::Layout self, int spacing) 974set_spacing (CFPlus::Layout self, int spacing)
820 CODE: 975 CODE:
821 pango_layout_set_spacing (self->pl, spacing * PANGO_SCALE); 976 pango_layout_set_spacing (self->pl, spacing * PANGO_SCALE);
822 977
823void 978void
824set_ellipsise (CFClient::Layout self, int ellipsise) 979set_ellipsise (CFPlus::Layout self, int ellipsise)
825 CODE: 980 CODE:
826 pango_layout_set_ellipsize (self->pl, 981 pango_layout_set_ellipsize (self->pl,
827 ellipsise == 1 ? PANGO_ELLIPSIZE_START 982 ellipsise == 1 ? PANGO_ELLIPSIZE_START
828 : ellipsise == 2 ? PANGO_ELLIPSIZE_MIDDLE 983 : ellipsise == 2 ? PANGO_ELLIPSIZE_MIDDLE
829 : ellipsise == 3 ? PANGO_ELLIPSIZE_END 984 : ellipsise == 3 ? PANGO_ELLIPSIZE_END
830 : PANGO_ELLIPSIZE_NONE 985 : PANGO_ELLIPSIZE_NONE
831 ); 986 );
832 987
833void 988void
834set_single_paragraph_mode (CFClient::Layout self, int spm) 989set_single_paragraph_mode (CFPlus::Layout self, int spm)
835 CODE: 990 CODE:
836 pango_layout_set_single_paragraph_mode (self->pl, !!spm); 991 pango_layout_set_single_paragraph_mode (self->pl, !!spm);
837 992
838void 993void
839size (CFClient::Layout self) 994size (CFPlus::Layout self)
840 PPCODE: 995 PPCODE:
841{ 996{
842 int w, h; 997 int w, h;
843 998
844 layout_get_pixel_size (self, &w, &h); 999 layout_get_pixel_size (self, &w, &h);
847 PUSHs (sv_2mortal (newSViv (w))); 1002 PUSHs (sv_2mortal (newSViv (w)));
848 PUSHs (sv_2mortal (newSViv (h))); 1003 PUSHs (sv_2mortal (newSViv (h)));
849} 1004}
850 1005
851int 1006int
1007descent (CFPlus::Layout self)
1008 CODE:
1009{
1010 PangoRectangle rect;
1011 PangoLayoutLine *line = pango_layout_get_line (self->pl, 0);
1012 pango_layout_line_get_pixel_extents (line, 0, &rect);
1013 RETVAL = PANGO_DESCENT (rect);
1014}
1015 OUTPUT:
1016 RETVAL
1017
1018int
852xy_to_index (CFClient::Layout self, int x, int y) 1019xy_to_index (CFPlus::Layout self, int x, int y)
853 CODE: 1020 CODE:
854{ 1021{
855 int index, trailing; 1022 int index, trailing;
856 pango_layout_xy_to_index (self->pl, x * PANGO_SCALE, y * PANGO_SCALE, &index, &trailing); 1023 pango_layout_xy_to_index (self->pl, x * PANGO_SCALE, y * PANGO_SCALE, &index, &trailing);
857 RETVAL = index; 1024 RETVAL = index;
858} 1025}
859 OUTPUT: 1026 OUTPUT:
860 RETVAL 1027 RETVAL
861 1028
862void 1029void
863cursor_pos (CFClient::Layout self, int index) 1030cursor_pos (CFPlus::Layout self, int index)
864 PPCODE: 1031 PPCODE:
865{ 1032{
866 PangoRectangle strong_pos; 1033 PangoRectangle strong_pos;
867 pango_layout_get_cursor_pos (self->pl, index, &strong_pos, 0); 1034 pango_layout_get_cursor_pos (self->pl, index, &strong_pos, 0);
868 1035
871 PUSHs (sv_2mortal (newSViv (strong_pos.y / PANGO_SCALE))); 1038 PUSHs (sv_2mortal (newSViv (strong_pos.y / PANGO_SCALE)));
872 PUSHs (sv_2mortal (newSViv (strong_pos.height / PANGO_SCALE))); 1039 PUSHs (sv_2mortal (newSViv (strong_pos.height / PANGO_SCALE)));
873} 1040}
874 1041
875void 1042void
876render (CFClient::Layout self) 1043render (CFPlus::Layout self, float x, float y, int flags = 0)
877 PPCODE: 1044 PPCODE:
878{ 1045 pango_opengl_render_layout_subpixel (
879 SV *retval; 1046 self->pl,
880 int w, h; 1047 x * PANGO_SCALE, y * PANGO_SCALE,
1048 self->r, self->g, self->b, self->a,
1049 flags
1050 );
881 1051
882 layout_get_pixel_size (self, &w, &h); 1052MODULE = CFPlus PACKAGE = CFPlus::Texture
883 1053
884 if (self->rgba) 1054void
1055pad2pot (SV *data_, SV *w_, SV *h_)
1056 CODE:
1057{
1058 int ow = SvIV (w_);
1059 int oh = SvIV (h_);
1060
1061 if (ow && oh)
885 { 1062 {
886 cairo_surface_t *surface; 1063 int nw = minpot (ow);
887 cairo_t *cairo; 1064 int nh = minpot (oh);
888 1065
889 retval = newSV (w * h * 4); 1066 if (nw != ow || nh != oh)
890 SvPOK_only (retval);
891 SvCUR_set (retval, w * h * 4);
892
893 memset (SvPVX (retval), 0, w * h * 4);
894
895 surface = cairo_image_surface_create_for_data (
896 (void*)SvPVX (retval), CAIRO_FORMAT_ARGB32, w, h, w * 4);
897 cairo = cairo_create (surface);
898 cairo_set_source_rgba (cairo, self->r, self->g, self->b, self->a);
899
900 pango_cairo_show_layout (cairo, self->pl);
901
902 cairo_destroy (cairo);
903 cairo_surface_destroy (surface);
904
905 // what a mess, and its premultiplied, too :(
906 { 1067 {
907 uint32_t *p = (uint32_t *)SvPVX (retval); 1068 if (SvOK (data_))
908 uint32_t *e = p + w * h;
909
910 while (p < e)
911 { 1069 {
912 uint32_t rgba = *p; 1070 STRLEN datalen;
913 rgba = (rgba >> 24) | (rgba << 8); 1071 char *data = SvPVbyte (data_, datalen);
914#if 0 1072 int bpp = datalen / (ow * oh);
915#ifdef _WIN32 1073 SV *result_ = sv_2mortal (newSV (nw * nh * bpp));
916 {//D
917 uint8_t r = rgba >> 24;
918 uint8_t g = rgba >> 16;
919 uint8_t b = rgba >> 8;
920 uint8_t a = rgba >> 0;
921 1074
922 rgba = (rgba & 0xffffff00) | a; 1075 SvPOK_only (result_);
1076 SvCUR_set (result_, nw * nh * bpp);
1077
1078 memset (SvPVX (result_), 0, nw * nh * bpp);
1079 while (oh--)
1080 memcpy (SvPVX (result_) + oh * nw * bpp, data + oh * ow * bpp, ow * bpp);
1081
1082 sv_setsv (data_, result_);
923 } 1083 }
924#endif 1084
925#endif 1085 sv_setiv (w_, nw);
926 rgba = SDL_SwapBE32 (rgba); 1086 sv_setiv (h_, nh);
927 *p++ = rgba;
928 } 1087 }
929 }
930
931 EXTEND (SP, 5);
932 PUSHs (sv_2mortal (newSViv (w)));
933 PUSHs (sv_2mortal (newSViv (h)));
934 PUSHs (sv_2mortal (retval));
935 PUSHs (sv_2mortal (newSViv (GL_RGBA)));
936 PUSHs (sv_2mortal (newSViv (GL_RGBA)));
937 } 1088 }
938 else
939 {
940 FT_Bitmap bitmap;
941
942 retval = newSV (w * h);
943 SvPOK_only (retval);
944 SvCUR_set (retval, w * h);
945
946 bitmap.rows = h;
947 bitmap.width = w;
948 bitmap.pitch = w;
949 bitmap.buffer = (unsigned char*)SvPVX (retval);
950 bitmap.num_grays = 256;
951 bitmap.pixel_mode = FT_PIXEL_MODE_GRAY;
952
953 memset (bitmap.buffer, 0, w * h);
954
955 pango_ft2_render_layout (&bitmap, self->pl, 0 * PANGO_SCALE, 0 * PANGO_SCALE);
956
957 EXTEND (SP, 5);
958 PUSHs (sv_2mortal (newSViv (w)));
959 PUSHs (sv_2mortal (newSViv (h)));
960 PUSHs (sv_2mortal (retval));
961 PUSHs (sv_2mortal (newSViv (GL_ALPHA)));
962 PUSHs (sv_2mortal (newSViv (GL_ALPHA)));
963 }
964} 1089}
965 1090
966MODULE = CFClient PACKAGE = CFClient::Texture
967
968void 1091void
969draw_quad (SV *self, float x, float y, float w = 0, float h = 0) 1092draw_quad (SV *self, float x, float y, float w = 0., float h = 0.)
970 PROTOTYPE: $$$;$$ 1093 PROTOTYPE: $$$;$$
971 ALIAS: 1094 ALIAS:
972 draw_quad_alpha = 1 1095 draw_quad_alpha = 1
973 draw_quad_alpha_premultiplied = 2 1096 draw_quad_alpha_premultiplied = 2
974 CODE: 1097 CODE:
990 1113
991 if (ix == 2) 1114 if (ix == 2)
992 glBlendFunc (GL_ONE, GL_ONE_MINUS_SRC_ALPHA); 1115 glBlendFunc (GL_ONE, GL_ONE_MINUS_SRC_ALPHA);
993 else 1116 else
994 gl_BlendFuncSeparate (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA, 1117 gl_BlendFuncSeparate (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA,
995 GL_ONE_MINUS_DST_ALPHA, GL_ONE); 1118 GL_ONE , GL_ONE_MINUS_SRC_ALPHA);
996 1119
997 glEnable (GL_ALPHA_TEST); 1120 glEnable (GL_ALPHA_TEST);
998 glAlphaFunc (GL_GREATER, 0.01f); 1121 glAlphaFunc (GL_GREATER, 0.01f);
999 } 1122 }
1000 1123
1012 glDisable (GL_ALPHA_TEST); 1135 glDisable (GL_ALPHA_TEST);
1013 glDisable (GL_BLEND); 1136 glDisable (GL_BLEND);
1014 } 1137 }
1015} 1138}
1016 1139
1017MODULE = CFClient PACKAGE = CFClient::Map 1140MODULE = CFPlus PACKAGE = CFPlus::Map
1018 1141
1019CFClient::Map 1142CFPlus::Map
1020new (SV *class, int map_width, int map_height) 1143new (SV *class, int map_width, int map_height)
1021 CODE: 1144 CODE:
1022 New (0, RETVAL, 1, struct map); 1145 New (0, RETVAL, 1, struct map);
1023 RETVAL->x = 0; 1146 RETVAL->x = 0;
1024 RETVAL->y = 0; 1147 RETVAL->y = 0;
1034 RETVAL->row = 0; 1157 RETVAL->row = 0;
1035 OUTPUT: 1158 OUTPUT:
1036 RETVAL 1159 RETVAL
1037 1160
1038void 1161void
1039DESTROY (CFClient::Map self) 1162DESTROY (CFPlus::Map self)
1040 CODE: 1163 CODE:
1041{ 1164{
1042 map_clear (self); 1165 map_clear (self);
1043 Safefree (self->face); 1166 Safefree (self->face);
1167 Safefree (self->tex);
1044 Safefree (self); 1168 Safefree (self);
1045} 1169}
1046 1170
1047void 1171void
1048clear (CFClient::Map self) 1172clear (CFPlus::Map self)
1049 CODE: 1173 CODE:
1050 map_clear (self); 1174 map_clear (self);
1051 1175
1052void 1176void
1053set_face (CFClient::Map self, int face, int texid) 1177set_face (CFPlus::Map self, int face, int texid)
1054 CODE: 1178 CODE:
1055{ 1179{
1056 while (self->faces <= face) 1180 while (self->faces <= face)
1057 { 1181 {
1058 Append (mapface, self->face, self->faces, self->faces); 1182 Append (mapface, self->face, self->faces, self->faces);
1061 1185
1062 self->face [face] = texid; 1186 self->face [face] = texid;
1063} 1187}
1064 1188
1065void 1189void
1066set_texture (CFClient::Map self, int texid, int name, int w, int h, float s, float t, int r, int g, int b, int a) 1190set_texture (CFPlus::Map self, int texid, int name, int w, int h, float s, float t, int r, int g, int b, int a)
1067 CODE: 1191 CODE:
1068{ 1192{
1069 while (self->texs <= texid) 1193 while (self->texs <= texid)
1070 { 1194 {
1071 Append (maptex, self->tex, self->texs, self->texs); 1195 Append (maptex, self->tex, self->texs, self->texs);
1095 // from transparent color bleeding and ugly wrapping effects. 1219 // from transparent color bleeding and ugly wrapping effects.
1096 glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); 1220 glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
1097} 1221}
1098 1222
1099int 1223int
1100ox (CFClient::Map self) 1224ox (CFPlus::Map self)
1101 ALIAS: 1225 ALIAS:
1102 oy = 1 1226 oy = 1
1103 x = 2 1227 x = 2
1104 y = 3 1228 y = 3
1105 w = 4 1229 w = 4
1116 } 1240 }
1117 OUTPUT: 1241 OUTPUT:
1118 RETVAL 1242 RETVAL
1119 1243
1120void 1244void
1121scroll (CFClient::Map self, int dx, int dy) 1245scroll (CFPlus::Map self, int dx, int dy)
1122 CODE: 1246 CODE:
1123{ 1247{
1124 if (dx > 0) 1248 if (dx > 0)
1125 map_blank (self, self->x, self->y, dx - 1, self->h); 1249 map_blank (self, self->x, self->y, dx - 1, self->h);
1126 else if (dx < 0) 1250 else if (dx < 0)
1142 self->y += MAP_EXTEND_Y; 1266 self->y += MAP_EXTEND_Y;
1143 } 1267 }
1144} 1268}
1145 1269
1146void 1270void
1147map1a_update (CFClient::Map self, SV *data_) 1271map1a_update (CFPlus::Map self, SV *data_, int extmap)
1148 CODE: 1272 CODE:
1149{ 1273{
1150 uint8_t *data = (uint8_t *)SvPVbyte_nolen (data_); 1274 uint8_t *data = (uint8_t *)SvPVbyte_nolen (data_);
1151 uint8_t *data_end = (uint8_t *)SvEND (data_); 1275 uint8_t *data_end = (uint8_t *)SvEND (data_);
1152 mapcell *cell; 1276 mapcell *cell;
1153 int x, y, flags; 1277 int x, y, flags;
1154 1278
1155 while (data < data_end) 1279 while (data < data_end - 1)
1156 { 1280 {
1157 flags = (data [0] << 8) + data [1]; data += 2; 1281 flags = (data [0] << 8) + data [1]; data += 2;
1158 1282
1159 x = ((flags >> 10) & 63) + self->x; 1283 x = self->x + ((flags >> 10) & 63);
1160 y = ((flags >> 4) & 63) + self->y; 1284 y = self->y + ((flags >> 4) & 63);
1161 1285
1162 cell = map_get_cell (self, x, y); 1286 cell = map_get_cell (self, x, y);
1163 1287
1164 if (flags & 15) 1288 if (flags & 15)
1165 { 1289 {
1166 if (cell->darkness < 0) 1290 if (!cell->darkness)
1167 { 1291 {
1168 cell->darkness = 0; 1292 cell->darkness = 256;
1169 cell->face [0] = 0; 1293 cell->face [0] = 0;
1170 cell->face [1] = 0; 1294 cell->face [1] = 0;
1171 cell->face [2] = 0; 1295 cell->face [2] = 0;
1172 } 1296 }
1173 1297
1174 cell->darkness = flags & 8 ? *data++ : 255;
1175
1176 //TODO: don't trust server data to be in-range(!) 1298 //TODO: don't trust server data to be in-range(!)
1299
1300 if (flags & 8)
1301 {
1302 if (extmap)
1303 {
1304 uint8_t ext, cmd;
1305
1306 do
1307 {
1308 ext = *data++;
1309 cmd = ext & 0x3f;
1310
1311 if (cmd < 4)
1312 cell->darkness = 255 - ext * 64 + 1;
1313 else if (cmd == 5) // health
1314 {
1315 cell->stat_width = 1;
1316 cell->stat_hp = *data++;
1317 }
1318 else if (cmd == 6) // monster width
1319 cell->stat_width = *data++ + 1;
1320 else if (cmd == 0x47) // monster width
1321 {
1322 if (*data == 4)
1323 ; // decode player tag
1324
1325 data += *data + 1;
1326 }
1327 else if (cmd == 8) // cell flags
1328 cell->flags = *data++;
1329 else if (ext & 0x40) // unknown, multibyte => skip
1330 data += *data + 1;
1331 else
1332 data++;
1333 }
1334 while (ext & 0x80);
1335 }
1336 else
1337 cell->darkness = *data++ + 1;
1338 }
1177 1339
1178 if (flags & 4) 1340 if (flags & 4)
1179 { 1341 {
1180 cell->face [0] = self->face [(data [0] << 8) + data [1]]; data += 2; 1342 cell->face [0] = self->face [(data [0] << 8) + data [1]]; data += 2;
1181 } 1343 }
1189 { 1351 {
1190 cell->face [2] = self->face [(data [0] << 8) + data [1]]; data += 2; 1352 cell->face [2] = self->face [(data [0] << 8) + data [1]]; data += 2;
1191 } 1353 }
1192 } 1354 }
1193 else 1355 else
1356 {
1194 cell->darkness = -1; 1357 cell->darkness = 0;
1358 cell->stat_hp = 0;
1359 }
1195 } 1360 }
1196} 1361}
1197 1362
1198SV * 1363SV *
1199mapmap (CFClient::Map self, int x0, int y0, int w, int h) 1364mapmap (CFPlus::Map self, int x0, int y0, int w, int h)
1200 CODE: 1365 CODE:
1201{ 1366{
1202 int x1, x; 1367 int x1, x;
1203 int y1, y; 1368 int y1, y;
1204 int z; 1369 int z;
1254} 1419}
1255 OUTPUT: 1420 OUTPUT:
1256 RETVAL 1421 RETVAL
1257 1422
1258void 1423void
1259draw (CFClient::Map self, int shift_x, int shift_y, int x0, int y0, int sw, int sh) 1424draw (CFPlus::Map self, int shift_x, int shift_y, int x0, int y0, int sw, int sh)
1260 PPCODE: 1425 CODE:
1261{ 1426{
1262 int vx, vy; 1427 int vx, vy;
1263 int x, y, z; 1428 int x, y, z;
1264 int last_name; 1429 int last_name;
1265 mapface face; 1430 mapface face;
1266 int sw4 = (sw + 3) & ~3;
1267 SV *darkness_sv = sv_2mortal (newSV (sw4 * sh));
1268 uint8_t *darkness = (uint8_t *)SvPVX (darkness_sv);
1269 1431
1270 memset (darkness, 255, sw4 * sh);
1271 SvPOK_only (darkness_sv);
1272 SvCUR_set (darkness_sv, sw4 * sh);
1273
1274 vx = self->x + (self->w - sw) / 2 - shift_x; 1432 vx = self->x + self->w / 2 - sw / 2 - shift_x;
1275 vy = self->y + (self->h - sh) / 2 - shift_y; 1433 vy = self->y + self->h / 2 - sh / 2 - shift_y;
1276 1434
1277 /* 1435 /*
1278 int vx = self->vx = self->w >= sw 1436 int vx = self->vx = self->w >= sw
1279 ? self->x + (self->w - sw) / 2 1437 ? self->x + (self->w - sw) / 2
1280 : MIN (self->x, MAX (self->x + self->w - sw + 1, self->vx)); 1438 : MIN (self->x, MAX (self->x + self->w - sw + 1, self->vx));
1284 : MIN (self->y, MAX (self->y + self->h - sh + 1, self->vy)); 1442 : MIN (self->y, MAX (self->y + self->h - sh + 1, self->vy));
1285 */ 1443 */
1286 1444
1287 glColor4ub (255, 255, 255, 255); 1445 glColor4ub (255, 255, 255, 255);
1288 1446
1447 glEnable (GL_BLEND);
1289 glBlendFunc (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); 1448 glBlendFunc (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
1290 glEnable (GL_BLEND);
1291 glEnable (GL_TEXTURE_2D); 1449 glEnable (GL_TEXTURE_2D);
1292 glTexEnvi (GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE); 1450 glTexEnvi (GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
1293 1451
1294 glBegin (GL_QUADS); 1452 glBegin (GL_QUADS);
1295 1453
1304 for (x = 0; x < sw; x++) 1462 for (x = 0; x < sw; x++)
1305 if (row->c0 <= x + vx && x + vx < row->c1) 1463 if (row->c0 <= x + vx && x + vx < row->c1)
1306 { 1464 {
1307 mapcell *cell = row->col + (x + vx - row->c0); 1465 mapcell *cell = row->col + (x + vx - row->c0);
1308 1466
1309 darkness[y * sw4 + x] = cell->darkness < 0
1310 ? 255 - FOW_DARKNESS
1311 : 255 - cell->darkness;
1312
1313 face = cell->face [z]; 1467 face = cell->face [z];
1314 1468
1315 if (face) 1469 if (face && face < self->texs)
1316 { 1470 {
1317 maptex tex = self->tex [face]; 1471 maptex tex = self->tex [face];
1318
1319 int px = (x + 1) * 32 - tex.w; 1472 int px = (x + 1) * 32 - tex.w;
1320 int py = (y + 1) * 32 - tex.h; 1473 int py = (y + 1) * 32 - tex.h;
1321 1474
1322 if (last_name != tex.name) 1475 if (last_name != tex.name)
1323 { 1476 {
1324 glEnd (); 1477 glEnd ();
1325 last_name = tex.name;
1326 glBindTexture (GL_TEXTURE_2D, last_name); 1478 glBindTexture (GL_TEXTURE_2D, last_name = tex.name);
1327 glBegin (GL_QUADS); 1479 glBegin (GL_QUADS);
1328 } 1480 }
1329 1481
1330 glTexCoord2f (0 , 0 ); glVertex2f (px , py ); 1482 glTexCoord2f (0 , 0 ); glVertex2f (px , py );
1331 glTexCoord2f (0 , tex.t); glVertex2f (px , py + tex.h); 1483 glTexCoord2f (0 , tex.t); glVertex2f (px , py + tex.h);
1332 glTexCoord2f (tex.s, tex.t); glVertex2f (px + tex.w, py + tex.h); 1484 glTexCoord2f (tex.s, tex.t); glVertex2f (px + tex.w, py + tex.h);
1333 glTexCoord2f (tex.s, 0 ); glVertex2f (px + tex.w, py ); 1485 glTexCoord2f (tex.s, 0 ); glVertex2f (px + tex.w, py );
1334 } 1486 }
1487
1488 if (cell->flags && z == 2)
1489 {
1490 if (cell->flags & 1)
1491 {
1492 maptex tex = self->tex [1];
1493 int px = (x + 1) * 32 - tex.w + 2;
1494 int py = (y + 1) * 32 - tex.h - 6;
1495
1496 glEnd ();
1497 glBindTexture (GL_TEXTURE_2D, last_name = tex.name);
1498 glBegin (GL_QUADS);
1499
1500 glTexCoord2f (0 , 0 ); glVertex2f (px , py );
1501 glTexCoord2f (0 , tex.t); glVertex2f (px , py + tex.h);
1502 glTexCoord2f (tex.s, tex.t); glVertex2f (px + tex.w, py + tex.h);
1503 glTexCoord2f (tex.s, 0 ); glVertex2f (px + tex.w, py );
1504 }
1505 }
1335 } 1506 }
1336 } 1507 }
1337 1508
1338 glEnd (); 1509 glEnd ();
1339 1510
1340 glDisable (GL_TEXTURE_2D); 1511 glDisable (GL_TEXTURE_2D);
1341 glDisable (GL_BLEND); 1512 glDisable (GL_BLEND);
1513
1514 // top layer: overlays such as the health bar
1515 for (y = 0; y < sh; y++)
1516 if (0 <= y + vy && y + vy < self->rows)
1517 {
1518 maprow *row = self->row + (y + vy);
1519
1520 for (x = 0; x < sw; x++)
1521 if (row->c0 <= x + vx && x + vx < row->c1)
1522 {
1523 mapcell *cell = row->col + (x + vx - row->c0);
1524
1525 int px = x * 32;
1526 int py = y * 32;
1527
1528 if (cell->stat_hp)
1529 {
1530 int width = cell->stat_width * 32;
1531 int thick = sh / 28 + 1 + cell->stat_width;
1532
1533 glColor3ub (0, 0, 0);
1534 glRectf (px + 1, py - thick - 2,
1535 px + width - 1, py);
1536
1537 glColor3ub (cell->stat_hp, 255 - cell->stat_hp, 0);
1538 glRectf (px + 2,
1539 py - thick - 1,
1540 px + width - 2 - cell->stat_hp * (width - 4) / 255, py - 1);
1541 }
1542 }
1543 }
1544}
1545
1546void
1547draw_magicmap (CFPlus::Map self, int dx, int dy, int w, int h, unsigned char *data)
1548 CODE:
1549{
1550 static float color[16][3] = {
1551 { 0.00F, 0.00F, 0.00F },
1552 { 1.00F, 1.00F, 1.00F },
1553 { 0.00F, 0.00F, 0.55F },
1554 { 1.00F, 0.00F, 0.00F },
1555
1556 { 1.00F, 0.54F, 0.00F },
1557 { 0.11F, 0.56F, 1.00F },
1558 { 0.93F, 0.46F, 0.00F },
1559 { 0.18F, 0.54F, 0.34F },
1560
1561 { 0.56F, 0.73F, 0.56F },
1562 { 0.80F, 0.80F, 0.80F },
1563 { 0.55F, 0.41F, 0.13F },
1564 { 0.99F, 0.77F, 0.26F },
1565
1566 { 0.74F, 0.65F, 0.41F },
1567
1568 { 0.00F, 1.00F, 1.00F },
1569 { 1.00F, 0.00F, 1.00F },
1570 { 1.00F, 1.00F, 0.00F },
1571 };
1572
1573 int x, y;
1574
1575 glEnable (GL_TEXTURE_2D);
1576 glTexEnvi (GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
1577 glEnable (GL_BLEND);
1578 glBlendFunc (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
1579 glBegin (GL_QUADS);
1580
1581 for (y = 0; y < h; y++)
1582 for (x = 0; x < w; x++)
1583 {
1584 unsigned char m = data [x + y * w];
1585
1586 if (m)
1587 {
1588 float *c = color [m & 15];
1589
1590 float tx1 = m & 0x40 ? 0.5 : 0.;
1591 float tx2 = tx1 + 0.5;
1592
1593 glColor4f (c[0], c[1], c[2], 0.75);
1594 glTexCoord2f (tx1, 0.); glVertex2i (x , y );
1595 glTexCoord2f (tx1, 1.); glVertex2i (x , y + 1);
1596 glTexCoord2f (tx2, 1.); glVertex2i (x + 1, y + 1);
1597 glTexCoord2f (tx2, 0.); glVertex2i (x + 1, y );
1598 }
1599 }
1600
1601 glEnd ();
1602 glDisable (GL_BLEND);
1603 glDisable (GL_TEXTURE_2D);
1604}
1605
1606void
1607fow_texture (CFPlus::Map self, int shift_x, int shift_y, int x0, int y0, int sw, int sh)
1608 PPCODE:
1609{
1610 int vx, vy;
1611 int x, y;
1612 int sw4 = (sw + 3) & ~3;
1613 SV *darkness_sv = sv_2mortal (newSV (sw4 * sh));
1614 uint8_t *darkness = (uint8_t *)SvPVX (darkness_sv);
1615
1616 memset (darkness, 255, sw4 * sh);
1617 SvPOK_only (darkness_sv);
1618 SvCUR_set (darkness_sv, sw4 * sh);
1619
1620 vx = self->x + (self->w - sw + 1) / 2 - shift_x;
1621 vy = self->y + (self->h - sh + 1) / 2 - shift_y;
1622
1623 for (y = 0; y < sh; y++)
1624 if (0 <= y + vy && y + vy < self->rows)
1625 {
1626 maprow *row = self->row + (y + vy);
1627
1628 for (x = 0; x < sw; x++)
1629 if (row->c0 <= x + vx && x + vx < row->c1)
1630 {
1631 mapcell *cell = row->col + (x + vx - row->c0);
1632
1633 darkness[y * sw4 + x] = cell->darkness
1634 ? 255 - (cell->darkness - 1)
1635 : 255 - FOW_DARKNESS;
1636 }
1637 }
1342 1638
1343 EXTEND (SP, 3); 1639 EXTEND (SP, 3);
1344 PUSHs (sv_2mortal (newSViv (sw4))); 1640 PUSHs (sv_2mortal (newSViv (sw4)));
1345 PUSHs (sv_2mortal (newSViv (sh))); 1641 PUSHs (sv_2mortal (newSViv (sh)));
1346 PUSHs (darkness_sv); 1642 PUSHs (darkness_sv);
1347} 1643}
1348 1644
1349SV * 1645SV *
1350get_rect (CFClient::Map self, int x0, int y0, int w, int h) 1646get_rect (CFPlus::Map self, int x0, int y0, int w, int h)
1351 CODE: 1647 CODE:
1352{ 1648{
1353 int x, y, x1, y1; 1649 int x, y, x1, y1;
1354 SV *data_sv = newSV (w * h * 7 + 5); 1650 SV *data_sv = newSV (w * h * 7 + 5);
1355 uint8_t *data = (uint8_t *)SvPVX (data_sv); 1651 uint8_t *data = (uint8_t *)SvPVX (data_sv);
1417} 1713}
1418 OUTPUT: 1714 OUTPUT:
1419 RETVAL 1715 RETVAL
1420 1716
1421void 1717void
1422set_rect (CFClient::Map self, int x0, int y0, uint8_t *data) 1718set_rect (CFPlus::Map self, int x0, int y0, uint8_t *data)
1423 PPCODE: 1719 PPCODE:
1424{ 1720{
1425 int x, y, z; 1721 int x, y, z;
1426 int w, h; 1722 int w, h;
1427 int x1, y1; 1723 int x1, y1;
1459 1755
1460 if (flags & 1) { face[0] = *data++ << 8; face[0] |= *data++; } 1756 if (flags & 1) { face[0] = *data++ << 8; face[0] |= *data++; }
1461 if (flags & 2) { face[1] = *data++ << 8; face[1] |= *data++; } 1757 if (flags & 2) { face[1] = *data++ << 8; face[1] |= *data++; }
1462 if (flags & 4) { face[2] = *data++ << 8; face[2] |= *data++; } 1758 if (flags & 4) { face[2] = *data++ << 8; face[2] |= *data++; }
1463 1759
1464 if (cell->darkness <= 0) 1760 if (cell->darkness == 0)
1465 { 1761 {
1466 cell->darkness = -1; 1762 cell->darkness = 0;
1467 1763
1468 for (z = 0; z <= 2; z++) 1764 for (z = 0; z <= 2; z++)
1469 { 1765 {
1470 cell->face[z] = face[z]; 1766 cell->face[z] = face[z];
1471 1767
1476 } 1772 }
1477 } 1773 }
1478 } 1774 }
1479} 1775}
1480 1776
1481MODULE = CFClient PACKAGE = CFClient::MixChunk 1777MODULE = CFPlus PACKAGE = CFPlus::MixChunk
1482 1778
1483CFClient::MixChunk 1779CFPlus::MixChunk
1484new_from_file (SV *class, char *path) 1780new_from_file (SV *class, char *path)
1485 CODE: 1781 CODE:
1486 RETVAL = Mix_LoadWAV (path); 1782 RETVAL = Mix_LoadWAV (path);
1487 OUTPUT: 1783 OUTPUT:
1488 RETVAL 1784 RETVAL
1489 1785
1490void 1786void
1491DESTROY (CFClient::MixChunk self) 1787DESTROY (CFPlus::MixChunk self)
1492 CODE: 1788 CODE:
1493 Mix_FreeChunk (self); 1789 Mix_FreeChunk (self);
1494 1790
1495int 1791int
1496volume (CFClient::MixChunk self, int volume = -1) 1792volume (CFPlus::MixChunk self, int volume = -1)
1497 CODE: 1793 CODE:
1498 RETVAL = Mix_VolumeChunk (self, volume); 1794 RETVAL = Mix_VolumeChunk (self, volume);
1499 OUTPUT: 1795 OUTPUT:
1500 RETVAL 1796 RETVAL
1501 1797
1502int 1798int
1503play (CFClient::MixChunk self, int channel = -1, int loops = 0, int ticks = -1) 1799play (CFPlus::MixChunk self, int channel = -1, int loops = 0, int ticks = -1)
1504 CODE: 1800 CODE:
1505 RETVAL = Mix_PlayChannelTimed (channel, self, loops, ticks); 1801 RETVAL = Mix_PlayChannelTimed (channel, self, loops, ticks);
1506 OUTPUT: 1802 OUTPUT:
1507 RETVAL 1803 RETVAL
1508 1804
1509MODULE = CFClient PACKAGE = CFClient::MixMusic 1805MODULE = CFPlus PACKAGE = CFPlus::MixMusic
1510 1806
1511int 1807int
1512volume (int volume = -1) 1808volume (int volume = -1)
1513 CODE: 1809 CODE:
1514 RETVAL = Mix_VolumeMusic (volume); 1810 RETVAL = Mix_VolumeMusic (volume);
1515 OUTPUT: 1811 OUTPUT:
1516 RETVAL 1812 RETVAL
1517 1813
1518CFClient::MixMusic 1814CFPlus::MixMusic
1519new_from_file (SV *class, char *path) 1815new_from_file (SV *class, char *path)
1520 CODE: 1816 CODE:
1521 RETVAL = Mix_LoadMUS (path); 1817 RETVAL = Mix_LoadMUS (path);
1522 OUTPUT: 1818 OUTPUT:
1523 RETVAL 1819 RETVAL
1524 1820
1525void 1821void
1526DESTROY (CFClient::MixMusic self) 1822DESTROY (CFPlus::MixMusic self)
1527 CODE: 1823 CODE:
1528 Mix_FreeMusic (self); 1824 Mix_FreeMusic (self);
1529 1825
1530int 1826int
1531play (CFClient::MixMusic self, int loops = -1) 1827play (CFPlus::MixMusic self, int loops = -1)
1532 CODE: 1828 CODE:
1533 RETVAL = Mix_PlayMusic (self, loops); 1829 RETVAL = Mix_PlayMusic (self, loops);
1534 OUTPUT: 1830 OUTPUT:
1535 RETVAL 1831 RETVAL
1536 1832
1537MODULE = CFClient PACKAGE = CFClient::OpenGL 1833MODULE = CFPlus PACKAGE = CFPlus::OpenGL
1538 1834
1539BOOT: 1835BOOT:
1540{ 1836{
1541 HV *stash = gv_stashpv ("CFClient::OpenGL", 1); 1837 HV *stash = gv_stashpv ("CFPlus::OpenGL", 1);
1542 static const struct { 1838 static const struct {
1543 const char *name; 1839 const char *name;
1544 IV iv; 1840 IV iv;
1545 } *civ, const_iv[] = { 1841 } *civ, const_iv[] = {
1546# define const_iv(name) { # name, (IV)name } 1842# define const_iv(name) { # name, (IV)name }
1553 const_iv (GL_SCISSOR_TEST), 1849 const_iv (GL_SCISSOR_TEST),
1554 const_iv (GL_DEPTH_TEST), 1850 const_iv (GL_DEPTH_TEST),
1555 const_iv (GL_ALPHA_TEST), 1851 const_iv (GL_ALPHA_TEST),
1556 const_iv (GL_NORMALIZE), 1852 const_iv (GL_NORMALIZE),
1557 const_iv (GL_RESCALE_NORMAL), 1853 const_iv (GL_RESCALE_NORMAL),
1854 const_iv (GL_FRONT),
1855 const_iv (GL_BACK),
1558 const_iv (GL_AND), 1856 const_iv (GL_AND),
1559 const_iv (GL_ONE), 1857 const_iv (GL_ONE),
1560 const_iv (GL_ZERO), 1858 const_iv (GL_ZERO),
1561 const_iv (GL_SRC_ALPHA), 1859 const_iv (GL_SRC_ALPHA),
1562 const_iv (GL_SRC_ALPHA_SATURATE), 1860 const_iv (GL_DST_ALPHA),
1563 const_iv (GL_ONE_MINUS_SRC_ALPHA), 1861 const_iv (GL_ONE_MINUS_SRC_ALPHA),
1564 const_iv (GL_ONE_MINUS_DST_ALPHA), 1862 const_iv (GL_ONE_MINUS_DST_ALPHA),
1863 const_iv (GL_SRC_ALPHA_SATURATE),
1565 const_iv (GL_RGB), 1864 const_iv (GL_RGB),
1566 const_iv (GL_RGBA), 1865 const_iv (GL_RGBA),
1866 const_iv (GL_RGBA4),
1867 const_iv (GL_RGBA8),
1868 const_iv (GL_RGB5_A1),
1567 const_iv (GL_UNSIGNED_BYTE), 1869 const_iv (GL_UNSIGNED_BYTE),
1568 const_iv (GL_UNSIGNED_SHORT), 1870 const_iv (GL_UNSIGNED_SHORT),
1569 const_iv (GL_UNSIGNED_INT), 1871 const_iv (GL_UNSIGNED_INT),
1570 const_iv (GL_ALPHA), 1872 const_iv (GL_ALPHA),
1571 const_iv (GL_INTENSITY), 1873 const_iv (GL_INTENSITY),
1603 const_iv (GL_SEPARABLE_2D), 1905 const_iv (GL_SEPARABLE_2D),
1604 const_iv (GL_CONVOLUTION_2D), 1906 const_iv (GL_CONVOLUTION_2D),
1605 const_iv (GL_CONVOLUTION_BORDER_MODE), 1907 const_iv (GL_CONVOLUTION_BORDER_MODE),
1606 const_iv (GL_CONSTANT_BORDER), 1908 const_iv (GL_CONSTANT_BORDER),
1607 const_iv (GL_LINES), 1909 const_iv (GL_LINES),
1910 const_iv (GL_LINE_STRIP),
1608 const_iv (GL_LINE_LOOP), 1911 const_iv (GL_LINE_LOOP),
1609 const_iv (GL_QUADS), 1912 const_iv (GL_QUADS),
1610 const_iv (GL_QUAD_STRIP), 1913 const_iv (GL_QUAD_STRIP),
1611 const_iv (GL_TRIANGLES), 1914 const_iv (GL_TRIANGLES),
1612 const_iv (GL_TRIANGLE_STRIP), 1915 const_iv (GL_TRIANGLE_STRIP),
1645 OUTPUT: 1948 OUTPUT:
1646 RETVAL 1949 RETVAL
1647 1950
1648int glGetError () 1951int glGetError ()
1649 1952
1953void glFinish ()
1954
1650void glClear (int mask) 1955void glClear (int mask)
1651 1956
1652void glClearColor (float r, float g, float b, float a = 1.0) 1957void glClearColor (float r, float g, float b, float a = 1.0)
1653 PROTOTYPE: @ 1958 PROTOTYPE: @
1654 1959
1677void glPushMatrix () 1982void glPushMatrix ()
1678 1983
1679void glPopMatrix () 1984void glPopMatrix ()
1680 1985
1681void glLoadIdentity () 1986void glLoadIdentity ()
1987
1988void glDrawBuffer (int buffer)
1989
1990void glReadBuffer (int buffer)
1682 1991
1683# near_ and far_ are due to microsofts buggy "c" compiler 1992# near_ and far_ are due to microsofts buggy "c" compiler
1684void glFrustum (double left, double right, double bottom, double top, double near_, double far_) 1993void glFrustum (double left, double right, double bottom, double top, double near_, double far_)
1685 1994
1686# near_ and far_ are due to microsofts buggy "c" compiler 1995# near_ and far_ are due to microsofts buggy "c" compiler
1716 r *= a; 2025 r *= a;
1717 g *= a; 2026 g *= a;
1718 b *= a; 2027 b *= a;
1719 } 2028 }
1720 // microsoft visual "c" rounds instead of truncating... 2029 // microsoft visual "c" rounds instead of truncating...
1721 glColor4ub (MIN ((int)(r * 256.f), 255), 2030 glColor4f (r, g, b, a);
1722 MIN ((int)(g * 256.f), 255),
1723 MIN ((int)(b * 256.f), 255),
1724 MIN ((int)(a * 256.f), 255));
1725 2031
1726void glInterleavedArrays (int format, int stride, char *data) 2032void glInterleavedArrays (int format, int stride, char *data)
1727 2033
1728void glDrawElements (int mode, int count, int type, char *indices) 2034void glDrawElements (int mode, int count, int type, char *indices)
1729 2035
1806 2112
1807void glEndList () 2113void glEndList ()
1808 2114
1809void glCallList (int list) 2115void glCallList (int list)
1810 2116
1811

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines