pose_painter.dart 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112
  1. import 'package:body_detection/models/pose.dart';
  2. import 'package:body_detection/models/pose_landmark.dart';
  3. import 'package:body_detection/models/pose_landmark_type.dart';
  4. import 'package:flutter/widgets.dart';
  5. class PosePainter extends CustomPainter {
  6. final Pose? pose;
  7. final Size imageSize;
  8. PosePainter({required this.pose, required this.imageSize, Key? key});
  9. final pointPaint = Paint()..color = const Color.fromRGBO(255, 255, 255, 0.8);
  10. final leftPointPaint = Paint()..color = const Color.fromRGBO(223, 157, 80, 1);
  11. final rightPointPaint = Paint()..color = const Color.fromRGBO(100, 208, 218, 1);
  12. final linePaint = Paint()
  13. ..color = const Color.fromARGB(228, 0, 0, 0)
  14. ..strokeWidth = 3;
  15. @override
  16. void paint(Canvas canvas, Size size) {
  17. if (pose == null) return;
  18. final double hRatio = imageSize.width == 0 ? 1 : size.width / imageSize.width;
  19. final double vRatio = imageSize.height == 0 ? 1 : size.height / imageSize.height;
  20. offsetForPart(PoseLandmark part) => Offset(part.position.x * hRatio, part.position.y * vRatio);
  21. // Landmark connections
  22. final landmarksByType = {for (final it in pose!.landmarks) it.type: it};
  23. for (final connection in connections) {
  24. if (landmarksByType[connection[0]] != null &&
  25. landmarksByType[connection[1]] != null &&
  26. landmarksByType[connection[0]]!.inFrameLikelihood > 0.5 &&
  27. landmarksByType[connection[1]]!.inFrameLikelihood > 0.5) {
  28. final point1 = offsetForPart(landmarksByType[connection[0]]!);
  29. final point2 = offsetForPart(landmarksByType[connection[1]]!);
  30. canvas.drawLine(point1, point2, linePaint);
  31. }
  32. }
  33. // for (final part in pose!.landmarks) {
  34. // // Landmark points
  35. // canvas.drawCircle(offsetForPart(part), 5, pointPaint);
  36. // if (part.type.isLeftSide) {
  37. // canvas.drawCircle(offsetForPart(part), 3, leftPointPaint);
  38. // } else if (part.type.isRightSide) {
  39. // canvas.drawCircle(offsetForPart(part), 3, rightPointPaint);
  40. // }
  41. // // Landmark labels
  42. // TextSpan span = TextSpan(
  43. // text: part.type.toString().substring(16),
  44. // style: const TextStyle(
  45. // color: Color.fromRGBO(0, 128, 255, 1),
  46. // fontSize: 10,
  47. // shadows: [
  48. // Shadow(
  49. // color: Color.fromRGBO(255, 255, 255, 1),
  50. // offset: Offset(1, 1),
  51. // blurRadius: 1,
  52. // ),
  53. // ],
  54. // ),
  55. // );
  56. // TextPainter tp = TextPainter(text: span, textAlign: TextAlign.left);
  57. // tp.textDirection = TextDirection.ltr;
  58. // tp.layout();
  59. // tp.paint(canvas, offsetForPart(part));
  60. // }
  61. }
  62. @override
  63. bool shouldRepaint(PosePainter oldDelegate) {
  64. return oldDelegate.pose != pose || oldDelegate.imageSize != imageSize;
  65. }
  66. static const List<List<PoseLandmarkType>> connections = [
  67. // [PoseLandmarkType.leftEar, PoseLandmarkType.leftEyeOuter],
  68. // [PoseLandmarkType.leftEyeOuter, PoseLandmarkType.leftEye],
  69. // [PoseLandmarkType.leftEye, PoseLandmarkType.leftEyeInner],
  70. // [PoseLandmarkType.leftEyeInner, PoseLandmarkType.nose],
  71. // [PoseLandmarkType.nose, PoseLandmarkType.rightEyeInner],
  72. // [PoseLandmarkType.rightEyeInner, PoseLandmarkType.rightEye],
  73. // [PoseLandmarkType.rightEye, PoseLandmarkType.rightEyeOuter],
  74. // [PoseLandmarkType.rightEyeOuter, PoseLandmarkType.rightEar],
  75. // [PoseLandmarkType.mouthLeft, PoseLandmarkType.mouthRight],
  76. [PoseLandmarkType.leftShoulder, PoseLandmarkType.rightShoulder],
  77. [PoseLandmarkType.leftShoulder, PoseLandmarkType.leftHip],
  78. [PoseLandmarkType.rightShoulder, PoseLandmarkType.rightHip],
  79. [PoseLandmarkType.rightShoulder, PoseLandmarkType.rightElbow],
  80. [PoseLandmarkType.rightWrist, PoseLandmarkType.rightElbow],
  81. // [PoseLandmarkType.rightWrist, PoseLandmarkType.rightThumb],
  82. // [PoseLandmarkType.rightWrist, PoseLandmarkType.rightIndexFinger],
  83. // [PoseLandmarkType.rightWrist, PoseLandmarkType.rightPinkyFinger],
  84. [PoseLandmarkType.leftHip, PoseLandmarkType.rightHip],
  85. [PoseLandmarkType.leftHip, PoseLandmarkType.leftKnee],
  86. [PoseLandmarkType.rightHip, PoseLandmarkType.rightKnee],
  87. [PoseLandmarkType.rightKnee, PoseLandmarkType.rightAnkle],
  88. [PoseLandmarkType.leftKnee, PoseLandmarkType.leftAnkle],
  89. [PoseLandmarkType.leftElbow, PoseLandmarkType.leftShoulder],
  90. [PoseLandmarkType.leftWrist, PoseLandmarkType.leftElbow],
  91. // [PoseLandmarkType.leftWrist, PoseLandmarkType.leftThumb],
  92. // [PoseLandmarkType.leftWrist, PoseLandmarkType.leftIndexFinger],
  93. // [PoseLandmarkType.leftWrist, PoseLandmarkType.leftPinkyFinger],
  94. // [PoseLandmarkType.leftAnkle, PoseLandmarkType.leftHeel],
  95. // [PoseLandmarkType.leftAnkle, PoseLandmarkType.leftToe],
  96. // [PoseLandmarkType.rightAnkle, PoseLandmarkType.rightHeel],
  97. // [PoseLandmarkType.rightAnkle, PoseLandmarkType.rightToe],
  98. // [PoseLandmarkType.rightHeel, PoseLandmarkType.rightToe],
  99. // [PoseLandmarkType.leftHeel, PoseLandmarkType.leftToe],
  100. // [PoseLandmarkType.rightIndexFinger, PoseLandmarkType.rightPinkyFinger],
  101. // [PoseLandmarkType.leftIndexFinger, PoseLandmarkType.leftPinkyFinger],
  102. ];
  103. }