gpt4 book ai didi

c++ - 使用 L_SaveFileOffset 保存多页 TIFF 文件

转载 作者:太空宇宙 更新时间:2023-11-04 13:22:56 27 4
gpt4 key购买 nike

我需要使用 L_SaveFileOffset 保存多页 TIFF 文件,因为我需要确保没有其他进程(包括 Windows 本身)可以在保存页面的过程中访问该文件,据我所知L_SaveFileOffsetLeadTools 中唯一用于保存图像的 API,它允许使用文件句柄保存图像。问题是,无论我做什么,都只会保存最后一页。请帮忙。

HANDLE hFile = ::CreateFile(L"ColorMaps.tif", GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);

if(NULL != hFile)
{
const wchar_t pathTemplate[] = {L"ColorMap%d.bmp"};
wchar_t tPath[sizeof(pathTemplate) / sizeof(pathTemplate[0])];
FILEINFO PageInfo;
SAVEFILEOPTION so;
LOADFILEOPTION tlo;
int i;
HDC hDc;
BITMAPHANDLE tBmp;
__int64 tSize;

memset(&tlo, 0, sizeof(LOADFILEOPTION));
tlo.uStructSize = sizeof(LOADFILEOPTION);
L_GetDefaultLoadFileOption(&tlo, sizeof(LOADFILEOPTION));
tlo.Flags |= ELO_ROTATED;
hDc = ::GetDC(NULL);
tlo.XResolution = ::GetDeviceCaps(hDc, LOGPIXELSX);
tlo.YResolution = ::GetDeviceCaps(hDc, LOGPIXELSY);
::ReleaseDC(NULL, hDc);
memset(&so, 0, sizeof(SAVEFILEOPTION));
so.uStructSize = sizeof(SAVEFILEOPTION);
so.Flags = ESO_INSERTPAGE;

memset(&tBmp, 0, sizeof(BITMAPHANDLE));
tBmp.uStructSize = sizeof(BITMAPHANDLE);
for(i = 1; i < 7; i++)
{
::StringCbPrintf(tPath, sizeof(tPath), pathTemplate, i);
L_FileInfo(tPath, &PageInfo, sizeof(FILEINFO), 0, &tlo);
L_LoadBitmap(tPath, &tBmp, sizeof(BITMAPHANDLE), 0, ORDER_RGBORGRAY, &tlo, &PageInfo);
if (TOP_LEFT != tBmp.ViewPerspective)
L_ChangeBitmapViewPerspective(NULL, &tBmp, sizeof(BITMAPHANDLE), TOP_LEFT);
L_SaveFileOffset((L_HFILE)hFile, 0, &tSize, &tBmp, FILE_TIF_PACKBITS, PageInfo.BitsPerPixel, 0, SAVEFILE_MULTIPAGE, NULL, NULL, &so);
so.PageNumber = i + 1;
}
::CloseHandle(hFile);
}

以上只是一个示例,可能有 1000 页被保存到 TIFF 文件中。

如果用户打开 Windows Explorer 并导航到保存文件的目录,问题就会自行显现,您可以看到 Windows 正在尝试重新绘制每个页面之间的文件图标保存,如果我使用 L_SaveBitmapL_SaveFile,有时它们会返回 -14 因为 Windows 正在读取文件并且 < strong>LeadTools 无法锁定它。

附言L_SaveFileOffset 为所有页面返回 1(SUCCESS),我使用的是 LeadTools Documents imaging 版本 17.5。。

谢谢

最佳答案

山姆,
完全控制文件句柄的一种方法是使用重定向 IO。锁定整个 6 页的 TIFF 文件所需的代码如下所示:

// 2 global variables
L_HFILE hFileMyTiff = NULL;
bool reallyClose = false;
L_HFILE EXT_CALLBACK MyOpen(L_TCHAR* pFile, L_INT nMode, L_INT nShare, L_VOID* pUserData)
{
if(!hFileMyTiff) //only open the file if it's not already open
hFileMyTiff = (L_HFILE)::CreateFile(pFile, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
return hFileMyTiff;
}
L_UINT EXT_CALLBACK MyRead(L_HFILE FD, L_UCHAR * pBuf, L_UINT uCount, L_VOID *pUserData)
{
DWORD dwRead = 0;
::ReadFile(FD, pBuf, uCount, &dwRead, NULL);
return dwRead;
}
L_UINT EXT_CALLBACK MyWrite(L_HFILE FD, L_UCHAR * pBuf, L_UINT uCount, L_VOID* pUserData)
{
DWORD dwWritten = 0;
::WriteFile(FD, pBuf, uCount, &dwWritten, NULL);
return dwWritten;
}
L_SSIZE_T EXT_CALLBACK MySeek(L_HFILE FD, L_SSIZE_T nPos, L_INT nOrigin, L_VOID* pUserData)
{
return ::SetFilePointer(FD, nPos, NULL, nOrigin);
}
L_INT EXT_CALLBACK MyClose (L_HFILE FD, L_VOID* pUserData)
{
if(reallyClose)
{
::CloseHandle(FD);
hFileMyTiff = NULL;
}
else
::SetFilePointer(FD, 0, NULL, FILE_BEGIN);
return TRUE;
}
void tst()
{
const wchar_t pathTemplate[] = {L"ColorMap%d.bmp"};
wchar_t tPath[sizeof(pathTemplate) / sizeof(pathTemplate[0])];
FILEINFO PageInfo;
LOADFILEOPTION tlo;
int i;
HDC hDc;
BITMAPHANDLE tBmp;
__int64 tSize;

memset(&tlo, 0, sizeof(LOADFILEOPTION));
tlo.uStructSize = sizeof(LOADFILEOPTION);
L_GetDefaultLoadFileOption(&tlo, sizeof(LOADFILEOPTION));
tlo.Flags |= ELO_ROTATED;
hDc = ::GetDC(NULL);
tlo.XResolution = ::GetDeviceCaps(hDc, LOGPIXELSX);
tlo.YResolution = ::GetDeviceCaps(hDc, LOGPIXELSY);
::ReleaseDC(NULL, hDc);

memset(&tBmp, 0, sizeof(BITMAPHANDLE));
tBmp.uStructSize = sizeof(BITMAPHANDLE);
for(i = 1; i < 7; i++)
{
::StringCbPrintf(tPath, sizeof(tPath), pathTemplate, i);
L_FileInfo(tPath, &PageInfo, sizeof(FILEINFO), 0, &tlo);
L_LoadBitmap(tPath, &tBmp, sizeof(BITMAPHANDLE), 0, ORDER_RGBORGRAY, &tlo, &PageInfo);
if (TOP_LEFT != tBmp.ViewPerspective)
L_ChangeBitmapViewPerspective(NULL, &tBmp, sizeof(BITMAPHANDLE), TOP_LEFT);

if(i==6) //only allow closing after the last page is saved
reallyClose = true;
else
reallyClose = false;
L_RedirectIO(MyOpen, MyRead, MyWrite, MySeek, MyClose, NULL); //use our own file I/O functions
L_SaveFile(L"ColorMaps.tif", &tBmp, FILE_TIF_PACKBITS, PageInfo.BitsPerPixel, 0, SAVEFILE_MULTIPAGE, NULL, NULL, NULL);
L_RedirectIO(NULL, NULL, NULL, NULL, NULL, NULL); //reset to default I/O so as not affect loading
}
}

请注意为简单起见使用了 2 个全局变量。一种更简洁的方法可能是定义您自己的数据结构并将其地址传递到 L_RedirectIO 的 pUserData 参数中。
我们已经使用 v17.5 测试了此解决方案并验证其工作正常。如果它不适合您的特定情况,请联系 LEADTOOLS 支持服务,我们将讨论其他替代方案。

更新:最新版本的 LEADTOOLS v19 现在支持使用 L_SaveFileOffset() 函数将页面附加到 TIFF 文件。

关于c++ - 使用 L_SaveFileOffset 保存多页 TIFF 文件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34426133/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com