Mask a UIView with a cut-out circle

前端 未结 3 1021
南旧
南旧 2020-12-13 21:09

I’ve created a UIView with a semi-transparent black background to sit across my main app’s view. What I’m aiming for here is to create a cut-out circle shape so

相关标签:
3条回答
  • 2020-12-13 21:22

    If You are using any Black Image to mask then

    -(void)maskWithImage:(UIImage*)maskImage toImageView:(UIImageView*)imgView{
    
        CALayer *aMaskLayer=[CALayer layer];
        aMaskLayer.contents=(id)maskImage.CGImage;
    
        imgView.layer.mask=aMaskLayer;
    
    
    }
    

    If you have any Custom ShapeLayer to mask

    -(void)maskWithShapeLayer:(CALayer*)layer toImageView:(UIImageView*)imgView{
        //Layer should be filled with Black color to mask those part of Image
        imgView.layer.mask=layer;
    }
    

    If You want to make ImageView Mask with Circle

    -(void)maskWithCircle:(UIImageView*)imgView{
    
        CAShapeLayer *aCircle=[CAShapeLayer layer];
        aCircle.path=[UIBezierPath bezierPathWithRoundedRect:imgView.bounds cornerRadius:imgView.frame.size.height/2].CGPath; // Considering the ImageView is square in Shape
    
        aCircle.fillColor=[UIColor blackColor].CGColor;
        imgView.layer.mask=aCircle;
    
    }
    
    0 讨论(0)
  • 2020-12-13 21:23

    So, here’s what I did. I created a custom UIView subclass called BlackoutView, like so:

    BlackoutView.h

    #import <UIKit/UIKit.h>
    
    @interface BlackoutView : UIView
    
    @property (nonatomic, retain) UIColor *fillColor;
    @property (nonatomic, retain) NSArray *framesToCutOut;
    
    @end
    

    BlackoutView.m

    #import "BlackoutView.h"
    
    @implementation BlackoutView
    
    - (void)drawRect:(CGRect)rect
    {
        [self.fillColor setFill];
        UIRectFill(rect);
    
        CGContextRef context = UIGraphicsGetCurrentContext();
        CGContextSetBlendMode(context, kCGBlendModeDestinationOut);
    
        for (NSValue *value in self.framesToCutOut) {
            CGRect pathRect = [value CGRectValue];
            UIBezierPath *path = [UIBezierPath bezierPathWithOvalInRect:pathRect];
            [path fill];
        }
    
        CGContextSetBlendMode(context, kCGBlendModeNormal);
    }
    
    @end
    

    I then instantiate it as normal, and set the properties to the colour of the mask I want to use, and the frames to be cut out of the mask:

    [blackMask setFillColor:[UIColor colorWithWhite:0.0f alpha:0.8f]];
    [blackMask setFramesToCutOut:@[[NSValue valueWithCGRect:buttonCircleA.frame],
                                   [NSValue valueWithCGRect:buttonCircleB.frame]]];
    

    This could be improved by allowing me to cut out other shapes besides ovals, but it’s fine for my purposes here and would be easily adapted to do so later. Hopefully this helps others!

    0 讨论(0)
  • 2020-12-13 21:26

    Here is my conversion of the accepted answer in Swift:

    public class OverlayView: UIView {
    let fillColor: UIColor = UIColor.grayColor()
    public var framesToCutOut: Array<NSValue> = [NSValue]()
    
    override public func drawRect(rect: CGRect) {
        super.drawRect(rect)
    
        fillColor.setFill()
        UIRectFill(rect)
    
        if let context: CGContextRef = UIGraphicsGetCurrentContext() {
            CGContextSetBlendMode(context, .DestinationOut)
    
            for value in framesToCutOut {
                let pathRect: CGRect = value.CGRectValue()
                let path: UIBezierPath = UIBezierPath(ovalInRect: pathRect)
                path.fill()
            }
    
            CGContextSetBlendMode(context, .Normal)
        }
    }
    
    0 讨论(0)
提交回复
热议问题