Add News widget and integrate NewsAPI for live updates
This commit is contained in:
parent
d638a8ae97
commit
9b5afc3d7c
77
apis/newsapi/main.go
Normal file
77
apis/newsapi/main.go
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
package news
|
||||||
|
|
||||||
|
import (
|
||||||
|
"ArinDash/config"
|
||||||
|
"encoding/json"
|
||||||
|
"io"
|
||||||
|
"log"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
const apiBaseURL = "https://newsapi.org/v2/top-headlines"
|
||||||
|
|
||||||
|
type configFile struct {
|
||||||
|
News newsConfig
|
||||||
|
}
|
||||||
|
|
||||||
|
type newsConfig struct {
|
||||||
|
ApiKey string
|
||||||
|
Sources string
|
||||||
|
}
|
||||||
|
|
||||||
|
type News struct {
|
||||||
|
Status string `json:"status"`
|
||||||
|
TotalResults int `json:"totalResults"`
|
||||||
|
Articles []Article `json:"articles"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Article struct {
|
||||||
|
Source Source `json:"source"`
|
||||||
|
Author string `json:"author"`
|
||||||
|
Title string `json:"title"`
|
||||||
|
Description string `json:"description"`
|
||||||
|
URL string `json:"url"`
|
||||||
|
URLToImage string `json:"urlToImage"`
|
||||||
|
PublishedAt string `json:"publishedAt"`
|
||||||
|
Content string `json:"content"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Source struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func FetchNews() News {
|
||||||
|
cfg := &configFile{}
|
||||||
|
config.LoadConfig(cfg)
|
||||||
|
client := &http.Client{}
|
||||||
|
|
||||||
|
req, err := http.NewRequest("GET", apiBaseURL+"?sources="+cfg.News.Sources+"&pageSize=100&apiKey="+cfg.News.ApiKey, nil)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return News{
|
||||||
|
Status: err.Error(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
respBody, err := io.ReadAll(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
return News{
|
||||||
|
Status: err.Error(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
news := News{}
|
||||||
|
|
||||||
|
err = json.Unmarshal([]byte(strings.ReplaceAll(strings.ReplaceAll(string(respBody), "\r", ""), "\u00A0", "")), &news)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return news
|
||||||
|
}
|
||||||
@ -4,7 +4,6 @@ import (
|
|||||||
"ArinDash/config"
|
"ArinDash/config"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -29,6 +28,7 @@ type Warning struct {
|
|||||||
Code []string `json:"code"`
|
Code []string `json:"code"`
|
||||||
Reference string `json:"reference"`
|
Reference string `json:"reference"`
|
||||||
Info []Info `json:"info"`
|
Info []Info `json:"info"`
|
||||||
|
Errormsg string `json:"errormsg"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type Info struct {
|
type Info struct {
|
||||||
@ -53,33 +53,45 @@ func FetchWarnings() []Warning {
|
|||||||
}
|
}
|
||||||
resp, err := client.Do(req)
|
resp, err := client.Do(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
return append(make([]Warning, 0), Warning{
|
||||||
|
Errormsg: err.Error(),
|
||||||
|
})
|
||||||
}
|
}
|
||||||
defer resp.Body.Close()
|
defer resp.Body.Close()
|
||||||
respBody, err := io.ReadAll(resp.Body)
|
respBody, err := io.ReadAll(resp.Body)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
return append(make([]Warning, 0), Warning{
|
||||||
|
Errormsg: err.Error(),
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
warnings := make([]Warning, 0)
|
warnings := make([]Warning, 0)
|
||||||
err = json.Unmarshal(respBody, &warnings)
|
err = json.Unmarshal(respBody, &warnings)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
return append(make([]Warning, 0), Warning{
|
||||||
|
Errormsg: err.Error(),
|
||||||
|
})
|
||||||
}
|
}
|
||||||
result := make([]Warning, 0)
|
result := make([]Warning, 0)
|
||||||
for _, warning := range warnings {
|
for _, warning := range warnings {
|
||||||
req, err := http.NewRequest("GET", apiBaseURL+"/warnings/"+warning.Id+".json", nil)
|
req, err := http.NewRequest("GET", apiBaseURL+"/warnings/"+warning.Id+".json", nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
return append(make([]Warning, 0), Warning{
|
||||||
|
Errormsg: err.Error(),
|
||||||
|
})
|
||||||
}
|
}
|
||||||
resp, err := client.Do(req)
|
resp, err := client.Do(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
return append(make([]Warning, 0), Warning{
|
||||||
|
Errormsg: err.Error(),
|
||||||
|
})
|
||||||
}
|
}
|
||||||
defer resp.Body.Close()
|
defer resp.Body.Close()
|
||||||
respBody, err := io.ReadAll(resp.Body)
|
respBody, err := io.ReadAll(resp.Body)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
return append(make([]Warning, 0), Warning{
|
||||||
|
Errormsg: err.Error(),
|
||||||
|
})
|
||||||
}
|
}
|
||||||
warning := Warning{}
|
warning := Warning{}
|
||||||
err = json.Unmarshal(respBody, &warning)
|
err = json.Unmarshal(respBody, &warning)
|
||||||
|
|||||||
@ -3,21 +3,22 @@ package openWeatherMap
|
|||||||
import "time"
|
import "time"
|
||||||
|
|
||||||
type CurrentWeather struct {
|
type CurrentWeather struct {
|
||||||
Base string `json:"base"`
|
Base string `json:"base"`
|
||||||
Visibility int `json:"visibility"`
|
Visibility int `json:"visibility"`
|
||||||
Dt int `json:"dt"`
|
Dt int `json:"dt"`
|
||||||
Timezone int `json:"timezone"`
|
Timezone int `json:"timezone"`
|
||||||
Id int `json:"id"`
|
Id int `json:"id"`
|
||||||
Name string `json:"name"`
|
Name string `json:"name"`
|
||||||
Cod int `json:"cod"`
|
Cod int `json:"cod"`
|
||||||
Coordinates Coordinates `json:"coord"`
|
Coordinates Coordinates `json:"coord"`
|
||||||
Weather []Weather `json:"weather"`
|
Weather []Weather `json:"weather"`
|
||||||
Wind Wind `json:"wind"`
|
Wind Wind `json:"wind"`
|
||||||
Clouds Clouds `json:"clouds"`
|
Clouds Clouds `json:"clouds"`
|
||||||
Rain Rain `json:"rain"`
|
Rain Rain `json:"rain"`
|
||||||
Snow Snow `json:"snow"`
|
Snow Snow `json:"snow"`
|
||||||
Sys Sys `json:"sys"`
|
Sys Sys `json:"sys"`
|
||||||
Main Main `json:"main"`
|
Main Main `json:"main"`
|
||||||
|
ErrorMessage string
|
||||||
}
|
}
|
||||||
|
|
||||||
type Coordinates struct {
|
type Coordinates struct {
|
||||||
|
|||||||
@ -4,7 +4,6 @@ import (
|
|||||||
"ArinDash/config"
|
"ArinDash/config"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -27,16 +26,22 @@ func FetchCurrentWeather() CurrentWeather {
|
|||||||
|
|
||||||
req, err := http.NewRequest("GET", apiBaseURL+"?id="+cfg.OpenWeatherMap.LocationId+"&units=metric&lang=en&APPID="+cfg.OpenWeatherMap.ApiKey, nil)
|
req, err := http.NewRequest("GET", apiBaseURL+"?id="+cfg.OpenWeatherMap.LocationId+"&units=metric&lang=en&APPID="+cfg.OpenWeatherMap.ApiKey, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
return CurrentWeather{
|
||||||
|
ErrorMessage: err.Error(),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
resp, err := client.Do(req)
|
resp, err := client.Do(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
return CurrentWeather{
|
||||||
|
ErrorMessage: err.Error(),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
defer resp.Body.Close()
|
defer resp.Body.Close()
|
||||||
respBody, err := io.ReadAll(resp.Body)
|
respBody, err := io.ReadAll(resp.Body)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
return CurrentWeather{
|
||||||
|
ErrorMessage: err.Error(),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
err = json.Unmarshal(respBody, currentWeather)
|
err = json.Unmarshal(respBody, currentWeather)
|
||||||
|
|||||||
@ -4,7 +4,6 @@ import (
|
|||||||
"ArinDash/config"
|
"ArinDash/config"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
@ -50,7 +49,7 @@ func (ph *PiHConnector) do(method string, endpoint string, body io.Reader) []byt
|
|||||||
req, err := http.NewRequest(method, requestString, body)
|
req, err := http.NewRequest(method, requestString, body)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
return make([]byte, 0)
|
||||||
}
|
}
|
||||||
|
|
||||||
req.Header.Add("X-FTL-SID", ph.Session.SID)
|
req.Header.Add("X-FTL-SID", ph.Session.SID)
|
||||||
@ -58,13 +57,13 @@ func (ph *PiHConnector) do(method string, endpoint string, body io.Reader) []byt
|
|||||||
|
|
||||||
resp, err := client.Do(req)
|
resp, err := client.Do(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
return make([]byte, 0)
|
||||||
}
|
}
|
||||||
defer resp.Body.Close()
|
defer resp.Body.Close()
|
||||||
|
|
||||||
respBody, err := io.ReadAll(resp.Body)
|
respBody, err := io.ReadAll(resp.Body)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
return make([]byte, 0)
|
||||||
}
|
}
|
||||||
|
|
||||||
return respBody
|
return respBody
|
||||||
@ -82,24 +81,24 @@ func Connect() PiHConnector {
|
|||||||
req, err := http.NewRequest("POST", "http://"+cfg.Pihole.Host+"/api/auth", strings.NewReader("{\"password\": \""+cfg.Pihole.Password+"\"}"))
|
req, err := http.NewRequest("POST", "http://"+cfg.Pihole.Host+"/api/auth", strings.NewReader("{\"password\": \""+cfg.Pihole.Password+"\"}"))
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
resp, err := client.Do(req)
|
resp, err := client.Do(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
defer resp.Body.Close()
|
defer resp.Body.Close()
|
||||||
|
|
||||||
respBody, err := io.ReadAll(resp.Body)
|
respBody, err := io.ReadAll(resp.Body)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
s := &PiHAuth{}
|
s := &PiHAuth{}
|
||||||
|
|
||||||
err = json.Unmarshal(respBody, s)
|
err = json.Unmarshal(respBody, s)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
connector = &PiHConnector{
|
connector = &PiHConnector{
|
||||||
Host: cfg.Pihole.Host,
|
Host: cfg.Pihole.Host,
|
||||||
|
|||||||
@ -17,3 +17,8 @@ ApiKey = "YourApiKey"
|
|||||||
[nina]
|
[nina]
|
||||||
#Code from https://www.xrepository.de/api/xrepository/urn:de:bund:destatis:bevoelkerungsstatistik:schluessel:rs_2021-07-31/download/Regionalschl_ssel_2021-07-31.json
|
#Code from https://www.xrepository.de/api/xrepository/urn:de:bund:destatis:bevoelkerungsstatistik:schluessel:rs_2021-07-31/download/Regionalschl_ssel_2021-07-31.json
|
||||||
gebietsCode = "000000000000"
|
gebietsCode = "000000000000"
|
||||||
|
|
||||||
|
[news]
|
||||||
|
#apiKey from newsapi.org
|
||||||
|
ApiKey = "ApiKey"
|
||||||
|
Sources = "comma,sepparated,sources,from,newsapi"
|
||||||
|
|||||||
13
main.go
13
main.go
@ -44,6 +44,7 @@ func initWidgets(ctx context.Context, term *tcell.Terminal) {
|
|||||||
widgets.New("PiHole", widgets.PiholeStats()),
|
widgets.New("PiHole", widgets.PiholeStats()),
|
||||||
widgets.New("PiHoleBlocked", widgets.PiholeBlocked()),
|
widgets.New("PiHoleBlocked", widgets.PiholeBlocked()),
|
||||||
widgets.New("NinaWarnings", widgets.NinaWarnings()),
|
widgets.New("NinaWarnings", widgets.NinaWarnings()),
|
||||||
|
widgets.New("News", widgets.News()),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -69,7 +70,7 @@ func layout() []container.Option {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
grid.ColWidthPerc(58,
|
grid.ColWidthPerc(62,
|
||||||
grid.RowHeightFixed(44,
|
grid.RowHeightFixed(44,
|
||||||
grid.ColWidthFixed(40,
|
grid.ColWidthFixed(40,
|
||||||
grid.RowHeightFixed(8,
|
grid.RowHeightFixed(8,
|
||||||
@ -125,8 +126,12 @@ func layout() []container.Option {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
grid.RowHeightFixed(20,
|
grid.RowHeightFixed(20,
|
||||||
grid.RowHeightFixed(25,
|
grid.RowHeightFixed(20,
|
||||||
grid.Widget(widgets.Get["empty"]),
|
grid.Widget(widgets.Get["News"],
|
||||||
|
container.BorderTitle("News"),
|
||||||
|
container.Border(linestyle.Light),
|
||||||
|
container.BorderColor(cell.ColorWhite),
|
||||||
|
),
|
||||||
),
|
),
|
||||||
grid.RowHeightFixed(20,
|
grid.RowHeightFixed(20,
|
||||||
grid.Widget(widgets.Get["NinaWarnings"],
|
grid.Widget(widgets.Get["NinaWarnings"],
|
||||||
@ -139,7 +144,7 @@ func layout() []container.Option {
|
|||||||
),
|
),
|
||||||
|
|
||||||
grid.ColWidthPerc(35,
|
grid.ColWidthPerc(35,
|
||||||
grid.RowHeightPerc(20,
|
grid.RowHeightFixed(20,
|
||||||
grid.Widget(widgets.Get["HTTPProber"],
|
grid.Widget(widgets.Get["HTTPProber"],
|
||||||
container.BorderTitle("Website Status"),
|
container.BorderTitle("Website Status"),
|
||||||
container.Border(linestyle.Light),
|
container.Border(linestyle.Light),
|
||||||
|
|||||||
114
widgets/news.go
Normal file
114
widgets/news.go
Normal file
@ -0,0 +1,114 @@
|
|||||||
|
package widgets
|
||||||
|
|
||||||
|
import (
|
||||||
|
news "ArinDash/apis/newsapi"
|
||||||
|
"ArinDash/util"
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/mum4k/termdash/cell"
|
||||||
|
"github.com/mum4k/termdash/keyboard"
|
||||||
|
"github.com/mum4k/termdash/terminal/terminalapi"
|
||||||
|
"github.com/mum4k/termdash/widgetapi"
|
||||||
|
"github.com/mum4k/termdash/widgets/text"
|
||||||
|
)
|
||||||
|
|
||||||
|
type NewsOptions struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func News() NewsOptions {
|
||||||
|
widgetOptions["NewsOptions"] = createNewsOptions
|
||||||
|
return NewsOptions{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func createNewsOptions(ctx context.Context, _ terminalapi.Terminal, _ interface{}) widgetapi.Widget {
|
||||||
|
widget := util.PanicOnErrorWithResult(NewNewsText(text.WrapAtWords()))
|
||||||
|
go util.Periodic(ctx, 1*time.Hour, func() error {
|
||||||
|
widget.newsArticles = news.FetchNews()
|
||||||
|
if widget.newsArticles.Status != "ok" {
|
||||||
|
if err := widget.Write(widget.newsArticles.Status, text.WriteCellOpts(cell.FgColor(cell.ColorRed))); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
go util.Periodic(ctx, 30*time.Second, func() error {
|
||||||
|
err := widget.drawNews()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if len(widget.newsArticles.Articles) > 0 {
|
||||||
|
widget.Selected = (widget.Selected + 1) % (len(widget.newsArticles.Articles) + 1)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
|
||||||
|
return widget
|
||||||
|
}
|
||||||
|
|
||||||
|
func (widget *NewsText) drawNews() error {
|
||||||
|
selected := widget.Selected
|
||||||
|
widget.Reset()
|
||||||
|
if len(widget.newsArticles.Articles) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if err := widget.Write(fmt.Sprintf("%d/%d\n", selected+1, len(widget.newsArticles.Articles)), text.WriteCellOpts(cell.FgColor(cell.ColorWhite))); err != nil {
|
||||||
|
widget.Write(fmt.Sprintf("Error: %s", err.Error()), text.WriteCellOpts(cell.FgColor(cell.ColorRed)))
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
article := widget.newsArticles.Articles[selected]
|
||||||
|
if err := widget.Write(fmt.Sprintf("%s\n", article.Author), text.WriteCellOpts(cell.FgColor(cell.ColorGray))); err != nil {
|
||||||
|
widget.Write(fmt.Sprintf("Error: %s", err.Error()), text.WriteCellOpts(cell.FgColor(cell.ColorRed)))
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := widget.Write(fmt.Sprintf("%s\n\n", article.Title), text.WriteCellOpts(cell.FgColor(cell.ColorWhite), cell.Bold())); err != nil {
|
||||||
|
widget.Write(fmt.Sprintf("Error: %s", err.Error()), text.WriteCellOpts(cell.FgColor(cell.ColorRed)))
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := widget.Write(fmt.Sprintf("%s\n\n", article.Description), text.WriteCellOpts(cell.FgColor(cell.ColorWhite))); err != nil {
|
||||||
|
widget.Write(fmt.Sprintf("Error: %s", err.Error()), text.WriteCellOpts(cell.FgColor(cell.ColorRed)))
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := widget.Write(fmt.Sprintf("%s\n", strings.ReplaceAll(article.Content, "\r", "")), text.WriteCellOpts(cell.FgColor(cell.ColorWhite))); err != nil {
|
||||||
|
widget.Write(fmt.Sprintf("Error: %s", err.Error()), text.WriteCellOpts(cell.FgColor(cell.ColorRed)))
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := widget.Write(fmt.Sprintf("%s\n\n", article.PublishedAt), text.WriteCellOpts(cell.FgColor(cell.ColorGray))); err != nil {
|
||||||
|
widget.Write(fmt.Sprintf("Error: %s", err.Error()), text.WriteCellOpts(cell.FgColor(cell.ColorRed)))
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := widget.Write(fmt.Sprintf("%s\n", article.URL)); err != nil {
|
||||||
|
widget.Write(fmt.Sprintf("Error: %s", err.Error()), text.WriteCellOpts(cell.FgColor(cell.ColorRed)))
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type NewsText struct {
|
||||||
|
*text.Text
|
||||||
|
Selected int
|
||||||
|
newsArticles news.News
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewNewsText(opts ...text.Option) (*NewsText, error) {
|
||||||
|
t, err := text.New(opts...)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &NewsText{Text: t}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *NewsText) Keyboard(k *terminalapi.Keyboard, _ *widgetapi.EventMeta) error {
|
||||||
|
if k.Key == keyboard.KeyArrowLeft {
|
||||||
|
if t.Selected == 0 {
|
||||||
|
t.Selected = len(t.newsArticles.Articles) - 1
|
||||||
|
} else {
|
||||||
|
t.Selected = t.Selected - 1
|
||||||
|
}
|
||||||
|
} else if k.Key == keyboard.KeyArrowRight {
|
||||||
|
t.Selected = (t.Selected + 1) % len(t.newsArticles.Articles)
|
||||||
|
}
|
||||||
|
return t.drawNews()
|
||||||
|
}
|
||||||
@ -27,6 +27,10 @@ func createWeather(ctx context.Context, _ terminalapi.Terminal, _ interface{}) w
|
|||||||
go util.Periodic(ctx, 1*time.Hour, func() error {
|
go util.Periodic(ctx, 1*time.Hour, func() error {
|
||||||
weather := openWeatherMap.FetchCurrentWeather()
|
weather := openWeatherMap.FetchCurrentWeather()
|
||||||
widget.Reset()
|
widget.Reset()
|
||||||
|
if weather.ErrorMessage != "" {
|
||||||
|
widget.Write(weather.ErrorMessage, text.WriteCellOpts(cell.FgColor(cell.ColorRed)))
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
weatherIcon := getIcon(weather.Weather[0].Icon)
|
weatherIcon := getIcon(weather.Weather[0].Icon)
|
||||||
|
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user