- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我已经实现了 RecyclerView.addOnScrolledToEnd
以使用异步任务从服务器获取下一组数据。仅使用 RecyclerView.addOnScrolledToEnd 时它工作正常。但是这里我的问题是一旦我到达终点并且我做了 SwipeRefreshLayout.setOnRefreshListener
(这里将数据重置为新记录) RecyclerView.addOnScrolledToEnd
没有调用并且分页没有在这种情况下工作。
fragment
class RecentNewsFragment : Fragment(), LoaderManager.LoaderCallbacks<Cursor>, NewsAdapter.OnNewsCallBack, OnRefreshCallback {
private var rlProgressContainer: View? = null
private var listContainer: View? = null
private var pbViewMore: ProgressBar? = null
private var swipeRefreshLayout: SwipeRefreshLayout? = null
private var srlNoMessageRefreshLayout: SwipeRefreshLayout? = null
private var newsAdapter: NewsAdapter? = null
private var pageLimit: Int = 10
private var pageOffset: Int = 0
private var totalNewsCount: Int = 0
private var isAsyncDone: Boolean = false
private val mRecentNewsBroadCast = object : BroadcastReceiver() {
override fun onReceive(context: Context, intent: Intent) {
// Get extra data included in the Intent
restartLoader()
}
}
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,
savedInstanceState: Bundle?): View? {
// Inflate the layout for this fragment
val view = inflater.inflate(R.layout.fragment_recent_news, container, false)
initView(view)
return view
}
override fun onResume() {
super.onResume()
if(!Helper.isConnected(context!!)) {
Toast.makeText(context!!, context?.getString(R.string.no_connection), Toast.LENGTH_SHORT).show()
}
}
private fun initView(view: View) {
rlProgressContainer = view.findViewById(R.id.rl_progressContainer)
listContainer = view.findViewById(R.id.ll_list_container)
pbViewMore = view.findViewById(R.id.pb_view_more)
newsAdapter = NewsAdapter(context!!, null, this)
val recyclerView = view.findViewById<RecyclerView>(R.id.recycler_view)
recyclerView.layoutManager = LinearLayoutManager(context)
recyclerView.adapter = newsAdapter
recyclerView.addItemDecoration(DividerItemDecoration(context!!, DividerItemDecoration.VERTICAL))
recyclerView.setHasFixedSize(true)
recyclerView.itemAnimator = DefaultItemAnimator()
pbViewMore!!.visibility = View.GONE
srlNoMessageRefreshLayout = view.findViewById(R.id.srl_activities_no_message)
srlNoMessageRefreshLayout!!.visibility = View.GONE
swipeRefreshLayout = view.findViewById(R.id.swipe_refresh_feeds_page)
val pullToRefresh = PullToRefresh()
swipeRefreshLayout!!.setColorSchemeResources(android.R.color.holo_green_dark, android.R.color.holo_red_dark, android.R.color.holo_orange_dark, android.R.color.holo_purple)
swipeRefreshLayout!!.setOnRefreshListener(pullToRefresh)
srlNoMessageRefreshLayout!!.setColorSchemeResources(android.R.color.holo_green_dark, android.R.color.holo_red_dark, android.R.color.holo_orange_dark, android.R.color.holo_purple)
srlNoMessageRefreshLayout!!.setOnRefreshListener(pullToRefresh)
listContainer!!.visibility = View.GONE
rlProgressContainer!!.visibility = View.VISIBLE
loaderManager.initLoader(Constants.URL_RECENT_NEWS_LOADER, arguments, this)
startAsync(pageLimit, pageOffset)
fun RecyclerView.addOnScrolledToEnd(onScrolledToEnd: () -> Unit){
this.addOnScrollListener(object: RecyclerView.OnScrollListener(){
private val visibleThreshold = 5
private var loading = true
private var previousTotal = 0
override fun onScrollStateChanged(recyclerView: RecyclerView,
newState: Int) {
with(layoutManager as LinearLayoutManager){
val visibleItemCount = childCount
val totalItemCount = itemCount
val firstVisibleItem = findFirstVisibleItemPosition()
if (loading && totalItemCount > previousTotal){
loading = false
previousTotal = totalItemCount
}
if(!loading && (totalItemCount - visibleItemCount) <= (firstVisibleItem + visibleThreshold)){
onScrolledToEnd()
loading = true
}
}
}
})
}
recyclerView.addOnScrolledToEnd {
//What you want to do once the end is reached
pbViewMore!!.visibility = View.VISIBLE
startAsync(pageLimit, pageOffset)
}
}
private inner class PullToRefresh : SwipeRefreshLayout.OnRefreshListener {
override fun onRefresh() {
pbViewMore!!.visibility = View.GONE
pageOffset = 0
startAsync(pageLimit, pageOffset)
srlNoMessageRefreshLayout!!.isRefreshing = false
swipeRefreshLayout!!.isRefreshing = false
}
}
private fun startAsync(pageLimit: Int, pageOffset: Int) {
if(Helper.isConnected(context!!)) {
val recentAsync = RecentNewsAsync(context = this, pageLimit = pageLimit, pageOffset = pageOffset)
recentAsync.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR)
} else {
isAsyncDone = true
}
}
fun restartLoader() {
if(isAdded)
loaderManager.restartLoader(Constants.URL_RECENT_NEWS_LOADER, null, this)
}
override fun onCreateLoader(id: Int, args: Bundle?): Loader<Cursor> {
return NewsVo.getCursorLoader(context!!, NewsVo.NewsType.RECENT, 0)
}
override fun onLoadFinished(loader: Loader<Cursor>, cursor: Cursor?) {
if (loader.id == Constants.URL_RECENT_NEWS_LOADER) {
if(isAsyncDone) {
listContainer!!.visibility = View.VISIBLE
rlProgressContainer!!.visibility = View.GONE
}
if (cursor != null) {
newsAdapter!!.swapCursor(cursor)
newsAdapter!!.notifyDataSetChanged()
if (isAsyncDone && cursor.count > 0) {
srlNoMessageRefreshLayout!!.visibility = View.GONE
swipeRefreshLayout!!.visibility = View.VISIBLE
} else if(isAsyncDone){
srlNoMessageRefreshLayout!!.visibility = View.VISIBLE
swipeRefreshLayout!!.visibility = View.GONE
}
}
}
}
override fun onLoaderReset(loader: Loader<Cursor>) {
newsAdapter!!.swapCursor(null)
}
private class RecentNewsAsync internal constructor(context: RecentNewsFragment, private val pageLimit: Int, private val pageOffset: Int) : AsyncTask<Void, Void, Response<ArrayList<NewsVo>>>() {
private val context: WeakReference<RecentNewsFragment> = WeakReference(context)
override fun doInBackground(vararg p0: Void?): Response<ArrayList<NewsVo>> {
var response = Response<ArrayList<NewsVo>>()
try {
val lphService = LPHServiceFactory.getCALFService(context.get()?.context!!)
response = lphService.recentNews(pageLimit, pageOffset)
} catch (e: LPHException) {
e.printStackTrace()
response.setThrowable(e)
} catch (e: JSONException) {
e.printStackTrace()
response.setThrowable(e)
} catch (e: IOException) {
e.printStackTrace()
response.setThrowable(e)
}
return response
}
override fun onPostExecute(response: Response<ArrayList<NewsVo>>) {
super.onPostExecute(response)
context.get()?.isAsyncDone = true
context.get()?.pbViewMore?.visibility = View.GONE
if (response.isSuccess()) {
context.get()?.totalNewsCount = response.getMetaData() as Int
if(context.get() != null)
context.get()!!.pageOffset += response.getResult()?.size!!
context.get()?.restartLoader()
}
}
}
override fun onFavoriteClick(newsId: Int, isFavorite: Boolean) {
if(Helper.isConnected(context!!)) {
val weakReferenceContext = WeakReference(this.context!!)
val markFavoriteAsync = Helper.MarkFavoriteAsync(weakReferenceContext, newsId, isFavorite, this, 0)
markFavoriteAsync.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR)
} else {
Helper.showConfirmationAlertTwoButton(context!!, context!!.getString(R.string.internet_warning), object : ConfirmationAlertCallback {
override fun onPositiveButtonClick() {
onFavoriteClick(newsId, isFavorite)
}
override fun onNegativeButtonClick() {
}
override fun onNeutralButtonClick() {
}
})
}
}
override fun setRead(newsId: Int, isRead: Boolean) {
if(Helper.isConnected(context!!)) {
val weakReferenceContext = WeakReference(context!!)
val markReadAsync = Helper.MarkReadAsync(weakReferenceContext, newsId, isRead, this, 0)
markReadAsync.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR)
} else {
Helper.showConfirmationAlertTwoButton(context!!, context!!.getString(R.string.internet_warning), object : ConfirmationAlertCallback {
override fun onPositiveButtonClick() {
setRead(newsId, isRead)
}
override fun onNegativeButtonClick() {
}
override fun onNeutralButtonClick() {
}
})
}
}
override fun onViewClick(newsId: Int) {
val newsDetailIntent = Intent(context, NewsDetailActivity::class.java)
newsDetailIntent.putExtra(Constants.BUNDLE_NEWS_ID, newsId)
startActivityForResult(newsDetailIntent, Constants.REQUEST_CODE_RECENT_NEWS)
}
override fun onActivityCreated(savedInstanceState: Bundle?) {
super.onActivityCreated(savedInstanceState)
activity!!.registerReceiver(mRecentNewsBroadCast, IntentFilter(Constants.BROADCAST_RECENT_NEWS))
}
override fun onDetach() {
super.onDetach()
activity!!.unregisterReceiver(mRecentNewsBroadCast)
}
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
super.onActivityResult(requestCode, resultCode, data)
if (requestCode == Constants.REQUEST_CODE_RECENT_NEWS && resultCode == Activity.RESULT_OK) {
restartLoader()
val intent1 = Intent(Constants.BROADCAST_FAVORITE_NEWS)
context!!.sendBroadcast(intent1)
val intent2 = Intent(Constants.BROADCAST_CATEGORIES)
context!!.sendBroadcast(intent2)
}
}
override fun onRefresh() {
restartLoader()
val intent1 = Intent(Constants.BROADCAST_FAVORITE_NEWS)
context!!.sendBroadcast(intent1)
val intent2 = Intent(Constants.BROADCAST_CATEGORIES)
context!!.sendBroadcast(intent2)
}
companion object {
/**
* Use this factory method to create a new instance of
* this fragment using the provided parameters.
*
* @return A new instance of fragment RecentNewsFragment.
*/
fun newInstance(): RecentNewsFragment {
return RecentNewsFragment()
}
}
}
布局文件:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<LinearLayout
android:id="@+id/ll_list_container"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="horizontal">
<android.support.v4.widget.SwipeRefreshLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/srl_activities_no_message"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:visibility="gone">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="horizontal">
<TextView
android:id="@+id/tv_no_data_found"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:enabled="false"
android:fontFamily="sans-serif-light"
android:gravity="center"
android:text="@string/no_data"
android:textAppearance="?android:attr/textAppearanceMedium"
android:textColor="@android:color/black" />
</LinearLayout>
</android.support.v4.widget.SwipeRefreshLayout>
<android.support.v4.widget.SwipeRefreshLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/swipe_refresh_feeds_page"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:fitsSystemWindows="false"
android:visibility="visible">
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
<android.support.v7.widget.RecyclerView
android:id="@+id/recycler_view"
android:layout_width="match_parent"
android:layout_height="match_parent" />
<ProgressBar
android:id="@+id/pb_view_more"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:layout_centerInParent="true"
android:layout_gravity="center_horizontal"
android:theme="@style/ProgressBarTheme"
android:visibility="gone" />
</RelativeLayout>
</android.support.v4.widget.SwipeRefreshLayout>
</LinearLayout>
<RelativeLayout
android:id="@+id/rl_progressContainer"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:visibility="gone">
<ProgressBar
android:layout_width="wrap_content"
android:layout_height="match_parent"
android:layout_centerInParent="true"
android:layout_centerVertical="true"
android:theme="@style/ProgressBarTheme" />
</RelativeLayout>
</LinearLayout>
最佳答案
这是 Recyclerview 的无限滚动,我从 here 得到它
public abstract class EndlessRecyclerOnScrollListener extends RecyclerView.OnScrollListener {
/**
* The total number of items in the dataset after the last load
*/
private int mPreviousTotal = 0;
/**
* True if we are still waiting for the last set of data to load.
*/
private boolean mLoading = true;
@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
super.onScrolled(recyclerView, dx, dy);
int visibleItemCount = recyclerView.getChildCount();
int totalItemCount = recyclerView.getLayoutManager().getItemCount();
int firstVisibleItem = ((LinearLayoutManager) recyclerView.getLayoutManager()).findFirstVisibleItemPosition();
if (mLoading) {
if (totalItemCount > mPreviousTotal) {
mLoading = false;
mPreviousTotal = totalItemCount;
}
}
int visibleThreshold = 5;
if (!mLoading && (totalItemCount - visibleItemCount)
<= (firstVisibleItem + visibleThreshold)) {
// End has been reached
onLoadMore();
mLoading = true;
}
}
public abstract void onLoadMore();
我在 Kotlin MVP 模式中使用它
rcyView_matches.addOnScrollListener(object : EndlessRecyclerOnScrollListener() {
override fun onLoadMore() {
item_progress_bar.setVisibility(View.VISIBLE)
offset = offset + 10
homeModel.matchesParametre(CommonUtility.getGlobalString(activity as Activity, "userId"), "" + offset)
}
})
和 Adapter Insilization,最初我膨胀空列表并在覆盖方法中添加列表之后。
override fun matchesList(list: ArrayList<ProfileSelectedData>) {
avlodder.hide()
iv_nodata.visibility = View.GONE
matcheslist.addAll(list)
adapter.notifyDataSetChanged()
item_progress_bar.setVisibility(View.GONE)
}
关于android - RecyclerView.addOnScrolledToEnd 和 SwipeRefreshLayout.setOnRefreshListener 在 Kotlin 中无法正常工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48880528/
我最近在/ drawable中添加了一些.gifs,以便可以将它们与按钮一起使用。这个工作正常(没有错误)。现在,当我重建/运行我的应用程序时,出现以下错误: Error: Gradle: Execu
Android 中有返回内部存储数据路径的方法吗? 我有 2 部 Android 智能手机(Samsung s2 和 s7 edge),我在其中安装了一个应用程序。我想使用位于这条路径中的 sqlit
这个问题在这里已经有了答案: What's the difference between "?android:" and "@android:" in an android layout xml f
我只想知道 android 开发手机、android 普通手机和 android root 手机之间的实际区别。 我们不能从实体店或除 android marketplace 以外的其他地方购买开发手
自Gradle更新以来,我正在努力使这个项目达到标准。这是一个团队项目,它使用的是android-apt插件。我已经进行了必要的语法更改(编译->实现和apt->注释处理器),但是编译器仍在告诉我存在
我是android和kotlin的新手,所以请原谅要解决的一个非常简单的问题! 我已经使用导航体系结构组件创建了一个基本应用程序,使用了底部的导航栏和三个导航选项。每个导航选项都指向一个专用片段,该片
我目前正在使用 Facebook official SDK for Android . 我现在正在使用高级示例应用程序,但我不知道如何让它获取应用程序墙/流/状态而不是登录的用户。 这可能吗?在那种情
我在下载文件时遇到问题, 我可以在模拟器中下载文件,但无法在手机上使用。我已经定义了上网和写入 SD 卡的权限。 我在服务器上有一个 doc 文件,如果用户单击下载。它下载文件。这在模拟器中工作正常但
这个问题在这里已经有了答案: What is the difference between gravity and layout_gravity in Android? (22 个答案) 关闭 9
任何人都可以告诉我什么是 android 缓存和应用程序缓存,因为当我们谈论缓存清理应用程序时,它的作用是,缓存清理概念是清理应用程序缓存还是像内存管理一样主存储、RAM、缓存是不同的并且据我所知,缓
假设应用程序 Foo 和 Eggs 在同一台 Android 设备上。任一应用程序都可以获取设备上所有应用程序的列表。一个应用程序是否有可能知道另一个应用程序是否已经运行以及运行了多长时间? 最佳答案
我有点困惑,我只看到了从 android 到 pc 或者从 android 到 pc 的例子。我需要制作一个从两部手机 (android) 连接的 android 应用程序进行视频聊天。我在想,我知道
用于使用 Android 以编程方式锁定屏幕。我从 Stackoverflow 之前关于此的问题中得到了一些好主意,并且我做得很好,但是当我运行该代码时,没有异常和错误。而且,屏幕没有锁定。请在这段代
文档说: android:layout_alignParentStart If true, makes the start edge of this view match the start edge
我不知道这两个属性和高度之间的区别。 以一个TextView为例,如果我将它的layout_width设置为wrap_content,并将它的width设置为50 dip,会发生什么情况? 最佳答案
这两个属性有什么关系?如果我有 android:noHistory="true",那么有 android:finishOnTaskLaunch="true" 有什么意义吗? 最佳答案 假设您的应用中有
我是新手,正在尝试理解以下 XML 代码: 查看 developer.android.com 上的文档,它说“starStyle”是 R.attr 中的常量, public static final
在下面的代码中,为什么当我设置时单选按钮的外观会发生变化 android:layout_width="fill_parent" 和 android:width="fill_parent" 我说的是
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 9
假设我有一个函数 fun myFunction(name:String, email:String){},当我调用这个函数时 myFunction('Ali', 'ali@test.com ') 如何
我是一名优秀的程序员,十分优秀!