How to use Label in SwiftUI custom view

⋅ 6 min read ⋅ SwiftUI Label

Table of Contents

Label is a SwiftUI view that provides a simple way to describe any other view. It contains text and an icon. A view that uses a label can choose whether to use it based on available space and environment.

We have learn all of this in SwiftUI Label: A standard way to label user interface items post. At the end of the post, I told you that we should consider using a label when building a custom view. And that is what we will do in this article, learn how to embrace a label into a custom view.

Use a label in a custom view

The label contains two pieces of information, text and an icon. But as we know, by observing built-in views, each view can control which piece of information to use.

Here is an example of the same label that render show different piece of information based on its container view.

struct ContentView: View {
// 1
let favoritesLabel = Label("Favorites", systemImage: "heart")

@State private var isPresented = false

var body: some View {
NavigationView {
VStack {
// 2
Button {
isPresented = true
} label: {
favoritesLabel
}
}.toolbar {
// 3
ToolbarItem(placement: .primaryAction) {
Button {

} label: {
favoritesLabel
}
}
}
.confirmationDialog(
"Test Confirmation",
isPresented: $isPresented,
titleVisibility: .visible
) {
// 4
Button {

} label: {
favoritesLabel
}

}
}
}
}

1 We create a shared label that will pass as an argument for each view.
2 A button uses an icon and text from label as its content.
3 A tool bar item use only icon as its content.
4 A button inside a confirmation dialog context use only text as its content.

View can pick a label information that fits its need.
View can pick a label information that fits its need.

Let's see how to control this in our custom view.

Default label style

To control which information to use, we used what we learn from our previous article, labelStyle(_:).

We created IconOnlyView that uses only an icon if a Label is passed as an argument.

// 1
struct IconOnlyView<Label>: View where Label: View {
let label: Label

// 2
init(@ViewBuilder _ label: () -> Label) {
self.label = label()
}

var body: some View {
HStack {
// 3
label
.labelStyle(.iconOnly)
Text("(Icon only)")
}
.border(.pink) // 4
}
}

1 We declare a view that accepts any view as a label. I use the same declaration style as a built-in SwiftUI view.
2 An initializer that accepts a view builder used to populate a label for our view.
3 This is where we enforce label style that matches the view, .iconOnly in this case.
4 Our custom view will put a label inside an HStack with a custom text and a pink border.

Users can provide any view as a label to our IconOnlyView. Only a Label view will be effect by .labelStyle.

struct ContentView: View {
let favoritesLabel = Label("Favorites", systemImage: "heart")

var body: some View {
VStack(spacing: 20) {
// 1
IconOnlyView {
Image(systemName: "heart")
}
IconOnlyView {
Text("Favorites")
}
IconOnlyView {
VStack {
Image(systemName: "heart")
Text("Favorites")
}
}
// 2
IconOnlyView {
favoritesLabel
}
}
}
}

1 Non-Label view won't get affect by .labelStyle and show as-is.
2 Only a Label will get an effect. Only an icon will be used as a label.

Users can provide any view as a label to our IconOnlyView. Only a Label view will be effect by label style.
Users can provide any view as a label to our IconOnlyView. Only a Label view will be effect by label style.

You can choose any style you want that matches your view. Here is an example of TitleOnlyView, which shows only a text, and NoLabelView, which shows nothing.

struct TitleOnlyView<Label>: View where Label: View {
let label: Label

init(@ViewBuilder _ label: () -> Label) {
self.label = label()
}

var body: some View {
HStack {
label
.labelStyle(.titleOnly)
Text("(Title only)")
}
.border(.pink)
}
}

struct NoLabelView<Label>: View where Label: View {
let label: Label

init(@ViewBuilder _ label: () -> Label) {
self.label = label()
}

var body: some View {
HStack {
// Don't use any label
Text("(No label)")
}
.border(.pink)
}
}

struct ContentView: View {
let favoritesLabel = Label("Favorites", systemImage: "heart")

var body: some View {
VStack(spacing: 20) {
TitleOnlyView {
favoritesLabel
}
NoLabelView {
favoritesLabel
}
}
}
}
Example views that use only a text label and no label at all.
Example views that use only a text label and no label at all.

Custom label style

The default label style will show an icon and text side-by-side, which might not suit your custom view. If you need to customize a layout of a label or make it adapt to environmental change, you would need to create a custom label style.

I will create an AdaptiveView view with a companion label style, AdaptiveLabelStyle. The view presents an icon and text label vertically in portrait and shows an icon and text horizontally in a landscape where vertical space is limited (compact vertical size class).

struct AdaptiveView<Label>: View where Label: View {
let label: Label

init(@ViewBuilder _ label: () -> Label) {
self.label = label()
}

var body: some View {
label
.labelStyle(AdaptiveLabelStyle())
}
}

struct AdaptiveLabelStyle: LabelStyle {
@Environment(\.verticalSizeClass) var verticalSizeClass: UserInterfaceSizeClass?

func makeBody(configuration: Configuration) -> some View {
if verticalSizeClass == .compact {
// 1
HStack {
configuration.icon
configuration.title
}
} else {
// 2
VStack {
configuration.icon
configuration.title
}
}
}
}

struct ContentView: View {
let favoritesLabel = Label("Favorites", systemImage: "heart")

var body: some View {
VStack(spacing: 20) {
AdaptiveView {
favoritesLabel
}
}
}
}

1 When vertical space is limited (landscape orientation), we show the icon and text horizontally.
2 In portrait mode, we show them vertically.

Adaptive label layout.
Adaptive label layout.

You can easily support sarunw.com by checking out this sponsor.

Sponsor sarunw.com and reach thousands of iOS developers.

Accessibility

We have learned how to show/hide an icon and text label, customize the layout, and adapt to environmental changes, but one problem remains with our IconOnlyView.

In IconOnlyView, I add an extra text, "Icon only", alongside a label. This extra text is used as a visual clue for demonstration only, and it doesn't mean for VoiceOver to read it out.

Using VoiceOver right now will read as two separate elements, "Favorites" and "Icon only".

The label read as
The label read as "Favorites".
The hard-coded text also treated as another accessibility element.
The hard-coded text also treated as another accessibility element.

This might not be behavior that users expect when providing a Label to a view. We should use label text as a description.

Custom accessibility data

There is no silver bullet for this custom layout. Each layout required a unique treatment, but I will show you how to fix mine.

Here is our implementation once again.

struct IconOnlyView<Label>: View where Label: View {
let label: Label

init(@ViewBuilder _ label: () -> Label) {
self.label = label()
}

var body: some View {
HStack {
label
.labelStyle(.iconOnly)
Text("(Icon only)")
}
.border(.pink)
}
}

The fix is easy in this case. I need to remove Text("(Icon only)") from system accessibility features. I can do this by applying accessibility(hidden:) to it.

struct IconOnlyView<Label>: View where Label: View {
let label: Label

init(@ViewBuilder _ label: () -> Label) {
self.label = label()
}

var body: some View {
HStack {
label
.labelStyle(.iconOnly)
Text("(Icon only)")
.accessibilityHidden(true)
}
.border(.pink)
}
}

VoiceOver will read our view as "Favourites" and ignore the text "Icon only" with this simple change.

Depending on your layout, you may or may not need to customize accessibility data. I just want to point out that when you create a custom label style, you opt out from the default behavior and make sure it works the way you want.

You can easily support sarunw.com by checking out this sponsor.

Sponsor sarunw.com and reach thousands of iOS developers.

Conclusion

Here is a recap on how to embrace a label in your custom view.

  • Create an initializer that accepts a label. To match built-in views, you should allow any view to use as a label. Only a Label view would get special treatment (Get an effect from .labelStyle).
  • Make sure accessibility data work the way you want if you implement a custom label style.

Read more article about SwiftUI, Label, or see all available topic

Enjoy the read?

If you enjoy this article, you can subscribe to the weekly newsletter.
Every Friday, you'll get a quick recap of all articles and tips posted on this site. No strings attached. Unsubscribe anytime.

Feel free to follow me on Twitter and ask your questions related to this post. Thanks for reading and see you next time.

If you enjoy my writing, please check out my Patreon https://www.patreon.com/sarunw and become my supporter. Sharing the article is also greatly appreciated.

Become a patron Buy me a coffee Tweet Share
Previous
How to draw custom paths and shapes in SwiftUI

Learn how to make hexagon-shaped profile pictures in SwiftUI.

Next
How to reset push notification permission on iOS

Learn how to make notification permission dialog popup again like the first time you run the app.

← Home