mirror of
https://github.com/ZDoom/Raze.git
synced 2025-01-06 09:10:48 +00:00
251 lines
5.5 KiB
Python
251 lines
5.5 KiB
Python
|
#!/usr/bin/python
|
||
|
|
||
|
import sys;
|
||
|
|
||
|
from numpy import array, zeros, ones, arange
|
||
|
from numpy import vstack, hstack, hsplit, dstack, dsplit
|
||
|
|
||
|
from PIL.Image import frombuffer
|
||
|
|
||
|
|
||
|
NBITS = 7;
|
||
|
RESIDBITS = 8-NBITS;
|
||
|
DIM = 1<<NBITS;
|
||
|
|
||
|
CONVFACT = 256;
|
||
|
|
||
|
### target hues
|
||
|
green = .39; # 11, 22
|
||
|
yellow = .17; # 23
|
||
|
red = .04; # 21
|
||
|
|
||
|
|
||
|
def genbasepal():
|
||
|
"Generate base palette for highpalookup system. \
|
||
|
All other palettes must be based on this one."
|
||
|
imint = zeros([DIM, DIM, DIM], 'uint32');
|
||
|
for i in range(DIM):
|
||
|
for j in range(DIM):
|
||
|
for k in range(DIM):
|
||
|
imint[i,j,k] = ((i<<16)|(k<<8)|j)<<RESIDBITS;
|
||
|
|
||
|
imint = hstack(dsplit(imint, DIM));
|
||
|
|
||
|
imbyte = zeros([DIM, DIM*DIM, 3], 'uint8');
|
||
|
|
||
|
for i in range(3):
|
||
|
imbyte[:,:,i] = (imint[:,:,0]>>(i*8))&255;
|
||
|
|
||
|
return imbyte;
|
||
|
|
||
|
|
||
|
def getdispimg(im):
|
||
|
"Get a reshaped version of the palette image IM suitable for viewing."
|
||
|
# 2^NBITS x 2^NBITS*2^NBITS --> 2^(NBITS + NBITS/2) x 2^NBITS*2^(NBITS - NBITS/2)
|
||
|
|
||
|
if (im.shape != (DIM, DIM*DIM, 3)):
|
||
|
raise ValueError("image must have shape (DIM, DIM*DIM, 3)");
|
||
|
|
||
|
dimfactor = 1<<(NBITS/2);
|
||
|
return vstack(hsplit(im, dimfactor));
|
||
|
|
||
|
|
||
|
def getPILimg(im):
|
||
|
sz = im.shape;
|
||
|
return frombuffer("RGB", [sz[1], sz[0]], im, "raw", "RGB", 0, 1);
|
||
|
|
||
|
|
||
|
def saveimage(im, filename):
|
||
|
getPILimg(im).save(filename);
|
||
|
|
||
|
def showpalimg(im):
|
||
|
getPILimg(getdispimg(im)).show();
|
||
|
|
||
|
|
||
|
### utility functions
|
||
|
|
||
|
## port of Octave's rbg2hsv
|
||
|
def rgb2hsv(im):
|
||
|
im = imitof(im);
|
||
|
|
||
|
r, g, b = im[..., 0], im[..., 1], im[..., 2];
|
||
|
s, v = im.min(2), im.max(2);
|
||
|
dif = v-s;
|
||
|
|
||
|
colored = (s != v);
|
||
|
|
||
|
h = zeros(r.shape, im.dtype);
|
||
|
|
||
|
# blue hue
|
||
|
idx = ((v==b) & colored);
|
||
|
h[idx] = 2./3 + 1./6*(r[idx]-g[idx])/dif[idx];
|
||
|
|
||
|
# green hue
|
||
|
idx = ((v==g) & colored);
|
||
|
h[idx] = 1./3 + 1./6*(b[idx]-r[idx])/dif[idx];
|
||
|
|
||
|
# red hue
|
||
|
idx = ((v==r) & colored);
|
||
|
h[idx] = 1./6*(g[idx]-b[idx])/dif[idx];
|
||
|
h[idx] %= 1;
|
||
|
|
||
|
s[~colored] = 0;
|
||
|
s[colored] = 1.0 - s[colored]/v[colored];
|
||
|
|
||
|
return dstack((h,s,v));
|
||
|
|
||
|
|
||
|
## port of Octave's hsv2rbg
|
||
|
def hsv2rgb(imh):
|
||
|
imh[imh<0] = 0;
|
||
|
imh[imh>1] = 1;
|
||
|
|
||
|
h, s, v = imh[..., 0], imh[..., 1], imh[..., 2];
|
||
|
|
||
|
rgb = v*(1.0-s);
|
||
|
rgb = dstack((rgb, rgb, rgb));
|
||
|
|
||
|
hue = dstack((h-2./3, h, h-1./3))%1;
|
||
|
f = s*v;
|
||
|
f = dstack((f, f, f));
|
||
|
|
||
|
rgb += f * (6.0 * (hue < 1./6)*hue
|
||
|
+ ((hue >= 1./6) & (hue < 1./2))
|
||
|
+ ((hue >= 1./2) & (hue < 2./3))*(4.0 - 6.0*hue));
|
||
|
|
||
|
return imftoi(rgb);
|
||
|
|
||
|
|
||
|
def imftoi(im):
|
||
|
im *= CONVFACT;
|
||
|
im[im>255] = 255;
|
||
|
return im.astype('uint8');
|
||
|
|
||
|
def imitof(im):
|
||
|
return im.astype('float32')/CONVFACT;
|
||
|
|
||
|
|
||
|
###
|
||
|
def genpal(basepal, basepalhsv, pal):
|
||
|
"Generate a highpalookup image for palette number PAL. \
|
||
|
BASEPALHSV should the precomputed HSV representation of BASEPAL."
|
||
|
|
||
|
if (basepal.dtype != 'uint8'):
|
||
|
raise TypeError('BASEPAL should be uint8.');
|
||
|
|
||
|
if (pal==0):
|
||
|
return basepal;
|
||
|
|
||
|
bph = basepalhsv;
|
||
|
h,s,v = bph[..., 0], bph[..., 1], bph[..., 2];
|
||
|
|
||
|
bluemask = (h>0.52) & (h<0.8) & (s>0.2);
|
||
|
|
||
|
# all true mask will be used unless overridden
|
||
|
mask = ones(h.shape, 'bool');
|
||
|
|
||
|
# plagman:
|
||
|
if (pal==1):
|
||
|
h[:] = 0.66;
|
||
|
|
||
|
elif (pal==6):
|
||
|
h[:] = 0.33;
|
||
|
v = 1.0 - v;
|
||
|
|
||
|
elif (pal==20):
|
||
|
m1 = ((h>0.6) & (h<0.7));
|
||
|
m2 = ((h>0.04) & (h<0.13));
|
||
|
m3 = ((h>0.7) & (h<0.9));
|
||
|
m4 = ((h>0.3) & (h<0.36));
|
||
|
mask = m1 | m2 | m3 | m4;
|
||
|
|
||
|
# blue to gray by removing all saturation
|
||
|
h[m1] = 0.0;
|
||
|
# orange and brown to blue
|
||
|
h[m2] = 0.66;
|
||
|
# purple and reddish to blue
|
||
|
h[m3] = 0.66
|
||
|
# green to blue
|
||
|
h[m4] = 0.66;
|
||
|
|
||
|
# helixhorned:
|
||
|
elif (pal==11):
|
||
|
mask = bluemask;
|
||
|
h[:] = green;
|
||
|
s += 0.1;
|
||
|
|
||
|
elif (pal==12):
|
||
|
mask = bluemask;
|
||
|
h[:] = 0.0;
|
||
|
s[:] = 0.0;
|
||
|
|
||
|
elif (pal==13):
|
||
|
mask = bluemask;
|
||
|
h[:] = 0.0;
|
||
|
s[:] = 0.0;
|
||
|
v *= 0.7;
|
||
|
|
||
|
elif (pal==16):
|
||
|
mask = bluemask;
|
||
|
s += 0.1;
|
||
|
v -= 0.1;
|
||
|
|
||
|
elif (pal==21):
|
||
|
mask = bluemask;
|
||
|
h[:] = red;
|
||
|
s += 0.3;
|
||
|
|
||
|
elif (pal==23):
|
||
|
mask = bluemask;
|
||
|
h[:] = yellow;
|
||
|
s += 0.12;
|
||
|
v *= 1.15;
|
||
|
|
||
|
# user:
|
||
|
# ...
|
||
|
|
||
|
else:
|
||
|
raise ValueError("unknown pal {0}!".format(pal));
|
||
|
|
||
|
# ---
|
||
|
newrgb = hsv2rgb(dstack((h, s, v)));
|
||
|
|
||
|
r = mask*newrgb[:,:,0] + (~mask)*basepal[:,:,0];
|
||
|
g = mask*newrgb[:,:,1] + (~mask)*basepal[:,:,1];
|
||
|
b = mask*newrgb[:,:,2] + (~mask)*basepal[:,:,2];
|
||
|
|
||
|
# PIL doesn't seem to like views/shallow copies
|
||
|
return dstack((r, g, b)).copy();
|
||
|
|
||
|
|
||
|
## main
|
||
|
if (__name__ == "__main__"):
|
||
|
|
||
|
argc = len(sys.argv);
|
||
|
if (argc == 1):
|
||
|
print "Usage: python prhighpal.py <palnum>"
|
||
|
sys.exit();
|
||
|
elif (argc > 2):
|
||
|
print "There's a weird bug when passing more than one palnum; \
|
||
|
processing only the first one.\n"
|
||
|
|
||
|
print "Generating base palette..."
|
||
|
bp = genbasepal();
|
||
|
bph = rgb2hsv(bp);
|
||
|
|
||
|
for i in [1]: #xrange(1, argc):
|
||
|
palnum = int(sys.argv[i]);
|
||
|
filename = "hipal{0}_gen.png".format(palnum);
|
||
|
print "Generating palnum", palnum, "image ...";
|
||
|
palimg = genpal(bp, bph, palnum);
|
||
|
print "Writing", filename, "...";
|
||
|
saveimage(palimg, filename);
|
||
|
|
||
|
print "\nDEF code:\n"
|
||
|
for i in xrange(1, argc):
|
||
|
palnum = int(sys.argv[i]);
|
||
|
if (palnum==0):
|
||
|
continue;
|
||
|
filename = "hipal{0}_gen.png".format(palnum);
|
||
|
print "highpalookup { pal", palnum, "file", filename, "}";
|