Re-design output to make skimming easier.

- We now always protect the left margin when printing task details
  (including tags) to make it easier to skim down that line.
- Also made the actual summary always follow immediately after the ID,
  to align to that skimmable line.
- Moved the information about the delegatee to the end of the summary,
  next to the tags, and changed the color of the delegatee to make it
  easier to distinguish.
- Added the `-G` option, to allow filtering out issues matching any of
  the provided tags.
- We now allow options to be passed to both the `delegate` and `help`
  command. Any options are ignored, but this allows the use of tools
  like `cmd_shell` which always wrap commands with the pre-given
  options.
This commit is contained in:
Jonathan Bernard 2022-07-31 20:01:39 -05:00
parent c7891de310
commit 7215b4969b
4 changed files with 88 additions and 45 deletions

View File

@ -1,6 +1,6 @@
# Package # Package
version = "4.20.0" version = "4.21.0"
author = "Jonathan Bernard" author = "Jonathan Bernard"
description = "Personal issue tracker." description = "Personal issue tracker."
license = "MIT" license = "MIT"

View File

@ -1,8 +1,9 @@
## Personal Issue Tracker CLI interface ## Personal Issue Tracker CLI interface
## ==================================== ## ====================================
import algorithm, cliutils, data_uri, docopt, json, logging, options, os, import std/algorithm, std/logging, std/options, std/os, std/sequtils,
sequtils, std/wordwrap, tables, terminal, times, timeutils, unicode, uuids std/wordwrap, std/tables, std/terminal, std/times, std/unicode
import cliutils, data_uri, docopt, json, timeutils, uuids
from nre import re from nre import re
import strutils except alignLeft, capitalize, strip, toUpper, toLower import strutils except alignLeft, capitalize, strip, toUpper, toLower
@ -71,46 +72,68 @@ proc formatIssue(ctx: CliContext, issue: Issue): string =
result &= termReset result &= termReset
proc formatSectionIssue(ctx: CliContext, issue: Issue, width: int, indent = "", proc formatSectionIssue(
verbose = false): string = ctx: CliContext,
issue: Issue,
result = "" width: int,
indent = "",
var showDetails = not issue.details.isEmptyOrWhitespace and verbose verbose = false): string =
var prefixLen = 0
var summaryIndentLen = indent.len + 7
if issue.hasProp("delegated-to"): prefixLen += issue["delegated-to"].len + 2 # space for the ':' and ' '
# Wrap and write the summary.
var wrappedSummary = ("+".repeat(prefixLen) & issue.summary).wrapWords(width - summaryIndentLen).indent(summaryIndentLen)
wrappedSummary = wrappedSummary[(prefixLen + summaryIndentLen)..^1]
result = (indent & ($issue.id)[0..<6]).withColor(fgBlack, true) & " " result = (indent & ($issue.id)[0..<6]).withColor(fgBlack, true) & " "
if issue.hasProp("delegated-to"): let showDetails = not issue.details.isEmptyOrWhitespace and verbose
result &= (issue["delegated-to"] & ": ").withColor(fgGreen)
result &= wrappedSummary.withColor(fgWhite) let summaryIndentLen = indent.len + 7
let summaryWidth = width - summaryIndentLen
let summaryLines = issue.summary
.wrapWords(summaryWidth)
.splitLines
result &= summaryLines[0].withColor(fgWhite)
for line in summaryLines[1..^1]:
result &= "\p" & line.indent(summaryIndentLen)
var lastLineLen = summaryLines[^1].len
if issue.hasProp("delegated-to"):
if lastLineLen + issue["delegated-to"].len + 1 < summaryWidth:
result &= " " & issue["delegated-to"].withColor(fgMagenta)
lastLineLen += issue["delegated-to"].len + 1
else:
result &= "\p" & issue["delegated-to"]
.withColor(fgMagenta)
.indent(summaryIndentLen)
lastLineLen = issue["delegated-to"].len
if issue.tags.len > 0: if issue.tags.len > 0:
let tagsStr = "(" & issue.tags.join(", ") & ")" let tagsStrLines = ("(" & issue.tags.join(", ") & ")")
if (result.splitLines[^1].len + tagsStr.len + 1) > (width - 2): .wrapWords(summaryWidth)
result &= "\n" & indent .splitLines
result &= " " & tagsStr.withColor(fgGreen)
if tagsStrLines.len == 1 and
(lastLineLen + tagsStrLines[0].len + 1) < summaryWidth:
result &= " " & tagsStrLines[0].withColor(fgGreen)
lastLineLen += tagsStrLines[0].len + 1
else:
result &= "\p" & tagsStrLines
.mapIt(it.indent(summaryIndentLen))
.join("\p")
.withColor(fgGreen)
lastLineLen = tagsStrLines[^1].len
if issue.hasProp("pending"): if issue.hasProp("pending"):
let startIdx = "Pending: ".len result &= "\p" & ("Pending: " & issue["pending"])
var pendingText = issue["pending"].wrapWords(width - startIdx - summaryIndentLen) .wrapwords(summaryWidth)
.indent(startIdx) .withColor(fgCyan)
pendingText = ("Pending: " & pendingText[startIdx..^1]).indent(summaryIndentLen) .indent(summaryIndentLen)
result &= "\n" & pendingText.withColor(fgCyan)
if showDetails: if showDetails:
result &= "\n" & issue.details.strip.indent(indent.len + 2).withColor(fgCyan) result &= "\p" & issue.details
.strip
.withColor(fgBlack, bright = true)
.indent(summaryIndentLen)
result &= termReset result &= termReset
@ -306,6 +329,7 @@ when isMainModule:
var propertiesOption = none(TableRef[string,string]) var propertiesOption = none(TableRef[string,string])
var exclPropsOption = none(TableRef[string,seq[string]]) var exclPropsOption = none(TableRef[string,seq[string]])
var tagsOption = none(seq[string]) var tagsOption = none(seq[string])
var exclTagsOption = none(seq[string])
if args["--properties"] or args["--context"]: if args["--properties"] or args["--context"]:
@ -333,6 +357,9 @@ when isMainModule:
if args["--tags"]: tagsOption = some(($args["--tags"]).split(",").mapIt(it.strip)) if args["--tags"]: tagsOption = some(($args["--tags"]).split(",").mapIt(it.strip))
if args["--excl-tags"]: exclTagsOption =
some(($args["--excl-tags"]).split(",").mapIt(it.strip))
## Actual command runners ## Actual command runners
if args["new"] or args["add"]: if args["new"] or args["add"]:
let state = let state =
@ -350,7 +377,7 @@ when isMainModule:
summary: $args["<summary>"], summary: $args["<summary>"],
properties: issueProps, properties: issueProps,
tags: tags:
if args["--tags"]: ($args["--tags"]).split(",").mapIt(it.strip) if tagsOption.isSome: tagsOption.get
else: newSeq[string]()) else: newSeq[string]())
ctx.tasksDir.store(issue, state) ctx.tasksDir.store(issue, state)
@ -376,9 +403,9 @@ when isMainModule:
else: edit(ctx.tasksDir.loadIssueById(editRef)) else: edit(ctx.tasksDir.loadIssueById(editRef))
elif args["tag"]: elif args["tag"]:
if not args["--tags"]: raise newException(Exception, "no tags given") if tagsOption.isNone: raise newException(Exception, "no tags given")
let newTags = ($args["--tags"]).split(",").mapIt(it.strip) let newTags = tagsOption.get
for id in @(args["<id>"]): for id in @(args["<id>"]):
var issue = ctx.tasksDir.loadIssueById(id) var issue = ctx.tasksDir.loadIssueById(id)
@ -387,7 +414,7 @@ when isMainModule:
elif args["untag"]: elif args["untag"]:
let tagsToRemove: seq[string] = let tagsToRemove: seq[string] =
if args["--tags"]: ($args["--tags"]).split(",").mapIt(it.strip) if tagsOption.isSome: tagsOption.get
else: @[] else: @[]
for id in @(args["<id>"]): for id in @(args["<id>"]):
@ -498,8 +525,12 @@ when isMainModule:
filter.properties["context"] = ctx.defaultContext.get filter.properties["context"] = ctx.defaultContext.get
filterOption = some(filter) filterOption = some(filter)
if args["--tags"]: if tagsOption.isSome:
filter.hasTags = ($args["--tags"]).split(',') filter.hasTags = tagsOption.get
filterOption = some(filter)
if exclTagsOption.isSome:
filter.exclTags = exclTagsOption.get
filterOption = some(filter) filterOption = some(filter)
# Finally, if the "context" is "all", don't filter on context # Finally, if the "context" is "all", don't filter on context

View File

@ -1,4 +1,4 @@
const PIT_VERSION* = "4.20.0" const PIT_VERSION* = "4.21.0"
const USAGE* = """Usage: const USAGE* = """Usage:
pit ( new | add) <summary> [<state>] [options] pit ( new | add) <summary> [<state>] [options]
@ -9,12 +9,12 @@ const USAGE* = """Usage:
pit tag <id>... [options] pit tag <id>... [options]
pit untag <id>... [options] pit untag <id>... [options]
pit reorder <state> [options] pit reorder <state> [options]
pit delegate <id> <delegated-to> pit delegate <id> <delegated-to> [options]
pit hide-until <id> <date> [options] pit hide-until <id> <date> [options]
pit ( delete | rm ) <id>... [options] pit ( delete | rm ) <id>... [options]
pit add-binary-property <id> <propName> <propSource> [options] pit add-binary-property <id> <propName> <propSource> [options]
pit get-binary-property <id> <propName> <propDest> [options] pit get-binary-property <id> <propName> <propDest> [options]
pit help pit help [options]
Options: Options:
@ -38,7 +38,13 @@ Options:
separate names. For example: -C ctx1,ctx2 separate names. For example: -C ctx1,ctx2
Shorthand for '-P context:<ctx>' Shorthand for '-P context:<ctx>'
-g, --tags <tags> Specify tags for an issue. -g, --tags <tags> Specify tags for an issue. Tags are specified as a
comma-delimited list. For example: -g tag1,tag2
-G, --excl-tags <tags> When used with the list command, exclude issues
that contain any of the provided tags. Tags are
specified as a comma-delimited list.
For example: -G tag1,tag2
-T, --today Limit to today's issues. -T, --today Limit to today's issues.

View File

@ -1,5 +1,6 @@
import cliutils, docopt, json, logging, langutils, options, os, import std/json, std/logging, std/options, std/os, std/sequtils, std/strformat,
sequtils, strformat, strutils, tables, times, timeutils, uuids std/strutils, std/tables, std/times
import cliutils, docopt, langutils, timeutils, uuids
import nre except toSeq import nre except toSeq
@ -23,6 +24,7 @@ type
completedRange*: Option[tuple[b, e: DateTime]] completedRange*: Option[tuple[b, e: DateTime]]
fullMatch*, summaryMatch*: Option[Regex] fullMatch*, summaryMatch*: Option[Regex]
hasTags*: seq[string] hasTags*: seq[string]
exclTags*: seq[string]
properties*: TableRef[string, string] properties*: TableRef[string, string]
exclProperties*: TableRef[string, seq[string]] exclProperties*: TableRef[string, seq[string]]
@ -115,6 +117,7 @@ proc initFilter*(): IssueFilter =
fullMatch: none(Regex), fullMatch: none(Regex),
summaryMatch: none(Regex), summaryMatch: none(Regex),
hasTags: @[], hasTags: @[],
exclTags: @[],
properties: newTable[string, string](), properties: newTable[string, string](),
exclProperties: newTable[string,seq[string]]()) exclProperties: newTable[string,seq[string]]())
@ -372,6 +375,9 @@ proc filter*(issues: seq[Issue], filter: IssueFilter): seq[Issue] =
for tag in filter.hasTags: for tag in filter.hasTags:
result = result.filterIt(it.tags.find(tag) >= 0) result = result.filterIt(it.tags.find(tag) >= 0)
for exclTag in filter.exclTags:
result = result.filterIt(it.tags.find(exclTag) < 0)
### Configuration utilities ### Configuration utilities
proc loadConfig*(args: Table[string, Value] = initTable[string, Value]()): PitConfig = proc loadConfig*(args: Table[string, Value] = initTable[string, Value]()): PitConfig =
let pitrcLocations = @[ let pitrcLocations = @[