diff --git a/app/src/main/java/it/reyboz/bustorino/fragments/NearbyArrivalsDownloader.java b/app/src/main/java/it/reyboz/bustorino/fragments/NearbyArrivalsDownloader.java new file mode 100644 index 0000000..0f2c81e --- /dev/null +++ b/app/src/main/java/it/reyboz/bustorino/fragments/NearbyArrivalsDownloader.java @@ -0,0 +1,122 @@ +package it.reyboz.bustorino.fragments; + +import android.content.Context; +import android.util.Log; +import com.android.volley.NetworkError; +import com.android.volley.ParseError; +import com.android.volley.Response; +import com.android.volley.VolleyError; +import it.reyboz.bustorino.backend.NetworkVolleyManager; +import it.reyboz.bustorino.backend.Palina; +import it.reyboz.bustorino.backend.Route; +import it.reyboz.bustorino.backend.Stop; +import it.reyboz.bustorino.backend.mato.MapiArrivalRequest; + +import java.util.ArrayList; +import java.util.Date; +import java.util.HashMap; +import java.util.List; + +class NearbyArrivalsDownloader implements Response.Listener<Palina>, Response.ErrorListener { + final static String DEBUG_TAG = "BusTO-NearbyArrivDowns"; + //final Map<String,List<Route>> routesToAdd = new HashMap<>(); + final ArrayList<Palina> nonEmptyPalinas = new ArrayList<>(); + final HashMap<String, Boolean> completedRequests = new HashMap<>(); + final static String REQUEST_TAG = "NearbyArrivals"; + final NetworkVolleyManager volleyManager; + int activeRequestCount = 0, reqErrorCount = 0, reqSuccessCount = 0; + + final ArrivalsListener listener; + + NearbyArrivalsDownloader(Context context, ArrivalsListener arrivalsListener) { + volleyManager = NetworkVolleyManager.getInstance(context); + + listener = arrivalsListener; + //flatProgressBar.setMax(numreq); + } + + public int requestArrivalsForStops(List<Stop> stops){ + int MAX_ARRIVAL_STOPS = 35; + Date currentDate = new Date(); + int timeRange = 3600; + int departures = 10; + int numreq = 0; + activeRequestCount = 0; + reqErrorCount = 0; + reqSuccessCount = 0; + nonEmptyPalinas.clear(); + completedRequests.clear(); + + for (Stop s : stops.subList(0, Math.min(stops.size(), MAX_ARRIVAL_STOPS))) { + + final MapiArrivalRequest req = new MapiArrivalRequest(s.ID, currentDate, timeRange, departures, this, this); + req.setTag(REQUEST_TAG); + volleyManager.addToRequestQueue(req); + activeRequestCount++; + numreq++; + completedRequests.put(s.ID, false); + } + listener.setProgress(reqErrorCount+reqSuccessCount, activeRequestCount); + return numreq; + } + + private int totalRequests(){ + return activeRequestCount + reqSuccessCount + reqErrorCount; + } + + + @Override + public void onErrorResponse(VolleyError error) { + if (error instanceof ParseError) { + //TODO + Log.w(DEBUG_TAG, "Parsing error for stop request"); + } else if (error instanceof NetworkError) { + String s; + if (error.networkResponse != null) + s = new String(error.networkResponse.data); + else s = ""; + Log.w(DEBUG_TAG, "Network error: " + s); + } else { + Log.w(DEBUG_TAG, "Volley Error: " + error.getMessage()); + } + if (error.networkResponse != null) { + Log.w(DEBUG_TAG, "Error status code: " + error.networkResponse.statusCode); + } + //counters + activeRequestCount--; + reqErrorCount++; + //flatProgressBar.setProgress(reqErrorCount + reqSuccessCount); + listener.setProgress(reqErrorCount + reqSuccessCount, activeRequestCount); + } + + @Override + public void onResponse(Palina palinaResult) { + //counter for requests + activeRequestCount--; + reqSuccessCount++; + listener.setProgress(reqErrorCount + reqSuccessCount, activeRequestCount); + + //add the palina to the successful one + if(palinaResult!=null) { + final List<Route> routes = palinaResult.queryAllRoutes(); + if (routes != null && !routes.isEmpty()) { + nonEmptyPalinas.add(palinaResult); + listener.showCompletedArrivals(nonEmptyPalinas); + } + } + } + + void cancelAllRequests() { + volleyManager.getRequestQueue().cancelAll(REQUEST_TAG); + //flatProgressBar.setVisibility(View.GONE); + listener.onAllRequestsCancelled(); + } + + public interface ArrivalsListener{ + void setProgress(int completedRequests, int pendingRequests); + + void onAllRequestsCancelled(); + + void showCompletedArrivals(ArrayList<Palina> completedPalinas); + } +} diff --git a/app/src/main/java/it/reyboz/bustorino/fragments/NearbyStopsFragment.java b/app/src/main/java/it/reyboz/bustorino/fragments/NearbyStopsFragment.java index 703152a..7bd2380 100644 --- a/app/src/main/java/it/reyboz/bustorino/fragments/NearbyStopsFragment.java +++ b/app/src/main/java/it/reyboz/bustorino/fragments/NearbyStopsFragment.java @@ -1,624 +1,591 @@ /* BusTO - Fragments components Copyright (C) 2018 Fabio Mazza This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see <http://www.gnu.org/licenses/>. */ package it.reyboz.bustorino.fragments; import android.content.Context; import android.content.SharedPreferences; import android.location.Location; import android.os.Bundle; import androidx.annotation.NonNull; import androidx.annotation.Nullable; import androidx.fragment.app.Fragment; import androidx.lifecycle.Observer; import androidx.lifecycle.ViewModelProvider; import androidx.core.util.Pair; import androidx.preference.PreferenceManager; import androidx.appcompat.widget.AppCompatButton; import androidx.recyclerview.widget.RecyclerView; import androidx.work.WorkInfo; import android.util.Log; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.ProgressBar; import android.widget.TextView; -import com.android.volley.*; import it.reyboz.bustorino.BuildConfig; import it.reyboz.bustorino.R; import it.reyboz.bustorino.adapters.ArrivalsStopAdapter; import it.reyboz.bustorino.backend.*; -import it.reyboz.bustorino.backend.mato.MapiArrivalRequest; import it.reyboz.bustorino.data.DatabaseUpdate; import it.reyboz.bustorino.middleware.AppLocationManager; import it.reyboz.bustorino.adapters.SquareStopAdapter; import it.reyboz.bustorino.middleware.AutoFitGridLayoutManager; import it.reyboz.bustorino.util.LocationCriteria; import it.reyboz.bustorino.util.StopSorterByDistance; import it.reyboz.bustorino.viewmodels.NearbyStopsViewModel; import org.jetbrains.annotations.NotNull; import java.util.*; public class NearbyStopsFragment extends Fragment { public enum FragType{ STOPS(1), ARRIVALS(2); private final int num; FragType(int num){ this.num = num; } public static FragType fromNum(int i){ switch (i){ case 1: return STOPS; case 2: return ARRIVALS; default: throw new IllegalArgumentException("type not recognized"); } } } private FragmentListenerMain mListener; private FragmentLocationListener fragmentLocationListener; private final static String DEBUG_TAG = "NearbyStopsFragment"; private final static String FRAGMENT_TYPE_KEY = "FragmentType"; //public final static int TYPE_STOPS = 19, TYPE_ARRIVALS = 20; private FragType fragment_type = FragType.STOPS; public final static String FRAGMENT_TAG="NearbyStopsFrag"; //data Bundle private final String BUNDLE_LOCATION = "location"; private final int LOADER_ID = 0; private RecyclerView gridRecyclerView; private SquareStopAdapter dataAdapter; private AutoFitGridLayoutManager gridLayoutManager; private GPSPoint lastPosition = null; private ProgressBar circlingProgressBar,flatProgressBar; private int distance = 10; protected SharedPreferences globalSharedPref; private SharedPreferences.OnSharedPreferenceChangeListener preferenceChangeListener; private TextView messageTextView,titleTextView, loadingTextView; private CommonScrollListener scrollListener; private AppCompatButton switchButton; private boolean firstLocForStops = true,firstLocForArrivals = true; public static final int COLUMN_WIDTH_DP = 250; private Integer MAX_DISTANCE = -3; private int MIN_NUM_STOPS = -1; private int TIME_INTERVAL_REQUESTS = -1; private AppLocationManager locManager; //These are useful for the case of nearby arrivals - private ArrivalsManager arrivalsManager = null; + private NearbyArrivalsDownloader arrivalsManager = null; private ArrivalsStopAdapter arrivalsStopAdapter = null; private boolean dbUpdateRunning = false; private ArrayList<Stop> currentNearbyStops = new ArrayList<>(); + private NearbyArrivalsDownloader nearbyArrivalsDownloader; + + private final NearbyArrivalsDownloader.ArrivalsListener arrivalsListener = new NearbyArrivalsDownloader.ArrivalsListener() { + @Override + public void setProgress(int completedRequests, int pendingRequests) { + if(flatProgressBar!=null) { + if (pendingRequests == 0) { + flatProgressBar.setIndeterminate(true); + flatProgressBar.setVisibility(View.GONE); + } else { + flatProgressBar.setIndeterminate(false); + flatProgressBar.setProgress(completedRequests); + } + } + } + + @Override + public void onAllRequestsCancelled() { + if(flatProgressBar!=null) flatProgressBar.setVisibility(View.GONE); + } + + @Override + public void showCompletedArrivals(ArrayList<Palina> completedPalinas) { + showArrivalsInRecycler(completedPalinas); + } + }; //ViewModel private NearbyStopsViewModel viewModel; public NearbyStopsFragment() { // Required empty public constructor } /** * Use this factory method to create a new instance of * this fragment using the provided parameters. * @return A new instance of fragment NearbyStopsFragment. */ public static NearbyStopsFragment newInstance(FragType type) { //if(fragmentType != TYPE_STOPS && fragmentType != TYPE_ARRIVALS ) // throw new IllegalArgumentException("WRONG KIND OF FRAGMENT USED"); NearbyStopsFragment fragment = new NearbyStopsFragment(); final Bundle args = new Bundle(1); args.putInt(FRAGMENT_TYPE_KEY,type.num); fragment.setArguments(args); return fragment; } @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); if (getArguments() != null) { setFragmentType(FragType.fromNum(getArguments().getInt(FRAGMENT_TYPE_KEY))); } locManager = AppLocationManager.getInstance(getContext()); fragmentLocationListener = new FragmentLocationListener(); if (getContext()!=null) { globalSharedPref = getContext().getSharedPreferences(getString(R.string.mainSharedPreferences), Context.MODE_PRIVATE); globalSharedPref.registerOnSharedPreferenceChangeListener(preferenceChangeListener); } + nearbyArrivalsDownloader = new NearbyArrivalsDownloader(getContext().getApplicationContext(), arrivalsListener); + } @Override public View onCreateView(@NonNull LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { // Inflate the layout for this fragment if (getContext() == null) throw new RuntimeException(); View root = inflater.inflate(R.layout.fragment_nearby_stops, container, false); gridRecyclerView = root.findViewById(R.id.stopGridRecyclerView); gridLayoutManager = new AutoFitGridLayoutManager(getContext().getApplicationContext(), Float.valueOf(utils.convertDipToPixels(getContext(),COLUMN_WIDTH_DP)).intValue()); gridRecyclerView.setLayoutManager(gridLayoutManager); gridRecyclerView.setHasFixedSize(false); circlingProgressBar = root.findViewById(R.id.circularProgressBar); flatProgressBar = root.findViewById(R.id.horizontalProgressBar); messageTextView = root.findViewById(R.id.messageTextView); titleTextView = root.findViewById(R.id.titleTextView); loadingTextView = root.findViewById(R.id.positionLoadingTextView); switchButton = root.findViewById(R.id.switchButton); scrollListener = new CommonScrollListener(mListener,false); switchButton.setOnClickListener(v -> switchFragmentType()); Log.d(DEBUG_TAG, "onCreateView"); DatabaseUpdate.watchUpdateWorkStatus(getContext(), this, new Observer<List<WorkInfo>>() { @Override public void onChanged(List<WorkInfo> workInfos) { if(workInfos.isEmpty()) return; WorkInfo wi = workInfos.get(0); if (wi.getState() == WorkInfo.State.RUNNING && locManager.isRequesterRegistered(fragmentLocationListener)) { locManager.removeLocationRequestFor(fragmentLocationListener); dbUpdateRunning = true; } else{ //start the request if(!locManager.isRequesterRegistered(fragmentLocationListener)) locManager.addLocationRequestFor(fragmentLocationListener); dbUpdateRunning = false; } } }); //observe the livedata viewModel.getStopsAtDistance().observe(getViewLifecycleOwner(), stops -> { if (!dbUpdateRunning && (stops.size() < MIN_NUM_STOPS && distance <= MAX_DISTANCE)) { distance = distance + 40; viewModel.requestStopsAtDistance(distance, true); //Log.d(DEBUG_TAG, "Doubling distance now!"); return; } if(!stops.isEmpty()) { Log.d(DEBUG_TAG, "Showing "+stops.size()+" stops nearby"); currentNearbyStops =stops; showStopsInViews(currentNearbyStops, lastPosition); } }); return root; } /** * Use this method to set the fragment type * @param type the type, TYPE_ARRIVALS or TYPE_STOPS */ private void setFragmentType(FragType type){ this.fragment_type = type; switch(type){ case ARRIVALS: TIME_INTERVAL_REQUESTS = 5*1000; break; case STOPS: TIME_INTERVAL_REQUESTS = 1000; } } @Override public void onAttach(@NonNull Context context) { super.onAttach(context); if (context instanceof FragmentListenerMain) { mListener = (FragmentListenerMain) context; } else { throw new RuntimeException(context + " must implement OnFragmentInteractionListener"); } Log.d(DEBUG_TAG, "OnAttach called"); viewModel = new ViewModelProvider(this).get(NearbyStopsViewModel.class); } @Override public void onPause() { super.onPause(); gridRecyclerView.setAdapter(null); locManager.removeLocationRequestFor(fragmentLocationListener); Log.d(DEBUG_TAG,"On paused called"); } @Override public void onResume() { super.onResume(); try{ if(!dbUpdateRunning && !locManager.isRequesterRegistered(fragmentLocationListener)) locManager.addLocationRequestFor(fragmentLocationListener); } catch (SecurityException ex){ //ignored //try another location provider } + //fix view if we were showing the stops or the arrivals + prepareForFragmentType(); switch(fragment_type){ case STOPS: if(dataAdapter!=null){ - gridRecyclerView.setAdapter(dataAdapter); + //gridRecyclerView.setAdapter(dataAdapter); circlingProgressBar.setVisibility(View.GONE); loadingTextView.setVisibility(View.GONE); } break; case ARRIVALS: if(arrivalsStopAdapter!=null){ - gridRecyclerView.setAdapter(arrivalsStopAdapter); + //gridRecyclerView.setAdapter(arrivalsStopAdapter); circlingProgressBar.setVisibility(View.GONE); loadingTextView.setVisibility(View.GONE); } } mListener.enableRefreshLayout(false); Log.d(DEBUG_TAG,"OnResume called"); if(getContext()==null){ Log.e(DEBUG_TAG, "NULL CONTEXT, everything is going to crash now"); MIN_NUM_STOPS = 5; MAX_DISTANCE = 600; return; } //Re-read preferences SharedPreferences shpr = PreferenceManager.getDefaultSharedPreferences(getContext().getApplicationContext()); //For some reason, they are all saved as strings MAX_DISTANCE = shpr.getInt(getString(R.string.pref_key_radius_recents),600); boolean isMinStopInt = true; try{ MIN_NUM_STOPS = shpr.getInt(getString(R.string.pref_key_num_recents), 5); } catch (ClassCastException ex){ isMinStopInt = false; } if(!isMinStopInt) try { MIN_NUM_STOPS = Integer.parseInt(shpr.getString(getString(R.string.pref_key_num_recents), "5")); } catch (NumberFormatException ex){ MIN_NUM_STOPS = 5; } if(BuildConfig.DEBUG) Log.d(DEBUG_TAG, "Max distance for stops: "+MAX_DISTANCE+ ", Min number of stops: "+MIN_NUM_STOPS); + } @Override public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) { super.onViewCreated(view, savedInstanceState); gridRecyclerView.setVisibility(View.INVISIBLE); gridRecyclerView.addOnScrollListener(scrollListener); } @Override public void onDetach() { super.onDetach(); mListener = null; if(arrivalsManager!=null) arrivalsManager.cancelAllRequests(); } /** * Display the stops, or run new set of requests for arrivals */ private void showStopsInViews(ArrayList<Stop> stops, GPSPoint location){ if (stops.isEmpty()) { setNoStopsLayout(); return; } double minDistance = Double.POSITIVE_INFINITY; for(Stop s: stops){ minDistance = Math.min(minDistance, s.getDistanceFromLocation(location.getLatitude(), location.getLongitude())); } //quick trial to hopefully always get the stops in the correct order Collections.sort(stops,new StopSorterByDistance(location)); switch (fragment_type){ case STOPS: showStopsInRecycler(stops); break; case ARRIVALS: - arrivalsManager = new ArrivalsManager(stops); - flatProgressBar.setVisibility(View.VISIBLE); + if(getContext()==null) break; //don't do anything if we're not attached + if(arrivalsManager==null) + arrivalsManager = new NearbyArrivalsDownloader(getContext().getApplicationContext(), arrivalsListener); + arrivalsManager.requestArrivalsForStops(stops); + /*flatProgressBar.setVisibility(View.VISIBLE); flatProgressBar.setProgress(0); flatProgressBar.setIndeterminate(false); + */ //for the moment, be satisfied with only one location //AppLocationManager.getInstance(getContext()).removeLocationRequestFor(fragmentLocationListener); break; default: } } /** * To enable targeting from the Button */ public void switchFragmentType(View v){ switchFragmentType(); } /** * Call when you need to switch the type of fragment */ private void switchFragmentType(){ - if(fragment_type==FragType.ARRIVALS){ - setFragmentType(FragType.STOPS); + switch (fragment_type){ + case ARRIVALS: + setFragmentType(FragType.STOPS); + break; + case STOPS: + setFragmentType(FragType.ARRIVALS); + break; + default: + } + prepareForFragmentType(); + fragmentLocationListener.lastUpdateTime = -1; + //locManager.removeLocationRequestFor(fragmentLocationListener); + //locManager.addLocationRequestFor(fragmentLocationListener); + showStopsInViews(currentNearbyStops, lastPosition); + } + + /** + * Prepare the views for the set fragment type + */ + private void prepareForFragmentType(){ + if(fragment_type==FragType.STOPS){ switchButton.setText(getString(R.string.show_arrivals)); titleTextView.setText(getString(R.string.nearby_stops_message)); if(arrivalsManager!=null) arrivalsManager.cancelAllRequests(); if(dataAdapter!=null) gridRecyclerView.setAdapter(dataAdapter); - } else if (fragment_type==FragType.STOPS){ - setFragmentType(FragType.ARRIVALS); + } else if (fragment_type==FragType.ARRIVALS){ titleTextView.setText(getString(R.string.nearby_arrivals_message)); switchButton.setText(getString(R.string.show_stops)); if(arrivalsStopAdapter!=null) gridRecyclerView.setAdapter(arrivalsStopAdapter); } - fragmentLocationListener.lastUpdateTime = -1; - //locManager.removeLocationRequestFor(fragmentLocationListener); - //locManager.addLocationRequestFor(fragmentLocationListener); - showStopsInViews(currentNearbyStops, lastPosition); } //useful methods /////// GUI METHODS //////// private void showStopsInRecycler(List<Stop> stops){ if(firstLocForStops) { dataAdapter = new SquareStopAdapter(stops, mListener, lastPosition); gridRecyclerView.setAdapter(dataAdapter); firstLocForStops = false; }else { dataAdapter.setStops(stops); dataAdapter.setUserPosition(lastPosition); } dataAdapter.notifyDataSetChanged(); //showRecyclerHidingLoadMessage(); if (gridRecyclerView.getVisibility() != View.VISIBLE) { circlingProgressBar.setVisibility(View.GONE); loadingTextView.setVisibility(View.GONE); gridRecyclerView.setVisibility(View.VISIBLE); } messageTextView.setVisibility(View.GONE); if(mListener!=null) mListener.readyGUIfor(FragmentKind.NEARBY_STOPS); } private void showArrivalsInRecycler(List<Palina> palinas){ Collections.sort(palinas,new StopSorterByDistance(lastPosition)); final ArrayList<Pair<Stop,Route>> routesPairList = new ArrayList<>(10); //int maxNum = Math.min(MAX_STOPS, stopList.size()); for(Palina p: palinas){ //if there are no routes available, skip stop - if(p.queryAllRoutes().size() == 0) continue; + if(p.queryAllRoutes().isEmpty()) continue; for(Route r: p.queryAllRoutes()){ //if there are no routes, should not do anything if (r.passaggi != null && !r.passaggi.isEmpty()) routesPairList.add(new Pair<>(p,r)); } } if (getContext()==null){ Log.e(DEBUG_TAG, "Trying to show arrivals in Recycler but we're not attached"); return; } if(firstLocForArrivals){ arrivalsStopAdapter = new ArrivalsStopAdapter(routesPairList,mListener,getContext(),lastPosition); gridRecyclerView.setAdapter(arrivalsStopAdapter); firstLocForArrivals = false; } else { arrivalsStopAdapter.setRoutesPairListAndPosition(routesPairList,lastPosition); } //arrivalsStopAdapter.notifyDataSetChanged(); showRecyclerHidingLoadMessage(); if(mListener!=null) mListener.readyGUIfor(FragmentKind.NEARBY_ARRIVALS); } private void setNoStopsLayout(){ messageTextView.setVisibility(View.VISIBLE); messageTextView.setText(R.string.no_stops_nearby); circlingProgressBar.setVisibility(View.GONE); loadingTextView.setVisibility(View.GONE); } /** * Does exactly what is says on the tin */ private void showRecyclerHidingLoadMessage(){ if (gridRecyclerView.getVisibility() != View.VISIBLE) { circlingProgressBar.setVisibility(View.GONE); loadingTextView.setVisibility(View.GONE); gridRecyclerView.setVisibility(View.VISIBLE); } messageTextView.setVisibility(View.GONE); } - class ArrivalsManager implements Response.Listener<Palina>, Response.ErrorListener{ - final HashMap<String,Palina> palinasDone = new HashMap<>(); - //final Map<String,List<Route>> routesToAdd = new HashMap<>(); - final static String REQUEST_TAG = "NearbyArrivals"; - final NetworkVolleyManager volleyManager; - int activeRequestCount = 0,reqErrorCount = 0, reqSuccessCount=0; - - ArrivalsManager(List<Stop> stops){ - volleyManager = NetworkVolleyManager.getInstance(getContext()); - - int MAX_ARRIVAL_STOPS = 35; - Date currentDate = new Date(); - int timeRange = 3600; - int departures = 10; - int numreq = 0; - for(Stop s: stops.subList(0,Math.min(stops.size(), MAX_ARRIVAL_STOPS))){ - - final MapiArrivalRequest req = new MapiArrivalRequest(s.ID, currentDate, timeRange, departures, this, this); - req.setTag(REQUEST_TAG); - volleyManager.addToRequestQueue(req); - activeRequestCount++; - numreq++; - } - flatProgressBar.setMax(numreq); - } - - - - @Override - public void onErrorResponse(VolleyError error) { - if(error instanceof ParseError){ - //TODO - Log.w(DEBUG_TAG,"Parsing error for stop request"); - } else if (error instanceof NetworkError){ - String s; - if(error.networkResponse!=null) - s = new String(error.networkResponse.data); - else s=""; - Log.w(DEBUG_TAG,"Network error: "+s); - }else { - Log.w(DEBUG_TAG,"Volley Error: "+error.getMessage()); - } - if(error.networkResponse!=null){ - Log.w(DEBUG_TAG, "Error status code: "+error.networkResponse.statusCode); - } - //counters - activeRequestCount--; - reqErrorCount++; - flatProgressBar.setProgress(reqErrorCount+reqSuccessCount); - } - - @Override - public void onResponse(Palina result) { - //counter for requests - activeRequestCount--; - reqSuccessCount++; - //final Palina palinaInMap = palinasDone.get(result.ID); - //palina cannot be null here - //sorry for the brutal crash when it happens - //if(palinaInMap == null) throw new IllegalStateException("Cannot get the palina from the map"); - //add the palina to the successful one - //TODO: Avoid redoing everything every time a new Result arrives - palinasDone.put(result.ID, result); - final ArrayList<Palina> outList = new ArrayList<>(); - for(Palina p: palinasDone.values()){ - final List<Route> routes = p.queryAllRoutes(); - if(routes!=null && routes.size()>0) outList.add(p); - } - showArrivalsInRecycler(outList); - flatProgressBar.setProgress(reqErrorCount+reqSuccessCount); - if(activeRequestCount==0) { - flatProgressBar.setIndeterminate(true); - flatProgressBar.setVisibility(View.GONE); - } - } - void cancelAllRequests(){ - volleyManager.getRequestQueue().cancelAll(REQUEST_TAG); - flatProgressBar.setVisibility(View.GONE); - } - } /** * Local locationListener, to use for the GPS */ class FragmentLocationListener implements AppLocationManager.LocationRequester{ private int oldLocStatus = -2; private LocationCriteria cr; private long lastUpdateTime = -1; @Override public void onLocationChanged(Location location) { //set adapter if(location==null){ Log.e(DEBUG_TAG, "Location is null, cannot request stops"); return; } else if(viewModel==null){ return; } - if(location.getAccuracy()<100 && !dbUpdateRunning) { + if(location.getAccuracy()<200 && !dbUpdateRunning) { if(viewModel.getDistanceMtLiveData().getValue()==null){ //never run request distance = 40; } lastPosition = new GPSPoint(location.getLatitude(), location.getLongitude()); viewModel.requestStopsAtDistance(location.getLatitude(), location.getLongitude(), distance, true); } lastUpdateTime = System.currentTimeMillis(); Log.d("BusTO:NearPositListen","can start request for stops: "+ !dbUpdateRunning); } @Override public void onLocationStatusChanged(int status) { switch(status){ case AppLocationManager.LOCATION_GPS_AVAILABLE: messageTextView.setVisibility(View.GONE); break; case AppLocationManager.LOCATION_UNAVAILABLE: messageTextView.setText(R.string.enableGpsText); messageTextView.setVisibility(View.VISIBLE); break; default: Log.e(DEBUG_TAG,"Location status not recognized"); } } @Override public @NotNull LocationCriteria getLocationCriteria() { return new LocationCriteria(200,TIME_INTERVAL_REQUESTS); } @Override public long getLastUpdateTimeMillis() { return lastUpdateTime; } void resetUpdateTime(){ lastUpdateTime = -1; } @Override public void onLocationProviderAvailable() { } @Override public void onLocationDisabled() { } } } diff --git a/app/src/main/res/layout/fragment_nearby_stops.xml b/app/src/main/res/layout/fragment_nearby_stops.xml index ec0108d..37a7020 100644 --- a/app/src/main/res/layout/fragment_nearby_stops.xml +++ b/app/src/main/res/layout/fragment_nearby_stops.xml @@ -1,85 +1,87 @@ <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="it.reyboz.bustorino.fragments.NearbyStopsFragment"> <TextView android:layout_width="match_parent" android:layout_height="wrap_content" android:text="@string/nearby_stops_message" android:id="@+id/titleTextView" android:textAppearance="@android:style/TextAppearance.Medium" android:layout_marginBottom="6dp" android:layout_marginTop="15dp" android:paddingTop="3dp" android:gravity="center_horizontal" android:textSize="23sp" android:layout_toLeftOf="@id/switchButton" android:layout_marginLeft="10dp" android:layout_marginStart="10dp" android:layout_marginRight="10dp" android:layout_marginEnd="10dp" /> <androidx.appcompat.widget.AppCompatButton xmlns:app="http://schemas.android.com/apk/res-auto" android:text="@string/show_arrivals" android:layout_width="wrap_content" android:layout_height="wrap_content" android:id="@+id/switchButton" android:layout_alignParentRight="true" android:layout_alignParentEnd="true" android:layout_margin="10dp" app:backgroundTint="@color/blue_500" android:textColor="@android:color/white" /> + <ProgressBar + style="@style/Base.Widget.AppCompat.ProgressBar.Horizontal" + android:layout_width="6dp" + android:layout_height="wrap_content" + android:id="@+id/horizontalProgressBar" + android:layout_alignParentEnd="true" + android:layout_alignParentStart="true" + android:layout_below="@id/titleTextView" + android:indeterminate="true" + android:visibility="gone" + android:layout_marginStart="5dp" + android:layout_marginEnd="5dp" + /> <androidx.recyclerview.widget.RecyclerView android:layout_width="match_parent" - android:layout_height="match_parent" android:layout_below="@id/titleTextView" + android:layout_height="match_parent" android:layout_below="@id/horizontalProgressBar" android:id="@+id/stopGridRecyclerView" android:clickable="true" android:focusable="true" android:visibility="gone" android:clipChildren="false" android:clipToPadding="true" android:horizontalSpacing="10dp" - android:layout_above="@id/horizontalProgressBar" - android:verticalSpacing="10dp"/> - <ProgressBar - style="@style/Base.Widget.AppCompat.ProgressBar.Horizontal" - android:layout_width="3dp" - android:layout_height="wrap_content" - android:id="@+id/horizontalProgressBar" - android:layout_alignParentRight="true" - android:layout_alignParentEnd="true" - android:layout_alignParentLeft="true" android:layout_alignParentBottom="true" - android:indeterminate="true" - android:visibility="gone" - /> + android:verticalSpacing="10dp"/> + <ProgressBar style="?android:attr/progressBarStyle" android:layout_width="wrap_content" android:layout_height="wrap_content" android:id="@+id/circularProgressBar" android:layout_marginTop="25dp" android:progressDrawable="@color/blue_620" android:indeterminate="true" android:indeterminateTint="@color/blue_620" android:layout_below="@+id/titleTextView" android:layout_centerHorizontal="true"/> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:id="@+id/positionLoadingTextView" android:layout_below="@id/circularProgressBar" android:text="@string/position_searching_message" android:layout_marginTop="8dp" android:textSize="15sp" android:layout_centerHorizontal="true" /> <TextView android:text="@string/enableGpsText" android:layout_width="wrap_content" android:layout_height="wrap_content" android:id="@+id/messageTextView" android:visibility="gone" android:textAppearance="@style/Base.ThemeOverlay.AppCompat.Light" android:layout_below="@+id/titleTextView" android:layout_centerHorizontal="true" android:textSize="17sp" android:layout_marginTop="20dp" /> </RelativeLayout> diff --git a/app/src/main/res/layout/stop_card.xml b/app/src/main/res/layout/stop_card.xml index c18c1c1..3771ff8 100644 --- a/app/src/main/res/layout/stop_card.xml +++ b/app/src/main/res/layout/stop_card.xml @@ -1,117 +1,117 @@ <?xml version="1.0" encoding="utf-8"?> <androidx.cardview.widget.CardView xmlns:card_view="http://schemas.android.com/apk/res-auto" xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/stop_cardView" android:foreground="?android:attr/selectableItemBackground" android:clickable="true" android:focusable="true" android:layout_width="match_parent" android:layout_height="wrap_content" card_view:cardCornerRadius="5dp" android:layout_margin="5dp" android:padding="4dp"> <LinearLayout android:orientation="horizontal" android:layout_width="match_parent" android:layout_height="match_parent" android:gravity="center_vertical"> <RelativeLayout android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_weight="1"> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="Title" android:id="@+id/stop_nameText" style="@style/TextAppearance.AppCompat.Medium" android:layout_alignParentLeft="true" android:layout_alignParentStart="true" android:paddingLeft="6dp" android:textColor="@color/primary_material_dark" android:layout_marginBottom="6dp" android:layout_marginTop="6dp" android:layout_marginLeft="6dp" android:layout_marginStart="6dp" android:layout_marginRight="6dp" android:layout_marginEnd="6dp"/> <TextView android:id="@+id/stop_numberText" android:layout_width="wrap_content" - android:layout_height="20dp" + android:layout_height="wrap_content" android:text="stopNumber" - android:gravity="center_vertical" - android:textAppearance="@style/TextAppearance.AppCompat.Medium" + android:textSize="16sp" android:layout_below="@id/stop_nameText" - android:textIsSelectable="true" android:layout_marginTop="8dp" - android:textColor="@color/orange_500" android:layout_marginLeft="12dp" - android:layout_marginStart="12dp" android:layout_marginRight="8dp" android:layout_marginEnd="8dp" + android:textIsSelectable="true" android:layout_marginTop="6dp" + android:textColor="@color/orange_700" + android:layout_marginStart="12dp" android:layout_marginEnd="8dp" android:layout_marginBottom="8dp"/> <TextView - android:text="TextView" + android:text="200m" android:layout_width="wrap_content" android:layout_height="wrap_content" - android:layout_marginRight="6dp" - android:layout_marginLeft="10dp" + android:layout_marginEnd="6dp" + android:layout_marginStart="10dp" android:minWidth="50dp" + android:textSize="16sp" android:id="@+id/stop_distanceTextView" android:layout_toEndOf="@+id/stop_numberText" android:layout_toRightOf="@id/stop_numberText" android:layout_alignTop="@+id/stop_numberText" /> <!-- <LinearLayout android:orientation="horizontal" android:layout_width="fill_parent" android:layout_height="fill_parent" android:id="@+id/row1" android:layout_below="@+id/stopTitle"> <TextView android:id="@+id/txtLine1_1" android:layout_width="wrap_content" android:layout_height="20dp" android:text="Name" android:gravity="center_vertical" android:textSize="14sp" android:layout_marginTop="10dp" android:layout_marginLeft="5dp" android:layout_margin="3dp"/> <TextView android:id="@+id/txtPass1_1" android:layout_width="wrap_content" android:layout_height="20dp" android:text="Name" android:gravity="center_vertical" android:layout_marginTop="10dp" android:layout_marginLeft="5dp" android:layout_margin="3dp"/> <TextView android:id="@+id/txtPass2" android:layout_width="wrap_content" android:layout_height="20dp" android:text="Name" android:gravity="center_vertical" android:textSize="14sp" android:layout_marginTop="10dp" android:layout_marginLeft="5dp" android:layout_margin="3dp"/> <TextView android:id="@+id/txtPass3" android:layout_width="wrap_content" android:layout_height="20dp" android:text="Name" android:gravity="center_vertical" android:textSize="14sp" android:layout_marginTop="10dp" android:layout_marginLeft="5dp" android:layout_margin="3dp"/> </LinearLayout> --> </RelativeLayout> <TextView android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="" android:id="@+id/stop_linesText" android:layout_toRightOf="@+id/stop_distanceTextView" android:minWidth="80dp" android:layout_alignParentRight="true" android:layout_alignParentEnd="true" android:layout_alignBottom="@+id/stop_distanceTextView" android:layout_alignTop="@+id/stop_nameText" android:nestedScrollingEnabled="true" android:gravity="center_horizontal|center" android:textColor="@color/blue_700" android:textSize="14sp" android:layout_weight="1" android:padding="5dp" android:maxWidth="100sp" android:layout_gravity="center_vertical|end" android:layout_marginRight="5dp"/> </LinearLayout> </androidx.cardview.widget.CardView> \ No newline at end of file