#include #include #include #include #include #include #include #include "common.hpp" // Define namespace to simplify code using namespace cv; using namespace cv::dnn; using namespace std; int threshold1 = 0; int threshold2 = 50; int blurAmount = 5; // Function to apply sigmoid activation static void sigmoid(Mat& input) { exp(-input, input); // e^-input input = 1.0 / (1.0 + input); // 1 / (1 + e^-input) } static void applyCanny(const Mat& image, Mat& result) { Mat gray; cvtColor(image, gray, COLOR_BGR2GRAY); Canny(gray, result, threshold1, threshold2); } // Load Model static void loadModel(const string modelPath, String backend, String target, Net &net, EngineType engine){ net = readNetFromONNX(modelPath, engine); net.setPreferableBackend(getBackendID(backend)); net.setPreferableTarget(getTargetID(target)); } static void setupCannyWindow(){ destroyWindow("Output"); namedWindow("Output", WINDOW_AUTOSIZE); moveWindow("Output", 200, 50); createTrackbar("thrs1", "Output", &threshold1, 255, nullptr); createTrackbar("thrs2", "Output", &threshold2, 255, nullptr); createTrackbar("blur", "Output", &blurAmount, 20, nullptr); } // Function to process the neural network output to generate edge maps static pair postProcess(const vector& output, int height, int width) { vector preds; preds.reserve(output.size()); for (const Mat &p : output) { Mat img; // Correctly handle 4D tensor assuming it's always in the format [1, 1, height, width] Mat processed; if (p.dims == 4 && p.size[0] == 1 && p.size[1] == 1) { // Use only the spatial dimensions processed = p.reshape(0, {p.size[2], p.size[3]}); } else { processed = p.clone(); } sigmoid(processed); normalize(processed, img, 0, 255, NORM_MINMAX, CV_8U); resize(img, img, Size(width, height)); // Resize to the original size preds.push_back(img); } Mat fuse = preds.back(); // Last element as the fused result // Calculate the average of the predictions Mat ave = Mat::zeros(height, width, CV_32F); for (Mat &pred : preds) { Mat temp; pred.convertTo(temp, CV_32F); ave += temp; } ave /= static_cast(preds.size()); ave.convertTo(ave, CV_8U); return {fuse, ave}; // Return both fused and average edge maps } static void applyDexined(Net &net, const Mat &image, Mat &result) { int originalWidth = image.cols; int originalHeight = image.rows; vector outputs; net.forward(outputs); pair res = postProcess(outputs, originalHeight, originalWidth); result = res.first; // or res.second for average edge map } int main(int argc, char** argv) { const string about = "This sample demonstrates edge detection with dexined and canny edge detection techniques.\n\n" "To run with canny:\n" "\t ./example_dnn_edge_detection --input=path/to/your/input/image/or/video (don't give --input flag if want to use device camera)\n" "With Dexined:\n" "\t ./example_dnn_edge_detection dexined --input=path/to/your/input/image/or/video\n\n" "For switching between deep learning based model(dexined) and canny edge detector, press space bar in case of video. In case of image, pass the argument --method for switching between dexined and canny.\n" "Model path can also be specified using --model argument. Download it using python download_models.py dexined from dnn samples directory\n\n"; const string param_keys = "{ help h | | Print help message. }" "{ @alias | | An alias name of model to extract preprocessing parameters from models.yml file. }" "{ zoo | ../dnn/models.yml | An optional path to file with preprocessing parameters }" "{ input i | | Path to input image or video file. Skip this argument to capture frames from a camera.}" "{ method | dexined | Choose method: dexined or canny. }" "{ model | | Path to the model file for using dexined. }"; const string backend_keys = format( "{ backend | default | Choose one of computation backends: " "default: automatically (by default), " "openvino: Intel's Deep Learning Inference Engine (https://software.intel.com/openvino-toolkit), " "opencv: OpenCV implementation, " "vkcom: VKCOM, " "cuda: CUDA, " "webnn: WebNN }"); const string target_keys = format( "{ target | cpu | Choose one of target computation devices: " "cpu: CPU target (by default), " "opencl: OpenCL, " "opencl_fp16: OpenCL fp16 (half-float precision), " "vpu: VPU, " "vulkan: Vulkan, " "cuda: CUDA, " "cuda_fp16: CUDA fp16 (half-float preprocess) }"); string keys = param_keys + backend_keys + target_keys; CommandLineParser parser(argc, argv, keys); if (parser.has("help")) { cout << about << endl; parser.printMessage(); return -1; } string modelName = parser.get("@alias"); string zooFile = parser.get("zoo"); const char* path = getenv("OPENCV_SAMPLES_DATA_PATH"); if ((path != NULL) || parser.has("@alias") || (parser.get("model") != "")) { modelName = "dexined"; zooFile = findFile(zooFile); } else{ cout<<"[WARN] set the environment variables or pass path to dexined.onnx model file using --model and models.yml file using --zoo for using dexined based edge detector. Continuing with canny edge detector\n\n"; } keys += genPreprocArguments(modelName, zooFile); parser = CommandLineParser(argc, argv, keys); int width = parser.get("width"); int height = parser.get("height"); float scale = parser.get("scale"); Scalar mean = parser.get("mean"); bool swapRB = parser.get("rgb"); String backend = parser.get("backend"); String target = parser.get("target"); string method = parser.get("method"); String sha1 = parser.get("sha1"); string model = findModel(parser.get("model"), sha1); EngineType engine = ENGINE_AUTO; if (backend != "default" || target != "cpu"){ engine = ENGINE_CLASSIC; } parser.about(about); VideoCapture cap; if (parser.has("input")) cap.open(samples::findFile(parser.get("input"))); else cap.open(0); if (!cap.isOpened()) { cerr << "Error: Video could not be opened." << endl; return -1; } namedWindow("Input", WINDOW_AUTOSIZE); namedWindow("Output", WINDOW_AUTOSIZE); moveWindow("Output", 200, 0); Net net; Mat image; if (model.empty()) { cout << "[WARN] Model file not provided, using canny instead. Pass model using --model=/path/to/dexined.onnx to use dexined model." << endl; method = "canny"; } if (method == "dexined") { loadModel(model, backend, target, net, engine); } else{ Mat dummy = Mat::zeros(512, 512, CV_8UC3); setupCannyWindow(); } cout<<"To switch between canny and dexined press space bar."<> image; if (image.empty()) { cout << "Press any key to exit" << endl; waitKey(); break; } Mat result; int kernelSize = 2 * blurAmount + 1; Mat blurred; GaussianBlur(image, blurred, Size(kernelSize, kernelSize), 0); if (method == "dexined") { Mat blob = blobFromImage(blurred, scale, Size(width, height), mean, swapRB, false, CV_32F); net.setInput(blob); applyDexined(net, image, result); } else if (method == "canny") { applyCanny(blurred, result); } imshow("Input", image); imshow("Output", result); int key = waitKey(30); if (key == ' ' && method == "canny") { if (!model.empty()){ method = "dexined"; if (net.empty()) loadModel(model, backend, target, net, engine); destroyWindow("Output"); namedWindow("Input", WINDOW_AUTOSIZE); namedWindow("Output", WINDOW_AUTOSIZE); moveWindow("Output", 200, 0); } else { cout << "[ERROR] Provide model file using --model to use dexined. Download model using python download_models.py dexined from dnn samples directory" << endl; } } else if (key == ' ' && method == "dexined") { method = "canny"; setupCannyWindow(); } else if (key == 27 || key == 'q') { // Escape key to exit break; } } destroyAllWindows(); return 0; }