【瞎折腾/opencv】用记事本打开视频(字符代替像素)

2023-12-24 07:10

本文主要是介绍【瞎折腾/opencv】用记事本打开视频(字符代替像素),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

  • 说在前面
  • Theory
  • SourceCode
  • Result

说在前面

  • opencv版本:4.0.1
  • 操作系统:win10
  • vs版本:2017
  • 其他说明:瞎折腾系列第一期,目前实现的效果并不是很好
  • 基础:【opencv/core module】(二)How to scan images, lookup tables and time measurement with OpenCV
    【opencv/videoio module】(一)Video Input with OpenCV and similarity measurement
    【opencv/videoio module】(二)Creating a video with OpenCV

Theory

  • 原理其实很简单
  1. 捕捉视频一帧,转为灰度图
  2. 降低颜色空间的大小
  3. 对每一个位置的像素用一个字符代替,字符所占的比例逐渐增大(例如 . ; + # @这样,相当于颜色逐渐加深)
  4. 最后,将转化后的由字符形成这一帧保存起来
  • 存在的问题
  1. 用哪些字符替代?

  2. 用哪种方式保存字符帧?

    这两问题我觉得有些联系,我在实验中用的是putText()函数,将字符直接输出到Mat上,然后使用VideoWriter写入到视频文件中,这样便于视频的生成;但是putText这个函数用的时候有些偏差,有些字符所占的宽度不是固定的,会导致最终结果emmm(っ °Д °;)っ
    在这里插入图片描述
    另一种保存方式,直接输出到文本文件中,这样就没有putText的问题,但是如何将其转换成视频?难道一帧一帧的截图 (不嫌麻烦可以一试,代码也有) ?(而且ascii码的话长宽是不等的)emmmm┗|`O′|┛
    在这里插入图片描述

SourceCode

主要是putText函数

void cv::putText 
(
InputOutputArray img, //输入到img上
const String & text, //要显示的字符
Point org, //要显示的位置(像素)
int fontFace, //字体类型
double fontScale, //字体缩放
Scalar color, //字体颜色
int thickness = 1, //字体粗细
int lineType = LINE_8, //线条类型
bool bottomLeftOrigin = false //是否将text的左下角作为起始位置,否的话左上角
)

字体类型(实验中用的是1,也就是FONT_HERSHEY_PLAIN,貌似最小的)

name含义
FONT_HERSHEY_SIMPLEXnormal size sans-serif font
FONT_HERSHEY_PLAINsmall size sans-serif font
FONT_HERSHEY_DUPLEXnormal size sans-serif font (more complex than FONT_HERSHEY_SIMPLEX)
FONT_HERSHEY_COMPLEXnormal size serif font
FONT_HERSHEY_TRIPLEXnormal size serif font (more complex than FONT_HERSHEY_COMPLEX)
FONT_HERSHEY_COMPLEX_SMALLsmaller version of FONT_HERSHEY_COMPLEX
FONT_HERSHEY_SCRIPT_SIMPLEXhand-writing style font
FONT_HERSHEY_SCRIPT_COMPLEXmore complex variant of FONT_HERSHEY_SCRIPT_SIMPLEX
FONT_ITALICflag for italic font

线条类型(用的0,填充)

name含义
FILLED
LINE_44-connected line
LINE_88-connected line
LINE_AAantialiased line

最后一个参数,在确定Point org后,相当于下面哪一个点(左边俩黑色)作为org
在这里插入图片描述

#include <iostream> // for standard I/O
#include <string>   // for strings
#include <opencv2/core.hpp>     // Basic OpenCV structures (cv::Mat)
#include <opencv2/videoio.hpp>  // Video write
#include <opencv2/imgproc.hpp>
#include <opencv2/highgui.hpp>
using namespace std;
using namespace cv;int CNT = 0;void ProcessText(Mat& I)
{CV_Assert(I.depth() == CV_8U && I.channels() == 1);string txt_name = "test_" + to_string(CNT) + ".txt";ofstream out(txt_name);for (int i = 15; i < I.rows - 10; ++i){char text[240] = { 0 };for (int j = 20; j < I.cols - 25; ++j){if (I.at<uchar>(i, j) < 25)text[j - 20] = '-';else if (I.at<uchar>(i, j) < 50)text[j - 20] = '-';else if (I.at<uchar>(i, j) < 75)text[j - 20] = '<';else if (I.at<uchar>(i, j) < 100)text[j - 20] = '>';else if (I.at<uchar>(i, j) < 125)text[j - 20] = '+';else if (I.at<uchar>(i, j) < 150)text[j - 20] = '+';else if (I.at<uchar>(i, j) < 175)text[j - 20] = '&';else if (I.at<uchar>(i, j) < 200)text[j - 20] = '&';else if (I.at<uchar>(i, j) < 225)text[j - 20] = '%';else if (I.at<uchar>(i, j) < 250)text[j - 20] = '#';elsetext[j - 20] = '@';}out << text << endl;}CNT++;out.close();
}int ggg = 0;void Process(Mat& I, Mat& Out)
{// accept only char type matricesCV_Assert(I.depth() == CV_8U && I.channels() == 1);for (int i = 15; i < I.rows - 10; ++i){char text[240] = { 0 };for (int j = 20; j < I.cols - 25; ++j){if (I.at<uchar>(i, j) < 25)text[j - 20] = '-';else if (I.at<uchar>(i, j) < 50)text[j - 20] = '-';else if (I.at<uchar>(i, j) < 75)text[j - 20] = '<';else if (I.at<uchar>(i, j) < 100)text[j - 20] = '>';else if (I.at<uchar>(i, j) < 125)text[j - 20] = '+';else if (I.at<uchar>(i, j) < 150)text[j - 20] = '+';else if (I.at<uchar>(i, j) < 175)text[j - 20] = '&';else if (I.at<uchar>(i, j) < 200)text[j - 20] = '&';else if (I.at<uchar>(i, j) < 225)text[j - 20] = '%';else if (I.at<uchar>(i, j) < 250)text[j - 20] = '#';elsetext[j - 20] = '@';}putText(Out, text, Point(0, (i - 14) * 10), 1, 1, Scalar(255), 1, 1);if (ggg == 0)cout << text << endl;}ggg = 1;}int main()
{const string source = "1.mp4";           // the source file nameconst string NAME = "test.avi";   // Form the new name with containerint divideWith = 25;uchar table[256];for (int i = 0; i < 256; ++i)table[i] = (uchar)(divideWith * (i / divideWith));Mat lookUpTable(1, 256, CV_8U);uchar* p = lookUpTable.ptr();for (int i = 0; i < 256; ++i)p[i] = table[i];VideoCapture inputVideo(source);              // Open inputif (!inputVideo.isOpened()){cout << "Could not open the input video: " << source << endl;return -1;}VideoWriter outputVideo(NAME, VideoWriter::fourcc('M', 'J', 'P', 'G'),inputVideo.get(CAP_PROP_FPS), Size(960*2, 540*2), true);if (!outputVideo.isOpened()){cout << "Could not open the output video for write: " << NAME << endl;return -1;}Mat src;while (true){inputVideo >> src;              // readif (src.empty()) break;         // check if at endpyrDown(src, src, Size(src.cols / 2, src.rows / 2));//缩放pyrDown(src, src, Size(src.cols / 2, src.rows / 2));//缩放pyrDown(src, src, Size(src.cols / 2, src.rows / 2));//缩放cvtColor(src, src, COLOR_BGR2GRAY);//转为灰度图LUT(src, lookUpTable, src);//降低颜色空间imshow("src", src);Mat res(Size(1920, 1080),CV_8UC1);//用于保存结果Process(src, res);//处理//pyrDown(res, res, Size(960, 540));//缩放cvtColor(res, res, COLOR_GRAY2BGR);//转化成BGR,VideoWriter的问题imshow("test", res);outputVideo << res;//写入//ProcessText(src);waitKey(15);}return 0;
}

Result

  • 上面原图,下面结果,都是gif

在这里插入图片描述
在这里插入图片描述


END

这篇关于【瞎折腾/opencv】用记事本打开视频(字符代替像素)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/530961

相关文章

解决IDEA报错:编码GBK的不可映射字符问题

《解决IDEA报错:编码GBK的不可映射字符问题》:本文主要介绍解决IDEA报错:编码GBK的不可映射字符问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录IDEA报错:编码GBK的不可映射字符终端软件问题描述原因分析解决方案方法1:将命令改为方法2:右下jav

SQL server配置管理器找不到如何打开它

《SQLserver配置管理器找不到如何打开它》最近遇到了SQLserver配置管理器打不开的问题,尝试在开始菜单栏搜SQLServerManager无果,于是将自己找到的方法总结分享给大家,对SQ... 目录方法一:桌面图标进入方法二:运行窗口进入方法三:查找文件路径方法四:检查 SQL Server 安

在PyCharm中安装PyTorch、torchvision和OpenCV详解

《在PyCharm中安装PyTorch、torchvision和OpenCV详解》:本文主要介绍在PyCharm中安装PyTorch、torchvision和OpenCV方式,具有很好的参考价值,... 目录PyCharm安装PyTorch、torchvision和OpenCV安装python安装PyTor

openCV中KNN算法的实现

《openCV中KNN算法的实现》KNN算法是一种简单且常用的分类算法,本文主要介绍了openCV中KNN算法的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录KNN算法流程使用OpenCV实现KNNOpenCV 是一个开源的跨平台计算机视觉库,它提供了各

OpenCV图像形态学的实现

《OpenCV图像形态学的实现》本文主要介绍了OpenCV图像形态学的实现,包括腐蚀、膨胀、开运算、闭运算、梯度运算、顶帽运算和黑帽运算,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起... 目录一、图像形态学简介二、腐蚀(Erosion)1. 原理2. OpenCV 实现三、膨胀China编程(

Android实现打开本地pdf文件的两种方式

《Android实现打开本地pdf文件的两种方式》在现代应用中,PDF格式因其跨平台、稳定性好、展示内容一致等特点,在Android平台上,如何高效地打开本地PDF文件,不仅关系到用户体验,也直接影响... 目录一、项目概述二、相关知识2.1 PDF文件基本概述2.2 android 文件访问与存储权限2.

基于Python和MoviePy实现照片管理和视频合成工具

《基于Python和MoviePy实现照片管理和视频合成工具》在这篇博客中,我们将详细剖析一个基于Python的图形界面应用程序,该程序使用wxPython构建用户界面,并结合MoviePy、Pill... 目录引言项目概述代码结构分析1. 导入和依赖2. 主类:PhotoManager初始化方法:__in

用js控制视频播放进度基本示例代码

《用js控制视频播放进度基本示例代码》写前端的时候,很多的时候是需要支持要网页视频播放的功能,下面这篇文章主要给大家介绍了关于用js控制视频播放进度的相关资料,文中通过代码介绍的非常详细,需要的朋友可... 目录前言html部分:JavaScript部分:注意:总结前言在javascript中控制视频播放

Python基于wxPython和FFmpeg开发一个视频标签工具

《Python基于wxPython和FFmpeg开发一个视频标签工具》在当今数字媒体时代,视频内容的管理和标记变得越来越重要,无论是研究人员需要对实验视频进行时间点标记,还是个人用户希望对家庭视频进行... 目录引言1. 应用概述2. 技术栈分析2.1 核心库和模块2.2 wxpython作为GUI选择的优

opencv图像处理之指纹验证的实现

《opencv图像处理之指纹验证的实现》本文主要介绍了opencv图像处理之指纹验证的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录一、简介二、具体案例实现1. 图像显示函数2. 指纹验证函数3. 主函数4、运行结果三、总结一、