diff --git a/samples/particle-shape-spin.html b/samples/particle-shape-spin.html
new file mode 100644
index 0000000..ef695b8
--- /dev/null
+++ b/samples/particle-shape-spin.html
@@ -0,0 +1,47 @@
+
+
+
+
+ Title
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/samples/particle-shape.html b/samples/particle-shape.html
index 58d27eb..aed667b 100644
--- a/samples/particle-shape.html
+++ b/samples/particle-shape.html
@@ -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'
],
})
})
diff --git a/samples/snow-shape-spin.html b/samples/snow-shape-spin.html
new file mode 100644
index 0000000..430d393
--- /dev/null
+++ b/samples/snow-shape-spin.html
@@ -0,0 +1,125 @@
+
+
+
+
+ Title
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/src/common/mask.ts b/src/common/mask.ts
index 1afe3de..0d81d40 100644
--- a/src/common/mask.ts
+++ b/src/common/mask.ts
@@ -6,12 +6,12 @@ export type modeMethodNames = 'modeNormal' | 'modeGhost'
export default abstract class Mask extends Base {
// 遮罩图像对象
- protected maskImage?: CanvasImageSource
+ protected maskImage?: ICanvasImageSource
// 扩展 mask 相关属性
protected readonly options!: Options &
CommonConfig & {
- mask?: string | CanvasImageSource
+ mask?: string | ICanvasImageSource
maskMode?: 'normal' | 'ghost'
}
@@ -41,7 +41,7 @@ export default abstract class Mask extends Base {
this.maskImage = image
})
} else {
- this.maskImage = mask as CanvasImageSource
+ this.maskImage = mask as ICanvasImageSource
}
}
diff --git a/src/common/shape.ts b/src/common/shape.ts
index c6de79a..01fe1ac 100644
--- a/src/common/shape.ts
+++ b/src/common/shape.ts
@@ -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'
@@ -12,13 +12,13 @@ export type NormalShapeType = Exclude, '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
// 当为图片类型时,图片加载完毕的源文件
- source?: CanvasImageSource
+ source?: ICanvasImageSource
// 当为图片类型时,图片是否加载完毕
isImageLoaded?: boolean
// 当为 star 类型时,星形的边数
@@ -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 ||
@@ -174,6 +173,7 @@ export default abstract class Shape extends Base {
r: number
color: string
shape: ShapeData
+ rotate: number
}): void {
const { type, isImageLoaded, source, sides, dent } = data.shape
@@ -181,38 +181,43 @@ export default abstract class Shape extends Base {
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()
}
diff --git a/src/particle.ts b/src/particle.ts
index 8437900..3e1547a 100644
--- a/src/particle.ts
+++ b/src/particle.ts
@@ -63,6 +63,15 @@ export default class Particle extends Shape {
// 视差强度,值越小视差效果越强烈
parallaxStrength: 3,
+
+ // 是否自旋
+ spin: false,
+
+ // 粒子最大运动角速度(0, 360)
+ spinMaxSpeed: 5,
+
+ // 粒子最小运动角速度(0, 360)
+ spinMinSpeed: 1,
}
protected elements!: IElement[]
@@ -185,7 +194,16 @@ export default class Particle extends Shape {
*/
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
@@ -204,6 +222,10 @@ export default class Particle extends Shape {
// 定义粒子视差的偏移值
parallaxOffsetX: 0,
parallaxOffsetY: 0,
+ // 定义粒子的旋转角度
+ rotate: spin ? randomInRange(0, 360) : 0,
+ // 粒子的旋转速度
+ rotateSpeed: randomSpeed(spinMaxSpeed, spinMinSpeed),
})
}
}
@@ -226,6 +248,10 @@ export default class Particle extends Shape {
// 绘制粒子
this.elements.forEach((dot) => {
const { x, y, parallaxOffsetX, parallaxOffsetY } = dot
+
+ // 更新粒子旋转角度
+ this.updateElementRotate(dot)
+
this.drawShape({
...dot,
x: x + parallaxOffsetX,
@@ -240,6 +266,23 @@ export default class Particle extends Shape {
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
+ }
+ }
+
/**
* 连接粒子,绘制线段
*/
diff --git a/src/snow.ts b/src/snow.ts
index 7514f3f..5416ed2 100644
--- a/src/snow.ts
+++ b/src/snow.ts
@@ -15,6 +15,9 @@ export default class Snow extends Shape {
swing: true,
swingInterval: 2000,
swingProbability: 0.06,
+ spin: false,
+ spinMaxSpeed: 5,
+ spinMinSpeed: 1,
}
protected elements!: IElement[]
@@ -41,8 +44,17 @@ export default class Snow extends Shape {
* 创建单个雪花,包含大小、位置、速度等信息
*/
private createSnowflake(): IElement {
- const { maxR, minR, maxSpeed, minSpeed } = this.options
+ const {
+ maxR,
+ minR,
+ maxSpeed,
+ minSpeed,
+ spin,
+ spinMaxSpeed,
+ spinMinSpeed,
+ } = this.options
const r = randomInRange(maxR, minR)
+
return {
r,
x: Math.random() * this.canvasWidth,
@@ -53,6 +65,10 @@ export default class Snow extends Shape {
color: this.getColor(),
swingAt: Date.now(),
shape: this.getShapeData(),
+ // 定义粒子的旋转角度
+ rotate: spin ? randomInRange(0, 360) : 0,
+ // 粒子的旋转速度
+ rotateSpeed: randomSpeed(spinMaxSpeed, spinMinSpeed),
}
}
@@ -74,14 +90,22 @@ export default class Snow extends Shape {
*/
protected draw(): void {
const { canvasWidth, canvasHeight, isPaused } = this
- const { maxR, swing, swingInterval, swingProbability, duration } =
- this.options
+ const {
+ maxR,
+ swing,
+ swingInterval,
+ swingProbability,
+ duration,
+ } = this.options
this.clearCanvasAndSetGlobalAttrs()
this.elements.forEach((snowflake, i, array) => {
const { x, y, r } = snowflake
+ // 更新旋转角度
+ this.updateElementRotate(snowflake)
+
this.drawShape(snowflake)
// 暂停(isPaused)且窗口改变(resize)的时候也会调用绘图方法
@@ -137,6 +161,23 @@ export default class Snow extends Shape {
}
}
+ /**
+ * 更新元素自旋数据
+ */
+ 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
+ }
+ }
+
/**
* 方法:当存在持续时间时,再次下雪
*/
diff --git a/src/types/global.d.ts b/src/types/global.d.ts
index bbce3af..e2f04fb 100644
--- a/src/types/global.d.ts
+++ b/src/types/global.d.ts
@@ -11,3 +11,8 @@ interface Window {
mozRequestAnimationFrame?: AnimationFrameProvider['requestAnimationFrame']
WebKitMutationObserver?: MutationObserver
}
+
+/**
+ * 排除没有 width 和 height 属性的元素
+ */
+type ICanvasImageSource = Exclude
diff --git a/src/types/particle.d.ts b/src/types/particle.d.ts
index c6c2c20..038614f 100644
--- a/src/types/particle.d.ts
+++ b/src/types/particle.d.ts
@@ -51,6 +51,15 @@ export interface Options extends Partial {
// 视差强度,值越小视差效果越强烈
parallaxStrength: number
+
+ // 粒子是否自旋
+ spin: boolean
+
+ // 粒子最大运动角速度(0, 360)
+ spinMaxSpeed: number
+
+ // 粒子最小运动角速度(0, 360)
+ spinMinSpeed: number
}
export interface IElement {
@@ -64,6 +73,10 @@ export interface IElement {
vx: number
// 每次绘制时,粒子在 y 轴方向的步进值,正负值
vy: number
+ // 粒子的旋转角度
+ rotate: number
+ // 粒子的旋转速度
+ rotateSpeed: number
// 粒子颜色
color: string
// 形状数据
diff --git a/src/types/snow.d.ts b/src/types/snow.d.ts
index 730efac..2633f08 100644
--- a/src/types/snow.d.ts
+++ b/src/types/snow.d.ts
@@ -22,6 +22,12 @@ export interface Options extends Partial {
swingInterval: number
// 变换方向的概率(达到时间间隔后),取值范围 [0, 1]
swingProbability: number
+ // 雪花是否自旋
+ spin: boolean
+ // 雪花最大运动角速度(0, 360)
+ spinMaxSpeed: number
+ // 雪花最小运动角速度(0, 360)
+ spinMinSpeed: number
}
export interface IElement {
@@ -41,4 +47,8 @@ export interface IElement {
swingAt: number
// 形状数据
shape: ShapeData
+ // 旋转角度
+ rotate: number
+ // 旋转速度
+ rotateSpeed: number
}