Post

Replies

Boosts

Views

Activity

NSMetadataQueryDidFinishGathering not being called enough
I have an app where each item in an NSTableView contains an NSMetadataQuery, to listen for updates to image files that contain the keyword associated with the item. If I write the keywords to, RAW files, everything works as expected and the number label in the row gets updated correctly. However, if I write the keywords to XMP sidecar files, the NSMetadataQueryDidFinishGathering callback isn't called enough times and the count of files affected isn't updated correctly. What is more, the number returned on the final call to the callback seems to vary depending on the time of day, weather, etc Here is the code that I have used to create the predicate for finding keywords in XMP files…       let xmpFileTypePredicate = NSPredicate(fromMetadataQueryString: "kMDItemKind = \"XMP sidecar*\"cdwt")!       let xmpKeywordPredicate = NSPredicate(fromMetadataQueryString: "kMDItemTextContent = \"<rdf:li>\(keyword)</rdf:li>\"cdw")!       let xmpKeywordsMetadataQueryPredicate = NSCompoundPredicate(andPredicateWithSubpredicates: [xmpFileTypePredicate, xmpKeywordPredicate]) Can anyone give me any ideas?
0
0
851
Aug ’22
Replace Threadsafe var with Actor
I have a long established app that uses a threadsafe var to protect an array of URLs from concurrency races, etc… class ThreadsafeVar<T> {   private var value = Optional<T>.none      func callAsFunction() -> T?   {     queue.sync(flags: .barrier) { [unowned self] in value }   }      func set(_ value: T?)   {     queue.async(flags: .barrier) { [unowned self] in self.value = value }   } } What I want to do is substitute an Actor, which I understand will be intrinsically threadsafe. So, I wrote one like this… actor UrlManager {   private var value = [URL]()   func callAsFunction() -> [URL]   {     value   }   func set(_ value: [URL])   {     self.value = value   } } The calling code is in one of four classes, all of which implement a protocol, which contains a read-write var… protocol FolderProtocol { …   var fileURLs: [URL] { get set } } In this particular class, the var is implemented as read-only…   private var urls = ThreadsafeVar<[URL]>()   var fileURLs: [URL]   {     get     {       urls()?.sorted(using: .finder, pathComponent: .fullPath) ?? []     }     set { }   } … the private var being updated from a metadata updating notification handler. Calling the set(_:[URL]) method on the actor poses no problem…       Task { await urls.set(foundURLs) } But, no matter what I try - continuation, try, await, etc, I don't seem to be able to read back the array of URLs from the actor's callAsFunction() method. The fileURLs var is called on the main thread in response to NSCollectionView delegate methods. Other implementing classes don't need the concurrency management as they are fixed once and never changed. Can anyone help me with this one?
1
0
1.1k
Jun ’22
QLThumbnailGenerator loading wrong images
I have an NSCollectionView which is showing thumbnails of images and I have just switched to using QLThumbnailGenerator to fetch them. There are over 6,000 possible images that can be viewed and, if I scroll too fast, I start to get the wrong thumbnails returned from the generator. Is this a bug, or is there something I can do to fix this? Here is the code that is written inside the NSCollectionViewItem derived class…   var request: QLThumbnailGenerator.Request?      func loadImage()   {     if imageView?.image != NSImage(imageLiteralResourceName: "Placeholder")     {       return     }          request = QLThumbnailGenerator.Request(fileAt: url!, size: imageSize, scale: 1.0, representationTypes: [.lowQualityThumbnail])        QLThumbnailGenerator.shared.generateBestRepresentation(for: request!)     {       (thumbnail: QLThumbnailRepresentation?, error: Error?) -> Void in              if let request = self.request       {         QLThumbnailGenerator.shared.cancel(request)       }              DispatchQueue.main.async       {         [unowned self] in                  if self.imageView?.image != NSImage(imageLiteralResourceName: "Placeholder")         {           return         }                  let transition = CATransition()                  transition.timingFunction = CAMediaTimingFunction(name: CAMediaTimingFunctionName.easeInEaseOut)                  transition.duration = 0.3                  imageView?.layer?.add(transition, forKey: nil)                  imageView?.image = thumbnail?.nsImage         …
2
0
1.1k
Jun ’21
Problems monitoring a file for metadata changes
I need to monitor a file for when its metadata changes. I'm using the following code I found. final class FileMonitor {   let url: URL   var fileHandle: FileHandle?   var source: DispatchSourceFileSystemObject?   var didReceiveChanges: (() - Void)?   func process(event: DispatchSource.FileSystemEvent)   {     print(String(describing: event))     didReceiveChanges?()   }   func eventHandler()   {     let event = self.source!.data     process(event: event)     createDispatchSource()   }   func createDispatchSource()   {     self.fileHandle = try? FileHandle(forReadingFrom: url)     source = DispatchSource.makeFileSystemObjectSource(fileDescriptor: fileHandle!.fileDescriptor, eventMask: [.attrib], queue: DispatchQueue.main)     source!.setEventHandler(handler: eventHandler)     source!.setCancelHandler     {       self.fileHandle!.closeFile()     }     source!.resume()   }   init(url: URL) throws   {     self.url = url     createDispatchSource()   }   deinit   {     source!.cancel()   } } The problem is that I am getting nearly 7,000 notifications every time a change is detected, which is causing the CPU to get rather warm. This subject seems to be very poorly documented and after days searching, I don't seem to be able to make things any better. Any ideas please? Joanna
3
0
1.4k
May ’21
How to save and restore position of secondary NSViewController
I have a macOS storyboard with a main Window/View Controller and a couple of "accessory" view controllers that are shown via segues from buttons. The problem is, even having set the restoration ID of both controllers in the storyboard, that they always start at the same position, even if they have been moved. Can anyone help me to get them to reopen in the same place they were when they were closed?
3
0
2.2k
May ’21
NSViewRepresentable control not being updated until resize of form
I have created a wrapped NSTextField to allow for keystroke validation. class ValidatableTextField: NSTextField, NSTextViewDelegate {   var validationClosure: ((ValidatableTextField, NSRange, String?) -> Bool)?   func textView(_ textView: NSTextView, shouldChangeTextIn affectedCharRange: NSRange, replacementString: String?) -> Bool   {     return validationClosure?(self, affectedCharRange, replacementString) ?? true   } } struct CurrencyTextField<rootT : AnyObject, valueT> : NSViewRepresentable {   private var object: rootT?   private var keyPath: ReferenceWritableKeyPath<rootT, valueT>?     func makeCoordinator() -> Coordinator   {     return Coordinator(object, keyPath: keyPath)   }   init(object: rootT?, keyPath: ReferenceWritableKeyPath<rootT, valueT>?)   {     self.object = object     self.keyPath = keyPath   }   func makeNSView(context: Context) -> ValidatableTextField   {     let textField = ValidatableTextField()     textField.validationClosure = context.coordinator.validationClosure        return textField   }   func updateNSView(_ nsView: ValidatableTextField, context: Context)   {     guard let object = object,           let keyPath = keyPath else     {       return     }     nsView.stringValue = "\(object[keyPath: keyPath])"   }   class Coordinator : NSObject   {     var object: rootT?     var keyPath: ReferenceWritableKeyPath<rootT, valueT>?     init(_ object: rootT?, keyPath: ReferenceWritableKeyPath<rootT, valueT>?)     {       self.object = object       self.keyPath = keyPath     }     lazy var validationClosure: ((ValidatableTextField, NSRange, String?) -> Bool) =     {       textField, range, replacementString in       if replacementString!.isEmpty       {         return true       }       return replacementString?.contains(where:       {         (char: Character) -> Bool in         return char.isNumber || String(char) == Locale.autoupdatingCurrent.decimalSeparator       }) ?? false     }   } } struct ContentView: View {   @ObservedObject var person: Person   var body: some View   {     VStack     {       Spacer()       Text(person.name)       HStack       {         Spacer(minLength: 100)         CurrencyTextField<Person, Double>(object: person, keyPath: \.bankBalance)         Text("\(person.bankBalance)")         Spacer(minLength: 100)       }       Button(action:       {         self.person.bankBalance = 65.43       })       {         Text("Change bank balance")       }       Spacer()     }   } } My problem is that I am setting the bankBalance property on the Person and the Text next to the TextField is being updated but my CurrencyTextField is not. I may have 30 years of programming experience, including quite a few with Swift but SwiftUI, beyond the simplest of examples, is proving a bit difficult. What am I missing here please?
0
0
771
Nov ’20
Why would using Swift DispatchGroup load CPU to 1000%?
I have some code that calls an external library asynchronously to read EXIF keywords from image files : &#9;func readKeywords() &#9;{ &#9;&#9;keywordsForSelectedItems.removeAll() &#9;&#9; &#9;&#9;let group: DispatchGroup = .init() &#9;&#9; &#9;&#9;let queue: DispatchQueue = .init(label: "com.keywords") &#9;&#9; &#9;&#9;collectionView.selectionIndexPaths.forEach &#9;&#9;{ &#9;&#9;&#9;indexPath in &#9;&#9;&#9; &#9;&#9;&#9;let item = self.collectionView.item(at: indexPath) as! CollectionViewItem &#9;&#9;&#9; &#9;&#9;&#9;let url = item.url! &#9;&#9;&#9; &#9;&#9;&#9;group.enter() &#9;&#9;&#9; &#9;&#9;&#9;queue.async(group: group) &#9;&#9;&#9;{ &#9;&#9;&#9;&#9;do &#9;&#9;&#9;&#9;{ &#9;&#9;&#9;&#9;&#9;try ExifReader.keywords(for: url) &#9;&#9;&#9;&#9;&#9;{ &#9;&#9;&#9;&#9;&#9;&#9;keywords in &#9;&#9;&#9;&#9;&#9;&#9; &#9;&#9;&#9;&#9;&#9;&#9;if let keywords = keywords, &#9;&#9;&#9;&#9;&#9;&#9;&#9; keywords.count > 0 &#9;&#9;&#9;&#9;&#9;&#9;{ &#9;&#9;&#9;&#9;&#9;&#9;&#9;self.keywordsForSelectedItems.append(keywords) &#9;&#9;&#9;&#9;&#9;&#9;} &#9;&#9;&#9;&#9;&#9;&#9; &#9;&#9;&#9;&#9;&#9;&#9;group.leave() &#9;&#9;&#9;&#9;&#9;} &#9;&#9;&#9;&#9;} &#9;&#9;&#9;&#9;catch &#9;&#9;&#9;&#9;{ &#9;&#9;&#9;&#9;&#9;print(error.localizedDescription) &#9;&#9;&#9;&#9;} &#9;&#9;&#9;} &#9;&#9;} &#9;&#9;group.notify(queue: .main) &#9;&#9;{ &#9;&#9;&#9;let firstValue: [String]? = self.keywordsForSelectedItems.first &#9;&#9;&#9;{ &#9;&#9;&#9;&#9;keywords in &#9;&#9;&#9;&#9; &#9;&#9;&#9;&#9;keywords.count > 0 &#9;&#9;&#9;} &#9;&#9;&#9;&#9; &#9;&#9;&#9;if firstValue != nil &#9;&#9;&#9;{ &#9;&#9;&#9;&#9;let keywords = self.keywordsForSelectedItems.reduce(Set(firstValue!)) &#9;&#9;&#9;&#9;{ &#9;&#9;&#9;&#9;&#9;result, item in &#9;&#9;&#9;&#9;&#9; &#9;&#9;&#9;&#9;&#9;result.union(item) &#9;&#9;&#9;&#9;} &#9;&#9;&#9;&#9; &#9;&#9;&#9;&#9;self.tokenView.objectValue = keywords.sorted() &#9;&#9;&#9;} &#9;&#9;} &#9;} Before I added this code, everything worked fine and images were displayed perfectly in a CollectionView. But, since adding this code, as soon as this code has been called, the CPU fan starts to run, anywhere up to 1000% at times and it can block other asynchronous processes that normally load images in the background. Anyone got any ideas?
1
0
584
Oct ’20
Incorrect localization for NSLocationWhenInUseUsageDescription
Greetings allI have an app whose development language is French, which is also its base localization. I also localise it to English.The app makes use of both location services and the calendar.I have created an InfoPlist.strings file and localized both NSCalendarsUsageDescription and NSLocationWhenInUseUsageDescription.My problem is that, although the Base localization (French) of NSCalendarsUsageDescription is correctly used when requesting permission, I can only ever see the French text for NSLocationWhenInUseUsageDescription if I remove the key from the English version of InfoPlist.strings.Many thanks for any help you can give me.
1
0
1.2k
Oct ’16