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.68 by root, Thu May 25 21:48:22 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}) }],
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,
309 anim => $self->{anim}, 288 anim => $self->{anim},
310 animspeed => $self->{animspeed}, # TODO# must be set at creation time 289 animspeed => $self->{animspeed}, # TODO# must be set at creation time
311 connect_button_down => $button_cb, 290 on_button_down => $button_cb,
312 ; 291 ;
313 $self->{face_widget}{face} = $self->{face}; 292 $self->{face_widget}{face} = $self->{face};
314 $self->{face_widget}{anim} = $self->{anim}; 293 $self->{face_widget}{anim} = $self->{anim};
315 $self->{face_widget}{animspeed} = $self->{animspeed}; 294 $self->{face_widget}{animspeed} = $self->{animspeed};
316 $self->{face_widget}->set_tooltip ( 295 $self->{face_widget}->set_tooltip (
323 $self->{desc_widget} ||= new CFClient::UI::Label 302 $self->{desc_widget} ||= new CFClient::UI::Label
324 can_events => 1, 303 can_events => 1,
325 can_hover => 1, 304 can_hover => 1,
326 ellipsise => 2, 305 ellipsise => 2,
327 align => -1, 306 align => -1,
328 connect_button_down => $button_cb, 307 on_button_down => $button_cb,
329 ; 308 ;
330 my $desc = CFClient::Item::desc_string $self; 309 my $desc = CFClient::Item::desc_string $self;
331 $self->{desc_widget}->set_text ($desc); 310 $self->{desc_widget}->set_text ($desc);
332 $self->{desc_widget}->set_tooltip ("<b>$desc</b>.\n$tooltip_std"); 311 $self->{desc_widget}->set_tooltip ("<b>$desc</b>.\n$tooltip_std");
333 312
334 $self->{weight_widget} ||= new CFClient::UI::Label 313 $self->{weight_widget} ||= new CFClient::UI::Label
335 can_events => 1, 314 can_events => 1,
336 can_hover => 1, 315 can_hover => 1,
337 ellipsise => 0, 316 ellipsise => 0,
338 align => 0, 317 align => 0,
339 connect_button_down => $button_cb, 318 on_button_down => $button_cb,
340 ; 319 ;
341 $self->{weight_widget}->set_text (CFClient::Item::weight_string $self); 320 $self->{weight_widget}->set_text (CFClient::Item::weight_string $self);
342 321
343 $self->{weight_widget}->set_tooltip ( 322 $self->{weight_widget}->set_tooltip (
344 "<b>Weight</b>.\n" 323 "<b>Weight</b>.\n"
346 . ($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. ")
347 . "\n\n$tooltip_std" 326 . "\n\n$tooltip_std"
348 ); 327 );
349} 328}
350 329
351package CFClient::Texture; 330package CFClient::Recorder;
352 331
353use strict; 332our $RECORD_WINDOW;
354 333
355use Scalar::Util; 334my $CMDBOX;
335my $CURRENT_CMDS;
336my $REC_BTN;
356 337
357use 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",
358 343
359my %TEXTURES; 344 (CFClient::SDLK_RSHIFT) => "RSHIFT",
345 (CFClient::SDLK_RCTRL ) => "RCTRL",
346 (CFClient::SDLK_RALT ) => "RALT",
347 (CFClient::SDLK_RMETA ) => "RMETA",
348);
360 349
361sub 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 {
362 my ($class, %data) = @_; 386 my ($cmd) = @_;
363 387
364 my $self = bless { 388 my $w = new CFClient::UI::FancyFrame
365 internalformat => GL_RGBA, 389 title => "Bind Action";
366 format => GL_RGBA,
367 type => GL_UNSIGNED_BYTE,
368 %data,
369 }, $class;
370 390
371 Scalar::Util::weaken ($TEXTURES{$self+0} = $self); 391 $w->add (my $vb = new CFClient::UI::VBox);
372 392 $vb->add (new CFClient::UI::Label
373 $self->upload; 393 text => "Press a modifier (CTRL, ALT and/or SHIFT) and a key."
374 394 ."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 ) 395 );
420} 396 $vb->add (my $entry = new CFClient::UI::Entry
397 text => "",
398 on_key_down => sub {
399 my ($entry, $ev) = @_;
421 400
422sub new_from_opengl { 401 my $mod = $ev->{mod};
423 my ($class, $w, $h, $cb) = @_; 402 my $sym = $ev->{sym};
424 403
425 $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
426} 405 if (grep { $_ == $sym } @ALLOWED_MODIFIER_KEYS) {
406 return;
407 }
427 408
428sub topot { 409 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] 410 and not $DIRECT_BIND_CHARS{chr ($ev->{unicode})}
430} 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 }
431 419
432sub 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 {
433 my ($self) = @_; 450 my ($list) = @_;
434 451
435 return unless $GL_VERSION; 452 return unless $CMDBOX;
436 453
437 my $data; 454 $CMDBOX->clear ();
455 $CURRENT_CMDS = $list;
438 456
439 if (exists $self->{data}) { 457 my $idx = 0;
440 $data = $self->{data};
441 458
442 } elsif (exists $self->{render_cb}) { 459 for (@$list) {
443 glViewport 0, 0, $self->{w}, $self->{h}; 460 $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 461
451 } else { 462 my $i = $idx;
452 ($self->{w}, $self->{h}, $data, $self->{internalformat}, $self->{format}, $self->{type}) 463 $hb->add (new CFClient::UI::Button
453 = 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++
454 } 474 }
475}
455 476
456 my ($tw, $th) = @$self{qw(w h)}; 477# if $show is 1 the recorder will be shown
478sub start {
479 my ($show) = @_;
457 480
458 unless ($tw > 0 && $th > 0) { 481 $RECORD_WINDOW->show if $show;
459 $tw = $th = 1; 482
460 $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;
461 } 504 }
505}
462 506
463 $self->{minified} = [CFClient::average $tw, $th, $data] 507sub make_window {
464 if $self->{minify}; 508 $RECORD_WINDOW = new CFClient::UI::FancyFrame
509 req_y => 1,
510 req_x => -1,
511 title => "Action Recorder";
465 512
466 unless ($GL_NPOT) { 513 $RECORD_WINDOW->add (my $vb = new CFClient::UI::VBox);
467 # TODO: does not work for zero-sized textures 514 $vb->add ($REC_BTN = new CFClient::UI::Button
468 $tw = topot $tw; 515 text => "start recording",
469 $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) = @_;
470 523
471 if (($tw != $self->{w} || $th != $self->{h}) && defined $data) { 524 unless ($btn->{recording}) {
472 my $bpp = (length $data) / ($self->{w} * $self->{h}); 525 start;
473 $data = pack "(a" . ($tw * $bpp) . ")*", 526 } else {
474 unpack "(a" . ($self->{w} * $bpp) . ")*", $data; 527 stop;
475 $data .= ("\x00" x ($tw * $bpp)) x ($th - $self->{h}); 528 }
476 } 529 });
477 } 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 });
478 537
479 $self->{s} = $self->{w} / $tw; 538 $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} 539}
541 540
5421; 5411;
543 542
544=back 543=back

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines