MFCで、CSVからデータを抽出・表示させる処理 6

前回までの記事

エディットボックス制御処理

まず初めに、前回作成したエディットボックスに関して
少しだけ制御処理を加えたいので実行します。

  • 入力桁数はエディットボックスいっぱいまで
  • 0以上100以下の数値が入力された場合、保存ボタン押下時にエラーメッセージを出す

まず一つ目。

これは簡単。
リソースビューからダイアログを開き、
エディットボックスのプロパティから自動HScrollをFalseに設定します。

前回最後に書きましたが、数字はTrueです。

これだけです。

続いて二つ目。
まず「保存&表示」ボタンと「保存」ボタンが押されたときに呼ばれる関数を追加します。

例によって、リソースビューから該当ボタンをダブルクリックです。

更に、チェック関数を追加します。

Condition.hにチェック関数を定義。

private:
	BOOL bControlCheck();			// 各コントロールのチェック

Condition.cppの方に処理を記述します。
ついでなので、チェックボックスに関する処理も追加しておきます。

内容は、「少なくともどれか一つはチェックが入っていること」

チェックボックスとエディットボックスの処理を記述してみました。

BOOL CCondition::bControlCheck()
{
	if (m_chk[SUBCON_JA].GetCheck() == BST_UNCHECKED &&
		m_chk[SUBCON_MA].GetCheck() == BST_UNCHECKED &&
		m_chk[SUBCON_EN].GetCheck() == BST_UNCHECKED &&
		m_chk[SUBCON_SC].GetCheck() == BST_UNCHECKED &&
		m_chk[SUBCON_SS].GetCheck() == BST_UNCHECKED)
	{
		// すべてのチェックボックスにチェックが入っていない
		AfxMessageBox(_T("少なくとも1つのチェックボックスに\r\nチェックを入れてください。"));
		return FALSE;
	}

	for (int iLoop = SUBCON_JA; iLoop < SUMCON_NUM; iLoop++)
	{
		//チェックが入っていない場合は入力チェック対象外
		if (m_chk[iLoop].GetCheck() == BST_UNCHECKED)
		{
			m_bCheck[iLoop] = FALSE;
			continue;
		}

		// 点以上のチェック
		int iScoreIJO = _ttoi(m_strIJO[iLoop]);	//int型に変換
		if (iScoreIJO < 0 || 100 < iScoreIJO)
		{
			AfxMessageBox(_T("0以上100以下の数字を入力してください。"));
			m_editIJO[iLoop].SetFocus();	//該当エディットボックスにセットフォーカス
			return FALSE;
		}

		//点以下のチェック
		int iScoreIKA = _ttoi(m_strIKA[iLoop]);	//int型に変換
		if (iScoreIKA < 0 || 100 < iScoreIKA)
		{
			AfxMessageBox(_T("0以上100以下の数字を入力してください。"));
			m_editIKA[iLoop].SetFocus();	//該当エディットボックスにセットフォーカス
			return FALSE;
		}

		//以下と以上の関係性チェック
		if (iScoreIKA < iScoreIJO)
		{
			AfxMessageBox(_T("数値の大小が正しくありません。"));
			m_editIJO[iLoop].SetFocus();	//該当エディットボックス(以上の方)にセットフォーカス
			return FALSE;
		}

		m_bCheck[iLoop] = TRUE;						//メンバ変数更新
		m_strIJO[iLoop].Format("%d", iScoreIJO);	//メンバ変数更新
		m_strIKA[iLoop].Format("%d", iScoreIKA);	//メンバ変数更新
	}

	//チェックは大丈夫
	return TRUE;
}

チェック方法は

  1. すべてのチェックボックスが未チェックな場合は問題あり
  2. 点以上の方の数値が0未満100越の場合は問題あり
  3. 点以下の方の数値が0未満100越の場合は問題あり
  4. 点以上が点以下の方より大きい場合は問題あり
  5. メンバ変数を更新

という感じです。
5を行う理由は後で説明します。

さて、今作った関数を、保存&表示ボタン、保存ボタン押下時の時に呼んでみます。

void CCondition::OnBnClickedButtonSavedisp()
{
	UpdateData(TRUE);	//コントロールからメンバ変数にデータ転送

	if (!bControlCheck())
		return;

	UpdateData(FALSE);	//メンバ変数からコントロールにデータ転送(念のため)

	CDialog::OnOK();	//画面を閉じる
}


void CCondition::OnBnClickedButtonSave()
{
	UpdateData(TRUE);	//コントロールからメンバ変数にデータ転送

	if (!bControlCheck())
		return;

	UpdateData(FALSE);	//メンバ変数からコントロールにデータ転送(念のため)

}

まずUpdateData(TRUE)で、エディットボックス内にある数値を
メンバ変数に入れます。

これが前回、DoDataExchange内にDDX_Textの文を加えた理由です。

そしてチェックが完了し、TRUEが返ってくると
UpdateData(FALSE)を呼応してメンバ変数をエディットボックスにセットします。

先ほどのメンバ変数更新処理ですが、
その更新後のメンバ変数がエディットボックスにセットされます。

なんのためにやっているかですが、
例えば01と入力された場合、保存ボタンを押すと

int型変換時には当然1となります。
それをメンバ変数更新で

文字列の1にしてやります。

後はUpdateData(FALSE)により、この「1」がエディットボックスにセットされ

先頭の0がなくなる、ただこれだけをやりたい話です。

因みに問題があった場合(bControlCheckがFALSEを返した場合)、
UpdateData(FALSE)は呼ばれませんので表示はそのままです。

とりあえず、チェック処理は以上にしようと思います。

修正入ります

「保存&表示」ボタンを押せば画面は閉じるのですが、
このままではデータが何もTestResultDlgの方にわたってきません。
また、条件変更ダイアログを開くたびに初期状態が設定されてしまいます。

この2つのダイアログでデータを受け渡す関数を用意してやります。

まず、一つ作り直さなくてはいけない処理があるのでそこから。

前回、enumを定義したときに
「共通ファイル用意した方がいいかなぁ」と言いましたが、

用意すべきでした。

用意しなくても処理を作るのは可能なのは可能ですが、
如何せんわかりにくいので。

ソリューションエクスプローラのヘッダーファイルを右クリックし、
追加→新しい項目を選択します。

ファイル名はConst.hにでもして、追加します。

追加したら、Const.hに、TestResultDlg.hに書いてあるSubjectのenumをそのままコピーします。

TestResultDlg.hのほうはもうこのenumはいらないので削除します。

当然、TestResultDlg.cpp内で昔作った、SUBJECTを利用している処理はエラーが起きます。

TestResultDlg.hの上部にインクルード文を入れます。

#include "Const.h"

これにより、Const.hのenumが使用可になるので、エラーは再度消えます。

同じことをCondition.hの方でも。

まず、ヘッダーファイルのSUBJECTCONDITIONのenumを削除します。
続いて先のinclude文を挿入。

そして、
Condition.hとCondition.cppでSUBJECTCONDITIONを使用していた個所を全部直します。

置換で一発変換。

6項目だけなのでなんとかなりました…。
次からはもうちょっと計画性のある開発にします(;´・ω・)

修正後のソースは例によって最後に掲載します。

ついでに、なんか綴りが間違っていたのでSUM_NUMをSUB_NUMに置換します。

データ受け渡し

やっとデータ受け渡しに入ります。

Condition.hに、セッターゲッター関数を定義。

public:
	void SetCheckBox(SUBJECT enSub, BOOL bCheck);			// チェックボックスセッター
	void SetEditBoxIJO(SUBJECT enSub, CString strBound);	// 点以上セッター
	void SetEditBoxIKA(SUBJECT enSub, CString strBound);	// 点以下セッター
	BOOL bGetCheckBox(SUBJECT enSub);						// チェックボックスゲッター
	CString strGetEditIJO(SUBJECT enSub);					// 点以上ゲッター
	CString strGetEditIKA(SUBJECT enSub);					// 点以下ゲッター

続いてこの6つの処理をCondition.cppに追加。

セッターは2番目の引数をメンバ変数に入れるだけで、
ゲッターはメンバ変数を返すだけです。

void CCondition::SetCheckBox(SUBJECT enSub, BOOL bCheck)
{
	m_bCheck[enSub] = bCheck;
}

void CCondition::SetEditBoxIJO(SUBJECT enSub, CString strBound)
{
	m_strIJO[enSub] = strBound;
}

void CCondition::SetEditBoxIKA(SUBJECT enSub, CString strBound)
{
	m_strIKA[enSub] = strBound;
}

BOOL CCondition::bGetCheckBox(SUBJECT enSub)
{
	return m_bCheck[enSub];
}

CString CCondition::strGetEditIJO(SUBJECT enSub)
{
	return m_strIJO[enSub];
}

CString CCondition::strGetEditIKA(SUBJECT enSub)
{
	return m_strIKA[enSub];
}

これで受け渡しの準備ができました。

(さっきの修正とは別に)久しぶりにTestResultDlgの方をいじります。

まずこちらにも、チェックボックスの状態や点以下以上のデータを格納するメンバ変数を用意します。

TestResultDlg.hに以下を定義。

private:
	BOOL m_bCheckTRDlg[SUB_NUM];		// 条件変更でセットしたチェックボックス状態
	CString m_strIJOTRDlg[SUB_NUM];		// 条件変更でセットした点以上
	CString m_strIKATRDlg[SUB_NUM];		// 条件変更でセットした点以下

続いてTestResultDlg.cppのOnInitDialogに初期値設定。

BOOL CTestResultDlg::OnInitDialog()
{
	CDialogEx::OnInitDialog();

	//略

	for (int iLoop = SUB_JA; iLoop < SUB_NUM; iLoop++)
	{
		m_bCheckTRDlg[iLoop] = TRUE;			// チェックボックス
		m_strIJOTRDlg[iLoop] = _T("0");			// 点以上
		m_strIKATRDlg[iLoop] = _T("100");		// 点以下
	}

	return TRUE;  // フォーカスをコントロールに設定した場合を除き、TRUE を返します。
}

最後に、条件変更ボタンが押されるたびに呼ばれるOnBnClickedButtonConditional内に処理を追加します。

void CTestResultDlg::OnBnClickedButtonConditional()
{
	CCondition dlgCondition;

	for (int iLoop = SUB_JA; iLoop < SUB_NUM; iLoop++)
	{
		//今の状態のデータを条件変更ダイアログへ送る
		dlgCondition.SetCheckBox((SUBJECT)iLoop, m_bCheckTRDlg[iLoop]);
		dlgCondition.SetEditBoxIJO((SUBJECT)iLoop, m_strIJOTRDlg[iLoop]);
		dlgCondition.SetEditBoxIKA((SUBJECT)iLoop, m_strIKATRDlg[iLoop]);
	}

	INT_PTR iRTN = dlgCondition.DoModal();
	if (iRTN == IDOK)
	{
		// 「保存&表示」ボタン
		for (int iLoop = SUB_JA; iLoop < SUB_NUM; iLoop++)
		{
			//条件変更ダイアログからデータを取得
			m_bCheckTRDlg[iLoop] = dlgCondition.bGetCheckBox((SUBJECT)iLoop);
			m_strIJOTRDlg[iLoop] = dlgCondition.strGetEditIJO((SUBJECT)iLoop);
			m_strIKATRDlg[iLoop] = dlgCondition.strGetEditIKA((SUBJECT)iLoop);
		}
	}
	else
	{
		// キャンセルボタン
	}
}

これにより、TestResultDlg.cppとCondition.cppとで、
チェックボックスの状態や点以下以上が受けわたります。

ビルドして動作確認。

初回表示時は

大丈夫です。

エラーチェックも問題なくできてるようです。

さて、以下の状態を作ります。

これで「保存&表示」ボタンを押して再度ダイアログを表示させると、

同じ画像に見えますが、こちらは条件変更を再表示させてます

全く同じ状態が初期表示で出てきました。

データの受け渡しができたので、今回はここまで。

終わりに

結構計画性がないので、やり直しが出ちゃいましたね。

いや、一度作ってからブログ書き始めたらこんなことにはならないんですけどね。

いつかはその方針でやりたいけど…。

成果物だけ載せて今回は終わります。

TestResultDlg.h


// TestResultDlg.h : ヘッダー ファイル
//

#pragma once
#include <afx.h>
#include <vector>
#include <string>
#include "Const.h"

typedef enum
{
	LEFT_AL = 0,	//左寄せ(今のところ使用予定なし)
	CENT_AL,		//中央寄せ
	RIGHT_AL		//右寄せ
}Alignment;

// CTestResultDlg ダイアログ
class CTestResultDlg : public CDialogEx
{
// コンストラクション
public:
	CTestResultDlg(CWnd* pParent = nullptr);	// 標準コンストラクター

// ダイアログ データ
#ifdef AFX_DESIGN_TIME
	enum { IDD = IDD_TESTRESULT_DIALOG };
#endif

	protected:
	virtual void DoDataExchange(CDataExchange* pDX);	// DDX/DDV サポート

protected:
	CFont m_font;

// 実装
private:
	BOOL bCSVFormatCheck();
	BOOL bCSVDataStore(CString filePath);
	void ButtonActive(BOOL bActive);	//条件変更ボタン・表示ボタン活性判定
	std::vector<CString> GetTableData(CString strKey); // データ取得関数
	CString strAlighnment(CString strData, Alignment enAli, int iByte);	//どこ揃え?
	CString strConvertHanZen(CString strNum);	//半角→全角
	void DispHead();		// ヘッダー表示
	void DispData();		// データ表示

private:
	CMap<CString, LPCTSTR, std::vector<CString>, std::vector<CString>&> m_tableData;
	const int m_iColNum = 6;	// CSV列数(ID、国語、数学、英語、理科、社会)
	int m_iCSVNum;				// CSV行数
	BOOL m_bCheckTRDlg[SUB_NUM];		// 条件変更でセットしたチェックボックス状態
	CString m_strIJOTRDlg[SUB_NUM];		// 条件変更でセットした点以上
	CString m_strIKATRDlg[SUB_NUM];		// 条件変更でセットした点以下
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 OnBnClickedButtonCsvfileread();
	afx_msg void OnBnClickedButtonConditional();
};

TestResultDlg.cpp


// TestResultDlg.cpp : 実装ファイル
//

#include "pch.h"
#include "framework.h"
#include "TestResult.h"
#include "TestResultDlg.h"
#include "afxdialogex.h"
#include "Condition.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()


// CTestResultDlg ダイアログ



CTestResultDlg::CTestResultDlg(CWnd* pParent /*=nullptr*/)
	: CDialogEx(IDD_TESTRESULT_DIALOG, pParent)
{
	m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}

void CTestResultDlg::DoDataExchange(CDataExchange* pDX)
{
	CDialogEx::DoDataExchange(pDX);
}

BEGIN_MESSAGE_MAP(CTestResultDlg, CDialogEx)
	ON_WM_SYSCOMMAND()
	ON_WM_PAINT()
	ON_WM_QUERYDRAGICON()
	ON_BN_CLICKED(IDC_BUTTON_CSVFILEREAD, &CTestResultDlg::OnBnClickedButtonCsvfileread)
	ON_BN_CLICKED(IDC_BUTTON_CONDITIONAL, &CTestResultDlg::OnBnClickedButtonConditional)
END_MESSAGE_MAP()


// CTestResultDlg メッセージ ハンドラー

BOOL CTestResultDlg::OnInitDialog()
{
	CDialogEx::OnInitDialog();

	// "バージョン情報..." メニューをシステム メニューに追加します。

	// MS ゴシックフォントを作成
	m_font.CreatePointFont(90, _T("MS ゴシック")); // 9ポイントのMS ゴシックフォントを作成

	// ダイアログ全体にフォントを適用
	SetFont(&m_font);

	// 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: 初期化をここに追加します。
	SetWindowText(_T("テスト結果表示ダイアログ"));	//タイトル
	ButtonActive(FALSE);	//ボタン非活性化
	for (int iLoop = SUB_JA; iLoop < SUB_NUM; iLoop++)
	{
		m_bCheckTRDlg[iLoop] = TRUE;			// チェックボックス
		m_strIJOTRDlg[iLoop] = _T("0");			// 点以上
		m_strIKATRDlg[iLoop] = _T("100");		// 点以下
	}

	return TRUE;  // フォーカスをコントロールに設定した場合を除き、TRUE を返します。
}

void CTestResultDlg::OnSysCommand(UINT nID, LPARAM lParam)
{
	if ((nID & 0xFFF0) == IDM_ABOUTBOX)
	{
		CAboutDlg dlgAbout;
		dlgAbout.DoModal();
	}
	else
	{
		CDialogEx::OnSysCommand(nID, lParam);
	}
}

// ダイアログに最小化ボタンを追加する場合、アイコンを描画するための
//  下のコードが必要です。ドキュメント/ビュー モデルを使う MFC アプリケーションの場合、
//  これは、Framework によって自動的に設定されます。

void CTestResultDlg::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 CTestResultDlg::OnQueryDragIcon()
{
	return static_cast<HCURSOR>(m_hIcon);
}

// ↓↓↓↓↓↓ここから自作処理↓↓↓↓↓↓

/**
 * @fn
 * @brief	取込ボタン押下時の処理
 * @param	なし
 * @return	なし
 */
void CTestResultDlg::OnBnClickedButtonCsvfileread()
{
	CString strTitle = _T("CSV読込処理");	//読込ダイアログタイトル
	static char BASED_CODE szFilter[] = "CSV(カンマ区切り)|*.csv|";
	CString strPath = _T("");	//ファイルパス

	// CFileDialog インスタンスの作成
	CFileDialog* dlgFile = new CFileDialog(TRUE, NULL, NULL, OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT, szFilter);
	dlgFile->m_ofn.lpstrTitle = strTitle;	// タイトル設定

	// ファイルダイアログの表示
	INT_PTR iRetFileDlg = dlgFile->DoModal();
	if (iRetFileDlg == IDOK)
	{
		// 開くボタンが押下された
		//ファイルパス取得
		strPath = dlgFile->GetPathName();	//パス取得

		if (bCSVFormatCheck())
		{
			// チェックして問題なかった
			if (bCSVDataStore(strPath))
			{
				// 格納が正常に行われた
				SetDlgItemText(IDC_EDIT_CSVPATH, strPath);	// パス設定
				DispHead();	//ヘッダー表示
				DispData();	//データ表示
				ButtonActive(TRUE);	//ボタンアクティブ化
			}
			else
			{
				// 問題があった
				return;
			}
		}
		else
		{
			// チェックして問題があった
			MessageBox(_T("CSVフォーマットに誤りがあります。"));
			return;
		}
	}
	else
	{
		// キャンセルボタンが押下された
	}
}

/**
 * @fn
 * @brief	CSVフォーマットチェック
 * @param	なし
 * @return	TRUE	:	チェック問題なし
 *			FALSE	:	チェック問題あり
 */
BOOL CTestResultDlg::bCSVFormatCheck()
{
	return TRUE;
}

/**
 * @fn
 * @brief	データ取り込み
 * @param	CString	strFilePath	CSVファイルパス
 * @return	TRUE	:	問題なし
 *			FALSE	:	問題あり
 */
BOOL CTestResultDlg::bCSVDataStore(CString strFilePath)
{
	BOOL bInit = TRUE;	//最初の行判定
	try
	{
		CStdioFile cFile;
		CFileException e;
		if (cFile.Open(strFilePath, CFile::modeRead | CFile::typeText, &e))
		{
			CString strLine = _T("");		// CSVの1行を格納
			m_iCSVNum = 0;		// CSV行数
			while (cFile.ReadString(strLine))
			{
				if (bInit)
				{
					// 最初の行はヘッダーなので無視
					bInit = FALSE;
					continue;
				}

				int iCurPos = 0;
				CString strToken = strLine.Tokenize(_T(","), iCurPos);
				std::vector<CString> tokens;
				while (!strToken.IsEmpty())
				{
					tokens.push_back(strToken);
					strToken = strLine.Tokenize(_T(","), iCurPos);
				}

				if (tokens.size() >= m_iColNum)
				{
					CString strKey = tokens[0];	// ID
					tokens.erase(tokens.begin());
					m_tableData[strKey] = tokens;	//IDに対しての各データ
				}

				m_iCSVNum++;
			}
			cFile.Close();
		}
		else
		{
			if (e.m_cause == CFileException::sharingViolation)
			{
				AfxMessageBox(_T("共有違反が発生しました。ファイルが他のプログラムで開かれている可能性があります。"));
				return FALSE;
			}
			else
			{
				CString errorMsg;
				e.GetErrorMessage(errorMsg.GetBuffer(256), 256);
				errorMsg.ReleaseBuffer();
				AfxMessageBox(_T("ファイルエラー: ") + errorMsg);
				return FALSE; // 処理を終了
			}
		}
	}
	catch (std::exception& e)
	{
		// 予期しない標準例外
		CString errorMsg(e.what());
		AfxMessageBox(_T("エラー: ") + errorMsg);
		return FALSE;
	}
	catch (...)
	{
		AfxMessageBox(_T("不明なエラーが発生しました。"));
		return FALSE;
	}

	return TRUE;
}

/**
 * @fn
 * @brief	ボタン活性非活性処理
 * @param	TRUE	:	活性化
 *			FALSE	:	非活性化
 * @return	なし
 */
void CTestResultDlg::ButtonActive(BOOL bActive)
{
	//条件変更ボタン
	CButton* pButton = (CButton*)GetDlgItem(IDC_BUTTON_CONDITIONAL);
	pButton->EnableWindow(bActive);

	//表示ボタン
	pButton = (CButton*)GetDlgItem(IDC_BUTTON_DISP);
	pButton->EnableWindow(bActive);
}

/**
 * @fn
 * @brief	データ取得処理
 * @param	CString	strKey	主キー
 * @return	vector<CString>		主キーに対する各データ
 */
std::vector<CString> CTestResultDlg::GetTableData(CString strKey)
{
	std::vector<CString> values;
	if (m_tableData.Lookup(strKey, values))	//キーに対するデータを取ってくるよ
	{
		return values;
	}
	return {};
}

/**
 * @fn
 * @brief	肥大r・中央・右寄席
 * @param	CString	s	trData	寄せたいデータ
 *			Alignment	enAli	どこ寄せ
 *			int			iStrNum	最終的な文字数
 * @return	CString	調整結果
 */
CString CTestResultDlg::strAlighnment(CString strData, Alignment enAli, int iStrNum)
{
	CString strRtn = strData;	//返却値

	size_t Length = _tcsclen(strData);	//文字数


	switch (enAli)
	{
	case LEFT_AL:
		for (int i = 0; iStrNum - Length; i++)
		{
			strRtn = strRtn + _T(" ");	//全角スペース
		}
		break;
	case CENT_AL:
		//左側
		for (int i = 0; i < (iStrNum - Length) / 2; i++)
		{
			strRtn = strRtn + _T(" ");	//全角スペース
		}
		//右側
		for (int i = 0; i < iStrNum - Length - (iStrNum - Length) / 2; i++)
		{
			strRtn = _T(" ") + strRtn;	//全角スペース
		}
		break;
	case RIGHT_AL:
		for (int i = 0; i < iStrNum - Length; i++)
		{
			strRtn = _T(" ") + strRtn;	//全角スペース
		}
		break;
	default:
		break;
	}

	return strRtn;
}

/**
 * @fn
 * @brief	半角を全角に
 * @param	CString	strNum	半角数値
 * @return	CString	全角文字
 */
CString CTestResultDlg::strConvertHanZen(CString strNum)
{
	CString strRtn = _T("");
	CString strTemp = strNum;
	while (!strTemp.IsEmpty())
	{
		if (IsDBCSLeadByte(strTemp[0]) != 0)
		{
			//全角文字が入っている
			AfxMessageBox(_T("データ不備"));
			return _T("");
		}

		CString strInit = strTemp.Mid(0, 1);	//最初の文字
		if (strInit == _T("0"))
		{
			strRtn += _T("0");
		}
		else if (strInit == _T("1"))
		{
			strRtn += _T("1");
		}
		else if (strInit == _T("2"))
		{
			strRtn += _T("2");
		}
		else if (strInit == _T("3"))
		{
			strRtn += _T("3");
		}
		else if (strInit == _T("4"))
		{
			strRtn += _T("4");
		}
		else if (strInit == _T("5"))
		{
			strRtn += _T("5");
		}
		else if (strInit == _T("6"))
		{
			strRtn += _T("6");
		}
		else if (strInit == _T("7"))
		{
			strRtn += _T("7");
		}
		else if (strInit == _T("8"))
		{
			strRtn += _T("8");
		}
		else if (strInit == _T("9"))
		{
			strRtn += _T("9");
		}
		else
		{
			// 数字以外
			AfxMessageBox(_T("データ不備"));
		}
		strTemp = strTemp.Mid(1);
	}

	return strRtn;
}

/**
 * @fn
 * @brief	ヘッダー表示処理
 * @param	なし
 * @return	なし
 */
void CTestResultDlg::DispHead()
{
	CString strHead = strAlighnment(_T("ID"), CENT_AL, 6) + 
				strAlighnment(_T("国語"), CENT_AL, 6) + 
				strAlighnment(_T("数学"), CENT_AL, 6) + 
				strAlighnment(_T("英語"), CENT_AL, 6) +
				strAlighnment(_T("理科"), CENT_AL, 6) + 
				strAlighnment(_T("社会"), CENT_AL, 6);

	SetDlgItemText(IDC_EDIT_DISP_HEADER, strHead);	// 表示
}

/**
 * @fn
 * @brief	データ表示処理
 * @param	なし
 * @return	なし
 */
void CTestResultDlg::DispData()
{
	CString strData = _T("");
	std::vector<CString> values;
	int iID = 0;
	CString strID = _T("");

	for (int iLoop = 0; iLoop < m_iCSVNum; iLoop++)
	{
		iID = iLoop + 1;				// ID
		strID.Format(_T("%d"), iID);	// ID CStringで
		values = GetTableData(strID);	// IDに対するデータ

		strData += strAlighnment(strConvertHanZen(strID), RIGHT_AL, 6);		// IDは全角に直して右詰

		for (int iSub = SUB_JA; iSub < SUB_NUM; iSub++)
		{
			strData += strAlighnment(strConvertHanZen(values[iSub]), RIGHT_AL, 6);	//各データを全角に直して右詰
		}
		
		if (iLoop != m_iCSVNum - 1)
			strData += _T("\r\n");	//最終行以外は改行
	}

	SetDlgItemText(IDC_EDIT_DISP_DATA, strData);	// 表示
}

/**
 * @fn
 * @brief	条件変更ボタン押下時の処理
 * @param	なし
 * @return	なし
 */
void CTestResultDlg::OnBnClickedButtonConditional()
{
	CCondition dlgCondition;

	for (int iLoop = SUB_JA; iLoop < SUB_NUM; iLoop++)
	{
		//今の状態のデータを条件変更ダイアログへ送る
		dlgCondition.SetCheckBox((SUBJECT)iLoop, m_bCheckTRDlg[iLoop]);
		dlgCondition.SetEditBoxIJO((SUBJECT)iLoop, m_strIJOTRDlg[iLoop]);
		dlgCondition.SetEditBoxIKA((SUBJECT)iLoop, m_strIKATRDlg[iLoop]);
	}

	INT_PTR iRTN = dlgCondition.DoModal();
	if (iRTN == IDOK)
	{
		// 「保存&表示」ボタン
		for (int iLoop = SUB_JA; iLoop < SUB_NUM; iLoop++)
		{
			//条件変更ダイアログからデータを取得
			m_bCheckTRDlg[iLoop] = dlgCondition.bGetCheckBox((SUBJECT)iLoop);
			m_strIJOTRDlg[iLoop] = dlgCondition.strGetEditIJO((SUBJECT)iLoop);
			m_strIKATRDlg[iLoop] = dlgCondition.strGetEditIKA((SUBJECT)iLoop);
		}
	}
	else
	{
		// キャンセルボタン
	}
}

Condition.h

#pragma once
#include "afxdialogex.h"
#include "Const.h"

// CCondition ダイアログ

class CCondition : public CDialog
{
	DECLARE_DYNAMIC(CCondition)

public:
	CCondition(CWnd* pParent = nullptr);   // 標準コンストラクター
	virtual ~CCondition();

// ダイアログ データ
#ifdef AFX_DESIGN_TIME
	enum { IDD = IDD_CONDITION };
#endif

private:
	BOOL m_bCheck[SUB_NUM];	// 各チェックボックスの状態
	CButton	m_chk[SUB_NUM];	// 各チェックボックス用メンバ変数

	CString m_strIJO[SUB_NUM];		// 各教科の点以上
	CString m_strIKA[SUB_NUM];		// 各教科の点以下
	CEdit m_editIJO[SUB_NUM];		// 各エディットボックス用メンバ変数(以上用)
	CEdit m_editIKA[SUB_NUM];		// 各エディットボックス用メンバ変数(以下用)

private:
	void EditEnableWrite();			// エディットボックス入力可能制御
	BOOL bControlCheck();			// 各コントロールのチェック

public:
	void SetCheckBox(SUBJECT enSub, BOOL bCheck);			// チェックボックスセッター
	void SetEditBoxIJO(SUBJECT enSub, CString strBound);	// 点以上セッター
	void SetEditBoxIKA(SUBJECT enSub, CString strBound);	// 点以下セッター
	BOOL bGetCheckBox(SUBJECT enSub);						// チェックボックスゲッター
	CString strGetEditIJO(SUBJECT enSub);					// 点以上ゲッター
	CString strGetEditIKA(SUBJECT enSub);					// 点以下ゲッター

protected:
	virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV サポート

	DECLARE_MESSAGE_MAP()
public:
	virtual BOOL OnInitDialog();
	afx_msg void OnBnClickedCheck1jp();			//国語チェックボックス
	afx_msg void OnBnClickedCheck2ma();			//数学チェックボックス
	afx_msg void OnBnClickedCheck3en();			//英語チェックボックス
	afx_msg void OnBnClickedCheck4sc();			//理科チェックボックス
	afx_msg void OnBnClickedCheck5ss();			//社会チェックボックス
	afx_msg void OnBnClickedButtonSavedisp();
	afx_msg void OnBnClickedButtonSave();
};

Condition.cpp

// Condition.cpp : 実装ファイル
//

#include "pch.h"
#include "TestResult.h"
#include "afxdialogex.h"
#include "Condition.h"


// CCondition ダイアログ

IMPLEMENT_DYNAMIC(CCondition, CDialog)

/**
 * @fn
 * @brief	コンストラクタ
 * @param	なし
 * @return	なし
 */
CCondition::CCondition(CWnd* pParent /*=nullptr*/)
	: CDialog(IDD_CONDITION, pParent)
{
	for (int iLoop = SUB_JA; iLoop < SUB_NUM; iLoop++)
	{
		m_bCheck[iLoop] = TRUE;			// チェックボックス状態初期化
		m_strIJO[iLoop] = _T("0");		// 以上初期化
		m_strIKA[iLoop] = _T("100");	// 以下初期化
	}
}

CCondition::~CCondition()
{
}

void CCondition::DoDataExchange(CDataExchange* pDX)
{
	CDialog::DoDataExchange(pDX);
	DDX_Control(pDX, IDC_CHECK_1JP, m_chk[SUB_JA]);	// 国語チェックボックス
	DDX_Control(pDX, IDC_CHECK_2MA, m_chk[SUB_MA]);	// 数学チェックボックス
	DDX_Control(pDX, IDC_CHECK_3EN, m_chk[SUB_EN]);	// 英語チェックボックス
	DDX_Control(pDX, IDC_CHECK_4SC, m_chk[SUB_SC]);	// 理科チェックボックス
	DDX_Control(pDX, IDC_CHECK_5SS, m_chk[SUB_SS]);	// 社会チェックボックス
	DDX_Control(pDX, IDC_EDIT_1JP_IJO, m_editIJO[SUB_JA]);	// 国語エディットボックス(以下)
	DDX_Control(pDX, IDC_EDIT_2MA_IJO, m_editIJO[SUB_MA]);	// 数学エディットボックス(以下)
	DDX_Control(pDX, IDC_EDIT_3EN_IJO, m_editIJO[SUB_EN]);	// 英語エディットボックス(以下)
	DDX_Control(pDX, IDC_EDIT_4SC_IJO, m_editIJO[SUB_SC]);	// 理科エディットボックス(以下)
	DDX_Control(pDX, IDC_EDIT_5SS_IJO, m_editIJO[SUB_SS]);	// 社会エディットボックス(以下)
	DDX_Control(pDX, IDC_EDIT_1JP_IKA, m_editIKA[SUB_JA]);	// 国語エディットボックス(以上)
	DDX_Control(pDX, IDC_EDIT_2MA_IKA, m_editIKA[SUB_MA]);	// 数学エディットボックス(以上)
	DDX_Control(pDX, IDC_EDIT_3EN_IKA, m_editIKA[SUB_EN]);	// 英語エディットボックス(以上)
	DDX_Control(pDX, IDC_EDIT_4SC_IKA, m_editIKA[SUB_SC]);	// 理科エディットボックス(以上)
	DDX_Control(pDX, IDC_EDIT_5SS_IKA, m_editIKA[SUB_SS]);	// 社会エディットボックス(以上)

	DDX_Text(pDX, IDC_EDIT_1JP_IJO, m_strIJO[SUB_JA]);		// 国語エディットボックス(以下)の数値をメンバ変数に
	DDX_Text(pDX, IDC_EDIT_2MA_IJO, m_strIJO[SUB_MA]);		// 国語エディットボックス(以下)の数値をメンバ変数に
	DDX_Text(pDX, IDC_EDIT_3EN_IJO, m_strIJO[SUB_EN]);		// 国語エディットボックス(以下)の数値をメンバ変数に
	DDX_Text(pDX, IDC_EDIT_4SC_IJO, m_strIJO[SUB_SC]);		// 国語エディットボックス(以下)の数値をメンバ変数に
	DDX_Text(pDX, IDC_EDIT_5SS_IJO, m_strIJO[SUB_SS]);		// 国語エディットボックス(以下)の数値をメンバ変数に
	DDX_Text(pDX, IDC_EDIT_1JP_IKA, m_strIKA[SUB_JA]);		// 国語エディットボックス(以上)の数値をメンバ変数に
	DDX_Text(pDX, IDC_EDIT_2MA_IKA, m_strIKA[SUB_MA]);		// 国語エディットボックス(以上)の数値をメンバ変数に
	DDX_Text(pDX, IDC_EDIT_3EN_IKA, m_strIKA[SUB_EN]);		// 国語エディットボックス(以上)の数値をメンバ変数に
	DDX_Text(pDX, IDC_EDIT_4SC_IKA, m_strIKA[SUB_SC]);		// 国語エディットボックス(以上)の数値をメンバ変数に
	DDX_Text(pDX, IDC_EDIT_5SS_IKA, m_strIKA[SUB_SS]);		// 国語エディットボックス(以上)の数値をメンバ変数に
}


BEGIN_MESSAGE_MAP(CCondition, CDialog)
	ON_BN_CLICKED(IDC_CHECK_1JP, &CCondition::OnBnClickedCheck1jp)
	ON_BN_CLICKED(IDC_CHECK_2MA, &CCondition::OnBnClickedCheck2ma)
	ON_BN_CLICKED(IDC_CHECK_3EN, &CCondition::OnBnClickedCheck3en)
	ON_BN_CLICKED(IDC_CHECK_4SC, &CCondition::OnBnClickedCheck4sc)
	ON_BN_CLICKED(IDC_CHECK_5SS, &CCondition::OnBnClickedCheck5ss)
	ON_BN_CLICKED(IDC_BUTTON_SAVEDISP, &CCondition::OnBnClickedButtonSavedisp)
	ON_BN_CLICKED(IDC_BUTTON_SAVE, &CCondition::OnBnClickedButtonSave)
END_MESSAGE_MAP()


// CCondition メッセージ ハンドラー


BOOL CCondition::OnInitDialog()
{
	CDialog::OnInitDialog();

	// TODO: ここに初期化を追加してください
	SetWindowText(_T("条件変更ダイアログ"));	//タイトル
	for (int iLoop = SUB_JA; iLoop < SUB_NUM; iLoop++)
	{ 
		int iCheck = BST_CHECKED;
		if (!m_bCheck[iLoop])
			iCheck = BST_UNCHECKED;
		m_chk[iLoop].SetCheck(iCheck);						// チェックボックスの状態
		m_editIJO[iLoop].SetWindowTextA(m_strIJO[iLoop]);	// 以上のエディットボックスに初期値セット
		m_editIKA[iLoop].SetWindowTextA(m_strIKA[iLoop]);	// 以下のエディットボックスに初期値セット
	}
	EditEnableWrite();	//制御チェック

	UpdateData(FALSE);
	return TRUE;  // return TRUE unless you set the focus to a control
	// 例外 : OCX プロパティ ページは必ず FALSE を返します。
}

void CCondition::SetCheckBox(SUBJECT enSub, BOOL bCheck)
{
	m_bCheck[enSub] = bCheck;
}

void CCondition::SetEditBoxIJO(SUBJECT enSub, CString strBound)
{
	m_strIJO[enSub] = strBound;
}

void CCondition::SetEditBoxIKA(SUBJECT enSub, CString strBound)
{
	m_strIKA[enSub] = strBound;
}

BOOL CCondition::bGetCheckBox(SUBJECT enSub)
{
	return m_bCheck[enSub];
}

CString CCondition::strGetEditIJO(SUBJECT enSub)
{
	return m_strIJO[enSub];
}

CString CCondition::strGetEditIKA(SUBJECT enSub)
{
	return m_strIKA[enSub];
}

/**
 * @fn
 * @brief	エディットボックス制御処理
 * @param	なし
 * @return	なし
 */
void CCondition::EditEnableWrite()
{
	for (int iLoop = SUB_JA; iLoop < SUB_NUM; iLoop++)
	{
		if (m_chk[iLoop].GetCheck() == BST_CHECKED)
		{
			// チェックが入っている状態
			// エディットボックス書き込み可
			m_editIJO[iLoop].SetReadOnly(FALSE);
			m_editIKA[iLoop].SetReadOnly(FALSE);
		}
		else
		{
			// チェックが入っていない状態
			// エディットボックス書き込み不可
			m_editIJO[iLoop].SetReadOnly(TRUE);
			m_editIKA[iLoop].SetReadOnly(TRUE);
		}
	}
}

/**
 * @fn
 * @brief	国語チェックボックス変更時の処理
 * @param	なし
 * @return	なし
 */
void CCondition::OnBnClickedCheck1jp()
{
	EditEnableWrite();
}

/**
 * @fn
 * @brief	数学チェックボックス変更時の処理
 * @param	なし
 * @return	なし
 */
void CCondition::OnBnClickedCheck2ma()
{
	EditEnableWrite();
}

/**
 * @fn
 * @brief	英語チェックボックス変更時の処理
 * @param	なし
 * @return	なし
 */
void CCondition::OnBnClickedCheck3en()
{
	EditEnableWrite();
}

/**
 * @fn
 * @brief	理科チェックボックス変更時の処理
 * @param	なし
 * @return	なし
 */
void CCondition::OnBnClickedCheck4sc()
{
	EditEnableWrite();
}

/**
 * @fn
 * @brief	社会チェックボックス変更時の処理
 * @param	なし
 * @return	なし
 */
void CCondition::OnBnClickedCheck5ss()
{
	EditEnableWrite();
}

/**
 * @fn
 * @brief	保存&表示ボタン押下時の処理
 * @param	なし
 * @return	なし
 */
void CCondition::OnBnClickedButtonSavedisp()
{
	UpdateData(TRUE);	//コントロールからメンバ変数にデータ転送
	// TODO: ここにコントロール通知ハンドラー コードを追加します。
	if (!bControlCheck())
		return;

	UpdateData(FALSE);	//メンバ変数からコントロールにデータ転送(念のため)

	CDialog::OnOK();	//画面を閉じる
}

/**
 * @fn
 * @brief	保存ボタン押下時の処理
 * @param	なし
 * @return	なし
 */
void CCondition::OnBnClickedButtonSave()
{
	UpdateData(TRUE);	//コントロールからメンバ変数にデータ転送
	// TODO: ここにコントロール通知ハンドラー コードを追加します。
	if (!bControlCheck())
		return;
	UpdateData(FALSE);	//メンバ変数からコントロールにデータ転送(念のため)

}

/**
 * @fn
 * @brief	入力チェック
 * @param	なし
 * @return	BOOL	TRUE	問題なし
 *					FALSE	問題あり
 */
BOOL CCondition::bControlCheck()
{
	if (m_chk[SUB_JA].GetCheck() == BST_UNCHECKED &&
		m_chk[SUB_MA].GetCheck() == BST_UNCHECKED &&
		m_chk[SUB_EN].GetCheck() == BST_UNCHECKED &&
		m_chk[SUB_SC].GetCheck() == BST_UNCHECKED &&
		m_chk[SUB_SS].GetCheck() == BST_UNCHECKED)
	{
		// すべてのチェックボックスにチェックが入っていない
		AfxMessageBox(_T("少なくとも1つのチェックボックスに\r\nチェックを入れてください。"));
		return FALSE;
	}

	for (int iLoop = SUB_JA; iLoop < SUB_NUM; iLoop++)
	{
		//チェックが入っていない場合は入力チェック対象外
		if (m_chk[iLoop].GetCheck() == BST_UNCHECKED)
		{
			m_bCheck[iLoop] = FALSE;
			continue;
		}

		// 点以上のチェック
		int iScoreIJO = _ttoi(m_strIJO[iLoop]);	//int型に変換
		if (iScoreIJO < 0 || 100 < iScoreIJO)
		{
			AfxMessageBox(_T("0以上100以下の数字を入力してください。"));
			m_editIJO[iLoop].SetFocus();	//該当エディットボックスにセットフォーカス
			return FALSE;
		}

		//点以下のチェック
		int iScoreIKA = _ttoi(m_strIKA[iLoop]);	//int型に変換
		if (iScoreIKA < 0 || 100 < iScoreIKA)
		{
			AfxMessageBox(_T("0以上100以下の数字を入力してください。"));
			m_editIKA[iLoop].SetFocus();	//該当エディットボックスにセットフォーカス
			return FALSE;
		}

		//以下と以上の関係性チェック
		if (iScoreIKA < iScoreIJO)
		{
			AfxMessageBox(_T("数値の大小が正しくありません。"));
			m_editIJO[iLoop].SetFocus();	//該当エディットボックス(以上の方)にセットフォーカス
			return FALSE;
		}

		m_bCheck[iLoop] = TRUE;						//メンバ変数更新	
		m_strIJO[iLoop].Format("%d", iScoreIJO);	//メンバ変数更新
		m_strIKA[iLoop].Format("%d", iScoreIKA);	//メンバ変数更新
	}

	//チェックは大丈夫
	return TRUE;
}
MFCで、CSVからデータを抽出・表示させる処理 7
前回までの記事ヘッダー部表示制御前回、条件変更処理で、どの列を表示させるかをチェックボックスのチェック状態で決めていました。その後、ゲッター関数でTestResultDlgの方にも状態が保存されていることも確認しました。今回はこれを使ってヘ...

コメント

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