mirror of
https://github.com/writeas/writefreely-swiftui-multiplatform.git
synced 2024-11-15 01:11:02 +00:00
a51bbd3abc
* Initial work on presenting alert on error
* Move Account-related error handling up the hierarchy
* Handle errors on logout
* Fix for temporary debugging
* Clean up WriteFreelyModel’s published vars
* Add error handling to top-level content view
* Set current error on API call failures
* Set current error on API call handlers
* Move User Defaults errors to ErrorConstants file
* Add default values for some error strings
* Handle purging post errors
* Add FIXME to track silent failure on fetching collections
As collections are fetched and added to the `list` property in the CollectionListModel’s initializer, it’s tricky to throw an error here: we call it as a property initializer in CollectionListView, which cannot throw.
Consider refactoring this logic such that we’re using, for example, a @FetchRequest in CollectionListView instead.
* Handle errors in (most) shared code
Two outliers to come back to are:
- the LocalStoreManager, where we can’t set a current error in the WriteFreelyModel in methods that can’t throw
- the CollectionListModel, where the initializer can’t throw because we use it as a property initializer in CollectionListView
* Add error handling to Mac app
* Revert "Add error handling to Mac app"
This reverts commit b1a8b8b29c
.
127 lines
4.7 KiB
Swift
127 lines
4.7 KiB
Swift
import SwiftUI
|
|
import CoreData
|
|
|
|
class PostListModel: ObservableObject {
|
|
func remove(_ post: WFAPost) {
|
|
withAnimation {
|
|
LocalStorageManager.standard.container.viewContext.delete(post)
|
|
LocalStorageManager.standard.saveContext()
|
|
}
|
|
}
|
|
|
|
func purgePublishedPosts() throws {
|
|
let fetchRequest: NSFetchRequest<NSFetchRequestResult> = NSFetchRequest(entityName: "WFAPost")
|
|
fetchRequest.predicate = NSPredicate(format: "status != %i", 0)
|
|
let deleteRequest = NSBatchDeleteRequest(fetchRequest: fetchRequest)
|
|
|
|
do {
|
|
try LocalStorageManager.standard.container.viewContext.executeAndMergeChanges(using: deleteRequest)
|
|
} catch {
|
|
throw LocalStoreError.couldNotPurgePosts("cached")
|
|
}
|
|
}
|
|
|
|
func getBodyPreview(of post: WFAPost) -> String {
|
|
var elidedPostBody: String = ""
|
|
|
|
// Strip any markdown from the post body.
|
|
let strippedPostBody = stripMarkdown(from: post.body)
|
|
|
|
// Extract lede from post.
|
|
elidedPostBody = extractLede(from: strippedPostBody)
|
|
|
|
return elidedPostBody
|
|
}
|
|
}
|
|
|
|
private extension PostListModel {
|
|
|
|
func stripMarkdown(from string: String) -> String {
|
|
var strippedString = string
|
|
strippedString = stripHeadingOctothorpes(from: strippedString)
|
|
strippedString = stripImages(from: strippedString, keepAltText: true)
|
|
return strippedString
|
|
}
|
|
|
|
func stripHeadingOctothorpes(from string: String) -> String {
|
|
let newLines = CharacterSet.newlines
|
|
var processedComponents: [String] = []
|
|
let components = string.components(separatedBy: newLines)
|
|
for component in components {
|
|
if component.isEmpty {
|
|
continue
|
|
}
|
|
var newString = component
|
|
while newString.first == "#" {
|
|
newString.removeFirst()
|
|
}
|
|
if newString.hasPrefix(" ") {
|
|
newString.removeFirst()
|
|
}
|
|
processedComponents.append(newString)
|
|
}
|
|
let headinglessString = processedComponents.joined(separator: "\n\n")
|
|
return headinglessString
|
|
}
|
|
|
|
func stripImages(from string: String, keepAltText: Bool = false) -> String {
|
|
let pattern = #"!\[[\"]?(.*?)[\"|]?\]\(.*?\)"#
|
|
var processedComponents: [String] = []
|
|
let components = string.components(separatedBy: .newlines)
|
|
for component in components {
|
|
if component.isEmpty { continue }
|
|
var processedString: String = component
|
|
if keepAltText {
|
|
let regex = try? NSRegularExpression(pattern: pattern, options: [])
|
|
if let matches = regex?.matches(
|
|
in: component, options: [], range: NSRange(location: 0, length: component.utf16.count)
|
|
) {
|
|
for match in matches {
|
|
if let range = Range(match.range(at: 1), in: component) {
|
|
processedString = "\(component[range])"
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
let range = component.startIndex..<component.endIndex
|
|
processedString = component.replacingOccurrences(
|
|
of: pattern,
|
|
with: "",
|
|
options: .regularExpression,
|
|
range: range
|
|
)
|
|
}
|
|
if processedString.isEmpty { continue }
|
|
processedComponents.append(processedString)
|
|
}
|
|
return processedComponents.joined(separator: "\n\n")
|
|
}
|
|
|
|
func extractLede(from string: String) -> String {
|
|
let truncatedString = string.prefix(80)
|
|
let terminatingPunctuation = ".。?"
|
|
let terminatingCharacters = CharacterSet(charactersIn: terminatingPunctuation).union(.newlines)
|
|
|
|
var lede: String = ""
|
|
let sentences = truncatedString.components(separatedBy: terminatingCharacters)
|
|
if let firstSentence = (sentences.filter { !$0.isEmpty }).first {
|
|
if truncatedString.count > firstSentence.count {
|
|
if terminatingPunctuation.contains(truncatedString[firstSentence.endIndex]) {
|
|
lede = String(truncatedString[...firstSentence.endIndex])
|
|
} else {
|
|
lede = firstSentence
|
|
}
|
|
} else if truncatedString.count == firstSentence.count {
|
|
if string.count > 80 {
|
|
if let endOfStringIndex = truncatedString.lastIndex(of: " ") {
|
|
lede = truncatedString[..<endOfStringIndex] + "…"
|
|
}
|
|
} else {
|
|
lede = firstSentence
|
|
}
|
|
}
|
|
}
|
|
return lede
|
|
}
|
|
}
|