MVVM & Observable

Module 1 of 11 0%

MVVM & Observable

MVVM vs Other iOS Architecture Patterns

MVVM (Model-View-ViewModel) is one of several architectural patterns used in iOS development. It has several advantages related to other patterns. Let’s take a look at it and compare with other patterns that can be used in iOS app development.

Core components of MVVM:

Key advantages of MVVM:

Let’s take a look at other architectural patterns commonly used in iOS development and compare them with MVVM.

MVC (Model-View-Controller)

Core components of MVC:

Comparison to MVVM:

MVP (Model-View-Presenter)

Core components of MVP:

Comparison to MVVM:

VIPER (View-Interactor-Presenter-Entity-Router)

Core components:

Comparison to MVVM:

VIP (View-Interactor-Presenter)

Core components:

Comparison to MVVM:

MVVM is a great choice when:

MVVM with Observable

SwiftUI and MVVM form a natural pairing, with SwiftUI’s declarative approach complementing MVVM’s separation of concerns. The @Observable macro, introduced in iOS 17, has further streamlined this relationship.

Let’s take a look at MVVM implementation in a typical SwiftUI app:

The @Observable macro provides a streamlined way to implement the ViewModel in MVVM:

@Observable class ProfileViewModel {
	var username = ""
	var bio = ""
	var isValid: Bool {
		!username.isEmpty && bio.count <= 160
	}

	func fetchProfile() {
		// Call repository/service to get data
		// Update properties when data arrives
	}

	func saveProfile() {
		// Business logic to validate and save
	}
}

What happens here:

This is how the MVVM components might be implemented in our example.

Model:

struct User {
	let id: UUID
	var name: String
	var email: String
	var bio: String
}

ViewModel:

@Observable class UserProfileViewModel {
	private var user: User?

	var name = ""
	var email = ""
	var bio = ""
	var isLoading = false
	var errorMessage = ""

	func loadUser(id: UUID) {
		isLoading = true

		// Simulated network call
		DispatchQueue.main.asyncAfter(deadline: .now() + 1) {
			self.user = User (id: id,
												name: "John Doe",
												email: "john@example.com",
												bio: "iOS Developer")
			self.name = self.user?.name ?? ""
			self.email = self.user?.email ?? ""
			self.bio = self.user?.bio ?? ""
			self.isLoading = false
		}
	}

	func saveUser() {
		guard let userId = user?.id else { return }

		// Update user model
		user = User (id: userId, name: name, email: email, bio: bio)

		// Send data to backend
	}
}

View

struct UserProfileView: View {
	@State privte var viewModel = UserProfileViewModel()
	let userId: UUID

	var body some View {
		Form {
			Section("Profile Details") {
				TextField("Name", text: $viewModel.name)
				TextField("Email", text: $viewModel.email)

				ZStack(alignment: .topLeading) {
					TextEditor(text: $viewModel.bio)
						.frame(height: 100)

					if viewModel.bio.isEmpty {
						Text("Bio")
							.foregroundColor(.gray.opacity(0.8))
							.padding(.top, 8)
							.padding(.leading, 4)
					}
				}
			}

			Button("Save Changes") {
				viewModel.saveUser()
			}
		}
		.overlay {
			if viewModel.isLoading {
				ProgressView()
					.background(Color.white.opacity(0.7))
			}
		}
		.onAppear {
			viewModel.loadUser(id: userId)
		}
	}
}

To get the same functionality without @Observable macro you have to use several property wrappers:

The @Observable macro simplifies things by eliminating the need for @Published markers, providing more fine-grained updates, requiring less boilerplate code.

Here’s the advantages of using MVVM pattern with @Observable:

MVVM with @Observable represents the most modern and efficient approach to architecting SwiftUI applications, combining the structural benefits of MVVM with SwiftUI’s reactive nature.

Homework