Skip to content

Commit

Permalink
feat: support spin effect of Particles & Snow
Browse files Browse the repository at this point in the history
  • Loading branch information
Barrior committed Mar 21, 2024
1 parent 683e158 commit e00208a
Show file tree
Hide file tree
Showing 10 changed files with 319 additions and 30 deletions.
47 changes: 47 additions & 0 deletions samples/particle-shape-spin.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
<meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=0">
<link rel="stylesheet" href="css/style.css">
<style>
</style>
</head>
<body>
<div id="instance1">
<div class="demo"></div>
<div class="btn-box">
<button class="btn btn-primary" type="button" data-open>
open
</button>
<button class="btn btn-danger" type="button" data-pause>
pause
</button>
</div>
</div>

<script src="js/event.js"></script>
<script>
bind('#instance1', function() {
return new JParticles.Particle('#instance1 .demo', {
num: 50,
maxR: 30,
minR: 30,
range: 0,
// 是否自旋
spin: true,
// 粒子形状
shape: [
'circle',
'triangle',
'star',
'star:4:0.5',
'star:30:0.5',
'https://raw.githubusercontent.com/Barrior/assets/main/smiling-face.gif'
],
})
})
</script>
</body>
</html>
2 changes: 1 addition & 1 deletion samples/particle-shape.html
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,7 @@
minR: 30,
shape: [
'circle', 'triangle', 'star', 'star:4:0.5', 'star:30:0.5',
'https://img30.360buyimg.com/ling/jfs/t1/42528/38/16521/191828/60ecf690E1b8b016d/060dbecad0b8042a.png'
'https://raw.githubusercontent.com/Barrior/assets/main/smiling-face.gif'
],
})
})
Expand Down
125 changes: 125 additions & 0 deletions samples/snow-shape-spin.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,125 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
<link rel="stylesheet" href="css/style.css">
<style>
[id*="instance"] .demo {
background: url(./img/merry_christmas.jpg) no-repeat;
background-size: 100% 100%;
}
</style>
</head>
<body>
<div id="instance">
<div class="demo"></div>
<div class="btn-box">
<button class="btn btn-primary" type="button" data-open>
open
</button>
<button class="btn btn-danger" type="button" data-pause>
pause
</button>
</div>
</div>

<div id="instance2">
<div class="demo"></div>
<div class="btn-box">
<button class="btn btn-primary" type="button" data-open>
open
</button>
<button class="btn btn-danger" type="button" data-pause>
pause
</button>
</div>
</div>

<div id="instance3">
<div class="demo"></div>
<div class="btn-box">
<button class="btn btn-primary" type="button" data-open>
open
</button>
<button class="btn btn-danger" type="button" data-pause>
pause
</button>
</div>
</div>

<div id="instance4">
<div class="demo"></div>
<div class="btn-box">
<button class="btn btn-primary" type="button" data-open>
open
</button>
<button class="btn btn-danger" type="button" data-pause>
pause
</button>
</div>
</div>

<script src="js/event.js"></script>
<script>
const base64PNG = ''

// 测试 base64 格式
bind('#instance', function() {
return new JParticles.Snow('#instance .demo', {
maxR: 30,
minR: 10,
maxSpeed: 0.2,
shape: base64PNG,
spin: true,
spinMaxSpeed: 20,
spinMinSpeed: 1,
})
})

// 测试「内置」形状
bind('#instance2', function() {
return new JParticles.Snow('#instance2 .demo', {
maxR: 10,
minR: 10,
maxSpeed: 0.2,
// star:[边数][凹值]
shape: ['circle', 'triangle', 'star', 'star:4:0.2', 'star:6:1.5'],
spin: true,
})
})

// 测试 HTTP 格式
bind('#instance3', function() {
return new JParticles.Snow('#instance3 .demo', {
maxR: 20,
minR: 20,
maxSpeed: 0.2,
minSpeed: 0.2,
swing: false,
shape: [
'https://img10.360buyimg.com/ling/jfs/t1/180952/11/13170/68465/60e6e46fE8d8e4f15/453c2896998eda6d.png',
'https://img10.360buyimg.com/ling/jfs/t1/182564/22/13408/6442/60e6ee1bE014c0d60/6ef57767e19999b2.png',
'https://raw.githubusercontent.com/Barrior/assets/main/chrome-logo.svg'
],
spin: true,
})
})

// 测试 HTMLImageElement 格式
JParticles.utils.loadImage('https://img10.360buyimg.com/ling/jfs/t1/180952/11/13170/68465/60e6e46fE8d8e4f15/453c2896998eda6d.png', (image) => {
bind('#instance4', function () {
return new JParticles.Snow('#instance4 .demo', {
maxR: 20,
minR: 20,
maxSpeed: 0.2,
minSpeed: 0.2,
shape: image,
spin: true,
})
})
})

</script>
</body>
</html>
6 changes: 3 additions & 3 deletions src/common/mask.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,12 +6,12 @@ export type modeMethodNames = 'modeNormal' | 'modeGhost'

export default abstract class Mask<Options> extends Base<Options> {
// 遮罩图像对象
protected maskImage?: CanvasImageSource
protected maskImage?: ICanvasImageSource

// 扩展 mask 相关属性
protected readonly options!: Options &
CommonConfig & {
mask?: string | CanvasImageSource
mask?: string | ICanvasImageSource
maskMode?: 'normal' | 'ghost'
}

Expand Down Expand Up @@ -41,7 +41,7 @@ export default abstract class Mask<Options> extends Base<Options> {
this.maskImage = image
})
} else {
this.maskImage = mask as CanvasImageSource
this.maskImage = mask as ICanvasImageSource
}
}

Expand Down
49 changes: 27 additions & 22 deletions src/common/shape.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import Base from '@src/common/base'
import { doublePi, regExp } from '@src/common/constants'
import { doublePi, piBy180, regExp } from '@src/common/constants'
import type { CommonConfig } from '@src/types/common-config'
import type { ValueOf } from '@src/types/utility-types'
import { isArray, isString, loadImage } from '@src/utils'
Expand All @@ -12,13 +12,13 @@ export type NormalShapeType = Exclude<ValueOf<typeof validShapeTypes>, 'image'>
// 1. `star:[边数][凹值]`, 示例:`star:5:0.5`, 表示五角星
// 2. 图片 HTTP 地址,示例:`https://xxx.com/a.jpg`
// 3. 图片 Base64 格式,示例:``
export type ShapeType = NormalShapeType | string | CanvasImageSource
export type ShapeType = NormalShapeType | string | ICanvasImageSource

export interface ShapeData {
// 形状类型
type: ValueOf<typeof validShapeTypes>
// 当为图片类型时,图片加载完毕的源文件
source?: CanvasImageSource
source?: ICanvasImageSource
// 当为图片类型时,图片是否加载完毕
isImageLoaded?: boolean
// 当为 star 类型时,星形的边数
Expand Down Expand Up @@ -75,7 +75,6 @@ function generateShapeData(shape: ShapeType): ShapeData {
// 处理 CanvasImageSource 类型
if (
shape instanceof HTMLImageElement ||
shape instanceof SVGImageElement ||
shape instanceof HTMLVideoElement ||
shape instanceof HTMLCanvasElement ||
shape instanceof ImageBitmap ||
Expand Down Expand Up @@ -174,45 +173,51 @@ export default abstract class Shape<Options> extends Base<Options> {
r: number
color: string
shape: ShapeData
rotate: number
}): void {
const { type, isImageLoaded, source, sides, dent } = data.shape

if (validShapeTypes.indexOf(type) === -1) {
return
}

if (type === 'image' && (!isImageLoaded || !source)) {
return
}

this.ctx.save()

// 旋转渲染
this.ctx.translate(data.x, data.y)
this.ctx.rotate(data.rotate * piBy180)

if (type === 'image') {
if (isImageLoaded) {
const width = data.r * 2
this.ctx.drawImage(
source!,
0,
0,
(source?.width as number) || width,
(source?.height as number) || width,
data.x - data.r,
data.y - data.r,
width,
width
)
}
const width = data.r * 2
this.ctx.drawImage(
source!,
0,
0,
source!.width || width,
source!.height || width,
-data.r,
-data.r,
width,
width
)
} else {
this.ctx.beginPath()

switch (data.shape.type) {
case 'circle':
this.ctx.arc(data.x, data.y, data.r, 0, doublePi)
this.ctx.arc(0, 0, data.r, 0, doublePi)
break
case 'triangle':
drawStar(this.ctx, data.x, data.y, data.r, 3, 0.5)
drawStar(this.ctx, 0, 0, data.r, 3, 0.5)
break
case 'star':
drawStar(this.ctx, data.x, data.y, data.r, sides!, dent!)
drawStar(this.ctx, 0, 0, data.r, sides!, dent!)
break
}

this.ctx.fillStyle = data.color
this.ctx.fill()
}
Expand Down
45 changes: 44 additions & 1 deletion src/particle.ts
Original file line number Diff line number Diff line change
Expand Up @@ -63,6 +63,15 @@ export default class Particle extends Shape<Options> {

// 视差强度,值越小视差效果越强烈
parallaxStrength: 3,

// 是否自旋
spin: false,

// 粒子最大运动角速度(0, 360)
spinMaxSpeed: 5,

// 粒子最小运动角速度(0, 360)
spinMinSpeed: 1,
}

protected elements!: IElement[]
Expand Down Expand Up @@ -185,7 +194,16 @@ export default class Particle extends Shape<Options> {
*/
private createDots(): void {
const { canvasWidth, canvasHeight, getColor } = this
const { maxR, minR, maxSpeed, minSpeed, parallaxLayer } = this.options
const {
maxR,
minR,
maxSpeed,
minSpeed,
parallaxLayer,
spin,
spinMaxSpeed,
spinMinSpeed,
} = this.options
const layerLength = parallaxLayer.length
let { num } = this.options

Expand All @@ -204,6 +222,10 @@ export default class Particle extends Shape<Options> {
// 定义粒子视差的偏移值
parallaxOffsetX: 0,
parallaxOffsetY: 0,
// 定义粒子的旋转角度
rotate: spin ? randomInRange(0, 360) : 0,
// 粒子的旋转速度
rotateSpeed: randomSpeed(spinMaxSpeed, spinMinSpeed),
})
}
}
Expand All @@ -226,6 +248,10 @@ export default class Particle extends Shape<Options> {
// 绘制粒子
this.elements.forEach((dot) => {
const { x, y, parallaxOffsetX, parallaxOffsetY } = dot

// 更新粒子旋转角度
this.updateElementRotate(dot)

this.drawShape({
...dot,
x: x + parallaxOffsetX,
Expand All @@ -240,6 +266,23 @@ export default class Particle extends Shape<Options> {
this.requestAnimationFrame()
}

/**
* 更新元素自旋数据
*/
private updateElementRotate(element: IElement) {
if (!this.options.spin || this.isPaused) {
return
}

// 更新旋转角度
element.rotate += element.rotateSpeed

// 大于等于 360 度时,回到 0-360 范围,避免变量数值一直累计超过 MAX_SAFE_INTEGER
if (element.rotate >= 360) {
element.rotate = element.rotate - 360
}
}

/**
* 连接粒子,绘制线段
*/
Expand Down
Loading

0 comments on commit e00208a

Please sign in to comment.