学习下自定义控件。从开源项目中挖了一个 理解模仿了一个
主要理解了下为什么不适用顺时针绘制。(当然也可以)
=====================================================
//绘制角为逆时针的
//界面角为顺时针的所以 绘制角度=360-界面(计算)角
//为什么不使用顺时针绘制
//不论顺时针逆时针 都可以得出 360-计算角=绘制角
//如果用顺时针,则绘制角与实际大小成反比 既 计算角270度时,绘制角为90 实际绘制270度。此时成反比不利于计算
//如果用逆时针,则绘制角与实际大小成正比 既 计算角270度时,绘制角为90 实际绘制90度。利于计算
=====================================================
代码如下
// // PLCircle.m // DrawCirclePercent // // Created by liu poolo on 14-4-1. // Copyright (c) 2014年 liu poolo. All rights reserved. // #define ToRad(deg) ( (M_PI * (deg)) / 180.0 ) #define ToDeg(rad) ( (180.0 * (rad)) / M_PI ) #define SQR(x) ( (x) * (x) ) #import "PLCircle.h" @interface PLCircle(){ int radius; } @end @implementation PLCircle - (id)initWithFrame:(CGRect)frame { radius=80; self.angle=0; self = [super initWithFrame:frame]; if (self) { // Initialization code self.backgroundColor=[UIColor clearColor]; } return self; } -(void)drawRect:(CGRect)rect{ [super drawRect:rect]; CGContextRef ctx=UIGraphicsGetCurrentContext(); //绘制背景 CGContextAddArc(ctx, CGRectGetMidX(self.frame), CGRectGetMidY(self.frame), radius, 0, M_PI*2, 1); [[UIColor blackColor] setStroke]; CGContextSetLineWidth(ctx, PL_SLIDER_BG_LINE_WIDTH); CGContextSetLineCap(ctx, kCGLineCapButt); CGContextDrawPath(ctx, kCGPathStroke); //绘制Mask 遮掩层 用于获取圆环选区 UIGraphicsBeginImageContext(CGSizeMake(PL_SLIDER_WIDTH, PL_SLIDER_WIDTH)); CGContextRef imgCtx=UIGraphicsGetCurrentContext(); CGContextAddArc(imgCtx, PL_SLIDER_WIDTH/2, PL_SLIDER_WIDTH/2, radius, 0, ToRad(self.angle), 0); //The clockwise parameter determines the direction in which the arc is created; the actual direction of the final path is dependent on the current transformation matrix of the graphics context. For example, on iOS, a UIView flips the Y-coordinate by scaling the Y values by -1. In a flipped coordinate system, specifying a clockwise arc results in a counterclockwise arc after the transformation is applied. //这是说,方向实际由graphics context的变换矩阵决定。比如在UIView中,Y轴被翻转之后,顺时针的将会变成逆时针的。所以你需要检查你实验用的环境中具体采取了什么变换。 //1与0 默认1 是顺时针 但是用到UIView之中后 1是逆时针 //绘制角为逆时针的 //界面角为顺时针的所以 绘制角度=360-界面(计算)角 //为什么不使用顺时针绘制 //不论顺时针逆时针 都可以得出 360-计算角=绘制角 //如果用顺时针,则绘制角与实际大小成反比 既 计算角270度时,绘制角为90 实际绘制270度。此时成反比不利于计算 //如果用逆时针,则绘制角与实际大小成正比 既 计算角270度时,绘制角为90 实际绘制90度。利于计算 CGContextSetShadowWithColor(imgCtx, CGSizeMake(0, 0), self.angle/20, [UIColor grayColor].CGColor); CGContextSetLineWidth(imgCtx, PL_SLIDER_LINE_WIDTH); CGContextDrawPath(imgCtx, kCGPathStroke); //保存context到图片掩盖层 CGImageRef mask = CGBitmapContextCreateImage(UIGraphicsGetCurrentContext()); UIGraphicsEndImageContext(); //获得切割后的Context CGContextSaveGState(ctx); CGContextClipToMask(ctx, self.bounds, mask); CGImageRelease(mask); //绘制渐变 CGFloat components[8] = { 0.78, 0.80, 0.82, 1.0, // 开始色 0.05, 0.05, 0.05, 1.0 }; // 结束色 //可配置4的倍数 配置改变时需要修改CGGradientCreateWithColorComponents最后的参数 CGColorSpaceRef baseSpace=CGColorSpaceCreateDeviceRGB(); CGGradientRef gradient =CGGradientCreateWithColorComponents(baseSpace, components, NULL, 2); CGColorSpaceRelease(baseSpace); baseSpace=NULL; //绘制渐变 CGPoint startPoint = CGPointMake(CGRectGetMidX(rect), CGRectGetMinY(rect)); CGPoint endPoint = CGPointMake(CGRectGetMidX(rect), CGRectGetMaxY(rect)); //绘制渐变 CGContextDrawLinearGradient(ctx, gradient, startPoint, endPoint, 0); CGGradientRelease(gradient), gradient = NULL; CGContextRestoreGState(ctx); [self drawTheHandle:ctx]; } #pragma mark - UIControl Override - /** Tracking is started **/ -(BOOL)beginTrackingWithTouch:(UITouch *)touch withEvent:(UIEvent *)event{ [super beginTrackingWithTouch:touch withEvent:event]; //We need to track continuously return YES; } /** Track continuos touch event (like drag) **/ -(BOOL)continueTrackingWithTouch:(UITouch *)touch withEvent:(UIEvent *)event{ [super continueTrackingWithTouch:touch withEvent:event]; //Get touch location CGPoint lastPoint = [touch locationInView:self]; //Use the location to design the Handle [self movehandle:lastPoint]; //Control value has changed, let's notify that [self sendActionsForControlEvents:UIControlEventValueChanged]; return YES; } /** Track is finished **/ -(void)endTrackingWithTouch:(UITouch *)touch withEvent:(UIEvent *)event{ [super endTrackingWithTouch:touch withEvent:event]; } /** Draw a white knob over the circle **/ -(void) drawTheHandle:(CGContextRef)ctx{ CGContextSaveGState(ctx); //I Love shadows CGContextSetShadowWithColor(ctx, CGSizeMake(0, 0), 3, [UIColor blackColor].CGColor); //Get the handle position CGPoint handleCenter = [self pointFromAngle: self.angle]; //Draw It! [[UIColor colorWithWhite:1.0 alpha:0.7]set]; CGContextFillEllipseInRect(ctx, CGRectMake(handleCenter.x, handleCenter.y, PL_SLIDER_LINE_WIDTH, PL_SLIDER_LINE_WIDTH)); CGContextRestoreGState(ctx); } #pragma mark - Math - /** Move the Handle **/ -(void)movehandle:(CGPoint)lastPoint{ //Get the center CGPoint centerPoint = CGPointMake(self.frame.size.width/2, self.frame.size.height/2); //Calculate the direction from a center point and a arbitrary position. float currentAngle = AngleFromNorth(centerPoint, lastPoint, NO); NSLog(@"currentAngle:%f ToRad(self.angle):%f",currentAngle,ToRad(self.angle)); int angleInt = floor(currentAngle); //Store the new angle self.angle = 360 - angleInt; // self.angle = angleInt; //Update the textfield // int result=self.angle*size/360; // _textField.text = [NSString stringWithFormat:@"%d", result]; //Redraw [self setNeedsDisplay]; } /** Given the angle, get the point position on circumference **/ -(CGPoint)pointFromAngle:(int)angleInt{ //Circle center CGPoint centerPoint = CGPointMake(self.frame.size.width/2 - PL_SLIDER_LINE_WIDTH/2, self.frame.size.height/2 - PL_SLIDER_LINE_WIDTH/2); //The point position on the circumference CGPoint result; result.y = round(centerPoint.y + radius * sin(ToRad(-angleInt))) ; result.x = round(centerPoint.x + radius * cos(ToRad(-angleInt))); return result; } //Sourcecode from Apple example clockControl //Calculate the direction in degrees from a center point to an arbitrary position. static inline float AngleFromNorth(CGPoint p1, CGPoint p2, BOOL flipped) { CGPoint v = CGPointMake(p2.x-p1.x,p2.y-p1.y); float vmag = sqrt(SQR(v.x) + SQR(v.y)), result = 0; v.x /= vmag; v.y /= vmag; double radians = atan2(v.y,v.x); result = ToDeg(radians); return (result >=0 ? result : result + 360.0); } @end
相关推荐
"ios-自定义封装日历--格式年月日.zip" 是一个开发者分享的项目,它提供了一个简单的自定义日历视图(View)实现,主要针对年、月、日的选择,并集成了系统自带的UIDatePicker。下面我们将详细探讨这个自定义日历...
综上所述,创建一个适用于表视图的自定义纵向滑动条涉及到了多个iOS开发的关键知识点,包括自定义视图、事件处理、布局管理以及性能优化。通过`CustomSliderView`的实现,开发者可以为自己的应用增添独特的视觉元素...
在iOS应用开发中,自定义TabBar是一种常见的需求,它能提供更加个性化和独特的用户体验。本文将深入探讨如何实现一个自定义的TabBar,并结合首页TableView实现类似QQ侧滑菜单的效果。这一技术栈主要涉及iOS UI设计、...
好用的自定义ActionSheet 内部使用了Masnory布局 所以使用前提是要导入Masnory 支持图片模式和标题模式 标题模式可以传入颜色数组控制标题颜色 简单易用 如果觉得好用,感谢给我点一颗星 ...
1. **自定义视图**: 在iOS中,我们可以创建自定义的UIView子类来实现自定义的Segment Control。这涉及到重写draw(_ rect: CGRect)方法,使用Core Graphics框架绘制下滑线和按钮的边框。 2. **按钮(UIButton)**: ...
iOS7.0后苹果提供了自定义转场动画的API,利用这些API我们可以改变 push和pop(navigation非模态),present和dismiss(模态),标签切换(tabbar)的默认转场动画。
在iOS开发中,自定义密码输入框是一种常见的需求,它允许开发者根据应用的特定设计风格创建独特的用户体验。本文将深入探讨如何在iOS中实现自定义密码输入框,并提供相关技术要点。 首先,我们来理解一下iOS系统...
1. **自定义颜色**: 开发者可以通过设置`setTitleTextAttributes:`方法来改变UISegmentedControl的文字颜色,以及通过`setTintColor:`来调整选中状态下的颜色。此外,可以使用`setBackgroundImage:forState:`来替换...
本项目“ios-Slider-双向选择渐变色滑动条.zip”提供了一个创新的滑动条实现,名为WWTwoSliderView,它具有双向选择和渐变色的特性,增加了交互的视觉效果和用户体验。 WWTwoSliderView是基于Swift编程语言编写的,...
这个名为"ios-iOS自定义横线导航-适配iPad.zip"的压缩包资源,显然提供了关于如何在iOS应用中创建一个自定义底部导航栏的解决方案,特别强调了对iPad设备的适配。下面,我们将深入探讨这一话题。 首先,我们要理解...
2. 设计外观:自定义滑动条的外观主要是通过修改滑块(thumb)和轨道(track)的图像。我们可以使用`setThumbImage:forState:`和`setMinimumTrackImage:forState:`、`setMaximumTrackImage:forState:`方法设置不同...
【标题】"IOS应用源码Demo-自定义风格UIPickerView-毕设学习"是一个针对iOS平台的应用程序源代码示例,重点展示了如何为UIPickerView定制独特样式,以适应不同设计需求或个人喜好。在iOS开发中,UIPickerView是常见...
在iOS开发中,自定义背景滑动特效选项条是一种常见的用户界面元素,它能提供更加丰富和个性化的交互体验。这种选项条通常用于展示多个可选项目,用户可以通过滑动来选择他们感兴趣的内容。本文将深入探讨如何在iOS中...
1. Swift或Objective-C编程:作为iOS开发的基础,这两门语言是实现自定义键盘的关键。 2. UIKit框架:UIKit提供了UIInputView和其子类,是构建自定义键盘的主要工具。 3. AutoLayout和Size Classes:用于处理不同...
在iOS开发中,自定义键盘是一项常见的需求,它允许开发者为用户提供更为个性化和高效的操作体验。Objective-C作为苹果平台的主要编程语言之一,是实现这一功能的重要工具。本篇将深入探讨如何在iOS应用中创建自定义...
这个名为“ios-iOS自定义一个左右滑动切换页面框架”的项目,提供了一个高效、灵活的解决方案,用于实现标题与内容页面间的互动切换。该框架的核心在于FSScrollContentView,它允许用户通过点击标题或滑动内容区域来...
在iOS应用开发中,自定义控件是提升用户体验和实现独特设计的重要手段。"超炫的自定义 iOS Filter Control" 是一个很好的实例,适合初学者和正在进行毕业设计的学生进行学习和参考。这个项目展示了如何创建一个...
1. **创建自定义插件** - 首先,你需要在你的项目中创建一个新的目录,用于存放插件代码。 - 使用 `cordova plugin add <path-to-plugin>` 命令将插件添加到你的项目中。这会将插件的结构和代码集成到 Cordova ...
本项目“ios-简单自定义圆形颜色渐变进度条”正是关注于这一方面,旨在创建一个具有颜色渐变效果的圆形进度条。通过这个项目,我们可以学习到如何在iOS应用中实现自定义视图和动画。 首先,我们需要理解`...