VC++:汉王人脸通SDK示例代码(五)获取考勤记录

同系列文章:

运行环境

下载安装Visual C++ Redistributable for Visual Studio 2015

演示程序源代码下载:

FaceIdDemo-VC2015-20161118.zip
注意:请将FaceId.dll和FaceId.lib拷贝到Release和Debug目录下

演示程序界面:
GetRecord-VC

示例代码
GetRecordDlg.h


// GetRecordDlg.h : 头文件
//

#pragma once


// CGetRecordDlg 对话框
class CGetRecordDlg : public CDialogEx
{
// 构造
public:
	CGetRecordDlg(CWnd* pParent = NULL);	// 标准构造函数

// 对话框数据
	enum { IDD = IDD_GETRECORD_DIALOG };

	protected:
	virtual void DoDataExchange(CDataExchange* pDX);	// DDX/DDV 支持


// 实现
protected:
	HICON m_hIcon;

	// 生成的消息映射函数
	virtual BOOL OnInitDialog();
	afx_msg void OnPaint();
	afx_msg HCURSOR OnQueryDragIcon();
	DECLARE_MESSAGE_MAP()
public:
	afx_msg void OnBnClickedButtonclear();
	afx_msg void OnBnClickedButtonexport();

private:
	CONST UINT DeviceCodePage = 936;	// 设备通信字符集为简体中文
};

GetRecordDlg.cpp

/* ----------------------------------------------------------
 * 文件名称:GetRecordDlg.cpp
 *
 * 作者:秦建辉
 *
 * QQ:36748897
 *
 * 博客:http://www.firstsolver.com/wordpress/
 *
 * 开发环境:
 *      Visual Studio V2013
 *
 * 版本历史:
 *	V1.1    2014年11月04日
 *           完善考勤记录提取正则表达式,使之能完整获取带照片的考勤记录
 *
 *	V1.0    2014年09月24日
 *           VC:人脸通SDK示例代码-获取考勤记录
 * ------------------------------------------------------------ */

// GetRecordDlg.cpp : 实现文件
//

#include "stdafx.h"
#include "GetRecord.h"
#include "GetRecordDlg.h"
#include "afxdialogex.h"
#include "..\\FaceId\\Source\\FaceId.h"
#include <regex>

typedef std::basic_regex<TCHAR> tregex;
typedef std::match_results<const TCHAR*> tmatch;

#ifdef _DEBUG
#define new DEBUG_NEW
#endif

#ifdef _DEBUG
#pragma comment(lib, "..\\Debug\\FaceId.lib")
#else
#pragma comment(lib, "..\\Release\\FaceId.lib")
#endif

// CGetRecordDlg 对话框


CGetRecordDlg::CGetRecordDlg(CWnd* pParent /*=NULL*/)
	: CDialogEx(CGetRecordDlg::IDD, pParent)
{
	m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}

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

BEGIN_MESSAGE_MAP(CGetRecordDlg, CDialogEx)
	ON_WM_PAINT()
	ON_WM_QUERYDRAGICON()
	ON_BN_CLICKED(IDC_BUTTONCLEAR, &CGetRecordDlg::OnBnClickedButtonclear)
	ON_BN_CLICKED(IDC_BUTTONEXPORT, &CGetRecordDlg::OnBnClickedButtonexport)
END_MESSAGE_MAP()


// CGetRecordDlg 消息处理程序

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

	// 设置此对话框的图标。  当应用程序主窗口不是对话框时,框架将自动
	//  执行此操作
	SetIcon(m_hIcon, TRUE);			// 设置大图标
	SetIcon(m_hIcon, FALSE);		// 设置小图标

	// TODO:  在此添加额外的初始化代码
	SetDlgItemText(IDC_EDITDEVICEIP, TEXT("192.168.1.18"));	// 设备地址
	SetDlgItemInt(IDC_EDITDEVICEPORT, 9922, FALSE);			// 设备端口

	return TRUE;  // 除非将焦点设置到控件,否则返回 TRUE
}

// 如果向对话框添加最小化按钮,则需要下面的代码
//  来绘制该图标。  对于使用文档/视图模型的 MFC 应用程序,
//  这将由框架自动完成。

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

// 清空文本
void CGetRecordDlg::OnBnClickedButtonclear()
{
	SetDlgItemText(IDC_EDITANSWER, TEXT(""));
}

// 导出考勤记录
void CGetRecordDlg::OnBnClickedButtonexport()
{
	// 执行命令
	FaceId *TcpClient = NULL;
	try
	{
		// 连接设备
		TCHAR DeviceIP[64];
		GetDlgItemText(IDC_EDITDEVICEIP, DeviceIP, _countof(DeviceIP));
		TcpClient = new FaceId(DeviceIP, GetDlgItemInt(IDC_EDITDEVICEPORT), AF_INET);

		// 设置通信密码
		TcpClient->SetSecretKey(NULL);	// 注意:需要和设备通信密码一致

		// 获取本地时间
		SYSTEMTIME lt;
		::GetLocalTime(&lt);

		// 生成命令字符串
		TCHAR Command[256];
		StringCchPrintf(Command, _countof(Command), TEXT("GetRecord(end_time=\"%d-%02d-%02d %02d:%02d:%02d\")"), lt.wYear, lt.wMonth, lt.wDay, lt.wHour, lt.wMinute, lt.wSecond);
		
		// 执行命令
		tstring Answer;
		INT nReadChars = TcpClient->Execute(Command, Answer, DeviceCodePage);
		if (nReadChars > 0)
		{	// 执行命令成功
			tregex rx(TEXT("\\b(time=.+\r\n(?:photo=\"[^\"]+\")*)"));
			tmatch m;

			// 搜索正则表达式匹配
			const TCHAR* Begin = Answer.c_str();
			const TCHAR* End = Begin + Answer.size();
			CString AttendRecords;
			while (std::regex_search(Begin, End, m, rx, std::regex_constants::match_default))
			{
				AttendRecords.AppendFormat(TEXT("%s\r\n"), m.str(1).c_str());
				Begin = m[0].second;
			}

			// 更新考勤记录内容
			SetDlgItemText(IDC_EDITANSWER, AttendRecords);
		}
		else
		{	// 显示错误代码
			TCHAR Information[256];
			StringCchPrintf(Information, _countof(Information), TEXT("错误代码:%d"), nReadChars);
			MessageBox(Information, TEXT("错误"), MB_OK | MB_ICONERROR | MB_DEFBUTTON1);
		}
	}
	catch (DWORD dwErrorCode)
	{	// 显示异常原因
		TCHAR Information[256];
		StringCchPrintf(Information, _countof(Information), TEXT("错误代码:%d"), dwErrorCode);
		MessageBox(Information, TEXT("错误"), MB_OK | MB_ICONERROR | MB_DEFBUTTON1);
	}
	catch (...)
	{

	}

	// 断开连接
	if (TcpClient != NULL)
	{
		TcpClient->Close();
		TcpClient = NULL;
	}
}

Comments are closed.