Fix #37: Privacy-Compliance + Onboarding-Kontaktpflicht

- NSContactsUsageDescription + NSPhotoLibraryUsageDescription in Info.plist
  ergänzt (App-Store-Review-Compliance für import Contacts/PhotosUI)
- Onboarding: Überspringen-Button entfernt, mindestens 1 Kontakt erforderlich
- Hinweistext wenn Kontaktliste leer: erklärt warum Weiter gesperrt ist
- Alert wenn Picker-Auswahl das Free-Tier-Limit von 3 überschreitet
- Lokalisierung (DE+EN) für alle neuen Strings

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-04-24 09:16:00 +02:00
parent 2c274ff4ae
commit e4c2293bec
3 changed files with 4405 additions and 4357 deletions
+4
View File
@@ -715,6 +715,8 @@
INFOPLIST_KEY_CFBundleDisplayName = nahbar;
INFOPLIST_KEY_NSCalendarsFullAccessUsageDescription = "nahbar liest deine Kalender, damit du beim Erstellen eines Termins den richtigen Kalender wählen kannst.";
INFOPLIST_KEY_NSCalendarsWriteOnlyAccessUsageDescription = "nahbar erstellt Kalendereinträge für geplante Treffen";
INFOPLIST_KEY_NSContactsUsageDescription = "nahbar öffnet den systemseitigen Kontakte-Picker, damit du Personen schnell aus deinem Adressbuch hinzufügen kannst. Die App liest deine Kontakte nicht selbstständig aus.";
INFOPLIST_KEY_NSPhotoLibraryUsageDescription = "nahbar öffnet den systemseitigen Foto-Picker, damit du ein Profilbild für eine Person auswählen kannst.";
INFOPLIST_KEY_UIApplicationSceneManifest_Generation = YES;
INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents = YES;
INFOPLIST_KEY_UILaunchScreen_Generation = YES;
@@ -756,6 +758,8 @@
INFOPLIST_KEY_CFBundleDisplayName = nahbar;
INFOPLIST_KEY_NSCalendarsFullAccessUsageDescription = "nahbar liest deine Kalender, damit du beim Erstellen eines Termins den richtigen Kalender wählen kannst.";
INFOPLIST_KEY_NSCalendarsWriteOnlyAccessUsageDescription = "nahbar erstellt Kalendereinträge für geplante Treffen";
INFOPLIST_KEY_NSContactsUsageDescription = "nahbar öffnet den systemseitigen Kontakte-Picker, damit du Personen schnell aus deinem Adressbuch hinzufügen kannst. Die App liest deine Kontakte nicht selbstständig aus.";
INFOPLIST_KEY_NSPhotoLibraryUsageDescription = "nahbar öffnet den systemseitigen Foto-Picker, damit du ein Profilbild für eine Person auswählen kannst.";
INFOPLIST_KEY_UIApplicationSceneManifest_Generation = YES;
INFOPLIST_KEY_UIApplicationSupportsIndirectInputEvents = YES;
INFOPLIST_KEY_UILaunchScreen_Generation = YES;
File diff suppressed because it is too large Load Diff
+29 -20
View File
@@ -31,8 +31,7 @@ struct OnboardingContainerView: View {
OnboardingContactImportView(
coordinator: coordinator,
onContinue: startPrivacyScreen,
onSkip: startPrivacyScreen
onContinue: startPrivacyScreen
)
.tag(1)
}
@@ -345,10 +344,10 @@ private struct OnboardingProfileView: View {
private struct OnboardingContactImportView: View {
@ObservedObject var coordinator: OnboardingCoordinator
let onContinue: () -> Void
let onSkip: () -> Void
@State private var showingPicker = false
@State private var showSkipConfirmation: Bool = false
@State private var showingLimitAlert = false
@State private var droppedByLimit = 0
private let maxContacts = 3
private var atLimit: Bool { coordinator.selectedContacts.count >= maxContacts }
@@ -407,29 +406,21 @@ private struct OnboardingContactImportView: View {
: "\(coordinator.selectedContacts.count) Kontakte ausgewählt. Weiter."
)
Button {
showSkipConfirmation = true
} label: {
Text("Überspringen")
.font(.subheadline)
if coordinator.selectedContacts.isEmpty {
Text("Füge mindestens eine Person hinzu, um fortzufahren sonst macht nahbar leider keinen Sinn.")
.font(.caption)
.foregroundStyle(.secondary)
.multilineTextAlignment(.center)
}
.accessibilityLabel("Kontakte überspringen")
.accessibilityHint("Zeigt eine Bestätigungsabfrage.")
}
.padding(.horizontal, 24)
.padding(.vertical, 16)
}
.confirmationDialog(
"Kontakte überspringen?",
isPresented: $showSkipConfirmation,
titleVisibility: .visible
) {
Button("Trotzdem überspringen", role: .destructive, action: onSkip)
Button("Abbrechen", role: .cancel) {}
.alert("Limit erreicht", isPresented: $showingLimitAlert) {
Button("OK", role: .cancel) {}
} message: {
Text("Du kannst Kontakte jederzeit später in der App hinzufügen.")
Text(limitAlertMessage)
}
.overlay(alignment: .center) {
// Invisible trigger finds the hosting UIViewController via
@@ -519,16 +510,34 @@ private struct OnboardingContactImportView: View {
// MARK: Merge helper
private var limitAlertMessage: String {
if droppedByLimit == 1 {
return String(localized: "1 Kontakt wurde nicht hinzugefügt. Im Free-Tier kannst du beim Onboarding bis zu 3 Personen auswählen.")
}
return String.localizedStringWithFormat(
String(localized: "%lld Kontakte wurden nicht hinzugefügt. Im Free-Tier kannst du beim Onboarding bis zu 3 Personen auswählen."),
droppedByLimit
)
}
/// Merges newly picked contacts into the existing selection (no duplicates).
private func mergeContacts(_ contacts: [CNContact]) {
var dropped = 0
for contact in contacts {
guard coordinator.selectedContacts.count < maxContacts else { break }
if coordinator.selectedContacts.count >= maxContacts {
dropped += 1
continue
}
let alreadySelected = coordinator.selectedContacts
.contains { $0.cnIdentifier == contact.identifier }
if !alreadySelected {
coordinator.selectedContacts.append(NahbarContact(from: contact))
}
}
if dropped > 0 {
droppedByLimit = dropped
showingLimitAlert = true
}
}
}