我一直在努力解决这个问题并且找不到任何解释!
背景
我正在使用VC9 + MFC在Windows上创建颜色管理照片编辑应用,并使用 WCS (Windows颜色系统)API转换照片嵌入颜色的像素配置文件到监视器的配置文件。
我的显示器已使用“Windows显示器校准”进行校准,并创建了一个名为“CalibratedDisplayProfile-x.icc”的配置文件。
问题:
当我将像素从“ProPhoto RGB”转换为显示器的配置文件时,暗区中的颜色会移动,色调会变为绿色。如果目标配置文件是sRGB,则在midtone / highlight中不会发生这种情况。这是屏幕截图。
测试
为了简化问题,我写了一些测试代码来翻译单一颜色,但测试结果确实让我感到困惑。源颜色“c0”是 RGB(0,0,65535),但输出颜色“c1”是 RGB(0,0,0) !!函数“CheckColor”失败,错误“无效参数”......
怎么会发生这种情况?我做错了吗?
您可以在此处下载两个配置文件:Color Profiles
非常感谢!
CString strProfilePath = _T("C:\\Windows\\System32\\spool\\drivers\\color\\");
CString strSrcProfile = strProfilePath + _T("ProPhoto.icm");
CString strDstProfile = strProfilePath + _T("CalibratedDisplayProfile-2.icc");
PROFILE pf = {0};
pf.dwType = PROFILE_FILENAME;
pf.pProfileData = (PVOID)strSrcProfile.GetBuffer();
pf.cbDataSize = (strSrcProfile.GetLength() + 1) * sizeof(TCHAR);
HPROFILE hSrcProfile = ::OpenColorProfile( &pf, PROFILE_READ, FILE_SHARE_READ, OPEN_EXISTING );
pf.pProfileData = (PVOID)strDstProfile.GetBuffer();
pf.cbDataSize = (strDstProfile.GetLength() + 1) * sizeof(TCHAR);
HPROFILE hDstProfile = ::OpenColorProfile( &pf, PROFILE_READ, FILE_SHARE_READ, OPEN_EXISTING );
HPROFILE hProfiles[2];
hProfiles[0] = hSrcProfile;
hProfiles[1] = hDstProfile;
DWORD dwIndents[2] = { INTENT_RELATIVE_COLORIMETRIC, INTENT_RELATIVE_COLORIMETRIC };
HTRANSFORM hTransform = ::CreateMultiProfileTransform( hProfiles, 2, dwIndents, 2, BEST_MODE, INDEX_DONT_CARE );
COLOR c0, c1;
c0.rgb.red = 0;
c0.rgb.green = 0;
c0.rgb.blue = 0xffff;
::TranslateColors( hTransform, &c0, 1, COLOR_RGB, &c1, COLOR_RGB );
BYTE btResult = 0;
::CheckColors( hTransform, &c0, 1, COLOR_RGB, &btResult );
::DeleteColorTransform( hTransform );
::CloseColorProfile( hSrcProfile );
::CloseColorProfile( hDstProfile );
答案 0 :(得分:0)
我在Windows 7 SP1 x64上遇到与您完全相同的问题。
似乎TranslateColors
函数要么被设计破坏,要么不应该以这种方式使用。我认为这是微软的错,因为他们可以在MSDN上写更多的WCS样本。
但我设法使用TranslateBitmapBits
函数来解决问题。这是一个示例:
bool translateColors(BYTE* srcRgbColors, BYTE* dstRgbColors, DWORD nBytes)
{
BOOL bResult = FALSE;
HPROFILE hSrcProfile = nullptr;
HPROFILE hDstProfile = nullptr;
HTRANSFORM hColorTransform = nullptr;
/* open source sRGB profile */
wchar_t* srcProfilePath = L"sRGB Color Space Profile.icm";
tagPROFILE targetProfile;
targetProfile.dwType = PROFILE_FILENAME;
targetProfile.pProfileData = srcProfilePath;
targetProfile.cbDataSize = sizeof(wchar_t) * (wcslen(srcProfilePath) + 1);
hSrcProfile = OpenColorProfile(&targetProfile, PROFILE_READ, FILE_SHARE_READ, OPEN_EXISTING);
if (nullptr == hSrcProfile) goto EXIT;
/* open destination monitor profile */
wchar_t* dstProfilePath = L"ActiveMonitorProfile.icm";
tagPROFILE destinationProfile;
destinationProfile.dwType = PROFILE_FILENAME;
destinationProfile.pProfileData = dstProfilePath;
destinationProfile.cbDataSize = sizeof(wchar_t) * (wcslen(dstProfilePath) + 1);
hDstProfile = OpenColorProfile(&destinationProfile, PROFILE_READ, FILE_SHARE_READ, OPEN_EXISTING);
if (nullptr == hDstProfile) goto EXIT;
/* create color transform */
DWORD dwIntent = (DWORD)-1;
HPROFILE hProfileList[2] = { hSrcProfile, hDstProfile };
hColorTransform = CreateMultiProfileTransform(
hProfileList,
2,
&dwIntent,
1,
NORMAL_MODE,
INDEX_DONT_CARE
);
if (nullptr == hColorTransform) goto EXIT;
/* transform colors */
DWORD dwWidth = nBytes / 3; // 3 channels per pixel, 8 bits per channel, RGB format
bResult = TranslateBitmapBits(
hColorTransform,
srcRgbColors,
BM_RGBTRIPLETS,
dwWidth, // bitmap width
1, // bitmap height
0,
dstRgbColors,
BM_RGBTRIPLETS,
0,
nullptr,
0
);
EXIT:
/* free resources */
if (nullptr != hColorTransform) {
DeleteColorTransform(hColorTransform);
}
if (nullptr != hSrcProfile) {
CloseColorProfile(hSrcProfile);
}
if (nullptr != hDstProfile) {
CloseColorProfile(hDstProfile);
}
return bResult == FALSE ? false : true;
}
/* example usage */
BYTE srcBitmapData[3];
srcBitmapData[0] = 0x1c;
srcBitmapData[1] = 0x1a;
srcBitmapData[2] = 0x1a;
BYTE dstOutputBitmapData[3];
bool bResult = traslateColors(srcBitmapData, dstOutputBitmapData, 3);
希望这有帮助。