Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: 🎸 [JIRA:HCPSDKFIORIUIKIT-2716]Step Progress Indicator #787

Merged
merged 11 commits into from
Sep 13, 2024
Original file line number Diff line number Diff line change
Expand Up @@ -51,34 +51,76 @@ struct StepProgressIndicatorExample_Previews: PreviewProvider {
}

struct SPIExampleWithIcon: View {
static let icon = Image(systemName: "app.dashed")
@State var title: String = ""
@State var steps = [StepItemData(title: "Sign In", icon: FioriIcon.arrows.initiative, state: .completed),
StepItemData(title: "User Info", icon: FioriIcon.people.personPlaceholder, state: .completed),
StepItemData(title: "Account Info", icon: FioriIcon.actions.edit, state: .completed),
StepItemData(title: "Settings", icon: FioriIcon.actions.actionSettings, state: .normal, substeps: [StepItemData(title: "Settings 1")]),
StepItemData(title: "Other0", icon: icon, state: .disabled),
StepItemData(title: "Other1", icon: icon, state: .completed),
StepItemData(title: "other2", icon: icon, state: .error)]
@State var selection: String = ""
@State var iconSteps = [StepItemData(title: "Sign In", node: .icon(FioriIcon.arrows.initiative), state: .completed),
StepItemData(title: "User Info", node: .icon(FioriIcon.people.personPlaceholder), state: .completed),
StepItemData(title: "Account Info", node: .icon(FioriIcon.actions.edit), state: .completed),
StepItemData(title: "Settings", node: .icon(FioriIcon.actions.actionSettings), state: .normal, substeps: [StepItemData(title: "Settings 1")]),
StepItemData(title: "Other0", state: .disabled),
StepItemData(title: "Other1", state: .completed),
StepItemData(title: "other2", state: .error)]
@State var mixtureSteps = [StepItemData(title: "Sign In", node: .text("No.1"), state: .completed),
StepItemData(title: "User Info", state: .completed),
StepItemData(title: "Account Info", node: .icon(FioriIcon.actions.edit), state: .completed),
StepItemData(title: "Settings", node: .icon(FioriIcon.actions.actionSettings), state: .normal, substeps: [StepItemData(title: "Settings 1")]),
StepItemData(title: "Other0", state: .disabled),
StepItemData(title: "Other1", state: .completed),
StepItemData(title: "other2", state: .error)]
@State var textSteps = [StepItemData(title: "Sign In", state: .completed),
StepItemData(title: "User Info", node: .text("AB"), state: .completed),
StepItemData(title: "Account Info", state: .completed),
StepItemData(title: "Settings", node: .text("AD"), state: .normal, substeps: [StepItemData(title: "Settings 1")]),
StepItemData(title: "Other0", state: .disabled),
StepItemData(title: "Other1", state: .completed),
StepItemData(title: "other2", state: .error)]
@State var iconSelection: String = ""
@State var mixtureSelection: String = ""
@State var textSelection: String = ""
var body: some View {
VStack(alignment: .leading) {
Text("Icon").bold()
StepProgressIndicator(selection: self.$selection,
stepItems: self.steps)
Text("Step: Only Icon Node").bold()
StepProgressIndicator(selection: self.$iconSelection,
stepItems: self.iconSteps)
{
Text(self.$title.wrappedValue).lineLimit(1)
} action: {
Button {} label: {
HStack(spacing: 2) {
Text("All Steps(\(self.steps.count))")
Text("All Steps(\(self.iconSteps.count))")
.foregroundStyle(Color.preferredColor(.tintColor))
FioriIcon.actions.slimArrowRight
.font(.fiori(forTextStyle: .subheadline, weight: .semibold))
.foregroundStyle(Color.preferredColor(.separator))
}
}
}
.stepProgressIndicatorNodeType(.icon)
.padding()
.onChange(of: self.iconSelection, perform: { _ in
self.updateCurrentStepName()
})
.onAppear {
self.updateCurrentStepName()
}

Text("Step: Mixture Node").bold()
StepProgressIndicator(selection: self.$mixtureSelection,
stepItems: self.mixtureSteps)
{
Text("Invariant title").lineLimit(1)
} action: {}
.stepProgressIndicatorNodeType(.mixture)
.padding()

Text("Step: Only Text Node").bold()
StepProgressIndicator(selection: self.$textSelection,
stepItems: self.textSteps)
{
Text("Invariant title").lineLimit(1)
} action: {}
.stepProgressIndicatorNodeType(.text)
.padding()

Spacer().padding(20)
Button {
self.completeStep()
Expand All @@ -87,19 +129,12 @@ struct SPIExampleWithIcon: View {
}
.padding(20)
}
.padding()
.onChange(of: self.selection, perform: { _ in
self.updateCurrentStepName()
})
.onAppear {
self.updateCurrentStepName()
}
}

func getStep() -> StepItem? {
func findStep(in data: [StepItem]) -> StepItem? {
for step in data {
if step.id == self.selection {
if step.id == self.iconSelection {
return step
}

Expand All @@ -114,7 +149,7 @@ struct SPIExampleWithIcon: View {

return nil
}
return findStep(in: self.steps)
return findStep(in: self.iconSteps)
}

func updateCurrentStepName() {
Expand All @@ -125,15 +160,15 @@ struct SPIExampleWithIcon: View {
}

func completeStep() {
for index in self.steps.indices {
if self.steps[index].id == self.selection {
self.steps[index].state = .completed
for index in self.iconSteps.indices {
if self.iconSteps[index].id == self.iconSelection {
self.iconSteps[index].state = .completed
} else {
let substeps = self.steps[index].substeps
let substeps = self.iconSteps[index].substeps
guard !substeps.isEmpty else { continue }
for subindex in substeps.indices {
if substeps[subindex].id == self.selection {
self.steps[index].substeps[subindex].state = .completed
if substeps[subindex].id == self.iconSelection {
self.iconSteps[index].substeps[subindex].state = .completed
}
}
}
Expand Down Expand Up @@ -528,22 +563,22 @@ struct StepItemData: StepItem {
var id: String = UUID().uuidString
/// Step title.
var title: String?
/// Node icon
var icon: Image?
/// Step node
var node: TextOrIcon?
/// Step state.
var state: StepProgressIndicatorState = .normal
/// Sub-steps for this one.
var substeps: [StepItem] = []

init(id: String = UUID().uuidString,
title: String? = nil,
icon: Image? = nil,
node: TextOrIcon? = nil,
state: StepProgressIndicatorState = [],
substeps: [StepItemData] = [])
{
self.id = id
self.title = title
self.icon = icon
self.node = node
self.state = state
self.substeps = substeps
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,16 @@ public protocol StepItem {
var state: StepProgressIndicatorState { get set }
/// Substeps for this one.
var substeps: [StepItem] { get set }
/// Node icon.
var icon: Image? { get }
/// Step node.
var node: TextOrIcon? { get }
billzhou0223 marked this conversation as resolved.
Show resolved Hide resolved
}

/// Node of Step Progress Indicator display type, default is `mixture`.
public enum StepProgressIndicatorNodeType {
KevinZK marked this conversation as resolved.
Show resolved Hide resolved
/// Text and icon.
case mixture
KevinZK marked this conversation as resolved.
Show resolved Hide resolved
/// Only text.
case text
/// Only icon.
case icon
}
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,26 @@ extension EnvironmentValues {
}
}

struct StepProgressIndicatorNodeTypeKey: EnvironmentKey {
static let defaultValue: StepProgressIndicatorNodeType = .mixture
}

extension EnvironmentValues {
var stepProgressIndicatorNodeType: StepProgressIndicatorNodeType {
get { self[StepProgressIndicatorNodeTypeKey.self] }
set { self[StepProgressIndicatorNodeTypeKey.self] = newValue }
}
}

public extension View {
/// The style of the node of `StepProgressIndicator`.
/// - Parameter type: `StepProgressIndicatorNodeType` enum values.
/// - Returns: A new `StepProgressIndicator` with specific node.
func stepProgressIndicatorNodeType(_ type: StepProgressIndicatorNodeType) -> some View {
self.environment(\.stepProgressIndicatorNodeType, type)
}
}

public extension View {
/// Step style for `StepProgressIndicator`.
/// - Parameter style: Style for `StepProgressIndicator`.
Expand Down
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import FioriThemeManager
import SwiftUI

/// Not used by developers.
Expand Down Expand Up @@ -52,6 +53,7 @@ struct DefaultSingleStep: View {
@Environment(\.dynamicTypeSize) var dynamicTypeSize
@Environment(\.stepAxis) var stepAxis
@Environment(\.stepFrames) var stepFrames
@Environment(\.stepProgressIndicatorNodeType) var type

var stepItem: StepItem
@Binding var selection: String
Expand Down Expand Up @@ -98,11 +100,33 @@ struct DefaultSingleStep: View {
} node: {
ZStack {
self.node(by: self.stepItem.state, isSelected: isSelected)
if self.stepItem.icon.isEmpty {
Text("\(self.index + 1)")
.font(Font.fiori(forTextStyle: .footnote))
} else {
self.stepItem.icon
self.noNodeStep()
switch self.type {
case .mixture:
switch self.stepItem.node {
case .text(let string):
Text(string)
.font(Font.fiori(forTextStyle: .footnote))
case .icon(let image):
image
case .none:
EmptyView()
}
case .icon:
switch self.stepItem.node {
case .icon(let image):
image
case .none, .text:
EmptyView()
}
case .text:
switch self.stepItem.node {
case .text(let string):
Text(string)
.font(Font.fiori(forTextStyle: .footnote))
case .icon, .none:
EmptyView()
}
}
}
.frame(width: self.sideLength, height: self.sideLength)
Expand All @@ -126,6 +150,19 @@ struct DefaultSingleStep: View {
}
}

@ViewBuilder
func noNodeStep() -> some View {
if self.stepItem.node == nil {
switch self.type {
case .mixture, .text:
Text("\(self.index + 1)")
.font(Font.fiori(forTextStyle: .footnote))
case .icon:
Image(systemName: "app.dashed")
}
}
}

@ViewBuilder
func singleSubstep() -> some View {
if self.stepItem.state.isSupported {
Expand Down
Loading