├── GPXRouteCreator.xcodeproj ├── project.pbxproj ├── project.xcworkspace │ ├── contents.xcworkspacedata │ ├── xcshareddata │ │ └── IDEWorkspaceChecks.plist │ └── xcuserdata │ │ └── dom.xcuserdatad │ │ └── xcdebugger │ │ └── Breakpoints_v2.xcbkptlist └── xcuserdata │ └── dom.xcuserdatad │ ├── xcdebugger │ └── Breakpoints_v2.xcbkptlist │ └── xcschemes │ └── xcschememanagement.plist ├── GPXRouteCreator ├── AppDelegate.swift ├── Assets.xcassets │ ├── AppIcon.appiconset │ │ └── Contents.json │ └── Contents.json ├── Base.lproj │ └── LaunchScreen.storyboard ├── ContentView.swift ├── DocumentPicker.swift ├── FileManagerExtension.swift ├── GPXRouteCreator.entitlements ├── Info.plist ├── MapView.swift ├── Model │ └── Location.swift ├── Preview Content │ └── Preview Assets.xcassets │ │ └── Contents.json └── SceneDelegate.swift ├── LICENSE ├── README.md └── screenshots └── 01.png /GPXRouteCreator.xcodeproj/project.pbxproj: -------------------------------------------------------------------------------- 1 | // !$*UTF8*$! 2 | { 3 | archiveVersion = 1; 4 | classes = { 5 | }; 6 | objectVersion = 50; 7 | objects = { 8 | 9 | /* Begin PBXBuildFile section */ 10 | F2A8EEBB23ED66EB003D0D4A /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = F2A8EEBA23ED66EB003D0D4A /* AppDelegate.swift */; }; 11 | F2A8EEBD23ED66EB003D0D4A /* SceneDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = F2A8EEBC23ED66EB003D0D4A /* SceneDelegate.swift */; }; 12 | F2A8EEBF23ED66EB003D0D4A /* ContentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = F2A8EEBE23ED66EB003D0D4A /* ContentView.swift */; }; 13 | F2A8EEC123ED66F5003D0D4A /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = F2A8EEC023ED66F5003D0D4A /* Assets.xcassets */; }; 14 | F2A8EEC423ED66F5003D0D4A /* Preview Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = F2A8EEC323ED66F5003D0D4A /* Preview Assets.xcassets */; }; 15 | F2A8EEC723ED66F5003D0D4A /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = F2A8EEC523ED66F5003D0D4A /* LaunchScreen.storyboard */; }; 16 | F2A8EED023ED6813003D0D4A /* MapView.swift in Sources */ = {isa = PBXBuildFile; fileRef = F2A8EECF23ED6813003D0D4A /* MapView.swift */; }; 17 | F2A8EED423ED9582003D0D4A /* FileManagerExtension.swift in Sources */ = {isa = PBXBuildFile; fileRef = F2A8EED323ED9582003D0D4A /* FileManagerExtension.swift */; }; 18 | F2A8EED623ED9CD7003D0D4A /* DocumentPicker.swift in Sources */ = {isa = PBXBuildFile; fileRef = F2A8EED523ED9CD7003D0D4A /* DocumentPicker.swift */; }; 19 | F2A8EEDB23EEFA37003D0D4A /* Location.swift in Sources */ = {isa = PBXBuildFile; fileRef = F2A8EEDA23EEFA36003D0D4A /* Location.swift */; }; 20 | /* End PBXBuildFile section */ 21 | 22 | /* Begin PBXFileReference section */ 23 | F2A8EEB723ED66EB003D0D4A /* GPXRouteCreator.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = GPXRouteCreator.app; sourceTree = BUILT_PRODUCTS_DIR; }; 24 | F2A8EEBA23ED66EB003D0D4A /* AppDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = ""; }; 25 | F2A8EEBC23ED66EB003D0D4A /* SceneDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SceneDelegate.swift; sourceTree = ""; }; 26 | F2A8EEBE23ED66EB003D0D4A /* ContentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ContentView.swift; sourceTree = ""; }; 27 | F2A8EEC023ED66F5003D0D4A /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = ""; }; 28 | F2A8EEC323ED66F5003D0D4A /* Preview Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = "Preview Assets.xcassets"; sourceTree = ""; }; 29 | F2A8EEC623ED66F5003D0D4A /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = ""; }; 30 | F2A8EEC823ED66F5003D0D4A /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; 31 | F2A8EECE23ED6726003D0D4A /* GPXRouteCreator.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = GPXRouteCreator.entitlements; sourceTree = ""; }; 32 | F2A8EECF23ED6813003D0D4A /* MapView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MapView.swift; sourceTree = ""; }; 33 | F2A8EED323ED9582003D0D4A /* FileManagerExtension.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FileManagerExtension.swift; sourceTree = ""; }; 34 | F2A8EED523ED9CD7003D0D4A /* DocumentPicker.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DocumentPicker.swift; sourceTree = ""; }; 35 | F2A8EEDA23EEFA36003D0D4A /* Location.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = Location.swift; sourceTree = ""; }; 36 | /* End PBXFileReference section */ 37 | 38 | /* Begin PBXFrameworksBuildPhase section */ 39 | F2A8EEB423ED66EB003D0D4A /* Frameworks */ = { 40 | isa = PBXFrameworksBuildPhase; 41 | buildActionMask = 2147483647; 42 | files = ( 43 | ); 44 | runOnlyForDeploymentPostprocessing = 0; 45 | }; 46 | /* End PBXFrameworksBuildPhase section */ 47 | 48 | /* Begin PBXGroup section */ 49 | F2A8EEAE23ED66EB003D0D4A = { 50 | isa = PBXGroup; 51 | children = ( 52 | F2A8EEB923ED66EB003D0D4A /* GPXRouteCreator */, 53 | F2A8EEB823ED66EB003D0D4A /* Products */, 54 | ); 55 | sourceTree = ""; 56 | }; 57 | F2A8EEB823ED66EB003D0D4A /* Products */ = { 58 | isa = PBXGroup; 59 | children = ( 60 | F2A8EEB723ED66EB003D0D4A /* GPXRouteCreator.app */, 61 | ); 62 | name = Products; 63 | sourceTree = ""; 64 | }; 65 | F2A8EEB923ED66EB003D0D4A /* GPXRouteCreator */ = { 66 | isa = PBXGroup; 67 | children = ( 68 | F2A8EECE23ED6726003D0D4A /* GPXRouteCreator.entitlements */, 69 | F2A8EEBA23ED66EB003D0D4A /* AppDelegate.swift */, 70 | F2A8EEBC23ED66EB003D0D4A /* SceneDelegate.swift */, 71 | F2A8EEBE23ED66EB003D0D4A /* ContentView.swift */, 72 | F2A8EECF23ED6813003D0D4A /* MapView.swift */, 73 | F2A8EED523ED9CD7003D0D4A /* DocumentPicker.swift */, 74 | F2A8EED323ED9582003D0D4A /* FileManagerExtension.swift */, 75 | F2A8EED923EEFA36003D0D4A /* Model */, 76 | F2A8EEC023ED66F5003D0D4A /* Assets.xcassets */, 77 | F2A8EEC523ED66F5003D0D4A /* LaunchScreen.storyboard */, 78 | F2A8EEC823ED66F5003D0D4A /* Info.plist */, 79 | F2A8EEC223ED66F5003D0D4A /* Preview Content */, 80 | ); 81 | path = GPXRouteCreator; 82 | sourceTree = ""; 83 | }; 84 | F2A8EEC223ED66F5003D0D4A /* Preview Content */ = { 85 | isa = PBXGroup; 86 | children = ( 87 | F2A8EEC323ED66F5003D0D4A /* Preview Assets.xcassets */, 88 | ); 89 | path = "Preview Content"; 90 | sourceTree = ""; 91 | }; 92 | F2A8EED923EEFA36003D0D4A /* Model */ = { 93 | isa = PBXGroup; 94 | children = ( 95 | F2A8EEDA23EEFA36003D0D4A /* Location.swift */, 96 | ); 97 | path = Model; 98 | sourceTree = ""; 99 | }; 100 | /* End PBXGroup section */ 101 | 102 | /* Begin PBXNativeTarget section */ 103 | F2A8EEB623ED66EB003D0D4A /* GPXRouteCreator */ = { 104 | isa = PBXNativeTarget; 105 | buildConfigurationList = F2A8EECB23ED66F5003D0D4A /* Build configuration list for PBXNativeTarget "GPXRouteCreator" */; 106 | buildPhases = ( 107 | F2A8EEB323ED66EB003D0D4A /* Sources */, 108 | F2A8EEB423ED66EB003D0D4A /* Frameworks */, 109 | F2A8EEB523ED66EB003D0D4A /* Resources */, 110 | ); 111 | buildRules = ( 112 | ); 113 | dependencies = ( 114 | ); 115 | name = GPXRouteCreator; 116 | productName = GPXRouteCreator; 117 | productReference = F2A8EEB723ED66EB003D0D4A /* GPXRouteCreator.app */; 118 | productType = "com.apple.product-type.application"; 119 | }; 120 | /* End PBXNativeTarget section */ 121 | 122 | /* Begin PBXProject section */ 123 | F2A8EEAF23ED66EB003D0D4A /* Project object */ = { 124 | isa = PBXProject; 125 | attributes = { 126 | LastSwiftUpdateCheck = 1140; 127 | LastUpgradeCheck = 1140; 128 | ORGANIZATIONNAME = dasdom; 129 | TargetAttributes = { 130 | F2A8EEB623ED66EB003D0D4A = { 131 | CreatedOnToolsVersion = 11.4; 132 | }; 133 | }; 134 | }; 135 | buildConfigurationList = F2A8EEB223ED66EB003D0D4A /* Build configuration list for PBXProject "GPXRouteCreator" */; 136 | compatibilityVersion = "Xcode 9.3"; 137 | developmentRegion = en; 138 | hasScannedForEncodings = 0; 139 | knownRegions = ( 140 | en, 141 | Base, 142 | ); 143 | mainGroup = F2A8EEAE23ED66EB003D0D4A; 144 | productRefGroup = F2A8EEB823ED66EB003D0D4A /* Products */; 145 | projectDirPath = ""; 146 | projectRoot = ""; 147 | targets = ( 148 | F2A8EEB623ED66EB003D0D4A /* GPXRouteCreator */, 149 | ); 150 | }; 151 | /* End PBXProject section */ 152 | 153 | /* Begin PBXResourcesBuildPhase section */ 154 | F2A8EEB523ED66EB003D0D4A /* Resources */ = { 155 | isa = PBXResourcesBuildPhase; 156 | buildActionMask = 2147483647; 157 | files = ( 158 | F2A8EEC723ED66F5003D0D4A /* LaunchScreen.storyboard in Resources */, 159 | F2A8EEC423ED66F5003D0D4A /* Preview Assets.xcassets in Resources */, 160 | F2A8EEC123ED66F5003D0D4A /* Assets.xcassets in Resources */, 161 | ); 162 | runOnlyForDeploymentPostprocessing = 0; 163 | }; 164 | /* End PBXResourcesBuildPhase section */ 165 | 166 | /* Begin PBXSourcesBuildPhase section */ 167 | F2A8EEB323ED66EB003D0D4A /* Sources */ = { 168 | isa = PBXSourcesBuildPhase; 169 | buildActionMask = 2147483647; 170 | files = ( 171 | F2A8EEBB23ED66EB003D0D4A /* AppDelegate.swift in Sources */, 172 | F2A8EED423ED9582003D0D4A /* FileManagerExtension.swift in Sources */, 173 | F2A8EEDB23EEFA37003D0D4A /* Location.swift in Sources */, 174 | F2A8EED023ED6813003D0D4A /* MapView.swift in Sources */, 175 | F2A8EEBD23ED66EB003D0D4A /* SceneDelegate.swift in Sources */, 176 | F2A8EEBF23ED66EB003D0D4A /* ContentView.swift in Sources */, 177 | F2A8EED623ED9CD7003D0D4A /* DocumentPicker.swift in Sources */, 178 | ); 179 | runOnlyForDeploymentPostprocessing = 0; 180 | }; 181 | /* End PBXSourcesBuildPhase section */ 182 | 183 | /* Begin PBXVariantGroup section */ 184 | F2A8EEC523ED66F5003D0D4A /* LaunchScreen.storyboard */ = { 185 | isa = PBXVariantGroup; 186 | children = ( 187 | F2A8EEC623ED66F5003D0D4A /* Base */, 188 | ); 189 | name = LaunchScreen.storyboard; 190 | sourceTree = ""; 191 | }; 192 | /* End PBXVariantGroup section */ 193 | 194 | /* Begin XCBuildConfiguration section */ 195 | F2A8EEC923ED66F5003D0D4A /* Debug */ = { 196 | isa = XCBuildConfiguration; 197 | buildSettings = { 198 | ALWAYS_SEARCH_USER_PATHS = NO; 199 | CLANG_ANALYZER_NONNULL = YES; 200 | CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; 201 | CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; 202 | CLANG_CXX_LIBRARY = "libc++"; 203 | CLANG_ENABLE_MODULES = YES; 204 | CLANG_ENABLE_OBJC_ARC = YES; 205 | CLANG_ENABLE_OBJC_WEAK = YES; 206 | CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; 207 | CLANG_WARN_BOOL_CONVERSION = YES; 208 | CLANG_WARN_COMMA = YES; 209 | CLANG_WARN_CONSTANT_CONVERSION = YES; 210 | CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; 211 | CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; 212 | CLANG_WARN_DOCUMENTATION_COMMENTS = YES; 213 | CLANG_WARN_EMPTY_BODY = YES; 214 | CLANG_WARN_ENUM_CONVERSION = YES; 215 | CLANG_WARN_INFINITE_RECURSION = YES; 216 | CLANG_WARN_INT_CONVERSION = YES; 217 | CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; 218 | CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; 219 | CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; 220 | CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; 221 | CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; 222 | CLANG_WARN_STRICT_PROTOTYPES = YES; 223 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 224 | CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; 225 | CLANG_WARN_UNREACHABLE_CODE = YES; 226 | CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; 227 | COPY_PHASE_STRIP = NO; 228 | DEBUG_INFORMATION_FORMAT = dwarf; 229 | ENABLE_STRICT_OBJC_MSGSEND = YES; 230 | ENABLE_TESTABILITY = YES; 231 | GCC_C_LANGUAGE_STANDARD = gnu11; 232 | GCC_DYNAMIC_NO_PIC = NO; 233 | GCC_NO_COMMON_BLOCKS = YES; 234 | GCC_OPTIMIZATION_LEVEL = 0; 235 | GCC_PREPROCESSOR_DEFINITIONS = ( 236 | "DEBUG=1", 237 | "$(inherited)", 238 | ); 239 | GCC_WARN_64_TO_32_BIT_CONVERSION = YES; 240 | GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; 241 | GCC_WARN_UNDECLARED_SELECTOR = YES; 242 | GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; 243 | GCC_WARN_UNUSED_FUNCTION = YES; 244 | GCC_WARN_UNUSED_VARIABLE = YES; 245 | IPHONEOS_DEPLOYMENT_TARGET = 13.4; 246 | MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE; 247 | MTL_FAST_MATH = YES; 248 | ONLY_ACTIVE_ARCH = YES; 249 | SDKROOT = iphoneos; 250 | SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG; 251 | SWIFT_OPTIMIZATION_LEVEL = "-Onone"; 252 | }; 253 | name = Debug; 254 | }; 255 | F2A8EECA23ED66F5003D0D4A /* Release */ = { 256 | isa = XCBuildConfiguration; 257 | buildSettings = { 258 | ALWAYS_SEARCH_USER_PATHS = NO; 259 | CLANG_ANALYZER_NONNULL = YES; 260 | CLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE; 261 | CLANG_CXX_LANGUAGE_STANDARD = "gnu++14"; 262 | CLANG_CXX_LIBRARY = "libc++"; 263 | CLANG_ENABLE_MODULES = YES; 264 | CLANG_ENABLE_OBJC_ARC = YES; 265 | CLANG_ENABLE_OBJC_WEAK = YES; 266 | CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES; 267 | CLANG_WARN_BOOL_CONVERSION = YES; 268 | CLANG_WARN_COMMA = YES; 269 | CLANG_WARN_CONSTANT_CONVERSION = YES; 270 | CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES; 271 | CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR; 272 | CLANG_WARN_DOCUMENTATION_COMMENTS = YES; 273 | CLANG_WARN_EMPTY_BODY = YES; 274 | CLANG_WARN_ENUM_CONVERSION = YES; 275 | CLANG_WARN_INFINITE_RECURSION = YES; 276 | CLANG_WARN_INT_CONVERSION = YES; 277 | CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES; 278 | CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES; 279 | CLANG_WARN_OBJC_LITERAL_CONVERSION = YES; 280 | CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR; 281 | CLANG_WARN_RANGE_LOOP_ANALYSIS = YES; 282 | CLANG_WARN_STRICT_PROTOTYPES = YES; 283 | CLANG_WARN_SUSPICIOUS_MOVE = YES; 284 | CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; 285 | CLANG_WARN_UNREACHABLE_CODE = YES; 286 | CLANG_WARN__DUPLICATE_METHOD_MATCH = YES; 287 | COPY_PHASE_STRIP = NO; 288 | DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; 289 | ENABLE_NS_ASSERTIONS = NO; 290 | ENABLE_STRICT_OBJC_MSGSEND = YES; 291 | GCC_C_LANGUAGE_STANDARD = gnu11; 292 | GCC_NO_COMMON_BLOCKS = YES; 293 | GCC_WARN_64_TO_32_BIT_CONVERSION = YES; 294 | GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR; 295 | GCC_WARN_UNDECLARED_SELECTOR = YES; 296 | GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE; 297 | GCC_WARN_UNUSED_FUNCTION = YES; 298 | GCC_WARN_UNUSED_VARIABLE = YES; 299 | IPHONEOS_DEPLOYMENT_TARGET = 13.4; 300 | MTL_ENABLE_DEBUG_INFO = NO; 301 | MTL_FAST_MATH = YES; 302 | SDKROOT = iphoneos; 303 | SWIFT_COMPILATION_MODE = wholemodule; 304 | SWIFT_OPTIMIZATION_LEVEL = "-O"; 305 | VALIDATE_PRODUCT = YES; 306 | }; 307 | name = Release; 308 | }; 309 | F2A8EECC23ED66F5003D0D4A /* Debug */ = { 310 | isa = XCBuildConfiguration; 311 | buildSettings = { 312 | ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; 313 | CODE_SIGN_ENTITLEMENTS = GPXRouteCreator/GPXRouteCreator.entitlements; 314 | CODE_SIGN_STYLE = Automatic; 315 | DEVELOPMENT_ASSET_PATHS = "\"GPXRouteCreator/Preview Content\""; 316 | DEVELOPMENT_TEAM = Q6PD97MRX4; 317 | ENABLE_PREVIEWS = YES; 318 | INFOPLIST_FILE = GPXRouteCreator/Info.plist; 319 | IPHONEOS_DEPLOYMENT_TARGET = 13.0; 320 | LD_RUNPATH_SEARCH_PATHS = ( 321 | "$(inherited)", 322 | "@executable_path/Frameworks", 323 | ); 324 | PRODUCT_BUNDLE_IDENTIFIER = de.dasdom.GPXRouteCreator; 325 | PRODUCT_NAME = "$(TARGET_NAME)"; 326 | SUPPORTS_MACCATALYST = YES; 327 | SWIFT_VERSION = 5.0; 328 | TARGETED_DEVICE_FAMILY = 2; 329 | }; 330 | name = Debug; 331 | }; 332 | F2A8EECD23ED66F5003D0D4A /* Release */ = { 333 | isa = XCBuildConfiguration; 334 | buildSettings = { 335 | ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; 336 | CODE_SIGN_ENTITLEMENTS = GPXRouteCreator/GPXRouteCreator.entitlements; 337 | CODE_SIGN_STYLE = Automatic; 338 | DEVELOPMENT_ASSET_PATHS = "\"GPXRouteCreator/Preview Content\""; 339 | DEVELOPMENT_TEAM = Q6PD97MRX4; 340 | ENABLE_PREVIEWS = YES; 341 | INFOPLIST_FILE = GPXRouteCreator/Info.plist; 342 | IPHONEOS_DEPLOYMENT_TARGET = 13.0; 343 | LD_RUNPATH_SEARCH_PATHS = ( 344 | "$(inherited)", 345 | "@executable_path/Frameworks", 346 | ); 347 | PRODUCT_BUNDLE_IDENTIFIER = de.dasdom.GPXRouteCreator; 348 | PRODUCT_NAME = "$(TARGET_NAME)"; 349 | SUPPORTS_MACCATALYST = YES; 350 | SWIFT_VERSION = 5.0; 351 | TARGETED_DEVICE_FAMILY = 2; 352 | }; 353 | name = Release; 354 | }; 355 | /* End XCBuildConfiguration section */ 356 | 357 | /* Begin XCConfigurationList section */ 358 | F2A8EEB223ED66EB003D0D4A /* Build configuration list for PBXProject "GPXRouteCreator" */ = { 359 | isa = XCConfigurationList; 360 | buildConfigurations = ( 361 | F2A8EEC923ED66F5003D0D4A /* Debug */, 362 | F2A8EECA23ED66F5003D0D4A /* Release */, 363 | ); 364 | defaultConfigurationIsVisible = 0; 365 | defaultConfigurationName = Release; 366 | }; 367 | F2A8EECB23ED66F5003D0D4A /* Build configuration list for PBXNativeTarget "GPXRouteCreator" */ = { 368 | isa = XCConfigurationList; 369 | buildConfigurations = ( 370 | F2A8EECC23ED66F5003D0D4A /* Debug */, 371 | F2A8EECD23ED66F5003D0D4A /* Release */, 372 | ); 373 | defaultConfigurationIsVisible = 0; 374 | defaultConfigurationName = Release; 375 | }; 376 | /* End XCConfigurationList section */ 377 | }; 378 | rootObject = F2A8EEAF23ED66EB003D0D4A /* Project object */; 379 | } 380 | -------------------------------------------------------------------------------- /GPXRouteCreator.xcodeproj/project.xcworkspace/contents.xcworkspacedata: -------------------------------------------------------------------------------- 1 | 2 | 4 | 6 | 7 | 8 | -------------------------------------------------------------------------------- /GPXRouteCreator.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | IDEDidComputeMac32BitWarning 6 | 7 | 8 | 9 | -------------------------------------------------------------------------------- /GPXRouteCreator.xcodeproj/project.xcworkspace/xcuserdata/dom.xcuserdatad/xcdebugger/Breakpoints_v2.xcbkptlist: -------------------------------------------------------------------------------- 1 | 2 | 6 | 7 | 9 | 16 | 17 | 18 | 19 | 20 | -------------------------------------------------------------------------------- /GPXRouteCreator.xcodeproj/xcuserdata/dom.xcuserdatad/xcdebugger/Breakpoints_v2.xcbkptlist: -------------------------------------------------------------------------------- 1 | 2 | 6 | 7 | -------------------------------------------------------------------------------- /GPXRouteCreator.xcodeproj/xcuserdata/dom.xcuserdatad/xcschemes/xcschememanagement.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | SchemeUserState 6 | 7 | GPXRouteCreator.xcscheme_^#shared#^_ 8 | 9 | orderHint 10 | 0 11 | 12 | 13 | 14 | 15 | -------------------------------------------------------------------------------- /GPXRouteCreator/AppDelegate.swift: -------------------------------------------------------------------------------- 1 | // Created by dasdom on 07.02.20. 2 | // Copyright © 2020 dasdom. All rights reserved. 3 | // 4 | 5 | import UIKit 6 | 7 | @UIApplicationMain 8 | class AppDelegate: UIResponder, UIApplicationDelegate { 9 | 10 | 11 | 12 | func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool { 13 | // Override point for customization after application launch. 14 | return true 15 | } 16 | 17 | // MARK: UISceneSession Lifecycle 18 | 19 | func application(_ application: UIApplication, configurationForConnecting connectingSceneSession: UISceneSession, options: UIScene.ConnectionOptions) -> UISceneConfiguration { 20 | // Called when a new scene session is being created. 21 | // Use this method to select a configuration to create the new scene with. 22 | return UISceneConfiguration(name: "Default Configuration", sessionRole: connectingSceneSession.role) 23 | } 24 | 25 | func application(_ application: UIApplication, didDiscardSceneSessions sceneSessions: Set) { 26 | // Called when the user discards a scene session. 27 | // If any sessions were discarded while the application was not running, this will be called shortly after application:didFinishLaunchingWithOptions. 28 | // Use this method to release any resources that were specific to the discarded scenes, as they will not return. 29 | } 30 | 31 | 32 | } 33 | 34 | -------------------------------------------------------------------------------- /GPXRouteCreator/Assets.xcassets/AppIcon.appiconset/Contents.json: -------------------------------------------------------------------------------- 1 | { 2 | "images" : [ 3 | { 4 | "idiom" : "iphone", 5 | "scale" : "2x", 6 | "size" : "20x20" 7 | }, 8 | { 9 | "idiom" : "iphone", 10 | "scale" : "3x", 11 | "size" : "20x20" 12 | }, 13 | { 14 | "idiom" : "iphone", 15 | "scale" : "2x", 16 | "size" : "29x29" 17 | }, 18 | { 19 | "idiom" : "iphone", 20 | "scale" : "3x", 21 | "size" : "29x29" 22 | }, 23 | { 24 | "idiom" : "iphone", 25 | "scale" : "2x", 26 | "size" : "40x40" 27 | }, 28 | { 29 | "idiom" : "iphone", 30 | "scale" : "3x", 31 | "size" : "40x40" 32 | }, 33 | { 34 | "idiom" : "iphone", 35 | "scale" : "2x", 36 | "size" : "60x60" 37 | }, 38 | { 39 | "idiom" : "iphone", 40 | "scale" : "3x", 41 | "size" : "60x60" 42 | }, 43 | { 44 | "idiom" : "ipad", 45 | "scale" : "1x", 46 | "size" : "20x20" 47 | }, 48 | { 49 | "idiom" : "ipad", 50 | "scale" : "2x", 51 | "size" : "20x20" 52 | }, 53 | { 54 | "idiom" : "ipad", 55 | "scale" : "1x", 56 | "size" : "29x29" 57 | }, 58 | { 59 | "idiom" : "ipad", 60 | "scale" : "2x", 61 | "size" : "29x29" 62 | }, 63 | { 64 | "idiom" : "ipad", 65 | "scale" : "1x", 66 | "size" : "40x40" 67 | }, 68 | { 69 | "idiom" : "ipad", 70 | "scale" : "2x", 71 | "size" : "40x40" 72 | }, 73 | { 74 | "idiom" : "ipad", 75 | "scale" : "1x", 76 | "size" : "76x76" 77 | }, 78 | { 79 | "idiom" : "ipad", 80 | "scale" : "2x", 81 | "size" : "76x76" 82 | }, 83 | { 84 | "idiom" : "ipad", 85 | "scale" : "2x", 86 | "size" : "83.5x83.5" 87 | }, 88 | { 89 | "idiom" : "ios-marketing", 90 | "scale" : "1x", 91 | "size" : "1024x1024" 92 | } 93 | ], 94 | "info" : { 95 | "author" : "xcode", 96 | "version" : 1 97 | } 98 | } 99 | -------------------------------------------------------------------------------- /GPXRouteCreator/Assets.xcassets/Contents.json: -------------------------------------------------------------------------------- 1 | { 2 | "info" : { 3 | "author" : "xcode", 4 | "version" : 1 5 | } 6 | } 7 | -------------------------------------------------------------------------------- /GPXRouteCreator/Base.lproj/LaunchScreen.storyboard: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | -------------------------------------------------------------------------------- /GPXRouteCreator/ContentView.swift: -------------------------------------------------------------------------------- 1 | // Created by dasdom on 07.02.20. 2 | // Copyright © 2020 dasdom. All rights reserved. 3 | // 4 | 5 | import SwiftUI 6 | import CoreLocation 7 | 8 | struct ContentView: View { 9 | 10 | @State private var locations: [Location] = [] 11 | @State private var tappedCoordinate: CLLocationCoordinate2D? { 12 | didSet { 13 | if let coordinate = tappedCoordinate { 14 | coordinateString = "\(coordinate.latitude), \(coordinate.longitude)" 15 | } 16 | } 17 | } 18 | @State private var coordinateString = "Tap a location you like to add." 19 | @State private var addTime = false 20 | @State private var showDocumentsPicker = false 21 | @State private var secondsBetween = 60 22 | private var timeDiffText: String { 23 | var strings: [String] = [] 24 | let minutes = secondsBetween / 60 25 | let seconds = secondsBetween % 60 26 | if minutes > 0 { 27 | strings.append("\(minutes) min") 28 | } 29 | if seconds > 0 { 30 | strings.append("\(seconds) s") 31 | } 32 | return strings.joined(separator: " ") 33 | } 34 | 35 | static let dateFormatter: DateFormatter = { 36 | let formatter = DateFormatter() 37 | formatter.dateStyle = .short 38 | formatter.timeStyle = .medium 39 | return formatter 40 | }() 41 | 42 | var body: some View { 43 | GeometryReader { geometry in 44 | HStack { 45 | ZStack(alignment: .bottomTrailing) { 46 | MapView(tappedCallback: { coordinate in 47 | self.tappedCoordinate = coordinate 48 | self.add(coordinate: coordinate) 49 | }, locations: self.$locations) 50 | 51 | VStack(alignment: .center, spacing: 10) { 52 | Text(self.coordinateString) 53 | Stepper(onIncrement: { 54 | if self.secondsBetween >= 120 { 55 | self.secondsBetween += 60 56 | } else { 57 | self.secondsBetween += 10 58 | } 59 | }, onDecrement: { 60 | if self.secondsBetween <= 10 { 61 | self.secondsBetween -= 1 62 | } else if self.secondsBetween < 0 { 63 | self.secondsBetween = 0 64 | } else { 65 | self.secondsBetween -= 10 66 | } 67 | }) { 68 | Text("\(self.timeDiffText) to previous entry") 69 | } 70 | } 71 | .padding() 72 | .frame(width: 400) 73 | .background(Color(UIColor.systemBackground.withAlphaComponent(0.7))) 74 | 75 | } 76 | VStack { 77 | // NavigationView { 78 | List { 79 | ForEach(self.locations) { location in 80 | VStack { 81 | Text("\(location.coordinate.latitude), \(location.coordinate.longitude)") 82 | Text("\(ContentView.dateFormatter.string(from: location.date))") 83 | } 84 | .foregroundColor(Color(UIColor.label)) 85 | .onLongPressGesture { 86 | if let index = self.locations.firstIndex(where: { (temp) -> Bool in 87 | temp.date == location.date 88 | }) { 89 | self.locations.remove(at: index) 90 | } 91 | } 92 | } 93 | // .onMove { from, to in 94 | // print("from: \(from), to: \(to)") 95 | // self.locations.move(fromOffsets: from, toOffset: to) 96 | // } 97 | } 98 | HStack { 99 | Button(action: { 100 | self.export() 101 | self.showDocumentsPicker = true 102 | #if targetEnvironment(macCatalyst) 103 | UIApplication.shared.windows[0].rootViewController!.present(UIDocumentPickerViewController(url: FileManager.trackURL(), in: .exportToService), animated: true) 104 | #endif 105 | }) { Image(systemName: "square.and.arrow.up") } 106 | } 107 | .padding() 108 | } 109 | .frame(width: geometry.size.width*0.3) 110 | } 111 | .overlay( 112 | VStack { 113 | if self.showDocumentsPicker { 114 | DocumentPicker(url: FileManager.trackURL()) 115 | } else { 116 | EmptyView() 117 | } 118 | } 119 | ) 120 | } 121 | } 122 | 123 | func add(coordinate: CLLocationCoordinate2D) { 124 | let lastDate = self.locations.last?.date ?? Date() 125 | let date = Date(timeInterval: TimeInterval(self.secondsBetween), since: lastDate) 126 | locations.append(Location(id: locations.count + 1, 127 | coordinate: coordinate, 128 | date: date)) 129 | } 130 | 131 | func export() { 132 | var exportStrings: [String] = [""] 133 | exportStrings.append("") 134 | 135 | let dateFormatter = DateFormatter() 136 | dateFormatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ss'Z'" 137 | 138 | locations.forEach { location in 139 | exportStrings.append(location.gpx) 140 | } 141 | 142 | exportStrings.append("\n") 143 | 144 | let data = exportStrings.joined(separator: "\n").data(using: .utf8) 145 | let url = FileManager.trackURL() 146 | do { 147 | try data?.write(to: url) 148 | print("did write to \(url)") 149 | } catch { 150 | print("error: \(error)") 151 | } 152 | } 153 | } 154 | 155 | struct ContentView_Previews: PreviewProvider { 156 | static var previews: some View { 157 | ContentView() 158 | } 159 | } 160 | -------------------------------------------------------------------------------- /GPXRouteCreator/DocumentPicker.swift: -------------------------------------------------------------------------------- 1 | // Created by dasdom on 07.02.20. 2 | // Copyright © 2020 dasdom. All rights reserved. 3 | // 4 | 5 | import UIKit 6 | import SwiftUI 7 | 8 | final class DocumentPicker : NSObject, UIViewControllerRepresentable { 9 | 10 | let url: URL 11 | 12 | init(url: URL) { 13 | 14 | self.url = url 15 | 16 | super.init() 17 | } 18 | 19 | func makeUIViewController(context: Context) -> UIDocumentPickerViewController { 20 | UIDocumentPickerViewController(url: url, in: .exportToService) 21 | } 22 | 23 | func updateUIViewController(_ uiViewController: UIDocumentPickerViewController, context: Context) { 24 | 25 | } 26 | } 27 | -------------------------------------------------------------------------------- /GPXRouteCreator/FileManagerExtension.swift: -------------------------------------------------------------------------------- 1 | // Created by dasdom on 25.01.20. 2 | // Copyright © 2020 dasdom. All rights reserved. 3 | // 4 | 5 | import Foundation 6 | 7 | extension FileManager { 8 | private static func documentsURL() -> URL { 9 | guard let url = FileManager.default.urls( 10 | for: .documentDirectory, 11 | in: .userDomainMask).first else { 12 | fatalError() 13 | } 14 | return url 15 | } 16 | 17 | static func trackURL() -> URL { 18 | return documentsURL().appendingPathComponent("track.gpx") 19 | } 20 | } 21 | -------------------------------------------------------------------------------- /GPXRouteCreator/GPXRouteCreator.entitlements: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | com.apple.security.app-sandbox 6 | 7 | com.apple.security.files.user-selected.read-write 8 | 9 | com.apple.security.network.client 10 | 11 | 12 | 13 | -------------------------------------------------------------------------------- /GPXRouteCreator/Info.plist: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | CFBundleDevelopmentRegion 6 | $(DEVELOPMENT_LANGUAGE) 7 | CFBundleExecutable 8 | $(EXECUTABLE_NAME) 9 | CFBundleIdentifier 10 | $(PRODUCT_BUNDLE_IDENTIFIER) 11 | CFBundleInfoDictionaryVersion 12 | 6.0 13 | CFBundleName 14 | $(PRODUCT_NAME) 15 | CFBundlePackageType 16 | $(PRODUCT_BUNDLE_PACKAGE_TYPE) 17 | CFBundleShortVersionString 18 | 1.0 19 | CFBundleVersion 20 | 1 21 | LSRequiresIPhoneOS 22 | 23 | UIApplicationSceneManifest 24 | 25 | UIApplicationSupportsMultipleScenes 26 | 27 | UISceneConfigurations 28 | 29 | UIWindowSceneSessionRoleApplication 30 | 31 | 32 | UISceneConfigurationName 33 | Default Configuration 34 | UISceneDelegateClassName 35 | $(PRODUCT_MODULE_NAME).SceneDelegate 36 | 37 | 38 | 39 | 40 | UILaunchStoryboardName 41 | LaunchScreen 42 | UIRequiredDeviceCapabilities 43 | 44 | armv7 45 | 46 | UISupportedInterfaceOrientations 47 | 48 | UIInterfaceOrientationPortrait 49 | UIInterfaceOrientationLandscapeLeft 50 | UIInterfaceOrientationLandscapeRight 51 | 52 | UISupportedInterfaceOrientations~ipad 53 | 54 | UIInterfaceOrientationPortrait 55 | UIInterfaceOrientationPortraitUpsideDown 56 | UIInterfaceOrientationLandscapeLeft 57 | UIInterfaceOrientationLandscapeRight 58 | 59 | 60 | 61 | -------------------------------------------------------------------------------- /GPXRouteCreator/MapView.swift: -------------------------------------------------------------------------------- 1 | // Created by dasdom on 07.02.20. 2 | // Copyright © 2020 dasdom. All rights reserved. 3 | // 4 | 5 | import SwiftUI 6 | import UIKit 7 | import MapKit 8 | 9 | struct MapView : UIViewRepresentable { 10 | 11 | // Tap handling from https://stackoverflow.com/a/56518293/498796 12 | var tappedCallback: ((CLLocationCoordinate2D) -> Void) 13 | @Binding var locations: [Location] 14 | let delegate = MapViewDelegate() 15 | 16 | func makeUIView(context: Context) -> MKMapView { 17 | let mapView = MKMapView() 18 | let longPressGesture = UILongPressGestureRecognizer(target: context.coordinator, 19 | action: #selector(Coordinator.addLocation)) 20 | 21 | mapView.addGestureRecognizer(longPressGesture) 22 | return mapView 23 | } 24 | 25 | func updateUIView(_ mapView: MKMapView, context: Context) { 26 | 27 | mapView.removeOverlays(mapView.overlays) 28 | mapView.removeAnnotations(mapView.annotations) 29 | 30 | let coordinates = locations.map({ $0.coordinate }) 31 | let overlay = MKPolyline(coordinates: coordinates, 32 | count: coordinates.count) 33 | mapView.delegate = delegate 34 | mapView.addOverlay(overlay) 35 | 36 | // locations.forEach { location in 37 | // mapView.addAnnotation(location.pointAnnotation) 38 | // } 39 | } 40 | 41 | class Coordinator : NSObject { 42 | var tappedCallback: ((CLLocationCoordinate2D) -> Void) 43 | 44 | init(tappedCallback: @escaping ((CLLocationCoordinate2D) -> Void)) { 45 | self.tappedCallback = tappedCallback 46 | } 47 | 48 | @objc func addLocation(_ sender: UILongPressGestureRecognizer) { 49 | 50 | print("sender state: \(sender.state)") 51 | if sender.state == .began { 52 | let point = sender.location(in: sender.view) 53 | if let mapView = sender.view as? MKMapView { 54 | let coordinate = mapView.convert(point, toCoordinateFrom: mapView) 55 | tappedCallback(coordinate) 56 | } 57 | } 58 | } 59 | } 60 | 61 | func makeCoordinator() -> Coordinator { 62 | Coordinator(tappedCallback: tappedCallback) 63 | } 64 | } 65 | 66 | class MapViewDelegate : NSObject, MKMapViewDelegate { 67 | func mapView(_ mapView: MKMapView, 68 | rendererFor overlay: MKOverlay) 69 | -> MKOverlayRenderer { 70 | 71 | if overlay is MKPolyline { 72 | let renderer = MKPolylineRenderer(overlay: overlay) 73 | renderer.strokeColor = .red 74 | renderer.lineWidth = 3 75 | return renderer 76 | } else { 77 | return MKOverlayRenderer(overlay: overlay) 78 | } 79 | } 80 | } 81 | 82 | -------------------------------------------------------------------------------- /GPXRouteCreator/Model/Location.swift: -------------------------------------------------------------------------------- 1 | // 2 | // Location.swift 3 | // GPXRouteCreator 4 | // 5 | // Created by Oliver Epper on 08.02.20. 6 | // Copyright © 2020 dasdom. All rights reserved. 7 | // 8 | 9 | import Foundation 10 | import CoreLocation 11 | import MapKit 12 | 13 | struct Location: Identifiable { 14 | var id: Int 15 | var coordinate: CLLocationCoordinate2D 16 | var date: Date 17 | 18 | static let dateFormatter: DateFormatter = { 19 | let formatter = DateFormatter() 20 | formatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ss'Z'" 21 | return formatter 22 | }() 23 | 24 | var pointAnnotation: MKPointAnnotation { 25 | get { 26 | let annotation = MKPointAnnotation() 27 | annotation.coordinate = coordinate 28 | annotation.title = String(id) 29 | 30 | return annotation 31 | } 32 | } 33 | 34 | var gpx: String { 35 | #""" 36 | 37 | 38 | 39 | """# 40 | } 41 | } 42 | -------------------------------------------------------------------------------- /GPXRouteCreator/Preview Content/Preview Assets.xcassets/Contents.json: -------------------------------------------------------------------------------- 1 | { 2 | "info" : { 3 | "author" : "xcode", 4 | "version" : 1 5 | } 6 | } 7 | -------------------------------------------------------------------------------- /GPXRouteCreator/SceneDelegate.swift: -------------------------------------------------------------------------------- 1 | // Created by dasdom on 07.02.20. 2 | // Copyright © 2020 dasdom. All rights reserved. 3 | // 4 | 5 | import UIKit 6 | import SwiftUI 7 | 8 | class SceneDelegate: UIResponder, UIWindowSceneDelegate { 9 | 10 | var window: UIWindow? 11 | 12 | 13 | func scene(_ scene: UIScene, willConnectTo session: UISceneSession, options connectionOptions: UIScene.ConnectionOptions) { 14 | // Use this method to optionally configure and attach the UIWindow `window` to the provided UIWindowScene `scene`. 15 | // If using a storyboard, the `window` property will automatically be initialized and attached to the scene. 16 | // This delegate does not imply the connecting scene or session are new (see `application:configurationForConnectingSceneSession` instead). 17 | 18 | // Create the SwiftUI view that provides the window contents. 19 | let contentView = ContentView() 20 | 21 | // Use a UIHostingController as window root view controller. 22 | if let windowScene = scene as? UIWindowScene { 23 | let window = UIWindow(windowScene: windowScene) 24 | window.rootViewController = UIHostingController(rootView: contentView) 25 | self.window = window 26 | window.makeKeyAndVisible() 27 | } 28 | } 29 | 30 | func sceneDidDisconnect(_ scene: UIScene) { 31 | // Called as the scene is being released by the system. 32 | // This occurs shortly after the scene enters the background, or when its session is discarded. 33 | // Release any resources associated with this scene that can be re-created the next time the scene connects. 34 | // The scene may re-connect later, as its session was not neccessarily discarded (see `application:didDiscardSceneSessions` instead). 35 | } 36 | 37 | func sceneDidBecomeActive(_ scene: UIScene) { 38 | // Called when the scene has moved from an inactive state to an active state. 39 | // Use this method to restart any tasks that were paused (or not yet started) when the scene was inactive. 40 | } 41 | 42 | func sceneWillResignActive(_ scene: UIScene) { 43 | // Called when the scene will move from an active state to an inactive state. 44 | // This may occur due to temporary interruptions (ex. an incoming phone call). 45 | } 46 | 47 | func sceneWillEnterForeground(_ scene: UIScene) { 48 | // Called as the scene transitions from the background to the foreground. 49 | // Use this method to undo the changes made on entering the background. 50 | } 51 | 52 | func sceneDidEnterBackground(_ scene: UIScene) { 53 | // Called as the scene transitions from the foreground to the background. 54 | // Use this method to save data, release shared resources, and store enough scene-specific state information 55 | // to restore the scene back to its current state. 56 | } 57 | 58 | 59 | } 60 | 61 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2020 Dominik Hauser 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # GPXRouteCreator 2 | 3 | Generate gpx files for iOS development. 4 | 5 | ![](screenshots/01.png) 6 | 7 | # Author 8 | 9 | [@dasdom](https://twitter.com/dasdom) 10 | 11 | # Licence 12 | 13 | MIT 14 | -------------------------------------------------------------------------------- /screenshots/01.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dasdom/GPXRouteCreator/d5f4c3646bb53ee5d6814257c13c26ace2786a10/screenshots/01.png --------------------------------------------------------------------------------