I want to give to my user a much better offline user experience, therefore, I want to build a Service
which stores all POST
, DELETE
,
You need to use a BroadcastReceiver to listen to network change events. Define a broadcastReciver in your AndroidManifest.xml with the following action.
<receiver android:name=".NetworkBroadcastReceiver" >
<intent-filter>
<action android:name="android.net.conn.CONNECTIVITY_CHANGE" />
</intent-filter>
</receiver>
Also add the following permissions as well to your manifest file -
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.INTERNET" />
NetworkBroadcastReceiver -
public class NetworkBroadcastReceiver extends BroadcastReceiver {
@Override
public void onReceive(final Context context, final Intent intent) {
if(isInternetConnected(context)) {
//Do something i.e. trigger an API call or start a IntentService etc.
}
}
public boolean isInternetConnected(Context context) {
ConnectivityManager connectivityManager
= (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo activeNetworkInfo = connectivityManager.getActiveNetworkInfo();
return activeNetworkInfo != null && activeNetworkInfo.isConnected();
}
}
That's all you need to do to make a request as soon as you get Internet.
To cache data, I'll advise you to parse the server response and keep it in your database for offline use. Every time you make a successful request save the new data in database and discard the old one. When user starts the app, load the data from database first and then trigger the Volley request, if request become successful then load the new data in app, store it in database and get rid of the old one. So in case if request fails, user will still be able to see the old data from previous successful request.
To handle inconsistency between data in app and server, you need to use SyncAdapter. SyncAdapter provide a great support for periodic sync in the background. Just put the syncing code in onPerformSync() method of SyncAdapter. It might not work in following two scenarios - 1. If user isn't connected to internet 2. If user device is turned off
To handle these scenario, use the BroadCastReceiver that I explained above in my answer to trigger the SyncAdapter. Also add the following actions to your receiver inside AndroidManifest.xml to listen to boot complete event of device.
<action android:name="android.intent.action.BOOT_COMPLETED" />
<action android:name="android.intent.action.QUICKBOOT_POWERON" />
Let me know if you need more in depth explanation with coding example. Hope it helps
Take a look at this library that I came across - Android Priority Job Queue.
You define your background tasks as Jobs and enqueue them to your JobManager instance. Job Manager will take care of prioritization, persistence, load balancing, delaying, network control, grouping etc. It also provides a nice lifecycle for your jobs to provide a better, consistent user experience.
Also, they monitor network connectivity. When a device is offline, jobs that require network connection won't run until network connectivity is restored.
you maybe do a networking validation method :
1. check the network connection
YES:
a) make the volley request
b) parse the adapter from volley request
c) and save the json into a private file
NO:
a) read the private file
b) make the adapter from json file downloaded.
********************** for example
if(isconnect){
JsonArrayRequest Req = new JsonArrayRequest(url,
new Response.Listener<JSONArray>() {
public void onResponse(JSONArray response) {
//this the request of your http request
Log.d(TAG, response.toString());
// Parsing json
for (int i = 0; i < response.length(); i++) {
try {
......adapter etc etc
}
adapter.notifyDataSetChanged();
//here write the httprequest (json file) in local memory
try {
String textjson= response.toString();
fos = context.openFileOutput("request.json",MODE_PRIVATE);
fos.write(textjson.getBytes("ISO-8859-1")); //encode
fos.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
swipeRefreshLayout.setRefreshing(false);
}
}, new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError error) {
// VolleyLog.d(TAG, "Error: " + error.getMessage());
} }); AppController.getInstance().addToRequestQueue(movieReq);
}
else{
//noconnection();
in the noconnection method u can parse the listview/recyclerview from json file to make the offline mode }
}else{
//parse the listview/recyclerview from local file. }