Flutter开发的高效图片压缩工具:让APP更加丝滑
.markdown-body { line-height: 1.75; font-weight: 400; font-size: 15px; overflow-x: hidden; color: rgba(43, 43, 43, 1); font-family: -apple-system, system-ui, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Hiragino Sans GB, Microsoft YaHei, Arial, sans-serif; background-image: linear-gradient(90deg, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0), linear-gradient(1turn, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0); background-size: 20px 20px; background-position: center }
.markdown-body h1, .markdown-body h2, .markdown-body h3, .markdown-body h4, .markdown-body h5, .markdown-body h6 { padding: 30px 0; margin-top: 35px; margin-bottom: 10px; color: rgba(77, 208, 225, 1) }
.markdown-body h1 { font-size: 30px; text-align: center; position: relative; width: max-content; margin: 0 auto }
.markdown-body h1:before { position: absolute; content: ""; z-index: -1; top: -20px; height: 100%; width: 100px; left: 0; right: 0; margin: 0 auto; background: url("") center / 64px 64px no-repeat; opacity: 0.84 }
.markdown-body h1:after { position: absolute; content: ""; width: 150%; left: -25%; height: 50%; bottom: 12px; border-radius: 50%; background: linear-gradient(rgba(0, 0, 0, 0) 80%, rgba(77, 208, 225, 0.8)); opacity: 0.6; animation: 6s linear infinite h1animate }
@keyframes h1Animate { 0% { background-position: right bottom } 50% { background-position: right } 100% { background-position: right bottom } }
.markdown-body h2 { display: block; border-bottom: 4px solid rgba(77, 208, 225, 1); position: relative; font-size: 24px; padding: 12px 32px; margin: 30px 0 }
.markdown-body h2:before { width: 24px; height: 24px; left: 0; top: 0; margin: auto; background-size: 24px 24px; background-image: url("") }
.markdown-body h2:after, .markdown-body h2:before { content: ""; display: block; position: absolute; bottom: 0 }
.markdown-body h2:after { right: 0; width: 400px; height: 10px; border-top-right-radius: 24px; background: linear-gradient(90deg, rgba(255, 255, 255, 1), rgba(77, 208, 225, 1)); max-width: 50vw }
.markdown-body h3 { margin: 30px 0; font-size: 18px; position: relative; padding: 4px 32px; width: max-content }
.markdown-body h3:before { border-bottom: 2px solid rgba(77, 208, 225, 1); width: 100%; content: ""; display: block; height: 28px; position: absolute; left: 0; top: 0; bottom: -2px; margin: auto; background-size: 28px 28px; background-image: url(""); background-repeat: no-repeat; animation: 2s infinite alternate h3animationbefore }
@keyframes h3AnimationBefore { 0% { width: 28px } 25% { width: 100% } 50% { width: 100% } 100% { width: 100% } }
.markdown-body h3:after { content: ""; display: block; width: 28px; height: 28px; position: absolute; border: 2px solid rgba(77, 208, 225, 1); border-radius: 50%; right: -15px; top: 0; bottom: 0; margin: auto; background-size: 28px 28px; background-image: url(""); animation: 2s infinite alternate h3animationafter }
@keyframes h3AnimationAfter { 0% { } 10% { } 50% { transform: rotate(-1turn) } 100% { transform: rotate(-1turn) } }
.markdown-body h4 { font-size: 16px }
.markdown-body h5 { font-size: 15px }
.markdown-body h6 { margin-top: 5px }
.markdown-body p { line-height: inherit; margin: 22px 0; letter-spacing: 2px; font-size: 14px; word-spacing: 2px }
.markdown-body img { max-width: 80%; border-radius: 6px; display: block; margin: 20px auto !important; object-fit: contain; box-shadow: 0 0 16px rgba(110, 110, 110, 0.45) }
.markdown-body figcaption { display: block; font-size: 13px; color: rgba(43, 43, 43, 1) }
.markdown-body figcaption:before { content: ""; background-image: url(""); display: inline-block; width: 18px; height: 18px; background-size: 18px; background-repeat: no-repeat; background-position: center; margin-right: 5px; margin-bottom: -5px }
.markdown-body hr { border-top: 1px solid rgba(77, 208, 225, 1); border-right: none; border-bottom: none; border-left: none; margin-top: 32px; margin-bottom: 32px }
.markdown-body del { color: rgba(77, 208, 225, 1) }
.markdown-body code { border-radius: 2px; overflow-x: auto; background-color: rgba(77, 208, 225, 0.08); color: rgba(38, 198, 218, 1); padding: 0.195em 0.4em }
.markdown-body pre { font-family: Menlo, Monaco, Consolas, Courier New, monospace; overflow: auto; position: relative; line-height: 1.75; box-shadow: 0 0 8px rgba(110, 110, 110, 0.45); border-radius: 4px; margin: 16px }
.markdown-body pre:before { content: ""; display: block; height: 30px; width: 100%; margin-bottom: -7px; background: url("") 10px 10px / 40px no-repeat }
.markdown-body pre>code { font-size: 12px; padding: 15px 12px; margin: 0; word-break: normal; display: block; overflow-x: auto; color: rgba(51, 51, 51, 1); background: rgba(248, 248, 248, 1) }
.markdown-body a { color: rgba(77, 208, 225, 1); border-bottom: 1px solid rgba(77, 208, 225, 1); font-weight: 400; text-decoration: none; margin: 0 4px }
.markdown-body a:active, .markdown-body a:hover { background-color: rgba(77, 208, 225, 0.1) }
.markdown-body strong { color: rgba(38, 198, 218, 1) }
.markdown-body strong:before { content: "「" }
.markdown-body strong:after { content: "」" }
.markdown-body em { font-style: normal; color: rgba(77, 208, 225, 1); font-weight: 700 }
.markdown-body table { display: inline-block !important; font-size: 12px; width: auto; max-width: 100%; overflow: auto; border: 1px solid rgba(246, 246, 246, 1) }
.markdown-body thead { background: rgba(246, 246, 246, 1); color: rgba(0, 0, 0, 1); text-align: left }
.markdown-body tr:nth-child(2n) { background-color: rgba(77, 208, 225, 0.05) }
.markdown-body td, .markdown-body th { padding: 12px 7px; line-height: 24px }
.markdown-body td { min-width: 120px }
.markdown-body blockquote { margin: 2em 0; padding: 24px 32px; border-left: 4px solid rgba(38, 198, 218, 1); background: rgba(77, 208, 225, 0.15); position: relative }
.markdown-body blockquote:before { content: "❝"; top: 8px; left: 8px; color: rgba(77, 208, 225, 1); font-size: 30px; line-height: 1; font-weight: 700; position: absolute; opacity: 0.7 }
.markdown-body blockquote:after { content: "❞"; font-size: 30px; position: absolute; right: 8px; bottom: 0; color: rgba(77, 208, 225, 1); opacity: 0.7 }
.markdown-body blockquote p { color: rgba(89, 89, 89, 1); line-height: 2 }
.markdown-body ol, .markdown-body ul { color: rgba(89, 89, 89, 1); padding-left: 28px }
.markdown-body ol li, .markdown-body ul li { margin-bottom: 0; list-style: inherit }
.markdown-body ol li .task-list-item, .markdown-body ul li .task-list-item { list-style: none }
.markdown-body ol li .task-list-item ol, .markdown-body ol li .task-list-item ul, .markdown-body ul li .task-list-item ol, .markdown-body ul li .task-list-item ul { margin-top: 0 }
.markdown-body ol ol, .markdown-body ol ul, .markdown-body ul ol, .markdown-body ul ul { margin-top: 3px }
.markdown-body ol li { padding-left: 6px }
@media (max-width: 720px) { .markdown-body h1 { font-size: 24px } .markdown-body h2 { font-size: 20px } .markdown-body h3 { font-size: 18px } }.markdown-body pre, .markdown-body pre>code.hljs { background: rgba(34, 34, 34, 1) }
.hljs-subst, .markdown-body pre, .markdown-body pre>code.hljs { color: rgba(170, 170, 170, 1) }
.hljs-section { color: rgba(255, 255, 255, 1) }
.hljs-comment, .hljs-meta, .hljs-quote { color: rgba(68, 68, 68, 1) }
.hljs-bullet, .hljs-regexp, .hljs-string, .hljs-symbol { color: rgba(255, 204, 51, 1) }
.hljs-addition, .hljs-number { color: rgba(0, 204, 102, 1) }
.hljs-attribute, .hljs-built_in, .hljs-builtin-name, .hljs-link, .hljs-literal, .hljs-template-variable, .hljs-type { color: rgba(50, 170, 238, 1) }
.hljs-keyword, .hljs-name, .hljs-selector-class, .hljs-selector-id, .hljs-selector-tag { color: rgba(102, 68, 170, 1) }
.hljs-deletion, .hljs-template-tag, .hljs-title, .hljs-variable { color: rgba(187, 17, 102, 1) }
.hljs-doctag, .hljs-section, .hljs-strong { font-weight: 700 }
.hljs-emphasis { font-style: italic }
我正在参加「掘金·启航计划」
简介
随着互联网时代的到来,图片作为一种重要的信息载体,已经渗透到了我们生活的方方面面。在开发移动应用时,经常需要对图片进行压缩,以减小图片的大小,从而减少应用程序对用户手机存储空间的占用和对网络带宽的占用。今天,我们将学习如何使用Flutter开发一款高效的图片压缩工具。
Flutter的基础知识介绍
Flutter是一种流行的跨平台移动应用开发框架,它可以同时支持Android和iOS两个平台。Flutter提供了丰富的UI组件,以及强大的开发工具和命令行接口。
Flutter新版本它增加了许多新特性和改进,包括:
- 更好的性能和稳定性
- 新的UI组件和动画效果
- 支持许多第三方库和代码包
- 优化了热重载和调试工具
开发步骤
在本节中,我们将学习如何使用Flutter开发图片压缩工具。整个开发过程分为以下几个步骤:
- 引入相关依赖库
- 编写图片压缩工具的主要功能
- 编写界面以及交互逻辑
- 优化和改进
1. 引入相关依赖库
在Flutter中,我们可以使用pubspec.yaml文件来管理我们的依赖库和代码包。在本例中,我们将使用两个依赖库来实现图片压缩功能。
第一个依赖库是image_picker,它可以让我们方便地从相册或相机中选择图片。
第二个依赖库是image,它是一个强大而灵活的图像处理库,支持多种格式的图像文件,并提供了通用的图像处理工具。
在pubspec.yaml文件中添加以下代码:
dependencies:
flutter:
sdk: flutter
image_picker: ^0.8.7+4
image: ^4.0.17
完成后,执行flutter packages get命令,等待依赖库下载完毕即可。
2. 编写图片压缩工具的主要功能
在本部分中,我们将编写图片压缩工具的主要功能,主要包括:
- 压缩图片尺寸
- 压缩图片质量
- 生成压缩后的图片文件。
2.1 压缩图片尺寸
图片压缩的第一个步骤是调整图片的尺寸。在Flutter中,我们可以使用image库中的Resize函数来完成这个步骤。
假设我们已经从相册选择了一张图片,我们可以将其转换成Image对象,并调整它的宽度和高度。
import 'dart:io';
import 'package:flutter/material.dart';
import 'package:image_picker/image_picker.dart';
import 'package:image/image.dart' as Img;
void main() => runApp(MyApp());
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Image Compressor',
theme: ThemeData(
primarySwatch: Colors.blue,
),
home: MyHomePage(title: 'Image Compressor'),
);
}
}
class MyHomePage extends StatefulWidget {
MyHomePage({Key? key, required this.title}) : super(key: key);
final String title;
@override
_MyHomePageState createState() => _MyHomePageState();
}
class _MyHomePageState extends State<MyHomePage> {
late File _imageFile;
TextEditingController _widthController = TextEditingController();
TextEditingController _heightController = TextEditingController();
Future<void> _pickImage(ImageSource source) async {
final pickedFile = await ImagePicker().pickImage(
source: source,
imageQuality: 100
);
setState(() {
if (pickedFile != null) {
_imageFile = File(pickedFile.path);
} else {
print(‘No image selected.’);
}
});
}
void _compressImage() async {
if (_imageFile == null) {
return;
}
int width = int.parse(_widthController.text);
int height = int.parse(_heightController.text);
Img.Image image = Img.decodeImage(await _imageFile.readAsBytes())!;
Img.Image resizedImage = Img.copyResize(image, width: width, height: height);
// 将调整大小的图像保存到临时文件
String tempPath = (await getTemporaryDirectory()).path;
File compressedImageFile = File('$tempPath/compressed_image.jpg');
compressedImageFile.writeAsBytesSync(Img.encodeJpg(resizedImage));
}
在上面的代码中,我们使用了一个名为resizeImage的辅助函数,它将图片缩小到指定的宽度和高度,并将结果作为一个新的Image对象返回。
注意,我们将压缩后的图片文件保存到了临时目录中。这是因为我们不想在用户的手机中存储多余的文件。
2.2 压缩图片质量
图片压缩的第二个步骤是调整图片的质量。在Flutter中,我们可以使用image库中的encodeJpg函数来完成这个步骤。
我们可以将这个函数的第二个参数quality设置为0-100之间的整数,来控制压缩后的图片质量。通常情况下,压缩后的图片质量设置为60-80之间是比较合理的。
Img.Image image = Img.decodeImage(await _imageFile.readAsBytes())!;
int quality = 80;
List<int> compressedImage = Img.encodeJpg(image, quality: quality);
2.3 生成压缩后的图片文件
最后一步是将压缩后的图片保存到文件中。在Flutter中,我们可以使用File类的writeAsBytesSync函数将字节数组写入文件。
String tempPath = (await getTemporaryDirectory()).path;
File compressedImageFile = File('$tempPath/compressed_image.jpg');
compressedImageFile.writeAsBytesSync(compressedImage);
3. 实现UI交互逻辑
我们可以使用Flutter的Widget构建器来创建一个简单的UI界面。在我们的案例中,我们需要一个按钮来选择图片、两个文本框来输入图片的宽度和高度、一个按钮来触发压缩图片的操作,以及一个图片查看器来查看压缩后的图片。
首先,我们需要添加一个FlatButton按钮,当用户点击这个按钮时,它将调用_pickImage函数,以便用户可以从相册或相机中选择图片。
FlatButton(
child: Text('Select Image'),
onPressed: () {
showDialog(
context: context,
builder: (BuildContext context) {
return AlertDialog(
title: Text('Select Image Source'),
actions: <Widget>[
FlatButton(
child: Text('Camera'),
onPressed: () {
Navigator.pop(context);
_pickImage(ImageSource.camera);
},
),
FlatButton(
child: Text('Gallery'),
onPressed: () {
Navigator.pop(context);
_pickImage(ImageSource.gallery);
},
),
],
);
},
);
},
),
接下来,我们需要添加两个文本框,让用户可以输入图片的宽度和高度。
TextField(
controller: _widthController,
keyboardType: TextInputType.number,
decoration: InputDecoration(
hintText: 'Enter Width',
),
),
TextField(
controller: _heightController,
keyboardType: TextInputType.number,
decoration: InputDecoration(
hintText: ‘Enter Height’,
),
),
我们还需要添加一个RaisedButton按钮,当用户点击这个按钮时,它将调用_compressImage函数,以便压缩图片。
RaisedButton(
child: Text('Compress Image'),
onPressed: () {
_compressImage();
},
),
最后,我们需要添加一个Image组件,当用户选择并压缩图片时,它将用于查看压缩后的图片。
Expanded(
child: Container(
padding: EdgeInsets.all(16.0),
decoration: BoxDecoration(
border: Border.all(
color: Colors.blue,
),
),
child: _imageFile == null
? Text('No image selected.')
: Image.file(_imageFile),
),
),
4. 优化和改进
4.1 压缩算法优化
我们可以进一步优化我们的压缩算法,以减小压缩文件的大小。例如,我们可以使用更复杂的算法来压缩图片,例如JPEG2000或WebP。
4.2 UI界面美化
我们也可以改进我们的UI界面,使它更美观和易于使用。例如,我们可以添加进度条来显示压缩进度,或向用户提供更多的压缩选项。
4.3 对其他格式的文件压缩支持
我们可以扩展我们的工具,以支持其他格式的文件压缩。例如,我们可以使用PDF压缩库来压缩PDF文件,或使用ZIP压缩库来压缩多个文件。
4.4 代码结构改进
我们可以改善我们的代码结构,使得它更易于维护和扩展。例如,我们可以将压缩算法封装在单独的类中,或将UI界面封装在单独的组件中。
总结
在本文中,我们学习了如何使用Flutter开发一款高效的图片压缩工具。我们学习了如何使用image_picker和image库来实现图片压缩的主要功能,以及如何使用Flutter的Material Design风格来设计UI界面。我们还讨论了如何优化和改进我们的图片压缩工具,并给出了一些可行的方案。
掘友们,快去动手试试。
Flutter开发的高效图片压缩工具:让APP更加丝滑的更多相关文章
- 分享一下怎么开发一款图片视频类App,秒拍和prisma
第一步,分解短视频App的功能 我们在秒拍官网看到如此描述: [视频拍摄及导入]支持直接拍摄及导入手机本地的视频 [照片电影]照片专属特效,轻松创作照片电影 [MV特效]10余款全新MV特效,让普通视 ...
- 尝试用python开发一款图片压缩工具1:尝试 pillow库
开发目的 我经常使用图片.公众号文章发文也好,还是生活中要使用素材.图片是一种比文字更加直观的载体.但是图片更加占用带宽,很多软件都对图片有大小限制.图片太大也会影响加载速度.我试过几款图片压缩工具, ...
- python 开发一款图片压缩工具(四):上传图床
上一篇使用了 pngquant 图片压缩工具进行压缩,并通过 click 命令行工具构建了 picom 包.这篇的主要功能是实现图片上传. 图片上传功能的实现 通过 pngquant 压缩图片后,得到 ...
- 开发一款图片压缩工具(三):使用 click 实现命令行
上一篇实现了图片的压缩函数.现在如果需要对图片进行压缩,可以调用实现的函数进行压缩: pngquant_compress('elephant.png', force=True, quality=20) ...
- 开发一款图片压缩工具(二):使用 pngquant 实现图片压缩
上一篇我尝试使用了 pillow 库对 png 图片进行了压缩,效果不好.这次我换用 pngquant 来压缩.pngquant 是用于 PNG 图像有损压缩的命令行实用程序和库.压缩程序会显著减小文 ...
- 用C#开发一个WinForm版的批量图片压缩工具
我们在实际项目开发过程中,曾经遇到过一个需求,就是要开发一个对大量图片进行整理(删除掉一些不符合要求的图片).归类(根据格式进行分类,比如jpg格式.bmp格式等).压缩(因为有的图片很大很占空间,看 ...
- 这些小工具让你的Android 开发更高效
在做Android 开发过程中,会遇到一些小的问题.尽管自己动手也能解决.可是有了一些小工具,解决这些问题就得心应手了,今天就为大家推荐一下Android 开发遇到的小工具,来让你的开发更高效. Vy ...
- 【如何快速的开发一个完整的iOS直播app】(原理篇)
原文转自:袁峥Seemygo 感谢分享.自我学习 目录 [如何快速的开发一个完整的iOS直播app](原理篇) [如何快速的开发一个完整的iOS直播app](播放篇) [如何快速的开发一个完整的 ...
- 如何快速的开发一个完整的iOS直播app(原理篇)
目录 [如何快速的开发一个完整的iOS直播app](原理篇) [如何快速的开发一个完整的iOS直播app](播放篇) [如何快速的开发一个完整的iOS直播app](采集篇) 前言 大半年没写博客了,但 ...
- 【如何快速的开发一个完整的iOS直播app】(美颜篇)
原文转自:袁峥Seemygo 感谢分享.自我学习 前言 在看这篇之前,如果您还不了解直播原理,请查看这篇文章如何快速的开发一个完整的iOS直播app(原理篇) 开发一款直播app,美颜功能是很重 ...
随机推荐
- python实现批量自动访问站点URL并获取内容,自动模拟打开电脑端及移动端URL访问站点,打开URL页面获取页面内容
问题描述:假设目前有多个网站URL,需要检查各站点keyword,description是否正常设置,如果人工逐个打开URL访问比较耗时,故采用python模拟电脑端和移动端自动打开网站URL访问,并 ...
- MySQL索引最左原则:从原理到实战的深度解析
MySQL索引最左原则:从原理到实战的深度解析 一.什么是索引最左原则? 索引最左原则是MySQL复合索引使用的核心规则,简单来说: "当使用复合索引(多列索引)时,查询条件必须从索引的最左 ...
- rust学习笔记(1)
参考 rust圣经 参考 通过例子学习rust cargo 是rust的包管理器+编译工具 创建新项目 使用下述指令创建一个新的项目 cargo new rust_learn 执行 使用 cargo ...
- 玩three.js的一点心得
契机: 3-4月份,有机会再次学了一遍高数,然后再一次从二,三重积分的坑里爬来爬去,其中有个直观的问题一直困扰着我就是一个函数在空间坐标系上的图像,所以当时就打算学完这些之后,自己在5月份的时候用th ...
- 想查看某些网站源码,结果发现网站F12被禁用,怎么解决?
当我们访问某些网站的时候,发现网站是禁用了F12和右键功能的.比如想保存网页上的一些文字或图片等, 新手不知道怎么破除. 下面分享给大家几种方法:1.打开网页后,鼠标点进浏览器地址栏,再按F12键,就 ...
- 业余无线电爱好者,自制天线比较容易上手天线“莫克森天线”Moxon
本文仅作为笔记分享,如有疑问可以留言交流. 莫克森天线尺寸计算软件:Moxon rectangle 高手门做的成品,参考资料: 英文文献资料:
- BUUCTF---这是什么
题目 题目给出apk 解题
- [源码系列:手写spring] IOC第七节:加载xml文件中定义的Bean
目录 主要内容 代码分支 核心代码 BeanDefinitionReader AbstractBeanDefinitionReader XmlBeanDefinitionReader 测试 bean定 ...
- 英语面试-Behavioral Question - second part
前言 希望我总结的行为面试问题和答案能够给大家帮助. 学习方法:每个问题都有三部分组成. 第一部分是语料积累,这里是根据视频中的内容总结而来: 第二部分是中文描述,这里主要根据我自己的经历结合问题做出 ...
- 史上最全EffectiveJava总结(二)
方法 49.检查参数的有效性 每次编写方法或构造函数时,都应该考虑参数存在哪些限制,并在文档中记录下来,然后在方法的开头显式地检查. 如果没有在方法开头就验证参数,可能会违反故障原子性.因为方法可能会 ...