CurryInfo: stylechecker-2.0.0 / Check.AST.Indent.Let

classes:

              
documentation:

              
name:
Check.AST.Indent.Let
operations:
checkExprIndent checkLet checkLet' checkLetDeclarations declarationIndent twoLines
sourcecode:
module Check.AST.Indent.Let where

import Control.Monad  ( unless )
import Curry.SpanInfo
import Curry.Span
import Curry.Position
import Curry.Types
import Text.Pretty

import Types

-- applies actual check on let constructs
checkLet :: Expression a -> Int -> CSM ()
checkLet e i =
  case e of
    (Let sI _ dcls expr) -> checkLet' sI dcls expr i
    _                    -> return ()

-- Checks indention of in expression first,
-- then checks if 
--  (a) `let` and `in` are in one line.
--  (b) they are aligned (indented) properly and the `let` declarations are aligned.
checkLet' :: SpanInfo -> [Decl a] -> Expression a -> Int -> CSM ()
checkLet' si (dcl:dcls) expr i = case si of 
  (SpanInfo
            (Span (Position l1 c1) (Position l2 c2))
            [Span (Position _  cl) _, 
             Span (Position li ci) _]
          ) -> do
    checkExprIndent expr li ci i
    unless (l1 == l2) $
      do
        twoLines (Span (Position l1 c1) (Position l2 c2)) cl ci
        if checkLetDeclarations (getCol (getSpanInfo dcl)) dcls
          then declarationIndent (Span (Position l1 c1) (Position l2 c2)) dcl i
          else report (Message
                        (Span (Position l1 c1) (Position l2 c2))
                        ( colorizeKey "let"
                          <+> text "declarations not aligned"
                        )
                        ( text "align"
                          <+> colorizeKey "let"
                          <+> text "declarations")
                      )
  _ -> return ()
checkLet' _ [] _ _ = return ()

-- Returns true if list of declarations are aligned with inputed position of first decl.
checkLetDeclarations :: Int -> [Decl a] -> Bool
checkLetDeclarations p (dcl:dcls@(_:_)) =
  (p == getCol (getSpanInfo dcl)) && checkLetDeclarations (getCol (getSpanInfo dcl)) dcls
checkLetDeclarations p [dcl]            = (p == getCol (getSpanInfo dcl))
checkLetDeclarations _ []               = True

-- Compares column positions of `let` and `in`.
twoLines :: Span -> Int -> Int -> CSM ()
twoLines sp x y =
  unless (x == y) $
    report (Message
              sp
              ( colorizeKey "let"
                <+> text "and"
                <+> colorizeKey "in"
                <+> text "not aligned"
              )
              ( text "align"
                <+> colorizeKey "let"
                <+> text "and"
                <+> colorizeKey "in"
              )
            )

-- `let` declarations should either be indented by 2 from let
-- or indentation edge.
declarationIndent :: Span -> Decl a -> Int -> CSM ()
declarationIndent sp dcl i =
  let sI = (getSpanInfo dcl)
      l = getLi (SpanInfo sp [])
      c = getCol (SpanInfo sp [])
  in
    unless ((getLi sI) == l)
      (unless ((getCol sI) == (c+2)||(getCol sI == (i+2))) $
        report (Message
                  sp
                  ( colorizeKey "let"
                    <+> text "declarations not properly indented"
                  )
                  ( text "indent by 2 from"
                    <+> colorizeKey "let"
                    <+> text "or"
                    <+> colorizeKey "outer block"
                  )
                )
      )

-- Checks if in expression is on the same line as `in`, if not, check if indented
-- by 2 from edge or `in`.
checkExprIndent :: Expression a -> Int -> Int -> Int -> CSM ()
checkExprIndent expr li ci i =
  do let sI = getSpanInfo expr
     unless (li == (getLi sI))
             (unless (((ci+2) == (getCol sI))||((i+2)==(getCol sI)))
                      (report (Message
                                (getSpan sI)
                                ( colorizeKey "in Expression"
                                  <+> text "not properly indented"
                                )
                                ( colorizeKey "in Expression"
                                  <+> text "should be indented by 2 from"
                                  <+> colorizeKey "in"
                                  <+> text "or"
                                  <+> colorizeKey "outer block"
                                )
                              )
                      )
             )
types:

              
unsafe:
safe