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 】
相关推荐:
php后缀怎么变mp4格式错误_修改扩展名提示格式不对怎么办【技巧】
Bootstrap整体框架之JavaScript插件架构
如何将凡科建站内容保存为本地文件?
如何快速搭建高效可靠的建站解决方案?
iOS验证手机号的正则表达式
如何在阿里云通过域名搭建网站?
Laravel如何实现API版本控制_Laravel版本化API设计方案
香港服务器租用每月最低只需15元?
制作旅游网站html,怎样注册旅游网站?
小视频制作网站有哪些,有什么看国内小视频的网站,求推荐?
Laravel中的withCount方法怎么高效统计关联模型数量
如何在沈阳梯子盘古建站优化SEO排名与功能模块?
Laravel怎么做缓存_Laravel Cache系统提升应用速度的策略与技巧
利用JavaScript实现拖拽改变元素大小
悟空浏览器如何设置小说背景色_悟空浏览器背景色设置【方法】
Laravel如何实现URL美化Slug功能_Laravel使用eloquent-sluggable生成别名【方法】
laravel怎么为API路由添加签名中间件保护_laravel API路由签名中间件保护方法
公司网站制作价格怎么算,公司办个官网需要多少钱?
Laravel怎么使用Intervention Image库处理图片上传和缩放
教你用AI将一段旋律扩展成一首完整的曲子
Windows11怎样设置电源计划_Windows11电源计划调整攻略【指南】
Zeus浏览器网页版官网入口 宙斯浏览器官网在线通道
Laravel怎么上传文件_Laravel图片上传及存储配置
网站制作软件免费下载安装,有哪些免费下载的软件网站?
如何确保西部建站助手FTP传输的安全性?
php485函数参数是什么意思_php485各参数详细说明【介绍】
phpredis提高消息队列的实时性方法(推荐)
Laravel队列任务超时怎么办_Laravel Queue Timeout设置详解
Windows家庭版如何开启组策略(gpedit.msc)?(安装方法)
Laravel Session怎么存储_Laravel Session驱动配置详解
网站制作价目表怎么做,珍爱网婚介费用多少?
如何在新浪SAE免费搭建个人博客?
Laravel Eloquent模型如何创建_Laravel ORM基础之Model创建与使用教程
Linux系统命令中tree命令详解
Laravel如何使用Spatie Media Library_Laravel图片上传管理与缩略图生成【步骤】
香港服务器WordPress建站指南:SEO优化与高效部署策略
香港服务器建站指南:外贸独立站搭建与跨境电商配置流程
Laravel如何实现全文搜索功能?(Scout和Algolia示例)
Laravel怎么实现搜索功能_Laravel使用Eloquent实现模糊查询与多条件搜索【实例】
Gemini手机端怎么发图片_Gemini手机端发图方法【步骤】
详解CentOS6.5 安装 MySQL5.1.71的方法
重庆市网站制作公司,重庆招聘网站哪个好?
如何快速搭建高效香港服务器网站?
Laravel如何实现用户注册和登录?(Auth脚手架指南)
如何用狗爹虚拟主机快速搭建网站?
Laravel的契約(Contracts)是什么_深入理解Laravel Contracts与依赖倒置
Laravel DB事务怎么使用_Laravel数据库事务回滚操作
Laravel安装步骤详细教程_Laravel环境搭建指南
Laravel如何创建和注册中间件_Laravel中间件编写与应用流程
Laravel的Blade指令怎么自定义_创建你自己的Laravel Blade Directives

