Android Flutter实现精灵图的使用详解
前言
在日常开发中遇到的图片展示一般是静态图和Gif图两种形式(静态和动态的不同)。与此同时当需要对图片做效果时让其动起来,常用方案是Gif图播放或者是帧动画(多种静态图轮询播放)。但在游戏开发中还有一种动图表现形式叫做Sprite图(雪碧图),其在前端开发中也是很常见。为什么需要使用精灵图,因为每张图片显示都需要去发起请求获取,若页面图片数量较多(一个页面有几十个小图)并发请求将是一个大数量级,可能会造成页面加载速度降低,精灵图其中一个特点就是减少服务器请求,一次性加载到所有图片。
如何使用精灵图
在游戏开发中精灵图会将一个人物所有动作放置在一张图片中,通过坐标定位选取其中一张图展示。根据精灵图配置信息来定位到不同图片再通过定时切换选取不同图片从而实现连贯动画效果。
自定义实现加载
自定义实习方式是对精灵图进行加载,因为每帧图片尺寸大小是一致的。
- 每帧图片尺寸固定后就能确认每一帧图片在整张图片的定位位置,因此能够根据X-Y坐标来找到相应图片。
- 创建定时器调整间隔时间计算出坐标位置切换需要展示图片。
- 再结合
Container+Positioned
形式来实现采用偏移量方式显示哪张图片。
class SpriteWidget extends StatefulWidget {
final Image image;
final Size spriteSize;
final Duration duration;
SpriteWidget({
Key key,
@required this.image,
@required this.spriteSize,
@required this.duration,
}) : super(key: key);
@override
_SpriteWidgetState createState() => _SpriteWidgetState();
}
class _SpriteWidgetState extends State<SpriteWidget> {
Image get image => widget.image;
Size get spriteSize => widget.spriteSize;
Duration get duration => widget.duration;
// 当前显示的图片位置
int currentIndex = 0;
int currentTimes = 0;
int startIndex = 0;
int endIndex = 1;
int playTimes = 0;
// 定时器用来更新精灵图加载
Timer timer;
@override
void initState() {
currentIndex = startIndex;
timer = Timer.periodic(duration, (timer) {
if (currentTimes <= playTimes) {
setState(() {
if (currentIndex >= endIndex) {
if (playTimes != 0) currentTimes++;
if (currentTimes < playTimes || playTimes == 0)
currentIndex = startIndex;
else
currentIndex = endIndex;
} else
currentIndex++;
});
}
});
super.initState();
}
@override
void dispose() {
super.dispose();
timer?.cancel();
}
@override
Widget build(BuildContext context) {
// 使用container+Positioned来限制图片显示区域
return Container(
width: spriteSize.width,
height: spriteSize.height,
child: Stack(
children: [
Positioned(
left: -spriteSize.width * currentIndex,
top: -spriteSize.height * currentIndex,
child: image)
],
),
);
}
}
Flame加载精灵图
除了自定义方式实现精灵图加载外,Flutter官方还提供了Flame
框架实现游戏内容制作帮助开发者更方便实现游戏相关功能开发。
首先在依赖库添加Flame
库,此外bonfire
是Flame
库的拓展封装了更多游戏开发相关功能接口。
bonfire: ^2.0.0
flame: ^1.0.0
实现角色资源加载类,bonfire
中已经帮助开发者实现了Sprite
功能只需要加载精灵图就能获取到精灵图加载能力。
abstract class BaseRole {
// 速度
double velocity;
late Sprite sprite;
BaseRole({
this.velocity = 10,
});
dynamic load();
void run();
}
class UserRole extends BaseRole {
Vector2 offset = Vector2(0.0, 0.0);
Vector2 size = Vector2(48.0, 48.0);
@override
void run() {
double x = (48 + offset.x) % 192;
double y = 0;
offset = Vector2(x, y);
sprite.class="lazy" data-srcPosition = offset;
sprite.class="lazy" data-srcSize = size;
}
@override
Future load() async {
sprite = await Sprite.load(
'user_role.png',
class="lazy" data-srcPosition: offset,
class="lazy" data-srcSize: size,
);
return sprite;
}
}
在bonfire
游戏开发中还有SpriteComponent
组件,开发者只需要继承它加载使用Sprite
即可。
class UserRoleComponent extends SpriteComponent {
late UserRole userRole;
UserRoleComponent()
: super(
size: Vector2.all(100),
);
double minDt = 1 / 8; // 30 fps
double dtOverflow = 0;
@override
Future<void>? onLoad() async {
userRole = UserRole();
sprite = await userRole.load();
super.onLoad();
}
@override
void update(double dt) {
dtOverflow += dt;
if (dtOverflow < minDt) {
return;
}
userRole.run();
dtOverflow = 0;
}
}
到此这篇关于Android Flutter实现精灵图的使用详解的文章就介绍到这了,更多相关Android Flutter精灵图内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!
免责声明:
① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。
② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341