initial commit
This commit is contained in:
parent
356708f6d7
commit
09bd1740de
@ -29,9 +29,22 @@
|
|||||||
EAC304C22D76693000D9006D /* EmployeeDirectoryUITests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = EmployeeDirectoryUITests.xctest; sourceTree = BUILT_PRODUCTS_DIR; };
|
EAC304C22D76693000D9006D /* EmployeeDirectoryUITests.xctest */ = {isa = PBXFileReference; explicitFileType = wrapper.cfbundle; includeInIndex = 0; path = EmployeeDirectoryUITests.xctest; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||||
/* End PBXFileReference section */
|
/* End PBXFileReference section */
|
||||||
|
|
||||||
|
/* Begin PBXFileSystemSynchronizedBuildFileExceptionSet section */
|
||||||
|
EA57F4B72D778F140015D7BC /* Exceptions for "EmployeeDirectory" folder in "EmployeeDirectory" target */ = {
|
||||||
|
isa = PBXFileSystemSynchronizedBuildFileExceptionSet;
|
||||||
|
membershipExceptions = (
|
||||||
|
Info.plist,
|
||||||
|
);
|
||||||
|
target = EAC304A72D76692F00D9006D /* EmployeeDirectory */;
|
||||||
|
};
|
||||||
|
/* End PBXFileSystemSynchronizedBuildFileExceptionSet section */
|
||||||
|
|
||||||
/* Begin PBXFileSystemSynchronizedRootGroup section */
|
/* Begin PBXFileSystemSynchronizedRootGroup section */
|
||||||
EAC304AA2D76692F00D9006D /* EmployeeDirectory */ = {
|
EAC304AA2D76692F00D9006D /* EmployeeDirectory */ = {
|
||||||
isa = PBXFileSystemSynchronizedRootGroup;
|
isa = PBXFileSystemSynchronizedRootGroup;
|
||||||
|
exceptions = (
|
||||||
|
EA57F4B72D778F140015D7BC /* Exceptions for "EmployeeDirectory" folder in "EmployeeDirectory" target */,
|
||||||
|
);
|
||||||
path = EmployeeDirectory;
|
path = EmployeeDirectory;
|
||||||
sourceTree = "<group>";
|
sourceTree = "<group>";
|
||||||
};
|
};
|
||||||
@ -399,6 +412,7 @@
|
|||||||
DEVELOPMENT_TEAM = 6R7KLBPBLZ;
|
DEVELOPMENT_TEAM = 6R7KLBPBLZ;
|
||||||
ENABLE_PREVIEWS = YES;
|
ENABLE_PREVIEWS = YES;
|
||||||
GENERATE_INFOPLIST_FILE = YES;
|
GENERATE_INFOPLIST_FILE = YES;
|
||||||
|
INFOPLIST_FILE = EmployeeDirectory/Info.plist;
|
||||||
INFOPLIST_KEY_UIApplicationSceneManifest_Generation = YES;
|
INFOPLIST_KEY_UIApplicationSceneManifest_Generation = YES;
|
||||||
INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents = YES;
|
INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents = YES;
|
||||||
INFOPLIST_KEY_UILaunchScreen_Generation = YES;
|
INFOPLIST_KEY_UILaunchScreen_Generation = YES;
|
||||||
@ -428,6 +442,7 @@
|
|||||||
DEVELOPMENT_TEAM = 6R7KLBPBLZ;
|
DEVELOPMENT_TEAM = 6R7KLBPBLZ;
|
||||||
ENABLE_PREVIEWS = YES;
|
ENABLE_PREVIEWS = YES;
|
||||||
GENERATE_INFOPLIST_FILE = YES;
|
GENERATE_INFOPLIST_FILE = YES;
|
||||||
|
INFOPLIST_FILE = EmployeeDirectory/Info.plist;
|
||||||
INFOPLIST_KEY_UIApplicationSceneManifest_Generation = YES;
|
INFOPLIST_KEY_UIApplicationSceneManifest_Generation = YES;
|
||||||
INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents = YES;
|
INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents = YES;
|
||||||
INFOPLIST_KEY_UILaunchScreen_Generation = YES;
|
INFOPLIST_KEY_UILaunchScreen_Generation = YES;
|
||||||
|
|||||||
@ -0,0 +1,6 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<Bucket
|
||||||
|
uuid = "BCD20A5B-B7B0-4775-A851-C44BE1718BD6"
|
||||||
|
type = "1"
|
||||||
|
version = "2.0">
|
||||||
|
</Bucket>
|
||||||
@ -1,24 +0,0 @@
|
|||||||
//
|
|
||||||
// ContentView.swift
|
|
||||||
// EmployeeDirectory
|
|
||||||
//
|
|
||||||
// Created by Matt Bruce on 3/3/25.
|
|
||||||
//
|
|
||||||
|
|
||||||
import SwiftUI
|
|
||||||
|
|
||||||
struct ContentView: View {
|
|
||||||
var body: some View {
|
|
||||||
VStack {
|
|
||||||
Image(systemName: "globe")
|
|
||||||
.imageScale(.large)
|
|
||||||
.foregroundStyle(.tint)
|
|
||||||
Text("Hello, world!")
|
|
||||||
}
|
|
||||||
.padding()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#Preview {
|
|
||||||
ContentView()
|
|
||||||
}
|
|
||||||
@ -11,7 +11,7 @@ import SwiftUI
|
|||||||
struct EmployeeDirectoryApp: App {
|
struct EmployeeDirectoryApp: App {
|
||||||
var body: some Scene {
|
var body: some Scene {
|
||||||
WindowGroup {
|
WindowGroup {
|
||||||
ContentView()
|
EmployeeListView()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
26
EmployeeDirectory/Extensions/String.swift
Normal file
26
EmployeeDirectory/Extensions/String.swift
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
//
|
||||||
|
// String.swift
|
||||||
|
// EmployeeDirectory
|
||||||
|
//
|
||||||
|
// Created by Matt Bruce on 3/4/25.
|
||||||
|
//
|
||||||
|
import Foundation
|
||||||
|
|
||||||
|
extension String {
|
||||||
|
/// Formats a string into a US phone number format (XXX-XXX-XXXX).
|
||||||
|
/// Non-numeric characters are removed, and formatting is applied based on the length of the string.
|
||||||
|
/// - Returns: A formatted phone number as a string.
|
||||||
|
internal func formatUSNumber() -> String {
|
||||||
|
let mask = "XXX-XXX-XXXX"
|
||||||
|
let digits = filter { $0.isNumber }
|
||||||
|
var index = digits.startIndex
|
||||||
|
return mask.reduce(into: "") { result, char in
|
||||||
|
if char == "X", index < digits.endIndex {
|
||||||
|
result.append(digits[index])
|
||||||
|
index = digits.index(after: index)
|
||||||
|
} else {
|
||||||
|
result.append(char)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
19
EmployeeDirectory/Info.plist
Normal file
19
EmployeeDirectory/Info.plist
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>NSAppTransportSecurity</key>
|
||||||
|
<dict>
|
||||||
|
<key>NSExceptionDomains</key>
|
||||||
|
<dict>
|
||||||
|
<key>dummyimage.com</key>
|
||||||
|
<dict>
|
||||||
|
<key>NSIncludesSubdomains</key>
|
||||||
|
<true/>
|
||||||
|
<key>NSExceptionAllowsInsecureHTTPLoads</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
||||||
50
EmployeeDirectory/Models/Employee.swift
Normal file
50
EmployeeDirectory/Models/Employee.swift
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
//
|
||||||
|
// Employee.swift
|
||||||
|
// EmployeeDirectory
|
||||||
|
//
|
||||||
|
// Created by Matt Bruce on 3/3/25.
|
||||||
|
//
|
||||||
|
import Foundation
|
||||||
|
|
||||||
|
/// Employee Object
|
||||||
|
/// JSON Object defintion
|
||||||
|
/// - https://square.github.io/microsite/mobile-interview-project/
|
||||||
|
public struct Employee: Identifiable, Hashable, Codable {
|
||||||
|
public var id: UUID { uuid }
|
||||||
|
|
||||||
|
/// The unique identifier for the employee. Represented as a UUID.
|
||||||
|
public let uuid: UUID
|
||||||
|
|
||||||
|
/// The first name of the employee.
|
||||||
|
public let firstName: String
|
||||||
|
|
||||||
|
/// The last name of the employee.
|
||||||
|
public let lastName: String
|
||||||
|
|
||||||
|
/// The phone number of the employee, sent as an unformatted string (eg, 5556661234).
|
||||||
|
public let phoneNumber: String?
|
||||||
|
|
||||||
|
/// The email address of the employee.
|
||||||
|
public let emailAddress: String
|
||||||
|
|
||||||
|
/// A short, tweet-length (~300 chars) string that the employee provided to describe themselves.
|
||||||
|
public let biography: String?
|
||||||
|
|
||||||
|
/// The URL of the employee’s small photo. Useful for list view.
|
||||||
|
public let photoURLSmall: String?
|
||||||
|
|
||||||
|
/// The URL of the employee’s full-size photo.
|
||||||
|
public let photoURLLarge: String?
|
||||||
|
|
||||||
|
private enum CodingKeys: String, CodingKey {
|
||||||
|
case uuid
|
||||||
|
case firstName = "first_name"
|
||||||
|
case lastName = "last_name"
|
||||||
|
case phoneNumber = "phone_number"
|
||||||
|
case emailAddress = "email_address"
|
||||||
|
case biography
|
||||||
|
case photoURLSmall = "photo_url_small"
|
||||||
|
case photoURLLarge = "photo_url_large"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
14
EmployeeDirectory/Models/Employees.swift
Normal file
14
EmployeeDirectory/Models/Employees.swift
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
//
|
||||||
|
// Employees.swift
|
||||||
|
// EmployeeDirectory
|
||||||
|
//
|
||||||
|
// Created by Matt Bruce on 3/3/25.
|
||||||
|
//
|
||||||
|
import Foundation
|
||||||
|
|
||||||
|
/// Wrapper JSON Class for the Employees
|
||||||
|
public struct Employees: Codable {
|
||||||
|
/// Array of Employees
|
||||||
|
public var result: [Employee]
|
||||||
|
public var hasNextPage: Bool = false
|
||||||
|
}
|
||||||
10
EmployeeDirectory/Protocols/EmployeeServiceProtocol.swift
Normal file
10
EmployeeDirectory/Protocols/EmployeeServiceProtocol.swift
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
//
|
||||||
|
// EmployeeServiceProtocol.swift
|
||||||
|
// EmployeeDirectory
|
||||||
|
//
|
||||||
|
// Created by Matt Bruce on 3/3/25.
|
||||||
|
//
|
||||||
|
|
||||||
|
public protocol EmployeeServiceProtocol {
|
||||||
|
func getUsers(page: Int?) async throws -> Employees
|
||||||
|
}
|
||||||
76
EmployeeDirectory/Services/EmployeeService.swift
Normal file
76
EmployeeDirectory/Services/EmployeeService.swift
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
//
|
||||||
|
// EmployeeService.swift
|
||||||
|
// EmployeeDirectory
|
||||||
|
//
|
||||||
|
// Created by Matt Bruce on 3/3/25.
|
||||||
|
//
|
||||||
|
import Foundation
|
||||||
|
|
||||||
|
public class EmployeeService: EmployeeServiceProtocol {
|
||||||
|
// MARK: - Properties
|
||||||
|
public static let shared = EmployeeService() // Default shared instance
|
||||||
|
|
||||||
|
// MARK: - Initializer
|
||||||
|
|
||||||
|
public init() {}
|
||||||
|
|
||||||
|
// MARK: - Public Methods
|
||||||
|
|
||||||
|
/// This will get a list of all employees
|
||||||
|
/// - Returns: An Employees struct
|
||||||
|
public func getUsers(page: Int? = nil) async throws -> Employees {
|
||||||
|
var endpoint = "https://my.api.mockaroo.com/users.json?key=f298b840"
|
||||||
|
if let page {
|
||||||
|
endpoint += "&page=\(page)"
|
||||||
|
}
|
||||||
|
|
||||||
|
//ensure a valid URL
|
||||||
|
guard let url = URL(string: endpoint) else {
|
||||||
|
throw URLError(.badURL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Perform network request
|
||||||
|
let (data, response) = try await URLSession.shared.data(from: url)
|
||||||
|
|
||||||
|
// Validate HTTP response
|
||||||
|
guard let httpResponse = response as? HTTPURLResponse,
|
||||||
|
200..<300 ~= httpResponse.statusCode else {
|
||||||
|
throw URLError(.badServerResponse)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Decode the response into the specified type
|
||||||
|
return try JSONDecoder().decode(Employees.self, from: data)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Mock Service Implementation for testing & previews.
|
||||||
|
struct MockEmployeeService: EmployeeServiceProtocol {
|
||||||
|
static let sample = createUser(page: 1, index: 1)
|
||||||
|
|
||||||
|
static func createUser(page: Int?, index: Int) -> Employee {
|
||||||
|
Employee(
|
||||||
|
uuid: UUID(),
|
||||||
|
firstName: "First \(index + ((page ?? 1) - 1) * 20)",
|
||||||
|
lastName: "Last \(index + ((page ?? 1) - 1) * 20)",
|
||||||
|
phoneNumber: "555555\(1000 + index)",
|
||||||
|
emailAddress: "user\(index)@example.com",
|
||||||
|
biography: "Biography for employee \(index)",
|
||||||
|
photoURLSmall: "https://example.com/photo_small.jpg",
|
||||||
|
photoURLLarge: "https://example.com/photo_large.jpg"
|
||||||
|
)
|
||||||
|
}
|
||||||
|
func getUsers(page: Int?) async throws -> Employees {
|
||||||
|
// Simulate network delay.
|
||||||
|
try await Task.sleep(nanoseconds: 500_000_000)
|
||||||
|
|
||||||
|
// Create dummy data (20 employees per page).
|
||||||
|
let dummyEmployees: [Employee] = (0..<20).map { i in
|
||||||
|
Self.createUser(page: page, index: i)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Simulate that there are more pages if page is less than 5.
|
||||||
|
let hasNext = (page ?? 1) < 5
|
||||||
|
|
||||||
|
return Employees(result: dummyEmployees, hasNextPage: hasNext)
|
||||||
|
}
|
||||||
|
}
|
||||||
35
EmployeeDirectory/ViewModels/EmployeeViewModel.swift
Normal file
35
EmployeeDirectory/ViewModels/EmployeeViewModel.swift
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
//
|
||||||
|
// EmployeeViewModel.swift
|
||||||
|
// EmployeeDirectory
|
||||||
|
//
|
||||||
|
// Created by Matt Bruce on 3/3/25.
|
||||||
|
//
|
||||||
|
import Foundation
|
||||||
|
import Combine
|
||||||
|
|
||||||
|
@MainActor
|
||||||
|
public class EmployeeViewModel {
|
||||||
|
// MARK: - Properties
|
||||||
|
|
||||||
|
private let employee: Employee
|
||||||
|
|
||||||
|
public private(set) var uuid: String
|
||||||
|
public private(set) var fullName: String
|
||||||
|
public private(set) var phoneNumber: String?
|
||||||
|
public private(set) var emailAddress: String
|
||||||
|
public private(set) var biography: String?
|
||||||
|
public private(set) var smallPhoto: String?
|
||||||
|
public private(set) var largePhoto: String?
|
||||||
|
|
||||||
|
// MARK: - Initializer
|
||||||
|
public init(employee: Employee) {
|
||||||
|
self.employee = employee
|
||||||
|
uuid = employee.uuid.uuidString
|
||||||
|
fullName = "\(employee.firstName) \(employee.lastName)"
|
||||||
|
phoneNumber = employee.phoneNumber?.formatUSNumber()
|
||||||
|
emailAddress = employee.emailAddress
|
||||||
|
biography = employee.biography
|
||||||
|
smallPhoto = employee.photoURLSmall
|
||||||
|
largePhoto = employee.photoURLLarge
|
||||||
|
}
|
||||||
|
}
|
||||||
40
EmployeeDirectory/ViewModels/EmployeesViewModel.swift
Normal file
40
EmployeeDirectory/ViewModels/EmployeesViewModel.swift
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
//
|
||||||
|
// EmployeesViewModel.swift
|
||||||
|
// EmployeeDirectory
|
||||||
|
//
|
||||||
|
// Created by Matt Bruce on 3/3/25.
|
||||||
|
//
|
||||||
|
import Foundation
|
||||||
|
import Combine
|
||||||
|
|
||||||
|
/// ViewModel that will be bound to an Employees model and used
|
||||||
|
/// specifically with the EmployeesViewController.
|
||||||
|
@MainActor
|
||||||
|
public class EmployeesViewModel: ObservableObject {
|
||||||
|
@Published var employees: [Employee] = []
|
||||||
|
@Published var isLoading = false
|
||||||
|
@Published var hasNextPage = false
|
||||||
|
|
||||||
|
private var currentPage = 1
|
||||||
|
let service: EmployeeServiceProtocol
|
||||||
|
|
||||||
|
init(service: EmployeeServiceProtocol) {
|
||||||
|
self.service = service
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Loads the next page of employees.
|
||||||
|
func loadEmployees() async {
|
||||||
|
guard !isLoading else { return }
|
||||||
|
isLoading = true
|
||||||
|
do {
|
||||||
|
let response = try await service.getUsers(page: currentPage)
|
||||||
|
employees.append(contentsOf: response.result)
|
||||||
|
hasNextPage = response.hasNextPage
|
||||||
|
currentPage += 1
|
||||||
|
} catch {
|
||||||
|
print("Error loading employees: \(error)")
|
||||||
|
}
|
||||||
|
isLoading = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
80
EmployeeDirectory/Views/ContactButtonView.swift
Normal file
80
EmployeeDirectory/Views/ContactButtonView.swift
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
//
|
||||||
|
// ContactButtonView.swift
|
||||||
|
// EmployeeDirectory
|
||||||
|
//
|
||||||
|
// Created by Matt Bruce on 3/4/25.
|
||||||
|
//
|
||||||
|
import SwiftUI
|
||||||
|
|
||||||
|
enum ContactType {
|
||||||
|
case phone, email
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ContactButtonView: View {
|
||||||
|
let contactType: ContactType
|
||||||
|
let contactValue: String
|
||||||
|
let enabled: Bool
|
||||||
|
|
||||||
|
init(contactType: ContactType, contactValue: String, enabled: Bool = true) {
|
||||||
|
self.contactType = contactType
|
||||||
|
self.contactValue = contactValue
|
||||||
|
self.enabled = enabled
|
||||||
|
}
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
if !enabled {
|
||||||
|
Label(labelText(for: contactValue), systemImage: iconName)
|
||||||
|
.foregroundColor(.gray)
|
||||||
|
.font(.caption)
|
||||||
|
} else {
|
||||||
|
Button(action: {
|
||||||
|
if let url = URL(string: urlString(for: contactValue)) {
|
||||||
|
UIApplication.shared.open(url)
|
||||||
|
}
|
||||||
|
}) {
|
||||||
|
Label(labelText(for: contactValue), systemImage: iconName)
|
||||||
|
.foregroundColor(.blue)
|
||||||
|
.font(.caption)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: - Helpers
|
||||||
|
|
||||||
|
private var iconName: String {
|
||||||
|
switch contactType {
|
||||||
|
case .phone:
|
||||||
|
return "phone.fill"
|
||||||
|
case .email:
|
||||||
|
return "envelope.fill"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private func labelText(for value: String) -> String {
|
||||||
|
switch contactType {
|
||||||
|
case .phone:
|
||||||
|
return "Call \(value)"
|
||||||
|
case .email:
|
||||||
|
return "Email \(value)"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private func urlString(for value: String) -> String {
|
||||||
|
switch contactType {
|
||||||
|
case .phone:
|
||||||
|
return "tel://\(value)"
|
||||||
|
case .email:
|
||||||
|
return "mailto:\(value)"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#Preview {
|
||||||
|
VStack (alignment: .leading) {
|
||||||
|
Text("Phone Number:")
|
||||||
|
ContactButtonView(contactType: .phone, contactValue: MockEmployeeService.sample.phoneNumber!.formatUSNumber()).padding()
|
||||||
|
Text("Email Address:")
|
||||||
|
ContactButtonView(contactType: .email, contactValue: MockEmployeeService.sample.emailAddress).padding()
|
||||||
|
}
|
||||||
|
}
|
||||||
43
EmployeeDirectory/Views/EmployeeDetailsView.swift
Normal file
43
EmployeeDirectory/Views/EmployeeDetailsView.swift
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
//
|
||||||
|
// EmployeeDetailsView.swift
|
||||||
|
// EmployeeDirectory
|
||||||
|
//
|
||||||
|
// Created by Matt Bruce on 3/4/25.
|
||||||
|
//
|
||||||
|
|
||||||
|
import SwiftUI
|
||||||
|
|
||||||
|
struct EmployeeDetailsView: View {
|
||||||
|
let viewModel: EmployeeViewModel
|
||||||
|
let defaultImage = "person.circle.fill" // SF Symbol as fallback
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
VStack(alignment: .leading, spacing: 12) {
|
||||||
|
// Profile Image with Default Placeholder
|
||||||
|
ProfileImageView(urlString: viewModel.largePhoto, size: 250) // Uses reusable component
|
||||||
|
.padding(.bottom)
|
||||||
|
Text(viewModel.fullName)
|
||||||
|
.font(.headline)
|
||||||
|
|
||||||
|
if let bio = viewModel.biography {
|
||||||
|
Text(bio)
|
||||||
|
.font(.footnote)
|
||||||
|
.foregroundColor(.gray)
|
||||||
|
.lineLimit(2)
|
||||||
|
}
|
||||||
|
|
||||||
|
ContactButtonView(contactType: .email, contactValue: viewModel.emailAddress)
|
||||||
|
|
||||||
|
if let phone = viewModel.phoneNumber {
|
||||||
|
ContactButtonView(contactType: .phone, contactValue: phone)
|
||||||
|
}
|
||||||
|
Spacer()
|
||||||
|
}
|
||||||
|
.padding(.all)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#Preview {
|
||||||
|
EmployeeDetailsView(viewModel: .init(employee: MockEmployeeService.sample)).padding(.all)
|
||||||
|
}
|
||||||
|
|
||||||
49
EmployeeDirectory/Views/EmployeeListView.swift
Normal file
49
EmployeeDirectory/Views/EmployeeListView.swift
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
//
|
||||||
|
// ContentView.swift
|
||||||
|
// EmployeeDirectory
|
||||||
|
//
|
||||||
|
// Created by Matt Bruce on 3/3/25.
|
||||||
|
//
|
||||||
|
|
||||||
|
import SwiftUI
|
||||||
|
|
||||||
|
@MainActor
|
||||||
|
struct EmployeeListView: View {
|
||||||
|
@StateObject public var viewModel: EmployeesViewModel
|
||||||
|
|
||||||
|
// Dependency injection via the initializer.
|
||||||
|
init(viewModel: EmployeesViewModel? = nil) {
|
||||||
|
_viewModel = StateObject(wrappedValue: viewModel ?? .init(service: EmployeeService()))
|
||||||
|
}
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
NavigationView {
|
||||||
|
List {
|
||||||
|
ForEach(viewModel.employees) { employee in
|
||||||
|
NavigationLink(destination: EmployeeDetailsView(viewModel: .init(employee: employee))) {
|
||||||
|
EmployeeRowView(viewModel: .init(employee: employee))
|
||||||
|
}
|
||||||
|
.listRowSeparator(.hidden)
|
||||||
|
}
|
||||||
|
if viewModel.hasNextPage {
|
||||||
|
ProgressView()
|
||||||
|
.frame(maxWidth: .infinity, alignment: .center)
|
||||||
|
.task {
|
||||||
|
await viewModel.loadEmployees()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.navigationTitle("Employees")
|
||||||
|
.listStyle(.plain)
|
||||||
|
}
|
||||||
|
.task {
|
||||||
|
if viewModel.employees.isEmpty {
|
||||||
|
await viewModel.loadEmployees()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#Preview {
|
||||||
|
EmployeeListView(viewModel: .init(service: MockEmployeeService()))
|
||||||
|
}
|
||||||
47
EmployeeDirectory/Views/EmployeeRowView.swift
Normal file
47
EmployeeDirectory/Views/EmployeeRowView.swift
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
//
|
||||||
|
// EmployeeRowView.swift
|
||||||
|
// EmployeeDirectory
|
||||||
|
//
|
||||||
|
// Created by Matt Bruce on 3/3/25.
|
||||||
|
//
|
||||||
|
|
||||||
|
import SwiftUI
|
||||||
|
|
||||||
|
struct EmployeeRowView: View {
|
||||||
|
let viewModel: EmployeeViewModel
|
||||||
|
let defaultImage = "person.circle.fill" // SF Symbol as fallback
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
HStack(alignment: .top, spacing: 12) {
|
||||||
|
// Profile Image with Default Placeholder
|
||||||
|
ProfileImageView(urlString: viewModel.smallPhoto, size: 50) // Uses reusable component
|
||||||
|
|
||||||
|
// Employee Info Stack
|
||||||
|
VStack(alignment: .leading, spacing: 4) {
|
||||||
|
Text(viewModel.fullName)
|
||||||
|
.font(.headline)
|
||||||
|
|
||||||
|
if let bio = viewModel.biography {
|
||||||
|
Text(bio)
|
||||||
|
.font(.footnote)
|
||||||
|
.foregroundColor(.gray)
|
||||||
|
.lineLimit(2)
|
||||||
|
}
|
||||||
|
|
||||||
|
ContactButtonView(contactType: .email, contactValue: viewModel.emailAddress, enabled: false)
|
||||||
|
|
||||||
|
if let phone = viewModel.phoneNumber {
|
||||||
|
ContactButtonView(contactType: .phone, contactValue: phone, enabled: false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Spacer() // Pushes everything to the left
|
||||||
|
}
|
||||||
|
.padding(.vertical, 8)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#Preview {
|
||||||
|
EmployeeRowView(viewModel: .init(employee: MockEmployeeService.sample)).padding()
|
||||||
|
}
|
||||||
|
|
||||||
37
EmployeeDirectory/Views/ProfileImageView.swift
Normal file
37
EmployeeDirectory/Views/ProfileImageView.swift
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
//
|
||||||
|
// ProfileImageView.swift
|
||||||
|
// EmployeeDirectory
|
||||||
|
//
|
||||||
|
// Created by Matt Bruce on 3/3/25.
|
||||||
|
//
|
||||||
|
|
||||||
|
import SwiftUI
|
||||||
|
|
||||||
|
struct ProfileImageView: View {
|
||||||
|
let urlString: String?
|
||||||
|
let size: CGFloat // Allows us to customize the image size
|
||||||
|
|
||||||
|
private let defaultImage = "person.circle.fill" // SF Symbol
|
||||||
|
|
||||||
|
var body: some View {
|
||||||
|
AsyncImage(url: URL(string: urlString ?? "")) { phase in
|
||||||
|
if let image = phase.image {
|
||||||
|
image.resizable()
|
||||||
|
} else if phase.error != nil || urlString == nil {
|
||||||
|
Image(systemName: defaultImage)
|
||||||
|
.resizable()
|
||||||
|
.foregroundColor(.gray)
|
||||||
|
} else {
|
||||||
|
ProgressView() // Show loader while fetching
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.scaledToFill()
|
||||||
|
.frame(width: size, height: size)
|
||||||
|
.clipShape(Circle()) // Rounded shape
|
||||||
|
.overlay(Circle().stroke(Color.gray, lineWidth: 1)) // Border
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#Preview {
|
||||||
|
ProfileImageView(urlString: MockEmployeeService.sample.photoURLSmall, size: 100).padding()
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue
Block a user