1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94
| #include <opencv2/opencv.hpp> #include <vector> #include <map>
using namespace cv; using namespace std;
struct StaticObject { Rect2d bbox; int stayFrames; bool isWarning; };
int main() { VideoCapture cap("test.mp4"); if (!cap.isOpened()) return -1;
Ptr<BackgroundSubtractor> bgSubtractor = createBackgroundSubtractorMOG2(500, 16, false);
Mat frame, fgMask, bgImg; map<int, StaticObject> trackedObjects; int objectId = 0;
while (cap.read(frame)) { bgSubtractor->apply(frame, fgMask, 0.001);
morphologyEx(fgMask, fgMask, MORPH_OPEN, getStructuringElement(MORPH_RECT, Size(5,5))); morphologyEx(fgMask, fgMask, MORPH_CLOSE, getStructuringElement(MORPH_RECT, Size(15,15)));
vector<vector<Point>> contours; findContours(fgMask, contours, RETR_EXTERNAL, CHAIN_APPROX_SIMPLE);
vector<Rect> currentRects; for (const auto& contour : contours) { if (contourArea(contour) > 500) { Rect rect = boundingRect(contour); currentRects.push_back(rect); } }
map<int, StaticObject> newTrackedObjects; for (const auto& rect : currentRects) { bool matched = false; for (auto& obj : trackedObjects) { Rect2d trackedRect = obj.second.bbox; double intersectionArea = (trackedRect & rect).area(); double unionArea = trackedRect.area() + rect.area() - intersectionArea; if (intersectionArea / unionArea > 0.3) { obj.second.bbox = rect; obj.second.stayFrames++; newTrackedObjects[obj.first] = obj.second; matched = true; break; } } if (!matched) { StaticObject newObj{rect, 1, false}; newTrackedObjects[objectId++] = newObj; } } trackedObjects = newTrackedObjects;
for (auto& obj : trackedObjects) { if (obj.second.stayFrames > 100 && !obj.second.isWarning) { obj.second.isWarning = true; rectangle(frame, obj.second.bbox, Scalar(0,0,255), 2); putText(frame, "Warning: Static Object!", Point(10,30), FONT_HERSHEY_SIMPLEX, 1, Scalar(0,0,255), 2); } else if (obj.second.isWarning) { rectangle(frame, obj.second.bbox, Scalar(0,0,255), 2); } }
imshow("Frame", frame); imshow("Foreground Mask", fgMask); if (waitKey(30) == 27) break; }
cap.release(); destroyAllWindows(); return 0; }
|