Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/grandsir/InteractiveMap
An Interactive Map Library for Swift and SwiftUI!
https://github.com/grandsir/InteractiveMap
interactive interactive-map ios ios-swift ipados macos map mapbox mapkit mapping maps svg swift swift5 swiftui swiftui1 swiftui2 swiftui3
Last synced: 2 months ago
JSON representation
An Interactive Map Library for Swift and SwiftUI!
- Host: GitHub
- URL: https://github.com/grandsir/InteractiveMap
- Owner: grandsir
- License: mit
- Created: 2022-09-19T15:27:00.000Z (over 2 years ago)
- Default Branch: main
- Last Pushed: 2023-12-06T10:00:20.000Z (about 1 year ago)
- Last Synced: 2024-08-03T18:12:55.644Z (6 months ago)
- Topics: interactive, interactive-map, ios, ios-swift, ipados, macos, map, mapbox, mapkit, mapping, maps, svg, swift, swift5, swiftui, swiftui1, swiftui2, swiftui3
- Language: Swift
- Homepage: https://medium.com/better-programming/how-to-create-interactive-maps-using-swiftui-1c49732e2950
- Size: 3.02 MB
- Stars: 86
- Watchers: 5
- Forks: 11
- Open Issues: 4
-
Metadata Files:
- Readme: README.md
- License: LICENSE.md
Awesome Lists containing this project
- awesome-swiftui-libraries - InteractiveMap - A Library to draw interactive maps and SVGs in SwiftUI (Map / Content)
README
![Interactive Map](./Assets/interactiveMap.png)
# InteractiveMapA Library to use SVG Based Maps interactively in SwiftUI.
- Works **only with** .svg based maps
- Allows you to modify **all** the provinces in the map with the attributes that SwiftUI's `Shape` provides
- Drag, drop and animate the provinces, as well as the map itself.Note! SVG Parsing is not yet in its final form, so some SVGs may not be parsed correctly. But as far as I can see almost every map is drawn correctly. You can see the maps I tried in the [Maps](./Maps) section.
Maps are taken from [FSInteractiveMap](https://github.com/ArthurGuibert/FSInteractiveMap) Repository
Installation
Requires iOS 13+
InteractiveMap currently can only be installed through the Swift Package Manager.
Swift Package Manager
Add the Package URL:
```
https://github.com/GrandSir/InteractiveMap
```
# Examples & ShowCase
[3D Scaling Effect](./Examples/3DEffect.swift)![3d-map](https://user-images.githubusercontent.com/69051988/191280598-63e0f189-2391-4d3c-90c0-06b6ea8d72b3.gif)
[Creepy Map](./Examples/creepyMap.swift)
![creepy)](https://user-images.githubusercontent.com/69051988/191286548-91302f94-7b16-4427-82c5-503fe0a442e0.gif)
[Mihmandar Province Selection Screen](./Examples/Mihmandar.swift)
![mihmandar](https://user-images.githubusercontent.com/69051988/191287726-49ff6d5a-ce56-49be-a235-d0c03710b16f.gif)
Usage
# SwiftUI
To draw your svg map in SwiftUI, use `InteractiveMap` with a closure taking `PathData` as the parameter.`InteractiveMap` uses `InteractiveShape`s to draw all the paths defined in SVG. But it needs to know which `Path` will be drawn, that is, `InteractiveMap { pathData in }` works just like `ForEach { index in }` and returns an iterable closure returning `PathData` as parameter, which contains all the information about `Path`s defined inside svg.
```swift
import SwiftUI
import InteractiveMapstruct ContentView: View {
var body: some View {
InteractiveMap(svgName: "tr") { pathData in // or just use $0
InteractiveShape(pathData)
.initWithAttributes()
}
}
}
```
InteractiveMap resizes itself to the assigned frame, takes all available space by default, and resizes itself responsively on device rotations.# Customization
Instead of using default attributes, you can define your own attributes as well.
```swift
InteractiveMap(svgName: "tr") {
InteractiveShape($0)
.initWithAttributes(.init(strokeWidth: 2, strokeColor: .red, background: Color(white: 0.2)))
}
```## Advanced Customization
Even though `.initWithAttributes` saves time for simple customization, it is neither highly customizable nor editable.
Since `InteractiveShape` is a `Shape`, you can use any methods with `InteractiveShape` that you can use with `Shape`.
```swift
InteractiveMap(svgName: "tr") {
InteractiveShape($0)
.stroke(Color.cyan)
.shadow(color: .cyan, radius: 3, x: 0, y: 0)
.background(InteractiveShape($0).fill(Color(white: 0.15)))
}
```### Handling Clicks
`PathData` is a `Struct` that contains all the information about all the paths, in our map example, they are districts and provinces.It has 5 variables inside it. `id`, `path` and `name`, `boundingBox` and `svgBounds`
`id` is the Unique Identifier that's being parsed directly from SVG
Most of the Map SVG's (NOT ALL!) has the `id` attribute in all in their `` element like this:
``
`MapParser`, defined in `MapParser.swift` parses that element and stores them in `PathData` struct.
If there is not any `id` attribute in path, MapParser automatically creates an UUID String.
But if you're going to store that id in somewhere, be aware that UUID String is automaticaly regenerated every time InteractiveMap is being drawed.
```swift
import SwiftUI
import InteractiveMapstruct ContentView: View {
@State private var clickedPath = PathData()
var body: some View {
VStack {
Text(clickedPath.name.isEmpty ? "" : "\(clickedPath.name) is clicked!" )
.font(.largeTitle)
.padding(.bottom, 15)InteractiveMap(svgName: "tr") { pathData in // is a PathData
InteractiveShape(pathData)
.stroke(clickedPath == pathData ? .cyan : .red, lineWidth: 1)
.shadow(color: clickedPath == pathData ? .cyan : .red, radius: 3)
.shadow(color: clickedPath == pathData ? .cyan : .clear , radius: 3) // to increase the glow amount
.background(InteractiveShape(pathData).fill(Color(white: 0.15))) // filling the shapes
.shadow(color: clickedPath == pathData ? .black : .clear , radius: 5, y: 1) // for depth.onTapGesture {
clickedPath = pathData
}
.zIndex(clickedPath == pathData ? 2 : 1) // this is REQUIRED because InteractiveShapes overlap, resulting in an ugly appearance
.animation(.easeInOut(duration: 0.3), value: clickedPath)
}
}
}
}
```
`clickedPath == pathData` basically compares the id's of the PathDatas.![clickable-map](https://user-images.githubusercontent.com/69051988/191283653-00cd4ebe-5ccd-48a1-b55b-3f70edfa3b14.gif)
# UIKit
soon
# SpriteKit
soon