{-# OPTIONS -fno-warn-name-shadowing #-}
{-# LANGUAGE NamedFieldPuns #-}
{-# LANGUAGE PatternGuards #-}
{-# LANGUAGE TupleSections #-}
module Language.Haskell.Names.RecordWildcards
( patWcNames
, wcFieldName
, WcNames
, expWcNames
, wcFieldOrigName
) where
import qualified Language.Haskell.Names.GlobalSymbolTable as Global
import qualified Language.Haskell.Names.LocalSymbolTable as Local
import Language.Haskell.Names.SyntaxUtils
import Language.Haskell.Names.Types
import Control.Monad
import qualified Data.Map as Map
import Data.Maybe
import qualified Data.Set as Set
import Language.Haskell.Exts
type WcNames = [WcField]
data WcField = WcField
{ wcFieldName :: Name ()
, wcFieldOrigName :: OrigName
, wcExistsGlobalValue :: Bool
}
getElidedFields
:: Global.Table
-> QName l
-> [Name l]
-> WcNames
getElidedFields gt con fields =
let
givenFieldNames :: Map.Map (Name ()) ()
givenFieldNames =
Map.fromList . map ((, ()) . void) $ fields
(mbConOrigName, mbTypeOrigName) =
case Global.lookupValue con gt of
Global.Result info@SymConstructor{} ->
(Just $ sv_origName info, Just $ sv_typeName info)
_ -> (Nothing, Nothing)
allValueInfos :: Set.Set (SymValueInfo OrigName)
allValueInfos = Set.unions $ Map.elems $ Global.values gt
ourFieldInfos :: Set.Set (SymValueInfo OrigName)
ourFieldInfos =
case mbConOrigName of
Nothing -> Set.empty
Just conOrigName ->
flip Set.filter allValueInfos $ \v ->
case v of
SymSelector { sv_constructors }
| conOrigName `elem` sv_constructors -> True
_ -> False
existsGlobalValue :: Name () -> Bool
existsGlobalValue name =
case Global.lookupValue (UnQual () name) gt of
Global.Result info
| Just typeOrigName <- mbTypeOrigName
, SymSelector {} <- info
, sv_typeName info == typeOrigName
-> False
| otherwise -> True
_ -> False
ourFieldNames :: Map.Map (Name ()) WcField
ourFieldNames =
Map.fromList $
map
(
(\orig ->
let name = stringToName . gName . origGName $ orig in
(name, ) $
WcField
{ wcFieldName = name
, wcFieldOrigName = orig
, wcExistsGlobalValue = existsGlobalValue name
}
) . sv_origName
)
$ Set.toList ourFieldInfos
in Map.elems $ ourFieldNames `Map.difference` givenFieldNames
nameOfPatField :: PatField l -> Maybe (Name l)
nameOfPatField pf =
case pf of
PFieldPat _ qn _ -> Just $ qNameToName qn
PFieldPun _ qn -> Just $ qNameToName qn
PFieldWildcard {} -> Nothing
nameOfUpdField :: FieldUpdate l -> Maybe (Name l)
nameOfUpdField pf =
case pf of
FieldUpdate _ qn _ -> Just $ qNameToName qn
FieldPun _ qn -> Just $ qNameToName qn
FieldWildcard {} -> Nothing
patWcNames
:: Global.Table
-> QName l
-> [PatField l]
-> WcNames
patWcNames gt con patfs =
getElidedFields gt con $
mapMaybe nameOfPatField patfs
expWcNames
:: Global.Table
-> Local.Table
-> QName l
-> [FieldUpdate l]
-> WcNames
expWcNames gt lt con patfs =
filter isInScope $
getElidedFields gt con $
mapMaybe nameOfUpdField patfs
where
isInScope field
| Right {} <- Local.lookupValue qn lt = True
| otherwise = wcExistsGlobalValue field
where
qn = UnQual () $ wcFieldName field