diff --git a/TODO.md b/TODO.md index 48cfcd8..666e974 100644 --- a/TODO.md +++ b/TODO.md @@ -1,3 +1 @@ -* Allow the user to use the `-g` and `-G` flags to filter tasks to sum or list - by tags (currently only date and id ranges are supported). * Sync with web timestamper. diff --git a/ptk.nim b/ptk.nim index 085c311..71573d3 100644 --- a/ptk.nim +++ b/ptk.nim @@ -3,8 +3,9 @@ ## ## Simple time keeping CLI -import algorithm, docopt, json, langutils, logging, os, sequtils, strutils, - tempfile, terminal, times, timeutils, uuids +import algorithm, docopt, json, langutils, logging, os, nre, sequtils, + strutils, tempfile, terminal, times, timeutils, uuids + import ptkutil type @@ -15,7 +16,7 @@ const STOP_MSG = "STOP" let NO_MARK: Mark = ( id: parseUUID("00000000-0000-0000-0000-000000000000"), - time: getLocalTime(getTime()), + time: fromSeconds(0).getLocalTime, summary: "", notes: "", tags: @[]) const ISO_TIME_FORMAT = "yyyy:MM:dd'T'HH:mm:ss" @@ -67,6 +68,9 @@ proc loadTimeline(filename: string): Timeline = notes: markJson["notes"].getStr(), tags: markJson["tags"].getElems(@[]).map(proc (t: JsonNode): string = t.getStr()))) + timeline.marks = timeline.marks.sorted( + proc(a, b: Mark): int = cmp(a.time, b.time)) + return timeline proc saveTimeline(timeline: Timeline, location: string): void = @@ -86,54 +90,63 @@ proc flexFormat(i: TimeInterval): string = return i.format(fmt) -proc writeMarks(marks: seq[Mark], includeNotes = false): void = +type WriteData = tuple[idx: int, mark: Mark, prefixLen: int, interval: TimeInterval] + +proc writeMarks(timeline: Timeline, indices: seq[int], includeNotes = false): void = + let marks = timeline.marks let now = getLocalTime(getTime()) + var idxs = indices.sorted( + proc(a, b: int): int = cmp(marks[a].time, marks[b].time)) + let timeFormat = - if now - marks.first.time > 1.years: "yyyy-MM-dd HH:mm" - elif now - marks.first.time > 7.days: "MMM dd HH:mm" - elif now - marks.first.time > 1.days: "ddd HH:mm" + if now - marks[idxs.first].time > 1.years: "yyyy-MM-dd HH:mm" + elif now - marks[idxs.first].time > 7.days: "MMM dd HH:mm" + elif now - marks[idxs.first].time > 1.days: "ddd HH:mm" else: "HH:mm" - var intervals: seq[TimeInterval] = @[] - for i in 0.. longestPrefix: longestPrefix = prefix.len - for i in 0.. 0: + if w.mark.tags.len > 0: setForegroundColor(stdout, fgGreen) - write(stdout, " (" & mark.tags.join(", ") & ")") + write(stdout, " (" & w.mark.tags.join(", ") & ")") resetAttributes(stdout) writeLine(stdout, "") - if includeNotes and len(mark.notes.strip) > 0: - writeLine(stdout, spaces(longestPrefix) & mark.notes) + if includeNotes and len(w.mark.notes.strip) > 0: + writeLine(stdout, spaces(longestPrefix) & w.mark.notes) writeLine(stdout, "") proc formatMark(mark: Mark, nextMark = NO_MARK, timeFormat = ISO_TIME_FORMAT, includeNotes = false): string = @@ -215,6 +228,46 @@ proc edit(mark: var Mark): void = finally: close(tempFile) +proc filterMarkIndices(timeline: Timeline, args: Table[string, Value]): seq[int] = + let marks = timeline.marks + result = sequtils.toSeq(0.."]: + let idx = marks.findById($args[""]) + if idx > 0: result = result.filterIt(it >= idx) + + if args[""]: + let idx = marks.findById($args[""]) + if (idx > 0): result = result.filterIt(it <= idx) + + if args["--after"]: + var startTime: TimeInfo + try: startTime = parseTime($args["--after"]) + except: raise newException(ValueError, + "invalid value for --after: " & getCurrentExceptionMsg()) + result = result.filterIt(marks[it].time > startTime) + + if args["--before"]: + var endTime: TimeInfo + try: endTime = parseTime($args["--before"]) + except: raise newException(ValueError, + "invalid value for --before: " & getCurrentExceptionMsg()) + result = result.filterIt(marks[it].time < endTime) + + if args["--tags"]: + let tags = (args["--tags"] ?: "").split({',', ';'}) + result = result.filter(proc (i: int): bool = + tags.allIt(marks[i].tags.contains(it))) + + if args["--remove-tags"]: + let tags = (args["--remove-tags"] ?: "").split({',', ';'}) + result = result.filter(proc (i: int): bool = + not tags.allIt(marks[i].tags.contains(it))) + + if args["--matching"]: + let pattern = re(args["--matching"] ?: "") + result = result.filterIt(marks[it].summary.find(pattern).isSome) + when isMainModule: try: let doc = """ @@ -244,6 +297,7 @@ Options: -g --tags Add the given tags (comma-separated) to the selected marks. -G --remove-tags Remove the given tag from the selected marks. -h --help Print this usage information. + -m --matching Restric the selection to marks matching . -n --notes For add and amend, set the notes for a time mark. -t --time