合并中的RxSwift`ActivityIndi​​cator`功能

如何解决合并中的RxSwift`ActivityIndi​​cator`功能

我已经与RxSwift合作了几年,并且开始探索与SwiftUI结合使用,并且在尝试从Combine中的RxSwift复制某些功能时遇到一些麻烦。

在RxSwift GitHub上,有一个示例名为ActivityIndicator.swift

基本用法如下:

class Foo {
  let activityIndicator = ActivityIndicator()

  lazy var activity = activityIndicator.asDriver()

  var disposeBag = DisposeBag()

  func doSomething() {
    Observable
      .just("this is something")
      .trackActivity(activityIndicator)
      .subscribe()
      .disposed(by: disposeBag)
  }
}

它的作用是让您从activity驱动程序中脱离出来,每当有订阅或完成订阅时,它将发出布尔值。

然后您可以使用RxCocoa直接驱动类似UIActivityIndi​​catorView的isAnimating属性之类的东西。

我一直在尝试找出如何在Combine中创建与此类似的内容,但是没有任何运气。

说我有一个如下所示的viewModel:

class ViewModel: ObservableObject {
  @Published var isActive = false

  func doSomething() -> AnyPublisher<Void,Never> {
    Just(())
      .delay(for: 2.0,scheduler: RunLoop.main)
      .eraseToAnyPublisher()
  }
}

我想做的是为Publisher创建一个运算符,该运算符的作用类似于Rx运算符的工作原理,即我可以通过链转发订阅中的事件,但更改{{1} }每次订阅/完成/取消都会有价值。

在SwiftUI视图中,我将启动isActive函数并沉入其中,同时还能够使用已发布的doSomething属性显示/隐藏isActive

类似的东西:

ProgressView

我已经完全想念了吗?

如果没有,我该如何在Combine中复制RxSwift功能?

预先感谢您的帮助。

解决方法

嗯... ActivityIndicator类的关键是Observable.using(_:observableFactory:)运算符。不幸的是,我不认为Combine中有等效的运算符。

在订阅Observable时,using运算符创建资源,然后在Observable发送stop事件(完成或错误)时处理资源。这确保了资源的生命周期。在这种情况下,资源只会在创建时增加一个Int值,而在处置时减少它。

我认为您可以通过以下方式模仿行为:

extension Publisher {
    func trackActivity(_ activityIndicator: CombineActivityIndicator) -> some Publisher {
        return activityIndicator.trackActivity(of: self)
    }
}

final class CombineActivityIndicator {
    var counter = CurrentValueSubject<Int,Never>(0)
    var cancelables = Set<AnyCancellable>()

    func trackActivity<Source: Publisher>(of source: Source) -> some Publisher {
        let sharedSource = source.share()
        counter.value += 1
        sharedSource
            .sink(
                receiveCompletion: { [unowned self] _ in
                    self.counter.value -= 1
                },receiveValue: { _ in }
            )
            .store(in: &cancelables)
        return sharedSource
    }

    var asPublisher: AnyPublisher<Bool,Never> {
        counter
            .map { $0 > 0 }
            .eraseToAnyPublisher()
    }
}

但是,上面的类将使Publisher发热,因此您可能会错过发出的值。使用自担风险,除非您绝望,否则我不建议您使用以上内容。

也许有人为Publisher编写了using运算符,并且愿意共享。

,

看起来有人创建了一个组合版本。我不知道它是否与@Daniel T. 讨论的问题相同,但看起来很有希望。 https://github.com/duyquang91/ActivityIndicator

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


依赖报错 idea导入项目后依赖报错,解决方案:https://blog.csdn.net/weixin_42420249/article/details/81191861 依赖版本报错:更换其他版本 无法下载依赖可参考:https://blog.csdn.net/weixin_42628809/a
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下 2021-12-03 13:33:33.927 ERROR 7228 [ main] o.s.b.d.LoggingFailureAnalysisReporter : *************************** APPL
错误1:gradle项目控制台输出为乱码 # 解决方案:https://blog.csdn.net/weixin_43501566/article/details/112482302 # 在gradle-wrapper.properties 添加以下内容 org.gradle.jvmargs=-Df
错误还原:在查询的过程中,传入的workType为0时,该条件不起作用 &lt;select id=&quot;xxx&quot;&gt; SELECT di.id, di.name, di.work_type, di.updated... &lt;where&gt; &lt;if test=&qu
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct redisServer’没有名为‘server_cpulist’的成员 redisSetCpuAffinity(server.server_cpulist); ^ server.c: 在函数‘hasActiveC
解决方案1 1、改项目中.idea/workspace.xml配置文件,增加dynamic.classpath参数 2、搜索PropertiesComponent,添加如下 &lt;property name=&quot;dynamic.classpath&quot; value=&quot;tru
删除根组件app.vue中的默认代码后报错:Module Error (from ./node_modules/eslint-loader/index.js): 解决方案:关闭ESlint代码检测,在项目根目录创建vue.config.js,在文件中添加 module.exports = { lin
查看spark默认的python版本 [root@master day27]# pyspark /home/software/spark-2.3.4-bin-hadoop2.7/conf/spark-env.sh: line 2: /usr/local/hadoop/bin/hadoop: No s
使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams[&#39;font.sans-serif&#39;] = [&#39;SimHei&#39;] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -&gt; systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping(&quot;/hires&quot;) public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate&lt;String
使用vite构建项目报错 C:\Users\ychen\work&gt;npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-