当前位置: 首页 > news >正文

网站建设性价比高网站集约化建设的问题

网站建设性价比高,网站集约化建设的问题,jsp网站开发的教材,海南省最新消息背景 项目中首页列表页需要统计每个item的曝光情况#xff0c;给产品运营提供数据报表分析用户行为#xff0c;于是封装了一个通用的列表Item曝光工具#xff0c;方便曝光埋点上报 源码分析 核心就是监听RecyclerView的滚动#xff0c;在滚动状态为SCROLL_STATE_IDLE的时…背景 项目中首页列表页需要统计每个item的曝光情况给产品运营提供数据报表分析用户行为于是封装了一个通用的列表Item曝光工具方便曝光埋点上报 源码分析 核心就是监听RecyclerView的滚动在滚动状态为SCROLL_STATE_IDLE的时候开始计算哪些item是可见的 private fun calculateVisibleItemInternal() {if (!isRecording) {return}val lastRange currVisibleRangeval currRange findItemVisibleRange()val newVisibleItemPosList createCurVisiblePosList(lastRange, currRange)visibleItemCheckTasks.forEach {it.updateVisibleRange(currRange)}if (newVisibleItemPosList.isNotEmpty()) {VisibleCheckTimerTask(newVisibleItemPosList, this, threshold).also {visibleItemCheckTasks.add(it)}.execute()}currVisibleRange currRange}根据LayoutManager找出当前可见item的范围剔除掉显示不到80%的item private fun findItemVisibleRange(): IntRange {return when (val lm currRecyclerView?.layoutManager) {is GridLayoutManager - {val first lm.findFirstVisibleItemPosition()val last lm.findLastVisibleItemPosition()return fixCurRealVisibleRange(first, last)}is LinearLayoutManager - {val first lm.findFirstVisibleItemPosition()val last lm.findLastVisibleItemPosition()return fixCurRealVisibleRange(first, last)}is StaggeredGridLayoutManager - {val firstItems IntArray(lm.spanCount)lm.findFirstVisibleItemPositions(firstItems)val lastItems IntArray(lm.spanCount)lm.findLastVisibleItemPositions(lastItems)val first when (RecyclerView.NO_POSITION) {firstItems[0] - {firstItems[lm.spanCount - 1]}firstItems[lm.spanCount - 1] - {firstItems[0]}else - {min(firstItems[0], firstItems[lm.spanCount - 1])}}val last when (RecyclerView.NO_POSITION) {lastItems[0] - {lastItems[lm.spanCount - 1]}lastItems[lm.spanCount - 1] - {lastItems[0]}else - {max(lastItems[0], lastItems[lm.spanCount - 1])}}return fixCurRealVisibleRange(first, last)}else - {IntRange.EMPTY}}}对可见的item进行分组形成一个位置列表上一次和这一次的分开组队 private fun createCurVisiblePosList(lastRange: IntRange, currRange: IntRange): ListInt {val result mutableListOfInt()currRange.forEach { pos -if (pos !in lastRange) {result.add(pos)}}return result}将分组好的列表装进一个定时的task在延迟一个阈值的时间后执行onVisibleCheck class VisibleCheckTimerTask(input: ListInt,private val callback: VisibleCheckCallback,private val delay: Long ) : Runnable {private val visibleList mutableListOfInt()private var isExecuted falseinit {visibleList.addAll(input)}fun updateVisibleRange(keyRange: IntRange) {val iterator visibleList.iterator()while (iterator.hasNext()) {val entry iterator.next()if (entry !in keyRange) {iterator.remove()}}}override fun run() {callback.onVisibleCheck( this, visibleList)}fun execute() {if (isExecuted) {return}mHandler.postDelayed(this, delay)isExecuted true}fun cancel() {mHandler.removeCallbacks(this)}interface VisibleCheckCallback {fun onVisibleCheck(task: VisibleCheckTimerTask, visibleList: ListInt)} }达到阈值后再获取一遍可见item的范围对于仍然可见的item回调onItemShow override fun onVisibleCheck(task: VisibleCheckTimerTask, visibleList: ListInt) {val visibleRange findItemVisibleRange()visibleList.forEach {if (it in visibleRange) {notifyItemShow(it)}}visibleItemCheckTasks.remove(task)}完整源码 val mHandler Handler(Looper.getMainLooper())class ListItemExposeUtil(private val threshold: Long 100): RecyclerView.OnScrollListener(), VisibleCheckTimerTask.VisibleCheckCallback {private var currRecyclerView: RecyclerView? nullprivate var isRecording falseprivate var currVisibleRange: IntRange IntRange.EMPTYprivate val visibleItemCheckTasks mutableListOfVisibleCheckTimerTask()private val itemShowListeners mutableListOfOnItemShowListener()fun attachTo(recyclerView: RecyclerView) {recyclerView.addOnScrollListener(this)currRecyclerView recyclerView}fun start() {isRecording truecurrRecyclerView?.post {calculateVisibleItemInternal()}}fun stop() {visibleItemCheckTasks.forEach {it.cancel()}visibleItemCheckTasks.clear()currVisibleRange IntRange.EMPTYisRecording false}fun detach() {if (isRecording) {stop()}itemShowListeners.clear()currRecyclerView?.removeOnScrollListener(this)currRecyclerView null}override fun onScrollStateChanged(recyclerView: RecyclerView, newState: Int) {if (newState RecyclerView.SCROLL_STATE_IDLE) {calculateVisibleItemInternal()}}private fun calculateVisibleItemInternal() {if (!isRecording) {return}val lastRange currVisibleRangeval currRange findItemVisibleRange()val newVisibleItemPosList createCurVisiblePosList(lastRange, currRange)visibleItemCheckTasks.forEach {it.updateVisibleRange(currRange)}if (newVisibleItemPosList.isNotEmpty()) {VisibleCheckTimerTask(newVisibleItemPosList, this, threshold).also {visibleItemCheckTasks.add(it)}.execute()}currVisibleRange currRange}private fun findItemVisibleRange(): IntRange {return when (val lm currRecyclerView?.layoutManager) {is GridLayoutManager - {val first lm.findFirstVisibleItemPosition()val last lm.findLastVisibleItemPosition()return fixCurRealVisibleRange(first, last)}is LinearLayoutManager - {val first lm.findFirstVisibleItemPosition()val last lm.findLastVisibleItemPosition()return fixCurRealVisibleRange(first, last)}is StaggeredGridLayoutManager - {val firstItems IntArray(lm.spanCount)lm.findFirstVisibleItemPositions(firstItems)val lastItems IntArray(lm.spanCount)lm.findLastVisibleItemPositions(lastItems)val first when (RecyclerView.NO_POSITION) {firstItems[0] - {firstItems[lm.spanCount - 1]}firstItems[lm.spanCount - 1] - {firstItems[0]}else - {min(firstItems[0], firstItems[lm.spanCount - 1])}}val last when (RecyclerView.NO_POSITION) {lastItems[0] - {lastItems[lm.spanCount - 1]}lastItems[lm.spanCount - 1] - {lastItems[0]}else - {max(lastItems[0], lastItems[lm.spanCount - 1])}}return fixCurRealVisibleRange(first, last)}else - {IntRange.EMPTY}}}/*** 检查该item是否真实可见* view区域80%显示出来就算*/private fun checkItemCurrRealVisible(pos: Int): Boolean {val holder currRecyclerView?.findViewHolderForAdapterPosition(pos)return if (holder null) {false} else {val rect Rect()holder.itemView.getGlobalVisibleRect(rect)if (holder.itemView.width 0 holder.itemView.height 0) {return (rect.width() * rect.height() / (holder.itemView.width * holder.itemView.height).toFloat()) 0.8f} else {return false}}}/*** 双指针寻找真实可见的item的范围有一些item没显示完整剔除*/private fun fixCurRealVisibleRange(first: Int, last: Int): IntRange {return if (first 0 last 0) {var realFirst firstwhile (!checkItemCurrRealVisible(realFirst) realFirst last) {realFirst}var realLast lastwhile (!checkItemCurrRealVisible(realLast) realLast realFirst) {realLast--}if (realFirst realLast) {realFirst..realLast} else {IntRange.EMPTY}} else {IntRange.EMPTY}}/*** 创建当前可见的item位置列表*/private fun createCurVisiblePosList(lastRange: IntRange, currRange: IntRange): ListInt {val result mutableListOfInt()currRange.forEach { pos -if (pos !in lastRange) {result.add(pos)}}return result}/*** 达到阈值后再获取一遍可见item的范围对于仍然可见的item回调onItemShow*/override fun onVisibleCheck(task: VisibleCheckTimerTask, visibleList: ListInt) {val visibleRange findItemVisibleRange()visibleList.forEach {if (it in visibleRange) {notifyItemShow(it)}}visibleItemCheckTasks.remove(task)}private fun notifyItemShow(pos: Int) {itemShowListeners.forEach {it.onItemShow(pos)}}fun addItemShowListener(listener: OnItemShowListener) {itemShowListeners.add(listener)} }interface OnItemShowListener {fun onItemShow(pos: Int) }class VisibleCheckTimerTask(input: ListInt,private val callback: VisibleCheckCallback,private val delay: Long) : Runnable {private val visibleList mutableListOfInt()private var isExecuted falseinit {visibleList.addAll(input)}fun updateVisibleRange(keyRange: IntRange) {val iterator visibleList.iterator()while (iterator.hasNext()) {val entry iterator.next()if (entry !in keyRange) {iterator.remove()}}}override fun run() {callback.onVisibleCheck(this, visibleList)}fun execute() {if (isExecuted) {return}mHandler.postDelayed(this, delay)isExecuted true}fun cancel() {mHandler.removeCallbacks(this)}interface VisibleCheckCallback {fun onVisibleCheck(task: VisibleCheckTimerTask, visibleList: ListInt)} }测试代码 运行结果
http://www.w-s-a.com/news/162341/

相关文章:

  • 下载flash网站网站设计书的结构
  • 水利建设公共服务平台网站放心网络营销定制
  • 设计网站过程wordpress+分页静态
  • 临海网站制作好了如何上线如果安装wordpress
  • 长沙 学校网站建设网站制作价格上海
  • 九江网站推广徽hyhyk1国家住房部和城乡建设部 网站首页
  • 阿克苏网站建设咨询动漫设计与制作属于什么大类
  • 网站编辑做多久可以升职wordpress版权修改
  • 网站开发维护成本计算国外外贸平台
  • 简单的招聘网站怎么做购物网站功能报价
  • 哪个网站做中高端衣服建设自己网站的流程
  • 网站建设概况做网站的是怎么赚钱的
  • 网站发布信息的基本流程现在都不用dw做网站了吗
  • 赣州热门网站深圳龙岗做网站的公司
  • 中国最大的建站平台广告传媒公司取名
  • 深圳网站设计公司专业吗学动漫设计后悔死了
  • 企业网站形象建设网站开发入职转正申请书
  • 网站设计步骤济南建设网中标公告
  • 石佛营网站建设wordpress关健词
  • 您的网站空间即将过期建站 discuz
  • 上海简站商贸有限公司福州哪家专业网站设计制作最好
  • 博客网站开发流程苏州专业做网站的公司哪家好
  • 四川手机网站建设西安 网站 高端 公司
  • 织梦大气绿色大气农业能源化工机械产品企业网站源码模版建筑工程知识零基础
  • 广州番禺网站公司v2017网站开发
  • 微信公众号怎么做微网站wordpress和dz
  • 西部数码网站管理助手 301福州搜索优化实力
  • 响应式网站介绍页面模板功能找不到
  • 公司网站如何seo自己做资讯网站
  • 天津网站建设软件开发招聘企业信用信息查询公示系统上海