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.67 by root, Thu May 25 21:36:44 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}) }],
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,
309 anim => $self->{anim}, 283 anim => $self->{anim},
310 animspeed => $self->{animspeed}, # TODO# must be set at creation time 284 animspeed => $self->{animspeed}, # TODO# must be set at creation time
311 connect_button_down => $button_cb, 285 on_button_down => $button_cb,
312 ; 286 ;
313 $self->{face_widget}{face} = $self->{face}; 287 $self->{face_widget}{face} = $self->{face};
314 $self->{face_widget}{anim} = $self->{anim}; 288 $self->{face_widget}{anim} = $self->{anim};
315 $self->{face_widget}{animspeed} = $self->{animspeed}; 289 $self->{face_widget}{animspeed} = $self->{animspeed};
316 $self->{face_widget}->set_tooltip ( 290 $self->{face_widget}->set_tooltip (
322 296
323 $self->{desc_widget} ||= new CFClient::UI::Label 297 $self->{desc_widget} ||= new CFClient::UI::Label
324 can_events => 1, 298 can_events => 1,
325 can_hover => 1, 299 can_hover => 1,
326 ellipsise => 2, 300 ellipsise => 2,
327 xalign => 0, 301 align => -1,
328 connect_button_down => $button_cb, 302 on_button_down => $button_cb,
329 ; 303 ;
330 my $desc = CFClient::Item::desc_string $self; 304 my $desc = CFClient::Item::desc_string $self;
331 $self->{desc_widget}->set_text ($desc); 305 $self->{desc_widget}->set_text ($desc);
332 $self->{desc_widget}->set_tooltip ("<b>$desc</b>.\n$tooltip_std"); 306 $self->{desc_widget}->set_tooltip ("<b>$desc</b>.\n$tooltip_std");
333 307
334 $self->{weight_widget} ||= new CFClient::UI::Label 308 $self->{weight_widget} ||= new CFClient::UI::Label
335 can_events => 1, 309 can_events => 1,
336 can_hover => 1, 310 can_hover => 1,
337 ellipsise => 0, 311 ellipsise => 0,
338 xalign => 0.5, 312 align => 0,
339 connect_button_down => $button_cb, 313 on_button_down => $button_cb,
340 ; 314 ;
341 $self->{weight_widget}->set_text (CFClient::Item::weight_string $self); 315 $self->{weight_widget}->set_text (CFClient::Item::weight_string $self);
342 316
343 $self->{weight_widget}->set_tooltip ( 317 $self->{weight_widget}->set_tooltip (
344 "<b>Weight</b>.\n" 318 "<b>Weight</b>.\n"
346 . ($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. ")
347 . "\n\n$tooltip_std" 321 . "\n\n$tooltip_std"
348 ); 322 );
349} 323}
350 324
351package CFClient::Texture; 325package CFClient::Recorder;
352 326
353use strict; 327our $RECORD_WINDOW;
354 328
355use Scalar::Util; 329my $CMDBOX;
330my $CURRENT_CMDS;
331my $REC_BTN;
356 332
357use 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",
358 338
359my %TEXTURES; 339 (CFClient::SDLK_RSHIFT) => "RSHIFT",
340 (CFClient::SDLK_RCTRL ) => "RCTRL",
341 (CFClient::SDLK_RALT ) => "RALT",
342 (CFClient::SDLK_RMETA ) => "RMETA",
343);
360 344
361sub 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 {
362 my ($class, %data) = @_; 381 my ($cmd) = @_;
363 382
364 my $self = bless { 383 my $w = new CFClient::UI::FancyFrame
365 internalformat => GL_RGBA, 384 title => "Bind Action";
366 format => GL_RGBA,
367 type => GL_UNSIGNED_BYTE,
368 %data,
369 }, $class;
370 385
371 Scalar::Util::weaken ($TEXTURES{$self+0} = $self); 386 $w->add (my $vb = new CFClient::UI::VBox);
372 387 $vb->add (new CFClient::UI::Label
373 $self->upload; 388 text => "Press a modifier (CTRL, ALT and/or SHIFT) and a key."
374 389 ."You can only bind 0-9 and F1-F15 without modifiers."
375 $self
376}
377
378sub new_from_image {
379 my ($class, $image, %arg) = @_;
380
381 $class->new (image => $image, %arg)
382}
383
384sub new_from_file {
385 my ($class, $path, %arg) = @_;
386
387 open my $fh, "<:raw", $path
388 or die "$path: $!";
389
390 local $/;
391 $class->new_from_image (<$fh>, %arg)
392}
393
394#sub new_from_surface {
395# my ($class, $surface) = @_;
396#
397# $surface->rgba;
398#
399# $class->new (
400# data => $surface->pixels,
401# w => $surface->width,
402# h => $surface->height,
403# )
404#}
405
406sub new_from_layout {
407 my ($class, $layout, %arg) = @_;
408
409 my ($w, $h, $data, $format, $internalformat) = $layout->render;
410
411 $class->new (
412 w => $w,
413 h => $h,
414 data => $data,
415 format => $format,
416 internalformat => $format,
417 type => GL_UNSIGNED_BYTE,
418 %arg,
419 ) 390 );
420} 391 $vb->add (my $entry = new CFClient::UI::Entry
392 text => "",
393 on_key_down => sub {
394 my ($entry, $ev) = @_;
421 395
422sub new_from_opengl { 396 my $mod = $ev->{mod};
423 my ($class, $w, $h, $cb) = @_; 397 my $sym = $ev->{sym};
424 398
425 $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
426} 400 if (grep { $_ == $sym } @ALLOWED_MODIFIER_KEYS) {
401 return;
402 }
427 403
428sub topot { 404 if ($mod == CFClient::KMOD_NONE
429 (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})}
430} 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 }
431 414
432sub 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 {
433 my ($self) = @_; 445 my ($list) = @_;
434 446
435 return unless $GL_VERSION; 447 return unless $CMDBOX;
436 448
437 my $data; 449 $CMDBOX->clear ();
450 $CURRENT_CMDS = $list;
438 451
439 if (exists $self->{data}) { 452 my $idx = 0;
440 $data = $self->{data};
441 453
442 } elsif (exists $self->{render_cb}) { 454 for (@$list) {
443 glViewport 0, 0, $self->{w}, $self->{h}; 455 $CMDBOX->add (my $hb = new CFClient::UI::HBox);
444 glMatrixMode GL_PROJECTION;
445 glLoadIdentity;
446 glOrtho 0, $self->{w}, 0, $self->{h}, -10000, 10000;
447 glMatrixMode GL_MODELVIEW;
448 glLoadIdentity;
449 $self->{render_cb}->($self, $self->{w}, $self->{h});
450 456
451 } else { 457 my $i = $idx;
452 ($self->{w}, $self->{h}, $data, $self->{internalformat}, $self->{format}, $self->{type}) 458 $hb->add (new CFClient::UI::Button
453 = 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++
454 } 469 }
470}
455 471
456 my ($tw, $th) = @$self{qw(w h)}; 472# if $show is 1 the recorder will be shown
473sub start {
474 my ($show) = @_;
457 475
458 unless ($tw > 0 && $th > 0) { 476 $RECORD_WINDOW->show if $show;
459 $tw = $th = 1; 477
460 $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;
461 } 499 }
500}
462 501
463 $self->{minified} = [CFClient::average $tw, $th, $data] 502sub make_window {
464 if $self->{minify}; 503 $RECORD_WINDOW = new CFClient::UI::FancyFrame
504 req_y => 1,
505 req_x => -1,
506 title => "Action Recorder";
465 507
466 unless ($GL_NPOT) { 508 $RECORD_WINDOW->add (my $vb = new CFClient::UI::VBox);
467 # TODO: does not work for zero-sized textures 509 $vb->add ($REC_BTN = new CFClient::UI::Button
468 $tw = topot $tw; 510 text => "start recording",
469 $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) = @_;
470 518
471 if (($tw != $self->{w} || $th != $self->{h}) && defined $data) { 519 unless ($btn->{recording}) {
472 my $bpp = (length $data) / ($self->{w} * $self->{h}); 520 start;
473 $data = pack "(a" . ($tw * $bpp) . ")*", 521 } else {
474 unpack "(a" . ($self->{w} * $bpp) . ")*", $data; 522 stop;
475 $data .= ("\x00" x ($tw * $bpp)) x ($th - $self->{h}); 523 }
476 } 524 });
477 } 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 });
478 532
479 $self->{s} = $self->{w} / $tw; 533 $RECORD_WINDOW
480 $self->{t} = $self->{h} / $th;
481
482 $self->{name} ||= glGenTexture;
483
484 glBindTexture GL_TEXTURE_2D, $self->{name};
485
486 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP;
487 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP;
488
489 if ($::FAST) {
490 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST;
491 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST;
492 } elsif ($self->{mipmap} && $GL_VERSION >= 1.4) {
493 # alternatively check for 0x8191
494 glTexParameter GL_TEXTURE_2D, GL_GENERATE_MIPMAP, 1;
495 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR;
496 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR;
497 } else {
498 glTexParameter GL_TEXTURE_2D, GL_GENERATE_MIPMAP, $self->{mipmap};
499 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR;
500 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR;
501 }
502
503 glGetError;
504
505 if (defined $data) {
506 glTexImage2D GL_TEXTURE_2D, 0,
507 $self->{internalformat},
508 $tw, $th,
509 0,
510 $self->{format},
511 $self->{type},
512 $data;
513 CFClient::gl_check "uploading texture %dx%d if=%x f=%x t=%x",
514 $tw, $th, $self->{internalformat}, $self->{format}, $self->{type};
515 } else {
516 glCopyTexImage2D GL_TEXTURE_2D, 0,
517 $self->{internalformat},
518 0, 0,
519 $tw, $th,
520 0;
521 CFClient::gl_check "copying to texture %dx%d if=%x",
522 $tw, $th, $self->{internalformat};
523 }
524
525 glBindTexture GL_TEXTURE_2D, 0; # just to be on the safe side
526}
527
528sub DESTROY {
529 my ($self) = @_;
530
531 delete $TEXTURES{$self+0};
532
533 glDeleteTexture delete $self->{name}
534 if $self->{name};
535}
536
537sub restore_state {
538 $_->upload
539 for values %TEXTURES;
540} 534}
541 535
5421; 5361;
543 537
544=back 538=back

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines