----------------------------------------------------------------------------
-- |
-- Module      :  Emacs.Module
-- Copyright   :  (c) Sergey Vinokurov 2018
-- License     :  Apache-2.0 (see LICENSE)
-- Maintainer  :  serg.foo@gmail.com
--
-- This module is the entry point for writing Emacs extensions in
-- Haskell.
--
-- This package, though provides a lot of wrapping around Emacs's bare
-- C interface, still presumes some familiarity with said interface.
-- Thus, when developnig Emacs modules it's recommended to keep a
-- reference of the C interface around. One such reference is
-- <https://phst.github.io/emacs-modules.html>.
--
-- = Minimalistic example
--
-- Consider Emacs function
--
-- > (defun foo (f x y z &optional w t &rest quux)
-- >   (+ (funcall f (* x y z)) (* (or w 1) (or t 2)) (length quux)))
--
-- With help of this package, it may be defined as
--
--
-- > {-# LANGUAGE DataKinds   #-}
-- > {-# LANGUAGE QuasiQuotes #-}
-- >
-- > import Data.Maybe
-- > import Data.Emacs.Module.SymbolName.TH
-- > import Emacs.Module
--
-- > foo
-- >   :: MonadEmacs m v
-- >   => EmacsFunction ('S ('S ('S ('S 'Z)))) ('S ('S 'Z)) 'True m v s
-- > foo (R f (R x (R y (R z (O w (O t (Rest quux))))))) = do
-- >   x'    <- extractInt x
-- >   y'    <- extractInt y
-- >   z'    <- extractInt z
-- >   w'    <- traverse extractInt w
-- >   t'    <- traverse extractInt t
-- >
-- >   tmp   <- makeInt (x' * y' * z')
-- >   tmp'  <- extractInt =<< funcallSym "funcall" [f, tmp]
-- >
-- >   produceRef =<< makeInt (tmp' + fromMaybe 1 w' * fromMaybe 2 t' + length quux)
--
-- = Creating Emacs dynamic module
-- In order to make shared object or dll callable from Emacs,
-- a cabal project with foreign-library section has to be created.
-- Please refer to <https://github.com/sergv/emacs-module/tree/master/test>
-- for such a project.
--
-- Please note that this project will need a small C file for initialising
-- Haskell runtime. In the project mentioned before it's present as
-- <https://github.com/sergv/emacs-module/blob/master/test/cbits/emacs_wrapper.c>
----------------------------------------------------------------------------

module Emacs.Module
  (
    -- * Basic bindings
    MonadEmacs(..)

    -- ** Define functions callable by Emacs
  , EmacsFunction
  , Nat(..)
  , R(..)
  , O(..)
  , Rest(..)
  , Stop(..)

    -- ** Error types
  , EmacsError(..)
  , EmacsInternalError(..)
  , reportAllErrorsToEmacs

    -- * Reexports
  , module Emacs.Module.Functions
  , module Data.Emacs.Module.Value
  , Env

    -- * Third-party reexports
  , MonadThrow(..)
  ) where

import Control.Monad.Catch (MonadThrow(..))

import Data.Emacs.Module.Args
import Data.Emacs.Module.Env (Env)
import Data.Emacs.Module.Value
import Emacs.Module.Errors
import Emacs.Module.Functions
import Emacs.Module.Monad.Class