/[projects]/android/TrainInfo/src/dk/thoerup/traininfo/StationList.java
ViewVC logotype

Diff of /android/TrainInfo/src/dk/thoerup/traininfo/StationList.java

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 1244 by torben, Sat Mar 26 01:49:11 2011 UTC revision 1577 by torben, Sun Jul 10 07:45:38 2011 UTC
# Line 3  package dk.thoerup.traininfo; Line 3  package dk.thoerup.traininfo;
3  import static dk.thoerup.traininfo.R.string.app_name;  import static dk.thoerup.traininfo.R.string.app_name;
4  import static dk.thoerup.traininfo.R.string.generic_cancel;  import static dk.thoerup.traininfo.R.string.generic_cancel;
5  import static dk.thoerup.traininfo.R.string.generic_retry;  import static dk.thoerup.traininfo.R.string.generic_retry;
 import static dk.thoerup.traininfo.R.string.generic_search;  
6  import static dk.thoerup.traininfo.R.string.stationlist_accuracy;  import static dk.thoerup.traininfo.R.string.stationlist_accuracy;
7  import static dk.thoerup.traininfo.R.string.stationlist_addfavorite;  import static dk.thoerup.traininfo.R.string.stationlist_addfavorite;
8  import static dk.thoerup.traininfo.R.string.stationlist_favorites;  import static dk.thoerup.traininfo.R.string.stationlist_favorites;
 import static dk.thoerup.traininfo.R.string.stationlist_fetcherror;  
 import static dk.thoerup.traininfo.R.string.stationlist_findbyname;  
 import static dk.thoerup.traininfo.R.string.stationlist_findingnearby;  
9  import static dk.thoerup.traininfo.R.string.stationlist_gpsinfo;  import static dk.thoerup.traininfo.R.string.stationlist_gpsinfo;
10  import static dk.thoerup.traininfo.R.string.stationlist_gpstimeout;  import static dk.thoerup.traininfo.R.string.stationlist_gpstimeout;
11  import static dk.thoerup.traininfo.R.string.stationlist_latitude;  import static dk.thoerup.traininfo.R.string.stationlist_latitude;
 import static dk.thoerup.traininfo.R.string.stationlist_loadfavorites;  
12  import static dk.thoerup.traininfo.R.string.stationlist_locationinfo;  import static dk.thoerup.traininfo.R.string.stationlist_locationinfo;
13  import static dk.thoerup.traininfo.R.string.stationlist_longitude;  import static dk.thoerup.traininfo.R.string.stationlist_longitude;
14  import static dk.thoerup.traininfo.R.string.stationlist_nearbystations;  import static dk.thoerup.traininfo.R.string.stationlist_nearbystations;
15  import static dk.thoerup.traininfo.R.string.stationlist_nofavorites;  import static dk.thoerup.traininfo.R.string.stationlist_nofavorites;
 import static dk.thoerup.traininfo.R.string.stationlist_nolocation;  
16  import static dk.thoerup.traininfo.R.string.stationlist_nolocationprovider;  import static dk.thoerup.traininfo.R.string.stationlist_nolocationprovider;
 import static dk.thoerup.traininfo.R.string.stationlist_nostations;  
17  import static dk.thoerup.traininfo.R.string.stationlist_obtainedby;  import static dk.thoerup.traininfo.R.string.stationlist_obtainedby;
18  import static dk.thoerup.traininfo.R.string.stationlist_removefavorite;  import static dk.thoerup.traininfo.R.string.stationlist_removefavorite;
19  import static dk.thoerup.traininfo.R.string.stationlist_satellitecount;  import static dk.thoerup.traininfo.R.string.stationlist_satellitecount;
# Line 28  import static dk.thoerup.traininfo.R.str Line 21  import static dk.thoerup.traininfo.R.str
21  import static dk.thoerup.traininfo.R.string.stationlist_stationadded;  import static dk.thoerup.traininfo.R.string.stationlist_stationadded;
22  import static dk.thoerup.traininfo.R.string.stationlist_stationmap;  import static dk.thoerup.traininfo.R.string.stationlist_stationmap;
23  import static dk.thoerup.traininfo.R.string.stationlist_stationremoved;  import static dk.thoerup.traininfo.R.string.stationlist_stationremoved;
 import static dk.thoerup.traininfo.R.string.stationlist_stationsearch;  
 import static dk.thoerup.traininfo.R.string.stationlist_twocharmin;  
24  import static dk.thoerup.traininfo.R.string.stationlist_waitforlocation;  import static dk.thoerup.traininfo.R.string.stationlist_waitforlocation;
25    
26  import java.util.ArrayList;  import java.util.ArrayList;
# Line 44  import android.content.Intent; Line 35  import android.content.Intent;
35  import android.content.SharedPreferences;  import android.content.SharedPreferences;
36  import android.content.SharedPreferences.Editor;  import android.content.SharedPreferences.Editor;
37  import android.location.Location;  import android.location.Location;
 import android.os.AsyncTask;  
38  import android.os.Bundle;  import android.os.Bundle;
39  import android.os.Handler;  import android.os.Handler;
40  import android.os.Message;  import android.os.Message;
41    import android.text.Editable;
42    import android.text.TextWatcher;
43  import android.util.Log;  import android.util.Log;
44  import android.view.ContextMenu;  import android.view.ContextMenu;
45  import android.view.LayoutInflater;  import android.view.ContextMenu.ContextMenuInfo;
46  import android.view.Menu;  import android.view.Menu;
47  import android.view.MenuItem;  import android.view.MenuItem;
48  import android.view.View;  import android.view.View;
 import android.view.ContextMenu.ContextMenuInfo;  
49  import android.view.View.OnCreateContextMenuListener;  import android.view.View.OnCreateContextMenuListener;
50  import android.widget.AdapterView;  import android.widget.AdapterView;
 import android.widget.EditText;  
51  import android.widget.ListView;  import android.widget.ListView;
52    import android.widget.TextView;
53  import android.widget.Toast;  import android.widget.Toast;
54  import dk.thoerup.android.traininfo.common.StationBean;  import dk.thoerup.android.traininfo.common.StationBean;
55  import dk.thoerup.android.traininfo.common.StationBean.StationEntry;  import dk.thoerup.android.traininfo.common.StationEntry;
56    import dk.thoerup.traininfo.WelcomeScreen.ListType;
57  import dk.thoerup.traininfo.provider.ProviderFactory;  import dk.thoerup.traininfo.provider.ProviderFactory;
58  import dk.thoerup.traininfo.provider.StationProvider;  import dk.thoerup.traininfo.provider.StationProvider;
59  import dk.thoerup.traininfo.stationmap.GeoPair;  import dk.thoerup.traininfo.stationmap.GeoPair;
# Line 73  public class StationList extends ListAct Line 65  public class StationList extends ListAct
65    
66                    
67                    
68          public static final int OPTIONS_MAP = 2003;          public static final int OPTIONS_MAP = 103;
69          public static final int OPTIONS_GPSINFO = 2004;          public static final int OPTIONS_GPSINFO = 104;
70                    
71          public static final int DLG_PROGRESS = 3001;          public static final int DLG_PROGRESS = 3001;
         public static final int DLG_STATIONNAME = 3002;  
72                    
73                    
74          public static final int GPS_TIMEOUT_MS = 15000; //how long are we willing to wait for gps fix -in milliseconds          public static final int GPS_TIMEOUT_MS = 15000; //how long are we willing to wait for gps fix -in milliseconds
# Line 94  public class StationList extends ListAct Line 85  public class StationList extends ListAct
85          String dialogMessage = "";          String dialogMessage = "";
86          ProgressDialog dialog;          ProgressDialog dialog;
87          LocationLookup locationLookup = null;          LocationLookup locationLookup = null;
         FindStationsTask findStationsTask;  
88          StationsFetchedHandler stationsFetched = new StationsFetchedHandler();          StationsFetchedHandler stationsFetched = new StationsFetchedHandler();
89                    
90          //GeoPair location = new GeoPair();          //GeoPair location = new GeoPair();
# Line 122  public class StationList extends ListAct Line 112  public class StationList extends ListAct
112                  super.onCreate(savedInstanceState);                  super.onCreate(savedInstanceState);
113                  setContentView(R.layout.stationlist);                  setContentView(R.layout.stationlist);
114                                    
115                    listType = (WelcomeScreen.ListType) getIntent().getSerializableExtra("type");
116                                    
117                  adapter = new StationListAdapter(this);                  adapter = new StationListAdapter(this, (listType == ListType.ListNearest) );
118                  setListAdapter(adapter);                  setListAdapter(adapter);
119                                    
120                  ListView lv = getListView();                  ListView lv = getListView();
# Line 138  public class StationList extends ListAct Line 129  public class StationList extends ListAct
129                          favorites.fromString(favoriteString);                          favorites.fromString(favoriteString);
130                  }                  }
131                                    
                 listType = (WelcomeScreen.ListType) getIntent().getSerializableExtra("type");  
132                  setTitle();                  setTitle();
133                                    
134                  isLaunchedforShortcut = getIntent().getBooleanExtra("shortcut", false);                  isLaunchedforShortcut = getIntent().getBooleanExtra("shortcut", false);
135                                    
136                  ProviderFactory.purgeOldEntries(); //cleanup before fetching more data                  ProviderFactory.purgeOldEntries(); //cleanup before fetching more data
137                                    
138                    if (listType == ListType.ListSearch) {
139                            enableNamesearchField();
140                    }
141                    
142                  if (savedInstanceState == null) {                  if (savedInstanceState == null) {
143                    
                           
144                          switch (listType) {                          switch (listType) {
145                          case ListNearest:                          case ListNearest:
146                                  startNearestLookup();                                  startNearestLookup();
147                                  break;                                  break;
148                          case ListSearch:                                                          case ListSearch:                                
149                                  showDialog(DLG_STATIONNAME);                                  // do nothing here
150                                  break;                                  break;
151                          case ListFavorites:                          case ListFavorites:
152                                  startFavoriteLookup();                                  startFavoriteLookup();
# Line 163  public class StationList extends ListAct Line 156  public class StationList extends ListAct
156                          }                          }
157                                                    
158                  } else {                  } else {
159                            ((TextView)findViewById(R.id.stationsearch)).setText(  savedInstanceState.getString( "search") );
160                            
161                          stations = (StationBean) savedInstanceState.getSerializable("stations");                          stations = (StationBean) savedInstanceState.getSerializable("stations");
162                          adapter.setStations(stations);                          adapter.setStations(stations);
163                  }                  }
# Line 184  public class StationList extends ListAct Line 179  public class StationList extends ListAct
179                  if (locationLookup != null) {                  if (locationLookup != null) {
180                          locationLookup.stopSearch();                          locationLookup.stopSearch();
181                  }                  }
182                  if (findStationsTask != null) {          
                         findStationsTask.cancel(true);  
                 }                
183          }          }
184    
185                    
# Line 210  public class StationList extends ListAct Line 203  public class StationList extends ListAct
203                                    
204          }          }
205                    
206            
207            public void enableNamesearchField() {
208                    final TextView stationSearch = (TextView) findViewById(R.id.stationsearch);
209                    stationSearch.setVisibility( View.VISIBLE );
210                    stationSearch.addTextChangedListener( new TextWatcher() {
211    
212                            @Override
213                            public void afterTextChanged(Editable s) {
214                                    if (s.length() > 0) {
215                                            startNameLookup( s.toString() );
216                                    } else {
217                                            stations = new StationBean();
218                                            getListView().invalidateViews();
219                                            adapter.setStations( stations );        
220                                    }
221                            }
222    
223                            @Override
224                            public void beforeTextChanged(CharSequence s, int start, int count, int after) {
225                            }
226    
227                            @Override
228                            public void onTextChanged(CharSequence s, int start, int before, int count) {
229                            }
230                            
231                    });
232                    
233                    
234            }
235                                    
236    
237      @Override      @Override
# Line 218  public class StationList extends ListAct Line 240  public class StationList extends ListAct
240          if (dialog != null && dialog.isShowing())          if (dialog != null && dialog.isShowing())
241                  dialog.dismiss();                  dialog.dismiss();
242          outState.putSerializable("stations", (StationBean) stations);          outState.putSerializable("stations", (StationBean) stations);
243            outState.putString("search", ((TextView)findViewById(R.id.stationsearch)).getText().toString() );
244                    
245      }      }
246                    
# Line 231  public class StationList extends ListAct Line 254  public class StationList extends ListAct
254                  item.setIcon(android.R.drawable.ic_menu_mapmode);                  item.setIcon(android.R.drawable.ic_menu_mapmode);
255                                    
256                  item = menu.add(0, OPTIONS_GPSINFO, 0, getString(stationlist_gpsinfo));                  item = menu.add(0, OPTIONS_GPSINFO, 0, getString(stationlist_gpsinfo));
257                  item.setIcon(android.R.drawable.ic_menu_mapmode);                                item.setIcon(android.R.drawable.ic_menu_info_details);
258                    boolean hasLoc = (locationLookup.getLocation() != null);
259                    item.setEnabled(hasLoc);
260                                    
261                  return true;                  return true;
262          }          }
263            
264    
265    
266    
267          @Override          @Override
268          public boolean onOptionsItemSelected(MenuItem item) {          public boolean onOptionsItemSelected(MenuItem item) {
269                  boolean retval = true;                  boolean retval = true;
270    
271                  //TODO: Cleanup  
272                  switch (item.getItemId()) {                  switch (item.getItemId()) {
273                  case OPTIONS_MAP:                  case OPTIONS_MAP:
274                                                    
275                            if ( stations == null || stations.entries == null || stations.entries.size() == 0 ) {
276                                    Toast.makeText(this, "No stations to show on map", Toast.LENGTH_SHORT).show(); //TODO: Translate
277                                    return true;
278                            }
279                            
280                          Intent intent = new Intent(this,StationMapView.class);                          Intent intent = new Intent(this,StationMapView.class);
281                                                    
282                          ArrayList<GeoPair> stationPoints = new ArrayList<GeoPair>();                          ArrayList<GeoPair> stationPoints = new ArrayList<GeoPair>();
# Line 259  public class StationList extends ListAct Line 292  public class StationList extends ListAct
292                          Location loc = locationLookup.getLocation();                          Location loc = locationLookup.getLocation();
293                          StringBuffer message = new StringBuffer();                          StringBuffer message = new StringBuffer();
294                          message.append( getString(stationlist_locationinfo) ).append(":\n");                          message.append( getString(stationlist_locationinfo) ).append(":\n");
295                          if (loc != null) {                          
296                                  message.append( getString(stationlist_obtainedby) ).append( loc.getProvider() ).append("\n");                          message.append( getString(stationlist_obtainedby) ).append( loc.getProvider() ).append("\n");
297                                  message.append( getString(stationlist_accuracy) ).append( (int)loc.getAccuracy()).append("m\n");                          message.append( getString(stationlist_accuracy) ).append( (int)loc.getAccuracy()).append("m\n");
298                                  message.append( getString(stationlist_latitude) ).append( (float)loc.getLatitude()).append("\n");                          message.append( getString(stationlist_latitude) ).append( (float)loc.getLatitude()).append("\n");
299                                  message.append( getString(stationlist_longitude) ).append( (float)loc.getLongitude() ).append("\n");                          message.append( getString(stationlist_longitude) ).append( (float)loc.getLongitude() ).append("\n");
300                          } else {                                                  
                                 message.append( getString(stationlist_nolocation) );  
                         }                        
301                                                    
302                          MessageBox.showMessage(this, message.toString(), false);                          MessageBox.showMessage(this, message.toString(), false);
303                          break;                          break;
# Line 311  public class StationList extends ListAct Line 342  public class StationList extends ListAct
342                          dlg.setMessage( getString(stationlist_waitforlocation) );                          dlg.setMessage( getString(stationlist_waitforlocation) );
343                          dlg.setCancelable(false);                          dlg.setCancelable(false);
344                          return dlg;                                              return dlg;                    
                 case DLG_STATIONNAME:  
                         LayoutInflater factory = LayoutInflater.from(this);  
                         final View rootView = factory.inflate(R.layout.textinput, null);  
                           
                           
                         AlertDialog.Builder builder = new AlertDialog.Builder(this);  
                           
                         builder.setTitle( getString(stationlist_stationsearch) );  
                         builder.setView(rootView);  
                         builder.setCancelable(true);  
                         builder.setPositiveButton( getString(generic_search), new DialogInterface.OnClickListener() {  
                                 public void onClick(DialogInterface dialog, int which) {  
                                         EditText et = (EditText) rootView.findViewById(R.id.EditText);  
                                         dialog.dismiss();  
                                         String search = et.getText().toString().trim();  
                                         if (search.length() >= 2) {  
                                                 startNameLookup(search);  
                                         } else {  
                                                 showMessageAndClose( getString(stationlist_twocharmin) );  
                                         }  
                                 }  
                         });  
                         builder.setNegativeButton(getString(generic_cancel), new DialogInterface.OnClickListener() {  
                                 public void onClick(DialogInterface dialog, int which) {  
                                         dialog.dismiss();  
                                         StationList.this.finish(); // Close this Activity  
                                 }  
                         });                      
                         return builder.create();  
345                                                    
346                  default:                  default:
347                          return super.onCreateDialog(id);                                          return super.onCreateDialog(id);                
# Line 392  public class StationList extends ListAct Line 394  public class StationList extends ListAct
394          }          }
395                    
396          void startNameLookup(String name) {          void startNameLookup(String name) {
                 dialogMessage = getString( stationlist_findbyname );  
                 showDialog(DLG_PROGRESS);  
   
                 findStationsTask = new FindStationsTask();  
                 findStationsTask.searchByName(name);  
                 findStationsTask.execute();  
397                                    
398                    stations = stationProvider.lookupStationsByName(name);
399                    getListView().invalidateViews();
400                    adapter.setStations( stations );                                
401          }          }
402                    
403          public void startFavoriteLookup() {          public void startFavoriteLookup() {
404                                    
405                  if (favorites.size() > 0) {                  if (favorites.size() > 0) {
406                          dialogMessage = getString( stationlist_loadfavorites );                          
407                          showDialog(DLG_PROGRESS);                          stations = stationProvider.lookupStationsByIds( favorites.toString() );
408                            getListView().invalidateViews();
409                          findStationsTask = new FindStationsTask();                          adapter.setStations( stations );                
410                          findStationsTask.searchByIds( favorites.toString() );                          
411                          findStationsTask.execute();                          
412                  } else {                  } else {
413                          showMessageAndClose( getString( stationlist_nofavorites ) );                          showMessageAndClose( getString( stationlist_nofavorites ) );
414                  }                  }
# Line 417  public class StationList extends ListAct Line 416  public class StationList extends ListAct
416    
417    
418                    
419          void startLocatorTask()          void startNearestLookupPhase2()
420          {          {      
421                  dialogMessage = getString( stationlist_findingnearby );                  stations = stationProvider.lookupStationsByLocation( locationLookup.getLocation() );
422                  showDialog(DLG_PROGRESS);                                getListView().invalidateViews();
423                                    adapter.setStations( stations );
                 findStationsTask = new FindStationsTask();  
                 findStationsTask.searchByLocation( locationLookup.getLocation() );  
                 findStationsTask.execute();      
424          }          }
425                                    
426                    
# Line 442  public class StationList extends ListAct Line 438  public class StationList extends ListAct
438                          case GOTLOCATION:                          case GOTLOCATION:
439                                  dismissDialog(DLG_PROGRESS);                                  dismissDialog(DLG_PROGRESS);
440    
441                                  startLocatorTask();                                  startNearestLookupPhase2();
442    
443                                  return;                                  return;
444    
# Line 461  public class StationList extends ListAct Line 457  public class StationList extends ListAct
457    
458    
459                          if (locationLookup.elapsedTime() >=  GPS_TIMEOUT_MS) {                          if (locationLookup.elapsedTime() >=  GPS_TIMEOUT_MS) {
460                                  try {                                  dismissDialog(DLG_PROGRESS);
461                                          dismissDialog(DLG_PROGRESS);  
                                 } catch (IllegalArgumentException ex) {  
                                         // I get  stacktraces that reports an exception is thrown here -  
                                         // but i can not recreate the situation that causes said exception  
                                         Log.e("TrainInfo", "Why the f*** is this exception thrown here ?? " + ex.getMessage() );  
                                 }  
462    
463                                  locationLookup.stopSearch();                                  locationLookup.stopSearch();
464    
465                                  if (locationLookup.hasLocation()) {                                  if (locationLookup.hasLocation()) {
466                                          startLocatorTask();                                          startNearestLookupPhase2();
467                                  } else {                                                                                  } else {                                                
468                                          AlertDialog.Builder builder = new AlertDialog.Builder(StationList.this);                                                                                          AlertDialog.Builder builder = new AlertDialog.Builder(StationList.this);                                                
469                                          builder.setMessage(  getString( stationlist_gpstimeout) );                                          builder.setMessage(  getString( stationlist_gpstimeout) );
# Line 508  public class StationList extends ListAct Line 499  public class StationList extends ListAct
499    
500    
501                    
         class FindStationsTask extends AsyncTask<Void,Void,Void> {  
                   
                 LookupMethod method = LookupMethod.MethodNone;  
                 String name;  
                 Location loc;  
                 String ids;  
                   
                 public void searchByName(String n) {  
                           
                         method = LookupMethod.ByName;  
                         name = n;  
                 }  
                   
                 public void searchByLocation(Location l) {  
                         method = LookupMethod.ByLocation;  
                         loc = l;  
                 }  
                   
                 public void searchByIds(String id) {  
                           
                         method = LookupMethod.ByList;  
                         ids = id;  
                 }  
                   
                 @Override  
                 protected void onPreExecute() {  
   
                         if (method.equals(LookupMethod.MethodNone))  
                                 throw new RuntimeException("Method not set");  
                         super.onPreExecute();  
                 }  
                   
                 @Override  
                 protected Void doInBackground(Void... params) {  
   
                         switch (method) {  
                         case ByLocation:  
                                 stations = stationProvider.lookupStationsByLocation(loc);  
                                 break;  
                         case ByName:  
                                 stations = stationProvider.lookupStationsByName(name);  
                                 break;  
                         case ByList:  
                                 stations = stationProvider.lookupStationsByIds(ids);  
                                 break;  
                         default:  
                                 stations = null; // not possible          
                         }  
                           
                           
                         return null;  
                 }  
   
                 @Override  
                 protected void onPostExecute(Void result) {  
                         super.onPostExecute(result);  
                         dialog.dismiss();  
                           
                           
                         if (stations != null) {                          
                                 if (stations.entries.size() == 0) {  
                                         showMessageAndClose(getString(stationlist_nostations));  
                                 }  
   
                                 StationList.this.getListView().invalidateViews();  
                                 adapter.setStations( stations );                                  
                                   
                                   
                         } else { //communication or parse errors  
                                 AlertDialog.Builder builder = new AlertDialog.Builder(StationList.this);                                                  
                                 builder.setMessage(getString(stationlist_fetcherror));                            
                                 builder.setCancelable(true);  
                                 builder.setPositiveButton(getString(generic_retry), new DialogInterface.OnClickListener() {  
                                         public void onClick(DialogInterface dialog, int id) {  
                                                 dialog.dismiss();  
                                                   
                                                 Runnable runner = null;  
                                                 switch (method) {  
                                                 case ByLocation:  
                                                         runner = new Runnable() {  
                                                                 @Override  
                                                                 public void run() {  
                                                                         startLocatorTask();                                                              
                                                                 }  
                                                         };  
                                                         break;  
                                                 case ByName:  
                                                         runner = new Runnable() {  
                                                                 @Override  
                                                                 public void run() {  
                                                                         startNameLookup( FindStationsTask.this.name );  
                                                                 }  
                                                         };  
                                                         break;  
                                                 case ByList:  
                                                         runner = new Runnable() {  
                                                                 @Override  
                                                                 public void run() {  
                                                                         startFavoriteLookup();  
                                                                 }  
                                                         };  
                                                         break;  
                                                 }  
                                                   
                                                 stationsFetched.post( runner );  
                                         }  
                                 });  
                                 builder.setNegativeButton(getString(generic_cancel), new DialogInterface.OnClickListener() {  
                                         public void onClick(DialogInterface dialog, int id) {  
                                                 dialog.dismiss();  
                                                 StationList.this.finish();  
                                         }                                                        
                                 });  
                                   
                                 builder.show();  
                         }  
                 }  
         }  
           
           
502          class FavoritesMenu implements OnCreateContextMenuListener {          class FavoritesMenu implements OnCreateContextMenuListener {
503                  private final static int FAVORITES_ADD = 9001;                  private final static int FAVORITES_ADD = 9001;
504                  private final static int FAVORITES_REMOVE = 9002;                  private final static int FAVORITES_REMOVE = 9002;

Legend:
Removed from v.1244  
changed lines
  Added in v.1577

  ViewVC Help
Powered by ViewVC 1.1.20