Преобразование цвета из ProPhoto RGB в профиль монитора

Я много дней боролся с этой проблемой и не могу найти никакого объяснения!

Фон:

Я создаю цвет управляемый приложение для редактирования фотографий в Windows с VC9 + MFC и использование WCS(Windows Color System) API-интерфейсы для преобразования пикселей из встроенного цветового профиля фотографии в профиль монитора.

Мой монитор был откалиброван с помощью «Калибровки дисплея Windows» и создал профиль с именем «CalibratedDisplayProfile-x.icc».

Проблема:

Когда я конвертирую пиксели из «ProPhoto RGB» в профиль монитора, цвет в темной области смещается, оттенок становится зеленым. Это не происходит в средних тональных сигналах / выделениях, если целевой профиль sRGB. Вот скриншоты.

правильный & изображения ошибок

Тестовое задание:

Чтобы упростить задачу, я написал несколько тестовых кодов для перевода одного цвета, но результат теста действительно запутывает меня. Исходный цвет «c0» RGB (0,0,65535), но выходной цвет «с1» RGB (0,0,0)!! И функция «CheckColor» завершается с ошибкой «Invalid Arguments» …

Как такое могло произойти? Я делаю что-то неправильно?

Вы можете скачать два профиля здесь: Цветовые профили

Спасибо большое!

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 );

2

Решение

У меня была та же проблема, что и у вас на Windows 7 SP1 x64.
Кажется, что TranslateColors Функция либо нарушена по замыслу, либо не должна использоваться таким образом. Я думаю, что это вина Microsoft, поскольку они могли бы написать больше образцов WCS на MSDN.

Но мне удалось решить проблему с помощью 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);

Надеюсь это поможет.

0

Другие решения

Других решений пока нет …

По вопросам рекламы [email protected]