From 895d580db207ab387082c7a31cd19921345ac718 Mon Sep 17 00:00:00 2001 From: Sergei Vorobyov Date: Thu, 16 Feb 2023 02:27:20 +0300 Subject: [PATCH] Fix a bug when heic files doubled when browsing a folder --- src/JPEGView/FileList.cpp | 126 ++++++++++++++++------------------ src/JPEGView/JPEGView.vcxproj | 4 +- src/JPEGView/stdafx.h | 3 +- 3 files changed, 63 insertions(+), 70 deletions(-) diff --git a/src/JPEGView/FileList.cpp b/src/JPEGView/FileList.cpp index 3b9e1d69..e0346549 100644 --- a/src/JPEGView/FileList.cpp +++ b/src/JPEGView/FileList.cpp @@ -5,6 +5,7 @@ #include "DirectoryWatcher.h" #include "Shlwapi.h" #include "LepLoader.h" +#include /////////////////////////////////////////////////////////////////////////////////// // Helpers @@ -124,19 +125,12 @@ void CFileDesc::SetModificationDate(const FILETIME& lastModDate) { // Public interface /////////////////////////////////////////////////////////////////////////////////// -// image file types supported internally (there are additional endings for RAW and WIC - these come from INI file) -// NOTE: when adding more supported filetypes, update installer to add another extension for "SupportedTypes" -static const int cnNumEndingsInternal = 14; -static const TCHAR* csFileEndingsInternal[cnNumEndingsInternal] = {_T("jpg"), _T("jpeg"), _T("bmp"), _T("png"), - _T("tif"), _T("tiff"), _T("gif"), _T("webp"), _T("jxl"), _T("avif"), _T("heif"), _T("heic"), _T("tga"), _T("qoi")}; +// Image file types supported internally and come from INI file. +typedef std::set file_endings_type; + // supported camera RAW formats static const TCHAR* csFileEndingsRAW = _T("*.pef;*.dng;*.crw;*.nef;*.cr2;*.mrw;*.rw2;*.orf;*.x3f;*.arw;*.kdc;*.nrw;*.dcr;*.sr2;*.raf"); - -static const int MAX_ENDINGS = 48; -static int nNumEndings; -static LPCTSTR* sFileEndings; - __declspec(dllimport) bool __stdcall WICPresent(void); // Check if Windows Image Codecs library is present @@ -149,48 +143,41 @@ static bool WICPresentGuarded(void) { } // Parses a semicolon separated list of file endings of the form "*.nef;*.cr2;*.dng" -static void ParseAndAddFileEndings(LPCTSTR sEndings) { - if (_tcslen(sEndings) > 2) { - LPTSTR buffer = new TCHAR[_tcslen(sEndings) + 1]; // this buffer will not be freed deliberately! - _tcscpy(buffer, sEndings); - LPTSTR sStart = buffer, sCurrent = buffer; - while (*sCurrent != 0 && nNumEndings < MAX_ENDINGS) { - while (*sCurrent != 0 && *sCurrent != _T(';')) { - sCurrent++; - } - if (*sCurrent == _T(';')) { - *sCurrent = 0; - sCurrent++; - } - if (_tcslen(sStart) > 2) { - sFileEndings[nNumEndings++] = sStart + 2; - } - sStart = sCurrent; - } +template +static void ParseAndAddFileEndings(const std::wstring& input, OUT_ITER output) { + if (input.size() > 2) + { + std::wstringstream ss(input); + std::wstring item; + while (std::getline(ss, item, L';')) + output = item.substr(2); } } // Gets all supported file endings, including the ones from WIC. // The length of the returned list is nNumEndings -static LPCTSTR* GetSupportedFileEndingList() { - if (sFileEndings == NULL) { - sFileEndings = new LPCTSTR[MAX_ENDINGS]; - for (nNumEndings = 0; nNumEndings < cnNumEndingsInternal; nNumEndings++) { - sFileEndings[nNumEndings] = csFileEndingsInternal[nNumEndings]; - } +static file_endings_type& GetSupportedFileEndingList() +{ + static file_endings_type file_endings; + if (!file_endings.empty()) + return file_endings; - LPCTSTR sFileEndingsWIC = CSettingsProvider::This().FilesProcessedByWIC(); - if (_tcslen(sFileEndingsWIC) > 2 && WICPresentGuarded()) { - ParseAndAddFileEndings(sFileEndingsWIC); - } + // Gets all supported file endings, including the ones from WIC. + file_endings = { _T("jpg"), _T("jpeg"), _T("bmp"), _T("png"), + _T("tif"), _T("tiff"), _T("gif"), _T("webp"), _T("jxl"), _T("avif"), _T("heif"), _T("heic"), _T("tga"), _T("qoi") }; - if (LepLoader::LeptonToolPresent()) { - ParseAndAddFileEndings(CSettingsProvider::This().FilesProcessedByLepton()); - } + LPCTSTR sFileEndingsWIC = CSettingsProvider::This().FilesProcessedByWIC(); + if (_tcslen(sFileEndingsWIC) > 2 && WICPresentGuarded()) { + ParseAndAddFileEndings(sFileEndingsWIC, std::inserter(file_endings, file_endings.end())); + } - ParseAndAddFileEndings(CSettingsProvider::This().FileEndingsRAW()); + if (LepLoader::LeptonToolPresent()) { + ParseAndAddFileEndings(CSettingsProvider::This().FilesProcessedByLepton(), std::inserter(file_endings, file_endings.end())); } - return sFileEndings; + + ParseAndAddFileEndings(CSettingsProvider::This().FileEndingsRAW(), std::inserter(file_endings, file_endings.end())); + + return file_endings; } CFileList::CFileList(const CString & sInitialFile, CDirectoryWatcher & directoryWatcher, @@ -212,7 +199,7 @@ CFileList::CFileList(const CString & sInitialFile, CDirectoryWatcher & directory bool bImageFile = !bIsDirectory && IsImageFile(sExtensionInitialFile); if (!bIsDirectory && !bImageFile && !sExtensionInitialFile.IsEmpty() && _tcsstr(csFileEndingsRAW, _T("*.") + sExtensionInitialFile) != NULL) { // initial file is a supported camera raw file but was excluded in INI file - add temporarily to file ending list - ParseAndAddFileEndings(_T("*.") + sExtensionInitialFile); + ParseAndAddFileEndings((LPCWSTR)(_T("*.") + sExtensionInitialFile), std::inserter(GetSupportedFileEndingList(), GetSupportedFileEndingList().end())); CSettingsProvider::This().AddTemporaryRAWFileEnding(sExtensionInitialFile); bImageFile = true; } @@ -248,15 +235,22 @@ CFileList::~CFileList() { m_fileList.clear(); } -CString CFileList::GetSupportedFileEndings() { - LPCTSTR* allFileEndings = GetSupportedFileEndingList(); - CString sList; - for (int i = 0; i < nNumEndings; i++) { - sList += _T("*."); - sList += allFileEndings[i]; - if (i+1 < nNumEndings) sList += _T(";"); +CString CFileList::GetSupportedFileEndings() +{ + auto extensions = GetSupportedFileEndingList(); + std::wostringstream str; + bool first = true; + for each (auto extension in extensions) + { + if (!first) + str << L";"; + else + first = false; + + str << L"*." << extension; } - return sList; + + return CString(str.str().c_str()); } void CFileList::Reload(LPCTSTR sFileName, bool clearForwardHistory) { @@ -822,12 +816,14 @@ void CFileList::FindFiles() { m_fileList.clear(); if (!m_sDirectory.IsEmpty()) { CFindFile fileFind; - LPCTSTR* allFileEndings = GetSupportedFileEndingList(); - for (int i = 0; i < nNumEndings; i++) { - if (fileFind.FindFile(m_sDirectory + _T("\\*.") + allFileEndings[i])) { - AddToFileList(m_fileList, fileFind, allFileEndings[i]); + auto extensions = GetSupportedFileEndingList(); + for each(auto extension in extensions) + { + const auto c_extension = extension.c_str(); + if (fileFind.FindFile(m_sDirectory + _T("\\*.") + c_extension)) { + AddToFileList(m_fileList, fileFind, c_extension); while (fileFind.FindNextFile()) { - AddToFileList(m_fileList, fileFind, allFileEndings[i]); + AddToFileList(m_fileList, fileFind, c_extension); } } } @@ -851,13 +847,8 @@ void CFileList::VerifyFiles() { bool CFileList::IsImageFile(const CString & sEnding) { CString sEndingLC = sEnding; sEndingLC.MakeLower(); - LPCTSTR* allFileEndings = GetSupportedFileEndingList(); - for (int i = 0; i < nNumEndings; i++) { - if (allFileEndings[i] == sEndingLC) { - return true; - } - } - return false; + auto allFileEndings = GetSupportedFileEndingList(); + return allFileEndings.find((LPCWSTR)sEndingLC) != allFileEndings.end(); } bool CFileList::TryReadingSlideShowList(const CString & sSlideShowFile) { @@ -972,9 +963,10 @@ bool CFileList::TryReadingSlideShowList(const CString & sSlideShowFile) { pStart--; } else { // try to find file name with relative path - LPCTSTR* allFileEndings = GetSupportedFileEndingList(); - for (int i = 0; i < nNumEndings; i++) { - pStart = Helpers::stristr(lineBuff, (LPCTSTR)(CString(_T(".")) + allFileEndings[i])); + auto extensions = GetSupportedFileEndingList(); + for each (auto extension in extensions) + { + pStart = Helpers::stristr(lineBuff, (LPCTSTR)(CString(_T(".")) + extension.c_str())); if (pStart != NULL) { while (pStart >= lineBuff && *pStart != _T('"') && *pStart != _T('>')) pStart--; pStart++; diff --git a/src/JPEGView/JPEGView.vcxproj b/src/JPEGView/JPEGView.vcxproj index cda37c5c..aaed4ea9 100644 --- a/src/JPEGView/JPEGView.vcxproj +++ b/src/JPEGView/JPEGView.vcxproj @@ -112,7 +112,7 @@ EditAndContinue true false - stdcpp17 + stdcpp20 _DEBUG;%(PreprocessorDefinitions) @@ -285,7 +285,7 @@ Windows false WICLoader.dll;jxl_dec.dll;jxl_threads.dll;heif.dll;avif.dll;lcms2.dll - UseFastLinkTimeCodeGeneration + UseLinkTimeCodeGeneration /IGNORE:4099 %(AdditionalOptions) diff --git a/src/JPEGView/stdafx.h b/src/JPEGView/stdafx.h index 6decc5a9..599cdcc3 100644 --- a/src/JPEGView/stdafx.h +++ b/src/JPEGView/stdafx.h @@ -38,7 +38,8 @@ extern CAppModule _Module; #pragma warning(pop) // STL stuff -#include +#include +#include #define _SILENCE_STDEXT_HASH_DEPRECATION_WARNINGS