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

Comparing deliantra/Deliantra-Client/DC.pm (file contents):
Revision 1.28 by root, Wed Apr 12 20:06:36 2006 UTC vs.
Revision 1.59 by elmex, Tue May 23 17:30:18 2006 UTC

19 19
20 use XSLoader; 20 use XSLoader;
21 XSLoader::load "CFClient", $VERSION; 21 XSLoader::load "CFClient", $VERSION;
22} 22}
23 23
24use SDL::OpenGL; 24use Carp ();
25use AnyEvent ();
26use BerkeleyDB;
27
28use CFClient::OpenGL;
25 29
26our %GL_EXT; 30our %GL_EXT;
27our $GL_VERSION; 31our $GL_VERSION;
28 32
29our $GL_NPOT; 33our $GL_NPOT;
34our $GL_DEBUG = 1;
30 35
31sub gl_init { 36sub gl_init {
32 $GL_VERSION = gl_version * 1; 37 $GL_VERSION = gl_version * 1;
33 %GL_EXT = map +($_ => 1), split /\s+/, gl_extensions; 38 %GL_EXT = map +($_ => 1), split /\s+/, gl_extensions;
34 39
35 $GL_NPOT = $GL_EXT{GL_ARB_texture_non_power_of_two} || $GL_VERSION >= 2; 40 $GL_NPOT = $GL_EXT{GL_ARB_texture_non_power_of_two} || $GL_VERSION >= 2;
41 $GL_NPOT = 0 if gl_vendor =~ /ATI Technologies/; # ATI doesn't get it right...
36 42
37 glClearColor 0.45, 0.45, 0.45, 1;
38
39 glEnable GL_TEXTURE_2D;
40 glEnable GL_COLOR_MATERIAL; 43 glDisable GL_COLOR_MATERIAL;
41 glShadeModel GL_FLAT; 44 glShadeModel GL_FLAT;
45 glDisable GL_DITHER;
42 glDisable GL_DEPTH_TEST; 46 glDisable GL_DEPTH_TEST;
43 glBlendFunc GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA; 47 glDepthMask 0;
44
45 glHint GL_PERSPECTIVE_CORRECTION_HINT, GL_FASTEST; 48 glHint GL_PERSPECTIVE_CORRECTION_HINT, GL_FASTEST;
46 49
47 CFClient::Texture::restore_state (); 50 CFClient::Texture::restore_state ();
51}
52
53sub gl_check {
54 return unless $GL_DEBUG;
55
56 if (my $error = glGetError) {
57 Carp::cluck sprintf "opengl error %x while %s", $error, &sprintf(@_);
58 }
48} 59}
49 60
50sub find_rcfile($) { 61sub find_rcfile($) {
51 my $path; 62 my $path;
52 63
53 for (@INC) { 64 for (grep !ref, @INC) {
54 $path = "$_/CFClient/resources/$_[0]"; 65 $path = "$_/CFClient/resources/$_[0]";
55 return $path if -r $path; 66 return $path if -r $path;
56 } 67 }
57 68
58 die "FATAL: can't find required file $_[0]\n"; 69 die "FATAL: can't find required file $_[0]\n";
88 } 99 }
89 100
90 close CFG; 101 close CFG;
91} 102}
92 103
104mkdir "$Crossfire::VARDIR/pclient", 0777;
105
106our $DB_ENV = new BerkeleyDB::Env
107 -Home => "$Crossfire::VARDIR/pclient",
108 -Cachesize => 1_000_000,
109 -ErrFile => "$Crossfire::VARDIR/pclient/errorlog.txt",
110# -ErrPrefix => "DATABASE",
111 -Verbose => 1,
112 -Flags => DB_CREATE | DB_RECOVER | DB_INIT_MPOOL | DB_INIT_LOCK | DB_INIT_TXN,
113 or die "unable to create/open database home $Crossfire::VARDIR/pclient: $BerkeleyDB::Error";
114
115sub db_table($) {
116 my ($table) = @_;
117
118 $table =~ s/([^a-zA-Z0-9_\-])/sprintf "=%x=", ord $1/ge;
119
120 new CFClient::Database
121 -Env => $DB_ENV,
122 -Filename => $table,
123# -Filename => "database",
124# -Subname => $table,
125 -Property => DB_CHKSUM,
126 -Flags => DB_CREATE | DB_UPGRADE,
127 or die "unable to create/open database table $_[0]: $BerkeleyDB::Error";
128}
129
130sub pod_to_pango($) {
131 my ($pom) = @_;
132
133 $pom->present ("CFClient::PodToPango")
134}
135
136package CFClient::PodToPango;
137
138use base Pod::POM::View::Text;
139
140our $indent = 0;
141
142*view_seq_code =
143*view_seq_bold = sub { "<b>$_[1]</b>" };
144*view_seq_italic = sub { "<i>$_[1]</i>" };
145*view_seq_space =
146*view_seq_link =
147*view_seq_index = sub { CFClient::UI::Label::escape ($_[1]) };
148
149sub view_seq_text {
150 my $text = $_[1];
151 $text =~ s/\s+/ /g;
152 CFClient::UI::Label::escape ($text)
153}
154
155sub view_item {
156 ("\t" x ($indent / 4))
157 . $_[1]->title->present ($_[0])
158 . "\n"
159 . $_[1]->content->present ($_[0])
160}
161
162sub view_verbatim {
163 (join "",
164 map +("\t" x ($indent / 2)) . "$_\n",
165 split /\n/, CFClient::UI::Label::escape ($_[1]))
166 . "\n"
167}
168
169sub view_textblock {
170 ("\t" x ($indent / 2)) . "$_[1]\n\n"
171}
172
173sub view_head2 {
174 "<big>" . $_[1]->title->present ($_[0]) . "</big>\n\n"
175 . $_[1]->content->present ($_[0])
176};
177
178sub view_over {
179 local $indent = $indent + $_[1]->indent;
180 $_[1]->content->present ($_[0])
181}
182
183package CFClient::Database;
184
185our @ISA = BerkeleyDB::Btree::;
186
187sub get($$) {
188 my $data;
189
190 $_[0]->db_get ($_[1], $data) == 0
191 ? $data
192 : ()
193}
194
195my %DB_SYNC;
196
197sub put($$$) {
198 my ($db, $key, $data) = @_;
199
200 $DB_SYNC{$db} = AnyEvent->timer (after => 5, cb => sub { $db->db_sync });
201
202 $db->db_put ($key => $data)
203}
204
93package CFClient::Texture; 205package CFClient::Texture;
94 206
95use strict; 207use strict;
96 208
97use Scalar::Util; 209use Scalar::Util;
98 210
99use SDL::OpenGL; 211use CFClient::OpenGL;
100 212
101my @textures; 213my %TEXTURES;
102 214
103sub new { 215sub new {
104 my ($class, %data) = @_; 216 my ($class, %data) = @_;
105 217
106 my $self = bless { 218 my $self = bless {
107 internalformat => GL_RGBA, 219 internalformat => GL_RGBA,
108 format => GL_RGBA, 220 format => GL_RGBA,
221 type => GL_UNSIGNED_BYTE,
109 %data, 222 %data,
110 }, $class; 223 }, $class;
111 224
112 push @textures, $self; 225 Scalar::Util::weaken ($TEXTURES{$self+0} = $self);
113 Scalar::Util::weaken $textures[-1];
114 226
115 $self->upload; 227 $self->upload;
116 228
117 $self 229 $self
118} 230}
119 231
120sub new_from_image { 232sub new_from_image {
121 my ($class, $image) = @_; 233 my ($class, $image, %arg) = @_;
122 234
123 $class->new (image => $image) 235 $class->new (image => $image, %arg)
124} 236}
125 237
126sub new_from_file { 238sub new_from_file {
127 my ($class, $path) = @_; 239 my ($class, $path, %arg) = @_;
128 240
129 open my $fh, "<:raw", $path 241 open my $fh, "<:raw", $path
130 or die "$path: $!"; 242 or die "$path: $!";
131 243
132 local $/; 244 local $/;
133 $class->new_from_image (<$fh>) 245 $class->new_from_image (<$fh>, %arg)
134} 246}
135 247
136#sub new_from_surface { 248#sub new_from_surface {
137# my ($class, $surface) = @_; 249# my ($class, $surface) = @_;
138# 250#
144# h => $surface->height, 256# h => $surface->height,
145# ) 257# )
146#} 258#}
147 259
148sub new_from_layout { 260sub new_from_layout {
149 my ($class, $layout) = @_; 261 my ($class, $layout, %arg) = @_;
150 262
151 my ($w, $h, $data) = $layout->render; 263 my ($w, $h, $data, $format, $internalformat) = $layout->render;
152 264
153 $class->new ( 265 $class->new (
154 w => $w, 266 w => $w,
155 h => $h, 267 h => $h,
156 data => $data, 268 data => $data,
157 internalformat => GL_ALPHA4,
158 format => GL_ALPHA, 269 format => $format,
270 internalformat => $format,
271 type => GL_UNSIGNED_BYTE,
272 %arg,
159 ) 273 )
160} 274}
161 275
162sub new_from_opengl { 276sub new_from_opengl {
163 my ($class, $w, $h, $cb) = @_; 277 my ($class, $w, $h, $cb) = @_;
164 278
165 $class->new (w => $w, h => $h, render_cb => $cb) 279 $class->new (w => $w || 1, h => $h || 1, render_cb => $cb)
166} 280}
167 281
168sub topot { 282sub topot {
169 (grep $_ >= $_[0], 1, 2, 4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096, 8192, 16384, 32768)[0] 283 (grep $_ >= $_[0], 1, 2, 4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096, 8192, 16384, 32768)[0]
170} 284}
179 if (exists $self->{data}) { 293 if (exists $self->{data}) {
180 $data = $self->{data}; 294 $data = $self->{data};
181 295
182 } elsif (exists $self->{render_cb}) { 296 } elsif (exists $self->{render_cb}) {
183 glViewport 0, 0, $self->{w}, $self->{h}; 297 glViewport 0, 0, $self->{w}, $self->{h};
184 glOrtho 0, $self->{w}, 0, $self->{h}, -10000, 10000;
185 glMatrixMode GL_PROJECTION; 298 glMatrixMode GL_PROJECTION;
186 glLoadIdentity; 299 glLoadIdentity;
300 glOrtho 0, $self->{w}, 0, $self->{h}, -10000, 10000;
187 glMatrixMode GL_MODELVIEW; 301 glMatrixMode GL_MODELVIEW;
188 glLoadIdentity; 302 glLoadIdentity;
189 glClear GL_COLOR_BUFFER_BIT;
190 $self->{render_cb}->($self, $self->{w}, $self->{h}); 303 $self->{render_cb}->($self, $self->{w}, $self->{h});
191 304
192 } else { 305 } else {
193 use Gtk2;#d# TODO kill 306 ($self->{w}, $self->{h}, $data, $self->{internalformat}, $self->{format}, $self->{type})
194 my $pb = new Gtk2::Gdk::PixbufLoader; 307 = CFClient::load_image_inline $self->{image};
195 $pb->write ($self->{image});
196 $pb->close;
197
198 $pb = $pb->get_pixbuf;
199 $pb = $pb->add_alpha (0, 0, 0, 0);
200
201 $self->{w} = $pb->get_width;
202 $self->{h} = $pb->get_height;
203
204 $data = $pb->get_pixels;
205 } 308 }
206 309
207 my ($tw, $th) = @$self{qw(w h)}; 310 my ($tw, $th) = @$self{qw(w h)};
208 311
209 unless ($tw && $th) { 312 unless ($tw > 0 && $th > 0) {
210 $tw = $th = 1; 313 $tw = $th = 1;
211 $data = "\x00" x 64; 314 $data = "\x00" x 64;
212 } 315 }
316
317 $self->{minified} = [CFClient::average $tw, $th, $data]
318 if $self->{minify};
213 319
214 unless ($GL_NPOT) { 320 unless ($GL_NPOT) {
215 # TODO: does not work for zero-sized textures 321 # TODO: does not work for zero-sized textures
216 $tw = topot $tw; 322 $tw = topot $tw;
217 $th = topot $th; 323 $th = topot $th;
218 324
219 if ($tw != $self->{w} || $th != $self->{h} && defined $data) { 325 if (($tw != $self->{w} || $th != $self->{h}) && defined $data) {
220 my $bpp = (length $data) / ($self->{w} * $self->{h}); 326 my $bpp = (length $data) / ($self->{w} * $self->{h});
221 $data = pack "(a" . ($tw * $bpp) . ")*", 327 $data = pack "(a" . ($tw * $bpp) . ")*",
222 unpack "(a" . ($self->{w} * $bpp) . ")*", $data; 328 unpack "(a" . ($self->{w} * $bpp) . ")*", $data;
223 $data .= ("\x00" x ($tw * $bpp)) x ($th - $self->{h}); 329 $data .= ("\x00" x ($tw * $bpp)) x ($th - $self->{h});
224 } 330 }
225 } 331 }
226 332
227 $self->{s} = $self->{w} / $tw; 333 $self->{s} = $self->{w} / $tw;
228 $self->{t} = $self->{h} / $th; 334 $self->{t} = $self->{h} / $th;
229 335
230 $self->{name} ||= (glGenTextures 1)->[0]; 336 $self->{name} ||= glGenTexture;
231 337
232 glBindTexture GL_TEXTURE_2D, $self->{name}; 338 glBindTexture GL_TEXTURE_2D, $self->{name};
233 339
234 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, $::FAST ? GL_NEAREST : GL_LINEAR;
235 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, $::FAST ? GL_NEAREST : GL_LINEAR;
236 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP; 340 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP;
237 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP; 341 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP;
342
343 if ($::FAST) {
344 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST;
345 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST;
346 } elsif ($self->{mipmap} && $GL_VERSION >= 1.4) {
347 # alternatively check for 0x8191
348 glTexParameter GL_TEXTURE_2D, GL_GENERATE_MIPMAP, 1;
349 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR;
350 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR;
351 } else {
352 glTexParameter GL_TEXTURE_2D, GL_GENERATE_MIPMAP, $self->{mipmap};
353 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR;
354 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR;
355 }
238 356
357 glGetError;
358
239 if (defined $data) { 359 if (defined $data) {
240 glTexImage2D GL_TEXTURE_2D, 0, 360 glTexImage2D GL_TEXTURE_2D, 0,
241 $self->{internalformat}, 361 $self->{internalformat},
242 $tw, $th, # need to pad texture first 362 $tw, $th,
243 0, 363 0,
244 $self->{format}, 364 $self->{format},
245 GL_UNSIGNED_BYTE, 365 $self->{type},
246 $data; 366 $data;
247 glGetError and die; 367 CFClient::gl_check "uploading texture %dx%d if=%x f=%x t=%x",
368 $tw, $th, $self->{internalformat}, $self->{format}, $self->{type};
248 } else { 369 } else {
249 glCopyTexImage2D GL_TEXTURE_2D, 0, 370 glCopyTexImage2D GL_TEXTURE_2D, 0,
250 $self->{internalformat}, 371 $self->{internalformat},
251 0, 0, 372 0, 0,
252 $tw, $th, 373 $tw, $th,
253 0; 374 0;
254 glGetError and die; 375 CFClient::gl_check "copying to texture %dx%d if=%x",
376 $tw, $th, $self->{internalformat};
255 } 377 }
378
379 glBindTexture GL_TEXTURE_2D, 0; # just to be on the safe side
256} 380}
257 381
258sub DESTROY { 382sub DESTROY {
259 my ($self) = @_; 383 my ($self) = @_;
260 384
261 return unless exists $self->{name}; 385 delete $TEXTURES{$self+0};
262 386
263 glDeleteTextures delete $self->{name}; 387 glDeleteTexture delete $self->{name}
388 if $self->{name};
264} 389}
265 390
266sub restore_state{ 391sub restore_state {
267 $_->upload 392 $_->upload
268 for grep $_, @textures; 393 for values %TEXTURES;
269}; 394}
270 395
2711; 3961;
272 397
273=back 398=back
274 399

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines