Compare commits
	
		
			38 Commits
		
	
	
		
			v0.0.20
			...
			7acfb84122
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 7acfb84122 | |||
| cbada7e4a4 | |||
| 4a542a4a1f | |||
| f25355ee3e | |||
| e18a9c63e6 | |||
| 5fcadbe8d8 | |||
| 5afb646381 | |||
| d0e837377b | |||
| d94c69c545 | |||
| 9e595ad933 | |||
| 53d56f3e30 | |||
| f39d02f967 | |||
| 94a7adac6c | |||
| 4d99715447 | |||
| 48555e7429 | |||
| 8b24876fd1 | |||
| c832461f29 | |||
| 6f1a019d4f | |||
| e6ccb7078b | |||
| 84839c5a02 | |||
| 9850e949c3 | |||
| 5fc25a3c39 | |||
| f76cd603f3 | |||
| 67f748244f | |||
| 66fa55e8c3 | |||
| f4729a2de7 | |||
| 40edd923c3 | |||
| 6ad749eeed | |||
| c20ebf3d63 | |||
| 6facde6e0b | |||
| 55b0a1b793 | |||
| 39df97f4d1 | |||
| 43aa26a107 | |||
| badb8ff919 | |||
| 290baec64e | |||
| 3710a476e8 | |||
| cdc9b0ecd1 | |||
| 097abc5f29 | 
| @@ -6,7 +6,7 @@ on: | |||||||
|       - frontend/** |       - frontend/** | ||||||
|  |  | ||||||
|  |  | ||||||
| name: Build and release APK | name: Build and release debug APK | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   build: |   build: | ||||||
| @@ -55,7 +55,7 @@ jobs: | |||||||
|         ls -lah android |         ls -lah android | ||||||
|       working-directory: ./frontend |       working-directory: ./frontend | ||||||
|  |  | ||||||
|     - run: flutter build apk --release --split-per-abi --build-number=${{ gitea.run_number }} |     - run: flutter build apk --debug --split-per-abi --build-number=${{ gitea.run_number }} | ||||||
|       working-directory: ./frontend |       working-directory: ./frontend | ||||||
|  |  | ||||||
|     - name: Upload APKs to artifacts |     - name: Upload APKs to artifacts | ||||||
|   | |||||||
| @@ -32,4 +32,4 @@ jobs: | |||||||
|     - name: Deploy to k8s |     - name: Deploy to k8s | ||||||
|       run: | |       run: | | ||||||
|         kubectl apply -k backend/deployment/overlays/${{ inputs.overlay }} --kubeconfig=kubeconfig |         kubectl apply -k backend/deployment/overlays/${{ inputs.overlay }} --kubeconfig=kubeconfig | ||||||
|         kubectl -n anyway-backend rollout restart deployment/anyway-backend-${{ inputs.overlay }} |         kubectl -n anyway-backend rollout restart deployment/anyway-backend-${{ inputs.overlay }} --kubeconfig=kubeconfig | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | |||||||
|  | { | ||||||
|  |     "cmake.ignoreCMakeListsMissing": true | ||||||
|  | } | ||||||
| @@ -9,9 +9,9 @@ name = "pypi" | |||||||
| numpy = "*" | numpy = "*" | ||||||
| fastapi = "*" | fastapi = "*" | ||||||
| pydantic = "*" | pydantic = "*" | ||||||
| geopy = "*" |  | ||||||
| shapely = "*" | shapely = "*" | ||||||
| scipy = "*" | scipy = "*" | ||||||
| osmpythontools = "*" | osmpythontools = "*" | ||||||
| pywikibot = "*" | pywikibot = "*" | ||||||
| pymemcache = "*" | pymemcache = "*" | ||||||
|  | fastapi-cli = "*" | ||||||
|   | |||||||
							
								
								
									
										2208
									
								
								backend/Pipfile.lock
									
									
									
										generated
									
									
									
								
							
							
						
						| @@ -1,12 +1,37 @@ | |||||||
| # Backend | # Backend | ||||||
|  |  | ||||||
| This repository contains the backend code for the application. It utilizes FastAPI that allows to quickly create a RESTful API that exposes the endpoints of the route optimizer. | This repository contains the backend code for the application. It utilizes **FastAPI** to quickly create a RESTful API that exposes the endpoints of the route optimizer. | ||||||
|  |  | ||||||
|  |  | ||||||
| ## Getting Started | ## Getting Started | ||||||
| - The code of the python application is located in the `src` directory. |  | ||||||
| - Package management is handled with `pipenv` and the dependencies are listed in the `Pipfile`. | ### Directory Structure | ||||||
| - Since the application is aimed to be deployed in a container, the `Dockerfile` is provided to build the image. | - The code for the Python application is located in the `src` directory. | ||||||
|  | - Package management is handled with **pipenv**, and the dependencies are listed in the `Pipfile`. | ||||||
|  | - Since the application is designed to be deployed in a container, the `Dockerfile` is provided to build the image. | ||||||
|  |  | ||||||
|  | ### Setting Up the Development Environment | ||||||
|  |  | ||||||
|  | To set up your development environment using **pipenv**, follow these steps: | ||||||
|  |  | ||||||
|  | 1. Install `pipenv` by running: | ||||||
|  |     ```bash | ||||||
|  |     sudo apt install pipenv | ||||||
|  |     ``` | ||||||
|  |  | ||||||
|  | 2. Create and activate a virtual environment: | ||||||
|  |     ```bash | ||||||
|  |     pipenv shell | ||||||
|  |     ``` | ||||||
|  |  | ||||||
|  | 3. Install the dependencies listed in the `Pipfile`: | ||||||
|  |     ```bash | ||||||
|  |     pipenv install | ||||||
|  |     ``` | ||||||
|  |  | ||||||
|  | 4. The virtual environment will be created under: | ||||||
|  |     ```bash | ||||||
|  |     ~/.local/share/virtualenvs/... | ||||||
|  |     ``` | ||||||
|  |  | ||||||
| ### Deployment | ### Deployment | ||||||
| To deploy the backend docker container, we use kubernetes. Modifications to the backend are automatically pushed to a two-stage environment through the CI pipeline. See [deployment/README](deployment/README.md] for further information. | To deploy the backend docker container, we use kubernetes. Modifications to the backend are automatically pushed to a two-stage environment through the CI pipeline. See [deployment/README](deployment/README.md] for further information. | ||||||
|   | |||||||
| @@ -16,7 +16,7 @@ OSM_CACHE_DIR = Path(cache_dir_string) | |||||||
|  |  | ||||||
| import logging | import logging | ||||||
| # if we are in a debug session, set verbose and rich logging | # if we are in a debug session, set verbose and rich logging | ||||||
| if os.getenv('DEBUG', False): | if os.getenv('DEBUG', "false") == "true": | ||||||
|     from rich.logging import RichHandler |     from rich.logging import RichHandler | ||||||
|     logging.basicConfig( |     logging.basicConfig( | ||||||
|         level=logging.DEBUG, |         level=logging.DEBUG, | ||||||
|   | |||||||
| @@ -63,7 +63,7 @@ def new_trip(preferences: Preferences, start: tuple[float, float], end: tuple[fl | |||||||
|     refined_tour = refiner.refine_optimization(landmarks, base_tour, preferences.max_time_minute, preferences.detour_tolerance_minute) |     refined_tour = refiner.refine_optimization(landmarks, base_tour, preferences.max_time_minute, preferences.detour_tolerance_minute) | ||||||
|  |  | ||||||
|     linked_tour = LinkedLandmarks(refined_tour) |     linked_tour = LinkedLandmarks(refined_tour) | ||||||
|     # upon creation of the trip, persistence of both the trip and its landmarks is ensured. Ca |     # upon creation of the trip, persistence of both the trip and its landmarks is ensured | ||||||
|     trip = Trip.from_linked_landmarks(linked_tour, cache_client) |     trip = Trip.from_linked_landmarks(linked_tour, cache_client) | ||||||
|     return trip |     return trip | ||||||
|  |  | ||||||
| @@ -84,4 +84,4 @@ def get_landmark(landmark_uuid: str) -> Landmark: | |||||||
|         landmark = cache_client.get(f"landmark_{landmark_uuid}") |         landmark = cache_client.get(f"landmark_{landmark_uuid}") | ||||||
|         return landmark |         return landmark | ||||||
|     except KeyError: |     except KeyError: | ||||||
|         raise HTTPException(status_code=404, detail="Landmark not found") |         raise HTTPException(status_code=404, detail="Landmark not found") | ||||||
|   | |||||||
| @@ -1,3 +1,6 @@ | |||||||
|  | # Tags were picked mostly arbitrarily, based on the OSM wiki and the OSM tags page. | ||||||
|  | # See https://taginfo.openstreetmap.org for more inspiration. | ||||||
|  |  | ||||||
| nature: | nature: | ||||||
|   leisure: park |   leisure: park | ||||||
|   geological: '' |   geological: '' | ||||||
| @@ -11,7 +14,24 @@ nature: | |||||||
|     - alpine_hut |     - alpine_hut | ||||||
|     - viewpoint |     - viewpoint | ||||||
|     - zoo |     - zoo | ||||||
|   waterway: waterfall |     - resort | ||||||
|  |     - picnic_site | ||||||
|  |   water: | ||||||
|  |     - pond | ||||||
|  |     - lake | ||||||
|  |     - river | ||||||
|  |     - basin | ||||||
|  |     - stream | ||||||
|  |     - lagoon | ||||||
|  |     - rapids | ||||||
|  |   waterway: | ||||||
|  |     - waterfall | ||||||
|  |     - river | ||||||
|  |     - canal | ||||||
|  |     - dam | ||||||
|  |     - dock | ||||||
|  |     - boatyard | ||||||
|  |  | ||||||
|  |  | ||||||
| shopping: | shopping: | ||||||
|   shop: |   shop: | ||||||
| @@ -23,10 +43,47 @@ sightseeing: | |||||||
|     - museum |     - museum | ||||||
|     - attraction |     - attraction | ||||||
|     - gallery |     - gallery | ||||||
|  |     - artwork | ||||||
|  |     - aquarium | ||||||
|   historic: '' |   historic: '' | ||||||
|   amenity: |   amenity: | ||||||
|     - planetarium |     - planetarium | ||||||
|     - place_of_worship |     - place_of_worship | ||||||
|     - fountain |     - fountain | ||||||
|  |     - townhall | ||||||
|   water: |   water: | ||||||
|     - reflecting_pool |     - reflecting_pool | ||||||
|  |   bridge: | ||||||
|  |     - aqueduct | ||||||
|  |     - viaduct | ||||||
|  |     - boardwalk | ||||||
|  |     - cantilever | ||||||
|  |     - abandoned | ||||||
|  |   building: | ||||||
|  |     - church | ||||||
|  |     - chapel | ||||||
|  |     - mosque | ||||||
|  |     - synagogue | ||||||
|  |     - ruins | ||||||
|  |     - temple | ||||||
|  |     - government | ||||||
|  |     - cathedral | ||||||
|  |     - castle | ||||||
|  |     - museum | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # to be used later on | ||||||
|  | restauration: | ||||||
|  |   shop: | ||||||
|  |     - coffee | ||||||
|  |     - bakery | ||||||
|  |     - restaurant | ||||||
|  |     - pastry | ||||||
|  |   amenity: | ||||||
|  |     - restaurant | ||||||
|  |     - cafe | ||||||
|  |     - ice_cream | ||||||
|  |     - food_court | ||||||
|  |     - biergarten | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| city_bbox_side: 7500 #m | city_bbox_side: 7500 #m | ||||||
| radius_close_to: 50 | radius_close_to: 50 | ||||||
| church_coeff: 0.75 | church_coeff: 0.5 | ||||||
| nature_coeff: 1.25 | nature_coeff: 1.25 | ||||||
| overall_coeff: 10 | overall_coeff: 10 | ||||||
| tag_exponent: 1.15 | tag_exponent: 1.15 | ||||||
|   | |||||||
| @@ -21,8 +21,8 @@ if constants.MEMCACHED_HOST_PATH is None: | |||||||
| else: | else: | ||||||
|     client = Client( |     client = Client( | ||||||
|         constants.MEMCACHED_HOST_PATH, |         constants.MEMCACHED_HOST_PATH, | ||||||
|         timeout=1, |         timeout = 1, | ||||||
|         allow_unicode_keys=True, |         allow_unicode_keys = True, | ||||||
|         encoding='utf-8', |         encoding = 'utf-8', | ||||||
|         serde=serde.pickle_serde |         serde = serde.pickle_serde | ||||||
|     ) |     ) | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ from uuid import uuid4 | |||||||
|  |  | ||||||
| # Output to frontend | # Output to frontend | ||||||
| class Landmark(BaseModel) : | class Landmark(BaseModel) : | ||||||
|      |  | ||||||
|     # Properties of the landmark |     # Properties of the landmark | ||||||
|     name : str |     name : str | ||||||
|     type: Literal['sightseeing', 'nature', 'shopping', 'start', 'finish'] |     type: Literal['sightseeing', 'nature', 'shopping', 'start', 'finish'] | ||||||
| @@ -14,31 +14,39 @@ class Landmark(BaseModel) : | |||||||
|     osm_id : int |     osm_id : int | ||||||
|     attractiveness : int |     attractiveness : int | ||||||
|     n_tags : int |     n_tags : int | ||||||
|     image_url : Optional[str] = None                            # TODO future |     image_url : Optional[str] = None | ||||||
|     website_url : Optional[str] = None |     website_url : Optional[str] = None | ||||||
|     wikipedia_url : Optional[str] = None |  | ||||||
|     description : Optional[str] = None                          # TODO future |     description : Optional[str] = None                          # TODO future | ||||||
|     duration : Optional[int] = 0                                # TODO future |     duration : Optional[int] = 0 | ||||||
|     name_en : Optional[str] = None |     name_en : Optional[str] = None | ||||||
|  |  | ||||||
|     # Unique ID of a given landmark |     # Unique ID of a given landmark | ||||||
|     uuid: str = Field(default_factory=uuid4)                    # TODO implement this ASAP |     uuid: str = Field(default_factory=uuid4) | ||||||
|      |  | ||||||
|     # Additional properties depending on specific tour |     # Additional properties depending on specific tour | ||||||
|     must_do : Optional[bool] = False |     must_do : Optional[bool] = False | ||||||
|     must_avoid : Optional[bool] = False |     must_avoid : Optional[bool] = False | ||||||
|     is_secondary : Optional[bool] = False                       # TODO future    |     is_secondary : Optional[bool] = False                       # TODO future    | ||||||
|      |  | ||||||
|     time_to_reach_next : Optional[int] = 0                      # TODO fix this in existing code |  | ||||||
|     next_uuid : Optional[str] = None                            # TODO implement this ASAP |  | ||||||
|  |  | ||||||
|     def __hash__(self) -> int: |     time_to_reach_next : Optional[int] = 0 | ||||||
|         return self.uuid.int |     next_uuid : Optional[str] = None | ||||||
|      |  | ||||||
|     def __str__(self) -> str: |     def __str__(self) -> str: | ||||||
|         time_to_next_str = f", time_to_next={self.time_to_reach_next}" if self.time_to_reach_next else "" |         time_to_next_str = f", time_to_next={self.time_to_reach_next}" if self.time_to_reach_next else "" | ||||||
|         is_secondary_str = f", secondary" if self.is_secondary else "" |         is_secondary_str = f", secondary" if self.is_secondary else "" | ||||||
|         type_str = '(' + self.type + ')' |         type_str = '(' + self.type + ')' | ||||||
|         if self.type in ["start", "finish", "nature", "shopping"] : type_str += '\t ' |         if self.type in ["start", "finish", "nature", "shopping"] : type_str += '\t ' | ||||||
|         return f'Landmark{type_str}: [{self.name} @{self.location}, score={self.attractiveness}{time_to_next_str}{is_secondary_str}]' |         return f'Landmark{type_str}: [{self.name} @{self.location}, score={self.attractiveness}{time_to_next_str}{is_secondary_str}]' | ||||||
|      |  | ||||||
|  |     def distance(self, value: 'Landmark') -> float: | ||||||
|  |         return (self.location[0] - value.location[0])**2 + (self.location[1] - value.location[1])**2 | ||||||
|  |  | ||||||
|  |     def __hash__(self) -> int: | ||||||
|  |         return hash(self.name) | ||||||
|  |  | ||||||
|  |     def __eq__(self, value: 'Landmark') -> bool: | ||||||
|  |         # eq and hash must be consistent | ||||||
|  |         # in particular, if two objects are equal, their hash must be equal | ||||||
|  |         # uuid and osm_id are just shortcuts to avoid comparing all the properties | ||||||
|  |         # if they are equal, we know that the name is also equal and in turn the hash is equal | ||||||
|  |         return self.uuid == value.uuid or self.osm_id == value.osm_id or (self.name == value.name and self.distance(value) < 0.001) | ||||||
|   | |||||||
| @@ -22,9 +22,10 @@ class Trip(BaseModel): | |||||||
|  |  | ||||||
|         # Store the trip in the cache |         # Store the trip in the cache | ||||||
|         cache_client.set(f"trip_{trip.uuid}", trip) |         cache_client.set(f"trip_{trip.uuid}", trip) | ||||||
|         cache_client.set_many({f"landmark_{landmark.uuid}": landmark for landmark in landmarks}, expire=3600) |         # make sure to await the result (noreply=False). Otherwise the cache might not be inplace when the trip is actually requested | ||||||
|  |         cache_client.set_many({f"landmark_{landmark.uuid}": landmark for landmark in landmarks}, expire=3600, noreply=False) | ||||||
|         # is equivalent to: |         # is equivalent to: | ||||||
|         # for landmark in landmarks: |         # for landmark in landmarks: | ||||||
|         #     cache_client.set(f"landmark_{landmark.uuid}", landmark, expire=3600) |         #     cache_client.set(f"landmark_{landmark.uuid}", landmark, expire=3600) | ||||||
|  |  | ||||||
|         return trip |         return trip | ||||||
|   | |||||||
| @@ -23,7 +23,7 @@ def test(start_coords: tuple[float, float], finish_coords: tuple[float, float] = | |||||||
|         sightseeing=Preference(type='sightseeing', score = 5), |         sightseeing=Preference(type='sightseeing', score = 5), | ||||||
|         nature=Preference(type='nature', score = 5), |         nature=Preference(type='nature', score = 5), | ||||||
|         shopping=Preference(type='shopping', score = 5), |         shopping=Preference(type='shopping', score = 5), | ||||||
|         max_time_minute=100, |         max_time_minute=15, | ||||||
|         detour_tolerance_minute=0 |         detour_tolerance_minute=0 | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
| @@ -74,6 +74,7 @@ def test(start_coords: tuple[float, float], finish_coords: tuple[float, float] = | |||||||
| # test(tuple((48.8344400, 2.3220540)))       # Café Chez César  | # test(tuple((48.8344400, 2.3220540)))       # Café Chez César  | ||||||
| # test(tuple((48.8375946, 2.2949904)))       # Point random | # test(tuple((48.8375946, 2.2949904)))       # Point random | ||||||
| # test(tuple((47.377859, 8.540585)))         # Zurich HB | # test(tuple((47.377859, 8.540585)))         # Zurich HB | ||||||
| # test(tuple((45.758217, 4.831814)))      # Lyon Bellecour | # test(tuple((45.758217, 4.831814)))         # Lyon Bellecour | ||||||
| test(tuple((48.5848435, 7.7332974)))      # Strasbourg Gare | # test(tuple((48.5848435, 7.7332974)))       # Strasbourg Gare | ||||||
| # test(tuple((48.2067858, 16.3692340)))      # Vienne | # test(tuple((48.2067858, 16.3692340)))      # Vienne | ||||||
|  | test(tuple((48.084588, 7.280405)))         # Turckheim  | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| import yaml | import yaml | ||||||
| from geopy.distance import geodesic | from math import sin, cos, sqrt, atan2, radians | ||||||
|  |  | ||||||
| import constants | import constants | ||||||
|  |  | ||||||
| @@ -8,6 +8,7 @@ with constants.OPTIMIZER_PARAMETERS_PATH.open('r') as f: | |||||||
|     DETOUR_FACTOR = parameters['detour_factor'] |     DETOUR_FACTOR = parameters['detour_factor'] | ||||||
|     AVERAGE_WALKING_SPEED = parameters['average_walking_speed'] |     AVERAGE_WALKING_SPEED = parameters['average_walking_speed'] | ||||||
|  |  | ||||||
|  | EARTH_RADIUS_KM = 6373 | ||||||
|  |  | ||||||
| def get_time(p1: tuple[float, float], p2: tuple[float, float]) -> int: | def get_time(p1: tuple[float, float], p2: tuple[float, float]) -> int: | ||||||
|     """ |     """ | ||||||
| @@ -16,24 +17,34 @@ def get_time(p1: tuple[float, float], p2: tuple[float, float]) -> int: | |||||||
|     Args: |     Args: | ||||||
|         p1 (Tuple[float, float]): Coordinates of the starting location. |         p1 (Tuple[float, float]): Coordinates of the starting location. | ||||||
|         p2 (Tuple[float, float]): Coordinates of the destination. |         p2 (Tuple[float, float]): Coordinates of the destination. | ||||||
|         detour (float): Detour factor affecting the distance. |  | ||||||
|         speed (float): Walking speed in kilometers per hour. |  | ||||||
|  |  | ||||||
|     Returns: |         Returns: | ||||||
|         int: Time to travel from p1 to p2 in minutes. |         int: Time to travel from p1 to p2 in minutes. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|  |  | ||||||
|     # Compute the straight-line distance in km |     if p1 == p2: | ||||||
|     if p1 == p2 : |  | ||||||
|         return 0 |         return 0 | ||||||
|     else:  |     else: | ||||||
|         dist = geodesic(p1, p2).kilometers |         # Compute the distance in km along the surface of the Earth | ||||||
|  |         # (assume spherical Earth) | ||||||
|  |         # this is the haversine formula, stolen from stackoverflow | ||||||
|  |         # in order to not use any external libraries | ||||||
|  |         lat1, lon1 = radians(p1[0]), radians(p1[1]) | ||||||
|  |         lat2, lon2 = radians(p2[0]), radians(p2[1]) | ||||||
|  |  | ||||||
|     # Consider the detour factor for average cityto deterline walking distance (in km) |         dlon = lon2 - lon1 | ||||||
|     walk_dist = dist*DETOUR_FACTOR |         dlat = lat2 - lat1 | ||||||
|  |  | ||||||
|  |         a = sin(dlat / 2)**2 + cos(lat1) * cos(lat2) * sin(dlon / 2)**2 | ||||||
|  |         c = 2 * atan2(sqrt(a), sqrt(1 - a)) | ||||||
|  |  | ||||||
|  |         distance = EARTH_RADIUS_KM * c | ||||||
|  |  | ||||||
|  |     # Consider the detour factor for average an average city | ||||||
|  |     walk_distance = distance * DETOUR_FACTOR | ||||||
|  |  | ||||||
|     # Time to walk this distance (in minutes) |     # Time to walk this distance (in minutes) | ||||||
|     walk_time = walk_dist/AVERAGE_WALKING_SPEED*60 |     walk_time = walk_distance / AVERAGE_WALKING_SPEED * 60 | ||||||
|  |  | ||||||
|     return round(walk_time) |     return round(walk_time) | ||||||
|   | |||||||
| @@ -1,20 +1,17 @@ | |||||||
| import math as m | import math | ||||||
| import yaml | import yaml | ||||||
| import logging | import logging | ||||||
|  |  | ||||||
| from OSMPythonTools.overpass import Overpass, overpassQueryBuilder | from OSMPythonTools.overpass import Overpass, overpassQueryBuilder | ||||||
| from OSMPythonTools.cachingStrategy import CachingStrategy, JSON | from OSMPythonTools.cachingStrategy import CachingStrategy, JSON | ||||||
| from pywikibot import ItemPage, Site |  | ||||||
| from pywikibot import config |  | ||||||
| config.put_throttle = 0 |  | ||||||
| config.maxlag = 0 |  | ||||||
|  |  | ||||||
| from structs.preferences import Preferences, Preference | from structs.preferences import Preferences | ||||||
| from structs.landmark import Landmark | from structs.landmark import Landmark | ||||||
| from .take_most_important import take_most_important | from .take_most_important import take_most_important | ||||||
| import constants | import constants | ||||||
|  |  | ||||||
|  | # silence the overpass logger | ||||||
|  | logging.getLogger('OSMPythonTools').setLevel(level=logging.CRITICAL) | ||||||
|  |  | ||||||
|  |  | ||||||
| class LandmarkManager: | class LandmarkManager: | ||||||
| @@ -46,7 +43,7 @@ class LandmarkManager: | |||||||
|             self.viewpoint_bonus = parameters['viewpoint_bonus'] |             self.viewpoint_bonus = parameters['viewpoint_bonus'] | ||||||
|             self.pay_bonus = parameters['pay_bonus'] |             self.pay_bonus = parameters['pay_bonus'] | ||||||
|             self.N_important = parameters['N_important'] |             self.N_important = parameters['N_important'] | ||||||
|              |  | ||||||
|         with constants.OPTIMIZER_PARAMETERS_PATH.open('r') as f: |         with constants.OPTIMIZER_PARAMETERS_PATH.open('r') as f: | ||||||
|             parameters = yaml.safe_load(f) |             parameters = yaml.safe_load(f) | ||||||
|             self.walking_speed = parameters['average_walking_speed'] |             self.walking_speed = parameters['average_walking_speed'] | ||||||
| @@ -69,87 +66,44 @@ class LandmarkManager: | |||||||
|         preferences (Preferences): The user's preference settings that influence the landmark selection. |         preferences (Preferences): The user's preference settings that influence the landmark selection. | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             tuple[list[Landmark], list[Landmark]]: |         tuple[list[Landmark], list[Landmark]]: | ||||||
|                 - A list of all existing landmarks. |         - A list of all existing landmarks. | ||||||
|                 - A list of the most important landmarks based on the user's preferences. |         - A list of the most important landmarks based on the user's preferences. | ||||||
|         """ |         """ | ||||||
|  |  | ||||||
|         max_walk_dist = (preferences.max_time_minute/2)/60*self.walking_speed*1000/self.detour_factor |         max_walk_dist = (preferences.max_time_minute/2)/60*self.walking_speed*1000/self.detour_factor | ||||||
|         reachable_bbox_side = min(max_walk_dist, self.max_bbox_side) |         reachable_bbox_side = min(max_walk_dist, self.max_bbox_side) | ||||||
|  |  | ||||||
|         L = [] |         # use set to avoid duplicates, this requires some __methods__ to be set in Landmark | ||||||
|  |         all_landmarks = set() | ||||||
|  |  | ||||||
|         bbox = self.create_bbox(center_coordinates, reachable_bbox_side) |         bbox = self.create_bbox(center_coordinates, reachable_bbox_side) | ||||||
|         # list for sightseeing |         # list for sightseeing | ||||||
|         if preferences.sightseeing.score != 0: |         if preferences.sightseeing.score != 0: | ||||||
|             score_function = lambda score: int(score*10*preferences.sightseeing.score/5)   # self.count_elements_close_to(loc) + |             score_function = lambda score: score * 10 * preferences.sightseeing.score / 5 | ||||||
|             L1 = self.fetch_landmarks(bbox, self.amenity_selectors['sightseeing'], preferences.sightseeing.type, score_function) |             current_landmarks = self.fetch_landmarks(bbox, self.amenity_selectors['sightseeing'], preferences.sightseeing.type, score_function) | ||||||
|             L += L1 |             all_landmarks.update(current_landmarks) | ||||||
|  |  | ||||||
|         # list for nature |         # list for nature | ||||||
|         if preferences.nature.score != 0: |         if preferences.nature.score != 0: | ||||||
|             score_function = lambda score: int(score*10*self.nature_coeff*preferences.nature.score/5)   # self.count_elements_close_to(loc) + |             score_function = lambda score: score * 10 * self.nature_coeff * preferences.nature.score / 5 | ||||||
|             L2 = self.fetch_landmarks(bbox, self.amenity_selectors['nature'], preferences.nature.type, score_function) |             current_landmarks = self.fetch_landmarks(bbox, self.amenity_selectors['nature'], preferences.nature.type, score_function) | ||||||
|             L += L2 |             all_landmarks.update(current_landmarks) | ||||||
|  |  | ||||||
|         # list for shopping |         # list for shopping | ||||||
|         if preferences.shopping.score != 0: |         if preferences.shopping.score != 0: | ||||||
|             score_function = lambda score: int(score*10*preferences.shopping.score/5)   # self.count_elements_close_to(loc) + |             score_function = lambda score: score * 10 * preferences.shopping.score / 5 | ||||||
|             L3 = self.fetch_landmarks(bbox, self.amenity_selectors['shopping'], preferences.shopping.type, score_function) |             current_landmarks = self.fetch_landmarks(bbox, self.amenity_selectors['shopping'], preferences.shopping.type, score_function) | ||||||
|             L += L3 |             # set time for all shopping activites : | ||||||
|  |             for landmark in current_landmarks : landmark.duration = 45 | ||||||
|  |             all_landmarks.update(current_landmarks) | ||||||
|  |  | ||||||
|  |  | ||||||
|         L = self.remove_duplicates(L) |         landmarks_constrained = take_most_important(all_landmarks, self.N_important) | ||||||
|         # self.correct_score(L, preferences) |         self.logger.info(f'Generated {len(all_landmarks)} landmarks around {center_coordinates}, and constrained to {len(landmarks_constrained)} most important ones.') | ||||||
|  |  | ||||||
|         L_constrained = take_most_important(L, self.N_important) |         return all_landmarks, landmarks_constrained | ||||||
|         self.logger.info(f'Generated {len(L)} landmarks around {center_coordinates}, and constrained to {len(L_constrained)} most important ones.') |  | ||||||
|  |  | ||||||
|         return L, L_constrained |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     def remove_duplicates(self, landmarks: list[Landmark]) -> list[Landmark]: |  | ||||||
|         """ |  | ||||||
|         Removes duplicate landmarks based on their names from the given list. Only retains the landmark with highest score |  | ||||||
|  |  | ||||||
|         Parameters: |  | ||||||
|         landmarks (list[Landmark]): A list of Landmark objects. |  | ||||||
|  |  | ||||||
|         Returns: |  | ||||||
|         list[Landmark]: A list of unique Landmark objects based on their names. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         L_clean = [] |  | ||||||
|         names = [] |  | ||||||
|  |  | ||||||
|         for landmark in landmarks: |  | ||||||
|             if landmark.name in names:  |  | ||||||
|                 continue   |  | ||||||
|             else: |  | ||||||
|                 names.append(landmark.name) |  | ||||||
|                 L_clean.append(landmark) |  | ||||||
|          |  | ||||||
|         return L_clean |  | ||||||
|          |  | ||||||
|  |  | ||||||
|     def correct_score(self, landmarks: list[Landmark], preferences: Preferences) -> None: |  | ||||||
|         """ |  | ||||||
|         Adjust the attractiveness score of each landmark in the list based on user preferences. |  | ||||||
|  |  | ||||||
|         This method updates the attractiveness of each landmark by scaling it according to the user's preference score. |  | ||||||
|         The score adjustment is computed using a simple linear transformation based on the preference score. |  | ||||||
|  |  | ||||||
|         Args: |  | ||||||
|             landmarks (list[Landmark]): A list of landmarks whose scores need to be corrected. |  | ||||||
|             preferences (Preferences): The user's preference settings that influence the attractiveness score adjustment. |  | ||||||
|         """ |  | ||||||
|  |  | ||||||
|         score_dict = { |  | ||||||
|             preferences.sightseeing.type: preferences.sightseeing.score, |  | ||||||
|             preferences.nature.type: preferences.nature.score, |  | ||||||
|             preferences.shopping.type: preferences.shopping.score |  | ||||||
|         } |  | ||||||
|         for landmark in landmarks: |  | ||||||
|             landmark.attractiveness = int(landmark.attractiveness * score_dict[landmark.type] / 5)         |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     def count_elements_close_to(self, coordinates: tuple[float, float]) -> int: |     def count_elements_close_to(self, coordinates: tuple[float, float]) -> int: | ||||||
| @@ -172,7 +126,7 @@ class LandmarkManager: | |||||||
|  |  | ||||||
|         radius = self.radius_close_to |         radius = self.radius_close_to | ||||||
|  |  | ||||||
|         alpha = (180*radius) / (6371000*m.pi) |         alpha = (180 * radius) / (6371000 * math.pi) | ||||||
|         bbox = {'latLower':lat-alpha,'lonLower':lon-alpha,'latHigher':lat+alpha,'lonHigher': lon+alpha} |         bbox = {'latLower':lat-alpha,'lonLower':lon-alpha,'latHigher':lat+alpha,'lonHigher': lon+alpha} | ||||||
|  |  | ||||||
|         # Build the query to find elements within the radius |         # Build the query to find elements within the radius | ||||||
| @@ -216,7 +170,7 @@ class LandmarkManager: | |||||||
|  |  | ||||||
|         # Convert distance to degrees |         # Convert distance to degrees | ||||||
|         lat_diff = half_side_length_km / 111  # 1 degree latitude is approximately 111 km |         lat_diff = half_side_length_km / 111  # 1 degree latitude is approximately 111 km | ||||||
|         lon_diff = half_side_length_km / (111 * m.cos(m.radians(lat)))  # Adjust for longitude based on latitude |         lon_diff = half_side_length_km / (111 * math.cos(math.radians(lat)))  # Adjust for longitude based on latitude | ||||||
|  |  | ||||||
|         # Calculate bbox |         # Calculate bbox | ||||||
|         min_lat = lat - lat_diff |         min_lat = lat - lat_diff | ||||||
| @@ -255,32 +209,32 @@ class LandmarkManager: | |||||||
|             query = overpassQueryBuilder( |             query = overpassQueryBuilder( | ||||||
|                 bbox = bbox, |                 bbox = bbox, | ||||||
|                 elementType = ['way', 'relation'], |                 elementType = ['way', 'relation'], | ||||||
|  |                 # selector can in principle be a list already, | ||||||
|  |                 # but it generates the intersection of the queries | ||||||
|  |                 # we want the union | ||||||
|                 selector = sel, |                 selector = sel, | ||||||
|                 # conditions = [], |                 conditions = ['count_tags()>5'], | ||||||
|                 includeCenter = True, |                 includeCenter = True, | ||||||
|                 out = 'body' |                 out = 'body' | ||||||
|                 ) |                 ) | ||||||
|  |             self.logger.debug(f"Query: {query}") | ||||||
|  |  | ||||||
|             try: |             try: | ||||||
|                 result = self.overpass.query(query) |                 result = self.overpass.query(query) | ||||||
|             except Exception as e: |             except Exception as e: | ||||||
|                 self.logger.error(f"Error fetching landmarks: {e}") |                 self.logger.error(f"Error fetching landmarks: {e}") | ||||||
|                 return |                 continue | ||||||
|              |  | ||||||
|             for elem in result.elements(): |             for elem in result.elements(): | ||||||
|  |  | ||||||
|                 name = elem.tag('name')                             # Add name |                 name = elem.tag('name') | ||||||
|                 location = (elem.centerLat(), elem.centerLon())     # Add coordinates (lat, lon) |                 location = (elem.centerLat(), elem.centerLon()) | ||||||
|  |  | ||||||
|                 # TODO: exclude these from the get go |                 # TODO: exclude these from the get go | ||||||
|                 # skip if unprecise location |                 # skip if unprecise location | ||||||
|                 if name is None or location[0] is None: |                 if name is None or location[0] is None: | ||||||
|                     continue |                     continue | ||||||
|  |  | ||||||
|                 # skip if unused |  | ||||||
|                 # if 'disused:leisure' in elem.tags().keys(): |  | ||||||
|                 #     continue |  | ||||||
|                  |  | ||||||
|                 # skip if part of another building |                 # skip if part of another building | ||||||
|                 if 'building:part' in elem.tags().keys() and elem.tag('building:part') == 'yes': |                 if 'building:part' in elem.tags().keys() and elem.tag('building:part') == 'yes': | ||||||
|                     continue |                     continue | ||||||
| @@ -291,36 +245,32 @@ class LandmarkManager: | |||||||
|                 n_tags = len(elem.tags().keys())    # Add number of tags |                 n_tags = len(elem.tags().keys())    # Add number of tags | ||||||
|                 score = n_tags**self.tag_exponent   # Add score |                 score = n_tags**self.tag_exponent   # Add score | ||||||
|                 website_url = None |                 website_url = None | ||||||
|                 wikpedia_url = None |  | ||||||
|                 image_url = None |                 image_url = None | ||||||
|                 name_en = None |                 name_en = None | ||||||
|  |  | ||||||
|                 # remove specific tags |                 # Adjust scoring | ||||||
|                 skip = False |                 skip = False | ||||||
|                 for tag in elem.tags().keys(): |                 for tag in elem.tags().keys(): | ||||||
|                     if "pay" in tag: |                     if "pay" in tag: | ||||||
|                         score += self.pay_bonus             # discard payment options for tags |                         # payment options are misleading and should not count for the scoring. | ||||||
|  |                         score += self.pay_bonus | ||||||
|  |  | ||||||
|                     if "disused" in tag: |                     if "disused" in tag: | ||||||
|                         skip = True             # skip disused amenities |                         # skip disused amenities | ||||||
|  |                         skip = True | ||||||
|                         break |                         break | ||||||
|  |  | ||||||
|                     if "wiki" in tag: |                     if "wiki" in tag: | ||||||
|                         score += self.wikipedia_bonus             # wikipedia entries count more |                         # wikipedia entries count more | ||||||
|                          |                         score += self.wikipedia_bonus | ||||||
|                     # if tag == "wikidata": |  | ||||||
|                     #     Q = elem.tag('wikidata') |  | ||||||
|                     #     site = Site("wikidata", "wikidata") |  | ||||||
|                     #     item = ItemPage(site, Q) |  | ||||||
|                     #     item.get() |  | ||||||
|                     #     n_languages = len(item.labels) |  | ||||||
|                     #     n_tags += n_languages/10 |  | ||||||
|  |  | ||||||
|                     if "viewpoint" in tag: |                     if "viewpoint" in tag: | ||||||
|  |                         # viewpoints must count more | ||||||
|                         score += self.viewpoint_bonus |                         score += self.viewpoint_bonus | ||||||
|                         duration = 10 |                         duration = 10 | ||||||
|  |  | ||||||
|                     if "image" in tag: |                     if "image" in tag: | ||||||
|  |                         # images must count more | ||||||
|                         score += self.image_bonus |                         score += self.image_bonus | ||||||
|  |  | ||||||
|                     if elem_type != "nature": |                     if elem_type != "nature": | ||||||
| @@ -335,47 +285,43 @@ class LandmarkManager: | |||||||
|                         if tag == "building" and elem.tag('building') in ['retail', 'supermarket', 'parking']: |                         if tag == "building" and elem.tag('building') in ['retail', 'supermarket', 'parking']: | ||||||
|                             skip = True |                             skip = True | ||||||
|                             break |                             break | ||||||
|                      |  | ||||||
|                     # Get additional information |                     # Extract image, website and english name | ||||||
|                     # if tag == 'wikipedia' : |                     if tag in ['website', 'contact:website']: | ||||||
|                     #     wikpedia_url = elem.tag('wikipedia') |  | ||||||
|                     if tag in ['website', 'contact:website'] : |  | ||||||
|                         website_url = elem.tag(tag) |                         website_url = elem.tag(tag) | ||||||
|                     if tag == 'image' : |                     if tag == 'image': | ||||||
|                         image_url = elem.tag('image') |                         image_url = elem.tag('image') | ||||||
|                     if tag =='name:en' : |                     if tag =='name:en': | ||||||
|                         name_en = elem.tag('name:en') |                         name_en = elem.tag('name:en') | ||||||
|  |  | ||||||
|                 if skip: |                 if skip: | ||||||
|                     continue |                     continue | ||||||
|  |  | ||||||
|                 score = score_function(score) |                 score = score_function(score) | ||||||
|                 if "place_of_worship" in elem.tags().values() : |                 if "place_of_worship" in elem.tags().values(): | ||||||
|                     score = int(score*self.church_coeff) |                     score = score * self.church_coeff | ||||||
|                     duration = 15 |                     duration = 10 | ||||||
|                  |                  | ||||||
|                 elif "museum" in elem.tags().values() : |                 elif "museum" in elem.tags().values() or "aquarium" in elem.tags().values() or "planetarium" in elem.tags().values(): | ||||||
|                     score = int(score*self.church_coeff) |  | ||||||
|                     duration = 60 |                     duration = 60 | ||||||
|                  |                  | ||||||
|                 else :  |                 else: | ||||||
|                     duration = 5 |                     duration = 5 | ||||||
|  |  | ||||||
|                 # Generate the landmark and append it to the list |                 # finally create our own landmark object | ||||||
|                 landmark = Landmark( |                 landmark = Landmark( | ||||||
|                     name=name, |                     name = name, | ||||||
|                     type=elem_type, |                     type = elem_type, | ||||||
|                     location=location, |                     location = location, | ||||||
|                     osm_type=osm_type, |                     osm_type = osm_type, | ||||||
|                     osm_id=osm_id, |                     osm_id = osm_id, | ||||||
|                     attractiveness=score, |                     attractiveness = int(score), | ||||||
|                     must_do=False, |                     must_do = False, | ||||||
|                     n_tags=int(n_tags), |                     n_tags = int(n_tags), | ||||||
|                     duration = duration,  |                     duration = int(duration), | ||||||
|                     name_en=name_en, |                     name_en = name_en, | ||||||
|                     image_url=image_url, |                     image_url = image_url, | ||||||
|                     # wikipedia_url=wikpedia_url, |                     website_url = website_url | ||||||
|                     website_url=website_url |  | ||||||
|                 ) |                 ) | ||||||
|                 return_list.append(landmark) |                 return_list.append(landmark) | ||||||
|          |          | ||||||
| @@ -399,7 +345,7 @@ def dict_to_selector_list(d: dict) -> list: | |||||||
|     for key, value in d.items(): |     for key, value in d.items(): | ||||||
|         if type(value) == list: |         if type(value) == list: | ||||||
|             val = '|'.join(value) |             val = '|'.join(value) | ||||||
|             return_list.append(f'{key}~"{val}"') |             return_list.append(f'{key}~"^({val})$"') | ||||||
|         elif type(value) == str and len(value) == 0: |         elif type(value) == str and len(value) == 0: | ||||||
|             return_list.append(f'{key}') |             return_list.append(f'{key}') | ||||||
|         else: |         else: | ||||||
|   | |||||||
| @@ -3,7 +3,6 @@ import numpy as np | |||||||
|  |  | ||||||
| from scipy.optimize import linprog | from scipy.optimize import linprog | ||||||
| from collections import defaultdict, deque | from collections import defaultdict, deque | ||||||
| from geopy.distance import geodesic |  | ||||||
|  |  | ||||||
| from structs.landmark import Landmark | from structs.landmark import Landmark | ||||||
| from .get_time_separation import get_time | from .get_time_separation import get_time | ||||||
| @@ -488,7 +487,7 @@ class Optimizer: | |||||||
|  |  | ||||||
|         # Raise error if no solution is found |         # Raise error if no solution is found | ||||||
|         if not res.success : |         if not res.success : | ||||||
|             raise ArithmeticError("No solution could be found, the problem is overconstrained. Please adapt your must_dos") |             raise ArithmeticError("No solution could be found, the problem is overconstrained. Try with a longer trip (>30 minutes).") | ||||||
|  |  | ||||||
|         # If there is a solution, we're good to go, just check for connectiveness |         # If there is a solution, we're good to go, just check for connectiveness | ||||||
|         order, circles = self.is_connected(res.x) |         order, circles = self.is_connected(res.x) | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ import yaml, logging | |||||||
|  |  | ||||||
| from shapely import buffer, LineString, Point, Polygon, MultiPoint, concave_hull | from shapely import buffer, LineString, Point, Polygon, MultiPoint, concave_hull | ||||||
| from math import pi | from math import pi | ||||||
|  | from typing import List | ||||||
|  |  | ||||||
| from structs.landmark import Landmark | from structs.landmark import Landmark | ||||||
| from . import take_most_important, get_time_separation | from . import take_most_important, get_time_separation | ||||||
| @@ -133,6 +134,21 @@ class Refiner : | |||||||
|             i += 1 |             i += 1 | ||||||
|      |      | ||||||
|         return tour |         return tour | ||||||
|  |      | ||||||
|  |     def integrate_landmarks(self, sub_list: List[Landmark], main_list: List[Landmark]) : | ||||||
|  |         """ | ||||||
|  |         Inserts 'sub_list' of Landmarks inside the 'main_list' by leaving the ends untouched. | ||||||
|  |          | ||||||
|  |         Args:  | ||||||
|  |             sub_list    : the list of Landmarks to be inserted inside of the 'main_list'. | ||||||
|  |             main_list   : the original list with start and finish. | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             the full list. | ||||||
|  |         """ | ||||||
|  |         sub_list.append(main_list[-1])          # add finish back | ||||||
|  |         return main_list[:-1] + sub_list        # create full set of possible landmarks | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     def find_shortest_path_through_all_landmarks(self, landmarks: list[Landmark]) -> tuple[list[Landmark], Polygon]: |     def find_shortest_path_through_all_landmarks(self, landmarks: list[Landmark]) -> tuple[list[Landmark], Polygon]: | ||||||
| @@ -253,6 +269,11 @@ class Refiner : | |||||||
|         except : |         except : | ||||||
|             better_tour_poly = concave_hull(MultiPoint(coords))  # Create concave hull with "core" of tour leaving out start and finish |             better_tour_poly = concave_hull(MultiPoint(coords))  # Create concave hull with "core" of tour leaving out start and finish | ||||||
|             xs, ys = better_tour_poly.exterior.xy |             xs, ys = better_tour_poly.exterior.xy | ||||||
|  |             """  | ||||||
|  |             ERROR HERE :  | ||||||
|  |                 Exception has occurred: AttributeError | ||||||
|  |                 'LineString' object has no attribute 'exterior' | ||||||
|  |             """ | ||||||
|  |  | ||||||
|  |  | ||||||
|         # reverse the xs and ys |         # reverse the xs and ys | ||||||
| @@ -315,26 +336,30 @@ class Refiner : | |||||||
|  |  | ||||||
|         self.logger.info(f"Using {len(minor_landmarks)} minor landmarks around the predicted path") |         self.logger.info(f"Using {len(minor_landmarks)} minor landmarks around the predicted path") | ||||||
|  |  | ||||||
|         # full set of visitable landmarks |         # Full set of visitable landmarks. | ||||||
|         full_set = base_tour[:-1] + minor_landmarks   # create full set of possible landmarks (without finish) |         full_set = self.integrate_landmarks(minor_landmarks, base_tour)     # could probably be optimized with less overhead | ||||||
|         full_set.append(base_tour[-1])                # add finish back |  | ||||||
|  |  | ||||||
|         # get a new tour |         # Generate a new tour with the optimizer. | ||||||
|         new_tour = self.optimizer.solve_optimization( |         new_tour = self.optimizer.solve_optimization( | ||||||
|             max_time = max_time + detour, |             max_time = max_time + detour, | ||||||
|             landmarks = full_set,  |             landmarks = full_set,  | ||||||
|             max_landmarks = self.max_landmarks_refiner |             max_landmarks = self.max_landmarks_refiner | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |         # If unsuccessful optimization, use the base_tour. | ||||||
|         if new_tour is None: |         if new_tour is None: | ||||||
|             self.logger.warning("No solution found for the refined tour. Returning the initial tour.") |             self.logger.warning("No solution found for the refined tour. Returning the initial tour.") | ||||||
|             new_tour = base_tour |             new_tour = base_tour | ||||||
|  |  | ||||||
|  |         # If only one landmark, return it. | ||||||
|  |         if len(new_tour) < 4 : | ||||||
|  |             return new_tour | ||||||
|  |  | ||||||
|         # Find shortest path using the nearest neighbor heuristic |         # Find shortest path using the nearest neighbor heuristic. | ||||||
|         better_tour, better_poly = self.find_shortest_path_through_all_landmarks(new_tour) |         better_tour, better_poly = self.find_shortest_path_through_all_landmarks(new_tour) | ||||||
|  |  | ||||||
|         # Fix the tour using Polygons if the path looks weird |         # Fix the tour using Polygons if the path looks weird.  | ||||||
|  |         # Conditions : circular trip and invalid polygon. | ||||||
|         if base_tour[0].location == base_tour[-1].location and not better_poly.is_valid : |         if base_tour[0].location == base_tour[-1].location and not better_poly.is_valid : | ||||||
|             better_tour = self.fix_using_polygon(better_tour) |             better_tour = self.fix_using_polygon(better_tour) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,38 +1,16 @@ | |||||||
| from structs.landmark import Landmark | from structs.landmark import Landmark | ||||||
|  |  | ||||||
| def take_most_important(landmarks: list[Landmark], N_important) -> list[Landmark] : | def take_most_important(landmarks: list[Landmark], n_important) -> list[Landmark]: | ||||||
|     L = len(landmarks) |     """ | ||||||
|     L_copy = [] |     Given a list of landmarks, return the n_important most important landmarks | ||||||
|     L_clean = [] |     Parameters: | ||||||
|     scores = [0]*len(landmarks) |     landmarks: list[Landmark] - list of landmarks | ||||||
|     names = [] |     n_important: int - number of most important landmarks to return | ||||||
|     name_id = {} |     Returns: | ||||||
|  |     list[Landmark] - list of the n_important most important landmarks | ||||||
|  |     """ | ||||||
|  |  | ||||||
|     for i, elem in enumerate(landmarks) : |     # Sort landmarks by attractiveness (descending) | ||||||
|         if elem.name not in names : |     sorted_landmarks = sorted(landmarks, key=lambda x: x.attractiveness, reverse=True) | ||||||
|             names.append(elem.name) |  | ||||||
|             name_id[elem.name] = [i] |  | ||||||
|             L_copy.append(elem) |  | ||||||
|         else : |  | ||||||
|             name_id[elem.name] += [i] |  | ||||||
|             scores = [] |  | ||||||
|             for j in name_id[elem.name] : |  | ||||||
|                 scores.append(L[j].attractiveness) |  | ||||||
|             best_id = max(range(len(scores)), key=scores.__getitem__) |  | ||||||
|             t = name_id[elem.name][best_id] |  | ||||||
|             if t == i : |  | ||||||
|                 for old in L_copy : |  | ||||||
|                     if old.name == elem.name : |  | ||||||
|                         old.attractiveness = L[t].attractiveness |  | ||||||
|      |  | ||||||
|     scores = [0]*len(L_copy) |  | ||||||
|     for i, elem in enumerate(L_copy) : |  | ||||||
|         scores[i] = elem.attractiveness |  | ||||||
|  |  | ||||||
|     res = sorted(range(len(scores)), key = lambda sub: scores[sub])[-(N_important-L):] |     return sorted_landmarks[:n_important] | ||||||
|  |  | ||||||
|     for i, elem in enumerate(L_copy) : |  | ||||||
|         if i in res : |  | ||||||
|             L_clean.append(elem) |  | ||||||
|  |  | ||||||
|     return L_clean |  | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| on: | on: | ||||||
|   push: |   push: | ||||||
|     branches: |     tags: | ||||||
|       - main |       - 'v*' | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   build: |   build: | ||||||
| @@ -23,6 +23,13 @@ jobs: | |||||||
|  |  | ||||||
|       - name: Setup android SDK |       - name: Setup android SDK | ||||||
|         uses: android-actions/setup-android@v3 |         uses: android-actions/setup-android@v3 | ||||||
|  |        | ||||||
|  |       - name: Install Flutter | ||||||
|  |         uses: subosito/flutter-action@v2 | ||||||
|  |         with: | ||||||
|  |           channel: stable | ||||||
|  |           flutter-version: 3.22.0 | ||||||
|  |           cache: true | ||||||
|  |  | ||||||
|       - name: Infer version number from git tag |       - name: Infer version number from git tag | ||||||
|         id: version |         id: version | ||||||
| @@ -30,13 +37,25 @@ jobs: | |||||||
|           REF_NAME: ${{ github.ref_name }} |           REF_NAME: ${{ github.ref_name }} | ||||||
|         run: |         run: | ||||||
|           # remove the 'v' prefix from the tag name |           # remove the 'v' prefix from the tag name | ||||||
|           echo "VERSION=${REF_NAME//v}" >> $GITHUB_OUTPUT |           echo "BUILD_NAME=${REF_NAME//v}" >> $GITHUB_ENV | ||||||
|  |  | ||||||
|       - name: Load secrets from github |       - name: Load secrets | ||||||
|  |         id: load-secrets | ||||||
|  |         uses: hashicorp/vault-action@v3 | ||||||
|  |         with: | ||||||
|  |           url: https://api.hashicorp.com | ||||||
|  |           token: ${{ secrets.VAULT_TOKEN }} | ||||||
|  |           secrets: | | ||||||
|  |             secret/release GOOGLE_MAPS_API_KEY | GOOGLE_MAPS_API_KEY ; | ||||||
|  |             secret/release ANDROID_SECRET_PROPERTIES_BASE64 | ANDROID_SECRET_PROPERTIES_BASE64 ; | ||||||
|  |             secret/release ANDROID_GOOGLE_PLAY_JSON_BASE64 | ANDROID_GOOGLE_PLAY_JSON_BASE64 ; | ||||||
|  |             secret/release ANDROID_KEYSTORE_BASE64 | ANDROID_KEYSTORE_BASE64 ; | ||||||
|  |  | ||||||
|  |       - name: Put selected secrets into files | ||||||
|         run: | |         run: | | ||||||
|           echo "${{ secrets.ANDROID_SECRET_PROPERTIES }}" > secrets.properties |           echo "${{ steps.load-secrets.outputs.ANDROID_SECRET_PROPERTIES_BASE64 }}" | base64 -d > secrets.properties | ||||||
|           echo "${{ secrets.ANDROID_KEYSTORE }}" > release.keystore |           echo "${{ steps.load-secrets.outputs.ANDROID_GOOGLE_PLAY_JSON_BASE64 }}" | base64 -d > google-key.json | ||||||
|           echo "${{ secrets.ANDROID_GOOGLE_PLAY_JSON }}" > google-key.json |           echo "${{ steps.load-secrets.outputs.ANDROID_KEYSTORE_BASE64 }}" | base64 -d > release.keystore | ||||||
|         working-directory: android |         working-directory: android | ||||||
|  |  | ||||||
|       - name: Install fastlane |       - name: Install fastlane | ||||||
| @@ -44,7 +63,9 @@ jobs: | |||||||
|         working-directory: android |         working-directory: android | ||||||
|  |  | ||||||
|       - name: Run fastlane lane |       - name: Run fastlane lane | ||||||
|         run: bundle exec fastlane deploy_testing |         run: bundle exec fastlane deploy_release | ||||||
|         working-directory: android |         working-directory: android | ||||||
|         env: |         env: | ||||||
|           VERSION_NAME: ${{ steps.version.VERSION }} |           BUILD_NUMBER: ${{ github.run_number }} | ||||||
|  |           # BUILD_NAME is implicitly available | ||||||
|  |           GOOGLE_MAPS_API_KEY: ${{ steps.load-secrets.outputs.GOOGLE_MAPS_API_KEY }} | ||||||
|   | |||||||
							
								
								
									
										65
									
								
								frontend/.github/workflows/build_app_ios.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,65 @@ | |||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     tags: | ||||||
|  |       - 'v*' | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   build: | ||||||
|  |     runs-on: macos-latest | ||||||
|  |     steps: | ||||||
|  |       - uses: actions/checkout@v4 | ||||||
|  |  | ||||||
|  |       - name: Set up ruby env | ||||||
|  |         uses: ruby/setup-ruby@v1 | ||||||
|  |         with: | ||||||
|  |           ruby-version: 3.2.1 | ||||||
|  |           bundler-cache: true | ||||||
|  |  | ||||||
|  |       - name: Install Flutter | ||||||
|  |         uses: subosito/flutter-action@v2 | ||||||
|  |         with: | ||||||
|  |           channel: stable | ||||||
|  |           flutter-version: 3.22.0 | ||||||
|  |           cache: true | ||||||
|  |  | ||||||
|  |       - name: Infer version number from git tag | ||||||
|  |         id: version | ||||||
|  |         env:  | ||||||
|  |           REF_NAME: ${{ github.ref_name }} | ||||||
|  |         run: | ||||||
|  |           # remove the 'v' prefix from the tag name | ||||||
|  |           echo "BUILD_NAME=${REF_NAME//v}" >> $GITHUB_ENV | ||||||
|  |  | ||||||
|  |       - name: Load secrets | ||||||
|  |         id: load-secrets | ||||||
|  |         uses: hashicorp/vault-action@v3 | ||||||
|  |         with: | ||||||
|  |           url: https://api.hashicorp.com | ||||||
|  |           token: ${{ secrets.VAULT_TOKEN }} | ||||||
|  |           secrets: | | ||||||
|  |             secret/release GOOGLE_MAPS_API_KEY | GOOGLE_MAPS_API_KEY ; | ||||||
|  |             secret/release IOS_ASC_KEY_ID | IOS_ASC_KEY_ID ; | ||||||
|  |             secret/release IOS_ASC_ISSUER_ID | IOS_ASC_ISSUER_ID ; | ||||||
|  |             secret/release IOS_ASC_KEY | IOS_ASC_KEY ; | ||||||
|  |             secret/release IOS_MATCH_REPO_SSH_KEY_BASE64 | IOS_MATCH_REPO_SSH_KEY_BASE64 ; | ||||||
|  |  | ||||||
|  |       - name: Setup SSH key for match git repo | ||||||
|  |         run: echo "$MATCH_REPO_SSH_KEY" | base64 --decode > ~/.ssh/id_rsa && chmod 600 ~/.ssh/id_rsa | ||||||
|  |         env: | ||||||
|  |           MATCH_REPO_SSH_KEY: ${{ steps.load-secrets.outputs.IOS_MATCH_REPO_SSH_KEY_BASE64 }} | ||||||
|  |  | ||||||
|  |       - name: Install fastlane | ||||||
|  |         run: bundle install | ||||||
|  |         working-directory: ios | ||||||
|  |  | ||||||
|  |       - name: Run fastlane lane | ||||||
|  |         run: bundle exec fastlane deploy_release | ||||||
|  |         working-directory: ios | ||||||
|  |         env: | ||||||
|  |           BUILD_NUMBER: ${{ github.run_number }} | ||||||
|  |           # BUILD_NAME is implicitly available | ||||||
|  |           GOOGLE_MAPS_API_KEY: ${{ steps.load-secrets.outputs.GOOGLE_MAPS_API_KEY }} | ||||||
|  |           IOS_ASC_KEY_ID: ${{ steps.load-secrets.outputs.IOS_ASC_KEY_ID }} | ||||||
|  |           IOS_ASC_ISSUER_ID: ${{ steps.load-secrets.outputs.IOS_ASC_ISSUER_ID }} | ||||||
|  |           IOS_ASC_KEY: ${{ steps.load-secrets.outputs.IOS_ASC_KEY }} | ||||||
|  |           MATCH_PASSWORD: ${{ steps.load-secrets.outputs.IOS_MATCH_PASSWORD }} | ||||||
| @@ -46,12 +46,17 @@ bundle exec fastlane <lane> | |||||||
| ``` | ``` | ||||||
| This is reused in the CI/CD pipeline to automate the deployment process. | This is reused in the CI/CD pipeline to automate the deployment process. | ||||||
|  |  | ||||||
| Fastlane assumes mutliple secrets to be present as files in the platform directories. These are: | Secrets used by fastlane are stored on hashicorp vault and are fetched by the CI/CD pipeline. See below. | ||||||
| - for android: |  | ||||||
|     - `secrets.properties` used by gradle to load secrets needed at execution time |  | ||||||
|     - `release.keystore` used by gradle to sign the apk |  | ||||||
|     - `google-key.json` used by fastlane to authenticate with the Google Play Store |  | ||||||
| - for ios: |  | ||||||
|     - TODO |  | ||||||
|  |  | ||||||
| These files are stored as secrets in the GitHub repository so that the CI pipeline can access them. | ## Secrets | ||||||
|  | These are mostly used by the CI/CD pipeline to deploy the application. The main usage for github actions is documented under [https://github.com/hashicorp/vault-action](https://github.com/hashicorp/vault-action). | ||||||
|  | **Global secrets** are used for both versions of the app (android and ios).  | ||||||
|  | - `GOOGLE_MAPS_API_KEY` is used to authenticate with the Google Maps API | ||||||
|  |  | ||||||
|  | **Platform-specific secrets** are used by the CI/CD pipeline to deploy to the respective app stores. | ||||||
|  | - `ANDROID_KEYSTORE` is used to sign the android apk | ||||||
|  | - `ANDROID_GOOGLE_KEY` is used to authenticate with the Google Play Store api | ||||||
|  | - `IOS_GOOGLE_...` | ||||||
|  | - `IOS_GOOGLE_...` | ||||||
|  | - `IOS_GOOGLE_...` | ||||||
|  | - `IOS_GOOGLE_...` | ||||||
							
								
								
									
										3
									
								
								frontend/android/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,3 +1,6 @@ | |||||||
|  | gradlew | ||||||
|  | gradlew.bat | ||||||
|  | gradle/ | ||||||
| /.gradle | /.gradle | ||||||
| /captures/ | /captures/ | ||||||
| /local.properties | /local.properties | ||||||
|   | |||||||
| @@ -63,11 +63,3 @@ Compared to the flutter template application, a few changes have to be made: | |||||||
|         } |         } | ||||||
|     ``` |     ``` | ||||||
|  |  | ||||||
|  |  | ||||||
| ### Using the credentials in CI |  | ||||||
| - Add the secret files to the repository secrets (e.g. `ANDROID_SECRETS_PROPERTIES`). |  | ||||||
|  |  | ||||||
| - temporarily write them back to files during the CI execution: |  | ||||||
|     ```bash |  | ||||||
|     echo {{ secrets.ANDROID_SECRETS }} >> android/secrets.properties |  | ||||||
|     ``` |  | ||||||
|   | |||||||
| @@ -30,14 +30,19 @@ if (flutterVersionName == null) { | |||||||
|  |  | ||||||
|  |  | ||||||
| def secretPropertiesFile = rootProject.file('secrets.properties') | def secretPropertiesFile = rootProject.file('secrets.properties') | ||||||
|  | def fallbackPropertiesFile = rootProject.file('fallback.properties') | ||||||
| def secretProperties = new Properties() | def secretProperties = new Properties() | ||||||
|  |  | ||||||
| if (secretPropertiesFile.exists()) { | if (secretPropertiesFile.exists()) { | ||||||
|     secretPropertiesFile.withReader('UTF-8') { reader -> |     secretPropertiesFile.withReader('UTF-8') { reader -> | ||||||
|         secretProperties.load(reader) |         secretProperties.load(reader) | ||||||
|     } |     } | ||||||
|  | } else if (fallbackPropertiesFile.exists()) { | ||||||
|  |     fallbackPropertiesFile.withReader('UTF-8') { reader -> | ||||||
|  |         secretProperties.load(reader) | ||||||
|  |     } | ||||||
| } else { | } else { | ||||||
|     throw new GradleException("Secrets file secrets.properties not found") |     throw new GradleException("Secrets file (secrets.properties, fallback.properties) not found") | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -60,7 +65,7 @@ android { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     defaultConfig { |     defaultConfig { | ||||||
|         // TODO: Specify your own unique Application ID (https://developer.android.com/studio/build/application-id.html). |    | ||||||
|         applicationId "com.anydev.anyway" |         applicationId "com.anydev.anyway" | ||||||
|         // You can update the following values to match your application needs. |         // You can update the following values to match your application needs. | ||||||
|         // For more information, see: https://docs.flutter.dev/deployment/android#reviewing-the-gradle-build-configuration. |         // For more information, see: https://docs.flutter.dev/deployment/android#reviewing-the-gradle-build-configuration. | ||||||
| @@ -72,7 +77,7 @@ android { | |||||||
|         versionCode flutterVersionCode.toInteger() |         versionCode flutterVersionCode.toInteger() | ||||||
|         versionName flutterVersionName |         versionName flutterVersionName | ||||||
|         // // Placeholders of keys that are replaced by the build system. |         // // Placeholders of keys that are replaced by the build system. | ||||||
|         manifestPlaceholders += ['MAPS_API_KEY': secretProperties.getProperty('MAPS_API_KEY')] |         manifestPlaceholders += ['MAPS_API_KEY': System.getenv('GOOGLE_MAPS_API_KEY')] | ||||||
|  |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1 +1,3 @@ | |||||||
|  | # This file mirrors the state of secrets.properties as a reference for the developer. | ||||||
|  | # And as a fallback for build.gradle | ||||||
| MAPS_API_KEY=Key | MAPS_API_KEY=Key | ||||||
| @@ -1,38 +1,52 @@ | |||||||
| # Uncomment the line if you want fastlane to automatically update itself |  | ||||||
| # update_fastlane |  | ||||||
|  |  | ||||||
| default_platform(:android) | default_platform(:android) | ||||||
|  |  | ||||||
| platform :android do | platform :android do | ||||||
|  |  | ||||||
|   desc "Deploy a new version as a preview version" |   desc "Deploy a new version to closed testing (play store)" | ||||||
|   lane :deploy_testing do |   lane :deploy_testing do | ||||||
|  |     build_name = ENV["BUILD_NAME"] | ||||||
|  |     build_number = ENV["BUILD_NUMBER"] | ||||||
|  |  | ||||||
|     sh( |     sh( | ||||||
|       "flutter build appbundle --release", |       "flutter", | ||||||
|       "--build-name=", |       "build", | ||||||
|       ENV["VERSION_NAME"], |       "appbundle", | ||||||
|  |       "--release", | ||||||
|  |       "--build-name=#{build_name}", | ||||||
|  |       "--build-number=#{build_number}", | ||||||
|       ) |       ) | ||||||
|      |      | ||||||
|     upload_to_play_store( |     upload_to_play_store( | ||||||
|       track: 'alpha', |       track: 'alpha', | ||||||
|       skip_upload_apk: true, |       skip_upload_apk: true, | ||||||
|       skip_upload_changelogs: true, |       skip_upload_changelogs: true, | ||||||
|  |       aab: "../build/app/outputs/bundle/release/app-release.aab", | ||||||
|  |       # this is the default output of flutter build ... --release | ||||||
|  |       # in particular this the build folder lies in the flutter root folder | ||||||
|  |       # this is the parent folder for the android folder | ||||||
|       ) |       ) | ||||||
|   end |   end | ||||||
|  |  | ||||||
|  |  | ||||||
|   desc "Deploy a new version as a full release" |   desc "Deploy a new version as a full release" | ||||||
|   lane :deploy_release do |   lane :deploy_release do | ||||||
|     gradle( |     build_name = ENV["BUILD_NAME"] | ||||||
|       task: "clean assembleRelease", |     build_number = ENV["BUILD_NUMBER"] | ||||||
|       properties: { |  | ||||||
|         # loaded from environment |     sh( | ||||||
|         "android.injected.version.name" => ENV["VERSION_NAME"], |       "flutter", | ||||||
|       } |       "build", | ||||||
|  |       "appbundle", | ||||||
|  |       "--release", | ||||||
|  |       "--build-name=#{build_name}", | ||||||
|  |       "--build-number=#{build_number}", | ||||||
|       ) |       ) | ||||||
|  |      | ||||||
|     upload_to_play_store( |     upload_to_play_store( | ||||||
|       track: "production", |       track: 'production', | ||||||
|       skip_upload_apk: true, |       skip_upload_apk: true, | ||||||
|       skip_upload_changelogs: true, |       skip_upload_changelogs: true, | ||||||
|     ) |       aab: "../build/app/outputs/bundle/release/app-release.aab", | ||||||
|  |       ) | ||||||
|   end |   end | ||||||
| end | end | ||||||
|   | |||||||
| Before Width: | Height: | Size: 106 KiB | 
| Before Width: | Height: | Size: 1.3 MiB | 
| Before Width: | Height: | Size: 637 KiB | 
| Before Width: | Height: | Size: 573 KiB | 
| Before Width: | Height: | Size: 175 KiB | 
| Before Width: | Height: | Size: 360 KiB | 
| Before Width: | Height: | Size: 106 KiB | 
| Before Width: | Height: | Size: 1.3 MiB | 
| Before Width: | Height: | Size: 637 KiB | 
| Before Width: | Height: | Size: 573 KiB | 
| Before Width: | Height: | Size: 175 KiB | 
| Before Width: | Height: | Size: 360 KiB | 
							
								
								
									
										6
									
								
								frontend/ios/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,3 +1,9 @@ | |||||||
|  | # fastlane secret | ||||||
|  | .env | ||||||
|  | secret.env | ||||||
|  | *.mobileprovision | ||||||
|  | report.xml | ||||||
|  |  | ||||||
| **/dgph | **/dgph | ||||||
| *.mode1v3 | *.mode1v3 | ||||||
| *.mode2v3 | *.mode2v3 | ||||||
|   | |||||||
| @@ -1 +1,2 @@ | |||||||
|  | #include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig" | ||||||
| #include "Generated.xcconfig" | #include "Generated.xcconfig" | ||||||
|   | |||||||
| @@ -1 +1,2 @@ | |||||||
|  | #include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig" | ||||||
| #include "Generated.xcconfig" | #include "Generated.xcconfig" | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								frontend/ios/Gemfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,5 @@ | |||||||
|  | source "https://rubygems.org" | ||||||
|  |  | ||||||
|  | gem "fastlane" | ||||||
|  | gem "cocoapods" | ||||||
|  |  | ||||||
							
								
								
									
										288
									
								
								frontend/ios/Gemfile.lock
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,288 @@ | |||||||
|  | GEM | ||||||
|  |   remote: https://rubygems.org/ | ||||||
|  |   specs: | ||||||
|  |     CFPropertyList (3.0.7) | ||||||
|  |       base64 | ||||||
|  |       nkf | ||||||
|  |       rexml | ||||||
|  |     activesupport (5.2.8.1) | ||||||
|  |       concurrent-ruby (~> 1.0, >= 1.0.2) | ||||||
|  |       i18n (>= 0.7, < 2) | ||||||
|  |       minitest (~> 5.1) | ||||||
|  |       tzinfo (~> 1.1) | ||||||
|  |     addressable (2.8.7) | ||||||
|  |       public_suffix (>= 2.0.2, < 7.0) | ||||||
|  |     algoliasearch (1.27.5) | ||||||
|  |       httpclient (~> 2.8, >= 2.8.3) | ||||||
|  |       json (>= 1.5.1) | ||||||
|  |     artifactory (3.0.17) | ||||||
|  |     atomos (0.1.3) | ||||||
|  |     aws-eventstream (1.3.0) | ||||||
|  |     aws-partitions (1.1004.0) | ||||||
|  |     aws-sdk-core (3.212.0) | ||||||
|  |       aws-eventstream (~> 1, >= 1.3.0) | ||||||
|  |       aws-partitions (~> 1, >= 1.992.0) | ||||||
|  |       aws-sigv4 (~> 1.9) | ||||||
|  |       jmespath (~> 1, >= 1.6.1) | ||||||
|  |     aws-sdk-kms (1.95.0) | ||||||
|  |       aws-sdk-core (~> 3, >= 3.210.0) | ||||||
|  |       aws-sigv4 (~> 1.5) | ||||||
|  |     aws-sdk-s3 (1.170.1) | ||||||
|  |       aws-sdk-core (~> 3, >= 3.210.0) | ||||||
|  |       aws-sdk-kms (~> 1) | ||||||
|  |       aws-sigv4 (~> 1.5) | ||||||
|  |     aws-sigv4 (1.10.1) | ||||||
|  |       aws-eventstream (~> 1, >= 1.0.2) | ||||||
|  |     babosa (1.0.4) | ||||||
|  |     base64 (0.2.0) | ||||||
|  |     claide (1.1.0) | ||||||
|  |     cocoapods (1.10.2) | ||||||
|  |       addressable (~> 2.6) | ||||||
|  |       claide (>= 1.0.2, < 2.0) | ||||||
|  |       cocoapods-core (= 1.10.2) | ||||||
|  |       cocoapods-deintegrate (>= 1.0.3, < 2.0) | ||||||
|  |       cocoapods-downloader (>= 1.4.0, < 2.0) | ||||||
|  |       cocoapods-plugins (>= 1.0.0, < 2.0) | ||||||
|  |       cocoapods-search (>= 1.0.0, < 2.0) | ||||||
|  |       cocoapods-trunk (>= 1.4.0, < 2.0) | ||||||
|  |       cocoapods-try (>= 1.1.0, < 2.0) | ||||||
|  |       colored2 (~> 3.1) | ||||||
|  |       escape (~> 0.0.4) | ||||||
|  |       fourflusher (>= 2.3.0, < 3.0) | ||||||
|  |       gh_inspector (~> 1.0) | ||||||
|  |       molinillo (~> 0.6.6) | ||||||
|  |       nap (~> 1.0) | ||||||
|  |       ruby-macho (~> 1.4) | ||||||
|  |       xcodeproj (>= 1.19.0, < 2.0) | ||||||
|  |     cocoapods-core (1.10.2) | ||||||
|  |       activesupport (> 5.0, < 6) | ||||||
|  |       addressable (~> 2.6) | ||||||
|  |       algoliasearch (~> 1.0) | ||||||
|  |       concurrent-ruby (~> 1.1) | ||||||
|  |       fuzzy_match (~> 2.0.4) | ||||||
|  |       nap (~> 1.0) | ||||||
|  |       netrc (~> 0.11) | ||||||
|  |       public_suffix | ||||||
|  |       typhoeus (~> 1.0) | ||||||
|  |     cocoapods-deintegrate (1.0.5) | ||||||
|  |     cocoapods-downloader (1.6.3) | ||||||
|  |     cocoapods-plugins (1.0.0) | ||||||
|  |       nap | ||||||
|  |     cocoapods-search (1.0.1) | ||||||
|  |     cocoapods-trunk (1.6.0) | ||||||
|  |       nap (>= 0.8, < 2.0) | ||||||
|  |       netrc (~> 0.11) | ||||||
|  |     cocoapods-try (1.2.0) | ||||||
|  |     colored (1.2) | ||||||
|  |     colored2 (3.1.2) | ||||||
|  |     commander (4.6.0) | ||||||
|  |       highline (~> 2.0.0) | ||||||
|  |     concurrent-ruby (1.3.4) | ||||||
|  |     declarative (0.0.20) | ||||||
|  |     digest-crc (0.6.5) | ||||||
|  |       rake (>= 12.0.0, < 14.0.0) | ||||||
|  |     domain_name (0.6.20240107) | ||||||
|  |     dotenv (2.8.1) | ||||||
|  |     emoji_regex (3.2.3) | ||||||
|  |     escape (0.0.4) | ||||||
|  |     ethon (0.16.0) | ||||||
|  |       ffi (>= 1.15.0) | ||||||
|  |     excon (0.112.0) | ||||||
|  |     faraday (1.10.4) | ||||||
|  |       faraday-em_http (~> 1.0) | ||||||
|  |       faraday-em_synchrony (~> 1.0) | ||||||
|  |       faraday-excon (~> 1.1) | ||||||
|  |       faraday-httpclient (~> 1.0) | ||||||
|  |       faraday-multipart (~> 1.0) | ||||||
|  |       faraday-net_http (~> 1.0) | ||||||
|  |       faraday-net_http_persistent (~> 1.0) | ||||||
|  |       faraday-patron (~> 1.0) | ||||||
|  |       faraday-rack (~> 1.0) | ||||||
|  |       faraday-retry (~> 1.0) | ||||||
|  |       ruby2_keywords (>= 0.0.4) | ||||||
|  |     faraday-cookie_jar (0.0.7) | ||||||
|  |       faraday (>= 0.8.0) | ||||||
|  |       http-cookie (~> 1.0.0) | ||||||
|  |     faraday-em_http (1.0.0) | ||||||
|  |     faraday-em_synchrony (1.0.0) | ||||||
|  |     faraday-excon (1.1.0) | ||||||
|  |     faraday-httpclient (1.0.1) | ||||||
|  |     faraday-multipart (1.0.4) | ||||||
|  |       multipart-post (~> 2) | ||||||
|  |     faraday-net_http (1.0.2) | ||||||
|  |     faraday-net_http_persistent (1.2.0) | ||||||
|  |     faraday-patron (1.0.0) | ||||||
|  |     faraday-rack (1.0.0) | ||||||
|  |     faraday-retry (1.0.3) | ||||||
|  |     faraday_middleware (1.2.1) | ||||||
|  |       faraday (~> 1.0) | ||||||
|  |     fastimage (2.3.1) | ||||||
|  |     fastlane (2.225.0) | ||||||
|  |       CFPropertyList (>= 2.3, < 4.0.0) | ||||||
|  |       addressable (>= 2.8, < 3.0.0) | ||||||
|  |       artifactory (~> 3.0) | ||||||
|  |       aws-sdk-s3 (~> 1.0) | ||||||
|  |       babosa (>= 1.0.3, < 2.0.0) | ||||||
|  |       bundler (>= 1.12.0, < 3.0.0) | ||||||
|  |       colored (~> 1.2) | ||||||
|  |       commander (~> 4.6) | ||||||
|  |       dotenv (>= 2.1.1, < 3.0.0) | ||||||
|  |       emoji_regex (>= 0.1, < 4.0) | ||||||
|  |       excon (>= 0.71.0, < 1.0.0) | ||||||
|  |       faraday (~> 1.0) | ||||||
|  |       faraday-cookie_jar (~> 0.0.6) | ||||||
|  |       faraday_middleware (~> 1.0) | ||||||
|  |       fastimage (>= 2.1.0, < 3.0.0) | ||||||
|  |       fastlane-sirp (>= 1.0.0) | ||||||
|  |       gh_inspector (>= 1.1.2, < 2.0.0) | ||||||
|  |       google-apis-androidpublisher_v3 (~> 0.3) | ||||||
|  |       google-apis-playcustomapp_v1 (~> 0.1) | ||||||
|  |       google-cloud-env (>= 1.6.0, < 2.0.0) | ||||||
|  |       google-cloud-storage (~> 1.31) | ||||||
|  |       highline (~> 2.0) | ||||||
|  |       http-cookie (~> 1.0.5) | ||||||
|  |       json (< 3.0.0) | ||||||
|  |       jwt (>= 2.1.0, < 3) | ||||||
|  |       mini_magick (>= 4.9.4, < 5.0.0) | ||||||
|  |       multipart-post (>= 2.0.0, < 3.0.0) | ||||||
|  |       naturally (~> 2.2) | ||||||
|  |       optparse (>= 0.1.1, < 1.0.0) | ||||||
|  |       plist (>= 3.1.0, < 4.0.0) | ||||||
|  |       rubyzip (>= 2.0.0, < 3.0.0) | ||||||
|  |       security (= 0.1.5) | ||||||
|  |       simctl (~> 1.6.3) | ||||||
|  |       terminal-notifier (>= 2.0.0, < 3.0.0) | ||||||
|  |       terminal-table (~> 3) | ||||||
|  |       tty-screen (>= 0.6.3, < 1.0.0) | ||||||
|  |       tty-spinner (>= 0.8.0, < 1.0.0) | ||||||
|  |       word_wrap (~> 1.0.0) | ||||||
|  |       xcodeproj (>= 1.13.0, < 2.0.0) | ||||||
|  |       xcpretty (~> 0.3.0) | ||||||
|  |       xcpretty-travis-formatter (>= 0.0.3, < 2.0.0) | ||||||
|  |     fastlane-sirp (1.0.0) | ||||||
|  |       sysrandom (~> 1.0) | ||||||
|  |     ffi (1.17.0) | ||||||
|  |     ffi (1.17.0-x86_64-darwin) | ||||||
|  |     fourflusher (2.3.1) | ||||||
|  |     fuzzy_match (2.0.4) | ||||||
|  |     gh_inspector (1.1.3) | ||||||
|  |     google-apis-androidpublisher_v3 (0.54.0) | ||||||
|  |       google-apis-core (>= 0.11.0, < 2.a) | ||||||
|  |     google-apis-core (0.11.3) | ||||||
|  |       addressable (~> 2.5, >= 2.5.1) | ||||||
|  |       googleauth (>= 0.16.2, < 2.a) | ||||||
|  |       httpclient (>= 2.8.1, < 3.a) | ||||||
|  |       mini_mime (~> 1.0) | ||||||
|  |       representable (~> 3.0) | ||||||
|  |       retriable (>= 2.0, < 4.a) | ||||||
|  |       rexml | ||||||
|  |     google-apis-iamcredentials_v1 (0.17.0) | ||||||
|  |       google-apis-core (>= 0.11.0, < 2.a) | ||||||
|  |     google-apis-playcustomapp_v1 (0.13.0) | ||||||
|  |       google-apis-core (>= 0.11.0, < 2.a) | ||||||
|  |     google-apis-storage_v1 (0.31.0) | ||||||
|  |       google-apis-core (>= 0.11.0, < 2.a) | ||||||
|  |     google-cloud-core (1.7.1) | ||||||
|  |       google-cloud-env (>= 1.0, < 3.a) | ||||||
|  |       google-cloud-errors (~> 1.0) | ||||||
|  |     google-cloud-env (1.6.0) | ||||||
|  |       faraday (>= 0.17.3, < 3.0) | ||||||
|  |     google-cloud-errors (1.4.0) | ||||||
|  |     google-cloud-storage (1.47.0) | ||||||
|  |       addressable (~> 2.8) | ||||||
|  |       digest-crc (~> 0.4) | ||||||
|  |       google-apis-iamcredentials_v1 (~> 0.1) | ||||||
|  |       google-apis-storage_v1 (~> 0.31.0) | ||||||
|  |       google-cloud-core (~> 1.6) | ||||||
|  |       googleauth (>= 0.16.2, < 2.a) | ||||||
|  |       mini_mime (~> 1.0) | ||||||
|  |     googleauth (1.8.1) | ||||||
|  |       faraday (>= 0.17.3, < 3.a) | ||||||
|  |       jwt (>= 1.4, < 3.0) | ||||||
|  |       multi_json (~> 1.11) | ||||||
|  |       os (>= 0.9, < 2.0) | ||||||
|  |       signet (>= 0.16, < 2.a) | ||||||
|  |     highline (2.0.3) | ||||||
|  |     http-cookie (1.0.7) | ||||||
|  |       domain_name (~> 0.5) | ||||||
|  |     httpclient (2.8.3) | ||||||
|  |     i18n (1.14.6) | ||||||
|  |       concurrent-ruby (~> 1.0) | ||||||
|  |     jmespath (1.6.2) | ||||||
|  |     json (2.8.1) | ||||||
|  |     jwt (2.9.3) | ||||||
|  |       base64 | ||||||
|  |     mini_magick (4.13.2) | ||||||
|  |     mini_mime (1.1.5) | ||||||
|  |     minitest (5.25.1) | ||||||
|  |     molinillo (0.6.6) | ||||||
|  |     multi_json (1.15.0) | ||||||
|  |     multipart-post (2.4.1) | ||||||
|  |     nanaimo (0.4.0) | ||||||
|  |     nap (1.1.0) | ||||||
|  |     naturally (2.2.1) | ||||||
|  |     netrc (0.11.0) | ||||||
|  |     nkf (0.2.0) | ||||||
|  |     optparse (0.6.0) | ||||||
|  |     os (1.1.4) | ||||||
|  |     plist (3.7.1) | ||||||
|  |     public_suffix (6.0.1) | ||||||
|  |     rake (13.2.1) | ||||||
|  |     representable (3.2.0) | ||||||
|  |       declarative (< 0.1.0) | ||||||
|  |       trailblazer-option (>= 0.1.1, < 0.2.0) | ||||||
|  |       uber (< 0.2.0) | ||||||
|  |     retriable (3.1.2) | ||||||
|  |     rexml (3.3.9) | ||||||
|  |     rouge (2.0.7) | ||||||
|  |     ruby-macho (1.4.0) | ||||||
|  |     ruby2_keywords (0.0.5) | ||||||
|  |     rubyzip (2.3.2) | ||||||
|  |     security (0.1.5) | ||||||
|  |     signet (0.19.0) | ||||||
|  |       addressable (~> 2.8) | ||||||
|  |       faraday (>= 0.17.5, < 3.a) | ||||||
|  |       jwt (>= 1.5, < 3.0) | ||||||
|  |       multi_json (~> 1.10) | ||||||
|  |     simctl (1.6.10) | ||||||
|  |       CFPropertyList | ||||||
|  |       naturally | ||||||
|  |     sysrandom (1.0.5) | ||||||
|  |     terminal-notifier (2.0.0) | ||||||
|  |     terminal-table (3.0.2) | ||||||
|  |       unicode-display_width (>= 1.1.1, < 3) | ||||||
|  |     thread_safe (0.3.6) | ||||||
|  |     trailblazer-option (0.1.2) | ||||||
|  |     tty-cursor (0.7.1) | ||||||
|  |     tty-screen (0.8.2) | ||||||
|  |     tty-spinner (0.9.3) | ||||||
|  |       tty-cursor (~> 0.7) | ||||||
|  |     typhoeus (1.4.1) | ||||||
|  |       ethon (>= 0.9.0) | ||||||
|  |     tzinfo (1.2.11) | ||||||
|  |       thread_safe (~> 0.1) | ||||||
|  |     uber (0.1.0) | ||||||
|  |     unicode-display_width (2.6.0) | ||||||
|  |     word_wrap (1.0.0) | ||||||
|  |     xcodeproj (1.27.0) | ||||||
|  |       CFPropertyList (>= 2.3.3, < 4.0) | ||||||
|  |       atomos (~> 0.1.3) | ||||||
|  |       claide (>= 1.0.2, < 2.0) | ||||||
|  |       colored2 (~> 3.1) | ||||||
|  |       nanaimo (~> 0.4.0) | ||||||
|  |       rexml (>= 3.3.6, < 4.0) | ||||||
|  |     xcpretty (0.3.0) | ||||||
|  |       rouge (~> 2.0.7) | ||||||
|  |     xcpretty-travis-formatter (1.0.1) | ||||||
|  |       xcpretty (~> 0.2, >= 0.0.7) | ||||||
|  |  | ||||||
|  | PLATFORMS | ||||||
|  |   ruby | ||||||
|  |   x86_64-darwin-23 | ||||||
|  |  | ||||||
|  | DEPENDENCIES | ||||||
|  |   cocoapods | ||||||
|  |   fastlane | ||||||
|  |  | ||||||
|  | BUNDLED WITH | ||||||
|  |    2.5.23 | ||||||
							
								
								
									
										44
									
								
								frontend/ios/Podfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,44 @@ | |||||||
|  | # Uncomment this line to define a global platform for your project | ||||||
|  | # platform :ios, '12.0' | ||||||
|  |  | ||||||
|  | # CocoaPods analytics sends network stats synchronously affecting flutter build latency. | ||||||
|  | ENV['COCOAPODS_DISABLE_STATS'] = 'true' | ||||||
|  |  | ||||||
|  | project 'Runner', { | ||||||
|  |   'Debug' => :debug, | ||||||
|  |   'Profile' => :release, | ||||||
|  |   'Release' => :release, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | def flutter_root | ||||||
|  |   generated_xcode_build_settings_path = File.expand_path(File.join('..', 'Flutter', 'Generated.xcconfig'), __FILE__) | ||||||
|  |   unless File.exist?(generated_xcode_build_settings_path) | ||||||
|  |     raise "#{generated_xcode_build_settings_path} must exist. If you're running pod install manually, make sure flutter pub get is executed first" | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   File.foreach(generated_xcode_build_settings_path) do |line| | ||||||
|  |     matches = line.match(/FLUTTER_ROOT\=(.*)/) | ||||||
|  |     return matches[1].strip if matches | ||||||
|  |   end | ||||||
|  |   raise "FLUTTER_ROOT not found in #{generated_xcode_build_settings_path}. Try deleting Generated.xcconfig, then run flutter pub get" | ||||||
|  | end | ||||||
|  |  | ||||||
|  | require File.expand_path(File.join('packages', 'flutter_tools', 'bin', 'podhelper'), flutter_root) | ||||||
|  |  | ||||||
|  | flutter_ios_podfile_setup | ||||||
|  |  | ||||||
|  | target 'Runner' do | ||||||
|  |   use_frameworks! | ||||||
|  |   use_modular_headers! | ||||||
|  |  | ||||||
|  |   flutter_install_all_ios_pods File.dirname(File.realpath(__FILE__)) | ||||||
|  |   target 'RunnerTests' do | ||||||
|  |     inherit! :search_paths | ||||||
|  |   end | ||||||
|  | end | ||||||
|  |  | ||||||
|  | post_install do |installer| | ||||||
|  |   installer.pods_project.targets.each do |target| | ||||||
|  |     flutter_additional_ios_build_settings(target) | ||||||
|  |   end | ||||||
|  | end | ||||||
							
								
								
									
										89
									
								
								frontend/ios/Podfile.lock
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,89 @@ | |||||||
|  | PODS: | ||||||
|  |   - Flutter (1.0.0) | ||||||
|  |   - geocoding_ios (1.0.5): | ||||||
|  |     - Flutter | ||||||
|  |   - geolocator_apple (1.2.0): | ||||||
|  |     - Flutter | ||||||
|  |   - Google-Maps-iOS-Utils (6.0.0): | ||||||
|  |     - GoogleMaps (~> 9.0) | ||||||
|  |   - google_maps_flutter_ios (0.0.1): | ||||||
|  |     - Flutter | ||||||
|  |     - Google-Maps-iOS-Utils (< 7.0, >= 5.0) | ||||||
|  |     - GoogleMaps (< 10.0, >= 8.4) | ||||||
|  |   - GoogleMaps (9.1.1): | ||||||
|  |     - GoogleMaps/Maps (= 9.1.1) | ||||||
|  |   - GoogleMaps/Base (9.1.1) | ||||||
|  |   - GoogleMaps/Maps (9.1.1): | ||||||
|  |     - GoogleMaps/Base | ||||||
|  |   - map_launcher (0.0.1): | ||||||
|  |     - Flutter | ||||||
|  |   - path_provider_foundation (0.0.1): | ||||||
|  |     - Flutter | ||||||
|  |     - FlutterMacOS | ||||||
|  |   - permission_handler_apple (9.3.0): | ||||||
|  |     - Flutter | ||||||
|  |   - shared_preferences_foundation (0.0.1): | ||||||
|  |     - Flutter | ||||||
|  |     - FlutterMacOS | ||||||
|  |   - sqflite (0.0.3): | ||||||
|  |     - Flutter | ||||||
|  |     - FlutterMacOS | ||||||
|  |   - url_launcher_ios (0.0.1): | ||||||
|  |     - Flutter | ||||||
|  |  | ||||||
|  | DEPENDENCIES: | ||||||
|  |   - Flutter (from `Flutter`) | ||||||
|  |   - geocoding_ios (from `.symlinks/plugins/geocoding_ios/ios`) | ||||||
|  |   - geolocator_apple (from `.symlinks/plugins/geolocator_apple/ios`) | ||||||
|  |   - google_maps_flutter_ios (from `.symlinks/plugins/google_maps_flutter_ios/ios`) | ||||||
|  |   - map_launcher (from `.symlinks/plugins/map_launcher/ios`) | ||||||
|  |   - path_provider_foundation (from `.symlinks/plugins/path_provider_foundation/darwin`) | ||||||
|  |   - permission_handler_apple (from `.symlinks/plugins/permission_handler_apple/ios`) | ||||||
|  |   - shared_preferences_foundation (from `.symlinks/plugins/shared_preferences_foundation/darwin`) | ||||||
|  |   - sqflite (from `.symlinks/plugins/sqflite/darwin`) | ||||||
|  |   - url_launcher_ios (from `.symlinks/plugins/url_launcher_ios/ios`) | ||||||
|  |  | ||||||
|  | SPEC REPOS: | ||||||
|  |   trunk: | ||||||
|  |     - Google-Maps-iOS-Utils | ||||||
|  |     - GoogleMaps | ||||||
|  |  | ||||||
|  | EXTERNAL SOURCES: | ||||||
|  |   Flutter: | ||||||
|  |     :path: Flutter | ||||||
|  |   geocoding_ios: | ||||||
|  |     :path: ".symlinks/plugins/geocoding_ios/ios" | ||||||
|  |   geolocator_apple: | ||||||
|  |     :path: ".symlinks/plugins/geolocator_apple/ios" | ||||||
|  |   google_maps_flutter_ios: | ||||||
|  |     :path: ".symlinks/plugins/google_maps_flutter_ios/ios" | ||||||
|  |   map_launcher: | ||||||
|  |     :path: ".symlinks/plugins/map_launcher/ios" | ||||||
|  |   path_provider_foundation: | ||||||
|  |     :path: ".symlinks/plugins/path_provider_foundation/darwin" | ||||||
|  |   permission_handler_apple: | ||||||
|  |     :path: ".symlinks/plugins/permission_handler_apple/ios" | ||||||
|  |   shared_preferences_foundation: | ||||||
|  |     :path: ".symlinks/plugins/shared_preferences_foundation/darwin" | ||||||
|  |   sqflite: | ||||||
|  |     :path: ".symlinks/plugins/sqflite/darwin" | ||||||
|  |   url_launcher_ios: | ||||||
|  |     :path: ".symlinks/plugins/url_launcher_ios/ios" | ||||||
|  |  | ||||||
|  | SPEC CHECKSUMS: | ||||||
|  |   Flutter: e0871f40cf51350855a761d2e70bf5af5b9b5de7 | ||||||
|  |   geocoding_ios: bcbdaa6bddd7d3129c9bcb8acddc5d8778689768 | ||||||
|  |   geolocator_apple: d981750b9f47dbdb02427e1476d9a04397beb8d9 | ||||||
|  |   Google-Maps-iOS-Utils: cfe6a0239c7ca634b7e001ad059a6707143dc8dc | ||||||
|  |   google_maps_flutter_ios: 0291eb2aa252298a769b04d075e4a9d747ff7264 | ||||||
|  |   GoogleMaps: 80ea184ed6bf44139f383a8b0e248ba3ec1cc8c9 | ||||||
|  |   map_launcher: fe43bda6720bb73c12fcc1bdd86123ff49a4d4d6 | ||||||
|  |   path_provider_foundation: 080d55be775b7414fd5a5ef3ac137b97b097e564 | ||||||
|  |   permission_handler_apple: 4ed2196e43d0651e8ff7ca3483a069d469701f2d | ||||||
|  |   shared_preferences_foundation: 9e1978ff2562383bd5676f64ec4e9aa8fa06a6f7 | ||||||
|  |   sqflite: c35dad70033b8862124f8337cc994a809fcd9fa3 | ||||||
|  |   url_launcher_ios: 694010445543906933d732453a59da0a173ae33d | ||||||
|  |  | ||||||
|  | PODFILE CHECKSUM: 819463e6a0290f5a72f145ba7cde16e8b6ef0796 | ||||||
|  |  | ||||||
|  | COCOAPODS: 1.10.2 | ||||||
| @@ -11,9 +11,11 @@ | |||||||
| 		331C808B294A63AB00263BE5 /* RunnerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 331C807B294A618700263BE5 /* RunnerTests.swift */; }; | 		331C808B294A63AB00263BE5 /* RunnerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 331C807B294A618700263BE5 /* RunnerTests.swift */; }; | ||||||
| 		3B3967161E833CAA004F5970 /* AppFrameworkInfo.plist in Resources */ = {isa = PBXBuildFile; fileRef = 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */; }; | 		3B3967161E833CAA004F5970 /* AppFrameworkInfo.plist in Resources */ = {isa = PBXBuildFile; fileRef = 3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */; }; | ||||||
| 		74858FAF1ED2DC5600515810 /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 74858FAE1ED2DC5600515810 /* AppDelegate.swift */; }; | 		74858FAF1ED2DC5600515810 /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 74858FAE1ED2DC5600515810 /* AppDelegate.swift */; }; | ||||||
|  | 		8F724AF5AC92A8A68D89C67E /* Pods_Runner.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 03CCEF89D4BD42ADA86AEDF9 /* Pods_Runner.framework */; }; | ||||||
| 		97C146FC1CF9000F007C117D /* Main.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FA1CF9000F007C117D /* Main.storyboard */; }; | 		97C146FC1CF9000F007C117D /* Main.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FA1CF9000F007C117D /* Main.storyboard */; }; | ||||||
| 		97C146FE1CF9000F007C117D /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FD1CF9000F007C117D /* Assets.xcassets */; }; | 		97C146FE1CF9000F007C117D /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FD1CF9000F007C117D /* Assets.xcassets */; }; | ||||||
| 		97C147011CF9000F007C117D /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */; }; | 		97C147011CF9000F007C117D /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 97C146FF1CF9000F007C117D /* LaunchScreen.storyboard */; }; | ||||||
|  | 		CDD1C9EB82AEC89C2181F722 /* Pods_RunnerTests.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 4CB8B4133CEB7949B7EEBD81 /* Pods_RunnerTests.framework */; }; | ||||||
| /* End PBXBuildFile section */ | /* End PBXBuildFile section */ | ||||||
|  |  | ||||||
| /* Begin PBXContainerItemProxy section */ | /* Begin PBXContainerItemProxy section */ | ||||||
| @@ -40,14 +42,20 @@ | |||||||
| /* End PBXCopyFilesBuildPhase section */ | /* End PBXCopyFilesBuildPhase section */ | ||||||
|  |  | ||||||
| /* Begin PBXFileReference section */ | /* Begin PBXFileReference section */ | ||||||
|  | 		03CCEF89D4BD42ADA86AEDF9 /* Pods_Runner.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Pods_Runner.framework; sourceTree = BUILT_PRODUCTS_DIR; }; | ||||||
| 		1498D2321E8E86230040F4C2 /* GeneratedPluginRegistrant.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = GeneratedPluginRegistrant.h; sourceTree = "<group>"; }; | 		1498D2321E8E86230040F4C2 /* GeneratedPluginRegistrant.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = GeneratedPluginRegistrant.h; sourceTree = "<group>"; }; | ||||||
| 		1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = GeneratedPluginRegistrant.m; sourceTree = "<group>"; }; | 		1498D2331E8E89220040F4C2 /* GeneratedPluginRegistrant.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = GeneratedPluginRegistrant.m; sourceTree = "<group>"; }; | ||||||
|  | 		282EA28E78AB3F765E4BA719 /* Pods-RunnerTests.profile.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.profile.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.profile.xcconfig"; sourceTree = "<group>"; }; | ||||||
|  | 		3023467726A2A8275ED51C3E /* Pods-Runner.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.debug.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig"; sourceTree = "<group>"; }; | ||||||
| 		331C807B294A618700263BE5 /* RunnerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RunnerTests.swift; sourceTree = "<group>"; }; | 		331C807B294A618700263BE5 /* RunnerTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RunnerTests.swift; sourceTree = "<group>"; }; | ||||||
| 		331C8081294A63A400263BE5 /* RunnerTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = RunnerTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; | 		331C8081294A63A400263BE5 /* RunnerTests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = RunnerTests.xctest; sourceTree = BUILT_PRODUCTS_DIR; }; | ||||||
| 		3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = AppFrameworkInfo.plist; path = Flutter/AppFrameworkInfo.plist; sourceTree = "<group>"; }; | 		3B3967151E833CAA004F5970 /* AppFrameworkInfo.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = AppFrameworkInfo.plist; path = Flutter/AppFrameworkInfo.plist; sourceTree = "<group>"; }; | ||||||
|  | 		4CB8B4133CEB7949B7EEBD81 /* Pods_RunnerTests.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Pods_RunnerTests.framework; sourceTree = BUILT_PRODUCTS_DIR; }; | ||||||
|  | 		5F8BB7E700693DEAB89BBE69 /* Pods-Runner.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.release.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig"; sourceTree = "<group>"; }; | ||||||
| 		74858FAD1ED2DC5600515810 /* Runner-Bridging-Header.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "Runner-Bridging-Header.h"; sourceTree = "<group>"; }; | 		74858FAD1ED2DC5600515810 /* Runner-Bridging-Header.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; path = "Runner-Bridging-Header.h"; sourceTree = "<group>"; }; | ||||||
| 		74858FAE1ED2DC5600515810 /* AppDelegate.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = "<group>"; }; | 		74858FAE1ED2DC5600515810 /* AppDelegate.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = "<group>"; }; | ||||||
| 		7AFA3C8E1D35360C0083082E /* Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; name = Release.xcconfig; path = Flutter/Release.xcconfig; sourceTree = "<group>"; }; | 		7AFA3C8E1D35360C0083082E /* Release.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; name = Release.xcconfig; path = Flutter/Release.xcconfig; sourceTree = "<group>"; }; | ||||||
|  | 		7B8A81C772249160491754F9 /* Pods-Runner.profile.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-Runner.profile.xcconfig"; path = "Target Support Files/Pods-Runner/Pods-Runner.profile.xcconfig"; sourceTree = "<group>"; }; | ||||||
| 		9740EEB21CF90195004384FC /* Debug.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Debug.xcconfig; path = Flutter/Debug.xcconfig; sourceTree = "<group>"; }; | 		9740EEB21CF90195004384FC /* Debug.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Debug.xcconfig; path = Flutter/Debug.xcconfig; sourceTree = "<group>"; }; | ||||||
| 		9740EEB31CF90195004384FC /* Generated.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Generated.xcconfig; path = Flutter/Generated.xcconfig; sourceTree = "<group>"; }; | 		9740EEB31CF90195004384FC /* Generated.xcconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xcconfig; name = Generated.xcconfig; path = Flutter/Generated.xcconfig; sourceTree = "<group>"; }; | ||||||
| 		97C146EE1CF9000F007C117D /* Runner.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Runner.app; sourceTree = BUILT_PRODUCTS_DIR; }; | 		97C146EE1CF9000F007C117D /* Runner.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = Runner.app; sourceTree = BUILT_PRODUCTS_DIR; }; | ||||||
| @@ -55,19 +63,43 @@ | |||||||
| 		97C146FD1CF9000F007C117D /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = "<group>"; }; | 		97C146FD1CF9000F007C117D /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = "<group>"; }; | ||||||
| 		97C147001CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = "<group>"; }; | 		97C147001CF9000F007C117D /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = "<group>"; }; | ||||||
| 		97C147021CF9000F007C117D /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; }; | 		97C147021CF9000F007C117D /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; }; | ||||||
|  | 		A565AAB9FE158487ABF3A5BF /* Pods-RunnerTests.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.release.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.release.xcconfig"; sourceTree = "<group>"; }; | ||||||
|  | 		DC475F5210027479529644C3 /* Pods-RunnerTests.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RunnerTests.debug.xcconfig"; path = "Target Support Files/Pods-RunnerTests/Pods-RunnerTests.debug.xcconfig"; sourceTree = "<group>"; }; | ||||||
| /* End PBXFileReference section */ | /* End PBXFileReference section */ | ||||||
|  |  | ||||||
| /* Begin PBXFrameworksBuildPhase section */ | /* Begin PBXFrameworksBuildPhase section */ | ||||||
|  | 		03EC59CC2AABC9D86B4ABFD7 /* Frameworks */ = { | ||||||
|  | 			isa = PBXFrameworksBuildPhase; | ||||||
|  | 			buildActionMask = 2147483647; | ||||||
|  | 			files = ( | ||||||
|  | 				CDD1C9EB82AEC89C2181F722 /* Pods_RunnerTests.framework in Frameworks */, | ||||||
|  | 			); | ||||||
|  | 			runOnlyForDeploymentPostprocessing = 0; | ||||||
|  | 		}; | ||||||
| 		97C146EB1CF9000F007C117D /* Frameworks */ = { | 		97C146EB1CF9000F007C117D /* Frameworks */ = { | ||||||
| 			isa = PBXFrameworksBuildPhase; | 			isa = PBXFrameworksBuildPhase; | ||||||
| 			buildActionMask = 2147483647; | 			buildActionMask = 2147483647; | ||||||
| 			files = ( | 			files = ( | ||||||
|  | 				8F724AF5AC92A8A68D89C67E /* Pods_Runner.framework in Frameworks */, | ||||||
| 			); | 			); | ||||||
| 			runOnlyForDeploymentPostprocessing = 0; | 			runOnlyForDeploymentPostprocessing = 0; | ||||||
| 		}; | 		}; | ||||||
| /* End PBXFrameworksBuildPhase section */ | /* End PBXFrameworksBuildPhase section */ | ||||||
|  |  | ||||||
| /* Begin PBXGroup section */ | /* Begin PBXGroup section */ | ||||||
|  | 		1C946B8D83A95663C2489C91 /* Pods */ = { | ||||||
|  | 			isa = PBXGroup; | ||||||
|  | 			children = ( | ||||||
|  | 				3023467726A2A8275ED51C3E /* Pods-Runner.debug.xcconfig */, | ||||||
|  | 				5F8BB7E700693DEAB89BBE69 /* Pods-Runner.release.xcconfig */, | ||||||
|  | 				7B8A81C772249160491754F9 /* Pods-Runner.profile.xcconfig */, | ||||||
|  | 				DC475F5210027479529644C3 /* Pods-RunnerTests.debug.xcconfig */, | ||||||
|  | 				A565AAB9FE158487ABF3A5BF /* Pods-RunnerTests.release.xcconfig */, | ||||||
|  | 				282EA28E78AB3F765E4BA719 /* Pods-RunnerTests.profile.xcconfig */, | ||||||
|  | 			); | ||||||
|  | 			path = Pods; | ||||||
|  | 			sourceTree = "<group>"; | ||||||
|  | 		}; | ||||||
| 		331C8082294A63A400263BE5 /* RunnerTests */ = { | 		331C8082294A63A400263BE5 /* RunnerTests */ = { | ||||||
| 			isa = PBXGroup; | 			isa = PBXGroup; | ||||||
| 			children = ( | 			children = ( | ||||||
| @@ -76,6 +108,15 @@ | |||||||
| 			path = RunnerTests; | 			path = RunnerTests; | ||||||
| 			sourceTree = "<group>"; | 			sourceTree = "<group>"; | ||||||
| 		}; | 		}; | ||||||
|  | 		3ECCC9BD7D0792871219624C /* Frameworks */ = { | ||||||
|  | 			isa = PBXGroup; | ||||||
|  | 			children = ( | ||||||
|  | 				03CCEF89D4BD42ADA86AEDF9 /* Pods_Runner.framework */, | ||||||
|  | 				4CB8B4133CEB7949B7EEBD81 /* Pods_RunnerTests.framework */, | ||||||
|  | 			); | ||||||
|  | 			name = Frameworks; | ||||||
|  | 			sourceTree = "<group>"; | ||||||
|  | 		}; | ||||||
| 		9740EEB11CF90186004384FC /* Flutter */ = { | 		9740EEB11CF90186004384FC /* Flutter */ = { | ||||||
| 			isa = PBXGroup; | 			isa = PBXGroup; | ||||||
| 			children = ( | 			children = ( | ||||||
| @@ -94,6 +135,8 @@ | |||||||
| 				97C146F01CF9000F007C117D /* Runner */, | 				97C146F01CF9000F007C117D /* Runner */, | ||||||
| 				97C146EF1CF9000F007C117D /* Products */, | 				97C146EF1CF9000F007C117D /* Products */, | ||||||
| 				331C8082294A63A400263BE5 /* RunnerTests */, | 				331C8082294A63A400263BE5 /* RunnerTests */, | ||||||
|  | 				1C946B8D83A95663C2489C91 /* Pods */, | ||||||
|  | 				3ECCC9BD7D0792871219624C /* Frameworks */, | ||||||
| 			); | 			); | ||||||
| 			sourceTree = "<group>"; | 			sourceTree = "<group>"; | ||||||
| 		}; | 		}; | ||||||
| @@ -128,8 +171,10 @@ | |||||||
| 			isa = PBXNativeTarget; | 			isa = PBXNativeTarget; | ||||||
| 			buildConfigurationList = 331C8087294A63A400263BE5 /* Build configuration list for PBXNativeTarget "RunnerTests" */; | 			buildConfigurationList = 331C8087294A63A400263BE5 /* Build configuration list for PBXNativeTarget "RunnerTests" */; | ||||||
| 			buildPhases = ( | 			buildPhases = ( | ||||||
|  | 				F27C1B361CA1B045C8D36B3B /* [CP] Check Pods Manifest.lock */, | ||||||
| 				331C807D294A63A400263BE5 /* Sources */, | 				331C807D294A63A400263BE5 /* Sources */, | ||||||
| 				331C807F294A63A400263BE5 /* Resources */, | 				331C807F294A63A400263BE5 /* Resources */, | ||||||
|  | 				03EC59CC2AABC9D86B4ABFD7 /* Frameworks */, | ||||||
| 			); | 			); | ||||||
| 			buildRules = ( | 			buildRules = ( | ||||||
| 			); | 			); | ||||||
| @@ -145,12 +190,15 @@ | |||||||
| 			isa = PBXNativeTarget; | 			isa = PBXNativeTarget; | ||||||
| 			buildConfigurationList = 97C147051CF9000F007C117D /* Build configuration list for PBXNativeTarget "Runner" */; | 			buildConfigurationList = 97C147051CF9000F007C117D /* Build configuration list for PBXNativeTarget "Runner" */; | ||||||
| 			buildPhases = ( | 			buildPhases = ( | ||||||
|  | 				2116AEE9DABFBBDED304ABEB /* [CP] Check Pods Manifest.lock */, | ||||||
| 				9740EEB61CF901F6004384FC /* Run Script */, | 				9740EEB61CF901F6004384FC /* Run Script */, | ||||||
| 				97C146EA1CF9000F007C117D /* Sources */, | 				97C146EA1CF9000F007C117D /* Sources */, | ||||||
| 				97C146EB1CF9000F007C117D /* Frameworks */, | 				97C146EB1CF9000F007C117D /* Frameworks */, | ||||||
| 				97C146EC1CF9000F007C117D /* Resources */, | 				97C146EC1CF9000F007C117D /* Resources */, | ||||||
| 				9705A1C41CF9048500538489 /* Embed Frameworks */, | 				9705A1C41CF9048500538489 /* Embed Frameworks */, | ||||||
| 				3B06AD1E1E4923F5004D2608 /* Thin Binary */, | 				3B06AD1E1E4923F5004D2608 /* Thin Binary */, | ||||||
|  | 				FE4BAF74959AF0624BA808EE /* [CP] Embed Pods Frameworks */, | ||||||
|  | 				EE58653D94051600FD646EBE /* [CP] Copy Pods Resources */, | ||||||
| 			); | 			); | ||||||
| 			buildRules = ( | 			buildRules = ( | ||||||
| 			); | 			); | ||||||
| @@ -222,6 +270,28 @@ | |||||||
| /* End PBXResourcesBuildPhase section */ | /* End PBXResourcesBuildPhase section */ | ||||||
|  |  | ||||||
| /* Begin PBXShellScriptBuildPhase section */ | /* Begin PBXShellScriptBuildPhase section */ | ||||||
|  | 		2116AEE9DABFBBDED304ABEB /* [CP] Check Pods Manifest.lock */ = { | ||||||
|  | 			isa = PBXShellScriptBuildPhase; | ||||||
|  | 			buildActionMask = 2147483647; | ||||||
|  | 			files = ( | ||||||
|  | 			); | ||||||
|  | 			inputFileListPaths = ( | ||||||
|  | 			); | ||||||
|  | 			inputPaths = ( | ||||||
|  | 				"${PODS_PODFILE_DIR_PATH}/Podfile.lock", | ||||||
|  | 				"${PODS_ROOT}/Manifest.lock", | ||||||
|  | 			); | ||||||
|  | 			name = "[CP] Check Pods Manifest.lock"; | ||||||
|  | 			outputFileListPaths = ( | ||||||
|  | 			); | ||||||
|  | 			outputPaths = ( | ||||||
|  | 				"$(DERIVED_FILE_DIR)/Pods-Runner-checkManifestLockResult.txt", | ||||||
|  | 			); | ||||||
|  | 			runOnlyForDeploymentPostprocessing = 0; | ||||||
|  | 			shellPath = /bin/sh; | ||||||
|  | 			shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n    # print error to STDERR\n    echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n    exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n"; | ||||||
|  | 			showEnvVarsInLog = 0; | ||||||
|  | 		}; | ||||||
| 		3B06AD1E1E4923F5004D2608 /* Thin Binary */ = { | 		3B06AD1E1E4923F5004D2608 /* Thin Binary */ = { | ||||||
| 			isa = PBXShellScriptBuildPhase; | 			isa = PBXShellScriptBuildPhase; | ||||||
| 			alwaysOutOfDate = 1; | 			alwaysOutOfDate = 1; | ||||||
| @@ -253,6 +323,62 @@ | |||||||
| 			shellPath = /bin/sh; | 			shellPath = /bin/sh; | ||||||
| 			shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" build"; | 			shellScript = "/bin/sh \"$FLUTTER_ROOT/packages/flutter_tools/bin/xcode_backend.sh\" build"; | ||||||
| 		}; | 		}; | ||||||
|  | 		EE58653D94051600FD646EBE /* [CP] Copy Pods Resources */ = { | ||||||
|  | 			isa = PBXShellScriptBuildPhase; | ||||||
|  | 			buildActionMask = 2147483647; | ||||||
|  | 			files = ( | ||||||
|  | 			); | ||||||
|  | 			inputFileListPaths = ( | ||||||
|  | 				"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-resources-${CONFIGURATION}-input-files.xcfilelist", | ||||||
|  | 			); | ||||||
|  | 			name = "[CP] Copy Pods Resources"; | ||||||
|  | 			outputFileListPaths = ( | ||||||
|  | 				"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-resources-${CONFIGURATION}-output-files.xcfilelist", | ||||||
|  | 			); | ||||||
|  | 			runOnlyForDeploymentPostprocessing = 0; | ||||||
|  | 			shellPath = /bin/sh; | ||||||
|  | 			shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-resources.sh\"\n"; | ||||||
|  | 			showEnvVarsInLog = 0; | ||||||
|  | 		}; | ||||||
|  | 		F27C1B361CA1B045C8D36B3B /* [CP] Check Pods Manifest.lock */ = { | ||||||
|  | 			isa = PBXShellScriptBuildPhase; | ||||||
|  | 			buildActionMask = 2147483647; | ||||||
|  | 			files = ( | ||||||
|  | 			); | ||||||
|  | 			inputFileListPaths = ( | ||||||
|  | 			); | ||||||
|  | 			inputPaths = ( | ||||||
|  | 				"${PODS_PODFILE_DIR_PATH}/Podfile.lock", | ||||||
|  | 				"${PODS_ROOT}/Manifest.lock", | ||||||
|  | 			); | ||||||
|  | 			name = "[CP] Check Pods Manifest.lock"; | ||||||
|  | 			outputFileListPaths = ( | ||||||
|  | 			); | ||||||
|  | 			outputPaths = ( | ||||||
|  | 				"$(DERIVED_FILE_DIR)/Pods-RunnerTests-checkManifestLockResult.txt", | ||||||
|  | 			); | ||||||
|  | 			runOnlyForDeploymentPostprocessing = 0; | ||||||
|  | 			shellPath = /bin/sh; | ||||||
|  | 			shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n    # print error to STDERR\n    echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n    exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n"; | ||||||
|  | 			showEnvVarsInLog = 0; | ||||||
|  | 		}; | ||||||
|  | 		FE4BAF74959AF0624BA808EE /* [CP] Embed Pods Frameworks */ = { | ||||||
|  | 			isa = PBXShellScriptBuildPhase; | ||||||
|  | 			buildActionMask = 2147483647; | ||||||
|  | 			files = ( | ||||||
|  | 			); | ||||||
|  | 			inputFileListPaths = ( | ||||||
|  | 				"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks-${CONFIGURATION}-input-files.xcfilelist", | ||||||
|  | 			); | ||||||
|  | 			name = "[CP] Embed Pods Frameworks"; | ||||||
|  | 			outputFileListPaths = ( | ||||||
|  | 				"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks-${CONFIGURATION}-output-files.xcfilelist", | ||||||
|  | 			); | ||||||
|  | 			runOnlyForDeploymentPostprocessing = 0; | ||||||
|  | 			shellPath = /bin/sh; | ||||||
|  | 			shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-Runner/Pods-Runner-frameworks.sh\"\n"; | ||||||
|  | 			showEnvVarsInLog = 0; | ||||||
|  | 		}; | ||||||
| /* End PBXShellScriptBuildPhase section */ | /* End PBXShellScriptBuildPhase section */ | ||||||
|  |  | ||||||
| /* Begin PBXSourcesBuildPhase section */ | /* Begin PBXSourcesBuildPhase section */ | ||||||
| @@ -327,6 +453,7 @@ | |||||||
| 				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | 				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | ||||||
| 				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | 				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | ||||||
| 				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | 				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | ||||||
|  | 				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||||
| 				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | 				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | ||||||
| 				CLANG_WARN_STRICT_PROTOTYPES = YES; | 				CLANG_WARN_STRICT_PROTOTYPES = YES; | ||||||
| 				CLANG_WARN_SUSPICIOUS_MOVE = YES; | 				CLANG_WARN_SUSPICIOUS_MOVE = YES; | ||||||
| @@ -361,27 +488,45 @@ | |||||||
| 			buildSettings = { | 			buildSettings = { | ||||||
| 				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | 				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | ||||||
| 				CLANG_ENABLE_MODULES = YES; | 				CLANG_ENABLE_MODULES = YES; | ||||||
| 				CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; | 				CODE_SIGN_IDENTITY = "Apple Development"; | ||||||
|  | 				"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; | ||||||
|  | 				CODE_SIGN_STYLE = Manual; | ||||||
|  | 				CURRENT_PROJECT_VERSION = 3; | ||||||
|  | 				DEVELOPMENT_TEAM = ""; | ||||||
|  | 				"DEVELOPMENT_TEAM[sdk=iphoneos*]" = L32Y3D8V83; | ||||||
| 				ENABLE_BITCODE = NO; | 				ENABLE_BITCODE = NO; | ||||||
| 				INFOPLIST_FILE = Runner/Info.plist; | 				INFOPLIST_FILE = Runner/Info.plist; | ||||||
|  | 				INFOPLIST_KEY_CFBundleDisplayName = Any.Way; | ||||||
|  | 				INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.travel"; | ||||||
|  | 				IPHONEOS_DEPLOYMENT_TARGET = 15.6; | ||||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | 				LD_RUNPATH_SEARCH_PATHS = ( | ||||||
| 					"$(inherited)", | 					"$(inherited)", | ||||||
| 					"@executable_path/Frameworks", | 					"@executable_path/Frameworks", | ||||||
| 				); | 				); | ||||||
| 				PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation; | 				MARKETING_VERSION = 1.0.0; | ||||||
|  | 				PRODUCT_BUNDLE_IDENTIFIER = info.anydev.anyway; | ||||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||||
|  | 				PROVISIONING_PROFILE_SPECIFIER = "match AppStore info.anydev.anyway"; | ||||||
|  | 				"PROVISIONING_PROFILE_SPECIFIER[sdk=iphoneos*]" = "match AppStore info.anydev.anyway"; | ||||||
|  | 				SUPPORTED_PLATFORMS = "iphoneos iphonesimulator"; | ||||||
|  | 				SUPPORTS_MACCATALYST = NO; | ||||||
|  | 				SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = NO; | ||||||
|  | 				SUPPORTS_XR_DESIGNED_FOR_IPHONE_IPAD = NO; | ||||||
| 				SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; | 				SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; | ||||||
| 				SWIFT_VERSION = 5.0; | 				SWIFT_VERSION = 5.0; | ||||||
|  | 				TARGETED_DEVICE_FAMILY = "1,2"; | ||||||
| 				VERSIONING_SYSTEM = "apple-generic"; | 				VERSIONING_SYSTEM = "apple-generic"; | ||||||
| 			}; | 			}; | ||||||
| 			name = Profile; | 			name = Profile; | ||||||
| 		}; | 		}; | ||||||
| 		331C8088294A63A400263BE5 /* Debug */ = { | 		331C8088294A63A400263BE5 /* Debug */ = { | ||||||
| 			isa = XCBuildConfiguration; | 			isa = XCBuildConfiguration; | ||||||
|  | 			baseConfigurationReference = DC475F5210027479529644C3 /* Pods-RunnerTests.debug.xcconfig */; | ||||||
| 			buildSettings = { | 			buildSettings = { | ||||||
| 				BUNDLE_LOADER = "$(TEST_HOST)"; | 				BUNDLE_LOADER = "$(TEST_HOST)"; | ||||||
| 				CODE_SIGN_STYLE = Automatic; | 				CODE_SIGN_STYLE = Automatic; | ||||||
| 				CURRENT_PROJECT_VERSION = 1; | 				CURRENT_PROJECT_VERSION = 3; | ||||||
|  | 				DEVELOPMENT_TEAM = L32Y3D8V83; | ||||||
| 				GENERATE_INFOPLIST_FILE = YES; | 				GENERATE_INFOPLIST_FILE = YES; | ||||||
| 				MARKETING_VERSION = 1.0; | 				MARKETING_VERSION = 1.0; | ||||||
| 				PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation.RunnerTests; | 				PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation.RunnerTests; | ||||||
| @@ -395,10 +540,12 @@ | |||||||
| 		}; | 		}; | ||||||
| 		331C8089294A63A400263BE5 /* Release */ = { | 		331C8089294A63A400263BE5 /* Release */ = { | ||||||
| 			isa = XCBuildConfiguration; | 			isa = XCBuildConfiguration; | ||||||
|  | 			baseConfigurationReference = A565AAB9FE158487ABF3A5BF /* Pods-RunnerTests.release.xcconfig */; | ||||||
| 			buildSettings = { | 			buildSettings = { | ||||||
| 				BUNDLE_LOADER = "$(TEST_HOST)"; | 				BUNDLE_LOADER = "$(TEST_HOST)"; | ||||||
| 				CODE_SIGN_STYLE = Automatic; | 				CODE_SIGN_STYLE = Automatic; | ||||||
| 				CURRENT_PROJECT_VERSION = 1; | 				CURRENT_PROJECT_VERSION = 3; | ||||||
|  | 				DEVELOPMENT_TEAM = L32Y3D8V83; | ||||||
| 				GENERATE_INFOPLIST_FILE = YES; | 				GENERATE_INFOPLIST_FILE = YES; | ||||||
| 				MARKETING_VERSION = 1.0; | 				MARKETING_VERSION = 1.0; | ||||||
| 				PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation.RunnerTests; | 				PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation.RunnerTests; | ||||||
| @@ -410,10 +557,12 @@ | |||||||
| 		}; | 		}; | ||||||
| 		331C808A294A63A400263BE5 /* Profile */ = { | 		331C808A294A63A400263BE5 /* Profile */ = { | ||||||
| 			isa = XCBuildConfiguration; | 			isa = XCBuildConfiguration; | ||||||
|  | 			baseConfigurationReference = 282EA28E78AB3F765E4BA719 /* Pods-RunnerTests.profile.xcconfig */; | ||||||
| 			buildSettings = { | 			buildSettings = { | ||||||
| 				BUNDLE_LOADER = "$(TEST_HOST)"; | 				BUNDLE_LOADER = "$(TEST_HOST)"; | ||||||
| 				CODE_SIGN_STYLE = Automatic; | 				CODE_SIGN_STYLE = Automatic; | ||||||
| 				CURRENT_PROJECT_VERSION = 1; | 				CURRENT_PROJECT_VERSION = 3; | ||||||
|  | 				DEVELOPMENT_TEAM = L32Y3D8V83; | ||||||
| 				GENERATE_INFOPLIST_FILE = YES; | 				GENERATE_INFOPLIST_FILE = YES; | ||||||
| 				MARKETING_VERSION = 1.0; | 				MARKETING_VERSION = 1.0; | ||||||
| 				PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation.RunnerTests; | 				PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation.RunnerTests; | ||||||
| @@ -447,6 +596,7 @@ | |||||||
| 				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | 				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | ||||||
| 				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | 				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | ||||||
| 				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | 				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | ||||||
|  | 				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||||
| 				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | 				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | ||||||
| 				CLANG_WARN_STRICT_PROTOTYPES = YES; | 				CLANG_WARN_STRICT_PROTOTYPES = YES; | ||||||
| 				CLANG_WARN_SUSPICIOUS_MOVE = YES; | 				CLANG_WARN_SUSPICIOUS_MOVE = YES; | ||||||
| @@ -504,6 +654,7 @@ | |||||||
| 				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | 				CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; | ||||||
| 				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | 				CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; | ||||||
| 				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | 				CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; | ||||||
|  | 				CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES; | ||||||
| 				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | 				CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; | ||||||
| 				CLANG_WARN_STRICT_PROTOTYPES = YES; | 				CLANG_WARN_STRICT_PROTOTYPES = YES; | ||||||
| 				CLANG_WARN_SUSPICIOUS_MOVE = YES; | 				CLANG_WARN_SUSPICIOUS_MOVE = YES; | ||||||
| @@ -540,18 +691,34 @@ | |||||||
| 			buildSettings = { | 			buildSettings = { | ||||||
| 				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | 				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | ||||||
| 				CLANG_ENABLE_MODULES = YES; | 				CLANG_ENABLE_MODULES = YES; | ||||||
| 				CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; | 				CODE_SIGN_IDENTITY = "Apple Development"; | ||||||
|  | 				"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; | ||||||
|  | 				CODE_SIGN_STYLE = Manual; | ||||||
|  | 				CURRENT_PROJECT_VERSION = 3; | ||||||
|  | 				DEVELOPMENT_TEAM = ""; | ||||||
|  | 				"DEVELOPMENT_TEAM[sdk=iphoneos*]" = L32Y3D8V83; | ||||||
| 				ENABLE_BITCODE = NO; | 				ENABLE_BITCODE = NO; | ||||||
| 				INFOPLIST_FILE = Runner/Info.plist; | 				INFOPLIST_FILE = Runner/Info.plist; | ||||||
|  | 				INFOPLIST_KEY_CFBundleDisplayName = Any.Way; | ||||||
|  | 				INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.travel"; | ||||||
|  | 				IPHONEOS_DEPLOYMENT_TARGET = 15.6; | ||||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | 				LD_RUNPATH_SEARCH_PATHS = ( | ||||||
| 					"$(inherited)", | 					"$(inherited)", | ||||||
| 					"@executable_path/Frameworks", | 					"@executable_path/Frameworks", | ||||||
| 				); | 				); | ||||||
| 				PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation; | 				MARKETING_VERSION = 1.0.0; | ||||||
|  | 				PRODUCT_BUNDLE_IDENTIFIER = info.anydev.anyway; | ||||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||||
|  | 				PROVISIONING_PROFILE_SPECIFIER = "match AppStore info.anydev.anyway"; | ||||||
|  | 				"PROVISIONING_PROFILE_SPECIFIER[sdk=iphoneos*]" = "match AppStore info.anydev.anyway"; | ||||||
|  | 				SUPPORTED_PLATFORMS = "iphoneos iphonesimulator"; | ||||||
|  | 				SUPPORTS_MACCATALYST = NO; | ||||||
|  | 				SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = NO; | ||||||
|  | 				SUPPORTS_XR_DESIGNED_FOR_IPHONE_IPAD = NO; | ||||||
| 				SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; | 				SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; | ||||||
| 				SWIFT_OPTIMIZATION_LEVEL = "-Onone"; | 				SWIFT_OPTIMIZATION_LEVEL = "-Onone"; | ||||||
| 				SWIFT_VERSION = 5.0; | 				SWIFT_VERSION = 5.0; | ||||||
|  | 				TARGETED_DEVICE_FAMILY = "1,2"; | ||||||
| 				VERSIONING_SYSTEM = "apple-generic"; | 				VERSIONING_SYSTEM = "apple-generic"; | ||||||
| 			}; | 			}; | ||||||
| 			name = Debug; | 			name = Debug; | ||||||
| @@ -562,17 +729,33 @@ | |||||||
| 			buildSettings = { | 			buildSettings = { | ||||||
| 				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | 				ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; | ||||||
| 				CLANG_ENABLE_MODULES = YES; | 				CLANG_ENABLE_MODULES = YES; | ||||||
| 				CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)"; | 				CODE_SIGN_IDENTITY = "Apple Development"; | ||||||
|  | 				"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; | ||||||
|  | 				CODE_SIGN_STYLE = Manual; | ||||||
|  | 				CURRENT_PROJECT_VERSION = 3; | ||||||
|  | 				DEVELOPMENT_TEAM = ""; | ||||||
|  | 				"DEVELOPMENT_TEAM[sdk=iphoneos*]" = L32Y3D8V83; | ||||||
| 				ENABLE_BITCODE = NO; | 				ENABLE_BITCODE = NO; | ||||||
| 				INFOPLIST_FILE = Runner/Info.plist; | 				INFOPLIST_FILE = Runner/Info.plist; | ||||||
|  | 				INFOPLIST_KEY_CFBundleDisplayName = Any.Way; | ||||||
|  | 				INFOPLIST_KEY_LSApplicationCategoryType = "public.app-category.travel"; | ||||||
|  | 				IPHONEOS_DEPLOYMENT_TARGET = 15.6; | ||||||
| 				LD_RUNPATH_SEARCH_PATHS = ( | 				LD_RUNPATH_SEARCH_PATHS = ( | ||||||
| 					"$(inherited)", | 					"$(inherited)", | ||||||
| 					"@executable_path/Frameworks", | 					"@executable_path/Frameworks", | ||||||
| 				); | 				); | ||||||
| 				PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation; | 				MARKETING_VERSION = 1.0.0; | ||||||
|  | 				PRODUCT_BUNDLE_IDENTIFIER = info.anydev.anyway; | ||||||
| 				PRODUCT_NAME = "$(TARGET_NAME)"; | 				PRODUCT_NAME = "$(TARGET_NAME)"; | ||||||
|  | 				PROVISIONING_PROFILE_SPECIFIER = "match AppStore info.anydev.anyway"; | ||||||
|  | 				"PROVISIONING_PROFILE_SPECIFIER[sdk=iphoneos*]" = "match AppStore info.anydev.anyway"; | ||||||
|  | 				SUPPORTED_PLATFORMS = "iphoneos iphonesimulator"; | ||||||
|  | 				SUPPORTS_MACCATALYST = NO; | ||||||
|  | 				SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = NO; | ||||||
|  | 				SUPPORTS_XR_DESIGNED_FOR_IPHONE_IPAD = NO; | ||||||
| 				SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; | 				SWIFT_OBJC_BRIDGING_HEADER = "Runner/Runner-Bridging-Header.h"; | ||||||
| 				SWIFT_VERSION = 5.0; | 				SWIFT_VERSION = 5.0; | ||||||
|  | 				TARGETED_DEVICE_FAMILY = "1,2"; | ||||||
| 				VERSIONING_SYSTEM = "apple-generic"; | 				VERSIONING_SYSTEM = "apple-generic"; | ||||||
| 			}; | 			}; | ||||||
| 			name = Release; | 			name = Release; | ||||||
|   | |||||||
| @@ -4,4 +4,7 @@ | |||||||
|    <FileRef |    <FileRef | ||||||
|       location = "group:Runner.xcodeproj"> |       location = "group:Runner.xcodeproj"> | ||||||
|    </FileRef> |    </FileRef> | ||||||
|  |    <FileRef | ||||||
|  |       location = "group:Pods/Pods.xcodeproj"> | ||||||
|  |    </FileRef> | ||||||
| </Workspace> | </Workspace> | ||||||
|   | |||||||
| @@ -1,12 +1,16 @@ | |||||||
| import UIKit | import UIKit | ||||||
| import Flutter | import Flutter | ||||||
|  | import GoogleMaps | ||||||
|  |  | ||||||
| @UIApplicationMain | @main | ||||||
| @objc class AppDelegate: FlutterAppDelegate { | @objc class AppDelegate: FlutterAppDelegate { | ||||||
|   override func application( |   override func application( | ||||||
|     _ application: UIApplication, |     _ application: UIApplication, | ||||||
|     didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]? |     didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]? | ||||||
|   ) -> Bool { |   ) -> Bool { | ||||||
|  |     // load the key from env | ||||||
|  |     let key = ProcessInfo.processInfo.environment["GOOGLE_MAPS_API_KEY"]! | ||||||
|  |     GMSServices.provideAPIKey(key) | ||||||
|     GeneratedPluginRegistrant.register(with: self) |     GeneratedPluginRegistrant.register(with: self) | ||||||
|     return super.application(application, didFinishLaunchingWithOptions: launchOptions) |     return super.application(application, didFinishLaunchingWithOptions: launchOptions) | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -2,6 +2,8 @@ | |||||||
| <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> | <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> | ||||||
| <plist version="1.0"> | <plist version="1.0"> | ||||||
| <dict> | <dict> | ||||||
|  | 	<key>CADisableMinimumFrameDurationOnPhone</key> | ||||||
|  | 	<true/> | ||||||
| 	<key>CFBundleDevelopmentRegion</key> | 	<key>CFBundleDevelopmentRegion</key> | ||||||
| 	<string>$(DEVELOPMENT_LANGUAGE)</string> | 	<string>$(DEVELOPMENT_LANGUAGE)</string> | ||||||
| 	<key>CFBundleDisplayName</key> | 	<key>CFBundleDisplayName</key> | ||||||
| @@ -24,6 +26,8 @@ | |||||||
| 	<string>$(FLUTTER_BUILD_NUMBER)</string> | 	<string>$(FLUTTER_BUILD_NUMBER)</string> | ||||||
| 	<key>LSRequiresIPhoneOS</key> | 	<key>LSRequiresIPhoneOS</key> | ||||||
| 	<true/> | 	<true/> | ||||||
|  | 	<key>UIApplicationSupportsIndirectInputEvents</key> | ||||||
|  | 	<true/> | ||||||
| 	<key>UILaunchStoryboardName</key> | 	<key>UILaunchStoryboardName</key> | ||||||
| 	<string>LaunchScreen</string> | 	<string>LaunchScreen</string> | ||||||
| 	<key>UIMainStoryboardFile</key> | 	<key>UIMainStoryboardFile</key> | ||||||
| @@ -41,9 +45,5 @@ | |||||||
| 		<string>UIInterfaceOrientationLandscapeLeft</string> | 		<string>UIInterfaceOrientationLandscapeLeft</string> | ||||||
| 		<string>UIInterfaceOrientationLandscapeRight</string> | 		<string>UIInterfaceOrientationLandscapeRight</string> | ||||||
| 	</array> | 	</array> | ||||||
| 	<key>CADisableMinimumFrameDurationOnPhone</key> |  | ||||||
| 	<true/> |  | ||||||
| 	<key>UIApplicationSupportsIndirectInputEvents</key> |  | ||||||
| 	<true/> |  | ||||||
| </dict> | </dict> | ||||||
| </plist> | </plist> | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								frontend/ios/fastlane/Appfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,8 @@ | |||||||
|  | app_identifier("info.anydev.testing") # The bundle identifier of your app | ||||||
|  | apple_id("me@moll.re") # Your Apple Developer Portal username | ||||||
|  |  | ||||||
|  | itc_team_id("127439860") # App Store Connect Team ID | ||||||
|  | team_id("L32Y3D8V83") # Developer Portal Team ID | ||||||
|  |  | ||||||
|  | # For more information about the Appfile, see: | ||||||
|  | #     https://docs.fastlane.tools/advanced/#appfile | ||||||
							
								
								
									
										91
									
								
								frontend/ios/fastlane/Fastfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,91 @@ | |||||||
|  | default_platform(:ios) | ||||||
|  |  | ||||||
|  | platform :ios do | ||||||
|  |  | ||||||
|  |   desc "Load the App Store Connect API token" | ||||||
|  |   lane :load_asc_api_token do | ||||||
|  |     app_store_connect_api_key( | ||||||
|  |       key_id: ENV["IOS_ASC_KEY_ID"], | ||||||
|  |       issuer_id: ENV["IOS_ASC_ISSUER_ID"], | ||||||
|  |       key_content: ENV["IOS_ASC_KEY"], | ||||||
|  |       is_key_content_base64: true, | ||||||
|  |       in_house: false | ||||||
|  |     ) | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   desc "Deploy a new version to closed testing (testflight)" | ||||||
|  |   lane :deploy_testing do | ||||||
|  |     build_name = ENV["BUILD_NAME"] | ||||||
|  |     build_number = ENV["BUILD_NUMBER"] | ||||||
|  |      | ||||||
|  |     load_asc_api_token | ||||||
|  |     api_key = lane_context[SharedValues::APP_STORE_CONNECT_API_KEY] | ||||||
|  |  | ||||||
|  |     sync_code_signing( | ||||||
|  |       api_key: api_key, | ||||||
|  |       type: "appstore", | ||||||
|  |       readonly: true, | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     sh( | ||||||
|  |       "flutter", | ||||||
|  |       "build", | ||||||
|  |       "ipa", | ||||||
|  |       "--release", | ||||||
|  |       "--build-name=#{build_name}", | ||||||
|  |       "--build-number=#{build_number}", | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     # sign the app (whithout rebuilding it) | ||||||
|  |     build_app( | ||||||
|  |       skip_build_archive: true, | ||||||
|  |       archive_path: "../build/ios/archive/Runner.xcarchive" | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     upload_to_testflight | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |  | ||||||
|  |   desc "Deploy a new version as a full release" | ||||||
|  |   lane :deploy_release do | ||||||
|  |     build_name = ENV["BUILD_NAME"] | ||||||
|  |     build_number = ENV["BUILD_NUMBER"] | ||||||
|  |      | ||||||
|  |     load_asc_api_token | ||||||
|  |     api_key = lane_context[SharedValues::APP_STORE_CONNECT_API_KEY] | ||||||
|  |  | ||||||
|  |     sync_code_signing( | ||||||
|  |       api_key: api_key, | ||||||
|  |       type: "appstore", | ||||||
|  |       readonly: true, | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     sh( | ||||||
|  |       "flutter", | ||||||
|  |       "build", | ||||||
|  |       "ipa", | ||||||
|  |       "--release", | ||||||
|  |       "--build-name=#{build_name}", | ||||||
|  |       "--build-number=#{build_number}", | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     # sign the app (whithout rebuilding it) | ||||||
|  |     build_app( | ||||||
|  |       skip_build_archive: true, | ||||||
|  |       archive_path: "../build/ios/archive/Runner.xarchive" | ||||||
|  |     ) | ||||||
|  |    | ||||||
|  |     upload_to_app_store( | ||||||
|  |       skip_screenshots: true, | ||||||
|  |       skip_metadata: true, | ||||||
|  |       skip_app_rating_config: true, | ||||||
|  |       skip_app_review_information: true, | ||||||
|  |       skip_submission: false, | ||||||
|  |       # automatically submit the app for review | ||||||
|  |       automatic_release: true, | ||||||
|  |       # automatically release the app after review | ||||||
|  |     ) | ||||||
|  |   end | ||||||
|  | end | ||||||
							
								
								
									
										8
									
								
								frontend/ios/fastlane/Matchfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,8 @@ | |||||||
|  | git_url("ssh://git@git.kluster.moll.re:2222/anydev/anyway-app-secrets.git") | ||||||
|  |  | ||||||
|  | storage_mode("git") | ||||||
|  |  | ||||||
|  | type("appstore") # The default type, can be: appstore, adhoc, enterprise or development | ||||||
|  |  | ||||||
|  | app_identifier(["info.anydev.anyway"]) | ||||||
|  | username("me@moll.re") # Your Apple Developer Portal username | ||||||
							
								
								
									
										48
									
								
								frontend/ios/fastlane/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,48 @@ | |||||||
|  | fastlane documentation | ||||||
|  | ---- | ||||||
|  |  | ||||||
|  | # Installation | ||||||
|  |  | ||||||
|  | Make sure you have the latest version of the Xcode command line tools installed: | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | xcode-select --install | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | For _fastlane_ installation instructions, see [Installing _fastlane_](https://docs.fastlane.tools/#installing-fastlane) | ||||||
|  |  | ||||||
|  | # Available Actions | ||||||
|  |  | ||||||
|  | ## iOS | ||||||
|  |  | ||||||
|  | ### ios load_asc_api_token | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | [bundle exec] fastlane ios load_asc_api_token | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Load the App Store Connect API token | ||||||
|  |  | ||||||
|  | ### ios beta | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | [bundle exec] fastlane ios beta | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Push a new beta build to TestFlight | ||||||
|  |  | ||||||
|  | ### ios deploy_testing | ||||||
|  |  | ||||||
|  | ```sh | ||||||
|  | [bundle exec] fastlane ios deploy_testing | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ---- | ||||||
|  |  | ||||||
|  | This README.md is auto-generated and will be re-generated every time [_fastlane_](https://fastlane.tools) is run. | ||||||
|  |  | ||||||
|  | More information about _fastlane_ can be found on [fastlane.tools](https://fastlane.tools). | ||||||
|  |  | ||||||
|  | The documentation of _fastlane_ can be found on [docs.fastlane.tools](https://docs.fastlane.tools). | ||||||
							
								
								
									
										10
									
								
								frontend/ios/fastlane/local.env.sample
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | |||||||
|  | # SAMPLE env file that replicates the env in the CI/CD pipeline | ||||||
|  | # DO NOT EDIT THIS FILE | ||||||
|  | # Copy this file to local.env and edit the values to match your local environment | ||||||
|  | IOS_ASC_KEY_ID="sample" | ||||||
|  | IOS_ASC_KEY="sample" | ||||||
|  | IOS_ASC_ISSUER_ID="sample" | ||||||
|  | SIGNING_KEY_FILE_PATH="sample" | ||||||
|  | SIGNING_KEY_PASSWORD="sample" | ||||||
|  | BUILD_NAME="sample" | ||||||
|  | BUILD_NUMBER="sample" | ||||||
| @@ -3,6 +3,7 @@ import 'package:flutter/material.dart'; | |||||||
| const String APP_NAME = 'AnyWay'; | const String APP_NAME = 'AnyWay'; | ||||||
|  |  | ||||||
| String API_URL_BASE = 'https://anyway.anydev.info'; | String API_URL_BASE = 'https://anyway.anydev.info'; | ||||||
|  | String API_URL_DEBUG = 'https://anyway.anydev.info'; | ||||||
| String PRIVACY_URL = 'https://anydev.info/privacy'; | String PRIVACY_URL = 'https://anydev.info/privacy'; | ||||||
|  |  | ||||||
| const String MAP_ID = '41c21ac9b81dbfd8'; | const String MAP_ID = '41c21ac9b81dbfd8'; | ||||||
|   | |||||||
							
								
								
									
										37
									
								
								frontend/lib/modules/current_trip_error_message.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,37 @@ | |||||||
|  | import 'package:anyway/structs/trip.dart'; | ||||||
|  | import 'package:auto_size_text/auto_size_text.dart'; | ||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  |  | ||||||
|  | class CurrentTripErrorMessage extends StatefulWidget { | ||||||
|  |   final Trip trip; | ||||||
|  |   const CurrentTripErrorMessage({ | ||||||
|  |     super.key, | ||||||
|  |     required this.trip, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   State<CurrentTripErrorMessage> createState() => _CurrentTripErrorMessageState(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class _CurrentTripErrorMessageState extends State<CurrentTripErrorMessage> { | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context) => Center( | ||||||
|  |     child: Row( | ||||||
|  |       mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |       children: [ | ||||||
|  |         const Icon( | ||||||
|  |           Icons.error_outline, | ||||||
|  |           color: Colors.red, | ||||||
|  |           size: 50, | ||||||
|  |         ), | ||||||
|  |         const Padding( | ||||||
|  |           padding: EdgeInsets.only(left: 10), | ||||||
|  |         ), | ||||||
|  |         AutoSizeText( | ||||||
|  |           'Error: ${widget.trip.errorDescription}', | ||||||
|  |           maxLines: 3, | ||||||
|  |         ), | ||||||
|  |       ], | ||||||
|  |     ) | ||||||
|  |   ); | ||||||
|  | } | ||||||
| @@ -1,111 +1,50 @@ | |||||||
| import 'dart:developer'; | import 'package:flutter/material.dart'; | ||||||
|  |  | ||||||
| import 'package:anyway/constants.dart'; |  | ||||||
| import 'package:anyway/structs/trip.dart'; |  | ||||||
| import 'package:auto_size_text/auto_size_text.dart'; | import 'package:auto_size_text/auto_size_text.dart'; | ||||||
|  |  | ||||||
| import 'package:flutter/material.dart'; | import 'package:anyway/pages/current_trip.dart'; | ||||||
|  | import 'package:anyway/structs/trip.dart'; | ||||||
|  |  | ||||||
| class Greeter extends StatefulWidget { |  | ||||||
|  | class CurrentTripGreeter extends StatefulWidget { | ||||||
|   final Trip trip; |   final Trip trip; | ||||||
|  |  | ||||||
|   Greeter({ |   CurrentTripGreeter({ | ||||||
|  |     super.key, | ||||||
|     required this.trip, |     required this.trip, | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   State<Greeter> createState() => _GreeterState(); |   State<CurrentTripGreeter> createState() => _CurrentTripGreeterState(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| class _GreeterState extends State<Greeter> { | class _CurrentTripGreeterState extends State<CurrentTripGreeter> { | ||||||
|    |  | ||||||
|   Widget greeterBuilder (BuildContext context, Widget? child) { |  | ||||||
|     final Shader textGradient = APP_GRADIENT.createShader(Rect.fromLTWH(0.0, 0.0, 200.0, 70.0)); |  | ||||||
|     TextStyle greeterStyle = TextStyle( |  | ||||||
|       foreground: Paint()..shader = textGradient, |  | ||||||
|       fontWeight: FontWeight.bold, |  | ||||||
|       fontSize: 26 |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     Widget topGreeter; |  | ||||||
|  |  | ||||||
|     if (widget.trip.uuid != 'pending') { |  | ||||||
|       topGreeter = FutureBuilder( |  | ||||||
|         future: widget.trip.cityName, |  | ||||||
|         builder: (BuildContext context, AsyncSnapshot<String> snapshot) { |  | ||||||
|           if (snapshot.hasData) { |  | ||||||
|             return AutoSizeText( |  | ||||||
|               maxLines: 1, |  | ||||||
|               'Welcome to ${snapshot.data}!', |  | ||||||
|               style: greeterStyle |  | ||||||
|             ); |  | ||||||
|           } else if (snapshot.hasError) { |  | ||||||
|             log('Error while fetching city name'); |  | ||||||
|             return AutoSizeText( |  | ||||||
|               maxLines: 1, |  | ||||||
|               'Welcome to your trip!', |  | ||||||
|               style: greeterStyle |  | ||||||
|             ); |  | ||||||
|           } else { |  | ||||||
|             return AutoSizeText( |  | ||||||
|               maxLines: 1, |  | ||||||
|               'Welcome to ...', |  | ||||||
|               style: greeterStyle |  | ||||||
|             ); |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|       ); |  | ||||||
|     } else { |  | ||||||
|       // still awaiting the trip |  | ||||||
|       // We can hopefully infer the city name from the cityName future |  | ||||||
|       // Show a linear loader at the bottom and an info message above |  | ||||||
|       topGreeter = Column( |  | ||||||
|         mainAxisAlignment: MainAxisAlignment.end, |  | ||||||
|         children: [ |  | ||||||
|           FutureBuilder( |  | ||||||
|             future: widget.trip.cityName, |  | ||||||
|             builder: (BuildContext context, AsyncSnapshot<String> snapshot) { |  | ||||||
|               if (snapshot.hasData) { |  | ||||||
|                 return AutoSizeText( |  | ||||||
|                   maxLines: 1, |  | ||||||
|                   'Generating your trip to ${snapshot.data}...', |  | ||||||
|                   style: greeterStyle |  | ||||||
|                 ); |  | ||||||
|               } else if (snapshot.hasError) { |  | ||||||
|                 // the exact error is shown in the central part of the trip overview. No need to show it here |  | ||||||
|                 return AutoSizeText( |  | ||||||
|                   maxLines: 1, |  | ||||||
|                   'Error while loading trip.', |  | ||||||
|                   style: greeterStyle |  | ||||||
|                   ); |  | ||||||
|               } |  | ||||||
|               return AutoSizeText( |  | ||||||
|                   maxLines: 1, |  | ||||||
|                   'Generating your trip...', |  | ||||||
|                   style: greeterStyle |  | ||||||
|                   ); |  | ||||||
|             } |  | ||||||
|           ), |  | ||||||
|           Padding( |  | ||||||
|             padding: EdgeInsets.all(5), |  | ||||||
|             child: const LinearProgressIndicator() |  | ||||||
|           ) |  | ||||||
|         ] |  | ||||||
|       ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return Center( |  | ||||||
|       child: topGreeter, |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) => Center( | ||||||
|     return ListenableBuilder( |     child: FutureBuilder( | ||||||
|       listenable: widget.trip, |       future: widget.trip.cityName, | ||||||
|       builder: greeterBuilder, |       builder: (BuildContext context, AsyncSnapshot<String> snapshot) { | ||||||
|     ); |         if (snapshot.hasData) { | ||||||
|   } |           return AutoSizeText( | ||||||
|  |             maxLines: 1, | ||||||
|  |             'Welcome to ${snapshot.data}!', | ||||||
|  |             style: greeterStyle | ||||||
|  |           ); | ||||||
|  |         } else if (snapshot.hasError) { | ||||||
|  |           return AutoSizeText( | ||||||
|  |             maxLines: 1, | ||||||
|  |             'Welcome to your trip!', | ||||||
|  |             style: greeterStyle | ||||||
|  |           ); | ||||||
|  |         } else { | ||||||
|  |           return AutoSizeText( | ||||||
|  |             maxLines: 1, | ||||||
|  |             'Welcome to ...', | ||||||
|  |             style: greeterStyle | ||||||
|  |           ); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     ) | ||||||
|  |   ); | ||||||
|  |  | ||||||
| } | } | ||||||
							
								
								
									
										60
									
								
								frontend/lib/modules/current_trip_loading_indicator.dart
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,60 @@ | |||||||
|  | import 'package:flutter/material.dart'; | ||||||
|  | import 'package:auto_size_text/auto_size_text.dart'; | ||||||
|  |  | ||||||
|  | import 'package:anyway/structs/trip.dart'; | ||||||
|  | import 'package:anyway/pages/current_trip.dart'; | ||||||
|  |  | ||||||
|  | class CurrentTripLoadingIndicator extends StatefulWidget { | ||||||
|  |   final Trip trip; | ||||||
|  |   const CurrentTripLoadingIndicator({ | ||||||
|  |     super.key, | ||||||
|  |     required this.trip, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   State<CurrentTripLoadingIndicator> createState() => _CurrentTripLoadingIndicatorState(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | class _CurrentTripLoadingIndicatorState extends State<CurrentTripLoadingIndicator> { | ||||||
|  |   @override | ||||||
|  |   Widget build(BuildContext context) => Center( | ||||||
|  |     child: FutureBuilder( | ||||||
|  |       future: widget.trip.cityName, | ||||||
|  |       builder: (BuildContext context, AsyncSnapshot<String> snapshot) { | ||||||
|  |         Widget greeter; | ||||||
|  |         Widget loadingIndicator = const Padding( | ||||||
|  |           padding: EdgeInsets.only(top: 10), | ||||||
|  |           child: CircularProgressIndicator() | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         if (snapshot.hasData) { | ||||||
|  |           greeter = AutoSizeText( | ||||||
|  |             maxLines: 1, | ||||||
|  |             'Generating your trip to ${snapshot.data}...', | ||||||
|  |             style: greeterStyle, | ||||||
|  |           ); | ||||||
|  |         } else if (snapshot.hasError) { | ||||||
|  |           // the exact error is shown in the central part of the trip overview. No need to show it here | ||||||
|  |           greeter = AutoSizeText( | ||||||
|  |             maxLines: 1, | ||||||
|  |             'Error while loading trip.', | ||||||
|  |             style: greeterStyle, | ||||||
|  |             ); | ||||||
|  |         } else { | ||||||
|  |           greeter = AutoSizeText( | ||||||
|  |             maxLines: 1, | ||||||
|  |             'Generating your trip...', | ||||||
|  |             style: greeterStyle, | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |         return Column( | ||||||
|  |           mainAxisAlignment: MainAxisAlignment.center, | ||||||
|  |           children: [ | ||||||
|  |             greeter, | ||||||
|  |             loadingIndicator, | ||||||
|  |           ], | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |     ) | ||||||
|  |   ); | ||||||
|  | } | ||||||
| @@ -1,4 +1,6 @@ | |||||||
| import 'package:anyway/constants.dart'; | import 'package:anyway/constants.dart'; | ||||||
|  | import 'package:anyway/modules/current_trip_error_message.dart'; | ||||||
|  | import 'package:anyway/modules/current_trip_loading_indicator.dart'; | ||||||
| import 'package:flutter/material.dart'; | import 'package:flutter/material.dart'; | ||||||
|  |  | ||||||
| import 'package:anyway/structs/trip.dart'; | import 'package:anyway/structs/trip.dart'; | ||||||
| @@ -28,16 +30,36 @@ class _CurrentTripPanelState extends State<CurrentTripPanel> { | |||||||
|     return ListenableBuilder( |     return ListenableBuilder( | ||||||
|       listenable: widget.trip, |       listenable: widget.trip, | ||||||
|       builder: (context, child) { |       builder: (context, child) { | ||||||
|         if (widget.trip.uuid != 'pending' && widget.trip.uuid != 'error') { |         if (widget.trip.uuid == 'error') { | ||||||
|  |           return Align( | ||||||
|  |               alignment: Alignment.topCenter, | ||||||
|  |               child: SizedBox( | ||||||
|  |                 // reuse the exact same height as the panel has when collapsed | ||||||
|  |                 // this way the greeter will be centered when the panel is collapsed | ||||||
|  |                 height: MediaQuery.of(context).size.height * TRIP_PANEL_MIN_HEIGHT - 20, | ||||||
|  |                 child: CurrentTripErrorMessage(trip: widget.trip) | ||||||
|  |               ), | ||||||
|  |             ); | ||||||
|  |         } else if (widget.trip.uuid == 'pending') { | ||||||
|  |             return Align( | ||||||
|  |               alignment: Alignment.topCenter, | ||||||
|  |               child: SizedBox( | ||||||
|  |                 // reuse the exact same height as the panel has when collapsed | ||||||
|  |                 // this way the greeter will be centered when the panel is collapsed | ||||||
|  |                 height: MediaQuery.of(context).size.height * TRIP_PANEL_MIN_HEIGHT - 20, | ||||||
|  |                 child: CurrentTripLoadingIndicator(trip: widget.trip), | ||||||
|  |               ), | ||||||
|  |             ); | ||||||
|  |         } else { | ||||||
|           return ListView( |           return ListView( | ||||||
|             controller: widget.controller, |             controller: widget.controller, | ||||||
|             padding: const EdgeInsets.only(bottom: 30, left: 5, right: 5), |             padding: const EdgeInsets.only(bottom: 30), | ||||||
|             children: [ |             children: [ | ||||||
|               SizedBox( |               SizedBox( | ||||||
|                 // reuse the exact same height as the panel has when collapsed |                 // reuse the exact same height as the panel has when collapsed | ||||||
|                 // this way the greeter will be centered when the panel is collapsed |                 // this way the greeter will be centered when the panel is collapsed | ||||||
|                 height: MediaQuery.of(context).size.height * TRIP_PANEL_MIN_HEIGHT - 20, |                 height: MediaQuery.of(context).size.height * TRIP_PANEL_MIN_HEIGHT - 20, | ||||||
|                 child: Greeter(trip: widget.trip), |                 child: CurrentTripGreeter(trip: widget.trip), | ||||||
|               ), |               ), | ||||||
|  |  | ||||||
|               const Padding(padding: EdgeInsets.only(top: 10)), |               const Padding(padding: EdgeInsets.only(top: 10)), | ||||||
| @@ -53,28 +75,6 @@ class _CurrentTripPanelState extends State<CurrentTripPanel> { | |||||||
|               Center(child: saveButton(widget.trip)), |               Center(child: saveButton(widget.trip)), | ||||||
|             ], |             ], | ||||||
|           ); |           ); | ||||||
|         } else if(widget.trip.uuid == 'pending') { |  | ||||||
|           return SizedBox( |  | ||||||
|             // reuse the exact same height as the panel has when collapsed |  | ||||||
|             // this way the greeter will be centered when the panel is collapsed |  | ||||||
|             height: MediaQuery.of(context).size.height * TRIP_PANEL_MIN_HEIGHT - 20, |  | ||||||
|             child: Greeter(trip: widget.trip) |  | ||||||
|           ); |  | ||||||
|         } else { |  | ||||||
|           return Row( |  | ||||||
|             mainAxisAlignment: MainAxisAlignment.center, |  | ||||||
|             children: [ |  | ||||||
|               const Icon( |  | ||||||
|                 Icons.error_outline, |  | ||||||
|                 color: Colors.red, |  | ||||||
|                 size: 50, |  | ||||||
|               ), |  | ||||||
|               Padding( |  | ||||||
|                 padding: const EdgeInsets.only(left: 10), |  | ||||||
|                 child: Text('Error: ${widget.trip.errorDescription}'), |  | ||||||
|               ), |  | ||||||
|             ], |  | ||||||
|           ); |  | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     ); |     ); | ||||||
|   | |||||||
| @@ -34,7 +34,7 @@ class _NewTripButtonState extends State<NewTripButton> { | |||||||
|         } |         } | ||||||
|         return FloatingActionButton.extended( |         return FloatingActionButton.extended( | ||||||
|           onPressed: onPressed, |           onPressed: onPressed, | ||||||
|           icon: const Icon(Icons.add), |           icon: const Icon(Icons.directions), | ||||||
|           label: AutoSizeText('Start planning!'), |           label: AutoSizeText('Start planning!'), | ||||||
|         );  |         );  | ||||||
|       } |       } | ||||||
|   | |||||||
| @@ -6,6 +6,12 @@ import 'package:anyway/structs/trip.dart'; | |||||||
| import 'package:anyway/modules/current_trip_map.dart'; | import 'package:anyway/modules/current_trip_map.dart'; | ||||||
| import 'package:anyway/modules/current_trip_panel.dart'; | import 'package:anyway/modules/current_trip_panel.dart'; | ||||||
|  |  | ||||||
|  | final Shader textGradient = APP_GRADIENT.createShader(Rect.fromLTWH(0.0, 0.0, 200.0, 70.0)); | ||||||
|  | TextStyle greeterStyle = TextStyle( | ||||||
|  |   foreground: Paint()..shader = textGradient, | ||||||
|  |   fontWeight: FontWeight.bold, | ||||||
|  |   fontSize: 26 | ||||||
|  | ); | ||||||
|  |  | ||||||
|  |  | ||||||
| class TripPage extends StatefulWidget { | class TripPage extends StatefulWidget { | ||||||
| @@ -35,7 +41,7 @@ class _TripPageState extends State<TripPage> { | |||||||
|         maxHeight: MediaQuery.of(context).size.height * TRIP_PANEL_MAX_HEIGHT, |         maxHeight: MediaQuery.of(context).size.height * TRIP_PANEL_MAX_HEIGHT, | ||||||
|         // padding in this context is annoying: it offsets the notion of vertical alignment. |         // padding in this context is annoying: it offsets the notion of vertical alignment. | ||||||
|         // children that want to be centered vertically need to have their size adjusted by 2x the padding |         // children that want to be centered vertically need to have their size adjusted by 2x the padding | ||||||
|         padding: const EdgeInsets.only(top: 10), |         padding: const EdgeInsets.all(10.0), | ||||||
|         // Panel snapping should not be disabled because it significantly improves the user experience |         // Panel snapping should not be disabled because it significantly improves the user experience | ||||||
|         // panelSnapping: false |         // panelSnapping: false | ||||||
|         borderRadius: const BorderRadius.only(topLeft: Radius.circular(25), topRight: Radius.circular(25)), |         borderRadius: const BorderRadius.only(topLeft: Radius.circular(25), topRight: Radius.circular(25)), | ||||||
|   | |||||||
| @@ -63,7 +63,7 @@ class _NewTripPreferencesPageState extends State<NewTripPreferencesPage> { | |||||||
|       margin: const EdgeInsets.only(left: 10, right: 10, top: 10, bottom: 0), |       margin: const EdgeInsets.only(left: 10, right: 10, top: 10, bottom: 0), | ||||||
|       shadowColor: Colors.grey, |       shadowColor: Colors.grey, | ||||||
|       child: ListTile( |       child: ListTile( | ||||||
|         leading: Icon(Icons.timer), |         leading: preferences.maxTime.icon, | ||||||
|         title: Text(preferences.maxTime.description), |         title: Text(preferences.maxTime.description), | ||||||
|         subtitle: CupertinoTimerPicker( |         subtitle: CupertinoTimerPicker( | ||||||
|           mode: CupertinoTimerPickerMode.hm, |           mode: CupertinoTimerPickerMode.hm, | ||||||
|   | |||||||
| @@ -61,9 +61,7 @@ class _SettingsPageState extends State<SettingsPage> { | |||||||
|                     return AlertDialog( |                     return AlertDialog( | ||||||
|                       title: Text('Debug mode - use a custom API endpoint'), |                       title: Text('Debug mode - use a custom API endpoint'), | ||||||
|                       content: TextField( |                       content: TextField( | ||||||
|                         decoration: InputDecoration( |                         controller: TextEditingController(text: API_URL_DEBUG), | ||||||
|                           hintText: 'https://anyway-stg.anydev.info' |  | ||||||
|                         ), |  | ||||||
|                         onChanged: (value) { |                         onChanged: (value) { | ||||||
|                           setState(() { |                           setState(() { | ||||||
|                             API_URL_BASE = value; |                             API_URL_BASE = value; | ||||||
|   | |||||||
| @@ -38,10 +38,18 @@ fetchTrip( | |||||||
|   String dataString = jsonEncode(data); |   String dataString = jsonEncode(data); | ||||||
|   log(dataString); |   log(dataString); | ||||||
|  |  | ||||||
|   final response = await dio.post( |   late Response response; | ||||||
|     "/trip/new", |   try { | ||||||
|     data: data |      response = await dio.post( | ||||||
|   ); |       "/trip/new", | ||||||
|  |       data: data | ||||||
|  |     ); | ||||||
|  |   } catch (e) { | ||||||
|  |     trip.updateUUID("error"); | ||||||
|  |     trip.updateError(e.toString()); | ||||||
|  |     log(e.toString()); | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   // handle errors |   // handle errors | ||||||
|   if (response.statusCode != 200) { |   if (response.statusCode != 200) { | ||||||
|   | |||||||
| @@ -1 +1,2 @@ | |||||||
|  | #include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig" | ||||||
| #include "ephemeral/Flutter-Generated.xcconfig" | #include "ephemeral/Flutter-Generated.xcconfig" | ||||||
|   | |||||||
| @@ -1 +1,2 @@ | |||||||
|  | #include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig" | ||||||
| #include "ephemeral/Flutter-Generated.xcconfig" | #include "ephemeral/Flutter-Generated.xcconfig" | ||||||
|   | |||||||
							
								
								
									
										43
									
								
								frontend/macos/Podfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,43 @@ | |||||||
|  | platform :osx, '10.14' | ||||||
|  |  | ||||||
|  | # CocoaPods analytics sends network stats synchronously affecting flutter build latency. | ||||||
|  | ENV['COCOAPODS_DISABLE_STATS'] = 'true' | ||||||
|  |  | ||||||
|  | project 'Runner', { | ||||||
|  |   'Debug' => :debug, | ||||||
|  |   'Profile' => :release, | ||||||
|  |   'Release' => :release, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | def flutter_root | ||||||
|  |   generated_xcode_build_settings_path = File.expand_path(File.join('..', 'Flutter', 'ephemeral', 'Flutter-Generated.xcconfig'), __FILE__) | ||||||
|  |   unless File.exist?(generated_xcode_build_settings_path) | ||||||
|  |     raise "#{generated_xcode_build_settings_path} must exist. If you're running pod install manually, make sure \"flutter pub get\" is executed first" | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   File.foreach(generated_xcode_build_settings_path) do |line| | ||||||
|  |     matches = line.match(/FLUTTER_ROOT\=(.*)/) | ||||||
|  |     return matches[1].strip if matches | ||||||
|  |   end | ||||||
|  |   raise "FLUTTER_ROOT not found in #{generated_xcode_build_settings_path}. Try deleting Flutter-Generated.xcconfig, then run \"flutter pub get\"" | ||||||
|  | end | ||||||
|  |  | ||||||
|  | require File.expand_path(File.join('packages', 'flutter_tools', 'bin', 'podhelper'), flutter_root) | ||||||
|  |  | ||||||
|  | flutter_macos_podfile_setup | ||||||
|  |  | ||||||
|  | target 'Runner' do | ||||||
|  |   use_frameworks! | ||||||
|  |   use_modular_headers! | ||||||
|  |  | ||||||
|  |   flutter_install_all_macos_pods File.dirname(File.realpath(__FILE__)) | ||||||
|  |   target 'RunnerTests' do | ||||||
|  |     inherit! :search_paths | ||||||
|  |   end | ||||||
|  | end | ||||||
|  |  | ||||||
|  | post_install do |installer| | ||||||
|  |   installer.pods_project.targets.each do |target| | ||||||
|  |     flutter_additional_macos_build_settings(target) | ||||||
|  |   end | ||||||
|  | end | ||||||