diff --git a/doc/ios-share-extension-git-commit-guide.md b/doc/ios-share-extension-git-commit-guide.md new file mode 100644 index 0000000000..a0652d457f --- /dev/null +++ b/doc/ios-share-extension-git-commit-guide.md @@ -0,0 +1,139 @@ +# iOS Share Extension - Git Commit Guide + +**Date:** 2025-01-27 +**Purpose:** Clarify which Xcode manual changes should be committed to the repository + +## Quick Answer + +**YES, most manual Xcode changes SHOULD be committed.** The only exceptions are user-specific settings that are already gitignored. + +## What Gets Modified (and Should Be Committed) + +When you create the Share Extension target and configure App Groups in Xcode, the following files are modified: + +### 1. `ios/App/App.xcodeproj/project.pbxproj` ✅ **COMMIT THIS** + +This is the main Xcode project file that tracks: +- **New targets** (Share Extension target) +- **File references** (which files belong to which targets) +- **Build settings** (compiler flags, deployment targets, etc.) +- **Build phases** (compile sources, link frameworks, etc.) +- **Capabilities** (App Groups configuration) +- **Target dependencies** + +**This file IS tracked in git** (not in `.gitignore`), so changes should be committed. + +### 2. Entitlements Files ✅ **COMMIT THESE** + +When you enable App Groups capability, Xcode creates/modifies: +- `ios/App/App/App.entitlements` (for main app) +- `ios/App/TimeSafariShareExtension/TimeSafariShareExtension.entitlements` (for extension) + +These files contain the App Group identifiers and should be committed. + +### 3. Share Extension Source Files ✅ **ALREADY COMMITTED** + +The following files are already in the repo: +- `ios/App/TimeSafariShareExtension/ShareViewController.swift` +- `ios/App/TimeSafariShareExtension/Info.plist` +- `ios/App/App/ShareImageBridge.swift` + +These should already be committed (they were created as part of the implementation). + +## What Should NOT Be Committed + +### 1. User-Specific Settings ❌ **ALREADY GITIGNORED** + +These are in `ios/.gitignore`: +- `xcuserdata/` - User-specific scheme selections, breakpoints, etc. +- `*.xcuserstate` - User's current Xcode state + +### 2. Signing Identities ❌ **USER-SPECIFIC** + +While the **App Groups capability** should be committed (it's in `project.pbxproj` and entitlements), your **personal signing identity/team** is user-specific and Xcode handles this automatically per developer. + +## What Happens When You Commit + +When you commit the changes: + +1. **Other developers** who pull the changes will: + - ✅ Get the new Share Extension target automatically + - ✅ Get the App Groups capability configuration + - ✅ Get file references and build settings + - ✅ See the Share Extension in their Xcode project + +2. **They will still need to:** + - Configure their own signing team/identity (Xcode prompts for this) + - Build the project (which may trigger CocoaPods updates) + - But they **won't** need to manually create the target or configure App Groups + +## Step-by-Step: What to Commit + +After completing the Xcode setup steps: + +```bash +# Check what changed +git status + +# You should see: +# - ios/App/App.xcodeproj/project.pbxproj (modified) +# - ios/App/App/App.entitlements (new or modified) +# - ios/App/TimeSafariShareExtension/TimeSafariShareExtension.entitlements (new) +# - Possibly other project-related files + +# Review the changes +git diff ios/App/App.xcodeproj/project.pbxproj + +# Commit the changes +git add ios/App/App.xcodeproj/project.pbxproj +git add ios/App/App/App.entitlements +git add ios/App/TimeSafariShareExtension/TimeSafariShareExtension.entitlements +git commit -m "Add iOS Share Extension target and App Groups configuration" +``` + +## Important Notes + +### Merge Conflicts in project.pbxproj + +The `project.pbxproj` file can have merge conflicts because: +- It's auto-generated by Xcode +- Multiple developers might modify it +- It uses UUIDs that can conflict + +**If you get merge conflicts:** +1. Open the project in Xcode +2. Xcode will often auto-resolve conflicts +3. Or manually resolve by keeping both sets of changes +4. Test that the project builds + +### Team/Developer IDs + +The `DEVELOPMENT_TEAM` setting in `project.pbxproj` might be user-specific: +- Some teams commit this (if everyone uses the same team) +- Some teams use `.xcconfig` files to override per developer +- Check with your team's practices + +If you see `DEVELOPMENT_TEAM = GM3FS5JQPH;` in the project file, this is already committed, so your team likely commits team IDs. + +## Verification + +After committing, verify that: +1. The Share Extension target appears in Xcode for other developers +2. App Groups capability is configured +3. The project builds successfully +4. No user-specific files were accidentally committed + +## Summary + +| Change Type | Commit? | Reason | +|------------|---------|--------| +| New target creation | ✅ Yes | Modifies `project.pbxproj` | +| App Groups capability | ✅ Yes | Creates/modifies entitlements files | +| File target membership | ✅ Yes | Modifies `project.pbxproj` | +| Build settings | ✅ Yes | Modifies `project.pbxproj` | +| Source files (Swift, plist) | ✅ Yes | Already in repo | +| User scheme selections | ❌ No | In `xcuserdata/` (gitignored) | +| Personal signing identity | ⚠️ Maybe | Depends on team practice | + +**Bottom line:** Commit all the Xcode project configuration changes. Other developers will get the Share Extension target automatically when they pull, and they'll only need to configure their personal signing settings. + diff --git a/doc/ios-share-extension-setup.md b/doc/ios-share-extension-setup.md new file mode 100644 index 0000000000..83f49007a3 --- /dev/null +++ b/doc/ios-share-extension-setup.md @@ -0,0 +1,140 @@ +# iOS Share Extension Setup Instructions + +**Date:** 2025-01-27 +**Purpose:** Step-by-step instructions for setting up the iOS Share Extension in Xcode + +## Prerequisites + +- Xcode installed +- iOS project already set up with Capacitor +- Access to Apple Developer account (for App Groups) + +## Step 1: Create Share Extension Target + +1. Open `ios/App/App.xcodeproj` in Xcode +2. In the Project Navigator, select the **App** project (top-level item) +3. Click the **+** button at the bottom of the Targets list +4. Select **iOS** → **Share Extension** +5. Click **Next** +6. Configure: + - **Product Name:** `TimeSafariShareExtension` + - **Bundle Identifier:** `app.timesafari.shareextension` (must match main app's bundle ID with `.shareextension` suffix) + - **Language:** Swift +7. Click **Finish** + +## Step 2: Configure Share Extension Files + +The following files have been created in `ios/App/TimeSafariShareExtension/`: + +- `ShareViewController.swift` - Main extension logic +- `Info.plist` - Extension configuration + +**Verify these files exist and are added to the Share Extension target.** + +## Step 3: Configure App Groups + +App Groups allow the Share Extension and main app to share data. + +### For Main App Target: + +1. Select the **App** target in Xcode +2. Go to **Signing & Capabilities** tab +3. Click **+ Capability** +4. Select **App Groups** +5. Click **+** to add a new group +6. Enter: `group.app.timesafari` +7. Ensure it's checked/enabled + +### For Share Extension Target: + +1. Select the **TimeSafariShareExtension** target +2. Go to **Signing & Capabilities** tab +3. Click **+ Capability** +4. Select **App Groups** +5. Click **+** to add a new group +6. Enter: `group.app.timesafari` (same as main app) +7. Ensure it's checked/enabled + +**Important:** Both targets must use the **exact same** App Group identifier. + +## Step 4: Configure Share Extension Info.plist + +The `Info.plist` file should already be configured, but verify: + +1. Select `TimeSafariShareExtension/Info.plist` in Xcode +2. Ensure it contains: + - `NSExtensionPointIdentifier` = `com.apple.share-services` + - `NSExtensionPrincipalClass` = `$(PRODUCT_MODULE_NAME).ShareViewController` + - `NSExtensionActivationSupportsImageWithMaxCount` = `1` + +## Step 5: Add ShareImageBridge to Main App + +1. The file `ios/App/App/ShareImageBridge.swift` has been created +2. Ensure it's added to the **App** target (not the Share Extension target) +3. In Xcode, select the file and check the **Target Membership** in the File Inspector + +## Step 6: Build and Test + +1. Select the **App** scheme (not the Share Extension scheme) +2. Build and run on a device or simulator +3. Open Photos app +4. Select an image +5. Tap **Share** button +6. Look for **TimeSafari Share** in the share sheet +7. Select it +8. The app should open and navigate to the shared photo view + +## Step 7: Troubleshooting + +### Share Extension doesn't appear in share sheet + +- Verify the Share Extension target builds successfully +- Check that `Info.plist` is correctly configured +- Ensure the extension's bundle identifier follows the pattern: `{main-app-bundle-id}.shareextension` +- Clean build folder (Product → Clean Build Folder) + +### App Group access fails + +- Verify both targets have the same App Group identifier +- Check that App Groups capability is enabled for both targets +- Ensure you're signed in with a valid Apple Developer account +- For development, you may need to enable App Groups in your Apple Developer account + +### Shared image not appearing + +- Check Xcode console for errors +- Verify `ShareViewController.swift` is correctly implemented +- Ensure the deep link `timesafari://shared-photo` is being handled +- Check that the native bridge method is being called + +### Build errors + +- Ensure Swift version matches between targets +- Check that all required frameworks are linked +- Verify deployment targets match between main app and extension + +## Step 8: Native Bridge Implementation (TODO) + +Currently, the JavaScript code needs a way to call the native `getSharedImageData()` method. This requires one of: + +1. **Option A:** Create a minimal Capacitor plugin +2. **Option B:** Use Capacitor's existing bridge mechanisms +3. **Option C:** Expose the method via a custom URL scheme parameter + +The current implementation in `main.capacitor.ts` has a placeholder that needs to be completed. + +## Next Steps + +After the Share Extension is set up and working: + +1. Complete the native bridge implementation to read from App Group +2. Test end-to-end flow: Share image → Extension stores → App reads → Displays +3. Implement Android version +4. Add error handling and edge cases + +## References + +- [Apple Share Extensions Documentation](https://developer.apple.com/documentation/social) +- [App Groups Documentation](https://developer.apple.com/documentation/xcode/configuring-app-groups) +- [Capacitor Native Bridge](https://capacitorjs.com/docs/guides/building-plugins) + diff --git a/doc/ios-share-implementation-status.md b/doc/ios-share-implementation-status.md new file mode 100644 index 0000000000..2b182b7bd9 --- /dev/null +++ b/doc/ios-share-implementation-status.md @@ -0,0 +1,93 @@ +# iOS Share Extension Implementation Status + +**Date:** 2025-01-27 +**Status:** In Progress - Native Code Complete, Bridge Pending + +## Completed + +✅ **Share Extension Files Created:** +- `ios/App/TimeSafariShareExtension/ShareViewController.swift` - Handles image sharing +- `ios/App/TimeSafariShareExtension/Info.plist` - Extension configuration + +✅ **Native Bridge Created:** +- `ios/App/App/ShareImageBridge.swift` - Native method to read from App Group + +✅ **JavaScript Integration Started:** +- `src/services/nativeShareHandler.ts` - Service to handle native shared images +- `src/main.capacitor.ts` - Updated to check for native shared images on deep link + +✅ **Documentation:** +- `doc/native-share-target-implementation.md` - Complete implementation guide +- `doc/ios-share-extension-setup.md` - Xcode setup instructions + +## Pending + +⚠️ **Xcode Configuration (Manual Steps Required):** +1. Create Share Extension target in Xcode +2. Configure App Groups for both main app and extension +3. Add ShareImageBridge.swift to App target +4. Build and test + +⚠️ **JavaScript-Native Bridge:** +The current implementation has a placeholder for calling the native `ShareImageBridge.getSharedImageData()` method from JavaScript. This needs to be completed using one of: + +**Option A: Minimal Capacitor Plugin** (Recommended for Option 1) +- Create a small plugin that exposes the method +- Clean and maintainable +- Follows Capacitor patterns + +**Option B: Direct Bridge Call** +- Use Capacitor's executePlugin or similar mechanism +- Requires understanding Capacitor's internal bridge +- Less maintainable + +**Option C: AppDelegate Integration** +- Have AppDelegate check on launch and expose via a different mechanism +- Workaround approach +- Less clean but functional + +## Next Steps + +1. **Complete Xcode Setup:** + - Follow `doc/ios-share-extension-setup.md` + - Create Share Extension target + - Configure App Groups + - Build and verify extension appears in share sheet + +2. **Implement JavaScript-Native Bridge:** + - Choose one of the options above + - Complete the `checkAndStoreNativeSharedImage()` function in `main.capacitor.ts` + - Test end-to-end flow + +3. **Testing:** + - Share image from Photos app + - Verify Share Extension appears + - Verify app opens and displays shared image + - Test "Record Gift" and "Save as Profile" flows + +## Current Flow + +1. ✅ User shares image → Share Extension receives +2. ✅ Share Extension converts to base64 +3. ✅ Share Extension stores in App Group UserDefaults +4. ✅ Share Extension opens app with `timesafari://shared-photo?fileName=...` +5. ⚠️ App receives deep link (handled) +6. ⚠️ App checks App Group UserDefaults (bridge needed) +7. ⚠️ App stores in temp database (pending bridge) +8. ✅ SharedPhotoView reads from temp database (already works) + +## Code Locations + +- **Share Extension:** `ios/App/TimeSafariShareExtension/` +- **Native Bridge:** `ios/App/App/ShareImageBridge.swift` +- **JavaScript Handler:** `src/services/nativeShareHandler.ts` +- **Deep Link Integration:** `src/main.capacitor.ts` +- **View Component:** `src/views/SharedPhotoView.vue` (already complete) + +## Notes + +- The Share Extension code is complete and ready to use +- The main missing piece is the JavaScript-to-native bridge +- Once the bridge is complete, the entire flow should work end-to-end +- The existing `SharedPhotoView.vue` doesn't need changes - it already handles images from temp storage + diff --git a/doc/native-share-target-implementation.md b/doc/native-share-target-implementation.md new file mode 100644 index 0000000000..7a30979c37 --- /dev/null +++ b/doc/native-share-target-implementation.md @@ -0,0 +1,507 @@ +# Native Share Target Implementation Guide + +**Date:** 2025-01-27 +**Purpose:** Enable TimeSafari native iOS and Android apps to receive shared images from other apps + +## Current State + +The app currently supports **PWA/web share target** functionality: +- Service worker intercepts POST to `/share-target` +- Images stored in temp database as base64 +- `SharedPhotoView.vue` processes and displays shared images + +**This does NOT work for native iOS/Android builds** because: +- Service workers don't run in native app contexts +- Native platforms use different sharing mechanisms (Share Extensions on iOS, Intent Filters on Android) + +## Required Changes + +### 1. iOS Implementation + +#### 1.1 Create Share Extension Target + +1. Open `ios/App/App.xcodeproj` in Xcode +2. File → New → Target +3. Select "Share Extension" template +4. Name it "TimeSafariShareExtension" +5. Bundle Identifier: `app.timesafari.shareextension` +6. Language: Swift + +#### 1.2 Configure Share Extension Info.plist + +Add to `ios/App/TimeSafariShareExtension/Info.plist`: + +```xml +NSExtension + + NSExtensionPointIdentifier + com.apple.share-services + NSExtensionPrincipalClass + $(PRODUCT_MODULE_NAME).ShareViewController + NSExtensionActivationRule + + NSExtensionActivationSupportsImageWithMaxCount + 1 + NSExtensionActivationSupportsFileWithMaxCount + 1 + + +``` + +#### 1.3 Implement ShareViewController + +Create `ios/App/TimeSafariShareExtension/ShareViewController.swift`: + +```swift +import UIKit +import Social +import MobileCoreServices +import Capacitor + +class ShareViewController: SLComposeServiceViewController { + + override func viewDidLoad() { + super.viewDidLoad() + self.title = "Share to TimeSafari" + } + + override func isContentValid() -> Bool { + return true + } + + override func didSelectPost() { + guard let extensionItem = extensionContext?.inputItems.first as? NSExtensionItem, + let itemProvider = extensionItem.attachments?.first else { + self.extensionContext?.completeRequest(returningItems: nil, completionHandler: nil) + return + } + + // Handle image sharing + if itemProvider.hasItemConformingToTypeIdentifier(kUTTypeImage as String) { + itemProvider.loadItem(forTypeIdentifier: kUTTypeImage as String, options: nil) { [weak self] (item, error) in + guard let self = self else { return } + + if let url = item as? URL { + // Handle file URL + self.handleSharedImage(url: url) + } else if let image = item as? UIImage { + // Handle UIImage directly + self.handleSharedImage(image: image) + } else if let data = item as? Data { + // Handle image data + self.handleSharedImage(data: data) + } + } + } + } + + private func handleSharedImage(url: URL? = nil, image: UIImage? = nil, data: Data? = nil) { + var imageData: Data? + var fileName: String? + + if let url = url { + imageData = try? Data(contentsOf: url) + fileName = url.lastPathComponent + } else if let image = image { + imageData = image.jpegData(compressionQuality: 0.8) + fileName = "shared-image.jpg" + } else if let data = data { + imageData = data + fileName = "shared-image.jpg" + } + + guard let imageData = imageData else { + self.extensionContext?.completeRequest(returningItems: nil, completionHandler: nil) + return + } + + // Convert to base64 + let base64String = imageData.base64EncodedString() + + // Store in shared UserDefaults (accessible by main app) + let userDefaults = UserDefaults(suiteName: "group.app.timesafari") + userDefaults?.set(base64String, forKey: "sharedPhotoBase64") + userDefaults?.set(fileName ?? "shared-image.jpg", forKey: "sharedPhotoFileName") + userDefaults?.synchronize() + + // Open main app with deep link + let url = URL(string: "timesafari://shared-photo?fileName=\(fileName?.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed) ?? "shared-image.jpg")")! + var responder = self as UIResponder? + while responder != nil { + if let application = responder as? UIApplication { + application.open(url, options: [:], completionHandler: nil) + break + } + responder = responder?.next + } + + // Close share extension + self.extensionContext?.completeRequest(returningItems: nil, completionHandler: nil) + } + + override func configurationItems() -> [Any]! { + return [] + } +} +``` + +#### 1.4 Configure App Groups + +1. In Xcode, select main app target → Signing & Capabilities +2. Add "App Groups" capability +3. Create group: `group.app.timesafari` +4. Repeat for Share Extension target with same group name + +#### 1.5 Update Main App to Read from App Group + +The main app needs to check for shared images on launch. This should be added to `AppDelegate.swift` or handled in JavaScript. + +### 2. Android Implementation + +#### 2.1 Update AndroidManifest.xml + +Add intent filter to `MainActivity` in `android/app/src/main/AndroidManifest.xml`: + +```xml + + + ... existing intent filters ... + + + + + + + + + + + + + + + +``` + +#### 2.2 Handle Intent in MainActivity + +Update `android/app/src/main/java/app/timesafari/MainActivity.java`: + +```java +package app.timesafari; + +import android.content.Intent; +import android.net.Uri; +import android.os.Bundle; +import android.util.Base64; +import android.util.Log; +import com.getcapacitor.BridgeActivity; +import com.getcapacitor.Plugin; +import java.io.InputStream; +import java.io.ByteArrayOutputStream; + +public class MainActivity extends BridgeActivity { + private static final String TAG = "MainActivity"; + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + handleShareIntent(getIntent()); + } + + @Override + protected void onNewIntent(Intent intent) { + super.onNewIntent(intent); + setIntent(intent); + handleShareIntent(intent); + } + + private void handleShareIntent(Intent intent) { + if (intent == null) return; + + String action = intent.getAction(); + String type = intent.getType(); + + if (Intent.ACTION_SEND.equals(action) && type != null && type.startsWith("image/")) { + Uri imageUri = intent.getParcelableExtra(Intent.EXTRA_STREAM); + if (imageUri != null) { + handleSharedImage(imageUri, intent.getStringExtra(Intent.EXTRA_TEXT)); + } + } else if (Intent.ACTION_SEND_MULTIPLE.equals(action) && type != null && type.startsWith("image/")) { + // Handle multiple images (optional - for now just take first) + java.util.ArrayList imageUris = intent.getParcelableArrayListExtra(Intent.EXTRA_STREAM); + if (imageUris != null && !imageUris.isEmpty()) { + handleSharedImage(imageUris.get(0), null); + } + } + } + + private void handleSharedImage(Uri imageUri, String fileName) { + try { + // Read image data + InputStream inputStream = getContentResolver().openInputStream(imageUri); + if (inputStream == null) { + Log.e(TAG, "Failed to open input stream for shared image"); + return; + } + + ByteArrayOutputStream buffer = new ByteArrayOutputStream(); + byte[] data = new byte[8192]; + int nRead; + while ((nRead = inputStream.read(data, 0, data.length)) != -1) { + buffer.write(data, 0, nRead); + } + buffer.flush(); + byte[] imageBytes = buffer.toByteArray(); + + // Convert to base64 + String base64String = Base64.encodeToString(imageBytes, Base64.NO_WRAP); + + // Extract filename from URI or use default + String actualFileName = fileName; + if (actualFileName == null || actualFileName.isEmpty()) { + String path = imageUri.getPath(); + if (path != null) { + int lastSlash = path.lastIndexOf('/'); + if (lastSlash >= 0 && lastSlash < path.length() - 1) { + actualFileName = path.substring(lastSlash + 1); + } + } + if (actualFileName == null || actualFileName.isEmpty()) { + actualFileName = "shared-image.jpg"; + } + } + + // Store in SharedPreferences (accessible by JavaScript via Capacitor) + android.content.SharedPreferences prefs = getSharedPreferences("TimeSafariShared", MODE_PRIVATE); + android.content.SharedPreferences.Editor editor = prefs.edit(); + editor.putString("sharedPhotoBase64", base64String); + editor.putString("sharedPhotoFileName", actualFileName); + editor.apply(); + + // Trigger JavaScript event or navigate to shared-photo route + // This will be handled by JavaScript checking for shared data on app launch + Log.d(TAG, "Shared image stored, filename: " + actualFileName); + + } catch (Exception e) { + Log.e(TAG, "Error handling shared image", e); + } + } +} +``` + +#### 2.3 Add Required Permissions + +Ensure `AndroidManifest.xml` has: + +```xml + + +``` + +### 3. JavaScript Layer Updates + +#### 3.1 Create Native Share Handler + +Create `src/services/nativeShareHandler.ts`: + +```typescript +/** + * Native Share Handler + * Handles shared images from native iOS and Android platforms + */ + +import { Capacitor } from "@capacitor/core"; +import { App } from "@capacitor/app"; +import { Filesystem, Directory, Encoding } from "@capacitor/filesystem"; +import { logger } from "../utils/logger"; +import { SHARED_PHOTO_BASE64_KEY } from "../libs/util"; +import { PlatformServiceMixin } from "@/utils/PlatformServiceMixin"; + +/** + * Check for shared images from native platforms and store in temp database + */ +export async function checkForNativeSharedImage( + platformService: InstanceType +): Promise { + if (!Capacitor.isNativePlatform()) { + return false; + } + + try { + if (Capacitor.getPlatform() === "ios") { + return await checkIOSSharedImage(platformService); + } else if (Capacitor.getPlatform() === "android") { + return await checkAndroidSharedImage(platformService); + } + } catch (error) { + logger.error("Error checking for native shared image:", error); + } + + return false; +} + +/** + * Check for shared image on iOS (from App Group UserDefaults) + */ +async function checkIOSSharedImage( + platformService: InstanceType +): Promise { + // iOS uses App Groups to share data between extension and main app + // We need to use a Capacitor plugin or native code to read from App Group + // For now, this is a placeholder - requires native plugin implementation + + // Option 1: Use Capacitor plugin to read from App Group + // Option 2: Use native code bridge + + logger.debug("Checking for iOS shared image (not yet implemented)"); + return false; +} + +/** + * Check for shared image on Android (from SharedPreferences) + */ +async function checkAndroidSharedImage( + platformService: InstanceType +): Promise { + // Android stores in SharedPreferences + // We need a Capacitor plugin to read from SharedPreferences + // For now, this is a placeholder - requires native plugin implementation + + logger.debug("Checking for Android shared image (not yet implemented)"); + return false; +} + +/** + * Store shared image in temp database + */ +async function storeSharedImage( + base64Data: string, + fileName: string, + platformService: InstanceType +): Promise { + try { + const existing = await platformService.$getTemp(SHARED_PHOTO_BASE64_KEY); + + if (existing) { + await platformService.$updateEntity( + "temp", + { blobB64: base64Data }, + "id = ?", + [SHARED_PHOTO_BASE64_KEY] + ); + } else { + await platformService.$insertEntity( + "temp", + { id: SHARED_PHOTO_BASE64_KEY, blobB64: base64Data }, + ["id", "blobB64"] + ); + } + + logger.debug("Stored shared image in temp database"); + } catch (error) { + logger.error("Error storing shared image:", error); + throw error; + } +} +``` + +#### 3.2 Update main.capacitor.ts + +Add check for shared images on app launch: + +```typescript +// In main.capacitor.ts, after app mount: + +import { checkForNativeSharedImage } from "./services/nativeShareHandler"; + +// Check for shared images when app becomes active +App.addListener("appStateChange", async (state) => { + if (state.isActive) { + // Check for native shared images + const hasSharedImage = await checkForNativeSharedImage(/* platformService */); + if (hasSharedImage) { + // Navigate to shared-photo view + await router.push({ + name: "shared-photo", + query: { source: "native" } + }); + } + } +}); + +// Also check on initial launch +App.getLaunchUrl().then((result) => { + if (result?.url) { + // Handle deep link + } else { + // Check for shared image + checkForNativeSharedImage(/* platformService */).then((hasShared) => { + if (hasShared) { + router.push({ name: "shared-photo", query: { source: "native" } }); + } + }); + } +}); +``` + +#### 3.3 Update SharedPhotoView.vue + +The existing `SharedPhotoView.vue` should work as-is, but we may want to add detection for native vs web sources. + +### 4. Alternative Approach: Capacitor Plugin + +Instead of implementing native code directly, consider creating a Capacitor plugin: + +1. **Create plugin**: `@capacitor-community/share-target` or custom plugin +2. **Plugin methods**: + - `checkForSharedImage()`: Returns shared image data if available + - `clearSharedImage()`: Clears shared image data after processing + +This would be cleaner and more maintainable. + +### 5. Testing Checklist + +- [ ] Test sharing image from Photos app on iOS +- [ ] Test sharing image from Gallery app on Android +- [ ] Test sharing from other apps (Safari, Chrome, etc.) +- [ ] Verify image appears in SharedPhotoView +- [ ] Test "Record Gift" flow with shared image +- [ ] Test "Save as Profile" flow with shared image +- [ ] Test cancel flow +- [ ] Verify temp storage cleanup +- [ ] Test app launch with shared image pending +- [ ] Test app already running when image is shared + +### 6. Implementation Priority + +**Phase 1: Android (Simpler)** +1. Update AndroidManifest.xml +2. Implement MainActivity intent handling +3. Create JavaScript handler +4. Test end-to-end + +**Phase 2: iOS (More Complex)** +1. Create Share Extension target +2. Implement ShareViewController +3. Configure App Groups +4. Create JavaScript handler +5. Test end-to-end + +### 7. Notes + +- **App Groups (iOS)**: Required for sharing data between Share Extension and main app +- **SharedPreferences (Android)**: Standard way to share data between app components +- **Base64 Encoding**: Both platforms convert images to base64 for JavaScript compatibility +- **File Size Limits**: Consider large image handling and memory management +- **Permissions**: Android 13+ requires `READ_MEDIA_IMAGES` instead of `READ_EXTERNAL_STORAGE` + +### 8. References + +- [iOS Share Extensions](https://developer.apple.com/documentation/social) +- [Android Share Targets](https://developer.android.com/training/sharing/receive) +- [Capacitor App Plugin](https://capacitorjs.com/docs/apis/app) +- [Capacitor Native Bridge](https://capacitorjs.com/docs/guides/building-plugins) + diff --git a/doc/xcode-26-cocoapods-workaround.md b/doc/xcode-26-cocoapods-workaround.md new file mode 100644 index 0000000000..2db40f4411 --- /dev/null +++ b/doc/xcode-26-cocoapods-workaround.md @@ -0,0 +1,76 @@ +# Xcode 26 / CocoaPods Compatibility Workaround + +**Date:** 2025-01-27 +**Issue:** CocoaPods `xcodeproj` gem (1.27.0) doesn't support Xcode 26's project format version 70 + +## The Problem + +Xcode 26.1.1 uses project format version 70, but the `xcodeproj` gem (1.27.0) only supports up to version 56. This causes CocoaPods to fail with: + +``` +ArgumentError - [Xcodeproj] Unable to find compatibility version string for object version `70`. +``` + +## Solutions + +### Option 1: Temporarily Downgrade Project Format (Recommended for Now) + +**Before running `pod install` or `npm run build:ios`:** + +1. Edit `ios/App/App.xcodeproj/project.pbxproj` +2. Change line 6 from: `objectVersion = 70;` to: `objectVersion = 56;` +3. Run your build/sync command +4. Change it back to: `objectVersion = 70;` (Xcode will likely change it back automatically) + +**Warning:** Xcode may automatically upgrade the format back to 70 when you open the project. This is okay - just repeat the process when needed. + +### Option 2: Wait for xcodeproj Update + +The `xcodeproj` gem maintainers will eventually release a version that supports format 70. You can: +- Check for updates: `bundle update xcodeproj` +- Monitor: https://github.com/CocoaPods/Xcodeproj/issues + +### Option 3: Use Xcode Directly (Bypass CocoaPods for Now) + +Since the Share Extension is already set up: +1. Open the project in Xcode +2. Build directly from Xcode (Product → Build) +3. Skip `npm run build:ios` for now +4. Test the Share Extension functionality + +### Option 4: Automated Workaround (Integrated into Build Script) ✅ + +The workaround is now **automatically integrated** into `scripts/build-ios.sh`. When you run: + +```bash +npm run build:ios +``` + +The build script will: +1. Automatically detect if the project format is version 70 +2. Temporarily downgrade to version 56 +3. Run `pod install` +4. Restore to version 70 +5. Continue with the build + +**No manual steps required!** The workaround is transparent and only applies when needed. + +To remove the workaround in the future: +1. Check if `xcodeproj` gem supports format 70: `bundle exec gem list xcodeproj` +2. Test if `pod install` works without the workaround +3. If it works, remove the `run_pod_install_with_workaround()` function from `scripts/build-ios.sh` +4. Replace it with a simple `pod install` call + +## Current Status + +- ✅ Share Extension target exists +- ✅ Share Extension files are in place +- ✅ Workaround integrated into build script +- ✅ `npm run build:ios` works automatically + +## Recommendation + +**Use `npm run build:ios`** - the workaround is handled automatically. No manual intervention needed. + +Once `xcodeproj` is updated to support format 70, the workaround can be removed from the build script. + diff --git a/ios/App/App.xcodeproj/project.pbxproj b/ios/App/App.xcodeproj/project.pbxproj index ce46d1f6c5..7214d4a44c 100644 --- a/ios/App/App.xcodeproj/project.pbxproj +++ b/ios/App/App.xcodeproj/project.pbxproj @@ -3,7 +3,7 @@ archiveVersion = 1; classes = { }; - objectVersion = 54; + objectVersion = 70; objects = { /* Begin PBXBuildFile section */ @@ -15,8 +15,35 @@ 504EC3121FED79650016851F /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 504EC3101FED79650016851F /* LaunchScreen.storyboard */; }; 50B271D11FEDC1A000F3C39B /* public in Resources */ = {isa = PBXBuildFile; fileRef = 50B271D01FEDC1A000F3C39B /* public */; }; 97EF2DC6FD76C3643D680B8D /* Pods_App.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 90DCAFB4D8948F7A50C13800 /* Pods_App.framework */; }; + C86585DF2ED456DE00824752 /* TimeSafariShareExtension.appex in Embed Foundation Extensions */ = {isa = PBXBuildFile; fileRef = C86585D52ED456DE00824752 /* TimeSafariShareExtension.appex */; settings = {ATTRIBUTES = (RemoveHeadersOnCopy, ); }; }; + C86585E82ED45A3E00824752 /* ShareImageBridge.swift in Sources */ = {isa = PBXBuildFile; fileRef = C86585E72ED45A3D00824752 /* ShareImageBridge.swift */; }; + C8D7E2CC2ED46A3B00DD738D /* ShareImagePlugin.swift in Sources */ = {isa = PBXBuildFile; fileRef = C8D7E2CB2ED46A3B00DD738D /* ShareImagePlugin.swift */; }; /* End PBXBuildFile section */ +/* Begin PBXContainerItemProxy section */ + C86585DD2ED456DE00824752 /* PBXContainerItemProxy */ = { + isa = PBXContainerItemProxy; + containerPortal = 504EC2FC1FED79650016851F /* Project object */; + proxyType = 1; + remoteGlobalIDString = C86585D42ED456DE00824752; + remoteInfo = TimeSafariShareExtension; + }; +/* End PBXContainerItemProxy section */ + +/* Begin PBXCopyFilesBuildPhase section */ + C86585E02ED456DE00824752 /* Embed Foundation Extensions */ = { + isa = PBXCopyFilesBuildPhase; + buildActionMask = 2147483647; + dstPath = ""; + dstSubfolderSpec = 13; + files = ( + C86585DF2ED456DE00824752 /* TimeSafariShareExtension.appex in Embed Foundation Extensions */, + ); + name = "Embed Foundation Extensions"; + runOnlyForDeploymentPostprocessing = 0; + }; +/* End PBXCopyFilesBuildPhase section */ + /* Begin PBXFileReference section */ 2FAD9762203C412B000D30F8 /* config.xml */ = {isa = PBXFileReference; lastKnownFileType = text.xml; path = config.xml; sourceTree = ""; }; 50379B222058CBB4000EE86E /* capacitor.config.json */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.json; path = capacitor.config.json; sourceTree = ""; }; @@ -28,10 +55,28 @@ 504EC3131FED79650016851F /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; 50B271D01FEDC1A000F3C39B /* public */ = {isa = PBXFileReference; lastKnownFileType = folder; path = public; sourceTree = ""; }; 90DCAFB4D8948F7A50C13800 /* Pods_App.framework */ = {isa = PBXFileReference; explicitFileType = wrapper.framework; includeInIndex = 0; path = Pods_App.framework; sourceTree = BUILT_PRODUCTS_DIR; }; + C86585D52ED456DE00824752 /* TimeSafariShareExtension.appex */ = {isa = PBXFileReference; explicitFileType = "wrapper.app-extension"; includeInIndex = 0; path = TimeSafariShareExtension.appex; sourceTree = BUILT_PRODUCTS_DIR; }; + C86585E52ED4577F00824752 /* App.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = App.entitlements; sourceTree = ""; }; + C86585E72ED45A3D00824752 /* ShareImageBridge.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ShareImageBridge.swift; sourceTree = ""; }; + C8D7E2CB2ED46A3B00DD738D /* ShareImagePlugin.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ShareImagePlugin.swift; sourceTree = ""; }; E2E9297D5D02C549106C77F9 /* Pods-App.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-App.release.xcconfig"; path = "Target Support Files/Pods-App/Pods-App.release.xcconfig"; sourceTree = ""; }; EAEC6436E595F7CD3A1C9E96 /* Pods-App.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-App.debug.xcconfig"; path = "Target Support Files/Pods-App/Pods-App.debug.xcconfig"; sourceTree = ""; }; /* End PBXFileReference section */ +/* Begin PBXFileSystemSynchronizedBuildFileExceptionSet section */ + C86585E32ED456DE00824752 /* PBXFileSystemSynchronizedBuildFileExceptionSet */ = { + isa = PBXFileSystemSynchronizedBuildFileExceptionSet; + membershipExceptions = ( + Info.plist, + ); + target = C86585D42ED456DE00824752 /* TimeSafariShareExtension */; + }; +/* End PBXFileSystemSynchronizedBuildFileExceptionSet section */ + +/* Begin PBXFileSystemSynchronizedRootGroup section */ + C86585D62ED456DE00824752 /* TimeSafariShareExtension */ = {isa = PBXFileSystemSynchronizedRootGroup; exceptions = (C86585E32ED456DE00824752 /* PBXFileSystemSynchronizedBuildFileExceptionSet */, ); explicitFileTypes = {}; explicitFolders = (); path = TimeSafariShareExtension; sourceTree = ""; }; +/* End PBXFileSystemSynchronizedRootGroup section */ + /* Begin PBXFrameworksBuildPhase section */ 504EC3011FED79650016851F /* Frameworks */ = { isa = PBXFrameworksBuildPhase; @@ -41,6 +86,13 @@ ); runOnlyForDeploymentPostprocessing = 0; }; + C86585D22ED456DE00824752 /* Frameworks */ = { + isa = PBXFrameworksBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; /* End PBXFrameworksBuildPhase section */ /* Begin PBXGroup section */ @@ -56,6 +108,7 @@ isa = PBXGroup; children = ( 504EC3061FED79650016851F /* App */, + C86585D62ED456DE00824752 /* TimeSafariShareExtension */, 504EC3051FED79650016851F /* Products */, BA325FFCDCE8D334E5C7AEBE /* Pods */, 4B546315E668C7A13939F417 /* Frameworks */, @@ -66,6 +119,7 @@ isa = PBXGroup; children = ( 504EC3041FED79650016851F /* App.app */, + C86585D52ED456DE00824752 /* TimeSafariShareExtension.appex */, ); name = Products; sourceTree = ""; @@ -73,6 +127,9 @@ 504EC3061FED79650016851F /* App */ = { isa = PBXGroup; children = ( + C8D7E2CB2ED46A3B00DD738D /* ShareImagePlugin.swift */, + C86585E72ED45A3D00824752 /* ShareImageBridge.swift */, + C86585E52ED4577F00824752 /* App.entitlements */, 50379B222058CBB4000EE86E /* capacitor.config.json */, 504EC3071FED79650016851F /* AppDelegate.swift */, 504EC30B1FED79650016851F /* Main.storyboard */, @@ -108,16 +165,40 @@ 012076E8FFE4BF260A79B034 /* Fix Privacy Manifest */, 3525031ED1C96EF4CF6E9959 /* [CP] Embed Pods Frameworks */, 96A7EF592DF3366D00084D51 /* Fix Privacy Manifest */, + C86585E02ED456DE00824752 /* Embed Foundation Extensions */, ); buildRules = ( ); dependencies = ( + C86585DE2ED456DE00824752 /* PBXTargetDependency */, ); name = App; productName = App; productReference = 504EC3041FED79650016851F /* App.app */; productType = "com.apple.product-type.application"; }; + C86585D42ED456DE00824752 /* TimeSafariShareExtension */ = { + isa = PBXNativeTarget; + buildConfigurationList = C86585E42ED456DE00824752 /* Build configuration list for PBXNativeTarget "TimeSafariShareExtension" */; + buildPhases = ( + C86585D12ED456DE00824752 /* Sources */, + C86585D22ED456DE00824752 /* Frameworks */, + C86585D32ED456DE00824752 /* Resources */, + ); + buildRules = ( + ); + dependencies = ( + ); + fileSystemSynchronizedGroups = ( + C86585D62ED456DE00824752 /* TimeSafariShareExtension */, + ); + name = TimeSafariShareExtension; + packageProductDependencies = ( + ); + productName = TimeSafariShareExtension; + productReference = C86585D52ED456DE00824752 /* TimeSafariShareExtension.appex */; + productType = "com.apple.product-type.app-extension"; + }; /* End PBXNativeTarget section */ /* Begin PBXProject section */ @@ -125,7 +206,7 @@ isa = PBXProject; attributes = { BuildIndependentTargetsInParallel = YES; - LastSwiftUpdateCheck = 920; + LastSwiftUpdateCheck = 2610; LastUpgradeCheck = 1630; TargetAttributes = { 504EC3031FED79650016851F = { @@ -133,6 +214,9 @@ LastSwiftMigration = 1100; ProvisioningStyle = Automatic; }; + C86585D42ED456DE00824752 = { + CreatedOnToolsVersion = 26.1.1; + }; }; }; buildConfigurationList = 504EC2FF1FED79650016851F /* Build configuration list for PBXProject "App" */; @@ -149,6 +233,7 @@ projectRoot = ""; targets = ( 504EC3031FED79650016851F /* App */, + C86585D42ED456DE00824752 /* TimeSafariShareExtension */, ); }; /* End PBXProject section */ @@ -167,6 +252,13 @@ ); runOnlyForDeploymentPostprocessing = 0; }; + C86585D32ED456DE00824752 /* Resources */ = { + isa = PBXResourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; /* End PBXResourcesBuildPhase section */ /* Begin PBXShellScriptBuildPhase section */ @@ -253,12 +345,29 @@ isa = PBXSourcesBuildPhase; buildActionMask = 2147483647; files = ( + C8D7E2CC2ED46A3B00DD738D /* ShareImagePlugin.swift in Sources */, + C86585E82ED45A3E00824752 /* ShareImageBridge.swift in Sources */, 504EC3081FED79650016851F /* AppDelegate.swift in Sources */, ); runOnlyForDeploymentPostprocessing = 0; }; + C86585D12ED456DE00824752 /* Sources */ = { + isa = PBXSourcesBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + runOnlyForDeploymentPostprocessing = 0; + }; /* End PBXSourcesBuildPhase section */ +/* Begin PBXTargetDependency section */ + C86585DE2ED456DE00824752 /* PBXTargetDependency */ = { + isa = PBXTargetDependency; + target = C86585D42ED456DE00824752 /* TimeSafariShareExtension */; + targetProxy = C86585DD2ED456DE00824752 /* PBXContainerItemProxy */; + }; +/* End PBXTargetDependency section */ + /* Begin PBXVariantGroup section */ 504EC30B1FED79650016851F /* Main.storyboard */ = { isa = PBXVariantGroup; @@ -402,6 +511,7 @@ baseConfigurationReference = EAEC6436E595F7CD3A1C9E96 /* Pods-App.debug.xcconfig */; buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; + CODE_SIGN_ENTITLEMENTS = App/App.entitlements; CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 48; DEVELOPMENT_TEAM = GM3FS5JQPH; @@ -429,6 +539,7 @@ baseConfigurationReference = E2E9297D5D02C549106C77F9 /* Pods-App.release.xcconfig */; buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; + CODE_SIGN_ENTITLEMENTS = App/App.entitlements; CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 48; DEVELOPMENT_TEAM = GM3FS5JQPH; @@ -450,6 +561,80 @@ }; name = Release; }; + C86585E12ED456DE00824752 /* Debug */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_OBJC_WEAK = YES; + CODE_SIGN_ENTITLEMENTS = TimeSafariShareExtension/TimeSafariShareExtension.entitlements; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = GM3FS5JQPH; + GCC_C_LANGUAGE_STANDARD = gnu17; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = TimeSafariShareExtension/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = TimeSafariShareExtension; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 26.1; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MARKETING_VERSION = 1.0; + MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE; + MTL_FAST_MATH = YES; + PRODUCT_BUNDLE_IDENTIFIER = app.timesafari.TimeSafariShareExtension; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + STRING_CATALOG_GENERATE_SYMBOLS = YES; + SWIFT_ACTIVE_COMPILATION_CONDITIONS = "DEBUG $(inherited)"; + SWIFT_APPROACHABLE_CONCURRENCY = YES; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = Debug; + }; + C86585E22ED456DE00824752 /* Release */ = { + isa = XCBuildConfiguration; + buildSettings = { + ASSETCATALOG_COMPILER_GENERATE_SWIFT_ASSET_SYMBOL_EXTENSIONS = YES; + CLANG_CXX_LANGUAGE_STANDARD = "gnu++20"; + CLANG_ENABLE_OBJC_WEAK = YES; + CODE_SIGN_ENTITLEMENTS = TimeSafariShareExtension/TimeSafariShareExtension.entitlements; + CODE_SIGN_STYLE = Automatic; + CURRENT_PROJECT_VERSION = 1; + DEVELOPMENT_TEAM = GM3FS5JQPH; + GCC_C_LANGUAGE_STANDARD = gnu17; + GENERATE_INFOPLIST_FILE = YES; + INFOPLIST_FILE = TimeSafariShareExtension/Info.plist; + INFOPLIST_KEY_CFBundleDisplayName = TimeSafariShareExtension; + INFOPLIST_KEY_NSHumanReadableCopyright = ""; + IPHONEOS_DEPLOYMENT_TARGET = 26.1; + LD_RUNPATH_SEARCH_PATHS = ( + "$(inherited)", + "@executable_path/Frameworks", + "@executable_path/../../Frameworks", + ); + LOCALIZATION_PREFERS_STRING_CATALOGS = YES; + MARKETING_VERSION = 1.0; + MTL_FAST_MATH = YES; + PRODUCT_BUNDLE_IDENTIFIER = app.timesafari.TimeSafariShareExtension; + PRODUCT_NAME = "$(TARGET_NAME)"; + SKIP_INSTALL = YES; + STRING_CATALOG_GENERATE_SYMBOLS = YES; + SWIFT_APPROACHABLE_CONCURRENCY = YES; + SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_UPCOMING_FEATURE_MEMBER_IMPORT_VISIBILITY = YES; + SWIFT_VERSION = 5.0; + TARGETED_DEVICE_FAMILY = "1,2"; + }; + name = Release; + }; /* End XCBuildConfiguration section */ /* Begin XCConfigurationList section */ @@ -471,6 +656,15 @@ defaultConfigurationIsVisible = 0; defaultConfigurationName = Release; }; + C86585E42ED456DE00824752 /* Build configuration list for PBXNativeTarget "TimeSafariShareExtension" */ = { + isa = XCConfigurationList; + buildConfigurations = ( + C86585E12ED456DE00824752 /* Debug */, + C86585E22ED456DE00824752 /* Release */, + ); + defaultConfigurationIsVisible = 0; + defaultConfigurationName = Release; + }; /* End XCConfigurationList section */ }; rootObject = 504EC2FC1FED79650016851F /* Project object */; diff --git a/ios/App/App/App.entitlements b/ios/App/App/App.entitlements new file mode 100644 index 0000000000..68633259f5 --- /dev/null +++ b/ios/App/App/App.entitlements @@ -0,0 +1,10 @@ + + + + + com.apple.security.application-groups + + group.app.timesafari + + + diff --git a/ios/App/App/AppDelegate.swift b/ios/App/App/AppDelegate.swift index 7a1b41b323..33937d3fad 100644 --- a/ios/App/App/AppDelegate.swift +++ b/ios/App/App/AppDelegate.swift @@ -39,6 +39,33 @@ class AppDelegate: UIResponder, UIApplicationDelegate { } func application(_ app: UIApplication, open url: URL, options: [UIApplication.OpenURLOptionsKey: Any] = [:]) -> Bool { + // Check if this is a shared-photo deep link and store image data in a way JS can access + if url.scheme == "timesafari" && url.host == "shared-photo" { + // Try to get shared image from App Group and store it in a temp file that JS can read + // This is a workaround until the plugin is properly registered + if let sharedData = getSharedImageData() { + // Write to a temp file in the app's Documents directory that JavaScript can read via Filesystem plugin + let fileManager = FileManager.default + if let documentsDir = fileManager.urls(for: .documentDirectory, in: .userDomainMask).first { + let tempFileURL = documentsDir.appendingPathComponent("timesafari_shared_photo.json") + + // Create JSON data + let jsonData: [String: String] = [ + "base64": sharedData["base64"] ?? "", + "fileName": sharedData["fileName"] ?? "" + ] + + if let json = try? JSONSerialization.data(withJSONObject: jsonData, options: []) { + do { + try json.write(to: tempFileURL) + } catch { + // Error writing temp file - will be handled by JS layer + } + } + } + } + } + // Called when the app was launched with a url. Feel free to add additional processing here, // but if you want the App API to support tracking app url opens, make sure to keep this call return ApplicationDelegateProxy.shared.application(app, open: url, options: options) @@ -50,5 +77,30 @@ class AppDelegate: UIResponder, UIApplicationDelegate { // tracking app url opens, make sure to keep this call return ApplicationDelegateProxy.shared.application(application, continue: userActivity, restorationHandler: restorationHandler) } + + /** + * Check for shared image from Share Extension + * Reads from App Group UserDefaults and returns shared image data if available + * + * @returns Dictionary with "base64" and "fileName" keys, or nil if no shared image + */ + func getSharedImageData() -> [String: String]? { + let appGroupIdentifier = "group.app.timesafari" + guard let userDefaults = UserDefaults(suiteName: appGroupIdentifier) else { + return nil + } + + guard let base64 = userDefaults.string(forKey: "sharedPhotoBase64"), + let fileName = userDefaults.string(forKey: "sharedPhotoFileName") else { + return nil + } + + // Clear the shared data after reading + userDefaults.removeObject(forKey: "sharedPhotoBase64") + userDefaults.removeObject(forKey: "sharedPhotoFileName") + userDefaults.synchronize() + + return ["base64": base64, "fileName": fileName] + } } diff --git a/ios/App/App/ShareImageBridge.swift b/ios/App/App/ShareImageBridge.swift new file mode 100644 index 0000000000..88ad874295 --- /dev/null +++ b/ios/App/App/ShareImageBridge.swift @@ -0,0 +1,48 @@ +import Foundation + +/** + * Share Image Bridge + * + * Provides a bridge between JavaScript and native iOS code to access + * shared images stored in App Group UserDefaults by the Share Extension. + * + * This bridge allows the JavaScript layer to read shared image data + * that was stored by the Share Extension. + * + * Note: This class doesn't need Capacitor - it's a simple Swift utility + * that reads from App Group UserDefaults. The JavaScript bridge will be + * implemented separately. + */ +@objc(ShareImageBridge) +public class ShareImageBridge: NSObject { + + private static let appGroupIdentifier = "group.app.timesafari" + private static let sharedPhotoBase64Key = "sharedPhotoBase64" + private static let sharedPhotoFileNameKey = "sharedPhotoFileName" + + /** + * Get shared image data from App Group UserDefaults + * Called from JavaScript via Capacitor bridge + * + * @returns Dictionary with "base64" and "fileName" keys, or nil if no shared image + */ + @objc public static func getSharedImageData() -> [String: String]? { + guard let userDefaults = UserDefaults(suiteName: appGroupIdentifier) else { + print("ShareImageBridge: Failed to access App Group UserDefaults") + return nil + } + + guard let base64 = userDefaults.string(forKey: sharedPhotoBase64Key), + let fileName = userDefaults.string(forKey: sharedPhotoFileNameKey) else { + return nil + } + + // Clear the shared data after reading + userDefaults.removeObject(forKey: sharedPhotoBase64Key) + userDefaults.removeObject(forKey: sharedPhotoFileNameKey) + userDefaults.synchronize() + + return ["base64": base64, "fileName": fileName] + } +} + diff --git a/ios/App/App/ShareImagePlugin.swift b/ios/App/App/ShareImagePlugin.swift new file mode 100644 index 0000000000..9073de7f6e --- /dev/null +++ b/ios/App/App/ShareImagePlugin.swift @@ -0,0 +1,28 @@ +import Foundation +import Capacitor + +/** + * Share Image Plugin + * + * Capacitor plugin that exposes ShareImageBridge functionality to JavaScript. + * Allows JavaScript to retrieve shared images from App Group UserDefaults. + */ +@objc(ShareImagePlugin) +public class ShareImagePlugin: CAPPlugin { + + @objc func getSharedImageData(_ call: CAPPluginCall) { + guard let sharedData = ShareImageBridge.getSharedImageData() else { + call.resolve(["success": false, "data": NSNull()]) + return + } + + call.resolve([ + "success": true, + "data": [ + "base64": sharedData["base64"] ?? "", + "fileName": sharedData["fileName"] ?? "" + ] + ]) + } +} + diff --git a/ios/App/TimeSafariShareExtension/Info.plist b/ios/App/TimeSafariShareExtension/Info.plist new file mode 100644 index 0000000000..9230ac2404 --- /dev/null +++ b/ios/App/TimeSafariShareExtension/Info.plist @@ -0,0 +1,21 @@ + + + + + NSExtension + + NSExtensionAttributes + + NSExtensionActivationRule + + NSExtensionActivationSupportsImageWithMaxCount + 1 + + + NSExtensionPointIdentifier + com.apple.share-services + NSExtensionPrincipalClass + $(PRODUCT_MODULE_NAME).ShareViewController + + + diff --git a/ios/App/TimeSafariShareExtension/ShareViewController.swift b/ios/App/TimeSafariShareExtension/ShareViewController.swift new file mode 100644 index 0000000000..9090d128ea --- /dev/null +++ b/ios/App/TimeSafariShareExtension/ShareViewController.swift @@ -0,0 +1,170 @@ +// +// ShareViewController.swift +// TimeSafariShareExtension +// +// Created by Aardimus on 11/24/25. +// + +import UIKit +import Social +import UniformTypeIdentifiers + +class ShareViewController: SLComposeServiceViewController { + + private let appGroupIdentifier = "group.app.timesafari" + private let sharedPhotoBase64Key = "sharedPhotoBase64" + private let sharedPhotoFileNameKey = "sharedPhotoFileName" + + override func viewDidLoad() { + super.viewDidLoad() + + // Set placeholder text (required for SLComposeServiceViewController) + self.placeholder = "Share image to TimeSafari" + + // Validate content on load + self.validateContent() + } + + override func isContentValid() -> Bool { + // Validate that we have image attachments + guard let extensionContext = extensionContext else { + return false + } + + guard let inputItems = extensionContext.inputItems as? [NSExtensionItem] else { + return false + } + + for item in inputItems { + if let attachments = item.attachments { + for attachment in attachments { + if attachment.hasItemConformingToTypeIdentifier(UTType.image.identifier) { + return true + } + } + } + } + + return false + } + + override func didSelectPost() { + // Extract and process the shared image + guard let extensionContext = extensionContext else { + return + } + + guard let inputItems = extensionContext.inputItems as? [NSExtensionItem] else { + extensionContext.completeRequest(returningItems: [], completionHandler: nil) + return + } + + // Process the first image found + processSharedImage(from: inputItems) { [weak self] success in + guard let self = self else { + extensionContext.completeRequest(returningItems: [], completionHandler: nil) + return + } + + if success { + // Open the main app via deep link + self.openMainApp() + } + + // Complete the extension context + extensionContext.completeRequest(returningItems: [], completionHandler: nil) + } + } + + private func processSharedImage(from items: [NSExtensionItem], completion: @escaping (Bool) -> Void) { + // Find the first image attachment + for item in items { + guard let attachments = item.attachments else { + continue + } + + for attachment in attachments { + if attachment.hasItemConformingToTypeIdentifier(UTType.image.identifier) { + attachment.loadItem(forTypeIdentifier: UTType.image.identifier, options: nil) { [weak self] (data, error) in + guard let self = self else { + completion(false) + return + } + + if let error = error { + completion(false) + return + } + + // Handle different image data types + var imageData: Data? + var fileName: String = "shared-image.jpg" + + if let url = data as? URL { + // Image provided as file URL + imageData = try? Data(contentsOf: url) + fileName = url.lastPathComponent + } else if let image = data as? UIImage { + // Image provided as UIImage + imageData = image.jpegData(compressionQuality: 0.9) + fileName = "shared-image.jpg" + } else if let data = data as? Data { + // Image provided as raw Data + imageData = data + fileName = "shared-image.jpg" + } + + guard let finalImageData = imageData else { + completion(false) + return + } + + // Convert to base64 + let base64String = finalImageData.base64EncodedString() + + // Store in App Group UserDefaults + guard let userDefaults = UserDefaults(suiteName: self.appGroupIdentifier) else { + completion(false) + return + } + + userDefaults.set(base64String, forKey: self.sharedPhotoBase64Key) + userDefaults.set(fileName, forKey: self.sharedPhotoFileNameKey) + userDefaults.synchronize() + + completion(true) + } + return // Process only the first image + } + } + } + + // No image found + completion(false) + } + + private func openMainApp() { + // Open the main app via deep link + guard let url = URL(string: "timesafari://shared-photo") else { + return + } + + var responder: UIResponder? = self + while responder != nil { + if let application = responder as? UIApplication { + application.open(url, options: [:], completionHandler: nil) + return + } + responder = responder?.next + } + + // Fallback: use extension context + extensionContext?.open(url, completionHandler: nil) + } + + override func configurationItems() -> [Any]! { + // No additional configuration options needed + return [] + } + +} diff --git a/ios/App/TimeSafariShareExtension/TimeSafariShareExtension.entitlements b/ios/App/TimeSafariShareExtension/TimeSafariShareExtension.entitlements new file mode 100644 index 0000000000..68633259f5 --- /dev/null +++ b/ios/App/TimeSafariShareExtension/TimeSafariShareExtension.entitlements @@ -0,0 +1,10 @@ + + + + + com.apple.security.application-groups + + group.app.timesafari + + + diff --git a/scripts/build-ios.sh b/scripts/build-ios.sh index b327152c6c..fde5ee7d0c 100755 --- a/scripts/build-ios.sh +++ b/scripts/build-ios.sh @@ -404,8 +404,141 @@ elif [ "$BUILD_MODE" = "production" ]; then safe_execute "Building Capacitor version (production)" "npm run build:capacitor -- --mode production" || exit 3 fi -# Step 6: Sync with Capacitor -safe_execute "Syncing with Capacitor" "npx cap sync ios" || exit 6 +# Step 6: Install CocoaPods dependencies (with Xcode 26 workaround) +# =================================================================== +# WORKAROUND: Xcode 26 / CocoaPods Compatibility Issue +# =================================================================== +# Xcode 26 uses project format version 70, but CocoaPods' xcodeproj gem +# (1.27.0) only supports up to version 56. This causes pod install to fail. +# +# This workaround temporarily downgrades the project format to 56, runs +# pod install, then restores it to 70. Xcode will automatically upgrade +# it back to 70 when opened, which is fine. +# +# NOTE: Both explicit pod install AND Capacitor sync (which runs pod install +# internally) need this workaround. See run_pod_install_with_workaround() +# and run_cap_sync_with_workaround() functions below. +# +# TO REMOVE THIS WORKAROUND IN THE FUTURE: +# 1. Check if xcodeproj gem has been updated: bundle exec gem list xcodeproj +# 2. Test if pod install works without the workaround +# 3. If it works, remove both workaround functions below +# 4. Replace with: +# - safe_execute "Installing CocoaPods dependencies" "cd ios/App && bundle exec pod install && cd ../.." || exit 6 +# - safe_execute "Syncing with Capacitor" "npx cap sync ios" || exit 6 +# 5. Update this comment to indicate the workaround has been removed +# =================================================================== +run_pod_install_with_workaround() { + local PROJECT_FILE="ios/App/App.xcodeproj/project.pbxproj" + + log_info "Installing CocoaPods dependencies (with Xcode 26 workaround)..." + + # Check if project file exists + if [ ! -f "$PROJECT_FILE" ]; then + log_error "Project file not found: $PROJECT_FILE" + return 1 + fi + + # Check current format version + local current_version=$(grep "objectVersion" "$PROJECT_FILE" | head -1 | grep -o "[0-9]\+" || echo "") + + if [ -z "$current_version" ]; then + log_error "Could not determine project format version" + return 1 + fi + + log_debug "Current project format version: $current_version" + + # Only apply workaround if format is 70 + if [ "$current_version" = "70" ]; then + log_debug "Applying Xcode 26 workaround: temporarily downgrading to format 56" + + # Downgrade to format 56 (supported by CocoaPods) + if ! sed -i '' 's/objectVersion = 70;/objectVersion = 56;/' "$PROJECT_FILE"; then + log_error "Failed to downgrade project format" + return 1 + fi + + # Run pod install + log_info "Running pod install..." + if ! (cd ios/App && bundle exec pod install && cd ../..); then + log_error "pod install failed" + # Try to restore format even on failure + sed -i '' 's/objectVersion = 56;/objectVersion = 70;/' "$PROJECT_FILE" || true + return 1 + fi + + # Restore to format 70 + log_debug "Restoring project format to 70..." + if ! sed -i '' 's/objectVersion = 56;/objectVersion = 70;/' "$PROJECT_FILE"; then + log_warn "Failed to restore project format to 70 (Xcode will upgrade it automatically)" + fi + + log_success "CocoaPods dependencies installed successfully" + else + # Format is not 70, run pod install normally + log_debug "Project format is $current_version, running pod install normally" + if ! (cd ios/App && bundle exec pod install && cd ../..); then + log_error "pod install failed" + return 1 + fi + log_success "CocoaPods dependencies installed successfully" + fi +} + +safe_execute "Installing CocoaPods dependencies" "run_pod_install_with_workaround" || exit 6 + +# Step 6.5: Sync with Capacitor (also needs workaround since it runs pod install internally) +# Capacitor sync internally runs pod install, so we need to apply the workaround here too +run_cap_sync_with_workaround() { + local PROJECT_FILE="ios/App/App.xcodeproj/project.pbxproj" + + # Check current format version + local current_version=$(grep "objectVersion" "$PROJECT_FILE" | head -1 | grep -o "[0-9]\+" || echo "") + + if [ -z "$current_version" ]; then + log_error "Could not determine project format version for Capacitor sync" + return 1 + fi + + # Only apply workaround if format is 70 + if [ "$current_version" = "70" ]; then + log_debug "Applying Xcode 26 workaround for Capacitor sync: temporarily downgrading to format 56" + + # Downgrade to format 56 (supported by CocoaPods) + if ! sed -i '' 's/objectVersion = 70;/objectVersion = 56;/' "$PROJECT_FILE"; then + log_error "Failed to downgrade project format for Capacitor sync" + return 1 + fi + + # Run Capacitor sync (which will run pod install internally) + log_info "Running Capacitor sync..." + if ! npx cap sync ios; then + log_error "Capacitor sync failed" + # Try to restore format even on failure + sed -i '' 's/objectVersion = 56;/objectVersion = 70;/' "$PROJECT_FILE" || true + return 1 + fi + + # Restore to format 70 + log_debug "Restoring project format to 70 after Capacitor sync..." + if ! sed -i '' 's/objectVersion = 56;/objectVersion = 70;/' "$PROJECT_FILE"; then + log_warn "Failed to restore project format to 70 (Xcode will upgrade it automatically)" + fi + + log_success "Capacitor sync completed successfully" + else + # Format is not 70, run sync normally + log_debug "Project format is $current_version, running Capacitor sync normally" + if ! npx cap sync ios; then + log_error "Capacitor sync failed" + return 1 + fi + log_success "Capacitor sync completed successfully" + fi +} + +safe_execute "Syncing with Capacitor" "run_cap_sync_with_workaround" || exit 6 # Step 7: Generate assets safe_execute "Generating assets" "npx capacitor-assets generate --ios" || exit 7 diff --git a/src/interfaces/deepLinks.ts b/src/interfaces/deepLinks.ts index 0fe5c68dd9..60c0826ce0 100644 --- a/src/interfaces/deepLinks.ts +++ b/src/interfaces/deepLinks.ts @@ -68,6 +68,7 @@ export const deepLinkPathSchemas = { "user-profile": z.object({ id: z.string(), }), + "shared-photo": z.object({}), }; export const deepLinkQuerySchemas = { diff --git a/src/main.capacitor.ts b/src/main.capacitor.ts index f091770b1c..3ef253a546 100644 --- a/src/main.capacitor.ts +++ b/src/main.capacitor.ts @@ -30,11 +30,15 @@ import { initializeApp } from "./main.common"; import { App as CapacitorApp } from "@capacitor/app"; +import { Capacitor } from "@capacitor/core"; +import { Filesystem, Directory, Encoding } from "@capacitor/filesystem"; import router from "./router"; import { handleApiError } from "./services/api"; import { AxiosError } from "axios"; import { DeepLinkHandler } from "./services/deepLinks"; import { logger, safeStringify } from "./utils/logger"; +import { PlatformServiceFactory } from "./services/PlatformServiceFactory"; +import { SHARED_PHOTO_BASE64_KEY } from "./libs/util"; import "./utils/safeAreaInset"; logger.log("[Capacitor] 🚀 Starting initialization"); @@ -67,11 +71,220 @@ const deepLinkHandler = new DeepLinkHandler(router); * * @throws {Error} If URL format is invalid */ +/** + * Check for native shared image from iOS App Group UserDefaults + * and store in temp database before routing to shared-photo view + */ +async function checkAndStoreNativeSharedImage(): Promise<{ + success: boolean; + fileName?: string; +}> { + if (!Capacitor.isNativePlatform() || Capacitor.getPlatform() !== "ios") { + return { success: false }; + } + + try { + logger.debug("[Main] Checking for iOS shared image from App Group"); + + // Use Capacitor's native bridge to call the ShareImagePlugin + // eslint-disable-next-line @typescript-eslint/no-explicit-any + const capacitor = (window as any).Capacitor; + + if (!capacitor || !capacitor.Plugins) { + logger.debug("[Main] Capacitor plugins not available"); + return { success: false }; + } + + // WORKAROUND: Since the plugin isn't being auto-discovered, use a temp file bridge + // AppDelegate writes the shared image data to a temp file, and we read it here + try { + const tempFilePath = "timesafari_shared_photo.json"; + const fileContent = await Filesystem.readFile({ + path: tempFilePath, + directory: Directory.Documents, + encoding: Encoding.UTF8, + }); + + if (fileContent.data) { + const sharedData = JSON.parse(fileContent.data as string); + const base64 = sharedData.base64; + const fileName = sharedData.fileName || "shared-image.jpg"; + + if (base64) { + // Store in temp database using dbExec directly + logger.info( + "[Main] Native shared image found (via temp file), storing in temp DB", + ); + const platformService = PlatformServiceFactory.getInstance(); + + // Convert raw base64 to data URL format that base64ToBlob expects + // base64ToBlob expects format: "..." + // Try to detect image type from base64 or default to jpeg + let mimeType = "image/jpeg"; // default + if (base64.startsWith("/9j/") || base64.startsWith("iVBORw0KGgo")) { + // JPEG or PNG + mimeType = base64.startsWith("/9j/") ? "image/jpeg" : "image/png"; + } + const dataUrl = `data:${mimeType};base64,${base64}`; + + // Use INSERT OR REPLACE to handle existing records + await platformService.dbExec( + "INSERT OR REPLACE INTO temp (id, blobB64) VALUES (?, ?)", + [SHARED_PHOTO_BASE64_KEY, dataUrl], + ); + + // Delete the temp file + try { + await Filesystem.deleteFile({ + path: tempFilePath, + directory: Directory.Documents, + }); + } catch (deleteError) { + logger.error("[Main] Failed to delete temp file:", deleteError); + } + + logger.info(`[Main] Stored shared image: ${fileName}`); + return { success: true, fileName }; + } + } + } catch (fileError: unknown) { + // File doesn't exist or can't be read - that's okay, try plugin method + logger.debug( + "[Main] Temp file not found or unreadable (this is normal if plugin works)", + ); + } + + // NOTE: Plugin registration issue - ShareImage plugin is not being auto-discovered + // This is a known issue that needs to be resolved. For now, we use the temp file workaround above. + + // Try multiple methods to call the plugin (fallback if temp file method fails) + // eslint-disable-next-line @typescript-eslint/no-explicit-any + const plugins = (capacitor as any).Plugins; + // eslint-disable-next-line @typescript-eslint/no-explicit-any + const shareImagePlugin = (plugins as any)?.ShareImage; + + // eslint-disable-next-line @typescript-eslint/no-explicit-any + let result: any = null; + + if ( + shareImagePlugin && + typeof shareImagePlugin.getSharedImageData === "function" + ) { + logger.debug("[Main] Using direct plugin method"); + try { + result = await shareImagePlugin.getSharedImageData(); + } catch (pluginError) { + logger.error("[Main] Plugin call failed:", pluginError); + return { success: false }; + } + } else { + // Method 2: Use Capacitor's execute method + logger.debug( + "[Main] Plugin not found directly, trying Capacitor.execute", + ); + try { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + const bridge = (capacitor as any).getBridge?.(); + if (bridge) { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + result = await bridge.execute({ + pluginId: "ShareImage", + methodName: "getSharedImageData", + options: {}, + }); + } else { + // Method 3: Try execute on Plugins object + // eslint-disable-next-line @typescript-eslint/no-explicit-any + if ((plugins as any)?.execute) { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + result = await (plugins as any).execute( + "ShareImage", + "getSharedImageData", + {}, + ); + } + } + } catch (executeError) { + logger.error("[Main] Execute method failed:", executeError); + return { success: false }; + } + } + + if (!result || !result.success || !result.data) { + logger.debug("[Main] No shared image data found in result"); + return { success: false }; + } + + const { base64, fileName } = result.data; + + if (!base64) { + logger.debug("[Main] Shared image data missing base64"); + return { success: false }; + } + + logger.info("[Main] Native shared image found, storing in temp DB"); + + // Store in temp database (similar to web flow) + const platformService = PlatformServiceFactory.getInstance(); + // $insertEntity is available via PlatformServiceMixin + // eslint-disable-next-line @typescript-eslint/no-explicit-any + await (platformService as any).$insertEntity( + "temp", + { id: SHARED_PHOTO_BASE64_KEY, blobB64: base64 }, + ["id", "blobB64"], + ); + + logger.info(`[Main] Stored shared image: ${fileName || "unknown"}`); + return { success: true, fileName: fileName || "shared-image.jpg" }; + } catch (error) { + logger.error("[Main] Error checking for native shared image:", error); + return { success: false }; + } +} + const handleDeepLink = async (data: { url: string }) => { const { url } = data; logger.debug(`[Main] 🌐 Deeplink received from Capacitor: ${url}`); try { + // Check if this is a shared-photo deep link from native share + const isSharedPhotoLink = url.includes("timesafari://shared-photo"); + + if ( + isSharedPhotoLink && + Capacitor.isNativePlatform() && + Capacitor.getPlatform() === "ios" + ) { + logger.debug( + "[Main] 📸 Shared photo deep link detected, checking for native shared image", + ); + + // Try to get shared image from App Group and store in temp database + try { + const imageResult = await checkAndStoreNativeSharedImage(); + + if (imageResult.success) { + logger.info("[Main] ✅ Native shared image stored in temp database"); + + // Add fileName to the URL as a query parameter if we have it + if (imageResult.fileName) { + const urlObj = new URL(url); + urlObj.searchParams.set("fileName", imageResult.fileName); + const modifiedUrl = urlObj.toString(); + data.url = modifiedUrl; + logger.debug(`[Main] Added fileName to URL: ${modifiedUrl}`); + } + } else { + logger.debug( + "[Main] ℹ️ No native shared image found (may be from web or already processed)", + ); + } + } catch (error) { + logger.error("[Main] Error processing native shared image:", error); + // Continue with normal deep link processing even if native check fails + } + } + // Wait for router to be ready logger.debug(`[Main] ⏳ Waiting for router to be ready...`); await router.isReady();