Weex扩展-原生刷新功能

Weex SDK中提供了下拉刷新和上拉加载的组件,但是功能比较单一,不能做到跟手滑动操作,还会导致weex刷新和原生刷新不统一的问题,所以最好的操作是扩展导出原生的刷新组件。

refresh组件是WeexSDK官方提供的,但是不好在这个组件内进行扩展,为了方便前端的使用和客户端的扩展,最终在list组件的基础上进行扩展,因为几乎所有的刷新和加载都是基于list组件的。

iOS端

为了方便扩展,我选择继承官方的WXListComponent类,先整体了解一下整个类中的代码

@interface MWSListComponent () 

@property (nonatomic, strong) MJRefreshGifHeader *refreshHeader;
@property (nonatomic, strong) MJRefreshAutoFooter *refreshFooter;
@property (nonatomic, assign) BOOL refresh;             /**< 是否开启下拉刷新 */
@property (nonatomic, assign) BOOL loading;             /**< 是否开启上拉加载 */
@property (nonatomic, assign) BOOL showLoading;         /**< 控制loading是否显示 */
@property (nonatomic, assign) BOOL refreshEvent;
@property (nonatomic, assign) BOOL loadingEvent;

@end

@implementation MWSListComponent

WX_EXPORT_METHOD(@selector(endRefreshing));
WX_EXPORT_METHOD(@selector(endLoading));
WX_EXPORT_METHOD(@selector(noticeNoMoreData));

- (void)dealloc
{
    _refreshFooter = nil;
    _refreshHeader = nil;
}

- (instancetype)initWithRef:(NSString *)ref type:(NSString *)type styles:(NSDictionary *)styles attributes:(NSDictionary *)attributes events:(NSArray *)events weexInstance:(WXSDKInstance *)weexInstance
{
    self = [super initWithRef:ref type:type styles:styles attributes:attributes events:events weexInstance:weexInstance];
    if (self) {
        _refresh = [WXConvert BOOL:attributes[@"refresh"]];
        _loading = [WXConvert BOOL:attributes[@"loading"]];
        _showLoading = [attributes.allKeys containsObject:@"showLoading"] ? [WXConvert BOOL:attributes[@"showLoading"]] : YES;
    }
    return self;
}

- (void)viewDidLoad
{
    [super viewDidLoad];
    [self _updateRefreshHeader];
    [self _updateRefreshFooter];
    [self _updateLoadingState];
}


- (void)addEvent:(NSString *)eventName
{
    [super addEvent:eventName];
    if ([eventName isEqualToString:@"refresh"]) {
        _refreshEvent = YES;
    }
    if ([eventName isEqualToString:@"loading"]) {
        _loadingEvent = YES;
    }
}

- (void)removeEvent:(NSString *)eventName
{
    [super removeEvent:eventName];
    if ([eventName isEqualToString:@"refresh"]) {
        _refreshEvent = NO;
    }
    if ([eventName isEqualToString:@"loading"]) {
        _loadingEvent = NO;
    }
}

- (void)updateAttributes:(NSDictionary *)attributes
{
    [super updateAttributes:attributes];
    if ([attributes.allKeys containsObject:@"refresh"]) {
        _refresh = [WXConvert BOOL:attributes[@"refresh"]];
        [self _updateRefreshHeader];
    }
    if ([attributes.allKeys containsObject:@"loading"]) {
        _loading = [WXConvert BOOL:attributes[@"loading"]];
        [self _updateRefreshFooter];
    }
    if ([attributes.allKeys containsObject:@"showLoading"]) {
        _showLoading = [WXConvert BOOL:attributes[@"showLoading"]];
        [self _updateLoadingState];
    }
}

#pragma mark - Private Method
- (void)_addRefreshHeader
{
    if (!_refreshHeader) {
        __weak typeof(self) weakSelf = self;
        _refreshHeader = [MJRefreshGifHeader headerWithRefreshingBlock:^{
            [weakSelf refreshData];
        }];
        UIScrollView *scrollView = (UIScrollView *)self.view;
        scrollView.header = _refreshHeader;
    }
}

- (void)refreshData
{
    if (_refreshEvent) {
        [self fireEvent:@"refresh" params:nil];
    }
}

- (void)_removeRefreshHeader
{
    if (_refreshHeader) {
        UIScrollView *scrollView = (UIScrollView *)self.view;
        scrollView.header = nil;
        _refreshHeader = nil;
    }
}

- (void)_addRefreshFooter
{
    if (!_refreshFooter) {
        __weak typeof(self) weakSelf = self;
        _refreshFooter = [MDRefreshAutoFooter footerWithRefreshingBlock:^{
            [weakSelf loadingData];
        }];
        UIScrollView *scrollView = (UIScrollView *)self.view;
        scrollView.footer = _refreshFooter;
        [self _updateLoadingState];
    }
}

- (void)loadingData
{
    if (_loadingEvent) {
        [self fireEvent:@"loading" params:nil];
    }
}

- (void)_removeRefreshFooter
{
    if (_refreshFooter) {
        UIScrollView *scrollView = (UIScrollView *)self.view;
        scrollView.footer = nil;
        _refreshFooter = nil;
    }
}

- (void)_updateRefreshHeader
{
    if (_refresh) {
        [self _addRefreshHeader];
    }
    else {
        [self _removeRefreshHeader];
    }
}

- (void)_updateRefreshFooter
{
    if (_loading) {
        [self _addRefreshFooter];
    }
    else {
        [self _removeRefreshFooter];
    }
}

- (void)_updateLoadingState
{
    self.refreshFooter.hidden = !_showLoading;
}

#pragma mark - JS call method
- (void)endRefreshing
{
    if (_refreshHeader) {
        UIScrollView *scrollView = (UIScrollView *)self.view;
        [scrollView.header endRefreshing];
        [UIView animateWithDuration:0.4 animations:^{ [(UIScrollView *)self.view setContentOffset:CGPointZero]; }];
    }
}

- (void)endLoading
{
    if (_refreshFooter) {
        UIScrollView *scrollView = (UIScrollView *)self.view;
        [scrollView.footer endRefreshing];
        scrollView.footer.hidden = YES;
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.25 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            scrollView.footer.hidden = NO;
        });
    }
}

- (void)noticeNoMoreData
{
    if (_refreshFooter) {
        UIScrollView *scrollView = (UIScrollView *)self.view;
        [scrollView.footer noticeNoMoreData];
    }
}

@end

refresh {boolean}: 可选值为true/false,默认是false。此值决定 list 是否开启下拉刷新功能。

loading {boolean}:可选值为true/false,默认是false。此值决定 list 是否开启上拉加载功能。

下拉刷新的时候,会触发list组件的refresh方法,上拉加载的时候,会触发list组件的loading方法。当数据返回的时候,我们需要手动去关闭刷新或者加载,因为数据什么时候回来只有weex端才知道,所以扩展了一下几个方法:

WX_EXPORT_METHOD(@selector(endRefreshing));    // 停止刷新

- (void)endRefreshing
{
   if (_refreshHeader) {
       UIScrollView *scrollView = (UIScrollView *)self.view;
       [scrollView.header endRefreshing];
       // weex在调用停止刷新后,scroll并不会回到顶部,需要我们手动进行设置一下。
       [UIView animateWithDuration:0.4 animations:^{ [(UIScrollView *)self.view setContentOffset:CGPointZero]; }];
   }
}
WX_EXPORT_METHOD(@selector(endLoading));    // 停止上拉加载

- (void)endLoading
{
    if (_refreshFooter) {
        UIScrollView *scrollView = (UIScrollView *)self.view;
        [scrollView.footer endRefreshing];
        scrollView.footer.hidden = YES;
        // 此处的处理因为数据返回时候,weex的render需要一定的时间,所有绝大部分时候,是先看到footer,然后cell才会一个一个进行渲染,所以此处先隐藏footer,0.25s之后再显示
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.25 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            scrollView.footer.hidden = NO;
        });
    }
}

当没有更多数据的时候,我们可以调用这个方法进行设置

WX_EXPORT_METHOD(@selector(noticeNoMoreData));  // 提示没有更多数据了

- (void)noticeNoMoreData
{
    if (_refreshFooter) {
        UIScrollView *scrollView = (UIScrollView *)self.view;
        [scrollView.footer noticeNoMoreData];
    }
}

前端

在设置样式的模块代码如下

<template>
    <div>
        <list 
          ref="list"
          refresh="true"
          @refresh="_refreshData"
          loading="true"
          @loading="_loadingData"
        >
        </list>
    </div>
</template>

ref为了拿到list组件,上述代码开启了上拉加载和下拉刷新的功能,下面是script模块的代码

<script>
export default {
    methods: {
        _refreshData() {
            this._fetchData(true)
        },
        _loadingData() {
            this._fetchData(false)
        },
        _fetchData(isRefresh) {
            // 请求数据,结束后
            if (_platform==='iOS') {
                let list = this.$refs.list;
                if (list) {
                    if (isRefresh) {
                        list.endRefreshing();
                    }
                    if (this.noMoreData) {
                        list.noticeNoMoreData();
                    } else {
                        list.endLoading();  
                    }
                }
            }
        }
    }
}
</script>

通过上述操作,就可以将原生的代码桥接到weex中使用啦。

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,050评论 25 707
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,497评论 18 139
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 11,945评论 4 60
  • 请你看看下面三个黑体加粗的词,你知道其中的“扛”、“拾”、“量”的正确读音在这里分别是什么吗? 扛鼎之作 拾级而上...
    朗读者晟焕阅读 1,042评论 0 1
  • 现在的市面上,流行各种有关时间管理的书籍,我也曾和大多数人一样,看了不少,但多半于事无补。面对时光匆匆流逝,却依然...
    戈多不多阅读 460评论 0 0