How to get feeds from public Facebook page without using Facebook Login in a Fragment

This post basically helps the hacker get message and photos of the posts of a public Facebook page in a fragment.Will soon update the code to allow reading older posts,seeing comments and likes on individual posts.Hope Facebook doesn’t ban it!

Well,this is my first hack in my android development career.So I hope the viewers forgive me for any amateurish approach towards the problem.
It all started when I was developing my college app and I wanted to include feeds from my college’s confession page.But due to recent Facebook updates,it only allowed feeds to be extracted after loging into Facebook.I found it a tedious job.
After some exhaustive hacking on my laptop and a couple of redbulls,I finally figured out a way to extract the feeds in the JSON format.
I have used android volley network library to make http calls and download the images. As volley comes with powerful image caching mechanism, we dont have to much worry about caching the requests and images. I would like to credit Ravi Tamada (Android Hive) for his excellent tutorials.
For the Github repo click here.
The following steps explain them-

Step 1

Create a Facebook developer account and create a new App Id.

Step 2

Get the App Id and the App secret

Step 3

Get an access token by making a GET request to this url:

https://graph.facebook.com/oauth/access_token?client_id="APP_ID"&client_secret="APP_SECRET"&grant_type=client_credential

#NOTE: Remove the quotes too!

Step 4

Go to your fan page on facebook and look at the url. Once you have that, make a GET request to this url:

https://graph.facebook.com/{fan-page-name}?access_token={access-token}

#NOTE: Remove the curly brackets too!

Step 5

After obtaining the page Id, make a GET request to this url for the JSON:

https://graph.facebook.com/{page id}/feed?fields=id,full_picture,message,story,created_time,link&access_token={access token} Example JSON

Step 6

Add the following library in your dependencies in your build.gradle

1
compile 'com.mcxiaoke.volley:library-aar:1.0.0'

Step 7

Open res -> values -> dimens.xml and add following dimensions. If you dont have dimens.xml, create a new file and add these values.

dimens.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
<?xml version="1.0" encoding="utf-8"?>
<resources>

<!-- Default screen margins, per the Android Design guidelines. -->
<dimen name="activity_horizontal_margin">16dp</dimen>
<dimen name="activity_vertical_margin">16dp</dimen>


<dimen name="feed_item_margin">10dp</dimen>
<dimen name="feed_item_padding_top_bottom">20dp</dimen>
<dimen name="feed_item_padding_left_right">15dp</dimen>
<dimen name="feed_item_profile_pic">50dp</dimen>
<dimen name="feed_item_profile_info_padd">10dp</dimen>
<dimen name="feed_item_profile_name">15dp</dimen>
<dimen name="feed_item_timestamp">13dp</dimen>
<dimen name="feed_item_status_pad_left_right">15dp</dimen>
<dimen name="feed_item_status_pad_top">13dp</dimen>
<dimen name="feed_item_corner_radius">3dp</dimen>
<dimen name="feed_item_border_width">1dp</dimen>

</resources>

Step 8

Also create another xml file named colors.xml under res -> values and add following colors.

colors.xml
1
2
3
4
5
6
7
8
9
<?xml version="1.0" encoding="utf-8"?>
<resources>
<color name="white">#ffffff</color>
<color name="feed_bg">#d3d6db</color>
<color name="feed_item_bg">#ffffff</color>
<color name="feed_item_border">#c2c3c8</color>
<color name="link">#0a80d1</color>
<color name="timestamp">#a0a3a7</color>
</resources>

Step 9

Under res -> drawable, create a new file called bg_parent_rounded_corner.xml and paste the below code. This xml will give a rounded corner background to feed item.

bg_parent_rounded_corner.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"
android:shape="rectangle" >


<!-- view background color -->
<solid android:color="@color/feed_item_bg" >
</solid>

<!-- view border color and width -->
<stroke
android:width="@dimen/feed_item_border_width"
android:color="@color/feed_item_border" >

</stroke>

<!-- Here is the corner radius -->
<corners android:radius="@dimen/feed_item_corner_radius" >
</corners>

</shape>

Step 10

Now open your AndroidManifest.xml file and add Application.java class in tag. Also we need to add INTERNET permission as this app makes network calls.

AndroidManifest.xml
1
2
3
<uses-permission android:name="android.permission.INTERNET" />

<application android:name=".AppController"> ....</application>

Step 11

Open your layout for your fragment (yourfragmentclass.xml) and add a list view element for the feed list.

yourfragmentclass.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" >


<ListView
android:id="@+id/list"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:divider="@null" />


</LinearLayout>

Step 12

Create another layout file named feed_item.xml under res -> layout folder. This layout file represents each individual feed item row in the list view.

feed_item.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@color/feed_bg"
android:orientation="vertical" >


<LinearLayout
android:layout_width="fill_parent"
android:layout_height="fill_parent"

android:layout_marginLeft="@dimen/feed_item_margin"
android:layout_marginRight="@dimen/feed_item_margin"
android:layout_marginTop="@dimen/feed_item_margin"
android:background="@drawable/bg_parent_rounded_corner"
android:orientation="vertical"
android:paddingBottom="@dimen/feed_item_padding_top_bottom"
android:paddingTop="@dimen/feed_item_padding_top_bottom" >


<LinearLayout
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:paddingLeft="@dimen/feed_item_padding_left_right"
android:paddingRight="@dimen/feed_item_padding_left_right" >


<com.android.volley.toolbox.NetworkImageView
android:id="@+id/profilePic"
android:layout_width="@dimen/feed_item_profile_pic"
android:layout_height="@dimen/feed_item_profile_pic"
android:scaleType="fitCenter" >

</com.android.volley.toolbox.NetworkImageView>

<LinearLayout
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:paddingLeft="@dimen/feed_item_profile_info_padd" >


<TextView
android:id="@+id/name"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:textSize="@dimen/feed_item_profile_name"
android:textStyle="bold" />


<TextView
android:id="@+id/timestamp"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:textColor="@color/timestamp"
android:textSize="@dimen/feed_item_timestamp" />

</LinearLayout>
</LinearLayout>

<TextView
android:id="@+id/txtStatusMsg"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:paddingBottom="5dp"
android:paddingLeft="@dimen/feed_item_status_pad_left_right"
android:paddingRight="@dimen/feed_item_status_pad_left_right"
android:paddingTop="@dimen/feed_item_status_pad_top" />


<TextView
android:id="@+id/txtUrl"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:linksClickable="true"
android:paddingBottom="10dp"
android:paddingLeft="@dimen/feed_item_status_pad_left_right"
android:paddingRight="@dimen/feed_item_status_pad_left_right"
android:textColorLink="@color/link" />


<YOURPACKAGENAMEHERE.FeedImageView
android:id="@+id/feedImage1"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@color/white"
android:scaleType="fitXY"
android:visibility="visible" />

</LinearLayout>

</LinearLayout>

Step 13

Since we have the JSON, we just need to parse it through volley. Create the following classes in your Android Studio.

AppController.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
import android.app.Application;
import android.text.TextUtils;

import com.android.volley.Request;
import com.android.volley.RequestQueue;
import com.android.volley.toolbox.ImageLoader;
import com.android.volley.toolbox.Volley;

public class AppController extends Application {

public static final String TAG = AppController.class.getSimpleName();

private RequestQueue mRequestQueue;
private ImageLoader mImageLoader;
LruBitmapCache mLruBitmapCache;

private static AppController mInstance;

@Override
public void onCreate() {
super.onCreate();
mInstance = this;
}

public static synchronized AppController getInstance() {
return mInstance;
}

public RequestQueue getRequestQueue() {
if (mRequestQueue == null) {
mRequestQueue = Volley.newRequestQueue(getApplicationContext());
}

return mRequestQueue;
}

public ImageLoader getImageLoader() {
getRequestQueue();
if (mImageLoader == null) {
getLruBitmapCache();
mImageLoader = new ImageLoader(this.mRequestQueue, mLruBitmapCache);
}

return this.mImageLoader;
}

public LruBitmapCache getLruBitmapCache() {
if (mLruBitmapCache == null)
mLruBitmapCache = new LruBitmapCache();
return this.mLruBitmapCache;
}

public <T> void addToRequestQueue(Request<T> req, String tag) {
req.setTag(TextUtils.isEmpty(tag) ? TAG : tag);
getRequestQueue().add(req);
}

public <T> void addToRequestQueue(Request<T> req) {
req.setTag(TAG);
getRequestQueue().add(req);
}

public void cancelPendingRequests(Object tag) {
if (mRequestQueue != null) {
mRequestQueue.cancelAll(tag);
}
}
}
LruBitmapCache.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
import android.app.Application;
import android.graphics.Bitmap;
import android.support.v4.util.LruCache;

import com.android.volley.toolbox.ImageLoader.ImageCache;

public class LruBitmapCache extends LruCache<String, Bitmap> implements
ImageCache {

public static int getDefaultLruCacheSize() {
final int maxMemory = (int) (Runtime.getRuntime().maxMemory() / 1024);
final int cacheSize = maxMemory / 8;

return cacheSize;
}

public LruBitmapCache() {
this(getDefaultLruCacheSize());
}

public LruBitmapCache(int sizeInKiloBytes) {
super(sizeInKiloBytes);
}

@Override
protected int sizeOf(String key, Bitmap value) {
return value.getRowBytes() * value.getHeight() / 1024;
}

@Override
public Bitmap getBitmap(String url) {
return get(url);
}

@Override
public void putBitmap(String url, Bitmap bitmap) {
put(url, bitmap);
}
}
FeedItem.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
public class FeedItem {

private String id,status, image, timeStamp, url;

public FeedItem() {
}

public FeedItem(String id, String image, String status, String timeStamp, String url) {
super();
this.id = id;
this.image = image;
this.status = status;
this.timeStamp = timeStamp;
this.url = url;
}

public String getId() {
return id;
}

public void setId(String id) {
this.id = id;
}

public String getImge() {
return image;
}

public void setImge(String image) {
this.image = image;
}

public String getStatus() {
return status;
}

public void setStatus(String status) {
this.status = status;
}

public String getTimeStamp() {
return timeStamp;
}

public void setTimeStamp(String timeStamp) {
this.timeStamp = timeStamp;
}

public String getUrl() {
return url;
}

public void setUrl(String url) {
this.url = url;
}
}
FeedListAdapter.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
import android.app.Activity;
import android.content.Context;
import android.text.Html;
import android.text.TextUtils;
import android.text.format.DateUtils;
import android.text.method.LinkMovementMethod;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.TextView;

import com.android.volley.toolbox.ImageLoader;

import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.List;


public class FeedListAdapter extends BaseAdapter {
private Activity activity;
private LayoutInflater inflater;
private List<FeedItem> feedItems;
ImageLoader imageLoader = AppController.getInstance().getImageLoader();

public FeedListAdapter(Activity activity, List<FeedItem> feedItems) {
this.activity = activity;
this.feedItems = feedItems;
}

@Override
public int getCount() {
return feedItems.size();
}

@Override
public Object getItem(int location) {
return feedItems.get(location);
}

@Override
public long getItemId(int position) {
return position;
}

@Override
public View getView(int position, View convertView, ViewGroup parent) {

if (inflater == null)
inflater = (LayoutInflater) activity
.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
if (convertView == null)
convertView = inflater.inflate(R.layout.feed_item, null);

if (imageLoader == null)
imageLoader = AppController.getInstance().getImageLoader();


TextView timestamp = (TextView) convertView
.findViewById(R.id.timestamp);
TextView statusMsg = (TextView) convertView
.findViewById(R.id.txtStatusMsg);
TextView url = (TextView) convertView.findViewById(R.id.txtUrl);

FeedImageView feedImageView = (FeedImageView) convertView
.findViewById(R.id.feedImage1);

FeedItem item = feedItems.get(position);



// Converting timestamp into x ago format
String a=item.getTimeStamp();
a=a.replace("T"," ").replace("+",".");

SimpleDateFormat dateFormat=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSS");
try {
Date timeCreatedDate = dateFormat.parse(a);
String timeAgo =(String) DateUtils.getRelativeTimeSpanString(
(timeCreatedDate.getTime()),
System.currentTimeMillis(), DateUtils.SECOND_IN_MILLIS);
timestamp.setText(timeAgo);
} catch (ParseException e) {
e.printStackTrace();
}


// Chcek for empty status message
if (!TextUtils.isEmpty(item.getStatus())) {
statusMsg.setText(item.getStatus());
statusMsg.setVisibility(View.VISIBLE);
} else {
// status is empty, remove from view
statusMsg.setVisibility(View.GONE);
}

// Checking for null feed url
if (item.getUrl() != null) {
url.setText(Html.fromHtml("<a href=\"" + item.getUrl() + "\">"
+ item.getUrl() + "</a> "));

// Making url clickable
url.setMovementMethod(LinkMovementMethod.getInstance());
url.setVisibility(View.VISIBLE);
} else {
// url is null, remove from the view
url.setVisibility(View.GONE);
}


// Feed image
if (item.getImge() != null) {
feedImageView.setImageUrl(item.getImge(), imageLoader);
feedImageView.setVisibility(View.VISIBLE);
feedImageView
.setResponseObserver(new FeedImageView.ResponseObserver() {
@Override
public void onError() {
}

@Override
public void onSuccess() {
}
});
} else {
feedImageView.setVisibility(View.GONE);
}

return convertView;
}

}
FeedImageView.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
import android.content.Context;
import android.text.TextUtils;
import android.util.AttributeSet;
import android.widget.ImageView;
import android.widget.LinearLayout.LayoutParams;

import com.android.volley.VolleyError;
import com.android.volley.toolbox.ImageLoader;
import com.android.volley.toolbox.ImageLoader.ImageContainer;
import com.android.volley.toolbox.ImageLoader.ImageListener;

public class FeedImageView extends ImageView {

public interface ResponseObserver {
public void onError();

public void onSuccess();
}

private ResponseObserver mObserver;

public void setResponseObserver(ResponseObserver observer) {
mObserver = observer;
}

/**
* The URL of the network image to load
*/

private String mUrl;

/**
* Resource ID of the image to be used as a placeholder until the network
* image is loaded.
*/

private int mDefaultImageId;

/**
* Resource ID of the image to be used if the network response fails.
*/

private int mErrorImageId;

/**
* Local copy of the ImageLoader.
*/

private ImageLoader mImageLoader;

/**
* Current ImageContainer. (either in-flight or finished)
*/

private ImageContainer mImageContainer;

public FeedImageView(Context context) {
this(context, null);
}

public FeedImageView(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}

public FeedImageView(Context context, AttributeSet attrs,
int defStyle)
{

super(context, attrs, defStyle);
}

/**
* Sets URL of the image that should be loaded into this view. Note that
* calling this will immediately either set the cached image (if available)
* or the default image specified by

*
* @param url
* The URL that should be loaded into this ImageView.
* @param imageLoader
* ImageLoader that will be used to make the request.
*/

public void setImageUrl(String url, ImageLoader imageLoader) {
mUrl = url;
mImageLoader = imageLoader;
// The URL has potentially changed. See if we need to load it.
loadImageIfNecessary(false);
}

/**
* Sets the default image resource ID to be used for this view until the
* attempt to load it completes.
*/

public void setDefaultImageResId(int defaultImage) {
mDefaultImageId = defaultImage;
}

/**
* Sets the error image resource ID to be used for this view in the event
* that the image requested fails to load.
*/

public void setErrorImageResId(int errorImage) {
mErrorImageId = errorImage;
}

/**
* Loads the image for the view if it isn't already loaded.
*
* @param isInLayoutPass
* True if this was invoked from a layout pass, false otherwise.
*/

private void loadImageIfNecessary(final boolean isInLayoutPass) {
final int width = getWidth();
int height = getHeight();

boolean isFullyWrapContent = getLayoutParams() != null
&& getLayoutParams().height == LayoutParams.WRAP_CONTENT
&& getLayoutParams().width == LayoutParams.WRAP_CONTENT;
// if the view's bounds aren't known yet, and this is not a
// wrap-content/wrap-content
// view, hold off on loading the image.
if (width == 0 && height == 0 && !isFullyWrapContent) {
return;
}

// if the URL to be loaded in this view is empty, cancel any old
// requests and clear the
// currently loaded image.
if (TextUtils.isEmpty(mUrl)) {
if (mImageContainer != null) {
mImageContainer.cancelRequest();
mImageContainer = null;
}
setDefaultImageOrNull();
return;
}

// if there was an old request in this view, check if it needs to be
// canceled.
if (mImageContainer != null && mImageContainer.getRequestUrl() != null) {
if (mImageContainer.getRequestUrl().equals(mUrl)) {
// if the request is from the same URL, return.
return;
} else {
// if there is a pre-existing request, cancel it if it's
// fetching a different URL.
mImageContainer.cancelRequest();
setDefaultImageOrNull();
}
}

// The pre-existing content of this view didn't match the current URL.
// Load the new image
// from the network.
ImageContainer newContainer = mImageLoader.get(mUrl,
new ImageListener() {
@Override
public void onErrorResponse(VolleyError error) {
if (mErrorImageId != 0) {
setImageResource(mErrorImageId);
}

if (mObserver != null) {
mObserver.onError();
}
}

@Override
public void onResponse(final ImageContainer response,
boolean isImmediate)
{

// If this was an immediate response that was delivered
// inside of a layout
// pass do not set the image immediately as it will
// trigger a requestLayout
// inside of a layout. Instead, defer setting the image
// by posting back to
// the main thread.
if (isImmediate && isInLayoutPass) {
post(new Runnable() {
@Override
public void run() {
onResponse(response, false);
}
});
return;
}

int bWidth = 0, bHeight = 0;
if (response.getBitmap() != null) {

setImageBitmap(response.getBitmap());
bWidth = response.getBitmap().getWidth();
bHeight = response.getBitmap().getHeight();
adjustImageAspect(bWidth, bHeight);

} else if (mDefaultImageId != 0) {
setImageResource(mDefaultImageId);
}

if (mObserver != null) {
mObserver.onSuccess();

}
}
});

// update the ImageContainer to be the new bitmap container.
mImageContainer = newContainer;

}

private void setDefaultImageOrNull() {
if (mDefaultImageId != 0) {
setImageResource(mDefaultImageId);
} else {
setImageBitmap(null);
}
}

@Override
protected void onLayout(boolean changed, int left, int top, int right,
int bottom)
{

super.onLayout(changed, left, top, right, bottom);
loadImageIfNecessary(true);
}

@Override
protected void onDetachedFromWindow() {
if (mImageContainer != null) {
// If the view was bound to an image request, cancel it and clear
// out the image from the view.
mImageContainer.cancelRequest();
setImageBitmap(null);
// also clear out the container so we can reload the image if
// necessary.
mImageContainer = null;
}
super.onDetachedFromWindow();
}

@Override
protected void drawableStateChanged() {
super.drawableStateChanged();
invalidate();
}

/*
* Adjusting imageview height
* */

private void adjustImageAspect(int bWidth, int bHeight) {
LayoutParams params = (LayoutParams) getLayoutParams();

if (bWidth == 0 || bHeight == 0)
return;

int swidth = getWidth();
int new_height = 0;
new_height = swidth * bHeight / bWidth;
params.width = swidth;
params.height = new_height;
setLayoutParams(params);
}
}
YourFagmentClass.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
import android.support.annotation.Nullable;
import android.support.v4.app.Fragment;
import android.support.v4.widget.SwipeRefreshLayout;
import android.os.Bundle;

import java.io.UnsupportedEncodingException;
import java.util.ArrayList;
import java.util.List;

import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;

import android.annotation.SuppressLint;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ListView;
import android.widget.Toast;

import com.android.volley.Cache;
import com.android.volley.Cache.Entry;
import com.android.volley.Request.Method;
import com.android.volley.Response;
import com.android.volley.VolleyError;
import com.android.volley.toolbox.JsonObjectRequest;

public class YourFragmentClass extends Fragment {


private ListView listView;
private FeedListAdapter listAdapter;
private List<FeedItem> feedItems;
SwipeRefreshLayout mSwipeRefreshLayout;
private String URL_FEED = "ENTER YOUR JSON URL HERE";

@SuppressLint("NewApi")
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
View v =inflater.inflate(R.layout.yourfragmentclass,container,false);

listView = (ListView)v.findViewById(R.id.list);

feedItems = new ArrayList<FeedItem>();

listAdapter = new FeedListAdapter(getActivity(), feedItems);
listView.setAdapter(listAdapter);

// We first check for cached request
Cache cache = AppController.getInstance().getRequestQueue().getCache();
Entry entry = cache.get(URL_FEED);
if (entry != null) {
// fetch the data from cache
try {
String data = new String(entry.data, "UTF-8");
try {
parseJsonFeed(new JSONObject(data));
} catch (JSONException e) {
e.printStackTrace();
}
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}

} else {
// making fresh volley request and getting json
JsonObjectRequest jsonReq = new JsonObjectRequest(Method.GET,
URL_FEED, null, new Response.Listener<JSONObject>() {

@Override
public void onResponse(JSONObject response) {

if (response != null) {
parseJsonFeed(response);
}
}
}, new Response.ErrorListener() {

@Override
public void onErrorResponse(VolleyError error) {

Toast.makeText(getActivity(),"NO INTERNET",Toast.LENGTH_LONG).show();
}
});

// Adding request to volley request queue
AppController.getInstance().addToRequestQueue(jsonReq);
}

mSwipeRefreshLayout = (SwipeRefreshLayout)v.findViewById(R.id.swipeContainer);
mSwipeRefreshLayout.setOnRefreshListener(new SwipeRefreshLayout.OnRefreshListener() {
@Override
public void onRefresh() {
//Refreshing data on server
feedItems.clear();
JsonObjectRequest jsonReq = new JsonObjectRequest(Method.GET,
URL_FEED, null, new Response.Listener<JSONObject>() {

@Override
public void onResponse(JSONObject response) {

if (response != null) {
parseJsonFeed(response);
}
}
}, new Response.ErrorListener() {

@Override
public void onErrorResponse(VolleyError error) {

mSwipeRefreshLayout.setRefreshing(false);
}
});

// Adding request to volley request queue
AppController.getInstance().addToRequestQueue(jsonReq);


}
});

return v;
}

/**
* Parsing json reponse and passing the data to feed view list adapter
* */

private void parseJsonFeed(JSONObject response) {
mSwipeRefreshLayout.setRefreshing(true);
try {
JSONArray feedArray = response.getJSONArray("data");

for (int i = 0; i < feedArray.length(); i++) {
JSONObject feedObj = (JSONObject) feedArray.get(i);

FeedItem item = new FeedItem();
item.setId(feedObj.getString("id"));


// Image might be null sometimes
String image = feedObj.isNull("full_picture") ? null : feedObj
.getString("full_picture");
item.setImge(image);

if(feedObj.opt("message")!=null)
item.setStatus(feedObj.getString("message"));
else
item.setStatus(feedObj.getString("story"));

item.setTimeStamp(feedObj.getString("created_time"));

// url might be null sometimes
String feedUrl = feedObj.isNull("link") ? null : feedObj
.getString("link");
item.setUrl(feedUrl);

feedItems.add(item);
}

// notify data changes to list adapter
listAdapter.notifyDataSetChanged();
mSwipeRefreshLayout.setRefreshing(false);
} catch (JSONException e) {
e.printStackTrace();
}
}


}

Step 14 ABRA-KADABRA!!!

NOTE: Take care of ENTERYOURPACKAGEHERE!!!

Share Comments