/[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

android/TrainInfo/src/dk/thoerup/traininfo/TrainInfoList.java revision 252 by torben, Mon Aug 10 10:49:43 2009 UTC android/TrainInfo/src/dk/thoerup/traininfo/StationList.java revision 554 by torben, Wed Jan 27 05:51:45 2010 UTC
# Line 1  Line 1 
1  package dk.thoerup.traininfo;  package dk.thoerup.traininfo;
2    
3    import java.util.ArrayList;
4    import java.util.List;
5    
6    
7    import android.app.AlertDialog;
8  import android.app.Dialog;  import android.app.Dialog;
9  import android.app.ListActivity;  import android.app.ListActivity;
10  import android.app.ProgressDialog;  import android.app.ProgressDialog;
11    import android.content.DialogInterface;
12  import android.content.Intent;  import android.content.Intent;
13    import android.content.SharedPreferences;
14    import android.content.SharedPreferences.Editor;
15    import android.location.Location;
16  import android.os.AsyncTask;  import android.os.AsyncTask;
17  import android.os.Bundle;  import android.os.Bundle;
18  import android.os.Handler;  import android.os.Handler;
19  import android.os.Message;  import android.os.Message;
20    
21    import android.view.ContextMenu;
22    import android.view.LayoutInflater;
23    import android.view.Menu;
24    import android.view.MenuItem;
25  import android.view.View;  import android.view.View;
26    import android.view.ContextMenu.ContextMenuInfo;
27    import android.view.View.OnCreateContextMenuListener;
28    import android.widget.AdapterView;
29    import android.widget.EditText;
30  import android.widget.ListView;  import android.widget.ListView;
31    import android.widget.Toast;
32    import dk.thoerup.traininfo.provider.ProviderFactory;
33    import dk.thoerup.traininfo.provider.StationProvider;
34    import dk.thoerup.traininfo.stationmap.GeoPair;
35    import dk.thoerup.traininfo.stationmap.StationMapView;
36    import dk.thoerup.traininfo.util.IntSet;
37  import dk.thoerup.traininfo.util.MessageBox;  import dk.thoerup.traininfo.util.MessageBox;
38    
39  public class TrainInfoList extends ListActivity  {  public class StationList extends ListActivity  {
40          public static final int GOTLOCATION = 1;          public static final int GOTLOCATION = 1001;
41          public static final int GOTSTATIONLIST = 2;          public static final int GOTSTATIONLIST = 1002;
42          public static final int NOPROVIDER = 3;          public static final int NOPROVIDER = 1003;
43          public static final int FIXTIMEOUT = 4;          public static final int LOCATIONFIXTIMEOUT = 1004;
44          public static final int LOOKUPSTATIONFAILED = 5;          
45            public static final int OPTIONS_MAP = 2003;
46            public static final int OPTIONS_GPSINFO = 2004;
47    
48            
49    
50            
51            public static final int DLG_PROGRESS = 3001;
52            public static final int DLG_STATIONNAME = 3002;
53            
54            static enum LookupMethod {
55                    ByLocation,
56                    ByName,
57                    ByList,
58                    MethodNone
59            }
60                    
         public static final int DLG_PROGRESS = 1;  
61                    
62          /** Called when the activity is first created. */          String dialogMessage = "";
63          ProgressDialog dialog;          ProgressDialog dialog;
64          StationLocator locator = null;          LocationLookup locationLookup = null;
65          LocatorTask locatorTask = new LocatorTask();          FindStationsTask findStationsTask;
66            StationsFetchedHandler stationsFetched = new StationsFetchedHandler();
67                    
68          boolean isRunning;          GeoPair location = new GeoPair();
69            
70            boolean isRunning = false;
71            List<StationBean> stations = new ArrayList<StationBean>();
72            
73            StationProvider stationProvider = ProviderFactory.getStationProvider();
74                    
75          StationListAdapter adapter = null;          StationListAdapter adapter = null;
76            
77            FavoritesMenu contextMenu = new FavoritesMenu();
78            IntSet favorites = new IntSet();
79    
80            WelcomeScreen.ListType listType;
81            SharedPreferences prefs;
82            
83            ///////////////////////////////////////////////////////////////////////////////////////////
84            //Activity call backs
85            
86            @SuppressWarnings("unchecked")
87          @Override          @Override
88          public void onCreate(Bundle savedInstanceState) {          public void onCreate(Bundle savedInstanceState) {
89                  super.onCreate(savedInstanceState);                  super.onCreate(savedInstanceState);
90                  setContentView(R.layout.main);                  setContentView(R.layout.stationlist);
91                                    
                 StationLocator.removeMockLocation(this);  
                 //StationLocator.injectMockLocation(this);  
92                                    
93                  adapter = new StationListAdapter(this);                  adapter = new StationListAdapter(this);
94                  setListAdapter(adapter);                  setListAdapter(adapter);
95                                    
96                  locator = new StationLocator(this, stationsFetched);                  ListView lv = getListView();
97                    lv.setOnCreateContextMenuListener(contextMenu);
98                    
99                    locationLookup = new LocationLookup(this, stationsFetched);
100                    
101    
102                    prefs = getSharedPreferences("TrainStation", 0);
103                    String favoriteString = prefs.getString("favorites", "");
104                    if (! favoriteString.equals("") ) {
105                            favorites.fromString(favoriteString);
106                    }
107                    
108                    listType = (WelcomeScreen.ListType) getIntent().getSerializableExtra("type");
109                    setTitle();
110                    
111                    if (savedInstanceState == null) {
112    
113                            
114                            switch (listType) {
115                            case ListNearest:
116                                    startLookup();
117                                    break;
118                            case ListSearch:
119                                    this.showDialog(DLG_STATIONNAME);
120                                    break;
121                            case ListFavorites:
122                                    startFavoriteLookup();
123                                    break;
124                            default:
125                                    // Not possible !?!
126                            }
127                            
128                    } else {
129                            stations = (ArrayList<StationBean>) savedInstanceState.getSerializable("stations");
130                            adapter.setStations(stations);
131                            location = (GeoPair) savedInstanceState.getSerializable("location");
132                    }
133                                    
                 startLookup();  
134          }          }
135            protected void setTitle() {
136                    String dialogTitle = getResources().getString(R.string.app_name);
137                    switch (listType) {
138                    case ListNearest:
139                            dialogTitle += " - Nearby stations";
140                            break;
141                    case ListSearch:
142                            dialogTitle += " - Search";
143                            break;
144                    case ListFavorites:
145                            dialogTitle += " - Favorites";
146                            break;
147                    default:
148                            dialogTitle = "";//not possible                                
149                    }
150            
151                    setTitle(dialogTitle);
152            }
153            
154                    
155    
156    
157      @Override      @Override
158      public void onSaveInstanceState(Bundle outState)      public void onSaveInstanceState(Bundle outState)
159      {      {
160          if (dialog.isShowing())          if (dialog != null && dialog.isShowing())
161                  dialog.dismiss();                  dialog.dismiss();
162            outState.putSerializable("stations", (ArrayList<StationBean>) stations);
163            outState.putSerializable("location", location);
164            
165      }      }
166                    
167                    
168    
169          @Override          @Override
170            public boolean onCreateOptionsMenu(Menu menu) {
171                    MenuItem item;
172                                    
173                    item = menu.add(0, OPTIONS_MAP, 0, "Station map");
174                    item.setIcon(android.R.drawable.ic_menu_mapmode);
175                    
176                    item = menu.add(0, OPTIONS_GPSINFO, 0, "GPS Info");
177                    item.setIcon(android.R.drawable.ic_menu_mapmode);              
178                    
179                    return true;
180            }
181    
182            @Override
183            public boolean onOptionsItemSelected(MenuItem item) {
184                    boolean retval = true;
185    
186                    //TODO: Cleanup
187                    switch (item.getItemId()) {
188                    case OPTIONS_MAP:
189                            
190                            Intent intent = new Intent(this,StationMapView.class);
191                            
192                            ArrayList<GeoPair> stationPoints = new ArrayList<GeoPair>();
193                            for (StationBean st : stations ) {
194                                    stationPoints.add( new GeoPair(st.getLatitude(), st.getLongitude(), st.getName()) );
195                            }
196                            
197                            intent.putExtra("stations", stationPoints);
198                            
199                            startActivity(intent);
200                            break;
201                    case OPTIONS_GPSINFO:
202                            Location loc = locationLookup.getLocation();
203                            StringBuffer message = new StringBuffer();
204                            message.append("Location info:\n");
205                            if (loc != null) {
206                                    message.append("-Obtained by: ").append( loc.getProvider() ).append("\n");
207                                    message.append("-Accuracy: ").append( (int)loc.getAccuracy()).append("m\n");
208                                    message.append("-Latitude: ").append( loc.getLatitude()).append("\n");
209                                    message.append("-Longitude: ").append( loc.getLongitude() ).append("\n");
210                            } else {
211                                    message.append(" - No location data!");
212                            }
213                                    
214                            MessageBox.showMessage(this, message.toString());
215                            break;
216                    default:
217                            retval = super.onOptionsItemSelected(item);
218                    }
219                    
220                    return retval;
221            }
222            
223            
224    
225            @Override
226            public boolean onContextItemSelected(MenuItem item) {
227                    contextMenu.onContextItemSelected(item);
228                    return true;
229    
230    
231            }
232    
233    
234    
235    
236            @Override
237          protected Dialog onCreateDialog(int id) {          protected Dialog onCreateDialog(int id) {
238                  switch (id) {                  switch (id) {
239                  case DLG_PROGRESS:                  case DLG_PROGRESS:
240                          ProgressDialog dlg = new ProgressDialog(this);                          ProgressDialog dlg = new ProgressDialog(this);
241                          dlg.setMessage("Wait for location fix");                          dlg.setMessage("Wait for location fix");
242                          dlg.setCancelable(false);                          dlg.setCancelable(false);
243                          return dlg;                          return dlg;                    
244                    case DLG_STATIONNAME:
245                            LayoutInflater factory = LayoutInflater.from(this);
246                            final View rootView = factory.inflate(R.layout.textinput, null);
247                            
248                            
249                            AlertDialog.Builder builder = new AlertDialog.Builder(this);
250                            
251                            builder.setTitle("Station search");
252                            builder.setView(rootView);
253                            builder.setCancelable(true);
254                            builder.setPositiveButton("Search", new DialogInterface.OnClickListener() {
255                                    public void onClick(DialogInterface dialog, int which) {
256                                            EditText et = (EditText) rootView.findViewById(R.id.EditText);
257                                            dialog.dismiss();
258                                            if (et.getText().toString().length() >= 2) {
259                                                    startNameSearch(et.getText().toString());
260                                            } else {
261                                                    MessageBox.showMessage(StationList.this, "Two characters minimum" );
262                                            }
263                                    }
264                            });
265                            builder.setNegativeButton("Cancel", new DialogInterface.OnClickListener() {
266                                    public void onClick(DialogInterface dialog, int which) {
267                                            dialog.dismiss();
268                                    }
269                            });                    
270                            return builder.create();
271                            
272                  default:                  default:
273                          return super.onCreateDialog(id);                                          return super.onCreateDialog(id);                
274                  }                  }
# Line 69  public class TrainInfoList extends ListA Line 276  public class TrainInfoList extends ListA
276          }          }
277                    
278                    
   
279          @Override          @Override
280          protected void onPrepareDialog(int id, Dialog dialog) {          protected void onPrepareDialog(int id, Dialog dialog) {
281                  super.onPrepareDialog(id, dialog);                  super.onPrepareDialog(id, dialog);
282                  switch (id) {                  switch (id) {
283                  case DLG_PROGRESS:                  case DLG_PROGRESS:
284                          this.dialog = (ProgressDialog) dialog;                          this.dialog = (ProgressDialog) dialog;
285                            if (!dialogMessage.equals("")) {
286                                    this.dialog.setMessage(dialogMessage);
287                                    dialogMessage = "";
288                            }
289                          break;                          break;
290                  }                  }
291          }          }
292            
293            @Override
294            protected void onListItemClick(ListView l, View v, int position, long id) {
295                    super.onListItemClick(l, v, position, id);
296                                    
297                    StationBean station = stations.get(position);
298    
299                    double latitude = station.getLatitude();
300                    double longitude = station.getLongitude();
301    
302    
303                    
304                    Intent intent = new Intent(this, DepartureList.class);
305                    intent.putExtra("name", station.getName());
306                    intent.putExtra("distance", station.getDistance());
307                    intent.putExtra("latitude", latitude);
308                    intent.putExtra("longitude", longitude);
309                    intent.putExtra("stationid", station.getId());
310                    intent.putExtra("address", station.getAddress());
311                    intent.putExtra("isregional", station.isRegional());
312                    intent.putExtra("isstrain", station.isSTrain());
313                    intent.putExtra("ismetro", station.isMetro());
314                    startActivity(intent);
315            }
316    
317            /////////////////////////////////////////////////////////////
318            //
319    
320          public void startLookup() {          public void startLookup() {
321                  isRunning = true;                  isRunning = true;
322                    dialogMessage = "Wait for location fix";
323                  showDialog(DLG_PROGRESS);                  showDialog(DLG_PROGRESS);
324                                    
325                  locator.locateStations();                  locationLookup.locateStations();
326                  stationsFetched.sendEmptyMessageDelayed(FIXTIMEOUT, 20000);                              stationsFetched.sendEmptyMessageDelayed(LOCATIONFIXTIMEOUT, 20000);
327          }          }
328            
329            void startNameSearch(String name) {
330                    dialogMessage = "Finding stations by name";
331                    showDialog(DLG_PROGRESS);
332    
333                    findStationsTask = new FindStationsTask();
334                    findStationsTask.searchByName(name, locationLookup.getLocation());
335                    findStationsTask.execute();
336                    
337            }
338            
339            public void startFavoriteLookup() {
340                    
341                    if (favorites.size() > 0) {
342                            dialogMessage = "Loading favorites";
343                            showDialog(DLG_PROGRESS);
344    
345          Handler stationsFetched = new Handler() {                          findStationsTask = new FindStationsTask();
346                            findStationsTask.searchByIds(favorites.toString(), locationLookup.getLocation());
347                            findStationsTask.execute();
348                    } else {
349                            MessageBox.showMessage(this, "Favorite list is empty");
350                    }
351            }
352    
353    
354            
355            void startLocatorTask()
356            {
357                    dialogMessage = "Finding nearby stations";
358                    showDialog(DLG_PROGRESS);
359                    
360                    findStationsTask = new FindStationsTask();
361                    findStationsTask.searchByLocation( locationLookup.getLocation() );
362                    findStationsTask.execute();    
363            }
364            
365    
366            /* TODO: Remove this no longer needed function
367            String lookupAddress(double latitude, double longitude) {
368                    
369                    Geocoder coder = new Geocoder(this, new Locale("da"));
370                    StringBuilder sb = new StringBuilder();
371                    Log.i("lookupaddr", "" + latitude + "/" + longitude);
372                    try {
373                            List<Address> addressList = coder.getFromLocation(latitude, longitude, 1);
374                            Address addr = addressList.get(0);
375                            
376                            
377                            int max = addr.getMaxAddressLineIndex();
378                            for (int i=0; i<max; i++) {
379                                    if (i>0)
380                                            sb.append(", ");
381                                    
382                                    sb.append(addr.getAddressLine(i));
383                            }
384                            
385                            
386                    } catch (Exception e) {
387                            Log.e("DepartureList", "geocoder failed", e);
388                    }
389                    
390                    return sb.toString();
391            }*/
392            
393            
394            ////////////////////////////////////////////////////////////////////////////
395            // Inner classes
396    
397            class StationsFetchedHandler extends Handler {
398                  @Override                  @Override
399                  public void handleMessage(Message msg) {                  public void handleMessage(Message msg) {
400                            
401                          switch (msg.what) {                          switch (msg.what) {
402                          case GOTLOCATION:                          case GOTLOCATION:
403                                  dialog.setMessage("Finding nearby stations");                                  dismissDialog(DLG_PROGRESS);
404                                  locatorTask.execute();                                  
405                                  break;                                  startLocatorTask();
406                          case GOTSTATIONLIST:                                  location = GeoPair.fromLocation( locationLookup.getLocation() );
407                                  dialog.dismiss();                                  
                                 if (locator.getStations().size() == 0)  
                                         MessageBox.showMessage(TrainInfoList.this,"Error loading station list!");  
                                 adapter.setStations( locator.getStations() );  
408                                  break;                                  break;
409    
410                          case NOPROVIDER:                          case NOPROVIDER:
411                                  dialog.dismiss();                                  dismissDialog(DLG_PROGRESS);
412                                  MessageBox.showMessage(TrainInfoList.this,"No location provider enabled. Plase enable gps.");                                  MessageBox.showMessage(StationList.this,"No location provider enabled. Plase enable gps.");
413                                  break;                                  break;
414                          case FIXTIMEOUT:                          case LOCATIONFIXTIMEOUT:                                
                                 dialog.dismiss();  
415                                  if (isRunning) {                                  if (isRunning) {
416                                          locator.abortLocationListener();                                          locationLookup.stopSearch();
417                                          MessageBox.showMessage(TrainInfoList.this,"GPS fix timed out");                                          if (locationLookup.hasLocation()) {
418                                                    stationsFetched.sendEmptyMessage( GOTLOCATION );
419                                            } else {                                                
420                                                    dismissDialog(DLG_PROGRESS);
421                                                    
422                                                    AlertDialog.Builder builder = new AlertDialog.Builder(StationList.this);                                                
423                                                    builder.setMessage("Location fix timed out");
424                                                    builder.setCancelable(true);
425                                                    builder.setPositiveButton("Retry", new DialogInterface.OnClickListener() {
426                                                            public void onClick(DialogInterface dialog, int id) {
427                                                                    dialog.dismiss();
428                                                                    startLookup();
429                                                                    
430                                                            }
431                                                    });
432                                                    builder.setNegativeButton("Cancel", new DialogInterface.OnClickListener() {
433                                                            public void onClick(DialogInterface dialog, int id) {
434                                                                    dialog.dismiss();
435                                                            }                                                      
436                                                    });                                                                                            
437                                                    builder.show();
438    
439                                            }
440                                  }                                  }
441                                  break;                                  break;
                         case LOOKUPSTATIONFAILED:  
                                 dialog.dismiss();  
                                 MessageBox.showMessage(TrainInfoList.this,"Error on finding nearby stations");  
                                 break;  
442                          }                          }
                           
443                          isRunning = false;                          isRunning = false;
444                  }                  }
445          };          };
446    
447                    
448                    class FindStationsTask extends AsyncTask<Void,Void,Void> {
449                            
450          @Override                  LookupMethod method = LookupMethod.MethodNone;
451          protected void onListItemClick(ListView l, View v, int position, long id) {                  boolean success;
452                  super.onListItemClick(l, v, position, id);                  String name;
453                    Location loc;
454                    String ids;
455                                    
456                  StationBean station = adapter.getStation(position);                  public void searchByName(String n, Location l) {
457                            
458                            method = LookupMethod.ByName;
459                            loc = l;
460                            name = n;
461                    }
462                                    
463                    public void searchByLocation(Location l) {
464                            method = LookupMethod.ByLocation;
465                            loc = l;
466                    }
467                    
468                    public void searchByIds(String id, Location l) {
469                            
470                            method = LookupMethod.ByList;
471                            loc = l;
472                            ids = id;
473                    }
474                                    
                 Intent intent = new Intent(this, DepartureList.class);  
                 intent.putExtra("name", station.getName());  
                 intent.putExtra("address", station.getAddress());  
                 intent.putExtra("distance", station.getDistance());  
                 intent.putExtra("latitude", station.getLatitude());  
                 intent.putExtra("longitude", station.getLongitude());  
                 startActivity(intent);  
         }  
   
           
         class LocatorTask extends AsyncTask<Void,Void,Void> {  
475                  @Override                  @Override
476                  protected void onPreExecute() {                  protected void onPreExecute() {
477    
478                            if (method.equals(LookupMethod.MethodNone))
479                                    throw new RuntimeException("Method not set");
480                          super.onPreExecute();                          super.onPreExecute();
481                  }                  }
482                                    
483                  @Override                  @Override
484                  protected Void doInBackground(Void... params) {                  protected Void doInBackground(Void... params) {
485                          locator.findNearestStations();  
486                            switch (method) {
487                            case ByLocation:
488                                    success = stationProvider.lookupStations(loc);
489                                    break;
490                            case ByName:
491                                    success = stationProvider.lookupStationsByName(name);
492                                    break;
493                            case ByList:
494                                    success = stationProvider.lookupStationsByIds(ids);
495                                    break;
496                            default:
497                                    success = false; // not possible        
498                            }
499                            
500                            
501                            Location dummy = new Location("gps");
502                            List<StationBean> stations = stationProvider.getStations();
503                            
504                            for (StationBean station : stations) {
505                                                                    
506                                    if (method.equals(LookupMethod.ByName) || method.equals(LookupMethod.ByList)) {
507                                            if (loc != null) { //only do the distance calc if we have a location
508                                                    dummy.setLatitude(station.getLatitude());
509                                                    dummy.setLongitude(station.getLongitude());
510                                                    station.setDistance( (int)loc.distanceTo(dummy) );
511                                            } else {
512                                                    station.setDistance(0);
513                                            }
514                                    }
515    
516                            }                                              
517                            
518                          return null;                          return null;
519                  }                  }
520    
521                  @Override                  @Override
522                  protected void onPostExecute(Void result) {                  protected void onPostExecute(Void result) {
523                          super.onPostExecute(result);                          super.onPostExecute(result);
524                            dialog.dismiss();
525                            
526                            
527                            if (success) {                          
528                                    if (stationProvider.getStations().size() == 0)
529                                            MessageBox.showMessage(StationList.this, "No stations found!"); // this should not be possible !?!
530                                    stations = stationProvider.getStations();
531                                    adapter.setStations( stations );                                
532                                    
533                            } else { //communication or parse errors
534                                    AlertDialog.Builder builder = new AlertDialog.Builder(StationList.this);                                                
535                                    builder.setMessage("Error on finding nearby stations");
536                                    builder.setCancelable(true);
537                                    builder.setPositiveButton("Retry", new DialogInterface.OnClickListener() {
538                                            public void onClick(DialogInterface dialog, int id) {
539                                                    dialog.dismiss();
540                                                    
541                                                    stationsFetched.post( new Runnable() {
542                                                            @Override
543                                                            public void run() {
544                                                                    startLocatorTask();                                                            
545                                                            }
546                                                    });
547                                            }
548                                    });
549                                    builder.setNegativeButton("Cancel", new DialogInterface.OnClickListener() {
550                                            public void onClick(DialogInterface dialog, int id) {
551                                                    dialog.dismiss();
552                                            }                                                      
553                                    });                                                                                            
554                                    builder.show();                        
555                            }
556                    }
557            }
558            
559            
560            class FavoritesMenu implements OnCreateContextMenuListener {
561                    private final static int FAVORITES_ADD = 9001;
562                    private final static int FAVORITES_REMOVE = 9002;
563                    
564                    private int selectedPosition;
565                    
566                    
567                    @Override
568                    public void onCreateContextMenu(ContextMenu menu, View v, ContextMenuInfo menuInfo) {
569                                                    
570                            AdapterView.AdapterContextMenuInfo info = (AdapterView.AdapterContextMenuInfo) menuInfo;
571                            selectedPosition = info.position;
572                            int stationID = stations.get(selectedPosition).getId();
573    
574                            if (!favorites.contains(stationID)) {
575                                    menu.add(0, FAVORITES_ADD, 0, "Add to favorites");
576                            } else {
577                                    menu.add(0, FAVORITES_REMOVE, 0, "Remove from favorites");
578                            }
579                            
580                    }
581                    
582                    public void onContextItemSelected(MenuItem item) {
583                            StationBean sb = stations.get(selectedPosition);
584                            
585                            int stationID = sb.getId();
586                            if (item.getItemId() == FAVORITES_ADD) {
587                                    favorites.add(stationID);
588                                    Toast.makeText(StationList.this, "Station added", Toast.LENGTH_SHORT).show();
589                            } else {
590                                    
591                                    favorites.remove(stationID);
592                                    Toast.makeText(StationList.this, "Station removed", Toast.LENGTH_SHORT).show();
593                                    
594                                    
595                                    if (listType.equals( WelcomeScreen.ListType.ListFavorites) ) {
596                                            stations.remove(selectedPosition);
597                                            adapter.notifyDataSetChanged();
598                                    }
599                            }
600                            Editor ed = prefs.edit();
601                            ed.putString("favorites", favorites.toString());
602                            ed.commit();
603                  }                  }
604          }          }
605  }  }

Legend:
Removed from v.252  
changed lines
  Added in v.554

  ViewVC Help
Powered by ViewVC 1.1.20