- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我正在尝试为我的应用制作一个“记录按钮”,它是一个带有手势识别器的 UIView。现在我正在实现这样一个功能,即当我单击 View 时,我想缩小内圈(红色圆圈),但它最终会发生意想不到的转变。我使用 UIView.animate 函数来执行此操作,下面是我的相关代码:
import UIKit
@IBDesignable
class RecordView: UIView {
@IBInspectable
var borderColor: UIColor = UIColor.clear {
didSet {
self.layer.borderColor = borderColor.cgColor
}
}
@IBInspectable
var borderWidth: CGFloat = 20 {
didSet {
layer.borderWidth = borderWidth
}
}
@IBInspectable
var cornerRadius: CGFloat = 100 {
didSet {
layer.cornerRadius = cornerRadius
}
}
private var fillView = UIView()
private func setupFillView() {
let radius = (self.cornerRadius - self.borderWidth) * 0.95
fillView.frame = CGRect(origin: CGPoint.zero, size: CGSize(width: radius * 2, height: radius * 2))
fillView.center = CGPoint(x: self.bounds.midX, y: self.bounds.midY)
fillView.layer.cornerRadius = radius
fillView.backgroundColor = UIColor.red
self.addSubview(fillView)
}
override func layoutSubviews() {
super.layoutSubviews()
setupFillView()
}
func didClick() {
UIView.animate(withDuration: 1.0, animations: {
self.fillView.transform = CGAffineTransform(scaleX: 0.6, y: 0.6)
}) { (true) in
print()
}
}
}
在我的 ViewController 中,我有:
@IBAction func recoreViewTapped(_ sender: UITapGestureRecognizer) {
recordView.didClick()
}
然而最终效果是这样的: https://media.giphy.com/media/3ohjUQrWt7vfIxzBrG/giphy.gif .我是初学者,真的不知道我的代码有什么问题?
最佳答案
问题是你正在应用一个转换,它再次触发 layoutSubviews
,所以 fillView
的 frame
的重置正在应用于转换后的 View 。
至少有两种选择:
您可以只转换整个 RecordButton
View :
@IBDesignable
class RecordView: UIView {
@IBInspectable
var borderColor: UIColor = .clear { didSet { layer.borderColor = borderColor.cgColor } }
@IBInspectable
var borderWidth: CGFloat = 20 { didSet { layer.borderWidth = borderWidth; setNeedsLayout() } }
private var fillView = UIView()
override init(frame: CGRect) {
super.init(frame: frame)
configure()
}
required init?(coder aDecoder: NSCoder) {
super.init(coder: aDecoder)
configure()
}
private func configure() {
fillView.backgroundColor = .red
self.addSubview(fillView)
}
override func layoutSubviews() {
super.layoutSubviews()
let radius = min(bounds.width, bounds.height) / 2 - borderWidth
fillView.frame = CGRect(origin: CGPoint(x: bounds.midX - radius, y: bounds.midY - radius),
size: CGSize(width: radius * 2, height: radius * 2))
fillView.layer.cornerRadius = radius
}
func didClick() {
UIView.animate(withDuration: 1.0, animations: {
self.transform = CGAffineTransform(scaleX: 0.6, y: 0.6)
}, completion: { _ in
print("completion", #function)
})
}
}
或者您可以将 fillView
放入容器中,然后 fillView
的转换不会触发 layoutSubviews
RecordView
的:
@IBDesignable
class RecordView: UIView {
@IBInspectable
var borderColor: UIColor = .clear { didSet { layer.borderColor = borderColor.cgColor } }
@IBInspectable
var borderWidth: CGFloat = 20 { didSet { layer.borderWidth = borderWidth; setNeedsLayout() } }
private var fillView = UIView()
private var containerView = UIView()
override init(frame: CGRect) {
super.init(frame: frame)
configure()
}
required init?(coder aDecoder: NSCoder) {
super.init(coder: aDecoder)
configure()
}
private func configure() {
fillView.backgroundColor = .red
self.addSubview(containerView)
containerView.addSubview(fillView)
}
override func layoutSubviews() {
super.layoutSubviews()
containerView.frame = bounds
let radius = min(bounds.width, bounds.height) / 2 - borderWidth
fillView.frame = CGRect(origin: CGPoint(x: bounds.midX - radius, y: bounds.midY - radius),
size: CGSize(width: radius * 2, height: radius * 2))
fillView.layer.cornerRadius = radius
}
func didClick() {
UIView.animate(withDuration: 1.0, animations: {
self.fillView.transform = CGAffineTransform(scaleX: 0.6, y: 0.6)
}, completion: { _ in
print("completion", #function)
})
}
}
顺便说一下其他一些小事:
正如您和我在别处讨论的那样,初始配置(例如添加 subview )应该从 init
调用。但是任何 frame
/bounds
偶然的东西都应该从 layoutSubviews
中调用。
完全不相关,但是传递给UIView.animate(withDuration:animations:completion:)
的completion
的参数是一个 bool 值,表示动画是否为finished
与否。如果您不打算使用该 bool 值,则应使用 _
,而不是 (true)
。
从长远来看,我建议将“触摸”/手势相关内容移至 RecordButton。我可以很容易地想象你变得更加狂热:例如一个动画是你“按下”(缩小按钮以呈现“按下”的感觉),另一个是“抬起”(例如,取消缩小按钮并将圆形“记录”按钮转换为方形“停止”按钮)。然后,您可以让按钮在发生重大事件时通知 View Controller (例如记录或停止识别),但会降低 View Controller 本身的复杂性。
protocol RecordViewDelegate: class {
func didTapRecord()
func didTapStop()
}
@IBDesignable
class RecordView: UIView {
@IBInspectable
var borderColor: UIColor = .clear { didSet { layer.borderColor = borderColor.cgColor } }
@IBInspectable
var borderWidth: CGFloat = 20 { didSet { layer.borderWidth = borderWidth; setNeedsLayout() } }
weak var delegate: RecordViewDelegate?
var isRecordButton = true
private var fillView = UIView()
private var containerView = UIView()
override init(frame: CGRect) {
super.init(frame: frame)
configure()
}
required init?(coder aDecoder: NSCoder) {
super.init(coder: aDecoder)
configure()
}
private func configure() {
fillView.backgroundColor = .red
self.addSubview(containerView)
containerView.addSubview(fillView)
}
private var radius: CGFloat { return min(bounds.width, bounds.height) / 2 - borderWidth }
override func layoutSubviews() {
super.layoutSubviews()
containerView.frame = bounds
fillView.frame = CGRect(origin: CGPoint(x: bounds.midX - radius, y: bounds.midY - radius),
size: CGSize(width: radius * 2, height: radius * 2))
fillView.layer.cornerRadius = isRecordButton ? radius : 0
}
override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
UIView.animate(withDuration: 0.25) {
self.fillView.transform = CGAffineTransform(scaleX: 0.8, y: 0.8)
self.fillView.backgroundColor = UIColor(red: 0.75, green: 0, blue: 0, alpha: 1)
}
}
override func touchesEnded(_ touches: Set<UITouch>, with event: UIEvent?) {
if let touch = touches.first, containerView.bounds.contains(touch.location(in: containerView)) {
if isRecordButton {
delegate?.didTapRecord()
} else {
delegate?.didTapStop()
}
isRecordButton = !isRecordButton
}
UIView.animate(withDuration: 0.25) {
self.fillView.transform = .identity
self.fillView.backgroundColor = UIColor.red
self.fillView.layer.cornerRadius = self.isRecordButton ? self.radius : 0
}
}
}
产生:
关于ios - Swift:UIView.animate 工作意外,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47615855/
我有一个 uiview,其中包含其他自定义绘制的 uiview。现在我希望,当用户单击此父 View 时,它将扩展到带有动画的新定义的框架。我使用动画 block 取得了一些成功。问题是展开的 Vie
我正在创建当前显示的 UIView 的屏幕截图。在此过程中,我遇到了控件扭曲和拉伸(stretch)片刻(0.2-0.5 秒)的问题。它仅适用于 iPhone 6、6+。 我的简单代码在这里 (Xam
我想让主 UIView 负责将多个 UIView 之一添加为 subview 。由于任何时候只有一个 subview 处于事件状态,而 subview 完全覆盖了主视图,所以我还不如更换主视图。但这需
这真的只是好奇,但是标签属性有最大值吗?我假设标签是一个 UINT,但我不确定。 最佳答案 “标签”属性是一个 NSInteger。来自 Apple 的 docs : NSInteger Used t
我的布局中有一个 UIView 以便进行一些剪辑和分组,但是自动布局会在缩小时调整它的大小。我想给它一个固定的高度,但唯一的选择是设置顶部和底部空间。 有没有办法设置明确的高度约束? 最佳答案 我刚刚
使用 NSLayoutConstraint类,可以创建基于 View 基线 ( NSLayoutAttributeBaseline ) 的约束。但是,我还没有看到任何描述 UIView 的文档。实际上
我正在尝试创建一个 UIView,它显示一个半透明的圆圈,其边界内有一个不透明的边框。我希望能够通过两种方式更改边界 - 在 -[UIView animateWithDuration:animatio
我目前正在 iOS 中开发一个项目(使用 XCode 和 Swift)。我正在尝试为登录 View 实现以下 UITextFields: 我正在考虑不同的方法来做这件事,但它们看起来都很复杂。如果有人
我正在使用最新的 SDK 开发 iOS 应用程序。 我创建了一个自定义 UIView,我需要向该 UIView 添加一些控件。我也在使用 storyboards,并且我必须向其中添加这个 UIView
我已经通过界面生成器在 super View 中启用了Autoresize Subviews。有没有办法以编程方式禁用一个 subview View 的自动调整大小? 最佳答案 我假设您确实在使用自动
我正在寻找用 Swift 2.0 编写的适用于 iOS 8.0 或更高版本的答案。我有一个圆形的 UIView,我想扩展它但保留圆形的属性。 要设置 UIView,这是代码。 let frame: C
我有一个有趣的问题,但我不确定如何解决它。我正在尝试使用 @IBInspectable 扩展 UIView。但是,使用这种方法,拐角半径似乎是从 nib 文件的默认端设置的,而不是 View 的实际大
我想将 UIView 移动到圆圈内。 UIView 移动圆内的每个点,但不接触圆的边界线。我正在计算圆和 UIView 之间的距离。 var distance = sqrt( pow((tou
A 是一个自定义的 UIView B是UIImagePickerController.view,这个UIImagePickerController.showsCameraControls = NO A
我无法在启动时修改我的 UIView 高度。 我必须使用 UIView,我希望其中一个屏幕大小 * 70,另一个填补空白。 这是我的 @IBOutlet weak var questionFrame
我一直在想这个问题。 我的情况是我需要在 View 中显示很多网格,我有两个计划。 1、在UIView上画很多矩形。我试过了,但是当我放大 View 时,这条线很模糊。 2、给UIView添加很多CA
我可以通过简单地将容器拖动到新的 UIView 来成功地将第一个 UIView 嵌入到容器中,但是要添加过渡 View ,我无法在按键输入事件上用新 View 替换当前的嵌入 View 。 这是我的代
问题是... 我有一条路径,我创建了一个在其中绘制它的 View 。 View 与路径具有相同的维度。然后,我创建第二个 View ,在其中重写 sizeThatFit: 方法,以便缩放第一个 Vie
我正致力于快速设计自定义 UIimageview。我想使用类似于此的 beizerpath 创建一个 UIimageview 编码应该是快速的。任何帮助表示赞赏。谢谢 最佳答案 创建一个 CAShap
我有一个 iOS 应用程序可以在 UIView 上动态绘制形状(通过 drawRect),我正在研究将该应用程序(或其中的一小部分)移植到 Apple Watch 的可能性。不幸的是,在阅读相关帖子后
我是一名优秀的程序员,十分优秀!