Callouts are small, transient popups that display information or user interface controls. As part of my Swift node based calculator experiment, I wanted to add two new features, a numeric dial and a delete node button, which are ideal candidates to be contained in a callout container.

I've used a different approach for both new features and this post describes how I've implemented them.

Popovers



First off, the numeric dial is launched from a UIBarButtonItem in my Toolbar class

The numeric dial is my own component that I blogged about here. In this implementation, I've updated its layers to extend CAShapeLayer and draw its value line immediately through a drawValueCurve() method in the NumericDialTrack class.

The dial itself knows nothing of my presentation model; in keeping with the strategy I've used for years, my low-level UI components are framework and domain agnostic. My NumericDialViewController, which houses the control, does know about the presentation model and has a few observers on it:


        NodesPM.addObserver(self, selector: "nodeChangeHandler", notificationType: NodeNotificationTypes.NodeSelected)
        NodesPM.addObserver(self, selector: "nodeChangeHandler", notificationType: NodeNotificationTypes.NodeUpdated)

nodeChangeHandler() does little more than normalise the current node's value and update the control to reflect the relevant node's value:


    func nodeChangeHandler()
    {
        if let selectedNode = NodesPM.selectedNode
        {
            let value = selectedNode.value
            
            ignoreDialChangeEvents = true
                
            numericDial.currentValue = value / 100
                
            ignoreDialChangeEvents = false
        }
    }

When the user changes the dial position, the controller acts on a .ValueChanged control event on the dial and notifies the presentation model:


    func dialChangeHandler(numericDial: NumericDial)
    {
        let dialValue = Double(Int(numericDial.currentValue * 100))
        
        if !ignoreDialChangeEvents
        {
            NodesPM.changeSelectedNodeValue(dialValue)
        }
    }

Up in the Toolbar class, I create an instance of my NumericDialController and an instance of UIPopoverController with the numeric dial controller set as its contentViewController:


    let numericDialViewController: NumericDialViewController
    let popoverController: UIPopoverController

    [...]

    numericDialViewController = NumericDialViewController()
    popoverControllerUIPopoverController(contentViewController: numericDialViewController)

I want to know when the dial callout has been closed and to do that my toolbar implements the UIPopoverControllerDelegate and sets itself to the popover controller's delegate:


    popoverController.delegate = self

The user launches the numeric dial callout by clicking a UIBarButtonItem:


    let numericButtonShowDial = UIBarButtonItem(title: "Dial", style: UIBarButtonItemStyle.Plain, target: self, action: "showDial:")

The showDial() method actually displays the callout. It does this by checking we have a rootViewControler and then simply invoking presentPopoverFromBarButton():


    popoverController.presentPopoverFromBarButtonItem(value, permittedArrowDirections: UIPopoverArrowDirection.Any, animated: true)

One thing I found was that although the callout is meant to be modal, the little calculator buttons were still clickable. Changing the toolbar's userInteractionEnabled to false didn't seem to help, so I created a little function that manually enables or disables the individual button bar buttons:


    func enableItems(enable: Bool)
    {
        if let barButtonItems = items
        {
            for barButtonItem:AnyObject in barButtonItems
            {
                (barButtonItem as UIBarButtonItem).enabled = enable;
            }
        }
    }

This is invoked when the callout is opened and, because toolbar is the popover controller's delegate, invoked again when the callout is closed through the popoverControllerDidDismissPopover() method:


    func popoverControllerDidDismissPopover(popoverController: UIPopoverController)
    {
        enableItems(true)
    }

Action Sheets



Next up is an ActionSheet which is launched from my MenuButton class.

I could have used the same technique for the menu button, but UIKit offers a simpler alternative for creating action sheets, UIAlertController

The UIController contains a set of UIAlertAction instances. In my case, I have one for deleting the selected node and two others to toggle nodes between numeric and operator types. 

My first steps are to instantiate the UIAlertController and add the actions:


    var alertController = UIAlertController(title: nil, message: nil, preferredStyle: UIAlertControllerStyle.ActionSheet)

    [...]

    makeOperatorAction = UIAlertAction(title: NodeTypes.Operator.toRaw(), style: UIAlertActionStyle.Default, handler: changeNodeType)
    makeNumericAction = UIAlertAction(title: NodeTypes.Number.toRaw(), style: UIAlertActionStyle.Default, handler: changeNodeType)
    deleteAlertAction = UIAlertAction(title: "Delete Selected Node", style: UIAlertActionStyle.Default, handler: deleteSelectedNode)

    [...]

    alertController.addAction(deleteAlertAction)
    alertController.addAction(makeNumericAction)
    alertController.addAction(makeOperatorAction)

Now that the actions have been created, launching them is simply a matter of invoking presentViewController() on the root view controller in the menu button's overridden touchesBegan() method:


    override func touchesBegan(touches: NSSet, withEvent event: UIEvent)
    {
        if let viewController = UIApplication.sharedApplication().keyWindow.rootViewController
        {
            if let popoverPresentationController = alertController.popoverPresentationController
            {
                popoverPresentationController.sourceRect = frame
                popoverPresentationController.sourceView = viewController.view
                
                viewController.presentViewController(alertController, animated: true, completion: nil)
            }
        }
    }

A little note on NSTimer and userInfo

When I first plugged the numeric dial into this application, performance was pretty grim. Because of the frequent updates to the screen, the dial was very juddery. One of the approaches I took to help with this was to use an NSTimer inside the recursive nodeUpdate() method inside my presentation model.

Although I didn't end up using this approach, I did have to scratch my head for a few minutes to figure out how to pass data from a timer to its selector. I needed to pass a NodeVO and I did this by creating a NSMutableDictionary containing my node and setting that as the timer's userInfo property:


var dictionary = NSMutableDictionary()
    dictionary.setValue(candidateNode, forKeyPath: "node")
    

    var timer = NSTimer(timeInterval: timeInterval, target: self, selector: "timerComplete:", userInfo: dictionary, repeats: false)

Then, timerComplete() uses valeForKey() to extract the node from its argument:


    func timerComplete(value: AnyObject)
    {
        let srcTimer: NSTimer = value as NSTimer
        
        let node: NodeVO = srcTimer.userInfo?.valueForKey("node") as NodeVO
        
        NodesPM.nodeUpdate(node)
    }

All the updated source code is available in my GitHub repository. Please note: this code has been built under Xcode 6.0 and may not work under 6.1.

0

Add a comment

Primordial Particle System in SideFX Houdini
5
"Sparse Vermiform" Rendering of Fluids with Pressure Based Color
3
Particle Advection by Gray Scott Reaction Diffusion Revisited
Particle Advection by Reaction Diffusion in SideFX Houdini
1
Revisiting Mitosis in SideFX Houdini
4
Faux Grain / Fluid Interaction in Houdini
1
Parametric Fibonacci Spheres in Houdini
Houdini Grain Solver with Custom VEX Forces
Animating Rayleigh-Taylor Instability in SideFX Houdini
1
Faking Toroidal Eddies in Side FX Houdini
Mixing Fluids in Houdini
Animating Gravitational Tides with Houdini FLIP Fluids
1
Houdini FLIP Fluid & Radial Gravity
1
Reaction Diffusion in SideFX Houdini
2
Animating Planet Engulfment in Houdini
1
Simulating Accretion with Houdini's Grains Solver
Chaotic Magnetic Pendulum with Custom Radial Forces in Houdini
More Chaos in Houdini: Simulating a Double Pendulum
Creating a Geometric Structure from Mitosis
2
Simulating Mitosis in Houdini
4
Stripy Viscous Fluid Impacts in Houdini
1
Using Houdini VOPs to Deform Geometry
1
Creating a Swarm Chemistry Digital Asset in Houdini
Swarm Chemistry in SideFX Houdini
1
Randomly Transforming Scattered Cones in Houdini
1
Metaball Morphogenesis in Houdini
Melting Geometry in Houdini
2
Simulating Belousov-Zhabotinsky Reaction in Houdini
Chaos in Houdini! Modeling Strange Attractors with Particles
Experimenting with Impacts in SideFX Houdini
Chaos in Swift! Visualising the Lorenz Attractor with Metal
Swift 3.0 for Core Image Developers
4
A Core Image Transverse Chromatic Aberration Filter in Swift
1
Random Numbers in Core Image Kernel Language
Core Image for Swift Version 1.3
Core Image for Swift Version 1.3
Nodality for AudioKit: Node Based Synth for iPad
Histogram Equalisation with Metal Performance Shaders
A Histogram Display Component in Swift for iOS
Simulating Depth-of-Field with Variable Bokeh in Core Image
Simulating Bokeh with Metal Performance Shaders
1
Creating a Lens Flare Filter in Core Image
Loading, Filtering & Saving Videos in Swift
3
vImage Histogram Functions Part II: Specification
Histogram Functions in Accelerate vImage
1
New Core Image Procedural Noise Generators for Filterpedia
Recreating Kai's Power Tools Goo in Swift
11
Creating Procedural Normal Maps for SceneKit
New Custom Core Image Filters
1
A Look at Perspective Transform & Correction with Core Image
8
Creating a Custom Variable Blur Filter in Core Image
3
Core Image for Swift v1.2 Released!
3
Creating a Selective HSL Adjustment Filter in Core Image
Creating a Bulging Eyes Purikura Effect with Core Image
1
Sweetcorn: A Node Based Core Image Kernel Builder
1
Core Image for Swift: Advanced Image Processing for iOS
2
Properly Typed Selectors in Xcode 7.3 beta 4
7
Core Image for Swift Available for Pre-Order!
10
Playing with Interpolation Functions in Swift
1
Metal Kernel Functions as Core Image Filter Engines
New Custom Core Image Filters Added to Filterpedia
5
Computational Fluid Dynamics in CoreImage with CIKernel
3
Creating a Slide Show App with Core Image Transitions
Filterpedia: Core Image Filter Explorer
5
CartoonEyes: Compositing Cartoon Eyes over Face from Front Camera in Swift
3
Apple Pencil Controlled Christmas Tree Bowling with SceneKit
1
BristlePaint: Embossed Painting with Individual Bristles using SpriteKit Normal Mapping
1
Scribe: A Handwriting Recognition Component for iOS
2
MercurialPaint: Globular Embossed Painting with Metal & Core Image
MercurialText: Embossed Type using SceneKit and CIShadedMaterial
1
FurrySketch: Hirsute Drawing with an Apple Pencil
PencilSynth - An Apple Pencil Controlled Synthesiser
PencilController - Using Apple Pencil as a 3D Controller for Image Editing
5
A Look at Agents, Goals & Behaviours in GameplayKit
PencilScale - Using an Apple Pencil with an iPad Pro as an Electronic Scale
2
A First Look at Metal Performance on the iPad Pro
Smooth Drawing for iOS in Swift with Hermite Spline Interpolation
3
Introducing Image Processing in Metal
Swift Hierarchical Selector Component based on UIPickerView & UICollectionView
The Plum-O-Meter: Weighing Plums Using 3D Touch in Swift
27
Book Review: Swift Documentation Markup by Erica Sadun
1
3D Touch in Swift: A Retrospective
3D ReTouch: An Experimental Retouching App Using 3D Touch
1
ForceZoom: Popup Image Detail View using 3D Touch Peek
Globular: Colourful Metaballs Controlled by 3D Touch
2
ForceSketch: A 3D Touch Drawing App using CIImageAccumulator
DeepPressGestureRecognizer - A 3D Touch Custom Gesture Recogniser in Swift
2
Rotatable: A Swift Protocol Extension to Rotate any UIView
ChromaTouch: a 3D Touch Colour Picker in Swift
3
3D Touch in Swift: Implementing Peek & Pop
1
A First Look at Metal Performance on the iPhone 6s
Applying Gaussian Blur to UIViews with Swift Protocol Extensions
7
Advanced Touch Handling in iOS9: Coalescing and Prediction
3
A Swift Node Based User Interface Component for iOS
2
Using an iPhone as a 3D Mouse with Multipeer Connectivity in Swift
6
A Swift Nixie Tube Display Component
CoreMotion Controlled 3D Sketching on an iPhone with Swift
2
iOS Live Camera Controlled Particles in Swift & Metal
6
Metal Performance Shaders & Fallback Copy Allocators
Event Dispatching in Swift with Protocol Extensions
1
Hybrid Marking Menu / Radial Slider Swift Component for iOS
About Me
About Me
Labels
Labels
Blog Archive
Loading