using AutoMapper; using FLY.Thick.Base.Common; using FLY.Thick.Base.Server; using FLY.Thick.Blowing; using FLY.Thick.Blowing.Common; using FLY.Thick.Blowing.IService; using FLY.Thick.Blowing.Server; using FLY.Thick.Blowing.Server.Model; using FLY.Thick.BlowingScan.Common; using FLY.Thick.BlowingScan.IService; using FlyADBase; using FObjBase; using FObjBase.Reflect; using MathNet.Numerics.LinearAlgebra.Double; using Misc; using Newtonsoft.Json; using System; using System.Collections.Generic; using System.ComponentModel; using System.IO; using System.Linq; using HistoryDb = FLY.Thick.BlowingScan.Server.Model.HistoryDb; namespace FLY.Thick.BlowingScan.Server { public class GM_BlowingScan : GM_ScanMotion, IBlowingScanService { #region 延时操作,markno const int MARKNO_PROFILE_ADD = 87; const int MARKNO_PROFILE_CHANGED = 88; #endregion /// <summary> /// 当前扫描号 /// </summary> public int ScanNo = 0; class ScanPosCell { public DateTime dt = DateTime.MinValue; public int pos;//位于膜的位置,不是机架 public double thk;//两层的厚度 //由pos转换!! public double angle1;//上层 位于膜泡的角度 public double angle2;//下层 位于膜泡的角度 public Misc.DIRECTION direction;//只有一整副数据都是同一个方向,才能解方程 public ScanPosCell Clone() { ScanPosCell scanpos = new ScanPosCell(); scanpos.dt = dt; scanpos.pos = pos; scanpos.thk = thk; scanpos.angle1 = angle1; scanpos.angle2 = angle2; scanpos.direction = direction; return scanpos; } } class EquationCell { public Dictionary<int, int> boltPower = new Dictionary<int, int>();//key=boltindex, value=power public double thk = 0; public int mainBolt;//主分区号 public void Add(int boltindex1, int boltindex2, double thk) { if (boltPower.ContainsKey(boltindex1)) { boltPower[boltindex1]++; } else { boltPower.Add(boltindex1, 1); } if (boltPower.ContainsKey(boltindex2)) { boltPower[boltindex2]++; } else { boltPower.Add(boltindex2, 1); } this.thk += thk; } public override string ToString() { string str = ""; foreach (var b in boltPower) { str += "[" + b.Key.ToString() + "," + b.Value.ToString() + "] "; } return str; } public double[] GetPower(int nbolts) { double[] power = new double[nbolts]; for (int i = 0; i < power.Count(); i++) power[i] = 0; foreach (var kv in boltPower) { power[kv.Key] = kv.Value; } return power; } public double GetAvg() { int cnt = 0; for (int i = 0; i < boltPower.Count(); i++) { cnt += boltPower.Values.ElementAt(i); } if (cnt > 0) return thk / cnt; else return double.NaN; } } class ScanInfoCell { public ScanInfoCell Clone() { ScanInfoCell scaninfo = new ScanInfoCell(ScanNo, thks.Count()); scaninfo.Direction = Direction; scaninfo.ScanNo = ScanNo; scaninfo.IsFull = IsFull; scaninfo.IndexOfToGetTime = IndexOfToGetTime; foreach (ScanPosCell scanpos in frame) { scaninfo.frame.Add(scanpos.Clone()); } scaninfo.width = width; scaninfo.beginPos = beginPos; scaninfo.endPos = endPos; scaninfo.lockbm = lockbm; return scaninfo; } public enum GetAngleResult { /// <summary> /// 全部grid 数据已经获取,而且都成功转为方程 /// </summary> AllGet, /// <summary> /// 部分grid 数据已经获取,这次转化了一些方程 /// </summary> PartGet, /// <summary> /// 什么都没干,没有转化为任何方程 /// </summary> NothingToDo } public GetAngleResult GetAngle(BlowingDetect rdetect, int orgboltno, int sensorWidth) { bool bHasNewData = false; for (; IndexOfToGetAngle < IndexOfToGetTime; IndexOfToGetAngle++) { int i = IndexOfToGetAngle; ScanPosCell scanpos = frame[i]; FilmInfo filminfo; if (scanpos.dt == DateTime.MinValue) { //异常,跳过 continue; } int ret = rdetect.GetFilmInfo( out filminfo, scanpos.dt, width, scanpos.pos ); if (ret == 1) { //发生在未来,等下一次Poll if (IsFull) return GetAngleResult.NothingToDo; else { if (bHasNewData) return GetAngleResult.PartGet; else return GetAngleResult.NothingToDo; } } if (filminfo == null)//异常,没回复 { continue; } //成功获取 scanpos.angle1 = filminfo.angle1; scanpos.angle2 = filminfo.angle2; scanpos.direction = filminfo.direction; if (IndexOfToGetAngle == 0) { Direction = scanpos.direction; } else { if (Direction != scanpos.direction) { Direction = DIRECTION.FIX; } } //探头直径转为角度范围 double sensor_angle = 180.0 * sensorWidth / width; if (AddEquation(i, orgboltno, sensor_angle)) bHasNewData = true; } if (IsFull) return GetAngleResult.AllGet; else { if (bHasNewData) return GetAngleResult.PartGet; else return GetAngleResult.NothingToDo; } } /// <summary> /// 角度转分区序号, 角度范围[0~360°) /// </summary> /// <param name="a">膜泡角度[0~360°)</param> /// <param name="orgboltno">复位区号</param> /// <returns></returns> int GetBoltIndexFromAngle(double a, int orgboltno) { int boltno1st = 1; int nbolts = thks.Count(); int boltindex = (int)(a / 360 * nbolts); boltindex += orgboltno - boltno1st; //100%确保输出的序号在有效范围内 while (boltindex < 0) boltindex += nbolts;//正常情况不会调用 while (boltindex>=nbolts) boltindex -= nbolts;//极端情况,a=360.2°, 这个会调用2次 return boltindex; } int boltindex1_last = -1; /// <summary> /// 完成了一些方程,返回true /// </summary> /// <param name="scanpos_idx"></param> /// <param name="orgboltno">复位区号</param> /// <param name="sensor_angle">探头直径对应的角度范围</param> /// <returns></returns> bool AddEquation(int scanpos_idx, int orgboltno, double sensor_angle) { ScanPosCell scanpos = frame[scanpos_idx]; if (equationList.Count() == 0)//复位记录 boltindex1_last = -1; EquationCell equation = new EquationCell(); equationList.Add(equation); int boltindex1 = GetBoltIndexFromAngle(scanpos.angle1, orgboltno); equation.mainBolt = boltindex1; equation.Add( GetBoltIndexFromAngle(scanpos.angle1, orgboltno), GetBoltIndexFromAngle(scanpos.angle2, orgboltno), scanpos.thk); //添加直径范围内的方程 for (int i = 1; i < sensor_angle / 2; i++) { equation.Add( GetBoltIndexFromAngle(scanpos.angle1 + i, orgboltno), GetBoltIndexFromAngle(scanpos.angle2 + i, orgboltno), scanpos.thk); equation.Add( GetBoltIndexFromAngle(scanpos.angle1 - i, orgboltno), GetBoltIndexFromAngle(scanpos.angle2 - i, orgboltno), scanpos.thk); } if (boltindex1_last != boltindex1)//记录不同,通知外部是时候解方程了 { boltindex1_last = boltindex1; return true; } return false; } public double GetAvg() { if (equationList.Count() < 0) return double.NaN; double sum = 0; int cnt = 0; for (int i = 0; i < equationList.Count(); i++) { EquationCell equation = equationList[i]; double a = equation.GetAvg(); if (!double.IsNaN(a)) { sum += a; cnt++; } } if (cnt > 0) return sum / cnt; else return double.NaN; } //public void SaveEquation(StreamWriter sw) //{ // for (int i = 0; i < equationList.Count(); i++) // { // EquationCell equation = equationList[i]; // string s = ScanNo.ToString(); // s +=","+ Direction.ToString(); // s += "," + StartTime.ToString(); // s+=","+equation.thick.ToString(); // for (int j = 0; j < equation.boltPower.Count(); j++) // { // int boltindex = equation.boltPower.Keys.ElementAt(j); // s += "," + boltindex.ToString(); // s += "," + equation.boltPower[boltindex].ToString(); // } // sw.WriteLine(s); // } //} //public static void SaveEquation_Header(StreamWriter sw) //{ // string s = "序号,方向,时间,总厚度,序号1,数量1,序号2,数量2"; // sw.WriteLine(s); //} public ScanInfoCell(int scanNo, int nbolts) { Reset(scanNo); thks = new double[nbolts]; for (int i = 0; i < nbolts; i++) { thks[i] = double.NaN; } } /// <summary> /// 开始测量时间 /// </summary> public DateTime StartTime; /// <summary> /// 结束测量时间 /// </summary> public DateTime EndTime; /// <summary> /// 旋转架旋转的方向,当同一次扫描中,有两个方向,Direction= Fix; /// </summary> public Misc.DIRECTION Direction; /// <summary> /// 扫描号,必须唯一!!!!! /// </summary> public int ScanNo; /// <summary> /// grid数据,已经全部放入!!!!! /// </summary> public bool IsFull; /// <summary> /// 准备要去获取时间的ScanPosCell序号, -1, 整个scaninfo 都发生在过去,作废 /// </summary> public int IndexOfToGetTime; /// <summary> /// 准备要去转角度的ScanPosCell序号 /// </summary> public int IndexOfToGetAngle; /// <summary> /// 快速解方法,equationList序号 /// </summary> public int IndexOfToSolveEquationQuick; public void Reset(int scanNo) { ScanNo = scanNo; frame.Clear(); equationList.Clear(); IsFull = false; IndexOfToGetTime = 0; IndexOfToGetAngle = 0; Direction = DIRECTION.FORWARD; } /// <summary> /// 复位数据;一幅grid成功生成,重新获取pos,thick,time,angle,equation /// </summary> /// <param name="beginPos"></param> /// <param name="endPos"></param> public void Reset_IsFull(int beginPos, int endPos) { this.beginPos = beginPos; this.endPos = endPos; this.width = endPos - beginPos + 1;//膜宽 this.IsFull = true; equationList.Clear(); frame.Clear(); IndexOfToGetTime = 0; IndexOfToGetAngle = 0; } void ClearFrame() { equationList.Clear(); frame.Clear(); IndexOfToGetTime = 0; IndexOfToGetAngle = 0; } public void Reset2GetAngle() { IndexOfToGetAngle = 0; equationList.Clear(); for (int i = 0; i < thks.Count(); i++) thks[i] = double.NaN; } /// <summary> /// 原始数据(时间,位置,总厚度 /// </summary> public List<ScanPosCell> frame = new List<ScanPosCell>(); /// <summary> /// 膜的宽度 /// </summary> public int width = -1; /// <summary> /// 膜边界左 /// </summary> public int beginPos; /// <summary> /// 膜边界右 /// </summary> public int endPos; /// <summary> /// 方程列表 /// </summary> public List<EquationCell> equationList = new List<EquationCell>(); /// <summary> /// 下面解方程得到的一幅数据对应的数据,与 StartTime 不一样 /// </summary> public DateTime thicks_dt; /// <summary> /// 一幅数据 /// </summary> public double[] thks; /// <summary> /// 经过分区表后的结果 /// </summary> public double[] thks_map; /// <summary> /// 锁定 bookmark,只要解过一次方程,bookmark 就会被锁定,不能再修改 /// </summary> public bool lockbm = false; public class SolveQuickResult { /// <summary> /// 新得到数据开始序号 /// </summary> public int start_index; /// <summary> /// 新得到数据长度,这是一个环形队列 /// </summary> public int len = 0; } /// <summary> /// 快速解方程 /// </summary> /// <param name="org_thks">上一幅完成的数据</param> /// <param name="power">上一幅图所占的权重,当前只占1</param> /// <returns></returns> public SolveQuickResult SolveQuick(double[] org_thks, int power) { List<int> updateboltindex = new List<int>(); for (int i = IndexOfToSolveEquationQuick; i < equationList.Count; i++, IndexOfToSolveEquationQuick++) { EquationCell equation = equationList[i]; int mainpower = equation.boltPower[equation.mainBolt]; double sum = 0; for (int j = 0; j < equation.boltPower.Count; j++) { int boltindex = equation.boltPower.Keys.ElementAt(j); if (boltindex != equation.mainBolt) { if (org_thks == null) { mainpower += equation.boltPower[boltindex]; } else if (double.IsNaN(org_thks[boltindex])) { mainpower += equation.boltPower[boltindex]; } else { sum += equation.boltPower[boltindex] * org_thks[boltindex]; } } } double mainBolt_value = (org_thks != null) ? org_thks[equation.mainBolt] : double.NaN; if (!double.IsNaN(mainBolt_value)) { //这个位置有效时,获取这个位置从现在到以前的均值 thks[equation.mainBolt] = (mainBolt_value * power + ((equation.thk - sum) / mainpower)) / (power + 1); } else { //这个位置无效时,获取这个位置以前的均值 thks[equation.mainBolt] = (equation.thk - sum) / mainpower; } updateboltindex.Add(equation.mainBolt); } SolveQuickResult result = new SolveQuickResult(); if (updateboltindex.Count > 0) { result.start_index = updateboltindex.Min(); result.len = updateboltindex.Max() - updateboltindex.Min() + 1; } return result; } } /// <summary> /// 每次扫描的数据, /// 不是每一幅数据都能解出方程, /// 因为要连续3幅才能解出方程 /// </summary> private RList<ScanInfoCell> mScanInfoList; /// <summary> /// 与 mScanInfoList 基本一样 /// </summary> public RList<FlyData_BlowingScan> mBufList; #region IRenZiJiaService #region 分区设置 /// <summary> /// 加热通道数 /// </summary> public int ChannelCnt { get; set; } = 88; /// <summary> /// 分区数/加热通道数 /// </summary> public int BPC { get; set; } = 1; /// <summary> /// 风环螺丝数 /// </summary> public int NBolts => ChannelCnt * BPC; /// <summary> /// 人字架复位时,对应的螺丝号,0° 对应的 螺丝号 /// </summary> public int OrgBoltNo { get; set; } = 1; private int smooth = 1; /// <summary> /// 平滑 /// </summary> public int Smooth { get { return smooth; } set { if (value < 0) value = 0; if (smooth != value) { smooth = value; } } } /// <summary> /// 使用分区表 /// </summary> public bool IsUsedMap { get; set; } bool IsNeedMap() { if (IsUsedMap && Map != null && Map.Count() > 0) return true; else return false; } private BoltMapCell[] map; /// <summary> /// 分区表,必须保证不能为null /// </summary> public BoltMapCell[] Map { get { return map; } set { //检测是否不一样 if (!IsEquals(map, value)) { if (!IsValid(value)) { IsUsedMap = false; map = null; } else { map = value; } } } } /// <summary> /// 检测通过,返回true /// </summary> /// <returns></returns> bool IsValid(BoltMapCell[] mapCells) { bool map_err = false; bool g_o = false; bool g_n = false; if (mapCells == null) return true; //Map 检测 //OldNo,NewNo不能 <1 >NBolts //Map OldNo 必须从小到大环形排列 //Map NewNo 必须从小到大环形排列 for (int i = 0; i < mapCells.Count(); i++) { var mapCell = mapCells[i]; if (mapCell.NewNo < 1 || mapCell.NewNo > NBolts) { map_err = true; break; } if (mapCell.OldNo < 1 || mapCell.OldNo > NBolts) { map_err = true; break; } int i_next = i + 1; if (i_next >= mapCells.Count()) i_next = 0; var mapCell_next = mapCells[i_next]; if (mapCell.OldNo >= mapCell_next.OldNo) { if (!g_o) g_o = true; else { map_err = true; break; } } if (mapCell.NewNo >= mapCell_next.NewNo) { if (!g_n) g_n = true; else { map_err = true; break; } } } if (map_err) { return false; } else { return true; } } bool IsEquals(BoltMapCell c1, BoltMapCell c2) { if (c1 == c2) { return true; } if (c1 == null && c2 != null) return false; if (c1 != null && c2 == null) return false; if (c1.NewNo != c2.NewNo) return false; if (c1.OldNo != c2.OldNo) return false; return true; } bool IsEquals(BoltMapCell[] mapCells1, BoltMapCell[] mapCells2) { if (mapCells1 == mapCells2) return true; if (mapCells1 == null && mapCells2 != null) return false; if (mapCells1 != null && mapCells2 == null) return false; if (mapCells1.Count() != mapCells2.Count()) return false; //mapCells1 与 mapCells2 肯定都不是 null 且数量一样 for (int i = 0; i < mapCells1.Count(); i++) { var c1 = mapCells1[i]; var c2 = mapCells2[i]; if (!IsEquals(c1, c2)) return false; } return true; } #endregion /// <summary> /// 测厚仪类型, 追边 or 扫描 /// </summary> public BlowingType BType { get; } = BlowingType.Scan; /// <summary> /// 测厚仪测量点位置方向:Left, Right (也就是三角形在左还是右) /// </summary> public bool IsProbeRight { get; set; } = true; /// <summary> /// 正方向是顺时钟; /// 信号0->信号1 为正方向; /// 风环加热棒排列方向为 正方向; /// 默认情况 风环加热棒排列是逆时钟 /// </summary> public bool IsForwCW { get; set; } /// <summary> /// 应用 /// </summary> public void Apply() { Save(); } [Push(typeof(RenZiJiaDataEventArgs))] public event EventHandler DataEvent; #endregion /// <summary> /// 缓冲区数据量 /// </summary> public int BufListCount { get; private set; } = 0; /// <summary> /// 缓冲区最后一个记录序号 /// </summary> public int BufListLastNo { get; private set; } = -1; private int sensorwidth = 250; /// <summary> /// 探头宽度,用于解方程,可以与实际不符合 单位脉冲 /// </summary> public int SensorWidth { get { return sensorwidth; } set { if (value < 1) value = 1; if (sensorwidth != value) { sensorwidth = value; } } } private int solvecnt = 4; /// <summary> /// 多少次扫描解一次方程 /// </summary> public int SolveCnt { get { return solvecnt; } set { if (value < 3) value = 3; if (solvecnt != value) { solvecnt = value; } } } void checkParamErr() { if (ChannelCnt < 20 || ChannelCnt > 160) { ChannelCnt = 44; } if (BPC < 1 || BPC > 4) { BPC = 2; } if (OrgBoltNo < 1 || OrgBoltNo > NBolts) { OrgBoltNo = 1; } if (SolveCnt > 20) { SolveCnt = 5; } if (!IsValid(Map)) { IsUsedMap = false; Map = null; } } /// <summary> ///边界查找模块 /// </summary> private BorderSearch mBorderSearch; /// <summary> ///人字架位置检测器 /// </summary> public BlowingDetect mPDetect; /// <summary> /// AD值转厚度值 /// </summary> private AD2ThkHandler Ad2Thk; BlowingFixProfileParam mProfileParam; HistoryDb mHistoryDb; BulkDb bulkDb; /// <summary> /// 动态数据组合 /// </summary> private DynArea mDynArea; /// <summary> /// 扫描脉冲检测 /// </summary> private ScanPosDetect mScanPosDetect; InitParam initParam; OBJComponents.Server.WarningSystem2 warningSystem; public GM_BlowingScan() { mPDetect = new BlowingDetect(); mScanInfoList = new RList<ScanInfoCell>(30); mBufList = new RList<FlyData_BlowingScan>(100); mScanPosDetect = new ScanPosDetect(); ChannelCnt = 44;//加热棒数量 BPC = 2;//分区数量/加热棒数量 OrgBoltNo = 10;//复位时,测厚仪对应的螺丝号 SolveCnt = 5; mPDetect.Load(); Load(); //检测参数 mPDetect.CheckParamErr(); checkParamErr(); if (SolveCnt < 3) SolveCnt = 3; PropertyChanged += new PropertyChangedEventHandler(RenZiJiaScan_PropertyChanged); mPDetect.PropertyChanged += (s, e) => { NotifyPropertyChanged(e.PropertyName); }; mPDetect.FilmInfoChangedEvent += new FilmInfoChangedEventHandler(mPDetect_FilmInfoChangedEvent); mPDetect.ClearEvent += new ClearEventHandler(mPDetect_ClearEvent); } public void Init( FlyAD7 flyAd, GSample gsample, InitParam initParam, OBJComponents.Server.WarningSystem2 warningSystem, BorderSearch borderSearch, DynArea dynArea, AD2ThkHandler ad2thk, BlowingFixProfileParam profileParam, HistoryDb historyDb, BulkDb bulkDb ) { base.Init(flyAd, gsample); this.initParam = initParam; this.warningSystem = warningSystem; Ad2Thk = ad2thk; mBorderSearch = borderSearch; mDynArea = dynArea; mProfileParam = profileParam; mHistoryDb = historyDb; this.bulkDb = bulkDb; mPDetect.Init(mFlyAD); mScanPosDetect.Init(mFlyAD); //接收PollScan的推送 DataPosEvent += new EndEventHandler(pollScan_DataPosEvent); EndEvent += new EndEventHandler(pollScan_EndEvent); //记录调试数据 //MarkDebugInit(); } #region 记录原始数据 //DateTime markdebug_dt = DateTime.MinValue; //int markdebug_lastBm = -1; //void MarkDebugInit() //{ // if (markdebug_dt == DateTime.MinValue) // markdebug_dt = DateTime.Now; // //启动1分钟定时器 // FObjBase.PollModule.Current.Poll_Config( // FObjBase.PollModule.POLL_CONFIG.ADD, // () => // { // mPDetect.GetSignList(markdebug_dt, (asyncState, retData) => // { // var reponse = retData as GetSignListReponse; // var cells = reponse.datas; // if (cells.Count > 0) // { // mHistoryDb.AddSignData( // new Lc_Sign() // { // Time = cells.First().Time, // Signs = cells.ToArray() // }); // } // }, null); // mPDetect.GetRollList(markdebug_dt, (asyncState, retData) => // { // var reponse = retData as GetRollListReponse; // var cells = reponse.datas; // if (cells.Count > 0) // { // mHistoryDb.AddRollData( // new Lc_Roll() // { // Time = cells.First().dt, // Signs = cells.ToArray() // }); // } // }, null); // int index = mBufList.No2Index(markdebug_lastBm); // index++; // if (index < 0) // index = 0; // if (index < mBufList.Count()) // { // //有新的数据 // List<Model.Lc_OrgScanData> lc_OrgScanDatas = new List<Model.Lc_OrgScanData>(); // for (int i = index; i < mBufList.Count(); i++) // { // var data = mBufList[i]; // var times = data.ThicksDt.Where(dt => dt != DateTime.MinValue); // if (times.Count() == 0) // continue; // Model.Lc_OrgScanData lcOrgScanData = new Model.Lc_OrgScanData(); // lcOrgScanData.Time = times.Min(); // lcOrgScanData.FilmBegin = data.FilmRange.Begin; // lcOrgScanData.FilmEnd = data.FilmRange.End; // lcOrgScanData.PosOfGrid = data.PosOfGrid; // lcOrgScanData.K = mProfileParam.K; // lcOrgScanData.Thicks = ToRealThicks(data.Thicks); // lcOrgScanData.ThicksDt = data.ThicksDt; // lc_OrgScanDatas.Add(lcOrgScanData); // } // mHistoryDb.AddOrgScanDatas(lc_OrgScanDatas); // }; // markdebug_dt = DateTime.Now; // }, // TimeSpan.FromMinutes(1)); //} #endregion double[] Round(IEnumerable<double> frame, int digits) { return frame.Select(t => { if (!double.IsNaN(t)) return Math.Round(t, digits); else return double.NaN; }).ToArray(); } void RenZiJiaScan_PropertyChanged(object sender, PropertyChangedEventArgs e) { if (e.PropertyName == nameof(NBolts)) { //所有数据删除 mScanInfoList.Clear(); //TODO!!!!! //停下来 } else if (e.PropertyName == nameof(OrgBoltNo)) { for (int i = 0; i < mScanInfoList.Count(); i++) { mScanInfoList[i].Reset2GetAngle(); } } } void mPDetect_FilmInfoChangedEvent(object sender, FilmInfoChangedEventArgs e) { // 找到 e.Time 对应的 scaninfo for (int i = 0; i < mScanInfoList.Count; i++) { ScanInfoCell scaninfo = mScanInfoList[i]; if (scaninfo.IndexOfToGetTime > 0) { DateTime dt_start = scaninfo.frame[0].dt; DateTime dt_end = scaninfo.frame[scaninfo.IndexOfToGetTime - 1].dt; if (e.Time < dt_end)//找到了 { for (int j = i; j < mScanInfoList.Count; j++) { scaninfo = mScanInfoList[j]; scaninfo.IndexOfToGetAngle = 0; } return; } } } } void mPDetect_ClearEvent(object sender) { //已经很久没数据,清空全部!!!!!!! //scanposdetect mScanPosDetect.ScanNo = ScanNo; if (mScanInfoList.Count > 0) { mScanInfoList.Clear(); //scaninfo ScanInfoCell scaninfo = new ScanInfoCell(ScanNo, NBolts); //scaninfo.bookmark = bookmark; mScanInfoList.Add(scaninfo); } } #region 1.grid 数据 转 ScanPosCell.(pos,thick) 放入 mScanInfoList.Last() /// <summary> /// 动态获取!!!! /// </summary> /// <param name="obj"></param> /// <param name="e"></param> void pollScan_DataPosEvent(object obj, FlyADBase.MiniGridEventArgs e) { DIRECTION direction = e.direction; int posOfGrid = e.posOfGrid; int start_idx = e.grid_start; int start_pos = e.grid_start * e.posOfGrid; int[] dat = e.buf; //------------------------------------- //DIR mDynArea.AD = dat[0]; mDynArea.Thk = Ad2Thk(dat[0]); if (mScanInfoList.Count == 0) return; ScanInfoCell scaninfocell = mScanInfoList.Last(); if (scaninfocell.width <= 0) { //刚开始,连膜边都还没找到 return; } //找边界前,要对数据进行校正 //尽可能不要两个边的数据,N2是边界向内减, 只是不要更多的数据而已 int bp = scaninfocell.beginPos + mBorderSearch.N2; int ep = scaninfocell.endPos - mBorderSearch.N2; //对应的grid号 int bp_idx = bp / posOfGrid; int ep_idx = ep / posOfGrid; if (bp_idx < start_idx) bp_idx = start_idx; if (ep_idx > (start_idx + dat.Length - 1)) ep_idx = start_idx + dat.Length - 1; for (int i = bp_idx; i <= ep_idx; i++) { int idx = i - start_idx; int pos = i * posOfGrid + posOfGrid / 2;//一个grid的中间脉冲,也就是大概而已 double thk = double.NaN; if (!double.IsNaN(dat[idx])) thk = Ad2Thk(dat[idx]) * 2; pos -= scaninfocell.beginPos;//膜上面的位置 scaninfocell.frame.Add(new ScanPosCell() { pos = pos, thk = thk }); } OnPoll_GetTime2(); } void pollScan_EndEvent(object obj, FlyADBase.MiniGridEventArgs e) { DIRECTION direction = e.direction; int posOfGrid = e.posOfGrid; int start_idx = e.grid_start; int start_pos = e.grid_start * e.posOfGrid; int[] dat = e.buf; //int scaninfo_idx = mScanInfoList.FindIndex(sil => sil.IndexOfToGetTime < sil.frame.Count); //if ((scaninfo_idx>0) && (mScanInfoList.Count - scaninfo_idx > 1) ) //{ // //有问题 //} ScanNo++; mScanPosDetect.ScanNo = ScanNo; if (gsample.Samples[0].AD <= 0) { //无温修AD if (mScanInfoList.Count == 0) return; ScanInfoCell scaninfo = mScanInfoList.Last(); //清空动态数据 bulkDb.SetTempFrame(scaninfo.StartTime, scaninfo.EndTime, 0, null); scaninfo.Reset(ScanNo); return; } //找边界 bool ret = mBorderSearch.FindFilm(direction, posOfGrid, start_idx, dat); if (!ret)//不能找到边界!!!没膜!!!! { mBorderSearch.Reset(); } //设置扫描范围 //POLLSCAN---------------------------------------- Range scanrange = mBorderSearch.GetScanRange(); ScanStart = scanrange.Begin;// beginPos; ScanEnd = scanrange.End;// endPos; if (!ret) { if (mScanInfoList.Count == 0) return; ScanInfoCell scaninfo = mScanInfoList.Last(); //清空动态数据 bulkDb.SetTempFrame(scaninfo.StartTime, scaninfo.EndTime, 0, null); scaninfo.Reset(ScanNo); return; } //必须设置好 mBorderSearch.SensorWidth,它就是探头直径 Range border = mBorderSearch.GetBorder(direction); if (mScanInfoList.Count == 0)//安全检查,不可能发生的!!!!! return; ScanInfoCell scaninfocell = mScanInfoList.Last(); //膜宽,和位置一起计算旋转架上的角度 scaninfocell.Reset_IsFull(border.Begin, border.End); //尽可能不要两个边的数据,N2是边界向内减, 只是不要更多的数据而已 Range boltrange = mBorderSearch.GetBoltRange(direction); //对应的grid号 int bp_idx = boltrange.Begin / posOfGrid; int ep_idx = boltrange.End / posOfGrid; if (bp_idx < start_idx) bp_idx = start_idx; if (ep_idx > (start_idx + dat.Length - 1)) ep_idx = start_idx + dat.Length - 1; for (int i = bp_idx; i <= ep_idx; i++) { int idx = i - start_idx; int pos = i * posOfGrid + posOfGrid / 2;//一个grid的中间脉冲,也就是大概而已 double thk = double.NaN; if (Misc.MyBase.ISVALIDATA(dat[idx])) thk = Ad2Thk(dat[idx]) * 2; pos -= border.Begin;//膜上面的位置 scaninfocell.frame.Add(new ScanPosCell() { pos = pos, thk = thk }); } //或取每个数据对应的时间点 OnPoll_GetTime2(); //把扫描数据放多一份到 mBufList PushToBufList(e, border, scaninfocell.ScanNo); //插入纵向数据 #region sqlite3 历史数据保存 { List<double> boltValues = new List<double>(); double boltwidth = scaninfocell.width * 1.0 / (NBolts / 2); int boltIdx = 0; double boltSum = 0; int boltCnt = 0; for (int i = 0; i < scaninfocell.frame.Count(); i++) { int idx = (int)(scaninfocell.frame[i].pos / boltwidth); if (boltIdx != idx) { if (boltCnt > 0) { boltValues.Add(boltSum / boltCnt); boltSum = 0; boltCnt = 0; } } boltIdx = idx; if (!double.IsNaN(scaninfocell.frame[i].thk)) { boltSum += scaninfocell.frame[i].thk / 2; boltCnt++; } } if (boltCnt > 0) { double v = boltSum / boltCnt; v = Math.Round(v, 2);//限制小数点 boltValues.Add(v); } mHistoryDb.AddTrendData(new Lc_TrendData() { Time = DateTime.Now, K = mProfileParam.K, Thicks = boltValues.ToArray() }); } #endregion //创建一个新的 scaninfocell //为下一趟即时扫描作准备 scaninfocell = new ScanInfoCell(ScanNo, NBolts); direction = (direction == DIRECTION.FORWARD) ? DIRECTION.BACKWARD : DIRECTION.FORWARD; border = mBorderSearch.GetBorder(direction); scaninfocell.beginPos = border.Begin; scaninfocell.endPos = border.End; scaninfocell.width = border.Width;//膜宽 scaninfocell.IsFull = false; mScanInfoList.RAdd(scaninfocell); } void PushToBufList(MiniGridEventArgs e, Range border, int scanNo) { DIRECTION direction = e.direction; int posOfGrid = e.posOfGrid; int start_idx = e.grid_start; int start_pos = e.grid_start * e.posOfGrid; int[] dat = e.buf; FlyData_BlowingScan f = new FlyData_BlowingScan(); f.FilmRange = border; f.PosOfGrid = posOfGrid; f.Thks = new double[mFlyAD.PosLen / posOfGrid]; f.ThicksDt = new DateTime[f.Thks.Count()]; //对应的grid号 for (int i = 0; i < f.Thks.Count(); i++) { int idx = i - start_idx; int pos = i * posOfGrid + posOfGrid / 2;//一个grid的中间脉冲,也就是大概而已 double thk = double.NaN; if (idx >= 0 && idx < dat.Count()) { if (!double.IsNaN(dat[idx])) { thk = Ad2Thk(dat[idx]) * 2; thk = Math.Round(thk, 2);//保存到 FlyData_BlowingScan 的, 需要通讯, 限制小数点 } } f.Thks[i] = thk; DateTime dt = DateTime.MinValue; if (!double.IsNaN(thk)) { int ret = mScanPosDetect.GetTime(pos, scanNo, out dt); } f.ThicksDt[i] = dt; } f.RotateCnt = mPDetect.RotationCnt; if (mPDetect.mLimitList.Count() > 0) { LimitCell lc = mPDetect.mLimitList.Last(); f.SignNo = lc.no; f.Time = lc.dt_begin; f.LeaveTime = lc.dt_end; } else { f.SignNo = -1; f.Time = DateTime.MinValue; f.LeaveTime = DateTime.MinValue; } f.LimitSignTime = mPDetect.LimitSignTime; f.RotatePeriod = mPDetect.RenZiJiaPeriod; f.Velocity = mPDetect.FilmVelocity; mBufList.RAdd(f); BufListLastNo = mBufList.GetLastNo(); BufListCount = mBufList.Count(); } #endregion #region 2.填入 ScanPosCell.(dt) /// <summary> /// 找到一幅数据,每个grid 对应的时间点 /// </summary> void OnPoll_GetTime2() { ScanInfoCell scaninfo = mScanInfoList.Last(); for (; scaninfo.IndexOfToGetTime < scaninfo.frame.Count; scaninfo.IndexOfToGetTime++) { int i = scaninfo.IndexOfToGetTime; ScanPosCell scanpos = scaninfo.frame[i]; //还原为机架位置 int pos = scaninfo.beginPos + scanpos.pos; DateTime dt; int ret = mScanPosDetect.GetTime(pos, scaninfo.ScanNo, out dt); if (ret == -2) { //发生在过去,不正常 } else if (ret == 1)//这个是未来的,等下一次Poll { return; } else if (dt == DateTime.MinValue) { //出错, 把scaninfo 删除掉 //mScanInfoList.RemoveAt(scaninfo_idx); //scaninfo_idx--; break; } //成功获取 scanpos.dt = dt; if (i == 0) { scaninfo.StartTime = dt; scaninfo.thicks_dt = dt; } scaninfo.EndTime = dt; } } #endregion #region 3.填入 ScanPosCell.(angle1,angle2), 且转为 Equation void OnPoll_GetAngle() { int scaninfo_idx = mScanInfoList.FindIndex( sil => (sil.IndexOfToGetAngle < sil.IndexOfToGetTime)); if (scaninfo_idx == -1) { return;//全部都转换完成!! } while (true) { ScanInfoCell scaninfo = mScanInfoList[scaninfo_idx]; ScanInfoCell.GetAngleResult result = scaninfo.GetAngle(mPDetect, OrgBoltNo, SensorWidth); if (result == ScanInfoCell.GetAngleResult.NothingToDo) { //getangle 失败,发生在未来,等下一次Poll break; } //解方程,画数据!! ToThicks(scaninfo_idx); //全部GetAngle 成功,下一个scaninfo scaninfo_idx++; if (scaninfo_idx >= mScanInfoList.Count) break;//已经没有了 } } class CalSolveCnt { List<int> list = new List<int>(); const int listcap = 5; void Add(int cnt) { list.Add(cnt); if (list.Count > listcap) list.RemoveAt(0); } public int Max() { if (list.Count > 0) return list.Max(); else return 4; } Misc.DIRECTION direction = DIRECTION.FIX; int cnt = 0; public bool Check(Misc.DIRECTION direction) { if (this.direction != direction) { if ((this.direction != Misc.DIRECTION.FIX) && (cnt > 3)) Add(cnt); cnt = 0; } cnt++; this.direction = direction; return OK(); } public bool OK() { if (list.Count >= listcap) { return true; } return false; } } CalSolveCnt mCalSolveCnt = new CalSolveCnt(); /// <summary> /// 解方程 /// </summary> /// <param name="scaninfo_idx"></param> /// <returns></returns> bool ToThicks_SolveEquation(int scaninfo_idx) { ScanInfoCell scaninfo = mScanInfoList[scaninfo_idx]; if (scaninfo.Direction == DIRECTION.FIX)//当前是Fix 不解方程 return false; //解方程数量检测,免得一直数量不足,解不了方程 //if (mCalSolveCnt.Check(scaninfo.Direction)) //{ // if (SolveCnt > (mCalSolveCnt.Max()-1)) // { // SolveCnt = (mCalSolveCnt.Max() - 1); // } //} if (scaninfo_idx < (SolveCnt - 1)) return false; //全部方向都必须一致,当然也不能是Fix //for (int i = 0; i < (SolveCnt - 1); i++) //{ // if (scaninfo.Direction != mScanInfoList[scaninfo_idx - 1 - i].Direction) // return false; //} double avg = scaninfo.GetAvg(); if (double.IsNaN(avg))//平均值不合法!!!! return false; if (avg < 5)//小于5um ,太小!!! return false; List<EquationCell> equationlist = new List<EquationCell>(); ////全部平均值相差不能超过 5% //for (int i = 0; i < SolveCnt; i++) //{ // int a = mScanInfoList[scaninfo_idx - 1 - i].GetAvg(); // if (Math.Abs((a - avg) / avg) > 0.05) // return false; // equationlist.AddRange(mScanInfoList[scaninfo_idx - i].equationList); //} int cnt = 0; for (int i = scaninfo_idx; i >= 0; i--) { if (mScanInfoList[i].Direction == DIRECTION.FIX) continue; cnt++; double a = mScanInfoList[i].GetAvg(); double p = (a - avg) / avg; if (Math.Abs(p) > 0.05) return false; equationlist.AddRange(mScanInfoList[i].equationList); if (cnt >= SolveCnt) break; } if (cnt < SolveCnt)//方程不够!!!! return false; if (equationlist.Count < 100)//方程数太少 return false; //2105 条方程,耗时 0.3s SolveEquation(mScanInfoList[scaninfo_idx].thks, equationlist); //不能有任何一个NULL_VALUE for (int i = 0; i < scaninfo.thks.Count(); i++) { if (double.IsNaN(scaninfo.thks[i])) return false; if (scaninfo.thks[i] > 10000) return false; if (scaninfo.thks[i] < -100) return false; } //检测 { double sigma = scaninfo.thks.Sigma(); double avg2 = scaninfo.thks.Average(); double p = 1.0 * (avg2 - avg) / avg; if (Math.Abs(p) > 0.05)//变化太大,不可能!!! return false; p = 1.0 * sigma / avg2; if (Math.Abs(p) > 0.5) //波动太大,异常 return false; } //最后处理,滤波,平滑 if (Smooth > 0) { double[] data = new double[scaninfo.thks.Count()]; for (int i = 0; i < scaninfo.thks.Count(); i++) { double sum = 0; for (int j = 0; j < (Smooth * 2 + 1); j++) { int index = i + j - Smooth; if (index < 0) index += scaninfo.thks.Count(); else if (index >= scaninfo.thks.Count()) index -= scaninfo.thks.Count(); sum += scaninfo.thks[index]; } data[i] = (sum / (Smooth * 2 + 1)); } Array.Copy(data, scaninfo.thks, data.Count()); } scaninfo.thicks_dt = mScanInfoList[scaninfo_idx - (SolveCnt - 1)].StartTime; return true; } enum BindingMotion { Add, Remove } DateTime lastTempBeginTime; bool ToThicks(int scaninfo_idx) { ScanInfoCell scaninfo = mScanInfoList[scaninfo_idx]; if (scaninfo.IsFull)//所有grid数据都获取完,解方程 { if (scaninfo.lockbm) { //不需要再解方程,算thicks 的值!!! return false; } if (ToThicks_SolveEquation(scaninfo_idx)) { //解方程成功 //重新分区 if (IsNeedMap()) { //上面已经检查了 Map == null 的情况 Dictionary<int, int> map = new Dictionary<int, int>(); for (int i = 0; i < Map.Count(); i++) map.Add(Map[i].OldNo - 1, Map[i].NewNo - 1); scaninfo.thks_map = Misc.MyMath.Map(scaninfo.thks, map); } //更新数据库内容 { double[] frame; List<BoltMapCell> map; if (IsNeedMap()) { //上面已经检查了 Map == null 的情况 frame = scaninfo.thks_map; map = new List<BoltMapCell>(); foreach (var m in Map) map.Add(m.Clone()); } else { frame = scaninfo.thks; map = null; } var realthicks = Round(frame, 2); var renZiJiaDataEventArgs = new RenZiJiaDataEventArgs() { Time = scaninfo.thicks_dt, EndTime = scaninfo.EndTime, IsBackw = scaninfo.Direction == DIRECTION.BACKWARD, RPeriod = mPDetect.RenZiJiaPeriod, RCnt = mPDetect.RotationCnt, OrgBoltNo = OrgBoltNo, RAngle = mPDetect.RAngle, FilmLength = mPDetect.FilmLength, FilmVelocity = mPDetect.FilmVelocity, K = mProfileParam.K, Thicks = realthicks, Boltmap = map }; //TODO 只有最后一幅图推送 //if(scaninfo_idx == mScanInfoList.Count() - 1) //居然还有一幅更新的,但里面数据是空 { bulkDb.SetTempFrame(scaninfo.StartTime, scaninfo.EndTime, 0, Round(scaninfo.thks, 2)); if (!mDynArea.SecuteLock) //授权过期, 没有数据 { DataEvent?.Invoke(this, renZiJiaDataEventArgs); //TODO 需要检查是否重复写入了!!!!! #region sqlite3 历史数据保存 var lc_scandata = Lc_AutoMapperProfile.Mapper.Map<Lc_ScanData>(renZiJiaDataEventArgs); mHistoryDb.AddScanData(lc_scandata); } #endregion } } } else { //失败,删除!!! for (int i = 0; i < scaninfo.thks.Count(); i++) scaninfo.thks[i] = double.NaN; } scaninfo.lockbm = true; return true; } else { //还没获取完所有grid, 简易解方程,只认为 mainBolt 是未知的,其它都是已知的,用上一幅数据的值 //这样的数据不需要放在数据库中!!!!!! double[] org_thks = null; if (scaninfo_idx != 0) org_thks = mScanInfoList[scaninfo_idx - 1].thks; ScanInfoCell.SolveQuickResult result = scaninfo.SolveQuick(org_thks, SolveCnt - 1); if (mScanInfoList.Count() - 1 == scaninfo_idx) { //最后一幅图,需要画出来 if (result.len > 0) { if (scaninfo.StartTime != lastTempBeginTime) { //新的,全部发送 bulkDb.SetTempFrame(scaninfo.StartTime, scaninfo.EndTime, 0, Round(scaninfo.thks, 2)); } else { int cnt = scaninfo.thks.Count() - result.start_index; if (result.len < cnt) { //没有环路 var thks = scaninfo.thks.Skip(result.start_index).Take(result.len); bulkDb.SetTempFrame(scaninfo.StartTime, scaninfo.EndTime, result.start_index, Round(thks, 2)); } else { //转了个圈,只能全部发送 bulkDb.SetTempFrame(scaninfo.StartTime, scaninfo.EndTime, 0, Round(scaninfo.thks, 2)); } } lastTempBeginTime = scaninfo.StartTime; } } } return false; } /// <summary> /// 解方程 /// </summary> /// <param name="equationList">N条方程</param> /// <param name="frame_out">输出的结果</param> void SolveEquation(double[] frame_out, List<EquationCell> equationList) { int nrow = equationList.Count(); //A~AX = A~B //转换为 DenseMatrix 与 DenseVector DenseMatrix matrixA = new DenseMatrix(nrow, NBolts); DenseMatrix matrixAT = new DenseMatrix(NBolts, nrow); DenseVector vectorB = new DenseVector(nrow); DenseVector vectorATB = new DenseVector(NBolts); int rowindex = 0; foreach (EquationCell ec in equationList) { double[] datas = ec.GetPower(NBolts); matrixA.SetRow(rowindex, datas); matrixAT.SetColumn(rowindex, datas); vectorB[rowindex] = ec.thk; rowindex++; } DenseMatrix ATA = (DenseMatrix)matrixA.TransposeThisAndMultiply(matrixA); DenseVector ATB = (DenseVector)matrixA.TransposeThisAndMultiply(vectorB); DenseVector X = (DenseVector)ATA.Solve(ATB); for (int i = 0; i < NBolts; i++) { frame_out[i] = X[i]; } } #endregion /// <summary> /// 开始扫描 /// </summary> /// <returns></returns> public override void Start() { base.Start(); if (IsRunning == false) { warningSystem.Add( ERRNOs.Instance.SCAN_ERRNO_FATAL.Code, ERRNOs.Instance.SCAN_ERRNO_FATAL.Descrption + " :AD卡连接断开"); return; } //设置扫描范围 int beginPos, endPos; mBorderSearch.Reset(); mBorderSearch.GetScanRange(out beginPos, out endPos); ScanNo++; //scanposdetect mScanPosDetect.ScanNo = ScanNo; //scaninfo ScanInfoCell scaninfo = new ScanInfoCell(ScanNo, NBolts); mScanInfoList.Add(scaninfo); //清空临时数据 bulkDb.SetTempFrame(DateTime.MinValue, DateTime.MinValue, 0, null); #region sqlite3 历史数据保存 CheckProfile(); mProfileParam.PropertyChanged += MProfileParam_PropertyChanged; #endregion base.Start2(beginPos, endPos); } #region sqlite3 历史数据保存 string[] profile_propertynames_add = new string[]{ "PName", "OrderNo", "Number" }; string[] profile_propertynames_update = new string[]{ "Target", "TolerancePercent" }; /// <summary> /// 检查产品信息更新情况,是否需要新加单,或者更新单信息 /// </summary> void CheckProfile() { if (mHistoryDb.localDb.IsProfileFinished) { //上一单结束了,添加新单 AddProfile(); } else { if (mHistoryDb.localDb.CurrProfile != null) { Db_Profile db_Profile = mHistoryDb.localDb.CurrProfile; if ((db_Profile.PName != mProfileParam.PName) || (db_Profile.OrderNo != mProfileParam.OrderNo) || (db_Profile.Number != mProfileParam.Number)) { //产品名称/订单号/卷号改变了,需要新加单 AddProfile(); } else { //继续上一次生产!!!!更新订单信息 UpdateProfile(); } } else { //异常,居然没有订单信息 AddProfile(); } } } void AddProfile() { mHistoryDb.AddProfile(new Db_Profile() { PName = mProfileParam.PName, OrderNo = mProfileParam.OrderNo, Number = mProfileParam.Number, Target = mProfileParam.Target, TolerancePercent = mProfileParam.TolerancePercent, StartTime = DateTime.Now, EndTime = DateTime.Now }); } void UpdateProfile() { mHistoryDb.UpdateProfile(new Db_Profile() { PName = mProfileParam.PName, OrderNo = mProfileParam.OrderNo, Number = mProfileParam.Number, Target = mProfileParam.Target, TolerancePercent = mProfileParam.TolerancePercent, }); } private void MProfileParam_PropertyChanged(object sender, System.ComponentModel.PropertyChangedEventArgs e) { if (profile_propertynames_add.Contains(e.PropertyName)) { FObjBase.PollModule.Current.Poll_JustOnce( () => { if (!mHistoryDb.localDb.IsProfileFinished) { mHistoryDb.FinishProfile(); } AddProfile();//新 }, this, MARKNO_PROFILE_ADD); } else if (profile_propertynames_update.Contains(e.PropertyName)) { FObjBase.PollModule.Current.Poll_JustOnce( () => { UpdateProfile();//修改 }, this, MARKNO_PROFILE_CHANGED); } } #endregion /// <summary> /// 停止扫描 /// </summary> public override void Stop() { #region sqlite3 历史数据保存 mProfileParam.PropertyChanged -= MProfileParam_PropertyChanged; #endregion base.Stop(); //已经停止 //删除全部 mScanInfoList.Clear(); } /// <summary> /// 周期动作处理 /// </summary> protected override void OnPoll() { base.OnPoll(); OnPoll_GetAngle(); } /// <summary> /// 获取 缓冲区列表 /// </summary> /// <param name="asyncDelegate"></param> /// <param name="asyncContext"></param> public void GetBufList(int no, AsyncCBHandler asyncDelegate, object asyncContext) { GetBufListReponse p = new GetBufListReponse(); p.no = no; int index = mBufList.No2Index(no); if (index >= 0 && index < mBufList.Count()) { p.data = mBufList[index]; } else { //已经被删除,不存在了 } asyncDelegate(asyncContext, p); } string file_path = "renzijiascan.json"; bool Load() { if (BlowingScanJsonDb.Load(this, file_path)) { return true; } else//只是为了版本兼容,以后就会删除 { var jsonDb = FLY.Thick.BlowingScan.Server.Update.GM_BlowingScan.ToJsonDb(); if (jsonDb == null) return false; string json = Newtonsoft.Json.JsonConvert.SerializeObject(jsonDb); File.WriteAllText(file_path, json); return BlowingScanJsonDb.Load(this, file_path); } } bool Save() { return BlowingScanJsonDb.Save(this, file_path); } } /// <summary> /// 扫描位置检测 /// 里面的列表会缓存一些旧数据,如1次扫描前的位置信息。 /// 什么时候删除,由ScanPosDetect 自行决定,使用者不用关心 /// </summary> public class ScanPosDetect { /// <summary> /// IOStatus 推送出来扫描位置与时间 /// </summary> class PosCell { public DateTime sysTime; public int position; public override string ToString() { return $"{sysTime} pos={position}"; } } class PosCellList { /// <summary> /// 这次扫描 的开始电脑时间 /// </summary> public DateTime beginTime; /// <summary> /// 这次扫描 的开始AD盒时间 /// </summary> public DateTime beginSysTime; /// <summary> /// 扫描号 /// </summary> public int ScanNo { get; private set; } public PosCellList(int scanNo, DateTime beginTime, DateTime beginSysTime) { this.ScanNo = scanNo; this.beginSysTime = beginSysTime; this.beginTime = beginTime; } private List<PosCell> mPosList = new List<PosCell>(); /// <summary> /// /// </summary> /// <param name="position">脉冲</param> /// <param name="systime">AD盒的时间,用于时间偏移量</param> /// <param name="dt">电脑的时间</param> public void Add(int position, DateTime systime) { if (mPosList.Count() == 0) { mPosList.Add(new PosCell() { position = position, sysTime = systime }); return; } if (mPosList.Count() > 0) { if (mPosList.Last().position == position) { mPosList.Last().sysTime = systime; return; } else { mPosList.Add(new PosCell() { position = position, sysTime = systime }); } } } private DateTime GetTime(int position, int idx) { var timeOffset = beginTime - beginSysTime; double d_pos = mPosList[idx].position - mPosList[idx + 1].position; TimeSpan ts = mPosList[idx].sysTime - mPosList[idx + 1].sysTime; long ticks = (long)((position - mPosList[idx + 1].position) / d_pos * ts.Ticks); return mPosList[idx + 1].sysTime + new TimeSpan(ticks) + timeOffset; } /// <summary> /// 获取位置对应的数据点. /// 返回 /// -1:position在列表的前面; /// 0:position在列表的里面 /// 1:position在列表的后面 /// </summary> /// <param name="position"></param> /// <param name="dt"></param> /// <returns></returns> public int GetTime(int position, out DateTime dt) { dt = DateTime.MinValue; //TODO if (mPosList.Count < 2) return -1; //通过列表最前面的数据,与最后面的数据,确认方向 DIRECTION direction; if (mPosList.First().position < mPosList.Last().position) direction = DIRECTION.FORWARD; else direction = DIRECTION.BACKWARD; int idx; if (direction == DIRECTION.FORWARD) { idx = mPosList.FindIndex(p => position <= p.position); } else { idx = mPosList.FindIndex(p => position >= p.position); } if (idx == -1) { //在列表后面 dt = GetTime(position, mPosList.Count - 2); return 1; } else if (idx == 0) { //在列表前面 dt = GetTime(position, 0); return -1; } else { //在列表里面 dt = GetTime(position, idx - 1); return 0; } } } //TODO private List<PosCellList> mPosLists = new List<PosCellList>(); private int lastGetTimeScanNo = -1; private int scanNo;//当前扫描号。每扫描一次,scanNo++ public int ScanNo { get { return scanNo; } set { if (scanNo != value) { //Clear(ScanNo, -2); scanNo = value; } } } /// <summary> /// 删除scanNo 偏移 offset 之前的数据 /// </summary> /// <param name="scanNo"></param> /// <param name="offset"></param> public void Clear(int scanNo, int offset) { if (mPosLists.Count == 0)//没数据 return; int index = mPosLists.FindIndex(pl => pl.ScanNo == scanNo); if (index == -1) return;//可能还没出现 index += offset; if (index < 0) return;//没有之前的数据 mPosLists.RemoveRange(0, index + 1); } public void Clear() { mPosLists.Clear(); } public ScanPosDetect() { } public void Init(IFlyAD flyad) { flyad.IStatusChangedEvent += new IStatusChangedEventHandler(flyad_IStatusChangedEvent); } void flyad_IStatusChangedEvent(object sender, IStatusChangedEventArgs e) { PosCellList poslist; if (mPosLists.Count == 0) mPosLists.Add(new PosCellList(ScanNo, DateTime.Now, e.Time)); else { poslist = mPosLists.Last(); if (poslist.ScanNo != ScanNo) { mPosLists.Add(new PosCellList(ScanNo, DateTime.Now, e.Time)); } } poslist = mPosLists.Last(); poslist.Add(e.Position, e.Time); } /// <summary> /// 获取pos 对应的时间; /// 返回 -1 数据在 列表前; /// 返回 0 数据在列表内; /// 返回 1 数据在列表外; /// </summary> /// <param name="position">位置</param> /// <param name="scanNo">扫描号</param> /// <returns></returns> public int GetTime(int position, int scanNo, out DateTime dt) { if (lastGetTimeScanNo != scanNo) { //TODO Clear(lastGetTimeScanNo, -8); lastGetTimeScanNo = scanNo; } dt = DateTime.MinValue; if (mPosLists.Count == 0)//没数据 return 1; int d = scanNo - mPosLists[0].ScanNo; if (d < 0) { //发生在过去 //不正常 return -2; } int index = mPosLists.FindIndex(pl => pl.ScanNo == scanNo); if (index == -1) return 1;//可能还没出现 PosCellList poslist = mPosLists[index]; int ret = poslist.GetTime(position, out dt); if ((ret == 1) && (index != (mPosLists.Count - 1))) return 0; else if ((ret == -1) && (index != 0)) return 0; else return ret; } } public class BlowingScanJsonDb { static Mapper Mapper { get; } = new AutoMapper.Mapper(new MapperConfiguration(c => { c.CreateMap<GM_BlowingScan, BlowingScanJsonDb>().ReverseMap(); })); public static bool Load(GM_BlowingScan src, string file_path) { if (File.Exists(file_path)) { try { string json = File.ReadAllText(file_path); var jsonDb = JsonConvert.DeserializeObject<BlowingScanJsonDb>(json); Mapper.Map(jsonDb, src); return true; } catch { //异常,没有json 解码失败 } } return false; } public static bool Save(GM_BlowingScan src, string file_path) { try { var jsonDb = Mapper.Map<BlowingScanJsonDb>(src); string json = JsonConvert.SerializeObject(jsonDb, Formatting.Indented); File.WriteAllText(file_path, json); return true; } catch { //异常,没有json 编码失败 } return false; } /// <summary> /// 加热通道数 /// </summary> public int ChannelCnt { get; set; } = 88; /// <summary> /// 分区数/加热通道数 /// </summary> public int BPC { get; set; } = 1; /// <summary> /// 复位时,探头测量的位置对应分区号 /// </summary> public int OrgBoltNo { get; set; } = 1; /// <summary> /// 使用分区表 /// </summary> public bool IsUsedMap { get; set; } = false; /// <summary> /// 探头宽度,用于解方程,可以与实际不符合 单位脉冲 /// </summary> public int SensorWidth { get; set; } = 300; /// <summary> /// 多少次扫描解一次方程 /// </summary> public int SolveCnt { get; set; } = 4; /// <summary> /// 平滑 /// </summary> public int Smooth { get; set; } = 1; /// <summary> /// 分区表,必须保证不能为null /// </summary> public BoltMapCell[] Map { get; set; } /// <summary> /// 测厚仪测量点位置方向:Left, Right (也就是三角形在左还是右) /// </summary> public bool IsProbeRight { get; set; } = true; public bool IsForwCW { get; set; } } } namespace FLY.Thick.BlowingScan.Server.Update { public class GM_BlowingScan : ISaveToXml { public static Mapper Mapper { get; } = new AutoMapper.Mapper(new MapperConfiguration(c => { c.CreateMap<GM_BlowingScan, BlowingScanJsonDb>() .ForMember(bs => bs.Map, opt => { opt.MapFrom(s => s.Map.ToArray()); }); })); /// <summary> /// 加热通道数 /// </summary> public int ChannelCnt { get; set; } = 88; /// <summary> /// 分区数/加热通道数 /// </summary> public int BPC { get; set; } = 1; /// <summary> /// 复位时,探头测量的位置对应分区号 /// </summary> public int OrgBoltNo { get; set; } = 1; /// <summary> /// 使用分区表 /// </summary> public bool IsUsedMap { get; set; } = false; /// <summary> /// 探头宽度,用于解方程,可以与实际不符合 单位脉冲 /// </summary> public int SensorWidth { get; set; } = 300; /// <summary> /// 多少次扫描解一次方程 /// </summary> public int SolveCnt { get; set; } = 4; /// <summary> /// 分区表,必须保证不能为null /// </summary> public List<BoltMapCell> Map { get; } = new List<BoltMapCell>(); static GM_BlowingScan() { Misc.SaveToXmlHepler.Regist(typeof(BoltMapCell)); } #region ISaveToXml public string[] GetSavePropertyNames() { return new string[]{ nameof(ChannelCnt), nameof(BPC), nameof(OrgBoltNo), nameof(SolveCnt), nameof(SensorWidth), nameof(IsUsedMap), nameof(Map) }; } #endregion public static BlowingScanJsonDb ToJsonDb() { if (File.Exists("renzijiascan.xml")) {//只是为了版本兼容,以后就会删除 var blowingScan = new GM_BlowingScan(); Misc.SaveToXmlHepler.Load("renzijiascan.xml", blowingScan); return Mapper.Map<BlowingScanJsonDb>(blowingScan); } else return null; } } }