Dart-通过注解自动生成代码(SourceGen、BuildRunner)

使用过Dart自动Json序列化的都用过自动生成代码,但是你知道如何自己自定义注解并生成代码吗?

  1. 需要使用 source_gen、build_runner 这两个库;
  2. 定义注解类,配置注解到自动代码的Generator类及Builder类;

综上所述,咱需要构建两个Library库来使用注解自动化代码生成。

Library1: 注解类定义

Library2: 注解类的代码自动生成Generator及Builder

下面我们就说说怎么来做吧!

首先,假如我们需要实现一个自动生成MethodChannel调用Native方法的一个功能,这里我们叫它NativeCall吧。

  1. 新建NativeCall库,编写NativeCall注解的类。

    import 'package:native_call/native_method_call_info.dart';
    
    class NativeCall {
      /// ChannelName
      final String channelName;
      const NativeCall({this.channelName}); //注解类,构造函数必须是Const的
    }
    
  1. 新建NativeCallGen库,编写NativeCallGenerator, NativeCallBuilder,配置builder.yaml,该库需要依赖上面的NativeCall,需要引用第三方的库(即yaml文件中配置:SourceGenBuildRunner)

    dependencies:
      flutter:
        sdk: flutter
      source_gen: ^0.9.6 #需要包含代码自动库
      native_call:  #需要包含注解的库
        path: ../native_call 
    
    dev_dependencies:
      flutter_test:
        sdk: flutter
      build_runner: ^1.10.0  #需要引用该库
    

NativeCallGenerator,就是用来生成代码的,可以通过设置模板代码字符串等来输出。
下面通过解析ClassElement以及其内部的MethodElement来进行解析并得到需要自动化的代码内容字符串。
自定义类需要继承GeneratorForAnnotation类,并实现generateForAnnotationElement方法,该方法内即可编写相应的代码生成逻辑。

library native_call_gen;

import 'package:path/path.dart' as Path;
import 'package:native_call/native_call.dart';
import 'package:source_gen/source_gen.dart';
import 'package:analyzer/dart/element/element.dart';
// ignore: implementation_imports
import 'package:build/src/builder/build_step.dart' show BuildStep;

class NativeCallGenerator extends GeneratorForAnnotation<NativeCall> {
  @override
  generateForAnnotatedElement(
      Element element, ConstantReader annotation, BuildStep buildStep) {
    if (element is! ClassElement)
      throw InvalidGenerationSourceError('注解未使用在类上');

    String className = element.displayName;
    String path = buildStep.inputId.path;
    String channelName = annotation.peek('channelName').stringValue;

    StringBuffer sb = StringBuffer();

    (element as ClassElement).methods?.forEach((el) {
      var mName =
          String.fromCharCode(el.displayName.codeUnitAt(0)).toUpperCase() +
              el.displayName.substring(1);
      StringBuffer paramsStr = StringBuffer();
      StringBuffer params = StringBuffer();
      el.parameters?.forEach((param) {
        paramsStr.write('${param.type} ${param.name},');
        params.write('\'${param.name}\':' + param.name + ',');
      });
      sb.writeln('''
      ${el.hasImplicitReturnType == true ? 'Future<dynamic>' : 'void'} \$$className$mName(${paramsStr.toString()}) =>
       _methodChannel.invokeMethod('${el.displayName}', { ${params.toString()} });
      ''');
    });
    return '''
    part of '${Path.basename(buildStep.inputId.path)}';
    
    final MethodChannel _methodChannel = const MethodChannel('$channelName');


    ${sb.toString()}

    

    ''';
  }
}

NativeCallBuilder 生成代码的构造器,到时候build.yaml文件中会用到。

import 'package:native_call_gen/native_call_generator.dart';
import 'package:source_gen/source_gen.dart';
import 'package:build/build.dart';

Builder nativeCallBuilder(BuilderOptions options) =>
    LibraryBuilder(NativeCallGenerator(), generatedExtension: '.nc.g.dart');

build.yaml文件的配置

targets:
  $default: #定义目标库,关键字$default默认为当前库
    builders:
     natice_call_gen|native_call:  
        enabled: true #可选,是否将构建器应用于此目标
     source_gen|combining_builder:
      enabled: true

builders:
  native_call_builder:
    target: ":native_call_gen" #目标库
    import: 'package:native_call_gen/native_call_builder.dart'  #build文件
    builder_factories: ['nativeCallBuilder']
    build_extensions: {'.dart': ['.nc.g.dart']}
    auto_apply: dependents #将此Builder应用于包,直接依赖于公开构建起的包,也可以是root_package
    build_to: source #输出到注解的目标类的代码同目录中,或者输出转到隐藏的构建缓存,不会发布(cache)
    applies_builders: ["source_gen|combining_builder"] #指定是否可以延迟运行构建器
    # 以上参数具体参考 https://github.com/dart-lang/build/blob/master/build_config/README.md
  1. 新建一个测试项目,并引入NativeCallNativeCallGen库,配置如下:

    dependencies:
      flutter:
        sdk: flutter
      cupertino_icons: ^0.1.3
      native_call:  #引入NativeCall
        path: ../native_call/
    
    dev_dependencies:
      flutter_test:
        sdk: flutter
      build_runner: ^1.10.0 #引入builder_runner
      native_call_gen:  #引入NativeCallGen
        path: ../native_call_gen/
    

    测试类:AppNativeUtils

    import 'package:native_call/native_call.dart';
    import 'package:flutter/services.dart';
    
    part 'app_native_utils.nc.g.dart';  //包含该自动生成的part代码
    
    @NativeCall(channelName: 'com.china.mrper/utils/app_utils')  //使用NativeCall注解
    class AppNativeUtils {
      void showToast(String message, [int length]) =>  //定义toast方法
          $AppNativeUtilsShowToast(message, length);
    
      void showAlertDialog(String title, {message: String}) => //定义showAlertDialog的方法
          $AppNativeUtilsShowAlertDialog(title, message);
    }
    
    

    终端执行命令行,如下:**flutter package pub run build_runner build **

    生成如下代码:app_native_utils.nc.g.dart

    // GENERATED CODE - DO NOT MODIFY BY HAND
    
    // **************************************************************************
    // NativeCallGenerator
    // **************************************************************************
    
    part of 'app_native_utils.dart';
    
    final MethodChannel _methodChannel =
        const MethodChannel('com.china.mrper/utils/app_utils');
    
    void $AppNativeUtilsShowToast(
      String message,
      int length,
    ) =>
        _methodChannel.invokeMethod('showToast', {
          'message': message,
          'length': length,
        });
    
    void $AppNativeUtilsShowAlertDialog(
      String title,
      dynamic message,
    ) =>
        _methodChannel.invokeMethod('showAlertDialog', {
          'title': title,
          'message': message,
        });
    
    

至此,所有的流程已经完成,如果你觉得对你有用,请点个赞!
其他:转载请注明出处,感谢!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352