Compare commits
	
		
			38 Commits
		
	
	
		
			v0.0.21
			...
			c2f3ef1d32
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| c2f3ef1d32 | |||
| 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
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -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
 | 
				
			||||||
    )
 | 
					    )
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -14,26 +14,22 @@ 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
 | 
					    time_to_reach_next : Optional[int] = 0
 | 
				
			||||||
    next_uuid : Optional[str] = None                            # TODO implement this ASAP
 | 
					    next_uuid : Optional[str] = None
 | 
				
			||||||
 | 
					 | 
				
			||||||
    def __hash__(self) -> int:
 | 
					 | 
				
			||||||
        return self.uuid.int
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    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 ""
 | 
				
			||||||
@@ -42,3 +38,15 @@ class Landmark(BaseModel) :
 | 
				
			|||||||
        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,7 +22,8 @@ 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)
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -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
 | 
				
			||||||
    )
 | 
					    )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -75,5 +75,6 @@ def test(start_coords: tuple[float, float], finish_coords: tuple[float, float] =
 | 
				
			|||||||
# 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:
 | 
				
			||||||
@@ -77,79 +74,36 @@ class LandmarkManager:
 | 
				
			|||||||
        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":
 | 
				
			||||||
@@ -336,46 +286,42 @@ class LandmarkManager:
 | 
				
			|||||||
                            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
 | 
				
			||||||
@@ -134,6 +135,21 @@ class Refiner :
 | 
				
			|||||||
    
 | 
					    
 | 
				
			||||||
        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)
 | 
					    return sorted_landmarks[:n_important]
 | 
				
			||||||
    for i, elem in enumerate(L_copy) :
 | 
					 | 
				
			||||||
        scores[i] = elem.attractiveness
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    res = sorted(range(len(scores)), key = lambda sub: scores[sub])[-(N_important-L):]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    for i, elem in enumerate(L_copy) :
 | 
					 | 
				
			||||||
        if i in res :
 | 
					 | 
				
			||||||
            L_clean.append(elem)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    return L_clean
 | 
					 | 
				
			||||||
 
 | 
				
			|||||||
@@ -37,14 +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_NAME=${REF_NAME//v}" >> $GITHUB_ENV
 | 
					          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_GOOGLE_PLAY_JSON }}" > google-key.json
 | 
					          echo "${{ steps.load-secrets.outputs.ANDROID_GOOGLE_PLAY_JSON_BASE64 }}" | base64 -d > google-key.json
 | 
				
			||||||
          # decode the base64 encoded google key
 | 
					          echo "${{ steps.load-secrets.outputs.ANDROID_KEYSTORE_BASE64 }}" | base64 -d > release.keystore
 | 
				
			||||||
          base64 -d ${{ secrets.ANDROID_KEYSTORE_BASE64 }} > release.keystore
 | 
					 | 
				
			||||||
        working-directory: android
 | 
					        working-directory: android
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      - name: Install fastlane
 | 
					      - name: Install fastlane
 | 
				
			||||||
@@ -52,6 +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
 | 
				
			||||||
        # the environment variable VERSION_NAME is implicitly available
 | 
					        env:
 | 
				
			||||||
 | 
					          BUILD_NUMBER: ${{ github.run_number }}
 | 
				
			||||||
 | 
					          # BUILD_NAME is implicitly available
 | 
				
			||||||
 | 
					          GOOGLE_MAPS_API_KEY: ${{ steps.load-secrets.outputs.GOOGLE_MAPS_API_KEY }}
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										64
									
								
								frontend/.github/workflows/build_app_ios.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						@@ -0,0 +1,64 @@
 | 
				
			|||||||
 | 
					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_P8 | IOS_ASC_KEY_P8 ;
 | 
				
			||||||
 | 
					            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: ${{ GOOGLE_MAPS_API_KEY.IOS_ASC_KEY_ID }}
 | 
				
			||||||
 | 
					          IOS_ASC_ISSUER_ID: ${{ GOOGLE_MAPS_API_KEY.IOS_ASC_ISSUER_ID }}
 | 
				
			||||||
 | 
					          IOS_ASC_KEY_P8: ${{ GOOGLE_MAPS_API_KEY.IOS_ASC_KEY_P8 }}
 | 
				
			||||||
@@ -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_...`
 | 
				
			||||||
@@ -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,42 +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
 | 
				
			||||||
    version_name = ENV["VERSION_NAME"]
 | 
					    build_name = ENV["BUILD_NAME"]
 | 
				
			||||||
 | 
					    build_number = ENV["BUILD_NUMBER"]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    sh(
 | 
					    sh(
 | 
				
			||||||
      "flutter",
 | 
					      "flutter",
 | 
				
			||||||
      "build",
 | 
					      "build",
 | 
				
			||||||
      "appbundle",
 | 
					      "appbundle",
 | 
				
			||||||
      "--release",
 | 
					      "--release",
 | 
				
			||||||
      "--build-name=#{version_name}",
 | 
					      "--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.provideAPI(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: "testflight",
 | 
				
			||||||
 | 
					      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,37 +1,27 @@
 | 
				
			|||||||
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> {
 | 
				
			||||||
  
 | 
					  @override
 | 
				
			||||||
  Widget greeterBuilder (BuildContext context, Widget? child) {
 | 
					  Widget build(BuildContext context) => Center(
 | 
				
			||||||
    final Shader textGradient = APP_GRADIENT.createShader(Rect.fromLTWH(0.0, 0.0, 200.0, 70.0));
 | 
					    child: FutureBuilder(
 | 
				
			||||||
    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,
 | 
					      future: widget.trip.cityName,
 | 
				
			||||||
      builder: (BuildContext context, AsyncSnapshot<String> snapshot) {
 | 
					      builder: (BuildContext context, AsyncSnapshot<String> snapshot) {
 | 
				
			||||||
        if (snapshot.hasData) {
 | 
					        if (snapshot.hasData) {
 | 
				
			||||||
@@ -41,7 +31,6 @@ class _GreeterState extends State<Greeter> {
 | 
				
			|||||||
            style: greeterStyle
 | 
					            style: greeterStyle
 | 
				
			||||||
          );
 | 
					          );
 | 
				
			||||||
        } else if (snapshot.hasError) {
 | 
					        } else if (snapshot.hasError) {
 | 
				
			||||||
            log('Error while fetching city name');
 | 
					 | 
				
			||||||
          return AutoSizeText(
 | 
					          return AutoSizeText(
 | 
				
			||||||
            maxLines: 1,
 | 
					            maxLines: 1,
 | 
				
			||||||
            'Welcome to your trip!',
 | 
					            'Welcome to your trip!',
 | 
				
			||||||
@@ -55,57 +44,7 @@ class _GreeterState extends State<Greeter> {
 | 
				
			|||||||
          );
 | 
					          );
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    } 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
 | 
					 | 
				
			||||||
  Widget build(BuildContext context) {
 | 
					 | 
				
			||||||
    return ListenableBuilder(
 | 
					 | 
				
			||||||
      listenable: widget.trip,
 | 
					 | 
				
			||||||
      builder: greeterBuilder,
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
							
								
								
									
										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;
 | 
				
			||||||
 | 
					  try {
 | 
				
			||||||
 | 
					     response = await dio.post(
 | 
				
			||||||
      "/trip/new",
 | 
					      "/trip/new",
 | 
				
			||||||
      data: data
 | 
					      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
 | 
				
			||||||