- [DEVEXPRESS] Hỗ trợ tìm kiếm highlight không dấu và không khoảng cách trên Gridview Filter
- [C#] Chia sẻ source code phần mềm Image Downloader tải hàng loạt hình ảnh từ danh sách link url
- [C#] Chụp hình và quay video từ camera trên winform
- [C#] Chia sẽ full source code tách file Pdf thành nhiều file với các tùy chọn
- Giới thiệu về Stock Tracker Widget - Công cụ theo dõi cổ phiếu và cảnh báo giá tăng giảm bằng C# và WPF
- [VB.NET] Chia sẻ công cụ nhập số tiền tự động định dạng tiền tệ Việt Nam
- [VB.NET] Hướng dẫn fill dữ liệu từ winform vào Microsoft word
- [VB.NET] Hướng dẫn chọn nhiều dòng trên Datagridview
- Hướng Dẫn Đăng Nhập Nhiều Tài Khoản Zalo Trên Máy Tính Cực Kỳ Đơn Giản
- [C#] Chia sẻ source code phần mềm đếm số trang tập tin file PDF
- [C#] Cách Sử Dụng DeviceId trong C# Để Tạo Khóa Cho Ứng Dụng
- [SQLSERVER] Loại bỏ Restricted User trên database MSSQL
- [C#] Hướng dẫn tạo mã QRcode Style trên winform
- [C#] Hướng dẫn sử dụng temp mail service api trên winform
- [C#] Hướng dẫn tạo mã thanh toán VietQR Pay không sử dụng API trên winform
- [C#] Hướng Dẫn Tạo Windows Service Đơn Giản Bằng Topshelf
- [C#] Chia sẻ source code đọc dữ liệu từ Google Sheet trên winform
- [C#] Chia sẻ source code tạo mã QR MOMO đa năng Winform
- [C#] Chia sẻ source code phần mềm lên lịch tự động chạy ứng dụng Scheduler Task Winform
- [C#] Hướng dẫn download file từ Minio Server Winform
[C#] Hướng dẫn download dữ liệu máy chấm công
Xin chào các bạn, bài viết hôm nay mình sẻ tiếp tục hướng dẫn các bạn cách tải dữ liệu chấm công từ máy chấm công trong lập trình C# winform.
[C#] Phần mềm tải dữ liệu máy chấm công

Trong bài viết này, chương trình demo của mình đang kết nối với máy chấm công Ronald Jack, một thương hiệu máy chấm công phổ biến trên thị trường.
Máy chấm công này có cung cấp cho chúng ta thư viện Interop.zkemkeeper để làm việc với máy chấm công.
Với thư viện, này chúng ta có thể làm nhiều việc trên máy chấm công: thêm nhân viên mới, xóa nhân viên, khóa nhân viên...
Tuy nhiên trong bài viết này, mình chỉ demo cách tải dữ liệu chấm công trên máy về GridView trên Winform.
Dưới đây là giao diện demo ứng dụng tools tải dữ liệu chấm công:

Phần mềm này, mình tích hợp chạy đa luồng, nên thời gian tải dữ liệu trên nhiều máy sẽ rất nhanh, như hình demo các bạn thấy.
Thông số port mặc định của máy thường là: 4370, nếu máy của bạn cấu hình trên port khác, các bạn có thể thay đổi port ở code bên dưới.
Đầu tiên các bạn tạo cho mình một class AttLogHelper.cs
using System;
using System.Collections.Generic;
using System.Data;
using System.IO;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using zkemkeeper;
namespace AttLog
{
    public static class AttLogHelper
    {
        public const int TCP_PORT = 4370;
        private const int CONNECT_TIMEOUT = 4000;
        private const int DATA_FILE_FLAG = 1; // Log data file
        private const int RECORD_COUNT_FLAG = 6; // Log data file
        private static int iMachineNumber = 1;//In fact,when you are using the tcp/ip communication,this parameter will be ignored,that is any integer will all right.Here we use 1.
        private static readonly Random m_random = new Random();
        private static readonly string HOB_TEMP_CHAMCONG_DIR = Path.GetTempPath() + "hob_chamcong\\";
        //1 dòng dữ liệu LOG máy chấm công cố định là 16 byte
        private const int ROW_LOG_BYTE_LENGTH = 16; 
        //1 dòng dữ liệu USER máy chấm công cố định là 28 byte
        private const int ROW_USER_BYTE_LENGTH = 28;
        private const string CHAMCONG_LOG_FILENAME = "extlog.dat";
        private const string CHAMCONG_USER_FILENAME = "user.dat"; //file lưu mã nhân viên - seri thẻ
        private const long MAX_SERI = 4294967295;
        //return false nghĩa là dữ liệu máy chấm công có lỗi cấu trúc
        public static bool ParseLog2Datatable(ref byte[] buff, DataTable outDt)
        {
            int manv = 0;
            DateTime ngaygio = default(DateTime);
            int startIndex = 0;
            var Today = DateTime.Now;
            DataRow row = default(DataRow);
            outDt.Clear();
            outDt.Columns.Add("manv", typeof(int));
            outDt.Columns.Add("ngaygio", typeof(DateTime));
            //1 dòng dữ liệu LOG máy chấm công cố định là 16 byte
            //buff.Length > 0 AndAlso
            if (buff.Length % ROW_LOG_BYTE_LENGTH == 0)
            {
                for (var r = 0; r <= (buff.Length / ROW_LOG_BYTE_LENGTH) - 1; r++)
                {
                    startIndex = ROW_LOG_BYTE_LENGTH * r;
                    manv = ReadManvFromByteArr(buff, startIndex);
                    ngaygio = ReadDateFromByteArr(buff, startIndex + 4);
                    if (!validDateLog(ngaygio, Today))
                    {
                        //lỗi cấu trúc file log, phải chuyển sang dùng zkem thay thế
                        return false;
                    }
                    row = outDt.NewRow();
                    row["manv"] = manv;
                    row["ngaygio"] = ngaygio;
                    outDt.Rows.Add(row);
                }
                return true;
            }
            return false;
        }
        //return false nghĩa là dữ liệu máy chấm công có lỗi cấu trúc
        public static bool ParseUser2Datatable(ref byte[] buff, DataTable outDt)
        {
            int startIndex = 0;
            DataRow row = default(DataRow);
            outDt.Clear();
            outDt.Columns.Add("manv", typeof(int));
            outDt.Columns.Add("masothe", typeof(long));
            outDt.Columns.Add("khoathe", typeof(bool));
            //1 dòng dữ liệu USER máy chấm công cố định là 28 byte
            //buff.Length > 0 AndAlso
            if (buff.Length % ROW_USER_BYTE_LENGTH == 0)
            {
                for (var r = 0; r <= (buff.Length / ROW_USER_BYTE_LENGTH) - 1; r++)
                {
                    startIndex = ROW_USER_BYTE_LENGTH * r;
                    row = outDt.NewRow();
                    row["manv"] = ReadManvFromByteArr(buff, startIndex);
                    row["masothe"] = ReadMasotheFromByteArr(buff, startIndex + 16);
                    row["khoathe"] = buff[startIndex + 21] == 0;
                    outDt.Rows.Add(row);
                }
                return true;
            }
            return false;
        }
        //Chuyển ngược datatable USER thành mảng Byte
        public static byte[] ConvertTbl_UserToByteArr(DataTable dt)
        {
            byte[] buff = new byte[] { };
            int r = 0;
            int startIndex = 0;
            byte b1 = 0;
            byte b2 = 0;
            int manv = 0;
            long masothe = 0;
            bool khoathe = false;
            if (dt.Rows.Count > 0)
            {
                buff = new byte[dt.Rows.Count * ROW_USER_BYTE_LENGTH - 1 + 1];
                foreach (DataRow row in dt.Rows)
                {
                    startIndex = r * ROW_USER_BYTE_LENGTH;
                    r++;
                    manv = Convert.ToInt32(row["manv"]);
                    b1 = Convert.ToByte(manv >> 0 & 0xFF);
                    b2 = Convert.ToByte(manv >> 8 & 0xFF);
                    buff[startIndex] = b1;
                    buff[startIndex + 1] = b2;
                    buff[startIndex + 24] = b1;
                    buff[startIndex + 25] = b2;
                    masothe = Convert.ToInt64(row["masothe"]);
                    buff[startIndex + 16] = Convert.ToByte(masothe >> 0 & 0xFF);
                    buff[startIndex + 17] = Convert.ToByte(masothe >> 8 & 0xFF);
                    buff[startIndex + 18] = Convert.ToByte(masothe >> 16 & 0xFF);
                    buff[startIndex + 19] = Convert.ToByte(masothe >> 24 & 0xFF);
                    khoathe = Convert.ToBoolean(row["khoathe"]);
                    buff[startIndex + 21] = Convert.ToByte(khoathe);
                }
            }
            return buff;
        }
        public static int getRecordCount(zkemkeeper.CZKEM axCZKEM1)
        {
            int result = -1;
            if (axCZKEM1 == null)
            {
                return result;
            }
            axCZKEM1.EnableDevice(iMachineNumber, false);//disable the device
            if (!axCZKEM1.GetDeviceStatus(iMachineNumber, RECORD_COUNT_FLAG, ref result))
            {
                result = -1;
            }
            axCZKEM1.EnableDevice(iMachineNumber, true);//enable the device
            return result;
        }
        private static string genRandomFilename()
        {
            DateTime now = DateTime.Now;
            DateTime sd = new DateTime(1970, 1, 1);
            string filename = (int)now.Subtract(sd).TotalDays + "." + (int)now.TimeOfDay.TotalMilliseconds +
                "." + m_random.Next(10000, 99999) + ".data.dat";
            return filename;
        }
        public static byte[] downloadDataFile(zkemkeeper.CZKEM axCZKEM1)
        {
            if (axCZKEM1 == null)
            {
                return null;
            }
            string filename = HOB_TEMP_CHAMCONG_DIR + genRandomFilename();
            if (!Directory.Exists(HOB_TEMP_CHAMCONG_DIR))
            {
                try
                {
                    Directory.CreateDirectory(HOB_TEMP_CHAMCONG_DIR);
                }
                catch (System.Exception)
                {
                    return null;
                }
            }
            bool isSuccess = axCZKEM1.GetDataFile(iMachineNumber, DATA_FILE_FLAG, filename);
            if (isSuccess)
            {
                byte[] bt = null;
                try
                {
                    bt = File.ReadAllBytes(filename);
                    File.Delete(filename);
                }
                catch (Exception)
                {
                }
                return bt;
            }
            else
            {
                return null;
            }
        }
        private static bool validDateLog(DateTime testDate, DateTime today)
        {
            return today.Date >= testDate.Date && testDate > today.AddDays(-35);
        }
        private static int ReadManvFromByteArr(byte[] buff, int startIndex)
        {
            return buff[startIndex] + buff[startIndex + 1] * 256;
        }
        public static async Task<CZKEM> Connect(string ip)
        {
            zkemkeeper.CZKEM axCZKEM1 = new zkemkeeper.CZKEM();
            await Task.Factory.StartNew(new Action(() => {
                axCZKEM1.Connect_Net(ip, TCP_PORT);
            }));
            return axCZKEM1;
         
        }
        private static DateTime ReadDateFromByteArr(byte[] buff, int startIndex)
        {
            long dateInt = 0;
            dateInt = System.Convert.ToInt64(buff[startIndex] + buff[startIndex + 1] * 256 +
                buff[startIndex + 2] * 65536 + buff[startIndex + 3] * Convert.ToInt64(16777216));
            int Year = 0;
            int Month = 0;
            int Day = 0;
            int Hour = 0;
            int Minute = 0;
            int Second = 0;
            Second = (int)(dateInt % 60);
            dateInt /= 60;
            Minute = (int)(dateInt % 60);
            dateInt /= 60;
            Hour = (int)(dateInt % 24);
            dateInt /= 24;
            Day = (int)(dateInt % 31 + 1);
            dateInt /= 31;
            Month = (int)(dateInt % 12 + 1);
            dateInt /= 12;
            Year = System.Convert.ToInt32(dateInt + 2000);
            return new DateTime(Year, Month, Day, Hour, Minute, Second);
        }
        private static long ReadMasotheFromByteArr(byte[] buff, int startIndex)
        {
            long Masothe = System.Convert.ToInt64(buff[startIndex] + buff[startIndex + 1] * 256 +
                buff[startIndex + 2] * 65536 + buff[startIndex + 3] * Convert.ToInt64(16777216));
            return Masothe;
        }
        private static string ReadUsernameFRomByteArr(byte[] buff, int startIndex)
        {
            if (buff[startIndex] == 0)
            {
                return "";
            }
            else
            {
                var nameLen = 1;
                for (var r = 1; r <= 7; r++)
                {
                    if (buff[startIndex + r] == 0)
                    {
                        break;
                    }
                    else
                    {
                        nameLen++;
                    }
                }
                return Encoding.Default.GetString(buff, startIndex, nameLen);
            }
        }
    }
}
Và source code cho Form Main:
using DevExpress.XtraEditors;
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;
using zkemkeeper;
namespace AttLog
{
    public partial class Form1 : DevExpress.XtraEditors.XtraForm
    {
        private int _soMayChon;
        DataTable dataTableResult;
        public Form1()
        {
            InitializeComponent();
        }
       
        private void Form1_Load(object sender, EventArgs e)
        {
            var dataTable = new DataTable();
            dataTable.Columns.Add("ip");
            dataTable.Columns.Add("tinhtrang");
            dataTable.Columns.Add("noidung");
            dataTable.Rows.Add("172.16.0.91");
            dataTable.Rows.Add("172.16.0.92");
            dataTable.Rows.Add("172.16.0.93");
            dataTable.Rows.Add("172.16.0.94");
            gridControl1.DataSource = dataTable;
            dataTableResult = new DataTable();
            dataTableResult.Columns.Add("manv");
            dataTableResult.Columns.Add("ngaygio");
            gridControl2.DataSource = dataTableResult;
        }
        private void ZKEM_download_chamcong(IProgress<int> progress)
        {
            gridView1.CloseEditor();
            gridView1.PostEditor();
            var selectedRows = gridView1.GetSelectedRows();
            _soMayChon = selectedRows.Count();            
            int countFinish = 0;
            BeginInvoke(new Action(() =>
            {
                btn_DownloadAtt.Enabled = false;
                foreach (var r in selectedRows)
                {
                    gridView1.SetRowCellValue(r, "tinhtrang", "");
                    gridView1.SetRowCellValue(r, "noidung", "");
                }
            }));
            Parallel.ForEach(selectedRows, async rowHandle =>
            {
                string deviceIp = Convert.ToString(gridView1.GetDataRow(rowHandle)["ip"]);
                TinhTrangKetNoi(rowHandle, "Đang kết nối ZKEM...");
                CZKEM izkem = await AttLogHelper.Connect(deviceIp);
                if (izkem != null)
                {
                    int recordCount = AttLogHelper.getRecordCount(izkem);
                    if (recordCount == 0)
                    {
                        TinhTrangKetNoi(rowHandle, "Máy không có dữ liệu");
                    }
                    else
                    {
                        TinhTrangKetNoi(rowHandle, "Đã kết nối. Đang tải...");
                        byte[] data = AttLogHelper.downloadDataFile(izkem);
                        izkem.Disconnect();
                        if (data != null)
                        {
                            var dt = new DataTable();
                            if (AttLogHelper.ParseLog2Datatable(ref data, dt))
                            {
                                gridControl2.BeginInvoke(new Action(() =>
                                {
                                   foreach(DataRow dr in dt.Rows)
                                    {
                                        var workRow = dataTableResult.NewRow();
                                        workRow[0] = dr["manv"];
                                        workRow[1] = dr["ngaygio"];                                      
                                        dataTableResult.Rows.Add(workRow);
                                    }
                                }));
                                NoidungKetNoi(rowHandle, "Tìm thấy " + dt.Rows.Count + " dòng");                           
                            }
                        }
                        else
                        {
                            TinhTrangKetNoi(rowHandle, "Tải file chấm công thất bại");
                        }
                    }
                }
                else
                {
                    TinhTrangKetNoi(rowHandle, "Lỗi kết nối. Thử lại sau.");
                }                
                Interlocked.Increment(ref countFinish);
                var percent = Convert.ToInt32(Convert.ToDouble(countFinish) / _soMayChon * 100d);
                progress.Report(percent);
            });
        }
        private void TinhTrangKetNoi(int rowHandle, string msg)
        {
            gridControl1.BeginInvoke((Action)(() =>
            {
                gridView1.SetRowCellValue(rowHandle, "tinhtrang", msg);
            }));
        }
        private void NoidungKetNoi(int rowHandle, string msg)
        {
            gridControl1.BeginInvoke((Action)(() =>
            {
                gridView1.SetRowCellValue(rowHandle, "noidung", msg);
            }));
        }
        private async void btnDownload_Click(object sender, EventArgs e)
        {
            if (gridView1.SelectedRowsCount > 0)
            {
                progressBarControl1.EditValue = 0;
                var progress = new Progress<int>(percent =>
                {
                    BeginInvoke(new Action(() =>
                    {
                        progressBarControl1.EditValue = percent;
                        if(percent == 100)
                        {
                            btn_DownloadAtt.Enabled = true;
                        }
                    }));
                });
                await Task.Run(() => ZKEM_download_chamcong(progress));
            }
            else
            {
                XtraMessageBox.Show("Bạn vui lòng chọn các máy chấm công để tải dữ liệu kết nối.");
            }
        }
    }
}
Thanks for watching!

![[C#] Hướng dẫn download dữ liệu máy chấm công](https://laptrinhvb.net/uploads/users/9a8cb514e4428e85fb4ca07588e9103f.png)

![[C#] Hướng dẫn tạo hiệu ứng rung chuyển form (Attention or buzz effect winform)](https://laptrinhvb.net/uploads/source/csharp/attention_effect_csharp_thumb.jpg)
![[C#] Chuyển chế độ màn hình GrayScale trắng đen một click chuột](https://laptrinhvb.net/uploads/source/new_image_baiviet/window_grayscale.png)

![[C#] Hướng dẫn định dạng tiền tệ trong lập trình Csharp](https://laptrinhvb.net/uploads/source/csharp/format_string_currency.jpg)
![[C#] Hướng dẫn lưu và load hình ảnh xuống database MS Access](https://laptrinhvb.net/uploads/source/vbnet/db_access_csharp.jpg)
![[C#] Giải pháp thay thế Web Browser Control mặc định bằng Awesomium của bộ Visual Studio](https://laptrinhvb.net/uploads/source/image_baiviet/0ddd54d36f9c30418c8465205ee441d2.png)
![[C#] Hướng dẫn soạn thảo các biểu thức toán học vào lưu vào dữ liệu Sqlserver](https://laptrinhvb.net/uploads/source/image_baiviet/d4429d1083839a831926d54c1658cc32.png)
![[C#] Hướng dẫn xóa cache trình duyệt WebBrowser trong Csharp](https://laptrinhvb.net/uploads/source/image_baiviet/1567e1065489af572f9410510ca6a0cf.jpg)
![[C#] Garbage Collector (GC) là gì? Cách giải phóng bộ nhớ trên ứng dụng Winform khi các đối tượng không còn sử dụng](https://laptrinhvb.net/uploads/source/new_image_baiviet/Garbage%20Collector%20csharp.png)
![[C#] Hướng dẫn cách viết ứng dụng lập lịch chạy tự động sử dụng thư viện Quartz.NET](https://laptrinhvb.net/uploads/source/image_baiviet/0bec151c11c53dd094691f07905ae002.jpg)
![[C#] Xem dung lượng RAM đang sử dụng trong ứng dụng Winform](https://laptrinhvb.net/uploads/source/csharp/ram_using_csharp_thumb.png)
![[C#] Hướng dẫn tạo dữ liệu test hàng loạt với thư viện Bogus for .NET](https://laptrinhvb.net/uploads/source/new_image_baiviet/bogus_fake_data.png)
![[C#] Chia sẻ tài liệu, sdk và source code máy chấm công dòng máy ZKTeco](https://laptrinhvb.net/uploads/source/new_image_baiviet/SDK_ZKTECO.png)

![[C#] Hướng dẫn viết ứng dụng đọc văn bản từ textbox](https://laptrinhvb.net/uploads/source/image_baiviet/e450fbeba60b43294a219ea6f64c719f.jpg)
![[C#] Hướng dẫn Overlay Icon dưới Taskbar của ứng dụng - lap trinh csharp](https://laptrinhvb.net/uploads/source/image_baiviet/0ed99f089b242555e56f6b8ae748c1bb.gif)
![[C#] Chia sẽ thư viện MCoreLib.dll để lập trình AT Command gởi, nhận tin nhắn SMS  và các chức năng khác](https://laptrinhvb.net/uploads/source/csharp/sms_lib_csharp.jpg)
![[C#] Chia sẽ source code Firework Effect trong Winform](https://laptrinhvb.net/uploads/source/devexpress/firework_effect.jpg)
![[C#] Hướng dẫn chỉnh màu Gamma màn hình hệ thống (Monitor Calibration)](https://laptrinhvb.net/uploads/source/csharp/gamma_thumb.jpg)
![[C#] Đặt mật khẩu bảo vệ cho database Sqlite](https://laptrinhvb.net/uploads/source/image_baiviet/271fd22eef8ba8835c6cc16e94677853.jpg)
![[C#] Hướng dẫn lấy thông tin Your ID và Password của Teamviewer Winform](https://laptrinhvb.net/uploads/source/vbnet/id_password_teamviewer_thumb.jpg)
![[C#] Theo dõi Monitoring all tất cả các Request Http từ ứng dụng .NET](https://laptrinhvb.net/uploads/source/new_image_baiviet/ObserveWrapHttp.png)
![[C#] Hiển thị hình ảnh GIF vào button, picturebox, label winform](https://laptrinhvb.net/uploads/source/csharp/gif_button_csharp.gif)
![[C#] Hướng dẫn sử dụng Factory trong Design Pattern](https://laptrinhvb.net/uploads/source/csharp/factory_pattern_thumb.png)
![[C#] Hướng dẫn hiện thị progress bar window khi copy file hoặc folder sử dụng thư viện visualbasic](https://laptrinhvb.net/uploads/source/image_baiviet/2017a10730b82d5561a3b1931e71befe.jpg)
