xmlplus组件设计系列之下拉刷新(PullRefresh)(6)

发布时间 - 2026-01-11 00:56:35    点击率:

“下拉刷新”由著名设计师 Loren Brichter 设计,并应用于 Twitter 第三方应用 Tweetie 中。2010年4月,Twitter 收购 Tweetie 开发商 Atebits 后,该专利归 Twitter 所有。这一章我们就来看看如何实现一个简单的下拉刷新组件。

目标组件分析

和前面在设计组件时的做法一样,我们先想想看最终的成品组件是如何使用的,这需要点想像力。下拉刷新组件看成一个容器组件是合理的,用户可以对容器的内容进行下拉操作。如果用户完成了完整的下拉触发操作,该组件应该会有下拉完成的事件反馈,假定这个事件名为 ready。根据以上的分析,我们很有可能得到下面的一个该组件的应用示例。

Example1: {
 xml: `<PullRefresh id='example'>
    <h1>Twitter</h1>
    <h2>Loren Brichter</h2>
   </PullRefresh>`,
 fun: function (sys, items, opts) {
  sys.example.on("ready", () => console.log("ready"));
 }
}

示例中的使用方式是非常简洁的,但我们还漏了一点。如果你用过一些新闻客户端,在某些情况下,此客户端会自动触发下拉刷新操作。比如,刚进入客户端页面或者由于软件推送机制产生的被动列表更新,这都将导致客户端下拉刷新操作的触发。所以如上的 PullRefresh 组件还应该提供一个触发自动刷新的操作接口。好了,下面是加入下拉刷新接口的应用示例。

Example2: {
 xml: `<PullRefresh id='example'>
    <h1>Twitter</h1>
    <h2>Loren Brichter</h2>
    <button id='refresh'>click</button>
   </PullRefresh>`,
 fun: function (sys, items, opts) {
  sys.example.on("ready", () => console.log("ready"));
  sys.refresh.on("click", items.example.refresh);
 }
}

基本框架

现在让我们把目光转移到下拉刷新组件的内部,看看该如何去实现。观察文章开始部分的大图,很自然地我们可以将整个组件划分为三个子组件,如下面的 XML 文档所示。

<div id="refresh">
 <Status id="status"/>
 <div id="content"></div>
</div>

外围 div 元素包含两个子组件:其中一个是状态指示条,用于显示“下拉刷新”、“松开刷新”、“加载中...”以及“刷新成功”四个状态提示,这里暂时使用未定义的 Status 组件替代;另一个 div 元素用于容纳下拉刷新组件的包含内容。到现在,大概可以想得出该组件的工作逻辑了,于是我们可以给出下面的一个基本的组件框架。

PullRefresh: {
 css: "#refresh { position: relative; height: 100%;...}",
 xml: `<div id="refresh">
   <Status id="status"/>
   <div id="content"/>
   </div>`,
 map: { appendTo: "content" },
 fun: function (sys, items, opts) {
  sys.content.on("touchstart", e => {
   // 侦听 touchmove 和 touchend事件
  });
  function touchmove(e) {
   // 1 处理状态条与内容内面跟随触点移动
   // 2 根据触点移动的距离显示相当的状态条内容
  }
  function touchend(e) {
   // 1 移除 touchmove 和 touchend 事件
   // 2 根据触点移动的距离决定返回原始状态或者进入刷新状态并派发事件
  }
 }
}

状态条的实现

如前面提到的,状态条组件包含四个状态提示,并且每一时刻仅显示一个状态。对于状态的切换,这里会先用到我们下一章将讲到的路由组件 ViewStack,这里仅需要了解如何使用即可。组件 ViewStack 对外只显示子级的一个子组件,同时侦听一个 switch 事件,该事件的派发者携带了一个切换到的目标对象的名称,也就是 ID。该组件根据这个 ID 来切换到目标视图。下面是状态条组件的完整实现。

Status: {
 css: "#statusbar { height: 2.5em; line-height: 2.5em; text-align: center; }",
 xml: <ViewStack id="statusbar">
   <span id="pull">下拉刷新</span>
   <span id="ready">松开刷新</span>
   <span id="loading">加载中...</span>
   <span id="success">刷新成功</span>
   </ViewStack>,
 fun: function (sys, items, opts) {
  var stat = "pull";
  function getValue() {
   return stat;
  }
  function setValue(value) {
   sys.statusbar.trigger("switch", stat = value);
  }
  return Object.defineProperty({}, "value", { get: getValue, set: setValue });
 }
}

该组件提供一个 value 接口用户设置与获取组件的显示状态。父级组件可根据不同的时机调用该接口。

最终实现

有了上面的储备,让我们来填充完下拉刷新组件的细节。下拉刷新过程中会涉及到动画,对于动画目前一般有两种选择,可以使用 JQuery 动画函数,也可以是 css3,这需要看各人喜好了。这里我们选择使用 css3 来实现。为清晰起见,下面的实现仅给出函数部分,其余部分同上。

PullRefresh: {
 fun: function (sys, items, opts) {
  var startY, height = sys.status.height();
  sys.content.on("stouchstart", e => {
   if (items.status.value == "pull") {
    startY = e.y;
    sys.content.on("touchmove", touchmove).on("touchend", touchend);
    sys.content.css("transition", "").prev().css("transition", "");
   }
  });
  function touchmove(e) {
   var offset = e.y - startY;
   if ( offset > 0 ) {
    sys.content.css("top", offset + "px"); 
    sys.status.css("top", (offset - height) + "px");
    items.status(offset > height ? "ready" : "pull");
   }
  }
  function touchend (e) {
   var offset = e.y - startY;
   sys.content.off("touchmove").off("touchend");
   sys.content.css("transition", "all 0.3s ease-in 0s").prev().css("transition", "all 0.3s ease-in 0s");
   if ( offset < height ) {
    sys.content.css("top", "0").prev().css("top", -height + "px");
   } else {
    items.status.value = "release";
    sys.refresh.once("complete", complete);
    sys.content.css("top", height + "px").prev().css("top", "0").trigger("ready");
   }
  }
  function complete() {
   items.status.value = "message";
   setTimeout(() => {
    sys.content.css("top", "0").prev().css("top", -height + "px");
    sys.content.once("webkitTransitionEnd", e => items.status.value = "pull");
   }, 300);
  }
 }
}

对于稍微有点复杂的组件,需要注意组件的组织归类,尽量把具有相近功能的组件放在一起。为了便于叙述,上述所列出的组件示意总把它们视作是同一目录,这一点读者应该能看出来。

本系列文章基于 xmlplus 框架。如果你对 xmlplus 没有多少了解,可以访问 www.xmlplus.cn。这里有详尽的入门文档可供参考。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


# xmlplus  # 下拉刷新  # xmlplus组件设计系列之网格(DataGrid)(10)  # xmlplus组件设计系列之文本框(TextBox)(3)  # xmlplus组件设计系列之选项卡(Tabbar)(5)  # xmlplus组件设计系列之路由(ViewStack)(7)  # xmlplus组件设计系列之分隔框(DividedBox)(8)  # xmlplus组件设计系列之树(Tree)(9)  # xmlplus组件设计系列之按钮(2)  # xmlplus组件设计系列之列表(4)  # xmlplus组件设计系列之图标(ICON)(1)  # 客户端  # 让我们  # 我们可以  # 提供一个  # 如何使用  # 切换到  # 如果你  # 文档  # 会有  # 加载中  # 好了  # 你对  # 只显示  # 可以使用  # 可供  # 用过  # 该如何  # 能看  # 所示  # 其中一个 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 成都网站制作公司哪家好,四川省职工服务网是做什么用?  广州网站制作公司哪家好一点,广州欧莱雅百库网络科技有限公司官网?  大同网页,大同瑞慈医院官网?  Linux后台任务运行方法_nohup与&使用技巧【技巧】  Laravel怎么使用Collection集合方法_Laravel数组操作高级函数pluck与map【手册】  Laravel怎么实现搜索高亮功能_Laravel结合Scout与Algolia全文检索【实战】  重庆市网站制作公司,重庆招聘网站哪个好?  JavaScript数据类型有哪些_如何准确判断一个变量的类型  Laravel如何配置.env文件管理环境变量_Laravel环境变量使用与安全管理  如何用JavaScript实现文本编辑器_光标和选区怎么处理  详解Huffman编码算法之Java实现  EditPlus中的正则表达式 实战(1)  Android自定义控件实现温度旋转按钮效果  HTML5建模怎么导出为FBX格式_FBX格式兼容性及导出步骤【指南】  中山网站制作网页,中山新生登记系统登记流程?  Laravel与Inertia.js怎么结合_使用Laravel和Inertia构建现代单页应用  如何用虚拟主机快速搭建网站?详细步骤解析  敲碗10年!Mac系列传将迎来「触控与联网」双革新  Laravel如何配置Horizon来管理队列?(安装和使用)  如何在云服务器上快速搭建个人网站?  韩国服务器如何优化跨境访问实现高效连接?  javascript基本数据类型及类型检测常用方法小结  Laravel如何使用软删除(Soft Deletes)功能_Eloquent软删除与数据恢复方法  矢量图网站制作软件,用千图网的一张矢量图做公司app首页,该网站并未说明版权等问题,这样做算不算侵权?应该如何解决?  ,在苏州找工作,上哪个网站比较好?  悟空识字如何进行跟读录音_悟空识字开启麦克风权限与录音  Laravel如何实现API速率限制?(Rate Limiting教程)  微博html5版本怎么弄发超话_超话进入入口及发帖格式要求【教程】  Laravel如何使用Passport实现OAuth2?(完整配置步骤)  Laravel如何使用.env文件管理环境变量?(最佳实践)  Laravel表单请求验证类怎么用_Laravel Form Request分离验证逻辑教程  Laravel如何升级到最新版本?(升级指南和步骤)  香港服务器租用费用高吗?如何避免常见误区?  Laravel如何实现用户注册和登录?(Auth脚手架指南)  JS中对数组元素进行增删改移的方法总结  如何快速查询网站的真实建站时间?  Laravel如何连接多个数据库_Laravel多数据库连接配置与切换教程  Laravel怎么使用Blade模板引擎_Laravel模板继承与Component组件复用【手册】  php8.4header发送头信息失败怎么办_php8.4header函数问题解决【解答】  如何在 React 中条件性地遍历数组并渲染元素  canvas 画布在主流浏览器中的尺寸限制详细介绍  Laravel怎么进行数据库事务处理_Laravel DB Facade事务操作确保数据一致性  Laravel怎么实现一对多关联查询_Laravel Eloquent模型关系定义与预加载【实战】  制作电商网页,电商供应链怎么做?  Laravel如何自定义分页视图?(Pagination示例)  Laravel如何使用Eloquent进行子查询  php json中文编码为null的解决办法  Laravel怎么实现前端Toast弹窗提示_Laravel Session闪存数据Flash传递给前端【方法】  Laravel中的Facade(门面)到底是什么原理  网页设计与网站制作内容,怎样注册网站?