-
-
Save kasimok/f742e1bfa155f045aae313f066808888 to your computer and use it in GitHub Desktop.
An IBInspectable Calculator Construction Set
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// CalculatorView.swift | |
// as seen in http://nshipster.com/ibinspectable-ibdesignable/ | |
// | |
// (c) 2015 Nate Cook, licensed under the MIT license | |
import UIKit | |
/// The alignment for drawing an String inside a bounding rectangle. | |
enum NCStringAlignment { | |
case LeftTop | |
case CenterTop | |
case RightTop | |
case LeftCenter | |
case Center | |
case RightCenter | |
case LeftBottom | |
case CenterBottom | |
case RightBottom | |
} | |
extension String { | |
/// Draw the `String` inside the bounding rectangle with a given alignment. | |
func drawAtPointInRect(rect: CGRect, withAttributes attributes: [NSAttributedString.Key: Any]?, andAlignment alignment: NCStringAlignment) { | |
let size = (self as NSString).size(withAttributes: attributes) | |
let x, y: CGFloat | |
switch alignment { | |
case .LeftTop, .LeftCenter, .LeftBottom: | |
x = rect.minX | |
case .CenterTop, .Center, .CenterBottom: | |
x = rect.midX - size.width / 2 | |
case .RightTop, .RightCenter, .RightBottom: | |
x = rect.maxX - size.width | |
} | |
switch alignment { | |
case .LeftTop, .CenterTop, .RightTop: | |
y = rect.minY | |
case .LeftCenter, .Center, .RightCenter: | |
y = rect.midY - size.height / 2 | |
case .LeftBottom, .CenterBottom, .RightBottom: | |
y = rect.maxY - size.height | |
} | |
self.draw(at: CGPoint(x: x, y: y), withAttributes: attributes) | |
} | |
} | |
extension CGRect { | |
/// Convert a rect to a pixel-aligned version, rounding position and size | |
func pixelAligned() -> CGRect { | |
return CGRect(x: round(origin.x), y: round(origin.y), | |
width: round(size.width), height: round(size.height)) | |
} | |
} | |
@IBDesignable | |
class CalculatorView: UIView { | |
let (columns, rows) = (4, 6) as (CGFloat, CGFloat) | |
// basic buttons: all 1x1 unit | |
let buttonRows = [ ["C", "=", "/", "*"], ["7", "8", "9", "-"], ["4", "5", "6", "+"], ["1", "2", "3"] ] | |
// special cases: non-square or out of position | |
let specialButtons = [("=", CGPoint(x: 3, y: 4), CGSize(width: 1, height: 2)), | |
("0", CGPoint(x: 0, y: 5), CGSize(width: 2, height: 1)), | |
(".", CGPoint(x: 2, y: 5), CGSize(width: 1, height: 1))] | |
// MARK: - IBInspectable | |
@IBInspectable var faceColor: UIColor = UIColor(red: 211/255, green: 211/255, blue: 211/255, alpha: 1) | |
@IBInspectable var textColor: UIColor = UIColor.black | |
@IBInspectable var textSize: CGFloat = 12 | |
@IBInspectable var buttonColor: UIColor = UIColor.white | |
@IBInspectable var edgeColor: UIColor = UIColor.black | |
@IBInspectable var edgeWidth: CGFloat = 1 | |
@IBInspectable var shadowColor: UIColor = UIColor.black | |
@IBInspectable var shadowOffset: CGFloat = 2 | |
@IBInspectable var cornerRadius: CGFloat = 7 | |
@IBInspectable var paddingX: CGFloat = 7 | |
@IBInspectable var paddingY: CGFloat = 8 | |
@IBInspectable var spacingX: CGFloat = 3 | |
@IBInspectable var spacingY: CGFloat = 5 | |
@IBInspectable var titleHeight: CGFloat = 20 | |
@IBInspectable var titleColor: UIColor = UIColor.white | |
@IBInspectable var titleBarColor: UIColor = UIColor.black | |
// MARK: - Drawing | |
override func draw(_ rect: CGRect) { | |
// initial drawing setup | |
// determine unit height/width from view size, padding and button spacing | |
let unitHeight = (bounds.height - 2 * paddingY - (rows - 1) * spacingY - titleHeight) / rows | |
let unitWidth = (bounds.width - 2 * paddingX - (columns - 1) * spacingX) / columns | |
// set up text attributes | |
let textFont = UIFont.systemFont(ofSize: textSize) | |
let textAttributes = [NSAttributedString.Key.font: textFont, NSAttributedString.Key.foregroundColor: textColor] as [NSAttributedString.Key : Any] | |
// get the graphics context | |
let context = UIGraphicsGetCurrentContext() | |
// helper functions | |
// maps a point and size (in units) to an actual rect in the view's coordinates for drawing | |
func rectForPosition(position: CGPoint, andSize size: CGSize) -> CGRect { | |
let x = paddingX + unitWidth * position.x + spacingX * position.x | |
let y = titleHeight + paddingY + unitHeight * position.y + spacingY * position.y | |
let width = size.width * unitWidth + (size.width - 1) * spacingX | |
let height = size.height * unitHeight + (size.height - 1) * spacingY | |
return CGRect(x: x, y: y, width: width, height: height).pixelAligned() | |
} | |
// draw a button in the specified rect | |
func drawButtonInRect(rect: CGRect, withText text: String) { | |
shadowColor.setFill() | |
UIRectFill(rect.offsetBy(dx: shadowOffset, dy: shadowOffset)) | |
buttonColor.setFill() | |
edgeColor.setStroke() | |
context!.setLineWidth(edgeWidth) | |
UIRectFill(rect) | |
UIRectFrame(rect) | |
var textRect = rect | |
if textRect.width != unitWidth { | |
textRect.size.width = unitWidth | |
} | |
if textRect.height != unitHeight { | |
textRect.origin.y = textRect.maxY - unitHeight | |
textRect.size.height = unitHeight | |
} | |
text.drawAtPointInRect(rect: textRect, withAttributes: textAttributes as [NSAttributedString.Key : Any]?, andAlignment: .Center) | |
} | |
// draw the display | |
func drawDisplayInRect(rect: CGRect, withText text: String) { | |
buttonColor.setFill() | |
edgeColor.setStroke() | |
context!.setLineWidth(edgeWidth) | |
UIRectFill(rect) | |
UIRectFrame(rect) | |
text.drawAtPointInRect(rect: rect.offsetBy(dx: unitWidth / -3, dy: 0), withAttributes: textAttributes as [NSAttributedString.Key : AnyObject]?, andAlignment: .RightCenter) | |
} | |
// calculator "window" - paint the color of the title bar | |
let window = UIBezierPath(roundedRect: bounds, cornerRadius: cornerRadius) | |
titleBarColor.setFill() | |
window.fill() | |
// center the title in the title bar | |
let title = "Calculator" | |
let (titleRect, _) = bounds.divided(atDistance: titleHeight, from: CGRectEdge.minYEdge) | |
let titleAttributes = [NSAttributedString.Key.font: UIFont.boldSystemFont(ofSize: textSize + 2), NSAttributedString.Key.foregroundColor: titleColor] as [NSAttributedString.Key : Any] | |
title.drawAtPointInRect(rect: titleRect, withAttributes: titleAttributes as [NSAttributedString.Key : Any]?, andAlignment: .Center) | |
// calculator "face" - covers most of the window | |
let face = UIBezierPath(roundedRect: bounds.insetBy(dx: 0, dy: titleHeight / 2).offsetBy(dx: 0, dy: titleHeight / 2), cornerRadius: cornerRadius) | |
faceColor.setFill() | |
face.fill() | |
// draw the display | |
let displayRect = rectForPosition(position: CGPoint(x: 0, y: 0), andSize: CGSize(width: 4, height: 1)) | |
drawDisplayInRect(rect: displayRect, withText: "3") | |
// draw basic buttons | |
for (rowNumber, row) in buttonRows.enumerated() { | |
for (columnNumber, text) in row.enumerated() { | |
drawButtonInRect(rect: rectForPosition(position: CGPoint(x: columnNumber, y: rowNumber + 1), andSize: CGSize(width: 1, height: 1)), withText: text) | |
} | |
} | |
// draw special buttons | |
for buttonInfo in specialButtons { | |
drawButtonInRect(rect: rectForPosition(position: buttonInfo.1, andSize: buttonInfo.2), withText: buttonInfo.0) | |
} | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Update to SWIFT5