Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/dntzhang/cax
Canvas 渲染引擎,支持 SVG,兼容了小程序、小游戏和 Web
https://github.com/dntzhang/cax
2d canvas minigame miniprogram render svg weapp web wegame
Last synced: 2 days ago
JSON representation
Canvas 渲染引擎,支持 SVG,兼容了小程序、小游戏和 Web
- Host: GitHub
- URL: https://github.com/dntzhang/cax
- Owner: dntzhang
- Created: 2012-09-29T02:49:41.000Z (over 12 years ago)
- Default Branch: master
- Last Pushed: 2023-07-26T01:14:13.000Z (over 1 year ago)
- Last Synced: 2025-01-15T13:00:55.821Z (4 days ago)
- Topics: 2d, canvas, minigame, miniprogram, render, svg, weapp, web, wegame
- Language: JavaScript
- Homepage:
- Size: 6.3 MB
- Stars: 2,082
- Watchers: 139
- Forks: 322
- Open Issues: 49
-
Metadata Files:
- Readme: README.CN.md
- Changelog: change-log.md
Awesome Lists containing this project
- awesome-github-star - cax
README
简体中文 | [English](./README.md)
# Cax [![](https://img.shields.io/npm/v/cax.svg)](https://www.npmjs.com/package/cax)
> Canvas 渲染引擎,支持SVG,兼容了小程序、小游戏和 Web
* [Wechart by Cax![](./asset/hot.png)](https://github.com/dntzhang/wechart)
* Web DEMO & [→ Source Code](https://github.com/dntzhang/cax/tree/master/packages/cax/examples)
* [Simple](https://dntzhang.github.io/cax)
* [Animation](https://dntzhang.github.io/cax/packages/cax/examples/to/)
* [Clip](https://dntzhang.github.io/cax/packages/cax/examples/clip/)
* [Clip Transform](https://dntzhang.github.io/cax/packages/cax/examples/clip-transform/)
* [Clip Transition](https://dntzhang.github.io/cax/packages/cax/examples/clip-transition/)
* [To2To Animate](https://dntzhang.github.io/cax/packages/cax/examples/to-animate/)
* [Swing](https://dntzhang.github.io/cax/packages/to/examples/swing/)
* [Cax + Matter](https://dntzhang.github.io/cax/packages/cax/examples/matter/)
* [Pie Chart](https://dntzhang.github.io/wechart/packages/pie/examples/simple/)
* [To + Shape](https://dntzhang.github.io/cax/packages/cax/examples/to-shape/)
* [Vision](https://dntzhang.github.io/cax/packages/cax/examples/vision/)
* [Cache](https://dntzhang.github.io/cax/packages/cax/examples/cache/)
* [Filter](https://dntzhang.github.io/cax/packages/cax/examples/filter/)
* [SVG](https://dntzhang.github.io/wechart/packages/path/examples/man/)
* [Graphics](https://dntzhang.github.io/cax/packages/cax/examples/graphics/)
* [Composite Operation](http://dntzhang.github.io/cax/packages/cax/examples/composite-operation/)
* [Cax + Dragonbones](https://dntzhang.github.io/wechart/packages/cax-dragonbones/examples/simple/index.html)## 特性
* Learn Once, Write Anywhere(小程序、小游戏、PC Web、Mobile Web)
* Write Once, Run Anywhere(小程序、小游戏、Web只需要修改`new Stage`传入参数即可)
* 支持小程序、小游戏以及 Web 浏览器渲染
* 小程序、小游戏和 Web 拥有相同简洁轻巧的 API
* 高性能且松耦合的渲染架构
* 超轻量级的代码体积
* 支持 Canvas 元素管理
* 支持 Canvas 元素事件体系
* 图灵完备的 group 嵌套体系
* 内置 [to2to](https://github.com/dntzhang/cax/tree/master/packages/to) 的跨平台运动引擎
* 支持可以变形的 clip 裁剪体系
* 内置文本、位图、序列帧、绘图对象和多种矢量绘制对象
* 支持 SVG Path 渲染
* 支持几乎全部的 CSS 滤镜和其他常用滤镜
* 内置图片加载器
---- [一分钟入门小程序 cax 使用](#一分钟入门小程序-cax-使用)
- [一分钟入门小游戏 cax 使用](#一分钟入门小游戏-cax-使用)
- [一分钟入门 Web cax 使用](#一分钟入门-web-cax-使用)
- [内置对象](#内置对象)
- [Group](#group)
- [Group 方法](#group-方法)
- [add](#add)
- [remove](#remove)
- [empty](#empty)
- [replace](#replace)
- [Stage](#stage)
- [Stage 方法](#stage-方法)
- [update](#update)
- [scaleEventPoint](#scaleeventpoint)
- [Stage 属性](#stage-属性)
- [disableMoveDetection](#disablemovedetection)
- [moveDetectionInterval](#movedetectioninterval)
- [Bitmap](#bitmap)
- [Sprite](#sprite)
- [Sprite 方法](#sprite-方法)
- [gotoAndPlay](#gotoandplay)
- [gotoAndStop](#gotoandstop)
- [gotoAndPlayOnce](#gotoandplayonce)
- [Text](#text)
- [Text 方法](#text-方法)
- [getWidth](#getwidth)
- [Graphics](#graphics)
- [Shape](#shape)
- [Rect](#rect)
- [Circle](#circle)
- [Ellipse](#ellipse)
- [Element](#element)
- [Button](#button)
- [属性](#属性)
- [Transform](#transform)
- [Alpha](#alpha)
- [CompositeOperation](#compositeoperation)
- [Cursor](#cursor)
- [Fixed](#fixed)
- [Shadow](#shadow)
- [Stage](#stage-1)
- [方法](#方法)
- [destroy](#destroy)
- [事件](#事件)
- [小程序事件](#小程序事件)
- [Web 事件](#web-事件)
- [运动](#运动)
- [裁剪](#裁剪)
- [自定义对象](#自定义对象)
- [自定义 Shape](#自定义-shape)
- [自定义 Element](#自定义-element)
- [图片加载器](#图片加载器)
- [谁在使用?](#谁在使用)
- [微信交流群](#微信交流群2)
- [License](#license)## 一分钟入门小程序 cax 使用
到 GitHub [下载 cax 自定义组件](https://github.com/dntzhang/cax/tree/master/packages/cax-weapp),然后小程序引入 cax 自定义组件:
```
└── cax
├── cax.js
├── cax.json
├── cax.wxml
├── cax.wxss
└── index.js
```在 page 或者 component 里声明依赖:
```json
{
"usingComponents": {
"cax":"../cax/cax"
}
}
```在的 wxml 里引入 cax 标签:
```html
```
在 js 里渲染逻辑:
```js
import cax from '../cax/index'Page({
onLoad: function () {
//比 web 里使用 cax 多传递 this,this 代表 Page 或 Component 的实例
const stage = new cax.Stage(200, 200, 'myCanvas', this)
const rect = new cax.Rect(100, 100, {
fillStyle: 'black'
})
rect.originX = 50
rect.originY = 50
rect.x = 100
rect.y = 100
rect.rotation = 30rect.on('tap', () => {
console.log('tap')
})stage.add(rect)
stage.update()
}
})
```效果如下所示:
![](./asset/demo.jpg)
除了 tap 事件,也可以帮 rect 绑定其他触摸事件:
```js
rect.on('touchstart', () => {
console.log('touchstart')
})rect.on('touchmove', () => {
console.log('touchmove')
})rect.on('touchend', () => {
console.log('touchend')
})
```主要,小程序事件触发是根据元素的矩形区域,所以需要设置 width 和 height 才能绑定触发事件。
## 一分钟入门小游戏 cax 使用
到 GitHub [下载 cax 小游戏示例](https://github.com/dntzhang/cax/tree/master/packages/cax-wegame),目录结构和运行效果如下:
![](./asset/cax-wegame.png)
``` js
const stage = new cax.Stage()
```和小程序以及 Web 不同的是,小游戏创建 Stage 不需要传任何参数。
## 一分钟入门 Web cax 使用
通过 npm 或者 CDN 获取:
``` bash
npm i cax
```* [https://unpkg.com/cax@latest/dist/cax.min.js](https://unpkg.com/cax@latest/dist/cax.min.js)
* [https://unpkg.com/cax@latest/dist/cax.js](https://unpkg.com/cax@latest/dist/cax.js)使用方法:
``` js
import cax from 'cax'const stage = new cax.Stage(200, 200, 'body')
cax.loadImgs({
imgs: ['./test.jpg'],
complete: (imgs) => {
const img = imgs[0]
const bitmap = new cax.Bitmap(img)bitmap.x = stage.width / 2
bitmap.y = stage.height / 2
bitmap.rotation = -10
bitmap.originX = img.width / 2
bitmap.originY = img.height / 2
bitmap.filter('blur(10px)')stage.add(bitmap)
stage.update()
}
})
```除了 Stage 构造函数比小程序第四个参数 `this`,其他使用方式都一样。
## 内置对象
### Group
用于分组, group 也可以嵌套 group,父容器的属性会叠加在子属性上, 比如:
* group 的 x 是 100, group 里的 bitmap 的 x 是 200, 最后 bitmap 渲染到 stage 上的 x 是 300
* group 的 alpha 是 0.7, group 里的 bitmap 的 alpha 是 0.6, 最后 bitmap 渲染到 stage 上的 alpha 是 0.42```js
const group = new cax.Group()
const rect = new cax.Rect(100, 100, {
fillStyle: 'black'
})
group.add(rect)
stage.add(group)
stage.update()
```group 拥有常用的 add 和 remove 方法进行元素的增加和删除。先 add 的会先绘制,所有后 add 的会盖在先 add 的上面。
#### Group 方法
##### add
添加对象
``` js
groupObj.add (child)
```##### remove
移除对象
``` js
groupObj.remove (child)
```##### empty
清空子对象
``` js
groupObj.empty ()
```##### replace
使用一个对象替代子对象
```js
groupObj.replace(current, prev)
```### Stage
最大的顶层容器,继承自 Group,所以 Group 拥有的方法它全都有。
#### Stage 方法
##### update
任何元素添加到舞台上是看不到的,你必须执行 update 方法。
```js
stage.update()
```任何元素属性的修改请执行 stage.update() 来更新舞台,或者放在定时器里:
```js
cax.tick(stage.update.bind(stage))
```##### scaleEventPoint
当 canvas 的宽高 和 canvas 的像素不对应的时候,事件触发位置是不准确的,你可以使用 scaleEventPoint 方法使事件校正准确。
```js
//canvas 宽高是像素的一半
stage.scaleEventPoint(0.5, 0.5)
```例子: https://github.com/dntzhang/cax/blob/master/packages/cax/examples/pie/main.js#L218-L220
#### Stage 属性
##### disableMoveDetection
是否禁用鼠标或触摸移动的事件检测。
```js
stage.disableMoveDetection = true
```这里需要注意,Web 的 disableMoveDetection 默认是 false,微信小游戏默认是 true。所以微信小游戏想要监听 touchmove 和 drag 事件的话需要在创建完 Stage 之后执行下面代码开启监听:
```js
stage.disableMoveDetection = false
```##### moveDetectionInterval
设置 touchmove 和 mousemove 检测的间隔
```js
//每秒检测两次
stage.moveDetectionInterval = 500
```### Bitmap
```js
const bitmap = new cax.Bitmap(img)
stage.add(bitmap)
stage.update()
```如果只传 url 而不是 Image 对象的实例,需要这样:
```js
const bitmap = new cax.Bitmap('./wepay.png', ()=>{
stage.update()
})
stage.add(bitmap)
```这里需要注意小程序需要配置 downloadFile 需要配置合法域名才能正常加载到图片。
可以设置图片裁剪显示区域,和其他 transform 属性:
```js
bitmap.rect = [0, 0, 170, 140]
bitmap.x = 200
```### Sprite
序列帧动画组件,可以把任意图片的任意区域组合成一串动画。
```js
const sprite = new cax.Sprite({
framerate: 7,
imgs: ['./mario-sheet.png'],
frames: [
// x, y, width, height, originX, originY ,imageIndex
[0, 0, 32, 32],
[32 * 1, 0, 32, 32],
[32 * 2, 0, 32, 32],
[32 * 3, 0, 32, 32],
[32 * 4, 0, 32, 32],
[32 * 5, 0, 32, 32],
[32 * 6, 0, 32, 32],
[32 * 7, 0, 32, 32],
[32 * 8, 0, 32, 32],
[32 * 9, 0, 32, 32],
[32 * 10, 0, 32, 32],
[32 * 11, 0, 32, 32],
[32 * 12, 0, 32, 32],
[32 * 13, 0, 32, 32],
[32 * 14, 0, 32, 32]
],
animations: {
walk: {
frames: [0, 1]
},
happy: {
frames: [5, 6, 7, 8, 9]
},
win: {
frames: [12]
}
},
playOnce: false,
currentAnimation: "walk",
animationEnd: function () {}
});
```#### Sprite 方法
##### gotoAndPlay
跳到当前 animationName 并开始播放
```js
spriteObj.gotoAndPlay(animationName)
```##### gotoAndStop
跳到当前 animationName 并开始停止
```js
spriteObj.gotoAndStop(animationName)
```##### gotoAndPlayOnce
跳到当前 animationName 并开始播放,播完一轮销毁自己。常用于爆炸
```js
spriteObj.gotoAndPlayOnce(animationName)
```### Text
文本对象
``` js
const text = new cax.Text('Hello World', {
font: '20px Arial',
color: '#ff7700',
baseline: 'top'
})
```#### Text 方法
##### getWidth
获取文本宽度
```js
textObj.getWidth()
```### Graphics
绘图对象,用于使用基本的连缀方式的 Canvas 指令绘制图形。
``` js
const graphics = new cax.Graphics()
graphics
.beginPath()
.arc(0, 0, 10, 0, Math.PI * 2)
.closePath()
.fillStyle('#f4862c')
.fill()
.strokeStyle('black')
.stroke()graphics.x = 100
graphics.y = 200stage.add(graphics)
```特别注意,如果你在某个循环中执行 graphics 连缀绘制操作,请务必加上 clear() 方法,不然路径叠加到你的浏览器不堪重负:
```js
cax.setInterval(function(){
graphics
.clear()
.beginPath()
.arc(0, 0, 10, 0, Math.PI * 2)
.stroke()
}, 16)
```### Shape
与 Graphics 不同的是, Shape 一般拥有有限的宽高,所以可以使用离屏 Canvas 进行缓存。下面这些属于 Shape。
#### Rect
``` js
const rect = new cax.Rect(200, 100, {
fillStyle: 'black'
})
```#### Circle
``` js
const circle = new cax.Circle(10)
```#### Ellipse
``` js
const ellipse = new cax.Ellipse(120, 20)
```注意:从技术上小游戏和 Web 可以离屏 Canvas,小程序不行,因为小程序不支持动态创建离屏 Canvas。
### Element
Element 是多种元素的组合,如 Bitmap、Group、 Text、 Shape 等混合起来的图像。
#### Button
``` js
const button = new cax.Button({
width: 100,
height: 40,
text: "Click Me!"
})
```## 属性
### Transform
|属性名 |描述 |
|---|---|
| x | 水平偏移 |
| y | 竖直偏移 |
| scaleX | 水平缩放 |
| scaleY | 竖直缩放 |
| scale | 同时设置或读取 scaleX 和 scaleY |
| rotation | 旋转 |
| skewX | 歪斜 X |
| skewY | 歪斜 Y |
| originX | 旋转基点 X |
| originY | 旋转基点 Y |### Alpha
|属性名 |描述 |
|---|---|
| alpha | 元素的透明度 |注意这里父子都设置了 alpha 会进行乘法叠加。
### compositeOperation
|属性名 |描述 |
|---|---|
| compositeOperation | 源图像绘制到目标图像上的叠加模式 |注意这里如果自身没有定义 compositeOperation 会进行向上查找,找到最近的定义了 compositeOperation 的父容器作为自己的 compositeOperation。
### Cursor
|属性名 |描述 |
|---|---|
| cursor | 鼠标移上去的形状 |### Fixed
|属性名 |描述 |
|---|---|
| fixed | 是否固定定位,默认是 false 设置成 true 不会叠加祖辈们的 transform 属性|### Shadow
|属性名 |描述 |
|---|---|
| shadow | 阴影|使用方式:
```js
obj.shadow = {
color: '#42B035',
offsetX: -5,
offsetY: 5,
blur: 10
}
```### Stage
|Name |Describe |
|---|---|
| stage |或者自己所在的 stage|使用方式:
```js
obj.stage
```## 方法
### destroy
销毁自己
``` js
obj.destroy()
```## 事件
### 小程序事件
|事件名 |描述 |
|---|---|
| tap | 手指触摸后马上离开 |
| touchstart | 手指触摸动作开始 |
| touchmove | 手指触摸后移动 |
| touchend | 手指触摸动作结束 |
| drag | 拖拽 |### Web 事件
|事件名 |描述 |
|---|---|
| click | 元素上发生点击时触发 |
| mousedown | 当元素上按下鼠标按钮时触发 |
| mousemove | 当鼠标指针移动到元素上时触发 |
| mouseup | 当在元素上释放鼠标按钮时触发 |
| mouseover | 当鼠标指针移动到元素上时触发 |
| mouseout | 当鼠标指针移出元素时触发 |
| tap | 手指触摸后马上离开 |
| touchstart | 手指触摸动作开始 |
| touchmove | 手指触摸后移动 |
| touchend | 手指触摸动作结束 |
| drag | 拖拽 |## 运动
cax 内置了 to 的能力以连缀的方式写运动效果:
``` js
const easing = cax.To.easing.elasticInOutcax.To.get(bitmap)
.to({ y: 340, rotation: 240 }, 2000, easing)
.begin(() => {
console.log("Task one has began!")
})
.progress(() => {
console.log("Task one is progressing!")
})
.end(() => {
console.log("Task one has completed!")
})
.wait(500)
.to()
.rotation(0, 1400, easing)
.begin(() => {
console.log("Task two has began!")
})
.progress(() => {
console.log("Task two is progressing!")
})
.end(() => {
console.log("Task two has completed!")
})
.start();
```* `to` 和 `to` 之间的是并行
* `to` 和 `wait` 之前的是串行
* `to` 和 `to` 之间的 与 下一个 `to` 和 `to` 之间的是串行有点绕,但是很直观,慢慢体会。
当然,也可以通过 set 方法支持任意属性的运动,如:
``` js
.set('y', 240, 2000, cax.easing.elasticInOut)
```等同于
``` js
.y(240, 2000, cax.easing.elasticInOut)
```如果想要循环播放的话可以使用 `cycle` 方法:
``` js
cax.To.get(bitmap)
.to()
.y(340, 2000, cax.easing.elasticInOut)
.to
.y(0, 2000, cax.easing.elasticInOut)
.cycle()
.start()
```* [→ 运动演示地址](http://dntzhang.github.io/cax/packages/cax/examples/to/)
* [→ 去 to2to 一看究竟](https://github.com/dntzhang/cax/tree/master/packages/to)
* [→ 看看 cax.To.easing 种类](http://tweenjs.github.io/tween.js/examples/03_graphs.html)这里需要注意,和 tween.js 不同,cax 把 easing 命名全改成了一个单词的驼峰命名。如 Cubic.In 变成了 cubicIn。
## 裁剪
```js
const stage = new cax.Stage(600, 400, 'body')
const bitmap = new cax.Bitmap('./wepay-diy.jpg', () => {
stage.update()
})
const clipPath = new cax.Graphics()
clipPath.arc(40, 40, 25, 0, Math.PI * 2)
bitmap.clip(clipPath)
stage.add(bitmap)
```使用下面的代码可以得到同样的效果:
```js
const clipPath = new cax.Graphics()
clipPath.x = 40
clipPath.y = 40
clipPath.arc(0, 0, 25, 0, Math.PI * 2)
```所以,裁剪区域也是支持所有 transform 属性(x,y,scaleX,scaleY,rotation,skewX,skewY,originX,originY)。
[→ 裁剪演示地址](http://dntzhang.github.io/cax/packages/cax/examples/clip/)
## 自定义对象
### 自定义 Shape
自定义 Shape 继承自 cax.Shape:
``` js
class Sector extends cax.Shape {
constructor (r, from, to, option) {
super()this.option = option || {}
this.r = r
this.from = from
this.to = to
}draw () {
this.beginPath()
.moveTo(0, 0)
.arc(0, 0, this.r, this.from, this.to)
.closePath()
.fillStyle(this.option.fillStyle)
.fill()
.strokeStyle(this.option.strokeStyle)
.lineWidth(this.option.lineWidth)
.stroke()
}
}
```使用 Shape:
``` js
const sector = new Sector(10, 0, Math.PI/6, {
fillStyle: 'red'
lineWidth: 2
})
stage.add(sector)
stage.update()
```### 自定义 Element
自定义 Element 继承自 cax.Group:
``` js
class Button extends cax.Group {
constructor (option) {
super()
this.width = option.width
this.roundedRect = new cax.RoundedRect(option.width, option.height, option.r)
this.text = new cax.Text(option.text, {
font: option.font,
color: option.color
})this.text.x = option.width / 2 - this.text.getWidth() / 2 * this.text.scaleX
this.text.y = option.height / 2 - 10 + 5 * this.text.scaleY
this.add(this.roundedRect, this.text)
}
}export default Button
```使用:
``` js
const button = new cax.Button({
width: 100,
height: 40,
text: "Click Me!"
})
```一般情况下,稍微复杂组合体都建议使用继承自 Group,这样利于扩展也方便管理自身内部的元件。
小游戏的 DEMO 里的 [Player、Bullet、Enemy、Background](https://github.com/dntzhang/cax/tree/master/packages/cax-wegame/js) 全都是继承自 Group。
[Wechart 的所有图表](https://github.com/dntzhang/wechart/tree/master/packages)全都是继承自 Group。## 图片加载器
``` js
cax.loadImg({
img: './a.png',
complete: function(img){}
})
```加载多张图片:
```js
cax.loadImgs({
img: ['./a.png','./b.png'],
progress: function(progress){
console.log(progress) //0.5 and then 1
},
complete: function(imgs){
console.log(imgs[0]) //Image(a.png)
console.log(imgs[1]) //Image(b.png)
}
})
```## 小程序案例
![星球部落](./asset/ss1.png)![星球部落](./asset/ss3.png)
## 微信小游戏案例
![你在画什么](https://github.com/dntzhang/cax/raw/master/asset/draw.png)
![点十消除](./asset/wegame.jpg)
## License
MIT