前言

Pixi.js使用WebGL,是一个超快的HTML5 2D渲染引擎。作为一个Javascript的2D渲染器,Pixi.js的目标是提供一个快速的、轻量级而且是兼任所有设备的2D库。提供无缝 Canvas 回退,支持主流浏览器,包括桌面和移动。 Pixi渲染器可以开发者享受到硬件加速,但并不需要了解WebGL。 网上暂时没有找到关于

pixi.js的中文说明文档,我这里仅是记录项目中使用的经验,即记录利用pixi在制作精灵动画时的一些方法。

=========================================================================

概述

需要说明的是,有些示例需要使用http://方式才能有效果,也就是说file://的方式有些示例是无法显示的。

下面我们来看一下,用pixi创建一个精灵要经历哪些步骤:

创建一个PIXI的实例,并展示出来,通常需要以下几步:

  1. 创建一个舞台 (stage)
  2. 创建一个画布 (render)
  3. 把画布插入DOM中
  4. 创建一个精灵 (sprite)
  5. 把精灵加入画布
  6. 把画布加入舞台
  7. 刷新舞台

概念解释: 

舞台(stage):

// A Stage represents the root of the display tree. Everything connected to the stage is rendered
所有要渲染的对象都必须连接到舞台中才能被显示出来,舞台处于整个树形展示结构的最底层,可以理解为背景
 
 demo :

 var stage = new PIXI.Stage(0xFFFFFF); //接收一个16进制的值,用于背景的颜色

  画布(renderer):
 选用canvas或webGL进行渲染的一个区域
 demo:

 var renderer = PIXI.autoDetectRenderer(400, 300); 


它的后面还有两个可选参数,分别 用来指定现有的canvas元素作为容器,canvas背景是否透明,

材质(texture):

可以理解成一种承载图片的结构,它本身不能直接用于显示,需要通过精灵(sprite)才能显示,有点类似于dom中的临时碎片(DocumentFragment)

demo:

  var texture = PIXI.Texture.fromImage("bunny.png");

精灵(sprite):

可以直接用于舞台显示的对象,可以理解为DOM中的element.
精灵可以直接用图片创建,也可以先创建材质,再用材质创建精灵

  demo :

var sprite = new PIXI.Sprite.fromImage('assets/image.png');//直接由图片创建
//...
var texture = PIXI.Texture.fromImage("bunny.png");
var sprite = new PIXI.Sprite(texture);//由材质创建

//  sprite.position.x = 30; //没有单位

//  sprite.position.y = 20; // 这就和给element设置样式是一样的道理。

 事件(event):
PIXI库在精灵和舞台上提供了事件,用于交互.
 demo:

 stage.click = function(data){
var event = data.originalEvent }
sprite.click = function(data){
var event = data.originalEvent
var target = event.target
//阻止默认行为,sprite.buttonMode = false;
event.preventDefault();
}

此外,还有.mousedown,.mouseover,.mouseout,.mousemove,.mouseup ,.mouseupoutside , .touchstart,.touchend,.tap

注意事项:

  1. 对于有事件的精灵,通常需要设置sprite.interactive = true , 对于作按钮用的精灵,需要设置sprite.buttonMode = true;
  2. 与DOM的事件不同的是,精灵中的事件,是进行了二次封装的,
  3. 通常需要用var event = data.originalEvent来与普通dom中的event对象保持一至

不过又不完全相同,如data.originalEvent.stopPropagation()就不起作用.

关于stopPropagation的问题,我问过pixijs的开发者,他答复说stopPropagation是dom中的概念,在canvas/webGL中是没有的,所以不能说是pixi的问题。

  下面使用PIXI库显示一张图片,简单的演示下pixi的用法:

var stage = new PIXI.Stage();
var renderer = PIXI.autoDetectRenderer(400, 300); document.body.appendChild(renderer.view);
var sprite = new PIXI.Sprite.fromImage('bunny.png');
stage.addChild(sprite);

直接贴出上面的代码,在网页上只能看到漆黑的背景, 这是因为精灵的创建是异步的,必须等加载完图片之后才能把精灵加入舞台中
 
我在这里给出两种解决方式:

1:不断的去刷新舞台:

requestAnimFrame(animate);
function animate(){
renderer.render(stage);
requestAnimFrame(animate);
}

2. 在图片加载完成的回调中执行。这仅限于不需要刷新的精灵、

 sprite.texture.baseTexture.on('loaded',function(){
renderer.render(stage);
})

 

在项目中实践的时候发现用基于baseTexture的loaded事件有两个问题
1:如果图片加载失败怎么办?
2:同时加载多张图片的时候,有些没有触发图片的loaded回调

经过实践,找到一个更加稳妥的办法:

 var img = new Image();
img.src = 'bunny.png';
img.onload = function(){   
var baseTexture = new PIXI.BaseTexture(this);
  var texture = new PIXI.Texture(baseTexture);   
var sprite = new PIXI.Sprite(texture);   
stage.addChild(sprite);   
renderer.render(stage);
}

此方案对于批量加载图片也是适用的。最后再重复一遍,请用localhost://xxxxxx的方式进浏览

细心的比较也许你会发现,我和官方的示例不相同,官方使用了requestAnimFrame( animate ),而然我这里没有用,为什么?因为我只是要显示一张图片而已,没有必要不断的去刷新舞台。requestAnimFrame( animate );这个是在精灵有变化的时候才需要用到。当然,如果不考虑性能,对一张静态的图不断的刷新也没有什么问题。

完整的示例:

        var stage = new PIXI.Stage(0xFFFFFF);
var renderer = PIXI.autoDetectRenderer(400, 300); document.body.appendChild(renderer.view); var sprite = new PIXI.Sprite.fromImage('bunny.png');
stage.addChild(sprite); sprite.texture.baseTexture.on('loaded',function(){
renderer.render(stage);
}) sprite.interactive = true stage.click = stage.tap = function(data){
console.log(data.originalEvent.target);
//alert('you hit the stage')
} sprite.mousedown = sprite.touchstart =function(data){
// stop the default event...
data.originalEvent.preventDefault();
this.data = data;
this.alpha = 0.9;
this.dragging = true;
console.log('mousedown');
} sprite.mouseup = sprite.mouseupoutside = sprite.touchend = function(data){
this.alpha = 1
this.dragging = false;
// set the interaction data to null
this.data = null;
console.log('mouseup');
}; sprite.mousemove = sprite.touchmove = function(data){
if(this.dragging){
var newPosition = this.data.getLocalPosition(this.parent);
this.position.x = newPosition.x;
this.position.y = newPosition.y;
console.log('mousemove')
renderer.render(stage);
}
} //requestAnimFrame(animate);
function animate(){
renderer.render(stage);
requestAnimFrame(animate);
}

精灵放大2倍

 var sprite = new PIXI.Sprite.fromImage('bunny.png');
var sx = sprite.scale.x;
var px = sprite.position.x; sprite.scale.x += 1;
sprite.position.x -= px;
//Y方向同理
...
//刷新一下
requestAnimFrame(animate);

精灵的层级问题

在canvas中,由于不能用css的层级样式表来控制层级关系,所以只能另想办法进行模拟。pixi.js中没有提供现成的方法,不过我们研究发现,只要控制精灵加入画布的顺序就可以实现类似的层级效果。它的规则是这样的,先加入画布的,出现在最下面,后加入的,出现在最上面。于是自然的就想到了先用一个数组收集这些精灵,排好序之后再循环加入画布。可是问题就来了,有时候,我们事先往往不能确定层级,或者有时候是用户触发某一个动作之后才改变层级的,这怎么实现呢?这个我想到了给精灵人为的加一个zindex属性。然后遍历这个属性,进行排序,最后再刷新一下舞台。

    /**
* 对象按指定的属性排序
* @param {array} arr 对象集合
* @return null
*/
function pixiSort(arr){
var ln = arr.length;
var tmp;
for(var i = 0;i<ln-1;i++){
for(var j=i+1;j<ln;j++){
tmp = arr[i];
if(arr[i].sortIndex > arr[j].sortIndex){
arr[i] = arr[j];
arr[j] = tmp;
}
}
}
}

受html dom操作的影响,有时候我们需要精灵按照指定的层级进行排列显示,比如给精灵指定index属性。要做到对zindex的控制,可以用到如下方法

function updateZindex(ob,stage){
Object.defineProperty(ob,'zIndex',{
set : function(value){
if(this.sortIndex !== value){
this.sortIndex = value;
pixiSort(stage.children);
}
},
get : function(){
return this.sortIndex;
}
});
}

说白了,就是控制精灵的先后顺序来达到显示层级的效果。

最后是如何销毁:

//移除全部精灵
stage.removeChildren()
//移除整个舞台的引用
stage.removeStageReference()

以上创建的是建静止的精灵,及在精灵上邦定事件。这个在官方没有找到示例,特补充一下。(或许pixi就是为动画而生的,所以没有考虑这种简单的需求了)

如果需要动画效果的精灵,官方提供了多种方法。这里就不再细述了。我目前正在研究这一块,有兴趣的可以和我(@frog)一起交流!

在线资源

Pixi.js的GitHub主页上,列举了很多在线Demo。

GitHub:https://github.com/GoodBoyDigital/pixi.js/

官网: http://www.pixijs.com/

API :http://pixijs.download/release/docs/index.html

官方教程:https://github.com/kittykatattack/learningPixi

https://github.com/GoodBoyDigital/pixi.js

 本篇为原创博文,转载请注明出处.  278500368@qq.com

pixi.js教程中文版--基础篇的更多相关文章

  1. .net 开源模板引擎jntemplate 教程:基础篇之语法

    一.基本概念 上一篇我们简单的介绍了jntemplate并写了一个hello world(如果没有看过的,点击查看),本文将继续介绍jntemplate的模板语法. 我们在讲解语法前,首先要了解一下标 ...

  2. .net 开源模板引擎jntemplate 教程:基础篇之在ASP.NET MVC中使用Jntemplate

    在ASP.NET MVC 中使用Jntemplate 上一篇我们详细介绍了jntemplate的标签语法,本篇文章将继续介绍如何在ASP.NET MVC 中使用Jntemplate. 一.使用Jnte ...

  3. 【MongoDB】NoSQL Manager for MongoDB 教程(基础篇)

    前段时间,学习了一下mongodb,在客户端工具方面,个人认为 NoSQL Manager for MongoDB 是体验比较好的一个,功能也较齐全.可惜在找教程的时候,发现很难找到比较详细的教程,也 ...

  4. vscode教程(基础篇)

    转载:https://segmentfault.com/a/1190000017949680 本文主要介绍vscode在工作中常用的快捷键及插件,目标在于提高工作效率 本文的快捷键是基于mac的,wi ...

  5. 【转】Git使用教程之基础篇

    Git使用教程 一:Git是什么? Git是目前世界上最先进的分布式版本控制系统. 二:SVN与Git的最主要的区别? SVN是集中式版本控制系统,版本库是集中放在中央服务器的,而干活的时候,用的都是 ...

  6. windows安装TortoiseGit详细使用教程【基础篇】

    标签:tortoisegit 环境:win8.1 64bit 安装准备: 首先你得安装windows下的git msysgit1.9.5 安装版本控制器客户端tortoisegit  tortoise ...

  7. Git安装和TortoiseGit详细使用教程【基础篇】

    标签:tortoisegit 环境:win8.1 64bit 安装准备: 首先你得安装windows下的git msysgit1.9.5 安装版本控制器客户端tortoisegit  tortoise ...

  8. SpringBoot图文教程「概念+案例 思维导图」「基础篇上」

    有天上飞的概念,就要有落地的实现 概念+代码实现是本文的特点,教程将涵盖完整的图文教程,代码案例 每个知识点配套自测面试题,学完技术自我测试 本文初学向,所以希望文中所有的代码案例都能敲一遍 大哥大姐 ...

  9. js基础到精通全面教程--JS教程

    适合阅读范围:对JavaScript一无所知-离精通只差一步之遥的人 基础知识:HTML JavaScript就这么回事1:基础知识 1 创建脚本块 1: <script language=”J ...

随机推荐

  1. Banner插件版

    条件:使用JQ. 使用情况:当目标元素调用该插件时,插件产生的元素会替换该目标元素,并且在目标元素位置生成.需要输入一组图片地址数组(对象还没有实现,慢慢改善)默认宽高是600*400,可在后面的参数 ...

  2. Python之路【第七篇】python基础 之socket网络编程

    本篇文章大部分借鉴 http://www.cnblogs.com/nulige/p/6235531.html python socket  网络编程 一.服务端和客户端 BS架构 (腾讯通软件:ser ...

  3. Java ArrayList和Vector、LinkedList与ArrayList、数组(Array)和列表集合(ArrayList)的区别

    ArrayList和Vector的区别ArrayList与Vector主要从二方面来说.  一.同步性:   Vector是线程安全的,也就是说是同步的,而ArrayList是线程序不安全的,不是同步 ...

  4. 初学者-PHP笔记

    PHP介绍 PHP 是 "PHP Hypertext Preprocessor" 的首字母缩略词 PHP 是一种被广泛使用的开源脚本语言 PHP 脚本在服务器上执行 PHP 没有成 ...

  5. 关于IOS浏览器:document,body的click事件触发规则

    今天做了个手机页面,点击某个按钮->弹出菜单,再点击菜单以外的任意位置->关闭菜单,在其他浏览器里面没有问题,但是在IOS浏览器中并不会关闭. 网上解决这个bug的帖子很多,这篇帖子主要是 ...

  6. PHP基础知识之魔术方法

    __construct(), __destruct(), __call(), __callStatic(), __get(), __set(), __isset(), __unset(), __sle ...

  7. Ajax的js库分析简化版

    Ajax jquery的库的简化版本 (function(){    //面向外界的唯一变量接口!    var myajax = window.myajax = {};    //作者.版本号等等信 ...

  8. jQuery中设置form表单中action值与js有什么不同。。。。

    jQuery中设置form表单中action值与js有什么不同.... HTML代码如下: <form action="" method="post" i ...

  9. Hadoop学习笔记—6.Hadoop Eclipse插件的使用

    开篇:Hadoop是一个强大的并行软件开发框架,它可以让任务在分布式集群上并行处理,从而提高执行效率.但是,它也有一些缺点,如编码.调试Hadoop程序的难度较大,这样的缺点直接导致开发人员入门门槛高 ...

  10. 扩展Bootstrap Tooltip插件使其可交互

    最近在公司某项目开发中遇见一特殊需求,请笔者帮助,因此有了本文的插件.在前端开发中tooltip是一个极其常用的插件,它能更好向使用者展示更多的文档等帮助信息.它们通常都是一些静态文本信息.但同事他们 ...