aboutsummaryrefslogtreecommitdiffstats
path: root/app/Issue/Sort.hs
blob: 1a0b3ffd4a581a91323319de8855c7510be9d61a (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
module Issue.Sort
  ( Sort,
    Order,
    SortBy,
    applySort,
    sortArg,
  )
where

import Data.List (sort, sortBy)
import Data.Maybe (mapMaybe)
import Data.Ord (comparing)
import Data.Text (Text)
import Data.Text qualified as T
import Issue (Issue (..))
import Issue.Tag (Tag (..))
import Options.Applicative qualified as O

data Sort = Sort Order SortBy
  deriving (Show)

data Order
  = Asc
  | Desc
  deriving (Show, Eq)

data SortBy
  = SortByTag Text
  deriving (Show)

defaultSort :: Sort
defaultSort = Sort Asc (SortByTag "@createdAt")

sortArg :: O.Parser [Sort]
sortArg =
  O.many
    ( O.option
        (O.maybeReader (parse . T.pack))
        ( O.long "sort"
            <> O.short 's'
            <> O.metavar "SORT"
            <> O.help "Sort selected issues. (Defaults: `-@createdAt`)"
        )
    )
  where
    parse s
      | "@" `T.isPrefixOf` s = Just (Sort Asc (SortByTag (T.drop 1 s)))
      | "-@" `T.isPrefixOf` s = Just (Sort Desc (SortByTag (T.drop 2 s)))
      | otherwise = Nothing

applySort :: [Sort] -> [Issue] -> [Issue]
applySort cs = compose (defaultSort : cs)
  where
    compose :: [Sort] -> ([Issue] -> [Issue])
    compose = foldr (.) id . map toSort

    toSort :: Sort -> ([Issue] -> [Issue])
    toSort (Sort order sortBy') =
      (if order == Desc then reverse else id)
        . sortBy (comparing (applySortBy sortBy'))

applySortBy :: SortBy -> Issue -> [Text]
applySortBy (SortByTag k) i =
  sort
    ( mapMaybe
        (\(Tag k' v) -> if k' == k then Just v else Nothing)
        (i.tags ++ i.internalTags)
    )