前回までの記事
フォルダ階層説明
今更ですが予定パスはこんな感じ

C:\Users\tatsu\work\VS
までは手動で作ります。
Temporaryは、コピーするexeを作るプロジェクトファイルです。
プロジェクトを今後作っていきますが、作成したときに出来上がるフォルダ内に、
今回作成するCopyTool.exeを配置します。
Temporaryソリューションをビルドし、exeがいくつかDebugフォルダまたはReleaseフォルダに出来上がります。
そのexeすべてを「フォルダA」に移動させる、
もしくはSeparateフォルダ内のX環境とY環境に分別コピーする、という感じです。
Constantヘッダー追加
いよいよコピー処理を書いていきます。
今回は、いったんテキストファイルのコピーができるかの確認をします。
実際にはexeをコピーし、コピー元となるパスはまたその時設定しますが、
コピー先のパスはもう今から設定できるので、今のうちに設定しておきます。
コピー先パスなど、処理内で不変の値が分かる場合、私はConstantヘッダーを用いるようにしています。

「ヘッダーファイル」を右クリック→「追加」→「新しい項目」を選択します。

「ヘッダーファイル(.h)」を選択し、名前をConstant.hに変えて追加ボタン。

Constant.hが追加されたのを確認できました。
Constant.hを開き、定義を書いていきます。
まずは、「フォルダA」へのパス。
#define PATH_DIR_FOLDER_A _T("C:\\Users\\tatsu\\フォルダA")
ただし、この状態でこの定数をCopyToolDlg.cppで使用した場合、コンパイルエラーになります。

定義されてないって文句言われてますね。
なので、CopyToolDlg.cppで以下を宣言します。
#include "Constant.h"挿入箇所は、上部に色々includeしている箇所があるので、その一番下で。

これでConstant.hに定義した定数が使えるようになりました。

120行目は実験のために書いたもので今後使わないので、元に戻しておきます。
ディレクトリ作成
さて、いよいよテキストファイルを先の「C:\Users\tatsu\フォルダA」にコピーするのですが。
当然、コピー先のディレクトリがなければコピーなんぞできないので、
フォルダ存在チェックを追加します。
BOOL型の関数を作り、「存在する」ならばTRUEを返すように。
仮に「存在しない」ならばディレクトリを作成します。
「ディレクトリ作成成功」ならばTRUEを、「作成失敗」ならばFALSEを返すようにします。
CopyToolDlg.hに関数定義。
BOOL bDirExistCheck(Mode enMode);CopyToolDlg.cppに関数処理を書きます。
とりあえず、「フォルダAに移動」の処理だけ記載します。
後で「分別用」の処理も書いちゃいましょう。
BOOL CCopyToolDlg::bDirExistCheck(Mode enMode)
{
CString strPath = _T("");
if (enMode == enCopyFolderA)
{
strPath = PATH_DIR_FOLDER_A; //フォルダAへのパス
if (!::PathFileExists(strPath))
{
//パスが存在しない
CString strMes = strPath + _T("が存在しません。\n作成しますか?");
int iRes = AfxMessageBox(strMes, MB_YESNO); //はい or いいえメッセージボックス
if (iRes == IDOK) //「はい」が選択される
{
if (!::CreateDirectory(strPath, NULL)) //ディレクトリ作成
{
AfxMessageBox(_T("フォルダ作成に失敗しました。"));
return FALSE; //作成失敗の場合はFALSE
}
}
else
{
return FALSE; //「いいえ」の場合はFALSE
}
}
}
else
{
return FALSE;
}
return TRUE;
}PathFileExistsは、引数のパスが存在するかチェックしてくれます。
存在したらTRUE、しなかったらFALSE。
FALSEの場合、メッセージボックスを出すようにしました。
そしてメッセージボックスのボタンが「はい」を押された時、
CreateDirectoryで引数に指定されたパスのフォルダを作成します。
こいつを、ボタンが押下されたときに呼ばれる共通処理CopyStart()内で呼応します。
もう一つ、共通処理CopyStart()の一番最後に、メッセージボックスを出しておきます。
処理がちゃんと通ったかの確認も兼ねてます。
void CCopyToolDlg::CopyStart()
{
if (!bDirExistCheck(m_enRadioBtnMode))
{
return; //存在チェックでFALSEが帰ってきた場合は以降の処理を行わない
}
AfxMessageBox(_T("処理が終了しました。"));
}
さて、ビルドして動かしてみます。
まず、すでに「フォルダA」は作ってあるとき、処理終了のメッセージが出てきました。

続いて、手動で「フォルダA」を削除して再度ボタンを押下すると

フォルダ作成するか聞かれました。
「はい」を選択すると先の「処理が終了しました。」のメッセージが出現し、
「いいえ」を選択すると何のメッセージも出てこないので、大丈夫そうです。
因みに、ラジオボタンが「分別用」の処理はまだ何も記載していないので、
そちらを選択後ボタンを押下してもまだ何もメッセージは出てきません。
さて、肝心の作成しますか?で「はい」を押下すると、
フォルダもちゃんと作成されているのを確認でき…
なかった!!!Σ(´∀`;)
デバッグで確認
記事書きながら作業進めるとそういうこともあるよなー。
これはネタとして意図して間違った処理を書いたのではなく、
本当に凡ミスでした。
すでにソースコードを書いた記事まで書いた後にexeを実施していたので、
記事を書き直そうかとも思いましたけど、せっかくなので直し方も載せようかと。
ビルドのモードを「Debug」に選択します。

というか、作成している段階ではどちらかというとDebugモードのほうが作りやすいので、
今後はDebugモードで作成することにします。
同じくビルドを行い、作成されたexeを開きます。
以前までは「C:\Users\tatsu\source\repos\CopyTool\x64\Release」の中にあるexeを開いていましたが、
今回からは「C:\Users\tatsu\source\repos\CopyTool\x64\Debug」の中のexeを開きます。

さて、Visual Studioの「デバッグ」→「プロセスにアタッチ」を選択します。

そして起動している「CopyTool.exe」を選択し、アタッチします。
先にexeを起動してないとこの画面にCopyTool.exeは出てこないので、
起動してからプロセスにアタッチ画面を開きます。

するとVisual Studioの下部が赤っぽく変化します

さて、「はい」を選択後の処理がどうなってるか確認するため、
int iRes = AfxMessageBox(strMes, MB_YESNO);の行にブレークポイントを張ってみます。
行番号の左側をクリックで簡単に張れます。

この時、ちゃんと赤色になっていればデバッグが可能です。
白色の丸になるときは、対処法は別のサイトを参考にしてください。
いざ、exeの「Debug」ボタンを押下
(「Release」ボタンでも当然いいです。)

処理が止まりました。
F10でステップオーバーを行い、先の「作成しますか?メッセージが出てくるので「はい」を選択。
するとiResには選択の戻り値が格納されます。

おや、iResには「6」が入ってます。
このままF10で処理を進めていくと、292行目のelseに入ってしまいました。
じゃあ「IDOK」ってなんだっけ?
IDOKを選択後、F12を押します。これで定義に飛べます。

IDOKは1でした。
間違えたわ。
メッセージボックスはMB_YESNOで表示して応えが「はい」か「いいえ」なんだから
条件はIDYESかどうかで判定しないと
修正しました。

これで大丈夫なはず
再度ビルドして作成質問に対して「はい」を選択すると、
ちゃんとフォルダが作成されているのを確認できました。
コピー処理
最後にコピー処理を書いていきます。
後で別関数に分けて処理を行いますが、とりあえず今はCopyStart()関数内に直接書いて
コピー処理が正しく実行されるか確認したいです。
CopyStart()内に以下を記述します。
CString strCopyPathFROM = m_strExecutePath; //コピー元パス 現在実行ディレクトリ
CString strCopyPathTO = PATH_DIR_FOLDER_A; //コピー先パス フォルダAへのパス
CString strFileName = _T("\\Test.txt"); //コピーするファイル
::CopyFile(strCopyPathFROM + strFileName, strCopyPathTO + strFileName, FALSE);
さぁ、ビルドして試してみます。

実行するexeの下に、Test.txtというものを事前に作成しておきます。
exeを開き、「Debug」ボタンを実行。
「処理が終了しました」のメッセージが出てきてくれました。
さて、フォルダAのほうを見に行くと…

ちゃんとコピー元のファイルが
コピーされていました(´▽`)
今回のまとめ
今回作成したヘッダーファイルとcppファイルを載せておきます。
// CopyToolDlg.h : ヘッダー ファイル
//
#pragma once
typedef enum tagMode
{
enDebug = 0, // Debugボタン押下された時用
enRelease, // Releaseボタン押下された時用
enCopyFolderA = 10, // フォルダAに移動ラジオボタン用
enSeparate // 分別用ラジオボタン用
}Mode;
// CCopyToolDlg ダイアログ
class CCopyToolDlg : public CDialogEx
{
// コンストラクション
public:
CCopyToolDlg(CWnd* pParent = nullptr); // 標準コンストラクター
// ダイアログ データ
#ifdef AFX_DESIGN_TIME
enum { IDD = IDD_COPYTOOL_DIALOG };
#endif
protected:
virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV サポート
// 実装
protected:
HICON m_hIcon;
// 生成された、メッセージ割り当て関数
virtual BOOL OnInitDialog();
afx_msg void OnSysCommand(UINT nID, LPARAM lParam);
afx_msg void OnPaint();
afx_msg HCURSOR OnQueryDragIcon();
DECLARE_MESSAGE_MAP()
public:
afx_msg void OnBnClickedButtonDebug(); // Debugボタン押下時に呼ばれる
afx_msg void OnBnClickedButtonRelease(); // Releaseボタン押下時に呼ばれる
afx_msg void OnBnClickedRadioFolderA(); // フォルダAに移動が選択されたときに呼ばれる
afx_msg void OnBnClickedRadioSeparate(); // 分別用が選択されたときに呼ばれる
private:
Mode m_enButtonMode; // ボタンモード
Mode m_enRadioBtnMode; // ラジオボタン選択モード
CString m_strExecutePath; // 現在実行パス
private:
void CopyStart(); // 共通処理
BOOL bGetCurrentPath(); // 現在実行パス取得関数
BOOL bDirExistCheck(Mode enMode);
};
// CopyToolDlg.cpp : 実装ファイル
//
#include "pch.h"
#include "framework.h"
#include "CopyTool.h"
#include "CopyToolDlg.h"
#include "afxdialogex.h"
#include "Constant.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#endif
// アプリケーションのバージョン情報に使われる CAboutDlg ダイアログ
class CAboutDlg : public CDialogEx
{
public:
CAboutDlg();
// ダイアログ データ
#ifdef AFX_DESIGN_TIME
enum { IDD = IDD_ABOUTBOX };
#endif
protected:
virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV サポート
// 実装
protected:
DECLARE_MESSAGE_MAP()
};
CAboutDlg::CAboutDlg() : CDialogEx(IDD_ABOUTBOX)
{
}
void CAboutDlg::DoDataExchange(CDataExchange* pDX)
{
CDialogEx::DoDataExchange(pDX);
}
BEGIN_MESSAGE_MAP(CAboutDlg, CDialogEx)
END_MESSAGE_MAP()
// CCopyToolDlg ダイアログ
CCopyToolDlg::CCopyToolDlg(CWnd* pParent /*=nullptr*/)
: CDialogEx(IDD_COPYTOOL_DIALOG, pParent)
{
m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}
void CCopyToolDlg::DoDataExchange(CDataExchange* pDX)
{
CDialogEx::DoDataExchange(pDX);
}
BEGIN_MESSAGE_MAP(CCopyToolDlg, CDialogEx)
ON_WM_SYSCOMMAND()
ON_WM_PAINT()
ON_WM_QUERYDRAGICON()
ON_BN_CLICKED(IDC_BUTTON_DEBUG, &CCopyToolDlg::OnBnClickedButtonDebug)
ON_BN_CLICKED(IDC_BUTTON_RELEASE, &CCopyToolDlg::OnBnClickedButtonRelease)
ON_BN_CLICKED(IDC_RADIO_FOLDER_A, &CCopyToolDlg::OnBnClickedRadioFolderA)
ON_BN_CLICKED(IDC_RADIO_SEPARATE, &CCopyToolDlg::OnBnClickedRadioSeparate)
END_MESSAGE_MAP()
// CCopyToolDlg メッセージ ハンドラー
BOOL CCopyToolDlg::OnInitDialog()
{
CDialogEx::OnInitDialog();
// "バージョン情報..." メニューをシステム メニューに追加します。
// IDM_ABOUTBOX は、システム コマンドの範囲内になければなりません。
ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
ASSERT(IDM_ABOUTBOX < 0xF000);
CMenu* pSysMenu = GetSystemMenu(FALSE);
if (pSysMenu != nullptr)
{
BOOL bNameValid;
CString strAboutMenu;
bNameValid = strAboutMenu.LoadString(IDS_ABOUTBOX);
ASSERT(bNameValid);
if (!strAboutMenu.IsEmpty())
{
pSysMenu->AppendMenu(MF_SEPARATOR);
pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);
}
}
// このダイアログのアイコンを設定します。アプリケーションのメイン ウィンドウがダイアログでない場合、
// Framework は、この設定を自動的に行います。
SetIcon(m_hIcon, TRUE); // 大きいアイコンの設定
SetIcon(m_hIcon, FALSE); // 小さいアイコンの設定
// TODO: 初期化をここに追加します。
BOOL bTemp = bGetCurrentPath();
CString strText = _T("現在実行パス\n");
strText += m_strExecutePath;
CStatic* pStatic = (CStatic*)GetDlgItem(IDC_STATIC_PATH);
pStatic->SetWindowTextW(strText);
CButton* pBtn = (CButton*)GetDlgItem(IDC_RADIO_FOLDER_A);
pBtn->SetCheck(1);
m_enRadioBtnMode = enCopyFolderA; //初期表示は「フォルダAへ移動」
SetWindowText(_T("コピーツール"));
return TRUE; // フォーカスをコントロールに設定した場合を除き、TRUE を返します。
}
void CCopyToolDlg::OnSysCommand(UINT nID, LPARAM lParam)
{
if ((nID & 0xFFF0) == IDM_ABOUTBOX)
{
CAboutDlg dlgAbout;
dlgAbout.DoModal();
}
else
{
CDialogEx::OnSysCommand(nID, lParam);
}
}
// ダイアログに最小化ボタンを追加する場合、アイコンを描画するための
// 下のコードが必要です。ドキュメント/ビュー モデルを使う MFC アプリケーションの場合、
// これは、Framework によって自動的に設定されます。
void CCopyToolDlg::OnPaint()
{
if (IsIconic())
{
CPaintDC dc(this); // 描画のデバイス コンテキスト
SendMessage(WM_ICONERASEBKGND, reinterpret_cast<WPARAM>(dc.GetSafeHdc()), 0);
// クライアントの四角形領域内の中央
int cxIcon = GetSystemMetrics(SM_CXICON);
int cyIcon = GetSystemMetrics(SM_CYICON);
CRect rect;
GetClientRect(&rect);
int x = (rect.Width() - cxIcon + 1) / 2;
int y = (rect.Height() - cyIcon + 1) / 2;
// アイコンの描画
dc.DrawIcon(x, y, m_hIcon);
}
else
{
CDialogEx::OnPaint();
}
}
// ユーザーが最小化したウィンドウをドラッグしているときに表示するカーソルを取得するために、
// システムがこの関数を呼び出します。
HCURSOR CCopyToolDlg::OnQueryDragIcon()
{
return static_cast<HCURSOR>(m_hIcon);
}
//ここから自作処理
/**
* @fn
* @brief Debugボタン押下時に呼応
* @param なし
* @return なし
*/
void CCopyToolDlg::OnBnClickedButtonDebug()
{
// TODO: ここにコントロール通知ハンドラー コードを追加します。
UpdateData(TRUE);
m_enButtonMode = enDebug;
CopyStart();
}
/**
* @fn
* @brief Releaseボタン押下時に呼応
* @param なし
* @return なし
*/
void CCopyToolDlg::OnBnClickedButtonRelease()
{
// TODO: ここにコントロール通知ハンドラー コードを追加します。
UpdateData(TRUE);
m_enButtonMode = enRelease;
CopyStart();
}
/**
* @fn
* @brief 「フォルダAへ移動」が選択されたときに呼応
* @param なし
* @return なし
*/
void CCopyToolDlg::OnBnClickedRadioFolderA()
{
// TODO: ここにコントロール通知ハンドラー コードを追加します。
UpdateData(TRUE);
m_enRadioBtnMode = enCopyFolderA; //「フォルダAへ移動」モード
}
/**
* @fn
* @brief 「分別用」が選択されたときに呼応
* @param なし
* @return なし
*/
void CCopyToolDlg::OnBnClickedRadioSeparate()
{
// TODO: ここにコントロール通知ハンドラー コードを追加します。
UpdateData(TRUE);
m_enRadioBtnMode = enSeparate; //「分別用」モード
}
/**
* @fn
* @brief コピー実行処理
* @param なし
* @return なし
*/
void CCopyToolDlg::CopyStart()
{
if (!bDirExistCheck(m_enRadioBtnMode))
{
return; //存在チェックでFALSEが帰ってきた場合は以降の処理を行わない
}
CString strCopyPathFROM = m_strExecutePath; //コピー元パス 現在実行ディレクトリ
CString strCopyPathTO = PATH_DIR_FOLDER_A; //コピー先パス フォルダAへのパス
CString strFileName = _T("\\Test.txt"); //コピーするファイル
::CopyFile(strCopyPathFROM + strFileName, strCopyPathTO + strFileName, FALSE);
AfxMessageBox(_T("処理が終了しました。"));
}
/**
* @fn
* @brief 現在実行パス取得処理
* @param なし
* @return BOOL TRUE 取得成功
* FALSE 取得失敗
*/
BOOL CCopyToolDlg::bGetCurrentPath()
{
CString strPath = (LPCTSTR)nullptr;
TCHAR path[_MAX_PATH], drive[_MAX_PATH], dir[_MAX_PATH], file[_MAX_PATH], ext[_MAX_PATH];
if (::GetModuleFileName(NULL, path, _MAX_PATH) != 0)
{
::_tsplitpath_s(path, drive, _MAX_PATH, dir, _MAX_PATH, file, _MAX_PATH, ext, _MAX_PATH);
strPath = ::PathCombine(path, drive, dir);
}
else
{
AfxMessageBox(_T("実行ファイルパスの取得に失敗しました"));
return FALSE;
}
m_strExecutePath = strPath;
return TRUE;
}
/**
* @fn
* @brief フォルダ存在チェック
* @param Mode enMode Debug or Release どちらのボタンが押下されたか
* @return BOOL TRUE 作成成功 or 既に存在
* FALSE 作成失敗
*/
BOOL CCopyToolDlg::bDirExistCheck(Mode enMode)
{
CString strPath = _T("");
if (enMode == enCopyFolderA)
{
strPath = PATH_DIR_FOLDER_A; //フォルダAへのパス
if (!::PathFileExists(strPath))
{
//パスが存在しない
CString strMes = strPath + _T("が存在しません。\n作成しますか?");
int iRes = AfxMessageBox(strMes, MB_YESNO); //はい or いいえメッセージボックス
if (iRes == IDYES) //「はい」が選択される
{
if (!::CreateDirectory(strPath, NULL)) //ディレクトリ作成
{
AfxMessageBox(_T("フォルダ作成に失敗しました。"));
return FALSE; //作成失敗の場合はFALSE
}
}
else
{
return FALSE; //「いいえ」の場合はFALSE
}
}
}
else
{
return FALSE;
}
return TRUE;
}

コメント