Bootstrap

Flutter: Websocket的使用与封装

1.引入相关插件库

  # websocket
  web_socket_channel: ^2.4.0
  # 引入rxdart 解决Bad state: Stream has already been listened to.的报错问题
  rxdart: ^0.27.7
  # 状态管理*
  provider: ^6.0.5

2.代码编写及封装

import 'dart:async';

import 'package:rxdart/subjects.dart';
import 'package:web_socket_channel/web_socket_channel.dart';
import 'package:web_socket_channel/status.dart' as status;

typedef WebsocketMessageCallback = void Function(dynamic message);

/// 注册流控制器需要在哪些页面使用
///
/// 目前分三种类型:
///
/// 1.[customerLoginPage]游客模式下也就是在未登录时候(用户处于登录相关页面)
/// 2.[customerMainPage]用户已登录,处于主页及其他登录后的页面下
/// 3.[chatRoomPage]用户处在聊天室里(游客下的在线客服聊天室、用户已登录下的在线客服聊天室、买卖用户之间的聊天室)
enum StreamControllerNameEnum {
  customerLoginPage,
  customerMainPage,
  chatRoomPage;
}

class WebsocketHelper {
  WebsocketHelper._();

  static WebsocketHelper? _singleton;

  factory WebsocketHelper() => _singleton ??= WebsocketHelper._();

/// 用于连接websocket的链接uri
  Uri? wsUri;

/// websocket连接后的对象
  WebSocketChannel? _webSocketChannel;

/// 指定的stream流控制器存放map
  Map<String, BehaviorSubject<String>>? streamControllerList;

/// 是否开启心跳
  bool isOpenHeartBeat = true;

/// 用于控制心跳轮询
  StreamSubscription<String>? _subscription;

/// 是否是用户主动触发关闭连接
  bool isDisconnectByUser = false;

/// 另辟一个单独的消息回调函数
  WebsocketMessageCallback? messageCallback;

/// 连接断开回调
  Function()? onDone;

/// 连接出错回调
  Function? onError;

  /// step one - ex: ws://localhost:1234
  initSocket({required String wsPath, bool isOpenHeartBeat = true}) {
    if (_webSocketChannel != null) {
      print("socket实例已存在, 请勿重复创建");
      return;
    }

    // 自己项目中后端需要前端拼一个登录令牌用于控制后端逻辑处理,这里使用的是登录后的token
    var authorization = "登录后的token";
    wsUri = Uri.tryParse("$wsPath?Authorization=$authorization");
    // wsUri = Uri.tryParse(wsPath);
    if (wsUri == null) return;
    this.isOpenHeartBeat = isOpenHeartBeat;
    _connectWebsocket();
  }

  /// [isRunForReConnect] 是否是由重连机制触发的此方法
  void _connectWebsocket({bool isRunForReConnect = false}) {
    _webSocketChannel = WebSocketChannel.connect(wsUri!);
    if (!isRunForReConnect) {
      isDisconnectByUser = false;
    }
  }

  /// step two - listen
  void listen(
      {WebsocketMessageCallback? messageCallback,
      Function()? onDone,
      Function? onError}) {
    this.messageCallback = messageCallback;
    this.onDone = onDone;
    this.onError = onError;
    streamControllerList ??= <String, BehaviorSubject<String>>{
      // StreamControllerNameEnum.customerLoginPage.name: BehaviorSubject(),
      // StreamControllerNameEnum.customerMainPage.name: BehaviorSubject(),
      // StreamControllerNameEnum.chatRoomPage.name: BehaviorSubject()
    };

    // 监听一系列连接情况(如收到消息、onDone:连接关闭、onError:接连异常)
    _webSocketChannel?.stream.listen((message) {
      print(
          "websocket onData message = ${message.toString()}, type = ${message.runtimeType}");
      if (message is String && message.isEmpty) {
        // 消息为空(可能得情况:心跳 or another)
        return;
      }
      // 通过流控制器把消息分发出去,在需要的页面监听此流的消息
      streamControllerList?.forEach((key, value) {
        // print("key = $key, value.isClosed = ${value.isClosed}");
        if (!value.isClosed) {
          value.sink.add(message);
        }
      });
      this.messageCallback?.call(message);
    }, onDone: () {
      print("websocket onDone ...");
      this.onDone?.call();
      // 掉线重连
      reConnect();
    }, onError: (Object error, StackTrace stackTrace) {
      print(
          "websocket onError error = ${error.toString()}, stackTrace = ${stackTrace.toString()}");
      showToast(msg: "连接服务器失败!");
      this.onError?.call(error, stackTrace);
    }, cancelOnError: false);
    // 连接建立成功时的回调通知,可在此做心跳操作
    _webSocketChannel?.ready.then((value) {
      print("webSocket ready");
      isDisconnectByUser = false;
      if (isOpenHeartBeat) {
        // 收到连接成功的回馈,开始执行心跳操作
        _startHeartBeat();
      }
    });
  }

  /// 掉线重连
  void reConnect() {
    if (isDisconnectByUser) return;
    Future.delayed(
      const Duration(seconds: 3),
      () {
        // disconnect();
        _subscription?.cancel();
        _subscription = null;
        _webSocketChannel?.sink.close(status.abnormalClosure, "掉线重连");
        _webSocketChannel = null;
        _connectWebsocket();
        listen(
            messageCallback: messageCallback, onDone: onDone, onError: onError);
      },
    );
  }

/// 发送消息
  void sendMessage({required String message, bool needDisplayMsg = true}) {
    print("websocket sendMessage message = $message");
    if (needDisplayMsg) {
      streamControllerList?.forEach((key, value) {
        if (!value.isClosed) {
          value.sink.add(message);
        }
      });
    }

    _webSocketChannel?.sink.add(message);
  }

/// 开启心跳
  void _startHeartBeat() {
    if (_subscription != null) {
      print("websocket startHeartBeat _subscription != null");
      return;
    }
    Future.delayed(
      const Duration(seconds: 30),
      () {
        var pollingStream = StreamTool().timedPolling(
            const Duration(seconds: 30), () => Future(() => ""), 100000000);
        //进行流内容监听
        _subscription = pollingStream.listen((result) {
          sendMessage(message: "heart beat", needDisplayMsg: false);
        });
      },
    );
  }

/// 断开连接并销毁
  void disconnect({bool isDisconnectByUser = false}) {
    this.isDisconnectByUser = isDisconnectByUser;
    _subscription?.cancel();
    _subscription = null;
    streamControllerList?.forEach((key, value) {
      value.close();
    });
    streamControllerList?.clear();
    _webSocketChannel?.sink.close(status.normalClosure, "用户退出聊天界面,聊天关闭");
    _webSocketChannel = null;
  }

/// 新建指定stream流控制器进行消息流回调
  setNewStreamController(StreamControllerNameEnum streamControllerName) {
    if (streamControllerList?.containsKey(streamControllerName.name) ?? false) {
      streamControllerList?[streamControllerName.name]?.close();
    }
    streamControllerList?[streamControllerName.name] = BehaviorSubject();
  }
}

3.提供一个轮询工具类StreamTool

import 'dart:async';

typedef FutureGenerator<T> = Future<T> Function();

class StreamTool {
  /// interval 轮询时间间隔
  /// maxCount 最大轮询数
  Stream<T> timedPolling<T>(Duration interval, FutureGenerator<T> future,
      [int maxCount = 1]) {
    StreamController<T>? controller;
    int counter = 0;
    bool polling = true;

    void stopTimer() {
      polling = false;
    }

    void tick() async {
      counter++;
      T result = await future();
      if (controller != null && !controller.isClosed) {
        controller.add(result);
      }
      if (counter == maxCount) {
        stopTimer();
        controller?.close();
      } else if (polling) {
        Future.delayed(interval, tick);
      }
    }

    void startTimer() {
      polling = true;
      tick();
    }

    //StreamSubscription调用pause,cancel时,stream里面的轮询也能响应暂停或取消
    controller = StreamController<T>(
      onListen: startTimer,
      onPause: stopTimer,
      onResume: startTimer,
      onCancel: stopTimer,
    );

    return controller.stream;
  }
}

4.新建全局的ChangeNotifier -> GlobalWebsocketVM

class GlobalWebsocketVM extends ChangeNotifier {
void startWebSocket() {
    WebsocketHelper()
      ..initSocket(wsPath: Api.wsUrlPath, isOpenHeartBeat: false)
      ..listen(
        messageCallback: (message) {
          // 延迟500毫秒,使listview进行滑动到底部
         // gotoListBottom();
        },
        onDone: () {},
      );
  }

 /// 获取socket实时数据流
  ///
  /// 每次都需要新绑定一个StreamController,避免数据流出现错乱情况
  Stream<String>? getMessageStream(
          StreamControllerNameEnum streamControllerName) =>
      (WebsocketHelper()..setNewStreamController(streamControllerName))
          .streamControllerList?[streamControllerName.name]
          ?.stream;
}

5.在入口类main.dart中MaterialApp中使用全局GlobalWebsocketVM

late GlobalWebsocketVM socketVM;

@override
void initState() {
  socketVM = GlobalWebsocketVM();
}

MaterialApp.router(
      debugShowCheckedModeBanner: false,
      onGenerateTitle: (context) => S.current.appName,
      theme: ThemeData(
        useMaterial3: true,
        colorScheme: ColorScheme.fromSeed(seedColor: Colors.white),
        appBarTheme: const AppBarTheme(
            color: Colors.white, surfaceTintColor: Colors.white),
        bottomAppBarTheme: BottomAppBarTheme.of(context)
            .copyWith(color: Colors.white, surfaceTintColor: Colors.white),
        scaffoldBackgroundColor: Colors.grey[200],
        cardTheme: const CardTheme(
            color: Colors.white, surfaceTintColor: Colors.white),
        progressIndicatorTheme:
            const ProgressIndicatorThemeData(color: AppColor.appThemeColor),
        // 统一修改输入框光标颜色、文本选中颜色
        textSelectionTheme: const TextSelectionThemeData(
          cursorColor: AppColor.appThemeColor,
          selectionColor: AppColor.appThemeColor,
          selectionHandleColor: AppColor.appThemeColor,
        ),
        // ios主题色设置
        cupertinoOverrideTheme:
            const CupertinoThemeData(primaryColor: AppColor.appThemeColor),
        iconButtonTheme: IconButtonThemeData(
          style: AppButtonStyle.stGlobalDefaultBtn,
        ),
        textButtonTheme: TextButtonThemeData(
          style: AppButtonStyle.stGlobalDefaultBtn,
        ),
        // primarySwatch: themeVM.theme,
      ),
      // locale: localeVM.getLocale(),
      builder: FlutterSmartDialog.init(
        builder: (context, mChild) {
          return MultiProvider(
            providers: [
              ChangeNotifierProvider<UserVM>(
                create: (_) => userVM,
              ),
              // 在这里为每个页面添加GlobalWebsocketVM绑定
              ChangeNotifierProvider<GlobalWebsocketVM>(
                create: (_) => socketVM,
              ),
            ],
            builder: (context, child) => mChild ?? const SizedBox.shrink(),
          );
          /*return ChangeNotifierProvider<UserVM>(
            create: (_) => userVM,
            builder: (context, child) =>
            mChild ?? const SizedBox.shrink(),
          );*/
        },
      ),
      localizationsDelegates: const [
        S.delegate,
        GlobalMaterialLocalizations.delegate,
        GlobalWidgetsLocalizations.delegate,
        GlobalCupertinoLocalizations.delegate,
      ],
      supportedLocales: S.delegate.supportedLocales,
      // localeResolutionCallback: (_locale, supportedLocales) {
      //   if (localeVM.getLocale() != null) {
      //     //如果已经选定语言,则不跟随系统
      //     return localeVM.getLocale();
      //   } else {
      //     //跟随系统
      //     Locale locale;
      //     if (supportedLocales.contains(_locale)) {
      //       locale = _locale!;
      //     } else {
      //       //如果系统语言不是中文简体或美国英语,则默认使用美国英语
      //       locale = const Locale('en', 'US');
      //     }
      //     return locale;
      //   }
      // },
      routerConfig: RouterHelper.router,
    );

6.页面中调用,在initState方法中建立连接,在build中使用StreamBuilder进行消息监听

@override
  void initState() {
    super.initState();
    // 建立连接ws global init
    context.read<GlobalWebsocketVM>().startWebSocket();
    
  }

@override
  Widget build(BuildContext context) {
    return Scaffold(
            resizeToAvoidBottomInset: false,
            appBar: TitleBar.build(title: "正与${model.titleContent}沟通"),
            // 监听聊天消息并刷新聊天列表
            body: StreamBuilder<String>(
          stream: context
        .read<GlobalWebsocketVM>()
        .getMessageStream(StreamControllerNameEnum.chatRoomPage),
          builder: (context, snapshot) {
            if (snapshot.connectionState == ConnectionState.active) {
              if (snapshot.data?.isEmpty ?? true) {
                return const SizedBox.shrink();
              }
              addMessageAndRefreshUI("orderNo", snapshot.data!);
              return const SizedBox.shrink();
            }
            return const SizedBox.shrink();
          },
          // catchError: (context, error) => error.toString(),
        ),,
          );
}

void addMessageAndRefreshUI(String tag, String message) {
    print("收到聊天消息:" + message);
}

7.发送消息

/// 在合适的地方(比如发送按钮点击发送聊天消息)
void sendChatMessage() {
WebsocketHelper().sendMessage(
      message: "我发送一条消息",
      needDisplayMsg: false,
    );
}

8.退出app断开websocket清理内存(可以在任何想断开websocket的地方调用销毁)

/// 通常在dispose中调用销毁,可以在任何想断开websocket的地方调用销毁
@override
  void dispose() {
    ScanHelper().dispose();
    WebsocketHelper().disconnect(isDisconnectByUser: true);
    super.dispose();
  }

;