zoukankan      html  css  js  c++  java
  • Core Graphics之CGContext详解 (转)

    CGContext又叫图形上下文,相当于一块画布,以堆栈形式存放,只有在当前 context上绘图才有效。iOS有分多种图形上下文,其中UIView自带提供的在drawRect:方法中通过 UIGraphicsGetCurrentContext获取,还有专门为图片处理context,UIGraphicsBeginImageContext函数生成,还有pdf的context等等。


    1.一共有3种使用context的场景,其中每种场景都有2种方法绘图

    场景1:

    //通过UIView的子类的drawRect:在上下文中绘制,该方法系统已准备好一个cgcontext,并放置在上下文栈顶,rect形参就是context的尺寸大小
    //当一个UIView的backgroundColor为nil和opaque为YES时,产生的context的背景就为黑色的

    - (void)drawRect:(CGRect)rect
    {
        //1.使用UIKit在context上绘制,UIKit的所有操作只会在当前栈顶的context,所以需要注意当前栈顶的context是否你需要操作的上下文
        //UIImage,NSString,UIBezierPath,UIColor等可以直接在当前context上操作

        UIImage* image = [UIImage imageNamed:@"test.png"];
        NSLog(@"size:%@",NSStringFromCGSize(image.size));
        //UIImage直接在context上操作,指定在context的哪个坐标上绘制,大小是原图的尺寸,如果大小超出了context的范围就会被截取掉
       // [image drawAtPoint:CGPointMake(100, 100)];
        //指定在context的哪个坐标上绘制,并指定绘制的图片尺寸大小,这样图片的尺寸就会被压缩,不会超出context范围
        [image drawInRect:CGRectMake(0, 0, rect.size.width/2, rect.size.height/2)];
      

       //2.使用Core Graphics的函数在context上绘制,Core Graphics的函数需要context作为参数,只绘制在指定使用的context上
        //功过UIGraphicsGetCurrentContext函数获取当前上下文栈顶的context,UIView系统已为其准备好context并存放在栈顶了

    //    CGContextRef context = UIGraphicsGetCurrentContext();
    //    //画一个椭圆
    //    CGContextAddEllipseInRect(context, CGRectMake(0,0,100,100));
    //    //填充颜色为蓝色
    //    CGContextSetFillColorWithColor(context, [UIColor blueColor].CGColor);
    //    //在context上绘制
    //    CGContextFillPath(context);

    场景2:

    //实现该方法,用于CALayer回调,CALayer通过它的代理类来进行绘图操作,切记千万不能把UIView作为CALayer的代理类,因为UIView自身有隐式的图层,若再把显式的图层赋给它会发生不知名错误的
    - (void)drawLayer:(CALayer*)layer inContext:(CGContextRef)ctx
    {
        //1.使用UIKit进行绘制,因为UIKit只会对当前上下文栈顶的context操作,所以要把形参中的context设置为当前上下文
        UIGraphicsPushContext(ctx);
        UIImage* image = [UIImage imageNamed:@"test.png"];
        //指定位置和大小绘制图片
        [image drawInRect:CGRectMake(0, 0,100 , 100)];
        UIGraphicsPopContext();
        
        //    UIGraphicsPushContext(ctx);
        //2.使用Core Graphics进行绘制,需要显式使用context
        //    //画一个椭圆
        //    CGContextAddEllipseInRect(ctx, CGRectMake(0,0,100,100));
        //    //填充颜色为蓝色
        //    CGContextSetFillColorWithColor(ctx, [UIColor blueColor].CGColor);
        //    //在context上绘制
        //    CGContextFillPath(ctx);
        //    UIGraphicsPopContext();
    }


        LayerDelegate* delegate = [[LayerDelegate alloc]init];
        CALayer* layer = [CALayer layer];
        layer.anchorPoint = CGPointMake(0, 0);
        layer.position = CGPointMake(100, 100);
        layer.bounds = CGRectMake(0, 0, 200, 200);
        layer.delegate = delegate;
        //需要显式调用setNeedsDisplay来刷新才会绘制layer
        [layer setNeedsDisplay];
        [self.view.layer addSublayer:layer];


    下图为场景1和场景2的显示效果:




    场景3:

     //通过自己创建一个context来绘制,通常用于对图片的处理
        /*
         解释一下UIGraphicsBeginImageContextWithOptions函数参数的含义:第一个参数表示所要创建的图片的尺寸;第二个参 数用来指定所生成图片的背景是否为不透明,如上我们使用YES而不是NO,则我们得到的图片背景将会是黑色,显然这不是我想要的;第三个参数指定生成图片 的缩放因子,这个缩放因子与UIImage的scale属性所指的含义是一致的。传入0则表示让图片的缩放因子根据屏幕的分辨率而变化,所以我们得到的图 片不管是在单分辨率还是视网膜屏上看起来都会很好。
         */
        //该函数会自动创建一个context,并把它push到上下文栈顶,坐标系也经处理和UIKit的坐标系相同

        UIGraphicsBeginImageContextWithOptions(rect.size, NO, 0);
        CGContextRef context = UIGraphicsGetCurrentContext();
        CGContextAddEllipseInRect(context, CGRectMake(0,0,100,100));
        //填充颜色为蓝色
        CGContextSetFillColorWithColor(context, [UIColor blueColor].CGColor);
        //在context上绘制
        CGContextFillPath(context);
        //把当前context的内容输出成一个UIImage图片
        UIImage* i = UIGraphicsGetImageFromCurrentImageContext();
        //上下文栈pop出创建的context
        UIGraphicsEndImageContext();
        [i drawInRect:CGRectMake(0, 0, 100, 100)];


    下图为绘制的显示效果:




    2.把整个屏幕转化为图片

        UIImageView* imageV = [[UIImageView alloc]initWithFrame:CGRectMake(0, 0, self.view.frame.size.width, self.view.frame.size.height)];
        UIGraphicsBeginImageContextWithOptions(imageV.frame.size, NO, 0);
        CGContextRef context = UIGraphicsGetCurrentContext();
        //把当前的整个画面导入到context中,然后通过context输出UIImage,这样就可以把整个屏幕转化为图片
        [self.view.layer renderInContext:context];
        UIImage* image = UIGraphicsGetImageFromCurrentImageContext();
        imageV.image = image;
        UIGraphicsEndImageContext();


    3.剪裁图片

       //对一张图片进行剪裁
        CGImageRef imageref = CGImageCreateWithImageInRect(image.CGImage, CGRectMake(100, 100, 200, 50));
        UIImageView* cropImage = [[UIImageView alloc]initWithFrame:CGRectMake(100, 300, 200, 50)];
        cropImage.image = [UIImage imageWithCGImage:imageref];

       CGImageRelease(imageref);

        [self.view addSubview:cropImage];


    下图下放的是剪裁后的图片效果:

    原文:http://blog.csdn.net/kingsley_cxz/article/details/9191479

  • 相关阅读:
    JavaScript中判断函数是new还是()调用
    IE6/7 and IE8/9(Q)中td的上下padding失效
    JQuery中html()方法使用不当带来的陷阱
    有name为action的表单元素时取form的属性action杯具了
    为非IE浏览器添加mouseenter,mouseleave事件
    各浏览器中querySelector和querySelectorAll的实现差异
    仅IE6/7/8中innerHTML返回值忽略英文空格
    各浏览器关键字/保留字作为对象属性的差异
    各浏览器中鼠标按键值的差异
    给body标签和document.body都添加点击事件后仅Firefox弹出了两次
  • 原文地址:https://www.cnblogs.com/songlei/p/3424847.html
Copyright © 2011-2022 走看看