Compare commits
1 Commits
7acfb84122
...
v0.0.21
Author | SHA1 | Date | |
---|---|---|---|
1e54ff45d5 |
@@ -6,7 +6,7 @@ on:
|
|||||||
- frontend/**
|
- frontend/**
|
||||||
|
|
||||||
|
|
||||||
name: Build and release debug APK
|
name: Build and release 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 --debug --split-per-abi --build-number=${{ gitea.run_number }}
|
- run: flutter build apk --release --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 }} --kubeconfig=kubeconfig
|
kubectl -n anyway-backend rollout restart deployment/anyway-backend-${{ inputs.overlay }}
|
||||||
|
3
.vscode/settings.json
vendored
@@ -1,3 +0,0 @@
|
|||||||
{
|
|
||||||
"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,37 +1,12 @@
|
|||||||
# Backend
|
# Backend
|
||||||
|
|
||||||
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.
|
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.
|
||||||
|
|
||||||
|
|
||||||
## Getting Started
|
## Getting Started
|
||||||
|
- The code of the python application is located in the `src` directory.
|
||||||
### Directory Structure
|
- Package management is handled with `pipenv` and the dependencies are listed in the `Pipfile`.
|
||||||
- The code for the Python application is located in the `src` directory.
|
- Since the application is aimed to be deployed in a container, the `Dockerfile` is provided to build the image.
|
||||||
- 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") == "true":
|
if os.getenv('DEBUG', False):
|
||||||
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
|
# upon creation of the trip, persistence of both the trip and its landmarks is ensured. Ca
|
||||||
trip = Trip.from_linked_landmarks(linked_tour, cache_client)
|
trip = Trip.from_linked_landmarks(linked_tour, cache_client)
|
||||||
return trip
|
return trip
|
||||||
|
|
||||||
@@ -84,4 +84,4 @@ def get_landmark(landmark_uuid: str) -> Landmark:
|
|||||||
landmark = cache_client.get(f"landmark_{landmark_uuid}")
|
landmark = cache_client.get(f"landmark_{landmark_uuid}")
|
||||||
return landmark
|
return landmark
|
||||||
except KeyError:
|
except KeyError:
|
||||||
raise HTTPException(status_code=404, detail="Landmark not found")
|
raise HTTPException(status_code=404, detail="Landmark not found")
|
@@ -1,6 +1,3 @@
|
|||||||
# 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: ''
|
||||||
@@ -14,24 +11,7 @@ nature:
|
|||||||
- alpine_hut
|
- alpine_hut
|
||||||
- viewpoint
|
- viewpoint
|
||||||
- zoo
|
- zoo
|
||||||
- resort
|
waterway: waterfall
|
||||||
- picnic_site
|
|
||||||
water:
|
|
||||||
- pond
|
|
||||||
- lake
|
|
||||||
- river
|
|
||||||
- basin
|
|
||||||
- stream
|
|
||||||
- lagoon
|
|
||||||
- rapids
|
|
||||||
waterway:
|
|
||||||
- waterfall
|
|
||||||
- river
|
|
||||||
- canal
|
|
||||||
- dam
|
|
||||||
- dock
|
|
||||||
- boatyard
|
|
||||||
|
|
||||||
|
|
||||||
shopping:
|
shopping:
|
||||||
shop:
|
shop:
|
||||||
@@ -43,47 +23,10 @@ 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.5
|
church_coeff: 0.75
|
||||||
nature_coeff: 1.25
|
nature_coeff: 1.25
|
||||||
overall_coeff: 10
|
overall_coeff: 10
|
||||||
tag_exponent: 1.15
|
tag_exponent: 1.15
|
||||||
|
@@ -21,8 +21,8 @@ if constants.MEMCACHED_HOST_PATH is None:
|
|||||||
else:
|
else:
|
||||||
client = Client(
|
client = Client(
|
||||||
constants.MEMCACHED_HOST_PATH,
|
constants.MEMCACHED_HOST_PATH,
|
||||||
timeout = 1,
|
timeout=1,
|
||||||
allow_unicode_keys = True,
|
allow_unicode_keys=True,
|
||||||
encoding = 'utf-8',
|
encoding='utf-8',
|
||||||
serde = serde.pickle_serde
|
serde=serde.pickle_serde
|
||||||
)
|
)
|
||||||
|
@@ -5,7 +5,7 @@ from uuid import uuid4
|
|||||||
|
|
||||||
# Output to frontend
|
# Output to frontend
|
||||||
class Landmark(BaseModel) :
|
class Landmark(BaseModel) :
|
||||||
|
|
||||||
# Properties of the landmark
|
# Properties of the landmark
|
||||||
name : str
|
name : str
|
||||||
type: Literal['sightseeing', 'nature', 'shopping', 'start', 'finish']
|
type: Literal['sightseeing', 'nature', 'shopping', 'start', 'finish']
|
||||||
@@ -14,39 +14,31 @@ class Landmark(BaseModel) :
|
|||||||
osm_id : int
|
osm_id : int
|
||||||
attractiveness : int
|
attractiveness : int
|
||||||
n_tags : int
|
n_tags : int
|
||||||
image_url : Optional[str] = None
|
image_url : Optional[str] = None # TODO future
|
||||||
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
|
duration : Optional[int] = 0 # TODO future
|
||||||
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)
|
uuid: str = Field(default_factory=uuid4) # TODO implement this ASAP
|
||||||
|
|
||||||
# Additional properties depending on specific tour
|
# Additional properties depending on specific tour
|
||||||
must_do : Optional[bool] = False
|
must_do : Optional[bool] = False
|
||||||
must_avoid : Optional[bool] = False
|
must_avoid : Optional[bool] = False
|
||||||
is_secondary : Optional[bool] = False # TODO future
|
is_secondary : Optional[bool] = False # TODO future
|
||||||
|
|
||||||
|
time_to_reach_next : Optional[int] = 0 # TODO fix this in existing code
|
||||||
|
next_uuid : Optional[str] = None # TODO implement this ASAP
|
||||||
|
|
||||||
time_to_reach_next : Optional[int] = 0
|
def __hash__(self) -> int:
|
||||||
next_uuid : Optional[str] = None
|
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 ""
|
||||||
is_secondary_str = f", secondary" if self.is_secondary else ""
|
is_secondary_str = f", secondary" if self.is_secondary else ""
|
||||||
type_str = '(' + self.type + ')'
|
type_str = '(' + self.type + ')'
|
||||||
if self.type in ["start", "finish", "nature", "shopping"] : type_str += '\t '
|
if self.type in ["start", "finish", "nature", "shopping"] : type_str += '\t '
|
||||||
return f'Landmark{type_str}: [{self.name} @{self.location}, score={self.attractiveness}{time_to_next_str}{is_secondary_str}]'
|
return f'Landmark{type_str}: [{self.name} @{self.location}, score={self.attractiveness}{time_to_next_str}{is_secondary_str}]'
|
||||||
|
|
||||||
def distance(self, value: 'Landmark') -> float:
|
|
||||||
return (self.location[0] - value.location[0])**2 + (self.location[1] - value.location[1])**2
|
|
||||||
|
|
||||||
def __hash__(self) -> int:
|
|
||||||
return hash(self.name)
|
|
||||||
|
|
||||||
def __eq__(self, value: 'Landmark') -> bool:
|
|
||||||
# eq and hash must be consistent
|
|
||||||
# in particular, if two objects are equal, their hash must be equal
|
|
||||||
# uuid and osm_id are just shortcuts to avoid comparing all the properties
|
|
||||||
# if they are equal, we know that the name is also equal and in turn the hash is equal
|
|
||||||
return self.uuid == value.uuid or self.osm_id == value.osm_id or (self.name == value.name and self.distance(value) < 0.001)
|
|
||||||
|
@@ -22,10 +22,9 @@ 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)
|
||||||
# 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)
|
||||||
cache_client.set_many({f"landmark_{landmark.uuid}": landmark for landmark in landmarks}, expire=3600, noreply=False)
|
|
||||||
# is equivalent to:
|
# is equivalent to:
|
||||||
# for landmark in landmarks:
|
# for landmark in landmarks:
|
||||||
# cache_client.set(f"landmark_{landmark.uuid}", landmark, expire=3600)
|
# cache_client.set(f"landmark_{landmark.uuid}", landmark, expire=3600)
|
||||||
|
|
||||||
return trip
|
return trip
|
@@ -23,7 +23,7 @@ def test(start_coords: tuple[float, float], finish_coords: tuple[float, float] =
|
|||||||
sightseeing=Preference(type='sightseeing', score = 5),
|
sightseeing=Preference(type='sightseeing', score = 5),
|
||||||
nature=Preference(type='nature', score = 5),
|
nature=Preference(type='nature', score = 5),
|
||||||
shopping=Preference(type='shopping', score = 5),
|
shopping=Preference(type='shopping', score = 5),
|
||||||
max_time_minute=15,
|
max_time_minute=100,
|
||||||
detour_tolerance_minute=0
|
detour_tolerance_minute=0
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -74,7 +74,6 @@ def test(start_coords: tuple[float, float], finish_coords: tuple[float, float] =
|
|||||||
# test(tuple((48.8344400, 2.3220540))) # Café Chez César
|
# test(tuple((48.8344400, 2.3220540))) # Café Chez César
|
||||||
# test(tuple((48.8375946, 2.2949904))) # Point random
|
# test(tuple((48.8375946, 2.2949904))) # Point random
|
||||||
# test(tuple((47.377859, 8.540585))) # Zurich HB
|
# test(tuple((47.377859, 8.540585))) # Zurich HB
|
||||||
# test(tuple((45.758217, 4.831814))) # Lyon Bellecour
|
# test(tuple((45.758217, 4.831814))) # Lyon Bellecour
|
||||||
# test(tuple((48.5848435, 7.7332974))) # Strasbourg Gare
|
test(tuple((48.5848435, 7.7332974))) # Strasbourg Gare
|
||||||
# test(tuple((48.2067858, 16.3692340))) # Vienne
|
# test(tuple((48.2067858, 16.3692340))) # Vienne
|
||||||
test(tuple((48.084588, 7.280405))) # Turckheim
|
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import yaml
|
import yaml
|
||||||
from math import sin, cos, sqrt, atan2, radians
|
from geopy.distance import geodesic
|
||||||
|
|
||||||
import constants
|
import constants
|
||||||
|
|
||||||
@@ -8,7 +8,6 @@ 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:
|
||||||
"""
|
"""
|
||||||
@@ -17,34 +16,24 @@ 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.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
|
||||||
if p1 == p2:
|
# Compute the straight-line distance in km
|
||||||
|
if p1 == p2 :
|
||||||
return 0
|
return 0
|
||||||
else:
|
else:
|
||||||
# Compute the distance in km along the surface of the Earth
|
dist = geodesic(p1, p2).kilometers
|
||||||
# (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])
|
|
||||||
|
|
||||||
dlon = lon2 - lon1
|
# Consider the detour factor for average cityto deterline walking distance (in km)
|
||||||
dlat = lat2 - lat1
|
walk_dist = dist*DETOUR_FACTOR
|
||||||
|
|
||||||
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_distance / AVERAGE_WALKING_SPEED * 60
|
walk_time = walk_dist/AVERAGE_WALKING_SPEED*60
|
||||||
|
|
||||||
return round(walk_time)
|
return round(walk_time)
|
||||||
|
@@ -1,17 +1,20 @@
|
|||||||
import math
|
import math as m
|
||||||
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
|
from structs.preferences import Preferences, Preference
|
||||||
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:
|
||||||
@@ -43,7 +46,7 @@ class LandmarkManager:
|
|||||||
self.viewpoint_bonus = parameters['viewpoint_bonus']
|
self.viewpoint_bonus = parameters['viewpoint_bonus']
|
||||||
self.pay_bonus = parameters['pay_bonus']
|
self.pay_bonus = parameters['pay_bonus']
|
||||||
self.N_important = parameters['N_important']
|
self.N_important = parameters['N_important']
|
||||||
|
|
||||||
with constants.OPTIMIZER_PARAMETERS_PATH.open('r') as f:
|
with constants.OPTIMIZER_PARAMETERS_PATH.open('r') as f:
|
||||||
parameters = yaml.safe_load(f)
|
parameters = yaml.safe_load(f)
|
||||||
self.walking_speed = parameters['average_walking_speed']
|
self.walking_speed = parameters['average_walking_speed']
|
||||||
@@ -66,44 +69,87 @@ class LandmarkManager:
|
|||||||
preferences (Preferences): The user's preference settings that influence the landmark selection.
|
preferences (Preferences): The user's preference settings that influence the landmark selection.
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
tuple[list[Landmark], list[Landmark]]:
|
tuple[list[Landmark], list[Landmark]]:
|
||||||
- A list of all existing landmarks.
|
- A list of all existing landmarks.
|
||||||
- A list of the most important landmarks based on the user's preferences.
|
- A list of the most important landmarks based on the user's preferences.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
max_walk_dist = (preferences.max_time_minute/2)/60*self.walking_speed*1000/self.detour_factor
|
max_walk_dist = (preferences.max_time_minute/2)/60*self.walking_speed*1000/self.detour_factor
|
||||||
reachable_bbox_side = min(max_walk_dist, self.max_bbox_side)
|
reachable_bbox_side = min(max_walk_dist, self.max_bbox_side)
|
||||||
|
|
||||||
# use set to avoid duplicates, this requires some __methods__ to be set in Landmark
|
L = []
|
||||||
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: score * 10 * preferences.sightseeing.score / 5
|
score_function = lambda score: int(score*10*preferences.sightseeing.score/5) # self.count_elements_close_to(loc) +
|
||||||
current_landmarks = self.fetch_landmarks(bbox, self.amenity_selectors['sightseeing'], preferences.sightseeing.type, score_function)
|
L1 = self.fetch_landmarks(bbox, self.amenity_selectors['sightseeing'], preferences.sightseeing.type, score_function)
|
||||||
all_landmarks.update(current_landmarks)
|
L += L1
|
||||||
|
|
||||||
# list for nature
|
# list for nature
|
||||||
if preferences.nature.score != 0:
|
if preferences.nature.score != 0:
|
||||||
score_function = lambda score: score * 10 * self.nature_coeff * preferences.nature.score / 5
|
score_function = lambda score: int(score*10*self.nature_coeff*preferences.nature.score/5) # self.count_elements_close_to(loc) +
|
||||||
current_landmarks = self.fetch_landmarks(bbox, self.amenity_selectors['nature'], preferences.nature.type, score_function)
|
L2 = self.fetch_landmarks(bbox, self.amenity_selectors['nature'], preferences.nature.type, score_function)
|
||||||
all_landmarks.update(current_landmarks)
|
L += L2
|
||||||
|
|
||||||
# list for shopping
|
# list for shopping
|
||||||
if preferences.shopping.score != 0:
|
if preferences.shopping.score != 0:
|
||||||
score_function = lambda score: score * 10 * preferences.shopping.score / 5
|
score_function = lambda score: int(score*10*preferences.shopping.score/5) # self.count_elements_close_to(loc) +
|
||||||
current_landmarks = self.fetch_landmarks(bbox, self.amenity_selectors['shopping'], preferences.shopping.type, score_function)
|
L3 = self.fetch_landmarks(bbox, self.amenity_selectors['shopping'], preferences.shopping.type, score_function)
|
||||||
# set time for all shopping activites :
|
L += L3
|
||||||
for landmark in current_landmarks : landmark.duration = 45
|
|
||||||
all_landmarks.update(current_landmarks)
|
|
||||||
|
|
||||||
|
|
||||||
landmarks_constrained = take_most_important(all_landmarks, self.N_important)
|
L = self.remove_duplicates(L)
|
||||||
self.logger.info(f'Generated {len(all_landmarks)} landmarks around {center_coordinates}, and constrained to {len(landmarks_constrained)} most important ones.')
|
# self.correct_score(L, preferences)
|
||||||
|
|
||||||
return all_landmarks, landmarks_constrained
|
L_constrained = take_most_important(L, self.N_important)
|
||||||
|
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:
|
||||||
@@ -126,7 +172,7 @@ class LandmarkManager:
|
|||||||
|
|
||||||
radius = self.radius_close_to
|
radius = self.radius_close_to
|
||||||
|
|
||||||
alpha = (180 * radius) / (6371000 * math.pi)
|
alpha = (180*radius) / (6371000*m.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
|
||||||
@@ -170,7 +216,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 * math.cos(math.radians(lat))) # Adjust for longitude based on latitude
|
lon_diff = half_side_length_km / (111 * m.cos(m.radians(lat))) # Adjust for longitude based on latitude
|
||||||
|
|
||||||
# Calculate bbox
|
# Calculate bbox
|
||||||
min_lat = lat - lat_diff
|
min_lat = lat - lat_diff
|
||||||
@@ -209,32 +255,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 = ['count_tags()>5'],
|
# conditions = [],
|
||||||
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}")
|
||||||
continue
|
return
|
||||||
|
|
||||||
for elem in result.elements():
|
for elem in result.elements():
|
||||||
|
|
||||||
name = elem.tag('name')
|
name = elem.tag('name') # Add name
|
||||||
location = (elem.centerLat(), elem.centerLon())
|
location = (elem.centerLat(), elem.centerLon()) # Add coordinates (lat, lon)
|
||||||
|
|
||||||
# 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
|
||||||
@@ -245,32 +291,36 @@ 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
|
||||||
|
|
||||||
# Adjust scoring
|
# remove specific tags
|
||||||
skip = False
|
skip = False
|
||||||
for tag in elem.tags().keys():
|
for tag in elem.tags().keys():
|
||||||
if "pay" in tag:
|
if "pay" in tag:
|
||||||
# payment options are misleading and should not count for the scoring.
|
score += self.pay_bonus # discard payment options for tags
|
||||||
score += self.pay_bonus
|
|
||||||
|
|
||||||
if "disused" in tag:
|
if "disused" in tag:
|
||||||
# skip disused amenities
|
skip = True # skip disused amenities
|
||||||
skip = True
|
|
||||||
break
|
break
|
||||||
|
|
||||||
if "wiki" in tag:
|
if "wiki" in tag:
|
||||||
# wikipedia entries count more
|
score += self.wikipedia_bonus # 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":
|
||||||
@@ -285,43 +335,47 @@ class LandmarkManager:
|
|||||||
if tag == "building" and elem.tag('building') in ['retail', 'supermarket', 'parking']:
|
if tag == "building" and elem.tag('building') in ['retail', 'supermarket', 'parking']:
|
||||||
skip = True
|
skip = True
|
||||||
break
|
break
|
||||||
|
|
||||||
# Extract image, website and english name
|
# Get additional information
|
||||||
if tag in ['website', 'contact:website']:
|
# if tag == 'wikipedia' :
|
||||||
|
# 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 = score * self.church_coeff
|
score = int(score*self.church_coeff)
|
||||||
duration = 10
|
duration = 15
|
||||||
|
|
||||||
elif "museum" in elem.tags().values() or "aquarium" in elem.tags().values() or "planetarium" in elem.tags().values():
|
elif "museum" in elem.tags().values() :
|
||||||
|
score = int(score*self.church_coeff)
|
||||||
duration = 60
|
duration = 60
|
||||||
|
|
||||||
else:
|
else :
|
||||||
duration = 5
|
duration = 5
|
||||||
|
|
||||||
# finally create our own landmark object
|
# Generate the landmark and append it to the list
|
||||||
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 = int(score),
|
attractiveness=score,
|
||||||
must_do = False,
|
must_do=False,
|
||||||
n_tags = int(n_tags),
|
n_tags=int(n_tags),
|
||||||
duration = int(duration),
|
duration = duration,
|
||||||
name_en = name_en,
|
name_en=name_en,
|
||||||
image_url = image_url,
|
image_url=image_url,
|
||||||
website_url = website_url
|
# wikipedia_url=wikpedia_url,
|
||||||
|
website_url=website_url
|
||||||
)
|
)
|
||||||
return_list.append(landmark)
|
return_list.append(landmark)
|
||||||
|
|
||||||
@@ -345,7 +399,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,6 +3,7 @@ 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
|
||||||
@@ -487,7 +488,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. Try with a longer trip (>30 minutes).")
|
raise ArithmeticError("No solution could be found, the problem is overconstrained. Please adapt your must_dos")
|
||||||
|
|
||||||
# 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,7 +2,6 @@ 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,21 +133,6 @@ class Refiner :
|
|||||||
i += 1
|
i += 1
|
||||||
|
|
||||||
return tour
|
return tour
|
||||||
|
|
||||||
def integrate_landmarks(self, sub_list: List[Landmark], main_list: List[Landmark]) :
|
|
||||||
"""
|
|
||||||
Inserts 'sub_list' of Landmarks inside the 'main_list' by leaving the ends untouched.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
sub_list : the list of Landmarks to be inserted inside of the 'main_list'.
|
|
||||||
main_list : the original list with start and finish.
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
the full list.
|
|
||||||
"""
|
|
||||||
sub_list.append(main_list[-1]) # add finish back
|
|
||||||
return main_list[:-1] + sub_list # create full set of possible landmarks
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def find_shortest_path_through_all_landmarks(self, landmarks: list[Landmark]) -> tuple[list[Landmark], Polygon]:
|
def find_shortest_path_through_all_landmarks(self, landmarks: list[Landmark]) -> tuple[list[Landmark], Polygon]:
|
||||||
@@ -269,11 +253,6 @@ 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
|
||||||
@@ -336,30 +315,26 @@ 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 = self.integrate_landmarks(minor_landmarks, base_tour) # could probably be optimized with less overhead
|
full_set = base_tour[:-1] + minor_landmarks # create full set of possible landmarks (without finish)
|
||||||
|
full_set.append(base_tour[-1]) # add finish back
|
||||||
|
|
||||||
# Generate a new tour with the optimizer.
|
# get a new tour
|
||||||
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,16 +1,38 @@
|
|||||||
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)
|
||||||
Given a list of landmarks, return the n_important most important landmarks
|
L_copy = []
|
||||||
Parameters:
|
L_clean = []
|
||||||
landmarks: list[Landmark] - list of landmarks
|
scores = [0]*len(landmarks)
|
||||||
n_important: int - number of most important landmarks to return
|
names = []
|
||||||
Returns:
|
name_id = {}
|
||||||
list[Landmark] - list of the n_important most important landmarks
|
|
||||||
"""
|
|
||||||
|
|
||||||
# Sort landmarks by attractiveness (descending)
|
for i, elem in enumerate(landmarks) :
|
||||||
sorted_landmarks = sorted(landmarks, key=lambda x: x.attractiveness, reverse=True)
|
if elem.name not in names :
|
||||||
|
names.append(elem.name)
|
||||||
|
name_id[elem.name] = [i]
|
||||||
|
L_copy.append(elem)
|
||||||
|
else :
|
||||||
|
name_id[elem.name] += [i]
|
||||||
|
scores = []
|
||||||
|
for j in name_id[elem.name] :
|
||||||
|
scores.append(L[j].attractiveness)
|
||||||
|
best_id = max(range(len(scores)), key=scores.__getitem__)
|
||||||
|
t = name_id[elem.name][best_id]
|
||||||
|
if t == i :
|
||||||
|
for old in L_copy :
|
||||||
|
if old.name == elem.name :
|
||||||
|
old.attractiveness = L[t].attractiveness
|
||||||
|
|
||||||
|
scores = [0]*len(L_copy)
|
||||||
|
for i, elem in enumerate(L_copy) :
|
||||||
|
scores[i] = elem.attractiveness
|
||||||
|
|
||||||
return sorted_landmarks[:n_important]
|
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,25 +37,14 @@ 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 "BUILD_NAME=${REF_NAME//v}" >> $GITHUB_ENV
|
echo "VERSION_NAME=${REF_NAME//v}" >> $GITHUB_ENV
|
||||||
|
|
||||||
- name: Load secrets
|
- name: Load secrets from github
|
||||||
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 "${{ steps.load-secrets.outputs.ANDROID_SECRET_PROPERTIES_BASE64 }}" | base64 -d > secrets.properties
|
echo "${{ secrets.ANDROID_SECRET_PROPERTIES }}" > secrets.properties
|
||||||
echo "${{ steps.load-secrets.outputs.ANDROID_GOOGLE_PLAY_JSON_BASE64 }}" | base64 -d > google-key.json
|
echo "${{ secrets.ANDROID_GOOGLE_PLAY_JSON }}" > google-key.json
|
||||||
echo "${{ steps.load-secrets.outputs.ANDROID_KEYSTORE_BASE64 }}" | base64 -d > release.keystore
|
# decode the base64 encoded google key
|
||||||
|
base64 -d ${{ secrets.ANDROID_KEYSTORE_BASE64 }} > release.keystore
|
||||||
working-directory: android
|
working-directory: android
|
||||||
|
|
||||||
- name: Install fastlane
|
- name: Install fastlane
|
||||||
@@ -63,9 +52,6 @@ jobs:
|
|||||||
working-directory: android
|
working-directory: android
|
||||||
|
|
||||||
- name: Run fastlane lane
|
- name: Run fastlane lane
|
||||||
run: bundle exec fastlane deploy_release
|
run: bundle exec fastlane deploy_testing
|
||||||
working-directory: android
|
working-directory: android
|
||||||
env:
|
# the environment variable VERSION_NAME is implicitly available
|
||||||
BUILD_NUMBER: ${{ github.run_number }}
|
|
||||||
# BUILD_NAME is implicitly available
|
|
||||||
GOOGLE_MAPS_API_KEY: ${{ steps.load-secrets.outputs.GOOGLE_MAPS_API_KEY }}
|
|
||||||
|
65
frontend/.github/workflows/build_app_ios.yaml
vendored
@@ -1,65 +0,0 @@
|
|||||||
on:
|
|
||||||
push:
|
|
||||||
tags:
|
|
||||||
- 'v*'
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
build:
|
|
||||||
runs-on: macos-latest
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Set up ruby env
|
|
||||||
uses: ruby/setup-ruby@v1
|
|
||||||
with:
|
|
||||||
ruby-version: 3.2.1
|
|
||||||
bundler-cache: true
|
|
||||||
|
|
||||||
- name: Install Flutter
|
|
||||||
uses: subosito/flutter-action@v2
|
|
||||||
with:
|
|
||||||
channel: stable
|
|
||||||
flutter-version: 3.22.0
|
|
||||||
cache: true
|
|
||||||
|
|
||||||
- name: Infer version number from git tag
|
|
||||||
id: version
|
|
||||||
env:
|
|
||||||
REF_NAME: ${{ github.ref_name }}
|
|
||||||
run:
|
|
||||||
# remove the 'v' prefix from the tag name
|
|
||||||
echo "BUILD_NAME=${REF_NAME//v}" >> $GITHUB_ENV
|
|
||||||
|
|
||||||
- name: Load secrets
|
|
||||||
id: load-secrets
|
|
||||||
uses: hashicorp/vault-action@v3
|
|
||||||
with:
|
|
||||||
url: https://api.hashicorp.com
|
|
||||||
token: ${{ secrets.VAULT_TOKEN }}
|
|
||||||
secrets: |
|
|
||||||
secret/release GOOGLE_MAPS_API_KEY | GOOGLE_MAPS_API_KEY ;
|
|
||||||
secret/release IOS_ASC_KEY_ID | IOS_ASC_KEY_ID ;
|
|
||||||
secret/release IOS_ASC_ISSUER_ID | IOS_ASC_ISSUER_ID ;
|
|
||||||
secret/release IOS_ASC_KEY | IOS_ASC_KEY ;
|
|
||||||
secret/release IOS_MATCH_REPO_SSH_KEY_BASE64 | IOS_MATCH_REPO_SSH_KEY_BASE64 ;
|
|
||||||
|
|
||||||
- name: Setup SSH key for match git repo
|
|
||||||
run: echo "$MATCH_REPO_SSH_KEY" | base64 --decode > ~/.ssh/id_rsa && chmod 600 ~/.ssh/id_rsa
|
|
||||||
env:
|
|
||||||
MATCH_REPO_SSH_KEY: ${{ steps.load-secrets.outputs.IOS_MATCH_REPO_SSH_KEY_BASE64 }}
|
|
||||||
|
|
||||||
- name: Install fastlane
|
|
||||||
run: bundle install
|
|
||||||
working-directory: ios
|
|
||||||
|
|
||||||
- name: Run fastlane lane
|
|
||||||
run: bundle exec fastlane deploy_release
|
|
||||||
working-directory: ios
|
|
||||||
env:
|
|
||||||
BUILD_NUMBER: ${{ github.run_number }}
|
|
||||||
# BUILD_NAME is implicitly available
|
|
||||||
GOOGLE_MAPS_API_KEY: ${{ steps.load-secrets.outputs.GOOGLE_MAPS_API_KEY }}
|
|
||||||
IOS_ASC_KEY_ID: ${{ steps.load-secrets.outputs.IOS_ASC_KEY_ID }}
|
|
||||||
IOS_ASC_ISSUER_ID: ${{ steps.load-secrets.outputs.IOS_ASC_ISSUER_ID }}
|
|
||||||
IOS_ASC_KEY: ${{ steps.load-secrets.outputs.IOS_ASC_KEY }}
|
|
||||||
MATCH_PASSWORD: ${{ steps.load-secrets.outputs.IOS_MATCH_PASSWORD }}
|
|
@@ -46,17 +46,12 @@ 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.
|
||||||
|
|
||||||
Secrets used by fastlane are stored on hashicorp vault and are fetched by the CI/CD pipeline. See below.
|
Fastlane assumes mutliple secrets to be present as files in the platform directories. These are:
|
||||||
|
- 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
|
||||||
|
|
||||||
## Secrets
|
These files are stored as secrets in the GitHub repository so that the CI pipeline can access them.
|
||||||
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,3 +63,11 @@ 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,19 +30,14 @@ 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, fallback.properties) not found")
|
throw new GradleException("Secrets file secrets.properties not found")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -65,7 +60,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.
|
||||||
@@ -77,7 +72,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': System.getenv('GOOGLE_MAPS_API_KEY')]
|
manifestPlaceholders += ['MAPS_API_KEY': secretProperties.getProperty('MAPS_API_KEY')]
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -1,3 +1 @@
|
|||||||
# 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,52 +1,42 @@
|
|||||||
|
# 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 to closed testing (play store)"
|
desc "Deploy a new version as a preview version"
|
||||||
lane :deploy_testing do
|
lane :deploy_testing do
|
||||||
build_name = ENV["BUILD_NAME"]
|
version_name = ENV["VERSION_NAME"]
|
||||||
build_number = ENV["BUILD_NUMBER"]
|
|
||||||
|
|
||||||
sh(
|
sh(
|
||||||
"flutter",
|
"flutter",
|
||||||
"build",
|
"build",
|
||||||
"appbundle",
|
"appbundle",
|
||||||
"--release",
|
"--release",
|
||||||
"--build-name=#{build_name}",
|
"--build-name=#{version_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
|
||||||
build_name = ENV["BUILD_NAME"]
|
gradle(
|
||||||
build_number = ENV["BUILD_NUMBER"]
|
task: "clean assembleRelease",
|
||||||
|
properties: {
|
||||||
sh(
|
# loaded from environment
|
||||||
"flutter",
|
"android.injected.version.name" => ENV["VERSION_NAME"],
|
||||||
"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
|
||||||
|
After Width: | Height: | Size: 106 KiB |
After Width: | Height: | Size: 1.3 MiB |
After Width: | Height: | Size: 637 KiB |
After Width: | Height: | Size: 573 KiB |
After Width: | Height: | Size: 175 KiB |
After Width: | Height: | Size: 360 KiB |
After Width: | Height: | Size: 106 KiB |
After Width: | Height: | Size: 1.3 MiB |
After Width: | Height: | Size: 637 KiB |
After Width: | Height: | Size: 573 KiB |
After Width: | Height: | Size: 175 KiB |
After Width: | Height: | Size: 360 KiB |
6
frontend/ios/.gitignore
vendored
@@ -1,9 +1,3 @@
|
|||||||
# fastlane secret
|
|
||||||
.env
|
|
||||||
secret.env
|
|
||||||
*.mobileprovision
|
|
||||||
report.xml
|
|
||||||
|
|
||||||
**/dgph
|
**/dgph
|
||||||
*.mode1v3
|
*.mode1v3
|
||||||
*.mode2v3
|
*.mode2v3
|
||||||
|
@@ -1,2 +1 @@
|
|||||||
#include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig"
|
|
||||||
#include "Generated.xcconfig"
|
#include "Generated.xcconfig"
|
||||||
|
@@ -1,2 +1 @@
|
|||||||
#include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig"
|
|
||||||
#include "Generated.xcconfig"
|
#include "Generated.xcconfig"
|
||||||
|
@@ -1,5 +0,0 @@
|
|||||||
source "https://rubygems.org"
|
|
||||||
|
|
||||||
gem "fastlane"
|
|
||||||
gem "cocoapods"
|
|
||||||
|
|
@@ -1,288 +0,0 @@
|
|||||||
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
|
|
@@ -1,44 +0,0 @@
|
|||||||
# 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
|
|
@@ -1,89 +0,0 @@
|
|||||||
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,11 +11,9 @@
|
|||||||
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 */
|
||||||
@@ -42,20 +40,14 @@
|
|||||||
/* 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; };
|
||||||
@@ -63,43 +55,19 @@
|
|||||||
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 = (
|
||||||
@@ -108,15 +76,6 @@
|
|||||||
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 = (
|
||||||
@@ -135,8 +94,6 @@
|
|||||||
97C146F01CF9000F007C117D /* Runner */,
|
97C146F01CF9000F007C117D /* Runner */,
|
||||||
97C146EF1CF9000F007C117D /* Products */,
|
97C146EF1CF9000F007C117D /* Products */,
|
||||||
331C8082294A63A400263BE5 /* RunnerTests */,
|
331C8082294A63A400263BE5 /* RunnerTests */,
|
||||||
1C946B8D83A95663C2489C91 /* Pods */,
|
|
||||||
3ECCC9BD7D0792871219624C /* Frameworks */,
|
|
||||||
);
|
);
|
||||||
sourceTree = "<group>";
|
sourceTree = "<group>";
|
||||||
};
|
};
|
||||||
@@ -171,10 +128,8 @@
|
|||||||
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 = (
|
||||||
);
|
);
|
||||||
@@ -190,15 +145,12 @@
|
|||||||
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 = (
|
||||||
);
|
);
|
||||||
@@ -270,28 +222,6 @@
|
|||||||
/* 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;
|
||||||
@@ -323,62 +253,6 @@
|
|||||||
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 */
|
||||||
@@ -453,7 +327,6 @@
|
|||||||
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;
|
||||||
@@ -488,45 +361,27 @@
|
|||||||
buildSettings = {
|
buildSettings = {
|
||||||
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
||||||
CLANG_ENABLE_MODULES = YES;
|
CLANG_ENABLE_MODULES = YES;
|
||||||
CODE_SIGN_IDENTITY = "Apple Development";
|
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
|
||||||
"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",
|
||||||
);
|
);
|
||||||
MARKETING_VERSION = 1.0.0;
|
PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation;
|
||||||
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 = 3;
|
CURRENT_PROJECT_VERSION = 1;
|
||||||
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;
|
||||||
@@ -540,12 +395,10 @@
|
|||||||
};
|
};
|
||||||
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 = 3;
|
CURRENT_PROJECT_VERSION = 1;
|
||||||
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;
|
||||||
@@ -557,12 +410,10 @@
|
|||||||
};
|
};
|
||||||
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 = 3;
|
CURRENT_PROJECT_VERSION = 1;
|
||||||
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;
|
||||||
@@ -596,7 +447,6 @@
|
|||||||
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;
|
||||||
@@ -654,7 +504,6 @@
|
|||||||
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;
|
||||||
@@ -691,34 +540,18 @@
|
|||||||
buildSettings = {
|
buildSettings = {
|
||||||
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
||||||
CLANG_ENABLE_MODULES = YES;
|
CLANG_ENABLE_MODULES = YES;
|
||||||
CODE_SIGN_IDENTITY = "Apple Development";
|
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
|
||||||
"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",
|
||||||
);
|
);
|
||||||
MARKETING_VERSION = 1.0.0;
|
PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation;
|
||||||
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;
|
||||||
@@ -729,33 +562,17 @@
|
|||||||
buildSettings = {
|
buildSettings = {
|
||||||
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
||||||
CLANG_ENABLE_MODULES = YES;
|
CLANG_ENABLE_MODULES = YES;
|
||||||
CODE_SIGN_IDENTITY = "Apple Development";
|
CURRENT_PROJECT_VERSION = "$(FLUTTER_BUILD_NUMBER)";
|
||||||
"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",
|
||||||
);
|
);
|
||||||
MARKETING_VERSION = 1.0.0;
|
PRODUCT_BUNDLE_IDENTIFIER = com.example.fastNetworkNavigation;
|
||||||
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,7 +4,4 @@
|
|||||||
<FileRef
|
<FileRef
|
||||||
location = "group:Runner.xcodeproj">
|
location = "group:Runner.xcodeproj">
|
||||||
</FileRef>
|
</FileRef>
|
||||||
<FileRef
|
|
||||||
location = "group:Pods/Pods.xcodeproj">
|
|
||||||
</FileRef>
|
|
||||||
</Workspace>
|
</Workspace>
|
||||||
|
@@ -1,16 +1,12 @@
|
|||||||
import UIKit
|
import UIKit
|
||||||
import Flutter
|
import Flutter
|
||||||
import GoogleMaps
|
|
||||||
|
|
||||||
@main
|
@UIApplicationMain
|
||||||
@objc class AppDelegate: FlutterAppDelegate {
|
@objc class AppDelegate: FlutterAppDelegate {
|
||||||
override func application(
|
override func application(
|
||||||
_ application: UIApplication,
|
_ application: UIApplication,
|
||||||
didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?
|
didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?
|
||||||
) -> Bool {
|
) -> Bool {
|
||||||
// load the key from env
|
|
||||||
let key = ProcessInfo.processInfo.environment["GOOGLE_MAPS_API_KEY"]!
|
|
||||||
GMSServices.provideAPIKey(key)
|
|
||||||
GeneratedPluginRegistrant.register(with: self)
|
GeneratedPluginRegistrant.register(with: self)
|
||||||
return super.application(application, didFinishLaunchingWithOptions: launchOptions)
|
return super.application(application, didFinishLaunchingWithOptions: launchOptions)
|
||||||
}
|
}
|
||||||
|
@@ -2,8 +2,6 @@
|
|||||||
<!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>
|
||||||
@@ -26,8 +24,6 @@
|
|||||||
<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>
|
||||||
@@ -45,5 +41,9 @@
|
|||||||
<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>
|
||||||
|
@@ -1,8 +0,0 @@
|
|||||||
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
|
|
@@ -1,91 +0,0 @@
|
|||||||
default_platform(:ios)
|
|
||||||
|
|
||||||
platform :ios do
|
|
||||||
|
|
||||||
desc "Load the App Store Connect API token"
|
|
||||||
lane :load_asc_api_token do
|
|
||||||
app_store_connect_api_key(
|
|
||||||
key_id: ENV["IOS_ASC_KEY_ID"],
|
|
||||||
issuer_id: ENV["IOS_ASC_ISSUER_ID"],
|
|
||||||
key_content: ENV["IOS_ASC_KEY"],
|
|
||||||
is_key_content_base64: true,
|
|
||||||
in_house: false
|
|
||||||
)
|
|
||||||
end
|
|
||||||
|
|
||||||
|
|
||||||
desc "Deploy a new version to closed testing (testflight)"
|
|
||||||
lane :deploy_testing do
|
|
||||||
build_name = ENV["BUILD_NAME"]
|
|
||||||
build_number = ENV["BUILD_NUMBER"]
|
|
||||||
|
|
||||||
load_asc_api_token
|
|
||||||
api_key = lane_context[SharedValues::APP_STORE_CONNECT_API_KEY]
|
|
||||||
|
|
||||||
sync_code_signing(
|
|
||||||
api_key: api_key,
|
|
||||||
type: "appstore",
|
|
||||||
readonly: true,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
sh(
|
|
||||||
"flutter",
|
|
||||||
"build",
|
|
||||||
"ipa",
|
|
||||||
"--release",
|
|
||||||
"--build-name=#{build_name}",
|
|
||||||
"--build-number=#{build_number}",
|
|
||||||
)
|
|
||||||
|
|
||||||
# sign the app (whithout rebuilding it)
|
|
||||||
build_app(
|
|
||||||
skip_build_archive: true,
|
|
||||||
archive_path: "../build/ios/archive/Runner.xcarchive"
|
|
||||||
)
|
|
||||||
|
|
||||||
upload_to_testflight
|
|
||||||
end
|
|
||||||
|
|
||||||
|
|
||||||
desc "Deploy a new version as a full release"
|
|
||||||
lane :deploy_release do
|
|
||||||
build_name = ENV["BUILD_NAME"]
|
|
||||||
build_number = ENV["BUILD_NUMBER"]
|
|
||||||
|
|
||||||
load_asc_api_token
|
|
||||||
api_key = lane_context[SharedValues::APP_STORE_CONNECT_API_KEY]
|
|
||||||
|
|
||||||
sync_code_signing(
|
|
||||||
api_key: api_key,
|
|
||||||
type: "appstore",
|
|
||||||
readonly: true,
|
|
||||||
)
|
|
||||||
|
|
||||||
sh(
|
|
||||||
"flutter",
|
|
||||||
"build",
|
|
||||||
"ipa",
|
|
||||||
"--release",
|
|
||||||
"--build-name=#{build_name}",
|
|
||||||
"--build-number=#{build_number}",
|
|
||||||
)
|
|
||||||
|
|
||||||
# sign the app (whithout rebuilding it)
|
|
||||||
build_app(
|
|
||||||
skip_build_archive: true,
|
|
||||||
archive_path: "../build/ios/archive/Runner.xarchive"
|
|
||||||
)
|
|
||||||
|
|
||||||
upload_to_app_store(
|
|
||||||
skip_screenshots: true,
|
|
||||||
skip_metadata: true,
|
|
||||||
skip_app_rating_config: true,
|
|
||||||
skip_app_review_information: true,
|
|
||||||
skip_submission: false,
|
|
||||||
# automatically submit the app for review
|
|
||||||
automatic_release: true,
|
|
||||||
# automatically release the app after review
|
|
||||||
)
|
|
||||||
end
|
|
||||||
end
|
|
@@ -1,8 +0,0 @@
|
|||||||
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
|
|
@@ -1,48 +0,0 @@
|
|||||||
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).
|
|
@@ -1,10 +0,0 @@
|
|||||||
# 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,7 +3,6 @@ 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';
|
||||||
|
@@ -1,37 +0,0 @@
|
|||||||
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,50 +1,111 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'dart:developer';
|
||||||
|
|
||||||
|
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:anyway/pages/current_trip.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:anyway/structs/trip.dart';
|
|
||||||
|
|
||||||
|
class Greeter extends StatefulWidget {
|
||||||
class CurrentTripGreeter extends StatefulWidget {
|
|
||||||
final Trip trip;
|
final Trip trip;
|
||||||
|
|
||||||
CurrentTripGreeter({
|
Greeter({
|
||||||
super.key,
|
|
||||||
required this.trip,
|
required this.trip,
|
||||||
});
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<CurrentTripGreeter> createState() => _CurrentTripGreeterState();
|
State<Greeter> createState() => _GreeterState();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class _CurrentTripGreeterState extends State<CurrentTripGreeter> {
|
class _GreeterState extends State<Greeter> {
|
||||||
@override
|
|
||||||
Widget build(BuildContext context) => Center(
|
Widget greeterBuilder (BuildContext context, Widget? child) {
|
||||||
child: FutureBuilder(
|
final Shader textGradient = APP_GRADIENT.createShader(Rect.fromLTWH(0.0, 0.0, 200.0, 70.0));
|
||||||
future: widget.trip.cityName,
|
TextStyle greeterStyle = TextStyle(
|
||||||
builder: (BuildContext context, AsyncSnapshot<String> snapshot) {
|
foreground: Paint()..shader = textGradient,
|
||||||
if (snapshot.hasData) {
|
fontWeight: FontWeight.bold,
|
||||||
return AutoSizeText(
|
fontSize: 26
|
||||||
maxLines: 1,
|
);
|
||||||
'Welcome to ${snapshot.data}!',
|
|
||||||
style: greeterStyle
|
|
||||||
);
|
|
||||||
} else if (snapshot.hasError) {
|
|
||||||
return AutoSizeText(
|
|
||||||
maxLines: 1,
|
|
||||||
'Welcome to your trip!',
|
|
||||||
style: greeterStyle
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return AutoSizeText(
|
|
||||||
maxLines: 1,
|
|
||||||
'Welcome to ...',
|
|
||||||
style: greeterStyle
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
|
Widget topGreeter;
|
||||||
|
|
||||||
|
if (widget.trip.uuid != 'pending') {
|
||||||
|
topGreeter = FutureBuilder(
|
||||||
|
future: widget.trip.cityName,
|
||||||
|
builder: (BuildContext context, AsyncSnapshot<String> snapshot) {
|
||||||
|
if (snapshot.hasData) {
|
||||||
|
return AutoSizeText(
|
||||||
|
maxLines: 1,
|
||||||
|
'Welcome to ${snapshot.data}!',
|
||||||
|
style: greeterStyle
|
||||||
|
);
|
||||||
|
} else if (snapshot.hasError) {
|
||||||
|
log('Error while fetching city name');
|
||||||
|
return AutoSizeText(
|
||||||
|
maxLines: 1,
|
||||||
|
'Welcome to your trip!',
|
||||||
|
style: greeterStyle
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return AutoSizeText(
|
||||||
|
maxLines: 1,
|
||||||
|
'Welcome to ...',
|
||||||
|
style: greeterStyle
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
// still awaiting the trip
|
||||||
|
// We can hopefully infer the city name from the cityName future
|
||||||
|
// Show a linear loader at the bottom and an info message above
|
||||||
|
topGreeter = Column(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.end,
|
||||||
|
children: [
|
||||||
|
FutureBuilder(
|
||||||
|
future: widget.trip.cityName,
|
||||||
|
builder: (BuildContext context, AsyncSnapshot<String> snapshot) {
|
||||||
|
if (snapshot.hasData) {
|
||||||
|
return AutoSizeText(
|
||||||
|
maxLines: 1,
|
||||||
|
'Generating your trip to ${snapshot.data}...',
|
||||||
|
style: greeterStyle
|
||||||
|
);
|
||||||
|
} else if (snapshot.hasError) {
|
||||||
|
// the exact error is shown in the central part of the trip overview. No need to show it here
|
||||||
|
return AutoSizeText(
|
||||||
|
maxLines: 1,
|
||||||
|
'Error while loading trip.',
|
||||||
|
style: greeterStyle
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return AutoSizeText(
|
||||||
|
maxLines: 1,
|
||||||
|
'Generating your trip...',
|
||||||
|
style: greeterStyle
|
||||||
|
);
|
||||||
|
}
|
||||||
|
),
|
||||||
|
Padding(
|
||||||
|
padding: EdgeInsets.all(5),
|
||||||
|
child: const LinearProgressIndicator()
|
||||||
|
)
|
||||||
|
]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Center(
|
||||||
|
child: topGreeter,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return ListenableBuilder(
|
||||||
|
listenable: widget.trip,
|
||||||
|
builder: greeterBuilder,
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
@@ -1,60 +0,0 @@
|
|||||||
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,6 +1,4 @@
|
|||||||
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';
|
||||||
@@ -30,36 +28,16 @@ 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 == 'error') {
|
if (widget.trip.uuid != 'pending' && 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),
|
padding: const EdgeInsets.only(bottom: 30, left: 5, right: 5),
|
||||||
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: CurrentTripGreeter(trip: widget.trip),
|
child: Greeter(trip: widget.trip),
|
||||||
),
|
),
|
||||||
|
|
||||||
const Padding(padding: EdgeInsets.only(top: 10)),
|
const Padding(padding: EdgeInsets.only(top: 10)),
|
||||||
@@ -75,6 +53,28 @@ 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.directions),
|
icon: const Icon(Icons.add),
|
||||||
label: AutoSizeText('Start planning!'),
|
label: AutoSizeText('Start planning!'),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@@ -6,12 +6,6 @@ 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 {
|
||||||
@@ -41,7 +35,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.all(10.0),
|
padding: const EdgeInsets.only(top: 10),
|
||||||
// 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: preferences.maxTime.icon,
|
leading: Icon(Icons.timer),
|
||||||
title: Text(preferences.maxTime.description),
|
title: Text(preferences.maxTime.description),
|
||||||
subtitle: CupertinoTimerPicker(
|
subtitle: CupertinoTimerPicker(
|
||||||
mode: CupertinoTimerPickerMode.hm,
|
mode: CupertinoTimerPickerMode.hm,
|
||||||
|
@@ -61,7 +61,9 @@ 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(
|
||||||
controller: TextEditingController(text: API_URL_DEBUG),
|
decoration: InputDecoration(
|
||||||
|
hintText: 'https://anyway-stg.anydev.info'
|
||||||
|
),
|
||||||
onChanged: (value) {
|
onChanged: (value) {
|
||||||
setState(() {
|
setState(() {
|
||||||
API_URL_BASE = value;
|
API_URL_BASE = value;
|
||||||
|
@@ -38,18 +38,10 @@ fetchTrip(
|
|||||||
String dataString = jsonEncode(data);
|
String dataString = jsonEncode(data);
|
||||||
log(dataString);
|
log(dataString);
|
||||||
|
|
||||||
late Response response;
|
final response = await dio.post(
|
||||||
try {
|
"/trip/new",
|
||||||
response = await dio.post(
|
data: data
|
||||||
"/trip/new",
|
);
|
||||||
data: data
|
|
||||||
);
|
|
||||||
} catch (e) {
|
|
||||||
trip.updateUUID("error");
|
|
||||||
trip.updateError(e.toString());
|
|
||||||
log(e.toString());
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// handle errors
|
// handle errors
|
||||||
if (response.statusCode != 200) {
|
if (response.statusCode != 200) {
|
||||||
|
@@ -1,2 +1 @@
|
|||||||
#include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.debug.xcconfig"
|
|
||||||
#include "ephemeral/Flutter-Generated.xcconfig"
|
#include "ephemeral/Flutter-Generated.xcconfig"
|
||||||
|
@@ -1,2 +1 @@
|
|||||||
#include? "Pods/Target Support Files/Pods-Runner/Pods-Runner.release.xcconfig"
|
|
||||||
#include "ephemeral/Flutter-Generated.xcconfig"
|
#include "ephemeral/Flutter-Generated.xcconfig"
|
||||||
|
@@ -1,43 +0,0 @@
|
|||||||
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
|
|