1
0
mirror of https://github.com/taigrr/wtf synced 2025-01-18 04:03:14 -08:00
wtf/modules/feedreader/widget.go
Chris Cummer 200dbcc03c
WTF-730 Fix missing color key config error (#738)
* WTF-730 Fix missing color key config error

Signed-off-by: Chris Cummer <chriscummer@me.com>

* WTF-730 Add Subheading color formatting to modules

Users can now set a `subheading` color in their config to change the
color of subheadings in widget display.

Defaults to `red`.

Signed-off-by: Chris Cummer <chriscummer@me.com>

* WTF-730 Fix oustanding color issues

Clean up missing color config changes not addressed in earlier commits.

Signed-off-by: Chris Cummer <chriscummer@me.com>

* Remove unused dependency

Signed-off-by: Chris Cummer <chriscummer@me.com>

* WTF-730 Base cleanup

Signed-off-by: Chris Cummer <chriscummer@me.com>

* WTF-730 Fix a few bugs related to color config changes

Signed-off-by: Chris Cummer <chriscummer@me.com>

* WTF-730 Fix issues with PagerDuty subheading display

Signed-off-by: Chris Cummer <chriscummer@me.com>

* WTF-730 Fix bug with Todo list colour rendering

Signed-off-by: Chris Cummer <chriscummer@me.com>
2019-11-09 11:21:45 -08:00

182 lines
4.0 KiB
Go

package feedreader
import (
"fmt"
"sort"
"time"
"github.com/mmcdole/gofeed"
"github.com/rivo/tview"
"github.com/wtfutil/wtf/utils"
"github.com/wtfutil/wtf/view"
)
const (
publishedDateLayout = "Mon, 02 2006 15:04:05"
)
// FeedItem represents an item returned from an RSS or Atom feed
type FeedItem struct {
item *gofeed.Item
viewed bool
}
// Widget is the container for RSS and Atom data
type Widget struct {
view.KeyboardWidget
view.ScrollableWidget
stories []*FeedItem
parser *gofeed.Parser
settings *Settings
err error
}
// NewWidget creates a new instance of a widget
func NewWidget(app *tview.Application, pages *tview.Pages, settings *Settings) *Widget {
widget := &Widget{
KeyboardWidget: view.NewKeyboardWidget(app, pages, settings.common),
ScrollableWidget: view.NewScrollableWidget(app, settings.common),
parser: gofeed.NewParser(),
settings: settings,
}
widget.SetRenderFunction(widget.Render)
widget.initializeKeyboardControls()
widget.View.SetInputCapture(widget.InputCapture)
widget.KeyboardWidget.SetView(widget.View)
return widget
}
/* -------------------- Exported Functions -------------------- */
// Fetch retrieves RSS and Atom feed data
func (widget *Widget) Fetch(feedURLs []string) ([]*FeedItem, error) {
data := []*FeedItem{}
for _, feedURL := range feedURLs {
feedItems, err := widget.fetchForFeed(feedURL)
if err != nil {
return nil, err
}
for _, feedItem := range feedItems {
data = append(data, feedItem)
}
}
data = widget.sort(data)
return data, nil
}
// Refresh updates the data in the widget
func (widget *Widget) Refresh() {
feedItems, err := widget.Fetch(widget.settings.feeds)
if err != nil {
widget.err = err
widget.stories = nil
widget.SetItemCount(0)
} else {
widget.err = nil
widget.stories = feedItems
widget.SetItemCount(len(feedItems))
}
widget.Render()
}
// Render sets up the widget data for redrawing to the screen
func (widget *Widget) Render() {
widget.Redraw(widget.content)
}
/* -------------------- Unexported Functions -------------------- */
func (widget *Widget) fetchForFeed(feedURL string) ([]*FeedItem, error) {
feed, err := widget.parser.ParseURL(feedURL)
if err != nil {
return nil, err
}
feedItems := []*FeedItem{}
for idx, gofeedItem := range feed.Items {
if widget.settings.feedLimit >= 1 && idx >= widget.settings.feedLimit {
// We only want to get the widget.settings.feedLimit latest articles,
// not all of them. To get all, set feedLimit to < 1
break
}
feedItem := &FeedItem{
item: gofeedItem,
viewed: false,
}
feedItems = append(feedItems, feedItem)
}
return feedItems, nil
}
func (widget *Widget) content() (string, string, bool) {
title := widget.CommonSettings().Title
if widget.err != nil {
return title, widget.err.Error(), true
}
data := widget.stories
if data == nil || len(data) == 0 {
return title, "No data", false
}
var str string
for idx, feedItem := range data {
rowColor := widget.RowColor(idx)
if feedItem.viewed {
// Grays out viewed items in the list, while preserving background highlighting when selected
rowColor = "gray"
if idx == widget.Selected {
rowColor = fmt.Sprintf("gray:%s", widget.settings.common.Colors.RowTheme.HighlightedBackground)
}
}
row := fmt.Sprintf(
"[%s]%2d. %s[white]",
rowColor,
idx+1,
feedItem.item.Title,
)
str += utils.HighlightableHelper(widget.View, row, idx, len(feedItem.item.Title))
}
return title, str, false
}
// feedItems are sorted by published date
func (widget *Widget) sort(feedItems []*FeedItem) []*FeedItem {
sort.Slice(feedItems, func(i, j int) bool {
iTime, _ := time.Parse(publishedDateLayout, feedItems[i].item.Published)
jTime, _ := time.Parse(publishedDateLayout, feedItems[j].item.Published)
return iTime.After(jTime)
})
return feedItems
}
func (widget *Widget) openStory() {
sel := widget.GetSelected()
if sel >= 0 && widget.stories != nil && sel < len(widget.stories) {
story := widget.stories[sel]
story.viewed = true
utils.OpenFile(story.item.Link)
}
}