Xamarin nicely provides a loading view to be used while the app is doing background processing. I found this and was quite happy I didn't have to figure out how to do it myself. I then proceeded to translate it to F# like so:
type LoadingOverlay(frame: RectangleF) as this =
inherit UIView(frame)
let labelHeight = 22.0f
let labelWidth = frame.Width - 20.0f
let centerX = frame.Width / 2.0f
let centerY = frame.Height / 2.0f
let activitySpinner = new UIActivityIndicatorView(UIActivityIndicatorViewStyle.WhiteLarge)
let loadingLabel = new UILabel(new RectangleF (centerX - (labelWidth / 2.0f),
centerY + 20.0f ,
labelWidth ,
labelHeight))
do this.BackgroundColor <- UIColor.Black
this.Alpha <- 0.75f
this.AutoresizingMask <- UIViewAutoresizing.FlexibleDimensions
activitySpinner.Frame <- new RectangleF (centerX - (activitySpinner.Frame.Width / 2.0f) ,
centerY - activitySpinner.Frame.Height - 20.0f ,
activitySpinner.Frame.Width ,
activitySpinner.Frame.Height)
activitySpinner.AutoresizingMask <- UIViewAutoresizing.FlexibleMargins
this.AddSubview (activitySpinner)
activitySpinner.StartAnimating ()
loadingLabel.BackgroundColor <- UIColor.Clear
loadingLabel.TextColor <- UIColor.White
loadingLabel.Text <- "Loading Data..."
loadingLabel.TextAlignment <- UITextAlignment.Center
loadingLabel.AutoresizingMask <- UIViewAutoresizing.FlexibleMargins
this.AddSubview (loadingLabel)
member public this.Hide () =
UIView.Animate(0.5, (fun() -> this.Alpha <- 0.0f), fun() -> this.RemoveFromSuperview())
inherit UIView(frame)
let labelHeight = 22.0f
let labelWidth = frame.Width - 20.0f
let centerX = frame.Width / 2.0f
let centerY = frame.Height / 2.0f
let activitySpinner = new UIActivityIndicatorView(UIActivityIndicatorViewStyle.WhiteLarge)
let loadingLabel = new UILabel(new RectangleF (centerX - (labelWidth / 2.0f),
centerY + 20.0f ,
labelWidth ,
labelHeight))
do this.BackgroundColor <- UIColor.Black
this.Alpha <- 0.75f
this.AutoresizingMask <- UIViewAutoresizing.FlexibleDimensions
activitySpinner.Frame <- new RectangleF (centerX - (activitySpinner.Frame.Width / 2.0f) ,
centerY - activitySpinner.Frame.Height - 20.0f ,
activitySpinner.Frame.Width ,
activitySpinner.Frame.Height)
activitySpinner.AutoresizingMask <- UIViewAutoresizing.FlexibleMargins
this.AddSubview (activitySpinner)
activitySpinner.StartAnimating ()
loadingLabel.BackgroundColor <- UIColor.Clear
loadingLabel.TextColor <- UIColor.White
loadingLabel.Text <- "Loading Data..."
loadingLabel.TextAlignment <- UITextAlignment.Center
loadingLabel.AutoresizingMask <- UIViewAutoresizing.FlexibleMargins
this.AddSubview (loadingLabel)
member public this.Hide () =
UIView.Animate(0.5, (fun() -> this.Alpha <- 0.0f), fun() -> this.RemoveFromSuperview())
And then tried it out:
This looks very nice until you try to reuse it. It only works once… After looking at the Hide code, it doesn't take long to realize that the the Alpha is getting set to 0 and never changed thereafter (it's initially set to 0.75 in the constructor). This is simple to fix, just add this.Alpha <- 0.75f to the callback passed to UIView.Animate. Voila, now I have a LoadingOverlay which works multiple times.
Next, I rotated the iPad simulator and tried again.
Oops, still not quite right. When the view is added to the parent view (in the Xamarin example), the bounds of the parent are passed to the view (loadingOverlay = new LoadingOverlay (UIScreen.MainScreen.Bounds); ) and then the frame of the LoadingOverlay is set with to it, so the size is appropriately set initially. Also, as is needed, the AutoresizingMask of the view is set like this: this.AutoresizingMask <- UIViewAutoresizing.FlexibleDimensions so that shouldn't be a problem. What then is the issue?
The problem lies here: fun() -> this.RemoveFromSuperview() . After the view is dismissed, it is removed from the parent view's hierarchy, so when the parent view rotates, it doesn't rotate the LoadingOverlay because it doesn't own it any longer. This is easy to fix, simply use this.Hidden instead of removing the view from it's parent. That way when the parent resizes, the LoadingOverlay is still part of the hierarchy. The final class for the view looks like this:
type LoadingOverlay(frame: RectangleF) as this =
inherit UIView(frame)
let labelHeight = 22.0f
let labelWidth = frame.Width - 20.0f
let centerX = frame.Width / 2.0f
let centerY = frame.Height / 2.0f
let activitySpinner = new UIActivityIndicatorView(UIActivityIndicatorViewStyle.WhiteLarge)
let loadingLabel = new UILabel(new RectangleF (centerX - (labelWidth / 2.0f),
centerY + 20.0f ,
labelWidth ,
labelHeight))
do this.Hidden <- true
this.BackgroundColor <- UIColor.Black
this.AutoresizingMask <- UIViewAutoresizing.FlexibleDimensions
activitySpinner.Frame <- new RectangleF (centerX - (activitySpinner.Frame.Width / 2.0f) ,
centerY - activitySpinner.Frame.Height - 20.0f ,
activitySpinner.Frame.Width ,
activitySpinner.Frame.Height)
activitySpinner.AutoresizingMask <- UIViewAutoresizing.FlexibleMargins
this.AddSubview (activitySpinner)
activitySpinner.StartAnimating ()
loadingLabel.BackgroundColor <- UIColor.Clear
loadingLabel.TextColor <- UIColor.White
loadingLabel.Text <- "Loading Data..."
loadingLabel.TextAlignment <- UITextAlignment.Center
loadingLabel.AutoresizingMask <- UIViewAutoresizing.FlexibleMargins
this.AddSubview (loadingLabel)
member public this.Hide () =
UIView.Animate(0.5, (fun() -> this.Alpha <- 0.0f), fun() -> this.Hidden <- true)
member public this.Show () =
this.Hidden <- false
UIView.Animate(0.5, (fun() -> this.Alpha <- 0.75f))
To use it, create the LoadingOverlay in the contractor of the parent, keep a reference to it, add it to the parent view in ViewDidAppear and then use Show/Hide to show and hide the view.