bug_table.go

  1package termui
  2
  3import (
  4	"bytes"
  5	"fmt"
  6
  7	"github.com/MichaelMure/git-bug/bug"
  8	"github.com/MichaelMure/git-bug/cache"
  9	"github.com/MichaelMure/git-bug/util"
 10	"github.com/dustin/go-humanize"
 11	"github.com/jroimartin/gocui"
 12)
 13
 14const bugTableView = "bugTableView"
 15const bugTableHeaderView = "bugTableHeaderView"
 16const bugTableFooterView = "bugTableFooterView"
 17const bugTableInstructionView = "bugTableInstructionView"
 18
 19const defaultRemote = "origin"
 20const defaultQuery = "status:open"
 21
 22type bugTable struct {
 23	repo         *cache.RepoCache
 24	queryStr     string
 25	query        *cache.Query
 26	allIds       []string
 27	bugs         []*cache.BugCache
 28	pageCursor   int
 29	selectCursor int
 30}
 31
 32func newBugTable(c *cache.RepoCache) *bugTable {
 33	query, err := cache.ParseQuery(defaultQuery)
 34	if err != nil {
 35		panic(err)
 36	}
 37
 38	return &bugTable{
 39		repo:         c,
 40		query:        query,
 41		queryStr:     defaultQuery,
 42		pageCursor:   0,
 43		selectCursor: 0,
 44	}
 45}
 46
 47func (bt *bugTable) layout(g *gocui.Gui) error {
 48	maxX, maxY := g.Size()
 49
 50	if maxY < 4 {
 51		// window too small !
 52		return nil
 53	}
 54
 55	v, err := g.SetView(bugTableHeaderView, -1, -1, maxX, 3)
 56
 57	if err != nil {
 58		if err != gocui.ErrUnknownView {
 59			return err
 60		}
 61
 62		v.Frame = false
 63	}
 64
 65	v.Clear()
 66	bt.renderHeader(v, maxX)
 67
 68	v, err = g.SetView(bugTableView, -1, 1, maxX, maxY-3)
 69
 70	if err != nil {
 71		if err != gocui.ErrUnknownView {
 72			return err
 73		}
 74
 75		v.Frame = false
 76		v.Highlight = true
 77		v.SelBgColor = gocui.ColorWhite
 78		v.SelFgColor = gocui.ColorBlack
 79
 80		// restore the cursor
 81		// window is too small to set the cursor properly, ignoring the error
 82		_ = v.SetCursor(0, bt.selectCursor)
 83	}
 84
 85	_, viewHeight := v.Size()
 86	err = bt.paginate(viewHeight)
 87	if err != nil {
 88		return err
 89	}
 90
 91	err = bt.cursorClamp(v)
 92	if err != nil {
 93		return err
 94	}
 95
 96	v.Clear()
 97	bt.render(v, maxX)
 98
 99	v, err = g.SetView(bugTableFooterView, -1, maxY-4, maxX, maxY)
100
101	if err != nil {
102		if err != gocui.ErrUnknownView {
103			return err
104		}
105
106		v.Frame = false
107	}
108
109	v.Clear()
110	bt.renderFooter(v, maxX)
111
112	v, err = g.SetView(bugTableInstructionView, -1, maxY-2, maxX, maxY)
113
114	if err != nil {
115		if err != gocui.ErrUnknownView {
116			return err
117		}
118
119		v.Frame = false
120		v.BgColor = gocui.ColorBlue
121
122		fmt.Fprintf(v, "[Esc] Quit [←↓↑→,hjkl] Navigation [enter] Open bug [n] New bug [i] Pull [o] Push")
123	}
124
125	_, err = g.SetCurrentView(bugTableView)
126	return err
127}
128
129func (bt *bugTable) keybindings(g *gocui.Gui) error {
130	// Quit
131	if err := g.SetKeybinding(bugTableView, gocui.KeyEsc, gocui.ModNone, quit); err != nil {
132		return err
133	}
134
135	// Down
136	if err := g.SetKeybinding(bugTableView, 'j', gocui.ModNone,
137		bt.cursorDown); err != nil {
138		return err
139	}
140	if err := g.SetKeybinding(bugTableView, gocui.KeyArrowDown, gocui.ModNone,
141		bt.cursorDown); err != nil {
142		return err
143	}
144	// Up
145	if err := g.SetKeybinding(bugTableView, 'k', gocui.ModNone,
146		bt.cursorUp); err != nil {
147		return err
148	}
149	if err := g.SetKeybinding(bugTableView, gocui.KeyArrowUp, gocui.ModNone,
150		bt.cursorUp); err != nil {
151		return err
152	}
153
154	// Previous page
155	if err := g.SetKeybinding(bugTableView, 'h', gocui.ModNone,
156		bt.previousPage); err != nil {
157		return err
158	}
159	if err := g.SetKeybinding(bugTableView, gocui.KeyArrowLeft, gocui.ModNone,
160		bt.previousPage); err != nil {
161		return err
162	}
163	if err := g.SetKeybinding(bugTableView, gocui.KeyPgup, gocui.ModNone,
164		bt.previousPage); err != nil {
165		return err
166	}
167	// Next page
168	if err := g.SetKeybinding(bugTableView, 'l', gocui.ModNone,
169		bt.nextPage); err != nil {
170		return err
171	}
172	if err := g.SetKeybinding(bugTableView, gocui.KeyArrowRight, gocui.ModNone,
173		bt.nextPage); err != nil {
174		return err
175	}
176	if err := g.SetKeybinding(bugTableView, gocui.KeyPgdn, gocui.ModNone,
177		bt.nextPage); err != nil {
178		return err
179	}
180
181	// New bug
182	if err := g.SetKeybinding(bugTableView, 'n', gocui.ModNone,
183		bt.newBug); err != nil {
184		return err
185	}
186
187	// Open bug
188	if err := g.SetKeybinding(bugTableView, gocui.KeyEnter, gocui.ModNone,
189		bt.openBug); err != nil {
190		return err
191	}
192
193	// Pull
194	if err := g.SetKeybinding(bugTableView, 'i', gocui.ModNone,
195		bt.pull); err != nil {
196		return err
197	}
198
199	// Push
200	if err := g.SetKeybinding(bugTableView, 'o', gocui.ModNone,
201		bt.push); err != nil {
202		return err
203	}
204
205	// Query
206	if err := g.SetKeybinding(bugTableView, 'q', gocui.ModNone,
207		bt.changeQuery); err != nil {
208		return err
209	}
210
211	return nil
212}
213
214func (bt *bugTable) disable(g *gocui.Gui) error {
215	if err := g.DeleteView(bugTableView); err != nil && err != gocui.ErrUnknownView {
216		return err
217	}
218	if err := g.DeleteView(bugTableHeaderView); err != nil && err != gocui.ErrUnknownView {
219		return err
220	}
221	if err := g.DeleteView(bugTableFooterView); err != nil && err != gocui.ErrUnknownView {
222		return err
223	}
224	if err := g.DeleteView(bugTableInstructionView); err != nil && err != gocui.ErrUnknownView {
225		return err
226	}
227	return nil
228}
229
230func (bt *bugTable) paginate(max int) error {
231	bt.allIds = bt.repo.QueryBugs(bt.query)
232
233	return bt.doPaginate(max)
234}
235
236func (bt *bugTable) doPaginate(max int) error {
237	// clamp the cursor
238	bt.pageCursor = maxInt(bt.pageCursor, 0)
239	bt.pageCursor = minInt(bt.pageCursor, len(bt.allIds))
240
241	nb := minInt(len(bt.allIds)-bt.pageCursor, max)
242
243	if nb < 0 {
244		bt.bugs = []*cache.BugCache{}
245		return nil
246	}
247
248	// slice the data
249	ids := bt.allIds[bt.pageCursor : bt.pageCursor+nb]
250
251	bt.bugs = make([]*cache.BugCache, len(ids))
252
253	for i, id := range ids {
254		b, err := bt.repo.ResolveBug(id)
255		if err != nil {
256			return err
257		}
258
259		bt.bugs[i] = b
260	}
261
262	return nil
263}
264
265func (bt *bugTable) getTableLength() int {
266	return len(bt.bugs)
267}
268
269func (bt *bugTable) getColumnWidths(maxX int) map[string]int {
270	m := make(map[string]int)
271	m["id"] = 10
272	m["status"] = 8
273
274	left := maxX - 5 - m["id"] - m["status"]
275
276	m["summary"] = maxInt(11, left/6)
277	left -= m["summary"]
278
279	m["lastEdit"] = maxInt(19, left/6)
280	left -= m["lastEdit"]
281
282	m["author"] = maxInt(left*2/5, 15)
283	m["title"] = maxInt(left-m["author"], 10)
284
285	return m
286}
287
288func (bt *bugTable) render(v *gocui.View, maxX int) {
289	columnWidths := bt.getColumnWidths(maxX)
290
291	for _, b := range bt.bugs {
292		person := bug.Person{}
293		snap := b.Snapshot()
294		if len(snap.Comments) > 0 {
295			create := snap.Comments[0]
296			person = create.Author
297		}
298
299		id := util.LeftPaddedString(snap.HumanId(), columnWidths["id"], 2)
300		status := util.LeftPaddedString(snap.Status.String(), columnWidths["status"], 2)
301		title := util.LeftPaddedString(snap.Title, columnWidths["title"], 2)
302		author := util.LeftPaddedString(person.Name, columnWidths["author"], 2)
303		summary := util.LeftPaddedString(snap.Summary(), columnWidths["summary"], 2)
304		lastEdit := util.LeftPaddedString(humanize.Time(snap.LastEditTime()), columnWidths["lastEdit"], 2)
305
306		fmt.Fprintf(v, "%s %s %s %s %s %s\n",
307			util.Cyan(id),
308			util.Yellow(status),
309			title,
310			util.Magenta(author),
311			summary,
312			lastEdit,
313		)
314	}
315}
316
317func (bt *bugTable) renderHeader(v *gocui.View, maxX int) {
318	columnWidths := bt.getColumnWidths(maxX)
319
320	id := util.LeftPaddedString("ID", columnWidths["id"], 2)
321	status := util.LeftPaddedString("STATUS", columnWidths["status"], 2)
322	title := util.LeftPaddedString("TITLE", columnWidths["title"], 2)
323	author := util.LeftPaddedString("AUTHOR", columnWidths["author"], 2)
324	summary := util.LeftPaddedString("SUMMARY", columnWidths["summary"], 2)
325	lastEdit := util.LeftPaddedString("LAST EDIT", columnWidths["lastEdit"], 2)
326
327	fmt.Fprintf(v, "\n")
328	fmt.Fprintf(v, "%s %s %s %s %s %s\n", id, status, title, author, summary, lastEdit)
329
330}
331
332func (bt *bugTable) renderFooter(v *gocui.View, maxX int) {
333	fmt.Fprintf(v, " \nShowing %d of %d bugs", len(bt.bugs), len(bt.allIds))
334}
335
336func (bt *bugTable) cursorDown(g *gocui.Gui, v *gocui.View) error {
337	_, y := v.Cursor()
338	y = minInt(y+1, bt.getTableLength()-1)
339
340	// window is too small to set the cursor properly, ignoring the error
341	_ = v.SetCursor(0, y)
342	bt.selectCursor = y
343
344	return nil
345}
346
347func (bt *bugTable) cursorUp(g *gocui.Gui, v *gocui.View) error {
348	_, y := v.Cursor()
349	y = maxInt(y-1, 0)
350
351	// window is too small to set the cursor properly, ignoring the error
352	_ = v.SetCursor(0, y)
353	bt.selectCursor = y
354
355	return nil
356}
357
358func (bt *bugTable) cursorClamp(v *gocui.View) error {
359	_, y := v.Cursor()
360
361	y = minInt(y, bt.getTableLength()-1)
362	y = maxInt(y, 0)
363
364	// window is too small to set the cursor properly, ignoring the error
365	_ = v.SetCursor(0, y)
366	bt.selectCursor = y
367
368	return nil
369}
370
371func (bt *bugTable) nextPage(g *gocui.Gui, v *gocui.View) error {
372	_, max := v.Size()
373
374	if bt.pageCursor+max >= len(bt.allIds) {
375		return nil
376	}
377
378	bt.pageCursor += max
379
380	return bt.doPaginate(max)
381}
382
383func (bt *bugTable) previousPage(g *gocui.Gui, v *gocui.View) error {
384	_, max := v.Size()
385
386	bt.pageCursor = maxInt(0, bt.pageCursor-max)
387
388	return bt.doPaginate(max)
389}
390
391func (bt *bugTable) newBug(g *gocui.Gui, v *gocui.View) error {
392	return newBugWithEditor(bt.repo)
393}
394
395func (bt *bugTable) openBug(g *gocui.Gui, v *gocui.View) error {
396	_, y := v.Cursor()
397	ui.showBug.SetBug(bt.bugs[y])
398	return ui.activateWindow(ui.showBug)
399}
400
401func (bt *bugTable) pull(g *gocui.Gui, v *gocui.View) error {
402	// Note: this is very hacky
403
404	ui.msgPopup.Activate("Pull from remote "+defaultRemote, "...")
405
406	go func() {
407		stdout, err := bt.repo.Fetch(defaultRemote)
408
409		if err != nil {
410			g.Update(func(gui *gocui.Gui) error {
411				ui.msgPopup.Activate(msgPopupErrorTitle, err.Error())
412				return nil
413			})
414		} else {
415			g.Update(func(gui *gocui.Gui) error {
416				ui.msgPopup.UpdateMessage(stdout)
417				return nil
418			})
419		}
420
421		var buffer bytes.Buffer
422		beginLine := ""
423
424		for merge := range bt.repo.MergeAll(defaultRemote) {
425			if merge.Status == bug.MsgMergeNothing {
426				continue
427			}
428
429			if merge.Err != nil {
430				g.Update(func(gui *gocui.Gui) error {
431					ui.msgPopup.Activate(msgPopupErrorTitle, err.Error())
432					return nil
433				})
434			} else {
435				fmt.Fprintf(&buffer, "%s%s: %s",
436					beginLine, util.Cyan(merge.Bug.HumanId()), merge.Status,
437				)
438
439				beginLine = "\n"
440
441				g.Update(func(gui *gocui.Gui) error {
442					ui.msgPopup.UpdateMessage(buffer.String())
443					return nil
444				})
445			}
446		}
447
448		fmt.Fprintf(&buffer, "%sdone", beginLine)
449
450		g.Update(func(gui *gocui.Gui) error {
451			ui.msgPopup.UpdateMessage(buffer.String())
452			return nil
453		})
454
455	}()
456
457	return nil
458}
459
460func (bt *bugTable) push(g *gocui.Gui, v *gocui.View) error {
461	ui.msgPopup.Activate("Push to remote "+defaultRemote, "...")
462
463	go func() {
464		// TODO: make the remote configurable
465		stdout, err := bt.repo.Push(defaultRemote)
466
467		if err != nil {
468			g.Update(func(gui *gocui.Gui) error {
469				ui.msgPopup.Activate(msgPopupErrorTitle, err.Error())
470				return nil
471			})
472		} else {
473			g.Update(func(gui *gocui.Gui) error {
474				ui.msgPopup.UpdateMessage(stdout)
475				return nil
476			})
477		}
478	}()
479
480	return nil
481}
482
483func (bt *bugTable) changeQuery(g *gocui.Gui, v *gocui.View) error {
484	return editQueryWithEditor(bt)
485}