图1.跟我女儿一起进行后期用户测试
这是一个关于灵感的故事,一个愿景的形成,以及来自最挑剔的客户——一个 2 岁女孩的快速反馈。
想跳到最终产品吗?现在就从App Store下载Aviator — Radar on your Phone吧!
启发
今年夏天,我带着我蹒跚学步的女儿出国了。
她太兴奋了。但是,为了确保她能够应付3个小时的飞行时间,我和我的妻子对她大肆宣传飞机旅行带来的奇妙体验。当我们不得不坐上去机场的出租车时,我的女儿的感到很不情愿,她说希望从我们家走到飞机场上飞机(她不喜欢坐出租车)。
-
如何显示方向。 -
采用什么尺寸(缩放的比例)。 -
要简单易用。
-
该应用程序需要保持正确的方向,与设备一起旋转,以便以正确的方向显示飞机。 -
该应用程序必须根据飞机的高度显示飞机的大小。 -
该应用程序必须有趣,并且感觉更像是一个复古的儿童玩具,而不是一个严肃的商业应用程序。
-
保持方向是区分产品的核心要求,因为现有解决方案中缺少这一点。我不从事详细的飞行信息——我只想做一个很酷的雷达!iOS Core Location API 为我们提供了帮助,每次用户重新调整设备方向时都会提供委托回调。 -
当然,最重要的组件是飞行数据API。OpenSky Network 正是我需要的。一个简单的 REST API,免费用于非商业用途,具有某个地区航班的实时数据。我们希望每隔几秒钟对这个端点执行一次 ping 操作,以获得真实的雷达扫描。 -
要调用 API,我们需要一些位置数据。核心位置再次为我们提供了保障——为了获得大量附近的航班,我们可以从用户的位置查询 +/- 1 度的纬度,精度为 0.1 度(约 10 公里),以确保用户的位置被充分混淆。我们也只需要在每个会话中获取一次此数据。 -
最后,也是最困难的事情,我们需要掸掉我们的三角学技能,将飞行位置数据与我们自己的定向坐标进行比较。这将使我们能够根据附近的飞机在天空中的相对位置,在正确的位置将附近的飞机吸附到屏幕上。
final
class
LocationManager
:
CLLocationManager
,
CLLocationManagerDelegate
{
static
let
shared =
LocationManager
()
private
(
set
)
var
rotationAngleSubject =
CurrentValueSubject
<
Double
,
Never
>(
0
)
override
private
init
() {
super
.
init
()
requestWhenInUseAuthorization()
delegate =
self
startUpdatingHeading()
}
func
locationManager
(
_
manager: CLLocationManager, didUpdateHeading newHeading: CLHeading)
{
rotationAngleSubject.send(-newHeading.magneticHeading)
}
}
@State
private
var
rotationAngle: Angle = .degrees(
0
)
var
body: some View {
ZStack {
ForEach(
0.
.<
36
) {
let
angle = Angle.degrees(Double($
0
*
10
)) +
rotationAngle
Rectangle
(
)
.
frame
(
width: $
0
==
0
?
16
:
8
, height: $
0
==
0
?
3
:
2
)
.
foregroundColor
(
$
0
==
0
? .red : .blue
)
.
rotationEffect
(
angle
)
.
offset
(
x:
120
* cos(CGFloat(angle.radians
)), y: 120 *
sin
(
CGFloat(angle.radians
)))
.
animation
(
.bouncy,
value
: rotationAngle
)
}
}
.
onReceive
(
LocationManager.shared.rotationAngleSubject
)
{ angle
in
rotationAngle = Angle.degrees(angle)
}
}
在我的设备上进行测试,它看起来相当不错,并且完美地响应了我的真实位置!
这就引出了一个问题,为什么谷歌地图无法计算出我面对的方向?
你会注意到一个有趣的视觉故障,因为动画逻辑将 0 度和 360 度视为单独的数字——当我经过正北时,所有的矩形都决定旋转——但这对 PoC 来说很好(因为我不太可能真正保留这个 UI)。
https:
/
/opensky-network.org/api
/states/all
?lamin=
51.0
&lamax=
52.0
&lomin=-
0
.
5
&lomax=
0
.
5
REST API 记录得很好,但具有未键控的结构,这意味着数据按顺序显示为列表属性。
来自 OpenSky Network API 的 JSON 响应文档
我们需要使用 an UnkeyedContainer 来解码它,它旨在按顺序解析 JSON 响应中的字段。
struct
Flight
:
Decodable
{
let
icao24:
String
let
callsign:
String
?
let
origin_country:
String
?
let
time_position:
Int
?
let
last_contact:
Int
let
longitude:
Double
let
latitude:
Double
// ...
init
(from decoder:
Decoder
)
throws
{
var
container =
try
decoder.unkeyedContainer()
icao24 =
try
container.decode(
String
.
self
)
callsign =
try
? container.decode(
String
?.
self
)
origin_country =
try
container.decode(
String
.
self
)
time_position =
try
? container.decode(
Int
?.
self
)
last_contact =
try
container.decode(
Int
.
self
)
longitude =
try
container.decode(
Double
.
self
)
latitude =
try
container.decode(
Double
.
self
)
// ...
}
}
我们可以编写一个简单的 API,根据用户的位置坐标执行 GET 请求。
final
class
FlightAPI {
func fetchLocalFlightData(coordinate: CLLocationCoordinate2D)
async
throws -> [Flight] {
let
lamin =
String
(format:
"%.1f"
, coordinate.latitude -
0.25
)
let
lamax =
String
(format:
"%.1f"
, coordinate.latitude +
0.25
)
let
lomin =
String
(format:
"%.1f"
, coordinate.longitude -
0.5
)
let
lomax =
String
(format:
"%.1f"
, coordinate.longitude +
0.5
)
let
url = URL(
string
:
"https://opensky-network.org/api/states/all?lamin=(lamin)&lamax=(lamax)&lomin=(lomin)&lomax=(lomax)"
)!
let
data =
try
await
URLSession.shared.data(
from
: url)
.0
return
try
JSONDecoder().decode([Flight].self,
from
: data)
}
}
您可能会注意到,我在此 API 调用中使用了 1 度的经度范围,但只有 0.5 度的纬度。这是因为在我的纬度英国,一个 0.5 纬度 x 1 经度的矩形大致显示为正方形。
绘制飞机
修改我的 LocationManager 以侦听重大位置更改并通过发布者发送这些坐标是非常微不足道的。
同样,在纯粹的 MV 架构风格中,我的视图通过侦听坐标, .onReceive 并用这些坐标调用我的 new FlightAPI 。结果呢?有关本地天空中高架飞机的数据。
现在,我们到达了我最初概念验证中最困难的部分:相对于我自己的位置,实际将飞机图标显示在正确的位置。
我的第一次迭代是一个钝器:我将相对纬度和经度乘以屏幕上的硬编码点值。
private
var
coordinates: CLLocationCoordinate2D?
private
var
flights: [Flight] = []
private
var
airplanes: some View {
ForEach(flights, id: .icao24) { flight
in
let
latDiff = coordinate.latitude - (flight.latitude ??
0
)
let
lngDiff = coordinate.longitude - (flight.longitude ??
0
)
Image(systemName:
"airplane"
)
.resizable()
.frame(width:
20
, height:
20
)
.rotationEffect(.degrees(flight.true_track ??
0
))
.foregroundColor(.red)
.offset(x:
250
* latDiff, y:
250
* lngDiff)
}
}
当然,这不可能是准确的,因为纬度或经度的绝对距离会随着您的地理位置而变化。但同样,这是一个很好的起点。
private var cameraPosition: MapCameraPosition = .camera(MapCamera(
centerCoordinate
:
CLLocationCoordinate2D(latitude: 51.0, longitude: 0.0),
distance
:
100_000,
heading
:
0))
var
body: some View {
ZStack
{
:
$cameraPosition) { }
airplanes
compass
}
}
这是我第一次深夜黑客马拉松的结果,与FlightRadar的预测进行了比较。
第 #1 天的结果,左边是我的应用程序,与右边的 FlightRadar 进行比较
import
MapKit
import
SwiftUI
struct
FlightMapView
:
View
{
@
Binding
var
cameraPosition:
MapCameraPosition
let
flights: [
Flight
]
var
body: some
View
{
Map
(position: $cameraPosition) {
planeMapAnnotations
}
.mapStyle(.imagery)
.allowsHitTesting(
false
)
}
}
@
State
private
var
rotationAngle:
Angle
= .degrees(
0
)
private
var
planeMapAnnotations: some
MapContent
{
ForEach
(flights, id: .icao24) { flight
in
Annotation
(flight.icao24, coordinate: flight.coordinate) {
let
rotation = rotationAngle.degrees + flight.true_track
let
scale =
min
(
2
,
max
(log10(height +
1
),
0.5
))
Image
(systemName:
"airplane"
)
.rotationEffect(.degrees(rotation))
.scaleEffect(scale)
}
}
.tint(.white)
}
}
min
(2,
max
(4
.7
-
log10
(
flight
.geo_altitude
+ 1), 0
.7
))
Scale
: 1
.0835408863965839
Scale
: 0
.8330645861650874
Scale
: 1
.095791123396205
Scale
: 1
.1077242935783653
Scale
: 2
.0
Scale
: 1
.4864702267977097
Scale
: 0
.7
private
func
fetchFlights
(at coordinate: CLLocationCoordinate2D, retries: Int =
3
)
async {
do
{
try
await api.fetchLocalFlightData(coordinate: coordinate)
}
catch
{
if
retries >
0
{
try
await fetchFlights(at: coordinate, retries: retries -
1
)
}
}
}
struct
FlightMapView
:
View
{
var
body: some
View
{
Map
(position: $cameraPosition) {
planeMapAnnotations
MapPolygon
(overlay(coordinate: coordinate))
}
.mapStyle(.imagery)
.allowsHitTesting(
false
)
}
// ...
private
func
rectangle
(around coordinate: CLLocationCoordinate2D)
-> [
CLLocationCoordinate2D
] {
[
CLLocationCoordinate2D
(latitude: coordinate.latitude -
1
, longitude: coordinate.longitude -
1
),
CLLocationCoordinate2D
(latitude: coordinate.latitude -
1
, longitude: coordinate.longitude +
1
),
CLLocationCoordinate2D
(latitude: coordinate.latitude +
1
, longitude: coordinate.longitude +
1
),
CLLocationCoordinate2D
(latitude: coordinate.latitude +
1
, longitude: coordinate.longitude -
1
)
]
}
private
func
overlay
(coordinate: CLLocationCoordinate2D)
->
MKPolygon
{
let
rectangle = rectangle(around: coordinate)
return
MKPolygon
(coordinates: rectangle,
count
: rectangle.
count
)
}
}
private
var radarLine: some View {
Circle()
.fill(
AngularGradient(
gradient
:
Gradient(colors: [
Color.black, Color.black, Color.black,
Color.black.opacity(0.6),
Color.black.opacity(0.2),
Color.clear, Color.clear, Color.clear,
Color.clear, Color.clear, Color.clear,
Color.clear, Color.clear, Color.green]),
center
:
.center,
startAngle
:
.degrees(rotationDegree),
endAngle
:
.degrees(rotationDegree + 360)
)
)
:
rotationDegree))
:
6).repeatForever(autoreverses: false), value: rotationDegree)
}
using
namespace
metal;
[[ stitchable ]]
half4
crtScreen
(
float2 position,
half4 color,
float
time
)
{
if
(all(
abs
(color.rgb - half3(
0.0
,
0.0
,
0.0
)) < half3(
0.01
,
0.01
,
0.01
))) {
return
color;
}
const
half scanlineIntensity =
0.2
;
const
half scanlineFrequency =
400.0
;
half scanlineValue =
sin
((position.y + time *
10.0
) * scanlineFrequency *
3.14159
h) * scanlineIntensity;
return
half4(color.rgb - scanlineValue, color.a);
}
extension
View
{
func
crtScreenEffect
(startTime: Date)
-> some
View
{
modifier(
CRTScreen
(startTime: startTime))
}
}
struct
CRTScreen
:
ViewModifier
{
let
startTime:
Date
func
body
(content: Content)
-> some
View
{
content
.colorEffect(
ShaderLibrary
.crtScreen(
.float(startTime.timeIntervalSinceNow)
)
)
}
}
private
func
fetchFlights
(coordinate: Coordinate, retries: Int =
2
)
async {
do
{
let
flights =
try
await api.fetchLocalFlightData(coordinate: coordinate)
await
MainActor
.run {
self
.flights = flights
AudioServicesPlaySystemSound
(
1052
)
hapticTrigger.toggle()
}
// ...
}
.sensoryFeedback
(
.levelChange
,
trigger
:
hapticTrigger
)
var
silentMode: Bool =
false
var
showMap: Bool =
false
var
userColor: Color = .green
private
func
toggleableIcon
(state: Bool, iconTrue: String, iconFalse: String)
-> some
View
{
Image
(systemName: state ? iconTrue : iconFalse)
.contentTransition(.symbolEffect(.replace))
// ...
}
// @State properties ...
var
body: some View {
ZStack {
if
let
coordinate = locationManager.coordinateSubject.
value
{
FlightMapView(
cameraPosition: $cameraPosition,
flights: flights,
rotationAngle: rotationAngle,
coordinate: coordinate
)
}
TimelineView(.animation) {
context
in
RadarView
(
)
.
crtScreenEffect
(
)
.
negativeHighlight
(
)
}
ControlsView
(
errorMessage: errorMessage
)
}
// onRecieve modifiers ...
}
呜。我首付 79 英镑,准备发布。
星期六上午
周六晚上
周日下午
而且它是在线的!
-
使用 OpenSky Network API 的高级版本来显示直升机、卫星和飞机尺寸等级。 -
切换飞机上的始发地和目的地国家/地区显示。 -
使用更高级的金属着色器改进 CRT 屏幕效果。 -
将所有控件重构为可调整大小的渐进式披露拉出模式,并带有棘爪。 -
实施滑块控制以过滤掉某些距离和高度 - 例如,隐藏所有低空,远处的飞机。 -
实现“滑稽模式”,在雷达上呈现不明飞行物、巨型虫子和外星人。
原文始发于微信公众号(黑客与极客):我为喜欢飞机的女儿做了一个雷达APP,来发现天空中的飞机
- 左青龙
- 微信扫一扫
-
- 右白虎
- 微信扫一扫
-
评论