OSDN Git Service

C++11のauto, nullptrを導入。
[tpi/lychee.git] / src / lychee / frm_main.cpp
index f37ae79..3962438 100644 (file)
@@ -45,15 +45,12 @@ bool g_fSortAscend;
 // MainFrame\r
 //******************************************************************************\r
 \r
-MainFrame::MainFrame(): wxFrame()\r
+MainFrame::MainFrame(): wxFrame(), gFrame(nullptr)\r
 {\r
 }\r
 \r
 MainFrame::~MainFrame()\r
 {\r
-       wxCommandEvent e;\r
-       this->OnArcClose(e);\r
-\r
        // 設定を記録。\r
        if (! this->IsIconized() && ! this->IsMaximized())\r
        {\r
@@ -65,7 +62,12 @@ MainFrame::~MainFrame()
                this->conf.WriteId(CONF_WINDOW_X, a);\r
                this->conf.WriteId(CONF_WINDOW_Y, b);\r
        }\r
-       this->conf.WriteId(CONF_WINDOW_SPLITTER_POS, this->window_splitter->GetSashPosition());\r
+\r
+       // ウインドウが表示されていないときは、GetSashPosition()が不正な値を返す。\r
+       if (this->IsShown())\r
+       {\r
+               this->conf.WriteId(CONF_WINDOW_SPLITTER_POS, this->window_splitter->GetSashPosition());\r
+       }\r
 \r
        // ツールバー/ステータスバー関連。\r
        this->conf.WriteId(CONF_WINDOW_STATUSBAR, this->statusbar->IsShown());\r
@@ -80,7 +82,7 @@ MainFrame::~MainFrame()
                this->conf.WriteId(CONF_LISTVIEW_C_PACKED,   this->list_ctrl->GetColumnWidth(2));\r
                this->conf.WriteId(CONF_LISTVIEW_C_RATIO,    this->list_ctrl->GetColumnWidth(3));\r
                this->conf.WriteId(CONF_LISTVIEW_C_METHOD,   this->list_ctrl->GetColumnWidth(4));\r
-               this->conf.WriteId(CONF_LISTVIEW_C_ATTR,         this->list_ctrl->GetColumnWidth(5));\r
+               this->conf.WriteId(CONF_LISTVIEW_C_ATTR,     this->list_ctrl->GetColumnWidth(5));\r
                this->conf.WriteId(CONF_LISTVIEW_C_LASTMOD,  this->list_ctrl->GetColumnWidth(6));\r
                this->conf.WriteId(CONF_LISTVIEW_C_PATH,     this->list_ctrl->GetColumnWidth(7));\r
                this->conf.WriteId(CONF_LISTVIEW_C_TYPE,     this->list_ctrl->GetColumnWidth(8));\r
@@ -90,7 +92,10 @@ MainFrame::~MainFrame()
                this->conf.WriteId(CONF_LISTVIEW_S_ASCEND,   g_fSortAscend);\r
        }\r
 \r
-       this->Close(true);\r
+       wxCommandEvent e;\r
+       delete this->gFrame;\r
+       this->Show(false);\r
+       this->OnArcClose(e);\r
 }\r
 \r
 //******************************************************************************\r
@@ -98,11 +103,14 @@ MainFrame::~MainFrame()
 //******************************************************************************\r
 \r
 BEGIN_EVENT_TABLE(MainFrame, wxFrame)\r
-       EVT_INIT_DIALOG(      MainFrame::OnInit)\r
+       EVT_INIT_DIALOG(MainFrame::OnInit)\r
+       EVT_SIZE(       MainFrame::OnSize)\r
+       EVT_CLOSE(      MainFrame::OnClose)\r
        // Menu\r
        EVT_MENU(XRCID("Arc_Create"),  MainFrame::OnArcCreate)\r
        EVT_MENU(XRCID("Arc_Open"),    MainFrame::OnArcOpen)\r
        EVT_MENU(XRCID("Arc_Close"),   MainFrame::OnArcClose)\r
+       EVT_MENU(XRCID("Arc_Clone"),   MainFrame::OnArcClone)\r
        EVT_MENU(XRCID("Arc_Add"),     MainFrame::OnArcAdd)\r
        EVT_MENU(XRCID("Arc_SFX"),     MainFrame::OnArcConvert)\r
        EVT_MENU(XRCID("Arc_UnSFX"),   MainFrame::OnArcConvert)\r
@@ -121,9 +129,13 @@ BEGIN_EVENT_TABLE(MainFrame, wxFrame)
        // TreeView\r
        EVT_TREE_SEL_CHANGED(XRCID("TreeView"), MainFrame::OnTreeChanged)\r
        EVT_TREE_BEGIN_DRAG( XRCID("TreeView"), MainFrame::OnTreeBeginDrag)\r
+       EVT_COMMAND_CONTEXT_MENU(XRCID("TreeView"), MainFrame::OnContextMenu)\r
        // ListView\r
-       EVT_LIST_ITEM_ACTIVATED(XRCID("ListView"), MainFrame::OnListItemDClick)\r
-       EVT_LIST_BEGIN_DRAG(    XRCID("ListView"), MainFrame::OnListBeginDrag)\r
+       EVT_LIST_ITEM_SELECTED(  XRCID("ListView"), MainFrame::OnListItemSelect)\r
+       EVT_LIST_ITEM_DESELECTED(XRCID("ListView"), MainFrame::OnListItemSelect)\r
+       EVT_LIST_ITEM_ACTIVATED( XRCID("ListView"), MainFrame::OnListItemDClick)\r
+       EVT_LIST_BEGIN_DRAG(     XRCID("ListView"), MainFrame::OnListBeginDrag)\r
+       EVT_COMMAND_CONTEXT_MENU(XRCID("ListView"), MainFrame::OnContextMenu)\r
        // Filter\r
        EVT_TEXT(XRCID("tcFilter"), MainFrame::OnFilter)\r
 END_EVENT_TABLE()\r
@@ -190,6 +202,33 @@ void MainFrame::OnInit(wxInitDialogEvent&)
        fShow = this->conf.ReadId(CONF_WINDOW_TOOLBAR, true);\r
        this->menubar->Check(XRCID("Exe_View_ToolBar"), fShow);\r
        this->toolbar->Show(fShow);\r
+\r
+       // ステータスバー上にプログレスバーを作成。\r
+       this->gFrame = new wxGauge(this->statusbar, -1, 0, wxPoint(0, 0), wxSize(200, 10));\r
+       this->gFrame->Show(false);\r
+}\r
+\r
+void MainFrame::OnClose(wxCloseEvent& e)\r
+{\r
+       if (e.CanVeto() && g_procDlg != nullptr)\r
+       {\r
+               g_procDlg->OnClose(e);\r
+               e.Veto();\r
+       }\r
+       else\r
+       {\r
+               this->Destroy();\r
+       }\r
+}\r
+\r
+void MainFrame::OnSize(wxSizeEvent& e)\r
+{\r
+       // プログレスバーの位置を変更。\r
+       if (this->gFrame != nullptr)\r
+       {\r
+               this->gFrame->SetPosition(wxPoint(this->statusbar->GetSize().GetWidth() - 230, 10));\r
+       }\r
+       e.Skip();\r
 }\r
 \r
 // MenuBar\r
@@ -202,36 +241,49 @@ void MainFrame::OnExit(wxCommandEvent&)
 void MainFrame::OnArcCreate(wxCommandEvent& e)\r
 {\r
        TPI_SWITCHES swInfo;\r
-       swInfo.pCustomSwitches = NULL;\r
+       swInfo.pCustomSwitches = nullptr;\r
 \r
        // 作成ダイアログを設定。\r
-       MakeDialog mkDlg;\r
-       ::wxXmlResource::Get()->Load(L_DIR_S_XRC wxT("dlg_make.xrc"));\r
-       ::wxXmlResource::Get()->LoadDialog(& mkDlg, this, wxT("dlg_make"));\r
-\r
-       if (e.GetClientData() == NULL)\r
+       MakeDialog mkDlg(this, TPI_COMMAND_CREATE);\r
+       if (e.GetClientData() == nullptr)\r
        {\r
-               // 処理対象のファイルを選択。\r
-               wxFileDialog fd(this, _("Choose files to compress"), this->conf.ReadHistory(CONF_HISTORY_PATH, 0));\r
-               fd.SetWindowStyleFlag(wxFD_OPEN | wxFD_FILE_MUST_EXIST | wxFD_MULTIPLE);\r
-               if (fd.ShowModal() == wxID_CANCEL)\r
+               if (::wxGetKeyState(WXK_SHIFT))\r
                {\r
-                       return;\r
+                       // 処理対象のフォルダを選択。\r
+                       wxDirDialog dd(this, _("Choose dir to compress"), this->conf.ReadHistory(CONF_HISTORY_PATH, 0), wxDD_DIR_MUST_EXIST);\r
+                       if (dd.ShowModal() == wxID_CANCEL)\r
+                       {\r
+                               return;\r
+                       }\r
+                       swInfo.fnDestinationDirectory = wxFileName(dd.GetPath());\r
+                       this->conf.WriteHistory(CONF_HISTORY_PATH, swInfo.fnDestinationDirectory.GetPath());\r
+                       mkDlg.files.Add(swInfo.fnDestinationDirectory.GetFullName());\r
+               }\r
+               else\r
+               {\r
+                       // 処理対象のファイルを選択。\r
+                       wxFileDialog fd(this, _("Choose files to compress"), this->conf.ReadHistory(CONF_HISTORY_PATH, 0), wxEmptyString, wxFileSelectorDefaultWildcardStr, wxFD_OPEN | wxFD_FILE_MUST_EXIST | wxFD_MULTIPLE);\r
+                       if (fd.ShowModal() == wxID_CANCEL)\r
+                       {\r
+                               return;\r
+                       }\r
+                       fd.GetFilenames(mkDlg.files);\r
+                       swInfo.fnDestinationDirectory = wxFileName::DirName(fd.GetDirectory());\r
+                       this->conf.WriteHistory(CONF_HISTORY_PATH, fd.GetDirectory());\r
                }\r
-               swInfo.fnDestinationDirectory = wxFileName::DirName(fd.GetDirectory());\r
-               this->conf.WriteHistory(CONF_HISTORY_PATH, fd.GetDirectory());\r
-               fd.GetFilenames(mkDlg.files);\r
        }\r
        else\r
        {\r
                mkDlg.files = * (wxArrayString *) e.GetClientData();\r
                swInfo.fnDestinationDirectory = wxFileName::DirName(wxFileName(mkDlg.files[0]).GetPath());\r
                // 相対パスに変換。\r
+//             for (auto s : mkDlg.files)\r
                for (size_t n = 0; n < mkDlg.files.GetCount(); n++)\r
                {\r
-                       wxFileName fn(mkDlg.files[n]);\r
+                       wxString & s = mkDlg.files[n];\r
+                       wxFileName fn(s);\r
                        fn.MakeRelativeTo(swInfo.fnDestinationDirectory.GetPath());\r
-                       mkDlg.files[n] = fn.GetFullPath();\r
+                       s = fn.GetFullPath();\r
                }\r
        }\r
 \r
@@ -249,8 +301,10 @@ void MainFrame::OnArcCreate(wxCommandEvent& e)
        }\r
 \r
        // 各種設定。\r
+       int nSelected = mkDlg.chType->GetSelection();\r
+       swInfo.nArchiveType         = mkDlg.afInfo[nSelected].nTypeId;\r
        swInfo.fStoreDirectoryPathes= ! mkDlg.cbIgnorePath->IsChecked();\r
-       swInfo.fMakeSFX                     = mkDlg.cbMakeSFX->IsChecked();\r
+       swInfo.fMakeSFX             = mkDlg.cbMakeSFX->IsChecked();\r
        swInfo.fSolid               = mkDlg.cbSolid->IsChecked();\r
        swInfo.fMMOptimize          = mkDlg.cbMMOptimize->IsChecked();\r
        swInfo.fEncryptHeader       = mkDlg.cbEncryptHeader->IsChecked();\r
@@ -263,34 +317,24 @@ void MainFrame::OnArcCreate(wxCommandEvent& e)
        mkDlg.cbSplitSize->GetValue().ToULongLong(& swInfo.nSplitSize);\r
 \r
        // TPIを読み込み。\r
-       int nSelected = mkDlg.chType->GetSelection();\r
        this->fnArchive = wxFileName(mkDlg.cbDir->GetValue(), mkDlg.cbFileName->GetValue());\r
-       if (! tpi.InitLibrary(mkDlg.afInfo[nSelected].szTPIName, this->fnArchive.GetFullPath(), mkDlg.afInfo[nSelected].nTypeId))\r
+       swInfo.szArcName            = this->fnArchive.GetFullPath();\r
+       if (! tpi.InitLibrary(mkDlg.afInfo[nSelected].szTPIName, this->fnArchive.GetFullPath(), TPICallbackProc, mkDlg.afInfo[nSelected].nTypeId))\r
        {\r
                this->ErrorCheck(tpi.nErrorCode, wxT("InitLibrary"));\r
                return;\r
        }\r
 \r
-       // コールバック関数を設定。\r
-       tpi.SetCallbackProc(TPICallbackProc);\r
-       this->ErrorCheck(tpi.nErrorCode, wxT("SetCallbackProc"));\r
-\r
        // 処理を行う。\r
        {\r
-               ProcessDialog procDlg;\r
-               procDlg.Show(true);\r
-\r
-               // コールバックを送信。\r
-               TPI_PROCESSINFO piInfo;\r
-               piInfo.eMessage = TPI_MESSAGE_STATUS;\r
-               piInfo.eStatus = 0x1000;\r
-               piInfo.fiInfo.fnFileName = this->fnArchive;\r
-               piInfo.fiInfo.nUnpackedSize = mkDlg.files.GetCount();\r
-               procDlg.CallbackProc(TPI_NOTIFY_COMMON, & piInfo);\r
-\r
-               tpi.Command(TPI_COMMAND_CREATE, & swInfo, this->fnArchive.GetFullPath(), mkDlg.files);\r
-               this->ErrorCheck(tpi.nErrorCode);\r
+               ProcessDialog procDlg(this, mkDlg.files.GetCount());\r
+               tpi.Command(TPI_COMMAND_CREATE, & swInfo, mkDlg.files);\r
                procDlg.Show(false);\r
+               if (this->ErrorCheck(tpi.nErrorCode) != TPI_ERROR_SUCCESS)\r
+               {\r
+                       tpi.FreeLibrary();\r
+                       return;\r
+               }\r
        }\r
        tpi.FreeLibrary();\r
 \r
@@ -307,17 +351,22 @@ void MainFrame::OnArcCreate(wxCommandEvent& e)
        }\r
 \r
        // 終了しない場合は書庫を開く。\r
-       this->OnArcOpen(e);\r
+       if (this->fnArchive.FileExists())\r
+       {\r
+               e.SetInt(1);\r
+               this->OnArcOpen(e);\r
+       }\r
 }\r
 \r
 void MainFrame::OnArcOpen(wxCommandEvent& e)\r
 {\r
+       // モード取得。通常は0, それ以外で開く場合は1, ファイルDnDなら2。\r
+       int nMode = e.GetInt();\r
+\r
        // 書庫を選択。\r
-       if (e.GetId() == XRCID("Arc_Open"))\r
+       if (nMode == 0)\r
        {\r
-               wxFileDialog fd(this);\r
-               fd.SetDirectory(this->conf.ReadHistory(CONF_HISTORY_PATH, 0));\r
-               fd.SetWindowStyleFlag(wxFD_OPEN | wxFD_FILE_MUST_EXIST);\r
+               wxFileDialog fd(this, _("Choose an archive"), this->conf.ReadHistory(CONF_HISTORY_PATH, 0), wxEmptyString, wxFileSelectorDefaultWildcardStr, wxFD_OPEN | wxFD_FILE_MUST_EXIST);\r
                if (fd.ShowModal() == wxID_CANCEL)\r
                {\r
                        return;\r
@@ -326,18 +375,18 @@ void MainFrame::OnArcOpen(wxCommandEvent& e)
                this->fnArchive = wxFileName(fd.GetPath());\r
        }\r
 \r
-       // 進捗ダイアログ表示。\r
-       ProcessDialog procDlg;\r
-       procDlg.Show(true);\r
-\r
-       // DnD以外で書庫を開く場合、TPIを読み込み。\r
+       // TPIを読み込み。\r
+       ProcessDialog procDlg(this);\r
        TPI_PROCESSINFO piInfo;\r
-       piInfo.fiInfo.nUnpackedSize = e.GetExtraLong();\r
-       if (e.GetInt() != 1 && ! this->LoadTPI(this->fnArchive.GetFullPath(), & piInfo.fiInfo.nUnpackedSize))\r
+       piInfo.fiInfo.nUnpackedSize = 0;\r
+       if (! this->LoadTPI(this->fnArchive.GetFullPath(), & piInfo.fiInfo.nUnpackedSize))\r
        {\r
-               procDlg.Show(false);\r
-               tpi.FreeLibrary();\r
-               if (this->IsShown())\r
+               if (nMode == 2)\r
+               {\r
+                       // DnDの場合は書庫を作成する。\r
+                       this->OnArcCreate(e);\r
+               }\r
+               else if (this->IsShown())\r
                {\r
                        wxBell();\r
                        this->statusbar->SetStatusText(_("No plug-in supporting this archive was found!"), 4);\r
@@ -350,12 +399,22 @@ void MainFrame::OnArcOpen(wxCommandEvent& e)
                return;\r
        }\r
 \r
-       // 最初のコールバックを送信。\r
+       // コマンドラインから書庫を開く場合にはウインドウが表示されていないので表示。\r
+       if (! this->IsShown())\r
+       {\r
+               this->Show();\r
+       }\r
+\r
        piInfo.eMessage = TPI_MESSAGE_STATUS;\r
-       piInfo.eStatus = 0x1000;\r
-       piInfo.nProcessedSize = 0;\r
-       piInfo.fiInfo.fnFileName = this->fnArchive;\r
-       procDlg.CallbackProc(TPI_NOTIFY_COMMON, & piInfo);\r
+       piInfo.eStatus = 0x1001;\r
+       if (this->ErrorCheck(procDlg.CallbackProc(TPI_NOTIFY_COMMON, & piInfo), wxT("Callback")) == TPI_CALLBACK_CANCEL)\r
+       {\r
+               procDlg.Show(false);\r
+               tpi.CloseArchive();\r
+               // OnArcClose処理はProcessDialogのデストラクタで行う。\r
+               // this->OnArcClose(e);\r
+               return;\r
+       }\r
 \r
        // 配列のサイズを確保。\r
        this->fileinfo.Alloc(piInfo.fiInfo.nUnpackedSize);\r
@@ -370,54 +429,102 @@ void MainFrame::OnArcOpen(wxCommandEvent& e)
        g_hIconT.Add(wxBitmap(L_DIR_S_ICO wxT("folder_open.png"), wxBITMAP_TYPE_ANY));\r
        this->tree_ctrl->SetImageList(& g_hIconT);\r
        wxTreeItemId\r
-               idRoot = this->tree_ctrl->AddRoot(wxEmptyString),\r
-#ifdef __WINDOWS__\r
-               idArchive = this->tree_ctrl->AppendItem(idRoot, piInfo.fiInfo.fnFileName.GetFullName(), g_hIconT.Add(GetFileTypeIcon(piInfo.fiInfo.fnFileName))),\r
-#else\r
-               idArchive = this->tree_ctrl->AppendItem(idRoot, piInfo.fiInfo.fnFileName.GetFullName(), g_hIconT.Add(GetFileTypeIcon(piInfo.fiInfo.fnFileName).ConvertToImage().Rescale(16, 16))),\r
-#endif\r
+               idRoot    = this->tree_ctrl->AddRoot(wxEmptyString),\r
+               idArchive = this->tree_ctrl->AppendItem(idRoot, this->fnArchive.GetFullName(), g_hIconT.Add(myRescaleIcon(GetFileTypeIcon(this->fnArchive)))),\r
                idArcRoot = this->tree_ctrl->AppendItem(idRoot, wxT("-----"), 0, 1);\r
 \r
+       // 巨大書庫のときにファイル名検査を省略するか。\r
+       bool bDTVCheck = true;//piInfo.fiInfo.nUnpackedSize < 10000 || AskDlg(_("This archive contains so many files that it takes long to check Directory Traversal Vulnerability(DTV) problem. If you are sure this archive is safe, you can skip this scanning process. Do you want to scan for DTV problem?"), this) == wxNO;\r
+\r
        // ファイル情報をロード。\r
        if (tpi.GetFileInformation(& piInfo.fiInfo, true))\r
        {\r
-               piInfo.eStatus = 0x1001;\r
+               piInfo.eStatus = 0x1002;\r
+               piInfo.nProcessedSize = 0;\r
                do\r
                {\r
                        piInfo.nProcessedSize++;\r
-                       if (this->ErrorCheck(procDlg.CallbackProc(TPI_NOTIFY_COMMON, & piInfo)) == TPI_CALLBACK_CANCEL)\r
+                       if (this->ErrorCheck(procDlg.CallbackProc(TPI_NOTIFY_COMMON, & piInfo), wxT("Callback")) == TPI_CALLBACK_CANCEL)\r
                        {\r
                                procDlg.Show(false);\r
                                tpi.CloseArchive();\r
-                               wxCommandEvent e;\r
-                               this->OnArcClose(e);\r
+                               // OnArcClose処理はProcessDialogのデストラクタで行う。\r
+                               // this->OnArcClose(e);\r
                                return;\r
                        }\r
 \r
-                       // ツリービューに反映。\r
-                       bool fDir = piInfo.fiInfo.dwAttribute & TPI_ATTRIBUTE_DIRECTORY;\r
-                       TreeView_CheckNewerItem(this->tree_ctrl, idArcRoot, fDir ? piInfo.fiInfo.fnFileName.GetFullPath() : piInfo.fiInfo.fnFileName.GetPath(), true);\r
+                       // 情報を保存してカウントアップ。\r
+                       this->fileinfo.Add(piInfo.fiInfo);\r
+               }\r
+               while (tpi.GetFileInformation(& piInfo.fiInfo));\r
+       }\r
 \r
-                       // セキュリティチェック。\r
+       // GetFileInformationがエラー終了した場合。\r
+       this->ErrorCheck(tpi.nErrorCode, wxT("GetFileInformation"));\r
+\r
+//     for (auto f : this->fileinfo)\r
+       for (size_t i = 0; i < this->fileinfo.GetCount(); i++)\r
+       {\r
+               TPI_FILEINFO & f = this->fileinfo[i];\r
+               // 拡張子のみ設定されている場合。\r
+               if (f.szStoredName.IsEmpty())\r
+               {\r
+                       f.szStoredName = this->fnArchive.GetName();\r
+                       if (f.fnFileName.HasExt())\r
+                       {\r
+                               f.szStoredName += wxT('.') + f.fnFileName.GetExt();\r
+                       }\r
+                       f.fnFileName = wxFileName(f.szStoredName);\r
+               }\r
+\r
+               // セキュリティチェック。\r
+               // ルート記号を削除。\r
+               wxString szPath = f.fnFileName.GetPathWithSep(wxPATH_UNIX);\r
+               if (szPath.StartsWith(wxT("/")) || szPath.StartsWith(wxT("./")))\r
+               {\r
+                       f.fnFileName = wxFileName(szPath.AfterFirst(wxT('/')), f.fnFileName.GetFullName(), wxPATH_DOS);\r
+               }\r
+               // ルートメンバの情報は無視する。\r
+               if (f.fnFileName.GetFullPath().IsEmpty() || f.fnFileName.GetFullPath() == wxT("."))\r
+               {\r
+                       continue;\r
+               }\r
+\r
+               // 改行文字/タブ文字などを削除。\r
+               if (f.szStoredName.Find(wxT('\r')) != wxNOT_FOUND\r
+               ||  f.szStoredName.Find(wxT('\n')) != wxNOT_FOUND\r
+               ||  f.szStoredName.Find(wxT('\t')) != wxNOT_FOUND)\r
+               {\r
+                       wxString sz = f.fnFileName.GetFullPath();\r
+                       sz.Replace(wxT("\r"), wxT(" "));\r
+                       sz.Replace(wxT("\n"), wxT(" "));\r
+                       sz.Replace(wxT("\t"), wxT(" "));\r
+                       f.eDanger = TRUE;\r
+                       f.fnFileName = wxFileName(sz);\r
+                       wxLogWarning(_("This archive may contain files whose name contains some special characters like CR(\\r), LF(\\n), Tab(\\t) and some problem would be happen if you extract these files. Don\'t extract these files carelessly.\nDanger file is:\n%s"), f.fnFileName.GetFullPath().c_str());\r
+               }\r
+\r
+               if (bDTVCheck)\r
+               {\r
                        // DTV検査。\r
-                       if (piInfo.fiInfo.fnFileName.GetPathWithSep(wxPATH_UNIX).Find(wxT("../")) != wxNOT_FOUND)\r
+                       if (f.fnFileName.GetPathWithSep(wxPATH_UNIX).Find(wxT("../")) != wxNOT_FOUND)\r
                        {\r
-                               piInfo.fiInfo.eDanger = TRUE;\r
-                               wxLogWarning(_("This archive may have Directory Traversal Vulnerability(DTV) problem, and some danger files may be extracted to the unexpected system directory! You should use the \"Ignore file pathes\" option when extracting this archive.\nDanger file is:\n%s"), piInfo.fiInfo.fnFileName.GetFullPath().c_str());\r
+                               f.eDanger = TRUE;\r
+                               wxLogWarning(_("This archive may have Directory Traversal Vulnerability(DTV) problem, and some danger files may be extracted to the unexpected system directory! You should use the \"Ignore file pathes\" option when extracting this archive.\nDanger file is:\n%s"), f.fnFileName.GetFullPath().c_str());\r
                        }\r
                        // 空白の連続による拡張子偽装を検査。\r
-                       if (piInfo.fiInfo.fnFileName.GetFullName().Find(wxT("        ")) != wxNOT_FOUND)\r
+                       if (f.fnFileName.GetFullName().Find(wxT("        ")) != wxNOT_FOUND)\r
                        {\r
-                               piInfo.fiInfo.eDanger = TRUE;\r
-                               wxLogWarning(_("This archive may contain extension-disguised files whose real extension is hidden by using many blank charactor and you may mistake that it is a \"safe\" file. Don\'t execute these files carelessly.\nDanger file is:\n%s"), piInfo.fiInfo.fnFileName.GetFullPath().c_str());\r
+                               f.eDanger = TRUE;\r
+                               wxLogWarning(_("This archive may contain extension-disguised files whose real extension is hidden by using many blank charactor and you may mistake that it is a \"safe\" file. Don\'t execute these files carelessly.\nDanger file is:\n%s"), f.fnFileName.GetFullPath().c_str());\r
                        }\r
                        // Unicode制御文字を検査。\r
                        for (wxChar c = 0x200c; c <= 0x206f; c++)\r
                        {\r
-                               if (piInfo.fiInfo.fnFileName.GetFullName().Find(c) != wxNOT_FOUND)\r
+                               if (f.fnFileName.GetFullName().Find(c) != wxNOT_FOUND)\r
                                {\r
-                                       piInfo.fiInfo.eDanger = TRUE;\r
-                                       wxLogWarning(_("This archive may contain extension-disguised files whose real extension is hidden by using Unicode control character and you may mistake that it is a \"safe\" file. Don\'t execute these files carelessly.\nDanger file is:\n%s"), piInfo.fiInfo.fnFileName.GetFullPath().c_str());\r
+                                       f.eDanger = TRUE;\r
+                                       wxLogWarning(_("This archive may contain extension-disguised files whose real extension is hidden by using Unicode control character and you may mistake that it is a \"safe\" file. Don\'t execute these files carelessly.\nDanger file is:\n%s"), f.fnFileName.GetFullPath().c_str());\r
                                }\r
                                switch (c)\r
                                {\r
@@ -425,35 +532,47 @@ void MainFrame::OnArcOpen(wxCommandEvent& e)
                                case 0x202e: c = 0x2060; break;\r
                                }\r
                        }\r
+               }\r
 \r
-                       // ディレクトリ属性を含むものについては情報を保存しない。\r
-                       if (fDir)\r
-                       {\r
-                               continue;\r
-                       }\r
+               // ツリービューに反映。\r
+               bool fDir = f.dwAttribute & TPI_ATTRIBUTE_DIRECTORY ? true : false;\r
+               TreeView_CheckNewerItem(this->tree_ctrl, idArcRoot, fDir ? f.fnFileName.GetFullPath() : f.fnFileName.GetPath(), true);\r
 \r
-                       // 情報を保存してカウントアップ。\r
-                       this->fileinfo.Add(piInfo.fiInfo);\r
+               // ディレクトリ属性を含むものについては情報を保存しない。\r
+               if (fDir)\r
+               {\r
+                       continue;\r
                }\r
-               while (tpi.GetFileInformation(& piInfo.fiInfo));\r
        }\r
 \r
-       // GetFileInformationがエラー終了した場合。\r
-       this->ErrorCheck(tpi.nErrorCode, wxT("GetFileInformation"));\r
-\r
        // 書庫の情報を取得。\r
        tpi.GetArchiveInformation(& this->aiArchive);\r
        this->ErrorCheck(tpi.nErrorCode, wxT("GetArchiveInformation"));\r
+       this->szPassword = procDlg.szPassword;\r
 \r
-       // 書庫を閉じる。\r
-       tpi.CloseArchive();\r
-       this->ErrorCheck(tpi.nErrorCode, wxT("CloseArchive"));\r
+       // Command()でハンドルを用いない場合は、ここで書庫を閉じる。\r
+       if (! tpi.bHandleOnCommand)\r
+       {\r
+               tpi.CloseArchive();\r
+               this->ErrorCheck(tpi.nErrorCode, wxT("CloseArchive"));\r
+       }\r
 \r
        // 以下、UI処理。\r
        this->fileinfo.Shrink();\r
-       this->tree_ctrl->ExpandAll();\r
+//     if (this->fileinfo.Count() < 10000)\r
+       {\r
+               this->tree_ctrl->ExpandAll();\r
+               this->tree_ctrl->SelectItem(idArchive);\r
+       }\r
+/*     else\r
+       {\r
+               // ファイル数が多いとソートに時間がかかるので、書庫のルートを表示させる。\r
+               this->tree_ctrl->Expand(idArcRoot);\r
+               this->tree_ctrl->SelectItem(idArcRoot);\r
+       }*/\r
+       // ツリービューの位置合わせ。\r
        this->tree_ctrl->ScrollTo(idArchive);\r
-       this->tree_ctrl->SelectItem(idArchive);\r
+       this->tree_ctrl->SetScrollPos(wxHORIZONTAL, 0);\r
        this->list_ctrl->atDangerItem.SetTextColour(* wxRED);\r
        this->list_ctrl->atEncryptedItem.SetTextColour(wxColour(wxT("forest green")));\r
 \r
@@ -463,16 +582,30 @@ void MainFrame::OnArcOpen(wxCommandEvent& e)
        this->statusbar->SetStatusText(wxString::Format(wxString("%" wxLongLongFmtSpec "uB -> %" wxLongLongFmtSpec "uB"), this->aiArchive.nUnpackedSize, this->aiArchive.nPackedSize), 2);\r
        this->statusbar->SetStatusText(wxString::Format(wxT("%3.1f%%"), this->aiArchive.wCompressRatio / 10.0), 3);\r
        this->statusbar->SetStatusText(this->fnArchive.GetFullPath(), 4);\r
-\r
-       // ツールバー・メニューバー設定。\r
+       this->statusbar->SetToolTip(\r
+               wxString::Format(\r
+                       _("%s(%s)\nTPI: %s(%s)\nAccess: %s\nModify: %s\nCreate: %s\nComment:\n%s"),\r
+                       this->fnArchive.GetFullName().c_str(), this->aiArchive.fiInfo.szTypeName.c_str(),\r
+                       this->aiArchive.fiInfo.szTPIName.c_str(), this->aiArchive.fiInfo.szEngineName.c_str(),\r
+                       this->aiArchive.tmAccess.Format(_("%Y/%m/%d %H:%M:%S")).c_str(),\r
+                       this->aiArchive.tmModify.Format(_("%Y/%m/%d %H:%M:%S")).c_str(),\r
+                       this->aiArchive.tmCreate.Format(_("%Y/%m/%d %H:%M:%S")).c_str(),\r
+                       this->aiArchive.szComment.c_str()\r
+               )\r
+       );\r
+\r
+       // ツールバー・メニューバー設定。ファイル選択時しか動作しない削除などは別に設定。\r
        SetMenuToolState("Arc_Close",   true);\r
-       SetMenuToolState("Arc_Add",     (this->aiArchive.fiInfo.eSupportedCommand & TPI_COMMAND_ADD)   == TPI_COMMAND_ADD   && this->aiArchive.fiInfo.fArchive);\r
+       SetMenuToolState("Arc_Add",     (this->aiArchive.fiInfo.eSupportedCommand & TPI_COMMAND_ADD)    == TPI_COMMAND_ADD   &&   this->aiArchive.fiInfo.fArchive);\r
        SetMenuToolState("Arc_SFX",     (this->aiArchive.fiInfo.eSupportedCommand & TPI_COMMAND_SFX)    == TPI_COMMAND_SFX   && ! this->aiArchive.fSFX);\r
-       SetMenuToolState("Arc_UnSFX",   (this->aiArchive.fiInfo.eSupportedCommand & TPI_COMMAND_UNSFX)  == TPI_COMMAND_UNSFX && this->aiArchive.fSFX);\r
+       SetMenuToolState("Arc_UnSFX",   (this->aiArchive.fiInfo.eSupportedCommand & TPI_COMMAND_UNSFX)  == TPI_COMMAND_UNSFX &&   this->aiArchive.fSFX);\r
        SetMenuToolState("Arc_Extract", (this->aiArchive.fiInfo.eSupportedCommand & TPI_COMMAND_EXTRACT)== TPI_COMMAND_EXTRACT);\r
-       SetMenuToolState("Arc_Delete",  (this->aiArchive.fiInfo.eSupportedCommand & TPI_COMMAND_DELETE) == TPI_COMMAND_DELETE && this->aiArchive.fiInfo.fArchive);\r
        SetMenuToolState("Arc_Test",    (this->aiArchive.fiInfo.eSupportedCommand & TPI_COMMAND_TEST)   == TPI_COMMAND_TEST);\r
        SetMenuToolState("Arc_Repair",  (this->aiArchive.fiInfo.eSupportedCommand & TPI_COMMAND_REPAIR) == TPI_COMMAND_REPAIR);\r
+       this->menubar->Enable(XRCID("Arc_Clone"), true);\r
+\r
+       // タイトルバー設定。\r
+       this->SetTitle(this->fnArchive.GetFullName() + wxT(" - Lychee"));\r
 \r
        procDlg.Show(false);\r
        this->Raise();\r
@@ -480,6 +613,11 @@ void MainFrame::OnArcOpen(wxCommandEvent& e)
 \r
 void MainFrame::OnArcClose(wxCommandEvent& e)\r
 {\r
+       if (tpi.bHandleOnCommand)\r
+       {\r
+               tpi.CloseArchive();\r
+       }\r
+\r
        // ツリービュー・リストビュー設定。\r
        this->tree_ctrl->DeleteAllItems();\r
        this->list_ctrl->DeleteAllItems();\r
@@ -494,12 +632,15 @@ void MainFrame::OnArcClose(wxCommandEvent& e)
        SetMenuToolState("Arc_Delete",  false);\r
        SetMenuToolState("Arc_Test",    false);\r
        SetMenuToolState("Arc_Repair",  false);\r
+       this->menubar->Enable(XRCID("Arc_Clone"), false);\r
 \r
        for (int i = 0; i < this->statusbar->GetFieldsCount(); i++)\r
        {\r
                this->statusbar->SetStatusText(wxEmptyString, i);\r
        }\r
+       this->statusbar->SetToolTip(wxEmptyString);\r
        this->fileinfo.Clear();\r
+       this->szPassword.Empty();\r
        this->aiArchive.szComment.Empty();\r
        this->aiArchive.fnArchive.Clear();\r
        this->aiArchive.fiInfo.szTypeName.Empty();\r
@@ -511,6 +652,9 @@ void MainFrame::OnArcClose(wxCommandEvent& e)
        g_hIconLL.RemoveAll();\r
        g_hIconLS.RemoveAll();\r
 \r
+       // タイトルバー設定。\r
+       this->SetTitle(wxT("Lychee"));\r
+\r
        // DnDで書庫を開くときは既に読み込まれているTPIを用いるので、解放してはいけない。\r
        if (e.GetExtraLong() == 0)\r
        {\r
@@ -518,24 +662,74 @@ void MainFrame::OnArcClose(wxCommandEvent& e)
        }\r
 }\r
 \r
-void MainFrame::OnArcAdd(wxCommandEvent& e)\r
+void MainFrame::OnArcClone(wxCommandEvent&)\r
 {\r
-       // 作成ダイアログを設定。\r
-       MakeDialog mkDlg;\r
-       ::wxXmlResource::Get()->Load(L_DIR_S_XRC wxT("dlg_make.xrc"));\r
-       ::wxXmlResource::Get()->LoadDialog(& mkDlg, this, wxT("dlg_make"));\r
-       mkDlg.uCommand = TPI_COMMAND_ADD;\r
-\r
-       // 処理対象のファイルを選択。\r
-       wxFileDialog fd(this, _("Choose files to add"), this->conf.ReadHistory(CONF_HISTORY_PATH, 0));\r
-       fd.SetWindowStyleFlag(wxFD_OPEN | wxFD_FILE_MUST_EXIST | wxFD_MULTIPLE);\r
+       // 保存先を尋ねる。\r
+       wxFileDialog fd(this, _("Clone archive"), this->fnArchive.GetPath(), this->fnArchive.GetFullName(), wxFileSelectorDefaultWildcardStr, wxFD_SAVE | wxFD_OVERWRITE_PROMPT);\r
        if (fd.ShowModal() == wxID_CANCEL)\r
        {\r
                return;\r
        }\r
-       fd.GetFilenames(mkDlg.files);\r
        this->conf.WriteHistory(CONF_HISTORY_PATH, fd.GetDirectory());\r
 \r
+       // コピー。\r
+       ::wxCopyFile(this->fnArchive.GetFullPath(), fd.GetPath());\r
+       wxFileName fn(fd.GetPath());\r
+       wxDateTime dtAccess, dtModify, dtCreate;\r
+       this->fnArchive.GetTimes(& dtAccess, & dtModify, & dtCreate);\r
+       fn.SetTimes(& dtAccess, & dtModify, & dtCreate);\r
+}\r
+\r
+void MainFrame::OnArcAdd(wxCommandEvent& e)\r
+{\r
+       // 作成ダイアログを設定。\r
+       MakeDialog mkDlg(this, TPI_COMMAND_ADD);\r
+\r
+       TPI_SWITCHES swInfo;\r
+       swInfo.pCustomSwitches       = nullptr;\r
+\r
+       if (e.GetClientData() == nullptr)\r
+       {\r
+               if (::wxGetKeyState(WXK_SHIFT))\r
+               {\r
+                       // 処理対象のフォルダを選択。\r
+                       wxDirDialog dd(this, _("Choose dir to add"), this->conf.ReadHistory(CONF_HISTORY_PATH, 0), wxDD_DIR_MUST_EXIST);\r
+                       if (dd.ShowModal() == wxID_CANCEL)\r
+                       {\r
+                               return;\r
+                       }\r
+                       swInfo.fnDestinationDirectory = wxFileName(dd.GetPath());\r
+                       this->conf.WriteHistory(CONF_HISTORY_PATH, swInfo.fnDestinationDirectory.GetPath());\r
+                       mkDlg.files.Add(swInfo.fnDestinationDirectory.GetFullName());\r
+               }\r
+               else\r
+               {\r
+                       // 処理対象のファイルを選択。\r
+                       wxFileDialog fd(this, _("Choose files to add"), this->conf.ReadHistory(CONF_HISTORY_PATH, 0), wxEmptyString, wxFileSelectorDefaultWildcardStr, wxFD_OPEN | wxFD_FILE_MUST_EXIST | wxFD_MULTIPLE);\r
+                       if (fd.ShowModal() == wxID_CANCEL)\r
+                       {\r
+                               return;\r
+                       }\r
+                       fd.GetFilenames(mkDlg.files);\r
+                       swInfo.fnDestinationDirectory = wxFileName::DirName(fd.GetDirectory());\r
+                       this->conf.WriteHistory(CONF_HISTORY_PATH, fd.GetDirectory());\r
+               }\r
+       }\r
+       else\r
+       {\r
+               mkDlg.files = * (wxArrayString *) e.GetClientData();\r
+               swInfo.fnDestinationDirectory = wxFileName::DirName(wxFileName(mkDlg.files[0]).GetPath());\r
+               // 相対パスに変換。\r
+//             for (auto s : mkDlg.files)\r
+               for (size_t n = 0; n < mkDlg.files.GetCount(); n++)\r
+               {\r
+                       wxString & s = mkDlg.files[n];\r
+                       wxFileName fn(s);\r
+                       fn.MakeRelativeTo(swInfo.fnDestinationDirectory.GetPath());\r
+                       s = fn.GetFullPath();\r
+               }\r
+       }\r
+\r
        // ダイアログを表示。\r
        if (mkDlg.ShowModal() == wxID_CANCEL)\r
        {\r
@@ -543,11 +737,9 @@ void MainFrame::OnArcAdd(wxCommandEvent& e)
        }\r
 \r
        // 各種設定。\r
-       TPI_SWITCHES swInfo;\r
-       swInfo.pCustomSwitches       = NULL;\r
-       swInfo.fMakeSFX              = false;\r
-       swInfo.fnDestinationDirectory= wxFileName::DirName(fd.GetDirectory());\r
-       swInfo.fStoreDirectoryPathes = ! mkDlg.cbIgnorePath->IsChecked();\r
+       swInfo.szArcName            = this->fnArchive.GetFullPath();\r
+       swInfo.fMakeSFX             = false;\r
+       swInfo.fStoreDirectoryPathes= ! mkDlg.cbIgnorePath->IsChecked();\r
        swInfo.fSolid               = mkDlg.cbSolid->IsChecked();\r
        swInfo.fMMOptimize          = mkDlg.cbMMOptimize->IsChecked();\r
        swInfo.fEncryptHeader       = mkDlg.cbEncryptHeader->IsChecked();\r
@@ -559,18 +751,8 @@ void MainFrame::OnArcAdd(wxCommandEvent& e)
 \r
        // 処理を行う。\r
        {\r
-               ProcessDialog procDlg;\r
-               procDlg.Show(true);\r
-\r
-               // コールバックを送信。\r
-               TPI_PROCESSINFO piInfo;\r
-               piInfo.eMessage = TPI_MESSAGE_STATUS;\r
-               piInfo.eStatus = 0x1000;\r
-               piInfo.fiInfo.fnFileName = this->fnArchive;\r
-               piInfo.fiInfo.nUnpackedSize = mkDlg.files.GetCount();\r
-               procDlg.CallbackProc(TPI_NOTIFY_COMMON, & piInfo);\r
-\r
-               tpi.Command(TPI_COMMAND_ADD, & swInfo, this->fnArchive.GetFullPath(), mkDlg.files);\r
+               ProcessDialog procDlg(this, mkDlg.files.GetCount(), this->szPassword);\r
+               tpi.Command(TPI_COMMAND_ADD, & swInfo, mkDlg.files);\r
                this->ErrorCheck(tpi.nErrorCode);\r
                procDlg.Show(false);\r
        }\r
@@ -588,17 +770,18 @@ void MainFrame::OnArcAdd(wxCommandEvent& e)
        }\r
 \r
        // 終了しない場合は書庫を再読み込み。\r
+       e.SetInt(1);\r
        this->OnArcOpen(e);\r
 }\r
 \r
 void MainFrame::OnArcConvert(wxCommandEvent& e)\r
 {\r
        TPI_SWITCHES swInfo;\r
-       swInfo.fMakeSFX = e.GetId() == XRCID("Arc_SFX");\r
+       swInfo.szArcName            = this->fnArchive.GetFullPath();\r
+       swInfo.fMakeSFX             = e.GetId() == XRCID("Arc_SFX");\r
 \r
        // 保存先を尋ねる。\r
-       wxFileDialog fd(this, swInfo.fMakeSFX ? _("Save as SFX") : _("Save as normal archive"), this->fnArchive.GetPath(), this->fnArchive.GetName() + (swInfo.fMakeSFX ? EXE_EXT : (wxString) wxEmptyString));\r
-       fd.SetWindowStyleFlag(wxFD_SAVE | wxFD_OVERWRITE_PROMPT);\r
+       wxFileDialog fd(this, swInfo.fMakeSFX ? _("Save as SFX") : _("Save as normal archive"), this->fnArchive.GetPath(), this->fnArchive.GetName() + (swInfo.fMakeSFX ? EXE_EXT : (wxString) wxEmptyString), wxFileSelectorDefaultWildcardStr, wxFD_SAVE | wxFD_OVERWRITE_PROMPT);\r
        if (fd.ShowModal() == wxID_CANCEL)\r
        {\r
                return;\r
@@ -609,9 +792,8 @@ void MainFrame::OnArcConvert(wxCommandEvent& e)
        wxArrayString files;\r
        files.Add(fd.GetPath());\r
 \r
-       ProcessDialog procDlg;\r
-       procDlg.Show(true);\r
-       tpi.Command(swInfo.fMakeSFX ? TPI_COMMAND_SFX : TPI_COMMAND_UNSFX, & swInfo, this->fnArchive.GetFullPath(), files);\r
+       ProcessDialog procDlg(this);\r
+       tpi.Command(swInfo.fMakeSFX ? TPI_COMMAND_SFX : TPI_COMMAND_UNSFX, & swInfo, files);\r
        this->ErrorCheck(tpi.nErrorCode);\r
        procDlg.Show(false);\r
 }\r
@@ -619,48 +801,35 @@ void MainFrame::OnArcConvert(wxCommandEvent& e)
 void MainFrame::OnArcExtract(wxCommandEvent& e)\r
 {\r
        TPI_SWITCHES swInfo;\r
-       swInfo.pCustomSwitches = NULL;\r
+       swInfo.szArcName            = this->fnArchive.GetFullPath();\r
+       swInfo.pCustomSwitches      = nullptr;\r
+       swInfo.szPassword           = this->szPassword;\r
 \r
-       // モード取得。通常は0, 実行なら1, ファイルDnDなら2、ディレクトリDnDなら3、クリップボードなら4。\r
+       // ã\83¢ã\83¼ã\83\89å\8f\96å¾\97ã\80\82é\80\9a常ã\81¯0, å®\9fè¡\8cã\81ªã\82\891, ã\83\95ã\82¡ã\82¤ã\83«DnDã\81ªã\82\89\80\81ã\83\87ã\82£ã\83¬ã\82¯ã\83\88ã\83ªDnDã\81ªã\82\89\80\81ã\82¯ã\83ªã\83\83ã\83\97ã\83\9cã\83¼ã\83\89ã\81ªã\82\89\80\81ã\82³ã\83³ã\83\86ã\82­ã\82¹ã\83\88ã\83¡ã\83\8bã\83¥ã\83¼ã\81\8bã\82\89ã\81ªã\82\89\80\82\r
        int nMode = e.GetInt();\r
+       if (e.GetInt() == 8)\r
+       {\r
+               nMode = 0;\r
+       }\r
        // 実行時のみ使用。\r
-       wxFileType * ftFile = NULL;\r
+       wxFileType * ftFile = nullptr;\r
 \r
        // 展開ダイアログを作成。DnDまたは実行時は表示しない。\r
-       MakeDialog mkDlg;\r
-       ::wxXmlResource::Get()->Load(L_DIR_S_XRC wxT("dlg_make.xrc"));\r
-       ::wxXmlResource::Get()->LoadDialog(& mkDlg, this, wxT("dlg_make"));\r
-       mkDlg.uCommand = TPI_COMMAND_EXTRACT;\r
-       mkDlg.files    = MakeTargetFileList(this, nMode == 1);\r
+       MakeDialog mkDlg(this, TPI_COMMAND_EXTRACT);\r
+       mkDlg.files    = MakeTargetFileList(this, nMode == 1, nMode == 0);\r
 \r
+       bool bMakeDir = false;\r
        if (nMode != 0)\r
        {\r
-               if (nMode == 1)\r
-               {\r
-                       // コマンドを取得。\r
-                       ftFile = wxTheMimeTypesManager->GetFileTypeFromExtension(wxFileName(mkDlg.files[0], wxPATH_DOS).GetExt());\r
-                       if (! ftFile)\r
-                       {\r
-                               // 種類が取得できないときは設定を読み込む。初期設定ではテキストとみなす。\r
-                               ftFile = wxTheMimeTypesManager->GetFileTypeFromExtension(this->conf.ReadId(CONF_DEFAULT_EXT, (wxString) wxT("txt")));\r
-                               if (! ftFile)\r
-                               {\r
-                                       wxLogError(_("Unable to get the file type!"));\r
-                                       return;\r
-                               }\r
-                       }\r
-               }\r
-\r
                // 作業ディレクトリ作成。\r
                swInfo.fStoreDirectoryPathes = false;\r
                wxString szDestDirBase = nMode == 3 ? this->tree_ctrl->GetItemText(this->tree_ctrl->GetSelection()) : wxT("tpi_tmp");\r
-               wxStandardPaths p;\r
                if (szDestDirBase == wxT("-----"))\r
                {\r
                        // 書庫ルートのときは書庫名にしておく。\r
                        szDestDirBase = this->fnArchive.GetName();\r
                }\r
-               swInfo.fnDestinationDirectory = MakeDirPath(wxFileName::DirName(p.GetTempDir()), szDestDirBase, true);\r
+               swInfo.fnDestinationDirectory = MakeDirPath(wxFileName::DirName(wxStandardPaths::Get().GetTempDir()), szDestDirBase, true);\r
                if (! swInfo.fnDestinationDirectory.IsOk())\r
                {\r
                        wxLogError(_("Unable to make the temporary directory!"));\r
@@ -681,7 +850,8 @@ void MainFrame::OnArcExtract(wxCommandEvent& e)
                swInfo.szKeyFile  = mkDlg.tcKeyfile->GetValue();\r
 \r
                // 必要なら書庫名でディレクトリを作成する。\r
-               if (WillMakeDirByArcName(this, & mkDlg))\r
+               bMakeDir = WillMakeDirByArcName(this, & mkDlg);\r
+               if (bMakeDir)\r
                {\r
                        swInfo.fnDestinationDirectory = MakeDirPath(swInfo.fnDestinationDirectory, this->fnArchive.GetName(), true);\r
                        if (! swInfo.fnDestinationDirectory.IsOk())\r
@@ -694,18 +864,8 @@ void MainFrame::OnArcExtract(wxCommandEvent& e)
 \r
        // 処理を行う。\r
        {\r
-               ProcessDialog procDlg;\r
-               procDlg.Show(true);\r
-\r
-               // コールバックを送信。\r
-               TPI_PROCESSINFO piInfo;\r
-               piInfo.eMessage = TPI_MESSAGE_STATUS;\r
-               piInfo.eStatus = 0x1000;\r
-               piInfo.fiInfo.fnFileName = this->fnArchive;\r
-               piInfo.fiInfo.nUnpackedSize = mkDlg.files.GetCount();\r
-               procDlg.CallbackProc(TPI_NOTIFY_COMMON, & piInfo);\r
-\r
-               tpi.Command(TPI_COMMAND_EXTRACT, & swInfo, this->fnArchive.GetFullPath(), mkDlg.files);\r
+               ProcessDialog procDlg(this, mkDlg.files.GetCount(), this->szPassword);\r
+               tpi.Command(TPI_COMMAND_EXTRACT, & swInfo, mkDlg.files);\r
                this->ErrorCheck(tpi.nErrorCode);\r
                procDlg.Show(false);\r
        }\r
@@ -715,6 +875,14 @@ void MainFrame::OnArcExtract(wxCommandEvent& e)
                if (mkDlg.cbOpenAfter->IsChecked())\r
                {\r
                        // 展開先を開く。\r
+                       if (mkDlg.chDirMake->GetSelection() == 2 && ! bMakeDir)\r
+                       {\r
+                               wxTreeItemId tiArcSub = this->tree_ctrl->GetLastChild(this->tree_ctrl->GetLastChild(this->tree_ctrl->GetRootItem()));\r
+                               if (tiArcSub.IsOk())\r
+                               {\r
+                                       swInfo.fnDestinationDirectory.SetFullName(this->tree_ctrl->GetItemText(tiArcSub));\r
+                               }\r
+                       }\r
                        ::wxExecute(DIR_APP + QuoteString(swInfo.fnDestinationDirectory.GetFullPath()));\r
                }\r
 \r
@@ -728,30 +896,43 @@ void MainFrame::OnArcExtract(wxCommandEvent& e)
        {\r
                if (nMode == 1)\r
                {\r
+                       // コマンドを取得。\r
+                       ftFile = wxTheMimeTypesManager->GetFileTypeFromExtension(wxFileName(mkDlg.files[0], wxPATH_DOS).GetExt());\r
+                       if (! ftFile || ftFile->GetOpenCommand(wxEmptyString).IsEmpty())\r
+                       {\r
+                               // 種類が取得できないときはテキストとみなす。\r
+                               ftFile = wxTheMimeTypesManager->GetFileTypeFromExtension(this->conf.ReadId(CONF_DEFAULT_EXT, (wxString) wxT("txt")));\r
+                       }\r
+\r
                        // コマンドを実行。\r
                        wxString szTempFile = swInfo.fnDestinationDirectory.GetPathWithSep() + wxFileName(mkDlg.files[0], wxPATH_DOS).GetFullName();\r
-                       if (tpi.nErrorCode == TPI_ERROR_SUCCESS)\r
+                       bool fSuccess = tpi.nErrorCode == TPI_ERROR_SUCCESS && ftFile != nullptr;\r
+                       auto pCallback = new myProcess(szTempFile, swInfo.fnDestinationDirectory.GetPath());\r
+                       if (fSuccess)\r
                        {\r
 #ifdef __LINUX__\r
                                // Linuxでは引用符で囲む必要がある。\r
-                               ::wxExecute(ftFile->GetOpenCommand(QuoteString(szTempFile)), wxEXEC_SYNC);\r
+                               fSuccess = ::wxExecute(wxT("xdg-open ") + QuoteString(szTempFile), wxEXEC_ASYNC, pCallback) > 0;\r
 #else\r
-                               ::wxExecute(ftFile->GetOpenCommand(szTempFile), wxEXEC_SYNC);\r
+                               fSuccess = ::wxExecute(ftFile->GetOpenCommand(szTempFile), wxEXEC_ASYNC, pCallback) > 0;\r
 #endif\r
                        }\r
-\r
-                       ::wxRemoveFile(szTempFile);\r
-                       ::wxRmdir(swInfo.fnDestinationDirectory.GetPath());\r
+                       if (! fSuccess)\r
+                       {\r
+                               pCallback->OnTerminate(0, 0);\r
+                       }\r
                }\r
                else\r
                {\r
                        // 展開対象を決定。\r
                        wxArrayString asFiles;\r
-                       myFileDataObject * objFile = new myFileDataObject();\r
+                       auto objFile = new myFileDataObject();\r
                        objFile->szTempDir = nMode == 3 ? swInfo.fnDestinationDirectory.GetPath() : swInfo.fnDestinationDirectory.GetFullPath();\r
-                       for (size_t i = 0; i < mkDlg.files.GetCount(); i++)\r
+//                     for (auto s : mkDlg.files)\r
+                       for (size_t n = 0; n < mkDlg.files.GetCount(); n++)\r
                        {\r
-                               wxString szFileName = swInfo.fnDestinationDirectory.GetPathWithSep() + wxFileName(mkDlg.files[i], wxPATH_DOS).GetFullName();\r
+                               wxString & s = mkDlg.files[n];\r
+                               wxString szFileName = swInfo.fnDestinationDirectory.GetPathWithSep() + wxFileName(s, wxPATH_DOS).GetFullName();\r
                                if (nMode == 3)\r
                                {\r
                                        asFiles.Add(szFileName);\r
@@ -774,7 +955,7 @@ void MainFrame::OnArcExtract(wxCommandEvent& e)
                        else\r
                        {\r
                                // 自身にドロップされると煩雑なので、一時的にドロップを受け付けないようにしておく。\r
-                               this->SetDropTarget(NULL);\r
+                               this->SetDropTarget(nullptr);\r
 \r
                                // DnD開始。\r
                                wxDropSource dropSource(* objFile, this);\r
@@ -791,9 +972,12 @@ void MainFrame::OnArcExtract(wxCommandEvent& e)
                                // ディレクトリDnDのときは、先にディレクトリの中のファイルを消しておく。\r
                                if (nMode == 3)\r
                                {\r
+//                                     for (auto s : asFiles)\r
                                        for (size_t i = 0; i < asFiles.GetCount(); i++)\r
                                        {\r
-                                               ::wxRemoveFile(asFiles[i]);\r
+                                               wxString & s = asFiles[i];\r
+                                               chmod(s.ToUTF8(), 0600);\r
+                                               ::wxRemoveFile(s);\r
                                        }\r
                                }\r
 \r
@@ -808,84 +992,55 @@ void MainFrame::OnArcDelete(wxCommandEvent& e)
        // 全ファイル削除は危険ではないかと。\r
        if (this->list_ctrl->GetSelectedItemCount() == 0)\r
        {\r
+               // wxのバグで自動では無効化できないので、実行しようとしたときに無効化する。\r
+               SetMenuToolState("Arc_Delete", false);\r
                return;\r
        }\r
 \r
-       if (::AskDlg(_("Are you sure to delete selected files?"), this) == wxNO)\r
+       if (AskDlg(_("Are you sure to delete selected files?"), this) == wxNO)\r
        {\r
                return;\r
        }\r
 \r
        // 処理を行う。\r
        {\r
+               wxArrayString asFiles = MakeTargetFileList(this);\r
+               ProcessDialog procDlg(this, asFiles.GetCount(), this->szPassword);\r
                TPI_SWITCHES swInfo;\r
-               ProcessDialog procDlg;\r
-               procDlg.Show(true);\r
-               wxArrayString asFiles = MakeTargetFileList(this, false);\r
-\r
-               // コールバックを送信。\r
-               TPI_PROCESSINFO piInfo;\r
-               piInfo.eMessage = TPI_MESSAGE_STATUS;\r
-               piInfo.eStatus = 0x1000;\r
-               piInfo.fiInfo.fnFileName = this->fnArchive;\r
-               piInfo.fiInfo.nUnpackedSize = asFiles.GetCount();\r
-               procDlg.CallbackProc(TPI_NOTIFY_COMMON, & piInfo);\r
-\r
-               tpi.Command(TPI_COMMAND_DELETE, & swInfo, this->fnArchive.GetFullPath(), asFiles);\r
+               swInfo.szArcName  = this->fnArchive.GetFullPath();\r
+               swInfo.szPassword = this->szPassword;\r
+               tpi.Command(TPI_COMMAND_DELETE, & swInfo, asFiles);\r
                this->ErrorCheck(tpi.nErrorCode);\r
                procDlg.Show(false);\r
        }\r
 \r
        // 書庫を再読み込みする。\r
+       e.SetInt(1);\r
        this->OnArcOpen(e);\r
 }\r
 \r
 void MainFrame::OnArcTest(wxCommandEvent&)\r
 {\r
        // 処理を行う。\r
-       ProcessDialog procDlg;\r
-       procDlg.Show(true);\r
-       wxArrayString asFiles = MakeTargetFileList(this, false);\r
-\r
-       // コールバックを送信。\r
-       TPI_PROCESSINFO piInfo;\r
-       piInfo.eMessage = TPI_MESSAGE_STATUS;\r
-       piInfo.eStatus = 0x1000;\r
-       piInfo.fiInfo.fnFileName = this->fnArchive;\r
-       piInfo.fiInfo.nUnpackedSize = asFiles.GetCount();\r
-       procDlg.CallbackProc(TPI_NOTIFY_COMMON, & piInfo);\r
-\r
+       wxArrayString asFiles = MakeTargetFileList(this);\r
+       ProcessDialog procDlg(this, asFiles.GetCount(), this->szPassword);\r
        TPI_SWITCHES swInfo;\r
-       bool bIsCorrect = tpi.Command(TPI_COMMAND_TEST, & swInfo, this->fnArchive.GetFullPath(), asFiles);\r
+       swInfo.szArcName  = this->fnArchive.GetFullPath();\r
+       swInfo.szPassword = this->szPassword;\r
+       tpi.Command(TPI_COMMAND_TEST, & swInfo, asFiles);\r
+       this->ErrorCheck(tpi.nErrorCode);\r
        procDlg.Show(false);\r
-\r
-       if (bIsCorrect)\r
-       {\r
-               wxLogMessage(_("This is a correct archive."));\r
-       }\r
-       else\r
-       {\r
-               this->ErrorCheck(tpi.nErrorCode);\r
-       }\r
 }\r
 \r
 void MainFrame::OnArcRepair(wxCommandEvent&)\r
 {\r
        // 処理を行う。\r
-       ProcessDialog procDlg;\r
-       procDlg.Show(true);\r
-       wxArrayString asFiles = MakeTargetFileList(this, false);\r
-\r
-       // コールバックを送信。\r
-       TPI_PROCESSINFO piInfo;\r
-       piInfo.eMessage = TPI_MESSAGE_STATUS;\r
-       piInfo.eStatus = 0x1000;\r
-       piInfo.fiInfo.fnFileName = this->fnArchive;\r
-       piInfo.fiInfo.nUnpackedSize = asFiles.GetCount();\r
-       procDlg.CallbackProc(TPI_NOTIFY_COMMON, & piInfo);\r
-\r
+       wxArrayString asFiles = MakeTargetFileList(this);\r
+       ProcessDialog procDlg(this, asFiles.GetCount(), this->szPassword);\r
        TPI_SWITCHES swInfo;\r
-       tpi.Command(TPI_COMMAND_REPAIR, & swInfo, this->fnArchive.GetFullPath(), asFiles);\r
+       swInfo.szArcName  = this->fnArchive.GetFullPath();\r
+       swInfo.szPassword = this->szPassword;\r
+       tpi.Command(TPI_COMMAND_REPAIR, & swInfo, asFiles);\r
        this->ErrorCheck(tpi.nErrorCode);\r
        procDlg.Show(false);    \r
 }\r
@@ -894,6 +1049,8 @@ void MainFrame::OnExeCopy(wxCommandEvent & e)
 {\r
        if (this->list_ctrl->GetSelectedItemCount() == 0)\r
        {\r
+               // wxのバグで自動では無効化できないので、実行しようとしたときに無効化する。\r
+               this->menubar->Enable(XRCID("Exe_Copy"), false);\r
                return;\r
        }\r
 \r
@@ -959,24 +1116,26 @@ void MainFrame::OnTreeChanged(wxTreeEvent& e)
        this->list_ctrl->SetImageList(& g_hIconLS, wxIMAGE_LIST_SMALL);\r
 \r
        // 配列と比較し、パスが一致しなければ消す。\r
+//     for (auto f : this->fileinfo)\r
        for (size_t i = 0; i < this->fileinfo.GetCount(); i++)\r
        {\r
                // パスを比較。\r
-               if (szNodePath == wxT("*") || szNodePath == this->fileinfo[i].fnFileName.GetPath())\r
+               TPI_FILEINFO & f = this->fileinfo[i];\r
+               if (szNodePath == wxT("*") || szNodePath == f.fnFileName.GetPath())\r
                {\r
                        // 項目がフォルダであるなら無視。\r
-                       if (this->fileinfo[i].fnFileName.IsDir() || ! TreeView_CheckNewerItem(this->tree_ctrl, this->tree_ctrl->GetLastChild(this->tree_ctrl->GetRootItem()), this->fileinfo[i].fnFileName.GetFullPath(), false))\r
+                       if (f.fnFileName.IsDir() || ! TreeView_CheckNewerItem(this->tree_ctrl, this->tree_ctrl->GetLastChild(this->tree_ctrl->GetRootItem()), f.fnFileName.GetFullPath(), false))\r
                        {\r
                                continue;\r
                        }\r
 \r
                        // フィルタにかからなければ無視。\r
-                       if (this->fileinfo[i].fnFileName.GetFullName().MakeLower().Find(this->tcFilter->GetValue().MakeLower()) == wxNOT_FOUND)\r
+                       if (f.fnFileName.GetFullName().MakeLower().Find(this->tcFilter->GetValue().MakeLower()) == wxNOT_FOUND)\r
                        {\r
                                continue;\r
                        }\r
 \r
-                       this->list_ctrl->apShowFile.Add(& this->fileinfo[i]);\r
+                       this->list_ctrl->apShowFile.Add(& f);\r
                }\r
        }\r
 \r
@@ -995,8 +1154,22 @@ void MainFrame::OnTreeBeginDrag(wxTreeEvent& e)
        this->OnArcExtract(e1);\r
 }\r
 \r
+void MainFrame::OnContextMenu(wxContextMenuEvent& e)\r
+{\r
+       wxPoint p = e.GetPosition();\r
+       this->list_ctrl->PopupMenu(this->menubar->GetMenu(1), p == wxDefaultPosition ? this->list_ctrl->GetPosition() : this->list_ctrl->ScreenToClient(p));\r
+}\r
+\r
 // ListView\r
 \r
+void MainFrame::OnListItemSelect(wxListEvent&)\r
+{\r
+       // ファイルに対する動作の設定。但し、選択解除時はwxのバグで呼び出されない。\r
+       bool fEnable = this->list_ctrl->GetSelectedItemCount() > 0;\r
+       SetMenuToolState("Arc_Delete", fEnable && (this->aiArchive.fiInfo.eSupportedCommand & TPI_COMMAND_DELETE) == TPI_COMMAND_DELETE && this->aiArchive.fiInfo.fArchive);\r
+       this->menubar->Enable(XRCID("Exe_Copy"), fEnable);\r
+}\r
+\r
 void MainFrame::OnListItemDClick(wxListEvent&)\r
 {\r
        wxCommandEvent e;\r
@@ -1041,18 +1214,8 @@ bool MainFrame::LoadTPI(const wxString & szFileName, wxULongLong_t * llFileCount
        {\r
                do\r
                {\r
-                       // 初期化。\r
-                       if (! tpi.InitLibrary(L_DIR_B_LIB + szTPIName, szFileName))\r
-                       {\r
-                               tpi.FreeLibrary();\r
-                               continue;\r
-                       }\r
-\r
-                       // コールバック関数を設定。\r
-                       tpi.SetCallbackProc(TPICallbackProc);\r
-\r
-                       // 対応確認。\r
-                       if (! tpi.OpenArchive(szFileName, llFileCount))\r
+                       // 初期化と対応確認。\r
+                       if (! tpi.InitLibrary(L_DIR_B_LIB + szTPIName, szFileName, TPICallbackProc) || ! tpi.OpenArchive(szFileName, llFileCount))\r
                        {\r
                                tpi.FreeLibrary();\r
                                * llFileCount = 0;\r
@@ -1078,7 +1241,6 @@ int MainFrame::ErrorCheck(int nErrorCode, const wxString & szAPIName)
                break;\r
        case TPI_ERROR_D_SKIPPED:\r
        case TPI_CALLBACK_CANCEL:\r
-               wxLogError(_("This operation is canceled by the user."));\r
                break;\r
        default:\r
                wxLogError(_("Error: %s()!\nError code: %d"), szAPIName.c_str(), nErrorCode);\r