DSCAnalysisTool/src/data/pointcalculate.cpp

462 lines
14 KiB
C++
Raw Normal View History

2025-03-26 09:30:02 +00:00
#include <QDebug>
2025-04-03 09:24:29 +00:00
#include "pointcalculate.h"
2025-03-31 09:24:48 +00:00
#include "logger.h"
2025-03-26 09:30:02 +00:00
2025-04-03 09:24:29 +00:00
QVector<FileManager::ExperimentData>PointCalculate:: _dataVtr;
QPointF PointCalculate::_peakPoint;
QPointF PointCalculate::_leftSelectedPoint,PointCalculate::_rightSelectedPoint;
2025-03-26 09:30:02 +00:00
2025-04-03 09:24:29 +00:00
void PointCalculate::setExperimentData(const QVector<FileManager::ExperimentData> &dataVtr)
2025-03-26 09:30:02 +00:00
{
_dataVtr = dataVtr;
2025-04-01 09:25:12 +00:00
2025-04-03 09:24:29 +00:00
if(_dataVtr.empty()) {
return;
}
2025-04-01 09:25:12 +00:00
FileManager::ExperimentData startPoint = _dataVtr.at(0);
FileManager::ExperimentData endPoint = _dataVtr.at(_dataVtr.size() - 1);
_leftSelectedPoint = QPointF(startPoint.sampleTemp,startPoint.dsc);
_rightSelectedPoint = QPointF(endPoint.sampleTemp,endPoint.dsc);
2025-03-26 09:30:02 +00:00
}
2025-04-03 09:24:29 +00:00
std::vector<float> PointCalculate::movingAverage(const std::vector<float>& data, int windowSize) {
std::vector<float> smoothedData;
for (size_t i = 0; i < data.size(); ++i) {
if (i + windowSize <= data.size()) {
float sum = std::accumulate(data.begin() + i, data.begin() + i + windowSize, 0.0);
smoothedData.push_back(sum / windowSize);
}
}
return smoothedData;
}
QPointF PointCalculate::getPeakPoint(){
2025-03-26 09:30:02 +00:00
int n = _dataVtr.size();
if (n < 3) {
return QPointF(); // 至少需要三个点才能找到波峰
}
QPointF uniquePeak;
2025-04-01 09:25:12 +00:00
float maxDiff = -std::numeric_limits<float>::infinity();
2025-03-26 09:30:02 +00:00
2025-04-01 09:25:12 +00:00
for (int i = 0; i < n; ++i) {
2025-03-27 09:31:19 +00:00
const float currentX = _dataVtr.at(i).sampleTemp;
const float currentY = _dataVtr.at(i).dsc;
2025-03-26 09:30:02 +00:00
2025-03-31 09:24:48 +00:00
if (currentX < _leftSelectedPoint.x()) {
2025-03-26 09:30:02 +00:00
continue;
}
2025-03-31 09:24:48 +00:00
if (currentX > _rightSelectedPoint.x()) {
2025-03-26 09:30:02 +00:00
break;
}
2025-04-01 09:25:12 +00:00
// 计算当前点与左选择点 y 值的差值
float diffLeft = std::abs(currentY - _leftSelectedPoint.y());
// 计算当前点与右选择点 y 值的差值
float diffRight = std::abs(currentY - _rightSelectedPoint.y());
// 取两个差值中的较大值
float currentDiff = std::max(diffLeft, diffRight);
2025-03-31 09:24:48 +00:00
2025-04-01 09:25:12 +00:00
if (currentDiff > maxDiff) {
maxDiff = currentDiff;
uniquePeak = QPointF(currentX, currentY);
2025-03-26 09:30:02 +00:00
}
}
_peakPoint = uniquePeak;
2025-03-31 09:24:48 +00:00
logde<<"peakPoint:"<<_peakPoint.x()<<","<<_peakPoint.y();
2025-03-26 09:30:02 +00:00
return uniquePeak;
}
2025-04-03 09:24:29 +00:00
QPair<QPointF, QPointF> PointCalculate::calculateMaxDiffPointDetail(
const PointCalculate::MaxDiffPointDetailType type)
2025-03-26 09:30:02 +00:00
{
#if 1
2025-03-27 09:31:19 +00:00
float maxDiff = std::numeric_limits<float>::min();
2025-03-26 09:30:02 +00:00
QPointF currentPoint,lastPoint;
for (int i = 0; i < _dataVtr.size() - 1; ++i) {
2025-03-27 09:31:19 +00:00
const float currentX = _dataVtr.at(i).sampleTemp;
const float currentY = _dataVtr.at(i).dsc;
2025-03-26 09:30:02 +00:00
if(type == MaxDiffPointDetailType::Left){
2025-03-31 09:24:48 +00:00
if(currentX <= _leftSelectedPoint.x()){
2025-03-26 09:30:02 +00:00
continue;
}
if(currentX >= _peakPoint.x()){
break;
}
}else{
if(currentX <= _peakPoint.x()){
continue;
}
2025-03-31 09:24:48 +00:00
if(currentX >= _rightSelectedPoint.x()){
2025-03-26 09:30:02 +00:00
break;
}
}
//
2025-03-27 09:31:19 +00:00
const float lastX = _dataVtr.at(i + 1).sampleTemp;
const float lastY = _dataVtr.at(i + 1).dsc;
float diff = std::abs(currentY - lastY);
2025-03-26 09:30:02 +00:00
if(diff > maxDiff){
maxDiff = diff;
currentPoint.setX(currentX);
currentPoint.setY(currentY);
lastPoint.setX(lastX);
lastPoint.setY(lastY);
}
}
#endif
return qMakePair(currentPoint,lastPoint);
}
2025-04-03 09:24:29 +00:00
QPair<QPointF, QPointF> PointCalculate::calculateMaxDiffPointLeft()
2025-03-26 09:30:02 +00:00
{
return calculateMaxDiffPointDetail(MaxDiffPointDetailType::Left);
}
2025-04-03 09:24:29 +00:00
QPair<QPointF, QPointF> PointCalculate::calculateMaxDiffPointRight()
2025-03-26 09:30:02 +00:00
{
return calculateMaxDiffPointDetail(MaxDiffPointDetailType::Right);
}
2025-04-01 09:25:12 +00:00
#if 0
2025-04-03 09:24:29 +00:00
QPointF PointCalculate::findClosestY(float targetX)
2025-03-26 09:30:02 +00:00
{
2025-03-27 09:31:19 +00:00
float minDiff = std::numeric_limits<float>::max();
2025-04-01 09:25:12 +00:00
QPointF resultPointF;
2025-03-26 09:30:02 +00:00
for(FileManager::ExperimentData &ed:_dataVtr){
2025-03-27 09:31:19 +00:00
float diff = std::abs(ed.sampleTemp - targetX);
2025-03-26 09:30:02 +00:00
if (diff < minDiff) {
minDiff = diff;
2025-04-01 09:25:12 +00:00
resultPointF = QPointF(ed.sampleTemp,ed.dsc);
2025-03-26 09:30:02 +00:00
}
}
2025-04-01 09:25:12 +00:00
return resultPointF;
2025-03-26 09:30:02 +00:00
}
2025-04-01 09:25:12 +00:00
#endif
2025-03-26 09:30:02 +00:00
2025-04-01 09:25:12 +00:00
#if 0
2025-04-03 09:24:29 +00:00
QPointF PointCalculate::findClosestPointByX(float x) {
2025-03-26 09:30:02 +00:00
int left = 0;
int right = _dataVtr.size() - 1;
QPointF targetPoint;
targetPoint.setX(_dataVtr.value(0).sampleTemp);
targetPoint.setY(_dataVtr.value(0).dsc);
while (left <= right) {
int mid = left + (right - left) / 2;
FileManager::ExperimentData& ed = _dataVtr[mid];
if (std::abs(ed.sampleTemp - x) < std::abs(targetPoint.x() - x)) {
targetPoint.setX(ed.sampleTemp);
targetPoint.setY(ed.dsc);
}
if(ed.sampleTemp < x){
left = mid + 1;
} else {
right = mid - 1;
}
}
return targetPoint;
}
2025-04-01 09:25:12 +00:00
#endif
2025-03-26 09:30:02 +00:00
2025-04-03 09:24:29 +00:00
void PointCalculate::setRegionPointX(const float left, const float right)
2025-03-26 09:30:02 +00:00
{
2025-04-01 09:25:12 +00:00
logde<<"select point param left,right:"<<left<<","<<right;
2025-04-03 09:24:29 +00:00
_leftSelectedPoint = getClosestPointByX(left);
_rightSelectedPoint = getClosestPointByX(right);
2025-03-26 09:30:02 +00:00
2025-04-03 09:24:29 +00:00
_peakPoint = getPeakPoint();
2025-04-08 09:30:33 +00:00
2025-04-01 09:25:12 +00:00
logde<<"peak point:"<<_peakPoint.x()<<","<<_peakPoint.y();
2025-03-27 09:31:19 +00:00
2025-04-01 09:25:12 +00:00
//根据峰谷重置选择点。
2025-03-27 09:31:19 +00:00
updateStartEndPoint();
2025-04-01 09:25:12 +00:00
logde<<"select point left:"<<_leftSelectedPoint.x()<<","<<_leftSelectedPoint.y();
logde<<"select point right:"<<_rightSelectedPoint.x()<<","<<_rightSelectedPoint.y();
2025-03-26 09:30:02 +00:00
}
2025-04-03 09:24:29 +00:00
QVector<QPointF> PointCalculate::getPeakPointGroup()
2025-03-27 09:31:19 +00:00
{
QVector<QPointF> pointVtr;
for(FileManager::ExperimentData& ed:_dataVtr) {
2025-04-08 09:30:33 +00:00
if(ed.sampleTemp >= _leftSelectedPoint.x() &&
ed.sampleTemp <= _rightSelectedPoint.x()){
2025-03-27 09:31:19 +00:00
pointVtr.push_back(QPointF(ed.sampleTemp,ed.dsc));
}
}
return pointVtr;
}
2025-04-08 09:30:33 +00:00
float PointCalculate::calculateArea() {
2025-03-31 05:19:02 +00:00
//getPoint group
QVector<QPointF> points = getPeakPointGroup();
//calculate Area
float integral = 0.0;
size_t n = points.size();
if (n < 2) {
return integral; // 至少需要两个点才能计算积分
}
//find a line.
2025-04-08 09:30:33 +00:00
float k = (_leftSelectedPoint.y() - _rightSelectedPoint.y()) /
2025-03-31 05:19:02 +00:00
(_leftSelectedPoint.x() - _rightSelectedPoint.x());
2025-04-08 09:30:33 +00:00
float b = _leftSelectedPoint.y() - k * _leftSelectedPoint.x();
2025-03-31 05:19:02 +00:00
for (size_t i = 0; i < n - 1; ++i) {
#if 1
2025-04-08 09:30:33 +00:00
float x1 = points[i].x();
float y1 = points[i].y();
2025-03-31 05:19:02 +00:00
2025-04-08 09:30:33 +00:00
float x2 = points[i + 1].x();
float y2 = points[i + 1].y();
2025-03-31 05:19:02 +00:00
#endif
2025-04-08 09:30:33 +00:00
float yLine1 = k * x1 + b;
float yLine2 = k * x2 + b;
2025-03-31 05:19:02 +00:00
2025-04-08 09:30:33 +00:00
float diff1 = y1 - yLine1;
float diff2 = y2 - yLine2;
2025-03-31 05:19:02 +00:00
2025-04-08 09:30:33 +00:00
float dx = x2 - x1;
2025-03-31 05:19:02 +00:00
2025-04-08 09:30:33 +00:00
float cellArea = (diff1 + diff2) * dx /2.0;
2025-03-31 05:19:02 +00:00
integral += std::abs(cellArea);
}
return integral;
}
2025-04-03 09:24:29 +00:00
QPair<QPointF, QPointF> PointCalculate::calculateStartAndEndPoint()
2025-03-27 09:31:19 +00:00
{
2025-04-03 09:24:29 +00:00
QPair<QPointF,QPointF> leftMaxDiffPointPair = PointCalculate::calculateMaxDiffPointLeft();
QPair<QPointF,QPointF> rightMaxDiffPointPair = PointCalculate::calculateMaxDiffPointRight();
#if 0
2025-03-31 09:24:48 +00:00
logde<<"b1:"<<leftMaxDiffPointPair.first.x()<<","<<leftMaxDiffPointPair.first.y();
logde<<"b2:"<<leftMaxDiffPointPair.second.x()<<","<<leftMaxDiffPointPair.second.y();
2025-04-01 09:25:12 +00:00
logde<<"b3:"<<rightMaxDiffPointPair.first.x()<<","<<rightMaxDiffPointPair.first.y();
logde<<"b4:"<<rightMaxDiffPointPair.second.x()<<","<<rightMaxDiffPointPair.second.y();
2025-04-03 09:24:29 +00:00
#endif
2025-03-27 09:31:19 +00:00
QPointF startPoint = calculateIntersection(_leftSelectedPoint,_rightSelectedPoint,
2025-04-01 09:25:12 +00:00
leftMaxDiffPointPair.first,
leftMaxDiffPointPair.second);
2025-03-27 09:31:19 +00:00
QPointF endPoint = calculateIntersection(_leftSelectedPoint,_rightSelectedPoint,
2025-04-01 09:25:12 +00:00
rightMaxDiffPointPair.first,
rightMaxDiffPointPair.second);
2025-03-27 09:31:19 +00:00
return qMakePair(startPoint,endPoint);
}
2025-04-03 09:24:29 +00:00
void PointCalculate::updateStartEndPoint()
2025-03-27 09:31:19 +00:00
{
//需要在a1和a2之间查询是否有高于a1和a2之间的点若存在则重新给a1、a2赋值
for(FileManager::ExperimentData& ed:_dataVtr){
if(ed.sampleTemp > _leftSelectedPoint.x() && ed.sampleTemp < _peakPoint.x()){
if(ed.dsc > _leftSelectedPoint.y()){
_leftSelectedPoint.setX(ed.sampleTemp);
_leftSelectedPoint.setY(ed.dsc);
}
}
if(ed.sampleTemp < _rightSelectedPoint.x() && ed.sampleTemp > _peakPoint.x()){
if(ed.dsc > _rightSelectedPoint.y()){
_rightSelectedPoint.setX(ed.sampleTemp);
_rightSelectedPoint.setY(ed.dsc);
}
}
}
}
2025-04-03 09:24:29 +00:00
QString PointCalculate::textFormatPeakPoint(const float enthalpyValue,
2025-04-08 09:30:33 +00:00
const float peakValue,
const float startPoint,
const float endPoint)
2025-03-26 09:30:02 +00:00
{
return QString("峰的综合信息:\n"
"焓值:%1 J/g \n"
"峰值:%2℃ \n"
"起始点:%3℃ \n"
"终止点:%4℃"
2025-03-31 09:24:48 +00:00
).arg(QString::number(enthalpyValue, 'f', 3))
.arg(QString::number(peakValue, 'f', 3))
.arg(QString::number(startPoint, 'f', 3))
.arg(QString::number(endPoint, 'f', 3));
2025-03-26 09:30:02 +00:00
}
// 计算两条直线的交点
2025-04-03 09:24:29 +00:00
QPointF PointCalculate::calculateIntersection(const QPointF p1,const QPointF p2,
2025-04-08 09:30:33 +00:00
const QPointF p3, const QPointF p4){
2025-03-26 09:30:02 +00:00
// 直线的一般式: A1x + B1y + C1 = 0 和 A2x + B2y + C2 = 0
2025-03-27 09:31:19 +00:00
float A1 = p2.y() - p1.y();
float B1 = p1.x() - p2.x();
float C1 = A1 * p1.x() + B1 * p1.y();
2025-03-26 09:30:02 +00:00
2025-03-27 09:31:19 +00:00
float A2 = p4.y() - p3.y();
float B2 = p3.x() - p4.x();
float C2 = A2 * p3.x() + B2 * p3.y();
2025-03-26 09:30:02 +00:00
2025-03-27 09:31:19 +00:00
float determinant = A1 * B2 - A2 * B1;
2025-03-26 09:30:02 +00:00
if (determinant == 0) {
// 两条直线平行或重合,无交点或无限交点
return {0, 0};
} else {
2025-03-27 09:31:19 +00:00
float x = (B2 * C1 - B1 * C2) / determinant;
float y = (A1 * C2 - A2 * C1) / determinant;
2025-03-26 09:30:02 +00:00
return {x, y};
}
}
2025-04-08 09:30:33 +00:00
QPair<float,float> PointCalculate::getCurveInflectionPointTangent(const float x1,const float x2)
{
std::vector<float> dataVtr;
for(FileManager::ExperimentData& ed:_dataVtr){
if(x1 < ed.sampleTemp && ed.sampleTemp < x2){
dataVtr.push_back(ed.sampleTemp);
}
}
//
std::vector<float> processedVtr = movingAverage(dataVtr,5);
// find max slope.
float maxSlope = 0.0;
float targetX = 0.0;
for (size_t i = 1; i < processedVtr.size(); ++i) {
float slope = processedVtr[i] - processedVtr[i - 1];
if(std::abs(slope) > maxSlope){
maxSlope = slope;
targetX = processedVtr[i];
}
}
// calculate line slope and axis Y space number b.
QPointF point = getClosestPointByX(targetX);
// 使用点斜式方程求y轴截距
// y = m * x + b
// double b = y - m * x;
float b = point.y() - maxSlope * point.x();
return qMakePair<float,float>(maxSlope,b);
}
QPointF PointCalculate::getIntersectionBySlope(const LineStruct& line1, const LineStruct& line2) {
float x = (line2.intercept - line1.intercept) / (line1.slope - line2.slope);
float y = line1.slope * x + line1.intercept;
// return {x, y};
return QPointF(x,y);
}
2025-04-03 09:24:29 +00:00
QPointF PointCalculate::getClosestPointByX(const float targetX)
2025-04-01 09:25:12 +00:00
{
2025-04-03 09:24:29 +00:00
#if 0
2025-04-01 09:25:12 +00:00
float minDiff = std::numeric_limits<float>::max();
2025-03-27 09:31:19 +00:00
2025-04-01 09:25:12 +00:00
for(FileManager::ExperimentData &ed:_dataVtr){
float diff = std::abs(ed.sampleTemp - targetX);
if (diff < minDiff) {
minDiff = diff;
resultPointF = QPointF(ed.sampleTemp,ed.dsc);
}
}
2025-04-03 09:24:29 +00:00
#endif
int left = 0;
int right = _dataVtr.size() - 1;
while (left < right) {
int mid = left + (right - left)/2;
const FileManager::ExperimentData& ed = _dataVtr.at(mid);
if(ed.sampleTemp > targetX){
right = mid;
}else{
left = mid + 1;
}
}
//
QPointF resultPointF(_dataVtr.at(left).sampleTemp,
_dataVtr.at(left).dsc);
if(left > 0 &&
std::abs(resultPointF.x() - targetX) >
std::abs(_dataVtr.at(left - 1).sampleTemp - targetX)){
2025-04-08 09:30:33 +00:00
//
2025-04-03 09:24:29 +00:00
resultPointF = QPointF(_dataVtr.at(left - 1).sampleTemp,
_dataVtr.at(left - 1).dsc);
}
2025-03-27 09:31:19 +00:00
2025-04-01 09:25:12 +00:00
return resultPointF;
}
2025-03-27 09:31:19 +00:00
2025-04-03 09:24:29 +00:00
QString PointCalculate::textFormatNumbericalLabel(const QPointF point)
2025-04-01 09:25:12 +00:00
{
return QString("数值:\n"
"%1℃,%2"
).arg(QString::number(point.x(), 'f', 3))
.arg(QString::number(point.y(), 'f', 3));
}
2025-03-27 09:31:19 +00:00
2025-04-03 09:24:29 +00:00
QPair<QPointF, QPointF> PointCalculate::getStartAndEndPoint()
2025-04-01 09:25:12 +00:00
{
2025-04-03 09:24:29 +00:00
if(_dataVtr.empty()){
2025-04-08 09:30:33 +00:00
return qMakePair(QPointF(), QPointF());
2025-04-03 09:24:29 +00:00
}
2025-04-08 09:30:33 +00:00
FileManager::ExperimentData startPoint = _dataVtr.at(0);
FileManager::ExperimentData endPoint = _dataVtr.at(_dataVtr.size() - 1);
2025-03-27 09:31:19 +00:00
2025-04-01 09:25:12 +00:00
return qMakePair<QPointF,QPointF>(
QPointF(startPoint.sampleTemp,startPoint.dsc),
QPointF(endPoint.sampleTemp,endPoint.dsc));
}
2025-04-03 09:24:29 +00:00
QString PointCalculate::textFormatStartPoint(const QPointF point)
{
return QString("外推起始点:\n"
"%1℃"
).arg(QString::number(point.x(), 'f', 3));
}
QString PointCalculate::textFormatEndPoint(const QPointF point)
{
return QString("外推终止点:\n"
"%1℃"
).arg(QString::number(point.x(), 'f', 3));
}
2025-04-08 09:30:33 +00:00
QPointF PointCalculate::getClosestPointByY(const float left,const float right,const float valueY)
2025-04-03 09:24:29 +00:00
{
2025-04-08 09:30:33 +00:00
float minValue = 100.0;
for(FileManager::ExperimentData& ed:_dataVtr){
if(left < ed.sampleTemp && ed.sampleTemp < right){
if(std::abs(ed.dsc - valueY) < minValue){
return QPointF(ed.sampleTemp,ed.dsc);
}
}
}
return QPointF();
2025-04-03 09:24:29 +00:00
}