You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
cointop/cointop/navigation.go

606 lines
13 KiB
Go

package cointop
import (
"math"
log "github.com/sirupsen/logrus"
)
// CurrentPage returns the current page
func (ct *Cointop) CurrentPage() int {
log.Debug("CurrentPage()")
return ct.State.page + 1
}
// CurrentDisplayPage returns the current page in human readable format
func (ct *Cointop) CurrentDisplayPage() int {
log.Debug("CurrentDisplayPage()")
return ct.State.page + 1
}
// TotalPages returns the number of total pages
func (ct *Cointop) TotalPages() int {
log.Debug("TotalPages()")
return ct.GetListCount() / ct.State.perPage
}
// TotalPagesDisplay returns the number of total pages in human readable format
func (ct *Cointop) TotalPagesDisplay() int {
log.Debug("TotalPagesDisplay()")
return ct.TotalPages() + 1
}
// TotalPerPage returns the number max rows per page
func (ct *Cointop) TotalPerPage() int {
return ct.State.perPage
}
// SetPage navigates to the selected page
func (ct *Cointop) SetPage(page int) int {
log.Debug("SetPage()")
if (page*ct.State.perPage) < ct.GetListCount() && page >= 0 {
ct.State.page = page
}
return ct.State.page
}
// CursorDown moves the cursor one row down
func (ct *Cointop) CursorDown() error {
log.Debug("CursorDown()")
// return if already at the bottom
if ct.IsLastRow() {
return nil
}
cx, cy := ct.Views.Table.Cursor()
y := cy + 1
if err := ct.Views.Table.SetCursor(cx, y); err != nil {
return err
}
ox, oy := ct.Views.Table.Origin()
h := ct.Views.Table.Height()
if y < 0 || y >= h {
// set origin scrolls
if err := ct.Views.Table.SetOrigin(ox, oy+1); err != nil {
return err
}
}
ct.RowChanged()
return nil
}
// CursorUp moves the cursor one row up
func (ct *Cointop) CursorUp() error {
log.Debug("CursorUp()")
// return if already at the top
if ct.IsFirstRow() {
return nil
}
ox, oy := ct.Views.Table.Origin()
cx, cy := ct.Views.Table.Cursor()
y := cy - 1
if err := ct.Views.Table.SetCursor(cx, y); err != nil {
return err
}
h := ct.Views.Table.Height()
if y < 0 || y >= h {
// set origin scrolls
if err := ct.Views.Table.SetOrigin(ox, oy-1); err != nil {
return err
}
}
ct.RowChanged()
return nil
}
// PageDown moves the cursor one page down
func (ct *Cointop) PageDown() error {
log.Debug("PageDown()")
// return if already at the bottom
if ct.IsLastRow() {
return nil
}
ox, oy := ct.Views.Table.Origin() // this is prev origin position
cx := ct.Views.Table.CursorX() // relative cursor position
sy := ct.Views.Table.Height() // rows in visible view
y := oy + sy
l := ct.TableRowsLen()
// end of table
if (oy + sy + sy) > l {
y = l - sy
}
// select last row if next jump is out of bounds
if y < 0 {
y = 0
sy = l
}
if err := ct.Views.Table.SetOrigin(ox, y); err != nil {
return err
}
// move cursor to last line if can't scroll further
if y == oy {
if err := ct.Views.Table.SetCursor(cx, sy-1); err != nil {
return err
}
}
ct.RowChanged()
return nil
}
// PageUp moves the cursor one page up
func (ct *Cointop) PageUp() error {
log.Debug("PageUp()")
// return if already at the top
if ct.IsFirstRow() {
return nil
}
ox, oy := ct.Views.Table.Origin()
cx := ct.Views.Table.CursorX() // relative cursor position
sy := ct.Views.Table.Height() // rows in visible view
k := oy - sy
if k < 0 {
k = 0
}
if err := ct.Views.Table.SetOrigin(ox, k); err != nil {
return err
}
// move cursor to first line if can't scroll further
if k == oy {
if err := ct.Views.Table.SetCursor(cx, 0); err != nil {
return err
}
}
ct.RowChanged()
return nil
}
// NavigateFirstLine moves the cursor to the first row of the table
func (ct *Cointop) NavigateFirstLine() error {
log.Debug("NavigateFirstLine()")
// return if already at the top
if ct.IsFirstRow() {
return nil
}
ox := ct.Views.Table.OriginX()
cx := ct.Views.Table.CursorX()
if err := ct.Views.Table.SetOrigin(ox, 0); err != nil {
return err
}
if err := ct.Views.Table.SetCursor(cx, 0); err != nil {
return err
}
ct.RowChanged()
return nil
}
// NavigateLastLine moves the cursor to the last row of the table
func (ct *Cointop) NavigateLastLine() error {
log.Debug("NavigateLastLine()")
// return if already at the bottom
if ct.IsLastRow() {
return nil
}
ox := ct.Views.Table.OriginX()
cx := ct.Views.Table.CursorX()
l := ct.TableRowsLen()
h := ct.Views.Table.Height()
h = int(math.Min(float64(h), float64(l)))
k := l - h
if k < 0 {
k = l
}
y := h - 1
if err := ct.Views.Table.SetCursor(cx, y); err != nil {
return err
}
// set origin scrolls
if err := ct.Views.Table.SetOrigin(ox, k); err != nil {
return err
}
ct.RowChanged()
return nil
}
// NavigatePageFirstLine moves the cursor to the visible first row of the table
func (ct *Cointop) NavigatePageFirstLine() error {
log.Debug("NavigatePageFirstLine()")
// return if already at the correct line
if ct.IsPageFirstLine() {
return nil
}
cx := ct.Views.Table.CursorX()
if err := ct.Views.Table.SetCursor(cx, 0); err != nil {
return err
}
ct.RowChanged()
return nil
}
// NavigatePageMiddleLine moves the cursor to the visible middle row of the table
func (ct *Cointop) NavigatePageMiddleLine() error {
log.Debug("NavigatePageMiddleLine()")
// return if already at the correct line
if ct.IsPageMiddleLine() {
return nil
}
cx := ct.Views.Table.CursorX()
sy := ct.Views.Table.Height()
if err := ct.Views.Table.SetCursor(cx, (sy/2)-1); err != nil {
return err
}
ct.RowChanged()
return nil
}
// NavigatePageLastLine moves the cursor to the visible last row of the table
func (ct *Cointop) navigatePageLastLine() error {
log.Debug("NavigatePageLastLine()")
// return if already at the correct line
if ct.IsPageLastLine() {
return nil
}
cx, _ := ct.Views.Table.Cursor()
sy := ct.Views.Table.Height()
if err := ct.Views.Table.SetCursor(cx, sy-1); err != nil {
return err
}
ct.RowChanged()
return nil
}
// NextPage navigates to the next page
func (ct *Cointop) NextPage() error {
log.Debug("NextPage()")
// return if already at the last page
if ct.IsLastPage() {
return nil
}
ct.SetPage(ct.State.page + 1)
ct.UpdateTable()
ct.RowChanged()
return nil
}
// PrevPage navigates to the previous page
func (ct *Cointop) PrevPage() error {
log.Debug("PrevPage()")
// return if already at the first page
if ct.IsFirstPage() {
return nil
}
ct.SetPage(ct.State.page - 1)
ct.UpdateTable()
ct.RowChanged()
return nil
}
// NextPageTop navigates to the first row of the next page
func (ct *Cointop) nextPageTop() error {
log.Debug("NextPageTop()")
ct.NextPage()
ct.NavigateFirstLine()
return nil
}
// PrevPageTop navigates to the first row of the previous page
func (ct *Cointop) PrevPageTop() error {
log.Debug("PrevtPageTop()")
ct.PrevPage()
ct.NavigateLastLine()
return nil
}
// FirstPage navigates to the first page
func (ct *Cointop) FirstPage() error {
log.Debug("FirstPage()")
// return if already at the first page
if ct.IsFirstPage() {
return nil
}
ct.State.page = 0
ct.UpdateTable()
ct.RowChanged()
return nil
}
// LastPage navigates to the last page
func (ct *Cointop) LastPage() error {
log.Debug("LastPage()")
// return if already at the last page
if ct.IsLastPage() {
return nil
}
ct.State.page = ct.GetListCount() / ct.State.perPage
ct.UpdateTable()
ct.RowChanged()
return nil
}
// IsFirstRow returns true if cursor is on first row
func (ct *Cointop) IsFirstRow() bool {
log.Debug("IsFirstRow()")
oy := ct.Views.Table.OriginY()
cy := ct.Views.Table.CursorY()
return (cy + oy) == 0
}
// IsLastRow returns true if cursor is on last row
func (ct *Cointop) IsLastRow() bool {
log.Debug("IsLastRow()")
oy := ct.Views.Table.OriginY()
cy := ct.Views.Table.CursorY()
numRows := ct.TableRowsLen() - 1
return (cy + oy + 1) > numRows
}
// IsFirstPage returns true if cursor is on the first page
func (ct *Cointop) IsFirstPage() bool {
log.Debug("IsFirstPage()")
return ct.State.page == 0
}
// IsLastPage returns true if cursor is on the last page
func (ct *Cointop) IsLastPage() bool {
log.Debug("IsLastPage()")
return ct.State.page == ct.TotalPages()-1
}
// IsPageFirstLine returns true if the cursor is on the visible first row
func (ct *Cointop) IsPageFirstLine() bool {
log.Debug("IsPageFirstLine()")
cy := ct.Views.Table.CursorY()
return cy == 0
}
// IsPageMiddleLine returns true if the cursor is on the visible middle row
func (ct *Cointop) IsPageMiddleLine() bool {
log.Debug("IsPageMiddleLine()")
cy := ct.Views.Table.CursorY()
sy := ct.Views.Table.Height()
return (sy/2)-1 == cy
}
// IsPageLastLine returns true if the cursor is on the visible last row
func (ct *Cointop) IsPageLastLine() bool {
log.Debug("IsPageLastLine()")
cy := ct.Views.Table.CursorY()
sy := ct.Views.Table.Height()
return cy+1 == sy
}
// GoToPageRowIndex navigates to the selected row index of the page
func (ct *Cointop) GoToPageRowIndex(idx int) error {
log.Debug("GoToPageRowIndex()")
if idx < 0 {
idx = 0
}
cx := ct.Views.Table.CursorX()
if err := ct.Views.Table.SetCursor(cx, idx); err != nil {
return err
}
ct.RowChanged()
return nil
}
// GoToGlobalIndex navigates to the selected row index of all page rows
func (ct *Cointop) GoToGlobalIndex(idx int) error {
log.Debug("GoToGlobalIndex()")
l := ct.TableRowsLen()
atpage := idx / l
ct.SetPage(atpage)
rowIndex := (idx % l)
ct.HighlightRow(rowIndex)
ct.UpdateTable()
return nil
}
// HighlightRow highlights the row at index within page
func (ct *Cointop) HighlightRow(pageRowIndex int) error {
if pageRowIndex < 0 {
pageRowIndex = 0
}
log.Debug("HighlightRow()")
ct.Views.Table.SetOrigin(0, 0)
ct.Views.Table.SetCursor(0, 0)
ox := ct.Views.Table.OriginX()
cx := ct.Views.Table.CursorX()
l := ct.TableRowsLen()
h := ct.Views.Table.Height()
h = int(math.Min(float64(h), float64(l)))
oy := 0
cy := 0
if h > 0 {
cy = pageRowIndex % h
oy = pageRowIndex - cy
// end of page
if pageRowIndex >= l-h {
oy = l - h
cy = h - (l - pageRowIndex)
}
}
log.Debugf("HighlightRow idx:%v h:%v cy:%v oy:%v", pageRowIndex, h, cy, oy)
ct.Views.Table.SetOrigin(ox, oy)
ct.Views.Table.SetCursor(cx, cy)
return nil
}
// GoToCoinRow navigates to the row of the matched coin
func (ct *Cointop) GoToCoinRow(coin *Coin) error {
log.Debug("GoToCoinRow()")
if coin == nil {
return nil
}
idx := ct.GetCoinRowIndex(coin)
return ct.GoToGlobalIndex(idx)
}
// GetGlobalCoinIndex returns the index of the coin in from the gloal coins list
func (ct *Cointop) GetGlobalCoinIndex(coin *Coin) int {
var idx int
for i, v := range ct.State.allCoins {
if v == coin {
idx = i
break
}
}
return idx
}
// GetCoinRowIndex returns the index of the coin in from the visible coins list
func (ct *Cointop) GetCoinRowIndex(coin *Coin) int {
var idx int
for i, v := range ct.State.coins {
if v == coin {
idx = i
break
}
}
return idx
}
// CursorDownOrNextPage moves the cursor down one row or goes to the next page if cursor is on the last row
func (ct *Cointop) CursorDownOrNextPage() error {
log.Debug("CursorDownOrNextPage()")
if ct.IsLastRow() {
if ct.IsLastPage() {
return nil
}
if err := ct.nextPageTop(); err != nil {
return err
}
return nil
}
if err := ct.CursorDown(); err != nil {
return err
}
return nil
}
// CursorUpOrPreviousPage moves the cursor up one row or goes to the previous page if cursor is on the first row
func (ct *Cointop) CursorUpOrPreviousPage() error {
log.Debug("CursorUpOrPreviousPage()")
if ct.IsFirstRow() {
if ct.IsFirstPage() {
return nil
}
if err := ct.PrevPageTop(); err != nil {
return err
}
return nil
}
if err := ct.CursorUp(); err != nil {
return err
}
return nil
}
// TableScrollLeft scrolls the table to the left
func (ct *Cointop) TableScrollLeft() error {
ct.State.tableOffsetX++
if ct.State.tableOffsetX >= 0 {
ct.State.tableOffsetX = 0
}
ct.UpdateTable()
return nil
}
// TableScrollRight scrolls the table to the right
func (ct *Cointop) TableScrollRight() error {
ct.State.tableOffsetX--
maxX := int(math.Min(float64(1-(ct.maxTableWidth-ct.Width())), 0))
if ct.State.tableOffsetX <= maxX {
ct.State.tableOffsetX = maxX
}
ct.UpdateTable()
return nil
}
// MouseRelease is called on mouse releae event
func (ct *Cointop) MouseRelease() error {
return nil
}
// MouseLeftClick is called on mouse left click event
func (ct *Cointop) MouseLeftClick() error {
return nil
}
// MouseMiddleClick is called on mouse middle click event
func (ct *Cointop) MouseMiddleClick() error {
return nil
}
// MouseRightClick is called on mouse right click event
func (ct *Cointop) MouseRightClick() error {
return ct.OpenLink()
}
// MouseWheelUp is called on mouse wheel up event
func (ct *Cointop) MouseWheelUp() error {
return nil
}
// MouseWheelDown is called on mouse wheel down event
func (ct *Cointop) MouseWheelDown() error {
return nil
}
// TableRowsLen returns the number of table row entries
func (ct *Cointop) TableRowsLen() int {
log.Debug("TableRowsLen()")
if ct.IsFavoritesVisible() {
return ct.FavoritesLen()
}
if ct.IsPortfolioVisible() {
return ct.PortfolioLen()
}
if ct.IsPriceAlertsVisible() {
return ct.ActivePriceAlertsLen()
}
return ct.TableCoinsLen()
}
// GetActiveTableSlice returns the rows slice for the active table
func (ct *Cointop) GetActiveTableSlice() []*Coin {
if ct.IsFavoritesVisible() {
return ct.GetFavoritesSlice()
}
if ct.IsPortfolioVisible() {
return ct.GetPortfolioSlice()
}
return ct.GetTableCoinsSlice()
}