corel_thumbnail.cpp 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  1. /*
  2. ** Copyright (C) http://srgb.googlecode.com All rights reserved.
  3. */
  4. #include "corel_thumbnail.h"
  5. #include <cstdio>
  6. #include <regex>
  7. #include <string>
  8. #include <atlimage.h>
  9. #define CASESENSITIVITY (0)
  10. #define WRITEBUFFERSIZE (8192)
  11. using std::string;
  12. bool zip_extract_currentfile(unzFile uf, LPCTSTR save_filename)
  13. {
  14. char szFilePathA[MAX_PATH];
  15. // 对于每个内部文件,可用 unzGetCurrentFileInfo64 来查内部文件名
  16. unz_file_info64 FileInfo;
  17. if (unzGetCurrentFileInfo64(uf, &FileInfo, szFilePathA, sizeof(szFilePathA), NULL, 0, NULL, 0) != UNZ_OK) {
  18. return false;
  19. }
  20. // printf("%s", szFilePathA); // 在压缩文件中的 文件名
  21. // 对于非目录的内部文件,用 unzOpenCurrentFile打开
  22. if (unzOpenCurrentFile(uf) != UNZ_OK) {
  23. return false;
  24. }
  25. HANDLE hFile = CreateFile(save_filename, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, 0, NULL);
  26. if (hFile == INVALID_HANDLE_VALUE) {
  27. return false;
  28. }
  29. // 然后 unzReadCurrentFile 读取文件内容,写入到真实文件中。
  30. // unzReadCurrentFile 返回 0 表示文件读取结束然后 unzReadCurrentFile 读取文件内容,写入到真实文件中。
  31. // unzReadCurrentFile 返回 0 表示文件读取结束
  32. char byBuffer[WRITEBUFFERSIZE];
  33. while (true) {
  34. int err = unzReadCurrentFile(uf, byBuffer, WRITEBUFFERSIZE);
  35. if (err < 0) {
  36. return false;
  37. } else if (err == 0) {
  38. break;
  39. } else {
  40. DWORD dwWritten = 0;
  41. if (!WriteFile(hFile, byBuffer, (DWORD)err, &dwWritten, NULL) || dwWritten != (DWORD)err) {
  42. return false;
  43. }
  44. }
  45. }
  46. CloseHandle(hFile);
  47. return true;
  48. }
  49. bool zip_extract_onefile(const char* zip_filename, const char* filename , const char* save_filename)
  50. {
  51. // 解压先使用 zipOpen64 来打开一个 ZIP 文件
  52. unzFile uf = unzOpen64(zip_filename);
  53. // 需要先使用 unzGetGlobalInfo64 来取得该文件的一些信息,来了解这个压缩包里一共包含了多少个文件,等等。
  54. unz_global_info64 gi;
  55. // if (unzGetGlobalInfo64(uf, &gi) != UNZ_OK) {
  56. // unzClose(uf);
  57. // return false;
  58. // }
  59. // 尝试zip文件中找到该文件szFileName。
  60. if (unzLocateFile(uf, filename, CASESENSITIVITY) != UNZ_OK) {
  61. // printf("file %s not found in the zipfile\n", filename);
  62. unzClose(uf);
  63. return false;
  64. }
  65. if (!zip_extract_currentfile(uf, save_filename)) {
  66. unzClose(uf);
  67. return false;
  68. }
  69. unzClose(uf);
  70. return true;
  71. }
  72. // 查阅: http://www.cppblog.com/Streamlet/archive/2013/05/12/127368.html
  73. /* 相关 API
  74. * 压缩相关:
  75. * zipOpen64 zipClose zipOpenNewFileInZip zipCloseFileInZip zipWriteInFileInZip
  76. *
  77. * 解压相关:
  78. * unzOpen64 unzClose unzGetGlobalInfo64 unzGoToNextFile
  79. * unzGetCurrentFileInfo64 unzOpenCurrentFile unzCloseCurrentFile unzReadCurrentFile
  80. *
  81. * 首先是压缩操作。使用 zipOpen64 来打开/创建一个 ZIP 文件,然后开始遍历要被放到压缩包中去的文件。
  82. * 针对每个文件,先调用一次 zipOpenNewFileInZip,然后开始读原始文件数据,使用 zipWriteInFileInZip 来写入到 ZIP 文件中去。
  83. * zipOpenNewFileInZip 的第三个参数是一个 zip_fileinfo 结构,该结构数据可全部置零,其中 dosDate 可用于填入一个时间(LastModificationTime)。
  84. * 它的第二个参数是 ZIP 中的文件名,若要保持目录结构,该参数中可以保留路径,如 foo/bar.txt。
  85. *
  86. * 解压操作稍微复杂一点点。打开一个 ZIP 文件后,需要先使用 unzGetGlobalInfo64 来取得该文件的一些信息,来了解这个压缩包里一共包含了多少个文件,等等。
  87. * 目前我们用得着的就是这个文件数目。然后开始遍历 ZIP 中的文件,初始时自动会定位在第一个文件,以后处理完一个后用 unzGoToNextFile 来跳到下一个文件。
  88. * 对于每个内部文件,可用 unzGetCurrentFileInfo64 来查内部文件名。这个文件名和刚才 zipOpenNewFileInZip 的第二个参数是一样的形式,所以有可能包含路径。
  89. * 也有可能会以路径分隔符(/)结尾,表明这是个目录项(其实压缩操作的时候也可以针对目录写入这样的内部文件,上面没有做)。所以接下来要根据情况创建(多级)目录。
  90. * unzGetCurrentFileInfo64 的第三个参数是 unz_file_info64 结构,其中也有一项包含了 dosDate 信息,可以还原文件时间。
  91. * 对于非目录的内部文件,用 unzOpenCurrentFile,打开,然后 unzReadCurrentFile 读取文件内容,写入到真实文件中。unzReadCurrentFile 返回 0 表示文件读取结束。
  92. *
  93. */
  94. bool cdr_extract_bmp(const char* cdr_filename, const char* bmp_filename)
  95. {
  96. if (! IsFileExist(cdr_filename))
  97. return false ;
  98. bool ret = false;
  99. int version = get_cdrfile_version(cdr_filename);
  100. // printf("\n%d\n", version);
  101. if (version >= 1400 ){
  102. ret = zip_extract_onefile(cdr_filename, "metadata/thumbnails/thumbnail.bmp", bmp_filename);
  103. if (!ret) ret = zip_extract_onefile(cdr_filename, "previews/thumbnail.png", bmp_filename);
  104. }
  105. if (version <= 1300 && version > 1)
  106. ret = cdr_riff_disp2bmp(cdr_filename, bmp_filename);
  107. return ret;
  108. }
  109. bool CorelThumbnail(const char* cdr_filename, const char* png_filename)
  110. {
  111. string file_ext(cdr_filename);
  112. string rs = "(.+)(\\.(?:cdr|CDR|Cdr|CDr|cdR))";
  113. std::regex expression(rs);
  114. bool ret = regex_match(file_ext, expression);
  115. if (!ret) {
  116. return ret ;
  117. }
  118. FILE* pfile = fopen(cdr_filename, "rb");
  119. if(NULL == pfile){
  120. // MessageBoxA(NULL, cdr_filename, "错误File", MB_OK);
  121. return false;
  122. }
  123. char temp_filename[128];
  124. const char* tmpBmpFile = tmpnam(temp_filename);
  125. strcat(temp_filename,".bmp");
  126. ret = cdr_extract_bmp(cdr_filename , tmpBmpFile);
  127. if (!ret)
  128. return false ;
  129. CImage image; // bmp 转换 png ,需要CImage类,头文件 atlimage.h
  130. image.Load(tmpBmpFile);
  131. image.Save(png_filename);
  132. if (remove(tmpBmpFile) != 0)
  133. perror("Error deleting file");
  134. return ret;
  135. }
  136. bool CorelThumbnail_W(const wchar_t* cdr_filename, const wchar_t* png_filename)
  137. {
  138. char fromfile[MAX_PATH] = {0};
  139. char tofile[MAX_PATH] = {0};
  140. WCHARTochar(fromfile, cdr_filename);
  141. WCHARTochar(tofile, png_filename);
  142. bool ret = CorelThumbnail(fromfile, tofile);
  143. return ret;
  144. }