Post

Replies

Boosts

Views

Activity

OSLog crashes on device but not on simulator
import Foundation import os.log extension OSLog { private static var subsystem = Bundle.main.bundleIdentifier! static let session = OSLog(subsystem: subsystem, category: "session") static func Debug(_ s: StaticString, args: CVarArg...) { os_log(s, log: OSLog.session, type: .debug, args) } static func Error(_ s: StaticString, args: CVarArg...) { os_log(s, log: OSLog.session, type: .error, args) } static func Info(_ s: StaticString, args: CVarArg...) { os_log(s, log: OSLog.session, type: .info, args) } }I defined this extension for using OSLog instead of NSLog in my project. I then wrote the following code which uses the extension:/// Instantiate an array of `Tweet`s from a property list. /// /// - Parameter url: The URL for a property list made up of one or more `Tweet`s. /// - Returns: Optinal `Tweet` array. If the property list exists, is not malformed, /// and contains all of the key/value pairs required to create one or /// more `Tweet` items, an array of said items will be returned. Othrwise, nil func tweetsWithContentsOfURL(_ url: URL) -> [Tweet] { do { let data = try Data(contentsOf: url) let decoder = PropertyListDecoder() let tweets = try decoder.decode([Tweet].self, from: data) return tweets } catch { OSLog.Error("Failed to load Tweet values from %@ with error: %@", args: String(describing: url), String(describing: error)) return [] } }Here are the arguments being passed into `OSLog.Error`:(lldb) po args ▿ 2 elements - 0 : "file:///var/mobile/Containers/Data/Application/AD339CF3-71DD-4CC3-876B-9668F51ECF2F/Documents/ClientTweets" - 1 : "Error Domain=NSCocoaErrorDomain Code=260 \"The file “ClientTweets” couldn’t be opened because there is no such file.\" UserInfo={NSFilePath=/var/mobile/Containers/Data/Application/AD339CF3-71DD-4CC3-876B-9668F51ECF2F/Documents/ClientTweets, NSUnderlyingError=0x2833b0b70 {Error Domain=NSPOSIXErrorDomain Code=2 \"No such file or directory\"}}" (lldb) po s "Failed to load Tweet values from %@ with error: %@"If the file I'm trying to load is empty, an error is thrown and an error is logged. I'm expecting this under certain test conditions. This code works just fine when running on the simulator. However, when runing on a device, invoking any of the status functions (Debug, Error, or Info) results in "Thread 1: EXC_BAD_ACCESS (code=1, address=0x10)". I check at runtime in the debugger to make sure that `subsystem`, `sessions` and my arguments are all properly initialized and make sense. They look good so I do not understand why this is crashing.What am I missing here?-Michael
1
0
2.5k
Apr ’19
TableView inside of a CollectionView cell does not draw on row deletion.
Ok, the setup for this is pretty complex. I'm doing maintenance on an app I did not write. The view structure is as follows: At the base you have a UICollectionViewController. Each UICollectionViewCell has an assigned UIViewController within it. Each of these controllers is attached to the view hierarchy as a child view controller. The main view of one of these child controllers is a UITableView/UIViewController pair. Everything seems to be hooked up properly and working well except for one case. Assuming the table-view has more than two rows, If I swipe-left to delete the second row (IndexPath 0,1), the content or row one (IndexPath 0,0) goes blank. I can see the disclosure accessory view but the two UILabels up and disappear. If I tap on the blank row the tap behavior is still intact. If I refresh the table-view, the missing content appears as it should. To make things a little more interesting, this table-view is implement using RxSwift. Here is the code: private func bind() {         // Cell for row at index path.         let curriedArgument: (Int, Assignment, AssessmentTableViewCell) - Void = { /* rowIndex */_, model, cell in             cell.assignment = model         }         let observer: Observable[Assignment]         // For the widget version of this controller, only display the first N         // items in a table-view that does not scroll. If more than N items exist         // the user must tap the `More` view. For the non-widget view, show all         // available items in a scrolling table-view.         if isWidget {             tableView.isScrollEnabled = false             observer = viewModel                 .assignments                 .asObservable()                 .map({ Array($0.prefix(CPAssignmentWidgetVC.numberOfItemsToDisplay))})         } else {             observer = viewModel                 .assignments                 .asObservable()         }         observer             .distinctUntilChanged()             .bind(to: tableView.rx.items(cellIdentifier: "AssessmentCell"), curriedArgument: curriedArgument)             .disposed(by: disposeBag)         // When something changes, update both the widget and the modal view as         // needed.         if isWidget {             Observable                 .combineLatest(viewModel.fetching.asObservable(),                                viewModel.assignments.asObservable())                 .subscribe(onNext: { [weak self] (isFetching, assignments) in                     self?.updateFooter(with: assignments, isFetching: isFetching)                 })                 .disposed(by: disposeBag)         } else {             viewModel.fetching.asObservable()                 .distinctUntilChanged()                 .debounce(.milliseconds(500), scheduler: MainScheduler.instance)                 .subscribe(onNext: { isFetching in                     if isFetching {                         SVProgressHUD.show()                     } else {                         SVProgressHUD.dismiss()                     }                 })                 .disposed(by: disposeBag)         }         // Select cell         tableView             .rx             .itemSelected             .subscribe(onNext: { [unowned self] indexPath in                 self.tableView.deselectRow(at: indexPath, animated: true)                 guard                     let cell = tableView.cellForRow(at: indexPath) as? AssessmentTableViewCell,                     let assignment = cell.assignment else { return }                 if cell.assignmentResult != nil {                     presentOptions(cell)                 } else {                     guard let patient = patient, patient.hasPermissionToModify(permissionType: .assessments) else {                         let title = NSLocalizedString("We're sorry!", comment: "Notification error title")                         let message = NSLocalizedString("You don't have permission to complete this survey", comment: "Notification error message")                         let dismiss = NSLocalizedString("Dismiss", comment: "Button title")                         LHDNotification.show(with: title, text: message, type: .error).setDismissButtonTitle(dismiss)                         return                     }                     presentAssignmentEntryForm(assignment)                 }             })             .disposed(by: disposeBag)         // Delete cell (remove associated backing model value)         tableView             .rx             .itemDeleted             .subscribe(onNext: { indexPath in                 self.viewModel.remove(at: indexPath.row)             })             .disposed(by: disposeBag)         // Display last cell         tableView             .rx             .willDisplayCell             .subscribe(onNext: { [unowned self] (/* cell */_, indexPath) in                 if indexPath.item + 1 == self.viewModel.numberOfItems {                     self.viewModel.loadMore()                 }             })             .disposed(by: disposeBag)     } } Question: Why is the first row content being wiped out? Assuming no one can know the answer to this I have a follow-up question. How can I get the contents of the entire tableview to redraw themselves after a successful delete? I would have assumed the binding to the table-view on line 27 would handle this automatically when the view-model updates.
3
0
987
Feb ’21
How to I cancel a task created by the new Swift `async` keyword and closure, after the fact?
I'm experimenting with async/await on some existing code and would like to cancel an async thumbnail download if the table-view cell needs to be re-used. The problem is that I don't know how to declare storage for the handle so I can cancel it later on. See the attached code. class PhotoCell: UITableViewCell {     @IBOutlet weak var albumLabel: UILabel!     @IBOutlet weak var photoIdLabel: UILabel!     @IBOutlet weak var thumbnailImageView: UIImageView!     @IBOutlet weak var titleLabel: UILabel!     private static var imageLoader = ImageLoader() //    private var task: Handle<(), Never>?     override func prepareForReuse() { //        task?.cancel()         albumLabel.text = ""         titleLabel.text = ""         photoIdLabel.text = ""     }     // MARK: - Public Methods     func configure(with photo: JPPhoto) {         albumLabel.text = "#\(photo.albumId.rawValue)"         titleLabel.text = photo.title photoIdLabel.text = "#\(photo.id.rawValue)"         thumbnailImageView.image = UIImage(systemName: "photo")         /* task = */async {             if let image = await PhotoCell.imageLoader.loadImage(from: photo.thumbnailUrl) {                 thumbnailImageView.image = image             }         }     } }
4
0
1.4k
Jun ’21
Unable to link to or even find USBDriverKit for iPadOS
According to Apple's documentation USBDriverKit for iPadOS has been available since DriverKit 19 and the M1 iPad were released. However, using Xcode 15.0.1, I am unable to link to or even find USBDriverKit from within Xcode. Is the documentation incorrect? If not what do I need to do in order to access and leverage USBDriverKit? From what I've read, USBSerialDriverKit is not available. As I have a USB based serial device that I want to use from the iPad, the next obvious step is to write my own USB device extension using USBDriverKit as a provider. If you have experience with this and can definitively say whether or not what I'm attempting is even within the realm of possibility, please chime in. Thanks. -Michael
2
0
990
Dec ’23
How to use Swift and AVFoundation to stream/record USB microphone input?
I have a custom USB device that includes a microphone. I can see the microphone on macOS when I plug in the device so I know that it is working with the kernel and AV subsystems. I can enumerate and reference the microphone using AVCaptureDevice but I have not been able to figure out how to use this device reference with AVAudioEngine. I'm trying to accomplish two things with this microphone. I want to stream audio from the microphone and have it rendered to the speakers on my MacBook Pro. I want to capture sound data from the microphone and forward it to a live streaming API. To my mind, from what I've read, I need AVAudioEngine to do this but I'm having trouble determining from the documentation just how to go about it on macOS. It seems that there is a lot more information for iOS or iPadOS but since USB-C support is sparsely documented on those operating systems, I'm focusing on the desktop (macOS) for now. Can I convert an AVCaptureDevice into and audio input for AVAudioEngine? If not, how can I accomplish what I'm trying to do using whatever is available on AVFoundation?
1
0
1.2k
Jan ’24
Settings bundle for DEXT loading app does not display content.
I have an app that loads a DEXT (driver). This app includes a settings bundle that allows me to activate/deactivate the driver. When I issue the API call to activate the driver, iOS switches to the Settings app and displays the page for my DEXT loading application but the switch to enable the driver, which is part of my settings bundle, does not appear. I'm using this API call: OSSystemExtensionRequest.activationRequest(forExtensionWithIdentifier: "driver-id", queue: .main) Here are the contents of my settings bundle Root.plist: ` Here are the contents of my Root.strings file: "Group" = "Group"; "Name" = "Name"; "none given" = "none given"; "Enabled" = "Enabled";
2
0
652
Mar ’24
Core Animation Layer in SwiftUI app does not update unless I rotate the device.
I am trying to create a near real-time drawing of waveform data from within a SwiftUI app. The data is streaming in from the hardware and I've verified that the draw(in ctx: CGContext) override in my custom CALayer is getting called. I have added this custom CALayer class as a sublayer to a UIView instance that I am making available via the UIViewRepresentable protocol. The only time I see updated output from the CALayer is when I rotate the device and layout happens (I assume). How can I force SwiftUI to update every time I render new data in my CALayer? More Info: I'm porting an app from the Windows desktop. Previously, I tried to make this work by simply generating a new UIImage from a CGContext every time I wanted to update the display. I quickly exhausted memory with that technique because a new context is being created every time I call UIGraphicsImageRenderer(size:).image { context in }. What I really wanted was something equivalent to a GDI WritableBitmap. Apparently this animal allows a programmer to continuously update and re-use the contents. I could not figure out how to do this in Swift without dropping down to the old CGBitmapContext stuff written in C and even then I wasn't sure if that would give me a reusable context that I could output in SwiftUI each time I refreshed it. CALayer seemed like the answer. I welcome any feedback on a better way to do what I'm trying to accomplish.
2
0
790
Apr ’24
AVCaptureVideoPreviewLayer transforms are not working. Need pan/zoom solution.
I am implementing pan and zoom features for an app using a custom USB camera device, in iPadOS. I am using an update function (shown below) to apply transforms for scale and translation but they are not working. By re-enabling the animation I can see that the scale translation seems to initially take effect but then the image animates back to its original scale. This all happens in a fraction of a second but I can see it. The translation transform seems to have no effect at all. Printing out the value of AVCaptureVideoPreviewLayer.transform before and after does show that my values have been applied. private func updateTransform() { #if false // Disable default animation. CATransaction.begin() CATransaction.setDisableActions(true) defer { CATransaction.commit() } #endif // Apply the transform. logger.debug("\(String(describing: self.videoPreviewLayer.transform))") let transform = CATransform3DIdentity let translate = CATransform3DTranslate(transform, translationX, translationY, 0) let scale = CATransform3DScale(transform, scale, scale, 1) videoPreviewLayer.transform = CATransform3DConcat(translate, scale) logger.debug("\(String(describing: self.videoPreviewLayer.transform))") } My question is this, how can I properly implement pan/zoom for an AVCaptureVideoPreviewLayer? Or even better, if you see a problem with my current approach or understand why the transforms I am applying do not work, please share that information.
0
0
507
May ’24
DriverKit: embedded.mobileprofile has the wildcard USB Vendor ID instead of my assigned Vendor ID
I've added my Vendor ID to the appropriate entitlement files but my binary fails validation when trying to upload it to the store for distribution. The embeded.mobileprovision file in the generated archive shows an asterisk instead of my approved Vendor ID. How can I make sure the embedded provisioning file has my Vendor ID?
4
0
914
May ’24
How does one create a provisioning profile for embedded DEXT for iPhoneOS that is signed with a distribution cert?
I've been developing a solution that has an embedded USB driver. I can build and run my solution just fine but I cannot pass verification for uploading to App Store Correct and TestFlight The problem is that the provisioning profile I am using (for development) does not have the explicit Vendor ID (idVendor) but is using the development value of asterisk "*". I've created a release version of my entitlements file with the proper Vendor ID and I have a distribution certificate for iOS. Further, I've created a provisioning profile for app-store distribution (not development) and imported it via Xcode. When I select this provisioning profile, I get the following errors from Xcode: Xcode 14 and later requires a DriverKit development profile enabled for iOS and macOS. Visit the developer website to create or download a DriverKit profile. Provisioning profile "MyProvisioningProfile - App Store" doesn't match the entitlements file's value for the com.apple.developer.driverkit.transport.usb entitlement. If I create and use a DriverKit profile, The Xcode UI errors go away on the "Signing & Capabilities" page. However, these profiles seem to be for development only. I then get an error, during compilation, telling me that the app and extension have two different signers, one for development (DEXT) and one for distribution (App). To sum up, using a DriverKit profile fails during the build process and using a distribution profile is a non-starter for Xcode. I can't even build. What do I need to do to get this to work?
2
0
707
May ’24
Internal Apple Developer team member does not appear in TestFlight. (User is in limbo)
I have a team member who was deleted. I tried to re-add that team member immediately after deletion but that failed. However, portal still recognizes that user as a member of the team. This results in the following behavior. When trying to re-add the user to the team, I get a message saying they are already on the team. No new invite can be sent because I can't complete the add-user operation. In TestFlight, I do not see this user on the list/roster of possible internal testers. Has anyone seen this behavior before and if so, how did you fix it?
1
0
548
May ’24
AVFoundation: Strange error while trying to switch camera formats with the touch of a single button.
I'm getting the following output from my iOS app's debug console, note the error on the last line: Capture format keys: ["600x600@25", "1200x1200@5", "1200x1200@30", "1600x1200@2", "1600x1200@30", "3200x2400@15", "3200x2400@2", "600x600@30"] Start capture session for 1600x1200@30: <AVCaptureSession: 0x303c70190 [AVCaptureSessionPresetPhoto]> Stop capture session: <AVCaptureSession: 0x303c70190 [AVCaptureSessionPresetInputPriority]> <AVCaptureDeviceInput: 0x303ebb720 [Medwand S3 Camera]>[vide] -> <AVCaptureVideoDataOutput: 0x303edf1e0> <AVCaptureDeviceInput: 0x303ebb720 [Medwand S3 Camera]>[vide] -> <AVCapturePhotoOutput: 0x303ee3e20> <AVCaptureDeviceInput: 0x303ebb720 [Medwand S3 Camera]>[vide] -> <AVCaptureVideoPreviewLayer: 0x3030b33c0> Start capture session for 600x600@30: <AVCaptureSession: 0x303c70190 [AVCaptureSessionPresetInputPriority]> <AVCaptureDeviceInput: 0x303ebb720 [Medwand S3 Camera]>[vide] -> <AVCaptureVideoDataOutput: 0x303edf1e0> <AVCaptureDeviceInput: 0x303ebb720 [Medwand S3 Camera]>[vide] -> <AVCapturePhotoOutput: 0x303ee3e20> <AVCaptureDeviceInput: 0x303ebb720 [Medwand S3 Camera]>[vide] -> <AVCaptureVideoPreviewLayer: 0x3030b33c0> <<<< FigSharedMemPool >>>> Fig assert: "blkHdr->useCount > 0" at (FigSharedMemPool.c:591) - (err=0) This is in response to trying to switch capture formats between the two key modes that must regularly be used by my application. Below you will find the functions that I use to start and stop capturing frames to my preview layer. I have a UI with three buttons. Off Mode 1 Mode 2 If I tap the Off button in between tapping Mode 1 or Mode 2 all is well; I can do this all day. However, attempt to jump between Mode 1andMode 2` directly I run into issues. I added a layer of software between the UI and the underlying functions so that I could make sure to turn off the Camera before turning it back on in the opposite mode and was surprised to get this output. Can someone at Apple please tell me what is going on here? For the rest of you, if anyone knows the magic incantation to safely switch camera formats, please paste that code here. Thanks. I've included my code below. func start(for deviceFormat: String) { sessionQueue.async { [unowned self] in logger.debug("Start capture session for \(deviceFormat): \(self.captureSession)") do { guard let format = formatDict[deviceFormat] else { throw Error.captureFormatNotFound } captureSession.stopRunning() captureSession.beginConfiguration() // May not be necessary. try captureDevice.lockForConfiguration() // Without this we get an error. captureDevice.activeFormat = format captureDevice.unlockForConfiguration() // Matching function: Necessary. captureSession.commitConfiguration() // Matching function: May not be necessary. captureSession.startRunning() } catch { logger.fault("Failed to start camera: \(error.localizedDescription)") errorPublisher.send(error) } } } func stop() { sessionQueue.async { [unowned self] in logger.debug("Stop capture session: \(self.captureSession)") captureSession.stopRunning() } }
0
0
464
May ’24