advanced east_SpriteKit Advanced —如何构建2,5D游戏(第二部分)

advanced east

by Luke Konior

卢克·科尼尔(Luke Konior)

SpriteKit Advanced —如何构建2,5D游戏(第二部分) (SpriteKit Advanced — How to build a 2,5D game (Part II))

介绍 (Intro)

This article shows how to write basic shaders in the SpriteKit. It’s split into two parts: first we play, then we learn.

本文介绍如何在SpriteKit中编写基本着色器。 它分为两个部分:首先我们玩,然后学习。

It also contains information how to use SKAttribute and SKAttributeValue classes that were added in iOS SDK 10.0.

它还包含有关如何使用iOS SDK 10.0中添加的SKAttributeSKAttributeValue类的信息。

If you haven’t already read it, here’s part 1 of this article series.

如果您还没有阅读它,请参阅本系列文章的第1部分 。

准备项目 (Prepare the project)

Let’s get quick and dirty.

让我们变得快速又肮脏。

  • Open XCode 8 and create a new project from template: iOS > Game.

    打开XCode 8,然后从以下模板创建一个新项目:iOS>游戏。
  • Open the GameScene.sks and remove the label in the center of the screen.

    打开GameScene.sks并删除屏幕中心的标签。

  • Download this and put it inside Assets.xcassets

    下载此文件并将其放入Assets.xcassets

  • Name it “Trees”

    命名为“树木”
  • Open the GameScene.m

    打开GameScene.m

  • remove all instance variables

    删除所有实例变量
  • remove all methods

    删除所有方法

片段着色器 (The Fragment Shader)

Now we create an empty fragment shader In XCode:

现在,我们在XCode中创建一个空的片段shader

  • In the Project Navigator select Supporting Files

    在项目浏览器中,选择“支持文件”
  • Choose: File > New > File…

    选择:文件>新建>文件…
  • Select: Other > Empty

    选择:其他>空
  • Name it “myShader.fsh” and press Create.

    将其命名为“ myShader.fsh ”,然后按创建。

  • Put this inside:

    放在里面:
// currently a boring pass-thru shader void main( void ) { vec4 color = texture2D(utexture, vtexcoord); // here will emerge something worthy glFragColor = color;}

Above fragment shader does nothing perceptible. Quick explanation:

片段shader上方没有任何可察觉的内容。 快速说明:

  • void main()

    void main()

    this function gets called for each pixel of the sprite and outputs color for that pixel

    为精灵的每个像素调用此函数,并输出该像素的颜色

    Gets input data from surrounding globals and must set the

    从周围的全局变量获取输入数据,并且必须设置

    gl_FragColor variable

    gl_FragColor变量

  • vec2, vec3and vec4 are the types similar to C's: float array[2], float array[3] and float array[4]

    vec2vec3vec4与C的类型相似: float array[2]float array[3]float array[4]

  • u_texture is a texture ID

    u_texture是纹理ID

    Leave it alone :-)

    不要管它 :-)

  • v_tex_coord is a vec2 which contains our current position in texture

    v_tex_coord是一个vec2 ,其中包含我们当前的纹理位置

  • texture2D(tex , p) is a function that returns color from texture tex in point p as vec4

    texture2D(tex , p)是一个函数,将point p处纹理tex颜色返回为vec4

    which contains rgba

    其中包含rgba

  • gl_FragColor is an output color

    gl_FragColor是输出颜色

    We must assign it a

    我们必须给它分配一个

    vec4

    vec4

正在加载代码 (Loading code)

What’s left is the loading code.

剩下的就是加载代码。

  • Open the GameScene.m

    打开GameScene.m

  • add method -didMoveToView:

    添加方法-didMoveToView:

- (void)didMoveToView:(SKView *)view {	// 1. load the shader's source from myShaderFile.fsh	NSString *file = [[NSBundle mainBundle] pathForResource:@"myShader" ofType:@"fsh"];	NSString *sourceString = [NSString stringWithContentsOfFile:file encoding:NSUTF8StringEncoding error:nil];	    	// 2. create the shader	SKShader *shader = [SKShader shaderWithSource:sourceString];		// 3. assign the shader to a newly created sprite node	SKSpriteNode *spriteNode = [SKSpriteNode spriteNodeWithImageNamed:@"Trees"];	spriteNode.shader = shader;		// 4. finally add the sprite to the scene	[self addChild:spriteNode];}

Ensure that myShader.fsh figures in ProjectFile > Target > Build Phases > Copy Bundle Resources!

确保myShader.fsh数字位于ProjectFile>目标>构建阶段>复制捆绑资源中!

You may now run the project on the iOS device. There shall be no errors in the XCode’s console and you should see a screen similar to this below:

您现在可以在iOS设备上运行该项目。 XCode的控制台中应该没有错误,并且您应该看到类似于以下的屏幕:

让我们玩吧! (Let’s play a bit!)

Now is the fun part. We’ll replace the shader’s main function.

现在是有趣的部分。 我们将替换着色器的主要功能。

颜色为红色,保留alpha (Color with red with alpha preservation)

void main( void ){    vec4 color = texture2D(u_texture, v_tex_coord);    float alpha = color.a;    gl_FragColor = vec4(1,0,0, 1.0) * alpha; //google "premultiplied alpha"}

缩小2倍 (Scale down by 2x)

void main( void ){    vec4 color = texture2D(u_texture, v_tex_coord * 2.0);    gl_FragColor = color;}

1秒后交换颜色 (Swap colors after 1 second)

void main( void ){    vec4 color = texture2D(u_texture, v_tex_coord);    float alpha = color.a;    float phase = mod(u_time, 3);    vec3 outputColor = color.rgb;    if (phase < 1.0) {        outputColor = color.bgr;    } else if (phase < 2.0) {        outputColor = color.brg;    }    gl_FragColor = vec4(outputColor, 1.0) * alpha;}

随时间着色 (Colorize over time)

void main( void ){    vec4 color = texture2D(u_texture, v_tex_coord);    float alpha = color.a;    float r = (sin(u_time+ 3.14 * 0.00)+1.0)*0.5;    float g = (sin(u_time+ 3.14 * 0.33)+1.0)*0.5;    float b = (sin(u_time+ 3.14 * 0.66)+1.0)*0.5;    gl_FragColor = vec4(r,g,b, 1.0) * alpha;}

波浪 (Waves)

void main( void ){    float deltaX = sin(v_tex_coord.y*3.14*10 + u_time * 4)*0.01;    vec2 coord = v_tex_coord;    coord.x = coord.x + deltaX;    vec4 color = texture2D(u_texture, coord);    gl_FragColor = color;}

新属性 (New Attributes)

At WWDC 2016 Apple introduced an important update to SpriteKit — the SKAttribute and SKAttributeValue classes.

苹果在WWDC 2016上对SpriteKit进行了重要更新,即SKAttributeSKAttributeValue类。

Before this SDK update, if we wanted to pass custom parameters into the shader program, we had to pass the data through a uniform value.

在此SDK更新之前,如果我们想将自定义参数传递给shader程序,则必须通过统一值传递数据。

This had two serious drawbacks:

这有两个严重的缺点:

  • every uniform change caused shader recompilation

    每次统一更改都会导致着色器重新编译
  • shader program handled every sprite in the exact same way

    着色器程序以完全相同的方式处理每个精灵

For example: if we wanted to dye a group of sprites red, and one of them blue, we had two ways. First we create two separate SKShader instances and change our custom myColor uniform.

例如:如果我们想将一组精灵染成红色,而其中一个染成蓝色,则有两种方法。 首先,我们创建两个单独的SKShader实例,并更改我们的自定义myColor制服。

Second we make one shader instance and change its uniform which causes a recompilation.

其次,我们制作一个shader实例并更改其统一性,从而导致重新编译。

Both ways cannot be drawn on same pass. And the second one requires complex management code.

两种方法不能同时绘制。 第二个要求复杂的管理代码。

SDK 10.0 introduced the SKAttribute and SKAttributeValue classes. These two allow (finally!) passing data to the shader programs without recompilation. The usage algorithm is simple:

SDK 10.0引入了SKAttributeSKAttributeValue类。 这两个允许(最终!)将数据传递到着色器程序,而无需重新编译。 用法算法很简单:

  • The shader part:

    着色器部分:
  1. Create a shader program

    创建一个着色器程序

    Create a shader programSKShader

    创建一个着色器程序SKShader

  2. Create an array of SKAttributes

    创建一个SKAttributes数组

  3. Assign array of attributes to the shader program

    将属性数组分配给着色器程序
  • The sprite part:

    sprite部分:

  1. Assign the shader program to a sprite

    将着色器程序分配给精灵
  2. Assign a dictionary of SKAttributeValues

    分配SKAttributeValues字典

属性示例 (Example with attributes)

In the last example, we’ll add two more sprites. Every one of them will have the same shader program and will differ only in attributes. Let’s modify the -didMoveToView: inGameScene.m:

在最后一个示例中,我们将再添加两个精灵。 它们中的每一个将具有相同的着色器程序,并且仅在属性上有所不同。 让我们修改- didMoveToView: inGameScene.m:

- (void)didMoveToView:(SKView *)view {    NSString *file = [[NSBundle mainBundle] pathForResource:@"myShader" ofType:@"fsh"];    NSString *sourceString = [NSString stringWithContentsOfFile:file encoding:NSUTF8StringEncoding error:nil];    SKShader *shader = [SKShader shaderWithSource:sourceString];        // 1. Add a custom attribute to shader    SKAttribute *attrProgress = [SKAttribute attributeWithName:@"THE_MIGHTY_DARK_FACTOR" type:SKAttributeTypeFloat];    shader.attributes = @[attrProgress];    // 2. Create tree sprites    NSArray *trees = @[                       [self createTreeWithShader:shader mightyFactor:0.3f zPosition:1],                       [self createTreeWithShader:shader mightyFactor:0.6f zPosition:2],                       [self createTreeWithShader:shader mightyFactor:0.9f zPosition:3],                       ];    for (SKSpriteNode *tree in trees) {        [self addChild:tree];    }}- (SKSpriteNode*)createTreeWithShader:(SKShader*)shader mightyFactor:(CGFloat)mightyFactor zPosition:(CGFloat)zPosition {    SKSpriteNode *treeNode = [SKSpriteNode spriteNodeWithImageNamed:@"Trees"];    treeNode.shader = shader;    // 3. Fill the custom attribute on the sprite    treeNode.attributeValues = @{@"THE_MIGHTY_DARK_FACTOR": [SKAttributeValue valueWithFloat:mightyFactor]};    treeNode.zPosition = zPosition;return treeNode;}

… and the shader program:

…和着色器程序:

void main( void ){    vec4 color = texture2D(u_texture, v_tex_coord * (2.5 * THE_MIGHTY_DARK_FACTOR));    float alpha = color.a;    vec3 baseColor = color.rgb * THE_MIGHTY_DARK_FACTOR;    gl_FragColor = vec4(baseColor, 1.0) * alpha;}

... and see the parameterized result!

...并查看参数化结果!

注意事项 (Caveats)

  • The shader’s source code is typically loaded from a .fsh file to a plain NSString

    着色器的源代码通常从.fsh文件加载到纯NSString

    This code must compile on the target device during the runtime

    此代码必须在运行时在目标设备上编译

    no buildtime checks!

    没有构建时间检查!

  • Older devices may use different version of OpenGL ES so beware GLSL syntax differences!

    较旧的设备可能使用不同版本的OpenGL ES,因此请注意GLSL语法的不同!

    In Raft Challenge’s case there was the need to replace

    在Raft Challenge的情况下,需要更换

    __constant (valid in OpenGL ES 3.0) to const for OpenGL ES 2.0.

    __constant (在OpenGL ES 3.0中有效)以const表示OpenGL ES 2.0。

  • It’s a good idea to keep a reference to SKShader object somewhere and reuse it as frequently as needed to avoid visible frame rate drop

    最好在某个地方保留对SKShader对象的引用,并根据需要重复使用它,以避免可见的帧速率下降

    While allocation and shader compilation takes less than 1/60 sec, it may become a huge burden in render loop

    虽然分配和着色器编译花费的时间少于1/60秒,但它可能会成为渲染循环中的巨大负担

  • When using SpriteKit’s Texture Atlases be cautious of vtexcoord

    使用SpriteKit的Texture vtexcoord谨慎使用vtexcoord

    XCode may rotate some textures which swap

    XCode可能会旋转一些交换的纹理

    X and Y axis

    XY

    X and Y axisColor modification is safe, geometry is not

    XY颜色修改是安全的,几何图形不是

摘要 (Summary)

We learned by examples how to use fragment shaders in the Sprite Kit. We added parameters to sprites so our shader program can render every instance in a different way without any performance loss.

我们通过示例学习了如何在Sprite Kit中使用片段着色器。 我们向精灵添加了参数,以便我们的着色器程序可以以不同的方式呈现每个实例而不会造成性能损失。

The complete project is available for a download.

完整的项目可下载。

You can read part 3 of this series here.

您可以在这里阅读本系列的第3部分 。

About the author: Kamil Ziętek is an iOS Developer at www.allinmobile.co

关于作者:KamilZiętek是www.allinmobile.co上的iOS开发人员

翻译自: https://www.freecodecamp.org/news/spritekit-advanced-how-to-build-a-2-5d-game-part-ii-30ddb613b568/

advanced east

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/395384.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

html原生上传,一个基于HTML5及原生JS的文件上传组件--JohnUploader

运行效果图一、组件介绍基本特点基于HTML5的FileReader和FormData可以完成多文件选择&#xff0c;并预览完成文件的异步上传原生XHR对象&#xff0c;适配多浏览器代码class JohnUploader{url;fileField;vollay;/**** param url 文件上传的地址* param fileField 一个"文件…

[20170617]vim中调用sqlplus.txt

[20170617]vim中调用sqlplus.txt --//以前写过一篇emacs下调用sqlplus的文章,一直想学emacs,受限制自己掌握vim,对学习它没有兴趣,原链接如下: --//http://blog.itpub.net/267265/viewspace-1309032/ --//实际上vim也有插件连接数据库,我觉得不好用,一直没这样用. --//今天在整…

centos redis验证_centos7中安装、配置、验证、卸载redis

本文介绍在centos7中安装、配置、验证、卸载redis等操作&#xff0c;以及在使用redis中的一些注意事项。一 安装redis1 创建redis的安装目录利用以下命令&#xff0c;切换到/usr/local路径cd /usr/local键入以下命令&#xff0c;新建一个redis目录&#xff0c;用于放置redis软件…

实习生解雇_我们解雇了我们的顶尖人才。 我们做出的最佳决定。

实习生解雇by Jonathan Solrzano-Hamilton乔纳森索洛萨诺汉密尔顿(JonathanSolrzano-Hamilton) 我们解雇了我们的顶尖人才。 我们做出的最佳决定。 (We fired our top talent. Best decision we ever made.) “You will never be able to understand any of what I’ve create…

微信企业号第三方应用开发[二]——创建应用

在应用套件里添加应用 当你创建完应用套件后&#xff0c;需要在套件配置应用&#xff0c;应用的信息填写如下。 基本信息&#xff1a; 信息项要求及说明应用Logo应用的Logo&#xff0c;小于2M&#xff0c;640*640&#xff0c;在授权页会被用于展示。应用名称应用的名称&#xf…

es6新增的html标签,javascript – 如何导入已在html中的标签中定义的es6模块?

我可以在我的html文件me.html中定义一个模块&#xff1a;import Atom from ./atom.js;console.log("definition of getAtom")export default function getAtom(){return new Atom(atom);}console.log("exported getAtom")另见>是否可以将该“匿名”模块…

jQ效果:简单的手风琴效果

实现效果如图所示&#xff1a; html结构&#xff1a; <div class"item_box box10"><div class"item_box_wp"><div class"voice_2"><ul><li class"li1" id"li1"><div class"fold"…

golang 日志分析_容器日志采集利器:Filebeat深度剖析与实践

在云原生时代和容器化浪潮中&#xff0c;容器的日志采集是一个看起来不起眼却又无法忽视的重要议题。对于容器日志采集我们常用的工具有filebeat和fluentd&#xff0c;两者对比各有优劣&#xff0c;相比基于ruby的fluentd&#xff0c;考虑到可定制性&#xff0c;我们一般默认选…

机器学习做自动聊天机器人_建立聊天机器人需要什么? 让我们找出答案。

机器学习做自动聊天机器人by Vanco Stojkov通过Vanco Stojkov 建立聊天机器人需要什么&#xff1f; 让我们找出答案。 (What does it take to build a chatbot? Let’s find out.) Without any delay, the image below shows what we are building:没有任何延迟&#xff0c;下…

UVA 11582 Colossal Fibonacci Numbers!【数学】

大一刚开始接触ACM就买了《算法竞赛入门经典》这本书&#xff0c;当时只能看懂前几章&#xff0c;而且题目也没做&#xff0c;粗鄙地以为这本书不适合自己。等到现在快大三了再回过头来看&#xff0c;发现刘老师还是很棒的&#xff01; 扯远了。。。 题意&#xff1a;问f[a^b]%…

Codeforces 919D Substring (拓扑图DP)

手动博客搬家: 本文发表于20180716 10:53:12, 原地址https://blog.csdn.net/suncongbo/article/details/81061500 给定一个\(n\)个点\(m\)条边的有向图&#xff08;不一定无环&#xff09;&#xff0c;每个点上有一个小写字母。要找一条路径&#xff0c;使得路径上出现次数最多…

layui自定义查询条件html页面,Layui的数据表格+springmvc实现搜索功能的例子_飛雲_前端开发者...

如下所示&#xff1a;主要在前端页面加&#xff1a;搜索ID&#xff1a;useridcontent搜索在reload:function () {var keyWord$("#keyWord").val();var keyType$("#key_type option:selected").val();table.reload(contenttable,{method:post,where:{keyWor…

mysql+keepalived 双主热备高可用

理论介绍&#xff1a;我们通常说的双机热备是指两台机器都在运行&#xff0c;但并不是两台机器都同时在提供服务。当提供服务的一台出现故障的时候&#xff0c;另外一台会马上自动接管并且提供服务&#xff0c;而且切换的时间非常短。MySQL双主复制&#xff0c;即互为Master-Sl…

java ldap userpassword 解密_Spring Boot中使用LDAP来统一管理用户信息

LDAP简介LDAP(轻量级目录访问协议&#xff0c;Lightweight Directory Access Protocol)是实现提供被称为目录服务的信息服务。目录服务是一种特殊的数据库系统&#xff0c;其专门针对读取&#xff0c;浏览和搜索操作进行了特定的优化。目录一般用来包含描述性的&#xff0c;基于…

第三章之枚举、注解

2019-01-22内容&#xff1a;枚举、注解一、自定义一个枚举类1 public class TestSeason {2 3 public static void main(String[] args) {4 Season spring Season.Spring;5 System.out.println(spring);6 }7 }8 public class Season {9 //将属性定…

html打开后默认浏览器页面,使用VBA打开默认浏览器中的html页面?

您可以使用Windows API函数ShellExecute来执行此操作&#xff1a;Option ExplicitPrivate Declare Function ShellExecute _Lib "shell32.dll" Alias "ShellExecuteA" ( _ByVal hWnd As Long, _ByVal Operation As String, _ByVal Filename As String, _Op…

数据科学r语言_您应该为数据科学学习哪些语言?

数据科学r语言Data science is an exciting field to work in, combining advanced statistical and quantitative skills with real-world programming ability. There are many potential programming languages that the aspiring data scientist might consider specializi…

Linux平台不同解压缩命令的使用方法

作者&#xff1a;郭孝星 微博&#xff1a;郭孝星的新浪微博 邮箱&#xff1a;allenwells163.com 博客&#xff1a;http://blog.csdn.net/allenwells github&#xff1a;https://github.com/AllenWell 一 .tar 解包 tar xvf FileName.tar 打包 tar cvf FileName.tar DirName 注意…

unity中怎么做河流_【干货】工作中怎么做工业设计的?(一)

最近在找工作&#xff0c;一直在看招聘信息。看到工业设计工资还是蛮高的。应届毕业生一般是4-6K&#xff0c;1-3年工作经验是6-8K&#xff0c;3年以后的差不多是8K以上了。我没有嫉妒羡慕恨&#xff0c;发誓&#xff0c;真的没有。工业设计已经被重视&#xff0c;未来的道路会…

[易学易懂系列|golang语言|零基础|快速入门|(一)]

golang编程语言&#xff0c;是google推出的一门语言。 主要应用在系统编程和高性能服务器编程&#xff0c;有广大的市场前景&#xff0c;目前整个生态也越来越强大&#xff0c;未来可能在企业应用和人工智能等领域占有越来越重要的地位。 本文章是【易学易懂系列|编程语言入门】…