ARkit图像识别和图像的AR可视化

时间:2018-07-05 08:20:36

标签: ios swift xcode arkit image-recognition

在我工作的公司,我需要进行此申请。我必须识别绘画的图像,并在识别后在AR中对其进行可视化(在实践中,我会找到真实的图片,并在AR中绘画的上方),对具有相关图像特征的文本或可选点进行视觉化。目前,我拥有用于​​AR的代码,该代码可以识别出有问题的图像,并在其上方可视化计划。您能帮我在图片上方创建具有上述功能的视图吗?

import ARKit
import SceneKit
import UIKit

class ViewController: UIViewController, ARSCNViewDelegate {

    @IBOutlet var sceneView: ARSCNView!

    @IBOutlet weak var blurView: UIVisualEffectView!

    /// The view controller that displays the status and "restart experience" UI.
    lazy var statusViewController: StatusViewController = {
        return childViewControllers.lazy.compactMap({ $0 as? StatusViewController }).first!
    }()

    /// A serial queue for thread safety when modifying the SceneKit node graph.
    let updateQueue = DispatchQueue(label: Bundle.main.bundleIdentifier! +
        ".serialSceneKitQueue")

    /// Convenience accessor for the session owned by ARSCNView.
    var session: ARSession {
        return sceneView.session
    }

    // MARK: - View Controller Life Cycle

    override func viewDidLoad() {
        super.viewDidLoad()

        sceneView.delegate = self
        sceneView.session.delegate = self

        // Hook up status view controller callback(s).
        statusViewController.restartExperienceHandler = { [unowned self] in
            self.restartExperience()
        }
    }

    override func viewDidAppear(_ animated: Bool) {
        super.viewDidAppear(animated)

        // Prevent the screen from being dimmed to avoid interuppting the AR experience.
        UIApplication.shared.isIdleTimerDisabled = true

        // Start the AR experience
        resetTracking()
    }

    override func viewWillDisappear(_ animated: Bool) {
        super.viewWillDisappear(animated)

        session.pause()
    }

    // MARK: - Session management (Image detection setup)

    /// Prevents restarting the session while a restart is in progress.
    var isRestartAvailable = true

    /// Creates a new AR configuration to run on the `session`.
    /// - Tag: ARReferenceImage-Loading
    func resetTracking() {

        guard let referenceImages = ARReferenceImage.referenceImages(inGroupNamed: "AR Resources", bundle: nil) else {
            fatalError("Missing expected asset catalog resources.")
        }

        let configuration = ARWorldTrackingConfiguration()
        configuration.detectionImages = referenceImages
        session.run(configuration, options: [.resetTracking, .removeExistingAnchors])

        statusViewController.scheduleMessage("Look around to detect images", inSeconds: 7.5, messageType: .contentPlacement)
    }

    // MARK: - ARSCNViewDelegate (Image detection results)
    /// - Tag: ARImageAnchor-Visualizing
    func renderer(_ renderer: SCNSceneRenderer, didAdd node: SCNNode, for anchor: ARAnchor) {
        guard let imageAnchor = anchor as? ARImageAnchor else { return }
        let referenceImage = imageAnchor.referenceImage
        updateQueue.async {

            // Create a plane to visualize the initial position of the detected image.
            let plane = SCNPlane(width: referenceImage.physicalSize.width,
                                 height: referenceImage.physicalSize.height)
            let planeNode = SCNNode(geometry: plane)
            planeNode.opacity = 0.25

            /*
             `SCNPlane` is vertically oriented in its local coordinate space, but
             `ARImageAnchor` assumes the image is horizontal in its local space, so
             rotate the plane to match.
             */
            planeNode.eulerAngles.x = -.pi / 2

            /*
             Image anchors are not tracked after initial detection, so create an
             animation that limits the duration for which the plane visualization appears.
             */
            planeNode.runAction(self.imageHighlightAction)

            // Add the plane visualization to the scene.
            node.addChildNode(planeNode)
        }

        DispatchQueue.main.async {
            let imageName = referenceImage.name ?? ""
            self.statusViewController.cancelAllScheduledMessages()
            self.statusViewController.showMessage("Detected image “\(imageName)”")
        }
    }

    var imageHighlightAction: SCNAction {
        return .sequence([
            .wait(duration: 0.25),
            .fadeOpacity(to: 0.85, duration: 0.25),
            .fadeOpacity(to: 0.15, duration: 0.25),
            .fadeOpacity(to: 0.85, duration: 0.25),
            .fadeOut(duration: 0.5),
            .removeFromParentNode()
        ])
    }
}

1 个答案:

答案 0 :(得分:0)

解决问题的一种方法是创建一个SKScene并将其呈现为SCNMaterial

这是为您提供完整注释的示例,该示例使用以下ARSCNViewDelegate方法:

//--------------------------
// MARK: - ARSCNViewDelegate
//--------------------------

extension ViewController: ARSCNViewDelegate {

    func renderer(_ renderer: SCNSceneRenderer, didAdd node: SCNNode, for anchor: ARAnchor) {

        //1. Check We Have An ARImageAnchor And Have Detected Our Reference Image
        guard let imageAnchor = anchor as? ARImageAnchor else { return }

        let referenceImage = imageAnchor.referenceImage

        //2. Get The Physical Width & Height Of Our Reference Image
        let width = CGFloat(referenceImage.physicalSize.width)
        let height = CGFloat(referenceImage.physicalSize.height)

        //3. Create An SKScene Which We Will Display Above Our Target
        let overlayNode = SCNNode()

        let spriteKitScene = SKScene(size: CGSize(width: 600, height: 300))
        spriteKitScene.backgroundColor = .clear

        let imageName = SKLabelNode(text: "Line Friends")
        imageName.position = CGPoint(x: 600/2, y: 240)
        imageName.horizontalAlignmentMode = .center
        imageName.fontSize = 60
        imageName.fontName = "San Fransisco"
        spriteKitScene.addChild(imageName)

        let artistName = SKLabelNode(text: "By Line Coorporation")
        artistName.position = CGPoint(x: 600/2, y: 180)
        artistName.horizontalAlignmentMode = .center
        artistName.fontSize = 60
        artistName.fontName = "San Fransisco"
        spriteKitScene.addChild(artistName)

        let creationDate = SKLabelNode(text: "Created 2011")
        creationDate.position = CGPoint(x: 600/2, y: 120)
        creationDate.horizontalAlignmentMode = .center
        creationDate.fontSize = 60
        creationDate.fontName = "San Fransisco"
        spriteKitScene.addChild(creationDate)

        let planeHeight = height/2
        let overlayGeometry = SCNPlane(width: width, height: planeHeight)
        overlayNode.geometry = overlayGeometry

        //4. Add The SpriteKit Scene As The SCNPlane's Geometry
        overlayGeometry.firstMaterial?.diffuse.contents = spriteKitScene

        //5. Rotate The Material Contents So It Isn't Backwards
        overlayGeometry.firstMaterial?.diffuse.contentsTransform = SCNMatrix4Translate(SCNMatrix4MakeScale(1, -1, 1), 0, 1, 0)

        //6. Rotate The Node
        overlayNode.transform = SCNMatrix4MakeRotation(-Float.pi / 2, 1, 0, 0)

        //7. Place It About The Target
        let zPosition = height - (planeHeight/2)
        overlayNode.position = SCNVector3(0, 0, -zPosition)

        //8. Add It To The Scene
        node.addChildNode(overlayNode)
    }

}

产生如下内容:

enter image description here

很明显,如果您有多个图像目标,则可以创建一个func以动态创建“信息”叠加层...

希望它会为您指明正确的方向...并为公司令人发指的拼写而道歉! ^ _______ *