Convert light frequency to RGB?

前端 未结 9 759
旧时难觅i
旧时难觅i 2020-12-02 04:43

Does anyone know of any formula for converting a light frequency to an RGB value?

相关标签:
9条回答
  • 2020-12-02 05:11

    I guess I might as well follow up my comment with a formal answer. The best option is to use the HSV colour space - though the hue represents the wavelength it is not a one-to-one comparison.

    0 讨论(0)
  • 2020-12-02 05:14

    For lazy guys (like me), here is an implementation in java of the code found in @user151323 's answer (that is, just a simple translation from pascal code found in Spectra Lab Report):

    static private final double Gamma = 0.80;
    static private final double IntensityMax = 255;
    
    /**
     * Taken from Earl F. Glynn's web page:
     * <a href="http://www.efg2.com/Lab/ScienceAndEngineering/Spectra.htm">Spectra Lab Report</a>
     */
    public static int[] waveLengthToRGB(double Wavelength) {
        double factor;
        double Red, Green, Blue;
    
        if((Wavelength >= 380) && (Wavelength < 440)) {
            Red = -(Wavelength - 440) / (440 - 380);
            Green = 0.0;
            Blue = 1.0;
        } else if((Wavelength >= 440) && (Wavelength < 490)) {
            Red = 0.0;
            Green = (Wavelength - 440) / (490 - 440);
            Blue = 1.0;
        } else if((Wavelength >= 490) && (Wavelength < 510)) {
            Red = 0.0;
            Green = 1.0;
            Blue = -(Wavelength - 510) / (510 - 490);
        } else if((Wavelength >= 510) && (Wavelength < 580)) {
            Red = (Wavelength - 510) / (580 - 510);
            Green = 1.0;
            Blue = 0.0;
        } else if((Wavelength >= 580) && (Wavelength < 645)) {
            Red = 1.0;
            Green = -(Wavelength - 645) / (645 - 580);
            Blue = 0.0;
        } else if((Wavelength >= 645) && (Wavelength < 781)) {
            Red = 1.0;
            Green = 0.0;
            Blue = 0.0;
        } else {
            Red = 0.0;
            Green = 0.0;
            Blue = 0.0;
        }
    
        // Let the intensity fall off near the vision limits
    
        if((Wavelength >= 380) && (Wavelength < 420)) {
            factor = 0.3 + 0.7 * (Wavelength - 380) / (420 - 380);
        } else if((Wavelength >= 420) && (Wavelength < 701)) {
            factor = 1.0;
        } else if((Wavelength >= 701) && (Wavelength < 781)) {
            factor = 0.3 + 0.7 * (780 - Wavelength) / (780 - 700);
        } else {
            factor = 0.0;
        }
    
    
        int[] rgb = new int[3];
    
        // Don't want 0^x = 1 for x <> 0
        rgb[0] = Red == 0.0 ? 0 : (int)Math.round(IntensityMax * Math.pow(Red * factor, Gamma));
        rgb[1] = Green == 0.0 ? 0 : (int)Math.round(IntensityMax * Math.pow(Green * factor, Gamma));
        rgb[2] = Blue == 0.0 ? 0 : (int)Math.round(IntensityMax * Math.pow(Blue * factor, Gamma));
    
        return rgb;
    }
    
    0 讨论(0)
  • 2020-12-02 05:16

    General idea:

    1. Use CEI color matching functions to convert wavelength to XYZ color.
    2. Convert XYZ to RGB
    3. Clip components to [0..1] and multiply by 255 to fit in the unsigned byte range.

    Steps 1 and 2 may vary.

    There are several color matching functions, available as tables or as analytic approximations (suggested by @Tarc and @Haochen Xie). Tables are best if you need a smooth preсise result.

    There is no single RGB color space. Multiple transformation matrices and different kinds of gamma correction may be used.

    Below is the C# code I came up with recently. It uses linear interpolation over the "CIE 1964 standard observer" table and sRGB matrix + gamma correction.

    static class RgbCalculator {
    
        const int
             LEN_MIN = 380,
             LEN_MAX = 780,
             LEN_STEP = 5;
    
        static readonly double[]
            X = {
                    0.000160, 0.000662, 0.002362, 0.007242, 0.019110, 0.043400, 0.084736, 0.140638, 0.204492, 0.264737,
                    0.314679, 0.357719, 0.383734, 0.386726, 0.370702, 0.342957, 0.302273, 0.254085, 0.195618, 0.132349,
                    0.080507, 0.041072, 0.016172, 0.005132, 0.003816, 0.015444, 0.037465, 0.071358, 0.117749, 0.172953,
                    0.236491, 0.304213, 0.376772, 0.451584, 0.529826, 0.616053, 0.705224, 0.793832, 0.878655, 0.951162,
                    1.014160, 1.074300, 1.118520, 1.134300, 1.123990, 1.089100, 1.030480, 0.950740, 0.856297, 0.754930,
                    0.647467, 0.535110, 0.431567, 0.343690, 0.268329, 0.204300, 0.152568, 0.112210, 0.081261, 0.057930,
                    0.040851, 0.028623, 0.019941, 0.013842, 0.009577, 0.006605, 0.004553, 0.003145, 0.002175, 0.001506,
                    0.001045, 0.000727, 0.000508, 0.000356, 0.000251, 0.000178, 0.000126, 0.000090, 0.000065, 0.000046,
                    0.000033
                },
    
            Y = {
                    0.000017, 0.000072, 0.000253, 0.000769, 0.002004, 0.004509, 0.008756, 0.014456, 0.021391, 0.029497,
                    0.038676, 0.049602, 0.062077, 0.074704, 0.089456, 0.106256, 0.128201, 0.152761, 0.185190, 0.219940,
                    0.253589, 0.297665, 0.339133, 0.395379, 0.460777, 0.531360, 0.606741, 0.685660, 0.761757, 0.823330,
                    0.875211, 0.923810, 0.961988, 0.982200, 0.991761, 0.999110, 0.997340, 0.982380, 0.955552, 0.915175,
                    0.868934, 0.825623, 0.777405, 0.720353, 0.658341, 0.593878, 0.527963, 0.461834, 0.398057, 0.339554,
                    0.283493, 0.228254, 0.179828, 0.140211, 0.107633, 0.081187, 0.060281, 0.044096, 0.031800, 0.022602,
                    0.015905, 0.011130, 0.007749, 0.005375, 0.003718, 0.002565, 0.001768, 0.001222, 0.000846, 0.000586,
                    0.000407, 0.000284, 0.000199, 0.000140, 0.000098, 0.000070, 0.000050, 0.000036, 0.000025, 0.000018,
                    0.000013
                },
    
            Z = {
                    0.000705, 0.002928, 0.010482, 0.032344, 0.086011, 0.197120, 0.389366, 0.656760, 0.972542, 1.282500,
                    1.553480, 1.798500, 1.967280, 2.027300, 1.994800, 1.900700, 1.745370, 1.554900, 1.317560, 1.030200,
                    0.772125, 0.570060, 0.415254, 0.302356, 0.218502, 0.159249, 0.112044, 0.082248, 0.060709, 0.043050,
                    0.030451, 0.020584, 0.013676, 0.007918, 0.003988, 0.001091, 0.000000, 0.000000, 0.000000, 0.000000,
                    0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000,
                    0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000,
                    0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000,
                    0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000, 0.000000,
                    0.000000
                };
    
        static readonly double[]
            MATRIX_SRGB_D65 = {
                 3.2404542, -1.5371385, -0.4985314,
                -0.9692660,  1.8760108,  0.0415560,
                 0.0556434, -0.2040259,  1.0572252
            };
    
        public static byte[] Calc(double len) {
            if(len < LEN_MIN || len > LEN_MAX)
                return new byte[3];
    
            len -= LEN_MIN;
            var index = (int)Math.Floor(len / LEN_STEP);
            var offset = len - LEN_STEP * index;
    
            var x = Interpolate(X, index, offset);
            var y = Interpolate(Y, index, offset);
            var z = Interpolate(Z, index, offset);
    
            var m = MATRIX_SRGB_D65;
    
            var r = m[0] * x + m[1] * y + m[2] * z;
            var g = m[3] * x + m[4] * y + m[5] * z;
            var b = m[6] * x + m[7] * y + m[8] * z;
    
            r = Clip(GammaCorrect_sRGB(r));
            g = Clip(GammaCorrect_sRGB(g));
            b = Clip(GammaCorrect_sRGB(b));
    
            return new[] { 
                (byte)(255 * r),
                (byte)(255 * g),
                (byte)(255 * b)
            };
        }
    
        static double Interpolate(double[] values, int index, double offset) {
            if(offset == 0)
                return values[index];
    
            var x0 = index * LEN_STEP;
            var x1 = x0 + LEN_STEP;
            var y0 = values[index];
            var y1 = values[1 + index];
    
            return y0 + offset * (y1 - y0) / (x1 - x0);
        }
    
        static double GammaCorrect_sRGB(double c) {
            if(c <= 0.0031308)
                return 12.92 * c;
    
            var a = 0.055;
            return (1 + a) * Math.Pow(c, 1 / 2.4) - a;
        }
    
        static double Clip(double c) {
            if(c < 0)
                return 0;
            if(c > 1)
                return 1;
            return c;
        }
    }
    

    Result for the 400-700 nm range:

    0 讨论(0)
  • 2020-12-02 05:17

    Project the wavelength's CIExy towards the D65 white onto the sRGB gamut

    #!/usr/bin/ghci
    ångstrømsfromTHz terahertz = 2997924.58 / terahertz
    tristimulusXYZfromÅngstrøms å=map(sum.map(stimulus))[
     [[1056,5998,379,310],[362,4420,160,267],[-65,5011,204,262]],
     [[821,5688,469,405],[286,5309,163,311]],
     [[1217,4370,118,360],[681,4590,260,138]]]
     where stimulus[ω,μ,ς,σ]=ω/1000*exp(-((å-μ)/if å<μ then ς else σ)^2/2)
    
    standardRGBfromTristimulusXYZ xyz=
     map(gamma.sum.zipWith(*)(gamutConfine xyz))[
     [3.2406,-1.5372,-0.4986],[-0.9689,1.8758,0.0415],[0.0557,-0.2040,1.057]]
    gamma u=if u<=0.0031308 then 12.92*u else (u**(5/12)*211-11)/200
    [red,green,blue,black]=
     [[0.64,0.33],[0.3,0.6],[0.15,0.06],[0.3127,0.3290,0]]
    ciexyYfromXYZ xyz=if xyz!!1==0 then black else map(/sum xyz)xyz
    cieXYZfromxyY[x,y,l]=if y==0 then black else[x*l/y,l,(1-x-y)*l/y]
    gamutConfine xyz=last$xyz:[cieXYZfromxyY[x0+t*(x1-x0),y0+t*(y1-y0),xyz!!1]|
     x0:y0:_<-[black],x1:y1:_<-[ciexyYfromXYZ xyz],i<-[0..2],
     [x2,y2]:[x3,y3]:_<-[drop i[red,green,blue,red]],
     det<-[(x0-x1)*(y2-y3)-(y0-y1)*(x2-x3)],
     t <-[((x0-x2)*(y2-y3)-(y0-y2)*(x2-x3))/det|det/=0],0<=t,t<=1]
    
    sRGBfromÅ=standardRGBfromTristimulusXYZ.tristimulusXYZfromÅngstrøms
    x s rgb=concat["\ESC[48;2;",
                   intercalate";"$map(show.(17*).round.(15*).max 0.min 1)rgb,
                   "m",s,"\ESC[49m"]
    spectrum=concatMap(x" ".sRGBfromÅ)$takeWhile(<7000)$iterate(+60)4000
    main=putStrLn spectrum
    
    0 讨论(0)
  • 2020-12-02 05:18

    I did a linear fit of known hue values and frequencies (dropping out red and violet because they extend so far in frequency values that they skew things a bit) and I got a rough conversion equation.

    It goes like
    frequency (in THz)=474+(3/4)(Hue Angle (in degrees))

    I've tried to look around and see if anyone has come up with this equation, but I haven't found anything as of May 2010.

    0 讨论(0)
  • 2020-12-02 05:21

    Here's a detailed explanation of the entire conversion process: http://www.fourmilab.ch/documents/specrend/. Source code included!

    0 讨论(0)
提交回复
热议问题