Mega Code Archive

 
Categories / Android / Network
 

Weather web service

//src\com\commonsware\android\service\WeatherPlus.java /***   Copyright (c) 2008-2009 CommonsWare, LLC      Licensed under the Apache License, Version 2.0 (the "License"); you may   not use this file except in compliance with the License. You may obtain   a copy of the License at     http://www.apache.org/licenses/LICENSE-2.0   Unless required by applicable law or agreed to in writing, software   distributed under the License is distributed on an "AS IS" BASIS,   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.   See the License for the specific language governing permissions and   limitations under the License. */ package com.commonsware.android.service; import android.app.Activity; import android.app.AlertDialog; import android.content.BroadcastReceiver; import android.content.ComponentName; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.content.ServiceConnection; import android.os.Bundle; import android.os.DeadObjectException; import android.os.RemoteException; import android.os.IBinder; import android.view.Menu; import android.view.MenuItem; import android.webkit.WebView; import android.widget.Toast; public class WeatherPlus extends Activity {   private static final int CLOSE_ID = Menu.FIRST+1;   private WebView browser;   private WeatherPlusService appService=null;      @Override   public void onCreate(Bundle savedInstanceState) {     super.onCreate(savedInstanceState);     setContentView(R.layout.main);          browser=(WebView)findViewById(R.id.webkit);     bindService(new Intent(this, WeatherPlusService.class),                 onService, Context.BIND_AUTO_CREATE);   }      @Override   public void onResume() {     super.onResume();     registerReceiver(receiver,           new IntentFilter(WeatherPlusService.BROADCAST_ACTION));   }      @Override   public void onPause() {     super.onPause();     unregisterReceiver(receiver);   }      @Override   public void onDestroy() {     super.onDestroy();          unbindService(onService);   }      @Override   public boolean onCreateOptionsMenu(Menu menu) {     menu.add(Menu.NONE, CLOSE_ID, Menu.NONE, "Close")         .setIcon(R.drawable.eject)         .setAlphabeticShortcut('c');     return(super.onCreateOptionsMenu(menu));   }   @Override   public boolean onOptionsItemSelected(MenuItem item) {     switch (item.getItemId()) {       case CLOSE_ID:         finish();         return(true);     }     return(super.onOptionsItemSelected(item));   }      private void updateForecast() {     try {       String page=appService.getForecastPage();              browser.loadDataWithBaseURL(null, page, "text/html",                                     "UTF-8", null);     }     catch (final Throwable t) {       goBlooey(t);     }   }      private void goBlooey(Throwable t) {     AlertDialog.Builder builder=new AlertDialog.Builder(this);          builder       .setTitle("Exception!")       .setMessage(t.toString())       .setPositiveButton("OK", null)       .show();   }      private BroadcastReceiver receiver=new BroadcastReceiver() {     public void onReceive(Context context, Intent intent) {       updateForecast();     }   };      private ServiceConnection onService=new ServiceConnection() {     public void onServiceConnected(ComponentName className,                                    IBinder rawBinder) {       appService=((WeatherPlusService.LocalBinder)rawBinder).getService();     }     public void onServiceDisconnected(ComponentName className) {       appService=null;     }   }; } //src\com\commonsware\android\service\WeatherPlusService.java /***   Copyright (c) 2008-2009 CommonsWare, LLC      Licensed under the Apache License, Version 2.0 (the "License"); you may   not use this file except in compliance with the License. You may obtain   a copy of the License at     http://www.apache.org/licenses/LICENSE-2.0   Unless required by applicable law or agreed to in writing, software   distributed under the License is distributed on an "AS IS" BASIS,   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.   See the License for the specific language governing permissions and   limitations under the License. */ package com.commonsware.android.service; import android.app.Service; import android.content.Context; import android.content.Intent; import android.content.SharedPreferences; import android.location.Criteria; import android.location.Location; import android.location.LocationListener; import android.location.LocationManager; import android.os.AsyncTask; import android.os.Binder; import android.os.Bundle; import android.os.IBinder; import java.io.StringReader; import java.util.concurrent.CopyOnWriteArraySet; import java.util.concurrent.ConcurrentLinkedQueue; import java.util.ArrayList; import java.util.HashMap; import java.util.HashSet; import java.util.List; import java.util.Map; import java.util.Set; import javax.xml.parsers.DocumentBuilder; import javax.xml.parsers.DocumentBuilderFactory; import org.apache.http.client.ResponseHandler; import org.apache.http.client.HttpClient; import org.apache.http.client.methods.HttpGet; import org.apache.http.impl.client.BasicResponseHandler; import org.apache.http.impl.client.DefaultHttpClient; import org.w3c.dom.Document; import org.w3c.dom.Element; import org.w3c.dom.NodeList; import org.xml.sax.InputSource; public class WeatherPlusService extends Service {   public static final int NOTIF_ID=1337;   public static final String SHUTDOWN="SHUTDOWN";   public static final String BROADCAST_ACTION=             "com.commonsware.android.service.ForecastUpdateEvent";   private LocationManager mgr=null;   private String forecast=null;   private HttpClient client=null;   private String format=null;   private Intent broadcast=new Intent(BROADCAST_ACTION);   private final Binder binder=new LocalBinder();      @Override   public void onCreate() {     super.onCreate();          client=new DefaultHttpClient();     format=getString(R.string.url);          mgr=(LocationManager)getSystemService(Context.LOCATION_SERVICE);     mgr.requestLocationUpdates(LocationManager.GPS_PROVIDER,                                 10000, 10000.0f, onLocationChange);   }      @Override   public IBinder onBind(Intent intent) {     return(binder);   }      @Override   public void onDestroy() {     super.onDestroy();          mgr.removeUpdates(onLocationChange);   }      synchronized public String getForecastPage() {     return(forecast);   }      private void updateForecast(Location loc) {     new FetchForecastTask().execute(loc);   }      List<Forecast> buildForecasts(String raw) throws Exception {     List<Forecast> forecasts=new ArrayList<Forecast>();     DocumentBuilder builder=DocumentBuilderFactory                               .newInstance()                               .newDocumentBuilder();     Document doc=builder.parse(new InputSource(new StringReader(raw)));     NodeList times=doc.getElementsByTagName("start-valid-time");          for (int i=0;i<times.getLength();i++) {       Element time=(Element)times.item(i);       Forecast forecast=new Forecast();              forecasts.add(forecast);       forecast.setTime(time.getFirstChild().getNodeValue());     }          NodeList temps=doc.getElementsByTagName("value");          for (int i=0;i<temps.getLength();i++) {       Element temp=(Element)temps.item(i);       Forecast forecast=forecasts.get(i);              forecast.setTemp(new Integer(temp.getFirstChild().getNodeValue()));     }          NodeList icons=doc.getElementsByTagName("icon-link");          for (int i=0;i<icons.getLength();i++) {       Element icon=(Element)icons.item(i);       Forecast forecast=forecasts.get(i);              forecast.setIcon(icon.getFirstChild().getNodeValue());     }          return(forecasts);   }      String generatePage(List<Forecast> forecasts) {     StringBuilder bufResult=new StringBuilder("<html><body><table>");          bufResult.append("<tr><th width=\"50%\">Time</th>"+                       "<th>Temperature</th><th>Forecast</th></tr>");          for (Forecast forecast : forecasts) {       bufResult.append("<tr><td align=\"center\">");       bufResult.append(forecast.getTime());       bufResult.append("</td><td align=\"center\">");       bufResult.append(forecast.getTemp());       bufResult.append("</td><td><img src=\"");       bufResult.append(forecast.getIcon());       bufResult.append("\"></td></tr>");     }          bufResult.append("</table></body></html>");          return(bufResult.toString());   }      LocationListener onLocationChange=new LocationListener() {     public void onLocationChanged(Location location) {       updateForecast(location);     }          public void onProviderDisabled(String provider) {       // required for interface, not used     }          public void onProviderEnabled(String provider) {       // required for interface, not used     }          public void onStatusChanged(String provider, int status,                                   Bundle extras) {       // required for interface, not used     }   };      public class LocalBinder extends Binder {     WeatherPlusService getService() {       return(WeatherPlusService.this);     }   }      class Forecast {     String time="";     Integer temp=null;     String iconUrl="";          String getTime() {       return(time);     }        void setTime(String time) {       this.time=time.substring(0,16).replace('T', ' ');     }          Integer getTemp() {       return(temp);     }          void setTemp(Integer temp) {       this.temp=temp;     }          String getIcon() {       return(iconUrl);     }          void setIcon(String iconUrl) {       this.iconUrl=iconUrl;     }   }      class FetchForecastTask extends AsyncTask<Location, Void, Void> {     @Override     protected Void doInBackground(Location... locs) {       Location loc=locs[0];       String url=String.format(format, loc.getLatitude(),                                 loc.getLongitude());       HttpGet getMethod=new HttpGet(url);              try {         ResponseHandler<String> responseHandler=new BasicResponseHandler();         String responseBody=client.execute(getMethod, responseHandler);         String page=generatePage(buildForecasts(responseBody));                  synchronized(this) {           forecast=page;         }                sendBroadcast(broadcast);       }       catch (Throwable t) {         android.util.Log.e("WeatherPlus",                             "Exception in updateForecast()", t);       }              return(null);     }          @Override     protected void onProgressUpdate(Void... unused) {       // not needed here     }          @Override     protected void onPostExecute(Void unused) {       // not needed here     }   } } //res\layout\main.xml <?xml version="1.0" encoding="utf-8"?> <WebView xmlns:android="http://schemas.android.com/apk/res/android"   android:id="@+id/webkit"   android:layout_width="fill_parent"   android:layout_height="fill_parent"   />