应用程序实用工具——为应用程序添加轨迹条 OpenCV v4.8.0

下一个教程使用 GDAL 读取地理空间光栅文件

原作者Ana Huamán
兼容性OpenCV >= 3.0
  • 在之前的教程中(关于 使用 OpenCV 添加(混合)两幅图像以及更改图像的对比度和亮度! ),您可能已经注意到,我们需要向我们的程序提供一些输入,例如 α 和 beta。我们通过终端机输入了这些数据。
  • 现在,是时候使用一些花哨的图形用户界面工具了。OpenCV 提供了一些图形用户界面工具(highgui 模块)。Trackbar 就是一个例子。

在这里插入图片描述

  • 在本教程中,我们将对之前的两个程序进行修改,使其能够从轨迹条中获取输入信息。

目标

在本教程中,您将学习如何

代码

让我们修改教程中的程序 使用 OpenCV 添加(混合)两幅图像。我们将让用户使用 Trackbar 输入 α 值。

C++
该教程的代码如下所示。您也可以从此处下载

#include "opencv2/imgcodecs.hpp"
#include "opencv2/highgui.hpp"
#include <iostream>
using namespace cv;
using std::cout;
const int alpha_slider_max = 100;
int alpha_slider;
double alpha;
double beta;
Mat src1;
Mat src2;
Mat dst;
static void on_trackbar( int, void* )
{
 alpha = (double) alpha_slider/alpha_slider_max ;
 beta = ( 1.0 - alpha );
 addWeighted( src1, alpha, src2, beta, 0.0, dst);
 imshow( "Linear Blend", dst );
}
int main( void )
{
 src1 = imread( samples::findFile("LinuxLogo.jpg") );
 src2 = imread( samples::findFile("WindowsLogo.jpg") );
 if( src1.empty() ) { cout << "Error loading src1 \n"; return -1; }
 if( src2.empty() ) { cout << "Error loading src2 \n"; return -1; }
 alpha_slider = 0;
 namedWindow("Linear Blend", WINDOW_AUTOSIZE); // 创建窗口
 char TrackbarName[50];
 snprintf( TrackbarName, sizeof(TrackbarName), "Alpha x %d", alpha_slider_max );
 createTrackbar( TrackbarName, "Linear Blend", &alpha_slider, alpha_slider_max, on_trackbar );
 on_trackbar( alpha_slider, 0 );
 waitKey(0);
 return 0;
}

Java
该教程的代码如下所示。您也可以从此处下载

import java.awt.BorderLayout;
import java.awt.Container;
import java.awt.Image;
import javax.swing.BoxLayout;
import javax.swing.ImageIcon;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JSlider;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
import org.opencv.core.Core;
import org.opencv.core.Mat;
import org.opencv.highgui.HighGui;
import org.opencv.imgcodecs.Imgcodecs;
public class AddingImagesTrackbar {
 private static final int ALPHA_SLIDER_MAX = 100;
 private int alphaVal = 0;
 private Mat matImgSrc1;
 private Mat matImgSrc2;
 private Mat matImgDst = new Mat();
 private JFrame frame;
 private JLabel imgLabel;
 public AddingImagesTrackbar(String[] args) {
 String imagePath1 = "../data/LinuxLogo.jpg";
 String imagePath2 = "../data/WindowsLogo.jpg";
 if (args.length > 1) {
 imagePath1 = args[0];
 imagePath2 = args[1];
 }
 matImgSrc1 = Imgcodecs.imread(imagePath1);
 matImgSrc2 = Imgcodecs.imread(imagePath2);
 if (matImgSrc1.empty()) {
 System.out.println("Empty image: " + imagePath1);
 System.exit(0);
 }
 if (matImgSrc2.empty()) {
 System.out.println("Empty image: " + imagePath2);
 System.exit(0);
 }
 // 创建并设置窗口。
 frame = new JFrame("Linear Blend");
 frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE)// 设置内容窗格。
 Image img = HighGui.toBufferedImage(matImgSrc2)addComponentsToPane(frame.getContentPane(), img)// 使用内容窗格的默认边框布局。无需
 // setLayout(new BorderLayout());
 // 显示窗口。
 frame.pack();
 frame.setVisible(true);
 }
 private void addComponentsToPane(Container pane, Image img) {
 if (!(pane.getLayout() instanceof BorderLayout)) {
 pane.add(new JLabel("Container doesn't use BorderLayout!"));
 return;
 }
 JPanel sliderPanel = new JPanel();
 sliderPanel.setLayout(new BoxLayout(sliderPanel, BoxLayout.PAGE_AXIS));
 sliderPanel.add(new JLabel(String.format("Alpha x %d", ALPHA_SLIDER_MAX)));
 JSlider slider = new JSlider(0, ALPHA_SLIDER_MAX, 0);
 slider.setMajorTickSpacing(20);
 slider.setMinorTickSpacing(5);
 slider.setPaintTicks(true);
 slider.setPaintLabels(true);
 slider.addChangeListener(new ChangeListener() {
 @Override
 public void stateChanged(ChangeEvent e) {
 JSlider source = (JSlider) e.getSource();
 alphaVal = source.getValue();
 update();
 }
 });
 sliderPanel.add(slider);
 pane.add(sliderPanel, BorderLayout.PAGE_START);
 imgLabel = new JLabel(new ImageIcon(img));
 pane.add(imgLabel, BorderLayout.CENTER);
 }
 private void update() {
 double alpha = alphaVal / (double) ALPHA_SLIDER_MAX;
 double beta = 1.0 - alpha;
 Core.addWeighted(matImgSrc1, alpha, matImgSrc2, beta, 0, matImgDst);
 Image img = HighGui.toBufferedImage(matImgDst);
 imgLabel.setIcon(new ImageIcon(img));
 frame.repaint();
 }
 public static void main(String[] args) {
 // 加载本地 OpenCV 库
 System.loadLibrary(Core.NATIVE_LIBRARY_NAME)// 为事件派发线程安排任务:
 // 创建并显示此应用程序的图形用户界面。
 javax.swing.SwingUtilities.invokeLater(new Runnable() {
 @Override
 public void run() {
 new AddingImagesTrackbar(args);
 }
 });
 }
}

Python
该教程的代码如下所示。您也可以从此处下载

from __future__ import print_function
from __future__ import division
import cv2 as cv
import argparse
alpha_slider_max = 100
title_window = 'Linear Blend'
def on_trackbar(val):
 alpha = val / alpha_slider_max
 beta = ( 1.0 - alpha )
 dst = cv.addWeighted(src1, alpha, src2, beta, 0.0)
 cv.imshow(title_window, dst)
parser = argparse.ArgumentParser(description='Code for Adding a Trackbar to our applications tutorial.')
parser.add_argument('--input1', help='Path to the first input image.', default='LinuxLogo.jpg')
parser.add_argument('--input2', help='Path to the second input image.', default='WindowsLogo.jpg')
args = parser.parse_args()
src1 = cv.imread(cv.samples.findFile(args.input1))
src2 = cv.imread(cv.samples.findFile(args.input2))
if src1 is None:
 print('Could not open or find the image: ', args.input1)
 exit(0)
if src2 is None:
 print('Could not open or find the image: ', args.input2)
 exit(0)
cv.namedWindow(title_window)
trackbar_name = 'Alpha x %d' % alpha_slider_max
cv.createTrackbar(trackbar_name, title_window , 0, alpha_slider_max, on_trackbar)
# 显示一些内容
on_trackbar(0)
# 等待用户按下某个键
cv.waitKey()

说明

我们只分析与 Trackbar 相关的代码:

  • 首先,我们加载两张将要混合的图片。
    C++
 src1 = imread( samples::findFile("LinuxLogo.jpg") );
 src2 = imread( samples::findFile("WindowsLogo.jpg") );

Java

 String imagePath1 = "../data/LinuxLogo.jpg";
 String imagePath2 = "../data/WindowsLogo.jpg";
 if (args.length > 1) {
 imagePath1 = args[0];
 imagePath2 = args[1];
 }
 matImgSrc1 = Imgcodecs.imread(imagePath1);
 matImgSrc2 = Imgcodecs.imread(imagePath2);

Python

# Read images ( both have to be of the same size and type )
src1 = cv.imread(cv.samples.findFile(args.input1))
src2 = cv.imread(cv.samples.findFile(args.input2))
  • 要创建轨迹条,首先要创建轨迹条所在的窗口。因此
    C++
 namedWindow("Linear Blend", WINDOW_AUTOSIZE); // Create Window

Java

 // Create and set up the window.
 frame = new JFrame("Linear Blend");
 frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
 // Set up the content pane.
 Image img = HighGui.toBufferedImage(matImgSrc2);
 addComponentsToPane(frame.getContentPane(), img);
 // Use the content pane's default BorderLayout. No need for
 // setLayout(new BorderLayout());
 // Display the window.
 frame.pack();
 frame.setVisible(true);

Python

cv.namedWindow(title_window)
  • 现在我们可以创建 Trackbar 了:
    C++
 char TrackbarName[50];
 snprintf( TrackbarName, sizeof(TrackbarName), "Alpha x %d", alpha_slider_max );
 createTrackbar( TrackbarName, "Linear Blend", &alpha_slider, alpha_slider_max, on_trackbar );

Java

 sliderPanel.add(new JLabel(String.format("Alpha x %d", ALPHA_SLIDER_MAX)));
 JSlider slider = new JSlider(0, ALPHA_SLIDER_MAX, 0);
 slider.setMajorTickSpacing(20);
 slider.setMinorTickSpacing(5);
 slider.setPaintTicks(true);
 slider.setPaintLabels(true);

Python

trackbar_name = 'Alpha x %d' % alpha_slider_max
cv.createTrackbar(trackbar_name, title_window , 0, alpha_slider_max, on_trackbar)

请注意以下内容(C++ 代码):

  • 我们的跟踪条有一个标签 TrackbarName
  • 跟踪栏位于名为 Linear Blend 的窗口中
  • Trackbar 的数值范围为 0 至 alpha_slider_max(最小值始终为 0)。
  • Trackbar 的数值存储在 alpha_slider 中。
  • 每当用户移动 Trackbar 时,就会调用回调函数 on_trackbar

最后,我们必须为 C++ 和 Python 代码定义回调函数 on_trackbar,在 Java 中使用匿名内部类监听器
C++


static void on_trackbar( int, void* )
{
 alpha = (double) alpha_slider/alpha_slider_max ;
 beta = ( 1.0 - alpha );
 addWeighted( src1, alpha, src2, beta, 0.0, dst);
 imshow( "Linear Blend", dst );
}

Java

 slider.addChangeListener(new ChangeListener() {
 @Override
 public void stateChanged(ChangeEvent e) {
 JSlider source = (JSlider) e.getSource();
 alphaVal = source.getValue();
 update();
 }
 });

Python

def on_trackbar(val):
 alpha = val / alpha_slider_max
 beta = ( 1.0 - alpha )
 dst = cv.addWeighted(src1, alpha, src2, beta, 0.0)
 cv.imshow(title_window, dst)

请注意(C++ 代码):

  • 我们使用 alpha_slider(整数)的值来获取 alpha 的双倍值。
  • 用户每次移动轨迹条时,alpha_slider 都会更新。
  • 我们将 src1、src2、dist、alpha、alpha_slider 和 beta 定义为全局变量,因此它们可以在任何地方使用。

运行结果

  • 我们的程序将产生以下输出结果:

在这里插入图片描述

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值