Ограничение уровней масштабирования UIPinchGestureRecognizer Цель C для Swift 3.0
iOS 10.2 Swift 3.0
Попытка перевести этот фрагмент кода из блога Пола Солта. Исправлено СО постером, обновлен код!
http://paulsolt.com/blog/2011/03/limiting-uipinchgesturerecognizer-zoom-levels
Образец кода
- (void)handlePinchGesture:(UIPinchGestureRecognizer *)gestureRecognizer {
if([gestureRecognizer state] == UIGestureRecognizerStateBegan) {
// Reset the last scale, necessary if there are multiple objects with different scales
lastScale = [gestureRecognizer scale];
}
if ([gestureRecognizer state] == UIGestureRecognizerStateBegan ||
[gestureRecognizer state] == UIGestureRecognizerStateChanged) {
CGFloat currentScale = [[[gestureRecognizer view].layer valueForKeyPath:@"transform.scale"] floatValue];
// Constants to adjust the max/min values of zoom
const CGFloat kMaxScale = 2.0;
const CGFloat kMinScale = 1.0;
CGFloat newScale = 1 - (lastScale - [gestureRecognizer scale]); // new scale is in the range (0-1)
newScale = MIN(newScale, kMaxScale / currentScale);
newScale = MAX(newScale, kMinScale / currentScale);
CGAffineTransform transform = CGAffineTransformScale([[gestureRecognizer view] transform], newScale, newScale);
[gestureRecognizer view].transform = transform;
lastScale = [gestureRecognizer scale]; // Store the previous scale factor for the next pinch gesture call
}
}
Почти готово, но не могу найти ссылку на ключ CATransform, используемый здесь для Swift 3.0. Мой код...
if sender.state == .began {
// Reset the last scale, necessary if there are multiple objects with different scales
lastScale = sender.scale
}
if sender.state == .began || sender.state == .changed {
// UPDATED
currentScaleX = self.image2P.transform.scaleX
currentScaleY = self.image2P.transform.scaleY
self.image2P.transform = self.image2P.transform.scaledBy(x: 1.1, y: 1.1)
// Constants to adjust the max/min values of zoom
let kMaxScale:CGFloat = 2.0;
let kMinScale:CGFloat = 1.0;
var newScale = 1 - (lastScale - sender.scale) // new scale is in the range (0-1)
newScale = min(newScale, kMaxScale / currentScaleX)
newScale = max(newScale, kMinScale / currentScaleY)
self.image2P.transform = self.image2P.transform.scaledBy(x: newScale, y: newScale)
lastScale = sender.scale // Store the previous scale factor for the next pinch gesture call
}
1 ответ
Вы должны хранить xScale
а также yScale
отдельно, потому что, вообще говоря, нет гарантии, что они равны.
extension CGAffineTransform {
var scaleX: CGFloat {
return (a > 0 ? 1 : -1) * sqrt (a*a + c*c)
}
var scaleY: CGFloat {
return (d > 0 ? 1 : -1) * sqrt (b*b + d*d)
}
}
Эти методы расширения будут работать, возвращая правильные масштабные коэффициенты, даже если ваше представление также повернуто и / или переведено:
view.transform = CGAffineTransform(scaleX: 1.5, y: 1.2).rotated(by: .pi/3.0).translatedBy(x: 50, y: 30)
print("scaleX = \(view.transform.scaleX), scaleY = \(view.transform.scaleY)")
Выход:
ScaleX = 1,5, ScaleY = 1,2
Это может быть не очевидно, но a
, b
, c
а также d
свойства являются элементами transform
матрица. Вы можете найти более подробную информацию в Quartz 2D Programming Guide. Также вы можете найти математические детали здесь. Просто знайте, что b
а также c
имена элементов меняются местами в этих двух источниках.