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.109 by root, Wed Jun 7 23:29:13 2006 UTC vs.
Revision 1.141 by root, Fri Aug 18 01:01:00 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);
126#ifdef FC_HINT_STYLE
91 FcPatternAddBool (pattern, FC_HINT_STYLE, FC_HINT_FULL); 127 FcPatternAddBool (pattern, FC_HINT_STYLE, FC_HINT_FULL);
92#ifdef _WIN32 128#endif
93 FcPatternAddBool (pattern, FC_AUTOHINT, 1);
94#else
95 FcPatternAddBool (pattern, FC_AUTOHINT, 0); 129 FcPatternAddBool (pattern, FC_AUTOHINT, 0);
96#endif
97} 130}
98 131
99static void 132static void
100layout_update_font (CFClient__Layout self) 133layout_update_font (CFPlus__Layout self)
101{ 134{
102 /* 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
103 * reasonably well with bitstream vera 136 * reasonably well with bitstream vera
104 */ 137 */
105 PangoFontDescription *font = self->font ? self->font : default_font; 138 PangoFontDescription *font = self->font ? self->font : default_font;
109 142
110 pango_layout_set_font_description (self->pl, font); 143 pango_layout_set_font_description (self->pl, font);
111} 144}
112 145
113static void 146static void
114layout_get_pixel_size (CFClient__Layout self, int *w, int *h) 147layout_get_pixel_size (CFPlus__Layout self, int *w, int *h)
115{ 148{
149 PangoRectangle rect;
150
151 // get_pixel_* wrongly rounds down
116 pango_layout_get_pixel_size (self->pl, w, h); 152 pango_layout_get_extents (self->pl, 0, &rect);
117 153
118 if (!*w) *w = 1; 154 rect.width = (rect.width + PANGO_SCALE - 1) / PANGO_SCALE;
119 if (!*h) *h = 1; 155 rect.height = (rect.height + PANGO_SCALE - 1) / PANGO_SCALE;
120 156
121 *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;
122} 162}
123 163
124typedef uint16_t mapface; 164typedef uint16_t mapface;
125 165
126typedef struct { 166typedef struct {
149 int texs; 189 int texs;
150 maptex *tex; 190 maptex *tex;
151 191
152 int32_t rows; 192 int32_t rows;
153 maprow *row; 193 maprow *row;
154} *CFClient__Map; 194} *CFPlus__Map;
155 195
156static char * 196static char *
157prepend (char *ptr, int sze, int inc) 197prepend (char *ptr, int sze, int inc)
158{ 198{
159 char *p; 199 char *p;
177 217
178#define Append(type,ptr,sze,inc) (ptr) = (type *)append ((char *)ptr, (sze) * sizeof (type), (inc) * sizeof (type)) 218#define Append(type,ptr,sze,inc) (ptr) = (type *)append ((char *)ptr, (sze) * sizeof (type), (inc) * sizeof (type))
179#define Prepend(type,ptr,sze,inc) (ptr) = (type *)prepend ((char *)ptr, (sze) * sizeof (type), (inc) * sizeof (type)) 219#define Prepend(type,ptr,sze,inc) (ptr) = (type *)prepend ((char *)ptr, (sze) * sizeof (type), (inc) * sizeof (type))
180 220
181static maprow * 221static maprow *
182map_get_row (CFClient__Map self, int y) 222map_get_row (CFPlus__Map self, int y)
183{ 223{
184 if (0 > y) 224 if (0 > y)
185 { 225 {
186 int extend = - y + MAP_EXTEND_Y; 226 int extend = - y + MAP_EXTEND_Y;
187 Prepend (maprow, self->row, self->rows, extend); 227 Prepend (maprow, self->row, self->rows, extend);
225 265
226 return row->col + (x - row->c0); 266 return row->col + (x - row->c0);
227} 267}
228 268
229static mapcell * 269static mapcell *
230map_get_cell (CFClient__Map self, int x, int y) 270map_get_cell (CFPlus__Map self, int x, int y)
231{ 271{
232 return row_get_cell (map_get_row (self, y), x); 272 return row_get_cell (map_get_row (self, y), x);
233} 273}
234 274
235static void 275static void
236map_clear (CFClient__Map self) 276map_clear (CFPlus__Map self)
237{ 277{
238 int r; 278 int r;
239 279
240 for (r = 0; r < self->rows; r++) 280 for (r = 0; r < self->rows; r++)
241 Safefree (self->row[r].col); 281 Safefree (self->row[r].col);
249 self->row = 0; 289 self->row = 0;
250 self->rows = 0; 290 self->rows = 0;
251} 291}
252 292
253static void 293static void
254map_blank (CFClient__Map self, int x0, int y0, int w, int h) 294map_blank (CFPlus__Map self, int x0, int y0, int w, int h)
255{ 295{
256 int x, y; 296 int x, y;
257 maprow *row; 297 maprow *row;
258 298
259 for (y = y0; y < y0 + h; y++) 299 for (y = y0; y < y0 + h; y++)
299 ev.data2 = 0; 339 ev.data2 = 0;
300 340
301 SDL_PushEvent ((SDL_Event *)&ev); 341 SDL_PushEvent ((SDL_Event *)&ev);
302} 342}
303 343
344static unsigned int
345minpot (unsigned int n)
346{
347 if (!n)
348 return 0;
349
350 --n;
351
352 n |= n >> 1;
353 n |= n >> 2;
354 n |= n >> 4;
355 n |= n >> 8;
356 n |= n >> 16;
357
358 return n + 1;
359}
360
304MODULE = CFClient PACKAGE = CFClient 361MODULE = CFPlus PACKAGE = CFPlus
305 362
306PROTOTYPES: ENABLE 363PROTOTYPES: ENABLE
307 364
308BOOT: 365BOOT:
309{ 366{
310 HV *stash = gv_stashpv ("CFClient", 1); 367 HV *stash = gv_stashpv ("CFPlus", 1);
311 static const struct { 368 static const struct {
312 const char *name; 369 const char *name;
313 IV iv; 370 IV iv;
314 } *civ, const_iv[] = { 371 } *civ, const_iv[] = {
315# define const_iv(name) { # name, (IV)name } 372# define const_iv(name) { # name, (IV)name }
416 473
417 for (civ = const_iv + sizeof (const_iv) / sizeof (const_iv [0]); civ-- > const_iv; ) 474 for (civ = const_iv + sizeof (const_iv) / sizeof (const_iv [0]); civ-- > const_iv; )
418 newCONSTSUB (stash, (char *)civ->name, newSViv (civ->iv)); 475 newCONSTSUB (stash, (char *)civ->name, newSViv (civ->iv));
419} 476}
420 477
478int
479in_destruct ()
480 CODE:
481 RETVAL = PL_main_cv == Nullcv;
482 OUTPUT:
483 RETVAL
484
485NV floor (NV x)
486
487NV ceil (NV x)
488
421void 489void
422pango_init () 490pango_init ()
423 CODE: 491 CODE:
424 // delayed, so it can pick up new fonts added by AddFontResourceEx
425{ 492{
426 {
427 ft2_fontmap = pango_ft2_font_map_new (); 493 opengl_fontmap = pango_opengl_font_map_new ();
428 pango_ft2_font_map_set_default_substitute ((PangoFT2FontMap *)ft2_fontmap, substitute_func, 0, 0); 494 pango_opengl_font_map_set_default_substitute ((PangoOpenGLFontMap *)opengl_fontmap, substitute_func, 0, 0);
429 ft2_context = pango_ft2_font_map_create_context ((PangoFT2FontMap *)ft2_fontmap); 495 opengl_context = pango_opengl_font_map_create_context ((PangoOpenGLFontMap *)opengl_fontmap);
430 }
431 {
432 cairo_font_options_t *fopt = cairo_font_options_create ();
433 cairo_fontmap = pango_cairo_font_map_get_default ();
434 cairo_context = pango_cairo_font_map_create_context ((PangoCairoFontMap *)cairo_fontmap);
435#ifdef _WIN32
436 // cairo looks like shit eaten twice on windows
437 cairo_font_options_set_antialias (fopt, CAIRO_ANTIALIAS_NONE);
438#else
439 cairo_font_options_set_antialias (fopt, CAIRO_ANTIALIAS_GRAY);
440#endif
441 cairo_font_options_set_hint_style (fopt, CAIRO_HINT_STYLE_FULL);
442 cairo_font_options_set_hint_metrics (fopt, CAIRO_HINT_METRICS_ON);
443 pango_cairo_context_set_font_options (cairo_context, fopt);
444 cairo_font_options_destroy (fopt);
445 }
446} 496}
447 497
448int 498int
449SDL_Init (U32 flags = SDL_INIT_VIDEO | SDL_INIT_AUDIO) 499SDL_Init (U32 flags = SDL_INIT_VIDEO | SDL_INIT_AUDIO | PARACHUTE)
450 500
451void 501void
452SDL_Quit () 502SDL_Quit ()
453 503
454void 504void
461 SDL_GL_SetAttribute (SDL_GL_GREEN_SIZE, 5); 511 SDL_GL_SetAttribute (SDL_GL_GREEN_SIZE, 5);
462 SDL_GL_SetAttribute (SDL_GL_BLUE_SIZE, 5); 512 SDL_GL_SetAttribute (SDL_GL_BLUE_SIZE, 5);
463 SDL_GL_SetAttribute (SDL_GL_ALPHA_SIZE, 1); 513 SDL_GL_SetAttribute (SDL_GL_ALPHA_SIZE, 1);
464 514
465 SDL_GL_SetAttribute (SDL_GL_BUFFER_SIZE, 15); 515 SDL_GL_SetAttribute (SDL_GL_BUFFER_SIZE, 15);
466 SDL_GL_SetAttribute (SDL_GL_DEPTH_SIZE, 16); 516 SDL_GL_SetAttribute (SDL_GL_DEPTH_SIZE, 0);
467 517
468 SDL_GL_SetAttribute (SDL_GL_ACCUM_RED_SIZE, 0); 518 SDL_GL_SetAttribute (SDL_GL_ACCUM_RED_SIZE, 0);
469 SDL_GL_SetAttribute (SDL_GL_ACCUM_GREEN_SIZE, 0); 519 SDL_GL_SetAttribute (SDL_GL_ACCUM_GREEN_SIZE, 0);
470 SDL_GL_SetAttribute (SDL_GL_ACCUM_BLUE_SIZE, 0); 520 SDL_GL_SetAttribute (SDL_GL_ACCUM_BLUE_SIZE, 0);
471 SDL_GL_SetAttribute (SDL_GL_ACCUM_ALPHA_SIZE, 0); 521 SDL_GL_SetAttribute (SDL_GL_ACCUM_ALPHA_SIZE, 0);
472 522
473 SDL_GL_SetAttribute (SDL_GL_DOUBLEBUFFER, 1); 523 SDL_GL_SetAttribute (SDL_GL_DOUBLEBUFFER, 1);
524#if SDL_VERSION_ATLEAST(1,2,10)
525 SDL_GL_SetAttribute (SDL_GL_ACCELERATED_VISUAL, 1);
526 SDL_GL_SetAttribute (SDL_GL_SWAP_CONTROL, 1);
527#endif
474 528
475 SDL_EnableUNICODE (1); 529 SDL_EnableUNICODE (1);
476 SDL_EnableKeyRepeat (SDL_DEFAULT_REPEAT_DELAY, SDL_DEFAULT_REPEAT_INTERVAL); 530 SDL_EnableKeyRepeat (SDL_DEFAULT_REPEAT_DELAY, SDL_DEFAULT_REPEAT_INTERVAL);
477 531
478 m = SDL_ListModes (0, SDL_FULLSCREEN | SDL_OPENGL); 532 m = SDL_ListModes (0, SDL_FULLSCREEN | SDL_OPENGL);
565 hv_store (hv, "data1", 5, newSViv ((IV)ev.user.data1), 0); 619 hv_store (hv, "data1", 5, newSViv ((IV)ev.user.data1), 0);
566 hv_store (hv, "data2", 5, newSViv ((IV)ev.user.data2), 0); 620 hv_store (hv, "data2", 5, newSViv ((IV)ev.user.data2), 0);
567 break; 621 break;
568 } 622 }
569 623
570 XPUSHs (sv_2mortal (newRV_noinc ((SV *)hv))); 624 XPUSHs (sv_2mortal (sv_bless (newRV_noinc ((SV *)hv), gv_stashpv ("CFPlus::UI::Event", 1))));
571 } 625 }
572} 626}
573 627
574int 628int
575Mix_OpenAudio (int frequency = 48000, int format = MIX_DEFAULT_FORMAT, int channels = 1, int chunksize = 2048) 629Mix_OpenAudio (int frequency = 48000, int format = MIX_DEFAULT_FORMAT, int channels = 1, int chunksize = 2048)
591#endif 645#endif
592 646
593void 647void
594add_font (char *file) 648add_font (char *file)
595 CODE: 649 CODE:
596 FcConfigAppFontAddFile (0, (const FcChar8 *)file); /* no idea wether this is required */ 650 FcConfigAppFontAddFile (0, (const FcChar8 *)file);
597#ifdef _WIN32
598 // cairo... sigh... requires win2000
599 AddFontResourceEx (file, FR_PRIVATE, 0);
600#endif
601 651
602void 652void
603load_image_inline (SV *image_) 653load_image_inline (SV *image_)
604 ALIAS: 654 ALIAS:
605 load_image_file = 1 655 load_image_file = 1
647 697
648 surface2 = SDL_ConvertSurface (surface, &fmt, SDL_SWSURFACE); 698 surface2 = SDL_ConvertSurface (surface, &fmt, SDL_SWSURFACE);
649 699
650 assert (surface2->pitch == surface2->w * 4); 700 assert (surface2->pitch == surface2->w * 4);
651 701
702 SDL_LockSurface (surface2);
652 EXTEND (SP, 5); 703 EXTEND (SP, 6);
653 PUSHs (sv_2mortal (newSViv (surface2->w))); 704 PUSHs (sv_2mortal (newSViv (surface2->w)));
654 PUSHs (sv_2mortal (newSViv (surface2->h))); 705 PUSHs (sv_2mortal (newSViv (surface2->h)));
655 SDL_LockSurface (surface2);
656 PUSHs (sv_2mortal (newSVpvn (surface2->pixels, surface2->h * surface2->pitch))); 706 PUSHs (sv_2mortal (newSVpvn (surface2->pixels, surface2->h * surface2->pitch)));
657 SDL_UnlockSurface (surface2);
658 PUSHs (sv_2mortal (newSViv (surface->flags & (SDL_SRCCOLORKEY | SDL_SRCALPHA) ? GL_RGBA : GL_RGB))); 707 PUSHs (sv_2mortal (newSViv (surface->flags & (SDL_SRCCOLORKEY | SDL_SRCALPHA) ? GL_RGBA : GL_RGB)));
659 PUSHs (sv_2mortal (newSViv (GL_RGBA))); 708 PUSHs (sv_2mortal (newSViv (GL_RGBA)));
660 PUSHs (sv_2mortal (newSViv (GL_UNSIGNED_BYTE))); 709 PUSHs (sv_2mortal (newSViv (GL_UNSIGNED_BYTE)));
710 SDL_UnlockSurface (surface2);
661 711
662 SDL_FreeSurface (surface); 712 SDL_FreeSurface (surface);
663 SDL_FreeSurface (surface2); 713 SDL_FreeSurface (surface2);
664} 714}
665 715
701 CODE: 751 CODE:
702 fprintf (stderr, "FATAL: %s\n", message); 752 fprintf (stderr, "FATAL: %s\n", message);
703#ifdef _WIN32 753#ifdef _WIN32
704 MessageBox (0, message, "Crossfire+ Fatal Error", MB_OK | MB_ICONERROR); 754 MessageBox (0, message, "Crossfire+ Fatal Error", MB_OK | MB_ICONERROR);
705#endif 755#endif
706 exit (1); 756 _exit (1);
707 757
758void
759_exit (int retval)
760 CODE:
761 _exit (retval);
762
708MODULE = CFClient PACKAGE = CFClient::Font 763MODULE = CFPlus PACKAGE = CFPlus::Font
709 764
710CFClient::Font 765CFPlus::Font
711new_from_file (SV *class, char *path, int id = 0) 766new_from_file (SV *class, char *path, int id = 0)
712 CODE: 767 CODE:
713{ 768{
714 int count; 769 int count;
715 FcPattern *pattern = FcFreeTypeQuery ((const FcChar8 *)path, id, 0, &count); 770 FcPattern *pattern = FcFreeTypeQuery ((const FcChar8 *)path, id, 0, &count);
718} 773}
719 OUTPUT: 774 OUTPUT:
720 RETVAL 775 RETVAL
721 776
722void 777void
723DESTROY (CFClient::Font self) 778DESTROY (CFPlus::Font self)
724 CODE: 779 CODE:
725 pango_font_description_free (self); 780 pango_font_description_free (self);
726 781
727void 782void
728make_default (CFClient::Font self) 783make_default (CFPlus::Font self)
729 CODE: 784 CODE:
730 default_font = self; 785 default_font = self;
731 786
732MODULE = CFClient PACKAGE = CFClient::Layout 787MODULE = CFPlus PACKAGE = CFPlus::Layout
733 788
734CFClient::Layout 789void
735new (SV *class, int rgba = 0) 790reset_glyph_cache ()
791 CODE:
792 tc_clear ();
793
794CFPlus::Layout
795new (SV *class)
736 CODE: 796 CODE:
737 New (0, RETVAL, 1, struct cf_layout); 797 New (0, RETVAL, 1, struct cf_layout);
738 798
739 RETVAL->pl = pango_layout_new (rgba ? cairo_context : ft2_context); 799 RETVAL->pl = pango_layout_new (opengl_context);
740 RETVAL->rgba = rgba;
741 RETVAL->r = 1.; 800 RETVAL->r = 1.;
742 RETVAL->g = 1.; 801 RETVAL->g = 1.;
743 RETVAL->b = 1.; 802 RETVAL->b = 1.;
744 RETVAL->a = 1.; 803 RETVAL->a = 1.;
745 RETVAL->base_height = MIN_FONT_HEIGHT; 804 RETVAL->base_height = MIN_FONT_HEIGHT;
749 layout_update_font (RETVAL); 808 layout_update_font (RETVAL);
750 OUTPUT: 809 OUTPUT:
751 RETVAL 810 RETVAL
752 811
753void 812void
754DESTROY (CFClient::Layout self) 813DESTROY (CFPlus::Layout self)
755 CODE: 814 CODE:
756 g_object_unref (self->pl); 815 g_object_unref (self->pl);
757 Safefree (self); 816 Safefree (self);
758 817
759int
760is_rgba (CFClient::Layout self)
761 CODE:
762 RETVAL = self->rgba;
763 OUTPUT:
764 RETVAL
765
766void 818void
767set_text (CFClient::Layout self, SV *text_) 819set_text (CFPlus::Layout self, SV *text_)
768 CODE: 820 CODE:
769{ 821{
770 STRLEN textlen; 822 STRLEN textlen;
771 char *text = SvPVutf8 (text_, textlen); 823 char *text = SvPVutf8 (text_, textlen);
772 824
773 pango_layout_set_text (self->pl, text, textlen); 825 pango_layout_set_text (self->pl, text, textlen);
774} 826}
775 827
776void 828void
777set_markup (CFClient::Layout self, SV *text_) 829set_markup (CFPlus::Layout self, SV *text_)
778 CODE: 830 CODE:
779{ 831{
780 STRLEN textlen; 832 STRLEN textlen;
781 char *text = SvPVutf8 (text_, textlen); 833 char *text = SvPVutf8 (text_, textlen);
782 834
783 pango_layout_set_markup (self->pl, text, textlen); 835 pango_layout_set_markup (self->pl, text, textlen);
784} 836}
785 837
838void
839set_shapes (CFPlus::Layout self, ...)
840 CODE:
841{
842 PangoAttrList *attrs = 0;
843 const char *text = pango_layout_get_text (self->pl);
844 const char *pos = text;
845 int arg = 4;
846
847 while (arg < items && (pos = strstr (pos, OBJ_STR)))
848 {
849 PangoRectangle inkrect, rect;
850 PangoAttribute *attr;
851
852 int x = SvIV (ST (arg - 3));
853 int y = SvIV (ST (arg - 2));
854 int w = SvIV (ST (arg - 1));
855 int h = SvIV (ST (arg ));
856
857 inkrect.x = 0;
858 inkrect.y = 0;
859 inkrect.width = 0;
860 inkrect.height = 0;
861
862 rect.x = x * PANGO_SCALE;
863 rect.y = y * PANGO_SCALE;
864 rect.width = w * PANGO_SCALE;
865 rect.height = h * PANGO_SCALE;
866
867 if (!attrs)
868 attrs = pango_layout_get_attributes (self->pl);
869
870 attr = pango_attr_shape_new (&inkrect, &rect);
871 attr->start_index = pos - text;
872 attr->end_index = attr->start_index + sizeof (OBJ_STR) - 1;
873 pango_attr_list_insert (attrs, attr);
874
875 arg += 4;
876 pos += sizeof (OBJ_STR) - 1;
877 }
878
879 if (attrs)
880 pango_layout_set_attributes (self->pl, attrs);
881}
882
883void
884get_shapes (CFPlus::Layout self)
885 PPCODE:
886{
887 PangoLayoutIter *iter = pango_layout_get_iter (self->pl);
888
889 do
890 {
891 PangoLayoutRun *run = pango_layout_iter_get_run (iter);
892
893 if (run && shape_attr_p (run))
894 {
895 PangoRectangle extents;
896 pango_layout_iter_get_run_extents (iter, 0, &extents);
897
898 EXTEND (SP, 2);
899 PUSHs (sv_2mortal (newSViv (PANGO_PIXELS (extents.x))));
900 PUSHs (sv_2mortal (newSViv (PANGO_PIXELS (extents.y))));
901 }
902 }
903 while (pango_layout_iter_next_run (iter));
904
905 pango_layout_iter_free (iter);
906}
907
908int
909has_wrapped (CFPlus::Layout self)
910 CODE:
911{
912 int lines = 1;
913 const char *text = pango_layout_get_text (self->pl);
914
915 while (*text)
916 lines += *text++ == '\n';
917
918 RETVAL = lines < pango_layout_get_line_count (self->pl);
919}
920 OUTPUT:
921 RETVAL
922
786SV * 923SV *
787get_text (CFClient::Layout self) 924get_text (CFPlus::Layout self)
788 CODE: 925 CODE:
789 RETVAL = newSVpv (pango_layout_get_text (self->pl), 0); 926 RETVAL = newSVpv (pango_layout_get_text (self->pl), 0);
790 sv_utf8_decode (RETVAL); 927 sv_utf8_decode (RETVAL);
791 OUTPUT: 928 OUTPUT:
792 RETVAL 929 RETVAL
793 930
794void 931void
795set_foreground (CFClient::Layout self, float r, float g, float b, float a = 1.) 932set_foreground (CFPlus::Layout self, float r, float g, float b, float a = 1.)
796 CODE: 933 CODE:
797 self->r = r; 934 self->r = r;
798 self->g = g; 935 self->g = g;
799 self->b = b; 936 self->b = b;
800 self->a = a; 937 self->a = a;
801 938
802void 939void
803set_font (CFClient::Layout self, CFClient::Font font = 0) 940set_font (CFPlus::Layout self, CFPlus::Font font = 0)
804 CODE: 941 CODE:
805 if (self->font != font) 942 if (self->font != font)
806 { 943 {
807 self->font = font; 944 self->font = font;
808 layout_update_font (self); 945 layout_update_font (self);
809 } 946 }
810 947
811void 948void
812set_height (CFClient::Layout self, int base_height) 949set_height (CFPlus::Layout self, int base_height)
813 CODE: 950 CODE:
814 if (self->base_height != base_height) 951 if (self->base_height != base_height)
815 { 952 {
816 self->base_height = base_height; 953 self->base_height = base_height;
817 layout_update_font (self); 954 layout_update_font (self);
818 } 955 }
819 956
820void 957void
821set_width (CFClient::Layout self, int max_width = -1) 958set_width (CFPlus::Layout self, int max_width = -1)
822 CODE: 959 CODE:
823 pango_layout_set_width (self->pl, max_width < 0 ? max_width : max_width * PANGO_SCALE); 960 pango_layout_set_width (self->pl, max_width < 0 ? max_width : max_width * PANGO_SCALE);
824 961
825void 962void
826set_indent (CFClient::Layout self, int indent) 963set_indent (CFPlus::Layout self, int indent)
827 CODE: 964 CODE:
828 pango_layout_set_indent (self->pl, indent * PANGO_SCALE); 965 pango_layout_set_indent (self->pl, indent * PANGO_SCALE);
829 966
830void 967void
831set_spacing (CFClient::Layout self, int spacing) 968set_spacing (CFPlus::Layout self, int spacing)
832 CODE: 969 CODE:
833 pango_layout_set_spacing (self->pl, spacing * PANGO_SCALE); 970 pango_layout_set_spacing (self->pl, spacing * PANGO_SCALE);
834 971
835void 972void
836set_ellipsise (CFClient::Layout self, int ellipsise) 973set_ellipsise (CFPlus::Layout self, int ellipsise)
837 CODE: 974 CODE:
838 pango_layout_set_ellipsize (self->pl, 975 pango_layout_set_ellipsize (self->pl,
839 ellipsise == 1 ? PANGO_ELLIPSIZE_START 976 ellipsise == 1 ? PANGO_ELLIPSIZE_START
840 : ellipsise == 2 ? PANGO_ELLIPSIZE_MIDDLE 977 : ellipsise == 2 ? PANGO_ELLIPSIZE_MIDDLE
841 : ellipsise == 3 ? PANGO_ELLIPSIZE_END 978 : ellipsise == 3 ? PANGO_ELLIPSIZE_END
842 : PANGO_ELLIPSIZE_NONE 979 : PANGO_ELLIPSIZE_NONE
843 ); 980 );
844 981
845void 982void
846set_single_paragraph_mode (CFClient::Layout self, int spm) 983set_single_paragraph_mode (CFPlus::Layout self, int spm)
847 CODE: 984 CODE:
848 pango_layout_set_single_paragraph_mode (self->pl, !!spm); 985 pango_layout_set_single_paragraph_mode (self->pl, !!spm);
849 986
850void 987void
851size (CFClient::Layout self) 988size (CFPlus::Layout self)
852 PPCODE: 989 PPCODE:
853{ 990{
854 int w, h; 991 int w, h;
855 992
856 layout_get_pixel_size (self, &w, &h); 993 layout_get_pixel_size (self, &w, &h);
859 PUSHs (sv_2mortal (newSViv (w))); 996 PUSHs (sv_2mortal (newSViv (w)));
860 PUSHs (sv_2mortal (newSViv (h))); 997 PUSHs (sv_2mortal (newSViv (h)));
861} 998}
862 999
863int 1000int
1001descent (CFPlus::Layout self)
1002 CODE:
1003{
1004 PangoRectangle rect;
1005 PangoLayoutLine *line = pango_layout_get_line (self->pl, 0);
1006 pango_layout_line_get_pixel_extents (line, 0, &rect);
1007 RETVAL = PANGO_DESCENT (rect);
1008}
1009 OUTPUT:
1010 RETVAL
1011
1012int
864xy_to_index (CFClient::Layout self, int x, int y) 1013xy_to_index (CFPlus::Layout self, int x, int y)
865 CODE: 1014 CODE:
866{ 1015{
867 int index, trailing; 1016 int index, trailing;
868 pango_layout_xy_to_index (self->pl, x * PANGO_SCALE, y * PANGO_SCALE, &index, &trailing); 1017 pango_layout_xy_to_index (self->pl, x * PANGO_SCALE, y * PANGO_SCALE, &index, &trailing);
869 RETVAL = index; 1018 RETVAL = index;
870} 1019}
871 OUTPUT: 1020 OUTPUT:
872 RETVAL 1021 RETVAL
873 1022
874void 1023void
875cursor_pos (CFClient::Layout self, int index) 1024cursor_pos (CFPlus::Layout self, int index)
876 PPCODE: 1025 PPCODE:
877{ 1026{
878 PangoRectangle strong_pos; 1027 PangoRectangle strong_pos;
879 pango_layout_get_cursor_pos (self->pl, index, &strong_pos, 0); 1028 pango_layout_get_cursor_pos (self->pl, index, &strong_pos, 0);
880 1029
883 PUSHs (sv_2mortal (newSViv (strong_pos.y / PANGO_SCALE))); 1032 PUSHs (sv_2mortal (newSViv (strong_pos.y / PANGO_SCALE)));
884 PUSHs (sv_2mortal (newSViv (strong_pos.height / PANGO_SCALE))); 1033 PUSHs (sv_2mortal (newSViv (strong_pos.height / PANGO_SCALE)));
885} 1034}
886 1035
887void 1036void
888render (CFClient::Layout self) 1037render (CFPlus::Layout self, float x, float y, int flags = 0)
889 PPCODE: 1038 PPCODE:
890{ 1039 pango_opengl_render_layout_subpixel (
891 SV *retval; 1040 self->pl,
892 int w, h; 1041 x * PANGO_SCALE, y * PANGO_SCALE,
1042 self->r, self->g, self->b, self->a,
1043 flags
1044 );
893 1045
894 layout_get_pixel_size (self, &w, &h); 1046MODULE = CFPlus PACKAGE = CFPlus::Texture
895 1047
896 if (self->rgba) 1048void
1049pad2pot (SV *data_, SV *w_, SV *h_)
1050 CODE:
1051{
1052 int ow = SvIV (w_);
1053 int oh = SvIV (h_);
1054
1055 if (ow && oh)
897 { 1056 {
898 cairo_surface_t *surface; 1057 int nw = minpot (ow);
899 cairo_t *cairo; 1058 int nh = minpot (oh);
900 1059
901 retval = newSV (w * h * 4); 1060 if (nw != ow || nh != oh)
902 SvPOK_only (retval);
903 SvCUR_set (retval, w * h * 4);
904
905 memset (SvPVX (retval), 0, w * h * 4);
906
907 surface = cairo_image_surface_create_for_data (
908 (void*)SvPVX (retval), CAIRO_FORMAT_ARGB32, w, h, w * 4);
909 cairo = cairo_create (surface);
910 cairo_set_source_rgba (cairo, self->r, self->g, self->b, self->a);
911
912 pango_cairo_show_layout (cairo, self->pl);
913
914 cairo_destroy (cairo);
915 cairo_surface_destroy (surface);
916
917 // what a mess, and its premultiplied, too :(
918 { 1061 {
919 uint32_t *p = (uint32_t *)SvPVX (retval); 1062 if (SvOK (data_))
920 uint32_t *e = p + w * h;
921
922 while (p < e)
923 { 1063 {
924 uint32_t rgba = *p; 1064 STRLEN datalen;
925 rgba = (rgba >> 24) | (rgba << 8); 1065 char *data = SvPVbyte (data_, datalen);
926#if 0 1066 int bpp = datalen / (ow * oh);
927#ifdef _WIN32 1067 SV *result_ = sv_2mortal (newSV (nw * nh * bpp));
928 {//D
929 uint8_t r = rgba >> 24;
930 uint8_t g = rgba >> 16;
931 uint8_t b = rgba >> 8;
932 uint8_t a = rgba >> 0;
933 1068
934 rgba = (rgba & 0xffffff00) | a; 1069 SvPOK_only (result_);
1070 SvCUR_set (result_, nw * nh * bpp);
1071
1072 memset (SvPVX (result_), 0, nw * nh * bpp);
1073 while (oh--)
1074 memcpy (SvPVX (result_) + oh * nw * bpp, data + oh * ow * bpp, ow * bpp);
1075
1076 sv_setsv (data_, result_);
935 } 1077 }
936#endif 1078
937#endif 1079 sv_setiv (w_, nw);
938 rgba = SDL_SwapBE32 (rgba); 1080 sv_setiv (h_, nh);
939 *p++ = rgba;
940 } 1081 }
941 }
942
943 EXTEND (SP, 5);
944 PUSHs (sv_2mortal (newSViv (w)));
945 PUSHs (sv_2mortal (newSViv (h)));
946 PUSHs (sv_2mortal (retval));
947 PUSHs (sv_2mortal (newSViv (GL_RGBA)));
948 PUSHs (sv_2mortal (newSViv (GL_RGBA)));
949 } 1082 }
950 else
951 {
952 FT_Bitmap bitmap;
953
954 retval = newSV (w * h);
955 SvPOK_only (retval);
956 SvCUR_set (retval, w * h);
957
958 bitmap.rows = h;
959 bitmap.width = w;
960 bitmap.pitch = w;
961 bitmap.buffer = (unsigned char*)SvPVX (retval);
962 bitmap.num_grays = 256;
963 bitmap.pixel_mode = FT_PIXEL_MODE_GRAY;
964
965 memset (bitmap.buffer, 0, w * h);
966
967 pango_ft2_render_layout (&bitmap, self->pl, 0 * PANGO_SCALE, 0 * PANGO_SCALE);
968
969 EXTEND (SP, 5);
970 PUSHs (sv_2mortal (newSViv (w)));
971 PUSHs (sv_2mortal (newSViv (h)));
972 PUSHs (sv_2mortal (retval));
973 PUSHs (sv_2mortal (newSViv (GL_ALPHA)));
974 PUSHs (sv_2mortal (newSViv (GL_ALPHA)));
975 }
976} 1083}
977 1084
978MODULE = CFClient PACKAGE = CFClient::Texture
979
980void 1085void
981draw_quad (SV *self, float x, float y, float w = 0, float h = 0) 1086draw_quad (SV *self, float x, float y, float w = 0., float h = 0.)
982 PROTOTYPE: $$$;$$ 1087 PROTOTYPE: $$$;$$
983 ALIAS: 1088 ALIAS:
984 draw_quad_alpha = 1 1089 draw_quad_alpha = 1
985 draw_quad_alpha_premultiplied = 2 1090 draw_quad_alpha_premultiplied = 2
986 CODE: 1091 CODE:
1024 glDisable (GL_ALPHA_TEST); 1129 glDisable (GL_ALPHA_TEST);
1025 glDisable (GL_BLEND); 1130 glDisable (GL_BLEND);
1026 } 1131 }
1027} 1132}
1028 1133
1029MODULE = CFClient PACKAGE = CFClient::Map 1134MODULE = CFPlus PACKAGE = CFPlus::Map
1030 1135
1031CFClient::Map 1136CFPlus::Map
1032new (SV *class, int map_width, int map_height) 1137new (SV *class, int map_width, int map_height)
1033 CODE: 1138 CODE:
1034 New (0, RETVAL, 1, struct map); 1139 New (0, RETVAL, 1, struct map);
1035 RETVAL->x = 0; 1140 RETVAL->x = 0;
1036 RETVAL->y = 0; 1141 RETVAL->y = 0;
1046 RETVAL->row = 0; 1151 RETVAL->row = 0;
1047 OUTPUT: 1152 OUTPUT:
1048 RETVAL 1153 RETVAL
1049 1154
1050void 1155void
1051DESTROY (CFClient::Map self) 1156DESTROY (CFPlus::Map self)
1052 CODE: 1157 CODE:
1053{ 1158{
1054 map_clear (self); 1159 map_clear (self);
1055 Safefree (self->face); 1160 Safefree (self->face);
1161 Safefree (self->tex);
1056 Safefree (self); 1162 Safefree (self);
1057} 1163}
1058 1164
1059void 1165void
1060clear (CFClient::Map self) 1166clear (CFPlus::Map self)
1061 CODE: 1167 CODE:
1062 map_clear (self); 1168 map_clear (self);
1063 1169
1064void 1170void
1065set_face (CFClient::Map self, int face, int texid) 1171set_face (CFPlus::Map self, int face, int texid)
1066 CODE: 1172 CODE:
1067{ 1173{
1068 while (self->faces <= face) 1174 while (self->faces <= face)
1069 { 1175 {
1070 Append (mapface, self->face, self->faces, self->faces); 1176 Append (mapface, self->face, self->faces, self->faces);
1073 1179
1074 self->face [face] = texid; 1180 self->face [face] = texid;
1075} 1181}
1076 1182
1077void 1183void
1078set_texture (CFClient::Map self, int texid, int name, int w, int h, float s, float t, int r, int g, int b, int a) 1184set_texture (CFPlus::Map self, int texid, int name, int w, int h, float s, float t, int r, int g, int b, int a)
1079 CODE: 1185 CODE:
1080{ 1186{
1081 while (self->texs <= texid) 1187 while (self->texs <= texid)
1082 { 1188 {
1083 Append (maptex, self->tex, self->texs, self->texs); 1189 Append (maptex, self->tex, self->texs, self->texs);
1107 // from transparent color bleeding and ugly wrapping effects. 1213 // from transparent color bleeding and ugly wrapping effects.
1108 glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); 1214 glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
1109} 1215}
1110 1216
1111int 1217int
1112ox (CFClient::Map self) 1218ox (CFPlus::Map self)
1113 ALIAS: 1219 ALIAS:
1114 oy = 1 1220 oy = 1
1115 x = 2 1221 x = 2
1116 y = 3 1222 y = 3
1117 w = 4 1223 w = 4
1128 } 1234 }
1129 OUTPUT: 1235 OUTPUT:
1130 RETVAL 1236 RETVAL
1131 1237
1132void 1238void
1133scroll (CFClient::Map self, int dx, int dy) 1239scroll (CFPlus::Map self, int dx, int dy)
1134 CODE: 1240 CODE:
1135{ 1241{
1136 if (dx > 0) 1242 if (dx > 0)
1137 map_blank (self, self->x, self->y, dx - 1, self->h); 1243 map_blank (self, self->x, self->y, dx - 1, self->h);
1138 else if (dx < 0) 1244 else if (dx < 0)
1154 self->y += MAP_EXTEND_Y; 1260 self->y += MAP_EXTEND_Y;
1155 } 1261 }
1156} 1262}
1157 1263
1158void 1264void
1159map1a_update (CFClient::Map self, SV *data_) 1265map1a_update (CFPlus::Map self, SV *data_, int extmap)
1160 CODE: 1266 CODE:
1161{ 1267{
1162 uint8_t *data = (uint8_t *)SvPVbyte_nolen (data_); 1268 uint8_t *data = (uint8_t *)SvPVbyte_nolen (data_);
1163 uint8_t *data_end = (uint8_t *)SvEND (data_); 1269 uint8_t *data_end = (uint8_t *)SvEND (data_);
1164 mapcell *cell; 1270 mapcell *cell;
1166 1272
1167 while (data < data_end) 1273 while (data < data_end)
1168 { 1274 {
1169 flags = (data [0] << 8) + data [1]; data += 2; 1275 flags = (data [0] << 8) + data [1]; data += 2;
1170 1276
1171 x = ((flags >> 10) & 63) + self->x; 1277 x = self->x + ((flags >> 10) & 63);
1172 y = ((flags >> 4) & 63) + self->y; 1278 y = self->y + ((flags >> 4) & 63);
1173 1279
1174 cell = map_get_cell (self, x, y); 1280 cell = map_get_cell (self, x, y);
1175 1281
1176 if (flags & 15) 1282 if (flags & 15)
1177 { 1283 {
1181 cell->face [0] = 0; 1287 cell->face [0] = 0;
1182 cell->face [1] = 0; 1288 cell->face [1] = 0;
1183 cell->face [2] = 0; 1289 cell->face [2] = 0;
1184 } 1290 }
1185 1291
1292 if (flags & 8)
1293 {
1294 fprintf (stderr, "oi\n");//D
1295 if (extmap)
1296 {
1297 uint8_t ext, cmd;
1298
1299 do
1300 {
1301
1302 ext = *data++;
1303 cmd = ext & 0x7f;
1304
1305 fprintf (stderr, "extcmd = %x\n", ext);//D
1306
1307 if (ext < 4)
1308 cell->darkness = 255 - ext * 64;
1309 }
1310 while (cmd & 0x80);
1311 }
1312 else
1186 cell->darkness = flags & 8 ? *data++ : 255; 1313 cell->darkness = flags & 8 ? *data++ : 255;
1314 }
1187 1315
1188 //TODO: don't trust server data to be in-range(!) 1316 //TODO: don't trust server data to be in-range(!)
1189 1317
1190 if (flags & 4) 1318 if (flags & 4)
1191 { 1319 {
1206 cell->darkness = -1; 1334 cell->darkness = -1;
1207 } 1335 }
1208} 1336}
1209 1337
1210SV * 1338SV *
1211mapmap (CFClient::Map self, int x0, int y0, int w, int h) 1339mapmap (CFPlus::Map self, int x0, int y0, int w, int h)
1212 CODE: 1340 CODE:
1213{ 1341{
1214 int x1, x; 1342 int x1, x;
1215 int y1, y; 1343 int y1, y;
1216 int z; 1344 int z;
1266} 1394}
1267 OUTPUT: 1395 OUTPUT:
1268 RETVAL 1396 RETVAL
1269 1397
1270void 1398void
1271draw (CFClient::Map self, int shift_x, int shift_y, int x0, int y0, int sw, int sh) 1399draw (CFPlus::Map self, int shift_x, int shift_y, int x0, int y0, int sw, int sh)
1272 PPCODE: 1400 CODE:
1273{ 1401{
1274 int vx, vy; 1402 int vx, vy;
1275 int x, y, z; 1403 int x, y, z;
1276 int last_name; 1404 int last_name;
1277 mapface face; 1405 mapface face;
1278 int sw4 = (sw + 3) & ~3;
1279 SV *darkness_sv = sv_2mortal (newSV (sw4 * sh));
1280 uint8_t *darkness = (uint8_t *)SvPVX (darkness_sv);
1281 1406
1282 memset (darkness, 255, sw4 * sh);
1283 SvPOK_only (darkness_sv);
1284 SvCUR_set (darkness_sv, sw4 * sh);
1285
1286 vx = self->x + (self->w - sw) / 2 - shift_x; 1407 vx = self->x + self->w / 2 - sw / 2 - shift_x;
1287 vy = self->y + (self->h - sh) / 2 - shift_y; 1408 vy = self->y + self->h / 2 - sh / 2 - shift_y;
1288 1409
1289 /* 1410 /*
1290 int vx = self->vx = self->w >= sw 1411 int vx = self->vx = self->w >= sw
1291 ? self->x + (self->w - sw) / 2 1412 ? self->x + (self->w - sw) / 2
1292 : MIN (self->x, MAX (self->x + self->w - sw + 1, self->vx)); 1413 : MIN (self->x, MAX (self->x + self->w - sw + 1, self->vx));
1296 : MIN (self->y, MAX (self->y + self->h - sh + 1, self->vy)); 1417 : MIN (self->y, MAX (self->y + self->h - sh + 1, self->vy));
1297 */ 1418 */
1298 1419
1299 glColor4ub (255, 255, 255, 255); 1420 glColor4ub (255, 255, 255, 255);
1300 1421
1422 glEnable (GL_BLEND);
1301 glBlendFunc (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); 1423 glBlendFunc (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
1302 glEnable (GL_BLEND);
1303 glEnable (GL_TEXTURE_2D); 1424 glEnable (GL_TEXTURE_2D);
1304 glTexEnvi (GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE); 1425 glTexEnvi (GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
1305 1426
1306 glBegin (GL_QUADS); 1427 glBegin (GL_QUADS);
1307 1428
1315 1436
1316 for (x = 0; x < sw; x++) 1437 for (x = 0; x < sw; x++)
1317 if (row->c0 <= x + vx && x + vx < row->c1) 1438 if (row->c0 <= x + vx && x + vx < row->c1)
1318 { 1439 {
1319 mapcell *cell = row->col + (x + vx - row->c0); 1440 mapcell *cell = row->col + (x + vx - row->c0);
1320
1321 darkness[y * sw4 + x] = cell->darkness < 0
1322 ? 255 - FOW_DARKNESS
1323 : 255 - cell->darkness;
1324 1441
1325 face = cell->face [z]; 1442 face = cell->face [z];
1326 1443
1327 if (face) 1444 if (face)
1328 { 1445 {
1349 1466
1350 glEnd (); 1467 glEnd ();
1351 1468
1352 glDisable (GL_TEXTURE_2D); 1469 glDisable (GL_TEXTURE_2D);
1353 glDisable (GL_BLEND); 1470 glDisable (GL_BLEND);
1471}
1472
1473void
1474draw_magicmap (CFPlus::Map self, int dx, int dy, int w, int h, unsigned char *data)
1475 CODE:
1476{
1477 static float color[16][3] = {
1478 { 0.00F, 0.00F, 0.00F },
1479 { 1.00F, 1.00F, 1.00F },
1480 { 0.00F, 0.00F, 0.55F },
1481 { 1.00F, 0.00F, 0.00F },
1482
1483 { 1.00F, 0.54F, 0.00F },
1484 { 0.11F, 0.56F, 1.00F },
1485 { 0.93F, 0.46F, 0.00F },
1486 { 0.18F, 0.54F, 0.34F },
1487
1488 { 0.56F, 0.73F, 0.56F },
1489 { 0.80F, 0.80F, 0.80F },
1490 { 0.55F, 0.41F, 0.13F },
1491 { 0.99F, 0.77F, 0.26F },
1492
1493 { 0.74F, 0.65F, 0.41F },
1494
1495 { 0.00F, 1.00F, 1.00F },
1496 { 1.00F, 0.00F, 1.00F },
1497 { 1.00F, 1.00F, 0.00F },
1498 };
1499
1500 int x, y;
1501
1502 glEnable (GL_TEXTURE_2D);
1503 glTexEnvi (GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_REPLACE);
1504 glEnable (GL_BLEND);
1505 glBlendFunc (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
1506 glBegin (GL_QUADS);
1507
1508 for (y = 0; y < h; y++)
1509 for (x = 0; x < w; x++)
1510 {
1511 unsigned char m = data [x + y * w];
1512
1513 if (m)
1514 {
1515 float *c = color [m & 15];
1516
1517 float tx1 = m & 0x40 ? 0.5 : 0.;
1518 float tx2 = tx1 + 0.5;
1519
1520 glColor4f (c[0], c[1], c[2], 0.75);
1521 glTexCoord2f (tx1, 0.); glVertex2i (x , y );
1522 glTexCoord2f (tx1, 1.); glVertex2i (x , y + 1);
1523 glTexCoord2f (tx2, 1.); glVertex2i (x + 1, y + 1);
1524 glTexCoord2f (tx2, 0.); glVertex2i (x + 1, y );
1525 }
1526 }
1527
1528 glEnd ();
1529 glDisable (GL_BLEND);
1530 glDisable (GL_TEXTURE_2D);
1531}
1532
1533void
1534fow_texture (CFPlus::Map self, int shift_x, int shift_y, int x0, int y0, int sw, int sh)
1535 PPCODE:
1536{
1537 int vx, vy;
1538 int x, y;
1539 int sw4 = (sw + 3) & ~3;
1540 SV *darkness_sv = sv_2mortal (newSV (sw4 * sh));
1541 uint8_t *darkness = (uint8_t *)SvPVX (darkness_sv);
1542
1543 memset (darkness, 255, sw4 * sh);
1544 SvPOK_only (darkness_sv);
1545 SvCUR_set (darkness_sv, sw4 * sh);
1546
1547 vx = self->x + (self->w - sw + 1) / 2 - shift_x;
1548 vy = self->y + (self->h - sh + 1) / 2 - shift_y;
1549
1550 for (y = 0; y < sh; y++)
1551 if (0 <= y + vy && y + vy < self->rows)
1552 {
1553 maprow *row = self->row + (y + vy);
1554
1555 for (x = 0; x < sw; x++)
1556 if (row->c0 <= x + vx && x + vx < row->c1)
1557 {
1558 mapcell *cell = row->col + (x + vx - row->c0);
1559
1560 darkness[y * sw4 + x] = cell->darkness < 0
1561 ? 255 - FOW_DARKNESS
1562 : 255 - cell->darkness;
1563 }
1564 }
1354 1565
1355 EXTEND (SP, 3); 1566 EXTEND (SP, 3);
1356 PUSHs (sv_2mortal (newSViv (sw4))); 1567 PUSHs (sv_2mortal (newSViv (sw4)));
1357 PUSHs (sv_2mortal (newSViv (sh))); 1568 PUSHs (sv_2mortal (newSViv (sh)));
1358 PUSHs (darkness_sv); 1569 PUSHs (darkness_sv);
1359} 1570}
1360 1571
1361SV * 1572SV *
1362get_rect (CFClient::Map self, int x0, int y0, int w, int h) 1573get_rect (CFPlus::Map self, int x0, int y0, int w, int h)
1363 CODE: 1574 CODE:
1364{ 1575{
1365 int x, y, x1, y1; 1576 int x, y, x1, y1;
1366 SV *data_sv = newSV (w * h * 7 + 5); 1577 SV *data_sv = newSV (w * h * 7 + 5);
1367 uint8_t *data = (uint8_t *)SvPVX (data_sv); 1578 uint8_t *data = (uint8_t *)SvPVX (data_sv);
1429} 1640}
1430 OUTPUT: 1641 OUTPUT:
1431 RETVAL 1642 RETVAL
1432 1643
1433void 1644void
1434set_rect (CFClient::Map self, int x0, int y0, uint8_t *data) 1645set_rect (CFPlus::Map self, int x0, int y0, uint8_t *data)
1435 PPCODE: 1646 PPCODE:
1436{ 1647{
1437 int x, y, z; 1648 int x, y, z;
1438 int w, h; 1649 int w, h;
1439 int x1, y1; 1650 int x1, y1;
1488 } 1699 }
1489 } 1700 }
1490 } 1701 }
1491} 1702}
1492 1703
1493MODULE = CFClient PACKAGE = CFClient::MixChunk 1704MODULE = CFPlus PACKAGE = CFPlus::MixChunk
1494 1705
1495CFClient::MixChunk 1706CFPlus::MixChunk
1496new_from_file (SV *class, char *path) 1707new_from_file (SV *class, char *path)
1497 CODE: 1708 CODE:
1498 RETVAL = Mix_LoadWAV (path); 1709 RETVAL = Mix_LoadWAV (path);
1499 OUTPUT: 1710 OUTPUT:
1500 RETVAL 1711 RETVAL
1501 1712
1502void 1713void
1503DESTROY (CFClient::MixChunk self) 1714DESTROY (CFPlus::MixChunk self)
1504 CODE: 1715 CODE:
1505 Mix_FreeChunk (self); 1716 Mix_FreeChunk (self);
1506 1717
1507int 1718int
1508volume (CFClient::MixChunk self, int volume = -1) 1719volume (CFPlus::MixChunk self, int volume = -1)
1509 CODE: 1720 CODE:
1510 RETVAL = Mix_VolumeChunk (self, volume); 1721 RETVAL = Mix_VolumeChunk (self, volume);
1511 OUTPUT: 1722 OUTPUT:
1512 RETVAL 1723 RETVAL
1513 1724
1514int 1725int
1515play (CFClient::MixChunk self, int channel = -1, int loops = 0, int ticks = -1) 1726play (CFPlus::MixChunk self, int channel = -1, int loops = 0, int ticks = -1)
1516 CODE: 1727 CODE:
1517 RETVAL = Mix_PlayChannelTimed (channel, self, loops, ticks); 1728 RETVAL = Mix_PlayChannelTimed (channel, self, loops, ticks);
1518 OUTPUT: 1729 OUTPUT:
1519 RETVAL 1730 RETVAL
1520 1731
1521MODULE = CFClient PACKAGE = CFClient::MixMusic 1732MODULE = CFPlus PACKAGE = CFPlus::MixMusic
1522 1733
1523int 1734int
1524volume (int volume = -1) 1735volume (int volume = -1)
1525 CODE: 1736 CODE:
1526 RETVAL = Mix_VolumeMusic (volume); 1737 RETVAL = Mix_VolumeMusic (volume);
1527 OUTPUT: 1738 OUTPUT:
1528 RETVAL 1739 RETVAL
1529 1740
1530CFClient::MixMusic 1741CFPlus::MixMusic
1531new_from_file (SV *class, char *path) 1742new_from_file (SV *class, char *path)
1532 CODE: 1743 CODE:
1533 RETVAL = Mix_LoadMUS (path); 1744 RETVAL = Mix_LoadMUS (path);
1534 OUTPUT: 1745 OUTPUT:
1535 RETVAL 1746 RETVAL
1536 1747
1537void 1748void
1538DESTROY (CFClient::MixMusic self) 1749DESTROY (CFPlus::MixMusic self)
1539 CODE: 1750 CODE:
1540 Mix_FreeMusic (self); 1751 Mix_FreeMusic (self);
1541 1752
1542int 1753int
1543play (CFClient::MixMusic self, int loops = -1) 1754play (CFPlus::MixMusic self, int loops = -1)
1544 CODE: 1755 CODE:
1545 RETVAL = Mix_PlayMusic (self, loops); 1756 RETVAL = Mix_PlayMusic (self, loops);
1546 OUTPUT: 1757 OUTPUT:
1547 RETVAL 1758 RETVAL
1548 1759
1549MODULE = CFClient PACKAGE = CFClient::OpenGL 1760MODULE = CFPlus PACKAGE = CFPlus::OpenGL
1550 1761
1551BOOT: 1762BOOT:
1552{ 1763{
1553 HV *stash = gv_stashpv ("CFClient::OpenGL", 1); 1764 HV *stash = gv_stashpv ("CFPlus::OpenGL", 1);
1554 static const struct { 1765 static const struct {
1555 const char *name; 1766 const char *name;
1556 IV iv; 1767 IV iv;
1557 } *civ, const_iv[] = { 1768 } *civ, const_iv[] = {
1558# define const_iv(name) { # name, (IV)name } 1769# define const_iv(name) { # name, (IV)name }
1565 const_iv (GL_SCISSOR_TEST), 1776 const_iv (GL_SCISSOR_TEST),
1566 const_iv (GL_DEPTH_TEST), 1777 const_iv (GL_DEPTH_TEST),
1567 const_iv (GL_ALPHA_TEST), 1778 const_iv (GL_ALPHA_TEST),
1568 const_iv (GL_NORMALIZE), 1779 const_iv (GL_NORMALIZE),
1569 const_iv (GL_RESCALE_NORMAL), 1780 const_iv (GL_RESCALE_NORMAL),
1781 const_iv (GL_FRONT),
1782 const_iv (GL_BACK),
1570 const_iv (GL_AND), 1783 const_iv (GL_AND),
1571 const_iv (GL_ONE), 1784 const_iv (GL_ONE),
1572 const_iv (GL_ZERO), 1785 const_iv (GL_ZERO),
1573 const_iv (GL_SRC_ALPHA), 1786 const_iv (GL_SRC_ALPHA),
1574 const_iv (GL_DST_ALPHA), 1787 const_iv (GL_DST_ALPHA),
1575 const_iv (GL_ONE_MINUS_SRC_ALPHA), 1788 const_iv (GL_ONE_MINUS_SRC_ALPHA),
1576 const_iv (GL_ONE_MINUS_DST_ALPHA), 1789 const_iv (GL_ONE_MINUS_DST_ALPHA),
1577 const_iv (GL_SRC_ALPHA_SATURATE), 1790 const_iv (GL_SRC_ALPHA_SATURATE),
1578 const_iv (GL_RGB), 1791 const_iv (GL_RGB),
1579 const_iv (GL_RGBA), 1792 const_iv (GL_RGBA),
1793 const_iv (GL_RGBA4),
1794 const_iv (GL_RGBA8),
1795 const_iv (GL_RGB5_A1),
1580 const_iv (GL_UNSIGNED_BYTE), 1796 const_iv (GL_UNSIGNED_BYTE),
1581 const_iv (GL_UNSIGNED_SHORT), 1797 const_iv (GL_UNSIGNED_SHORT),
1582 const_iv (GL_UNSIGNED_INT), 1798 const_iv (GL_UNSIGNED_INT),
1583 const_iv (GL_ALPHA), 1799 const_iv (GL_ALPHA),
1584 const_iv (GL_INTENSITY), 1800 const_iv (GL_INTENSITY),
1616 const_iv (GL_SEPARABLE_2D), 1832 const_iv (GL_SEPARABLE_2D),
1617 const_iv (GL_CONVOLUTION_2D), 1833 const_iv (GL_CONVOLUTION_2D),
1618 const_iv (GL_CONVOLUTION_BORDER_MODE), 1834 const_iv (GL_CONVOLUTION_BORDER_MODE),
1619 const_iv (GL_CONSTANT_BORDER), 1835 const_iv (GL_CONSTANT_BORDER),
1620 const_iv (GL_LINES), 1836 const_iv (GL_LINES),
1837 const_iv (GL_LINE_STRIP),
1621 const_iv (GL_LINE_LOOP), 1838 const_iv (GL_LINE_LOOP),
1622 const_iv (GL_QUADS), 1839 const_iv (GL_QUADS),
1623 const_iv (GL_QUAD_STRIP), 1840 const_iv (GL_QUAD_STRIP),
1624 const_iv (GL_TRIANGLES), 1841 const_iv (GL_TRIANGLES),
1625 const_iv (GL_TRIANGLE_STRIP), 1842 const_iv (GL_TRIANGLE_STRIP),
1658 OUTPUT: 1875 OUTPUT:
1659 RETVAL 1876 RETVAL
1660 1877
1661int glGetError () 1878int glGetError ()
1662 1879
1880void glFinish ()
1881
1663void glClear (int mask) 1882void glClear (int mask)
1664 1883
1665void glClearColor (float r, float g, float b, float a = 1.0) 1884void glClearColor (float r, float g, float b, float a = 1.0)
1666 PROTOTYPE: @ 1885 PROTOTYPE: @
1667 1886
1690void glPushMatrix () 1909void glPushMatrix ()
1691 1910
1692void glPopMatrix () 1911void glPopMatrix ()
1693 1912
1694void glLoadIdentity () 1913void glLoadIdentity ()
1914
1915void glDrawBuffer (int buffer)
1916
1917void glReadBuffer (int buffer)
1695 1918
1696# near_ and far_ are due to microsofts buggy "c" compiler 1919# near_ and far_ are due to microsofts buggy "c" compiler
1697void glFrustum (double left, double right, double bottom, double top, double near_, double far_) 1920void glFrustum (double left, double right, double bottom, double top, double near_, double far_)
1698 1921
1699# near_ and far_ are due to microsofts buggy "c" compiler 1922# near_ and far_ are due to microsofts buggy "c" compiler
1729 r *= a; 1952 r *= a;
1730 g *= a; 1953 g *= a;
1731 b *= a; 1954 b *= a;
1732 } 1955 }
1733 // microsoft visual "c" rounds instead of truncating... 1956 // microsoft visual "c" rounds instead of truncating...
1734 glColor4ub (MIN ((int)(r * 256.f), 255), 1957 glColor4f (r, g, b, a);
1735 MIN ((int)(g * 256.f), 255),
1736 MIN ((int)(b * 256.f), 255),
1737 MIN ((int)(a * 256.f), 255));
1738 1958
1739void glInterleavedArrays (int format, int stride, char *data) 1959void glInterleavedArrays (int format, int stride, char *data)
1740 1960
1741void glDrawElements (int mode, int count, int type, char *indices) 1961void glDrawElements (int mode, int count, int type, char *indices)
1742 1962
1819 2039
1820void glEndList () 2040void glEndList ()
1821 2041
1822void glCallList (int list) 2042void glCallList (int list)
1823 2043
1824

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines