diff --git a/res/layout/headlines_row_loadmore.xml b/res/layout/headlines_row_loadmore.xml
new file mode 100644
index 00000000..26e21ee5
--- /dev/null
+++ b/res/layout/headlines_row_loadmore.xml
@@ -0,0 +1,33 @@
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/src/org/fox/ttrss/Article.java b/src/org/fox/ttrss/Article.java
index 37b2137f..56f7adcf 100644
--- a/src/org/fox/ttrss/Article.java
+++ b/src/org/fox/ttrss/Article.java
@@ -24,6 +24,13 @@ public class Article implements Parcelable {
readFromParcel(in);
}
+ public Article(int id) {
+ this.id = id;
+ this.title = "";
+ this.link = "";
+ this.tags = new ArrayList();
+ }
+
@Override
public int describeContents() {
return 0;
diff --git a/src/org/fox/ttrss/HeadlinesFragment.java b/src/org/fox/ttrss/HeadlinesFragment.java
index 3f36d569..243ee3cc 100644
--- a/src/org/fox/ttrss/HeadlinesFragment.java
+++ b/src/org/fox/ttrss/HeadlinesFragment.java
@@ -23,6 +23,8 @@ import android.view.View;
import android.view.ContextMenu.ContextMenuInfo;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
+import android.widget.AbsListView;
+import android.widget.AbsListView.OnScrollListener;
import android.widget.AdapterView;
import android.widget.AdapterView.AdapterContextMenuInfo;
import android.widget.AdapterView.OnItemClickListener;
@@ -37,13 +39,18 @@ import com.google.gson.JsonArray;
import com.google.gson.JsonElement;
import com.google.gson.reflect.TypeToken;
-public class HeadlinesFragment extends Fragment implements OnItemClickListener {
+public class HeadlinesFragment extends Fragment implements OnItemClickListener, OnScrollListener {
public static enum ArticlesSelection { ALL, NONE, UNREAD };
+ public static final int HEADLINES_REQUEST_SIZE = 30;
+ public static final int HEADLINES_BUFFER_MAX = 500;
+
private final String TAG = this.getClass().getSimpleName();
private Feed m_feed;
private int m_activeArticleId;
+ private boolean m_refreshInProgress = false;
+ private boolean m_canLoadMore = false;
private ArticleListAdapter m_adapter;
private ArticleList m_articles = new ArticleList();
@@ -81,6 +88,7 @@ public class HeadlinesFragment extends Fragment implements OnItemClickListener {
m_articles = savedInstanceState.getParcelable("articles");
m_activeArticleId = savedInstanceState.getInt("activeArticleId");
m_selectedArticles = savedInstanceState.getParcelable("selectedArticles");
+ m_canLoadMore = savedInstanceState.getBoolean("canLoadMore");
}
View view = inflater.inflate(R.layout.headlines_fragment, container, false);
@@ -89,6 +97,7 @@ public class HeadlinesFragment extends Fragment implements OnItemClickListener {
m_adapter = new ArticleListAdapter(getActivity(), R.layout.headlines_row, (ArrayList)m_articles);
list.setAdapter(m_adapter);
list.setOnItemClickListener(this);
+ list.setOnScrollListener(this);
registerForContextMenu(list);
Log.d(TAG, "onCreateView, feed=" + m_feed);
@@ -114,15 +123,19 @@ public class HeadlinesFragment extends Fragment implements OnItemClickListener {
if (list != null) {
Article article = (Article)list.getItemAtPosition(position);
- m_articleOps.openArticle(article, 0);
+ if (article.id >= 0) {
+ m_articleOps.openArticle(article, 0);
- m_activeArticleId = article.id;
- m_adapter.notifyDataSetChanged();
+ m_activeArticleId = article.id;
+ m_adapter.notifyDataSetChanged();
+ }
}
}
@SuppressWarnings({ "unchecked", "serial" })
public void refresh(boolean append) {
+ m_refreshInProgress = true;
+
HeadlinesRequest req = new HeadlinesRequest(getActivity().getApplicationContext());
final String sessionId = ((MainActivity)getActivity()).getSessionId();
@@ -136,21 +149,21 @@ public class HeadlinesFragment extends Fragment implements OnItemClickListener {
}
if (skip == 0) skip = m_articles.size();
+ } else {
+ setLoadingStatus(R.string.blank, true);
}
final int fskip = skip;
req.setOffset(skip);
- setLoadingStatus(R.string.blank, true);
-
HashMap map = new HashMap() {
{
put("op", "getHeadlines");
put("sid", sessionId);
put("feed_id", String.valueOf(m_feed.id));
put("show_content", "true");
- put("limit", String.valueOf(30));
+ put("limit", String.valueOf(HEADLINES_REQUEST_SIZE));
put("offset", String.valueOf(0));
put("view_mode", showUnread ? "adaptive" : "all_articles");
put("skip", String.valueOf(fskip));
@@ -170,6 +183,7 @@ public class HeadlinesFragment extends Fragment implements OnItemClickListener {
out.putParcelable("articles", m_articles);
out.putInt("activeArticleId", m_activeArticleId);
out.putParcelable("selectedArticles", m_selectedArticles);
+ out.putBoolean("canLoadMore", m_canLoadMore);
}
public void setLoadingStatus(int status, boolean showProgress) {
@@ -203,31 +217,43 @@ public class HeadlinesFragment extends Fragment implements OnItemClickListener {
Type listType = new TypeToken>() {}.getType();
final List articles = new Gson().fromJson(content, listType);
+ while (m_articles.size() > HEADLINES_BUFFER_MAX)
+ m_articles.remove(0);
+
if (m_offset == 0)
m_articles.clear();
+ else
+ m_articles.remove(m_articles.size()-1); // remove previous placeholder
int last_position = m_articles.size();
for (Article f : articles)
m_articles.add(f);
-
+
+ if (articles.size() == HEADLINES_REQUEST_SIZE) {
+ Article placeholder = new Article(-1);
+ m_articles.add(placeholder);
+
+ m_canLoadMore = true;
+ } else {
+ m_canLoadMore = false;
+ }
+
m_adapter.notifyDataSetChanged();
-
- ListView list = (ListView)getView().findViewById(R.id.headlines);
+
+ /* ListView list = (ListView)getView().findViewById(R.id.headlines);
if (list != null && m_offset != 0 && articles.size() > 0) {
list.setSelection(last_position-1);
- }
-
- MainActivity activity = (MainActivity)getActivity();
- activity.setCanLoadMore(articles.size() >= 30);
- activity.initMainMenu();
+ } */
if (m_articles.size() == 0)
setLoadingStatus(R.string.no_headlines_to_display, false);
else
setLoadingStatus(R.string.blank, false);
+ m_refreshInProgress = false;
+
return;
}
@@ -242,6 +268,7 @@ public class HeadlinesFragment extends Fragment implements OnItemClickListener {
} else {
setLoadingStatus(getErrorMessage(), false);
}
+ m_refreshInProgress = false;
}
public void setOffset(int skip) {
@@ -255,8 +282,9 @@ public class HeadlinesFragment extends Fragment implements OnItemClickListener {
public static final int VIEW_NORMAL = 0;
public static final int VIEW_UNREAD = 1;
public static final int VIEW_SELECTED = 2;
+ public static final int VIEW_LOADMORE = 3;
- public static final int VIEW_COUNT = VIEW_SELECTED+1;
+ public static final int VIEW_COUNT = VIEW_LOADMORE+1;
public ArticleListAdapter(Context context, int textViewResourceId, ArrayList items) {
super(context, textViewResourceId, items);
@@ -271,7 +299,9 @@ public class HeadlinesFragment extends Fragment implements OnItemClickListener {
public int getItemViewType(int position) {
Article a = items.get(position);
- if (a.id == m_activeArticleId) {
+ if (a.id == -1) {
+ return VIEW_LOADMORE;
+ } else if (a.id == m_activeArticleId) {
return VIEW_SELECTED;
} else if (a.unread) {
return VIEW_UNREAD;
@@ -291,6 +321,9 @@ public class HeadlinesFragment extends Fragment implements OnItemClickListener {
int layoutId = R.layout.headlines_row;
switch (getItemViewType(position)) {
+ case VIEW_LOADMORE:
+ layoutId = R.layout.headlines_row_loadmore;
+ break;
case VIEW_UNREAD:
layoutId = R.layout.headlines_row_unread;
break;
@@ -439,4 +472,16 @@ public class HeadlinesFragment extends Fragment implements OnItemClickListener {
return tmp;
}
+ @Override
+ public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {
+ if (!m_refreshInProgress && m_canLoadMore && firstVisibleItem + visibleItemCount == m_articles.size()) {
+ refresh(true);
+ }
+ }
+
+ @Override
+ public void onScrollStateChanged(AbsListView view, int scrollState) {
+ // TODO Auto-generated method stub
+ }
+
}
diff --git a/src/org/fox/ttrss/MainActivity.java b/src/org/fox/ttrss/MainActivity.java
index ab2f59b1..0ede0c83 100644
--- a/src/org/fox/ttrss/MainActivity.java
+++ b/src/org/fox/ttrss/MainActivity.java
@@ -49,7 +49,7 @@ public class MainActivity extends FragmentActivity implements FeedsFragment.OnFe
private boolean m_smallScreenMode;
private boolean m_unreadOnly = true;
private boolean m_unreadArticlesOnly = true;
- private boolean m_canLoadMore = true;
+ //private boolean m_canLoadMore = true;
private boolean m_compatMode = false;
private boolean m_enableCats = false;
private int m_apiLevel = 0;
@@ -302,7 +302,7 @@ public class MainActivity extends FragmentActivity implements FeedsFragment.OnFe
m_activeFeed = savedInstanceState.getParcelable("activeFeed");
m_selectedArticle = savedInstanceState.getParcelable("selectedArticle");
m_unreadArticlesOnly = savedInstanceState.getBoolean("unreadArticlesOnly");
- m_canLoadMore = savedInstanceState.getBoolean("canLoadMore");
+ //m_canLoadMore = savedInstanceState.getBoolean("canLoadMore");
m_activeCategory = savedInstanceState.getParcelable("activeCategory");
m_apiLevel = savedInstanceState.getInt("apiLevel");
}
@@ -397,7 +397,7 @@ public class MainActivity extends FragmentActivity implements FeedsFragment.OnFe
out.putParcelable("activeFeed", m_activeFeed);
out.putParcelable("selectedArticle", m_selectedArticle);
out.putBoolean("unreadArticlesOnly", m_unreadArticlesOnly);
- out.putBoolean("canLoadMore", m_canLoadMore);
+ //out.putBoolean("canLoadMore", m_canLoadMore);
out.putParcelable("activeCategory", m_activeCategory);
out.putInt("apiLevel", m_apiLevel);
}
@@ -581,7 +581,6 @@ public class MainActivity extends FragmentActivity implements FeedsFragment.OnFe
dialog = builder.create();
dialog.show();
}
-
return true;
case R.id.headlines_mark_as_read:
if (hf != null) {
@@ -590,14 +589,7 @@ public class MainActivity extends FragmentActivity implements FeedsFragment.OnFe
for (Article a : articles)
a.unread = false;
- ApiRequest req = new ApiRequest(getApplicationContext()) {
- @Override
- protected void onPostExecute(JsonElement result) {
- if (result != null) {
- viewFeed(m_activeFeed, true);
- }
- }
- };
+ ApiRequest req = new ApiRequest(getApplicationContext());
final String articleIds = articlesToIdString(articles);
@@ -613,7 +605,6 @@ public class MainActivity extends FragmentActivity implements FeedsFragment.OnFe
};
req.execute(map);
-
}
return true;
case R.id.share_article:
@@ -736,9 +727,9 @@ public class MainActivity extends FragmentActivity implements FeedsFragment.OnFe
}
- public void setCanLoadMore(boolean canLoadMore) {
+ /* public void setCanLoadMore(boolean canLoadMore) {
m_canLoadMore = canLoadMore;
- }
+ } */
public void initMainMenu() {
if (m_menu != null) {