내 정보 UIView
가 사용자에게 표시 되는지 여부를 결정할 수 있습니까?
내보기로 추가 subview
에 여러 번 Tab Bar Controller
.
이보기의 각 인스턴스에는보기 NSTimer
를 업데이트하는가 있습니다.
그러나 사용자에게 표시되지 않는보기를 업데이트하고 싶지 않습니다.
이것이 가능한가?
감사
답변:
다음 사항을 확인할 수 있습니다.
view.superview != nil
내가 생각할 수있는 유일한 다른 것은 당신의 견해가 다른 사람들 뒤에 묻혀서 그 이유로 보이지 않는 경우입니다. 당신은 그들이 당신의 시야를 가리는지 확인하기 위해 이후에 오는 모든 견해를 검토해야 할 수도 있습니다.
여기서 끝나는 다른 사람을 위해 :
UIView의 오히려 검사보다 화면 어딘가에 있는지 확인하려면 superview != nil
,이 있는지 확인하는 것이 좋습니다 window != nil
. 전자의 경우 뷰에 수퍼 뷰가 있지만 수퍼 뷰가 화면에 없을 수 있습니다.
if (view.window != nil) {
// do stuff
}
물론 그것이 있는지 hidden
또는 alpha > 0
.
NSTimer
보기가 표시되지 않는 동안 실행하는 것을 원하지 않는 경우 가능하면 이러한보기를 수동으로 숨기고보기가 숨겨 질 때 타이머가 중지되도록해야합니다. 그러나 나는 당신이 무엇을하는지 전혀 확신하지 못합니다.
po [self.view recursiveDescription]
내 하위보기는보기 계층 구조에 표시되지만 view.window
항상 nil입니다.
self.view
에도 a window
가 nil 이라고 생각합니다 . 그럴까요? (늦은 답변 죄송합니다)
이것은 뷰의 프레임이 모든 수퍼 뷰의 경계 내에 있는지 (루트 뷰까지) 결정합니다. 실제 사용 사례 중 하나는 스크롤 뷰 내에서 하위 뷰가 (적어도 부분적으로) 표시되는지 확인하는 것입니다.
func isVisible(view: UIView) -> Bool {
func isVisible(view: UIView, inView: UIView?) -> Bool {
guard let inView = inView else { return true }
let viewFrame = inView.convert(view.bounds, from: view)
if viewFrame.intersects(inView.bounds) {
return isVisible(view: view, inView: inView.superview)
}
return false
}
return isVisible(view: view, inView: view.superview)
}
func isVisible(view: UIView) -> Bool {
func isVisible(view: UIView, inView: UIView?) -> Bool {
guard let inView = inView else { return true }
let viewFrame = inView.convertRect(view.bounds, fromView: view)
if CGRectIntersectsRect(viewFrame, inView.bounds) {
return isVisible(view, inView: inView.superview)
}
return false
}
return isVisible(view, inView: view.superview)
}
잠재적 개선 :
alpha
과 hidden
.clipsToBounds
뷰가 거짓 인 경우 수퍼 뷰의 경계를 초과 할 수 있으므로 존중하십시오 .나를 위해 일한 해결책은 먼저 뷰에 창이 있는지 확인한 다음 수퍼 뷰를 반복하고 다음 사항을 확인하는 것입니다.
지금까지 잘 작동하는 것 같습니다.
public func isVisible(view: UIView) -> Bool {
if view.window == nil {
return false
}
var currentView: UIView = view
while let superview = currentView.superview {
if (superview.bounds).intersects(currentView.frame) == false {
return false;
}
if currentView.isHidden {
return false
}
currentView = superview
}
return true
}
보기가 사용자에게 표시되는지 진정으로 알고 싶습니다. 다음 사항을 고려해야합니다.
특히 전면 뷰의 투명한 배경색은 프로그래밍 방식으로 확인하는 데 문제가 될 수 있습니다. 진정으로 확신하는 유일한 방법은 전체 화면의 스냅 샷과 함께 프레임 내에서 확인하고 비교하기 위해 뷰의 프로그래밍 방식 스냅 샷을 만드는 것입니다. 그러나 충분히 뚜렷하지 않은 뷰 (예 : 완전히 흰색)에는 작동하지 않습니다.
영감을 얻으려면 iOS Calabash-server 프로젝트의 isViewVisible 메소드를 참조하십시오.
테스트 된 솔루션.
func isVisible(_ view: UIView) -> Bool {
if view.isHidden || view.superview == nil {
return false
}
if let rootViewController = UIApplication.shared.keyWindow?.rootViewController,
let rootView = rootViewController.view {
let viewFrame = view.convert(view.bounds, to: rootView)
let topSafeArea: CGFloat
let bottomSafeArea: CGFloat
if #available(iOS 11.0, *) {
topSafeArea = rootView.safeAreaInsets.top
bottomSafeArea = rootView.safeAreaInsets.bottom
} else {
topSafeArea = rootViewController.topLayoutGuide.length
bottomSafeArea = rootViewController.bottomLayoutGuide.length
}
return viewFrame.minX >= 0 &&
viewFrame.maxX <= rootView.bounds.width &&
viewFrame.minY >= topSafeArea &&
viewFrame.maxY <= rootView.bounds.height - bottomSafeArea
}
return false
}
@Audrey M.과 @John Gibb의 솔루션을 벤치마킹했습니다.
그리고 @Audrey M. 그의 방식이 더 잘 수행되었습니다 (10 번).
그래서 나는 그것을 관찰 가능하게 만들기 위해 그것을 사용했습니다.
UIView가 표시 될 때 알림을 받기 위해 RxSwift Observable을 만들었습니다.
배너 '보기'이벤트를 트리거하려는 경우 유용 할 수 있습니다.
import Foundation
import UIKit
import RxSwift
extension UIView {
var isVisibleToUser: Bool {
if isHidden || alpha == 0 || superview == nil {
return false
}
guard let rootViewController = UIApplication.shared.keyWindow?.rootViewController else {
return false
}
let viewFrame = convert(bounds, to: rootViewController.view)
let topSafeArea: CGFloat
let bottomSafeArea: CGFloat
if #available(iOS 11.0, *) {
topSafeArea = rootViewController.view.safeAreaInsets.top
bottomSafeArea = rootViewController.view.safeAreaInsets.bottom
} else {
topSafeArea = rootViewController.topLayoutGuide.length
bottomSafeArea = rootViewController.bottomLayoutGuide.length
}
return viewFrame.minX >= 0 &&
viewFrame.maxX <= rootViewController.view.bounds.width &&
viewFrame.minY >= topSafeArea &&
viewFrame.maxY <= rootViewController.view.bounds.height - bottomSafeArea
}
}
extension Reactive where Base: UIView {
var isVisibleToUser: Observable<Bool> {
// Every second this will check `isVisibleToUser`
return Observable<Int>.interval(.milliseconds(1000),
scheduler: MainScheduler.instance)
.flatMap { [base] _ in
return Observable.just(base.isVisibleToUser)
}.distinctUntilChanged()
}
}
다음과 같이 사용하십시오.
import RxSwift
import UIKit
import Foundation
private let disposeBag = DisposeBag()
private func _checkBannerVisibility() {
bannerView.rx.isVisibleToUser
.filter { $0 }
.take(1) // Only trigger it once
.subscribe(onNext: { [weak self] _ in
// ... Do something
}).disposed(by: disposeBag)
}
이렇게하면 UIView가 최상위보기인지 파악하는 데 도움이됩니다. 도움이 될 수 있습니다.
let visibleBool = view.superview?.subviews.last?.isEqual(view)
//have to check first whether it's nil (bc it's an optional)
//as well as the true/false
if let visibleBool = visibleBool where visibleBool { value
//can be seen on top
} else {
//maybe can be seen but not the topmost view
}
이 시도:
func isDisplayedInScreen() -> Bool
{
if (self == nil) {
return false
}
let screenRect = UIScreen.main.bounds
//
let rect = self.convert(self.frame, from: nil)
if (rect.isEmpty || rect.isNull) {
return false
}
// 若view 隐藏
if (self.isHidden) {
return false
}
//
if (self.superview == nil) {
return false
}
//
if (rect.size.equalTo(CGSize.zero)) {
return false
}
//
let intersectionRect = rect.intersection(screenRect)
if (intersectionRect.isEmpty || intersectionRect.isNull) {
return false
}
return true
}
숨겨진 속성을 사용하는 경우 다음을 수행하십시오.
view.hidden (objective C) 또는 view.isHidden (swift)은 읽기 / 쓰기 속성입니다. 따라서 쉽게 읽고 쓸 수 있습니다.
신속한 3.0 용
if(view.isHidden){
print("Hidden")
}else{
print("visible")
}
.window
(mahboudz에 의해) 확인하는 대답이.superview
기술적으로 정확하지 않고 저에게 버그를 일으켰 기 때문에 가장 많은 찬성 투표를 한 대답 (walkingbrad에 의해) 에 대해 선택한 대답을 업데이트하는 것을 고려 하시겠습니까 ? .