162 lines
3.2 KiB
Go
162 lines
3.2 KiB
Go
package format
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"ewintr.nl/gte/internal/task"
|
|
)
|
|
|
|
var (
|
|
ErrFieldAlreadyUsed = errors.New("field was already used")
|
|
)
|
|
|
|
type Column int
|
|
|
|
const (
|
|
COL_ID Column = iota
|
|
COL_STATUS
|
|
COL_DUE
|
|
COL_ACTION
|
|
COL_PROJECT
|
|
)
|
|
|
|
var (
|
|
COL_ALL = []Column{COL_ID, COL_STATUS, COL_DUE, COL_ACTION, COL_PROJECT}
|
|
)
|
|
|
|
func FormatError(err error) string {
|
|
return fmt.Sprintf("could not perform command.\n\nerror: %s\n", err.Error())
|
|
}
|
|
|
|
func FormatTaskTable(tasks []*task.LocalTask, cols []Column) string {
|
|
if len(tasks) == 0 {
|
|
return "no tasks to display\n"
|
|
}
|
|
|
|
var data [][]string
|
|
for _, t := range tasks {
|
|
var line []string
|
|
for _, col := range cols {
|
|
switch col {
|
|
case COL_ID:
|
|
line = append(line, fmt.Sprintf("%d", t.LocalId))
|
|
case COL_STATUS:
|
|
var updated []string
|
|
if t.IsRecurrer() {
|
|
updated = append(updated, "r")
|
|
}
|
|
if t.LocalStatus == task.STATUS_UPDATED {
|
|
updated = append(updated, "u")
|
|
}
|
|
if !t.Due.IsZero() && task.Today().After(t.Due) {
|
|
updated = append(updated, "o")
|
|
}
|
|
line = append(line, strings.Join(updated, " "))
|
|
case COL_DUE:
|
|
if t.Due.IsZero() {
|
|
line = append(line, "")
|
|
continue
|
|
}
|
|
line = append(line, t.Due.Human())
|
|
case COL_ACTION:
|
|
line = append(line, t.Action)
|
|
case COL_PROJECT:
|
|
line = append(line, t.Project)
|
|
}
|
|
}
|
|
data = append(data, line)
|
|
}
|
|
|
|
return fmt.Sprintf("\n%s\n", FormatTable(data))
|
|
}
|
|
|
|
func FormatTask(t *task.LocalTask) string {
|
|
output := fmt.Sprintf(`folder: %s
|
|
action: %s
|
|
project: %s
|
|
due: %s
|
|
`, t.Folder, t.Action, t.Project, t.Due.String())
|
|
if t.IsRecurrer() {
|
|
output += fmt.Sprintf("recur:%s", t.Recur.String())
|
|
}
|
|
|
|
return fmt.Sprintf("%s\n", output)
|
|
}
|
|
|
|
func ParseTaskFieldArgs(args []string) (*task.LocalUpdate, error) {
|
|
lu := &task.LocalUpdate{}
|
|
|
|
action, fields := []string{}, []string{}
|
|
for _, f := range args {
|
|
split := strings.SplitN(f, ":", 2)
|
|
if len(split) == 2 {
|
|
switch split[0] {
|
|
case "project":
|
|
if lu.Project != "" {
|
|
return &task.LocalUpdate{}, fmt.Errorf("%w: %s", ErrFieldAlreadyUsed, task.FIELD_PROJECT)
|
|
}
|
|
lu.Project = split[1]
|
|
fields = append(fields, task.FIELD_PROJECT)
|
|
case "due":
|
|
if !lu.Due.IsZero() {
|
|
return &task.LocalUpdate{}, fmt.Errorf("%w: %s", ErrFieldAlreadyUsed, task.FIELD_DUE)
|
|
}
|
|
lu.Due = task.NewDateFromString(split[1])
|
|
fields = append(fields, task.FIELD_DUE)
|
|
}
|
|
} else {
|
|
if len(f) > 0 {
|
|
action = append(action, f)
|
|
}
|
|
}
|
|
}
|
|
|
|
if len(action) > 0 {
|
|
lu.Action = strings.Join(action, " ")
|
|
fields = append(fields, task.FIELD_ACTION)
|
|
}
|
|
|
|
lu.Fields = fields
|
|
|
|
return lu, nil
|
|
}
|
|
|
|
func FormatTable(data [][]string) string {
|
|
if len(data) == 0 {
|
|
return ""
|
|
}
|
|
max := make([]int, len(data[0]))
|
|
for _, line := range data {
|
|
for i, col := range line {
|
|
if len(col) > max[i] {
|
|
max[i] = len(col)
|
|
}
|
|
}
|
|
}
|
|
|
|
var output string
|
|
for r, line := range data {
|
|
if r%3 == 0 {
|
|
output += fmt.Sprintf("%s", "\x1b[48;5;237m")
|
|
}
|
|
for c, col := range line {
|
|
output += col
|
|
for s := 0; s < max[c]-len(col); s++ {
|
|
output += " "
|
|
}
|
|
if c != len(line)-1 {
|
|
output += " "
|
|
}
|
|
}
|
|
if r%3 == 0 {
|
|
output += fmt.Sprintf("%s", "\x1b[49m")
|
|
}
|
|
output += "\n"
|
|
|
|
}
|
|
|
|
return output
|
|
}
|