(转)iOS—添加文字水印(swift版)

原文链接(iOS—添加文字水印)
最近有个写水印需求,搜到了上面的文章,修复了一些小问题,改成swift实现,如下:

import Foundation
import UIKit

fileprivate let HORIZONTAL_SPACEING: CGFloat = 30 //水平间距
fileprivate let VERTICAL_SPACEING: CGFloat = 50 //竖直间距
fileprivate let CG_TRANSFORM_ROTATING = CGFloat(-Double.pi / 6) //旋转角度(正旋45度 || 反旋45度)

class WaterMarkView: UIView {
    public private(set) lazy var waterMarkImgV: UIImageView = {
        let imgV = UIImageView(frame: self.bounds)
        imgV.image = UIImage(named: "")
        return imgV
    }()
    
    private static let shared = WaterMarkView(frame: UIScreen.main.bounds)
    /// 水印文字
    public private(set) var markText: NSString = "默认水印文字"

    override init(frame: CGRect) {
        super.init(frame: frame)

        self.addSubview(waterMarkImgV)
        waterMarkImgV.frame = frame
    }

    required init?(coder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }

    /// 显示水印
    public static func show(view: UIView, text: String = "默认水印文字") {
        let markText = text as NSString
        let image = waterImage(size: view.bounds.size, text: markText)
        let shared =  WaterMarkView.shared
        shared.markText = markText
        shared.waterMarkImgV.image = image
        
        view.addSubview(shared)
    }
    
    /// 隐藏水印
    public static func hide() {
        let shared =  WaterMarkView.shared
        shared.removeFromSuperview()
    }

    /// 水印图片
    private static func waterImage(size: CGSize, text: NSString, font: UIFont = UIFont.systemFont(ofSize: 17), textColor: UIColor = .black) -> UIImage? {
        let imgW: CGFloat = size.width
        let imgH: CGFloat = size.height
        // 1.开启上下文
        UIGraphicsBeginImageContextWithOptions(size, false, 0)

        // 2.绘制图片
        //属性字符串
        let attr: [NSAttributedString.Key: Any] = [.font: font, .foregroundColor: textColor]
        let attrStr = NSMutableAttributedString(string: text as String, attributes: attr)

        //文字宽高
        let strW: CGFloat = attrStr.size().width
        let strH: CGFloat = attrStr.size().height

        //获取当前上下文
        guard let context = UIGraphicsGetCurrentContext() else { return nil }

        //调整矩阵锚点到中心
        context.concatenate(CGAffineTransform(translationX: imgW * 0.5, y: imgH * 0.5))
        //旋转矩阵
        context.concatenate(CGAffineTransform(rotationAngle: CG_TRANSFORM_ROTATING))
        //恢复锚点位置到左上角
        context.concatenate(CGAffineTransform(translationX: -imgW * 0.5, y: -imgH * 0.5))

        //对角线长度(实际绘制范围是对角线长度的正方形)
        let sqrtLength: CGFloat = sqrt(pow(imgW, 2) + pow(imgH, 2))

        //绘制的行数和列数
        let countHor = Int(sqrtLength / (strW + HORIZONTAL_SPACEING)) + 1
        let countVer = Int(sqrtLength / (strH + VERTICAL_SPACEING)) + 1

        //画布起点
        let orginX: CGFloat = -(sqrtLength - imgW) * 0.5
        let orginY: CGFloat = -(sqrtLength - imgH) * 0.5

        //双循环实现
        var i = 0
        var j = 0
        while i < countVer {
            j = 0
            let overlayOriginY = orginY + CGFloat(i) * (strH + VERTICAL_SPACEING)
            while j < countHor {
                let overlayOrginX = orginX + CGFloat(j) * (strW + HORIZONTAL_SPACEING)
                //绘制单个文字水印
                let drawRect = CGRect(x: overlayOrginX, y: overlayOriginY, width: strW, height: strH)
                text.draw(in: drawRect, withAttributes: attr)
                j += 1
            }
            i += 1
        }

        //3.上下文获取新图片
        let newImage = UIGraphicsGetImageFromCurrentImageContext()

        //4.关闭上下文
        UIGraphicsEndImageContext()
        context.restoreGState()

        return newImage
    }
    
    override func hitTest(_ point: CGPoint, with event: UIEvent?) -> UIView? {
        return nil
    }

}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,843评论 6 502
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,538评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,187评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,264评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,289评论 6 390
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,231评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,116评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,945评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,367评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,581评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,754评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,458评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,068评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,692评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,842评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,797评论 2 369
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,654评论 2 354

推荐阅读更多精彩内容