Skip to content

Commit 8337ed4

Browse files
authored
Merge pull request #24 from OWND-Project/release
Release -> develop
2 parents 42bce66 + d25c7a8 commit 8337ed4

File tree

7 files changed

+256
-161
lines changed

7 files changed

+256
-161
lines changed

tw2023_wallet/Feature/IssueCredential/ViewModels/CredentialOfferViewModel.swift

+19-2
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,22 @@
55
// Created by 若葉良介 on 2023/12/22.
66
//
77

8+
9+
/*
10+
-------------------------------------------------------------------------------
11+
Caution!
12+
13+
`CredentialOfferView` will be significantly revised in PR#17,
14+
which has been filed to make it ID1 compliant.
15+
This PR#17 will be merged soon, but will not be included in the next release.
16+
17+
With that in mind, the current content below contains a stopgap measure
18+
that was implemented to get through the next release.
19+
20+
-------------------------------------------------------------------------------
21+
*/
22+
23+
824
import Foundation
925
import SwiftUI
1026

@@ -14,7 +30,7 @@ enum CredentialOfferParseError: Error {
1430
case InvalidCredentialOffer
1531
}
1632

17-
class CredentialOfferViewModel {
33+
class CredentialOfferViewModel: ObservableObject {
1834
var dataModel: CredentialOfferModel = .init()
1935
var rawCredentialOfferString: String? = nil
2036

@@ -48,7 +64,6 @@ class CredentialOfferViewModel {
4864

4965
func sendRequest(userPin: String?) async throws {
5066
do {
51-
interpretMetadataAndOffer()
5267

5368
let vciClient = try await VCIClient(
5469
credentialOfferJson:
@@ -105,6 +120,8 @@ class CredentialOfferViewModel {
105120

106121
let credentialIssuer = credentialOffer!["credential_issuer"] as! String
107122
self.dataModel.metaData = try await retrieveAllMetadata(issuer: credentialIssuer)
123+
124+
interpretMetadataAndOffer()
108125

109126
dataModel.isLoading = false
110127
dataModel.hasLoadedData = true

tw2023_wallet/Feature/IssueCredential/Views/CredentialOffer.swift

+117-85
Original file line numberDiff line numberDiff line change
@@ -5,8 +5,20 @@
55
// Created by 若葉良介 on 2023/12/22.
66
//
77

8-
// AndroidのConfirmationFragment相当
9-
// https://github.com/datasign-inc/tw2023-wallet-android/blob/3655ace01d3c454529e34f1b78a4329f44508d23/app/src/main/java/com/ownd_project/tw2023_wallet_android/ui/confirmation/ConfirmationFragment.kt#L1
8+
9+
/*
10+
-------------------------------------------------------------------------------
11+
Caution!
12+
13+
`CredentialOfferView` will be significantly revised in PR#17,
14+
which has been filed to make it ID1 compliant.
15+
This PR#17 will be merged soon, but will not be included in the next release.
16+
17+
With that in mind, the current content below contains a stopgap measure
18+
that was implemented to get through the next release.
19+
20+
-------------------------------------------------------------------------------
21+
*/
1022

1123
import SwiftUI
1224

@@ -37,35 +49,19 @@ func getClaimNames(credentialSupported: CredentialSupported?) -> [String] {
3749
struct CredentialOfferView: View {
3850
@Environment(\.presentationMode) var presentationMode
3951
@Environment(CredentialOfferArgs.self) var args
40-
var viewModel: CredentialOfferViewModel
52+
@StateObject var viewModel: CredentialOfferViewModel = .init()
4153
@State private var navigateToHome = false
4254
@State private var navigateToPinInput = false
4355
@State private var showErrorDialog = false
4456

45-
init(viewModel: CredentialOfferViewModel = CredentialOfferViewModel()) {
46-
self.viewModel = viewModel
47-
}
48-
49-
var body: some View {
50-
NavigationStack {
51-
Group {
52-
if viewModel.dataModel.isLoading {
53-
ProgressView().progressViewStyle(CircularProgressViewStyle())
54-
}
55-
else {
56-
let issuerDisplayName = viewModel.dataModel.metaData?.display?.first?.name ?? ""
57-
let credentialSupported = viewModel.dataModel.metaData?.credentialsSupported
58-
.keys
59-
let firstCredentialName = credentialSupported?.first
60-
let targetCredential =
61-
firstCredentialName == nil
62-
? nil
63-
: viewModel.dataModel.metaData?.credentialsSupported[firstCredentialName!]
64-
65-
let credentialDisplayName = getCredentialDisplayName(
66-
credentialSupported: targetCredential)
67-
let displayNames: [String] = getClaimNames(
68-
credentialSupported: targetCredential)
57+
private func contentWithMetaData(
58+
issuerDisplayName: String,
59+
credentialDisplayName: String,
60+
displayNames: [String]
61+
) -> some View {
62+
GeometryReader { geometry in
63+
ScrollView {
64+
VStack {
6965
HStack {
7066
Button("Cancel") {
7167
presentationMode.wrappedValue.dismiss()
@@ -74,67 +70,103 @@ struct CredentialOfferView: View {
7470
.padding(.horizontal, 16)
7571
Spacer()
7672
}
77-
GeometryReader { geometry in
78-
ScrollView {
79-
VStack {
80-
Text(
81-
String(
82-
format: NSLocalizedString(
83-
"credentialOfferText", comment: ""), issuerDisplayName,
84-
credentialDisplayName)
85-
)
86-
.modifier(Title3Black())
87-
Image("issue_confirmation")
88-
.resizable()
89-
.scaledToFit()
90-
.frame(width: geometry.size.width * 0.65) // 横幅の65%に設定
91-
}
92-
Text("Items to be issued")
93-
.padding(.vertical, 16)
94-
.frame(maxWidth: .infinity, alignment: .leading) // 左寄せ
95-
.modifier(BodyGray())
96-
ForEach(displayNames, id: \.self) { displayName in
97-
CredentialSubjectLow(item: displayName)
73+
74+
Text(
75+
String(
76+
format: NSLocalizedString(
77+
"credentialOfferText", comment: ""), issuerDisplayName,
78+
credentialDisplayName)
79+
)
80+
.modifier(Title3Black())
81+
Image("issue_confirmation")
82+
.resizable()
83+
.scaledToFit()
84+
.frame(width: geometry.size.width * 0.65) // 横幅の65%に設定
85+
}
86+
Text("Items to be issued")
87+
.padding(.vertical, 16)
88+
.frame(maxWidth: .infinity, alignment: .leading) // 左寄せ
89+
.modifier(BodyGray())
90+
ForEach(displayNames, id: \.self) { displayName in
91+
CredentialSubjectLow(item: displayName)
92+
}
93+
Text("issuing_authority_information")
94+
.frame(maxWidth: .infinity, alignment: .leading) // 左寄せ
95+
.padding(.top, 32)
96+
.modifier(BodyBlack())
97+
98+
IssuerDetail(
99+
issuerMetadata: viewModel.dataModel.metaData, showTitle: false)
100+
ActionButtonBlack(
101+
title: "issue_credential",
102+
action: {
103+
let pinRequired = viewModel.checkIfPinIsRequired()
104+
if pinRequired {
105+
self.navigateToPinInput = true
106+
}
107+
else {
108+
Task {
109+
try await viewModel.sendRequest(userPin: nil)
110+
self.navigateToHome = true
98111
}
99-
Text("issuing_authority_information")
100-
.frame(maxWidth: .infinity, alignment: .leading) // 左寄せ
101-
.padding(.top, 32)
102-
.modifier(BodyBlack())
103-
104-
IssuerDetail(
105-
issuerMetadata: viewModel.dataModel.metaData, showTitle: false)
106-
ActionButtonBlack(
107-
title: "issue_credential",
108-
action: {
109-
let pinRequired = viewModel.checkIfPinIsRequired()
110-
if pinRequired {
111-
self.navigateToPinInput = true
112-
}
113-
else {
114-
Task {
115-
try await viewModel.sendRequest(userPin: nil)
116-
self.navigateToHome = true
117-
}
118-
}
119-
}
120-
)
121-
.padding(.vertical, 16)
122-
.navigationDestination(
123-
isPresented: $navigateToHome,
124-
destination: {
125-
Home()
126-
}
127-
)
128-
.navigationDestination(
129-
isPresented: $navigateToPinInput,
130-
destination: {
131-
PinCodeInput(viewModel: self.viewModel)
132-
}
133-
)
134112
}
135-
.padding(.horizontal, 16) // 左右に16dpのパディング
136-
.padding(.vertical, 16)
137113
}
114+
)
115+
.padding(.vertical, 16)
116+
.navigationDestination(
117+
isPresented: $navigateToHome,
118+
destination: {
119+
Home()
120+
}
121+
)
122+
.navigationDestination(
123+
isPresented: $navigateToPinInput,
124+
destination: {
125+
PinCodeInput(viewModel: self.viewModel)
126+
}
127+
)
128+
}
129+
.padding(.horizontal, 16) // 左右に16dpのパディング
130+
.padding(.vertical, 16)
131+
}
132+
}
133+
134+
@ViewBuilder
135+
private var content: some View {
136+
137+
if let issuerDisplayName = viewModel.dataModel.metaData?.display?.first?.name,
138+
let credentialName = viewModel.credential_vct,
139+
let credentialSupported = viewModel.dataModel.metaData?.credentialsSupported,
140+
let targetCredential = viewModel.dataModel.metaData?.credentialsSupported[
141+
credentialName]
142+
{
143+
144+
let credentialDisplayName = getCredentialDisplayName(
145+
credentialSupported: targetCredential)
146+
let displayNames: [String] = getClaimNames(
147+
credentialSupported: targetCredential)
148+
149+
contentWithMetaData(
150+
issuerDisplayName: issuerDisplayName, credentialDisplayName: credentialDisplayName,
151+
displayNames: displayNames)
152+
}
153+
else {
154+
EmptyView()
155+
.onAppear {
156+
print("Unable to load metadata")
157+
showErrorDialog = true
158+
}
159+
}
160+
}
161+
162+
var body: some View {
163+
NavigationStack {
164+
Group {
165+
if viewModel.dataModel.isLoading {
166+
ProgressView().progressViewStyle(CircularProgressViewStyle())
167+
}
168+
else {
169+
content
138170
}
139171
}
140172
.navigationBarTitle("", displayMode: .inline)

tw2023_wallet/Localizable.xcstrings

+41-2
Original file line numberDiff line numberDiff line change
@@ -614,6 +614,28 @@
614614
}
615615
}
616616
},
617+
"eventName" : {
618+
"extractionState" : "manual",
619+
"localizations" : {
620+
"ja" : {
621+
"stringUnit" : {
622+
"state" : "translated",
623+
"value" : "イベント名"
624+
}
625+
}
626+
}
627+
},
628+
"EventTicketCredential" : {
629+
"extractionState" : "manual",
630+
"localizations" : {
631+
"ja" : {
632+
"stringUnit" : {
633+
"state" : "translated",
634+
"value" : "イベントチケット"
635+
}
636+
}
637+
}
638+
},
617639
"failed_to_show_credential_offer" : {
618640
"localizations" : {
619641
"en" : {
@@ -1035,7 +1057,7 @@
10351057
"ja" : {
10361058
"stringUnit" : {
10371059
"state" : "translated",
1038-
"value" : "氏名"
1060+
"value" : "イベント名"
10391061
}
10401062
}
10411063
}
@@ -1672,6 +1694,23 @@
16721694
}
16731695
}
16741696
},
1697+
"ticketNo" : {
1698+
"extractionState" : "manual",
1699+
"localizations" : {
1700+
"en" : {
1701+
"stringUnit" : {
1702+
"state" : "translated",
1703+
"value" : "Ticket Number"
1704+
}
1705+
},
1706+
"ja" : {
1707+
"stringUnit" : {
1708+
"state" : "translated",
1709+
"value" : "チケット番号"
1710+
}
1711+
}
1712+
}
1713+
},
16751714
"TotalItems" : {
16761715
"extractionState" : "manual",
16771716
"localizations" : {
@@ -2056,4 +2095,4 @@
20562095
}
20572096
},
20582097
"version" : "1.0"
2059-
}
2098+
}

tw2023_wallet/Services/OID/OpenIdProvider.swift

+8-2
Original file line numberDiff line numberDiff line change
@@ -112,7 +112,13 @@ class OpenIdProvider {
112112
the Client Identifier MUST be a DNS name and match a dNSName Subject Alternative Name (SAN) [RFC5280] entry in the leaf certificate passed with the request.
113113
*/
114114
let (decoded, certificates) = verifedX5CJwt
115-
if isDomainInSAN(certificate: certificates[0], domain: _clientId) {
115+
116+
guard let url = URL(string: _clientId),
117+
let domainName = url.host else {
118+
return .failure(.authRequestInputError(reason: .compliantError(reason: "Unable to get host name")))
119+
}
120+
121+
if isDomainInSAN(certificate: certificates[0], domain: domainName) {
116122
print("verify san entry success")
117123
}
118124
else {
@@ -399,7 +405,7 @@ class OpenIdProvider {
399405
throw NetworkError.invalidResponse
400406
}
401407
}
402-
408+
403409
return PostResult(statusCode: statusCode, location: nil, cookies: nil)
404410
}
405411

tw2023_wallet/datastore/CredentialDataManager.swift

+3-4
Original file line numberDiff line numberDiff line change
@@ -118,10 +118,9 @@ extension Datastore_CredentialData {
118118
}
119119
}
120120

121-
private func getBackgroundImage() -> String? {
121+
private func getBackgroundImage(credentialType: String) -> String? {
122122
guard let metaData = self.parsedMetaData(),
123-
let supportedName = metaData.credentialsSupported.keys.first, // todo: 1つめを前提としている
124-
let supported = metaData.credentialsSupported[supportedName],
123+
let supported = metaData.credentialsSupported[credentialType],
125124
let displays = supported.display,
126125
let firstDisplay = displays.first, // todo: 1つめを前提としている
127126
let backgroundImageUrl = firstDisplay.backgroundImage
@@ -161,7 +160,7 @@ extension Datastore_CredentialData {
161160
issuer: issuer,
162161
issuerDisplayName: issuerName,
163162
issuedAt: iat,
164-
backgroundImageUrl: getBackgroundImage(),
163+
backgroundImageUrl: getBackgroundImage(credentialType: self.type),
165164
credentialType: self.type,
166165
disclosure: disclosure,
167166
qrDisplay: self.generateQRDisplay(),

0 commit comments

Comments
 (0)