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.66 by root, Thu May 25 21:18:47 2006 UTC vs.
Revision 1.76 by root, Mon May 29 21:20:15 2006 UTC

24use utf8; 24use utf8;
25 25
26use Carp (); 26use Carp ();
27use AnyEvent (); 27use AnyEvent ();
28use BerkeleyDB; 28use BerkeleyDB;
29
30use CFClient::OpenGL;
31
32our %GL_EXT;
33our $GL_VERSION;
34
35our $GL_NPOT;
36our $GL_DEBUG = 1;
37
38sub gl_init {
39 $GL_VERSION = gl_version * 1;
40 %GL_EXT = map +($_ => 1), split /\s+/, gl_extensions;
41
42 $GL_NPOT = $GL_EXT{GL_ARB_texture_non_power_of_two} || $GL_VERSION >= 2;
43 $GL_NPOT = 0 if gl_vendor =~ /ATI Technologies/; # ATI doesn't get it right...
44
45 glDisable GL_COLOR_MATERIAL;
46 glShadeModel GL_FLAT;
47 glDisable GL_DITHER;
48 glDisable GL_DEPTH_TEST;
49 glDepthMask 0;
50 glHint GL_PERSPECTIVE_CORRECTION_HINT, GL_FASTEST;
51
52 CFClient::Texture::restore_state ();
53}
54
55sub gl_check {
56 return unless $GL_DEBUG;
57
58 if (my $error = glGetError) {
59 my ($format, @args) = @_;
60 Carp::cluck sprintf "opengl error %x while $format", $error, @args;
61 }
62}
63 29
64sub find_rcfile($) { 30sub find_rcfile($) {
65 my $path; 31 my $path;
66 32
67 for (grep !ref, @INC) { 33 for (grep !ref, @INC) {
102 } 68 }
103 69
104 close CFG; 70 close CFG;
105} 71}
106 72
107mkdir "$Crossfire::VARDIR/pclient", 0777; 73mkdir "$Crossfire::VARDIR/cfplus", 0777;
108 74
75{
76 use strict;
77
109our $DB_ENV = new BerkeleyDB::Env 78 our $DB_ENV = new BerkeleyDB::Env
110 -Home => "$Crossfire::VARDIR/pclient", 79 -Home => "$Crossfire::VARDIR/cfplus",
111 -Cachesize => 1_000_000, 80 -Cachesize => 1_000_000,
112 -ErrFile => "$Crossfire::VARDIR/pclient/errorlog.txt", 81 -ErrFile => "$Crossfire::VARDIR/cfplus/errorlog.txt",
113# -ErrPrefix => "DATABASE", 82# -ErrPrefix => "DATABASE",
114 -Verbose => 1, 83 -Verbose => 1,
115 -Flags => DB_CREATE | DB_RECOVER | DB_INIT_MPOOL | DB_INIT_LOCK | DB_INIT_TXN, 84 -Flags => DB_CREATE | DB_RECOVER | DB_INIT_MPOOL | DB_INIT_LOCK | DB_INIT_TXN,
85 -SetFlags => DB_AUTO_COMMIT | DB_LOG_AUTOREMOVE | DB_TXN_WRITE_NOSYNC,
116 or die "unable to create/open database home $Crossfire::VARDIR/pclient: $BerkeleyDB::Error"; 86 or die "unable to create/open database home $Crossfire::VARDIR/cfplus: $BerkeleyDB::Error";
87}
117 88
118sub db_table($) { 89sub db_table($) {
119 my ($table) = @_; 90 my ($table) = @_;
120 91
121 $table =~ s/([^a-zA-Z0-9_\-])/sprintf "=%x=", ord $1/ge; 92 $table =~ s/([^a-zA-Z0-9_\-])/sprintf "=%x=", ord $1/ge;
122 93
123 new CFClient::Database 94 new CFClient::Database
124 -Env => $DB_ENV, 95 -Env => $DB_ENV,
125 -Filename => $table, 96 -Filename => $table,
126# -Filename => "database", 97# -Filename => "database",
127# -Subname => $table, 98# -Subname => $table,
128 -Property => DB_CHKSUM, 99 -Property => DB_CHKSUM,
129 -Flags => DB_CREATE | DB_UPGRADE, 100 -Flags => DB_CREATE | DB_UPGRADE,
130 or die "unable to create/open database table $_[0]: $BerkeleyDB::Error"; 101 or die "unable to create/open database table $_[0]: $BerkeleyDB::Error"
131} 102}
132 103
133sub pod_to_pango($) { 104sub pod_to_pango($) {
134 my ($pom) = @_; 105 my ($pom) = @_;
135 106
171 . $_[1]->content->present ($_[0]) 142 . $_[1]->content->present ($_[0])
172} 143}
173 144
174sub view_verbatim { 145sub view_verbatim {
175 (join "", 146 (join "",
176 map +("\t" x ($indent / 2)) . "$_\n", 147 map +("\t" x ($indent / 2)) . "<tt>$_</tt>\n",
177 split /\n/, CFClient::UI::Label::escape ($_[1])) 148 split /\n/, CFClient::UI::Label::escape ($_[1]))
178 . "\n" 149 . "\n"
179} 150}
180 151
181sub view_textblock { 152sub view_textblock {
182 ("\t" x ($indent / 2)) . "$_[1]\n\n" 153 ("\t" x ($indent / 2)) . "$_[1]\n\n"
183} 154}
184 155
185sub view_head1 { 156sub view_head1 {
186 "<span foreground='#ffff00' size='x-large'>" . $_[1]->title->present ($_[0]) . "</span>\n\n" 157 "\n\n<span foreground='#ffff00' size='x-large'>" . $_[1]->title->present ($_[0]) . "</span>\n\n"
187 . $_[1]->content->present ($_[0]) 158 . $_[1]->content->present ($_[0])
188}; 159};
189 160
190sub view_head2 { 161sub view_head2 {
191 "<span foreground='#ccccff' size='large'>" . $_[1]->title->present ($_[0]) . "</span>\n\n" 162 "\n<span foreground='#ccccff' size='large'>" . $_[1]->title->present ($_[0]) . "</span>\n\n"
163 . $_[1]->content->present ($_[0])
164};
165
166sub view_head3 {
167 "\n<span size='large'>" . $_[1]->title->present ($_[0]) . "</span>\n\n"
192 . $_[1]->content->present ($_[0]) 168 . $_[1]->content->present ($_[0])
193}; 169};
194 170
195sub view_over { 171sub view_over {
196 local $indent = $indent + $_[1]->indent; 172 local $indent = $indent + $_[1]->indent;
218 194
219 $db->db_put ($key => $data) 195 $db->db_put ($key => $data)
220} 196}
221 197
222package CFClient::Item; 198package CFClient::Item;
199
200use strict;
201use Crossfire::Protocol::Constants;
223 202
224sub desc_string { 203sub desc_string {
225 my ($self) = @_; 204 my ($self) = @_;
226 205
227 my $desc = 206 my $desc =
228 $self->{nrof} < 2 207 $self->{nrof} < 2
229 ? $self->{name} 208 ? $self->{name}
230 : "$self->{nrof} × $self->{name_pl}"; 209 : "$self->{nrof} × $self->{name_pl}";
231 210
232 $self->{flags} & Crossfire::Protocol::F_OPEN 211 $self->{flags} & F_OPEN
233 and $desc .= " (open)"; 212 and $desc .= " (open)";
234 $self->{flags} & Crossfire::Protocol::F_APPLIED 213 $self->{flags} & F_APPLIED
235 and $desc .= " (applied)"; 214 and $desc .= " (applied)";
236 $self->{flags} & Crossfire::Protocol::F_UNPAID 215 $self->{flags} & F_UNPAID
237 and $desc .= " (unpaid)"; 216 and $desc .= " (unpaid)";
238 $self->{flags} & Crossfire::Protocol::F_MAGIC 217 $self->{flags} & F_MAGIC
239 and $desc .= " (magic)"; 218 and $desc .= " (magic)";
240 $self->{flags} & Crossfire::Protocol::F_CURSED 219 $self->{flags} & F_CURSED
241 and $desc .= " (cursed)"; 220 and $desc .= " (cursed)";
242 $self->{flags} & Crossfire::Protocol::F_DAMNED 221 $self->{flags} & F_DAMNED
243 and $desc .= " (damned)"; 222 and $desc .= " (damned)";
244 $self->{flags} & Crossfire::Protocol::F_LOCKED 223 $self->{flags} & F_LOCKED
245 and $desc .= " *"; 224 and $desc .= " *";
246 225
247 $desc 226 $desc
248} 227}
249 228
277 my @menu_items = ( 256 my @menu_items = (
278 ["examine", sub { $::CONN->send ("examine $self->{tag}") }], 257 ["examine", sub { $::CONN->send ("examine $self->{tag}") }],
279 ["mark", sub { $::CONN->send ("mark ". pack "N", $self->{tag}) }], 258 ["mark", sub { $::CONN->send ("mark ". pack "N", $self->{tag}) }],
280 ["apply", sub { $::CONN->send ("apply $self->{tag}") }], 259 ["apply", sub { $::CONN->send ("apply $self->{tag}") }],
281 ( 260 (
282 $self->{flags} & Crossfire::Protocol::F_LOCKED 261 $self->{flags} & F_LOCKED
283 ? ( 262 ? (
284 ["unlock", sub { $::CONN->send ("lock " . pack "CN", 0, $self->{tag}) }], 263 ["unlock", sub { $::CONN->send ("lock " . pack "CN", 0, $self->{tag}) }],
285 ) 264 )
286 : ( 265 : (
287 ["lock", sub { $::CONN->send ("lock " . pack "CN", 1, $self->{tag}) }], 266 ["lock", sub { $::CONN->send ("lock " . pack "CN", 1, $self->{tag}) }],
304 . "</small>\n"; 283 . "</small>\n";
305 284
306 $self->{face_widget} ||= new CFClient::UI::Face 285 $self->{face_widget} ||= new CFClient::UI::Face
307 can_events => 1, 286 can_events => 1,
308 can_hover => 1, 287 can_hover => 1,
288 anim => $self->{anim},
309 animspeed => $self->{animspeed}, # TODO# must be set at creation time 289 animspeed => $self->{animspeed}, # TODO# must be set at creation time
310 connect_button_down => $button_cb, 290 on_button_down => $button_cb,
311 ; 291 ;
312 $self->{face_widget}{face} = $self->{face}; 292 $self->{face_widget}{face} = $self->{face};
313 $self->{face_widget}{anim} = $self->{anim}; 293 $self->{face_widget}{anim} = $self->{anim};
314 $self->{face_widget}{animspeed} = $self->{animspeed}; 294 $self->{face_widget}{animspeed} = $self->{animspeed};
315 $self->{face_widget}->set_tooltip ( 295 $self->{face_widget}->set_tooltip (
321 301
322 $self->{desc_widget} ||= new CFClient::UI::Label 302 $self->{desc_widget} ||= new CFClient::UI::Label
323 can_events => 1, 303 can_events => 1,
324 can_hover => 1, 304 can_hover => 1,
325 ellipsise => 2, 305 ellipsise => 2,
326 xalign => 0, 306 align => -1,
327 connect_button_down => $button_cb, 307 on_button_down => $button_cb,
328 ; 308 ;
329 my $desc = CFClient::Item::desc_string $self; 309 my $desc = CFClient::Item::desc_string $self;
330 $self->{desc_widget}->set_text ($desc); 310 $self->{desc_widget}->set_text ($desc);
331 $self->{desc_widget}->set_tooltip ("<b>$desc</b>.\n$tooltip_std"); 311 $self->{desc_widget}->set_tooltip ("<b>$desc</b>.\n$tooltip_std");
332 312
333 $self->{weight_widget} ||= new CFClient::UI::Label 313 $self->{weight_widget} ||= new CFClient::UI::Label
334 can_events => 1, 314 can_events => 1,
335 can_hover => 1, 315 can_hover => 1,
336 ellipsise => 0, 316 ellipsise => 0,
337 xalign => 0.5, 317 align => 0,
338 connect_button_down => $button_cb, 318 on_button_down => $button_cb,
339 ; 319 ;
340 $self->{weight_widget}->set_text (CFClient::Item::weight_string $self); 320 $self->{weight_widget}->set_text (CFClient::Item::weight_string $self);
341 321
342 $self->{weight_widget}->set_tooltip ( 322 $self->{weight_widget}->set_tooltip (
343 "<b>Weight</b>.\n" 323 "<b>Weight</b>.\n"
345 . ($self->{nrof} ? "You have $self->{nrof} of it. " : "Item cannot stack with others of it's kind. ") 325 . ($self->{nrof} ? "You have $self->{nrof} of it. " : "Item cannot stack with others of it's kind. ")
346 . "\n\n$tooltip_std" 326 . "\n\n$tooltip_std"
347 ); 327 );
348} 328}
349 329
350package CFClient::Texture; 330package CFClient::Recorder;
351 331
352use strict; 332our $RECORD_WINDOW;
353 333
354use Scalar::Util; 334my $CMDBOX;
335my $CURRENT_CMDS;
336my $REC_BTN;
355 337
356use CFClient::OpenGL; 338my @ALLOWED_MODIFIER_KEYS = (
339 (CFClient::SDLK_LSHIFT) => "LSHIFT",
340 (CFClient::SDLK_LCTRL ) => "LCTRL",
341 (CFClient::SDLK_LALT ) => "LALT",
342 (CFClient::SDLK_LMETA ) => "LMETA",
357 343
358my %TEXTURES; 344 (CFClient::SDLK_RSHIFT) => "RSHIFT",
345 (CFClient::SDLK_RCTRL ) => "RCTRL",
346 (CFClient::SDLK_RALT ) => "RALT",
347 (CFClient::SDLK_RMETA ) => "RMETA",
348);
359 349
360sub new { 350my %ALLOWED_MODIFIERS = (
351 (CFClient::KMOD_LSHIFT) => "LSHIFT",
352 (CFClient::KMOD_LCTRL ) => "LCTRL",
353 (CFClient::KMOD_LALT ) => "LALT",
354 (CFClient::KMOD_LMETA ) => "LMETA",
355
356 (CFClient::KMOD_RSHIFT) => "RSHIFT",
357 (CFClient::KMOD_RCTRL ) => "RCTRL",
358 (CFClient::KMOD_RALT ) => "RALT",
359 (CFClient::KMOD_RMETA ) => "RMETA",
360);
361
362my %DIRECT_BIND_CHARS = map { $_ => 1 } qw/0 1 2 3 4 5 6 7 8 9/;
363my @DIRECT_BIND_KEYS = (
364 CFClient::SDLK_F1,
365 CFClient::SDLK_F2,
366 CFClient::SDLK_F3,
367 CFClient::SDLK_F4,
368 CFClient::SDLK_F5,
369 CFClient::SDLK_F6,
370 CFClient::SDLK_F7,
371 CFClient::SDLK_F8,
372 CFClient::SDLK_F9,
373 CFClient::SDLK_F10,
374 CFClient::SDLK_F11,
375 CFClient::SDLK_F12,
376 CFClient::SDLK_F13,
377 CFClient::SDLK_F14,
378 CFClient::SDLK_F15,
379);
380
381# this binding dialog asks for a key-combo to be pressed
382# and if successful it binds the modifier+symbol to the
383# supplied actions in $cmd.
384# (Bindings are stored in $::CFG->{bindings}->{$mod}->{$sym})
385sub open_binding_dialog {
361 my ($class, %data) = @_; 386 my ($cmd) = @_;
362 387
363 my $self = bless { 388 my $w = new CFClient::UI::FancyFrame
364 internalformat => GL_RGBA, 389 title => "Bind Action";
365 format => GL_RGBA,
366 type => GL_UNSIGNED_BYTE,
367 %data,
368 }, $class;
369 390
370 Scalar::Util::weaken ($TEXTURES{$self+0} = $self); 391 $w->add (my $vb = new CFClient::UI::VBox);
371 392 $vb->add (new CFClient::UI::Label
372 $self->upload; 393 text => "Press a modifier (CTRL, ALT and/or SHIFT) and a key."
373 394 ."You can only bind 0-9 and F1-F15 without modifiers."
374 $self
375}
376
377sub new_from_image {
378 my ($class, $image, %arg) = @_;
379
380 $class->new (image => $image, %arg)
381}
382
383sub new_from_file {
384 my ($class, $path, %arg) = @_;
385
386 open my $fh, "<:raw", $path
387 or die "$path: $!";
388
389 local $/;
390 $class->new_from_image (<$fh>, %arg)
391}
392
393#sub new_from_surface {
394# my ($class, $surface) = @_;
395#
396# $surface->rgba;
397#
398# $class->new (
399# data => $surface->pixels,
400# w => $surface->width,
401# h => $surface->height,
402# )
403#}
404
405sub new_from_layout {
406 my ($class, $layout, %arg) = @_;
407
408 my ($w, $h, $data, $format, $internalformat) = $layout->render;
409
410 $class->new (
411 w => $w,
412 h => $h,
413 data => $data,
414 format => $format,
415 internalformat => $format,
416 type => GL_UNSIGNED_BYTE,
417 %arg,
418 ) 395 );
419} 396 $vb->add (my $entry = new CFClient::UI::Entry
397 text => "",
398 on_key_down => sub {
399 my ($entry, $ev) = @_;
420 400
421sub new_from_opengl { 401 my $mod = $ev->{mod};
422 my ($class, $w, $h, $cb) = @_; 402 my $sym = $ev->{sym};
423 403
424 $class->new (w => $w || 1, h => $h || 1, render_cb => $cb) 404 # XXX: This seems a little bit hackisch to me, but i have to ignore them
425} 405 if (grep { $_ == $sym } @ALLOWED_MODIFIER_KEYS) {
406 return;
407 }
426 408
427sub topot { 409 if ($mod == CFClient::KMOD_NONE
428 (grep $_ >= $_[0], 1, 2, 4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096, 8192, 16384, 32768)[0] 410 and not $DIRECT_BIND_CHARS{chr ($ev->{unicode})}
429} 411 and not grep { $sym == $_ } @DIRECT_BIND_KEYS)
412 {
413 $::STATUSBOX->add (
414 "Can't bind key ".CFClient::SDL_GetKeyName ($sym)
415 ." directly without modifier! It would damage the completer handling."
416 );
417 return;
418 }
430 419
431sub upload { 420 $entry->focus_out;
421
422 $::CFG->{bindings}->{$mod}->{$sym} = $cmd;
423 $::STATUSBOX->add ("Bound actions to '".keycombo_to_name ($mod, $sym)."'. Don't forget 'Save Config'!");
424
425 $w->destroy
426 });
427
428 $entry->focus_in;
429 $w->center;
430 $w->show;
431}
432
433sub keycombo_to_name {
434 my ($mod, $sym) = @_;
435
436 my $mods = join '+',
437 map { $ALLOWED_MODIFIERS{$_} }
438 grep { $_ & $mod }
439 keys %ALLOWED_MODIFIERS;
440 $mods .= "+" if $mods ne '';
441
442 return $mods . CFClient::SDL_GetKeyName ($sym);
443}
444
445sub clear_command_list {
446 $CMDBOX->clear () if $CMDBOX;
447}
448
449sub set_command_list {
432 my ($self) = @_; 450 my ($list) = @_;
433 451
434 return unless $GL_VERSION; 452 return unless $CMDBOX;
435 453
436 my $data; 454 $CMDBOX->clear ();
455 $CURRENT_CMDS = $list;
437 456
438 if (exists $self->{data}) { 457 my $idx = 0;
439 $data = $self->{data};
440 458
441 } elsif (exists $self->{render_cb}) { 459 for (@$list) {
442 glViewport 0, 0, $self->{w}, $self->{h}; 460 $CMDBOX->add (my $hb = new CFClient::UI::HBox);
443 glMatrixMode GL_PROJECTION;
444 glLoadIdentity;
445 glOrtho 0, $self->{w}, 0, $self->{h}, -10000, 10000;
446 glMatrixMode GL_MODELVIEW;
447 glLoadIdentity;
448 $self->{render_cb}->($self, $self->{w}, $self->{h});
449 461
450 } else { 462 my $i = $idx;
451 ($self->{w}, $self->{h}, $data, $self->{internalformat}, $self->{format}, $self->{type}) 463 $hb->add (new CFClient::UI::Button
452 = CFClient::load_image_inline $self->{image}; 464 text => "delete",
465 tooltip => "Deletes the action from the record",
466 on_activate => sub {
467 $CMDBOX->remove ($hb);
468 $list->[$i] = undef;
469 });
470
471 $hb->add (new CFClient::UI::Label text => $_);
472
473 $idx++
453 } 474 }
475}
454 476
455 my ($tw, $th) = @$self{qw(w h)}; 477# if $show is 1 the recorder will be shown
478sub start {
479 my ($show) = @_;
456 480
457 unless ($tw > 0 && $th > 0) { 481 $RECORD_WINDOW->show if $show;
458 $tw = $th = 1; 482
459 $data = "\x00" x 64; 483 $REC_BTN->set_text ("stop recording");
484 $REC_BTN->{recording} = 1;
485 clear_command_list;
486 $::CONN->start_record;
487}
488
489# if $autobind is 1 the recorder will be automatically
490# jump into the binding query and hide the recorder window
491sub stop {
492 my ($autobind) = @_;
493
494 $REC_BTN->set_text ("start recording");
495 $REC_BTN->{recording} = 0;
496
497 my $rec = $::CONN->stop_record;
498 return unless ref $rec eq 'ARRAY';
499 set_command_list ($rec);
500
501 if ($autobind) {
502 open_binding_dialog ([ grep { defined $_ } @$CURRENT_CMDS ]);
503 $RECORD_WINDOW->hide;
460 } 504 }
505}
461 506
462 $self->{minified} = [CFClient::average $tw, $th, $data] 507sub make_window {
463 if $self->{minify}; 508 $RECORD_WINDOW = new CFClient::UI::FancyFrame
509 req_y => 1,
510 req_x => -1,
511 title => "Action Recorder";
464 512
465 unless ($GL_NPOT) { 513 $RECORD_WINDOW->add (my $vb = new CFClient::UI::VBox);
466 # TODO: does not work for zero-sized textures 514 $vb->add ($REC_BTN = new CFClient::UI::Button
467 $tw = topot $tw; 515 text => "start recording",
468 $th = topot $th; 516 tooltip => "Start/Stops recording of actions."
517 ."(CTRL+Insert Starts the recorder, Insert Stops recorder and binds automatically)"
518 ."All subsequent actions after the recording started will be captured."
519 ."The actions are displayed after the record was stopped."
520 ."To bind the action you have to click on the 'Bind' button",
521 on_activate => sub {
522 my ($btn) = @_;
469 523
470 if (($tw != $self->{w} || $th != $self->{h}) && defined $data) { 524 unless ($btn->{recording}) {
471 my $bpp = (length $data) / ($self->{w} * $self->{h}); 525 start;
472 $data = pack "(a" . ($tw * $bpp) . ")*", 526 } else {
473 unpack "(a" . ($self->{w} * $bpp) . ")*", $data; 527 stop;
474 $data .= ("\x00" x ($tw * $bpp)) x ($th - $self->{h}); 528 }
475 } 529 });
476 } 530 $vb->add ($CMDBOX = new CFClient::UI::VBox);
531 $vb->add (new CFClient::UI::Button
532 text => "bind",
533 tooltip => "This opens a query where you have to press the key combination to bind the recorded actions",
534 on_activate => sub {
535 open_binding_dialog ([ grep { defined $_ } @$CURRENT_CMDS ]);
536 });
477 537
478 $self->{s} = $self->{w} / $tw; 538 $RECORD_WINDOW
479 $self->{t} = $self->{h} / $th;
480
481 $self->{name} ||= glGenTexture;
482
483 glBindTexture GL_TEXTURE_2D, $self->{name};
484
485 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP;
486 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP;
487
488 if ($::FAST) {
489 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST;
490 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST;
491 } elsif ($self->{mipmap} && $GL_VERSION >= 1.4) {
492 # alternatively check for 0x8191
493 glTexParameter GL_TEXTURE_2D, GL_GENERATE_MIPMAP, 1;
494 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR;
495 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR;
496 } else {
497 glTexParameter GL_TEXTURE_2D, GL_GENERATE_MIPMAP, $self->{mipmap};
498 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR;
499 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR;
500 }
501
502 glGetError;
503
504 if (defined $data) {
505 glTexImage2D GL_TEXTURE_2D, 0,
506 $self->{internalformat},
507 $tw, $th,
508 0,
509 $self->{format},
510 $self->{type},
511 $data;
512 CFClient::gl_check "uploading texture %dx%d if=%x f=%x t=%x",
513 $tw, $th, $self->{internalformat}, $self->{format}, $self->{type};
514 } else {
515 glCopyTexImage2D GL_TEXTURE_2D, 0,
516 $self->{internalformat},
517 0, 0,
518 $tw, $th,
519 0;
520 CFClient::gl_check "copying to texture %dx%d if=%x",
521 $tw, $th, $self->{internalformat};
522 }
523
524 glBindTexture GL_TEXTURE_2D, 0; # just to be on the safe side
525}
526
527sub DESTROY {
528 my ($self) = @_;
529
530 delete $TEXTURES{$self+0};
531
532 glDeleteTexture delete $self->{name}
533 if $self->{name};
534}
535
536sub restore_state {
537 $_->upload
538 for values %TEXTURES;
539} 539}
540 540
5411; 5411;
542 542
543=back 543=back

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines