Post

Replies

Boosts

Views

Activity

Reply to Trouble decoding array objects via NSKeyedArchiver / NSSecureCoding
Thanks πŸ™! I would've never guessed the issue was due to Objective-C types. With some trial and error I was able to get everything working, using decodeArrayOfObjects(ofClass:forKey:) for the simple arrays and decodeObject(of:forKey:) for the nested arrays. It was really important to list out NSNumber and NSArray, where appropriate, to avoid warnings. Essentially this is what I ended up with: let myIntegers = decodeArrayOfObjects(ofClass: NSNumber.self, forKey: "myIntegers") as! [Int] let myEmbedIntegers = decoder.decodeObject(of: [NSArray.self, NSNumber.self], forKey: "myEmbedIntegers") as! [[[Int]]] let myBooleans = decoder.decodeArrayOfObjects(ofClass: NSNumber.self, forKey: "myBooleans") as! [Bool] let myEmbedBooleans = decoder.decodeObject(of: [NSArray.self, NSNumber.self], forKey: "myEmbedBooleans") as! [[Bool]] And here's the whole amended class: import Foundation class MyData: NSObject, NSSecureCoding { public class var supportsSecureCoding: Bool { true } var id: Int? var isModified: Bool? var modifyDate: Date? var myIntegers: [Int]? var myEmbedIntegers: [[[Int]]]? var myBooleans: [Bool]? var myEmbedBooleans: [[Bool]]? init(_id: Int, _isModified: Bool, _modifyDate: Date, _myIntegers: [Int], _myEmbedIntegers: [[[Int]]], _myBooleans: [Bool], _myEmbedBooleans: [[Bool]]) { id = _id isModified = _isModified modifyDate = _modifyDate myIntegers = _myIntegers myEmbedIntegers = _myEmbedIntegers myBooleans = _myBooleans myEmbedBooleans = _myEmbedBooleans } required convenience init(coder decoder: NSCoder) { let id = decoder.decodeInteger(forKey: "id") let isModified = decoder.decodeBool(forKey: "isModified") let modifyDate = decoder.decodeObject(of: NSDate.self, forKey: "modifyDate")! as Date let myIntegers = decodeArrayOfObjects(ofClass: NSNumber.self, forKey: "myIntegers") as! [Int] let myEmbedIntegers = decoder.decodeObject(of: [NSArray.self, NSNumber.self], forKey: "myEmbedIntegers") as! [[[Int]]] let myBooleans = decoder.decodeArrayOfObjects(ofClass: NSNumber.self, forKey: "myBooleans") as! [Bool] let myEmbedBooleans = decoder.decodeObject(of: [NSArray.self, NSNumber.self], forKey: "myEmbedBooleans") as! [[Bool]] self.init(_id: id, _isModified: isModified, _modifyDate: modifyDate, _myIntegers: myIntegers, _myEmbedIntegers: myEmbedIntegers, _myBooleans: myBooleans, _myEmbedBooleans: myEmbedBooleans) } func encode(with coder: NSCoder) { coder.encode(id!, forKey: "id") coder.encode(isModified!, forKey: "isModified") coder.encode(modifyDate!, forKey: "modifyDate") coder.encode(myIntegers!, forKey: "myIntegers") coder.encode(myEmbedIntegers!, forKey: "myEmbedIntegers") coder.encode(myBooleans!, forKey: "myBooleans") coder.encode(myEmbedBooleans!, forKey: "myEmbedBooleans") } } Thanks again for the expert help! I can finally move on to more enjoyable challenges 🀣.
3w
Reply to Trouble decoding array objects via NSKeyedArchiver / NSSecureCoding
Certainly πŸ‘Œ! The full class is below. Note that this is a mock example, but it does reflect and represent my real class pretty identically. import Foundation class MyData: NSObject, NSSecureCoding { public class var supportsSecureCoding: Bool { true } var id: Int? var isModified: Bool? var modifyDate: Date? var myIntegers: [Int]? var myEmbedIntegers: [[[Int]]]? var myBooleans: [Bool]? var myEmbedBooleans: [[Bool]]? init(_id: Int, _isModified: Bool, _modifyDate: Date, _myIntegers: [Int], _myEmbedIntegers: [[[Int]]], _myBooleans: [Bool], _myEmbedBooleans: [[Bool]]) { id = _id isModified = _isModified modifyDate = _modifyDate myIntegers = _myIntegers myEmbedIntegers = _myEmbedIntegers myBooleans = _myBooleans myEmbedBooleans = _myEmbedBooleans } required convenience init(coder decoder: NSCoder) { let id = decoder.decodeInteger(forKey: "id") let isModified = decoder.decodeBool(forKey: "isModified") let modifyDate = decoder.decodeObject(of: NSDate.self, forKey: "modifyDate")! as Date // Here's where the errors happen: The four lines underneath return nil, which in this case cause crashes due to force unwrapping. let myIntegers = decoder.decodeObject(forKey: "myIntegers") as! [Int] let myEmbedIntegers = decoder.decodeObject(forKey: "myEmbedIntegers") as! [[[Int]]] let myBooleans = decoder.decodeObject(forKey: "myBooleans") as! [Bool] let myEmbedBooleans = decoder.decodeObject(forKey: "myEmbedBooleans") as! [[Bool]] self.init(_id: id, _isModified: isModified, _modifyDate: modifyDate, _myIntegers: myIntegers, _myEmbedIntegers: myEmbedIntegers, _myBooleans: myBooleans, _myEmbedBooleans: myEmbedBooleans) } func encode(with coder: NSCoder) { coder.encode(id!, forKey: "id") coder.encode(isModified!, forKey: "isModified") coder.encode(modifyDate!, forKey: "modifyDate") coder.encode(myIntegers!, forKey: "myIntegers") coder.encode(myEmbedIntegers!, forKey: "myEmbedIntegers") coder.encode(myBooleans!, forKey: "myBooleans") coder.encode(myEmbedBooleans!, forKey: "myEmbedBooleans") } } And they are called using these interfaces: public func loadMyData() -> [MyData]? { do { let data = try Data(contentsOf: myDataPath()) let myData = try NSKeyedUnarchiver.unarchivedArrayOfObjects(ofClass: MyData.self, from: data) return myData } catch {} return nil } public func saveMyData(_ mixData: [MyData]) { do { let data = try NSKeyedArchiver.archivedData(withRootObject: myData, requiringSecureCoding: false) try data.write(to: myDataPath()) } catch {} } private func myDataPath() -> URL { return (FileManager.default.urls(for: .documentDirectory, in: .userDomainMask).first! as URL).appendingPathComponent("MyData.plist") }
3w
Reply to Place UITextField behind SKNode
Well, there are differences between SKLabelNode and UITextField for sure, but most of the functionality is there with SKLabelNode. Frankly, you would need some special reason to not use SKLabelNode if your app is primarily in SpriteKit. I would say there's really only two cases where you want to mix UIKit and SpriteKit:You're making a UIKit app, which uses some element of SpriteKit almost like an embedded thing, for example a mini-game within a normal iPhone app.You're making a SpriteKit app, but you need some functionality from UIKit that SpriteKit doesn't boast (such as a UIWebView, though nowadays even that is deprecated and you should use WKWebView instead, or the automatic scrolling of a UITextField, although I would even recommend programming your own scrolling).Generally mixing should be done with some direct and thought-out goal in mind, because stuff can go wrong very quickly. For example UIKit and SpriteKit live on different times (SpriteKit has it's own game time that affects everything within the package, but wouldn't affect the UIKit elements) and xy-coordinate systems are different.In short, you really should be using SKLabelNode unless you have some exceptional reason to do otherwise. Feel free to present your case.About the rounded corners SKShapeNode issue, I would mainly recommend looking into 9-slice features of SKSpriteNode (https://developer.apple.com/library/archive/documentation/GraphicsAnimation/Conceptual/SpriteKit_PG/Sprites/Sprites.html#//apple_ref/doc/uid/TP40013043-CH9-SW10). Native 9-slice features in SpriteKit allow you to designate parts of an image as corners, sides and middle, so the whole element gets stretched dynamically.That said, I would also recommend considering if you really need dynamic stretching. I've done this stuff for years and years now with a good amount of success too πŸ™‚, so in my experience the simplest and most minimal solutions are usually the best. With apps and games users want understandable and simple, and you end up with a much easier workload yourself, too, with that approach. It's a win-win!
Feb ’20
Reply to Place UITextField behind SKNode
Since tutorialNotification is being added to the SKScene itself while txtName actually adds an entirely new Subview, the latter would always be on top of the former. Essentially you're adding a new container on top of the container that includes the SKScene, which will give you a hard time trying to hack depth.Is there a reason you're not using SKLabelNode for the text field? That would solve your issue right away and make it easier to control anyhow, as it would all be within the SKScene. I guess the overall question is: what is your reason for mixing UIKit and SpriteKit?On a side note, I always advise against using SKShapeNode. I violently ripped out all SKShapeNodes from my apps years ago, after I found out a number of issues with them. I was going on a shape bender at some point, but now I'm fully a SKSpriteNode and image assets guy.
Feb ’20