2019-04-15 22:03:05 +02:00
//
// M a s t e r T i m e l i n e V i e w C o n t r o l l e r . s w i f t
// N e t N e w s W i r e
//
// C r e a t e d b y M a u r i c e P a r k e r o n 4 / 8 / 1 9 .
// C o p y r i g h t © 2 0 1 9 R a n c h e r o S o f t w a r e . A l l r i g h t s r e s e r v e d .
//
import UIKit
import RSCore
import Account
import Articles
2019-07-27 21:49:07 +02:00
class MasterTimelineViewController : UITableViewController , UndoableCommandRunner {
2019-04-17 01:08:02 +02:00
2019-04-29 22:29:00 +02:00
private var numberOfTextLines = 0
2019-04-15 22:03:05 +02:00
2019-04-23 11:35:48 +02:00
@IBOutlet weak var markAllAsReadButton : UIBarButtonItem !
@IBOutlet weak var firstUnreadButton : UIBarButtonItem !
2019-04-22 22:31:34 +02:00
2019-08-30 21:17:05 +02:00
private lazy var dataSource = makeDataSource ( )
2019-08-31 18:50:34 +02:00
private let searchController = UISearchController ( searchResultsController : nil )
2019-09-01 19:43:07 +02:00
weak var coordinator : SceneCoordinator !
2019-04-21 20:57:23 +02:00
var undoableCommands = [ UndoableCommand ] ( )
2019-08-31 18:50:34 +02:00
2019-09-05 21:37:07 +02:00
private let keyboardManager = KeyboardManager ( type : . timeline )
2019-09-04 23:24:16 +02:00
override var keyCommands : [ UIKeyCommand ] ? {
return keyboardManager . keyCommands
}
2019-04-17 01:08:02 +02:00
override var canBecomeFirstResponder : Bool {
return true
}
2019-09-04 23:24:16 +02:00
2019-04-15 22:03:05 +02:00
override func viewDidLoad ( ) {
super . viewDidLoad ( )
2019-08-30 21:17:05 +02:00
2019-04-23 11:35:48 +02:00
NotificationCenter . default . addObserver ( self , selector : #selector ( unreadCountDidChange ( _ : ) ) , name : . UnreadCountDidChange , object : nil )
2019-04-15 22:03:05 +02:00
NotificationCenter . default . addObserver ( self , selector : #selector ( statusesDidChange ( _ : ) ) , name : . StatusesDidChange , object : nil )
NotificationCenter . default . addObserver ( self , selector : #selector ( feedIconDidBecomeAvailable ( _ : ) ) , name : . FeedIconDidBecomeAvailable , object : nil )
NotificationCenter . default . addObserver ( self , selector : #selector ( avatarDidBecomeAvailable ( _ : ) ) , name : . AvatarDidBecomeAvailable , object : nil )
2019-08-22 02:37:19 +02:00
NotificationCenter . default . addObserver ( self , selector : #selector ( faviconDidBecomeAvailable ( _ : ) ) , name : . FaviconDidBecomeAvailable , object : nil )
2019-04-29 22:29:00 +02:00
NotificationCenter . default . addObserver ( self , selector : #selector ( userDefaultsDidChange ( _ : ) ) , name : UserDefaults . didChangeNotification , object : nil )
2019-07-27 21:49:07 +02:00
NotificationCenter . default . addObserver ( self , selector : #selector ( progressDidChange ( _ : ) ) , name : . AccountRefreshProgressDidChange , object : nil )
2019-04-29 21:50:56 +02:00
NotificationCenter . default . addObserver ( self , selector : #selector ( contentSizeCategoryDidChange ) , name : UIContentSizeCategory . didChangeNotification , object : nil )
2019-04-21 20:57:23 +02:00
2019-08-31 18:50:34 +02:00
// S e t u p t h e S e a r c h C o n t r o l l e r
2019-09-02 19:40:14 +02:00
searchController . delegate = self
2019-08-31 18:50:34 +02:00
searchController . searchResultsUpdater = self
searchController . obscuresBackgroundDuringPresentation = false
2019-08-31 22:53:47 +02:00
searchController . searchBar . delegate = self
2019-08-31 18:50:34 +02:00
searchController . searchBar . placeholder = NSLocalizedString ( " Search Articles " , comment : " Search Articles " )
2019-08-31 22:53:47 +02:00
searchController . searchBar . scopeButtonTitles = [
NSLocalizedString ( " Here " , comment : " Here " ) ,
NSLocalizedString ( " All Articles " , comment : " All Articles " )
]
2019-08-31 18:50:34 +02:00
navigationItem . searchController = searchController
definesPresentationContext = true
2019-08-31 22:53:47 +02:00
2019-08-31 18:50:34 +02:00
// S e t u p t h e R e f r e s h C o n t r o l
2019-04-18 21:36:22 +02:00
refreshControl = UIRefreshControl ( )
refreshControl ! . addTarget ( self , action : #selector ( refreshAccounts ( _ : ) ) , for : . valueChanged )
2019-08-31 18:50:34 +02:00
// C o n f i g u r e t h e t a b l e
tableView . dataSource = dataSource
2019-04-29 22:29:00 +02:00
numberOfTextLines = AppDefaults . timelineNumberOfLines
2019-04-29 23:29:53 +02:00
resetEstimatedRowHeight ( )
2019-04-29 22:29:00 +02:00
2019-08-30 21:17:05 +02:00
applyChanges ( animate : false )
2019-04-23 01:00:26 +02:00
resetUI ( )
2019-04-22 22:31:34 +02:00
2019-04-15 22:03:05 +02:00
}
2019-08-02 17:25:47 +02:00
override func viewWillAppear ( _ animated : Bool ) {
clearsSelectionOnViewWillAppear = coordinator . isRootSplitCollapsed
super . viewWillAppear ( animated )
}
2019-04-17 01:08:02 +02:00
override func viewDidAppear ( _ animated : Bool ) {
super . viewDidAppear ( animated )
2019-07-27 21:49:07 +02:00
updateProgressIndicatorIfNeeded ( )
2019-04-17 01:08:02 +02:00
}
2019-06-20 18:58:26 +02:00
override func traitCollectionDidChange ( _ previousTraitCollection : UITraitCollection ? ) {
super . traitCollectionDidChange ( previousTraitCollection )
if traitCollection . userInterfaceStyle != previousTraitCollection ? . userInterfaceStyle {
appDelegate . authorAvatarDownloader . resetCache ( )
appDelegate . feedIconDownloader . resetCache ( )
appDelegate . faviconDownloader . resetCache ( )
2019-08-30 21:17:05 +02:00
// t r a i t C o l l e c t i o n D i d C h a n g e w i l l g e t c a l l e d o n a b a c k g r o u n d t h r e a d
DispatchQueue . main . async {
self . reloadAllVisibleCells ( )
2019-06-20 18:58:26 +02:00
}
}
}
2019-09-05 04:06:29 +02:00
// MARK: A c t i o n s
2019-04-15 22:03:05 +02:00
@IBAction func markAllAsRead ( _ sender : Any ) {
let title = NSLocalizedString ( " Mark All Read " , comment : " Mark All Read " )
let message = NSLocalizedString ( " Mark all articles in this timeline as read? " , comment : " Mark all articles " )
let alertController = UIAlertController ( title : title , message : message , preferredStyle : . alert )
let cancelTitle = NSLocalizedString ( " Cancel " , comment : " Cancel " )
let cancelAction = UIAlertAction ( title : cancelTitle , style : . cancel )
alertController . addAction ( cancelAction )
let markTitle = NSLocalizedString ( " Mark All Read " , comment : " Mark All Read " )
let markAction = UIAlertAction ( title : markTitle , style : . default ) { [ weak self ] ( action ) in
2019-07-06 19:31:07 +02:00
self ? . coordinator . markAllAsReadInTimeline ( )
2019-04-15 22:03:05 +02:00
}
2019-06-26 00:39:07 +02:00
2019-04-15 22:03:05 +02:00
alertController . addAction ( markAction )
present ( alertController , animated : true )
}
2019-04-23 11:35:48 +02:00
@IBAction func firstUnread ( _ sender : Any ) {
2019-08-04 00:07:43 +02:00
coordinator . selectNextUnread ( )
2019-04-23 01:00:26 +02:00
}
2019-09-05 04:06:29 +02:00
// MARK: K e y b o a r d s h o r t c u t s
@objc func selectNextUp ( _ sender : Any ? ) {
coordinator . selectPrevArticle ( )
}
@objc func selectNextDown ( _ sender : Any ? ) {
coordinator . selectNextArticle ( )
}
@objc func navigateToSidebar ( _ sender : Any ? ) {
coordinator . navigateToFeeds ( )
}
@objc func navigateToDetail ( _ sender : Any ? ) {
coordinator . navigateToDetail ( )
}
2019-08-25 18:38:04 +02:00
// MARK: A P I
func reinitializeArticles ( ) {
resetUI ( )
}
2019-08-30 21:42:33 +02:00
func reloadArticles ( animate : Bool ) {
applyChanges ( animate : animate ) { [ weak self ] in
self ? . updateArticleSelection ( animate : animate )
2019-08-25 18:38:04 +02:00
}
}
2019-08-30 21:42:33 +02:00
func updateArticleSelection ( animate : Bool ) {
2019-09-06 18:22:35 +02:00
guard ! coordinator . isRootSplitCollapsed && ! coordinator . articles . isEmpty else {
2019-09-06 18:11:28 +02:00
return
}
2019-08-25 18:38:04 +02:00
if let indexPath = coordinator . currentArticleIndexPath {
if tableView . indexPathForSelectedRow != indexPath {
2019-09-06 20:45:45 +02:00
tableView . selectRowAndScrollIfNotVisible ( at : indexPath , animated : true )
2019-08-25 18:38:04 +02:00
}
2019-09-05 04:06:29 +02:00
} else {
tableView . selectRow ( at : nil , animated : animate , scrollPosition : . none )
2019-08-25 18:38:04 +02:00
}
2019-09-06 17:38:02 +02:00
2019-08-25 18:38:04 +02:00
updateUI ( )
}
2019-09-02 00:41:46 +02:00
func showSearchAll ( ) {
navigationItem . searchController ? . isActive = true
navigationItem . searchController ? . searchBar . selectedScopeButtonIndex = 1
2019-09-06 17:29:00 +02:00
navigationItem . searchController ? . searchBar . becomeFirstResponder ( )
2019-09-02 00:41:46 +02:00
}
2019-09-05 04:06:29 +02:00
func focus ( ) {
becomeFirstResponder ( )
}
2019-04-15 22:03:05 +02:00
// MARK: - T a b l e v i e w
override func tableView ( _ tableView : UITableView , trailingSwipeActionsConfigurationForRowAt indexPath : IndexPath ) -> UISwipeActionsConfiguration ? {
2019-06-29 20:35:12 +02:00
let article = coordinator . articles [ indexPath . row ]
2019-04-15 22:03:05 +02:00
2019-04-29 13:01:53 +02:00
// S e t u p t h e r e a d a c t i o n
let readTitle = article . status . read ?
NSLocalizedString ( " Unread " , comment : " Unread " ) :
NSLocalizedString ( " Read " , comment : " Read " )
2019-04-15 22:03:05 +02:00
2019-04-29 13:01:53 +02:00
let readAction = UIContextualAction ( style : . normal , title : readTitle ) { [ weak self ] ( action , view , completionHandler ) in
2019-08-19 23:03:07 +02:00
self ? . coordinator . toggleRead ( for : indexPath )
2019-04-15 22:03:05 +02:00
completionHandler ( true )
}
2019-04-29 13:01:53 +02:00
readAction . image = AppAssets . circleClosedImage
2019-06-19 01:31:37 +02:00
readAction . backgroundColor = AppAssets . netNewsWireBlueColor
2019-04-15 22:03:05 +02:00
2019-04-29 13:01:53 +02:00
// S e t u p t h e s t a r a c t i o n
let starTitle = article . status . starred ?
NSLocalizedString ( " Unstar " , comment : " Unstar " ) :
NSLocalizedString ( " Star " , comment : " Star " )
2019-04-15 22:03:05 +02:00
2019-04-29 13:01:53 +02:00
let starAction = UIContextualAction ( style : . normal , title : starTitle ) { [ weak self ] ( action , view , completionHandler ) in
2019-08-19 23:03:07 +02:00
self ? . coordinator . toggleStar ( for : indexPath )
2019-04-15 22:03:05 +02:00
completionHandler ( true )
}
2019-04-29 13:01:53 +02:00
starAction . image = AppAssets . starClosedImage
starAction . backgroundColor = AppAssets . starColor
2019-04-15 22:03:05 +02:00
2019-08-19 00:34:53 +02:00
// S e t u p t h e r e a d a c t i o n
let moreTitle = NSLocalizedString ( " More " , comment : " More " )
let moreAction = UIContextualAction ( style : . normal , title : moreTitle ) { [ weak self ] ( action , view , completionHandler ) in
if let self = self {
let alert = UIAlertController ( title : nil , message : nil , preferredStyle : . actionSheet )
if let popoverController = alert . popoverPresentationController {
popoverController . sourceView = view
popoverController . sourceRect = CGRect ( x : view . frame . size . width / 2 , y : view . frame . size . height / 2 , width : 1 , height : 1 )
}
alert . addAction ( self . markOlderAsReadAlertAction ( indexPath : indexPath , completionHandler : completionHandler ) )
2019-08-19 22:45:52 +02:00
if let action = self . discloseFeedAlertAction ( indexPath : indexPath , completionHandler : completionHandler ) {
alert . addAction ( action )
}
2019-08-20 00:26:09 +02:00
if let action = self . markAllInFeedAsReadAlertAction ( indexPath : indexPath , completionHandler : completionHandler ) {
alert . addAction ( action )
}
2019-08-20 00:38:30 +02:00
if let action = self . openInBrowserAlertAction ( indexPath : indexPath , completionHandler : completionHandler ) {
alert . addAction ( action )
}
2019-08-20 01:09:38 +02:00
if let action = self . shareAlertAction ( indexPath : indexPath , completionHandler : completionHandler ) {
alert . addAction ( action )
}
2019-08-19 00:34:53 +02:00
let cancelTitle = NSLocalizedString ( " Cancel " , comment : " Cancel " )
alert . addAction ( UIAlertAction ( title : cancelTitle , style : . cancel ) { _ in
completionHandler ( true )
} )
self . present ( alert , animated : true )
}
}
2019-08-19 22:45:52 +02:00
moreAction . image = AppAssets . moreImage
2019-08-19 00:34:53 +02:00
moreAction . backgroundColor = UIColor . systemGray
let configuration = UISwipeActionsConfiguration ( actions : [ readAction , starAction , moreAction ] )
2019-04-15 22:03:05 +02:00
return configuration
}
2019-08-16 20:19:06 +02:00
override func tableView ( _ tableView : UITableView , contextMenuConfigurationForRowAt indexPath : IndexPath , point : CGPoint ) -> UIContextMenuConfiguration ? {
2019-08-19 22:45:52 +02:00
return UIContextMenuConfiguration ( identifier : nil , previewProvider : nil , actionProvider : { [ weak self ] suggestedActions in
2019-08-16 20:19:06 +02:00
2019-08-19 22:45:52 +02:00
guard let self = self else { return nil }
2019-08-16 20:19:06 +02:00
var actions = [ UIAction ] ( )
actions . append ( self . toggleArticleReadStatusAction ( indexPath : indexPath ) )
actions . append ( self . toggleArticleStarStatusAction ( indexPath : indexPath ) )
2019-08-19 00:34:53 +02:00
actions . append ( self . markOlderAsReadAction ( indexPath : indexPath ) )
2019-08-16 20:19:06 +02:00
2019-08-19 22:45:52 +02:00
if let action = self . discloseFeedAction ( indexPath : indexPath ) {
actions . append ( action )
}
2019-08-20 00:26:09 +02:00
if let action = self . markAllInFeedAsReadAction ( indexPath : indexPath ) {
actions . append ( action )
}
2019-08-20 00:38:30 +02:00
if let action = self . openInBrowserAction ( indexPath : indexPath ) {
actions . append ( action )
}
2019-08-20 01:09:38 +02:00
if let action = self . shareAction ( indexPath : indexPath ) {
actions . append ( action )
}
2019-08-19 22:49:42 +02:00
return UIMenu ( title : " " , children : actions )
2019-08-16 20:19:06 +02:00
} )
}
2019-04-15 22:03:05 +02:00
override func tableView ( _ tableView : UITableView , didSelectRowAt indexPath : IndexPath ) {
2019-09-04 23:24:16 +02:00
becomeFirstResponder ( )
2019-09-02 19:05:11 +02:00
coordinator . selectArticle ( indexPath , automated : false )
2019-04-15 22:03:05 +02:00
}
// MARK: N o t i f i c a t i o n s
2019-04-23 11:35:48 +02:00
@objc dynamic func unreadCountDidChange ( _ notification : Notification ) {
updateUI ( )
}
2019-04-15 22:03:05 +02:00
@objc func statusesDidChange ( _ note : Notification ) {
2019-09-06 20:45:45 +02:00
guard let updatedArticles = note . userInfo ? [ Account . UserInfoKey . articles ] as ? Set < Article > else {
2019-04-15 22:03:05 +02:00
return
}
2019-09-06 20:45:45 +02:00
let visibleArticles = tableView . indexPathsForVisibleRows ! . map { return coordinator . articles [ $0 . row ] }
let visibleUpdatedArticles = visibleArticles . filter { updatedArticles . contains ( $0 ) }
for article in visibleUpdatedArticles {
if let articleIndex = coordinator . indexForArticleID ( article . articleID ) {
if let cell = tableView . cellForRow ( at : IndexPath ( row : articleIndex , section : 0 ) ) as ? MasterTimelineTableViewCell {
configure ( cell , article : article )
}
}
}
2019-04-15 22:03:05 +02:00
}
@objc func feedIconDidBecomeAvailable ( _ note : Notification ) {
guard let feed = note . userInfo ? [ UserInfoKey . feed ] as ? Feed else {
return
}
2019-08-30 21:17:05 +02:00
tableView . indexPathsForVisibleRows ? . forEach { indexPath in
guard let article = coordinator . articles . articleAtRow ( indexPath . row ) else {
return
}
if article . feed = = feed , let cell = tableView . cellForRow ( at : indexPath ) as ? MasterTimelineTableViewCell , let image = avatarFor ( article ) {
cell . setAvatarImage ( image )
2019-04-15 22:03:05 +02:00
}
}
}
@objc func avatarDidBecomeAvailable ( _ note : Notification ) {
2019-06-29 20:35:12 +02:00
guard coordinator . showAvatars , let avatarURL = note . userInfo ? [ UserInfoKey . url ] as ? String else {
2019-04-15 22:03:05 +02:00
return
}
2019-08-30 21:17:05 +02:00
tableView . indexPathsForVisibleRows ? . forEach { indexPath in
guard let article = coordinator . articles . articleAtRow ( indexPath . row ) , let authors = article . authors , ! authors . isEmpty else {
return
}
for author in authors {
if author . avatarURL = = avatarURL , let cell = tableView . cellForRow ( at : indexPath ) as ? MasterTimelineTableViewCell , let image = avatarFor ( article ) {
cell . setAvatarImage ( image )
2019-04-15 22:03:05 +02:00
}
}
}
}
2019-08-22 02:37:19 +02:00
@objc func faviconDidBecomeAvailable ( _ note : Notification ) {
guard coordinator . showAvatars , let faviconURL = note . userInfo ? [ " faviconURL " ] as ? String else {
return
}
2019-08-30 21:17:05 +02:00
tableView . indexPathsForVisibleRows ? . forEach { indexPath in
guard let article = coordinator . articles . articleAtRow ( indexPath . row ) , let articleFaviconURL = article . feed ? . faviconURL else {
return
}
if faviconURL = = articleFaviconURL , let cell = tableView . cellForRow ( at : indexPath ) as ? MasterTimelineTableViewCell , let image = avatarFor ( article ) {
cell . setAvatarImage ( image )
return
2019-08-22 02:37:19 +02:00
}
2019-04-15 22:03:05 +02:00
}
}
2019-04-29 22:29:00 +02:00
@objc func userDefaultsDidChange ( _ note : Notification ) {
if numberOfTextLines != AppDefaults . timelineNumberOfLines {
numberOfTextLines = AppDefaults . timelineNumberOfLines
2019-04-29 23:29:53 +02:00
resetEstimatedRowHeight ( )
2019-08-30 21:17:05 +02:00
reloadAllVisibleCells ( )
2019-04-29 22:29:00 +02:00
}
}
2019-04-29 21:50:56 +02:00
@objc func contentSizeCategoryDidChange ( _ note : Notification ) {
2019-08-30 21:17:05 +02:00
reloadAllVisibleCells ( )
2019-04-29 21:50:56 +02:00
}
2019-07-27 21:49:07 +02:00
@objc func progressDidChange ( _ note : Notification ) {
updateProgressIndicatorIfNeeded ( )
}
2019-04-15 22:03:05 +02:00
// MARK: R e l o a d i n g
2019-08-30 21:17:05 +02:00
private func reloadAllVisibleCells ( ) {
let visibleArticles = tableView . indexPathsForVisibleRows ! . map { return coordinator . articles [ $0 . row ] }
reloadCells ( visibleArticles )
2019-04-15 22:03:05 +02:00
}
2019-08-30 21:17:05 +02:00
private func reloadCells ( _ articles : [ Article ] ) {
var snapshot = dataSource . snapshot ( )
snapshot . reloadItems ( articles )
dataSource . apply ( snapshot , animatingDifferences : false ) { [ weak self ] in
self ? . restoreSelectionIfNecessary ( )
}
2019-04-15 22:03:05 +02:00
}
2019-04-29 23:29:53 +02:00
// MARK: C e l l C o n f i g u r i n g
private func resetEstimatedRowHeight ( ) {
let longTitle = " But I must explain to you how all this mistaken idea of denouncing pleasure and praising pain was born and I will give you a complete account of the system, and expound the actual teachings of the great explorer of the truth, the master-builder of human happiness. No one rejects, dislikes, or avoids pleasure itself, because it is pleasure, but because those who do not know how to pursue pleasure rationally encounter consequences that are extremely painful. Nor again is there anyone who loves or pursues or desires to obtain pain of itself, because it is pain, but because occasionally circumstances occur in which toil and pain can procure him some great pleasure. To take a trivial example, which of us ever undertakes laborious physical exercise, except to obtain some advantage from it? But who has any right to find fault with a man who chooses to enjoy a pleasure that has no annoying consequences, or one who avoids a pain that produces no resultant pleasure? "
let prototypeID = " prototype "
let status = ArticleStatus ( articleID : prototypeID , read : false , starred : false , userDeleted : false , dateArrived : Date ( ) )
let prototypeArticle = Article ( accountID : prototypeID , articleID : prototypeID , feedID : prototypeID , uniqueID : prototypeID , title : longTitle , contentHTML : nil , contentText : nil , url : nil , externalURL : nil , summary : nil , imageURL : nil , bannerImageURL : nil , datePublished : nil , dateModified : nil , authors : nil , attachments : nil , status : status )
let prototypeCellData = MasterTimelineCellData ( article : prototypeArticle , showFeedName : true , feedName : " Prototype Feed Name " , avatar : nil , showAvatar : false , featuredImage : nil , numberOfLines : numberOfTextLines )
2019-04-30 00:19:08 +02:00
if UIApplication . shared . preferredContentSizeCategory . isAccessibilityCategory {
let layout = MasterTimelineAccessibilityCellLayout ( width : tableView . bounds . width , insets : tableView . safeAreaInsets , cellData : prototypeCellData )
tableView . estimatedRowHeight = layout . height
} else {
let layout = MasterTimelineDefaultCellLayout ( width : tableView . bounds . width , insets : tableView . safeAreaInsets , cellData : prototypeCellData )
tableView . estimatedRowHeight = layout . height
}
2019-04-29 23:29:53 +02:00
}
2019-04-15 22:03:05 +02:00
}
2019-08-31 22:53:47 +02:00
// MARK: S e a r c h i n g
2019-08-31 18:50:34 +02:00
2019-09-02 19:40:14 +02:00
extension MasterTimelineViewController : UISearchControllerDelegate {
func willPresentSearchController ( _ searchController : UISearchController ) {
coordinator . beginSearching ( )
searchController . searchBar . showsScopeBar = true
}
func willDismissSearchController ( _ searchController : UISearchController ) {
coordinator . endSearching ( )
searchController . searchBar . showsScopeBar = false
}
}
2019-08-31 18:50:34 +02:00
extension MasterTimelineViewController : UISearchResultsUpdating {
2019-08-31 22:53:47 +02:00
2019-08-31 18:50:34 +02:00
func updateSearchResults ( for searchController : UISearchController ) {
2019-08-31 22:53:47 +02:00
let searchScope = SearchScope ( rawValue : searchController . searchBar . selectedScopeButtonIndex ) !
coordinator . searchArticles ( searchController . searchBar . text ! , searchScope )
2019-08-31 18:50:34 +02:00
}
2019-08-31 22:53:47 +02:00
}
extension MasterTimelineViewController : UISearchBarDelegate {
func searchBar ( _ searchBar : UISearchBar , selectedScopeButtonIndexDidChange selectedScope : Int ) {
let searchScope = SearchScope ( rawValue : selectedScope ) !
coordinator . searchArticles ( searchBar . text ! , searchScope )
}
2019-08-31 18:50:34 +02:00
}
2019-04-15 22:03:05 +02:00
// MARK: P r i v a t e
private extension MasterTimelineViewController {
2019-04-22 23:25:16 +02:00
2019-04-18 21:36:22 +02:00
@objc private func refreshAccounts ( _ sender : Any ) {
2019-04-23 14:26:35 +02:00
refreshControl ? . endRefreshing ( )
2019-07-18 22:16:54 +02:00
// T h i s i s a h a c k t o m a k e s u r e t h a t a n e r r o r d i a l o g d o e s n ' t i n t e r f e r e w i t h d i s m i s s i n g t h e r e f r e s h C o n t r o l .
// I f t h e e r r o r d i a l o g a p p e a r s t o o c l o s e l y t o t h e c a l l t o e n d R e f r e s h i n g , t h e n t h e r e f r e s h C o n t r o l n e v e r d i s a p p e a r s .
DispatchQueue . main . asyncAfter ( deadline : . now ( ) + 0.5 ) {
AccountManager . shared . refreshAll ( errorHandler : ErrorHandler . present ( self ) )
}
2019-04-18 21:36:22 +02:00
}
2019-04-22 23:25:16 +02:00
2019-04-23 01:00:26 +02:00
func resetUI ( ) {
2019-04-22 23:25:16 +02:00
2019-06-29 20:35:12 +02:00
title = coordinator . timelineName
navigationController ? . title = coordinator . timelineName
2019-04-22 23:25:16 +02:00
2019-07-27 21:36:01 +02:00
tableView . selectRow ( at : nil , animated : false , scrollPosition : . top )
2019-06-29 20:35:12 +02:00
if coordinator . articles . count > 0 {
2019-04-22 23:25:16 +02:00
tableView . scrollToRow ( at : IndexPath ( row : 0 , section : 0 ) , at : . top , animated : false )
}
2019-04-23 11:35:48 +02:00
updateUI ( )
2019-04-23 01:00:26 +02:00
}
2019-04-23 11:35:48 +02:00
func updateUI ( ) {
2019-06-29 20:35:12 +02:00
markAllAsReadButton . isEnabled = coordinator . isTimelineUnreadAvailable
firstUnreadButton . isEnabled = coordinator . isTimelineUnreadAvailable
2019-04-22 23:25:16 +02:00
}
2019-04-18 21:36:22 +02:00
2019-07-27 21:49:07 +02:00
func updateProgressIndicatorIfNeeded ( ) {
if ! coordinator . isThreePanelMode {
navigationController ? . updateAccountRefreshProgressIndicator ( )
}
}
2019-08-30 21:17:05 +02:00
func applyChanges ( animate : Bool , completion : ( ( ) -> Void ) ? = nil ) {
var snapshot = NSDiffableDataSourceSnapshot < Int , Article > ( )
snapshot . appendSections ( [ 0 ] )
snapshot . appendItems ( coordinator . articles , toSection : 0 )
dataSource . apply ( snapshot , animatingDifferences : animate ) { [ weak self ] in
self ? . restoreSelectionIfNecessary ( )
completion ? ( )
}
}
func makeDataSource ( ) -> UITableViewDiffableDataSource < Int , Article > {
return MasterTimelineDataSource ( coordinator : coordinator , tableView : tableView , cellProvider : { [ weak self ] tableView , indexPath , article in
let cell = tableView . dequeueReusableCell ( withIdentifier : " Cell " , for : indexPath ) as ! MasterTimelineTableViewCell
self ? . configure ( cell , article : article )
return cell
} )
}
2019-07-27 21:49:07 +02:00
2019-08-30 21:17:05 +02:00
func configure ( _ cell : MasterTimelineTableViewCell , article : Article ) {
2019-04-15 22:03:05 +02:00
2019-04-29 12:51:47 +02:00
let avatar = avatarFor ( article )
2019-04-15 22:03:05 +02:00
let featuredImage = featuredImageFor ( article )
2019-06-29 20:35:12 +02:00
let showFeedNames = coordinator . showFeedNames
let showAvatar = coordinator . showAvatars && avatar != nil
2019-04-29 22:29:00 +02:00
cell . cellData = MasterTimelineCellData ( article : article , showFeedName : showFeedNames , feedName : article . feed ? . nameForDisplay , avatar : avatar , showAvatar : showAvatar , featuredImage : featuredImage , numberOfLines : numberOfTextLines )
2019-04-15 22:03:05 +02:00
}
2019-08-24 21:57:51 +02:00
func avatarFor ( _ article : Article ) -> RSImage ? {
2019-06-29 20:35:12 +02:00
if ! coordinator . showAvatars {
2019-04-15 22:03:05 +02:00
return nil
}
2019-08-24 21:57:51 +02:00
return article . avatarImage ( )
2019-04-15 22:03:05 +02:00
}
func featuredImageFor ( _ article : Article ) -> UIImage ? {
if let url = article . imageURL , let data = appDelegate . imageDownloader . image ( for : url ) {
return RSImage ( data : data )
}
return nil
}
2019-08-30 21:17:05 +02:00
func restoreSelectionIfNecessary ( ) {
2019-09-06 18:22:35 +02:00
guard ! coordinator . isRootSplitCollapsed else {
2019-08-30 09:39:54 +02:00
return
}
2019-09-03 20:43:59 +02:00
if let articleID = coordinator . currentArticle ? . articleID , let index = coordinator . indexForArticleID ( articleID ) {
2019-08-29 18:53:36 +02:00
let indexPath = IndexPath ( row : index , section : 0 )
2019-09-06 20:45:45 +02:00
tableView . selectRowAndScrollIfNotVisible ( at : indexPath , animated : false )
2019-04-15 22:03:05 +02:00
}
}
2019-08-16 20:19:06 +02:00
func toggleArticleReadStatusAction ( indexPath : IndexPath ) -> UIAction {
let article = coordinator . articles [ indexPath . row ]
let title = article . status . read ?
NSLocalizedString ( " Mark as Unread " , comment : " Mark as Unread " ) :
NSLocalizedString ( " Mark as Read " , comment : " Mark as Read " )
let image = article . status . read ? AppAssets . circleClosedImage : AppAssets . circleOpenImage
2019-08-19 00:34:53 +02:00
let action = UIAction ( title : title , image : image ) { [ weak self ] action in
2019-08-19 23:03:07 +02:00
self ? . coordinator . toggleRead ( for : indexPath )
2019-08-16 20:19:06 +02:00
}
return action
}
func toggleArticleStarStatusAction ( indexPath : IndexPath ) -> UIAction {
let article = coordinator . articles [ indexPath . row ]
let title = article . status . starred ?
NSLocalizedString ( " Mark as Unstarred " , comment : " Mark as Unstarred " ) :
NSLocalizedString ( " Mark as Starred " , comment : " Mark as Starred " )
let image = article . status . starred ? AppAssets . starOpenImage : AppAssets . starClosedImage
2019-08-19 00:34:53 +02:00
let action = UIAction ( title : title , image : image ) { [ weak self ] action in
2019-08-19 23:03:07 +02:00
self ? . coordinator . toggleStar ( for : indexPath )
2019-08-16 20:19:06 +02:00
}
return action
}
2019-08-19 00:34:53 +02:00
func markOlderAsReadAction ( indexPath : IndexPath ) -> UIAction {
let title = NSLocalizedString ( " Mark Older as Read " , comment : " Mark Older as Read " )
let image = coordinator . sortDirection = = . orderedDescending ? AppAssets . markOlderAsReadDownImage : AppAssets . markOlderAsReadUpImage
let action = UIAction ( title : title , image : image ) { [ weak self ] action in
2019-08-19 23:03:07 +02:00
self ? . coordinator . markAsReadOlderArticlesInTimeline ( indexPath )
2019-08-19 00:34:53 +02:00
}
return action
}
func markOlderAsReadAlertAction ( indexPath : IndexPath , completionHandler : @ escaping ( Bool ) -> Void ) -> UIAlertAction {
let title = NSLocalizedString ( " Mark Older as Read " , comment : " Mark Older as Read " )
let action = UIAlertAction ( title : title , style : . default ) { [ weak self ] action in
2019-08-19 23:03:07 +02:00
self ? . coordinator . markAsReadOlderArticlesInTimeline ( indexPath )
2019-08-19 00:34:53 +02:00
completionHandler ( true )
}
return action
}
2019-08-19 22:45:52 +02:00
func discloseFeedAction ( indexPath : IndexPath ) -> UIAction ? {
guard let feed = coordinator . articles [ indexPath . row ] . feed else {
return nil
}
let title = NSLocalizedString ( " Select Feed " , comment : " Select Feed " )
let action = UIAction ( title : title , image : AppAssets . openInSidebarImage ) { [ weak self ] action in
self ? . coordinator . discloseFeed ( feed )
}
return action
}
func discloseFeedAlertAction ( indexPath : IndexPath , completionHandler : @ escaping ( Bool ) -> Void ) -> UIAlertAction ? {
guard let feed = coordinator . articles [ indexPath . row ] . feed else {
return nil
}
let title = NSLocalizedString ( " Select Feed " , comment : " Select Feed " )
let action = UIAlertAction ( title : title , style : . default ) { [ weak self ] action in
self ? . coordinator . discloseFeed ( feed )
completionHandler ( true )
}
return action
}
2019-08-20 00:26:09 +02:00
func markAllInFeedAsReadAction ( indexPath : IndexPath ) -> UIAction ? {
guard let feed = coordinator . articles [ indexPath . row ] . feed else {
return nil
}
let articles = Array ( feed . fetchArticles ( ) )
guard articles . canMarkAllAsRead ( ) else {
return nil
}
let localizedMenuText = NSLocalizedString ( " Mark All as Read in “%@” " , comment : " Command " )
let title = NSString . localizedStringWithFormat ( localizedMenuText as NSString , feed . nameForDisplay ) as String
let action = UIAction ( title : title , image : AppAssets . markAllInFeedAsReadImage ) { [ weak self ] action in
self ? . coordinator . markAllAsRead ( articles )
}
return action
}
func markAllInFeedAsReadAlertAction ( indexPath : IndexPath , completionHandler : @ escaping ( Bool ) -> Void ) -> UIAlertAction ? {
guard let feed = coordinator . articles [ indexPath . row ] . feed else {
return nil
}
let articles = Array ( feed . fetchArticles ( ) )
guard articles . canMarkAllAsRead ( ) else {
return nil
}
2019-08-20 00:38:30 +02:00
let localizedMenuText = NSLocalizedString ( " Mark All as Read in “%@” " , comment : " Mark All as Read in Feed " )
2019-08-20 00:26:09 +02:00
let title = NSString . localizedStringWithFormat ( localizedMenuText as NSString , feed . nameForDisplay ) as String
let action = UIAlertAction ( title : title , style : . default ) { [ weak self ] action in
self ? . coordinator . markAllAsRead ( articles )
completionHandler ( true )
}
return action
}
2019-08-20 00:38:30 +02:00
func openInBrowserAction ( indexPath : IndexPath ) -> UIAction ? {
guard let preferredLink = coordinator . articles [ indexPath . row ] . preferredLink , let _ = URL ( string : preferredLink ) else {
return nil
}
let title = NSLocalizedString ( " Open in Browser " , comment : " Open in Browser " )
let action = UIAction ( title : title , image : AppAssets . safariImage ) { [ weak self ] action in
2019-09-04 23:24:16 +02:00
self ? . coordinator . showBrowserForArticle ( indexPath )
2019-08-20 00:38:30 +02:00
}
return action
}
func openInBrowserAlertAction ( indexPath : IndexPath , completionHandler : @ escaping ( Bool ) -> Void ) -> UIAlertAction ? {
guard let preferredLink = coordinator . articles [ indexPath . row ] . preferredLink , let _ = URL ( string : preferredLink ) else {
return nil
}
let title = NSLocalizedString ( " Open in Browser " , comment : " Open in Browser " )
let action = UIAlertAction ( title : title , style : . default ) { [ weak self ] action in
2019-09-04 23:24:16 +02:00
self ? . coordinator . showBrowserForArticle ( indexPath )
2019-08-20 00:38:30 +02:00
completionHandler ( true )
}
return action
}
2019-08-20 00:26:09 +02:00
2019-08-20 01:09:38 +02:00
func shareDialogForTableCell ( indexPath : IndexPath , url : URL , title : String ? ) {
let itemSource = ArticleActivityItemSource ( url : url , subject : title )
let activityViewController = UIActivityViewController ( activityItems : [ itemSource ] , applicationActivities : nil )
guard let cell = tableView . cellForRow ( at : indexPath ) else { return }
let popoverController = activityViewController . popoverPresentationController
popoverController ? . sourceView = cell
popoverController ? . sourceRect = CGRect ( x : 0 , y : 0 , width : cell . frame . size . width , height : cell . frame . size . height )
present ( activityViewController , animated : true )
}
func shareAction ( indexPath : IndexPath ) -> UIAction ? {
let article = coordinator . articles [ indexPath . row ]
guard let preferredLink = article . preferredLink , let url = URL ( string : preferredLink ) else {
return nil
}
let title = NSLocalizedString ( " Share " , comment : " Share " )
let action = UIAction ( title : title , image : AppAssets . shareImage ) { [ weak self ] action in
self ? . shareDialogForTableCell ( indexPath : indexPath , url : url , title : article . title )
}
return action
}
func shareAlertAction ( indexPath : IndexPath , completionHandler : @ escaping ( Bool ) -> Void ) -> UIAlertAction ? {
let article = coordinator . articles [ indexPath . row ]
guard let preferredLink = article . preferredLink , let url = URL ( string : preferredLink ) else {
return nil
}
let title = NSLocalizedString ( " Share " , comment : " Share " )
let action = UIAlertAction ( title : title , style : . default ) { [ weak self ] action in
completionHandler ( true )
self ? . shareDialogForTableCell ( indexPath : indexPath , url : url , title : article . title )
}
return action
}
2019-04-15 22:03:05 +02:00
}