461 lines
11 KiB
C++
461 lines
11 KiB
C++
|
// GUI.cpp
|
||
|
|
||
|
#include "StdAfx.h"
|
||
|
|
||
|
#include "../../../../C/Alloc.h"
|
||
|
|
||
|
#include "../../../Common/MyInitGuid.h"
|
||
|
|
||
|
#include "../../../Common/CommandLineParser.h"
|
||
|
#include "../../../Common/IntToString.h"
|
||
|
#include "../../../Common/MyException.h"
|
||
|
#include "../../../Common/StringConvert.h"
|
||
|
|
||
|
#include "../../../Windows/FileDir.h"
|
||
|
#include "../../../Windows/NtCheck.h"
|
||
|
#ifdef _WIN32
|
||
|
#include "../../../Windows/MemoryLock.h"
|
||
|
#endif
|
||
|
|
||
|
#include "../Common/ArchiveCommandLine.h"
|
||
|
#include "../Common/ExitCode.h"
|
||
|
|
||
|
#include "../FileManager/StringUtils.h"
|
||
|
#include "../FileManager/MyWindowsNew.h"
|
||
|
|
||
|
#include "BenchmarkDialog.h"
|
||
|
#include "ExtractGUI.h"
|
||
|
#include "HashGUI.h"
|
||
|
#include "UpdateGUI.h"
|
||
|
|
||
|
#include "ExtractRes.h"
|
||
|
|
||
|
using namespace NWindows;
|
||
|
|
||
|
HINSTANCE g_hInstance;
|
||
|
#ifndef _UNICODE
|
||
|
#endif
|
||
|
|
||
|
#if 0 // #ifndef UNDER_CE
|
||
|
|
||
|
DWORD g_ComCtl32Version;
|
||
|
|
||
|
static DWORD GetDllVersion(LPCTSTR dllName)
|
||
|
{
|
||
|
DWORD dwVersion = 0;
|
||
|
HINSTANCE hinstDll = LoadLibrary(dllName);
|
||
|
if (hinstDll)
|
||
|
{
|
||
|
DLLGETVERSIONPROC pDllGetVersion = (DLLGETVERSIONPROC)GetProcAddress(hinstDll, "DllGetVersion");
|
||
|
if (pDllGetVersion)
|
||
|
{
|
||
|
DLLVERSIONINFO dvi;
|
||
|
ZeroMemory(&dvi, sizeof(dvi));
|
||
|
dvi.cbSize = sizeof(dvi);
|
||
|
HRESULT hr = (*pDllGetVersion)(&dvi);
|
||
|
if (SUCCEEDED(hr))
|
||
|
dwVersion = MAKELONG(dvi.dwMinorVersion, dvi.dwMajorVersion);
|
||
|
}
|
||
|
FreeLibrary(hinstDll);
|
||
|
}
|
||
|
return dwVersion;
|
||
|
}
|
||
|
|
||
|
#endif
|
||
|
|
||
|
bool g_LVN_ITEMACTIVATE_Support = true;
|
||
|
|
||
|
static void ErrorMessage(LPCWSTR message)
|
||
|
{
|
||
|
MessageBoxW(NULL, message, L"7-Zip", MB_ICONERROR | MB_OK);
|
||
|
}
|
||
|
|
||
|
static void ErrorLangMessage(UINT resourceID)
|
||
|
{
|
||
|
ErrorMessage(LangString(resourceID));
|
||
|
}
|
||
|
|
||
|
static const char *kNoFormats = "7-Zip cannot find the code that works with archives.";
|
||
|
|
||
|
static int ShowMemErrorMessage()
|
||
|
{
|
||
|
ErrorLangMessage(IDS_MEM_ERROR);
|
||
|
return NExitCode::kMemoryError;
|
||
|
}
|
||
|
|
||
|
static int ShowSysErrorMessage(DWORD errorCode)
|
||
|
{
|
||
|
if (errorCode == E_OUTOFMEMORY)
|
||
|
return ShowMemErrorMessage();
|
||
|
ErrorMessage(HResultToMessage(errorCode));
|
||
|
return NExitCode::kFatalError;
|
||
|
}
|
||
|
|
||
|
static void ThrowException_if_Error(HRESULT res)
|
||
|
{
|
||
|
if (res != S_OK)
|
||
|
throw CSystemException(res);
|
||
|
}
|
||
|
|
||
|
static int Main2(int argc,TCHAR **argv)
|
||
|
{
|
||
|
UStringVector commandStrings;
|
||
|
#ifdef _WIN32
|
||
|
NCommandLineParser::SplitCommandLine(GetCommandLineW(), commandStrings);
|
||
|
#else
|
||
|
extern void mySplitCommandLineW(int numArguments,TCHAR **arguments,UStringVector &parts);
|
||
|
mySplitCommandLineW(argc,argv,commandStrings);
|
||
|
#endif
|
||
|
#ifndef UNDER_CE
|
||
|
if (commandStrings.Size() > 0)
|
||
|
commandStrings.Delete(0);
|
||
|
#endif
|
||
|
if (commandStrings.Size() == 0)
|
||
|
{
|
||
|
MessageBoxW(0, L"Specify command", L"7-Zip", 0);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
CArcCmdLineOptions options;
|
||
|
CArcCmdLineParser parser;
|
||
|
|
||
|
parser.Parse1(commandStrings, options);
|
||
|
parser.Parse2(options);
|
||
|
|
||
|
#if defined(_WIN32) && !defined(UNDER_CE)
|
||
|
NSecurity::EnablePrivilege_SymLink();
|
||
|
#ifdef _7ZIP_LARGE_PAGES
|
||
|
if (options.LargePages)
|
||
|
NSecurity::EnablePrivilege_LockMemory();
|
||
|
#endif
|
||
|
#endif
|
||
|
|
||
|
CREATE_CODECS_OBJECT
|
||
|
|
||
|
codecs->CaseSensitiveChange = options.CaseSensitiveChange;
|
||
|
codecs->CaseSensitive = options.CaseSensitive;
|
||
|
ThrowException_if_Error(codecs->Load());
|
||
|
|
||
|
bool isExtractGroupCommand = options.Command.IsFromExtractGroup();
|
||
|
|
||
|
if (codecs->Formats.Size() == 0 &&
|
||
|
(isExtractGroupCommand
|
||
|
|
||
|
|| options.Command.IsFromUpdateGroup()))
|
||
|
{
|
||
|
#ifdef EXTERNAL_CODECS
|
||
|
if (!codecs->MainDll_ErrorPath.IsEmpty())
|
||
|
{
|
||
|
UString s = L"7-Zip cannot load module ";
|
||
|
s += fs2us(codecs->MainDll_ErrorPath);
|
||
|
throw s;
|
||
|
}
|
||
|
#endif
|
||
|
throw kNoFormats;
|
||
|
}
|
||
|
|
||
|
CObjectVector<COpenType> formatIndices;
|
||
|
if (!ParseOpenTypes(*codecs, options.ArcType, formatIndices))
|
||
|
{
|
||
|
ErrorLangMessage(IDS_UNSUPPORTED_ARCHIVE_TYPE);
|
||
|
return NExitCode::kFatalError;
|
||
|
}
|
||
|
|
||
|
CIntVector excludedFormatIndices;
|
||
|
FOR_VECTOR (k, options.ExcludedArcTypes)
|
||
|
{
|
||
|
CIntVector tempIndices;
|
||
|
if (!codecs->FindFormatForArchiveType(options.ExcludedArcTypes[k], tempIndices)
|
||
|
|| tempIndices.Size() != 1)
|
||
|
{
|
||
|
ErrorLangMessage(IDS_UNSUPPORTED_ARCHIVE_TYPE);
|
||
|
return NExitCode::kFatalError;
|
||
|
}
|
||
|
excludedFormatIndices.AddToUniqueSorted(tempIndices[0]);
|
||
|
// excludedFormatIndices.Sort();
|
||
|
}
|
||
|
|
||
|
#ifdef EXTERNAL_CODECS
|
||
|
if (isExtractGroupCommand
|
||
|
|| options.Command.CommandType == NCommandType::kHash
|
||
|
|| options.Command.CommandType == NCommandType::kBenchmark)
|
||
|
ThrowException_if_Error(__externalCodecs.Load());
|
||
|
#endif
|
||
|
|
||
|
if (options.Command.CommandType == NCommandType::kBenchmark)
|
||
|
{
|
||
|
HRESULT res = Benchmark(EXTERNAL_CODECS_VARS_L options.Properties);
|
||
|
/*
|
||
|
if (res == S_FALSE)
|
||
|
{
|
||
|
stdStream << "\nDecoding Error\n";
|
||
|
return NExitCode::kFatalError;
|
||
|
}
|
||
|
*/
|
||
|
ThrowException_if_Error(res);
|
||
|
}
|
||
|
else if (isExtractGroupCommand)
|
||
|
{
|
||
|
UStringVector ArchivePathsSorted;
|
||
|
UStringVector ArchivePathsFullSorted;
|
||
|
|
||
|
CExtractCallbackImp *ecs = new CExtractCallbackImp;
|
||
|
CMyComPtr<IFolderArchiveExtractCallback> extractCallback = ecs;
|
||
|
|
||
|
#ifndef _NO_CRYPTO
|
||
|
ecs->PasswordIsDefined = options.PasswordEnabled;
|
||
|
ecs->Password = options.Password;
|
||
|
#endif
|
||
|
|
||
|
ecs->Init();
|
||
|
|
||
|
CExtractOptions eo;
|
||
|
(CExtractOptionsBase &)eo = options.ExtractOptions;
|
||
|
eo.StdInMode = options.StdInMode;
|
||
|
eo.StdOutMode = options.StdOutMode;
|
||
|
eo.YesToAll = options.YesToAll;
|
||
|
eo.TestMode = options.Command.IsTestCommand();
|
||
|
|
||
|
#ifndef _SFX
|
||
|
eo.Properties = options.Properties;
|
||
|
#endif
|
||
|
|
||
|
bool messageWasDisplayed = false;
|
||
|
|
||
|
#ifndef _SFX
|
||
|
CHashBundle hb;
|
||
|
CHashBundle *hb_ptr = NULL;
|
||
|
|
||
|
if (!options.HashMethods.IsEmpty())
|
||
|
{
|
||
|
hb_ptr = &hb;
|
||
|
ThrowException_if_Error(hb.SetMethods(EXTERNAL_CODECS_VARS_L options.HashMethods));
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
{
|
||
|
CDirItemsStat st;
|
||
|
HRESULT hresultMain = EnumerateDirItemsAndSort(
|
||
|
options.arcCensor,
|
||
|
NWildcard::k_RelatPath,
|
||
|
UString(), // addPathPrefix
|
||
|
ArchivePathsSorted,
|
||
|
ArchivePathsFullSorted,
|
||
|
st,
|
||
|
NULL // &scan: change it!!!!
|
||
|
);
|
||
|
if (hresultMain != S_OK)
|
||
|
{
|
||
|
/*
|
||
|
if (hresultMain != E_ABORT && messageWasDisplayed)
|
||
|
return NExitCode::kFatalError;
|
||
|
*/
|
||
|
throw CSystemException(hresultMain);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ecs->MultiArcMode = (ArchivePathsSorted.Size() > 1);
|
||
|
|
||
|
HRESULT result = ExtractGUI(codecs,
|
||
|
formatIndices, excludedFormatIndices,
|
||
|
ArchivePathsSorted,
|
||
|
ArchivePathsFullSorted,
|
||
|
options.Censor.Pairs.Front().Head,
|
||
|
eo,
|
||
|
#ifndef _SFX
|
||
|
hb_ptr,
|
||
|
#endif
|
||
|
options.ShowDialog, messageWasDisplayed, ecs);
|
||
|
if (result != S_OK)
|
||
|
{
|
||
|
if (result != E_ABORT && messageWasDisplayed)
|
||
|
return NExitCode::kFatalError;
|
||
|
throw CSystemException(result);
|
||
|
}
|
||
|
if (!ecs->IsOK())
|
||
|
return NExitCode::kFatalError;
|
||
|
}
|
||
|
else if (options.Command.IsFromUpdateGroup())
|
||
|
{
|
||
|
#ifndef _NO_CRYPTO
|
||
|
bool passwordIsDefined = options.PasswordEnabled && !options.Password.IsEmpty();
|
||
|
#endif
|
||
|
|
||
|
CUpdateCallbackGUI callback;
|
||
|
// callback.EnablePercents = options.EnablePercents;
|
||
|
|
||
|
#ifndef _NO_CRYPTO
|
||
|
callback.PasswordIsDefined = passwordIsDefined;
|
||
|
callback.AskPassword = options.PasswordEnabled && options.Password.IsEmpty();
|
||
|
callback.Password = options.Password;
|
||
|
#endif
|
||
|
|
||
|
// callback.StdOutMode = options.UpdateOptions.StdOutMode;
|
||
|
callback.Init();
|
||
|
|
||
|
if (!options.UpdateOptions.InitFormatIndex(codecs, formatIndices, options.ArchiveName) ||
|
||
|
!options.UpdateOptions.SetArcPath(codecs, options.ArchiveName))
|
||
|
{
|
||
|
ErrorLangMessage(IDS_UPDATE_NOT_SUPPORTED);
|
||
|
return NExitCode::kFatalError;
|
||
|
}
|
||
|
bool messageWasDisplayed = false;
|
||
|
HRESULT result = UpdateGUI(
|
||
|
codecs, formatIndices,
|
||
|
options.ArchiveName,
|
||
|
options.Censor,
|
||
|
options.UpdateOptions,
|
||
|
options.ShowDialog,
|
||
|
messageWasDisplayed,
|
||
|
&callback);
|
||
|
|
||
|
if (result != S_OK)
|
||
|
{
|
||
|
if (result != E_ABORT && messageWasDisplayed)
|
||
|
return NExitCode::kFatalError;
|
||
|
throw CSystemException(result);
|
||
|
}
|
||
|
if (callback.FailedFiles.Size() > 0)
|
||
|
{
|
||
|
if (!messageWasDisplayed)
|
||
|
throw CSystemException(E_FAIL);
|
||
|
return NExitCode::kWarning;
|
||
|
}
|
||
|
}
|
||
|
else if (options.Command.CommandType == NCommandType::kHash)
|
||
|
{
|
||
|
bool messageWasDisplayed = false;
|
||
|
HRESULT result = HashCalcGUI(EXTERNAL_CODECS_VARS_L
|
||
|
options.Censor, options.HashOptions, messageWasDisplayed);
|
||
|
|
||
|
if (result != S_OK)
|
||
|
{
|
||
|
if (result != E_ABORT && messageWasDisplayed)
|
||
|
return NExitCode::kFatalError;
|
||
|
throw CSystemException(result);
|
||
|
}
|
||
|
/*
|
||
|
if (callback.FailedFiles.Size() > 0)
|
||
|
{
|
||
|
if (!messageWasDisplayed)
|
||
|
throw CSystemException(E_FAIL);
|
||
|
return NExitCode::kWarning;
|
||
|
}
|
||
|
*/
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
throw "Unsupported command";
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
#define NT_CHECK_FAIL_ACTION ErrorMessage(L"Unsupported Windows version"); return NExitCode::kFatalError;
|
||
|
|
||
|
#ifdef _WIN32
|
||
|
int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE /* hPrevInstance */,
|
||
|
#ifdef UNDER_CE
|
||
|
LPWSTR
|
||
|
#else
|
||
|
LPSTR
|
||
|
#endif
|
||
|
/* lpCmdLine */, int /* nCmdShow */)
|
||
|
{
|
||
|
g_hInstance = hInstance;
|
||
|
#ifdef _WIN32
|
||
|
NT_CHECK
|
||
|
SetLargePageSize();
|
||
|
#endif
|
||
|
|
||
|
InitCommonControls();
|
||
|
|
||
|
#ifndef UNDER_CE
|
||
|
g_ComCtl32Version = ::GetDllVersion(TEXT("comctl32.dll"));
|
||
|
g_LVN_ITEMACTIVATE_Support = (g_ComCtl32Version >= MAKELONG(71, 4));
|
||
|
#endif
|
||
|
|
||
|
// OleInitialize is required for ProgressBar in TaskBar.
|
||
|
#ifndef UNDER_CE
|
||
|
OleInitialize(NULL);
|
||
|
#endif
|
||
|
|
||
|
LoadLangOneTime();
|
||
|
|
||
|
// setlocale(LC_COLLATE, ".ACP");
|
||
|
try
|
||
|
{
|
||
|
return Main2();
|
||
|
}
|
||
|
#else
|
||
|
int Main1(int argc,TCHAR **argv)
|
||
|
{
|
||
|
LoadLangOneTime();
|
||
|
|
||
|
// under MacOSX, a bundle does not keep the current directory
|
||
|
// between 7zFM and 7zG ...
|
||
|
// So, try to use the environment variable P7ZIP_CURRENT_DIR
|
||
|
const char *p7zip_current_dir = getenv("P7ZIP_CURRENT_DIR");
|
||
|
|
||
|
if (p7zip_current_dir)
|
||
|
{
|
||
|
UString currentDir = MultiByteToUnicodeString(p7zip_current_dir);
|
||
|
|
||
|
NWindows::NFile::NDir::SetCurrentDir(currentDir);
|
||
|
}
|
||
|
|
||
|
|
||
|
// setlocale(LC_COLLATE, ".ACP");
|
||
|
try
|
||
|
{
|
||
|
return Main2(argc,argv);
|
||
|
}
|
||
|
#endif
|
||
|
catch(const CNewException &)
|
||
|
{
|
||
|
return ShowMemErrorMessage();
|
||
|
}
|
||
|
catch(const CArcCmdLineException &e)
|
||
|
{
|
||
|
ErrorMessage(e);
|
||
|
return NExitCode::kUserError;
|
||
|
}
|
||
|
catch(const CSystemException &systemError)
|
||
|
{
|
||
|
if (systemError.ErrorCode == E_ABORT)
|
||
|
return NExitCode::kUserBreak;
|
||
|
return ShowSysErrorMessage(systemError.ErrorCode);
|
||
|
}
|
||
|
catch(const UString &s)
|
||
|
{
|
||
|
ErrorMessage(s);
|
||
|
return NExitCode::kFatalError;
|
||
|
}
|
||
|
catch(const AString &s)
|
||
|
{
|
||
|
ErrorMessage(GetUnicodeString(s));
|
||
|
return NExitCode::kFatalError;
|
||
|
}
|
||
|
catch(const wchar_t *s)
|
||
|
{
|
||
|
ErrorMessage(s);
|
||
|
return NExitCode::kFatalError;
|
||
|
}
|
||
|
catch(const char *s)
|
||
|
{
|
||
|
ErrorMessage(GetUnicodeString(s));
|
||
|
return NExitCode::kFatalError;
|
||
|
}
|
||
|
catch(int v)
|
||
|
{
|
||
|
wchar_t s[32];
|
||
|
ConvertUInt32ToString(v, s);
|
||
|
ErrorMessage(UString(L"Error: ") + s);
|
||
|
return NExitCode::kFatalError;
|
||
|
}
|
||
|
catch(...)
|
||
|
{
|
||
|
ErrorMessage(L"Unknown error");
|
||
|
return NExitCode::kFatalError;
|
||
|
}
|
||
|
}
|