服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - IOS - 在iOS中使用OpenGL ES实现绘画板的方法

在iOS中使用OpenGL ES实现绘画板的方法

2021-06-01 15:56Lyman''s Blog IOS

这篇文章主要介绍了在iOS中使用OpenGL ES实现绘画板的方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

今天我们使用 opengl es 来实现一个绘画板,主要介绍在 opengl es 中绘制平滑曲线的实现方案。

首先看一下最终效果:

在iOS中使用OpenGL ES实现绘画板的方法

在 ios 中,有很多种方式可以实现一个绘画板,比如我的另外一个项目 mfpaintview 就是基于 coregraphics 实现的。

然而,使用 opengl es 来实现可以获得更多的灵活性,比如我们可以自定义笔触的形状,这是其他实现方式做不到的。

我们知道,opengl es 中只有 点、直线、三角形 这三种图元。因此, 怎么在 opengl es 中绘制曲线 ,是我们第一个要解决的问题,也是最复杂的问题。

我们会使用比较大的篇幅来讲解这个问题。至于绘画板的其他功能实现,并不是说不重要,只是说其他的绘画板实现方式,也会有类似的逻辑,所以这部分会放在最后再简单介绍一下。

一、怎么绘制曲线

在 opengl es 中绘制曲线的方式,就是 将曲线拆分成点序列来绘制

因为要绘制点,所以我们采取的是 点图元 。即我们要把顶点数据当成 来绘制,并且每个点都要绘制出笔触的纹理。关键步骤如下:

指定图元类型:

?
1
gldrawarrays(gl_points, 0, self.vertexcount);

顶点着色器:

?
1
2
3
4
5
6
7
8
attribute vec4 position;
 
uniform float size;
 
void main (void) {
  gl_position = position;
  gl_pointsize = size;
}

片段着色器:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
precision highp float;
 
uniform float r;
uniform float g;
uniform float b;
uniform float a;
 
uniform sampler2d texture;
 
void main (void) {
  vec4 mask = texture2d(texture, vec2(gl_pointcoord.x, 1.0 - gl_pointcoord.y));
  gl_fragcolor = a * vec4(r, g, b, 1.0) * mask;
}

这里的关键点在于 gl_pointcoord 这个内置变量,当我们使用点图元的时候,可以通过这个变量获取到 当前像素在点图元中的归一化坐标

但是这个坐标的原点是在左上角,这和纹理坐标在竖直方向上是相反的。所以从纹理读取颜色的时候,要做一个 y 坐标的转换。

接下来,我们通过 uitouch 来获取触摸点的位置,然后算出归一化的顶点坐标。

?
1
2
3
4
5
- (void)touchesmoved:(nsset<uitouch *> *)touches withevent:(uievent *)event {
  [super touchesmoved:touches withevent:event];
  
  [self addpointwithtouches:touches];
}

但是由于 ios 系统触摸事件的派发频率有限,我们最终得到的只能是稀疏的点。如下图所示,每个触摸点之间的间隔会比较大。

在iOS中使用OpenGL ES实现绘画板的方法

二、怎么绘制密集的点

很容易想到,只需要在两个点之间,按照一定的密度进行插值,就可以绘制出连续的轨迹。

在iOS中使用OpenGL ES实现绘画板的方法

但是很明显,我们的绘制结果是折线,并不平滑。

三、怎么使曲线变平滑

解决点连接不平滑的问题,一般是使用贝塞尔曲线。这种方案在 mfpaintview 中也得到了很好的应用。

具体的做法是使用 两个顶点间的中点一个顶点 ,来构造一条贝塞尔曲线。如下图,图中的 3 个 红点 被用来构造一条贝塞尔曲线。

在iOS中使用OpenGL ES实现绘画板的方法

于是,我们的问题就变成了 怎么在 opengl es 中绘制贝塞尔曲线 。相当于已知贝塞尔曲线的 3 个关键点,反向来求曲线上的点序列。

我们知道贝塞尔曲线的方程是 p = (1 - t)^2 * p0 + 2 * t * (1 - t) * p1 + t^2 * p2t 是唯一的变量,其取值范围是 0 ~ 1

所以我们可以采取线性取值的方式,每一条贝塞尔曲线取 n 个点( n 是个确定的常量)。只要依次往方程中代入 1 / n 、 2 / n 、 ... n / n ,就可以得到一个点序列。

在iOS中使用OpenGL ES实现绘画板的方法

先将 n 取一个比较小的值,这样比较容易看出存在的问题。我们发现, 点序列的间隔并不均匀 。原因有两个:

  • 不同贝塞尔曲线的长度不一样,使用同一个 n 值,算出来的点的疏密程度肯定不同。
  • 由于贝塞尔曲线随着 t 增长,曲线长度的增长并不是线性的。按照我们上面的算法,最终会得到的结果是 两头比较稀疏,中间比较密集

四、怎么生成均匀的点序列

贝塞尔曲线生成均匀的点序列,涉及到了一个经典的「贝塞尔曲线匀速运动」问题。

这个问题的推导和计算比较复杂。如果你有兴趣,可以阅读一下文末的两篇文章。由于我还不能完全领悟,就不在这里误导大家了。

简单来说,就是我们通过一系列的骚操作,封装了一个方法,只需要传入贝塞尔曲线的 3 个关键点和笔触尺寸,就可以获取均匀的点序列。

?
1
2
3
4
+ (nsarray <nsvalue *>*)pointswithfrom:(cgpoint)from
                  to:(cgpoint)to
                control:(cgpoint)control
               pointsize:(cgfloat)pointsize;

下面我们固定贝塞尔曲线的 起始点控制点 ,只移动 终止点 ,来验证一下这个方法是否可靠。

在iOS中使用OpenGL ES实现绘画板的方法

可以看到,在移动过程中,点和点的距离基本是保持一致的,并且是均匀的。通过这个「神奇」的方法,我们终于画出了平滑且均匀的曲线。

在iOS中使用OpenGL ES实现绘画板的方法

五、绘画板功能实现

终于讲完了最麻烦的部分,接下来简单介绍一下绘画板基本功能的实现。

1、颜色混合

在以往的例子中,我们在开始一次渲染之前,都会调用 glclear(gl_color_buffer_bit) 来清除画布,因为我们不希望保留上次的渲染结果。

但是对于一个绘画板来说,我们要不断地往画布上画东西,所以是希望保留上次结果的。因此,在绘制之前不能执行清除的操作。

另外,由于我们的画笔可能是半透明的,所以新绘制的颜色需要和画布上已经存在的颜色进行混合。因此在绘制开始之前,需要开启混合选项。

?
1
2
glenable(gl_blend);
glblendfunc(gl_one, gl_one_minus_src_alpha);

2、笔触调整

笔触有 3 个属性可以调整: 颜色、尺寸、形状 。它们本质上都是对点图元的调整,通过 uniform 变量的形式,将颜色、尺寸、纹理传入着色器并应用。

3、橡皮擦

glpaintview 在初始化的时候,需要传入一个背景色参数,当用户切换到橡皮擦功能的时候,内部只是单纯地将画笔的颜色切换成背景色,于是就产生了橡皮擦的效果。

4、撤销重做

撤销重做功能需要依赖两个栈来实现。我们把用户的手指从 按下屏幕到离开屏幕 这一过程中产生的数据,定义为一个操作对象,这个操作对象保存了归一化后的点序列,以及点的属性。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
@interface mfpaintmodel : nsobject
 
/// 笔刷尺寸
@property (nonatomic, assign) cgfloat brushsize;
/// 笔刷颜色
@property (nonatomic, strong) uicolor *brushcolor;
/// 笔刷模式
@property (nonatomic, assign) glpaintviewbrushmode brushmode;
/// 笔触纹理图片文件名
@property (nonatomic, copy) nsstring *brushimagename;
/// 点序列
@property (nonatomic, copy) nsarray<nsvalue *> *points;
 
@end

撤销重做的代码实现大概像这样子:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
- (void)undo {
  if ([self.operationstack isempty]) {
    return;
  }
  mfpaintmodel *model = self.operationstack.topmodel;
  [self.operationstack popmodel];
  [self.undooperationstack pushmodel:model];
  
  [self redraw];
}
 
- (void)redo {
  if ([self.undooperationstack isempty]) {
    return;
  }
  mfpaintmodel *model = self.undooperationstack.topmodel;
  [self.undooperationstack popmodel];
  [self.operationstack pushmodel:model];
  
  [self drawmodel:model];
}

需要注意的是,由于 撤销操作 需要先清除画布,所以每次都需要重绘。而 重做操作 可以利用上次绘制的结果,所以每次只需要绘制一个步骤即可。

源码

请到 github 上查看完整代码。

到此这篇关于在ios中使用opengl es实现绘画板的方法的文章就介绍到这了,更多相关ios 绘画板内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:http://www.lymanli.com/2020/01/04/ios-opengles-paint/

延伸 · 阅读

精彩推荐
  • IOSiOS中tableview 两级cell的展开与收回的示例代码

    iOS中tableview 两级cell的展开与收回的示例代码

    本篇文章主要介绍了iOS中tableview 两级cell的展开与收回的示例代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧...

    J_Kang3862021-04-22
  • IOS解析iOS开发中的FirstResponder第一响应对象

    解析iOS开发中的FirstResponder第一响应对象

    这篇文章主要介绍了解析iOS开发中的FirstResponder第一响应对象,包括View的FirstResponder的释放问题,需要的朋友可以参考下...

    一片枫叶4662020-12-25
  • IOSiOS布局渲染之UIView方法的调用时机详解

    iOS布局渲染之UIView方法的调用时机详解

    在你刚开始开发 iOS 应用时,最难避免或者是调试的就是和布局相关的问题,下面这篇文章主要给大家介绍了关于iOS布局渲染之UIView方法调用时机的相关资料...

    windtersharp7642021-05-04
  • IOSiOS 雷达效果实例详解

    iOS 雷达效果实例详解

    这篇文章主要介绍了iOS 雷达效果实例详解的相关资料,需要的朋友可以参考下...

    SimpleWorld11022021-01-28
  • IOSIOS开发之字典转字符串的实例详解

    IOS开发之字典转字符串的实例详解

    这篇文章主要介绍了IOS开发之字典转字符串的实例详解的相关资料,希望通过本文能帮助到大家,让大家掌握这样的方法,需要的朋友可以参考下...

    苦练内功5832021-04-01
  • IOS关于iOS自适应cell行高的那些事儿

    关于iOS自适应cell行高的那些事儿

    这篇文章主要给大家介绍了关于iOS自适应cell行高的那些事儿,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的...

    daisy6092021-05-17
  • IOSIOS 屏幕适配方案实现缩放window的示例代码

    IOS 屏幕适配方案实现缩放window的示例代码

    这篇文章主要介绍了IOS 屏幕适配方案实现缩放window的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要...

    xiari5772021-06-01
  • IOSiOS通过逆向理解Block的内存模型

    iOS通过逆向理解Block的内存模型

    自从对 iOS 的逆向初窥门径后,我也经常通过它来分析一些比较大的应用,参考一下这些应用中某些功能的实现。这个探索的过程乐趣多多,不仅能满足自...

    Swiftyper12832021-03-03