ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/deliantra/Deliantra-Client/DC.pm
Revision: 1.28
Committed: Wed Apr 12 20:06:36 2006 UTC (18 years, 1 month ago) by root
Branch: MAIN
Changes since 1.27: +1 -0 lines
Log Message:
even less gtk

File Contents

# User Rev Content
1 root 1.1 =head1 NAME
2    
3 root 1.22 CFClient - undocumented utility garbage for our crossfire client
4 root 1.1
5     =head1 SYNOPSIS
6    
7 root 1.22 use CFClient;
8 root 1.1
9     =head1 DESCRIPTION
10    
11     =over 4
12    
13     =cut
14    
15 root 1.22 package CFClient;
16 root 1.1
17     BEGIN {
18     $VERSION = '0.1';
19    
20 root 1.2 use XSLoader;
21 root 1.22 XSLoader::load "CFClient", $VERSION;
22 root 1.1 }
23    
24 root 1.26 use SDL::OpenGL;
25    
26 root 1.19 our %GL_EXT;
27     our $GL_VERSION;
28    
29     our $GL_NPOT;
30    
31     sub gl_init {
32     $GL_VERSION = gl_version * 1;
33     %GL_EXT = map +($_ => 1), split /\s+/, gl_extensions;
34    
35     $GL_NPOT = $GL_EXT{GL_ARB_texture_non_power_of_two} || $GL_VERSION >= 2;
36    
37 root 1.26 glClearColor 0.45, 0.45, 0.45, 1;
38    
39     glEnable GL_TEXTURE_2D;
40     glEnable GL_COLOR_MATERIAL;
41     glShadeModel GL_FLAT;
42     glDisable GL_DEPTH_TEST;
43     glBlendFunc GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA;
44    
45     glHint GL_PERSPECTIVE_CORRECTION_HINT, GL_FASTEST;
46    
47 root 1.22 CFClient::Texture::restore_state ();
48 root 1.19 }
49    
50 root 1.5 sub find_rcfile($) {
51     my $path;
52    
53     for (@INC) {
54 root 1.22 $path = "$_/CFClient/resources/$_[0]";
55 root 1.5 return $path if -r $path;
56     }
57    
58     die "FATAL: can't find required file $_[0]\n";
59     }
60    
61     sub read_cfg {
62     my ($file) = @_;
63    
64     open CFG, $file
65     or return;
66    
67     my $CFG;
68    
69     local $/;
70     $CFG = eval <CFG>;
71    
72     $::CFG = $CFG;
73    
74     close CFG;
75     }
76    
77     sub write_cfg {
78     my ($file) = @_;
79    
80     open CFG, ">$file"
81     or return;
82    
83     {
84 elmex 1.9 require Data::Dumper;
85 root 1.5 local $Data::Dumper::Purity = 1;
86     $::CFG->{VERSION} = $::VERSION;
87     print CFG Data::Dumper->Dump ([$::CFG], [qw/CFG/]);
88     }
89    
90     close CFG;
91     }
92    
93 root 1.22 package CFClient::Texture;
94 root 1.3
95 root 1.25 use strict;
96    
97 root 1.3 use Scalar::Util;
98    
99     use SDL::OpenGL;
100    
101     my @textures;
102    
103 root 1.14 sub new {
104 root 1.4 my ($class, %data) = @_;
105    
106 root 1.14 my $self = bless {
107 root 1.15 internalformat => GL_RGBA,
108     format => GL_RGBA,
109 root 1.14 %data,
110     }, $class;
111 root 1.4
112     push @textures, $self;
113     Scalar::Util::weaken $textures[-1];
114    
115     $self->upload;
116    
117     $self
118     }
119    
120     sub new_from_image {
121     my ($class, $image) = @_;
122    
123 root 1.14 $class->new (image => $image)
124 root 1.4 }
125    
126 root 1.3 sub new_from_file {
127     my ($class, $path) = @_;
128    
129     open my $fh, "<:raw", $path
130     or die "$path: $!";
131    
132     local $/;
133 root 1.4 $class->new_from_image (<$fh>)
134 root 1.3 }
135    
136 root 1.14 #sub new_from_surface {
137     # my ($class, $surface) = @_;
138     #
139     # $surface->rgba;
140     #
141     # $class->new (
142     # data => $surface->pixels,
143 root 1.24 # w => $surface->width,
144     # h => $surface->height,
145 root 1.14 # )
146     #}
147    
148 root 1.21 sub new_from_layout {
149     my ($class, $layout) = @_;
150 root 1.14
151 root 1.21 my ($w, $h, $data) = $layout->render;
152 root 1.14
153     $class->new (
154 root 1.24 w => $w,
155     h => $h,
156 root 1.14 data => $data,
157 root 1.16 internalformat => GL_ALPHA4,
158 root 1.14 format => GL_ALPHA,
159 root 1.4 )
160 root 1.3 }
161    
162 root 1.8 sub new_from_opengl {
163     my ($class, $w, $h, $cb) = @_;
164    
165 root 1.24 $class->new (w => $w, h => $h, render_cb => $cb)
166 root 1.8 }
167    
168 root 1.19 sub topot {
169     (grep $_ >= $_[0], 1, 2, 4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096, 8192, 16384, 32768)[0]
170     }
171    
172 root 1.3 sub upload {
173     my ($self) = @_;
174    
175 root 1.27 return unless $GL_VERSION;
176 root 1.3
177 root 1.6 my $data;
178 root 1.3
179     if (exists $self->{data}) {
180 root 1.6 $data = $self->{data};
181 root 1.25
182 root 1.24 } elsif (exists $self->{render_cb}) {
183     glViewport 0, 0, $self->{w}, $self->{h};
184 root 1.25 glOrtho 0, $self->{w}, 0, $self->{h}, -10000, 10000;
185 root 1.12 glMatrixMode GL_PROJECTION;
186     glLoadIdentity;
187     glMatrixMode GL_MODELVIEW;
188     glLoadIdentity;
189 root 1.8 glClear GL_COLOR_BUFFER_BIT;
190 root 1.25 $self->{render_cb}->($self, $self->{w}, $self->{h});
191 root 1.8
192 root 1.3 } else {
193 root 1.28 use Gtk2;#d# TODO kill
194 root 1.3 my $pb = new Gtk2::Gdk::PixbufLoader;
195 root 1.4 $pb->write ($self->{image});
196 root 1.3 $pb->close;
197    
198     $pb = $pb->get_pixbuf;
199     $pb = $pb->add_alpha (0, 0, 0, 0);
200    
201 root 1.24 $self->{w} = $pb->get_width;
202     $self->{h} = $pb->get_height;
203 root 1.6
204     $data = $pb->get_pixels;
205 root 1.3 }
206    
207 root 1.24 my ($tw, $th) = @$self{qw(w h)};
208 root 1.19
209 root 1.25 unless ($tw && $th) {
210 root 1.24 $tw = $th = 1;
211     $data = "\x00" x 64;
212     }
213    
214     unless ($GL_NPOT) {
215     # TODO: does not work for zero-sized textures
216 root 1.19 $tw = topot $tw;
217     $th = topot $th;
218    
219 root 1.25 if ($tw != $self->{w} || $th != $self->{h} && defined $data) {
220 root 1.24 my $bpp = (length $data) / ($self->{w} * $self->{h});
221 root 1.20 $data = pack "(a" . ($tw * $bpp) . ")*",
222 root 1.24 unpack "(a" . ($self->{w} * $bpp) . ")*", $data;
223     $data .= ("\x00" x ($tw * $bpp)) x ($th - $self->{h});
224 root 1.19 }
225     }
226    
227 root 1.24 $self->{s} = $self->{w} / $tw;
228     $self->{t} = $self->{h} / $th;
229 root 1.19
230     $self->{name} ||= (glGenTextures 1)->[0];
231 root 1.3
232     glBindTexture GL_TEXTURE_2D, $self->{name};
233    
234 root 1.26 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 root 1.3 glTexParameter GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP;
237     glTexParameter GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP;
238    
239 root 1.8 if (defined $data) {
240     glTexImage2D GL_TEXTURE_2D, 0,
241 root 1.15 $self->{internalformat},
242 root 1.19 $tw, $th, # need to pad texture first
243 root 1.8 0,
244 root 1.14 $self->{format},
245 root 1.8 GL_UNSIGNED_BYTE,
246     $data;
247 root 1.16 glGetError and die;
248 root 1.8 } else {
249     glCopyTexImage2D GL_TEXTURE_2D, 0,
250 root 1.15 $self->{internalformat},
251 root 1.8 0, 0,
252 root 1.19 $tw, $th,
253 root 1.8 0;
254 root 1.25 glGetError and die;
255 root 1.8 }
256 root 1.3 }
257    
258     sub DESTROY {
259     my ($self) = @_;
260    
261     return unless exists $self->{name};
262    
263     glDeleteTextures delete $self->{name};
264     }
265    
266 root 1.19 sub restore_state{
267 root 1.3 $_->upload
268     for grep $_, @textures;
269     };
270    
271 root 1.1 1;
272    
273     =back
274    
275     =head1 AUTHOR
276    
277     Marc Lehmann <schmorp@schmorp.de>
278     http://home.schmorp.de/
279    
280     =cut
281