opencv 用puttext在视频上添加文字

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了opencv 用puttext在视频上添加文字相关的知识,希望对你有一定的参考价值。

vs2015+opencv3,在视频文件上添加字符“hello”,用videocapture读取文件,建立mat对象取视频,再用puttext显示字符却显示不出来,编译成功,这是怎么回事?

你需要确保在添加视频文字后没再提取过,试试把添加字符的语句放在显示语句的紧前面 参考技术A putText(img, "hello", Point(300, 100), FONT_HERSHEY_SIMPLEX, 0.8, Scalar(0, 0, 255), 2);

opencv文字旋转 putText旋转90°

文章目录

● opencv文字旋转 putText旋转90°

  • putText本身不支持对文字进行旋转,所以将文字写在图像上,将图像进行旋转后再贴到原如上(可设置掩膜),以实现文字旋转的效果

1. cv::getTextSize获取文字的尺寸textSize

std::string text="treeAndCat";
int fontFace=cv::FONT_HERSHEY_SIMPLEX;//字体
double fontScale=2;//缩放系数
cv::Scalar color=cv::Scalar(255,255,255);//颜色
int thickness = 4;//线条粗细
int lineType = 8;//线型
int baseLine;//相当于四线格的第三行高度
  • cv::Size textSize = cv::getTextSize(text, fontFace, fontScale, thickness, &baseLine);

2. 设置文字图像textImg尺寸格式

  • cv::Mat textImg=cv::Mat::zeros(textSize.height+baseLine+thickness,textSize.width,/*CV_8UC1*/img.type());
    • textSize.width 文字的宽度
    • textSize.height 文字的高度(相当于四线格的第一第二行高度)
    • baseLine 基线相对于最底部文本点的y坐标(相当于四线格的第三行高度)
      • 所以在设置文字的roi时高度应该是textSize.height+baseLine
      • 尤其是文字中含fgjpqy字母时不要忘记加baseLine
      • 虽然纯数字不含fgjpq但cv::getTextSize计算出的baseLine也不为0


3. 将文字绘制在文字图像textImg上

  • cv::putText(textImg,text,cv::Point(0,textSize.height+0.5*baseLine),fontFace,fontScale,color,thickness,lineType);
    • 注意putText第三个参数为左下角坐标不是左上角
    • 因为不含fgjpq不担心超出,这里再加上了0.5*baseLine,是为了在文字上下各留0.5*baseLine的空白,(倘若含fgjpq就不要加0.5*baseLine了,上方会超出图像)
  • 一开始误将坐标写为左上角坐标导致textImg基本没有写到文字

  • putText第三个参数的y坐标一开始设为textSize.height没加0.5*baseLine,导致文字底端紧贴图像底部,上方仍有空余不美观

  • 第三个参数设置为cv::Point(0,textSize.height+baseLine)的效果

  • 第三个参数设置为cv::Point(0,textSize.height+0.5*baseLine)的效果


4. 对文字图像textImg进行旋转

  • cv::rotate(textImg,textImg,cv::ROTATE_90_COUNTERCLOCKWISE);
    • cv::ROTATE_90_COUNTERCLOCKWISE 逆时针旋转90°
    • cv::ROTATE_90_CLOCKWISE 顺时针旋转90°

5. 在原图img上设置用来放置文字图像textImg的roi区域

5.1 矩形区域roi

  • cv::Mat textRoi=img(cv::Rect(leftTopPoint.x,leftTopPoint.y,width,height));
    • cv::Rect里的参数分别为 左上角坐标x,y值以及区域宽高

5.2 行列范围roi

  • cv::Mat textRoi=img(cv::Range(rowStart,rowEnd),cv::Range(colStart,colEnd));
    • cv::Range里的参数分别为起止行,列;注意不包含结束行/列,相当于左闭右开 [start,end)

6. 将文字图像textImg放置在roi区域上

6.1 copyTo

6.1.1 不设置掩膜

  • textImg.copyTo(textRoi);
    • 会将整个textImg贴在原图roi处,textImg上非文字部分的图像可能会遮住原图的部分信息,如下图

6.2.2 设置掩膜

  • textImg.copyTo(textRoi,textImg);
    • 由于textImg设置了黑底白字,所以可以直接用它本身当掩膜(textRoi上只显示textImg在掩膜上不为0的像素),即只将textImg图像上的文字贴在了原图上
  • 另一种掩膜思路(这个复杂一点 可结合思考copyTo的掩膜)
    • 将水印加在原图上,对于需要作为水印的图,首先将这张图由彩色图转换为单通道灰度图,再利用阈值将灰度图转换为二值图,那么这张二值图就会成为掩膜;在原图上取与掩膜图同样大小的roi区域;对两张图进行按位与bitwise_and(有0为0),掩膜上的黑色区域在roi上也是黑色的;其次对掩膜进行非运算取反操作bitwise_not(掩膜黑白颠倒),再将掩膜与水印图进行位与bitwise_and(掩膜上的黑色区域在水印图上也是黑色的,非黑色部分正好是roi上的黑色区域),最后将进行了掩膜的roi与水印图相加add()

6.2 + / add()

  • 图像+图像 img1=img2+img3;

    • 相加图像尺寸应相同,相同位置像素值相加(单通道灰度图为对应位置像素的灰度值相加,多通道为各通道的每个像素值相加)
  • 图像+标量 img1=img2+a;

    • img2各通道的每个像素值加a
  • +add()的区别

    • +sum=mod(a+b,256) (取模操作,超出255的取模,如和为300最后sum就为45;直接使加法时需谨慎,超出255的话很亮的区域反而变暗了)
    • cv::add()sum=a+b>255 ? 255:a+b(饱和操作,超出255的设为255)
void cv::add(InputArray src1,InputArray src2,OutputArray dst,InputArray mask = noArray(),int dtype = -1)	

7. 流程代码

//获取文字尺寸
cv::Size textSize = cv::getTextSize(text, fontFace, fontScale, thickness, &baseLine);
//创建文字图像
cv::Mat textImg=cv::Mat::zeros(textSize.height+baseLine+thickness,textSize.width,img.type());
//将文字写到图像上
cv::putText(textImg,text,cv::Point(0,textImg.size().height-thickness),fontFace,fontScale,color,thickness,lineType);
//对图像进行旋转
cv::rotate(textImg,textImg,cv::ROTATE_90_COUNTERCLOCKWISE);
//在原图上取图像大小roi
cv::Mat textRoi=img(cv::Rect(leftTopPoint.x,leftTopPoint.y,textImg.size().width,textImg.size().height));
//将文字图像贴到原图上
textImg.copyTo(textRoi,textImg);
//或者使用add()与上方代码效果相同(因为我的图是黑底白字的,其他颜色的可能会有些区别)
//textRoi=textRoi+textImg
  • 最终效果图 将文字横着写在图像竖线上方


8. 封装代码

  • 还是封装成一个函数吧,方便调用
/** @brief Rotate and put the text on image.对文字进行旋转贴在图像上
@param img input image 输入图像
//@param leftTopPoint the upper-left corner point of roi 左上角坐标
//感觉设置roi左上角坐标还需考虑文字的大小,参数还是设置为文字底部中点坐标吧,位置如最后图片说明
@param centerPoint the center point of the bottom line 文字底线中点坐标见下图
@param text the text you want to write on the image 输入文字
@param rotate the angle of rotation; negative <0,270°; zero =0,180°; positive >0,90°. 旋转角度,<0 逆时针旋转90°;>0 顺时针旋转90°;=0 旋转180°
**/
void putTextRotate(cv::Mat &img, const cv::Point2f /*&leftTopPoint*/&centerPoint, const std::string &text,const int &rotate=-1)

  //对于字体的参数就不设置形参了(形参太多也麻烦不过也可以设置成默认参数),大多数时候都是一样的,有需要的直接在这个函数内更改就行
  int fontFace=cv::FONT_HERSHEY_SIMPLEX;//字体
  double fontScale=2;//缩放系数
  cv::Scalar color=cv::Scalar(255,255,255);//颜色 如果输入的是单通道图 cv::Scalar()里参数须更改为一个数 如cv::Scalar(255)
  //另外如果你的原图是白底的导致文字不能设置为白色的话,后期的掩膜需要重新设置,有两种方法:1.在将旋转后的文字图像贴在roi上之前,新建一张图mask保存将文字图像进行阈值处理后文字设置为白色的图,将这张图作为掩膜 textImg.copyTo(textRoi,mask); 2.一开始就画两张图,一张为黑底你需要的颜色的字体的图textImg,一张为黑底白色字体的图mask,后面mask须跟着textImg一起旋转,最后就可以直接将mask作为掩膜textImg.copyTo(textRoi,mask);
  int thickness = 4;//线条粗细
  int lineType = 8;//线型
  int baseLine;//相当于四线格的第三行高度

  //获取文字尺寸
  cv::Size textSize = cv::getTextSize(text, fontFace, fontScale, thickness, &baseLine);
  // baseLine+=thickness;//加上线宽不然会有小部分线条被截断
  //将文字图像设置为黑底方便后期做掩膜
  cv::Mat textImg=cv::Mat::zeros(textSize.height+baseLine+thickness,textSize.width,img.type());
  //因为输入的文字可能含有gjpq为防止下端被截断,左下角y坐标就设置为textImg.size().height-thickness相当于textSize.height+baseLine,(-thickness是为了底部的部分线条不被截断)
  cv::putText(textImg,text,cv::Point(0,textImg.size().height-thickness),fontFace,fontScale,color,thickness,lineType);
  //对文字图像进行旋转
  //小于0 顺时针旋转270° 即逆时针旋转90°
  cv::Mat textRoi;
  if(rotate<0)
  
    cv::rotate(textImg,textImg,cv::ROTATE_90_COUNTERCLOCKWISE);

    textRoi=img(cv::Rect(centerPoint.x-textImg.size().width,centerPoint.y-0.5*textImg.size().height,textImg.size().width,textImg.size().height));
  
  //大于0 顺时针旋转90°
  else if(rotate>0)
  
    cv::rotate(textImg,textImg,cv::ROTATE_90_CLOCKWISE);

    textRoi=img(cv::Rect(centerPoint.x,centerPoint.y-0.5*textImg.size().height,textImg.size().width,textImg.size().height));
  
  //等于0 旋转180°
  else
  
    cv::rotate(textImg,textImg,cv::ROTATE_180);

    textRoi=img(cv::Rect(centerPoint.x-0.5*textImg.size().width,centerPoint.y,textImg.size().width,textImg.size().height));
  

  //取roi
  //cv::Mat textRoi=img(cv::Rect(leftTopPoint.x,leftTopPoint.y,textImg.size().width,textImg.size().height));

  //将文字贴在图像上
  textImg.copyTo(textRoi,textImg);
  //若原图不是黑底add和加法直接用不可行,要用的话须自己设置掩膜等
  //cv::add(textImg,textRoi,textRoi);
  //textRoi=textRoi+textImg;

8.1 centerPoint位置

  • 如图centerPoint位于文字底部中点即图中的红点,从第二张图起依次是旋转180°,顺时针旋转90°,逆时针旋转90°

9. 完整测试代码

  • 加几句写一个完整的测试程序,读取图像然后将文字逆时针旋转90°写在图像中央
#include <iostream>
#include <opencv2/core.hpp>
#include <opencv2/highgui.hpp>
#include <opencv2/imgproc.hpp>

/** @brief Rotate and put the text on image.对文字进行旋转贴在图像上
@param img input image 输入图像
@param centerPoint the center point of the bottom line 文字底线中点坐标见上图
@param text the text you want to write on the image 输入文字
@param rotate the angle of rotation; negative <0,270°; zero =0,180°; positive >0,90°. 旋转角度,<0 逆时针旋转90°;>0 顺时针旋转90°;=0 旋转180°
**/
void putTextRotate(cv::Mat &img, const cv::Point2f /*&leftTopPoint*/&centerPoint, const std::string &text,const int &rotate=-1)

  //对于字体的参数就不设置形参了(形参太多也麻烦不过也可以设置成默认参数),大多数时候都是一样的,有需要的直接在这个函数内更改就行
  int fontFace=cv::FONT_HERSHEY_SIMPLEX;//字体
  double fontScale=2;//缩放系数
  cv::Scalar color=cv::Scalar(255,255,255);//颜色
  int thickness = 4;//线条粗细
  int lineType = 8;//线型
  int baseLine;//相当于四线格的第三行高度

  //获取文字尺寸
  cv::Size textSize = cv::getTextSize(text, fontFace, fontScale, thickness, &baseLine);
  // baseLine+=thickness;
  //将文字图像设置为黑底方便后期做掩膜
  cv::Mat textImg=cv::Mat::zeros(textSize.height+baseLine+thickness,textSize.width,img.type());
  std::cout<<"textImgSize: "<<textImg.size().width<<" , "<<textImg.size().height<<std::endl;
  //因为输入的文字可能含有gjpq为防止下端被截断,左下角y坐标就设置为textSize.height+baseLine
  cv::putText(textImg,text,cv::Point(0,textImg.size().height-thickness),fontFace,fontScale,color,thickness,lineType);
  //对文字图像进行旋转
  //小于0 顺时针旋转270° 即逆时针旋转90°
  cv::Mat textRoi;
  if(rotate<0)
  
    cv::rotate(textImg,textImg,cv::ROTATE_90_COUNTERCLOCKWISE);

    textRoi=img(cv::Rect(centerPoint.x-textImg.size().width,centerPoint.y-0.5*textImg.size().height,textImg.size().width,textImg.size().height));
  
  //大于0 顺时针旋转90°
  else if(rotate>0)
  
    cv::rotate(textImg,textImg,cv::ROTATE_90_CLOCKWISE);

    textRoi=img(cv::Rect(centerPoint.x,centerPoint.y-0.5*textImg.size().height,textImg.size().width,textImg.size().height));
  
  //等于0 旋转180°
  else
  
    cv::rotate(textImg,textImg,cv::ROTATE_180);

    textRoi=img(cv::Rect(centerPoint.x-0.5*textImg.size().width,centerPoint.y,textImg.size().width,textImg.size().height));
  
  std::cout<<"textRoiSize: "<<textRoi.size().width<<" , "<<textRoi.size().height<<std::endl;

  //取roi
  //cv::Mat textRoi=img(cv::Rect(leftTopPoint.x,leftTopPoint.y,textImg.size().width,textImg.size().height));

  //将文字贴在图像上
  textImg.copyTo(textRoi,textImg);
  //这里由于原图不是黑底的所以add和加法直接用不可行,要用的话须自己设置掩膜等
  //cv::add(textImg,textRoi,textRoi);
  //textRoi=textRoi+textImg;


int main()

   cv::Mat img = cv::imread("E:/desktop/01.jpg");
    if(img.empty())
    
        std::cout<<"error!"<<std::endl;
        return 0;
    
    std::cout<<"imgSize: "<<img.size().width<<" , "<<img.size().height<<std::endl;
    cv::Point2f centerPoint=cv::Point2f(0.5*img.size().width,0.5*img.size().height);
    std::string text="tree&cat";
    putTextRotate(img,centerPoint,text,-1);
    cv::imwrite("tree&cat.jpg",img);
    return 0;

9.1 测试效果图

以上是关于opencv 用puttext在视频上添加文字的主要内容,如果未能解决你的问题,请参考以下文章

opencv文字旋转 putText旋转90°

OpenCV添加中文——

OpenCV-文字绘制cv::putText

OpenCV 完整例程23. 图像添加中文文字

利用OpenCV的函数putText()为图像添加数值型文本内容

使用 PIL 在 OpenCV Python 中更改字体系列