exeコピーツール作成 5

前回までの記事

フォルダ階層説明

今更ですが予定パスはこんな感じ

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を開きます。

現在実行パスが「Debug」になってる!!

さて、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;
}

コメント

タイトルとURLをコピーしました