17 Commits

Author SHA1 Message Date
Haik Aslanyan e02e761c2e changed version number to 1.3.0 2020-05-07 21:10:48 +04:00
Haik Aslanyan 9e5ddf0082 updated example project 2020-05-07 21:10:04 +04:00
Haik Aslanyan aace0fd52d added ScanningMode parameter 2020-05-07 21:08:40 +04:00
Haik Aslanyan 63fd478b58 modified scanning logic for dark background images 2020-05-07 21:08:08 +04:00
Haik Aslanyan 783ac44550 moved decoding calculation into different queue 2020-05-07 17:07:24 +04:00
Haik Aslanyan 05f71742a1 bumped to version 1.2.0 2020-05-07 14:45:25 +04:00
Haik Aslanyan d544551032 small change 2020-05-06 19:00:46 +04:00
Haik Aslanyan 255195d1b4 modified scanning area logic to support non centered images 2020-05-06 18:56:59 +04:00
Haik Aslanyan cbeea6ef44 removed redundant UIImagePickerControllerDelegate 2020-05-01 12:30:00 +04:00
Haik Aslanyan 3d584f7b23 updated example project 2020-04-30 21:50:34 +04:00
Haik Aslanyan e2a305dcf3 updated pixelThreshold range 2020-04-19 17:22:02 +04:00
Haik Aslanyan fb3fc03fe1 Merge pull request #4 from aslanyanhaik/develop
Develop
2020-04-19 16:56:00 +04:00
Haik Aslanyan 2e45c812b9 Merge branch 'develop' of https://github.com/aslanyanhaik/roundCode into develop 2020-04-19 16:53:45 +04:00
Haik Aslanyan 795bed4c5d updated to 1.1.1 version 2020-04-19 16:53:39 +04:00
Haik Aslanyan 843df78755 updated example project 2020-04-19 16:51:02 +04:00
Haik Aslanyan 5f88ff991e added color color validation for black background 2020-04-19 16:50:14 +04:00
Haik Aslanyan 395f608cee changed pixelThreshold from constant to range 2020-04-19 16:47:53 +04:00
8 changed files with 134 additions and 90 deletions
+13 -6
View File
@@ -43,7 +43,7 @@
<rect key="frame" x="20" y="93" width="374" height="32"/> <rect key="frame" x="20" y="93" width="374" height="32"/>
<segments> <segments>
<segment title="Orange"/> <segment title="Orange"/>
<segment title="Magenta"/> <segment title="Red"/>
<segment title="Green"/> <segment title="Green"/>
<segment title="Brown"/> <segment title="Brown"/>
<segment title="Cyan"/> <segment title="Cyan"/>
@@ -84,11 +84,18 @@
<viewLayoutGuide key="safeArea" id="aXm-cj-1eD"/> <viewLayoutGuide key="safeArea" id="aXm-cj-1eD"/>
</view> </view>
<navigationItem key="navigationItem" id="XgG-ZM-i9d"> <navigationItem key="navigationItem" id="XgG-ZM-i9d">
<barButtonItem key="leftBarButtonItem" title="Share" id="so2-d3-VCF"> <leftBarButtonItems>
<connections> <barButtonItem title="Share" id="so2-d3-VCF">
<action selector="share:" destination="0zP-ee-W0J" id="uNU-bV-PaS"/> <connections>
</connections> <action selector="share:" destination="0zP-ee-W0J" id="uNU-bV-PaS"/>
</barButtonItem> </connections>
</barButtonItem>
<barButtonItem title="Config" id="33c-Ng-bhx">
<connections>
<action selector="changeConfig:" destination="0zP-ee-W0J" id="izA-qU-8it"/>
</connections>
</barButtonItem>
</leftBarButtonItems>
<rightBarButtonItems> <rightBarButtonItems>
<barButtonItem title="Camera" id="Ckh-RB-Kad"> <barButtonItem title="Camera" id="Ckh-RB-Kad">
<connections> <connections>
+4 -2
View File
@@ -22,8 +22,8 @@
<true/> <true/>
<key>NSAppleMusicUsageDescription</key> <key>NSAppleMusicUsageDescription</key>
<string>To scan roundcodes</string> <string>To scan roundcodes</string>
<key>NSPhotoLibraryAddUsageDescription</key> <key>NSPhotoLibraryAddUsageDescription</key>
<string>To save RoundCode images</string> <string>To save RoundCode images</string>
<key>NSCameraUsageDescription</key> <key>NSCameraUsageDescription</key>
<string>To scan roundcodes</string> <string>To scan roundcodes</string>
<key>UIApplicationSceneManifest</key> <key>UIApplicationSceneManifest</key>
@@ -64,5 +64,7 @@
<string>UIInterfaceOrientationLandscapeLeft</string> <string>UIInterfaceOrientationLandscapeLeft</string>
<string>UIInterfaceOrientationLandscapeRight</string> <string>UIInterfaceOrientationLandscapeRight</string>
</array> </array>
<key>UIUserInterfaceStyle</key>
<string>Light</string>
</dict> </dict>
</plist> </plist>
+28 -6
View File
@@ -30,17 +30,39 @@ class ViewController: UIViewController {
@IBOutlet weak var messageLabel: UILabel! @IBOutlet weak var messageLabel: UILabel!
var coder = RCCoder() var coder = RCCoder()
var image = RCImage(message: "") var image = RCImage(message: "")
let colors: [[UIColor]] = [[.orange, .gray], [.orange, .magenta], [.systemGreen, .systemBlue], [.orange, .brown], [.purple, .cyan]] let colors: [[UIColor]] = [[.orange, .magenta], [.systemPink, .systemTeal], [.systemTeal, .systemGreen], [.orange, .brown], [.purple, .cyan]]
var isScanningFromLibrary = false var isScanningFromLibrary = false
} }
extension ViewController { extension ViewController {
override func viewDidLoad() { @IBAction func changeConfig(_ sender: Any) {
super.viewDidLoad() textField.text = ""
image.isTransparent = true image.message = ""
imageView.image = try? coder.encode(image)
let vc = UIAlertController(title: "Select Configuration", message: nil, preferredStyle: .actionSheet)
let uuidAction = UIAlertAction(title: "UUID", style: .default) { _ in
self.coder = RCCoder(configuration: .uuidConfiguration)
}
let numericAction = UIAlertAction(title: "Numeric", style: .default) { _ in
self.coder = RCCoder(configuration: .numericConfiguration)
}
let shortAction = UIAlertAction(title: "Short", style: .default) { _ in
self.coder = RCCoder(configuration: .shortConfiguration)
}
let defaultAction = UIAlertAction(title: "Default", style: .default) { _ in
self.coder = RCCoder(configuration: .defaultConfiguration)
}
let cancelAction = UIAlertAction(title: "Cancel", style: .cancel)
vc.addAction(uuidAction)
vc.addAction(numericAction)
vc.addAction(shortAction)
vc.addAction(defaultAction)
vc.addAction(cancelAction)
present(vc, animated: true)
} }
@IBAction func scanImage(_ sender: Any) { @IBAction func scanImage(_ sender: Any) {
isScanningFromLibrary = true isScanningFromLibrary = true
let vc = UIImagePickerController() let vc = UIImagePickerController()
@@ -53,18 +75,18 @@ extension ViewController {
@IBAction func scan(_ sender: Any) { @IBAction func scan(_ sender: Any) {
let vc = RCCameraViewController() let vc = RCCameraViewController()
coder.scanningMode = .lightBackground
vc.coder = coder
vc.delegate = self vc.delegate = self
present(vc, animated: true) present(vc, animated: true)
} }
@IBAction func share(_ sender: Any) { @IBAction func share(_ sender: Any) {
image.size = 1000 image.size = 1000
image.isTransparent = false
image.contentInsets = UIEdgeInsets(top: 50, left: 50, bottom: 50, right: 50) image.contentInsets = UIEdgeInsets(top: 50, left: 50, bottom: 50, right: 50)
let uiImage = try? coder.encode(image) let uiImage = try? coder.encode(image)
image.size = 300 image.size = 300
image.contentInsets = UIEdgeInsets(top: 8, left: 8, bottom: 8, right: 8) image.contentInsets = UIEdgeInsets(top: 8, left: 8, bottom: 8, right: 8)
image.isTransparent = true
let vc = UIActivityViewController.init(activityItems: [uiImage!], applicationActivities: nil) let vc = UIActivityViewController.init(activityItems: [uiImage!], applicationActivities: nil)
present(vc, animated: true) present(vc, animated: true)
} }
+1 -1
View File
@@ -1,6 +1,6 @@
Pod::Spec.new do |s| Pod::Spec.new do |s|
s.name = 'RoundCode' s.name = 'RoundCode'
s.version = '1.1.0' s.version = '1.3.0'
s.summary = 'Facebook messenger style custom barcode.' s.summary = 'Facebook messenger style custom barcode.'
s.description = <<-DESC s.description = <<-DESC
Encode and decode data into custom stylish barcode. Encode and decode data into custom stylish barcode.
+2 -1
View File
@@ -28,5 +28,6 @@ struct RCConstants {
static let dotSizeScale: CGFloat = 0.08 static let dotSizeScale: CGFloat = 0.08
static let dotPatterns: [CGFloat] = [6, 4, 2] static let dotPatterns: [CGFloat] = [6, 4, 2]
static let dotPointRange = (Float(1.3)...Float(2.5)) static let dotPointRange = (Float(1.3)...Float(2.5))
static let pixelThreshold = 180 static let lightBackgroundRange = UInt8(0)...UInt8(180)
static let darkBackgroundRange = UInt8(100)...UInt8(255)
} }
+67 -61
View File
@@ -27,37 +27,18 @@ struct RCImageDecoder {
internal let configuration: RCCoderConfiguration internal let configuration: RCCoderConfiguration
internal var size = 720 internal var size = 720
internal var bytesPerRow = 720 internal var bytesPerRow = 720
internal var padding = 0 internal var pixelThreshold = RCConstants.lightBackgroundRange
private var sectionSize: Int {
self.size / 5
}
} }
extension RCImageDecoder { extension RCImageDecoder {
func process(pointer: UnsafeMutablePointer<UInt8>) throws -> [RCBit] { func process(pointer: UnsafeMutablePointer<UInt8>) throws -> [RCBit] {
let bufferData = UnsafeMutableBufferPointer<UInt8>(start: pointer, count: size * bytesPerRow) let bufferData = UnsafeMutableBufferPointer<UInt8>(start: pointer, count: size * bytesPerRow)
let data = PixelContainer(rows: bytesPerRow, items: bufferData) let data = PixelContainer(rows: bytesPerRow, items: bufferData)
var points = [CGPoint]() let points = try scanControlPoints(for: data)
for side in Side.allCases {
switch side {
case .left:
let point = try scanControlPoint(for: data, region: (padding, (size - sectionSize) / 2), side: side)
points.append(point)
case .top:
let point = try scanControlPoint(for: data, region: ((size - sectionSize) / 2, padding), side: side)
points.append(point)
case .right:
let point = try scanControlPoint(for: data, region: (size - sectionSize - padding, (size - sectionSize) / 2), side: side)
points.append(point)
case .bottom:
let point = try scanControlPoint(for: data, region: ((size - sectionSize) / 2, size - sectionSize - padding), side: side)
points.append(point)
}
}
let transform = calculateTransform(from: points) let transform = calculateTransform(from: points)
let mapper = RCPointMapper(transform: transform, size: size) let mapper = RCPointMapper(transform: transform, size: size)
let locations = mapper.map(points: calculateBitLocations()) let locations = mapper.map(points: calculateBitLocations())
let bits = locations.map { data[Int($0.x), Int($0.y)] > RCConstants.pixelThreshold ? RCBit.zero : RCBit.one } let bits = locations.map { pixelThreshold.contains(data[Int($0.x), Int($0.y)]) ? RCBit.one : RCBit.zero }
return bits return bits
} }
@@ -72,47 +53,67 @@ extension RCImageDecoder {
} }
extension RCImageDecoder { extension RCImageDecoder {
private func scanControlPoint(for data: PixelContainer, region: (x: Int, y: Int), side: Side) throws -> CGPoint { private func scanControlPoints(for data: PixelContainer) throws -> [CGPoint] {
var horizontalPatterns = [PixelPattern]()
func scan(region: (x: Int, y: Int, size: Int), data: PixelContainer, coordinate: (Int) -> (x: Int, y: Int), comparison: (PixelPattern, (x: Int, y: Int)) -> Bool) -> [PixelPattern] { var verticalPatterns = [PixelPattern]()
var lastPattern = PixelPattern(bit: data[region.x, region.y] > RCConstants.pixelThreshold ? RCBit.zero : RCBit.one, x: region.x, y: region.y, count: 0) var points = [CGPoint]()
var pixelPatterns = [lastPattern] for side in Side.allCases {
var count = 0 switch side {
let maxSize = region.size * region.size case .left:
while count < maxSize { horizontalPatterns = scanPixelPattern(for: .horizontal, data: data)
let coordinate = coordinate(count) points.append(try controlPoint(for: horizontalPatterns, side: side))
let bit = data[coordinate.x, coordinate.y] > RCConstants.pixelThreshold ? RCBit.zero : RCBit.one case .right:
if comparison(lastPattern, coordinate), lastPattern.bit == bit { points.append(try controlPoint(for: horizontalPatterns, side: side))
lastPattern.count += 1 case .top:
pixelPatterns[pixelPatterns.count - 1] = lastPattern verticalPatterns = scanPixelPattern(for: .vertical, data: data)
} else { points.append(try controlPoint(for: verticalPatterns, side: side))
lastPattern = PixelPattern(bit: bit, x: coordinate.x, y: coordinate.y, count: 1) case .bottom:
pixelPatterns.append(lastPattern) points.append(try controlPoint(for: verticalPatterns, side: side))
}
count += 1
} }
return pixelPatterns
} }
return points
let pixelPatterns: [PixelPattern] }
switch side {
case .left, .right:
pixelPatterns = scan(region: (region.x, region.y, sectionSize), data: data, coordinate: { count in private func scanPixelPattern(for mode: ScanMode, data: PixelContainer) -> [PixelPattern] {
let x = count % sectionSize + region.x var lastPattern = PixelPattern.init(bit: pixelThreshold.contains((data[0, 0])) ? RCBit.one : RCBit.zero, x: 0, y: 0, count: 0)
let y = count / sectionSize + region.y var pixelPatterns = [lastPattern]
return (x, y) var count = 0
}, comparison: { (pattern, coordinate) in let maxSize = size * size
return pattern.y == coordinate.y switch mode {
}) case .horizontal:
case .top, .bottom: while count < maxSize {
pixelPatterns = scan(region: (region.x, region.y, sectionSize), data: data, coordinate: { count in let x = count % size
let x = count / sectionSize + region.x let y = count / size
let y = count % sectionSize + region.y let bit = pixelThreshold.contains(data[x, y]) ? RCBit.one : RCBit.zero
return (x, y) if lastPattern.y == y, lastPattern.bit == bit {
}, comparison: { (pattern, coordinate) in lastPattern.count += 1
return pattern.x == coordinate.x pixelPatterns[pixelPatterns.count - 1] = lastPattern
}) } else {
lastPattern = PixelPattern(bit: bit, x: x, y: y, count: 1)
pixelPatterns.append(lastPattern)
}
count += 1
}
case .vertical:
while count < maxSize {
let x = count / size
let y = count % size
let bit = pixelThreshold.contains(data[x, y]) ? RCBit.one : RCBit.zero
if lastPattern.x == x, lastPattern.bit == bit {
lastPattern.count += 1
pixelPatterns[pixelPatterns.count - 1] = lastPattern
} else {
lastPattern = PixelPattern(bit: bit, x: x, y: y, count: 1)
pixelPatterns.append(lastPattern)
}
count += 1
}
} }
return pixelPatterns
}
private func controlPoint(for pixelPatterns: [PixelPattern], side: Side) throws -> CGPoint {
let controlPoints = try pixelPatterns.withUnsafeBufferPointer { (pixelPatternsBuffer) -> [CGPoint] in let controlPoints = try pixelPatterns.withUnsafeBufferPointer { (pixelPatternsBuffer) -> [CGPoint] in
var points = [CGPoint]() var points = [CGPoint]()
guard pixelPatternsBuffer.count >= 5 else { throw RCError.decoding } guard pixelPatternsBuffer.count >= 5 else { throw RCError.decoding }
@@ -164,7 +165,7 @@ extension RCImageDecoder {
} }
}.first! }.first!
} }
private func calculateTransform(from points: [CGPoint]) -> CATransform3D { private func calculateTransform(from points: [CGPoint]) -> CATransform3D {
let perspective = RCTransformation(points) let perspective = RCTransformation(points)
let middleRect = CGRect(origin: .zero, size: CGSize(width: CGFloat(size), height: CGFloat(size))) let middleRect = CGRect(origin: .zero, size: CGSize(width: CGFloat(size), height: CGFloat(size)))
@@ -215,6 +216,11 @@ extension RCImageDecoder {
case right case right
} }
enum ScanMode {
case horizontal
case vertical
}
final class PixelContainer { final class PixelContainer {
let rows: Int let rows: Int
var columns: Int { data.count / rows } var columns: Int { data.count / rows }
+6 -12
View File
@@ -23,7 +23,7 @@
import UIKit import UIKit
import AVFoundation import AVFoundation
public final class RCCameraViewController: UIViewController, UIImagePickerControllerDelegate { public final class RCCameraViewController: UIViewController {
//MARK: Public properties //MARK: Public properties
public weak var delegate: RCCameraViewControllerDelegate? public weak var delegate: RCCameraViewControllerDelegate?
@@ -113,7 +113,6 @@ public extension RCCameraViewController {
super.viewDidLayoutSubviews() super.viewDidLayoutSubviews()
configureMaskLayer() configureMaskLayer()
videoPreviewLayer?.frame = view.bounds videoPreviewLayer?.frame = view.bounds
calculateScanArea()
} }
} }
@@ -157,12 +156,11 @@ extension RCCameraViewController {
guard let captureDevice = AVCaptureDevice.default(for: .video) else { return } guard let captureDevice = AVCaptureDevice.default(for: .video) else { return }
do { do {
captureSession.sessionPreset = .hd1280x720 captureSession.sessionPreset = .hd1280x720
calculateScanArea()
let input = try AVCaptureDeviceInput(device: captureDevice) let input = try AVCaptureDeviceInput(device: captureDevice)
input.device.activeVideoMaxFrameDuration = CMTimeMake(value: 1, timescale: 30) input.device.activeVideoMaxFrameDuration = CMTimeMake(value: 1, timescale: 30)
captureSession.addInput(input) captureSession.addInput(input)
let videoOutput = AVCaptureVideoDataOutput() let videoOutput = AVCaptureVideoDataOutput()
videoOutput.setSampleBufferDelegate(self, queue: .main) videoOutput.setSampleBufferDelegate(self, queue: DispatchQueue.global(qos: .userInteractive))
videoOutput.videoSettings = [kCVPixelBufferPixelFormatTypeKey as String: NSNumber(value: kCVPixelFormatType_420YpCbCr8BiPlanarFullRange)] videoOutput.videoSettings = [kCVPixelBufferPixelFormatTypeKey as String: NSNumber(value: kCVPixelFormatType_420YpCbCr8BiPlanarFullRange)]
captureSession.addOutput(videoOutput) captureSession.addOutput(videoOutput)
} catch { } catch {
@@ -170,12 +168,6 @@ extension RCCameraViewController {
} }
} }
private func calculateScanArea() {
let actualWidth = view.frame.height / 16 * 9
let sideArea = (actualWidth - view.frame.width * 0.9) / 2
coder.imageDecoder.padding = Int(sideArea / actualWidth * 720)
}
private func configureVideoPreview(orientation: AVCaptureVideoOrientation = .portrait) { private func configureVideoPreview(orientation: AVCaptureVideoOrientation = .portrait) {
videoPreviewLayer?.removeFromSuperlayer() videoPreviewLayer?.removeFromSuperlayer()
videoPreviewLayer = AVCaptureVideoPreviewLayer(session: captureSession) videoPreviewLayer = AVCaptureVideoPreviewLayer(session: captureSession)
@@ -205,8 +197,10 @@ extension RCCameraViewController: AVCaptureVideoDataOutputSampleBufferDelegate {
coder.imageDecoder.bytesPerRow = bytesPerRow coder.imageDecoder.bytesPerRow = bytesPerRow
if let message = try? coder.decode(buffer: lumaCopy.assumingMemoryBound(to: UInt8.self)) { if let message = try? coder.decode(buffer: lumaCopy.assumingMemoryBound(to: UInt8.self)) {
captureSession.stopRunning() captureSession.stopRunning()
delegate?.cameraViewController(didFinishScanning: message) DispatchQueue.main.async {[weak self] in
dismiss(animated: true) self?.delegate?.cameraViewController(didFinishScanning: message)
self?.dismiss(animated: true)
}
} }
lumaCopy.deallocate() lumaCopy.deallocate()
CVPixelBufferUnlockBaseAddress(pixelBuffer, .readOnly) CVPixelBufferUnlockBaseAddress(pixelBuffer, .readOnly)
+13 -1
View File
@@ -25,6 +25,11 @@ import UIKit
public final class RCCoder { public final class RCCoder {
public let configuration: RCCoderConfiguration public let configuration: RCCoderConfiguration
public var scanningMode = ScanningMode.lightBackground {
didSet {
imageDecoder.pixelThreshold = scanningMode == .lightBackground ? RCConstants.lightBackgroundRange : RCConstants.darkBackgroundRange
}
}
internal lazy var imageDecoder = RCImageDecoder(configuration: self.configuration) internal lazy var imageDecoder = RCImageDecoder(configuration: self.configuration)
internal lazy var imageEncoder = RCImageEncoder(configuration: self.configuration) internal lazy var imageEncoder = RCImageEncoder(configuration: self.configuration)
internal lazy var bitCoder = RCBitCoder(configuration: self.configuration) internal lazy var bitCoder = RCBitCoder(configuration: self.configuration)
@@ -44,7 +49,6 @@ public extension RCCoder {
func decode(_ image: UIImage) throws -> String { func decode(_ image: UIImage) throws -> String {
guard image.size.width == image.size.height else { throw RCError.wrongImageSize } guard image.size.width == image.size.height else { throw RCError.wrongImageSize }
imageDecoder.size = image.cgImage!.height imageDecoder.size = image.cgImage!.height
imageDecoder.padding = 0
imageDecoder.bytesPerRow = image.cgImage!.height imageDecoder.bytesPerRow = image.cgImage!.height
let bits = try imageDecoder.decode(image) let bits = try imageDecoder.decode(image)
let message = try bitCoder.decode(bits) let message = try bitCoder.decode(bits)
@@ -56,6 +60,13 @@ public extension RCCoder {
} }
} }
public extension RCCoder {
enum ScanningMode {
case lightBackground
case darkBackground
}
}
extension RCCoder { extension RCCoder {
func decode(buffer: UnsafeMutablePointer<UInt8>) throws -> String { func decode(buffer: UnsafeMutablePointer<UInt8>) throws -> String {
@@ -64,3 +75,4 @@ extension RCCoder {
return message return message
} }
} }