From e8a27bb203fe2aff70390a5eca002d7438da9b0f Mon Sep 17 00:00:00 2001
From: mrDarker <mr.darker@163.com>
Date: 星期三, 22 十月 2025 14:24:34 +0800
Subject: [PATCH] Merge branch 'clh' into liuyang

---
 SourceCode/Bond/Servo/CPageGlassList.cpp |  402 +++++++++++++++++++++++++++++++++++++++++++++------------
 1 files changed, 317 insertions(+), 85 deletions(-)

diff --git a/SourceCode/Bond/Servo/CPageGlassList.cpp b/SourceCode/Bond/Servo/CPageGlassList.cpp
index 0d0d5c1..cbe79ed 100644
--- a/SourceCode/Bond/Servo/CPageGlassList.cpp
+++ b/SourceCode/Bond/Servo/CPageGlassList.cpp
@@ -100,6 +100,9 @@
 // ====== 寮�鍏筹細1=鍚敤鍋囨暟鎹紙鍙浛鎹� DB 鏌ヨ锛夛紱0=鐢ㄧ湡瀹� DB ======
 #define USE_FAKE_DB_DEMO 0
 
+// ====== 寮�鍏筹細1=鍚敤妯℃嫙浼犳劅鍣ㄦ暟鎹敓鎴愶紱0=浣跨敤鐪熷疄鏁版嵁 ======
+#define USE_MOCK_SENSOR_DATA 0
+
 #if USE_FAKE_DB_DEMO
 #include <ctime>
 #include <atlconv.h>   // CStringA
@@ -1334,12 +1337,12 @@
     strSanitizedGlassId.Remove('>');
     strSanitizedGlassId.Remove('|');
 
-    strDefaultFileName.Format(_T("Glass_%s.json"), strSanitizedGlassId);
+    strDefaultFileName.Format(_T("Glass_%s.csv"), strSanitizedGlassId);
 
     // 鏂囦欢淇濆瓨瀵硅瘽妗嗭紝璁剧疆榛樿鏂囦欢鍚�
-    CFileDialog fileDialog(FALSE, _T("json"), strDefaultFileName,
+    CFileDialog fileDialog(FALSE, _T("csv"), strDefaultFileName,
         OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT,
-        _T("JSON Files (*.json)|*.json|CSV Files (*.csv)|*.csv||"));
+        _T("CSV Files (*.csv)|*.csv|JSON Files (*.json)|*.json||"));
 
     if (fileDialog.DoModal() != IDOK) return;
 
@@ -1347,98 +1350,214 @@
     CString fileExt = fileDialog.GetFileExt();
 
     if (fileExt.CompareNoCase(_T("json")) == 0) {
-        // 淇濆瓨涓� JSON
-        if (!row->pretty.empty()) {
-            CFile file;
-            if (file.Open(filePath, CFile::modeCreate | CFile::modeWrite)) {
-                file.Write(row->pretty.c_str(), row->pretty.length());
-                file.Close();
-
-                CString strSuccess;
-                strSuccess.Format(_T("璁板綍宸蹭繚瀛樹负JSON鏂囦欢锛歕n%s"), filePath);
-                AfxMessageBox(strSuccess);
-            }
-            else {
-                AfxMessageBox(_T("淇濆瓨鏂囦欢澶辫触"));
-            }
-        }
-        else {
-            AfxMessageBox(_T("璇ヨ褰曟病鏈塉SON鏁版嵁"));
-        }
+        ExportToJson(*row, filePath);
     }
     else {
-        // 淇濆瓨涓� CSV 鏍煎紡 - 鍒嗘寮�
-        CString csvContent;
+        ExportToCsv(*row, filePath);
+    }
+}
 
-        // === 绗竴閮ㄥ垎锛氬熀纭�淇℃伅 ===
-        csvContent += _T("=== 鍩虹淇℃伅 ===\n");
-        csvContent += _T("ID,Cassette搴忓垪鍙�,Job搴忓垪鍙�,Glass ID,鐗╂枡绫诲瀷,鐘舵��,寮�濮嬫椂闂�,缁撴潫鏃堕棿,缁戝畾Glass ID,AOI缁撴灉,璺緞\n");
+void CPageGlassList::ExportToJson(const GlassLogDb::Row& row, const CString& filePath)
+{
+    // 淇濆瓨涓� JSON
+    if (!row.pretty.empty()) {
+        CFile file;
+        if (file.Open(filePath, CFile::modeCreate | CFile::modeWrite)) {
+            file.Write(row.pretty.c_str(), row.pretty.length());
+            file.Close();
 
-        CString baseInfoRow;
-        baseInfoRow.Format(_T("%lld,%d,%d,%s,%d,%d,%s,%s,%s,%d,%s\n"),
-            row->id, row->cassetteSeqNo, row->jobSeqNo,
-            CString(row->classId.c_str()), row->materialType, row->state,
-            CString(row->tStart.c_str()), CString(row->tEnd.c_str()),
-            CString(row->buddyId.c_str()), row->aoiResult,
-            CString(row->path.c_str()));
-        csvContent += baseInfoRow;
-
-        // === 绗簩閮ㄥ垎锛氬伐鑹哄弬鏁� ===
-        csvContent += _T("\n=== 宸ヨ壓鍙傛暟 ===\n");
-
-        // 濡傛灉鏈� pretty 瀛楁锛岃В鏋愬伐鑹哄弬鏁�
-        if (!row->pretty.empty()) {
-            SERVO::CGlass tempGlass;
-            if (GlassJson::FromString(row->pretty, tempGlass)) {
-                auto& params = tempGlass.getParams();
-                if (!params.empty()) {
-                    // 宸ヨ壓鍙傛暟琛ㄥご - 璋冩暣鍚庣殑鍒�
-                    csvContent += _T("鍙傛暟鍚嶇О,鍙傛暟ID,鏁板��,鏈哄櫒鍗曞厓\n");
-
-                    // 宸ヨ壓鍙傛暟鏁版嵁 - 璋冩暣鍚庣殑鏍煎紡
-                    for (auto& param : params) {
-                        CString paramRow;
-                        CString valueStr;
-
-                        // 鏍规嵁鍙傛暟绫诲瀷鏍煎紡鍖栨暟鍊�
-                        if (param.getValueType() == PVT_INT) {
-                            valueStr.Format(_T("%d"), param.getIntValue());
-                        }
-                        else {
-                            valueStr.Format(_T("%.3f"), param.getDoubleValue());
-                        }
-
-                        // 璋冩暣鍚庣殑鏍煎紡锛氬幓鎺夋暟鍊肩被鍨嬪垪
-                        paramRow.Format(_T("%s,%s,%s,%s\n"),
-                            CString(param.getName().c_str()),
-                            CString(param.getId().c_str()),
-                            valueStr,
-                            CString(param.getUnit().c_str())); // 杩欓噷鏄剧ず鏈哄櫒鍗曞厓
-
-                        csvContent += paramRow;
-                    }
-                }
-                else {
-                    csvContent += _T("鏃犲伐鑹哄弬鏁版暟鎹甛n");
-                }
-            }
-            else {
-                csvContent += _T("鏃犳硶瑙f瀽宸ヨ壓鍙傛暟\n");
-            }
-        }
-        else {
-            csvContent += _T("鏃犲伐鑹哄弬鏁版暟鎹甛n");
-        }
-
-        // 浣跨敤杈呭姪鍑芥暟淇濆瓨涓� UTF-8 缂栫爜
-        if (WriteAnsiStringAsUtf8ToFile(csvContent, filePath)) {
             CString strSuccess;
-            strSuccess.Format(_T("璁板綍宸蹭繚瀛樹负CSV鏂囦欢锛歕n%s"), filePath);
+            strSuccess.Format(_T("璁板綍宸蹭繚瀛樹负JSON鏂囦欢锛歕n%s"), filePath);
             AfxMessageBox(strSuccess);
         }
         else {
             AfxMessageBox(_T("淇濆瓨鏂囦欢澶辫触"));
         }
+    }
+    else {
+        AfxMessageBox(_T("璇ヨ褰曟病鏈塉SON鏁版嵁"));
+    }
+}
+
+void CPageGlassList::ExportToCsv(const GlassLogDb::Row& row, const CString& filePath)
+{
+    CString csvContent;
+
+    // === 绗竴閮ㄥ垎锛氬熀纭�淇℃伅 ===
+    ExportBasicInfo(csvContent, row);
+
+    // === 绗簩閮ㄥ垎锛氬伐鑹哄弬鏁� ===
+    ExportProcessParams(csvContent, row);
+
+    // === 绗笁閮ㄥ垎锛氫紶鎰熷櫒鏁版嵁璇︽儏 ===
+    ExportSensorData(csvContent, row);
+
+    // 浣跨敤杈呭姪鍑芥暟淇濆瓨涓� UTF-8 缂栫爜
+    if (WriteAnsiStringAsUtf8ToFile(csvContent, filePath)) {
+        CString strSuccess;
+        strSuccess.Format(_T("璁板綍宸蹭繚瀛樹负CSV鏂囦欢锛歕n%s"), filePath);
+        AfxMessageBox(strSuccess);
+    }
+    else {
+        AfxMessageBox(_T("淇濆瓨鏂囦欢澶辫触"));
+    }
+}
+
+void CPageGlassList::ExportBasicInfo(CString& csvContent, const GlassLogDb::Row& row)
+{
+    csvContent += _T("=== 鍩虹淇℃伅 ===\n");
+    csvContent += _T("ID,Cassette搴忓垪鍙�,Job搴忓垪鍙�,Glass ID,鐗╂枡绫诲瀷,鐘舵��,寮�濮嬫椂闂�,缁撴潫鏃堕棿,缁戝畾Glass ID,AOI缁撴灉,璺緞\n");
+
+    CString baseInfoRow;
+    baseInfoRow.Format(_T("%lld,%d,%d,%s,%d,%d,%s,%s,%s,%d,%s\n"),
+        row.id, row.cassetteSeqNo, row.jobSeqNo,
+        CString(row.classId.c_str()), row.materialType, row.state,
+        CString(row.tStart.c_str()), CString(row.tEnd.c_str()),
+        CString(row.buddyId.c_str()), row.aoiResult,
+        CString(row.path.c_str()));
+    csvContent += baseInfoRow;
+}
+
+void CPageGlassList::ExportProcessParams(CString& csvContent, const GlassLogDb::Row& row)
+{
+    csvContent += _T("\n=== 宸ヨ壓鍙傛暟 ===\n");
+
+    // 濡傛灉鏈� pretty 瀛楁锛岃В鏋愬伐鑹哄弬鏁�
+    if (!row.pretty.empty()) {
+        SERVO::CGlass tempGlass;
+        if (GlassJson::FromString(row.pretty, tempGlass)) {
+            auto& params = tempGlass.getParams();
+            if (!params.empty()) {
+                // 宸ヨ壓鍙傛暟琛ㄥご
+                csvContent += _T("鍙傛暟鍚嶇О,鍙傛暟ID,鏁板��,鏈哄櫒鍗曞厓\n");
+
+                // 宸ヨ壓鍙傛暟鏁版嵁
+                for (auto& param : params) {
+                    CString paramRow;
+                    CString valueStr;
+
+                    // 鏍规嵁鍙傛暟绫诲瀷鏍煎紡鍖栨暟鍊�
+                    if (param.getValueType() == PVT_INT) {
+                        valueStr.Format(_T("%d"), param.getIntValue());
+                    }
+                    else {
+                        valueStr.Format(_T("%.3f"), param.getDoubleValue());
+                    }
+
+                    paramRow.Format(_T("%s,%s,%s,%s\n"),
+                        CString(param.getName().c_str()),
+                        CString(param.getId().c_str()),
+                        valueStr,
+                        CString(param.getUnit().c_str()));
+
+                    csvContent += paramRow;
+                }
+            }
+            else {
+                csvContent += _T("鏃犲伐鑹哄弬鏁版暟鎹甛n");
+            }
+        }
+        else {
+            csvContent += _T("鏃犳硶瑙f瀽宸ヨ壓鍙傛暟\n");
+        }
+    }
+    else {
+        csvContent += _T("鏃犲伐鑹哄弬鏁版暟鎹甛n");
+    }
+}
+
+void CPageGlassList::ExportSensorData(CString& csvContent, const GlassLogDb::Row& row)
+{
+    csvContent += _T("\n=== 浼犳劅鍣ㄦ暟鎹鎯� ===\n");
+
+    // 濡傛灉鏈� pretty 瀛楁锛岃В鏋愪紶鎰熷櫒鏁版嵁
+    if (!row.pretty.empty()) {
+        SERVO::CGlass tempGlass;
+        if (GlassJson::FromString(row.pretty, tempGlass)) {
+#if USE_MOCK_SENSOR_DATA
+            // 鐢熸垚妯℃嫙鐨凷VData鐢ㄤ簬娴嬭瘯
+            GenerateMockSVData(tempGlass);
+#endif
+            // 瀵规瘡涓満鍣ㄧ敓鎴愯〃鏍�
+            for (const auto& machinePair : tempGlass.getAllSVData()) {
+                int machineId = machinePair.first;
+                CString machineName = CString(SERVO::CServoUtilsTool::getEqName(machineId).c_str());
+
+                csvContent += _T("\n[") + machineName + _T("]\n");
+
+                // 鑾峰彇璇ユ満鍣ㄧ殑棰勫畾涔夊垪椤哄簭
+                auto columnOrder = getMachineColumnOrder(machineId);
+
+                if (columnOrder.empty()) {
+                    csvContent += _T("鏃犻瀹氫箟鍒楅厤缃甛n");
+                    continue;
+                }
+
+                // 鏋勫缓琛ㄥご - 鐩存帴浣跨敤涓枃鍒楀悕
+                CString header = _T("鏃堕棿鎴�(ms),鏈湴鏃堕棿");
+                for (const auto& dataType : columnOrder) {
+                    header += _T(",");
+                    header += CString(dataType.c_str()); // 鐩存帴浣跨敤涓枃鍒楀悕
+                }
+                header += _T("\n");
+                csvContent += header;
+
+                // 妫�鏌ユ槸鍚︽湁鏁版嵁
+                if (machinePair.second.empty()) {
+                    csvContent += _T("鏃犱紶鎰熷櫒鏁版嵁\n");
+                    continue;
+                }
+
+                // 浣跨敤绗竴涓暟鎹被鍨嬬殑鏃堕棿搴忓垪浣滀负鍩哄噯
+                const std::string& firstDataType = columnOrder[0];
+                auto firstDataTypeIt = machinePair.second.find(firstDataType);
+                if (firstDataTypeIt == machinePair.second.end() || firstDataTypeIt->second.empty()) {
+                    csvContent += _T("鏃犲熀鍑嗘暟鎹被鍨嬫暟鎹甛n");
+                    continue;
+                }
+
+                const auto& timeSeries = firstDataTypeIt->second;
+
+                // 瀵逛簬姣忎釜鏃堕棿鐐癸紝杈撳嚭涓�琛屾暟鎹�
+                for (size_t i = 0; i < timeSeries.size(); i++) {
+                    auto timestamp = timeSeries[i].timestamp;
+
+                    // 鏃堕棿鎴筹紙姣锛�
+                    auto ms = timePointToMs(timestamp);
+                    CString row;
+                    row.Format(_T("%lld,"), ms);
+
+                    // 鏈湴鏃堕棿瀛楃涓�
+                    CString localTime = CString(timePointToString(timestamp).c_str());
+                    row += localTime;
+
+                    // 鎸夌収棰勫畾涔夌殑鍒楅『搴忚緭鍑烘暟鎹�
+                    for (const auto& dataType : columnOrder) {
+                        row += _T(",");
+
+                        auto dataTypeIt = machinePair.second.find(dataType);
+                        if (dataTypeIt != machinePair.second.end() && i < dataTypeIt->second.size()) {
+                            // 鐩存帴鎸夌储寮曡幏鍙栨暟鎹�
+                            CString valueStr;
+                            valueStr.Format(_T("%.3f"), dataTypeIt->second[i].value);
+                            row += valueStr;
+                        }
+                        else {
+                            // 鐞嗚涓婁笉搴旇鍙戠敓锛屽洜涓烘偍璇存病鏈夌┖鍊�
+                            row += _T("N/A");
+                        }
+                    }
+                    row += _T("\n");
+                    csvContent += row;
+                }
+            }
+        }
+        else {
+            csvContent += _T("鏃犳硶瑙f瀽浼犳劅鍣ㄦ暟鎹甛n");
+        }
+    }
+    else {
+        csvContent += _T("鏃犱紶鎰熷櫒鏁版嵁\n");
     }
 }
 
@@ -1797,4 +1916,117 @@
     }
 
     return CDialogEx::PreTranslateMessage(pMsg);
+}
+
+// 鑾峰彇鏈哄櫒棰勫畾涔夌殑鍒楅『搴�
+std::vector<std::string> CPageGlassList::getMachineColumnOrder(int machineId)
+{
+    auto dataTypes = SERVO::CServoUtilsTool::getEqDataTypes();
+    auto it = dataTypes.find(machineId);
+    return it != dataTypes.end() ? it->second : std::vector<std::string>();
+}
+
+// 鏃堕棿鎴宠浆鎹负瀛楃涓�
+std::string CPageGlassList::timePointToString(const std::chrono::system_clock::time_point& tp) 
+{
+    auto time_t = std::chrono::system_clock::to_time_t(tp);
+    std::tm tm;
+    localtime_s(&tm, &time_t);
+    char buffer[20];
+    std::strftime(buffer, sizeof(buffer), "%Y-%m-%d %H:%M:%S", &tm);
+    return buffer;
+}
+
+// 鏃堕棿鎴宠浆鎹负姣
+int64_t CPageGlassList::timePointToMs(const std::chrono::system_clock::time_point& tp)
+{
+    return std::chrono::duration_cast<std::chrono::milliseconds>(tp.time_since_epoch()).count();
+}
+
+// 鐢熸垚妯℃嫙鐨凷VData鐢ㄤ簬娴嬭瘯
+void CPageGlassList::GenerateMockSVData(SERVO::CGlass& glass)
+{
+    // 鑾峰彇璁惧鏁版嵁绫诲瀷閰嶇疆
+    auto& dataTypes = SERVO::CServoUtilsTool::getEqDataTypes();
+    
+    // 涓烘瘡涓澶囩敓鎴愭ā鎷熸暟鎹�
+    for (const auto& machinePair : dataTypes) {
+        int machineId = machinePair.first;
+        const auto& dataTypeList = machinePair.second;
+        
+        // 鐢熸垚鏃堕棿搴忓垪锛氫粠褰撳墠鏃堕棿寰�鍓嶆帹10鍒嗛挓锛屾瘡1绉掍竴涓暟鎹偣
+        auto now = std::chrono::system_clock::now();
+        auto startTime = now - std::chrono::minutes(10);
+        
+        // 涓烘瘡涓暟鎹被鍨嬬敓鎴愭ā鎷熸暟鎹�
+        for (const auto& dataType : dataTypeList) {
+            std::vector<SERVO::SVDataItem> mockData;
+            
+            // 鐢熸垚600涓暟鎹偣锛�10鍒嗛挓 * 60涓偣/鍒嗛挓锛�
+            for (int i = 0; i < 600; ++i) {
+                auto timestamp = startTime + std::chrono::seconds(i * 1);
+                
+                // 鏍规嵁璁惧绫诲瀷鍜屾暟鎹被鍨嬬敓鎴愪笉鍚岀殑妯℃嫙鍊�
+                double value = GenerateMockValue(machineId, dataType, i);
+                
+                mockData.emplace_back(timestamp, value);
+            }
+            
+            // 灏嗘ā鎷熸暟鎹坊鍔犲埌glass瀵硅薄涓�
+            glass.addSVData(machineId, dataType, mockData);
+        }
+    }
+}
+
+// 鏍规嵁璁惧绫诲瀷鍜屾暟鎹被鍨嬬敓鎴愭ā鎷熸暟鍊�
+double CPageGlassList::GenerateMockValue(int machineId, const std::string& dataType, int index)
+{
+    // 鍩虹鍊艰寖鍥�
+    double baseValue = 0.0;
+    double variation = 0.0;
+    
+    // 鏍规嵁璁惧绫诲瀷璁剧疆鍩虹鍊�
+    switch (machineId) {
+        case EQ_ID_Bonder1:
+        case EQ_ID_Bonder2:
+            if (dataType.find("鍘嬪姏") != std::string::npos) {
+                baseValue = 50.0;  // 鍘嬪姏鍩虹鍊�
+                variation = 10.0;  // 鍘嬪姏鍙樺寲鑼冨洿
+            } else if (dataType.find("娓╁害") != std::string::npos) {
+                baseValue = 180.0; // 娓╁害鍩虹鍊�
+                variation = 5.0;   // 娓╁害鍙樺寲鑼冨洿
+            } else if (dataType.find("鎵╁睍鍊�") != std::string::npos) {
+                baseValue = 100.0; // 鎵╁睍鍊煎熀纭�鍊�
+                variation = 15.0;  // 鎵╁睍鍊煎彉鍖栬寖鍥�
+            }
+            break;
+            
+        case EQ_ID_VACUUMBAKE:
+            if (dataType.find("鎵╁睍鍊�") != std::string::npos) {
+                baseValue = 80.0;
+                variation = 12.0;
+            } else if (dataType.find("娓╁害") != std::string::npos) {
+                baseValue = 200.0;
+                variation = 8.0;
+            }
+            break;
+            
+        case EQ_ID_BAKE_COOLING:
+            if (dataType.find("娓╁害") != std::string::npos) {
+                baseValue = 25.0;  // 鍐峰嵈娓╁害
+                variation = 3.0;
+            }
+            break;
+            
+        default:
+            baseValue = 50.0;
+            variation = 5.0;
+            break;
+    }
+    
+    // 娣诲姞鏃堕棿鐩稿叧鐨勮秼鍔垮拰闅忔満鍙樺寲
+    double timeTrend = sin(index * 0.1) * 2.0;  // 姝e鸡娉㈣秼鍔�
+    double randomNoise = (rand() % 100 - 50) / 100.0 * variation * 0.3;  // 闅忔満鍣0
+    
+    return baseValue + timeTrend + randomNoise;
 }
\ No newline at end of file

--
Gitblit v1.9.3