Task rename added. Closes #3

- Moved title of TaskDetail panel to separate component
- Refactored indentation, formatting
- Updated status bar
This commit is contained in:
Anis Ahmad
2020-07-11 07:48:16 +06:00
parent 1508edce6f
commit 533c28f8bc
7 changed files with 116 additions and 23 deletions

View File

@@ -95,6 +95,8 @@ In case writing in a text input (e,g, new project/task, due date), you have to `
| Task Detail | `d` | Set Due date |
| Task Detail | `↓`/`↑` | Scroll Up/Down the note editor |
| Task Detail | `e` | Activate note editor for modification |
| Task Detail | `v` | Edit task details in external editor (default `vim`) _Have a Known Bug<sup>4</sup>_ |
| Task Detail | `r` | Rename Task Title |
| Active Note Editor | `Esc` | Deactivate note editor and save content |
**Tips about using shortcuts efficiently:**
@@ -172,6 +174,6 @@ You may :thumbsup: issues if you want to increase priority of a feature.
1. In my Macbook Air, 1.6 GHz Dual-Core Intel Core i5, RAM: 8 GB 1600 MHz DDR3
2. Using [monakai](https://github.com/sickill/vim-monokai) color scheme for markdown syntax
3. Habitica is a free habit and productivity app that treats your real life like a game
4. Known Bug: Mouse events (click) don't work after getting back from Editor.
---
> "This is the Book about which there is no doubt, a guidance for those conscious of Allah" - [Al-Quran](http://quran.com)

View File

@@ -47,7 +47,7 @@ func main() {
taskRepo = repo.NewTaskRepository(db)
titleText := tview.NewTextView().SetText("[lime::b]Geek-life [::-]- Task Manager for geeks!").SetDynamicColors(true)
cloudStatus := tview.NewTextView().SetText("[::d]Version: 0.0.3").SetTextAlign(tview.AlignRight).SetDynamicColors(true)
cloudStatus := tview.NewTextView().SetText("[::d]Version: 0.1.0").SetTextAlign(tview.AlignRight).SetDynamicColors(true)
titleBar := tview.NewFlex().
AddItem(titleText, 0, 2, false).

View File

@@ -31,7 +31,7 @@ func makeStatusBar(app *tview.Application) *StatusBar {
tview.NewGrid(). // Content will not be modified, So, no need to declare explicitly
SetColumns(0, 0, 0, 0).
SetRows(0).
AddItem(tview.NewTextView().SetText("Navigate List: ↓/↑"), 0, 0, 1, 1, 0, 0, false).
AddItem(tview.NewTextView().SetText("Navigate List: ↓,↑ / j,k"), 0, 0, 1, 1, 0, 0, false).
AddItem(tview.NewTextView().SetText("New Task/Project: n").SetTextAlign(tview.AlignCenter), 0, 1, 1, 1, 0, 0, false).
AddItem(tview.NewTextView().SetText("Step back: Esc").SetTextAlign(tview.AlignCenter), 0, 2, 1, 1, 0, 0, false).
AddItem(tview.NewTextView().SetText("Quit: Ctrl+C").SetTextAlign(tview.AlignRight), 0, 3, 1, 1, 0, 0, false),

View File

@@ -23,21 +23,22 @@ const dateLayoutHuman = "02 Jan, Monday"
// TaskDetailPane displays detailed info of a Task
type TaskDetailPane struct {
*tview.Flex
taskName, taskDateDisplay *tview.TextView
editorHint *tview.TextView
taskDate *tview.InputField
taskStatusToggle *tview.Button
taskDetailView *femto.View
colorScheme femto.Colorscheme
taskRepo repository.TaskRepository
task *model.Task
header *TaskDetailHeader
taskDateDisplay *tview.TextView
editorHint *tview.TextView
taskDate *tview.InputField
taskStatusToggle *tview.Button
taskDetailView *femto.View
colorScheme femto.Colorscheme
taskRepo repository.TaskRepository
task *model.Task
}
// NewTaskDetailPane initializes and configures a TaskDetailPane
func NewTaskDetailPane(taskRepo repository.TaskRepository) *TaskDetailPane {
pane := TaskDetailPane{
Flex: tview.NewFlex().SetDirection(tview.FlexRow),
taskName: tview.NewTextView().SetDynamicColors(true),
header: NewTaskDetailHeader(taskRepo),
taskDateDisplay: tview.NewTextView().SetDynamicColors(true),
taskStatusToggle: makeButton("Complete", nil).SetLabelColor(tcell.ColorLightGray),
taskRepo: taskRepo,
@@ -61,8 +62,7 @@ func NewTaskDetailPane(taskRepo repository.TaskRepository) *TaskDetailPane {
SetTextColor(tcell.ColorDimGray), 0, 1, false)
pane.
AddItem(pane.taskName, 2, 1, true).
AddItem(makeHorizontalLine(tcell.RuneS3, tcell.ColorGray), 1, 1, false).
AddItem(pane.header, 4, 1, true).
AddItem(blankCell, 1, 1, false).
AddItem(pane.makeDateRow(), 1, 1, true).
AddItem(blankCell, 1, 1, false).
@@ -130,8 +130,7 @@ func (td *TaskDetailPane) toggleTaskStatus() {
status := !td.task.Completed
if taskRepo.UpdateField(td.task, "Completed", status) == nil {
td.task.Completed = status
td.SetTask(td.task)
taskPane.list.SetItemText(taskPane.list.GetCurrentItem(), makeTaskListingTitle(*td.task), "")
taskPane.ReloadCurrentTask()
}
}
@@ -299,12 +298,11 @@ func (td *TaskDetailPane) handleShortcuts(event *tcell.EventKey) *tcell.EventKey
td.editInExternalEditor()
case 'd':
app.SetFocus(td.taskDate)
case 'h':
app.SetFocus(taskPane)
case 'r':
td.header.ShowRename()
case ' ':
td.toggleTaskStatus()
}
}
return event
@@ -314,7 +312,7 @@ func (td *TaskDetailPane) handleShortcuts(event *tcell.EventKey) *tcell.EventKey
func (td *TaskDetailPane) SetTask(task *model.Task) {
td.task = task
td.taskName.SetText(fmt.Sprintf("[%s::b]# %s", getTaskTitleColor(*td.task), td.task.Title))
td.header.SetTask(task)
td.taskDetailView.Buf = makeBufferFromString(td.task.Details)
td.taskDetailView.SetColorscheme(td.colorScheme)
td.taskDetailView.Start()

88
app/task_title.go Normal file
View File

@@ -0,0 +1,88 @@
package main
import (
"fmt"
"github.com/gdamore/tcell"
"github.com/rivo/tview"
"github.com/ajaxray/geek-life/model"
"github.com/ajaxray/geek-life/repository"
)
// TaskDetailHeader displays Task title and relevant action in TaskDetail pane
type TaskDetailHeader struct {
*tview.Flex
pages *tview.Pages
taskName *tview.TextView
taskRepo repository.TaskRepository
task *model.Task
renameText *tview.InputField
}
// NewTaskDetailHeader initializes and configures a TaskDetailHeader
func NewTaskDetailHeader(taskRepo repository.TaskRepository) *TaskDetailHeader {
header := TaskDetailHeader{
Flex: tview.NewFlex().SetDirection(tview.FlexRow),
pages: tview.NewPages(),
taskName: tview.NewTextView().SetDynamicColors(true),
taskRepo: taskRepo,
renameText: makeLightTextInput("Task title"),
}
header.pages.AddPage("title", header.taskName, true, true)
header.pages.AddPage("rename", header.renameText, true, false)
header.bindRenameEvent()
buttons := tview.NewFlex().
AddItem(tview.NewTextView().SetTextColor(tcell.ColorDimGray).SetText("r = Rename"), 0, 1, false).
AddItem(blankCell, 0, 1, false).
AddItem(makeButton("rename", func() { header.ShowRename() }), 8, 0, false)
header.
AddItem(header.pages, 1, 1, true).
AddItem(blankCell, 1, 0, true).
AddItem(buttons, 1, 1, false).
AddItem(makeHorizontalLine(tcell.RuneS3, tcell.ColorGray), 1, 1, false)
return &header
}
func (header *TaskDetailHeader) bindRenameEvent() *tview.InputField {
return header.renameText.SetDoneFunc(func(key tcell.Key) {
switch key {
case tcell.KeyEnter:
name := header.renameText.GetText()
if len(name) < 3 {
statusBar.showForSeconds("[red::]Task title should be at least 3 character", 5)
return
}
if err := header.taskRepo.UpdateField(header.task, "Title", name); err != nil {
statusBar.showForSeconds("Could not update Task Title: "+err.Error(), 5)
} else {
header.task.Title = name
statusBar.showForSeconds("[yellow::]Task Title Updated.", 5)
}
header.pages.SwitchToPage("title")
taskPane.ReloadCurrentTask()
case tcell.KeyEsc:
header.pages.SwitchToPage("title")
}
})
}
// SetTask set a task to the header
func (header *TaskDetailHeader) SetTask(task *model.Task) {
header.task = task
header.taskName.SetText(task.Title)
header.taskName.SetText(fmt.Sprintf("[%s::b]# %s", getTaskTitleColor(*task), task.Title))
}
// ShowRename activate edit option of task title
func (header *TaskDetailHeader) ShowRename() {
header.renameText.SetText(header.task.Title)
header.pages.SwitchToPage("rename")
app.SetFocus(header.renameText)
}

View File

@@ -63,7 +63,6 @@ func NewTaskPane(projectRepo repository.ProjectRepository, taskRepo repository.T
case tcell.KeyEsc:
app.SetFocus(pane)
}
})
pane.
@@ -196,6 +195,12 @@ func (pane *TaskPane) ClearCompletedTasks() {
statusBar.showForSeconds(fmt.Sprintf("[yellow]%d tasks cleared!", count), 5)
}
// ReloadCurrentTask Loads the current task - in Task details and listing
func (pane *TaskPane) ReloadCurrentTask() {
pane.list.SetItemText(pane.list.GetCurrentItem(), makeTaskListingTitle(*pane.activeTask), "")
taskDetailPane.SetTask(pane.activeTask)
}
func (pane TaskPane) setHintMessage() {
if len(projectPane.projects) == 0 {
pane.hint.SetText("Welcome to the organized life!\n------------------------------\n Create TaskList/Project at the bottom of Projects pane.\n (Press p,n) \n\nHelp - https://bit.ly/cli-task")

View File

@@ -97,11 +97,11 @@ func makeTaskListingTitle(task model.Task) string {
prefix := ""
if projectPane.GetActiveProject() == nil {
if project, err := projectRepo.GetByID(task.ProjectID); err == nil {
prefix = project.Title + ":"
prefix = project.Title + ": "
}
}
return fmt.Sprintf("[%s] %s %s %s", getTaskTitleColor(task), prefix, checkbox, task.Title)
return fmt.Sprintf("[%s]%s %s%s", getTaskTitleColor(task), checkbox, prefix, task.Title)
}
func findProjectByID(id int64) *model.Project {