欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

iOS新增繪制圓的方法實例代碼

 更新時間:2020年05月18日 08:42:19   作者:CobableKun  
這篇文章主要給大家介紹了關(guān)于iOS新增繪制圓的方法,文中通過示例代碼介紹的非常詳細(xì),對各位iOS開發(fā)者們具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧

iOS 的坐標(biāo)系和我們幾何課本中的二維坐標(biāo)系并不一樣!

# BezierPath繪制圓弧

使用 UIBezierPath 進(jìn)行繪制圓弧的方法,通常會直接使用 addArc :

addArc(withCenter:, radius:, startAngle:, endAngle:, clockwise:)

或者使用 addCurve 進(jìn)行擬圓弧:

addCurve(to:, controlPoint1:, controlPoint2:)

其實我們可以通過,兩個坐標(biāo)點(diǎn)(startPoint & endPoint),和兩點(diǎn)間的線段對應(yīng)的圓弧的弧度(angle/radian)就能確定這個圓的信息(半徑radius, center), 所以我們是不是可以封裝出只提供 start, end 和 angle 就能繪制 arc 的函數(shù)?

addArc(startPoint: , endPoint: , angle: , clockwise:)

# 計算兩點(diǎn)間的距離

這里邏輯很簡單不做贅述。

func calculateLineLength(_ point1: CGPoint, _ point2: CGPoint) -> CGFloat {
  let w = point1.x - point2.x
  let h = point1.y - point2.y
  return sqrt(w * w + h * h)
}

# 計算兩點(diǎn)間的夾角

計算 point 和 origin 連線在 iOS 坐標(biāo)系的角度

func calculateAngle(point: CGPoint, origin: CGPoint) -> Double {
  
  if point.y == origin.y {
    return point.x > origin.x ? 0.0 : -Double.pi
  }
  
  if point.x == origin.x {
    return point.y > origin.y ? Double.pi * 0.5 : Double.pi * -0.5
  }
  // Note: 修正標(biāo)準(zhǔn)坐標(biāo)系角度到 iOS 坐標(biāo)系
  let rotationAdjustment = Double.pi * 0.5
  
  let offsetX = point.x - origin.x
  let offsetY = point.y - origin.y
  // Note: 使用 -offsetY 是因為 iOS 坐標(biāo)系與標(biāo)準(zhǔn)坐標(biāo)系的區(qū)別
  if offsetY > 0 {
    return Double(atan(offsetX / -offsetY)) + rotationAdjustment
  } else {
    return Double(atan(offsetX / -offsetY)) - rotationAdjustment
  }
}

# 計算圓心的坐標(biāo)

如果你已經(jīng)將幾何知識丟的差不多了的話,我在這里畫了個大概的草圖,如下( angle 比較小時):

angle 比較大時:

所以我么可以寫出如下計算中心點(diǎn)的代碼

// Woring: 只計算從start到end **順時針** 計算對應(yīng)的 **小于π** 圓弧對應(yīng)的圓心
// Note: 計算逆時針(end到start)可以看做將傳入的start和end對調(diào)后計算順時針時的圓心位置
// Note: 計算大于π的叫相當(dāng)于將end和start對換后計算2π-angle的順時針圓心位置
// Note: 綜上傳入start,end,angle 右外部自行處理邏輯
func calculateCenterFor(startPoint start: CGPoint, endPoint end: CGPoint, radian: Double) -> CGPoint {
  guard radian <= Double.pi else {
    fatalError("Does not support radian calculations greater than π!")
  }
  
  guard start != end else {
    fatalError("Start position and end position cannot be equal!")
  }
  
  if radian == Double.pi {
    let centerX = (end.x - start.x) * 0.5 + start.x
    let centerY = (end.y - start.y) * 0.5 + start.y
    return CGPoint(x: centerX, y: centerY)
  }
  
  let lineAB = calculateLineLength(start, end)
  
  // 平行 Y 軸
  if start.x == end.x {
    let centerY = (end.y - start.y) * 0.5 + start.y
    let tanResult = CGFloat(tan(radian * 0.5))
    let offsetX = lineAB * 0.5 / tanResult
    let centerX = start.x + offsetX * (start.y > end.y ? 1.0 : -1.0)
    return CGPoint(x: centerX, y: centerY)
  }
  
  // 平行 X 軸
  if start.y == end.y {
    let centerX = (end.x - start.x) * 0.5 + start.x
    let tanResult = CGFloat(tan(radian * 0.5))
    let offsetY = lineAB * 0.5 / tanResult
    let centerY = start.y + offsetY * (start.x < end.x ? 1.0 : -1.0)
    return CGPoint(x: centerX, y: centerY)
  }
  
  // 普通情況
  
  // 計算半徑
  let radius = lineAB * 0.5 / CGFloat(sin(radian * 0.5))
  // 計算與 Y 軸的夾角
  let angleToYAxis = atan(abs(start.x - end.x) / abs(start.y - end.y))
  let cacluteAngle = CGFloat(Double.pi - radian) * 0.5 - angleToYAxis
  // 偏移量
  let offsetX = radius * sin(cacluteAngle)
  let offsetY = radius * cos(cacluteAngle)
  
  var centetX = end.x
  var centerY = end.y
  // 以 start 為原點(diǎn)判斷象限區(qū)間(iOS坐標(biāo)系)
  if end.x > start.x && end.y < start.y {
    // 第一象限
    centetX = end.x + offsetX
    centerY = end.y + offsetY
  } else if end.x > start.x && end.y > start.y {
    // 第二象限
    centetX = start.x - offsetX
    centerY = start.y + offsetY
  } else if end.x < start.x && end.y > start.y {
    // 第三象限
    centetX = end.x - offsetX
    centerY = end.y - offsetY
  } else if end.x < start.x && end.y < start.y {
    // 第四象限
    centetX = start.x + offsetX
    centerY = start.y - offsetY
  }
  
  return CGPoint(x: centetX, y: centerY)
}

這里附上一個逆時針繪制第一張圖中圓心位置的草圖,圖中已將 start 和 end 對換

如果你對其中計算時到底該使用 + 還是 - 有困惑的話也可以自己多畫些草圖大概驗證下,總之有疑惑多動手🤭

# 實現(xiàn)我們的目標(biāo)函數(shù)

在有了計算圓心位置,和兩點(diǎn)間角度的函數(shù)后我們很容易就能實現(xiàn) addArc(startPoint: , endPoint: , angle: , clockwise:) 了;

func addArc(startPoint start: CGPoint, endPoint end: CGPoint, angle: Double, clockwise: Bool) {
  
  guard start != end && (angle >= 0 && angle <= 2 * Double.pi) else {
    return
  }
  if angle == 0 {
    move(to: start)
    addLine(to: end)
    return
  }
  
  var tmpStart = start, tmpEnd = end, tmpAngle = angle
  // Note: 保證計算圓心時是從 start 到 end 順時針 小于 π 的角
  if tmpAngle > Double.pi {
    tmpAngle = 2 * Double.pi - tmpAngle
    (tmpStart, tmpEnd) = (tmpEnd, tmpStart)
  }
  if !clockwise {
    (tmpStart, tmpEnd) = (tmpEnd, tmpStart)
  }
  
  let center = calculateCenterFor(startPoint: tmpStart, endPoint: tmpEnd, radian: tmpAngle)
  let radius = calculateLineLength(start, center)
  
  var startAngle = calculateAngle(point: start, origin: center)
  var endAngle = calculateAngle(point: end, origin: center)
  // Note: 逆時針繪制則交換 startAngle 和 endAngle,并且將開始點(diǎn)移動的 end 位置
  if !clockwise {
    (startAngle, endAngle) = (endAngle, startAngle)
    move(to: end)
  }
  
  addArc(withCenter: center, radius: radius, startAngle: CGFloat(startAngle), endAngle: CGFloat(endAngle), clockwise: true)
  move(to: end)
}

# 完結(jié)

最后也不知道是你否會碰到相同的需求,這里附上源碼和一份樣例及運(yùn)行結(jié)果圖;

override func draw(_ rect: CGRect) {
  
  let path = UIBezierPath()
  var start = CGPoint(x: 160, y: 130)
  var end = CGPoint(x: 180, y: 200)
  path.move(to: start)
  path.addArc(startPoint: start, endPoint: end, angle: Double.pi * 1.6, clockwise: true)
  path.move(to: start)
  path.addArc(startPoint: start, endPoint: end, angle: Double.pi * 0.8, clockwise: true)
  
  start = CGPoint(x: 142, y: 130)
  end = CGPoint(x: 162, y: 200)
  path.move(to: start)
  path.addArc(startPoint: start, endPoint: end, angle: Double.pi * 0.4, clockwise: true)
  
  start = CGPoint(x: 140, y: 130)
  end = CGPoint(x: 160, y: 200)
  path.move(to: start)
  path.addArc(startPoint: start, endPoint: end, angle: Double.pi * 1.6, clockwise: false)
  path.move(to: start)
  path.addArc(startPoint: start, endPoint: end, angle: Double.pi * 0.8, clockwise: false)
  
  path.close()
  path.lineWidth = 1
  UIColor.red.setStroke()
  path.stroke()
}

ps: 每次都寫 Double.pi / x 很煩? 試試類似于 SwiftUI 提供的接口, 使用 度數(shù)(degress) 而非 弧度(radian)

struct Angle {
  private var degress: Double
  static func deggess(_ degress: Double) -> Angle {
    return .init(degress: degress)
  }
  // 弧度
  var radians: Double { Double.pi * degress / 180.0 }
}

// Angle.deggess(90).radians // 1.570796326794897
func addArc(startPoint start: CGPoint, endPoint end: CGPoint, angle: Angle, clockwise: Bool)

感謝閱讀,祝好祝順🥰

總結(jié)

到此這篇關(guān)于iOS新增繪制圓的文章就介紹到這了,更多相關(guān)iOS新增繪制圓內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • iOS中給自定義tabBar的按鈕添加點(diǎn)擊放大縮小的動畫效果

    iOS中給自定義tabBar的按鈕添加點(diǎn)擊放大縮小的動畫效果

    這篇文章主要介紹了iOS中給自定義tabBar的按鈕添加點(diǎn)擊放大縮小的動畫效果的相關(guān)資料,非常不錯,具有參考解決價值,需要的朋友可以參考下
    2016-11-11
  • 刪除xcode 中過期的描述性文件方法

    刪除xcode 中過期的描述性文件方法

    下面小編就為大家分享一篇刪除xcode 中過期的描述性文件方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-01-01
  • iOS實現(xiàn)百度地圖定位簽到功能

    iOS實現(xiàn)百度地圖定位簽到功能

    這篇文章主要給大家介紹了iOS實現(xiàn)百度地圖定位簽到功能的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-01-01
  • iOS與Unity交互筆記之參數(shù)傳遞

    iOS與Unity交互筆記之參數(shù)傳遞

    這篇文章主要給大家介紹了關(guān)于iOS與Unity交互筆記之參數(shù)傳遞的相關(guān)資料,需要的朋友可以參考下
    2019-04-04
  • iOS視頻添加背景音樂同時保留原音

    iOS視頻添加背景音樂同時保留原音

    本文主要介紹了iOS視頻添加背景音樂同時保留原音的實現(xiàn)方法。具有很好的參考價值,下面跟著小編一起來看下吧
    2017-03-03
  • IOS開發(fā)過程中的消息通知--小紅點(diǎn)

    IOS開發(fā)過程中的消息通知--小紅點(diǎn)

    本文主要介紹了IOS開發(fā)過程中的消息通知--小紅點(diǎn)的相關(guān)知識。大致分為兩種方法:系統(tǒng)方法和自定義方法。下面跟著小編一起來看下吧
    2017-04-04
  • IOS檢測指定路徑的文件是否存在

    IOS檢測指定路徑的文件是否存在

    本文給大家分享的是在IOS開發(fā)中檢測指定文件是否存在的方法,給大家匯總了4種,十分實用,小伙伴們根據(jù)自己的需求自由選擇吧。
    2015-05-05
  • iOS開發(fā)教程之Status Bar狀態(tài)欄設(shè)置的方法匯總

    iOS開發(fā)教程之Status Bar狀態(tài)欄設(shè)置的方法匯總

    iOS 的 Status Bar 狀態(tài)欄是一個比較坑的地方,所以下面這篇文章主要給大家介紹了關(guān)于iOS開發(fā)教程之Status Bar狀態(tài)欄設(shè)置的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2018-08-08
  • iOS實現(xiàn)動態(tài)的開屏廣告示例代碼

    iOS實現(xiàn)動態(tài)的開屏廣告示例代碼

    啟動圖是在iOS開發(fā)過程中必不可少的一個部分,很多app在啟動圖之后會有一張自定義的開屏廣告圖,但是有的時候需要讓啟動圖看起來就是一個廣告,而且還要這個廣告里面會動,iOS的啟動圖只能是靜態(tài)的,而且固定,為了實現(xiàn)看起來的動畫效果,只能進(jìn)行偽造了。下面來一起看看
    2016-09-09
  • 詳解ios中自定義cell,自定義UITableViewCell

    詳解ios中自定義cell,自定義UITableViewCell

    本篇文章主要介紹了ios中自定義cell,自定義UITableViewCell,非常具有實用價值,需要的朋友可以參考下。
    2016-12-12

最新評論