123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518 |
- import 'dart:ui';
- import 'dart:math' as math;
- import 'package:fis_measure/interfaces/date_types/point.dart';
- import 'package:fis_measure/interfaces/enums/items.dart';
- import 'package:fis_measure/interfaces/process/items/item.dart';
- import 'package:fis_measure/interfaces/process/items/item_metas.dart';
- import 'package:fis_measure/interfaces/process/workspace/point_info.dart';
- import 'package:fis_measure/process/calcuators/calculator.dart';
- import 'package:fis_measure/process/calcuators/curve.dart';
- import 'package:fis_measure/process/calcuators/formulas/general.dart';
- import 'package:fis_measure/process/workspace/urm/application.dart';
- import 'package:fis_measure/utils/canvas.dart';
- import 'package:flutter/rendering.dart';
- import 'package:path_drawing/path_drawing.dart';
- import 'package:vid/us/vid_us_unit.dart';
- import 'area_abstract.dart';
- class Ellipse extends AreaItemAbstract {
- Ellipse(ItemMeta meta, IMeasureItem? parent) : super(meta, parent);
- @override
- bool onExecuteMouse(PointInfo args) {
- if (state == ItemStates.finished) {
- if (args.pointType == PointInfoType.mouseDown) {
- state = ItemStates.waiting;
- }
- }
- if (state == ItemStates.waiting) {
- if (args.pointType == PointInfoType.mouseDown) {
- handleMouseDownWhileWaiting(args);
- }
- } else if (state == ItemStates.running) {
- if (feature == null) return false;
- final f = feature! as EllipseFeature;
- final activeIndex = f.activeIndex;
- switch (args.pointType) {
- case PointInfoType.mouseMove:
- f.innerPoints[activeIndex] = args;
- f.adjustPoints(args);
- break;
- case PointInfoType.mouseDown:
- if (activeIndex == 1) {
- if (f.xAxisEnd == f.xAxisStart) {
- break;
- }
- f.adjustPoints(args);
- f.activeIndex = 2;
- } else if (activeIndex == 2) {
- doFeatureFinish();
- }
- break;
- default:
- return false;
- }
- doCalculate();
- return true;
- }
- return false;
- }
- bool isFirstPointMove = true;
- DPoint touchStartPosition = DPoint(0, 0); // 相对位移起始触摸点
- DPoint currPointLastPosition = DPoint(0, 0); // 当前操作的点之前所在位置
- @override
- bool onExecuteTouch(PointInfo args) {
- if (state == ItemStates.finished) {
- if (args.pointType == PointInfoType.touchDown) {
- state = ItemStates.waiting;
- }
- }
- if (state == ItemStates.waiting) {
- if (args.pointType == PointInfoType.touchDown) {
- isFirstPointMove = true;
- handleTouchDownWhileWaiting(args);
- }
- } else if (state == ItemStates.running) {
- if (feature == null) return false;
- final f = feature! as EllipseFeature;
- if (args.pointType == PointInfoType.touchDown) {
- touchStartPosition = args;
- currPointLastPosition = f.innerPoints[f.activeIndex];
- }
- if (args.pointType == PointInfoType.touchMove) {
- if (isFirstPointMove) {
- PointInfo newStartPoint = args;
- newStartPoint.addOffset(0, -0.2);
- if (isMoveTargetOutOfRange(newStartPoint)) return true;
- for (var element in f.innerPoints) {
- element.update(newStartPoint);
- }
- } else {
- PointInfo newPoint = PointInfo.fromOffset(
- currPointLastPosition
- .clone()
- .addVector(args - touchStartPosition)
- .toOffset(),
- args.pointType);
- newPoint.hostVisualArea = args.hostVisualArea;
- if (isMoveTargetOutOfRange(newPoint)) return true;
- f.innerPoints[f.activeIndex] = newPoint;
- f.adjustPoints(newPoint);
- }
- }
- if (args.pointType == PointInfoType.touchUp) {
- if (isFirstPointMove) {
- isFirstPointMove = false;
- f.activeIndex = 1;
- }
- if (f.activeIndex == 1) {
- if (f.xAxisEnd != f.xAxisStart) {
- f.adjustPoints(args);
- f.activeIndex = 2;
- }
- } else if (f.activeIndex == 2) {
- doFeatureFinish();
- }
- }
- doCalculate();
- return true;
- }
- return false;
- }
- void handleMouseDownWhileWaiting(PointInfo args) {
- // TODO: 判断是否当前area
- // 转换为Area逻辑位置
- final point = args.toAreaLogicPoint();
- feature = EllipseFeature(this, point);
- if (args.hostVisualArea != null) {
- feature!.hostVisualArea = args.hostVisualArea;
- }
- feature!.activeIndex = 1;
- state = ItemStates.running;
- }
- void handleTouchDownWhileWaiting(PointInfo args) {
- // TODO: 判断是否当前area
- // 转换为Area逻辑位置
- final point = args.toAreaLogicPoint();
- feature = EllipseFeature(this, point);
- if (args.hostVisualArea != null) {
- feature!.hostVisualArea = args.hostVisualArea;
- }
- feature!.activeIndex = 0;
- state = ItemStates.running;
- }
- static Ellipse createAreaPerimeter(ItemMeta meta, [IMeasureItem? parent]) {
- final ellipse = Ellipse(meta, parent);
- ellipse.calculator = AreaPerimeterEllipseCal(ellipse);
- return ellipse;
- }
- static Ellipse createVolume(ItemMeta meta, [IMeasureItem? parent]) {
- final ellipse = Ellipse(meta, parent);
- ellipse.calculator = _EllipseVolumeCal(ellipse);
- return ellipse;
- }
- }
- class EllipseFeature extends AreaItemFeatureAbstract {
- EllipseFeature(Ellipse refItem, DPoint point) : super(refItem) {
- innerPoints.add(point.clone());
- innerPoints.add(point.clone());
- innerPoints.add(point.clone());
- innerPoints.add(point.clone());
- }
- /// 质心
- DPoint get centroid {
- final x = (xAxisStart.x + xAxisEnd.x) * 0.5;
- final y = (xAxisStart.y + xAxisEnd.y) * 0.5;
- return DPoint(x, y);
- }
- DPoint get xAxisStart => innerPoints[0];
- DPoint get xAxisEnd => innerPoints[1];
- DPoint get yAxisStart => innerPoints[2];
- DPoint get yAxisEnd => innerPoints[3];
- @override
- void paint(Canvas canvas, Size size) {
- if (innerPoints.isEmpty) return;
- drawId(canvas, size, idText);
- final xStartOffset = convert2ViewPoint(size, xAxisStart).toOffset();
- if (activeIndex < 1) {
- drawVertex(canvas, xStartOffset, true);
- return;
- } else {
- drawVertex(canvas, xStartOffset);
- }
- final xEndOffset = convert2ViewPoint(size, xAxisEnd).toOffset();
- canvas.drawDashLine(xStartOffset, xEndOffset, 1, 10, paintLinePan);
- final yStartOffset = convert2ViewPoint(size, innerPoints[2]).toOffset();
- final yEndOffset = convert2ViewPoint(size, innerPoints[3]).toOffset();
- final centroidOffset = convert2ViewPoint(size, centroid).toOffset();
- canvas.drawDashLine(yStartOffset, yEndOffset, 1, 10, paintLinePan);
- // canvas.drawDashLine(yStartOffset, xEndOffset, 1, 10, paintLinePan);
- // canvas.drawDashLine(yEndOffset, xEndOffset, 1, 10, paintLinePan);
- final radiusX = getRadiusX(size);
- final radiusY = getRadiusY(size);
- canvas.save();
- final path = Path();
- path.moveTo(xStartOffset.dx, xStartOffset.dy);
- // 以质心为原点旋转到水平方向
- canvas.translate(centroidOffset.dx, centroidOffset.dy);
- final rotateRad = (xEndOffset - xStartOffset).direction;
- canvas.rotate(rotateRad);
- canvas.translate(-centroidOffset.dx, -centroidOffset.dy);
- path.addOval(
- Rect.fromCenter(
- center: centroidOffset,
- width: radiusX * 2,
- height: radiusY * 2,
- ),
- );
- canvas.drawPath(
- dashPath(
- path,
- dashArray: CircularIntervalList<double>(<double>[2.0, 10.0]),
- ),
- paintPan,
- );
- canvas.restore();
- if (activeIndex == 1) {
- drawVertex(canvas, xEndOffset, true);
- drawVertex(canvas, yStartOffset);
- drawVertex(canvas, yEndOffset);
- } else if (activeIndex == 2) {
- drawVertex(canvas, xEndOffset);
- drawVertex(canvas, yEndOffset);
- drawVertex(canvas, yStartOffset, isActive);
- }
- }
- /// 绘制灌注图
- @override
- void paintPerfusion(Canvas canvas, Size size) {}
- void paintSharpOnly(Canvas canvas, Size size, {Paint? customPaintPen}) {
- if (innerPoints.isEmpty) return;
- final xStartOffset = convert2ViewPoint(size, xAxisStart).toOffset();
- final xEndOffset = convert2ViewPoint(size, xAxisEnd).toOffset();
- canvas.drawDashLine(
- xStartOffset, xEndOffset, 1, 10, customPaintPen ?? paintLinePan);
- final yStartOffset = convert2ViewPoint(size, innerPoints[2]).toOffset();
- final yEndOffset = convert2ViewPoint(size, innerPoints[3]).toOffset();
- final centroidOffset = convert2ViewPoint(size, centroid).toOffset();
- canvas.drawDashLine(
- yStartOffset, yEndOffset, 1, 10, customPaintPen ?? paintLinePan);
- final radiusX = getRadiusX(size);
- final radiusY = getRadiusY(size);
- canvas.save();
- final path = Path();
- path.moveTo(xStartOffset.dx, xStartOffset.dy);
- // 以质心为原点旋转到水平方向
- canvas.translate(centroidOffset.dx, centroidOffset.dy);
- final rotateRad = (xEndOffset - xStartOffset).direction;
- canvas.rotate(rotateRad);
- canvas.translate(-centroidOffset.dx, -centroidOffset.dy);
- path.addOval(
- Rect.fromCenter(
- center: centroidOffset,
- width: radiusX * 2,
- height: radiusY * 2,
- ),
- );
- canvas.drawPath(
- dashPath(
- path,
- dashArray: CircularIntervalList<double>(<double>[2.0, 10.0]),
- ),
- customPaintPen ?? paintPan,
- );
- canvas.restore();
- }
- void paintShapeFitInSize(Canvas canvas, Size size, Paint paintPan) {
- if (innerPoints.isEmpty) return;
- final xStartOffset = convert2ViewPoint(size, xAxisStart).toOffset();
- final xEndOffset = convert2ViewPoint(size, xAxisEnd).toOffset();
- final centroidOffset = convert2ViewPoint(size, centroid).toOffset();
- final radiusX = getRadiusX(size);
- final radiusY = getRadiusY(size);
- canvas.save();
- // 创建椭圆路径
- final path = Path();
- path.addOval(
- Rect.fromCenter(
- center: centroidOffset,
- width: radiusX * 2,
- height: radiusY * 2,
- ),
- );
- // 计算旋转角度
- final rotateRad = (xEndOffset - xStartOffset).direction;
- // 创建一个矩阵来应用旋转
- final rotateMatrix = Matrix4.rotationZ(rotateRad);
- // 获取旋转后的路径
- final rotatedPath = path.transform(rotateMatrix.storage);
- // 获取外接矩形
- final rect = rotatedPath.getBounds();
- // 计算缩放比例
- final scaleW = size.width / rect.width;
- final scaleH = size.height / rect.height;
- final scale = scaleW < scaleH ? scaleW : scaleH;
- // 计算平移距离
- final dx = (size.width - rect.width * scale) / 2 - rect.left * scale;
- final dy = (size.height - rect.height * scale) / 2 - rect.top * scale;
- // 应用变换
- canvas.translate(dx, dy);
- canvas.scale(scale, scale);
- // 调整画笔宽度
- paintPan.strokeWidth = 1 / scale;
- // 绘制路径
- canvas.drawPath(rotatedPath, paintPan);
- canvas.restore();
- }
- /// X轴半径
- double getRadiusX([Size? fitSize]) {
- if (innerPoints.length < 2) {
- return 0;
- }
- var p1 = xAxisStart;
- var p2 = xAxisEnd;
- if (fitSize != null) {
- p1 = convert2ViewPoint(fitSize, p1);
- p2 = convert2ViewPoint(fitSize, p2);
- // p1 = p1.scale2Size(fitSize);
- // p2 = p2.scale2Size(fitSize);
- }
- return (p2 - p1).length / 2;
- }
- // 获取未缩放的基准半径
- double getBaseRadiusX([Size? fitSize]) {
- if (innerPoints.length < 2) {
- return 0;
- }
- var p1 = xAxisStart;
- var p2 = xAxisEnd;
- if (fitSize != null) {
- p1 = p1.scale2Size(fitSize);
- p2 = p2.scale2Size(fitSize);
- }
- return (p2 - p1).length / 2;
- }
- /// Y轴半径
- double getRadiusY([Size? fitSize]) {
- if (activeIndex > 1) {
- var p = yAxisStart;
- var p2 = yAxisEnd;
- var c = centroid;
- if (fitSize != null) {
- p = convert2ViewPoint(fitSize, p);
- p2 = convert2ViewPoint(fitSize, p2);
- c = convert2ViewPoint(fitSize, c);
- // p = p.scale2Size(fitSize);
- // p2 = p2.scale2Size(fitSize);
- // c = c.scale2Size(fitSize);
- }
- return (p - p2).length / 2;
- } else {
- return getRadiusX(fitSize);
- }
- }
- /// 圆周长
- double getCircumference([Size? fitSize]) {
- double a = getRadiusX(fitSize), b = getRadiusY(fitSize);
- if (a < b) {
- a = b;
- b = a;
- }
- return 2 * math.pi * b + 4 * (a - b);
- }
- /// 面积
- double getArea([Size? fitSize]) {
- return math.pi * getRadiusX(fitSize) * getRadiusY(fitSize);
- }
- /// 计算Y轴坐标点
- void adjustPoints(DPoint point) {
- if (activeIndex < 1) return;
- Size refSize = refItem.application.displaySize;
- // 判断是否为 URMApp 如果是左右双幅,则需要宽度减半,此处其实是需要一个当前画幅长宽比来确定垂直位置
- if (refItem.application is URMApplication &&
- (refItem.application as URMApplication).urmDataProcessor.isLeftRight) {
- refSize = Size(refSize.width / 2, refSize.height);
- }
- final restoreSize = Size(1 / refSize.width, 1 / refSize.height);
- final p = point.scale2Size(refSize);
- final p1 = xAxisStart.scale2Size(refSize);
- final p2 = xAxisEnd.scale2Size(refSize);
- final double radius;
- if (activeIndex == 2) {
- radius = GeneralFormulas.distance2Line(p1, p2, p);
- } else {
- radius = getBaseRadiusX(refSize);
- }
- final crossPoints = findCrossPoints(p2, p1, radius);
- final p3Index = findNearest(p, crossPoints);
- innerPoints[2] = crossPoints[p3Index].scale2Size(restoreSize);
- innerPoints[3] = crossPoints[1 - p3Index].scale2Size(restoreSize);
- }
- List<DPoint> findCrossPoints(DPoint a, DPoint b, distance) {
- final dx = b.x - a.x;
- final dy = b.y - a.y;
- final len = (b - a).length;
- /// 单位长度轴上距离增加量
- final udx = dx / len;
- final udy = dy / len;
- final px = -udy;
- final py = udx;
- final centroidX = (a.x + b.x) * 0.5;
- final centroidY = (a.y + b.y) * 0.5;
- final centroid = DPoint(centroidX, centroidY);
- final x1 = centroid.x + px * distance;
- final y1 = centroid.y + py * distance;
- final out1 = DPoint(x1, y1);
- final x2 = centroid.x - px * distance;
- final y2 = centroid.y - py * distance;
- final out2 = DPoint(x2, y2);
- return [out1, out2];
- }
- int findNearest(DPoint p, List<DPoint> source) {
- int rst = 0;
- double minLen = (source[0] - p).length;
- for (var i = 1; i < source.length; i++) {
- final len = (source[i] - p).length;
- if (len < minLen) {
- minLen = len;
- rst = i;
- }
- }
- return rst;
- }
- }
- class _EllipseVolumeCal extends Calculator<Ellipse, double> {
- _EllipseVolumeCal(Ellipse ref) : super(ref);
- @override
- void calculate() {
- if (ref.feature == null) return;
- final feature = ref.feature! as EllipseFeature;
- final viewport = feature.hostVisualArea!.viewport!;
- final size = viewport.convertBoundary;
- double xAxis = feature.getRadiusX(size);
- double yAxis = feature.getRadiusY(size);
- var a = math.max(xAxis, yAxis);
- var b = math.min(xAxis, yAxis);
- var value = math.pi / 6.0 * a * b * b;
- updateFloatValue(value, unit: VidUsUnit.cm3, useRound: true);
- }
- }
- class EllipseFeatureForRoiCopy extends EllipseFeature {
- EllipseFeatureForRoiCopy(Ellipse refItem, DPoint point)
- : super(refItem, point);
- @override
- bool needRecordHistory = false; // 不记录历史
- }
|