在图像上添加标记和数字

文章介绍了如何在Windows11的MicrosoftVisualStudioCommunity2019环境下,利用VTK库的MyStyle类创建一个自定义交互式样式,实现在图像上添加标记和数字的功能。主要讲解了MyStyle类的实现,包括重写OnLeftButtonDown方法处理鼠标事件和使用vtkCaptionActor2D添加数字标记的过程。
摘要由CSDN通过智能技术生成

开发环境:

  1. Windows 11 家庭中文版
  2. Microsoft Visual Studio Community 2019
  3. VTK-9.3.0.rc0
  4. vtk-example

demo解决问题:在图像上添加标记和数字的功能。

在这里插入图片描述

关键点:

  1. MyStyle类继承自vtkActorStyleImage,用于创建图像渲染的自定义交互式样式。
  2. 在’MyStyle’类中,'OnLeftButtonDown’函数被重写以处理鼠标左键点击。它会获取用户点击的像素位置,在该位置添加一个标记,然后执行交互样式的默认行为。
  3. 'MyStyle’中的’AddNumber’函数负责在指定的2D位置添加一个数字作为标记。它使用vtkCaptionActor2D类创建一个文本标签,并将其附加到指定的位置上。注意:vtkCaptionActor2D 是一个混合的 2D/3D 角色,用于将文本与场景中的点(AttachmentPoint)关联。绘制的标题可带有矩形边框和连接标题与附件点的引线。可以选择在其端点处对引线进行字形化,以创建箭头或其他指示符。
  4. 'main’函数设置了图像渲染的流水线。它创建了一个空白图像,在其中绘制了一个红色圆圈,并设置了窗口、渲染器和交互器。

prj name: CaptionActor2D

#include <vtkAbstractPicker.h>
#include <vtkActor2D.h>
#include <vtkCaptionActor2D.h>
#include <vtkImageActor.h>
#include <vtkImageCanvasSource2D.h>
#include <vtkImageMapper3D.h>
#include <vtkInteractorStyleImage.h>
#include <vtkNamedColors.h>
#include <vtkNew.h>
#include <vtkObjectFactory.h>
#include <vtkProperty.h>
#include <vtkRenderWindow.h>
#include <vtkRenderWindowInteractor.h>
#include <vtkRenderer.h>
#include <vtkTextProperty.h>

#include <iostream>
#include <sstream>
#include <string>

namespace {
class MyStyle : public vtkInteractorStyleImage
{
public:
  static MyStyle* New();
  vtkTypeMacro(MyStyle, vtkInteractorStyleImage);

  std::vector<vtkActor2D*> Numbers;

  void OnLeftButtonDown()
  {
    // std::cout << "Picking pixel: " << this->Interactor->GetEventPosition()[0]
    // << " " << this->Interactor->GetEventPosition()[1] << std::endl;
    this->Interactor->GetPicker()->Pick(
        this->Interactor->GetEventPosition()[0],
        this->Interactor->GetEventPosition()[1],
        0, // always zero.
        // this->Interactor->GetRenderWindow()->GetRenderers()->GetFirstRenderer());
        this->CurrentRenderer);
    double picked[3];
    this->Interactor->GetPicker()->GetPickPosition(picked);
    // std::cout << "Picked point with coordinate: " << picked[0] << " " <<
    // picked[1] << " " << picked[2] << std::endl;

    this->AddNumber(picked);

    // Forward events
    vtkInteractorStyleImage::OnLeftButtonDown();

    // this->Interactor->GetRenderWindow()->Render();
    this->Interactor->Render();
  }

  void AddNumber(double p[3])
  {
    std::cout << "Adding marker at " << p[0] << " " << p[1]; //<< std::endl;

    // normally, with an image you would do
    // double* s = image->GetSpacing();
    // double* o = image->GetOrigin();
    // p[0] = static_cast<int>( (p[0] - o[0]) / s[0] + 0.5 );
    p[0] = static_cast<int>(p[0] + 0.5);
    p[1] = static_cast<int>(p[1] + 0.5);

    std::cout << " -> " << p[0] << " " << p[1] << std::endl;

    // Convert the current number to a string
    std::stringstream ss;
    ss << Numbers.size();
    std::cout << "Adding number: " << ss.str() << std::endl;

    // Create an actor for the text
    vtkNew<vtkCaptionActor2D> captionActor;
    captionActor->SetCaption(ss.str().c_str());
    captionActor->SetAttachmentPoint(p);
    captionActor->BorderOff();
    captionActor->GetCaptionTextProperty()->BoldOff();
    captionActor->GetCaptionTextProperty()->ItalicOff();
    captionActor->GetCaptionTextProperty()->ShadowOff();
    captionActor->ThreeDimensionalLeaderOff();

    this->CurrentRenderer->AddViewProp(captionActor);

    this->Numbers.push_back(captionActor);
  }
};

vtkStandardNewMacro(MyStyle);
} // namespace

int main(int, char*[])
{
  vtkNew<vtkNamedColors> colors;

  // Create a blank, black image.
  vtkNew<vtkImageCanvasSource2D> drawing;
  drawing->SetScalarTypeToUnsignedChar();
  drawing->SetNumberOfScalarComponents(3);
  drawing->SetExtent(0, 20, 0, 50, 0, 0);
  drawing->FillBox(0, 20, 0, 50);

  // Draw a red circle of radius 5 centered at (9,10).
  drawing->SetDrawColor(255, 0, 0, 0);
  drawing->DrawCircle(9, 10, 5);
  drawing->Update();

  vtkNew<vtkImageActor> actor;
  actor->GetMapper()->SetInputConnection(drawing->GetOutputPort());
  actor->InterpolateOff();

  vtkNew<vtkRenderer> renderer;
  vtkNew<vtkRenderWindow> renderWindow;
  renderWindow->AddRenderer(renderer);
  renderWindow->SetWindowName("CaptionActor2D");

  vtkNew<vtkRenderWindowInteractor> renderWindowInteractor;
  renderWindowInteractor->SetRenderWindow(renderWindow);

  renderer->AddActor(actor);
  renderer->SetBackground(colors->GetColor3d("Silver").GetData());

  renderer->GradientBackgroundOn();
  renderer->SetBackground2(colors->GetColor3d("Blue").GetData());

  renderWindow->Render();

  vtkNew<MyStyle> style;

  renderWindowInteractor->SetInteractorStyle(style);
  style->SetDefaultRenderer(renderer);
  style->SetCurrentRenderer(renderer);
  renderWindowInteractor->Start();

  return EXIT_SUCCESS;
}

  • 10
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,根据您的要求,我会给出具体的代码实现。 首先,您需要安装OpenCV库,然后导入以下所需的库: ```python import cv2 import numpy as np from matplotlib import pyplot as plt ``` 接下来,我们需要读入一张细胞图像,这里我们以细胞图像`cell.jpg`为例: ```python img = cv2.imread('cell.jpg') ``` 然后,我们需要进行预处理,包括灰度化、二值化等操作: ```python gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY) ret, thresh = cv2.threshold(gray,0,255,cv2.THRESH_BINARY_INV+cv2.THRESH_OTSU) ``` 然后,我们需要对图像进行形态学操作,以便于进行分割: ```python # 进行腐蚀操作 kernel = np.ones((3,3),np.uint8) opening = cv2.morphologyEx(thresh,cv2.MORPH_OPEN,kernel,iterations = 2) # 进行膨胀操作 sure_bg = cv2.dilate(opening,kernel,iterations=3) ``` 接下来,我们可以通过距离变换来获取前景区域: ```python # 距离变换 dist_transform = cv2.distanceTransform(opening,cv2.DIST_L2,5) ret, sure_fg = cv2.threshold(dist_transform,0.7*dist_transform.max(),255,0) # 找到未知区域 sure_fg = np.uint8(sure_fg) unknown = cv2.subtract(sure_bg,sure_fg) ``` 然后,我们可以通过分水岭算法来进行分割: ```python # 分水岭算法 ret, markers = cv2.connectedComponents(sure_fg) # 添加标记 markers = markers+1 # 将未知区域标记为0 markers[unknown==255] = 0 # 应用分水岭算法 markers = cv2.watershed(img,markers) img[markers == -1] = [255,0,0] ``` 最后,我们可以通过以下代码来标记细胞数字并显示图像: ```python # 标记细胞数字 font = cv2.FONT_HERSHEY_SIMPLEX count = 0 for i in range(1,ret): x,y,w,h = cv2.boundingRect(np.uint8(markers==i)) if w*h>100: count += 1 cv2.putText(img,str(count),(x,y-5), font, 0.5,(255,255,255),2,cv2.LINE_AA) cv2.rectangle(img,(x,y),(x+w,y+h),(0,255,0),2) # 显示图像 plt.imshow(cv2.cvtColor(img, cv2.COLOR_BGR2RGB)) plt.show() ``` 这样,我们就完成了细胞计数并在图上标记细胞数字的任务。完整代码如下: ```python import cv2 import numpy as np from matplotlib import pyplot as plt # 读入图像 img = cv2.imread('cell.jpg') # 灰度化和二值化处理 gray = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY) ret, thresh = cv2.threshold(gray,0,255,cv2.THRESH_BINARY_INV+cv2.THRESH_OTSU) # 形态学处理 kernel = np.ones((3,3),np.uint8) opening = cv2.morphologyEx(thresh,cv2.MORPH_OPEN,kernel,iterations = 2) sure_bg = cv2.dilate(opening,kernel,iterations=3) # 距离变换 dist_transform = cv2.distanceTransform(opening,cv2.DIST_L2,5) ret, sure_fg = cv2.threshold(dist_transform,0.7*dist_transform.max(),255,0) sure_fg = np.uint8(sure_fg) unknown = cv2.subtract(sure_bg,sure_fg) # 分水岭算法 ret, markers = cv2.connectedComponents(sure_fg) markers = markers+1 markers[unknown==255] = 0 markers = cv2.watershed(img,markers) img[markers == -1] = [255,0,0] # 标记细胞数字 font = cv2.FONT_HERSHEY_SIMPLEX count = 0 for i in range(1,ret): x,y,w,h = cv2.boundingRect(np.uint8(markers==i)) if w*h>100: count += 1 cv2.putText(img,str(count),(x,y-5), font, 0.5,(255,255,255),2,cv2.LINE_AA) cv2.rectangle(img,(x,y),(x+w,y+h),(0,255,0),2) # 显示图像 plt.imshow(cv2.cvtColor(img, cv2.COLOR_BGR2RGB)) plt.show() ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值