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

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines