fixed dark font bug by calling gr_set_mode from gr_init. don't really know why this...
[btb/d2x.git] / 2d / palette.c
1 /* $Id: palette.c,v 1.7 2002-10-11 03:37:58 btb Exp $ */
2 /*
3 THE COMPUTER CODE CONTAINED HEREIN IS THE SOLE PROPERTY OF PARALLAX
4 SOFTWARE CORPORATION ("PARALLAX").  PARALLAX, IN DISTRIBUTING THE CODE TO
5 END-USERS, AND SUBJECT TO ALL OF THE TERMS AND CONDITIONS HEREIN, GRANTS A
6 ROYALTY-FREE, PERPETUAL LICENSE TO SUCH END-USERS FOR USE BY SUCH END-USERS
7 IN USING, DISPLAYING,  AND CREATING DERIVATIVE WORKS THEREOF, SO LONG AS
8 SUCH USE, DISPLAY OR CREATION IS FOR NON-COMMERCIAL, ROYALTY OR REVENUE
9 FREE PURPOSES.  IN NO EVENT SHALL THE END-USER USE THE COMPUTER CODE
10 CONTAINED HEREIN FOR REVENUE-BEARING PURPOSES.  THE END-USER UNDERSTANDS
11 AND AGREES TO THE TERMS HEREIN AND ACCEPTS THE SAME BY USE OF THIS FILE.
12 COPYRIGHT 1993-1998 PARALLAX SOFTWARE CORPORATION.  ALL RIGHTS RESERVED.
13 */
14
15 /*
16  * Graphical routines for setting the palette
17  *
18  * Old Log:
19  * Revision 1.41  1995/02/02  14:26:31  john
20  * Made palette fades work better with gamma thingy..
21  *
22  * Revision 1.40  1994/12/08  19:03:46  john
23  * Made functions use cfile.
24  *
25  * Revision 1.39  1994/12/01  11:23:27  john
26  * Limited Gamma from 0-8.
27  *
28  * Revision 1.38  1994/11/28  01:31:08  mike
29  * optimize color lookup function, caching recently used colors.
30  *
31  * Revision 1.37  1994/11/18  22:50:18  john
32  * Changed shorts to ints in parameters.
33  *
34  * Revision 1.36  1994/11/15  17:54:59  john
35  * Made text palette fade in when game over.
36  *
37  * Revision 1.35  1994/11/10  19:53:14  matt
38  * Fixed error handling is gr_use_palette_table()
39  *
40  * Revision 1.34  1994/11/07  13:53:48  john
41  * Added better gamma stufff.
42  *
43  * Revision 1.33  1994/11/07  13:37:56  john
44  * Added gamma correction stuff.
45  *
46  * Revision 1.32  1994/11/05  13:20:14  john
47  * Fixed bug with find_closest_color_current not working.
48  *
49  * Revision 1.31  1994/11/05  13:08:09  john
50  * MAde it return 0 when palette already faded out.
51  *
52  * Revision 1.30  1994/11/05  13:05:34  john
53  * Added back in code to allow keys during fade.
54  *
55  * Revision 1.29  1994/11/05  12:49:50  john
56  * Fixed bug with previous comment..
57  *
58  * Revision 1.28  1994/11/05  12:48:46  john
59  * Made palette only fade in / out when its supposed to.
60  *
61  * Revision 1.27  1994/11/05  12:46:43  john
62  * Changed palette stuff a bit.
63  *
64  * Revision 1.26  1994/11/01  12:59:35  john
65  * Reduced palette.256 size.
66  *
67  * Revision 1.25  1994/10/26  23:55:35  john
68  * Took out roller; Took out inverse table.
69  *
70  * Revision 1.24  1994/10/04  22:03:05  matt
71  * Fixed bug: palette wasn't fading all the way out or in
72  *
73  * Revision 1.23  1994/09/22  16:08:40  john
74  * Fixed some palette stuff.
75  *
76  * Revision 1.22  1994/09/19  11:44:31  john
77  * Changed call to allocate selector to the dpmi module.
78  *
79  * Revision 1.21  1994/09/12  19:28:09  john
80  * Fixed bug with unclipped fonts clipping.
81  *
82  * Revision 1.20  1994/09/12  18:18:39  john
83  * Set 254 and 255 to fade to themselves in fadetable
84  *
85  * Revision 1.19  1994/09/12  14:40:10  john
86  * Neatend.
87  *
88  * Revision 1.18  1994/09/09  09:31:55  john
89  * Made find_closest_color not look at superx spot of 254
90  *
91  * Revision 1.17  1994/08/09  11:27:08  john
92  * Add cthru stuff.
93  *
94  * Revision 1.16  1994/08/01  11:03:51  john
95  * MAde it read in old/new palette.256
96  *
97  * Revision 1.15  1994/07/27  18:30:27  john
98  * Took away the blending table.
99  *
100  * Revision 1.14  1994/06/09  10:39:52  john
101  * In fade out.in functions, returned 1 if key was pressed...
102  *
103  * Revision 1.13  1994/05/31  19:04:16  john
104  * Added key to stop fade if desired.
105  *
106  * Revision 1.12  1994/05/06  12:50:20  john
107  * Added supertransparency; neatend things up; took out warnings.
108  *
109  * Revision 1.11  1994/05/03  19:39:02  john
110  * *** empty log message ***
111  *
112  * Revision 1.10  1994/04/22  11:16:07  john
113  * *** empty log message ***
114  *
115  * Revision 1.9  1994/04/08  16:59:40  john
116  * Add fading poly's; Made palette fade 32 instead of 16.
117  *
118  * Revision 1.8  1994/03/16  17:21:17  john
119  * Added slow palette searching options.
120  *
121  * Revision 1.7  1994/01/07  11:47:33  john
122  * made use cflib
123  *
124  * Revision 1.6  1993/12/21  11:41:04  john
125  * *** empty log message ***
126  *
127  * Revision 1.5  1993/12/09  15:02:47  john
128  * Changed palette stuff majorly
129  *
130  * Revision 1.4  1993/12/07  12:31:41  john
131  * moved bmd_palette to gr_palette
132  *
133  * Revision 1.3  1993/10/15  16:22:23  john
134  * *** empty log message ***
135  *
136  * Revision 1.2  1993/09/26  18:59:46  john
137  * fade stuff
138  *
139  * Revision 1.1  1993/09/08  11:44:03  john
140  * Initial revision
141  *
142  */
143
144 #ifdef HAVE_CONFIG_H
145 #include <conf.h>
146 #endif
147
148 #include <stdlib.h>
149 #include <stdio.h>
150
151 #include "pstypes.h"
152 #include "u_mem.h"
153 #include "gr.h"
154 #include "grdef.h"
155 #include "cfile.h"
156 #include "error.h"
157 #include "mono.h"
158 #include "fix.h"
159 //added/remove by dph on 1/9/99
160 //#include "key.h"
161 //end remove
162
163 #include "palette.h"
164
165 extern int gr_installed;
166
167 ubyte gr_palette[256*3];
168 ubyte gr_current_pal[256*3];
169 ubyte gr_fade_table[256*34];
170
171 ubyte gr_palette_gamma = 0;
172 int gr_palette_gamma_param = 0;
173 ubyte gr_palette_faded_out = 1;
174
175 int grd_fades_disabled=0;   // Used to skip fading for development
176
177 extern void gr_palette_load( ubyte * pal );
178
179 void gr_palette_set_gamma( int gamma )
180 {
181         if ( gamma < 0 ) gamma = 0;
182 //added/changed on 10/27/98 by Victor Rachels to increase brightness slider
183         if ( gamma > 16 ) gamma = 16;      //was 8
184 //end this section change - Victor Rachels
185
186         if (gr_palette_gamma_param != gamma )   {
187                 gr_palette_gamma_param = gamma;
188                 gr_palette_gamma = gamma;
189                 if (!gr_palette_faded_out)
190                         gr_palette_load( gr_palette );
191         }
192 }
193
194 int gr_palette_get_gamma()
195 {
196         return gr_palette_gamma_param;
197 }
198
199
200 void gr_use_palette_table( char * filename )
201 {
202         CFILE *fp;
203         int i,fsize;
204 #ifdef SWAP_0_255
205         ubyte c;
206 #endif
207
208         fp = cfopen( filename, "rb" );
209
210         // the following is a hack to enable the loading of d2 levels
211         // even if only the d2 mac shareware datafiles are present.
212         // However, if the pig file is present but the palette file isn't,
213         // the textures in the level will look wierd...
214         if ( fp==NULL)
215                 fp = cfopen( DEFAULT_LEVEL_PALETTE, "rb" );
216         if ( fp==NULL)
217                 Error("Can open neither palette file <%s> "
218                       "nor default palette file <"
219                       DEFAULT_LEVEL_PALETTE
220                       ">.\n",
221                       filename);
222
223         fsize   = cfilelength( fp );
224         Assert( fsize == 9472 );
225         cfread( gr_palette, 256*3, 1, fp );
226         cfread( gr_fade_table, 256*34, 1, fp );
227         cfclose(fp);
228
229         // This is the TRANSPARENCY COLOR
230         for (i=0; i<GR_FADE_LEVELS; i++ )       {
231                 gr_fade_table[i*256+255] = 255;
232         }
233
234 // swap colors 0 and 255 of the palette along with fade table entries
235
236 #ifdef SWAP_0_255
237         for (i = 0; i < 3; i++) {
238                 c = gr_palette[i];
239                 gr_palette[i] = gr_palette[765+i];
240                 gr_palette[765+i] = c;
241         }
242
243         for (i = 0; i < GR_FADE_LEVELS * 256; i++) {
244                 if (gr_fade_table[i] == 0)
245                         gr_fade_table[i] = 255;
246         }
247         for (i=0; i<GR_FADE_LEVELS; i++)
248                 gr_fade_table[i*256] = TRANSPARENCY_COLOR;
249 #endif
250 }
251
252 #define SQUARE(x) ((x)*(x))
253
254 #define MAX_COMPUTED_COLORS     32
255
256 int     Num_computed_colors=0;
257
258 typedef struct {
259         ubyte   r,g,b,color_num;
260 } color_record;
261
262 color_record Computed_colors[MAX_COMPUTED_COLORS];
263
264 //      Add a computed color (by gr_find_closest_color) to list of computed colors in Computed_colors.
265 //      If list wasn't full already, increment Num_computed_colors.
266 //      If was full, replace a random one.
267 void add_computed_color(int r, int g, int b, int color_num)
268 {
269         int     add_index;
270
271         if (Num_computed_colors < MAX_COMPUTED_COLORS) {
272                 add_index = Num_computed_colors;
273                 Num_computed_colors++;
274         } else
275                 add_index = (d_rand() * MAX_COMPUTED_COLORS) >> 15;
276
277         Computed_colors[add_index].r = r;
278         Computed_colors[add_index].g = g;
279         Computed_colors[add_index].b = b;
280         Computed_colors[add_index].color_num = color_num;
281 }
282
283 void init_computed_colors(void)
284 {
285         int     i;
286
287         for (i=0; i<MAX_COMPUTED_COLORS; i++)
288                 Computed_colors[i].r = 255;             //      Make impossible to match.
289 }
290
291 int gr_find_closest_color( int r, int g, int b )
292 {
293         int i, j;
294         int best_value, best_index, value;
295
296         if (Num_computed_colors == 0)
297                 init_computed_colors();
298
299         //      If we've already computed this color, return it!
300         for (i=0; i<Num_computed_colors; i++)
301                 if (r == Computed_colors[i].r)
302                         if (g == Computed_colors[i].g)
303                                 if (b == Computed_colors[i].b) {
304                                         if (i > 4) {
305                                                 color_record    trec;
306                                                 trec = Computed_colors[i-1];
307                                                 Computed_colors[i-1] = Computed_colors[i];
308                                                 Computed_colors[i] = trec;
309                                                 return Computed_colors[i-1].color_num;
310                                         }
311                                         return Computed_colors[i].color_num;
312                                 }
313
314 //      r &= 63;
315 //      g &= 63;
316 //      b &= 63;
317
318         best_value = SQUARE(r-gr_palette[0])+SQUARE(g-gr_palette[1])+SQUARE(b-gr_palette[2]);
319         best_index = 0;
320         if (best_value==0) {
321                 add_computed_color(r, g, b, best_index);
322                 return best_index;
323         }
324         j=0;
325         // only go to 255, 'cause we dont want to check the transparent color.
326         for (i=1; i<254; i++ )  {
327                 j += 3;
328                 value = SQUARE(r-gr_palette[j])+SQUARE(g-gr_palette[j+1])+SQUARE(b-gr_palette[j+2]);
329                 if ( value < best_value )       {
330                         if (value==0) {
331                                 add_computed_color(r, g, b, i);
332                                 return i;
333                         }
334                         best_value = value;
335                         best_index = i;
336                 }
337         }
338         add_computed_color(r, g, b, best_index);
339         return best_index;
340 }
341
342 int gr_find_closest_color_15bpp( int rgb )
343 {
344         return gr_find_closest_color( ((rgb>>10)&31)*2, ((rgb>>5)&31)*2, (rgb&31)*2 );
345 }
346
347
348 int gr_find_closest_color_current( int r, int g, int b )
349 {
350         int i, j;
351         int best_value, best_index, value;
352
353 //      r &= 63;
354 //      g &= 63;
355 //      b &= 63;
356
357         best_value = SQUARE(r-gr_current_pal[0])+SQUARE(g-gr_current_pal[1])+SQUARE(b-gr_current_pal[2]);
358         best_index = 0;
359         if (best_value==0)
360                 return best_index;
361
362         j=0;
363         // only go to 255, 'cause we dont want to check the transparent color.
364         for (i=1; i<254; i++ )  {
365                 j += 3;
366                 value = SQUARE(r-gr_current_pal[j])+SQUARE(g-gr_current_pal[j+1])+SQUARE(b-gr_current_pal[j+2]);
367                 if ( value < best_value )       {
368                         if (value==0)
369                                 return i;
370                         best_value = value;
371                         best_index = i;
372                 }
373         }
374         return best_index;
375 }
376
377 void gr_make_cthru_table(ubyte * table, ubyte r, ubyte g, ubyte b )
378 {
379         int i;
380         ubyte r1, g1, b1;
381
382         for (i=0; i<256; i++ )  {
383                 r1 = gr_palette[i*3+0] + r;
384                 if ( r1 > 63 ) r1 = 63;
385                 g1 = gr_palette[i*3+1] + g;
386                 if ( g1 > 63 ) g1 = 63;
387                 b1 = gr_palette[i*3+2] + b;
388                 if ( b1 > 63 ) b1 = 63;
389                 table[i] = gr_find_closest_color( r1, g1, b1 );
390         }
391 }
392