2024-08-23



import 'package:flutter/material.dart';
import 'package:flutter_markdown/flutter_markdown.dart';
 
class UserQueryService {
  final String userId;
  final String token;
 
  UserQueryService({@required this.userId, @required this.token});
 
  Future<String> getUserProfileMarkdown() async {
    // 模拟从服务器获取用户信息的函数
    return """
    | 用户ID | 用户名 | 邮箱 |
    | --- | --- | --- |
    | ${userId} | 张三 | zhangsan@example.com |
    """;
  }
 
  Future<Widget> getUserProfileWidget() async {
    String markdown = await getUserProfileMarkdown();
    return MarkdownBody(data: markdown);
  }
}
 
// 使用示例
void main() {
  runApp(MaterialApp(
    home: Scaffold(
      body: Center(
        child: FutureBuilder<Widget>(
          future: UserQueryService(userId: '123', token: 'token').getUserProfileWidget(),
          builder: (context, snapshot) {
            if (snapshot.hasData) {
              return snapshot.data;
            } else {
              return CircularProgressIndicator();
            }
          },
        ),
      ),
    ),
  ));
}

这个代码示例展示了如何使用Flutter\_markdown包来展示从模拟的用户查询服务中获取的Markdown格式的用户信息。在实际应用中,你需要替换getUserProfileMarkdown方法以实际从服务器获取信息,并在getUserProfileWidget中返回相应的Widget。

2024-08-23

Flutter Hooks 是一个实验性的特性,它允许在函数组件中使用状态管理。这是一个在不需要完全重建组件树的情况下,可以复用状态逻辑的方法。

以下是一个简单的使用 useState Hook 的 Flutter 示例:




import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
 
void main() {
  runApp(HookExampleApp());
}
 
class HookExampleApp extends HookWidget {
  @override
  Widget build(BuildContext context) {
    final counter = useState(0);
 
    return MaterialApp(
      home: Scaffold(
        body: Center(
          child: Text(
            'You have pushed the button this many times: ${counter.value}',
          ),
        ),
        floatingActionButton: FloatingActionButton(
          onPressed: () => counter.value++,
          child: Icon(Icons.add),
        ),
      ),
    );
  }
}

在这个例子中,我们创建了一个计数器应用,使用 useState Hook 来管理计数状态。每次点击浮动按钮,计数器的值就会增加,并且通过 Text 组件实时更新显示的计数值。这是一个简单的 Hooks 使用案例,展示了状态管理而不需要完全重建组件。

2024-08-23

在Flutter中,我们可以使用image_picker插件来选择图片,使用image包来处理图片,并使用flutter_image_compress来压缩图片。以下是一个简化的例子,展示了如何在Flutter中实现图片的缩放和裁剪功能:




import 'package:flutter/material.dart';
import 'package:image_picker/image_picker.dart';
import 'package:flutter_image_compress/flutter_image_compress.dart';
 
class ImageCropperPage extends StatefulWidget {
  @override
  _ImageCropperPageState createState() => _ImageCropperPageState();
}
 
class _ImageCropperPageState extends State<ImageCropperPage> {
  File _imageFile;
 
  Future<void> _cropImage() async {
    File croppedFile = await ImageCropper.cropImage(
      sourcePath: _imageFile.path,
      compressQuality: 10, // 图片压缩质量
      maxWidth: 1000, // 最大宽度
      maxHeight: 1000, // 最大高度
      aspectRatio: CropAspectRatio(ratio: 1 / 1), // 裁剪比例
    );
    setState(() {
      _imageFile = croppedFile;
    });
  }
 
  Future<void> _pickImage() async {
    final ImagePicker picker = ImagePicker();
    final PickedFile imageFile = await picker.getImage(source: ImageSource.gallery);
    if (imageFile != null) {
      setState(() {
        _imageFile = File(imageFile.path);
      });
    }
  }
 
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text('图片裁剪'),
      ),
      body: Center(
        child: _imageFile != null ? Image.file(_imageFile) : Text('未选择图片'),
      ),
      floatingActionButton: Column(
        crossAxisAlignment: CrossAxisAlignment.end,
        mainAxisAlignment: MainAxisAlignment.end,
        children: <Widget>[
          FloatingActionButton(
            onPressed: _pickImage,
            tooltip: 'Pick Image',
            child: Icon(Icons.add_a_photo),
          ),
          SizedBox(height: 10.0),
          FloatingActionButton(
            onPressed: _imageFile != null ? _cropImage : null,
            tooltip: 'Crop Image',
            child: Icon(Icons.crop),
          ),
        ],
      ),
    );
  }
}

在这个例子中,我们首先使用ImagePicker来选择图片,然后使用ImageCropper来裁剪图片。裁剪后的图片可以进一步进行压缩,以减少文件大小。这个例子提供了一个简单的用户界面,用户可以通过点击两个浮动按钮来选择图片和裁剪图片。裁剪后的图片会显示在屏幕中央。

2024-08-23



import 'package:flutter/material.dart';
import 'package:provider/provider.dart';
 
void main() {
  runApp(MyApp());
}
 
class MyApp extends StatelessWidget {
  // This widget is the root of your application.
  @override
  Widget build(BuildContext context) {
    return MultiProvider(
      providers: [
        ChangeNotifierProvider(create: (_) => Counter()),
      ],
      child: MaterialApp(
        title: 'Flutter Demo',
        home: MyHomePage(),
      ),
    );
  }
}
 
class Counter with ChangeNotifier {
  int _count = 0;
  int get count => _count;
 
  void increment() {
    _count++;
    notifyListeners();
  }
}
 
class MyHomePage extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text("Flutter Provider Example"),
      ),
      body: Center(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: <Widget>[
            Text(
              'You have pushed the button this many times:',
            ),
            Consumer<Counter>(
              builder: (context, counter, child) => Text(
                '${counter.count}',
                style: Theme.of(context).textTheme.headline4,
              ),
            ),
          ],
        ),
      ),
      floatingActionButton: FloatingActionButton(
        onPressed: () => Provider.of<Counter>(context, listen: false).increment(),
        tooltip: 'Increment',
        child: Icon(Icons.add),
      ),
    );
  }
}

这段代码使用了Provider包来管理状态,它展示了如何创建一个简单的计数器应用程序。在这个例子中,我们使用ChangeNotifier类来包装我们的状态,并在需要时通过notifyListeners()来通知监听器状态已更新。我们使用Consumer小部件来响应状态的变化,并且在FloatingActionButtononPressed回调中,我们通过Provider.of来获取Counter实例并调用increment方法。这是一个很基础的状态管理示例,但它展示了如何在实际应用程序中使用Provider来管理和响应状态的变化。

2024-08-23

Flutter 是一种创建跨平台应用的开源技术,它依赖于Dart语言。虽然Flutter本身不直接处理加密或安全性,但它可以集成各种加密库来保护数据。

在Flutter中处理加密的一种方法是使用第三方库,例如点(encrypt)和flutter\_secure\_storage。这些库提供了一种简单的方式来加密和存储敏感数据。

以下是使用encrypt库进行AES加密的一个例子:




import 'package:encrypt/encrypt.dart';
 
void main() {
  final key = Key.fromUtf8('your-256-bit-secret-key'); // 256-bit key for AES
  final encrypter = Encrypter(AES(key));
  
  final encrypted = encrypter.encrypt('Hello, World!');
  print(encrypted.base64);
  
  final decrypted = encrypter.decrypt(encrypted);
  print(decrypted);
}

另一个例子是使用flutter\_secure\_storage库来安全地存储数据:




import 'package:flutter_secure_storage/flutter_secure_storage.dart';
 
void main() async {
  final storage = FlutterSecureStorage();
  
  // 保存数据
  await storage.write(key: 'token', value: 'hello_world_secret');
  
  // 读取数据
  String token = await storage.read(key: 'token');
  print('Token: $token');
  
  // 删除数据
  await storage.delete(key: 'token');
}

这些库都提供了一些基本的加密功能,并且都是开源的,这意味着你可以查看它们的实现细节,以确保它们满足你的安全需求。

在实际应用中,你还需要考虑其他安全因素,如HTTPS通信、输入验证、错误处理等,以保障应用的安全性。

2024-08-23



# 使用Flutter官方提供的镜像作为基础镜像
FROM flutter:latest
 
# 设置工作目录
WORKDIR /app
 
# 复制项目文件到工作目录
COPY . /app
 
# 运行Flutter打包命令,生成发布版本的应用
RUN flutter build apk
 
# 使用官方提供的Google ChromeDocker镜像作为基础镜像
FROM google/chrome:latest
 
# 设置工作目录
WORKDIR /app
 
# 将Flutter打包生成的APK复制到Chrome容器中
COPY --from=0 /app/build/app/outputs/apk/release/app-release.apk /app/
 
# 运行Chrome容器,并启动Flutter应用
CMD ["/usr/bin/chromium-browser", "--app=http://localhost:8080", "--no-sandbox", "--user-data-dir"]

这个Dockerfile实例展示了如何将一个Flutter应用容器化,并且将生成的APK文件运行在Google Chrome容器中。这个例子简洁明了,并且清晰地展示了容器化的过程。

2024-08-23

Flutter 应用程序启动时可能会出现白屏或者闪屏问题,这是因为 Flutter 还没有完全初始化就已经显示了默认的窗口。flutter_native_splash 是一个第三方库,旨在解决这个问题。

使用 flutter_native_splash 库可以创建一个启动屏幕(splash screen),在 Flutter 应用程序完全启动之前,显示一个自定义的图像或颜色。

以下是如何使用 flutter_native_splash 库的基本步骤:

  1. flutter_native_splash 添加到你的 pubspec.yaml 文件的依赖中:



dependencies:
  flutter:
    sdk: flutter
  flutter_native_splash: ^1.3.1
  1. 在你的 lib/main.dart 文件中,确保你已经移除或注释掉了 native_splash.dart 中的配置代码。
  2. 运行 flutter pub get 来安装新的依赖。
  3. 使用 flutter_native_splash 提供的命令来生成启动屏幕配置文件。例如,对于 Android,你可以使用:



flutter pub run flutter_native_splash:create --image launch_image.png

这里的 launch_image.png 是你想要作为启动屏幕的图片文件路径。

  1. 根据提示完成其他必要的配置,例如在 AndroidManifest.xml 中添加必要的元数据。
  2. 确保你的 AndroidManifest.xmlInfo.plist 文件中没有任何与启动屏幕相关的不一致的配置。
  3. 运行你的 Flutter 应用程序,现在应该能看到你设置的启动屏幕。

注意:在实际使用中,你可能需要根据你的项目具体情况来调整上述步骤。例如,你可能需要为不同的设备分辨率和屏幕尺寸提供不同尺寸的启动屏幕图片。此外,flutter_native_splash 库可能不支持所有平台(如 web),你需要检查其文档以确认支持情况。

2024-08-23

在Flutter中,如果你想要禁止某些文本随系统字体大小缩放,你可以使用MediaQuery来获取当前的字体缩放比例,并手动应用这个比例来覆盖系统的字体缩放。

以下是一个简单的例子,展示如何在Text组件中禁止字体缩放:




import 'package:flutter/material.dart';
 
void main() => runApp(MyApp());
 
class MyApp extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      home: HomePage(),
    );
  }
}
 
class HomePage extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    // 获取当前系统的字体缩放比例
    double textScaleFactor = MediaQuery.of(context).textScaleFactor;
 
    return Scaffold(
      body: Center(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: <Widget>[
            // 应用系统字体缩放比例
            Text(
              '这段文字会缩放',
              style: TextStyle(
                fontSize: 16 * textScaleFactor,
              ),
            ),
            // 忽略系统字体缩放比例
            Text(
              '这段文字不会缩放',
              style: TextStyle(
                fontSize: 16,
              ),
              textScaleFactor: 1.0,
            ),
          ],
        ),
      ),
    );
  }
}

在这个例子中,第一个Text组件会根据系统的字体缩放比例自动调整字体大小,而第二个Text组件通过设置textScaleFactor: 1.0属性,直接忽略系统的字体缩放比例,保持其字体大小不变。

2024-08-23

在Flutter项目中配置极光推送,需要按照以下步骤进行:

  1. android/app/build.gradle中配置小米厂商通道:



android {
    ...
    defaultConfig {
        ...
        ndk {
            // 设置支持的SO库架构
            abiFilters 'armeabi-v7a', 'x86'
        }
        manifestPlaceholders = [
            JPUSH_APPKEY: "你的极光APPKEY", //JPush 上注册的包名对应的 Appkey.
            JPUSH_CHANNEL: "developer-default", //推送通道.
            // 小米厂商通道
            MI_APP_KEY: "你的小米APPKEY", // 小米开放平台获取的AppKey.
            MI_APP_ID: "你的小米APPID", // 小米开放平台获取的AppID.
        ]
    }
    ...
}
  1. android/app/src/main/AndroidManifest.xml中配置小米厂商通道权限和特性:



<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="你的包名">
    ...
    <uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.READ_PHONE_STATE" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
    
    ...
    <application
        ...
        android:name="${applicationName}">
        ...
        <!-- 小米厂商通道配置 -->
        <meta-data android:name="MI_APP_ID" android:value="${MI_APP_ID}"/>
        <meta-data android:name="MI_APP_KEY" android:value="${MI_APP_KEY}"/>
        
        ...
    </application>
</manifest>
  1. lib/main.dart或其他适当的位置初始化极光推送:



import 'package:jpush_flutter/jpush_flutter.dart';
 
void main() {
  // 初始化极光推送
  JPush jpush = new JPush();
  // 获取设备注册ID,用于注册极光推送服务
  jpush.getRegistrationID((rid) {
    print('JPush RegistrationID: $rid');
  });
 
  // 其他初始化代码
  runApp(MyApp());
}

确保在实际的项目中替换你的包名你的极光APPKEY你的小米APPKEY你的小米APPID为实际值。

以上步骤配置了小米厂商通道,确保您已经按照极光推送的官方文档完成了其他必要配置,例如在AndroidManifest.xml中配置了必要的权限和服务。如果您在实际操作过程中遇到任何问题,欢迎提问。

2024-08-23

在Flutter中接入现有App通常涉及以下步骤:

  1. 设置Flutter环境:确保你的开发环境中已安装Flutter SDK和所需的开发工具。
  2. 添加原生代码集成:如果你需要调用原生API或集成现有的原生模块,你需要在Android和iOS项目中添加相应的原生代码和配置。
  3. 使用Flutter插件:如果现有的功能可以通过Flutter插件来实现,那么你可以直接在Flutter项目中添加对应的插件依赖。
  4. Method Channel通信:如果需要在Flutter和原生代码之间进行更复杂的交互,可以使用Method Channel进行跨平台通信。
  5. 修改现有App的启动流程:根据你的需求,可能需要修改现有App的启动流程,以便在启动时集成Flutter模块。

MMKV原理:MMKV 是基于 mmap 内存映射的 key-value 存储引擎,它的设计目标是替代 iOS 上的 NSUserDefaults 与 Android 的 SharedPreferences,它具有以下特点:

  • 性能:使用 mmap 进行内存映射,读写数据不会导致 CPU 的上下文切换和磁盘 I/O,性能较高。
  • 多进程支持:MMKV 支持多进程同时读写,数据以文件形式保存在内存中。
  • 安全性:数据加密存储,提供了基于 AES CFB 的加密能力,保证了数据的安全性。
  • 小巧:库文件大小较小,iOS 版本不到 100KB,Android 版本不到 200KB。

集成MMKV到Flutter项目

  1. 添加MMKV依赖:在Android和iOS项目中添加MMKV的依赖。
  2. 初始化MMKV:在Flutter项目的入口点(如main.dart中)初始化MMKV。
  3. 使用MMKV进行数据存取:通过Method Channel调用MMKV的方法进行数据的存取。

示例代码:




// main.dart
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
 
void main() {
  WidgetsFlutterBinding.ensureInitialized();
  SystemChannels.lifecycle.setMessageHandler((message) {
    // 处理生命周期事件,例如在这里初始化MMKV
  });
 
  runApp(MyApp());
}
 
class MyApp extends StatelessWidget {
  // 使用MMKV进行数据存取的方法
  Future<void> setMMKV(String key, String value) async {
    const platform = MethodChannel('com.example.mmkv_plugin');
    try {
      await platform.invokeMethod('setMMKV', <String, dynamic>{
        'key': key,
        'value': value,
      });
    } on PlatformException catch (e) {
      print("Failed to set MMKV: '${e.message}'.");
    }
  }
 
  Future<String> getMMKV(String key) async {
    const platform = MethodChannel('com.example.mmkv_plugin');
    try {
      String value = await platform.invokeMethod('getMMKV', key);
      return value;
    } on PlatformException catch (e) {
      print("Failed to get MMKV: '${e.message}'.");
      return null;