- 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/
我正在尝试 react-native run-android 有这个问题吗?? Could not resolve all files for configuration ':react-native
关于我的环境的信息 Mac OS 大苏尔:11.4 VS for Mac:8.10.6(内部版本 10) Xamarin.Android:11.3.0.4 SDK 工具版本:26.1.1 SDK平台工
我正在开发一个 Android 应用程序。我将 listview 与 SwipefreshLayout 一起使用。但是当我使用 SwipefreshLayout 刷新 ListView 时,加载圈永远
我的应用程序包含一个 SwipeRefreshLayout ,其中包含一个 listView 。该列表从服务器接收数据。因此,当获取数据并且正确填充列表时,滑动刷新可以工作,但是当数据未传递到列表并且
我在使用 SwipeRefreshLayout 时遇到了内存泄漏问题,想知道是否有任何方法可以通过解决方法来解决此问题? 在一个空项目中,以滑动布局作为主要 Activity 的根,如果你向下滑动,并
我尝试为现有的 ListView 实现 SwipeRefreshLayout,但每当我运行该程序时,即使我通过拉动刷新它,它也只显示一项。但是,刷新动画有效。相关代码如下: items = (
这个问题在这里已经有了答案: When I am scrolling up SwipeRefreshLayout refreshing my app (4 个答案) 关闭 3 年前。 我有一个带有
我正在我的应用程序中实现 SwipeRefreshLayout。我已成功实现 OnRefreshListener,并且能够正确刷新 ListView 项目。 我的问题是,如果我向下滑动,当 Swipe
fragment 的onCreateView方法内部: swipeRefreshLayout.setOnRefreshListener(new SwipeRefreshLayout.OnRefresh
我有以下问题:我有很长的 ListView,但是当我向上滚动时会触发 onRefresh,而不是先滚动回顶部。我该如何解决这个问题? 我将以下 fragment 插入到 FrameLay
我有一个 SwipeRefreshLayout,其中包含用于聊天 Activity 的 RecyclerView。我将其底部限制在屏幕底部线性布局的顶部:
我在我的布局中使用了 SwipeRefreshLayout,我需要显示一个动画来调整已被下拉的距离。 到目前为止,我还没有找到任何方法来检测已拉动的实际距离。 SwipeRefreshListener
我喜欢新的 SwipeRefreshLayout!它看起来很棒,而且非常容易使用。但我想在两个方向上使用它。我有一个消息屏幕,我想通过从上到下滑动来加载旧消息,我想通过从下到上滑动来加载新消息。 这个
我正在使用支持库中的 SwipeRefreshListener。我正在从 AsyncTask 的 onPreExecute 和 onPostExecute 调用 setRefresh。然而,什么都没有
是否可以在不显示进度微调器的情况下执行 SwipeRefreshLayout 功能。现在,它的默认拉动刷新行为完美无缺地显示了 Progress 微调器和 onRefresh() 我隐藏了它。但我想将
我正在为 Google 支持库的 SwipeRefreshLayout 中的文档而苦苦挣扎。 当我使用 setOnRefreshListener 设置的回调接收到刷新调用时,我执行了我的操作,之后我发
有没有办法使用主题修改 SwipeRefreshLayout 中箭头的颜色? 我知道您可以通过编程方式使用此代码 public void setColorSchemeResources (int...
我已经实现了一个将 SwipeRefreshLayout 作为内容 View 的 fragment 。刷新动画在 onRefresh 时触发,但即使在从服务器检索数据后将 setRefreshing
我有一个向用户显示项目列表的 Activity ,它使用了分页库。我的问题是当用户向下滑动屏幕时我无法重新加载列表,以便它再次从服务器获取数据。 这是我的数据源工厂: public class Cou
我在我的 listView 中添加了一个“拉动刷新”,我还想在列表为空时添加一个空 View - 现在我收到了这个错误。我怎样才能使这项工作?如果我将 View 定位在 swipeRefresh 之外
我是一名优秀的程序员,十分优秀!