半透明混色算法
曾经遇到一个需求,PM 希望可以在设备上呈现一个自定义的图像,上面叠加一个半透明的 Logo 。
由于底部图像是自定义的,且 Logo 位置也非固定,这就需要自行生成一个叠加图像,给到设备去显示。这时一个问题便出现了。我们需要对 Logo 的半透明区域和背景图像的颜色做混色处理。
因此,我们要能够对每一个像素点的颜色进行混色。混色的原理其实十分简单,已知底图像素颜色为 c1
,半透明图像素点颜色为 c2
,透明度 alpha 值为 a
,不透明 alpha 值为 a'
。那么也就是混色区域的颜色值 c
,其 c2
颜色保留 a / a'
,c1
颜色保留 1 - a / a'
。那么就可以得出公式:
c = (c1 * (a' - a) + c2 * a) / a'
按照这种方式分别计算 RGB 三个通道的颜色值即可。
这里附上 VC 版本的
https://github.com/imlinhanchao/mix
。核心方法实现https://github.com/imlinhanchao/Mix/blob/811f3eabb197380f2c06d1e8886d4a164c5e92e8/Mix/MixDlg.cpp#L192
:CString CMixDlg::MixPicture( CString sPic1, CString sPic2, CPoint ptPos, CString sSave )
{
ATL::CImage img1, img2;
img1.Load(sPic1); img2.Load(sPic2);
int nWidth1 = img1.GetWidth(), nHeight1 = img1.GetHeight();
int nWidth2 = img2.GetWidth(), nHeight2 = img2.GetHeight();
for (int i = 0; i < nWidth2; i++)
{
int nX = ptPos.x + i;
if (nX > nWidth1) break;
for (int j = 0; j < nHeight2; j++)
{
int nY = ptPos.y + j;
if (nY > nHeight1) break;
// get color and alpha from image
COLORREF clr1 = img1.GetPixel(nX, nY);
COLORREF clr2 = img2.GetPixel(i, j);
BYTE* pAlpha = (BYTE*)img2.GetPixelAddress(i, j) + 3;
int nAlpha = (*pAlpha), nRAlpha = 255 - (*pAlpha);
// mixing RGB
DWORD dwR = (GetRValue(clr2) * nAlpha + GetRValue(clr1) * nRAlpha) / 255.0 + 0.5;
DWORD dwG = (GetGValue(clr2) * nAlpha + GetGValue(clr1) * nRAlpha) / 255.0 + 0.5;
DWORD dwB = (GetBValue(clr2) * nAlpha + GetBValue(clr1) * nRAlpha) / 255.0 + 0.5;
COLORREF clrFinal = RGB(dwR, dwG, dwB);
img1.SetPixel(nX, nY, clrFinal);
}
}
img1.Save(sSave);
return sSave;
}
本文首发于:
https://my.hancel.org/2021/02/25/color-mixing-algorithm.html
歪日 师傅 受我一拜!(告诉PM 这个需求有必要?)
跳佬不愧是跳佬
牛掰 跳跳大佬