Merge branch 'main' of https://github.com/Ranchero-Software/NetNewsWire into main
This commit is contained in:
commit
d821fbd761
4
.gitmodules
vendored
4
.gitmodules
vendored
@ -1,4 +0,0 @@
|
|||||||
[submodule "submodules/Sparkle"]
|
|
||||||
path = submodules/Sparkle
|
|
||||||
url = https://github.com/brentsimmons/Sparkle
|
|
||||||
branch = ui-separation-and-xpc
|
|
@ -8,6 +8,7 @@
|
|||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
import os.log
|
import os.log
|
||||||
|
import RSCore
|
||||||
import RSWeb
|
import RSWeb
|
||||||
import RSParser
|
import RSParser
|
||||||
import CloudKit
|
import CloudKit
|
||||||
|
@ -8,6 +8,7 @@
|
|||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
import os.log
|
import os.log
|
||||||
|
import RSCore
|
||||||
import RSParser
|
import RSParser
|
||||||
import RSWeb
|
import RSWeb
|
||||||
import CloudKit
|
import CloudKit
|
||||||
|
@ -8,6 +8,7 @@
|
|||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
import os.log
|
import os.log
|
||||||
|
import RSCore
|
||||||
import RSParser
|
import RSParser
|
||||||
import RSWeb
|
import RSWeb
|
||||||
import CloudKit
|
import CloudKit
|
||||||
|
@ -1,98 +0,0 @@
|
|||||||
//
|
|
||||||
// CloudKitError.swift
|
|
||||||
// Account
|
|
||||||
//
|
|
||||||
// Created by Maurice Parker on 3/26/20.
|
|
||||||
// Copyright © 2020 Ranchero Software, LLC. All rights reserved.
|
|
||||||
//
|
|
||||||
// Derived from https://github.com/caiyue1993/IceCream
|
|
||||||
|
|
||||||
import Foundation
|
|
||||||
import CloudKit
|
|
||||||
|
|
||||||
class CloudKitError: LocalizedError {
|
|
||||||
|
|
||||||
let error: Error
|
|
||||||
|
|
||||||
init(_ error: Error) {
|
|
||||||
self.error = error
|
|
||||||
}
|
|
||||||
|
|
||||||
public var errorDescription: String? {
|
|
||||||
guard let ckError = error as? CKError else {
|
|
||||||
return error.localizedDescription
|
|
||||||
}
|
|
||||||
|
|
||||||
switch ckError.code {
|
|
||||||
case .alreadyShared:
|
|
||||||
return NSLocalizedString("Already Shared: a record or share cannot be saved because doing so would cause the same hierarchy of records to exist in multiple shares.", comment: "Known iCloud Error")
|
|
||||||
case .assetFileModified:
|
|
||||||
return NSLocalizedString("Asset File Modified: the content of the specified asset file was modified while being saved.", comment: "Known iCloud Error")
|
|
||||||
case .assetFileNotFound:
|
|
||||||
return NSLocalizedString("Asset File Not Found: the specified asset file is not found.", comment: "Known iCloud Error")
|
|
||||||
case .badContainer:
|
|
||||||
return NSLocalizedString("Bad Container: the specified container is unknown or unauthorized.", comment: "Known iCloud Error")
|
|
||||||
case .badDatabase:
|
|
||||||
return NSLocalizedString("Bad Database: the operation could not be completed on the given database.", comment: "Known iCloud Error")
|
|
||||||
case .batchRequestFailed:
|
|
||||||
return NSLocalizedString("Batch Request Failed: the entire batch was rejected.", comment: "Known iCloud Error")
|
|
||||||
case .changeTokenExpired:
|
|
||||||
return NSLocalizedString("Change Token Expired: the previous server change token is too old.", comment: "Known iCloud Error")
|
|
||||||
case .constraintViolation:
|
|
||||||
return NSLocalizedString("Constraint Violation: the server rejected the request because of a conflict with a unique field.", comment: "Known iCloud Error")
|
|
||||||
case .incompatibleVersion:
|
|
||||||
return NSLocalizedString("Incompatible Version: your app version is older than the oldest version allowed.", comment: "Known iCloud Error")
|
|
||||||
case .internalError:
|
|
||||||
return NSLocalizedString("Internal Error: a nonrecoverable error was encountered by CloudKit.", comment: "Known iCloud Error")
|
|
||||||
case .invalidArguments:
|
|
||||||
return NSLocalizedString("Invalid Arguments: the specified request contains bad information.", comment: "Known iCloud Error")
|
|
||||||
case .limitExceeded:
|
|
||||||
return NSLocalizedString("Limit Exceeded: the request to the server is too large.", comment: "Known iCloud Error")
|
|
||||||
case .managedAccountRestricted:
|
|
||||||
return NSLocalizedString("Managed Account Restricted: the request was rejected due to a managed-account restriction.", comment: "Known iCloud Error")
|
|
||||||
case .missingEntitlement:
|
|
||||||
return NSLocalizedString("Missing Entitlement: the app is missing a required entitlement.", comment: "Known iCloud Error")
|
|
||||||
case .networkUnavailable:
|
|
||||||
return NSLocalizedString("Network Unavailable: the internet connection appears to be offline.", comment: "Known iCloud Error")
|
|
||||||
case .networkFailure:
|
|
||||||
return NSLocalizedString("Network Failure: the internet connection appears to be offline.", comment: "Known iCloud Error")
|
|
||||||
case .notAuthenticated:
|
|
||||||
return NSLocalizedString("Not Authenticated: to use the iCloud account, you must enable iCloud syncing. Go to device Settings, sign in to iCloud, then in the app settings, be sure the iCloud feature is enabled.", comment: "Known iCloud Error")
|
|
||||||
case .operationCancelled:
|
|
||||||
return NSLocalizedString("Operation Cancelled: the operation was explicitly canceled.", comment: "Known iCloud Error")
|
|
||||||
case .partialFailure:
|
|
||||||
return NSLocalizedString("Partial Failure: some items failed, but the operation succeeded overall.", comment: "Known iCloud Error")
|
|
||||||
case .participantMayNeedVerification:
|
|
||||||
return NSLocalizedString("Participant May Need Verification: you are not a member of the share.", comment: "Known iCloud Error")
|
|
||||||
case .permissionFailure:
|
|
||||||
return NSLocalizedString("Permission Failure: to use this app, you must enable iCloud syncing. Go to device Settings, sign in to iCloud, then in the app settings, be sure the iCloud feature is enabled.", comment: "Known iCloud Error")
|
|
||||||
case .quotaExceeded:
|
|
||||||
return NSLocalizedString("Quota Exceeded: saving would exceed your current iCloud storage quota.", comment: "Known iCloud Error")
|
|
||||||
case .referenceViolation:
|
|
||||||
return NSLocalizedString("Reference Violation: the target of a record's parent or share reference was not found.", comment: "Known iCloud Error")
|
|
||||||
case .requestRateLimited:
|
|
||||||
return NSLocalizedString("Request Rate Limited: transfers to and from the server are being rate limited at this time.", comment: "Known iCloud Error")
|
|
||||||
case .serverRecordChanged:
|
|
||||||
return NSLocalizedString("Server Record Changed: the record was rejected because the version on the server is different.", comment: "Known iCloud Error")
|
|
||||||
case .serverRejectedRequest:
|
|
||||||
return NSLocalizedString("Server Rejected Request", comment: "Known iCloud Error")
|
|
||||||
case .serverResponseLost:
|
|
||||||
return NSLocalizedString("Server Response Lost", comment: "Known iCloud Error")
|
|
||||||
case .serviceUnavailable:
|
|
||||||
return NSLocalizedString("Service Unavailable: Please try again.", comment: "Known iCloud Error")
|
|
||||||
case .tooManyParticipants:
|
|
||||||
return NSLocalizedString("Too Many Participants: a share cannot be saved because too many participants are attached to the share.", comment: "Known iCloud Error")
|
|
||||||
case .unknownItem:
|
|
||||||
return NSLocalizedString("Unknown Item: the specified record does not exist.", comment: "Known iCloud Error")
|
|
||||||
case .userDeletedZone:
|
|
||||||
return NSLocalizedString("User Deleted Zone: the user has deleted this zone from the settings UI.", comment: "Known iCloud Error")
|
|
||||||
case .zoneBusy:
|
|
||||||
return NSLocalizedString("Zone Busy: the server is too busy to handle the zone operation.", comment: "Known iCloud Error")
|
|
||||||
case .zoneNotFound:
|
|
||||||
return NSLocalizedString("Zone Not Found: the specified record zone does not exist on the server.", comment: "Known iCloud Error")
|
|
||||||
default:
|
|
||||||
return NSLocalizedString("Unhandled Error.", comment: "Unknown iCloud Error")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -1,687 +0,0 @@
|
|||||||
//
|
|
||||||
// CloudKitZone.swift
|
|
||||||
// Account
|
|
||||||
//
|
|
||||||
// Created by Maurice Parker on 3/21/20.
|
|
||||||
// Copyright © 2020 Ranchero Software, LLC. All rights reserved.
|
|
||||||
//
|
|
||||||
|
|
||||||
import CloudKit
|
|
||||||
import os.log
|
|
||||||
import RSWeb
|
|
||||||
|
|
||||||
enum CloudKitZoneError: LocalizedError {
|
|
||||||
case userDeletedZone
|
|
||||||
case invalidParameter
|
|
||||||
case unknown
|
|
||||||
|
|
||||||
var errorDescription: String? {
|
|
||||||
if case .userDeletedZone = self {
|
|
||||||
return NSLocalizedString("The iCloud data was deleted. Please delete the NetNewsWire iCloud account and add it again to continue using NetNewsWire's iCloud support.", comment: "User deleted zone.")
|
|
||||||
} else {
|
|
||||||
return NSLocalizedString("An unexpected CloudKit error occurred.", comment: "An unexpected CloudKit error occurred.")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protocol CloudKitZoneDelegate: class {
|
|
||||||
func cloudKitDidModify(changed: [CKRecord], deleted: [CloudKitRecordKey], completion: @escaping (Result<Void, Error>) -> Void);
|
|
||||||
}
|
|
||||||
|
|
||||||
typealias CloudKitRecordKey = (recordType: CKRecord.RecordType, recordID: CKRecord.ID)
|
|
||||||
|
|
||||||
protocol CloudKitZone: class {
|
|
||||||
|
|
||||||
static var zoneID: CKRecordZone.ID { get }
|
|
||||||
static var qualityOfService: QualityOfService { get }
|
|
||||||
|
|
||||||
var log: OSLog { get }
|
|
||||||
|
|
||||||
var container: CKContainer? { get }
|
|
||||||
var database: CKDatabase? { get }
|
|
||||||
var delegate: CloudKitZoneDelegate? { get set }
|
|
||||||
|
|
||||||
/// Reset the change token used to determine what point in time we are doing changes fetches
|
|
||||||
func resetChangeToken()
|
|
||||||
|
|
||||||
/// Generates a new CKRecord.ID using a UUID for the record's name
|
|
||||||
func generateRecordID() -> CKRecord.ID
|
|
||||||
|
|
||||||
/// Subscribe to changes at a zone level
|
|
||||||
func subscribeToZoneChanges()
|
|
||||||
|
|
||||||
/// Process a remove notification
|
|
||||||
func receiveRemoteNotification(userInfo: [AnyHashable : Any], completion: @escaping () -> Void)
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
extension CloudKitZone {
|
|
||||||
|
|
||||||
// My observation has been that QoS is treated differently for CloudKit operations on macOS vs iOS.
|
|
||||||
// .userInitiated is too aggressive on iOS and can lead the UI slowing down and appearing to block.
|
|
||||||
// .default (or lower) on macOS will sometimes hang for extended periods of time and appear to hang.
|
|
||||||
static var qualityOfService: QualityOfService {
|
|
||||||
#if os(macOS)
|
|
||||||
return .userInitiated
|
|
||||||
#else
|
|
||||||
return .default
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Reset the change token used to determine what point in time we are doing changes fetches
|
|
||||||
func resetChangeToken() {
|
|
||||||
changeToken = nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func generateRecordID() -> CKRecord.ID {
|
|
||||||
return CKRecord.ID(recordName: UUID().uuidString, zoneID: Self.zoneID)
|
|
||||||
}
|
|
||||||
|
|
||||||
func retryIfPossible(after: Double, block: @escaping () -> ()) {
|
|
||||||
let delayTime = DispatchTime.now() + after
|
|
||||||
DispatchQueue.main.asyncAfter(deadline: delayTime, execute: {
|
|
||||||
block()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func receiveRemoteNotification(userInfo: [AnyHashable : Any], completion: @escaping () -> Void) {
|
|
||||||
let note = CKRecordZoneNotification(fromRemoteNotificationDictionary: userInfo)
|
|
||||||
guard note?.recordZoneID?.zoneName == Self.zoneID.zoneName else {
|
|
||||||
completion()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
fetchChangesInZone() { result in
|
|
||||||
if case .failure(let error) = result {
|
|
||||||
os_log(.error, log: self.log, "%@ zone remote notification fetch error: %@", Self.zoneID.zoneName, error.localizedDescription)
|
|
||||||
}
|
|
||||||
completion()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Creates the zone record
|
|
||||||
func createZoneRecord(completion: @escaping (Result<Void, Error>) -> Void) {
|
|
||||||
guard let database = database else {
|
|
||||||
completion(.failure(CloudKitZoneError.unknown))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
database.save(CKRecordZone(zoneID: Self.zoneID)) { (recordZone, error) in
|
|
||||||
if let error = error {
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(CloudKitError(error)))
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.success(()))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Subscribes to zone changes
|
|
||||||
func subscribeToZoneChanges() {
|
|
||||||
let subscription = CKRecordZoneSubscription(zoneID: Self.zoneID)
|
|
||||||
|
|
||||||
let info = CKSubscription.NotificationInfo()
|
|
||||||
info.shouldSendContentAvailable = true
|
|
||||||
subscription.notificationInfo = info
|
|
||||||
|
|
||||||
save(subscription) { result in
|
|
||||||
if case .failure(let error) = result {
|
|
||||||
os_log(.error, log: self.log, "%@ zone subscribe to changes error: %@", Self.zoneID.zoneName, error.localizedDescription)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Issue a CKQuery and return the resulting CKRecords.s
|
|
||||||
func query(_ query: CKQuery, completion: @escaping (Result<[CKRecord], Error>) -> Void) {
|
|
||||||
guard let database = database else {
|
|
||||||
completion(.failure(CloudKitZoneError.unknown))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
database.perform(query, inZoneWith: Self.zoneID) { [weak self] records, error in
|
|
||||||
guard let self = self else {
|
|
||||||
completion(.failure(CloudKitZoneError.unknown))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
switch CloudKitZoneResult.resolve(error) {
|
|
||||||
case .success:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
if let records = records {
|
|
||||||
completion(.success(records))
|
|
||||||
} else {
|
|
||||||
completion(.failure(CloudKitZoneError.unknown))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
case .zoneNotFound:
|
|
||||||
self.createZoneRecord() { result in
|
|
||||||
switch result {
|
|
||||||
case .success:
|
|
||||||
self.query(query, completion: completion)
|
|
||||||
case .failure(let error):
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(error))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
case .retry(let timeToWait):
|
|
||||||
os_log(.error, log: self.log, "%@ zone query retry in %f seconds.", Self.zoneID.zoneName, timeToWait)
|
|
||||||
self.retryIfPossible(after: timeToWait) {
|
|
||||||
self.query(query, completion: completion)
|
|
||||||
}
|
|
||||||
case .userDeletedZone:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(CloudKitZoneError.userDeletedZone))
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(CloudKitError(error!)))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Fetch a CKRecord by using its externalID
|
|
||||||
func fetch(externalID: String?, completion: @escaping (Result<CKRecord, Error>) -> Void) {
|
|
||||||
guard let externalID = externalID else {
|
|
||||||
completion(.failure(CloudKitZoneError.invalidParameter))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
let recordID = CKRecord.ID(recordName: externalID, zoneID: Self.zoneID)
|
|
||||||
|
|
||||||
database?.fetch(withRecordID: recordID) { [weak self] record, error in
|
|
||||||
guard let self = self else {
|
|
||||||
completion(.failure(CloudKitZoneError.unknown))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
switch CloudKitZoneResult.resolve(error) {
|
|
||||||
case .success:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
if let record = record {
|
|
||||||
completion(.success(record))
|
|
||||||
} else {
|
|
||||||
completion(.failure(CloudKitZoneError.unknown))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
case .zoneNotFound:
|
|
||||||
self.createZoneRecord() { result in
|
|
||||||
switch result {
|
|
||||||
case .success:
|
|
||||||
self.fetch(externalID: externalID, completion: completion)
|
|
||||||
case .failure(let error):
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(error))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
case .retry(let timeToWait):
|
|
||||||
os_log(.error, log: self.log, "%@ zone fetch retry in %f seconds.", Self.zoneID.zoneName, timeToWait)
|
|
||||||
self.retryIfPossible(after: timeToWait) {
|
|
||||||
self.fetch(externalID: externalID, completion: completion)
|
|
||||||
}
|
|
||||||
case .userDeletedZone:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(CloudKitZoneError.userDeletedZone))
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(CloudKitError(error!)))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Save the CKRecord
|
|
||||||
func save(_ record: CKRecord, completion: @escaping (Result<Void, Error>) -> Void) {
|
|
||||||
modify(recordsToSave: [record], recordIDsToDelete: [], completion: completion)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Save the CKRecords
|
|
||||||
func save(_ records: [CKRecord], completion: @escaping (Result<Void, Error>) -> Void) {
|
|
||||||
modify(recordsToSave: records, recordIDsToDelete: [], completion: completion)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Saves or modifies the records as long as they are unchanged relative to the local version
|
|
||||||
func saveIfNew(_ records: [CKRecord], completion: @escaping (Result<Void, Error>) -> Void) {
|
|
||||||
let op = CKModifyRecordsOperation(recordsToSave: records, recordIDsToDelete: [CKRecord.ID]())
|
|
||||||
op.savePolicy = .ifServerRecordUnchanged
|
|
||||||
op.isAtomic = false
|
|
||||||
op.qualityOfService = Self.qualityOfService
|
|
||||||
|
|
||||||
op.modifyRecordsCompletionBlock = { [weak self] (_, _, error) in
|
|
||||||
|
|
||||||
guard let self = self else { return }
|
|
||||||
|
|
||||||
switch CloudKitZoneResult.resolve(error) {
|
|
||||||
case .success, .partialFailure:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.success(()))
|
|
||||||
}
|
|
||||||
|
|
||||||
case .zoneNotFound:
|
|
||||||
self.createZoneRecord() { result in
|
|
||||||
switch result {
|
|
||||||
case .success:
|
|
||||||
self.saveIfNew(records, completion: completion)
|
|
||||||
case .failure(let error):
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(error))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
case .userDeletedZone:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(CloudKitZoneError.userDeletedZone))
|
|
||||||
}
|
|
||||||
|
|
||||||
case .retry(let timeToWait):
|
|
||||||
self.retryIfPossible(after: timeToWait) {
|
|
||||||
self.saveIfNew(records, completion: completion)
|
|
||||||
}
|
|
||||||
|
|
||||||
case .limitExceeded:
|
|
||||||
|
|
||||||
let chunkedRecords = records.chunked(into: 300)
|
|
||||||
|
|
||||||
let group = DispatchGroup()
|
|
||||||
var errorOccurred = false
|
|
||||||
|
|
||||||
for chunk in chunkedRecords {
|
|
||||||
group.enter()
|
|
||||||
self.saveIfNew(chunk) { result in
|
|
||||||
if case .failure(let error) = result {
|
|
||||||
os_log(.error, log: self.log, "%@ zone modify records error: %@", Self.zoneID.zoneName, error.localizedDescription)
|
|
||||||
errorOccurred = true
|
|
||||||
}
|
|
||||||
group.leave()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
group.notify(queue: DispatchQueue.main) {
|
|
||||||
if errorOccurred {
|
|
||||||
completion(.failure(CloudKitZoneError.unknown))
|
|
||||||
} else {
|
|
||||||
completion(.success(()))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
default:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(CloudKitError(error!)))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
database?.add(op)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Save the CKSubscription
|
|
||||||
func save(_ subscription: CKSubscription, completion: @escaping (Result<CKSubscription, Error>) -> Void) {
|
|
||||||
database?.save(subscription) { [weak self] savedSubscription, error in
|
|
||||||
guard let self = self else {
|
|
||||||
completion(.failure(CloudKitZoneError.unknown))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
switch CloudKitZoneResult.resolve(error) {
|
|
||||||
case .success:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.success((savedSubscription!)))
|
|
||||||
}
|
|
||||||
case .zoneNotFound:
|
|
||||||
self.createZoneRecord() { result in
|
|
||||||
switch result {
|
|
||||||
case .success:
|
|
||||||
self.save(subscription, completion: completion)
|
|
||||||
case .failure(let error):
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(error))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
case .retry(let timeToWait):
|
|
||||||
os_log(.error, log: self.log, "%@ zone save subscription retry in %f seconds.", Self.zoneID.zoneName, timeToWait)
|
|
||||||
self.retryIfPossible(after: timeToWait) {
|
|
||||||
self.save(subscription, completion: completion)
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(CloudKitError(error!)))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Delete CKRecords using a CKQuery
|
|
||||||
func delete(ckQuery: CKQuery, completion: @escaping (Result<Void, Error>) -> Void) {
|
|
||||||
|
|
||||||
var records = [CKRecord]()
|
|
||||||
|
|
||||||
let op = CKQueryOperation(query: ckQuery)
|
|
||||||
op.qualityOfService = Self.qualityOfService
|
|
||||||
op.recordFetchedBlock = { record in
|
|
||||||
records.append(record)
|
|
||||||
}
|
|
||||||
|
|
||||||
op.queryCompletionBlock = { [weak self] (cursor, error) in
|
|
||||||
guard let self = self else {
|
|
||||||
completion(.failure(CloudKitZoneError.unknown))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
if let cursor = cursor {
|
|
||||||
self.delete(cursor: cursor, carriedRecords: records, completion: completion)
|
|
||||||
} else {
|
|
||||||
guard !records.isEmpty else {
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.success(()))
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
let recordIDs = records.map { $0.recordID }
|
|
||||||
self.modify(recordsToSave: [], recordIDsToDelete: recordIDs, completion: completion)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
database?.add(op)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Delete CKRecords using a CKQuery
|
|
||||||
func delete(cursor: CKQueryOperation.Cursor, carriedRecords: [CKRecord], completion: @escaping (Result<Void, Error>) -> Void) {
|
|
||||||
|
|
||||||
var records = [CKRecord]()
|
|
||||||
|
|
||||||
let op = CKQueryOperation(cursor: cursor)
|
|
||||||
op.qualityOfService = Self.qualityOfService
|
|
||||||
op.recordFetchedBlock = { record in
|
|
||||||
records.append(record)
|
|
||||||
}
|
|
||||||
|
|
||||||
op.queryCompletionBlock = { [weak self] (cursor, error) in
|
|
||||||
guard let self = self else {
|
|
||||||
completion(.failure(CloudKitZoneError.unknown))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
records.append(contentsOf: carriedRecords)
|
|
||||||
|
|
||||||
if let cursor = cursor {
|
|
||||||
self.delete(cursor: cursor, carriedRecords: records, completion: completion)
|
|
||||||
} else {
|
|
||||||
let recordIDs = records.map { $0.recordID }
|
|
||||||
self.modify(recordsToSave: [], recordIDsToDelete: recordIDs, completion: completion)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
database?.add(op)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Delete a CKRecord using its recordID
|
|
||||||
func delete(recordID: CKRecord.ID, completion: @escaping (Result<Void, Error>) -> Void) {
|
|
||||||
modify(recordsToSave: [], recordIDsToDelete: [recordID], completion: completion)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Delete CKRecords
|
|
||||||
func delete(recordIDs: [CKRecord.ID], completion: @escaping (Result<Void, Error>) -> Void) {
|
|
||||||
modify(recordsToSave: [], recordIDsToDelete: recordIDs, completion: completion)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Delete a CKRecord using its externalID
|
|
||||||
func delete(externalID: String?, completion: @escaping (Result<Void, Error>) -> Void) {
|
|
||||||
guard let externalID = externalID else {
|
|
||||||
completion(.failure(CloudKitZoneError.invalidParameter))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
let recordID = CKRecord.ID(recordName: externalID, zoneID: Self.zoneID)
|
|
||||||
modify(recordsToSave: [], recordIDsToDelete: [recordID], completion: completion)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Delete a CKSubscription
|
|
||||||
func delete(subscriptionID: String, completion: @escaping (Result<Void, Error>) -> Void) {
|
|
||||||
database?.delete(withSubscriptionID: subscriptionID) { [weak self] _, error in
|
|
||||||
guard let self = self else {
|
|
||||||
completion(.failure(CloudKitZoneError.unknown))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
switch CloudKitZoneResult.resolve(error) {
|
|
||||||
case .success:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.success(()))
|
|
||||||
}
|
|
||||||
case .retry(let timeToWait):
|
|
||||||
os_log(.error, log: self.log, "%@ zone delete subscription retry in %f seconds.", Self.zoneID.zoneName, timeToWait)
|
|
||||||
self.retryIfPossible(after: timeToWait) {
|
|
||||||
self.delete(subscriptionID: subscriptionID, completion: completion)
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(CloudKitError(error!)))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Modify and delete the supplied CKRecords and CKRecord.IDs
|
|
||||||
func modify(recordsToSave: [CKRecord], recordIDsToDelete: [CKRecord.ID], completion: @escaping (Result<Void, Error>) -> Void) {
|
|
||||||
guard !(recordsToSave.isEmpty && recordIDsToDelete.isEmpty) else {
|
|
||||||
completion(.success(()))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
let op = CKModifyRecordsOperation(recordsToSave: recordsToSave, recordIDsToDelete: recordIDsToDelete)
|
|
||||||
op.savePolicy = .changedKeys
|
|
||||||
op.isAtomic = true
|
|
||||||
op.qualityOfService = Self.qualityOfService
|
|
||||||
|
|
||||||
op.modifyRecordsCompletionBlock = { [weak self] (_, _, error) in
|
|
||||||
|
|
||||||
guard let self = self else {
|
|
||||||
completion(.failure(CloudKitZoneError.unknown))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
switch CloudKitZoneResult.resolve(error) {
|
|
||||||
case .success:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.success(()))
|
|
||||||
}
|
|
||||||
case .zoneNotFound:
|
|
||||||
self.createZoneRecord() { result in
|
|
||||||
switch result {
|
|
||||||
case .success:
|
|
||||||
self.modify(recordsToSave: recordsToSave, recordIDsToDelete: recordIDsToDelete, completion: completion)
|
|
||||||
case .failure(let error):
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(error))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
case .userDeletedZone:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(CloudKitZoneError.userDeletedZone))
|
|
||||||
}
|
|
||||||
case .retry(let timeToWait):
|
|
||||||
os_log(.error, log: self.log, "%@ zone modify retry in %f seconds.", Self.zoneID.zoneName, timeToWait)
|
|
||||||
self.retryIfPossible(after: timeToWait) {
|
|
||||||
self.modify(recordsToSave: recordsToSave, recordIDsToDelete: recordIDsToDelete, completion: completion)
|
|
||||||
}
|
|
||||||
case .limitExceeded:
|
|
||||||
let recordToSaveChunks = recordsToSave.chunked(into: 300)
|
|
||||||
let recordIDsToDeleteChunks = recordIDsToDelete.chunked(into: 300)
|
|
||||||
|
|
||||||
let group = DispatchGroup()
|
|
||||||
var errorOccurred = false
|
|
||||||
|
|
||||||
for chunk in recordToSaveChunks {
|
|
||||||
group.enter()
|
|
||||||
self.modify(recordsToSave: chunk, recordIDsToDelete: []) { result in
|
|
||||||
if case .failure(let error) = result {
|
|
||||||
os_log(.error, log: self.log, "%@ zone modify records error: %@", Self.zoneID.zoneName, error.localizedDescription)
|
|
||||||
errorOccurred = true
|
|
||||||
}
|
|
||||||
group.leave()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for chunk in recordIDsToDeleteChunks {
|
|
||||||
group.enter()
|
|
||||||
self.modify(recordsToSave: [], recordIDsToDelete: chunk) { result in
|
|
||||||
if case .failure(let error) = result {
|
|
||||||
os_log(.error, log: self.log, "%@ zone modify records error: %@", Self.zoneID.zoneName, error.localizedDescription)
|
|
||||||
errorOccurred = true
|
|
||||||
}
|
|
||||||
group.leave()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
group.notify(queue: DispatchQueue.global(qos: .background)) {
|
|
||||||
if errorOccurred {
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(CloudKitZoneError.unknown))
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.success(()))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
default:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(CloudKitError(error!)))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
database?.add(op)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Fetch all the changes in the CKZone since the last time we checked
|
|
||||||
func fetchChangesInZone(completion: @escaping (Result<Void, Error>) -> Void) {
|
|
||||||
|
|
||||||
var savedChangeToken = changeToken
|
|
||||||
|
|
||||||
var changedRecords = [CKRecord]()
|
|
||||||
var deletedRecordKeys = [CloudKitRecordKey]()
|
|
||||||
|
|
||||||
let zoneConfig = CKFetchRecordZoneChangesOperation.ZoneConfiguration()
|
|
||||||
zoneConfig.previousServerChangeToken = changeToken
|
|
||||||
let op = CKFetchRecordZoneChangesOperation(recordZoneIDs: [Self.zoneID], configurationsByRecordZoneID: [Self.zoneID: zoneConfig])
|
|
||||||
op.fetchAllChanges = true
|
|
||||||
op.qualityOfService = Self.qualityOfService
|
|
||||||
|
|
||||||
op.recordZoneChangeTokensUpdatedBlock = { zoneID, token, _ in
|
|
||||||
savedChangeToken = token
|
|
||||||
}
|
|
||||||
|
|
||||||
op.recordChangedBlock = { record in
|
|
||||||
changedRecords.append(record)
|
|
||||||
}
|
|
||||||
|
|
||||||
op.recordWithIDWasDeletedBlock = { recordID, recordType in
|
|
||||||
let recordKey = CloudKitRecordKey(recordType: recordType, recordID: recordID)
|
|
||||||
deletedRecordKeys.append(recordKey)
|
|
||||||
}
|
|
||||||
|
|
||||||
op.recordZoneFetchCompletionBlock = { zoneID ,token, _, _, error in
|
|
||||||
if case .success = CloudKitZoneResult.resolve(error) {
|
|
||||||
savedChangeToken = token
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
op.fetchRecordZoneChangesCompletionBlock = { [weak self] error in
|
|
||||||
guard let self = self else {
|
|
||||||
completion(.failure(CloudKitZoneError.unknown))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
switch CloudKitZoneResult.resolve(error) {
|
|
||||||
case .success:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
self.delegate?.cloudKitDidModify(changed: changedRecords, deleted: deletedRecordKeys) { result in
|
|
||||||
switch result {
|
|
||||||
case .success:
|
|
||||||
self.changeToken = savedChangeToken
|
|
||||||
completion(.success(()))
|
|
||||||
case .failure(let error):
|
|
||||||
completion(.failure(error))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
case .zoneNotFound:
|
|
||||||
self.createZoneRecord() { result in
|
|
||||||
switch result {
|
|
||||||
case .success:
|
|
||||||
self.fetchChangesInZone(completion: completion)
|
|
||||||
case .failure(let error):
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(error))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
case .userDeletedZone:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(CloudKitZoneError.userDeletedZone))
|
|
||||||
}
|
|
||||||
case .retry(let timeToWait):
|
|
||||||
os_log(.error, log: self.log, "%@ zone fetch changes retry in %f seconds.", Self.zoneID.zoneName, timeToWait)
|
|
||||||
self.retryIfPossible(after: timeToWait) {
|
|
||||||
self.fetchChangesInZone(completion: completion)
|
|
||||||
}
|
|
||||||
case .changeTokenExpired:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
self.changeToken = nil
|
|
||||||
self.fetchChangesInZone(completion: completion)
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.failure(CloudKitError(error!)))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
database?.add(op)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
private extension CloudKitZone {
|
|
||||||
|
|
||||||
var changeTokenKey: String {
|
|
||||||
return "cloudkit.server.token.\(Self.zoneID.zoneName)"
|
|
||||||
}
|
|
||||||
|
|
||||||
var changeToken: CKServerChangeToken? {
|
|
||||||
get {
|
|
||||||
guard let tokenData = UserDefaults.standard.object(forKey: changeTokenKey) as? Data else { return nil }
|
|
||||||
return try? NSKeyedUnarchiver.unarchivedObject(ofClass: CKServerChangeToken.self, from: tokenData)
|
|
||||||
}
|
|
||||||
set {
|
|
||||||
guard let token = newValue, let data = try? NSKeyedArchiver.archivedData(withRootObject: token, requiringSecureCoding: false) else {
|
|
||||||
UserDefaults.standard.removeObject(forKey: changeTokenKey)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
UserDefaults.standard.set(data, forKey: changeTokenKey)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var zoneConfiguration: CKFetchRecordZoneChangesOperation.ZoneConfiguration {
|
|
||||||
let config = CKFetchRecordZoneChangesOperation.ZoneConfiguration()
|
|
||||||
config.previousServerChangeToken = changeToken
|
|
||||||
return config
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -1,81 +0,0 @@
|
|||||||
//
|
|
||||||
// CloudKitResult.swift
|
|
||||||
// Account
|
|
||||||
//
|
|
||||||
// Created by Maurice Parker on 3/26/20.
|
|
||||||
// Copyright © 2020 Ranchero Software, LLC. All rights reserved.
|
|
||||||
//
|
|
||||||
|
|
||||||
import Foundation
|
|
||||||
import CloudKit
|
|
||||||
|
|
||||||
enum CloudKitZoneResult {
|
|
||||||
case success
|
|
||||||
case retry(afterSeconds: Double)
|
|
||||||
case limitExceeded
|
|
||||||
case changeTokenExpired
|
|
||||||
case partialFailure(errors: [AnyHashable: CKError])
|
|
||||||
case serverRecordChanged
|
|
||||||
case zoneNotFound
|
|
||||||
case userDeletedZone
|
|
||||||
case failure(error: Error)
|
|
||||||
|
|
||||||
static func resolve(_ error: Error?) -> CloudKitZoneResult {
|
|
||||||
|
|
||||||
guard error != nil else { return .success }
|
|
||||||
|
|
||||||
guard let ckError = error as? CKError else {
|
|
||||||
return .failure(error: error!)
|
|
||||||
}
|
|
||||||
|
|
||||||
switch ckError.code {
|
|
||||||
case .serviceUnavailable, .requestRateLimited, .zoneBusy:
|
|
||||||
if let retry = ckError.userInfo[CKErrorRetryAfterKey] as? NSNumber {
|
|
||||||
return .retry(afterSeconds: retry.doubleValue)
|
|
||||||
} else {
|
|
||||||
return .failure(error: CloudKitError(ckError))
|
|
||||||
}
|
|
||||||
case .zoneNotFound:
|
|
||||||
return .zoneNotFound
|
|
||||||
case .userDeletedZone:
|
|
||||||
return .userDeletedZone
|
|
||||||
case .changeTokenExpired:
|
|
||||||
return .changeTokenExpired
|
|
||||||
case .serverRecordChanged:
|
|
||||||
return .serverRecordChanged
|
|
||||||
case .partialFailure:
|
|
||||||
if let partialErrors = ckError.userInfo[CKPartialErrorsByItemIDKey] as? [AnyHashable: CKError] {
|
|
||||||
if let zoneResult = anyRequestErrors(partialErrors) {
|
|
||||||
return zoneResult
|
|
||||||
} else {
|
|
||||||
return .partialFailure(errors: partialErrors)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return .failure(error: CloudKitError(ckError))
|
|
||||||
}
|
|
||||||
case .limitExceeded:
|
|
||||||
return .limitExceeded
|
|
||||||
default:
|
|
||||||
return .failure(error: CloudKitError(ckError))
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
private extension CloudKitZoneResult {
|
|
||||||
|
|
||||||
static func anyRequestErrors(_ errors: [AnyHashable: CKError]) -> CloudKitZoneResult? {
|
|
||||||
if errors.values.contains(where: { $0.code == .changeTokenExpired } ) {
|
|
||||||
return .changeTokenExpired
|
|
||||||
}
|
|
||||||
if errors.values.contains(where: { $0.code == .zoneNotFound } ) {
|
|
||||||
return .zoneNotFound
|
|
||||||
}
|
|
||||||
if errors.values.contains(where: { $0.code == .userDeletedZone } ) {
|
|
||||||
return .userDeletedZone
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -102,40 +102,38 @@ final class RedditLinkData: Codable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if isVideo ?? false, let videoURL = media?.video?.hlsURL {
|
if isVideo ?? false, let videoURL = media?.video?.hlsURL {
|
||||||
var html = "<video "
|
var html = "<figure><video "
|
||||||
if let previewImageURL = preview?.images?.first?.source?.url {
|
if let previewImageURL = preview?.images?.first?.source?.url {
|
||||||
html += "poster=\"\(previewImageURL)\" "
|
html += "poster=\"\(previewImageURL)\" "
|
||||||
}
|
}
|
||||||
if let width = media?.video?.width, let height = media?.video?.height {
|
if let width = media?.video?.width, let height = media?.video?.height {
|
||||||
html += "width=\"\(width)\" height=\"\(height)\" "
|
html += "width=\"\(width)\" height=\"\(height)\" "
|
||||||
}
|
}
|
||||||
html += "src=\"\(videoURL)\"></video>"
|
html += "src=\"\(videoURL)\"></video></figure>"
|
||||||
return html
|
return html
|
||||||
}
|
}
|
||||||
|
|
||||||
if let imageVariantURL = preview?.images?.first?.variants?.mp4?.source?.url {
|
if let imageVariantURL = preview?.images?.first?.variants?.mp4?.source?.url {
|
||||||
var html = "<video class=\"nnwAnimatedGIF\" "
|
var html = "<figure><video class=\"nnwAnimatedGIF\" "
|
||||||
if let previewImageURL = preview?.images?.first?.source?.url {
|
if let previewImageURL = preview?.images?.first?.source?.url {
|
||||||
html += "poster=\"\(previewImageURL)\" "
|
html += "poster=\"\(previewImageURL)\" "
|
||||||
}
|
}
|
||||||
if let width = preview?.images?.first?.variants?.mp4?.source?.width, let height = preview?.images?.first?.variants?.mp4?.source?.height {
|
if let width = preview?.images?.first?.variants?.mp4?.source?.width, let height = preview?.images?.first?.variants?.mp4?.source?.height {
|
||||||
html += "width=\"\(width)\" height=\"\(height)\" "
|
html += "width=\"\(width)\" height=\"\(height)\" "
|
||||||
}
|
}
|
||||||
html += "src=\"\(imageVariantURL)\" autoplay muted loop></video>"
|
html += "src=\"\(imageVariantURL)\" autoplay muted loop></video></figure>"
|
||||||
html += linkURL(url)
|
|
||||||
return html
|
return html
|
||||||
}
|
}
|
||||||
|
|
||||||
if let videoPreviewURL = preview?.videoPreview?.url {
|
if let videoPreviewURL = preview?.videoPreview?.url {
|
||||||
var html = "<video class=\"nnwAnimatedGIF\" "
|
var html = "<figure><video class=\"nnwAnimatedGIF\" "
|
||||||
if let previewImageURL = preview?.images?.first?.source?.url {
|
if let previewImageURL = preview?.images?.first?.source?.url {
|
||||||
html += "poster=\"\(previewImageURL)\" "
|
html += "poster=\"\(previewImageURL)\" "
|
||||||
}
|
}
|
||||||
if let width = preview?.videoPreview?.width, let height = preview?.videoPreview?.height {
|
if let width = preview?.videoPreview?.width, let height = preview?.videoPreview?.height {
|
||||||
html += "width=\"\(width)\" height=\"\(height)\" "
|
html += "width=\"\(width)\" height=\"\(height)\" "
|
||||||
}
|
}
|
||||||
html += "src=\"\(videoPreviewURL)\" autoplay muted loop></video>"
|
html += "src=\"\(videoPreviewURL)\" autoplay muted loop></video></figure>"
|
||||||
html += linkURL(url)
|
|
||||||
return html
|
return html
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -144,15 +142,14 @@ final class RedditLinkData: Codable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if let imageSource = preview?.images?.first?.source, let imageURL = imageSource.url {
|
if let imageSource = preview?.images?.first?.source, let imageURL = imageSource.url {
|
||||||
var html = "<img src=\"\(imageURL)\" "
|
var html = "<figure><img src=\"\(imageURL)\" "
|
||||||
if postHint == "link" {
|
if postHint == "link" {
|
||||||
html += "class=\"nnw-nozoom\" "
|
html += "class=\"nnw-nozoom\" "
|
||||||
}
|
}
|
||||||
if let width = imageSource.width, let height = imageSource.height {
|
if let width = imageSource.width, let height = imageSource.height {
|
||||||
html += "width=\"\(width)\" height=\"\(height)\" "
|
html += "width=\"\(width)\" height=\"\(height)\" "
|
||||||
}
|
}
|
||||||
html += ">"
|
html += "></figure>"
|
||||||
html += linkURL(url, linkOutOnly: false)
|
|
||||||
return html
|
return html
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -167,25 +164,10 @@ final class RedditLinkData: Codable {
|
|||||||
html += "></figure>"
|
html += "></figure>"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
html += linkURL(url, linkOutOnly: false)
|
|
||||||
return html
|
return html
|
||||||
}
|
}
|
||||||
|
|
||||||
return linkURL(url)
|
return ""
|
||||||
}
|
|
||||||
|
|
||||||
func linkURL(_ url: String, linkOutOnly: Bool = true) -> String {
|
|
||||||
guard let urlComponents = URLComponents(string: url), let host = urlComponents.host else {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
guard !linkOutOnly || (!host.hasSuffix("reddit.com") && !host.hasSuffix("redd.it")) else {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
var displayURL = "\(urlComponents.host ?? "")\(urlComponents.path)"
|
|
||||||
if displayURL.count > 30 {
|
|
||||||
displayURL = "\(displayURL.prefix(30))..."
|
|
||||||
}
|
|
||||||
return "<div><a href=\"\(url)\">\(displayURL)</a></div>"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -16,6 +16,7 @@ import SyncDatabase
|
|||||||
import os.log
|
import os.log
|
||||||
|
|
||||||
extension NewsBlurAccountDelegate {
|
extension NewsBlurAccountDelegate {
|
||||||
|
|
||||||
func refreshFeeds(for account: Account, completion: @escaping (Result<Void, Error>) -> Void) {
|
func refreshFeeds(for account: Account, completion: @escaping (Result<Void, Error>) -> Void) {
|
||||||
os_log(.debug, log: log, "Refreshing feeds...")
|
os_log(.debug, log: log, "Refreshing feeds...")
|
||||||
|
|
||||||
@ -27,8 +28,6 @@ extension NewsBlurAccountDelegate {
|
|||||||
self.syncFeeds(account, feeds)
|
self.syncFeeds(account, feeds)
|
||||||
self.syncFeedFolderRelationship(account, folders)
|
self.syncFeedFolderRelationship(account, folders)
|
||||||
}
|
}
|
||||||
|
|
||||||
self.refreshProgress.completeTask()
|
|
||||||
completion(.success(()))
|
completion(.success(()))
|
||||||
case .failure(let error):
|
case .failure(let error):
|
||||||
completion(.failure(error))
|
completion(.failure(error))
|
||||||
|
@ -63,7 +63,7 @@ final class NewsBlurAccountDelegate: AccountDelegate {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func refreshAll(for account: Account, completion: @escaping (Result<Void, Error>) -> ()) {
|
func refreshAll(for account: Account, completion: @escaping (Result<Void, Error>) -> ()) {
|
||||||
self.refreshProgress.addToNumberOfTasksAndRemaining(5)
|
self.refreshProgress.addToNumberOfTasksAndRemaining(4)
|
||||||
|
|
||||||
refreshFeeds(for: account) { result in
|
refreshFeeds(for: account) { result in
|
||||||
self.refreshProgress.completeTask()
|
self.refreshProgress.completeTask()
|
||||||
@ -80,31 +80,21 @@ final class NewsBlurAccountDelegate: AccountDelegate {
|
|||||||
|
|
||||||
switch result {
|
switch result {
|
||||||
case .success:
|
case .success:
|
||||||
self.refreshStories(for: account) { result in
|
self.refreshMissingStories(for: account) { result in
|
||||||
self.refreshProgress.completeTask()
|
self.refreshProgress.completeTask()
|
||||||
|
|
||||||
switch result {
|
switch result {
|
||||||
case .success:
|
case .success:
|
||||||
self.refreshMissingStories(for: account) { result in
|
DispatchQueue.main.async {
|
||||||
self.refreshProgress.completeTask()
|
completion(.success(()))
|
||||||
|
|
||||||
switch result {
|
|
||||||
case .success:
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
completion(.success(()))
|
|
||||||
}
|
|
||||||
|
|
||||||
case .failure(let error):
|
|
||||||
DispatchQueue.main.async {
|
|
||||||
self.refreshProgress.clear()
|
|
||||||
let wrappedError = AccountError.wrappedError(error: error, account: account)
|
|
||||||
completion(.failure(wrappedError))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
case .failure(let error):
|
case .failure(let error):
|
||||||
completion(.failure(error))
|
DispatchQueue.main.async {
|
||||||
|
self.refreshProgress.clear()
|
||||||
|
let wrappedError = AccountError.wrappedError(error: error, account: account)
|
||||||
|
completion(.failure(wrappedError))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -246,7 +236,6 @@ final class NewsBlurAccountDelegate: AccountDelegate {
|
|||||||
caller.retrieveUnreadStoryHashes { result in
|
caller.retrieveUnreadStoryHashes { result in
|
||||||
switch result {
|
switch result {
|
||||||
case .success(let storyHashes):
|
case .success(let storyHashes):
|
||||||
self.refreshProgress.completeTask()
|
|
||||||
|
|
||||||
if let count = storyHashes?.count, count > 0 {
|
if let count = storyHashes?.count, count > 0 {
|
||||||
self.refreshProgress.addToNumberOfTasksAndRemaining((count - 1) / 100 + 1)
|
self.refreshProgress.addToNumberOfTasksAndRemaining((count - 1) / 100 + 1)
|
||||||
|
@ -95,11 +95,19 @@ struct AppAssets {
|
|||||||
}()
|
}()
|
||||||
|
|
||||||
static var filterActive: RSImage = {
|
static var filterActive: RSImage = {
|
||||||
return RSImage(named: "filterActive")!
|
if #available(macOS 11.0, *) {
|
||||||
|
return NSImage(systemSymbolName: "line.horizontal.3.decrease.circle.fill", accessibilityDescription: nil)!
|
||||||
|
} else {
|
||||||
|
return RSImage(named: "filterActive")!
|
||||||
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
static var filterInactive: RSImage = {
|
static var filterInactive: RSImage = {
|
||||||
return RSImage(named: "filterInactive")!
|
if #available(macOS 11.0, *) {
|
||||||
|
return NSImage(systemSymbolName: "line.horizontal.3.decrease.circle", accessibilityDescription: nil)!
|
||||||
|
} else {
|
||||||
|
return RSImage(named: "filterInactive")!
|
||||||
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
static var iconLightBackgroundColor: NSColor = {
|
static var iconLightBackgroundColor: NSColor = {
|
||||||
@ -249,6 +257,10 @@ struct AppAssets {
|
|||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
|
static var timelineSeparatorColor: NSColor = {
|
||||||
|
return NSColor(named: "timelineSeparatorColor")!
|
||||||
|
}()
|
||||||
|
|
||||||
static var timelineStarSelected: RSImage! = {
|
static var timelineStarSelected: RSImage! = {
|
||||||
return RSImage(named: "timelineStar")?.tinted(with: .white)
|
return RSImage(named: "timelineStar")?.tinted(with: .white)
|
||||||
}()
|
}()
|
||||||
|
@ -389,16 +389,3 @@ private extension AppDefaults {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: -
|
|
||||||
extension UserDefaults {
|
|
||||||
/// This property exists so that it can conveniently be observed via KVO
|
|
||||||
@objc var CorreiaSeparators: Bool {
|
|
||||||
get {
|
|
||||||
return bool(forKey: AppDefaults.Key.timelineShowsSeparators)
|
|
||||||
}
|
|
||||||
set {
|
|
||||||
set(newValue, forKey: AppDefaults.Key.timelineShowsSeparators)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -15,6 +15,7 @@ import Account
|
|||||||
import RSCore
|
import RSCore
|
||||||
import RSCoreResources
|
import RSCoreResources
|
||||||
import Secrets
|
import Secrets
|
||||||
|
import OSLog
|
||||||
|
|
||||||
// If we're not going to import Sparkle, provide dummy protocols to make it easy
|
// If we're not going to import Sparkle, provide dummy protocols to make it easy
|
||||||
// for AppDelegate to comply
|
// for AppDelegate to comply
|
||||||
@ -97,7 +98,6 @@ class AppDelegate: NSObject, NSApplicationDelegate, NSUserInterfaceValidations,
|
|||||||
private var keyboardShortcutsWindowController: WebViewWindowController?
|
private var keyboardShortcutsWindowController: WebViewWindowController?
|
||||||
private var inspectorWindowController: InspectorWindowController?
|
private var inspectorWindowController: InspectorWindowController?
|
||||||
private var crashReportWindowController: CrashReportWindowController? // For testing only
|
private var crashReportWindowController: CrashReportWindowController? // For testing only
|
||||||
private let log = Log()
|
|
||||||
private let appMovementMonitor = RSAppMovementMonitor()
|
private let appMovementMonitor = RSAppMovementMonitor()
|
||||||
#if !MAC_APP_STORE && !TEST
|
#if !MAC_APP_STORE && !TEST
|
||||||
private var softwareUpdater: SPUUpdater!
|
private var softwareUpdater: SPUUpdater!
|
||||||
@ -119,22 +119,6 @@ class AppDelegate: NSObject, NSApplicationDelegate, NSUserInterfaceValidations,
|
|||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - API
|
// MARK: - API
|
||||||
func logMessage(_ message: String, type: LogItem.ItemType) {
|
|
||||||
|
|
||||||
#if DEBUG
|
|
||||||
if type == .debug {
|
|
||||||
print("logMessage: \(message) - \(type)")
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
let logItem = LogItem(type: type, message: message)
|
|
||||||
log.add(logItem)
|
|
||||||
}
|
|
||||||
|
|
||||||
func logDebugMessage(_ message: String) {
|
|
||||||
logMessage(message, type: .debug)
|
|
||||||
}
|
|
||||||
|
|
||||||
func showAddFolderSheetOnWindow(_ window: NSWindow) {
|
func showAddFolderSheetOnWindow(_ window: NSWindow) {
|
||||||
addFolderWindowController = AddFolderWindowController()
|
addFolderWindowController = AddFolderWindowController()
|
||||||
addFolderWindowController!.runSheetOnWindow(window)
|
addFolderWindowController!.runSheetOnWindow(window)
|
||||||
@ -199,7 +183,7 @@ class AppDelegate: NSObject, NSApplicationDelegate, NSUserInterfaceValidations,
|
|||||||
AppDefaults.shared.registerDefaults()
|
AppDefaults.shared.registerDefaults()
|
||||||
let isFirstRun = AppDefaults.shared.isFirstRun
|
let isFirstRun = AppDefaults.shared.isFirstRun
|
||||||
if isFirstRun {
|
if isFirstRun {
|
||||||
logDebugMessage("Is first run.")
|
os_log(.debug, "Is first run.")
|
||||||
}
|
}
|
||||||
let localAccount = AccountManager.shared.defaultAccount
|
let localAccount = AccountManager.shared.defaultAccount
|
||||||
|
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<document type="com.apple.InterfaceBuilder3.Cocoa.Storyboard.XIB" version="3.0" toolsVersion="17154" targetRuntime="MacOSX.Cocoa" propertyAccessControl="none" useAutolayout="YES" initialViewController="B8D-0N-5wS">
|
<document type="com.apple.InterfaceBuilder3.Cocoa.Storyboard.XIB" version="3.0" toolsVersion="17506" targetRuntime="MacOSX.Cocoa" propertyAccessControl="none" useAutolayout="YES" initialViewController="B8D-0N-5wS">
|
||||||
<dependencies>
|
<dependencies>
|
||||||
<deployment identifier="macosx"/>
|
<deployment identifier="macosx"/>
|
||||||
<plugIn identifier="com.apple.InterfaceBuilder.CocoaPlugin" version="17154"/>
|
<plugIn identifier="com.apple.InterfaceBuilder.CocoaPlugin" version="17506"/>
|
||||||
<capability name="NSView safe area layout guides" minToolsVersion="12.0"/>
|
<capability name="NSView safe area layout guides" minToolsVersion="12.0"/>
|
||||||
<capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/>
|
<capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
@ -298,43 +298,43 @@
|
|||||||
<scene sceneID="Yae-mu-VsH">
|
<scene sceneID="Yae-mu-VsH">
|
||||||
<objects>
|
<objects>
|
||||||
<viewController id="XML-A3-pDn" userLabel="Sidebar View Controller" customClass="SidebarViewController" customModule="NetNewsWire" customModuleProvider="target" sceneMemberID="viewController">
|
<viewController id="XML-A3-pDn" userLabel="Sidebar View Controller" customClass="SidebarViewController" customModule="NetNewsWire" customModuleProvider="target" sceneMemberID="viewController">
|
||||||
<view key="view" wantsLayer="YES" id="bJZ-bH-vgc">
|
<view key="view" wantsLayer="YES" misplaced="YES" id="bJZ-bH-vgc">
|
||||||
<rect key="frame" x="0.0" y="0.0" width="240" height="531"/>
|
<rect key="frame" x="0.0" y="0.0" width="240" height="704"/>
|
||||||
<autoresizingMask key="autoresizingMask"/>
|
<autoresizingMask key="autoresizingMask"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
<scrollView borderType="none" autohidesScrollers="YES" horizontalLineScroll="28" horizontalPageScroll="10" verticalLineScroll="28" verticalPageScroll="10" hasHorizontalScroller="NO" horizontalScrollElasticity="none" translatesAutoresizingMaskIntoConstraints="NO" id="cJj-Wv-9ep">
|
<scrollView autohidesScrollers="YES" horizontalLineScroll="26" horizontalPageScroll="10" verticalLineScroll="26" verticalPageScroll="10" usesPredominantAxisScrolling="NO" translatesAutoresizingMaskIntoConstraints="NO" id="vEw-8O-WQc">
|
||||||
<rect key="frame" x="0.0" y="0.0" width="240" height="531"/>
|
<rect key="frame" x="0.0" y="0.0" width="240" height="996"/>
|
||||||
<clipView key="contentView" drawsBackground="NO" copiesOnScroll="NO" id="2eU-Wz-F9g">
|
<clipView key="contentView" drawsBackground="NO" id="WBm-nN-Xa0">
|
||||||
<rect key="frame" x="0.0" y="0.0" width="240" height="531"/>
|
<rect key="frame" x="1" y="1" width="238" height="994"/>
|
||||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
<outlineView verticalHuggingPriority="750" allowsExpansionToolTips="YES" columnAutoresizingStyle="firstColumnOnly" selectionHighlightStyle="sourceList" columnReordering="NO" columnResizing="NO" autosaveColumns="NO" typeSelect="NO" rowHeight="28" rowSizeStyle="systemDefault" viewBased="YES" floatsGroupRows="NO" indentationPerLevel="13" outlineTableColumn="ih9-mJ-EA7" id="cnV-kg-Dn2" customClass="SidebarOutlineView" customModule="NetNewsWire" customModuleProvider="target">
|
<outlineView verticalHuggingPriority="750" allowsExpansionToolTips="YES" columnAutoresizingStyle="lastColumnOnly" selectionHighlightStyle="sourceList" columnReordering="NO" columnResizing="NO" autosaveColumns="NO" typeSelect="NO" rowHeight="24" rowSizeStyle="systemDefault" viewBased="YES" floatsGroupRows="NO" indentationPerLevel="13" outlineTableColumn="rEJ-0k-dzS" id="Ksh-tg-xwv" customClass="SidebarOutlineView" customModule="NetNewsWire" customModuleProvider="target">
|
||||||
<rect key="frame" x="0.0" y="0.0" width="240" height="531"/>
|
<rect key="frame" x="0.0" y="0.0" width="238" height="994"/>
|
||||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||||
<size key="intercellSpacing" width="3" height="0.0"/>
|
<size key="intercellSpacing" width="3" height="2"/>
|
||||||
<color key="backgroundColor" name="_sourceListBackgroundColor" catalog="System" colorSpace="catalog"/>
|
<color key="backgroundColor" name="_sourceListBackgroundColor" catalog="System" colorSpace="catalog"/>
|
||||||
<color key="gridColor" name="gridColor" catalog="System" colorSpace="catalog"/>
|
<color key="gridColor" name="gridColor" catalog="System" colorSpace="catalog"/>
|
||||||
<tableColumns>
|
<tableColumns>
|
||||||
<tableColumn width="237" minWidth="23" maxWidth="1000" id="ih9-mJ-EA7">
|
<tableColumn width="235" minWidth="16" maxWidth="1000" id="rEJ-0k-dzS">
|
||||||
<tableHeaderCell key="headerCell" lineBreakMode="truncatingTail" borderStyle="border">
|
<tableHeaderCell key="headerCell" lineBreakMode="truncatingTail" borderStyle="border">
|
||||||
<color key="textColor" name="headerTextColor" catalog="System" colorSpace="catalog"/>
|
<color key="textColor" name="headerTextColor" catalog="System" colorSpace="catalog"/>
|
||||||
<color key="backgroundColor" name="headerColor" catalog="System" colorSpace="catalog"/>
|
<color key="backgroundColor" name="headerColor" catalog="System" colorSpace="catalog"/>
|
||||||
</tableHeaderCell>
|
</tableHeaderCell>
|
||||||
<textFieldCell key="dataCell" lineBreakMode="truncatingTail" selectable="YES" editable="YES" title="Text Cell" id="sXh-y7-12P">
|
<textFieldCell key="dataCell" lineBreakMode="truncatingTail" selectable="YES" editable="YES" title="Text Cell" id="BJ3-rV-boT">
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
<color key="textColor" name="controlTextColor" catalog="System" colorSpace="catalog"/>
|
<color key="textColor" name="controlTextColor" catalog="System" colorSpace="catalog"/>
|
||||||
<color key="backgroundColor" name="controlBackgroundColor" catalog="System" colorSpace="catalog"/>
|
<color key="backgroundColor" name="controlBackgroundColor" catalog="System" colorSpace="catalog"/>
|
||||||
</textFieldCell>
|
</textFieldCell>
|
||||||
<tableColumnResizingMask key="resizingMask" resizeWithTable="YES"/>
|
<tableColumnResizingMask key="resizingMask" resizeWithTable="YES" userResizable="YES"/>
|
||||||
<prototypeCellViews>
|
<prototypeCellViews>
|
||||||
<tableCellView identifier="HeaderCell" id="qkt-WA-5tB">
|
<tableCellView identifier="HeaderCell" id="Rcy-nR-n8V">
|
||||||
<rect key="frame" x="1" y="0.0" width="237" height="17"/>
|
<rect key="frame" x="1" y="1" width="235" height="17"/>
|
||||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
<textField verticalHuggingPriority="750" fixedFrame="YES" translatesAutoresizingMaskIntoConstraints="NO" id="fNJ-z1-0Up">
|
<textField verticalHuggingPriority="750" fixedFrame="YES" translatesAutoresizingMaskIntoConstraints="NO" id="UYV-gZ-amb">
|
||||||
<rect key="frame" x="0.0" y="1" width="145" height="14"/>
|
<rect key="frame" x="0.0" y="1" width="235" height="14"/>
|
||||||
<autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMinY="YES"/>
|
<autoresizingMask key="autoresizingMask" widthSizable="YES" flexibleMinY="YES"/>
|
||||||
<textFieldCell key="cell" lineBreakMode="truncatingTail" sendsActionOnEndEditing="YES" title="HEADER CELL" id="dRB-0K-qxz">
|
<textFieldCell key="cell" lineBreakMode="truncatingTail" sendsActionOnEndEditing="YES" title="HEADER CELL" id="6Au-5W-t8n">
|
||||||
<font key="font" metaFont="smallSystemBold"/>
|
<font key="font" metaFont="smallSystemBold"/>
|
||||||
<color key="textColor" name="headerColor" catalog="System" colorSpace="catalog"/>
|
<color key="textColor" name="headerColor" catalog="System" colorSpace="catalog"/>
|
||||||
<color key="backgroundColor" name="controlColor" catalog="System" colorSpace="catalog"/>
|
<color key="backgroundColor" name="controlColor" catalog="System" colorSpace="catalog"/>
|
||||||
@ -342,34 +342,30 @@
|
|||||||
</textField>
|
</textField>
|
||||||
</subviews>
|
</subviews>
|
||||||
<connections>
|
<connections>
|
||||||
<outlet property="textField" destination="fNJ-z1-0Up" id="jEh-Oo-s62"/>
|
<outlet property="textField" destination="UYV-gZ-amb" id="ITg-Si-8HC"/>
|
||||||
</connections>
|
</connections>
|
||||||
</tableCellView>
|
</tableCellView>
|
||||||
<tableCellView identifier="DataCell" id="HJn-Tm-YNO" customClass="SidebarCell" customModule="NetNewsWire" customModuleProvider="target">
|
<tableCellView identifier="DataCell" id="YA6-bT-fhL" customClass="SidebarCell" customModule="NetNewsWire" customModuleProvider="target">
|
||||||
<rect key="frame" x="1" y="17" width="237" height="17"/>
|
<rect key="frame" x="1" y="20" width="235" height="17"/>
|
||||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||||
</tableCellView>
|
</tableCellView>
|
||||||
</prototypeCellViews>
|
</prototypeCellViews>
|
||||||
</tableColumn>
|
</tableColumn>
|
||||||
</tableColumns>
|
</tableColumns>
|
||||||
<accessibility description="Feeds"/>
|
|
||||||
<connections>
|
<connections>
|
||||||
<outlet property="delegate" destination="XML-A3-pDn" id="fPE-cv-p5c"/>
|
<outlet property="delegate" destination="XML-A3-pDn" id="1rX-mk-rO4"/>
|
||||||
<outlet property="keyboardDelegate" destination="h5K-zR-cUa" id="BlT-aW-sea"/>
|
<outlet property="keyboardDelegate" destination="h5K-zR-cUa" id="bm3-GZ-HRO"/>
|
||||||
<outlet property="menu" destination="p3f-EZ-sSD" id="KTA-tl-UrO"/>
|
<outlet property="menu" destination="p3f-EZ-sSD" id="MMV-na-KIl"/>
|
||||||
</connections>
|
</connections>
|
||||||
</outlineView>
|
</outlineView>
|
||||||
</subviews>
|
</subviews>
|
||||||
<nil key="backgroundColor"/>
|
<nil key="backgroundColor"/>
|
||||||
</clipView>
|
</clipView>
|
||||||
<constraints>
|
<scroller key="horizontalScroller" hidden="YES" wantsLayer="YES" verticalHuggingPriority="750" horizontal="YES" id="crb-RU-OP7">
|
||||||
<constraint firstAttribute="width" relation="greaterThanOrEqual" constant="166" id="pzy-wh-tgi"/>
|
<rect key="frame" x="1" y="979" width="238" height="16"/>
|
||||||
</constraints>
|
|
||||||
<scroller key="horizontalScroller" hidden="YES" wantsLayer="YES" verticalHuggingPriority="750" horizontal="YES" id="vs5-5h-CXe">
|
|
||||||
<rect key="frame" x="-100" y="-100" width="238" height="15"/>
|
|
||||||
<autoresizingMask key="autoresizingMask"/>
|
<autoresizingMask key="autoresizingMask"/>
|
||||||
</scroller>
|
</scroller>
|
||||||
<scroller key="verticalScroller" hidden="YES" wantsLayer="YES" verticalHuggingPriority="750" horizontal="NO" id="FWV-kB-qct">
|
<scroller key="verticalScroller" hidden="YES" wantsLayer="YES" verticalHuggingPriority="750" horizontal="NO" id="Bag-2b-CQj">
|
||||||
<rect key="frame" x="224" y="17" width="15" height="102"/>
|
<rect key="frame" x="224" y="17" width="15" height="102"/>
|
||||||
<autoresizingMask key="autoresizingMask"/>
|
<autoresizingMask key="autoresizingMask"/>
|
||||||
</scroller>
|
</scroller>
|
||||||
@ -414,21 +410,21 @@
|
|||||||
</customView>
|
</customView>
|
||||||
</subviews>
|
</subviews>
|
||||||
<constraints>
|
<constraints>
|
||||||
<constraint firstItem="HZs-Zf-G8s" firstAttribute="top" secondItem="cJj-Wv-9ep" secondAttribute="bottom" id="0Zg-oW-o7U"/>
|
|
||||||
<constraint firstItem="cJj-Wv-9ep" firstAttribute="leading" secondItem="bJZ-bH-vgc" secondAttribute="leading" id="5Rs-9M-TKq"/>
|
|
||||||
<constraint firstItem="cJj-Wv-9ep" firstAttribute="top" secondItem="bJZ-bH-vgc" secondAttribute="top" id="A7C-VI-drt"/>
|
|
||||||
<constraint firstAttribute="trailing" secondItem="iyL-pW-cT6" secondAttribute="trailing" constant="20" id="Mnm-9S-Qpm"/>
|
<constraint firstAttribute="trailing" secondItem="iyL-pW-cT6" secondAttribute="trailing" constant="20" id="Mnm-9S-Qpm"/>
|
||||||
<constraint firstAttribute="bottom" secondItem="HZs-Zf-G8s" secondAttribute="bottom" constant="-28" id="UN9-Wa-uxb"/>
|
<constraint firstAttribute="bottom" secondItem="HZs-Zf-G8s" secondAttribute="bottom" constant="-28" id="UN9-Wa-uxb"/>
|
||||||
|
<constraint firstAttribute="trailing" secondItem="vEw-8O-WQc" secondAttribute="trailing" id="b0g-fS-PDC"/>
|
||||||
<constraint firstAttribute="trailing" secondItem="HZs-Zf-G8s" secondAttribute="trailing" id="iNE-nb-QEB"/>
|
<constraint firstAttribute="trailing" secondItem="HZs-Zf-G8s" secondAttribute="trailing" id="iNE-nb-QEB"/>
|
||||||
|
<constraint firstItem="vEw-8O-WQc" firstAttribute="leading" secondItem="bJZ-bH-vgc" secondAttribute="leading" id="rHx-Y1-mfZ"/>
|
||||||
<constraint firstItem="HZs-Zf-G8s" firstAttribute="leading" secondItem="bJZ-bH-vgc" secondAttribute="leading" id="tPp-xB-CgB"/>
|
<constraint firstItem="HZs-Zf-G8s" firstAttribute="leading" secondItem="bJZ-bH-vgc" secondAttribute="leading" id="tPp-xB-CgB"/>
|
||||||
<constraint firstAttribute="trailing" secondItem="cJj-Wv-9ep" secondAttribute="trailing" id="vo7-3F-Fd3"/>
|
<constraint firstItem="vEw-8O-WQc" firstAttribute="top" secondItem="bJZ-bH-vgc" secondAttribute="top" id="trl-kq-iNn"/>
|
||||||
|
<constraint firstItem="HZs-Zf-G8s" firstAttribute="top" secondItem="vEw-8O-WQc" secondAttribute="bottom" id="vam-28-uGK"/>
|
||||||
<constraint firstAttribute="trailing" secondItem="j0H-G3-rg2" secondAttribute="trailing" id="wWv-I7-qKm"/>
|
<constraint firstAttribute="trailing" secondItem="j0H-G3-rg2" secondAttribute="trailing" id="wWv-I7-qKm"/>
|
||||||
</constraints>
|
</constraints>
|
||||||
<viewLayoutGuide key="safeArea" id="j0H-G3-rg2"/>
|
<viewLayoutGuide key="safeArea" id="j0H-G3-rg2"/>
|
||||||
<viewLayoutGuide key="layoutMargins" id="mQg-Jg-Bfh"/>
|
<viewLayoutGuide key="layoutMargins" id="mQg-Jg-Bfh"/>
|
||||||
</view>
|
</view>
|
||||||
<connections>
|
<connections>
|
||||||
<outlet property="outlineView" destination="cnV-kg-Dn2" id="FVf-OT-E3h"/>
|
<outlet property="outlineView" destination="Ksh-tg-xwv" id="VbY-N9-ZQ0"/>
|
||||||
</connections>
|
</connections>
|
||||||
</viewController>
|
</viewController>
|
||||||
<customObject id="Jih-JO-hIE" userLabel="First Responder" customClass="NSResponder" sceneMemberID="firstResponder"/>
|
<customObject id="Jih-JO-hIE" userLabel="First Responder" customClass="NSResponder" sceneMemberID="firstResponder"/>
|
||||||
@ -465,74 +461,76 @@
|
|||||||
<autoresizingMask key="autoresizingMask"/>
|
<autoresizingMask key="autoresizingMask"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
<visualEffectView blendingMode="behindWindow" material="contentBackground" state="followsWindowActiveState" translatesAutoresizingMaskIntoConstraints="NO" id="PdS-jL-yH1">
|
<visualEffectView blendingMode="behindWindow" material="contentBackground" state="followsWindowActiveState" translatesAutoresizingMaskIntoConstraints="NO" id="PdS-jL-yH1">
|
||||||
<rect key="frame" x="0.0" y="120" width="375" height="26"/>
|
<rect key="frame" x="0.0" y="172" width="375" height="26"/>
|
||||||
|
<subviews>
|
||||||
|
<popUpButton verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="lSU-OC-sEC">
|
||||||
|
<rect key="frame" x="8" y="3" width="51" height="19"/>
|
||||||
|
<constraints>
|
||||||
|
<constraint firstAttribute="height" constant="18" id="DoO-KI-ena"/>
|
||||||
|
</constraints>
|
||||||
|
<popUpButtonCell key="cell" type="recessed" title="Sort" bezelStyle="recessed" alignment="center" lineBreakMode="truncatingTail" borderStyle="border" tag="1" imageScaling="proportionallyDown" inset="2" pullsDown="YES" id="bl0-6I-cH2">
|
||||||
|
<behavior key="behavior" pushIn="YES" lightByBackground="YES" lightByGray="YES" changeBackground="YES" changeGray="YES"/>
|
||||||
|
<font key="font" metaFont="smallSystemBold"/>
|
||||||
|
<menu key="menu" id="dN0-S2-uqU">
|
||||||
|
<items>
|
||||||
|
<menuItem title="Sort" tag="1" hidden="YES" id="4BZ-ya-evy">
|
||||||
|
<attributedString key="attributedTitle"/>
|
||||||
|
<modifierMask key="keyEquivalentModifierMask"/>
|
||||||
|
</menuItem>
|
||||||
|
<menuItem title="Newest Article on Top" state="on" tag="2" id="40c-kt-vhO">
|
||||||
|
<connections>
|
||||||
|
<action selector="sortByNewestArticleOnTop:" target="Ebq-4s-EwK" id="vYg-MZ-zve"/>
|
||||||
|
</connections>
|
||||||
|
</menuItem>
|
||||||
|
<menuItem title="Oldest Article on Top" tag="3" id="sOF-Ez-vIL">
|
||||||
|
<connections>
|
||||||
|
<action selector="sortByOldestArticleOnTop:" target="Ebq-4s-EwK" id="KFG-M7-blB"/>
|
||||||
|
</connections>
|
||||||
|
</menuItem>
|
||||||
|
<menuItem isSeparatorItem="YES" id="xQP-gm-iO9"/>
|
||||||
|
<menuItem title="Group by Feed" tag="4" id="YSR-5C-Yjd">
|
||||||
|
<connections>
|
||||||
|
<action selector="groupByFeedToggled:" target="Ebq-4s-EwK" id="4y9-5l-ToF"/>
|
||||||
|
</connections>
|
||||||
|
</menuItem>
|
||||||
|
</items>
|
||||||
|
</menu>
|
||||||
|
</popUpButtonCell>
|
||||||
|
</popUpButton>
|
||||||
|
<button verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="iA5-go-AO0">
|
||||||
|
<rect key="frame" x="350" y="6" width="13" height="14"/>
|
||||||
|
<buttonCell key="cell" type="bevel" bezelStyle="rounded" image="filterInactive" imagePosition="overlaps" alignment="center" imageScaling="proportionallyDown" inset="2" id="j7d-36-DO5">
|
||||||
|
<behavior key="behavior" pushIn="YES" lightByBackground="YES" lightByGray="YES"/>
|
||||||
|
<font key="font" metaFont="system"/>
|
||||||
|
</buttonCell>
|
||||||
|
<color key="contentTintColor" name="controlTextColor" catalog="System" colorSpace="catalog"/>
|
||||||
|
<connections>
|
||||||
|
<action selector="toggleReadArticlesFilter:" target="Ebq-4s-EwK" id="tcC-72-Npk"/>
|
||||||
|
</connections>
|
||||||
|
</button>
|
||||||
|
</subviews>
|
||||||
<constraints>
|
<constraints>
|
||||||
|
<constraint firstItem="iA5-go-AO0" firstAttribute="centerY" secondItem="PdS-jL-yH1" secondAttribute="centerY" id="0Iw-TM-gQz"/>
|
||||||
|
<constraint firstItem="iA5-go-AO0" firstAttribute="leading" relation="greaterThanOrEqual" secondItem="lSU-OC-sEC" secondAttribute="trailing" constant="8" id="HHM-Ls-lso"/>
|
||||||
|
<constraint firstAttribute="trailing" secondItem="iA5-go-AO0" secondAttribute="trailing" constant="12" id="VYe-w6-t0L"/>
|
||||||
<constraint firstAttribute="height" constant="26" id="ZMg-gZ-6aa"/>
|
<constraint firstAttribute="height" constant="26" id="ZMg-gZ-6aa"/>
|
||||||
|
<constraint firstItem="lSU-OC-sEC" firstAttribute="centerY" secondItem="PdS-jL-yH1" secondAttribute="centerY" id="a5Z-68-nkI"/>
|
||||||
|
<constraint firstItem="lSU-OC-sEC" firstAttribute="leading" secondItem="PdS-jL-yH1" secondAttribute="leading" constant="8" id="jDu-ra-lrz"/>
|
||||||
</constraints>
|
</constraints>
|
||||||
</visualEffectView>
|
</visualEffectView>
|
||||||
<popUpButton verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="lSU-OC-sEC">
|
|
||||||
<rect key="frame" x="8" y="123" width="51" height="19"/>
|
|
||||||
<constraints>
|
|
||||||
<constraint firstAttribute="height" constant="18" id="DoO-KI-ena"/>
|
|
||||||
</constraints>
|
|
||||||
<popUpButtonCell key="cell" type="recessed" title="Sort" bezelStyle="recessed" alignment="center" lineBreakMode="truncatingTail" borderStyle="border" tag="1" imageScaling="proportionallyDown" inset="2" pullsDown="YES" id="bl0-6I-cH2">
|
|
||||||
<behavior key="behavior" pushIn="YES" lightByBackground="YES" lightByGray="YES" changeBackground="YES" changeGray="YES"/>
|
|
||||||
<font key="font" metaFont="smallSystemBold"/>
|
|
||||||
<menu key="menu" id="dN0-S2-uqU">
|
|
||||||
<items>
|
|
||||||
<menuItem title="Sort" tag="1" hidden="YES" id="4BZ-ya-evy">
|
|
||||||
<attributedString key="attributedTitle"/>
|
|
||||||
<modifierMask key="keyEquivalentModifierMask"/>
|
|
||||||
</menuItem>
|
|
||||||
<menuItem title="Newest Article on Top" state="on" tag="2" id="40c-kt-vhO">
|
|
||||||
<connections>
|
|
||||||
<action selector="sortByNewestArticleOnTop:" target="Ebq-4s-EwK" id="vYg-MZ-zve"/>
|
|
||||||
</connections>
|
|
||||||
</menuItem>
|
|
||||||
<menuItem title="Oldest Article on Top" tag="3" id="sOF-Ez-vIL">
|
|
||||||
<connections>
|
|
||||||
<action selector="sortByOldestArticleOnTop:" target="Ebq-4s-EwK" id="KFG-M7-blB"/>
|
|
||||||
</connections>
|
|
||||||
</menuItem>
|
|
||||||
<menuItem isSeparatorItem="YES" id="xQP-gm-iO9"/>
|
|
||||||
<menuItem title="Group by Feed" tag="4" id="YSR-5C-Yjd">
|
|
||||||
<connections>
|
|
||||||
<action selector="groupByFeedToggled:" target="Ebq-4s-EwK" id="4y9-5l-ToF"/>
|
|
||||||
</connections>
|
|
||||||
</menuItem>
|
|
||||||
</items>
|
|
||||||
</menu>
|
|
||||||
</popUpButtonCell>
|
|
||||||
</popUpButton>
|
|
||||||
<button verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="iA5-go-AO0">
|
|
||||||
<rect key="frame" x="350" y="179" width="13" height="14"/>
|
|
||||||
<buttonCell key="cell" type="bevel" bezelStyle="rounded" image="filterInactive" imagePosition="overlaps" alignment="center" imageScaling="proportionallyDown" inset="2" id="j7d-36-DO5">
|
|
||||||
<behavior key="behavior" pushIn="YES" lightByBackground="YES" lightByGray="YES"/>
|
|
||||||
<font key="font" metaFont="system"/>
|
|
||||||
</buttonCell>
|
|
||||||
<color key="contentTintColor" name="controlTextColor" catalog="System" colorSpace="catalog"/>
|
|
||||||
<connections>
|
|
||||||
<action selector="toggleReadArticlesFilter:" target="Ebq-4s-EwK" id="tcC-72-Npk"/>
|
|
||||||
</connections>
|
|
||||||
</button>
|
|
||||||
<customView translatesAutoresizingMaskIntoConstraints="NO" id="Zpk-pq-9nW" customClass="TimelineContainerView" customModule="NetNewsWire" customModuleProvider="target">
|
<customView translatesAutoresizingMaskIntoConstraints="NO" id="Zpk-pq-9nW" customClass="TimelineContainerView" customModule="NetNewsWire" customModuleProvider="target">
|
||||||
<rect key="frame" x="0.0" y="0.0" width="375" height="120"/>
|
<rect key="frame" x="0.0" y="0.0" width="375" height="172"/>
|
||||||
</customView>
|
</customView>
|
||||||
</subviews>
|
</subviews>
|
||||||
<constraints>
|
<constraints>
|
||||||
<constraint firstItem="Zpk-pq-9nW" firstAttribute="top" secondItem="PdS-jL-yH1" secondAttribute="bottom" id="2dy-bB-DI2"/>
|
<constraint firstItem="Zpk-pq-9nW" firstAttribute="top" secondItem="PdS-jL-yH1" secondAttribute="bottom" id="2dy-bB-DI2"/>
|
||||||
<constraint firstAttribute="trailing" secondItem="Zpk-pq-9nW" secondAttribute="trailing" id="67d-pI-I9C"/>
|
<constraint firstAttribute="trailing" secondItem="Zpk-pq-9nW" secondAttribute="trailing" id="67d-pI-I9C"/>
|
||||||
<constraint firstAttribute="trailing" secondItem="iA5-go-AO0" secondAttribute="trailing" constant="12" id="9Dl-n9-vRI"/>
|
|
||||||
<constraint firstItem="lSU-OC-sEC" firstAttribute="leading" secondItem="Dnl-L5-xFP" secondAttribute="leading" constant="8" id="Ceb-sA-ECJ"/>
|
|
||||||
<constraint firstItem="PdS-jL-yH1" firstAttribute="trailing" secondItem="M3G-7s-D6y" secondAttribute="trailing" id="Eln-Tf-W9k"/>
|
<constraint firstItem="PdS-jL-yH1" firstAttribute="trailing" secondItem="M3G-7s-D6y" secondAttribute="trailing" id="Eln-Tf-W9k"/>
|
||||||
<constraint firstItem="lSU-OC-sEC" firstAttribute="centerY" secondItem="iA5-go-AO0" secondAttribute="centerY" id="OeL-Zp-iRT"/>
|
|
||||||
<constraint firstItem="Zpk-pq-9nW" firstAttribute="leading" secondItem="Dnl-L5-xFP" secondAttribute="leading" id="XF2-31-E1x"/>
|
<constraint firstItem="Zpk-pq-9nW" firstAttribute="leading" secondItem="Dnl-L5-xFP" secondAttribute="leading" id="XF2-31-E1x"/>
|
||||||
<constraint firstItem="PdS-jL-yH1" firstAttribute="top" secondItem="M3G-7s-D6y" secondAttribute="top" id="aB8-Pt-Szt"/>
|
<constraint firstItem="PdS-jL-yH1" firstAttribute="top" secondItem="M3G-7s-D6y" secondAttribute="top" id="aB8-Pt-Szt"/>
|
||||||
<constraint firstAttribute="bottom" secondItem="Zpk-pq-9nW" secondAttribute="bottom" id="fyv-EG-PC8"/>
|
<constraint firstAttribute="bottom" secondItem="Zpk-pq-9nW" secondAttribute="bottom" id="fyv-EG-PC8"/>
|
||||||
<constraint firstItem="PdS-jL-yH1" firstAttribute="leading" secondItem="M3G-7s-D6y" secondAttribute="leading" id="lSy-HN-fWB"/>
|
<constraint firstItem="PdS-jL-yH1" firstAttribute="leading" secondItem="M3G-7s-D6y" secondAttribute="leading" id="lSy-HN-fWB"/>
|
||||||
<constraint firstAttribute="leading" secondItem="Dnl-L5-xFP" secondAttribute="leading" id="pZU-jW-B1h"/>
|
<constraint firstAttribute="leading" secondItem="Dnl-L5-xFP" secondAttribute="leading" id="pZU-jW-B1h"/>
|
||||||
<constraint firstItem="iA5-go-AO0" firstAttribute="leading" relation="greaterThanOrEqual" secondItem="lSU-OC-sEC" secondAttribute="trailing" constant="8" id="yCg-gc-exN"/>
|
|
||||||
<constraint firstItem="lSU-OC-sEC" firstAttribute="top" secondItem="M3G-7s-D6y" secondAttribute="top" constant="4" id="zay-ZJ-od3"/>
|
|
||||||
</constraints>
|
</constraints>
|
||||||
<viewLayoutGuide key="safeArea" id="M3G-7s-D6y"/>
|
<viewLayoutGuide key="safeArea" id="M3G-7s-D6y"/>
|
||||||
<viewLayoutGuide key="layoutMargins" id="Ebd-af-pc9"/>
|
<viewLayoutGuide key="layoutMargins" id="Ebd-af-pc9"/>
|
||||||
@ -543,6 +541,7 @@
|
|||||||
<outlet property="newestToOldestMenuItem" destination="40c-kt-vhO" id="AGa-fX-EVy"/>
|
<outlet property="newestToOldestMenuItem" destination="40c-kt-vhO" id="AGa-fX-EVy"/>
|
||||||
<outlet property="oldestToNewestMenuItem" destination="sOF-Ez-vIL" id="qSg-ST-ww9"/>
|
<outlet property="oldestToNewestMenuItem" destination="sOF-Ez-vIL" id="qSg-ST-ww9"/>
|
||||||
<outlet property="readFilteredButton" destination="iA5-go-AO0" id="kQg-2g-zNZ"/>
|
<outlet property="readFilteredButton" destination="iA5-go-AO0" id="kQg-2g-zNZ"/>
|
||||||
|
<outlet property="sortAndFilterViewHeightConstraint" destination="ZMg-gZ-6aa" id="gH7-E6-ABU"/>
|
||||||
<outlet property="viewOptionsPopUpButton" destination="lSU-OC-sEC" id="Z8V-rm-n2m"/>
|
<outlet property="viewOptionsPopUpButton" destination="lSU-OC-sEC" id="Z8V-rm-n2m"/>
|
||||||
</connections>
|
</connections>
|
||||||
</viewController>
|
</viewController>
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<document type="com.apple.InterfaceBuilder3.Cocoa.Storyboard.XIB" version="3.0" toolsVersion="17505" targetRuntime="MacOSX.Cocoa" propertyAccessControl="none" useAutolayout="YES" initialViewController="mPU-HG-I4u">
|
<document type="com.apple.InterfaceBuilder3.Cocoa.Storyboard.XIB" version="3.0" toolsVersion="17506" targetRuntime="MacOSX.Cocoa" propertyAccessControl="none" useAutolayout="YES" initialViewController="mPU-HG-I4u">
|
||||||
<dependencies>
|
<dependencies>
|
||||||
<deployment identifier="macosx"/>
|
<deployment identifier="macosx"/>
|
||||||
<plugIn identifier="com.apple.InterfaceBuilder.CocoaPlugin" version="17505"/>
|
<plugIn identifier="com.apple.InterfaceBuilder.CocoaPlugin" version="17506"/>
|
||||||
<capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/>
|
<capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
<scenes>
|
<scenes>
|
||||||
@ -256,15 +256,15 @@
|
|||||||
<scene sceneID="z1G-rc-sP5">
|
<scene sceneID="z1G-rc-sP5">
|
||||||
<objects>
|
<objects>
|
||||||
<viewController storyboardIdentifier="Advanced" id="GNh-Wp-giO" customClass="AdvancedPreferencesViewController" customModule="NetNewsWire" customModuleProvider="target" sceneMemberID="viewController">
|
<viewController storyboardIdentifier="Advanced" id="GNh-Wp-giO" customClass="AdvancedPreferencesViewController" customModule="NetNewsWire" customModuleProvider="target" sceneMemberID="viewController">
|
||||||
<view key="view" id="Hij-7D-6Pw">
|
<view key="view" misplaced="YES" id="Hij-7D-6Pw">
|
||||||
<rect key="frame" x="0.0" y="0.0" width="450" height="290"/>
|
<rect key="frame" x="0.0" y="0.0" width="450" height="291"/>
|
||||||
<autoresizingMask key="autoresizingMask"/>
|
<autoresizingMask key="autoresizingMask"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
<customView translatesAutoresizingMaskIntoConstraints="NO" id="uJD-OF-YVY">
|
<customView translatesAutoresizingMaskIntoConstraints="NO" id="uJD-OF-YVY">
|
||||||
<rect key="frame" x="60" y="20" width="330" height="250"/>
|
<rect key="frame" x="60" y="20" width="330" height="251"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
<textField horizontalHuggingPriority="1000" verticalHuggingPriority="1000" horizontalCompressionResistancePriority="1000" verticalCompressionResistancePriority="1000" translatesAutoresizingMaskIntoConstraints="NO" id="EH5-aS-E55">
|
<textField horizontalHuggingPriority="1000" verticalHuggingPriority="1000" horizontalCompressionResistancePriority="1000" verticalCompressionResistancePriority="1000" translatesAutoresizingMaskIntoConstraints="NO" id="EH5-aS-E55">
|
||||||
<rect key="frame" x="-2" y="234" width="87" height="16"/>
|
<rect key="frame" x="-2" y="235" width="87" height="16"/>
|
||||||
<textFieldCell key="cell" scrollable="YES" lineBreakMode="clipping" sendsActionOnEndEditing="YES" alignment="right" title="App Updates:" id="zqG-X2-E9b">
|
<textFieldCell key="cell" scrollable="YES" lineBreakMode="clipping" sendsActionOnEndEditing="YES" alignment="right" title="App Updates:" id="zqG-X2-E9b">
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
<color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/>
|
<color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/>
|
||||||
@ -272,7 +272,7 @@
|
|||||||
</textFieldCell>
|
</textFieldCell>
|
||||||
</textField>
|
</textField>
|
||||||
<button verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="T4A-0o-p2w">
|
<button verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="T4A-0o-p2w">
|
||||||
<rect key="frame" x="89" y="233" width="149" height="18"/>
|
<rect key="frame" x="89" y="234" width="149" height="18"/>
|
||||||
<buttonCell key="cell" type="check" title="Check automatically" bezelStyle="regularSquare" imagePosition="left" state="on" inset="2" id="dm8-Xy-0Ba">
|
<buttonCell key="cell" type="check" title="Check automatically" bezelStyle="regularSquare" imagePosition="left" state="on" inset="2" id="dm8-Xy-0Ba">
|
||||||
<behavior key="behavior" changeContents="YES" doesNotDimImage="YES" lightByContents="YES"/>
|
<behavior key="behavior" changeContents="YES" doesNotDimImage="YES" lightByContents="YES"/>
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
@ -282,7 +282,7 @@
|
|||||||
</connections>
|
</connections>
|
||||||
</button>
|
</button>
|
||||||
<textField horizontalHuggingPriority="251" verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="Q6M-Iz-Ypx">
|
<textField horizontalHuggingPriority="251" verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="Q6M-Iz-Ypx">
|
||||||
<rect key="frame" x="17" y="206" width="68" height="16"/>
|
<rect key="frame" x="17" y="207" width="68" height="16"/>
|
||||||
<textFieldCell key="cell" lineBreakMode="clipping" alignment="right" title="Download:" id="6bb-c0-guo">
|
<textFieldCell key="cell" lineBreakMode="clipping" alignment="right" title="Download:" id="6bb-c0-guo">
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
<color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/>
|
<color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/>
|
||||||
@ -290,7 +290,7 @@
|
|||||||
</textFieldCell>
|
</textFieldCell>
|
||||||
</textField>
|
</textField>
|
||||||
<button verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="QCu-J4-0yV">
|
<button verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="QCu-J4-0yV">
|
||||||
<rect key="frame" x="89" y="205" width="114" height="18"/>
|
<rect key="frame" x="89" y="206" width="114" height="18"/>
|
||||||
<buttonCell key="cell" type="radio" title="Release builds" bezelStyle="regularSquare" imagePosition="left" alignment="left" inset="2" id="F8M-rS-und">
|
<buttonCell key="cell" type="radio" title="Release builds" bezelStyle="regularSquare" imagePosition="left" alignment="left" inset="2" id="F8M-rS-und">
|
||||||
<behavior key="behavior" changeContents="YES" doesNotDimImage="YES" lightByContents="YES"/>
|
<behavior key="behavior" changeContents="YES" doesNotDimImage="YES" lightByContents="YES"/>
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
@ -300,7 +300,7 @@
|
|||||||
</connections>
|
</connections>
|
||||||
</button>
|
</button>
|
||||||
<button verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="CeE-AE-hRG">
|
<button verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="CeE-AE-hRG">
|
||||||
<rect key="frame" x="89" y="183" width="92" height="18"/>
|
<rect key="frame" x="89" y="184" width="92" height="18"/>
|
||||||
<buttonCell key="cell" type="radio" title="Test builds" bezelStyle="regularSquare" imagePosition="left" alignment="left" inset="2" id="Fuf-rU-D6M">
|
<buttonCell key="cell" type="radio" title="Test builds" bezelStyle="regularSquare" imagePosition="left" alignment="left" inset="2" id="Fuf-rU-D6M">
|
||||||
<behavior key="behavior" changeContents="YES" doesNotDimImage="YES" lightByContents="YES"/>
|
<behavior key="behavior" changeContents="YES" doesNotDimImage="YES" lightByContents="YES"/>
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
@ -310,7 +310,7 @@
|
|||||||
</connections>
|
</connections>
|
||||||
</button>
|
</button>
|
||||||
<textField verticalHuggingPriority="750" horizontalCompressionResistancePriority="250" translatesAutoresizingMaskIntoConstraints="NO" id="MzL-QQ-2oL">
|
<textField verticalHuggingPriority="750" horizontalCompressionResistancePriority="250" translatesAutoresizingMaskIntoConstraints="NO" id="MzL-QQ-2oL">
|
||||||
<rect key="frame" x="-2" y="128" width="334" height="48"/>
|
<rect key="frame" x="-2" y="129" width="334" height="48"/>
|
||||||
<constraints>
|
<constraints>
|
||||||
<constraint firstAttribute="width" constant="330" id="jf8-5e-Eij"/>
|
<constraint firstAttribute="width" constant="330" id="jf8-5e-Eij"/>
|
||||||
</constraints>
|
</constraints>
|
||||||
@ -321,7 +321,7 @@
|
|||||||
</textFieldCell>
|
</textFieldCell>
|
||||||
</textField>
|
</textField>
|
||||||
<button verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="TKI-a9-bRX">
|
<button verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="TKI-a9-bRX">
|
||||||
<rect key="frame" x="84" y="93" width="148" height="32"/>
|
<rect key="frame" x="84" y="94" width="148" height="32"/>
|
||||||
<buttonCell key="cell" type="push" title="Check for Updates" bezelStyle="rounded" alignment="center" borderStyle="border" imageScaling="proportionallyDown" inset="2" id="AaA-Rr-UYD">
|
<buttonCell key="cell" type="push" title="Check for Updates" bezelStyle="rounded" alignment="center" borderStyle="border" imageScaling="proportionallyDown" inset="2" id="AaA-Rr-UYD">
|
||||||
<behavior key="behavior" pushIn="YES" lightByBackground="YES" lightByGray="YES"/>
|
<behavior key="behavior" pushIn="YES" lightByBackground="YES" lightByGray="YES"/>
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
@ -331,7 +331,7 @@
|
|||||||
</connections>
|
</connections>
|
||||||
</button>
|
</button>
|
||||||
<textField horizontalHuggingPriority="1000" verticalHuggingPriority="1000" horizontalCompressionResistancePriority="1000" verticalCompressionResistancePriority="1000" translatesAutoresizingMaskIntoConstraints="NO" id="SUN-k3-ZEb">
|
<textField horizontalHuggingPriority="1000" verticalHuggingPriority="1000" horizontalCompressionResistancePriority="1000" verticalCompressionResistancePriority="1000" translatesAutoresizingMaskIntoConstraints="NO" id="SUN-k3-ZEb">
|
||||||
<rect key="frame" x="12" y="52" width="73" height="16"/>
|
<rect key="frame" x="12" y="53" width="73" height="16"/>
|
||||||
<textFieldCell key="cell" scrollable="YES" lineBreakMode="clipping" sendsActionOnEndEditing="YES" alignment="right" title="Crash logs:" id="qcq-fU-Ks0">
|
<textFieldCell key="cell" scrollable="YES" lineBreakMode="clipping" sendsActionOnEndEditing="YES" alignment="right" title="Crash logs:" id="qcq-fU-Ks0">
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
<color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/>
|
<color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/>
|
||||||
@ -339,7 +339,7 @@
|
|||||||
</textFieldCell>
|
</textFieldCell>
|
||||||
</textField>
|
</textField>
|
||||||
<button horizontalHuggingPriority="1000" verticalHuggingPriority="1000" horizontalCompressionResistancePriority="1000" verticalCompressionResistancePriority="1000" translatesAutoresizingMaskIntoConstraints="NO" id="UHg-1l-FlD">
|
<button horizontalHuggingPriority="1000" verticalHuggingPriority="1000" horizontalCompressionResistancePriority="1000" verticalCompressionResistancePriority="1000" translatesAutoresizingMaskIntoConstraints="NO" id="UHg-1l-FlD">
|
||||||
<rect key="frame" x="89" y="51" width="142" height="18"/>
|
<rect key="frame" x="89" y="52" width="142" height="18"/>
|
||||||
<buttonCell key="cell" type="check" title="Send automatically" bezelStyle="regularSquare" imagePosition="left" state="on" inset="2" id="jnc-C5-4oI">
|
<buttonCell key="cell" type="check" title="Send automatically" bezelStyle="regularSquare" imagePosition="left" state="on" inset="2" id="jnc-C5-4oI">
|
||||||
<behavior key="behavior" changeContents="YES" doesNotDimImage="YES" lightByContents="YES"/>
|
<behavior key="behavior" changeContents="YES" doesNotDimImage="YES" lightByContents="YES"/>
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
@ -359,7 +359,7 @@
|
|||||||
</connections>
|
</connections>
|
||||||
</button>
|
</button>
|
||||||
<button verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="uuc-f2-OFX">
|
<button verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="uuc-f2-OFX">
|
||||||
<rect key="frame" x="84" y="-7" width="148" height="32"/>
|
<rect key="frame" x="84" y="-6" width="148" height="32"/>
|
||||||
<buttonCell key="cell" type="push" title="Privacy Policy" bezelStyle="rounded" alignment="center" borderStyle="border" imageScaling="proportionallyDown" inset="2" id="kSv-Wu-NYx">
|
<buttonCell key="cell" type="push" title="Privacy Policy" bezelStyle="rounded" alignment="center" borderStyle="border" imageScaling="proportionallyDown" inset="2" id="kSv-Wu-NYx">
|
||||||
<behavior key="behavior" pushIn="YES" lightByBackground="YES" lightByGray="YES"/>
|
<behavior key="behavior" pushIn="YES" lightByBackground="YES" lightByGray="YES"/>
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
@ -369,10 +369,10 @@
|
|||||||
</connections>
|
</connections>
|
||||||
</button>
|
</button>
|
||||||
<box verticalHuggingPriority="750" boxType="separator" translatesAutoresizingMaskIntoConstraints="NO" id="UD5-5N-W4F">
|
<box verticalHuggingPriority="750" boxType="separator" translatesAutoresizingMaskIntoConstraints="NO" id="UD5-5N-W4F">
|
||||||
<rect key="frame" x="0.0" y="77" width="330" height="5"/>
|
<rect key="frame" x="0.0" y="78" width="330" height="5"/>
|
||||||
</box>
|
</box>
|
||||||
<box verticalHuggingPriority="750" boxType="separator" translatesAutoresizingMaskIntoConstraints="NO" id="B1Q-jV-3Yl">
|
<box verticalHuggingPriority="750" boxType="separator" translatesAutoresizingMaskIntoConstraints="NO" id="B1Q-jV-3Yl">
|
||||||
<rect key="frame" x="0.0" y="33" width="330" height="5"/>
|
<rect key="frame" x="0.0" y="34" width="330" height="5"/>
|
||||||
</box>
|
</box>
|
||||||
</subviews>
|
</subviews>
|
||||||
<constraints>
|
<constraints>
|
||||||
@ -392,7 +392,7 @@
|
|||||||
<constraint firstAttribute="trailing" relation="greaterThanOrEqual" secondItem="QCu-J4-0yV" secondAttribute="trailing" id="QVh-z8-aNJ"/>
|
<constraint firstAttribute="trailing" relation="greaterThanOrEqual" secondItem="QCu-J4-0yV" secondAttribute="trailing" id="QVh-z8-aNJ"/>
|
||||||
<constraint firstItem="UHg-1l-FlD" firstAttribute="leading" secondItem="CeE-AE-hRG" secondAttribute="leading" id="QlP-bI-uga"/>
|
<constraint firstItem="UHg-1l-FlD" firstAttribute="leading" secondItem="CeE-AE-hRG" secondAttribute="leading" id="QlP-bI-uga"/>
|
||||||
<constraint firstItem="EH5-aS-E55" firstAttribute="top" secondItem="uJD-OF-YVY" secondAttribute="top" id="VDU-as-fdx"/>
|
<constraint firstItem="EH5-aS-E55" firstAttribute="top" secondItem="uJD-OF-YVY" secondAttribute="top" id="VDU-as-fdx"/>
|
||||||
<constraint firstAttribute="bottom" secondItem="uuc-f2-OFX" secondAttribute="bottom" id="YA7-Xm-cFO"/>
|
<constraint firstAttribute="bottom" secondItem="uuc-f2-OFX" secondAttribute="bottom" constant="1" id="YA7-Xm-cFO"/>
|
||||||
<constraint firstItem="Q6M-Iz-Ypx" firstAttribute="leading" relation="greaterThanOrEqual" secondItem="uJD-OF-YVY" secondAttribute="leading" id="Ygv-ha-RLn"/>
|
<constraint firstItem="Q6M-Iz-Ypx" firstAttribute="leading" relation="greaterThanOrEqual" secondItem="uJD-OF-YVY" secondAttribute="leading" id="Ygv-ha-RLn"/>
|
||||||
<constraint firstItem="UD5-5N-W4F" firstAttribute="top" secondItem="TKI-a9-bRX" secondAttribute="bottom" constant="20" symbolic="YES" id="b3Y-RX-O4j"/>
|
<constraint firstItem="UD5-5N-W4F" firstAttribute="top" secondItem="TKI-a9-bRX" secondAttribute="bottom" constant="20" symbolic="YES" id="b3Y-RX-O4j"/>
|
||||||
<constraint firstAttribute="trailing" relation="greaterThanOrEqual" secondItem="TKI-a9-bRX" secondAttribute="trailing" id="bLP-TU-TeL"/>
|
<constraint firstAttribute="trailing" relation="greaterThanOrEqual" secondItem="TKI-a9-bRX" secondAttribute="trailing" id="bLP-TU-TeL"/>
|
||||||
@ -442,16 +442,16 @@
|
|||||||
<autoresizingMask key="autoresizingMask"/>
|
<autoresizingMask key="autoresizingMask"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
<customView translatesAutoresizingMaskIntoConstraints="NO" id="7UM-iq-OLB" customClass="PreferencesTableViewBackgroundView" customModule="NetNewsWire" customModuleProvider="target">
|
<customView translatesAutoresizingMaskIntoConstraints="NO" id="7UM-iq-OLB" customClass="PreferencesTableViewBackgroundView" customModule="NetNewsWire" customModuleProvider="target">
|
||||||
<rect key="frame" x="20" y="44" width="180" height="314"/>
|
<rect key="frame" x="20" y="44" width="180" height="324"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
<scrollView borderType="none" autohidesScrollers="YES" horizontalLineScroll="26" horizontalPageScroll="10" verticalLineScroll="26" verticalPageScroll="10" hasHorizontalScroller="NO" horizontalScrollElasticity="none" translatesAutoresizingMaskIntoConstraints="NO" id="PaF-du-r3c">
|
<scrollView borderType="none" autohidesScrollers="YES" horizontalLineScroll="26" horizontalPageScroll="10" verticalLineScroll="26" verticalPageScroll="10" hasHorizontalScroller="NO" horizontalScrollElasticity="none" translatesAutoresizingMaskIntoConstraints="NO" id="PaF-du-r3c">
|
||||||
<rect key="frame" x="1" y="0.0" width="178" height="313"/>
|
<rect key="frame" x="1" y="0.0" width="178" height="323"/>
|
||||||
<clipView key="contentView" id="cil-Gq-akO">
|
<clipView key="contentView" id="cil-Gq-akO">
|
||||||
<rect key="frame" x="0.0" y="0.0" width="178" height="313"/>
|
<rect key="frame" x="0.0" y="0.0" width="178" height="323"/>
|
||||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
<tableView verticalHuggingPriority="750" allowsExpansionToolTips="YES" columnAutoresizingStyle="lastColumnOnly" columnReordering="NO" columnSelection="YES" columnResizing="NO" multipleSelection="NO" autosaveColumns="NO" rowHeight="24" viewBased="YES" id="aTp-KR-y6b">
|
<tableView verticalHuggingPriority="750" allowsExpansionToolTips="YES" columnAutoresizingStyle="lastColumnOnly" columnReordering="NO" columnSelection="YES" columnResizing="NO" multipleSelection="NO" autosaveColumns="NO" rowHeight="24" viewBased="YES" id="aTp-KR-y6b">
|
||||||
<rect key="frame" x="0.0" y="0.0" width="178" height="313"/>
|
<rect key="frame" x="0.0" y="0.0" width="178" height="323"/>
|
||||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||||
<size key="intercellSpacing" width="3" height="2"/>
|
<size key="intercellSpacing" width="3" height="2"/>
|
||||||
<color key="backgroundColor" name="controlBackgroundColor" catalog="System" colorSpace="catalog"/>
|
<color key="backgroundColor" name="controlBackgroundColor" catalog="System" colorSpace="catalog"/>
|
||||||
@ -469,7 +469,7 @@
|
|||||||
</textFieldCell>
|
</textFieldCell>
|
||||||
<tableColumnResizingMask key="resizingMask" resizeWithTable="YES" userResizable="YES"/>
|
<tableColumnResizingMask key="resizingMask" resizeWithTable="YES" userResizable="YES"/>
|
||||||
<prototypeCellViews>
|
<prototypeCellViews>
|
||||||
<tableCellView identifier="Cell" id="h2e-5a-qNO">
|
<tableCellView identifier="Cell" id="h2e-5a-qNO" customClass="AccountCell" customModule="NetNewsWire" customModuleProvider="target">
|
||||||
<rect key="frame" x="11" y="1" width="155" height="17"/>
|
<rect key="frame" x="11" y="1" width="155" height="17"/>
|
||||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
@ -558,7 +558,7 @@
|
|||||||
<rect key="frame" x="83" y="20" width="117" height="24"/>
|
<rect key="frame" x="83" y="20" width="117" height="24"/>
|
||||||
</customView>
|
</customView>
|
||||||
<customView translatesAutoresizingMaskIntoConstraints="NO" id="Y7D-xQ-wep">
|
<customView translatesAutoresizingMaskIntoConstraints="NO" id="Y7D-xQ-wep">
|
||||||
<rect key="frame" x="208" y="20" width="222" height="338"/>
|
<rect key="frame" x="208" y="20" width="222" height="348"/>
|
||||||
</customView>
|
</customView>
|
||||||
</subviews>
|
</subviews>
|
||||||
<constraints>
|
<constraints>
|
||||||
@ -613,16 +613,16 @@
|
|||||||
<autoresizingMask key="autoresizingMask"/>
|
<autoresizingMask key="autoresizingMask"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
<customView translatesAutoresizingMaskIntoConstraints="NO" id="pjs-G4-byk" customClass="PreferencesTableViewBackgroundView" customModule="NetNewsWire" customModuleProvider="target">
|
<customView translatesAutoresizingMaskIntoConstraints="NO" id="pjs-G4-byk" customClass="PreferencesTableViewBackgroundView" customModule="NetNewsWire" customModuleProvider="target">
|
||||||
<rect key="frame" x="20" y="44" width="180" height="307"/>
|
<rect key="frame" x="20" y="44" width="180" height="317"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
<scrollView borderType="none" autohidesScrollers="YES" horizontalLineScroll="26" horizontalPageScroll="10" verticalLineScroll="26" verticalPageScroll="10" hasHorizontalScroller="NO" horizontalScrollElasticity="none" translatesAutoresizingMaskIntoConstraints="NO" id="29T-r2-ckC">
|
<scrollView borderType="none" autohidesScrollers="YES" horizontalLineScroll="26" horizontalPageScroll="10" verticalLineScroll="26" verticalPageScroll="10" hasHorizontalScroller="NO" horizontalScrollElasticity="none" translatesAutoresizingMaskIntoConstraints="NO" id="29T-r2-ckC">
|
||||||
<rect key="frame" x="1" y="0.0" width="178" height="306"/>
|
<rect key="frame" x="1" y="0.0" width="178" height="316"/>
|
||||||
<clipView key="contentView" id="dXw-GY-TP8">
|
<clipView key="contentView" id="dXw-GY-TP8">
|
||||||
<rect key="frame" x="0.0" y="0.0" width="178" height="306"/>
|
<rect key="frame" x="0.0" y="0.0" width="178" height="316"/>
|
||||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
<tableView verticalHuggingPriority="750" allowsExpansionToolTips="YES" columnAutoresizingStyle="lastColumnOnly" columnReordering="NO" columnSelection="YES" columnResizing="NO" multipleSelection="NO" autosaveColumns="NO" rowHeight="24" viewBased="YES" id="dfn-Vn-oDp">
|
<tableView verticalHuggingPriority="750" allowsExpansionToolTips="YES" columnAutoresizingStyle="lastColumnOnly" columnReordering="NO" columnSelection="YES" columnResizing="NO" multipleSelection="NO" autosaveColumns="NO" rowHeight="24" viewBased="YES" id="dfn-Vn-oDp">
|
||||||
<rect key="frame" x="0.0" y="0.0" width="178" height="306"/>
|
<rect key="frame" x="0.0" y="0.0" width="178" height="316"/>
|
||||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||||
<size key="intercellSpacing" width="3" height="2"/>
|
<size key="intercellSpacing" width="3" height="2"/>
|
||||||
<color key="backgroundColor" name="controlBackgroundColor" catalog="System" colorSpace="catalog"/>
|
<color key="backgroundColor" name="controlBackgroundColor" catalog="System" colorSpace="catalog"/>
|
||||||
@ -725,7 +725,7 @@
|
|||||||
<rect key="frame" x="83" y="20" width="117" height="24"/>
|
<rect key="frame" x="83" y="20" width="117" height="24"/>
|
||||||
</customView>
|
</customView>
|
||||||
<customView translatesAutoresizingMaskIntoConstraints="NO" id="N1N-pE-gBL">
|
<customView translatesAutoresizingMaskIntoConstraints="NO" id="N1N-pE-gBL">
|
||||||
<rect key="frame" x="208" y="20" width="222" height="331"/>
|
<rect key="frame" x="208" y="20" width="222" height="341"/>
|
||||||
</customView>
|
</customView>
|
||||||
</subviews>
|
</subviews>
|
||||||
<constraints>
|
<constraints>
|
||||||
|
@ -642,6 +642,7 @@ extension MainWindowController: NSSearchFieldDelegate {
|
|||||||
let smartFeed = SmartFeed(delegate: SearchFeedDelegate(searchString: searchString))
|
let smartFeed = SmartFeed(delegate: SearchFeedDelegate(searchString: searchString))
|
||||||
timelineContainerViewController?.setRepresentedObjects([smartFeed], mode: .search)
|
timelineContainerViewController?.setRepresentedObjects([smartFeed], mode: .search)
|
||||||
searchSmartFeed = smartFeed
|
searchSmartFeed = smartFeed
|
||||||
|
updateWindowTitle()
|
||||||
}
|
}
|
||||||
|
|
||||||
func forceSearchToEnd() {
|
func forceSearchToEnd() {
|
||||||
@ -651,10 +652,12 @@ extension MainWindowController: NSSearchFieldDelegate {
|
|||||||
if let searchField = currentSearchField {
|
if let searchField = currentSearchField {
|
||||||
searchField.stringValue = ""
|
searchField.stringValue = ""
|
||||||
}
|
}
|
||||||
|
updateWindowTitle()
|
||||||
}
|
}
|
||||||
|
|
||||||
private func startSearchingIfNeeded() {
|
private func startSearchingIfNeeded() {
|
||||||
timelineSourceMode = .search
|
timelineSourceMode = .search
|
||||||
|
updateWindowTitle()
|
||||||
}
|
}
|
||||||
|
|
||||||
private func stopSearchingIfNeeded() {
|
private func stopSearchingIfNeeded() {
|
||||||
@ -662,6 +665,7 @@ extension MainWindowController: NSSearchFieldDelegate {
|
|||||||
lastSentSearchString = nil
|
lastSentSearchString = nil
|
||||||
timelineSourceMode = .regular
|
timelineSourceMode = .regular
|
||||||
timelineContainerViewController?.setRepresentedObjects(nil, mode: .search)
|
timelineContainerViewController?.setRepresentedObjects(nil, mode: .search)
|
||||||
|
updateWindowTitle()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -716,6 +720,7 @@ extension NSToolbarItem.Identifier {
|
|||||||
static let timelineTrackingSeparator = NSToolbarItem.Identifier("timelineTrackingSeparator")
|
static let timelineTrackingSeparator = NSToolbarItem.Identifier("timelineTrackingSeparator")
|
||||||
static let search = NSToolbarItem.Identifier("search")
|
static let search = NSToolbarItem.Identifier("search")
|
||||||
static let markAllAsRead = NSToolbarItem.Identifier("markAllAsRead")
|
static let markAllAsRead = NSToolbarItem.Identifier("markAllAsRead")
|
||||||
|
static let toggleReadArticlesFilter = NSToolbarItem.Identifier("toggleReadArticlesFilter")
|
||||||
static let nextUnread = NSToolbarItem.Identifier("nextUnread")
|
static let nextUnread = NSToolbarItem.Identifier("nextUnread")
|
||||||
static let markRead = NSToolbarItem.Identifier("markRead")
|
static let markRead = NSToolbarItem.Identifier("markRead")
|
||||||
static let markStar = NSToolbarItem.Identifier("markStar")
|
static let markStar = NSToolbarItem.Identifier("markStar")
|
||||||
@ -749,24 +754,17 @@ extension MainWindowController: NSToolbarDelegate {
|
|||||||
toolbarItem.menu = buildNewSidebarItemMenu()
|
toolbarItem.menu = buildNewSidebarItemMenu()
|
||||||
return toolbarItem
|
return toolbarItem
|
||||||
|
|
||||||
case .search:
|
|
||||||
let toolbarItem = NSSearchToolbarItem(itemIdentifier: .search)
|
|
||||||
let description = NSLocalizedString("Search", comment: "Search")
|
|
||||||
toolbarItem.toolTip = description
|
|
||||||
toolbarItem.label = description
|
|
||||||
return toolbarItem
|
|
||||||
|
|
||||||
case .markAllAsRead:
|
case .markAllAsRead:
|
||||||
let title = NSLocalizedString("Mark All as Read", comment: "Mark All as Read")
|
let title = NSLocalizedString("Mark All as Read", comment: "Mark All as Read")
|
||||||
return buildToolbarButton(.markAllAsRead, title, AppAssets.markAllAsReadImage, "markAllAsRead:")
|
return buildToolbarButton(.markAllAsRead, title, AppAssets.markAllAsReadImage, "markAllAsRead:")
|
||||||
|
|
||||||
|
case .toggleReadArticlesFilter:
|
||||||
|
let title = NSLocalizedString("Read Articles Filter", comment: "Read Articles Filter")
|
||||||
|
return buildToolbarButton(.toggleReadArticlesFilter, title, AppAssets.filterInactive, "toggleReadArticlesFilter:")
|
||||||
|
|
||||||
case .timelineTrackingSeparator:
|
case .timelineTrackingSeparator:
|
||||||
return NSTrackingSeparatorToolbarItem(identifier: .timelineTrackingSeparator, splitView: splitViewController!.splitView, dividerIndex: 1)
|
return NSTrackingSeparatorToolbarItem(identifier: .timelineTrackingSeparator, splitView: splitViewController!.splitView, dividerIndex: 1)
|
||||||
|
|
||||||
case .nextUnread:
|
|
||||||
let title = NSLocalizedString("Next Unread", comment: "Next Unread")
|
|
||||||
return buildToolbarButton(.nextUnread, title, AppAssets.nextUnreadImage, "nextUnread:")
|
|
||||||
|
|
||||||
case .markRead:
|
case .markRead:
|
||||||
let title = NSLocalizedString("Mark Read", comment: "Mark Read")
|
let title = NSLocalizedString("Mark Read", comment: "Mark Read")
|
||||||
return buildToolbarButton(.markRead, title, AppAssets.readClosedImage, "toggleRead:")
|
return buildToolbarButton(.markRead, title, AppAssets.readClosedImage, "toggleRead:")
|
||||||
@ -775,6 +773,10 @@ extension MainWindowController: NSToolbarDelegate {
|
|||||||
let title = NSLocalizedString("Star", comment: "Star")
|
let title = NSLocalizedString("Star", comment: "Star")
|
||||||
return buildToolbarButton(.markStar, title, AppAssets.starOpenImage, "toggleStarred:")
|
return buildToolbarButton(.markStar, title, AppAssets.starOpenImage, "toggleStarred:")
|
||||||
|
|
||||||
|
case .nextUnread:
|
||||||
|
let title = NSLocalizedString("Next Unread", comment: "Next Unread")
|
||||||
|
return buildToolbarButton(.nextUnread, title, AppAssets.nextUnreadImage, "nextUnread:")
|
||||||
|
|
||||||
case .readerView:
|
case .readerView:
|
||||||
let toolbarItem = RSToolbarItem(itemIdentifier: .readerView)
|
let toolbarItem = RSToolbarItem(itemIdentifier: .readerView)
|
||||||
toolbarItem.autovalidates = true
|
toolbarItem.autovalidates = true
|
||||||
@ -786,14 +788,21 @@ extension MainWindowController: NSToolbarDelegate {
|
|||||||
toolbarItem.view = button
|
toolbarItem.view = button
|
||||||
return toolbarItem
|
return toolbarItem
|
||||||
|
|
||||||
case .openInBrowser:
|
|
||||||
let title = NSLocalizedString("Open in Browser", comment: "Open in Browser")
|
|
||||||
return buildToolbarButton(.openInBrowser, title, AppAssets.openInBrowserImage, "openArticleInBrowser:")
|
|
||||||
|
|
||||||
case .share:
|
case .share:
|
||||||
let title = NSLocalizedString("Share", comment: "Share")
|
let title = NSLocalizedString("Share", comment: "Share")
|
||||||
return buildToolbarButton(.share, title, AppAssets.shareImage, "toolbarShowShareMenu:")
|
return buildToolbarButton(.share, title, AppAssets.shareImage, "toolbarShowShareMenu:")
|
||||||
|
|
||||||
|
case .openInBrowser:
|
||||||
|
let title = NSLocalizedString("Open in Browser", comment: "Open in Browser")
|
||||||
|
return buildToolbarButton(.openInBrowser, title, AppAssets.openInBrowserImage, "openArticleInBrowser:")
|
||||||
|
|
||||||
|
case .search:
|
||||||
|
let toolbarItem = NSSearchToolbarItem(itemIdentifier: .search)
|
||||||
|
let description = NSLocalizedString("Search", comment: "Search")
|
||||||
|
toolbarItem.toolTip = description
|
||||||
|
toolbarItem.label = description
|
||||||
|
return toolbarItem
|
||||||
|
|
||||||
case .cleanUp:
|
case .cleanUp:
|
||||||
let title = NSLocalizedString("Clean Up", comment: "Clean Up")
|
let title = NSLocalizedString("Clean Up", comment: "Clean Up")
|
||||||
return buildToolbarButton(.cleanUp, title, AppAssets.cleanUpImage, "cleanUp:")
|
return buildToolbarButton(.cleanUp, title, AppAssets.cleanUpImage, "cleanUp:")
|
||||||
@ -815,7 +824,7 @@ extension MainWindowController: NSToolbarDelegate {
|
|||||||
.newSidebarItemMenu,
|
.newSidebarItemMenu,
|
||||||
.sidebarTrackingSeparator,
|
.sidebarTrackingSeparator,
|
||||||
.markAllAsRead,
|
.markAllAsRead,
|
||||||
.search,
|
.toggleReadArticlesFilter,
|
||||||
.timelineTrackingSeparator,
|
.timelineTrackingSeparator,
|
||||||
.flexibleSpace,
|
.flexibleSpace,
|
||||||
.nextUnread,
|
.nextUnread,
|
||||||
@ -824,6 +833,7 @@ extension MainWindowController: NSToolbarDelegate {
|
|||||||
.readerView,
|
.readerView,
|
||||||
.openInBrowser,
|
.openInBrowser,
|
||||||
.share,
|
.share,
|
||||||
|
.search,
|
||||||
.cleanUp
|
.cleanUp
|
||||||
]
|
]
|
||||||
} else {
|
} else {
|
||||||
@ -854,15 +864,16 @@ extension MainWindowController: NSToolbarDelegate {
|
|||||||
.newSidebarItemMenu,
|
.newSidebarItemMenu,
|
||||||
.sidebarTrackingSeparator,
|
.sidebarTrackingSeparator,
|
||||||
.markAllAsRead,
|
.markAllAsRead,
|
||||||
.search,
|
.toggleReadArticlesFilter,
|
||||||
.timelineTrackingSeparator,
|
.timelineTrackingSeparator,
|
||||||
.flexibleSpace,
|
|
||||||
.nextUnread,
|
|
||||||
.markRead,
|
.markRead,
|
||||||
.markStar,
|
.markStar,
|
||||||
|
.nextUnread,
|
||||||
.readerView,
|
.readerView,
|
||||||
|
.share,
|
||||||
.openInBrowser,
|
.openInBrowser,
|
||||||
.share
|
.flexibleSpace,
|
||||||
|
.search
|
||||||
]
|
]
|
||||||
} else {
|
} else {
|
||||||
return [
|
return [
|
||||||
@ -1175,18 +1186,33 @@ private extension MainWindowController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func validateToggleReadArticles(_ item: NSValidatedUserInterfaceItem) -> Bool {
|
func validateToggleReadArticles(_ item: NSValidatedUserInterfaceItem) -> Bool {
|
||||||
guard let menuItem = item as? NSMenuItem else { return false }
|
|
||||||
|
|
||||||
let showCommand = NSLocalizedString("Show Read Articles", comment: "Command")
|
let showCommand = NSLocalizedString("Show Read Articles", comment: "Command")
|
||||||
let hideCommand = NSLocalizedString("Hide Read Articles", comment: "Command")
|
let hideCommand = NSLocalizedString("Hide Read Articles", comment: "Command")
|
||||||
|
|
||||||
if let isReadFiltered = timelineContainerViewController?.isReadFiltered {
|
guard let isReadFiltered = timelineContainerViewController?.isReadFiltered else {
|
||||||
menuItem.title = isReadFiltered ? showCommand : hideCommand
|
(item as? NSMenuItem)?.title = hideCommand
|
||||||
return true
|
if #available(macOS 11.0, *), let toolbarItem = item as? NSToolbarItem, let button = toolbarItem.view as? NSButton {
|
||||||
} else {
|
toolbarItem.toolTip = hideCommand
|
||||||
menuItem.title = showCommand
|
button.image = AppAssets.filterInactive
|
||||||
|
}
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if isReadFiltered {
|
||||||
|
(item as? NSMenuItem)?.title = showCommand
|
||||||
|
if #available(macOS 11.0, *), let toolbarItem = item as? NSToolbarItem, let button = toolbarItem.view as? NSButton {
|
||||||
|
toolbarItem.toolTip = showCommand
|
||||||
|
button.image = AppAssets.filterActive
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
(item as? NSMenuItem)?.title = hideCommand
|
||||||
|
if #available(macOS 11.0, *), let toolbarItem = item as? NSToolbarItem, let button = toolbarItem.view as? NSButton {
|
||||||
|
toolbarItem.toolTip = hideCommand
|
||||||
|
button.image = AppAssets.filterInactive
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: - Misc.
|
// MARK: - Misc.
|
||||||
@ -1212,6 +1238,15 @@ private extension MainWindowController {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func updateWindowTitle() {
|
func updateWindowTitle() {
|
||||||
|
guard timelineSourceMode != .search else {
|
||||||
|
let localizedLabel = NSLocalizedString("Search: %@", comment: "Search")
|
||||||
|
window?.title = NSString.localizedStringWithFormat(localizedLabel as NSString, searchString ?? "") as String
|
||||||
|
if #available(macOS 11.0, *) {
|
||||||
|
window?.subtitle = ""
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
func setSubtitle(_ count: Int) {
|
func setSubtitle(_ count: Int) {
|
||||||
let localizedLabel = NSLocalizedString("%d unread", comment: "Unread")
|
let localizedLabel = NSLocalizedString("%d unread", comment: "Unread")
|
||||||
let formattedLabel = NSString.localizedStringWithFormat(localizedLabel as NSString, count)
|
let formattedLabel = NSString.localizedStringWithFormat(localizedLabel as NSString, count)
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<document type="com.apple.InterfaceBuilder3.Cocoa.XIB" version="3.0" toolsVersion="15400" targetRuntime="MacOSX.Cocoa" propertyAccessControl="none" useAutolayout="YES" customObjectInstantitationMethod="direct">
|
<document type="com.apple.InterfaceBuilder3.Cocoa.XIB" version="3.0" toolsVersion="17506" targetRuntime="MacOSX.Cocoa" propertyAccessControl="none" useAutolayout="YES" customObjectInstantitationMethod="direct">
|
||||||
<dependencies>
|
<dependencies>
|
||||||
<deployment identifier="macosx"/>
|
<deployment identifier="macosx"/>
|
||||||
<plugIn identifier="com.apple.InterfaceBuilder.CocoaPlugin" version="15400"/>
|
<plugIn identifier="com.apple.InterfaceBuilder.CocoaPlugin" version="17506"/>
|
||||||
<capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/>
|
<capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
<objects>
|
<objects>
|
||||||
@ -19,10 +19,10 @@
|
|||||||
<autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMinY="YES"/>
|
<autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMinY="YES"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
<customView translatesAutoresizingMaskIntoConstraints="NO" id="gSF-Ze-XcY">
|
<customView translatesAutoresizingMaskIntoConstraints="NO" id="gSF-Ze-XcY">
|
||||||
<rect key="frame" x="50" y="20" width="400" height="77"/>
|
<rect key="frame" x="50" y="20" width="400" height="78"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
<textField verticalHuggingPriority="750" horizontalCompressionResistancePriority="250" translatesAutoresizingMaskIntoConstraints="NO" id="wGx-8H-BqE">
|
<textField verticalHuggingPriority="750" horizontalCompressionResistancePriority="250" translatesAutoresizingMaskIntoConstraints="NO" id="wGx-8H-BqE">
|
||||||
<rect key="frame" x="-2" y="29" width="404" height="48"/>
|
<rect key="frame" x="-2" y="30" width="404" height="48"/>
|
||||||
<textFieldCell key="cell" selectable="YES" id="IFj-4w-B03">
|
<textFieldCell key="cell" selectable="YES" id="IFj-4w-B03">
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
<string key="title">Choose a NetNewsWire 3 “Subscriptions.plist” file.
|
<string key="title">Choose a NetNewsWire 3 “Subscriptions.plist” file.
|
||||||
@ -33,7 +33,7 @@ Then choose the account to receive your imported subscriptions.</string>
|
|||||||
</textFieldCell>
|
</textFieldCell>
|
||||||
</textField>
|
</textField>
|
||||||
<textField horizontalHuggingPriority="251" verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="976-Ry-M6G">
|
<textField horizontalHuggingPriority="251" verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="976-Ry-M6G">
|
||||||
<rect key="frame" x="-2" y="3" width="63" height="16"/>
|
<rect key="frame" x="-2" y="5" width="63" height="16"/>
|
||||||
<textFieldCell key="cell" scrollable="YES" lineBreakMode="clipping" sendsActionOnEndEditing="YES" alignment="right" title="Account:" id="uoh-QY-7LX">
|
<textFieldCell key="cell" scrollable="YES" lineBreakMode="clipping" sendsActionOnEndEditing="YES" alignment="right" title="Account:" id="uoh-QY-7LX">
|
||||||
<font key="font" metaFont="systemBold"/>
|
<font key="font" metaFont="systemBold"/>
|
||||||
<color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/>
|
<color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/>
|
||||||
@ -41,10 +41,10 @@ Then choose the account to receive your imported subscriptions.</string>
|
|||||||
</textFieldCell>
|
</textFieldCell>
|
||||||
</textField>
|
</textField>
|
||||||
<popUpButton verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="M8B-pG-mg8" userLabel="Account Popup">
|
<popUpButton verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="M8B-pG-mg8" userLabel="Account Popup">
|
||||||
<rect key="frame" x="65" y="-3" width="338" height="25"/>
|
<rect key="frame" x="64" y="-2" width="340" height="25"/>
|
||||||
<popUpButtonCell key="cell" type="push" title="Item 1" bezelStyle="rounded" alignment="left" lineBreakMode="truncatingTail" state="on" borderStyle="borderAndBezel" imageScaling="proportionallyDown" inset="2" selectedItem="OAk-KA-y5i" id="ddF-fN-stL">
|
<popUpButtonCell key="cell" type="push" title="Item 1" bezelStyle="rounded" alignment="left" lineBreakMode="truncatingTail" state="on" borderStyle="borderAndBezel" imageScaling="proportionallyDown" inset="2" selectedItem="OAk-KA-y5i" id="ddF-fN-stL">
|
||||||
<behavior key="behavior" lightByBackground="YES" lightByGray="YES"/>
|
<behavior key="behavior" lightByBackground="YES" lightByGray="YES"/>
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="menu"/>
|
||||||
<menu key="menu" id="xya-TQ-koA">
|
<menu key="menu" id="xya-TQ-koA">
|
||||||
<items>
|
<items>
|
||||||
<menuItem title="Item 1" state="on" id="OAk-KA-y5i"/>
|
<menuItem title="Item 1" state="on" id="OAk-KA-y5i"/>
|
||||||
@ -60,7 +60,7 @@ Then choose the account to receive your imported subscriptions.</string>
|
|||||||
<constraint firstItem="wGx-8H-BqE" firstAttribute="top" secondItem="gSF-Ze-XcY" secondAttribute="top" id="1du-tK-vXo"/>
|
<constraint firstItem="wGx-8H-BqE" firstAttribute="top" secondItem="gSF-Ze-XcY" secondAttribute="top" id="1du-tK-vXo"/>
|
||||||
<constraint firstItem="M8B-pG-mg8" firstAttribute="leading" secondItem="976-Ry-M6G" secondAttribute="trailing" constant="8" symbolic="YES" id="HfV-5e-Qa6"/>
|
<constraint firstItem="M8B-pG-mg8" firstAttribute="leading" secondItem="976-Ry-M6G" secondAttribute="trailing" constant="8" symbolic="YES" id="HfV-5e-Qa6"/>
|
||||||
<constraint firstAttribute="width" constant="400" id="Pgs-f3-wRX"/>
|
<constraint firstAttribute="width" constant="400" id="Pgs-f3-wRX"/>
|
||||||
<constraint firstAttribute="bottom" secondItem="M8B-pG-mg8" secondAttribute="bottom" id="T2s-0I-bPf"/>
|
<constraint firstAttribute="bottom" secondItem="M8B-pG-mg8" secondAttribute="bottom" constant="2" id="T2s-0I-bPf"/>
|
||||||
<constraint firstItem="M8B-pG-mg8" firstAttribute="top" secondItem="wGx-8H-BqE" secondAttribute="bottom" constant="8" symbolic="YES" id="ZmZ-rR-1NZ"/>
|
<constraint firstItem="M8B-pG-mg8" firstAttribute="top" secondItem="wGx-8H-BqE" secondAttribute="bottom" constant="8" symbolic="YES" id="ZmZ-rR-1NZ"/>
|
||||||
<constraint firstAttribute="trailing" secondItem="wGx-8H-BqE" secondAttribute="trailing" id="aw0-OW-sHK"/>
|
<constraint firstAttribute="trailing" secondItem="wGx-8H-BqE" secondAttribute="trailing" id="aw0-OW-sHK"/>
|
||||||
<constraint firstItem="wGx-8H-BqE" firstAttribute="leading" secondItem="gSF-Ze-XcY" secondAttribute="leading" id="c4P-PJ-vd2"/>
|
<constraint firstItem="wGx-8H-BqE" firstAttribute="leading" secondItem="gSF-Ze-XcY" secondAttribute="leading" id="c4P-PJ-vd2"/>
|
||||||
|
@ -24,8 +24,8 @@ protocol SidebarDelegate: class {
|
|||||||
|
|
||||||
@objc class SidebarViewController: NSViewController, NSOutlineViewDelegate, NSMenuDelegate, UndoableCommandRunner {
|
@objc class SidebarViewController: NSViewController, NSOutlineViewDelegate, NSMenuDelegate, UndoableCommandRunner {
|
||||||
|
|
||||||
@IBOutlet var outlineView: SidebarOutlineView!
|
@IBOutlet weak var outlineView: NSOutlineView!
|
||||||
|
|
||||||
weak var delegate: SidebarDelegate?
|
weak var delegate: SidebarDelegate?
|
||||||
|
|
||||||
private let rebuildTreeAndRestoreSelectionQueue = CoalescingQueue(name: "Rebuild Tree Queue", interval: 1.0)
|
private let rebuildTreeAndRestoreSelectionQueue = CoalescingQueue(name: "Rebuild Tree Queue", interval: 1.0)
|
||||||
|
@ -12,7 +12,7 @@ struct TimelineCellAppearance: Equatable {
|
|||||||
|
|
||||||
let showIcon: Bool
|
let showIcon: Bool
|
||||||
|
|
||||||
let cellPadding = NSEdgeInsets(top: 8.0, left: 18.0, bottom: 10.0, right: 18.0)
|
let cellPadding: NSEdgeInsets
|
||||||
|
|
||||||
let feedNameFont: NSFont
|
let feedNameFont: NSFont
|
||||||
|
|
||||||
@ -55,6 +55,12 @@ struct TimelineCellAppearance: Equatable {
|
|||||||
self.textOnlyFont = NSFont.systemFont(ofSize: largeItemFontSize)
|
self.textOnlyFont = NSFont.systemFont(ofSize: largeItemFontSize)
|
||||||
|
|
||||||
self.showIcon = showIcon
|
self.showIcon = showIcon
|
||||||
|
|
||||||
|
if #available(macOS 11.0, *) {
|
||||||
|
cellPadding = NSEdgeInsets(top: 8.0, left: 4.0, bottom: 10.0, right: 4.0)
|
||||||
|
} else {
|
||||||
|
cellPadding = NSEdgeInsets(top: 8.0, left: 18.0, bottom: 10.0, right: 18.0)
|
||||||
|
}
|
||||||
|
|
||||||
let margin = self.cellPadding.left + self.unreadCircleDimension + self.unreadCircleMarginRight
|
let margin = self.cellPadding.left + self.unreadCircleDimension + self.unreadCircleMarginRight
|
||||||
self.boxLeftMargin = margin
|
self.boxLeftMargin = margin
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
// Copyright © 2015 Ranchero Software, LLC. All rights reserved.
|
// Copyright © 2015 Ranchero Software, LLC. All rights reserved.
|
||||||
//
|
//
|
||||||
|
|
||||||
import Foundation
|
import AppKit
|
||||||
import RSCore
|
import RSCore
|
||||||
|
|
||||||
class TimelineTableCellView: NSTableCellView {
|
class TimelineTableCellView: NSTableCellView {
|
||||||
@ -21,18 +21,11 @@ class TimelineTableCellView: NSTableCellView {
|
|||||||
private lazy var iconView = IconView()
|
private lazy var iconView = IconView()
|
||||||
|
|
||||||
private var starView = TimelineTableCellView.imageView(with: AppAssets.timelineStarUnselected, scaling: .scaleNone)
|
private var starView = TimelineTableCellView.imageView(with: AppAssets.timelineStarUnselected, scaling: .scaleNone)
|
||||||
private let separatorView = TimelineTableCellView.separatorView()
|
|
||||||
|
|
||||||
private lazy var textFields = {
|
private lazy var textFields = {
|
||||||
return [self.dateView, self.feedNameView, self.titleView, self.summaryView, self.textView]
|
return [self.dateView, self.feedNameView, self.titleView, self.summaryView, self.textView]
|
||||||
}()
|
}()
|
||||||
|
|
||||||
private var showsSeparator: Bool = AppDefaults.shared.timelineShowsSeparators {
|
|
||||||
didSet {
|
|
||||||
separatorView.isHidden = !showsSeparator
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var cellAppearance: TimelineCellAppearance! {
|
var cellAppearance: TimelineCellAppearance! {
|
||||||
didSet {
|
didSet {
|
||||||
if cellAppearance != oldValue {
|
if cellAppearance != oldValue {
|
||||||
@ -81,15 +74,6 @@ class TimelineTableCellView: NSTableCellView {
|
|||||||
self.init(frame: NSRect.zero)
|
self.init(frame: NSRect.zero)
|
||||||
}
|
}
|
||||||
|
|
||||||
override func prepareForReuse() {
|
|
||||||
super.prepareForReuse()
|
|
||||||
separatorView.isHidden = !showsSeparator
|
|
||||||
}
|
|
||||||
|
|
||||||
func timelineShowsSeparatorsDefaultDidChange() {
|
|
||||||
showsSeparator = AppDefaults.shared.timelineShowsSeparators
|
|
||||||
}
|
|
||||||
|
|
||||||
override func setFrameSize(_ newSize: NSSize) {
|
override func setFrameSize(_ newSize: NSSize) {
|
||||||
|
|
||||||
if newSize == self.frame.size {
|
if newSize == self.frame.size {
|
||||||
@ -123,7 +107,6 @@ class TimelineTableCellView: NSTableCellView {
|
|||||||
feedNameView.setFrame(ifNotEqualTo: layoutRects.feedNameRect)
|
feedNameView.setFrame(ifNotEqualTo: layoutRects.feedNameRect)
|
||||||
iconView.setFrame(ifNotEqualTo: layoutRects.iconImageRect)
|
iconView.setFrame(ifNotEqualTo: layoutRects.iconImageRect)
|
||||||
starView.setFrame(ifNotEqualTo: layoutRects.starRect)
|
starView.setFrame(ifNotEqualTo: layoutRects.starRect)
|
||||||
separatorView.setFrame(ifNotEqualTo: layoutRects.separatorRect)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -162,11 +145,6 @@ private extension TimelineTableCellView {
|
|||||||
return imageView
|
return imageView
|
||||||
}
|
}
|
||||||
|
|
||||||
static func separatorView() -> NSView {
|
|
||||||
|
|
||||||
return TimelineSeparatorView(frame: .zero)
|
|
||||||
}
|
|
||||||
|
|
||||||
func setFrame(for textField: NSTextField, rect: NSRect) {
|
func setFrame(for textField: NSTextField, rect: NSRect) {
|
||||||
|
|
||||||
if Int(floor(rect.height)) == 0 || Int(floor(rect.width)) == 0 {
|
if Int(floor(rect.height)) == 0 || Int(floor(rect.width)) == 0 {
|
||||||
@ -211,7 +189,6 @@ private extension TimelineTableCellView {
|
|||||||
addSubviewAtInit(feedNameView, hidden: true)
|
addSubviewAtInit(feedNameView, hidden: true)
|
||||||
addSubviewAtInit(iconView, hidden: true)
|
addSubviewAtInit(iconView, hidden: true)
|
||||||
addSubviewAtInit(starView, hidden: true)
|
addSubviewAtInit(starView, hidden: true)
|
||||||
addSubviewAtInit(separatorView, hidden: !AppDefaults.shared.timelineShowsSeparators)
|
|
||||||
|
|
||||||
makeTextFieldColorsNormal()
|
makeTextFieldColorsNormal()
|
||||||
}
|
}
|
||||||
@ -337,32 +314,3 @@ private extension TimelineTableCellView {
|
|||||||
updateIcon()
|
updateIcon()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: -
|
|
||||||
|
|
||||||
private class TimelineSeparatorView: NSView {
|
|
||||||
private static let backgroundColor = NSColor(named: "timelineSeparatorColor")!
|
|
||||||
|
|
||||||
override init(frame: NSRect) {
|
|
||||||
super.init(frame: frame)
|
|
||||||
self.wantsLayer = true
|
|
||||||
}
|
|
||||||
|
|
||||||
required init?(coder: NSCoder) {
|
|
||||||
fatalError("init(coder:) has not been implemented")
|
|
||||||
}
|
|
||||||
|
|
||||||
override func viewDidChangeEffectiveAppearance() {
|
|
||||||
super.viewDidChangeEffectiveAppearance()
|
|
||||||
needsDisplay = true
|
|
||||||
}
|
|
||||||
|
|
||||||
override var wantsUpdateLayer: Bool {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
override func updateLayer() {
|
|
||||||
super.updateLayer()
|
|
||||||
layer?.backgroundColor = TimelineSeparatorView.backgroundColor.cgColor
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -26,6 +26,7 @@ final class TimelineContainerViewController: NSViewController {
|
|||||||
|
|
||||||
@IBOutlet weak var readFilteredButton: NSButton!
|
@IBOutlet weak var readFilteredButton: NSButton!
|
||||||
@IBOutlet var containerView: TimelineContainerView!
|
@IBOutlet var containerView: TimelineContainerView!
|
||||||
|
@IBOutlet weak var sortAndFilterViewHeightConstraint: NSLayoutConstraint!
|
||||||
|
|
||||||
var currentTimelineViewController: TimelineViewController? {
|
var currentTimelineViewController: TimelineViewController? {
|
||||||
didSet {
|
didSet {
|
||||||
@ -69,6 +70,10 @@ final class TimelineContainerViewController: NSViewController {
|
|||||||
makeMenuItemTitleLarger(groupByFeedMenuItem)
|
makeMenuItemTitleLarger(groupByFeedMenuItem)
|
||||||
updateViewOptionsPopUpButton()
|
updateViewOptionsPopUpButton()
|
||||||
|
|
||||||
|
if #available(macOS 11.0, *) {
|
||||||
|
sortAndFilterViewHeightConstraint.constant = 0
|
||||||
|
}
|
||||||
|
|
||||||
NotificationCenter.default.addObserver(self, selector: #selector(userDefaultsDidChange(_:)), name: UserDefaults.didChangeNotification, object: nil)
|
NotificationCenter.default.addObserver(self, selector: #selector(userDefaultsDidChange(_:)), name: UserDefaults.didChangeNotification, object: nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -10,6 +10,8 @@ import AppKit
|
|||||||
|
|
||||||
class TimelineTableRowView : NSTableRowView {
|
class TimelineTableRowView : NSTableRowView {
|
||||||
|
|
||||||
|
private var separator: NSView?
|
||||||
|
|
||||||
override var isOpaque: Bool {
|
override var isOpaque: Bool {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
@ -23,6 +25,7 @@ class TimelineTableRowView : NSTableRowView {
|
|||||||
override var isSelected: Bool {
|
override var isSelected: Bool {
|
||||||
didSet {
|
didSet {
|
||||||
cellView?.isSelected = isSelected
|
cellView?.isSelected = isSelected
|
||||||
|
separator?.isHidden = isSelected
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -34,21 +37,6 @@ class TimelineTableRowView : NSTableRowView {
|
|||||||
super.init(coder: coder)
|
super.init(coder: coder)
|
||||||
}
|
}
|
||||||
|
|
||||||
override func drawBackground(in dirtyRect: NSRect) {
|
|
||||||
NSColor.alternatingContentBackgroundColors[0].setFill()
|
|
||||||
dirtyRect.fill()
|
|
||||||
}
|
|
||||||
|
|
||||||
override func drawSelection(in dirtyRect: NSRect) {
|
|
||||||
if isEmphasized {
|
|
||||||
NSColor.selectedContentBackgroundColor.setFill()
|
|
||||||
dirtyRect.fill()
|
|
||||||
} else {
|
|
||||||
NSColor.unemphasizedSelectedContentBackgroundColor.setFill()
|
|
||||||
dirtyRect.fill()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private var cellView: TimelineTableCellView? {
|
private var cellView: TimelineTableCellView? {
|
||||||
for oneSubview in subviews {
|
for oneSubview in subviews {
|
||||||
if let foundView = oneSubview as? TimelineTableCellView {
|
if let foundView = oneSubview as? TimelineTableCellView {
|
||||||
@ -58,4 +46,38 @@ class TimelineTableRowView : NSTableRowView {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override func viewDidMoveToSuperview() {
|
||||||
|
if #available(macOS 11.0, *) {
|
||||||
|
addSeparatorView()
|
||||||
|
} else {
|
||||||
|
if AppDefaults.shared.timelineShowsSeparators {
|
||||||
|
addSeparatorView()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private func addSeparatorView() {
|
||||||
|
guard let cellView = cellView, separator == nil else { return }
|
||||||
|
separator = NSView()
|
||||||
|
separator!.translatesAutoresizingMaskIntoConstraints = false
|
||||||
|
separator!.wantsLayer = true
|
||||||
|
separator!.layer?.backgroundColor = AppAssets.timelineSeparatorColor.cgColor
|
||||||
|
addSubview(separator!)
|
||||||
|
if #available(macOS 11.0, *) {
|
||||||
|
NSLayoutConstraint.activate([
|
||||||
|
separator!.leadingAnchor.constraint(equalTo: cellView.leadingAnchor, constant: 20),
|
||||||
|
separator!.trailingAnchor.constraint(equalTo: cellView.trailingAnchor, constant: -4),
|
||||||
|
separator!.heightAnchor.constraint(equalToConstant: 1),
|
||||||
|
separator!.bottomAnchor.constraint(equalTo: bottomAnchor, constant: 0)
|
||||||
|
])
|
||||||
|
} else {
|
||||||
|
NSLayoutConstraint.activate([
|
||||||
|
separator!.leadingAnchor.constraint(equalTo: cellView.leadingAnchor, constant: 34),
|
||||||
|
separator!.trailingAnchor.constraint(equalTo: cellView.trailingAnchor, constant: -28),
|
||||||
|
separator!.heightAnchor.constraint(equalToConstant: 1),
|
||||||
|
separator!.bottomAnchor.constraint(equalTo: bottomAnchor, constant: 0)
|
||||||
|
])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -194,7 +194,6 @@ final class TimelineViewController: NSViewController, UndoableCommandRunner, Unr
|
|||||||
convenience init(delegate: TimelineDelegate) {
|
convenience init(delegate: TimelineDelegate) {
|
||||||
self.init(nibName: "TimelineTableView", bundle: nil)
|
self.init(nibName: "TimelineTableView", bundle: nil)
|
||||||
self.delegate = delegate
|
self.delegate = delegate
|
||||||
self.startObservingUserDefaults()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
override func viewDidLoad() {
|
override func viewDidLoad() {
|
||||||
@ -208,6 +207,10 @@ final class TimelineViewController: NSViewController, UndoableCommandRunner, Unr
|
|||||||
tableView.setDraggingSourceOperationMask(.copy, forLocal: false)
|
tableView.setDraggingSourceOperationMask(.copy, forLocal: false)
|
||||||
tableView.keyboardDelegate = keyboardDelegate
|
tableView.keyboardDelegate = keyboardDelegate
|
||||||
|
|
||||||
|
if #available(macOS 11.0, *) {
|
||||||
|
tableView.style = .inset
|
||||||
|
}
|
||||||
|
|
||||||
if !didRegisterForNotifications {
|
if !didRegisterForNotifications {
|
||||||
NotificationCenter.default.addObserver(self, selector: #selector(statusesDidChange(_:)), name: .StatusesDidChange, object: nil)
|
NotificationCenter.default.addObserver(self, selector: #selector(statusesDidChange(_:)), name: .StatusesDidChange, object: nil)
|
||||||
NotificationCenter.default.addObserver(self, selector: #selector(webFeedIconDidBecomeAvailable(_:)), name: .WebFeedIconDidBecomeAvailable, object: nil)
|
NotificationCenter.default.addObserver(self, selector: #selector(webFeedIconDidBecomeAvailable(_:)), name: .WebFeedIconDidBecomeAvailable, object: nil)
|
||||||
@ -965,18 +968,6 @@ extension TimelineViewController: NSTableViewDelegate {
|
|||||||
// MARK: - Private
|
// MARK: - Private
|
||||||
|
|
||||||
private extension TimelineViewController {
|
private extension TimelineViewController {
|
||||||
|
|
||||||
func startObservingUserDefaults() {
|
|
||||||
assert(timelineShowsSeparatorsObserver == nil)
|
|
||||||
timelineShowsSeparatorsObserver = UserDefaults.standard.observe(\UserDefaults.CorreiaSeparators) { [weak self] (_, _) in
|
|
||||||
guard let self = self, self.isViewLoaded else { return }
|
|
||||||
self.tableView.enumerateAvailableRowViews { (rowView, index) in
|
|
||||||
if let cellView = rowView.view(atColumn: 0) as? TimelineTableCellView {
|
|
||||||
cellView.timelineShowsSeparatorsDefaultDidChange()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func fetchAndReplacePreservingSelection() {
|
func fetchAndReplacePreservingSelection() {
|
||||||
if let article = oneSelectedArticle, let account = article.account {
|
if let article = oneSelectedArticle, let account = article.account {
|
||||||
|
44
Mac/Preferences/Accounts/AccountCell.swift
Normal file
44
Mac/Preferences/Accounts/AccountCell.swift
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
//
|
||||||
|
// AccountCell.swift
|
||||||
|
// NetNewsWire
|
||||||
|
//
|
||||||
|
// Created by Maurice Parker on 11/19/20.
|
||||||
|
// Copyright © 2020 Ranchero Software. All rights reserved.
|
||||||
|
//
|
||||||
|
|
||||||
|
import AppKit
|
||||||
|
|
||||||
|
class AccountCell: NSTableCellView {
|
||||||
|
|
||||||
|
private var originalImage: NSImage?
|
||||||
|
|
||||||
|
var isImageTemplateCapable = true
|
||||||
|
|
||||||
|
override var backgroundStyle: NSView.BackgroundStyle {
|
||||||
|
didSet {
|
||||||
|
updateImage()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private extension AccountCell {
|
||||||
|
|
||||||
|
func updateImage() {
|
||||||
|
guard isImageTemplateCapable else { return }
|
||||||
|
|
||||||
|
if backgroundStyle != .normal {
|
||||||
|
guard !(imageView?.image?.isTemplate ?? false) else { return }
|
||||||
|
|
||||||
|
originalImage = imageView?.image
|
||||||
|
|
||||||
|
let templateImage = imageView?.image?.copy() as? NSImage
|
||||||
|
templateImage?.isTemplate = true
|
||||||
|
imageView?.image = templateImage
|
||||||
|
} else {
|
||||||
|
guard let originalImage = originalImage else { return }
|
||||||
|
imageView?.image = originalImage
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -1,7 +1,8 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<document type="com.apple.InterfaceBuilder3.Cocoa.XIB" version="3.0" toolsVersion="16096" targetRuntime="MacOSX.Cocoa" propertyAccessControl="none" useAutolayout="YES" customObjectInstantitationMethod="direct">
|
<document type="com.apple.InterfaceBuilder3.Cocoa.XIB" version="3.0" toolsVersion="17506" targetRuntime="MacOSX.Cocoa" propertyAccessControl="none" useAutolayout="YES" customObjectInstantitationMethod="direct">
|
||||||
<dependencies>
|
<dependencies>
|
||||||
<plugIn identifier="com.apple.InterfaceBuilder.CocoaPlugin" version="16096"/>
|
<deployment identifier="macosx"/>
|
||||||
|
<plugIn identifier="com.apple.InterfaceBuilder.CocoaPlugin" version="17506"/>
|
||||||
<capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/>
|
<capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
<objects>
|
<objects>
|
||||||
@ -27,7 +28,7 @@
|
|||||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||||
<subviews>
|
<subviews>
|
||||||
<gridView xPlacement="fill" yPlacement="center" rowAlignment="none" rowSpacing="9" translatesAutoresizingMaskIntoConstraints="NO" id="nVy-H3-bFO">
|
<gridView xPlacement="fill" yPlacement="center" rowAlignment="none" rowSpacing="9" translatesAutoresizingMaskIntoConstraints="NO" id="nVy-H3-bFO">
|
||||||
<rect key="frame" x="20" y="108" width="286" height="126"/>
|
<rect key="frame" x="20" y="90" width="286" height="144"/>
|
||||||
<rows>
|
<rows>
|
||||||
<gridRow id="yLs-SL-a1b"/>
|
<gridRow id="yLs-SL-a1b"/>
|
||||||
<gridRow yPlacement="top" id="etw-2m-nWZ"/>
|
<gridRow yPlacement="top" id="etw-2m-nWZ"/>
|
||||||
@ -41,7 +42,7 @@
|
|||||||
<gridCells>
|
<gridCells>
|
||||||
<gridCell row="yLs-SL-a1b" column="sMM-Ds-SKX" id="3ea-DE-T3i">
|
<gridCell row="yLs-SL-a1b" column="sMM-Ds-SKX" id="3ea-DE-T3i">
|
||||||
<textField key="contentView" horizontalHuggingPriority="251" verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="jiQ-KJ-SS0">
|
<textField key="contentView" horizontalHuggingPriority="251" verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="jiQ-KJ-SS0">
|
||||||
<rect key="frame" x="-2" y="110" width="44" height="16"/>
|
<rect key="frame" x="-2" y="128" width="44" height="16"/>
|
||||||
<textFieldCell key="cell" lineBreakMode="clipping" alignment="right" title="Type:" id="tC5-Vt-gBc">
|
<textFieldCell key="cell" lineBreakMode="clipping" alignment="right" title="Type:" id="tC5-Vt-gBc">
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
<color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/>
|
<color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/>
|
||||||
@ -51,7 +52,7 @@
|
|||||||
</gridCell>
|
</gridCell>
|
||||||
<gridCell row="yLs-SL-a1b" column="Fhf-h9-g0O" id="baI-Kp-tKF">
|
<gridCell row="yLs-SL-a1b" column="Fhf-h9-g0O" id="baI-Kp-tKF">
|
||||||
<textField key="contentView" horizontalHuggingPriority="251" verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="XYX-iz-hnq">
|
<textField key="contentView" horizontalHuggingPriority="251" verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="XYX-iz-hnq">
|
||||||
<rect key="frame" x="44" y="110" width="73" height="16"/>
|
<rect key="frame" x="44" y="128" width="73" height="16"/>
|
||||||
<textFieldCell key="cell" lineBreakMode="clipping" title="On My Mac" id="6yI-bV-1Sh">
|
<textFieldCell key="cell" lineBreakMode="clipping" title="On My Mac" id="6yI-bV-1Sh">
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
<color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/>
|
<color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/>
|
||||||
@ -62,7 +63,7 @@
|
|||||||
<gridCell row="etw-2m-nWZ" column="sMM-Ds-SKX" id="htf-Ca-Hpv"/>
|
<gridCell row="etw-2m-nWZ" column="sMM-Ds-SKX" id="htf-Ca-Hpv"/>
|
||||||
<gridCell row="etw-2m-nWZ" column="Fhf-h9-g0O" id="NrD-vV-1Y1">
|
<gridCell row="etw-2m-nWZ" column="Fhf-h9-g0O" id="NrD-vV-1Y1">
|
||||||
<button key="contentView" verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="mgt-uY-fuq">
|
<button key="contentView" verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="mgt-uY-fuq">
|
||||||
<rect key="frame" x="44" y="85" width="60" height="18"/>
|
<rect key="frame" x="44" y="102" width="64" height="18"/>
|
||||||
<buttonCell key="cell" type="check" title="Active" bezelStyle="regularSquare" imagePosition="left" state="on" inset="2" id="wxB-dX-nGt">
|
<buttonCell key="cell" type="check" title="Active" bezelStyle="regularSquare" imagePosition="left" state="on" inset="2" id="wxB-dX-nGt">
|
||||||
<behavior key="behavior" changeContents="YES" doesNotDimImage="YES" lightByContents="YES"/>
|
<behavior key="behavior" changeContents="YES" doesNotDimImage="YES" lightByContents="YES"/>
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
@ -74,7 +75,7 @@
|
|||||||
</gridCell>
|
</gridCell>
|
||||||
<gridCell row="3IT-3r-gEK" column="sMM-Ds-SKX" id="2yP-oZ-A6S">
|
<gridCell row="3IT-3r-gEK" column="sMM-Ds-SKX" id="2yP-oZ-A6S">
|
||||||
<textField key="contentView" horizontalHuggingPriority="251" verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="Ted-jN-oYR">
|
<textField key="contentView" horizontalHuggingPriority="251" verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="Ted-jN-oYR">
|
||||||
<rect key="frame" x="-2" y="60" width="44" height="16"/>
|
<rect key="frame" x="-2" y="76" width="44" height="16"/>
|
||||||
<textFieldCell key="cell" lineBreakMode="clipping" alignment="right" title="Name:" id="uyQ-Zi-QCr">
|
<textFieldCell key="cell" lineBreakMode="clipping" alignment="right" title="Name:" id="uyQ-Zi-QCr">
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
<color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/>
|
<color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/>
|
||||||
@ -84,7 +85,7 @@
|
|||||||
</gridCell>
|
</gridCell>
|
||||||
<gridCell row="3IT-3r-gEK" column="Fhf-h9-g0O" id="nCq-02-YVv">
|
<gridCell row="3IT-3r-gEK" column="Fhf-h9-g0O" id="nCq-02-YVv">
|
||||||
<textField key="contentView" verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="TT0-Kf-YTC">
|
<textField key="contentView" verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="TT0-Kf-YTC">
|
||||||
<rect key="frame" x="46" y="57" width="100" height="21"/>
|
<rect key="frame" x="46" y="73" width="100" height="21"/>
|
||||||
<textFieldCell key="cell" scrollable="YES" lineBreakMode="clipping" selectable="YES" editable="YES" sendsActionOnEndEditing="YES" borderStyle="bezel" drawsBackground="YES" id="7Vp-Hq-j6n">
|
<textFieldCell key="cell" scrollable="YES" lineBreakMode="clipping" selectable="YES" editable="YES" sendsActionOnEndEditing="YES" borderStyle="bezel" drawsBackground="YES" id="7Vp-Hq-j6n">
|
||||||
<font key="font" usesAppearanceFont="YES"/>
|
<font key="font" usesAppearanceFont="YES"/>
|
||||||
<color key="textColor" name="controlTextColor" catalog="System" colorSpace="catalog"/>
|
<color key="textColor" name="controlTextColor" catalog="System" colorSpace="catalog"/>
|
||||||
@ -95,9 +96,11 @@
|
|||||||
<gridCell row="Y4C-5M-ySp" column="sMM-Ds-SKX" id="dON-E7-yd2"/>
|
<gridCell row="Y4C-5M-ySp" column="sMM-Ds-SKX" id="dON-E7-yd2"/>
|
||||||
<gridCell row="Y4C-5M-ySp" column="Fhf-h9-g0O" id="i7Y-4k-5TF">
|
<gridCell row="Y4C-5M-ySp" column="Fhf-h9-g0O" id="i7Y-4k-5TF">
|
||||||
<textField key="contentView" verticalHuggingPriority="750" horizontalCompressionResistancePriority="250" translatesAutoresizingMaskIntoConstraints="NO" id="xp5-wk-PKc">
|
<textField key="contentView" verticalHuggingPriority="750" horizontalCompressionResistancePriority="250" translatesAutoresizingMaskIntoConstraints="NO" id="xp5-wk-PKc">
|
||||||
<rect key="frame" x="44" y="0.0" width="244" height="48"/>
|
<rect key="frame" x="44" y="0.0" width="244" height="64"/>
|
||||||
<textFieldCell key="cell" selectable="YES" title="The name appears in the sidebar. It can be anything you want. You can even use emoji. 🎸" id="MW0-mH-Gaa">
|
<textFieldCell key="cell" selectable="YES" id="MW0-mH-Gaa">
|
||||||
<font key="font" usesAppearanceFont="YES"/>
|
<font key="font" usesAppearanceFont="YES"/>
|
||||||
|
<string key="title">The name appears in the sidebar. It can be anything you want. You can even use emoji. 🎸
|
||||||
|
</string>
|
||||||
<color key="textColor" name="secondaryLabelColor" catalog="System" colorSpace="catalog"/>
|
<color key="textColor" name="secondaryLabelColor" catalog="System" colorSpace="catalog"/>
|
||||||
<color key="backgroundColor" name="textBackgroundColor" catalog="System" colorSpace="catalog"/>
|
<color key="backgroundColor" name="textBackgroundColor" catalog="System" colorSpace="catalog"/>
|
||||||
</textFieldCell>
|
</textFieldCell>
|
||||||
@ -106,7 +109,7 @@
|
|||||||
</gridCells>
|
</gridCells>
|
||||||
</gridView>
|
</gridView>
|
||||||
<button verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="gLh-gl-ZGQ">
|
<button verticalHuggingPriority="750" translatesAutoresizingMaskIntoConstraints="NO" id="gLh-gl-ZGQ">
|
||||||
<rect key="frame" x="109" y="60" width="109" height="32"/>
|
<rect key="frame" x="112" y="55" width="103" height="32"/>
|
||||||
<buttonCell key="cell" type="push" title="Credentials" bezelStyle="rounded" alignment="center" borderStyle="border" imageScaling="proportionallyDown" inset="2" id="vYg-ZC-o4W">
|
<buttonCell key="cell" type="push" title="Credentials" bezelStyle="rounded" alignment="center" borderStyle="border" imageScaling="proportionallyDown" inset="2" id="vYg-ZC-o4W">
|
||||||
<behavior key="behavior" pushIn="YES" lightByBackground="YES" lightByGray="YES"/>
|
<behavior key="behavior" pushIn="YES" lightByBackground="YES" lightByGray="YES"/>
|
||||||
<font key="font" metaFont="system"/>
|
<font key="font" metaFont="system"/>
|
||||||
@ -119,7 +122,7 @@
|
|||||||
<constraints>
|
<constraints>
|
||||||
<constraint firstItem="nVy-H3-bFO" firstAttribute="leading" secondItem="ft2-Mb-5LD" secondAttribute="leading" constant="20" symbolic="YES" id="SQe-pg-1hl"/>
|
<constraint firstItem="nVy-H3-bFO" firstAttribute="leading" secondItem="ft2-Mb-5LD" secondAttribute="leading" constant="20" symbolic="YES" id="SQe-pg-1hl"/>
|
||||||
<constraint firstAttribute="trailing" secondItem="nVy-H3-bFO" secondAttribute="trailing" constant="20" symbolic="YES" id="Wsq-ar-poP"/>
|
<constraint firstAttribute="trailing" secondItem="nVy-H3-bFO" secondAttribute="trailing" constant="20" symbolic="YES" id="Wsq-ar-poP"/>
|
||||||
<constraint firstItem="gLh-gl-ZGQ" firstAttribute="top" secondItem="nVy-H3-bFO" secondAttribute="bottom" constant="20" symbolic="YES" id="a0S-2S-3dR"/>
|
<constraint firstItem="gLh-gl-ZGQ" firstAttribute="top" secondItem="nVy-H3-bFO" secondAttribute="bottom" constant="8" id="a0S-2S-3dR"/>
|
||||||
<constraint firstItem="gLh-gl-ZGQ" firstAttribute="centerX" secondItem="ft2-Mb-5LD" secondAttribute="centerX" id="cW8-YT-BEn"/>
|
<constraint firstItem="gLh-gl-ZGQ" firstAttribute="centerX" secondItem="ft2-Mb-5LD" secondAttribute="centerX" id="cW8-YT-BEn"/>
|
||||||
<constraint firstItem="nVy-H3-bFO" firstAttribute="top" secondItem="ft2-Mb-5LD" secondAttribute="top" constant="20" symbolic="YES" id="sy2-s4-iEW"/>
|
<constraint firstItem="nVy-H3-bFO" firstAttribute="top" secondItem="ft2-Mb-5LD" secondAttribute="top" constant="20" symbolic="YES" id="sy2-s4-iEW"/>
|
||||||
</constraints>
|
</constraints>
|
||||||
|
@ -109,13 +109,17 @@ extension AccountsPreferencesViewController: NSTableViewDataSource {
|
|||||||
|
|
||||||
extension AccountsPreferencesViewController: NSTableViewDelegate {
|
extension AccountsPreferencesViewController: NSTableViewDelegate {
|
||||||
|
|
||||||
private static let cellIdentifier = NSUserInterfaceItemIdentifier(rawValue: "AccountCell")
|
|
||||||
|
|
||||||
func tableView(_ tableView: NSTableView, viewFor tableColumn: NSTableColumn?, row: Int) -> NSView? {
|
func tableView(_ tableView: NSTableView, viewFor tableColumn: NSTableColumn?, row: Int) -> NSView? {
|
||||||
if let cell = tableView.makeView(withIdentifier: NSUserInterfaceItemIdentifier(rawValue: "Cell"), owner: nil) as? NSTableCellView {
|
if let cell = tableView.makeView(withIdentifier: NSUserInterfaceItemIdentifier(rawValue: "Cell"), owner: nil) as? AccountCell {
|
||||||
|
|
||||||
let account = sortedAccounts[row]
|
let account = sortedAccounts[row]
|
||||||
cell.textField?.stringValue = account.nameForDisplay
|
cell.textField?.stringValue = account.nameForDisplay
|
||||||
cell.imageView?.image = account.smallIcon?.image
|
cell.imageView?.image = account.smallIcon?.image
|
||||||
|
|
||||||
|
if account.type == .feedbin {
|
||||||
|
cell.isImageTemplateCapable = false
|
||||||
|
}
|
||||||
|
|
||||||
return cell
|
return cell
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
|
@ -14,33 +14,24 @@ struct AddAccountHelpView: View {
|
|||||||
let accountTypes: [AccountType] = AddAccountSections.allOrdered.sectionContent
|
let accountTypes: [AccountType] = AddAccountSections.allOrdered.sectionContent
|
||||||
var delegate: AccountsPreferencesAddAccountDelegate?
|
var delegate: AccountsPreferencesAddAccountDelegate?
|
||||||
var helpText: String
|
var helpText: String
|
||||||
@State private var hoveringId: String? = nil
|
|
||||||
@State private var iCloudUnavailableError: Bool = false
|
@State private var iCloudUnavailableError: Bool = false
|
||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
VStack {
|
VStack {
|
||||||
HStack {
|
HStack {
|
||||||
ForEach(accountTypes, id: \.self) { account in
|
ForEach(accountTypes, id: \.self) { account in
|
||||||
account.image()
|
Button(action: {
|
||||||
.resizable()
|
if account == .cloudKit && AccountManager.shared.accounts.contains(where: { $0.type == .cloudKit }) {
|
||||||
.frame(width: 20, height: 20, alignment: .center)
|
iCloudUnavailableError = true
|
||||||
.onTapGesture {
|
} else {
|
||||||
if account == .cloudKit && AccountManager.shared.accounts.contains(where: { $0.type == .cloudKit }) {
|
delegate?.presentSheetForAccount(account)
|
||||||
iCloudUnavailableError = true
|
|
||||||
} else {
|
|
||||||
delegate?.presentSheetForAccount(account)
|
|
||||||
}
|
|
||||||
hoveringId = nil
|
|
||||||
}
|
}
|
||||||
.onHover(perform: { hovering in
|
}, label: {
|
||||||
if hovering {
|
account.image()
|
||||||
hoveringId = account.localizedAccountName()
|
.resizable()
|
||||||
} else {
|
.frame(width: 20, height: 20, alignment: .center)
|
||||||
hoveringId = nil
|
})
|
||||||
}
|
.buttonStyle(PlainButtonStyle())
|
||||||
})
|
|
||||||
.scaleEffect(hoveringId == account.localizedAccountName() ? 1.2 : 1)
|
|
||||||
.shadow(radius: hoveringId == account.localizedAccountName() ? 0.8 : 0)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -20,28 +20,18 @@ struct EnableExtensionPointHelpView: View {
|
|||||||
var helpText: String
|
var helpText: String
|
||||||
weak var preferencesController: ExtensionPointPreferencesViewController?
|
weak var preferencesController: ExtensionPointPreferencesViewController?
|
||||||
|
|
||||||
@State private var hoveringId: String?
|
|
||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
VStack {
|
VStack {
|
||||||
HStack {
|
HStack {
|
||||||
ForEach(0..<extensionPoints.count, content: { i in
|
ForEach(0..<extensionPoints.count, content: { i in
|
||||||
Image(nsImage: extensionPoints[i].image)
|
Button(action: {
|
||||||
.resizable()
|
preferencesController?.enableExtensionPointFromSelection(extensionPoints[i])
|
||||||
.frame(width: 20, height: 20, alignment: .center)
|
}, label: {
|
||||||
.onTapGesture {
|
Image(nsImage: extensionPoints[i].image)
|
||||||
preferencesController?.enableExtensionPointFromSelection(extensionPoints[i])
|
.resizable()
|
||||||
hoveringId = nil
|
.frame(width: 20, height: 20, alignment: .center)
|
||||||
}
|
})
|
||||||
.onHover(perform: { hovering in
|
.buttonStyle(PlainButtonStyle())
|
||||||
if hovering {
|
|
||||||
hoveringId = extensionPoints[i].title
|
|
||||||
} else {
|
|
||||||
hoveringId = nil
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.scaleEffect(hoveringId == extensionPoints[i].title ? 1.2 : 1)
|
|
||||||
.shadow(radius: hoveringId == extensionPoints[i].title ? 0.8 : 0)
|
|
||||||
})
|
})
|
||||||
|
|
||||||
if ExtensionPointManager.shared.availableExtensionPointTypes.count == 0 {
|
if ExtensionPointManager.shared.availableExtensionPointTypes.count == 0 {
|
||||||
|
@ -25,7 +25,7 @@ struct EnableExtensionPointView: View {
|
|||||||
|
|
||||||
var body: some View {
|
var body: some View {
|
||||||
VStack(alignment: .leading, spacing: 8) {
|
VStack(alignment: .leading, spacing: 8) {
|
||||||
Text("Choose an extension point to add...")
|
Text("Choose an extension to add...")
|
||||||
.font(.headline)
|
.font(.headline)
|
||||||
.padding()
|
.padding()
|
||||||
|
|
||||||
|
@ -1,21 +1,16 @@
|
|||||||
{
|
{
|
||||||
"info" : {
|
|
||||||
"version" : 1,
|
|
||||||
"author" : "xcode"
|
|
||||||
},
|
|
||||||
"colors" : [
|
"colors" : [
|
||||||
{
|
{
|
||||||
"idiom" : "universal",
|
|
||||||
"color" : {
|
"color" : {
|
||||||
"color-space" : "gray-gamma-22",
|
"color-space" : "gray-gamma-22",
|
||||||
"components" : {
|
"components" : {
|
||||||
"white" : "0.900",
|
"alpha" : "1.000",
|
||||||
"alpha" : "1.000"
|
"white" : "0.900"
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
|
"idiom" : "universal"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"idiom" : "universal",
|
|
||||||
"appearances" : [
|
"appearances" : [
|
||||||
{
|
{
|
||||||
"appearance" : "luminosity",
|
"appearance" : "luminosity",
|
||||||
@ -24,8 +19,13 @@
|
|||||||
],
|
],
|
||||||
"color" : {
|
"color" : {
|
||||||
"platform" : "osx",
|
"platform" : "osx",
|
||||||
"reference" : "gridColor"
|
"reference" : "quaternaryLabelColor"
|
||||||
}
|
},
|
||||||
|
"idiom" : "universal"
|
||||||
}
|
}
|
||||||
]
|
],
|
||||||
}
|
"info" : {
|
||||||
|
"author" : "xcode",
|
||||||
|
"version" : 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -2,6 +2,10 @@
|
|||||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
<plist version="1.0">
|
<plist version="1.0">
|
||||||
<dict>
|
<dict>
|
||||||
|
<key>AppGroup</key>
|
||||||
|
<string>group.$(ORGANIZATION_IDENTIFIER).NetNewsWire</string>
|
||||||
|
<key>AppIdentifierPrefix</key>
|
||||||
|
<string>$(AppIdentifierPrefix)</string>
|
||||||
<key>CFBundleDevelopmentRegion</key>
|
<key>CFBundleDevelopmentRegion</key>
|
||||||
<string>en</string>
|
<string>en</string>
|
||||||
<key>CFBundleExecutable</key>
|
<key>CFBundleExecutable</key>
|
||||||
@ -34,6 +38,10 @@
|
|||||||
</array>
|
</array>
|
||||||
<key>CFBundleVersion</key>
|
<key>CFBundleVersion</key>
|
||||||
<string>$(CURRENT_PROJECT_VERSION)</string>
|
<string>$(CURRENT_PROJECT_VERSION)</string>
|
||||||
|
<key>DeveloperEntitlements</key>
|
||||||
|
<string>$(DEVELOPER_ENTITLEMENTS)</string>
|
||||||
|
<key>FeedURLForTestBuilds</key>
|
||||||
|
<string>https://ranchero.com/downloads/netnewswire-beta.xml</string>
|
||||||
<key>LSApplicationCategoryType</key>
|
<key>LSApplicationCategoryType</key>
|
||||||
<string>public.app-category.news</string>
|
<string>public.app-category.news</string>
|
||||||
<key>LSMinimumSystemVersion</key>
|
<key>LSMinimumSystemVersion</key>
|
||||||
@ -43,10 +51,6 @@
|
|||||||
<key>NSAllowsArbitraryLoads</key>
|
<key>NSAllowsArbitraryLoads</key>
|
||||||
<true/>
|
<true/>
|
||||||
</dict>
|
</dict>
|
||||||
<key>NSUserActivityTypes</key>
|
|
||||||
<array>
|
|
||||||
<string>ReadArticle</string>
|
|
||||||
</array>
|
|
||||||
<key>NSAppleEventsUsageDescription</key>
|
<key>NSAppleEventsUsageDescription</key>
|
||||||
<string>NetNewsWire communicates with other apps on your Mac when you choose to share an article.</string>
|
<string>NetNewsWire communicates with other apps on your Mac when you choose to share an article.</string>
|
||||||
<key>NSAppleScriptEnabled</key>
|
<key>NSAppleScriptEnabled</key>
|
||||||
@ -57,21 +61,17 @@
|
|||||||
<string>Main</string>
|
<string>Main</string>
|
||||||
<key>NSPrincipalClass</key>
|
<key>NSPrincipalClass</key>
|
||||||
<string>NSApplication</string>
|
<string>NSApplication</string>
|
||||||
|
<key>NSUserActivityTypes</key>
|
||||||
|
<array>
|
||||||
|
<string>ReadArticle</string>
|
||||||
|
</array>
|
||||||
<key>OSAScriptingDefinition</key>
|
<key>OSAScriptingDefinition</key>
|
||||||
<string>NetNewsWire.sdef</string>
|
<string>NetNewsWire.sdef</string>
|
||||||
<key>SUFeedURL</key>
|
|
||||||
<string>https://ranchero.com/downloads/netnewswire-release.xml</string>
|
|
||||||
<key>FeedURLForTestBuilds</key>
|
|
||||||
<string>https://ranchero.com/downloads/netnewswire-beta.xml</string>
|
|
||||||
<key>UserAgent</key>
|
|
||||||
<string>NetNewsWire (RSS Reader; https://ranchero.com/netnewswire/)</string>
|
|
||||||
<key>OrganizationIdentifier</key>
|
<key>OrganizationIdentifier</key>
|
||||||
<string>$(ORGANIZATION_IDENTIFIER)</string>
|
<string>$(ORGANIZATION_IDENTIFIER)</string>
|
||||||
<key>AppGroup</key>
|
<key>SUFeedURL</key>
|
||||||
<string>group.$(ORGANIZATION_IDENTIFIER).NetNewsWire</string>
|
<string>https://ranchero.com/downloads/netnewswire-release.xml</string>
|
||||||
<key>AppIdentifierPrefix</key>
|
<key>UserAgent</key>
|
||||||
<string>$(AppIdentifierPrefix)</string>
|
<string>NetNewsWire (RSS Reader; https://ranchero.com/netnewswire/)</string>
|
||||||
<key>DeveloperEntitlements</key>
|
|
||||||
<string>$(DEVELOPER_ENTITLEMENTS)</string>
|
|
||||||
</dict>
|
</dict>
|
||||||
</plist>
|
</plist>
|
||||||
|
@ -0,0 +1,62 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>BuildMachineOSBuild</key>
|
||||||
|
<string>20C5048k</string>
|
||||||
|
<key>CFBundleDevelopmentRegion</key>
|
||||||
|
<string>en</string>
|
||||||
|
<key>CFBundleExecutable</key>
|
||||||
|
<string>org.sparkle-project.Downloader</string>
|
||||||
|
<key>CFBundleIdentifier</key>
|
||||||
|
<string>org.sparkle-project.Downloader</string>
|
||||||
|
<key>CFBundleInfoDictionaryVersion</key>
|
||||||
|
<string>6.0</string>
|
||||||
|
<key>CFBundleName</key>
|
||||||
|
<string>org.sparkle-project.Downloader</string>
|
||||||
|
<key>CFBundlePackageType</key>
|
||||||
|
<string>XPC!</string>
|
||||||
|
<key>CFBundleShortVersionString</key>
|
||||||
|
<string>2.0.0</string>
|
||||||
|
<key>CFBundleSignature</key>
|
||||||
|
<string>????</string>
|
||||||
|
<key>CFBundleSupportedPlatforms</key>
|
||||||
|
<array>
|
||||||
|
<string>MacOSX</string>
|
||||||
|
</array>
|
||||||
|
<key>CFBundleVersion</key>
|
||||||
|
<string>2.0.0</string>
|
||||||
|
<key>DTCompiler</key>
|
||||||
|
<string>com.apple.compilers.llvm.clang.1_0</string>
|
||||||
|
<key>DTPlatformBuild</key>
|
||||||
|
<string>12B45b</string>
|
||||||
|
<key>DTPlatformName</key>
|
||||||
|
<string>macosx</string>
|
||||||
|
<key>DTPlatformVersion</key>
|
||||||
|
<string>11.0</string>
|
||||||
|
<key>DTSDKBuild</key>
|
||||||
|
<string>20A2408</string>
|
||||||
|
<key>DTSDKName</key>
|
||||||
|
<string>macosx11.0</string>
|
||||||
|
<key>DTXcode</key>
|
||||||
|
<string>1220</string>
|
||||||
|
<key>DTXcodeBuild</key>
|
||||||
|
<string>12B45b</string>
|
||||||
|
<key>LSMinimumSystemVersion</key>
|
||||||
|
<string>10.9</string>
|
||||||
|
<key>NSAppTransportSecurity</key>
|
||||||
|
<dict>
|
||||||
|
<key>NSAllowsArbitraryLoads</key>
|
||||||
|
<false/>
|
||||||
|
</dict>
|
||||||
|
<key>NSHumanReadableCopyright</key>
|
||||||
|
<string>Copyright © 2016 Sparkle Project. All rights reserved.</string>
|
||||||
|
<key>XPCService</key>
|
||||||
|
<dict>
|
||||||
|
<key>RunLoopType</key>
|
||||||
|
<string>NSRunLoop</string>
|
||||||
|
<key>ServiceType</key>
|
||||||
|
<string>Application</string>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
Binary file not shown.
@ -0,0 +1,115 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>files</key>
|
||||||
|
<dict/>
|
||||||
|
<key>files2</key>
|
||||||
|
<dict/>
|
||||||
|
<key>rules</key>
|
||||||
|
<dict>
|
||||||
|
<key>^Resources/</key>
|
||||||
|
<true/>
|
||||||
|
<key>^Resources/.*\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1000</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/.*\.lproj/locversion.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1100</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/Base\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1010</real>
|
||||||
|
</dict>
|
||||||
|
<key>^version.plist$</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>rules2</key>
|
||||||
|
<dict>
|
||||||
|
<key>.*\.dSYM($|/)</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>11</real>
|
||||||
|
</dict>
|
||||||
|
<key>^(.*/)?\.DS_Store$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>2000</real>
|
||||||
|
</dict>
|
||||||
|
<key>^(Frameworks|SharedFrameworks|PlugIns|Plug-ins|XPCServices|Helpers|MacOS|Library/(Automator|Spotlight|LoginItems))/</key>
|
||||||
|
<dict>
|
||||||
|
<key>nested</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>10</real>
|
||||||
|
</dict>
|
||||||
|
<key>^.*</key>
|
||||||
|
<true/>
|
||||||
|
<key>^Info\.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^PkgInfo$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/.*\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1000</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/.*\.lproj/locversion.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1100</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/Base\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1010</real>
|
||||||
|
</dict>
|
||||||
|
<key>^[^/]+$</key>
|
||||||
|
<dict>
|
||||||
|
<key>nested</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>10</real>
|
||||||
|
</dict>
|
||||||
|
<key>^embedded\.provisionprofile$</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^version\.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
@ -0,0 +1,55 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>BuildMachineOSBuild</key>
|
||||||
|
<string>20C5048k</string>
|
||||||
|
<key>CFBundleDevelopmentRegion</key>
|
||||||
|
<string>en</string>
|
||||||
|
<key>CFBundleExecutable</key>
|
||||||
|
<string>org.sparkle-project.InstallerConnection</string>
|
||||||
|
<key>CFBundleIdentifier</key>
|
||||||
|
<string>org.sparkle-project.InstallerConnection</string>
|
||||||
|
<key>CFBundleInfoDictionaryVersion</key>
|
||||||
|
<string>6.0</string>
|
||||||
|
<key>CFBundleName</key>
|
||||||
|
<string>org.sparkle-project.InstallerConnection</string>
|
||||||
|
<key>CFBundlePackageType</key>
|
||||||
|
<string>XPC!</string>
|
||||||
|
<key>CFBundleShortVersionString</key>
|
||||||
|
<string>2.0.0</string>
|
||||||
|
<key>CFBundleSignature</key>
|
||||||
|
<string>????</string>
|
||||||
|
<key>CFBundleSupportedPlatforms</key>
|
||||||
|
<array>
|
||||||
|
<string>MacOSX</string>
|
||||||
|
</array>
|
||||||
|
<key>CFBundleVersion</key>
|
||||||
|
<string>2.0.0</string>
|
||||||
|
<key>DTCompiler</key>
|
||||||
|
<string>com.apple.compilers.llvm.clang.1_0</string>
|
||||||
|
<key>DTPlatformBuild</key>
|
||||||
|
<string>12B45b</string>
|
||||||
|
<key>DTPlatformName</key>
|
||||||
|
<string>macosx</string>
|
||||||
|
<key>DTPlatformVersion</key>
|
||||||
|
<string>11.0</string>
|
||||||
|
<key>DTSDKBuild</key>
|
||||||
|
<string>20A2408</string>
|
||||||
|
<key>DTSDKName</key>
|
||||||
|
<string>macosx11.0</string>
|
||||||
|
<key>DTXcode</key>
|
||||||
|
<string>1220</string>
|
||||||
|
<key>DTXcodeBuild</key>
|
||||||
|
<string>12B45b</string>
|
||||||
|
<key>LSMinimumSystemVersion</key>
|
||||||
|
<string>10.9</string>
|
||||||
|
<key>NSHumanReadableCopyright</key>
|
||||||
|
<string>Copyright © 2016 Sparkle Project. All rights reserved.</string>
|
||||||
|
<key>XPCService</key>
|
||||||
|
<dict>
|
||||||
|
<key>ServiceType</key>
|
||||||
|
<string>Application</string>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
Binary file not shown.
@ -0,0 +1,115 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>files</key>
|
||||||
|
<dict/>
|
||||||
|
<key>files2</key>
|
||||||
|
<dict/>
|
||||||
|
<key>rules</key>
|
||||||
|
<dict>
|
||||||
|
<key>^Resources/</key>
|
||||||
|
<true/>
|
||||||
|
<key>^Resources/.*\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1000</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/.*\.lproj/locversion.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1100</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/Base\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1010</real>
|
||||||
|
</dict>
|
||||||
|
<key>^version.plist$</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>rules2</key>
|
||||||
|
<dict>
|
||||||
|
<key>.*\.dSYM($|/)</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>11</real>
|
||||||
|
</dict>
|
||||||
|
<key>^(.*/)?\.DS_Store$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>2000</real>
|
||||||
|
</dict>
|
||||||
|
<key>^(Frameworks|SharedFrameworks|PlugIns|Plug-ins|XPCServices|Helpers|MacOS|Library/(Automator|Spotlight|LoginItems))/</key>
|
||||||
|
<dict>
|
||||||
|
<key>nested</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>10</real>
|
||||||
|
</dict>
|
||||||
|
<key>^.*</key>
|
||||||
|
<true/>
|
||||||
|
<key>^Info\.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^PkgInfo$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/.*\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1000</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/.*\.lproj/locversion.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1100</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/Base\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1010</real>
|
||||||
|
</dict>
|
||||||
|
<key>^[^/]+$</key>
|
||||||
|
<dict>
|
||||||
|
<key>nested</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>10</real>
|
||||||
|
</dict>
|
||||||
|
<key>^embedded\.provisionprofile$</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^version\.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
@ -0,0 +1,57 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>BuildMachineOSBuild</key>
|
||||||
|
<string>20C5048k</string>
|
||||||
|
<key>CFBundleDevelopmentRegion</key>
|
||||||
|
<string>en</string>
|
||||||
|
<key>CFBundleExecutable</key>
|
||||||
|
<string>org.sparkle-project.InstallerLauncher</string>
|
||||||
|
<key>CFBundleIdentifier</key>
|
||||||
|
<string>org.sparkle-project.InstallerLauncher</string>
|
||||||
|
<key>CFBundleInfoDictionaryVersion</key>
|
||||||
|
<string>6.0</string>
|
||||||
|
<key>CFBundleName</key>
|
||||||
|
<string>org.sparkle-project.InstallerLauncher</string>
|
||||||
|
<key>CFBundlePackageType</key>
|
||||||
|
<string>XPC!</string>
|
||||||
|
<key>CFBundleShortVersionString</key>
|
||||||
|
<string>2.0.0</string>
|
||||||
|
<key>CFBundleSignature</key>
|
||||||
|
<string>????</string>
|
||||||
|
<key>CFBundleSupportedPlatforms</key>
|
||||||
|
<array>
|
||||||
|
<string>MacOSX</string>
|
||||||
|
</array>
|
||||||
|
<key>CFBundleVersion</key>
|
||||||
|
<string>2.0.0</string>
|
||||||
|
<key>DTCompiler</key>
|
||||||
|
<string>com.apple.compilers.llvm.clang.1_0</string>
|
||||||
|
<key>DTPlatformBuild</key>
|
||||||
|
<string>12B45b</string>
|
||||||
|
<key>DTPlatformName</key>
|
||||||
|
<string>macosx</string>
|
||||||
|
<key>DTPlatformVersion</key>
|
||||||
|
<string>11.0</string>
|
||||||
|
<key>DTSDKBuild</key>
|
||||||
|
<string>20A2408</string>
|
||||||
|
<key>DTSDKName</key>
|
||||||
|
<string>macosx11.0</string>
|
||||||
|
<key>DTXcode</key>
|
||||||
|
<string>1220</string>
|
||||||
|
<key>DTXcodeBuild</key>
|
||||||
|
<string>12B45b</string>
|
||||||
|
<key>LSMinimumSystemVersion</key>
|
||||||
|
<string>10.9</string>
|
||||||
|
<key>NSHumanReadableCopyright</key>
|
||||||
|
<string>Copyright © 2016 Sparkle Project. All rights reserved.</string>
|
||||||
|
<key>XPCService</key>
|
||||||
|
<dict>
|
||||||
|
<key>JoinExistingSession</key>
|
||||||
|
<true/>
|
||||||
|
<key>ServiceType</key>
|
||||||
|
<string>Application</string>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
Binary file not shown.
@ -0,0 +1,52 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>BuildMachineOSBuild</key>
|
||||||
|
<string>20C5048k</string>
|
||||||
|
<key>CFBundleDevelopmentRegion</key>
|
||||||
|
<string>en</string>
|
||||||
|
<key>CFBundleExecutable</key>
|
||||||
|
<string>Updater</string>
|
||||||
|
<key>CFBundleIdentifier</key>
|
||||||
|
<string>org.sparkle-project.Sparkle.Updater</string>
|
||||||
|
<key>CFBundleInfoDictionaryVersion</key>
|
||||||
|
<string>6.0</string>
|
||||||
|
<key>CFBundleName</key>
|
||||||
|
<string>Updater</string>
|
||||||
|
<key>CFBundlePackageType</key>
|
||||||
|
<string>APPL</string>
|
||||||
|
<key>CFBundleShortVersionString</key>
|
||||||
|
<string>2.0.0</string>
|
||||||
|
<key>CFBundleSignature</key>
|
||||||
|
<string>????</string>
|
||||||
|
<key>CFBundleSupportedPlatforms</key>
|
||||||
|
<array>
|
||||||
|
<string>MacOSX</string>
|
||||||
|
</array>
|
||||||
|
<key>CFBundleVersion</key>
|
||||||
|
<string>2.0.0</string>
|
||||||
|
<key>DTCompiler</key>
|
||||||
|
<string>com.apple.compilers.llvm.clang.1_0</string>
|
||||||
|
<key>DTPlatformBuild</key>
|
||||||
|
<string>12B45b</string>
|
||||||
|
<key>DTPlatformName</key>
|
||||||
|
<string>macosx</string>
|
||||||
|
<key>DTPlatformVersion</key>
|
||||||
|
<string>11.0</string>
|
||||||
|
<key>DTSDKBuild</key>
|
||||||
|
<string>20A2408</string>
|
||||||
|
<key>DTSDKName</key>
|
||||||
|
<string>macosx11.0</string>
|
||||||
|
<key>DTXcode</key>
|
||||||
|
<string>1220</string>
|
||||||
|
<key>DTXcodeBuild</key>
|
||||||
|
<string>12B45b</string>
|
||||||
|
<key>LSMinimumSystemVersion</key>
|
||||||
|
<string>10.9</string>
|
||||||
|
<key>LSUIElement</key>
|
||||||
|
<string>1</string>
|
||||||
|
<key>NSPrincipalClass</key>
|
||||||
|
<string>NSApplication</string>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
Binary file not shown.
@ -0,0 +1 @@
|
|||||||
|
APPL????
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -0,0 +1 @@
|
|||||||
|
fr.lproj
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -0,0 +1 @@
|
|||||||
|
pt_BR.lproj
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -0,0 +1,802 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>files</key>
|
||||||
|
<dict>
|
||||||
|
<key>Resources/SUStatus.nib</key>
|
||||||
|
<data>
|
||||||
|
ECVWRExfxyDt5uvKRD+70wc9J6s=
|
||||||
|
</data>
|
||||||
|
<key>Resources/ar.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
9n6+2ab5/d3baNlcFRfSpztHdKc=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/ca.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
K1BEF6sG2vXMLgibwfo3j2h588E=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/cs.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
qmZIcgaZTr//z9PjOI776B5GQ3E=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/da.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
88FAIY52ex+k6CHvZHUHiYpaSdQ=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/de.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
FnTKeC2WOm3Wo79G5tYK17ssA4g=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/el.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
gQTKA4Zd4FpsXRLWTcEfqV3Czu0=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/en.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
HMDIP8J6ekyxwFQ6/Gn+q3WSTl4=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/es.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
saEdp9H51NgvY5tzYYY5QoM5dsg=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/fi.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
Xfk3iYvY4+ymcoVUpHQATY5FNLg=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/fr.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
N3afKcO8erR7VUa2Cq4bwqxw/DY=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/he.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
ONZyQ7mMihp025wvYCm+YH5p9t8=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/is.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
bKE7f6KUVWbXzh+cBrwa31j6sXU=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/it.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
PGQtWau2xbYKJPKZjSvkwnPSSJU=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/ja.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
iD89mxaGjEzXuqTCpr1SbfWzdyM=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/ko.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
36Fahhtf/RNpPA22ntiODYGqG30=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/nb.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
lxEVDkftYdIz5tpFIlCBRzjq1G8=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/nl.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
3esiRzch9B/dcmSDuZOlhGRmvhI=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/pl.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
5DAYxRDmzfZJHVzkzmq9B33cV+Q=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/pt_BR.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
9OEsTkc4OnLubR99mP0Br13Mflo=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/pt_PT.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
DXgfdoW9r94wdvH+tYnJNakKzDs=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/ro.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
Yk1UW9SBQyAtNbFvLmiIjW/UCcc=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/ru.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
Px2O36VmsQbjS8ywxoJ/Pp+xQiQ=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/sk.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
8A/scZSblfhf9/SAyz5Di2EqrqM=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/sl.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
YRXBwzauFczYTqobmqCxBBPR4DE=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/sv.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
K+ak+cmJ5S1D27ODU3IntD0wITI=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/th.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
anxUgZs0IJsgMZlzI1HUeCjvmrc=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/tr.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
4L5cXvWM1KkQdn5c+uYML/PX6xg=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/uk.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
uhJ3st+FckuLz8HIH0r/RtUVGsw=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/zh_CN.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
kFXz9LiX6VmEsvEWZcZOIMmUE5o=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/zh_TW.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
fq2MGchNCsDkfRX6i950z9hnHAM=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
<key>files2</key>
|
||||||
|
<dict>
|
||||||
|
<key>Resources/SUStatus.nib</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
ECVWRExfxyDt5uvKRD+70wc9J6s=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
AtY9YmPv7cUlbFWP2vCyVdi3/M+XQn98wOlrIES2Dgk=
|
||||||
|
</data>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/ar.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
9n6+2ab5/d3baNlcFRfSpztHdKc=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
kEBNsn9OraKT0YF/n5ZaJC14Y/+GW/HI/CjiahPHgwM=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/ca.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
K1BEF6sG2vXMLgibwfo3j2h588E=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
D01nO0KWUvaVR/PR0E95dLAlJCYEKPRh858t+lcxFto=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/cs.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
qmZIcgaZTr//z9PjOI776B5GQ3E=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
6sIHusRLkghCkCVemdyAqniiTfJ68E6t0qswH/A+Aac=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/da.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
88FAIY52ex+k6CHvZHUHiYpaSdQ=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
YtLfD1azWIUD2eqATgQak+tKys3x9ZFjo91mSYwSY68=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/de.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
FnTKeC2WOm3Wo79G5tYK17ssA4g=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
zG5B5gvBrmrL31eAFv8JQ0xYZrAGgvpcePzhSL9lRSI=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/el.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
gQTKA4Zd4FpsXRLWTcEfqV3Czu0=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
DpBU2fltmtw85+0U85gXwPH8qApgI0zbG6K0qIn2X0c=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/en.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
HMDIP8J6ekyxwFQ6/Gn+q3WSTl4=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
T0siv9/ri/ulfofXL+GzB1ClarT02vlzl4QRomTIy9A=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/es.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
saEdp9H51NgvY5tzYYY5QoM5dsg=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
Rv71G/XkSv/4JZd+ejfFkpu4HKXFsM0Nxe094rw3mAQ=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/fi.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
Xfk3iYvY4+ymcoVUpHQATY5FNLg=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
DwdjkY2nc5XvSzY7wbwHcwKnnCfJXwDl1bO6PbtoeUU=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/fr.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
N3afKcO8erR7VUa2Cq4bwqxw/DY=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
nGZJLdRUiRSWfcROzRsVZzoM/Pyl+C6y0c7WJdZ++ME=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/fr_CA.lproj</key>
|
||||||
|
<dict>
|
||||||
|
<key>symlink</key>
|
||||||
|
<string>fr.lproj</string>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/he.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
ONZyQ7mMihp025wvYCm+YH5p9t8=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
35ECtsAW7lQQpZTAtYBIKgel5ItYO6FvWJaSueWWqVU=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/is.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
bKE7f6KUVWbXzh+cBrwa31j6sXU=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
Dh4VgRSkntzRdCDvUFT0O91wxRUTyfKmsonwoD8JO3s=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/it.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
PGQtWau2xbYKJPKZjSvkwnPSSJU=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
6KWPm6/BMUnxP7kax40a/akTj6RVSNWSgXpS2+5bkMg=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/ja.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
iD89mxaGjEzXuqTCpr1SbfWzdyM=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
P8h6uv3ksdrzPVBgsLywrDU+NA6c3at5YNW9MyQ5+i0=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/ko.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
36Fahhtf/RNpPA22ntiODYGqG30=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
oX2Hsbm8fF05oGgMFXazS+rqg3KswApukPT1inQKxs8=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/nb.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
lxEVDkftYdIz5tpFIlCBRzjq1G8=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
j1Ga6bYhYJ7h65dfZiX0udIIngNspVWPJaqKaEZhdIY=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/nl.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
3esiRzch9B/dcmSDuZOlhGRmvhI=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
Ft3lAx+eG7MsySkCRtYN7wT7zRTPWDsJDJnghgcNWrA=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/pl.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
5DAYxRDmzfZJHVzkzmq9B33cV+Q=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
tv/j3ywfuO1E3J5/vmrVFQ3cbZPi3EudMtacnjqVqWA=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/pt.lproj</key>
|
||||||
|
<dict>
|
||||||
|
<key>symlink</key>
|
||||||
|
<string>pt_BR.lproj</string>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/pt_BR.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
9OEsTkc4OnLubR99mP0Br13Mflo=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
p12hYL8AHpuT+aXzheKTHwZEQFpPfc/qCoaYe7NmP6I=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/pt_PT.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
DXgfdoW9r94wdvH+tYnJNakKzDs=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
xjNkmadedPLED0QHUgWiGXlJ/d0rZeHWkUmAyGdURyA=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/ro.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
Yk1UW9SBQyAtNbFvLmiIjW/UCcc=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
IffqR5gxQdL9YEeJj/L9jauu1eduqT1taxe3hKDDXOk=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/ru.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
Px2O36VmsQbjS8ywxoJ/Pp+xQiQ=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
MBWSZcnNsYWJkCrv3YDWyANbEghjnWl8TFrApZqIh8c=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/sk.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
8A/scZSblfhf9/SAyz5Di2EqrqM=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
hKJVJbokW6LXrUqrf3FyGAxdnXJe+NAM1IzwtfMpPTs=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/sl.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
YRXBwzauFczYTqobmqCxBBPR4DE=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
mO9OxrL9L5y2wDXWsMt11pjcxa4wJrXVXM26w/TWqpE=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/sv.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
K+ak+cmJ5S1D27ODU3IntD0wITI=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
OXVaG3Vrb1xKlSXHj2qnMe/+X3r5r+huDymhPpx7j5w=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/th.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
anxUgZs0IJsgMZlzI1HUeCjvmrc=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
uFBTQa44/YKNE5qHbmLqdlZUuLF0Zfk0LepBeIQ7ZQ8=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/tr.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
4L5cXvWM1KkQdn5c+uYML/PX6xg=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
rOuDu7og0MYRXCQMAZ48ge5FRTN4+ZBYl9DxJEDnDaY=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/uk.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
uhJ3st+FckuLz8HIH0r/RtUVGsw=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
AdON9wb2iTlde8P8StWkzdTMy8iL7M6mj94hIj6ixA0=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/zh_CN.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
kFXz9LiX6VmEsvEWZcZOIMmUE5o=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
oT/+oPtd/EjVyWINXmlilXd0HUk9MdcNrJQsHA5Mfys=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>Resources/zh_TW.lproj/Sparkle.strings</key>
|
||||||
|
<dict>
|
||||||
|
<key>hash</key>
|
||||||
|
<data>
|
||||||
|
fq2MGchNCsDkfRX6i950z9hnHAM=
|
||||||
|
</data>
|
||||||
|
<key>hash2</key>
|
||||||
|
<data>
|
||||||
|
4bQfH6cx4JPlejfZbFtgdDFbRS9FENa0UFlKJqZqhtg=
|
||||||
|
</data>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
<key>rules</key>
|
||||||
|
<dict>
|
||||||
|
<key>^Resources/</key>
|
||||||
|
<true/>
|
||||||
|
<key>^Resources/.*\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1000</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/.*\.lproj/locversion.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1100</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/Base\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1010</real>
|
||||||
|
</dict>
|
||||||
|
<key>^version.plist$</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>rules2</key>
|
||||||
|
<dict>
|
||||||
|
<key>.*\.dSYM($|/)</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>11</real>
|
||||||
|
</dict>
|
||||||
|
<key>^(.*/)?\.DS_Store$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>2000</real>
|
||||||
|
</dict>
|
||||||
|
<key>^(Frameworks|SharedFrameworks|PlugIns|Plug-ins|XPCServices|Helpers|MacOS|Library/(Automator|Spotlight|LoginItems))/</key>
|
||||||
|
<dict>
|
||||||
|
<key>nested</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>10</real>
|
||||||
|
</dict>
|
||||||
|
<key>^.*</key>
|
||||||
|
<true/>
|
||||||
|
<key>^Info\.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^PkgInfo$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/.*\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1000</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/.*\.lproj/locversion.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1100</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/Base\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1010</real>
|
||||||
|
</dict>
|
||||||
|
<key>^[^/]+$</key>
|
||||||
|
<dict>
|
||||||
|
<key>nested</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>10</real>
|
||||||
|
</dict>
|
||||||
|
<key>^embedded\.provisionprofile$</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^version\.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
Binary file not shown.
@ -0,0 +1,134 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>files</key>
|
||||||
|
<dict/>
|
||||||
|
<key>files2</key>
|
||||||
|
<dict>
|
||||||
|
<key>MacOS/Autoupdate</key>
|
||||||
|
<dict>
|
||||||
|
<key>cdhash</key>
|
||||||
|
<data>
|
||||||
|
4u6dcWJ/FUGEoMzv+dGop57DXtw=
|
||||||
|
</data>
|
||||||
|
<key>requirement</key>
|
||||||
|
<string>cdhash H"658a70b9d2d389c086433126ff259ee263247b29" or cdhash H"e2ee9d71627f154184a0cceff9d1a8a79ec35edc" or cdhash H"86f6cbd23c573f1aab41c027d28c8c2ba4ea0124" or cdhash H"a666ab9b660affc2ddf89855f05983e54e78a422"</string>
|
||||||
|
</dict>
|
||||||
|
<key>MacOS/Updater.app</key>
|
||||||
|
<dict>
|
||||||
|
<key>cdhash</key>
|
||||||
|
<data>
|
||||||
|
Vd1bV7RYKTFw7rrVvq4vORweSNw=
|
||||||
|
</data>
|
||||||
|
<key>requirement</key>
|
||||||
|
<string>cdhash H"55c5f5c21d7a1b677908ef0951c008f170413490" or cdhash H"55dd5b57b458293170eebad5beae2f391c1e48dc" or cdhash H"934f6e08232316efc5e602b37471ff2825f20d1a" or cdhash H"cff78d762d3bf40e44394124a874ec43129f0879"</string>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
<key>rules</key>
|
||||||
|
<dict>
|
||||||
|
<key>^Resources/</key>
|
||||||
|
<true/>
|
||||||
|
<key>^Resources/.*\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1000</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/.*\.lproj/locversion.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1100</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/Base\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1010</real>
|
||||||
|
</dict>
|
||||||
|
<key>^version.plist$</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>rules2</key>
|
||||||
|
<dict>
|
||||||
|
<key>.*\.dSYM($|/)</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>11</real>
|
||||||
|
</dict>
|
||||||
|
<key>^(.*/)?\.DS_Store$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>2000</real>
|
||||||
|
</dict>
|
||||||
|
<key>^(Frameworks|SharedFrameworks|PlugIns|Plug-ins|XPCServices|Helpers|MacOS|Library/(Automator|Spotlight|LoginItems))/</key>
|
||||||
|
<dict>
|
||||||
|
<key>nested</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>10</real>
|
||||||
|
</dict>
|
||||||
|
<key>^.*</key>
|
||||||
|
<true/>
|
||||||
|
<key>^Info\.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^PkgInfo$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/.*\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1000</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/.*\.lproj/locversion.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1100</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/Base\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1010</real>
|
||||||
|
</dict>
|
||||||
|
<key>^[^/]+$</key>
|
||||||
|
<dict>
|
||||||
|
<key>nested</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>10</real>
|
||||||
|
</dict>
|
||||||
|
<key>^embedded\.provisionprofile$</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^version\.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
@ -0,0 +1,55 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>BuildMachineOSBuild</key>
|
||||||
|
<string>20C5048k</string>
|
||||||
|
<key>CFBundleDevelopmentRegion</key>
|
||||||
|
<string>en</string>
|
||||||
|
<key>CFBundleExecutable</key>
|
||||||
|
<string>org.sparkle-project.InstallerStatus</string>
|
||||||
|
<key>CFBundleIdentifier</key>
|
||||||
|
<string>org.sparkle-project.InstallerStatus</string>
|
||||||
|
<key>CFBundleInfoDictionaryVersion</key>
|
||||||
|
<string>6.0</string>
|
||||||
|
<key>CFBundleName</key>
|
||||||
|
<string>org.sparkle-project.InstallerStatus</string>
|
||||||
|
<key>CFBundlePackageType</key>
|
||||||
|
<string>XPC!</string>
|
||||||
|
<key>CFBundleShortVersionString</key>
|
||||||
|
<string>2.0.0</string>
|
||||||
|
<key>CFBundleSignature</key>
|
||||||
|
<string>????</string>
|
||||||
|
<key>CFBundleSupportedPlatforms</key>
|
||||||
|
<array>
|
||||||
|
<string>MacOSX</string>
|
||||||
|
</array>
|
||||||
|
<key>CFBundleVersion</key>
|
||||||
|
<string>2.0.0</string>
|
||||||
|
<key>DTCompiler</key>
|
||||||
|
<string>com.apple.compilers.llvm.clang.1_0</string>
|
||||||
|
<key>DTPlatformBuild</key>
|
||||||
|
<string>12B45b</string>
|
||||||
|
<key>DTPlatformName</key>
|
||||||
|
<string>macosx</string>
|
||||||
|
<key>DTPlatformVersion</key>
|
||||||
|
<string>11.0</string>
|
||||||
|
<key>DTSDKBuild</key>
|
||||||
|
<string>20A2408</string>
|
||||||
|
<key>DTSDKName</key>
|
||||||
|
<string>macosx11.0</string>
|
||||||
|
<key>DTXcode</key>
|
||||||
|
<string>1220</string>
|
||||||
|
<key>DTXcodeBuild</key>
|
||||||
|
<string>12B45b</string>
|
||||||
|
<key>LSMinimumSystemVersion</key>
|
||||||
|
<string>10.9</string>
|
||||||
|
<key>NSHumanReadableCopyright</key>
|
||||||
|
<string>Copyright © 2016 Sparkle Project. All rights reserved.</string>
|
||||||
|
<key>XPCService</key>
|
||||||
|
<dict>
|
||||||
|
<key>ServiceType</key>
|
||||||
|
<string>Application</string>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
Binary file not shown.
@ -0,0 +1,115 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
|
<plist version="1.0">
|
||||||
|
<dict>
|
||||||
|
<key>files</key>
|
||||||
|
<dict/>
|
||||||
|
<key>files2</key>
|
||||||
|
<dict/>
|
||||||
|
<key>rules</key>
|
||||||
|
<dict>
|
||||||
|
<key>^Resources/</key>
|
||||||
|
<true/>
|
||||||
|
<key>^Resources/.*\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1000</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/.*\.lproj/locversion.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1100</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/Base\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1010</real>
|
||||||
|
</dict>
|
||||||
|
<key>^version.plist$</key>
|
||||||
|
<true/>
|
||||||
|
</dict>
|
||||||
|
<key>rules2</key>
|
||||||
|
<dict>
|
||||||
|
<key>.*\.dSYM($|/)</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>11</real>
|
||||||
|
</dict>
|
||||||
|
<key>^(.*/)?\.DS_Store$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>2000</real>
|
||||||
|
</dict>
|
||||||
|
<key>^(Frameworks|SharedFrameworks|PlugIns|Plug-ins|XPCServices|Helpers|MacOS|Library/(Automator|Spotlight|LoginItems))/</key>
|
||||||
|
<dict>
|
||||||
|
<key>nested</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>10</real>
|
||||||
|
</dict>
|
||||||
|
<key>^.*</key>
|
||||||
|
<true/>
|
||||||
|
<key>^Info\.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^PkgInfo$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/.*\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>optional</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1000</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/.*\.lproj/locversion.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>omit</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1100</real>
|
||||||
|
</dict>
|
||||||
|
<key>^Resources/Base\.lproj/</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>1010</real>
|
||||||
|
</dict>
|
||||||
|
<key>^[^/]+$</key>
|
||||||
|
<dict>
|
||||||
|
<key>nested</key>
|
||||||
|
<true/>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>10</real>
|
||||||
|
</dict>
|
||||||
|
<key>^embedded\.provisionprofile$</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
<key>^version\.plist$</key>
|
||||||
|
<dict>
|
||||||
|
<key>weight</key>
|
||||||
|
<real>20</real>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</dict>
|
||||||
|
</plist>
|
@ -2,13 +2,13 @@
|
|||||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||||
<plist version="1.0">
|
<plist version="1.0">
|
||||||
<dict>
|
<dict>
|
||||||
<key>com.apple.security.app-sandbox</key>
|
<key>com.apple.security.app-sandbox</key>
|
||||||
<true/>
|
<true/>
|
||||||
<key>com.apple.security.files.user-selected.read-only</key>
|
<key>com.apple.security.files.user-selected.read-only</key>
|
||||||
<true/>
|
<true/>
|
||||||
<key>com.apple.security.application-groups</key>
|
<key>com.apple.security.application-groups</key>
|
||||||
<array>
|
<array>
|
||||||
<string>group.$(ORGANIZATION_IDENTIFIER).NetNewsWire</string>
|
<string>group.$(ORGANIZATION_IDENTIFIER).NetNewsWire-Evergreen</string>
|
||||||
</array>
|
</array>
|
||||||
</dict>
|
</dict>
|
||||||
</plist>
|
</plist>
|
||||||
|
@ -179,15 +179,6 @@ class AppDelegate: UIResponder, UIApplicationDelegate, UNUserNotificationCenterD
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func logMessage(_ message: String, type: LogItem.ItemType) {
|
|
||||||
print("logMessage: \(message) - \(type)")
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func logDebugMessage(_ message: String) {
|
|
||||||
logMessage(message, type: .debug)
|
|
||||||
}
|
|
||||||
|
|
||||||
func userNotificationCenter(_ center: UNUserNotificationCenter, willPresent notification: UNNotification, withCompletionHandler completionHandler: @escaping (UNNotificationPresentationOptions) -> Void) {
|
func userNotificationCenter(_ center: UNUserNotificationCenter, willPresent notification: UNNotification, withCompletionHandler completionHandler: @escaping (UNNotificationPresentationOptions) -> Void) {
|
||||||
completionHandler([.banner, .badge, .sound])
|
completionHandler([.banner, .badge, .sound])
|
||||||
}
|
}
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -51,8 +51,8 @@
|
|||||||
"repositoryURL": "https://github.com/Ranchero-Software/RSCore.git",
|
"repositoryURL": "https://github.com/Ranchero-Software/RSCore.git",
|
||||||
"state": {
|
"state": {
|
||||||
"branch": null,
|
"branch": null,
|
||||||
"revision": "1f72115989c05ca1e79fe694f25a17b9b731d0df",
|
"revision": "1017fe09c61bd9f75aa713381894aef979c994ef",
|
||||||
"version": "1.0.0-beta3"
|
"version": "1.0.0-beta6"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -91,6 +91,15 @@
|
|||||||
"version": "1.0.0-beta8"
|
"version": "1.0.0-beta8"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"package": "RSSparkle",
|
||||||
|
"repositoryURL": "https://github.com/Ranchero-Software/Sparkle-Binary.git",
|
||||||
|
"state": {
|
||||||
|
"branch": "main",
|
||||||
|
"revision": "67cd26321bdf4e77954cf6de7d9e6a20544f2030",
|
||||||
|
"version": null
|
||||||
|
}
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"package": "Swifter",
|
"package": "Swifter",
|
||||||
"repositoryURL": "https://github.com/httpswift/swifter.git",
|
"repositoryURL": "https://github.com/httpswift/swifter.git",
|
||||||
|
@ -30,8 +30,6 @@ You can build and test NetNewsWire without a paid developer account.
|
|||||||
|
|
||||||
```bash
|
```bash
|
||||||
git clone https://github.com/Ranchero-Software/NetNewsWire.git
|
git clone https://github.com/Ranchero-Software/NetNewsWire.git
|
||||||
cd NetNewsWire
|
|
||||||
git submodule update --init --recursive
|
|
||||||
```
|
```
|
||||||
|
|
||||||
You can locally override the Xcode settings for code signing
|
You can locally override the Xcode settings for code signing
|
||||||
|
@ -152,7 +152,19 @@ private extension ArticleRenderer {
|
|||||||
|
|
||||||
let title = titleOrTitleLink()
|
let title = titleOrTitleLink()
|
||||||
d["title"] = title
|
d["title"] = title
|
||||||
|
|
||||||
|
if let externalLink = article.externalURL, externalLink != article.preferredLink {
|
||||||
|
var displayLink = externalLink.strippingHTTPOrHTTPSScheme
|
||||||
|
if displayLink.count > 27 {
|
||||||
|
displayLink = displayLink.prefix(27).appending("...")
|
||||||
|
}
|
||||||
|
let regarding = NSLocalizedString("Link", comment: "Link")
|
||||||
|
let externalLinkString = "\(regarding): <a href=\"\(externalLink)\">\(displayLink)</a>"
|
||||||
|
d["external_link"] = externalLinkString
|
||||||
|
} else {
|
||||||
|
d["external_link"] = ""
|
||||||
|
}
|
||||||
|
|
||||||
d["body"] = body
|
d["body"] = body
|
||||||
|
|
||||||
var components = URLComponents()
|
var components = URLComponents()
|
||||||
|
@ -18,10 +18,27 @@ function stripStylesFromElement(element, propertiesToStrip) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Strip inline styles that could harm readability.
|
||||||
function stripStyles() {
|
function stripStyles() {
|
||||||
document.getElementsByTagName("body")[0].querySelectorAll("style, link[rel=stylesheet]").forEach(element => element.remove());
|
document.getElementsByTagName("body")[0].querySelectorAll("style, link[rel=stylesheet]").forEach(element => element.remove());
|
||||||
// Removing "background" and "font" will also remove properties that would be reflected in them, e.g., "background-color" and "font-family"
|
// Removing "background" and "font" will also remove properties that would be reflected in them, e.g., "background-color" and "font-family"
|
||||||
document.getElementsByTagName("body")[0].querySelectorAll("[style]").forEach(element => stripStylesFromElement(element, ["color", "background", "font", "max-width", "max-height"]));
|
document.getElementsByTagName("body")[0].querySelectorAll("[style]").forEach(element => stripStylesFromElement(element, ["color", "background", "font", "max-width", "max-height", "position"]));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Constrain the height of iframes whose heights are defined relative to the document body to be at most
|
||||||
|
// 50% of the viewport width.
|
||||||
|
function constrainBodyRelativeIframes() {
|
||||||
|
let iframes = document.getElementsByTagName("iframe");
|
||||||
|
|
||||||
|
for (iframe of iframes) {
|
||||||
|
if (iframe.offsetParent === document.body) {
|
||||||
|
let heightAttribute = iframe.style.height;
|
||||||
|
|
||||||
|
if (/%|vw|vh$/i.test(heightAttribute)) {
|
||||||
|
iframe.classList.add("nnw-constrained");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Convert all Feedbin proxy images to be used as src, otherwise change image locations to be absolute if not already
|
// Convert all Feedbin proxy images to be used as src, otherwise change image locations to be absolute if not already
|
||||||
@ -29,7 +46,7 @@ function convertImgSrc() {
|
|||||||
document.querySelectorAll("img").forEach(element => {
|
document.querySelectorAll("img").forEach(element => {
|
||||||
if (element.hasAttribute("data-canonical-src")) {
|
if (element.hasAttribute("data-canonical-src")) {
|
||||||
element.src = element.getAttribute("data-canonical-src")
|
element.src = element.getAttribute("data-canonical-src")
|
||||||
} else if (!element.src.match(/^[a-z]+\:\/\//i)) {
|
} else if (!/^[a-z]+\:\/\//i.test(element.src)) {
|
||||||
element.src = new URL(element.src, document.baseURI).href;
|
element.src = new URL(element.src, document.baseURI).href;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -136,6 +153,7 @@ function processPage() {
|
|||||||
wrapTables();
|
wrapTables();
|
||||||
inlineVideos();
|
inlineVideos();
|
||||||
stripStyles();
|
stripStyles();
|
||||||
|
constrainBodyRelativeIframes();
|
||||||
convertImgSrc();
|
convertImgSrc();
|
||||||
flattenPreElements();
|
flattenPreElements();
|
||||||
styleLocalFootnotes();
|
styleLocalFootnotes();
|
||||||
|
@ -98,7 +98,7 @@ body > .systemMessage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.articleDateline {
|
.articleDateline {
|
||||||
margin-bottom: 25px;
|
margin-bottom: 5px;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -107,7 +107,7 @@ body > .systemMessage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.articleDatelineTitle {
|
.articleDatelineTitle {
|
||||||
margin-bottom: 25px;
|
margin-bottom: 5px;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -115,9 +115,16 @@ body > .systemMessage {
|
|||||||
color: var(--article-title-color);
|
color: var(--article-title-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.externalLink {
|
||||||
|
margin-bottom: 5px;
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
|
|
||||||
.articleBody {
|
.articleBody {
|
||||||
|
margin-top: 20px;
|
||||||
line-height: 1.6em;
|
line-height: 1.6em;
|
||||||
}
|
}
|
||||||
|
|
||||||
h1 {
|
h1 {
|
||||||
line-height: 1.15em;
|
line-height: 1.15em;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
@ -197,6 +204,10 @@ iframe {
|
|||||||
margin: 0 auto;
|
margin: 0 auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
iframe.nnw-constrained {
|
||||||
|
max-height: 50vw;
|
||||||
|
}
|
||||||
|
|
||||||
figure {
|
figure {
|
||||||
margin-bottom: 1em;
|
margin-bottom: 1em;
|
||||||
margin-top: 1em;
|
margin-top: 1em;
|
||||||
|
@ -10,5 +10,6 @@
|
|||||||
<article>
|
<article>
|
||||||
<div class="articleTitle"><h1>[[title]]</h1></div>
|
<div class="articleTitle"><h1>[[title]]</h1></div>
|
||||||
<div class="[[dateline_style]]">[[date_medium]]</div>
|
<div class="[[dateline_style]]">[[date_medium]]</div>
|
||||||
|
<div class="externalLink">[[external_link]]</div>
|
||||||
<div class="articleBody">[[body]]</div>
|
<div class="articleBody">[[body]]</div>
|
||||||
</article>
|
</article>
|
||||||
|
32
Shared/Widget/WidgetData.swift
Normal file
32
Shared/Widget/WidgetData.swift
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
//
|
||||||
|
// WidgetData.swift
|
||||||
|
// NetNewsWire
|
||||||
|
//
|
||||||
|
// Created by Stuart Breckenridge on 18/11/20.
|
||||||
|
// Copyright © 2020 Ranchero Software. All rights reserved.
|
||||||
|
//
|
||||||
|
|
||||||
|
import Foundation
|
||||||
|
|
||||||
|
struct WidgetData: Codable {
|
||||||
|
|
||||||
|
let currentUnreadCount: Int
|
||||||
|
let currentTodayCount: Int
|
||||||
|
let currentStarredCount: Int
|
||||||
|
let unreadArticles: [LatestArticle]
|
||||||
|
let starredArticles: [LatestArticle]
|
||||||
|
let todayArticles: [LatestArticle]
|
||||||
|
let lastUpdateTime: Date
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
struct LatestArticle: Codable, Identifiable {
|
||||||
|
|
||||||
|
var id: String
|
||||||
|
let feedTitle: String
|
||||||
|
let articleTitle: String?
|
||||||
|
let articleSummary: String?
|
||||||
|
let feedIcon: Data? // Base64 encoded image data
|
||||||
|
let pubDate: String
|
||||||
|
|
||||||
|
}
|
36
Shared/Widget/WidgetDataDecoder.swift
Normal file
36
Shared/Widget/WidgetDataDecoder.swift
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
//
|
||||||
|
// WidgetDataDecoder.swift
|
||||||
|
// NetNewsWire
|
||||||
|
//
|
||||||
|
// Created by Stuart Breckenridge on 18/11/20.
|
||||||
|
// Copyright © 2020 Ranchero Software. All rights reserved.
|
||||||
|
//
|
||||||
|
|
||||||
|
import Foundation
|
||||||
|
|
||||||
|
struct WidgetDataDecoder {
|
||||||
|
|
||||||
|
static func decodeWidgetData() throws -> WidgetData {
|
||||||
|
let appGroup = Bundle.main.object(forInfoDictionaryKey: "AppGroup") as! String
|
||||||
|
let containerURL = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: appGroup)
|
||||||
|
let dataURL = containerURL?.appendingPathComponent("widget-data.json")
|
||||||
|
if FileManager.default.fileExists(atPath: dataURL!.path) {
|
||||||
|
let decodedWidgetData = try JSONDecoder().decode(WidgetData.self, from: Data(contentsOf: dataURL!))
|
||||||
|
return decodedWidgetData
|
||||||
|
} else {
|
||||||
|
return WidgetData(currentUnreadCount: 0, currentTodayCount: 0, currentStarredCount: 0, unreadArticles: [], starredArticles: [], todayArticles: [], lastUpdateTime: Date())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static func sampleData() -> WidgetData {
|
||||||
|
let pathToSample = Bundle.main.url(forResource: "widget-sample", withExtension: "json")
|
||||||
|
do {
|
||||||
|
let data = try Data(contentsOf: pathToSample!)
|
||||||
|
let decoded = try JSONDecoder().decode(WidgetData.self, from: data)
|
||||||
|
return decoded
|
||||||
|
} catch {
|
||||||
|
return WidgetData(currentUnreadCount: 0, currentTodayCount: 0, currentStarredCount: 0, unreadArticles: [], starredArticles: [], todayArticles: [], lastUpdateTime: Date())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
120
Shared/Widget/WidgetDataEncoder.swift
Normal file
120
Shared/Widget/WidgetDataEncoder.swift
Normal file
@ -0,0 +1,120 @@
|
|||||||
|
//
|
||||||
|
// WidgetDataEncoder.swift
|
||||||
|
// NetNewsWire
|
||||||
|
//
|
||||||
|
// Created by Stuart Breckenridge on 18/11/20.
|
||||||
|
// Copyright © 2020 Ranchero Software. All rights reserved.
|
||||||
|
//
|
||||||
|
|
||||||
|
import Foundation
|
||||||
|
import WidgetKit
|
||||||
|
import os.log
|
||||||
|
import UIKit
|
||||||
|
import RSCore
|
||||||
|
import Articles
|
||||||
|
|
||||||
|
|
||||||
|
public final class WidgetDataEncoder {
|
||||||
|
|
||||||
|
private let log = OSLog(subsystem: Bundle.main.bundleIdentifier!, category: "Application")
|
||||||
|
|
||||||
|
private var backgroundTaskID: UIBackgroundTaskIdentifier!
|
||||||
|
private lazy var appGroup = Bundle.main.object(forInfoDictionaryKey: "AppGroup") as! String
|
||||||
|
private lazy var containerURL = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: appGroup)
|
||||||
|
private lazy var dataURL = containerURL?.appendingPathComponent("widget-data.json")
|
||||||
|
|
||||||
|
static let shared = WidgetDataEncoder()
|
||||||
|
private init () {}
|
||||||
|
|
||||||
|
@available(iOS 14, *)
|
||||||
|
func encodeWidgetData() throws {
|
||||||
|
os_log(.debug, log: log, "Starting encoding widget data.")
|
||||||
|
|
||||||
|
do {
|
||||||
|
let unreadArticles = Array(try SmartFeedsController.shared.unreadFeed.fetchArticles()).sortedByDate(.orderedDescending)
|
||||||
|
|
||||||
|
let starredArticles = Array(try SmartFeedsController.shared.starredFeed.fetchArticles()).sortedByDate(.orderedDescending)
|
||||||
|
|
||||||
|
let todayArticles = Array(try SmartFeedsController.shared.todayFeed.fetchUnreadArticles()).sortedByDate(.orderedDescending)
|
||||||
|
|
||||||
|
var unread = [LatestArticle]()
|
||||||
|
var today = [LatestArticle]()
|
||||||
|
var starred = [LatestArticle]()
|
||||||
|
|
||||||
|
for article in unreadArticles {
|
||||||
|
let latestArticle = LatestArticle(id: article.sortableArticleID,
|
||||||
|
feedTitle: article.sortableName,
|
||||||
|
articleTitle: ArticleStringFormatter.truncatedTitle(article).isEmpty ? article.contentHTML?.strippingHTML().trimmingWhitespace : ArticleStringFormatter.truncatedTitle(article),
|
||||||
|
articleSummary: article.summary,
|
||||||
|
feedIcon: article.iconImage()?.image.dataRepresentation(),
|
||||||
|
pubDate: article.datePublished!.description)
|
||||||
|
unread.append(latestArticle)
|
||||||
|
if unread.count == 7 { break }
|
||||||
|
}
|
||||||
|
|
||||||
|
for article in starredArticles {
|
||||||
|
let latestArticle = LatestArticle(id: article.sortableArticleID,
|
||||||
|
feedTitle: article.sortableName,
|
||||||
|
articleTitle: ArticleStringFormatter.truncatedTitle(article).isEmpty ? article.contentHTML?.strippingHTML().trimmingWhitespace : ArticleStringFormatter.truncatedTitle(article),
|
||||||
|
articleSummary: article.summary,
|
||||||
|
feedIcon: article.iconImage()?.image.dataRepresentation(),
|
||||||
|
pubDate: article.datePublished!.description)
|
||||||
|
starred.append(latestArticle)
|
||||||
|
if starred.count == 7 { break }
|
||||||
|
}
|
||||||
|
|
||||||
|
for article in todayArticles {
|
||||||
|
let latestArticle = LatestArticle(id: article.sortableArticleID,
|
||||||
|
feedTitle: article.sortableName,
|
||||||
|
articleTitle: ArticleStringFormatter.truncatedTitle(article).isEmpty ? article.contentHTML?.strippingHTML().trimmingWhitespace : ArticleStringFormatter.truncatedTitle(article),
|
||||||
|
articleSummary: article.summary,
|
||||||
|
feedIcon: article.iconImage()?.image.dataRepresentation(),
|
||||||
|
pubDate: article.datePublished!.description)
|
||||||
|
today.append(latestArticle)
|
||||||
|
if today.count == 7 { break }
|
||||||
|
}
|
||||||
|
|
||||||
|
let latestData = WidgetData(currentUnreadCount: SmartFeedsController.shared.unreadFeed.unreadCount,
|
||||||
|
currentTodayCount: try! SmartFeedsController.shared.todayFeed.fetchUnreadArticles().count,
|
||||||
|
currentStarredCount: try! SmartFeedsController.shared.starredFeed.fetchArticles().count,
|
||||||
|
unreadArticles: unread,
|
||||||
|
starredArticles: starred,
|
||||||
|
todayArticles:today,
|
||||||
|
lastUpdateTime: Date())
|
||||||
|
|
||||||
|
|
||||||
|
DispatchQueue.global().async { [weak self] in
|
||||||
|
guard let self = self else { return }
|
||||||
|
|
||||||
|
self.backgroundTaskID = UIApplication.shared.beginBackgroundTask (withName: "com.ranchero.NetNewsWire.Encode") {
|
||||||
|
UIApplication.shared.endBackgroundTask(self.backgroundTaskID!)
|
||||||
|
self.backgroundTaskID = .invalid
|
||||||
|
}
|
||||||
|
let encodedData = try? JSONEncoder().encode(latestData)
|
||||||
|
|
||||||
|
os_log(.debug, log: self.log, "Finished encoding widget data.")
|
||||||
|
|
||||||
|
if self.fileExists() {
|
||||||
|
try? FileManager.default.removeItem(at: self.dataURL!)
|
||||||
|
os_log(.debug, log: self.log, "Removed widget data from container.")
|
||||||
|
}
|
||||||
|
if FileManager.default.createFile(atPath: self.dataURL!.path, contents: encodedData, attributes: nil) {
|
||||||
|
os_log(.debug, log: self.log, "Wrote widget data to container.")
|
||||||
|
WidgetCenter.shared.reloadAllTimelines()
|
||||||
|
UIApplication.shared.endBackgroundTask(self.backgroundTaskID!)
|
||||||
|
self.backgroundTaskID = .invalid
|
||||||
|
} else {
|
||||||
|
UIApplication.shared.endBackgroundTask(self.backgroundTaskID!)
|
||||||
|
self.backgroundTaskID = .invalid
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private func fileExists() -> Bool {
|
||||||
|
FileManager.default.fileExists(atPath: dataURL!.path)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
46
Shared/Widget/WidgetDeepLinks.swift
Normal file
46
Shared/Widget/WidgetDeepLinks.swift
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
//
|
||||||
|
// WidgetDeepLinks.swift
|
||||||
|
// NetNewsWire
|
||||||
|
//
|
||||||
|
// Created by Stuart Breckenridge on 18/11/20.
|
||||||
|
// Copyright © 2020 Ranchero Software. All rights reserved.
|
||||||
|
//
|
||||||
|
|
||||||
|
import Foundation
|
||||||
|
|
||||||
|
enum WidgetDeepLink {
|
||||||
|
|
||||||
|
case unread
|
||||||
|
case unreadArticle(id: String)
|
||||||
|
case today
|
||||||
|
case todayArticle(id: String)
|
||||||
|
case starred
|
||||||
|
case starredArticle(id: String)
|
||||||
|
case icon
|
||||||
|
|
||||||
|
var url: URL {
|
||||||
|
switch self {
|
||||||
|
case .unread:
|
||||||
|
return URL(string: "nnw://showunread")!
|
||||||
|
case .unreadArticle(let articleID):
|
||||||
|
var url = URLComponents(url: WidgetDeepLink.unread.url, resolvingAgainstBaseURL: false)!
|
||||||
|
url.queryItems = [URLQueryItem(name: "id", value: articleID)]
|
||||||
|
return url.url!
|
||||||
|
case .today:
|
||||||
|
return URL(string: "nnw://showtoday")!
|
||||||
|
case .todayArticle(let articleID):
|
||||||
|
var url = URLComponents(url: WidgetDeepLink.today.url, resolvingAgainstBaseURL: false)!
|
||||||
|
url.queryItems = [URLQueryItem(name: "id", value: articleID)]
|
||||||
|
return url.url!
|
||||||
|
case .starred:
|
||||||
|
return URL(string: "nnw://showstarred")!
|
||||||
|
case .starredArticle(let articleID):
|
||||||
|
var url = URLComponents(url: WidgetDeepLink.starred.url, resolvingAgainstBaseURL: false)!
|
||||||
|
url.queryItems = [URLQueryItem(name: "id", value: articleID)]
|
||||||
|
return url.url!
|
||||||
|
case .icon:
|
||||||
|
return URL(string: "nnw://icon")!
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
51
Technotes/Widgets.md
Normal file
51
Technotes/Widgets.md
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
# Widgets on iOS
|
||||||
|
|
||||||
|
There are _currently_ seven widgets available for iOS:
|
||||||
|
|
||||||
|
- 1x small widget that displays the current count of each of the Smart Feeds
|
||||||
|
- 3x medium widgets—one for each of the smart feeds.
|
||||||
|
- 3x large widgets—bigger versions of the medium widgets
|
||||||
|
|
||||||
|
## Widget Data
|
||||||
|
The widget does not have access to the parent app's database. To surface data to the widget, a small amount of article data is encoded to JSON (see `WidgetDataEncoder`) and saved to the AppGroup container.
|
||||||
|
|
||||||
|
Widget data is written at two points:
|
||||||
|
|
||||||
|
1. As part of a background refresh
|
||||||
|
2. When the scene enters the background
|
||||||
|
|
||||||
|
The widget timeline is refreshed—via `WidgetCenter.shared.reloadAllTimelines()`—after each of the above.
|
||||||
|
|
||||||
|
## Deep Links
|
||||||
|
The medium widgets support deep links for each of the articles that are surfaced.
|
||||||
|
|
||||||
|
If the user taps on an unread article in the unread widget, the widget opens the parent app with a deep link URL (see `WidgetDeepLink`), for example: `nnw://showunread?id={articeID}`. Once the app is opened, `scene(_ scene: UIScene, openURLContexts URLContexts: Set<UIOpenURLContext>)` is called and it is then determined what should be presented to the user based on the URL. If there is no `id` parameter—the user has tapped on a small widget or a non-linked item in a medium widget—the relevant smart feed controller is displayed.
|
||||||
|
|
||||||
|
|
||||||
|
## Data Models
|
||||||
|
```swift
|
||||||
|
struct WidgetData: Codable {
|
||||||
|
|
||||||
|
let currentUnreadCount: Int
|
||||||
|
let currentTodayCount: Int
|
||||||
|
let currentStarredCount: Int
|
||||||
|
let unreadArticles: [LatestArticle]
|
||||||
|
let starredArticles: [LatestArticle]
|
||||||
|
let todayArticles: [LatestArticle]
|
||||||
|
let lastUpdateTime: Date
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
struct LatestArticle: Codable, Identifiable {
|
||||||
|
|
||||||
|
var id: String // articleID
|
||||||
|
let feedTitle: String
|
||||||
|
let articleTitle: String?
|
||||||
|
let articleSummary: String?
|
||||||
|
let feedIcon: Data? // Base64 encoded image
|
||||||
|
let pubDate: String
|
||||||
|
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
|
38
Widget/Assets.xcassets/AccentColor.colorset/Contents.json
Normal file
38
Widget/Assets.xcassets/AccentColor.colorset/Contents.json
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
{
|
||||||
|
"colors" : [
|
||||||
|
{
|
||||||
|
"color" : {
|
||||||
|
"color-space" : "srgb",
|
||||||
|
"components" : {
|
||||||
|
"alpha" : "1.000",
|
||||||
|
"blue" : "0.933",
|
||||||
|
"green" : "0.416",
|
||||||
|
"red" : "0.031"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"idiom" : "universal"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"appearances" : [
|
||||||
|
{
|
||||||
|
"appearance" : "luminosity",
|
||||||
|
"value" : "dark"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"color" : {
|
||||||
|
"color-space" : "srgb",
|
||||||
|
"components" : {
|
||||||
|
"alpha" : "1.000",
|
||||||
|
"blue" : "0.945",
|
||||||
|
"green" : "0.502",
|
||||||
|
"red" : "0.176"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"idiom" : "universal"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"info" : {
|
||||||
|
"author" : "xcode",
|
||||||
|
"version" : 1
|
||||||
|
}
|
||||||
|
}
|
116
Widget/Assets.xcassets/AppIcon.appiconset/Contents.json
Normal file
116
Widget/Assets.xcassets/AppIcon.appiconset/Contents.json
Normal file
@ -0,0 +1,116 @@
|
|||||||
|
{
|
||||||
|
"images" : [
|
||||||
|
{
|
||||||
|
"filename" : "icon-41.png",
|
||||||
|
"idiom" : "iphone",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "20x20"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-60.png",
|
||||||
|
"idiom" : "iphone",
|
||||||
|
"scale" : "3x",
|
||||||
|
"size" : "20x20"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-58.png",
|
||||||
|
"idiom" : "iphone",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "29x29"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-87.png",
|
||||||
|
"idiom" : "iphone",
|
||||||
|
"scale" : "3x",
|
||||||
|
"size" : "29x29"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-80.png",
|
||||||
|
"idiom" : "iphone",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "40x40"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-121.png",
|
||||||
|
"idiom" : "iphone",
|
||||||
|
"scale" : "3x",
|
||||||
|
"size" : "40x40"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-120.png",
|
||||||
|
"idiom" : "iphone",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "60x60"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-180.png",
|
||||||
|
"idiom" : "iphone",
|
||||||
|
"scale" : "3x",
|
||||||
|
"size" : "60x60"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-20.png",
|
||||||
|
"idiom" : "ipad",
|
||||||
|
"scale" : "1x",
|
||||||
|
"size" : "20x20"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-42.png",
|
||||||
|
"idiom" : "ipad",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "20x20"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-29.png",
|
||||||
|
"idiom" : "ipad",
|
||||||
|
"scale" : "1x",
|
||||||
|
"size" : "29x29"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-59.png",
|
||||||
|
"idiom" : "ipad",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "29x29"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-40.png",
|
||||||
|
"idiom" : "ipad",
|
||||||
|
"scale" : "1x",
|
||||||
|
"size" : "40x40"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-81.png",
|
||||||
|
"idiom" : "ipad",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "40x40"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-76.png",
|
||||||
|
"idiom" : "ipad",
|
||||||
|
"scale" : "1x",
|
||||||
|
"size" : "76x76"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-152.png",
|
||||||
|
"idiom" : "ipad",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "76x76"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-167.png",
|
||||||
|
"idiom" : "ipad",
|
||||||
|
"scale" : "2x",
|
||||||
|
"size" : "83.5x83.5"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"filename" : "icon-1024.png",
|
||||||
|
"idiom" : "ios-marketing",
|
||||||
|
"scale" : "1x",
|
||||||
|
"size" : "1024x1024"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"info" : {
|
||||||
|
"author" : "xcode",
|
||||||
|
"version" : 1
|
||||||
|
}
|
||||||
|
}
|
BIN
Widget/Assets.xcassets/AppIcon.appiconset/icon-1024.png
Normal file
BIN
Widget/Assets.xcassets/AppIcon.appiconset/icon-1024.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 689 KiB |
BIN
Widget/Assets.xcassets/AppIcon.appiconset/icon-120.png
Normal file
BIN
Widget/Assets.xcassets/AppIcon.appiconset/icon-120.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 11 KiB |
BIN
Widget/Assets.xcassets/AppIcon.appiconset/icon-121.png
Normal file
BIN
Widget/Assets.xcassets/AppIcon.appiconset/icon-121.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 11 KiB |
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user