[카테고리:] <span>MFC 실습</span>

메인 대화상자

유튜브에 무료 동영상 강의를 업로드하였습니다.

EndDialogEventHandler.h

등록 대화상자는 하나가 뜬 상태에서 다시 뜨지 않게 처리할 거예요.

물론 등록 대화상자를 닫으면 다시 뜨게 할 수 있어야 합니다.

메인 대화상자에서는 등록 대화상자가 닫히는 것을 알아야 이를 처리할 수 있어요.

이를 위해 EndDilaogEventHandler 클래스를 정의합니다.

순수 가상 메서드로 대화상자가 종료되었음을 수신한 후에 처리할 메서드를 약속합니다.

#pragma once
class EndDialogEventHandler
{
public:
	virtual void EndedDialog() = 0;
};

도서 관리 프로그램Dlg.h

메인 대화상자의 디폴트 이름에는 프로젝트 이름이 들어갑니다.

메인 대화상자는 기반 클래스인 CDialogEx 외에도 이벤트 처리를 위해 EndDialogEventHandelr, AddBookEventHandler, ModifyEventHandler를 기반 형식으로 정의할게요.

다중 상속에서 기반 형식의 멤버의 충돌 등이 발생할 수 있어 코드를 모호하게 만들 수 있습니다. 이에 Java나 C#에서는 제공하지 않습니다. 하지만 여러 기반 인터페이스를 기반으로 구현 약속하는 것은 가능합니다.

여기에서 이벤트 핸들러는 순수 가상 메서드만 멤버로 두기 때문에 Java나 C#에서의 인터페이스 역할을 할 뿐이기 때문에 다중 상속이 갖는 모호한 단점은 나타나지 않습니다.

class C도서관리프로그램Dlg : public CDialogEx,
	public EndDialogEventHandler,
	public AddBookEventHandler,
	public ModifyBookEventHandler

등록 대화상자 개체를 기억할 멤버를 캡슐화합니다.

	MyRegDialog* mrd;

컨트롤 변수를 멤버로 캡슐화합니다.

여기에서는 IDC_LIST_BOOK과 매핑하는 컨트롤 변수로 blist를 캡슐화합니다.

그리고 IDC_BUTTON_REMVE와 매핑하는 컨트롤 변수 btn_remove를 캡슐화합니다.

	CListCtrl blist;
	CButton btn_remove;

등록 버튼(IDC_BUTTON_REG)을 클릭 메시지 처리기를 캡슐화합니다.

	afx_msg void OnBnClickedButtonReg();

등록 대화상자 종료 이벤트 핸들러를 캡슐화합니다.

	afx_msg LRESULT OnEndedReg(WPARAM, LPARAM);

등록 대화상자 종료 이벤트 핸들러에서 사용자 정의 이벤트를 사용합니다.

이 때 수행할 처리할 메시지 처리기를 캡슐화합니다.

	afx_msg LRESULT OnEndedReg(WPARAM, LPARAM);

도서 추가 이벤트 핸들러를 캡슐화합니다.

	virtual void AddedBook(Book* book);

자세히 보기 버튼 클릭 메시지 처리기를 캡슐화합니다.

	afx_msg void OnBnClickedButtonVerify();

리스트 항목 변경 메시지 처리기를 캡슐화합니다.

	afx_msg void OnLvnItemchangedListBook(NMHDR* pNMHDR, LRESULT* pResult);

삭제 버튼 클릭 및 변경 버튼 클릭 메시지 처리기를 캡슐화합니다.

	afx_msg void OnBnClickedButtonRemove();
	afx_msg void OnBnClickedButtonModify();

도서 변경 이벤트 핸들러를 캡슐화합니다.

	virtual void ModifiedBook(Book* book);

DestroyWindow 메서드도 재정의합니다.

	virtual BOOL DestroyWindow();

다음은 도서관리 프로그램 Dlg.h 파일의 소스 코드 내용입니다.


// 도서 관리 프로그램Dlg.h: 헤더 파일
//

#pragma once
#include "MyRegDialog.h"
#include "ModifyDialog.h"
#include "VerifyDilaog.h"
#include "EndDialogEventHandler.h"
#include "BookEventHandler.h"
#include "BookManager.h"
#include "Book.h"
// C도서관리프로그램Dlg 대화 상자
class C도서관리프로그램Dlg : public CDialogEx,
	public EndDialogEventHandler,
	public AddBookEventHandler,
	public ModifyBookEventHandler
{
	MyRegDialog* mrd;
// 생성입니다.
public:
	C도서관리프로그램Dlg(CWnd* pParent = nullptr);	// 표준 생성자입니다.

// 대화 상자 데이터입니다.
#ifdef AFX_DESIGN_TIME
	enum { IDD = IDD_DIALOG_MAIN};
#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()
	
	CListCtrl blist;
	CButton btn_remove;
public:
	afx_msg void OnBnClickedButtonReg();
	virtual void EndedDialog();
	afx_msg LRESULT OnEndedReg(WPARAM, LPARAM);

	virtual void AddedBook(Book* book);
	afx_msg void OnBnClickedButtonVerify();
	afx_msg void OnLvnItemchangedListBook(NMHDR* pNMHDR, LRESULT* pResult);

	afx_msg void OnBnClickedButtonRemove();
	afx_msg void OnBnClickedButtonModify();
	virtual void ModifiedBook(Book* book);
	virtual BOOL DestroyWindow();
};

도서 관리 프로그램 Dlg.cpp

등록 대화상자가 닫혔을 때 처리를 위해 사용자 정의 메시지를 정의합시다.

enum MY_W_MSG
{
	MWM_ENDED_REG = WM_APP+1
};

OnInitDialog

OnInitDialog 메서드의 TODO 주석 밑에 초기화 작업을 작성합시다.

먼저 등록 대화상자 개체를 기억할 mdr 멤버를 0으로 초기화합니다.

그리고 BookManager 개체에 도서 추가 이벤트 및 변경 이벤트를 수신하기 위해 이벤트 핸들러를 등록합니다.

	// TODO: 여기에 추가 초기화 작업을 추가합니다.
	mrd = 0;
	BookManager& bm = BookManager::GetBookManager();
	bm.AddABEventHandler(this);//도서 추가하면 나한테 알려줘
	bm.AddMBEventHandler(this);//도서 변경하면 나한테 알려줘

도서 목록 컨트롤에 컬럼을 추가합니다. 여기에서는 도서 번호, 제목, 일자를 컬럼으로 추가할게요.

	LPWSTR texts[3] = {TEXT("No"),TEXT("제목"),TEXT("일자")};
	int widths[3] = { 50,200,120 };
	LV_COLUMN col;
	col.mask = LVCF_FMT | LVCF_SUBITEM | LVCF_TEXT | LVCF_WIDTH;
	col.fmt = LVCFMT_CENTER;
	for (int i = 0; i < 3; i++)
	{
		col.pszText = texts[i];
		col.iSubItem = i;
		col.cx = widths[i];
		blist.InsertColumn(i, &col);
	}

도서 관리자 개체의 Load 메서드를 호출합니다.

	bm.Load();

OnInitDialog 소스 코드

BOOL C도서관리프로그램Dlg::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);
		}
	}

	// 이 대화 상자의 아이콘을 설정합니다.  응용 프로그램의 주 창이 대화 상자가 아닐 경우에는
	//  프레임워크가 이 작업을 자동으로 수행합니다.
	SetIcon(m_hIcon, TRUE);			// 큰 아이콘을 설정합니다.
	SetIcon(m_hIcon, FALSE);		// 작은 아이콘을 설정합니다.

	// TODO: 여기에 추가 초기화 작업을 추가합니다.
	mrd = 0;
	BookManager& bm = BookManager::GetBookManager();
	bm.AddABEventHandler(this);//도서 추가하면 나한테 알려줘
	bm.AddMBEventHandler(this);//도서 변경하면 나한테 알려줘

	LPWSTR texts[3] = {TEXT("No"),TEXT("제목"),TEXT("일자")};
	int widths[3] = { 50,200,120 };
	LV_COLUMN col;
	col.mask = LVCF_FMT | LVCF_SUBITEM | LVCF_TEXT | LVCF_WIDTH;
	col.fmt = LVCFMT_CENTER;
	for (int i = 0; i < 3; i++)
	{
		col.pszText = texts[i];
		col.iSubItem = i;
		col.cx = widths[i];
		blist.InsertColumn(i, &col);
	}
	bm.Load();
	return TRUE;  // 포커스를 컨트롤에 설정하지 않으면 TRUE를 반환합니다.
}

OnBnClickedButtonReg

등록 대화상자 개체가 없는지 확인하여 등록 대화상자를 띄웁니다.

그리고 등록 대화상자 종료 이벤트를 수신하기 위해 이벤트 핸들러를 등록합니다.

그리고 등록 대화상자의 Create 메서드와 ShowWindow 메서드를 호출합니다.

Create 메서드에 DesktopWindow를 두 번째 인자로 전달하면 메인 대화상자와 Sub 대화상자의 Z순서가 자연스럽게 동작합니다.

만약 이를 지정하지 않으면 메인 대화상자는 Sub 대화상자보다 언제나 Z순서가 밑에 가 있습니다.

void C도서관리프로그램Dlg::OnBnClickedButtonReg()
{
	// TODO: 여기에 컨트롤 알림 처리기 코드를 추가합니다.
	if (mrd == 0)
	{
		mrd = new MyRegDialog();
		mrd->AddEndedEventHandler(this);//너 죽을 때 나한테 알려줘
		mrd->Create(IDD_DIALOG_REG,CWnd::GetDesktopWindow());
		mrd->ShowWindow(SW_SHOW);
	}
}

EndedDialog

등록 대화상자 종료 이벤트 핸들러에서는 MWM_ENDED_REG 메시지를 붙여줍니다.

등록 대화상자 종료 이벤트 핸들러를 호출해 주는 것이 등록 대화상자 내부 메서드에서입니다.

여기에서는 등록 대화상자 개체를 소멸해야 하는데 딜레마에 빠집니다.

등록 대화상자 종료 이벤트 핸들러를 동작하고 있다는 것은 아직 이를 호출한 등록 대화상자 개체의 특정 메서드가 끝나지 않은 시점입니다. 이는 등록 대화상자 개체를 소멸할 수 없다는 것이죠.

이를 처리하기 위해 PostMessage로 사용자 정의 메시지를 보내고 해당 메시지 처리기에서 등록 대화상자 개체를 소멸하는 방식으로 딜레마를 해결할 수 있습니다.

void C도서관리프로그램Dlg::EndedDialog()
{
	PostMessage(MWM_ENDED_REG, 0, 0);
}

LRESULT C도서관리프로그램Dlg::OnEndedReg(WPARAM, LPARAM)
{
	if (mrd == 0)
	{
		delete mrd;
	}
	mrd = 0;
	return 0;
}

AddedBook

도서 추가 이벤트 핸들러에서는 입력 인자로 받은 도서 개체의 정보로 리스트 컨트롤에 항목을 추가합니다.

void C도서관리프로그램Dlg::AddedBook(Book* book)
{
	int index = blist.GetItemCount();
	wchar_t nobuf[256];
	wsprintf(nobuf, TEXT("%d"), book->GetNo());
	blist.InsertItem(index, nobuf);
	blist.SetItemText(index, 1, book->GetTitle());
	COleDateTime dt = book->GetPubdate();
	wchar_t dtbuf[256];
	wsprintf(dtbuf, TEXT("%d년 %d월 %d일"), dt.GetYear(), dt.GetMonth(), dt.GetDay());
	blist.SetItemText(index, 2, dtbuf);
}

OnBnClickedButtonVerify

자세히 보기 대화상자를 생성하고 시각화합니다.

이 프로그램에서 자세히 보기 대화상자는 동시에 여러 개를 띄울 수 있게 구현할 것입니다.

void C도서관리프로그램Dlg::OnBnClickedButtonVerify()
{
	// TODO: 여기에 컨트롤 알림 처리기 코드를 추가합니다.
	VerifyDilaog* vd = new VerifyDilaog();
	vd->Create(IDD_DIALOG_VERIFY, CWnd::GetDesktopWindow());
	vd->ShowWindow(SW_SHOW);
}

OnLvnItemchangedListBook

리스트 컨트롤은 여러 항목을 선택할 수도 있습니다.

선택 항목이 있는지 여부에 따라 삭제 버튼의 활성화를 결정합니다.

void C도서관리프로그램Dlg::OnLvnItemchangedListBook(NMHDR* pNMHDR, LRESULT* pResult)
{
	LPNMLISTVIEW pNMLV = reinterpret_cast<LPNMLISTVIEW>(pNMHDR);
	// TODO: 여기에 컨트롤 알림 처리기 코드를 추가합니다.
	*pResult = 0;
	
	if (blist.GetSelectedCount())
	{
		btn_remove.EnableWindow(TRUE);
	}
	else
	{
		btn_remove.EnableWindow(FALSE);
	}
}

OnBnClickedButtonRemove

리스트 컨트롤의 선택 항목을 얻어옵니다.

그리고 얻어온 정보로 도서 번호를 구하여 BookManager에게 도서 삭제를 요청합니다.

물론 리스트 컨트롤에서도 항목 삭제해야겠죠.

void C도서관리프로그램Dlg::OnBnClickedButtonRemove()
{
	// TODO: 여기에 컨트롤 알림 처리기 코드를 추가합니다.
	BookManager& bm = BookManager::GetBookManager();
	POSITION pos = blist.GetFirstSelectedItemPosition();
	int index = blist.GetNextSelectedItem(pos);
	if (index == -1)
	{
		return;
	}
	LVITEM li = { 0 };
	li.iItem = index;
	li.iSubItem = 0;
	wchar_t buf[256]=TEXT("");
	li.mask = LVIF_TEXT;
	li.pszText = buf;
	li.cchTextMax = 256;//버퍼 크기
	blist.GetItem(&li);
	int bno = _wtoi(buf);
	if (bm.RemoveBook(bno))
	{
		blist.DeleteItem(index);
	}
}

OnBnClickedButtonModify

변경 버튼을 클릭하면 변경 대화상자를 생성하고 시각화합니다.

void C도서관리프로그램Dlg::OnBnClickedButtonModify()
{	
	// TODO: 여기에 컨트롤 알림 처리기 코드를 추가합니다.
	ModifyDialog* md = new ModifyDialog();
	md->Create(IDD_DIALOG_MODIFY, CWnd::GetDesktopWindow());
	md->ShowWindow(SW_SHOW);
}

ModifiedBook

도서 변경 이벤트 핸들러를 정의합시다.

먼저 수정한 도서 개체의 도서 번호를 얻어옵니다.

그리고 리스트 컨트롤에서 같은 도서 번호인 항목을 찾아 제목과 출판일을 수정합니다.

void C도서관리프로그램Dlg::ModifiedBook(Book* book)
{
	int bno = book->GetNo();
	LVITEM li = { 0 };
	wchar_t buf[256] = TEXT("");
	int cnt = blist.GetItemCount();
	COleDateTime dt;
	wchar_t dtbuf[256]=TEXT("");
	for (int i = 0; i < cnt; i++)
	{
		li.iItem = i;
		li.iSubItem = 0;
		li.mask = LVIF_TEXT;
		li.pszText = buf;
		li.cchTextMax = 256;
		blist.GetItem(&li);
		if (bno == _wtoi(buf))
		{
			blist.SetItemText(i, 1, book->GetTitle());
			dt = book->GetPubdate();			
			wsprintf(dtbuf, TEXT("%d년 %d월 %d일"), dt.GetYear(), dt.GetMonth(), dt.GetDay());
			blist.SetItemText(i, 2, dtbuf);
			return;
		}
	}
}

DestroyWindow

BookManager에게 저장을 요청합니다.

BOOL C도서관리프로그램Dlg::DestroyWindow()
{
	// TODO: 여기에 특수화된 코드를 추가 및/또는 기본 클래스를 호출합니다.
	BookManager& bm = BookManager::GetBookManager();
	bm.Save();
	return __super::DestroyWindow();
}

도서 관리 프로그램Dlg.cpp 전체 소스 코드 내용


// 도서 관리 프로그램Dlg.cpp: 구현 파일
//

#include "pch.h"
#include "framework.h"
#include "도서 관리 프로그램.h"
#include "도서 관리 프로그램Dlg.h"
#include "afxdialogex.h"

#ifdef _DEBUG
#define new DEBUG_NEW
#endif

enum MY_W_MSG
{
	MWM_ENDED_REG = WM_APP+1
};


// 응용 프로그램 정보에 사용되는 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()


// C도서관리프로그램Dlg 대화 상자



C도서관리프로그램Dlg::C도서관리프로그램Dlg(CWnd* pParent /*=nullptr*/)
	: CDialogEx(IDD_DIALOG_MAIN, pParent)
{
	m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}

void C도서관리프로그램Dlg::DoDataExchange(CDataExchange* pDX)
{
	CDialogEx::DoDataExchange(pDX);
	DDX_Control(pDX, IDC_LIST_BOOK, blist);
	DDX_Control(pDX, IDC_BUTTON_REMOVE, btn_remove);
}

BEGIN_MESSAGE_MAP(C도서관리프로그램Dlg, CDialogEx)
	ON_WM_SYSCOMMAND()
	ON_WM_PAINT()
	ON_WM_QUERYDRAGICON()
	ON_BN_CLICKED(IDC_BUTTON_REG, &C도서관리프로그램Dlg::OnBnClickedButtonReg)	
	ON_MESSAGE(MWM_ENDED_REG, &C도서관리프로그램Dlg::OnEndedReg)
	ON_BN_CLICKED(IDC_BUTTON_VERIFY, &C도서관리프로그램Dlg::OnBnClickedButtonVerify)
	ON_NOTIFY(LVN_ITEMCHANGED, IDC_LIST_BOOK, &C도서관리프로그램Dlg::OnLvnItemchangedListBook)
	ON_BN_CLICKED(IDC_BUTTON_REMOVE, &C도서관리프로그램Dlg::OnBnClickedButtonRemove)
	ON_BN_CLICKED(IDC_BUTTON_MODIFY, &C도서관리프로그램Dlg::OnBnClickedButtonModify)
END_MESSAGE_MAP()


// C도서관리프로그램Dlg 메시지 처리기

BOOL C도서관리프로그램Dlg::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);
		}
	}

	// 이 대화 상자의 아이콘을 설정합니다.  응용 프로그램의 주 창이 대화 상자가 아닐 경우에는
	//  프레임워크가 이 작업을 자동으로 수행합니다.
	SetIcon(m_hIcon, TRUE);			// 큰 아이콘을 설정합니다.
	SetIcon(m_hIcon, FALSE);		// 작은 아이콘을 설정합니다.

	// TODO: 여기에 추가 초기화 작업을 추가합니다.
	mrd = 0;
	BookManager& bm = BookManager::GetBookManager();
	bm.AddABEventHandler(this);//도서 추가하면 나한테 알려줘
	bm.AddMBEventHandler(this);//도서 변경하면 나한테 알려줘

	LPWSTR texts[3] = {TEXT("No"),TEXT("제목"),TEXT("일자")};
	int widths[3] = { 50,200,120 };
	LV_COLUMN col;
	col.mask = LVCF_FMT | LVCF_SUBITEM | LVCF_TEXT | LVCF_WIDTH;
	col.fmt = LVCFMT_CENTER;
	for (int i = 0; i < 3; i++)
	{
		col.pszText = texts[i];
		col.iSubItem = i;
		col.cx = widths[i];
		blist.InsertColumn(i, &col);
	}
	bm.Load();
	return TRUE;  // 포커스를 컨트롤에 설정하지 않으면 TRUE를 반환합니다.
}

void C도서관리프로그램Dlg::OnSysCommand(UINT nID, LPARAM lParam)
{
	if ((nID & 0xFFF0) == IDM_ABOUTBOX)
	{
		CAboutDlg dlgAbout;
		dlgAbout.DoModal();
	}
	else
	{
		CDialogEx::OnSysCommand(nID, lParam);
	}
}

// 대화 상자에 최소화 단추를 추가할 경우 아이콘을 그리려면
//  아래 코드가 필요합니다.  문서/뷰 모델을 사용하는 MFC 애플리케이션의 경우에는
//  프레임워크에서 이 작업을 자동으로 수행합니다.

void C도서관리프로그램Dlg::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 C도서관리프로그램Dlg::OnQueryDragIcon()
{
	return static_cast<HCURSOR>(m_hIcon);
}



void C도서관리프로그램Dlg::OnBnClickedButtonReg()
{
	// TODO: 여기에 컨트롤 알림 처리기 코드를 추가합니다.
	if (mrd == 0)
	{
		mrd = new MyRegDialog();
		mrd->AddEndedEventHandler(this);//너 죽을 때 나한테 알려줘
		mrd->Create(IDD_DIALOG_REG,CWnd::GetDesktopWindow());
		mrd->ShowWindow(SW_SHOW);
	}
}
void C도서관리프로그램Dlg::EndedDialog()
{
	PostMessage(MWM_ENDED_REG, 0, 0);
}

LRESULT C도서관리프로그램Dlg::OnEndedReg(WPARAM, LPARAM)
{
	if (mrd == 0)
	{
		delete mrd;
	}
	mrd = 0;
	return 0;
}
void C도서관리프로그램Dlg::AddedBook(Book* book)
{
	int index = blist.GetItemCount();
	wchar_t nobuf[256];
	wsprintf(nobuf, TEXT("%d"), book->GetNo());
	blist.InsertItem(index, nobuf);
	blist.SetItemText(index, 1, book->GetTitle());
	COleDateTime dt = book->GetPubdate();
	wchar_t dtbuf[256];
	wsprintf(dtbuf, TEXT("%d년 %d월 %d일"), dt.GetYear(), dt.GetMonth(), dt.GetDay());
	blist.SetItemText(index, 2, dtbuf);
}

void C도서관리프로그램Dlg::OnBnClickedButtonVerify()
{
	// TODO: 여기에 컨트롤 알림 처리기 코드를 추가합니다.
	VerifyDilaog* vd = new VerifyDilaog();
	vd->Create(IDD_DIALOG_VERIFY, CWnd::GetDesktopWindow());
	vd->ShowWindow(SW_SHOW);
}


void C도서관리프로그램Dlg::OnLvnItemchangedListBook(NMHDR* pNMHDR, LRESULT* pResult)
{
	LPNMLISTVIEW pNMLV = reinterpret_cast<LPNMLISTVIEW>(pNMHDR);
	// TODO: 여기에 컨트롤 알림 처리기 코드를 추가합니다.
	*pResult = 0;
	
	if (blist.GetSelectedCount())
	{
		btn_remove.EnableWindow(TRUE);
	}
	else
	{
		btn_remove.EnableWindow(FALSE);
	}
}


void C도서관리프로그램Dlg::OnBnClickedButtonRemove()
{
	// TODO: 여기에 컨트롤 알림 처리기 코드를 추가합니다.
	BookManager& bm = BookManager::GetBookManager();
	POSITION pos = blist.GetFirstSelectedItemPosition();
	int index = blist.GetNextSelectedItem(pos);
	if (index == -1)
	{
		return;
	}
	LVITEM li = { 0 };
	li.iItem = index;
	li.iSubItem = 0;
	wchar_t buf[256]=TEXT("");
	li.mask = LVIF_TEXT;
	li.pszText = buf;
	li.cchTextMax = 256;//버퍼 크기
	blist.GetItem(&li);
	int bno = _wtoi(buf);
	if (bm.RemoveBook(bno))
	{
		blist.DeleteItem(index);
	}
}


void C도서관리프로그램Dlg::OnBnClickedButtonModify()
{	
	// TODO: 여기에 컨트롤 알림 처리기 코드를 추가합니다.
	ModifyDialog* md = new ModifyDialog();
	md->Create(IDD_DIALOG_MODIFY, CWnd::GetDesktopWindow());
	md->ShowWindow(SW_SHOW);
}
void C도서관리프로그램Dlg::ModifiedBook(Book* book)
{
	int bno = book->GetNo();
	LVITEM li = { 0 };
	wchar_t buf[256] = TEXT("");
	int cnt = blist.GetItemCount();
	COleDateTime dt;
	wchar_t dtbuf[256]=TEXT("");
	for (int i = 0; i < cnt; i++)
	{
		li.iItem = i;
		li.iSubItem = 0;
		li.mask = LVIF_TEXT;
		li.pszText = buf;
		li.cchTextMax = 256;
		blist.GetItem(&li);
		if (bno == _wtoi(buf))
		{
			blist.SetItemText(i, 1, book->GetTitle());
			dt = book->GetPubdate();			
			wsprintf(dtbuf, TEXT("%d년 %d월 %d일"), dt.GetYear(), dt.GetMonth(), dt.GetDay());
			blist.SetItemText(i, 2, dtbuf);
			return;
		}
	}
}

BOOL C도서관리프로그램Dlg::DestroyWindow()
{
	// TODO: 여기에 특수화된 코드를 추가 및/또는 기본 클래스를 호출합니다.
	BookManager& bm = BookManager::GetBookManager();
	bm.Save();
	return __super::DestroyWindow();
}

MFC 실습