Compare commits
3 Commits
dda2fc15b2
...
8abe1d1bc2
| Author | SHA1 | Date | |
|---|---|---|---|
| 8abe1d1bc2 | |||
| 9dd9313829 | |||
| fa850d5017 |
139
agents.md
139
agents.md
@ -1,115 +1,48 @@
|
|||||||
# Claude Code Agents
|
# Docster — Project Guide
|
||||||
|
|
||||||
This project uses specialized Claude Code agents for different types of Haskell refactoring. Each agent has focused expertise to provide targeted improvements.
|
Docster is a Haskell CLI tool: Markdown + embedded Mermaid diagrams → PDF, HTML, or DOCX.
|
||||||
|
|
||||||
## Available Agents
|
## Quick Commands
|
||||||
|
|
||||||
### haskell-refactoring-expert
|
|
||||||
**Purpose**: Basic code quality and structural improvements
|
|
||||||
|
|
||||||
**Expertise**:
|
|
||||||
- Type consistency (String vs Text vs ByteString)
|
|
||||||
- Module organization and file splitting (>150 lines)
|
|
||||||
- Naming conventions and clarity
|
|
||||||
- Dependency management
|
|
||||||
- Basic code structure improvements
|
|
||||||
|
|
||||||
**When to use**:
|
|
||||||
- Inconsistent type usage across the codebase
|
|
||||||
- Large files that need module organization
|
|
||||||
- Poor naming or unclear function responsibilities
|
|
||||||
- Mixed concerns in single modules
|
|
||||||
|
|
||||||
**Example**: Converting a 300-line Main.hs into proper module hierarchy
|
|
||||||
|
|
||||||
### haskell-higher-order
|
|
||||||
**Purpose**: Advanced functional programming patterns and architectural refactoring
|
|
||||||
|
|
||||||
**Expertise**:
|
|
||||||
- Monad transformer patterns (ExceptT, ReaderT, StateT)
|
|
||||||
- Pipeline composition with monadic operators
|
|
||||||
- Higher-order abstractions and strategy patterns
|
|
||||||
- Effect management and pure/IO separation
|
|
||||||
- Functional design patterns
|
|
||||||
|
|
||||||
**When to use**:
|
|
||||||
- Nested case statements handling Either values in IO
|
|
||||||
- Duplicated functions that differ only in specific steps
|
|
||||||
- Manual threading of configuration or state
|
|
||||||
- Imperative-style code that could be more functional
|
|
||||||
- Complex error handling that needs cleanup
|
|
||||||
|
|
||||||
**Example**: Converting nested Either/IO handling to ExceptT pipelines
|
|
||||||
|
|
||||||
## Agent Boundaries and Trade-offs
|
|
||||||
|
|
||||||
### Complementary Design
|
|
||||||
These agents are designed to work **sequentially**:
|
|
||||||
1. **First pass**: `haskell-refactoring-expert` for structural cleanup
|
|
||||||
2. **Second pass**: `haskell-higher-order` for functional patterns
|
|
||||||
|
|
||||||
### Why Separate Agents?
|
|
||||||
|
|
||||||
**Benefits**:
|
|
||||||
- **Focused expertise**: Each agent has deep knowledge in its domain
|
|
||||||
- **Clear boundaries**: Easy to know which agent to use
|
|
||||||
- **Manageable complexity**: Avoids instruction bloat in single agent
|
|
||||||
- **Progressive enhancement**: Apply increasingly sophisticated refactoring
|
|
||||||
- **Composability**: Can run both agents or just one as needed
|
|
||||||
|
|
||||||
**Trade-offs**:
|
|
||||||
- **Coordination overhead**: Need to run multiple agents
|
|
||||||
- **Context switching**: Each agent analyzes code independently
|
|
||||||
- **Potential overlap**: Some patterns might fit both agents
|
|
||||||
|
|
||||||
### Decision Framework
|
|
||||||
|
|
||||||
**Use haskell-refactoring-expert when you have**:
|
|
||||||
- ❌ Mixed String/Text types
|
|
||||||
- ❌ Large monolithic files (>150 lines)
|
|
||||||
- ❌ Unclear naming or responsibilities
|
|
||||||
- ❌ Basic structural issues
|
|
||||||
|
|
||||||
**Use haskell-higher-order when you have**:
|
|
||||||
- ❌ Nested error handling (Either in IO)
|
|
||||||
- ❌ Duplicated function structures
|
|
||||||
- ❌ Manual state/config threading
|
|
||||||
- ❌ Imperative-style patterns
|
|
||||||
|
|
||||||
**Use both agents when**:
|
|
||||||
- ❌ You want comprehensive refactoring
|
|
||||||
- ❌ Code has both structural and architectural issues
|
|
||||||
- ❌ You're doing major codebase improvements
|
|
||||||
|
|
||||||
## Usage Patterns
|
|
||||||
|
|
||||||
### Sequential Refactoring
|
|
||||||
```bash
|
```bash
|
||||||
# Run basic refactoring first
|
stack build # build
|
||||||
/agent haskell-refactoring-expert "Please refactor the Main.hs file"
|
stack test # run tests
|
||||||
|
stack exec docster -- -pdf file.md # convert to PDF
|
||||||
# Then apply advanced patterns
|
stack exec docster -- -html file.md # convert to HTML
|
||||||
/agent haskell-higher-order "Please improve the error handling patterns"
|
stack exec docster -- -docx file.md # convert to DOCX
|
||||||
|
stack exec docster -- -pdf sample.md # test with a single file
|
||||||
|
stack clean # clean build artifacts
|
||||||
```
|
```
|
||||||
|
|
||||||
### Targeted Improvements
|
## Structure
|
||||||
```bash
|
|
||||||
# Just structural cleanup
|
|
||||||
/agent haskell-refactoring-expert "Split this large module"
|
|
||||||
|
|
||||||
# Just functional patterns
|
```
|
||||||
/agent haskell-higher-order "Convert these nested cases to monadic style"
|
docster.cabal # package definition
|
||||||
|
stack.yaml # GHC 9.12.2, lts-24.34
|
||||||
|
app/Main.hs # everything — entry point + all logic (~70 lines)
|
||||||
|
test/ # HSpec tests (TransformSpec.hs)
|
||||||
```
|
```
|
||||||
|
|
||||||
## Evolution Strategy
|
## How It Works
|
||||||
|
|
||||||
These agents can evolve independently:
|
1. Parse Markdown via Pandoc AST
|
||||||
- **haskell-refactoring-expert**: Add more structural patterns, linting rules
|
2. Walk the AST, find Mermaid code blocks
|
||||||
- **haskell-higher-order**: Add more advanced patterns (free monads, effect systems)
|
3. Run `mmdc` (mermaid-cli) to render each block → SVG (for HTML) or high-res PNG (for PDF)
|
||||||
|
4. Replace code blocks with image references in the AST
|
||||||
|
5. Compile final output via Pandoc (LaTeX/XeLaTeX for PDF, native for HTML/DOCX)
|
||||||
|
|
||||||
New specialized agents could be added:
|
Key functions in `Main.hs`:
|
||||||
- **haskell-performance**: Optimization-focused refactoring
|
- `transformDoc` — AST walker
|
||||||
- **haskell-testing**: Test-driven refactoring and property-based testing
|
- `processMermaidBlock` — calls `mmdc`, returns image reference
|
||||||
- **haskell-domain**: Domain modeling and type design
|
- `compileToPDF` / `compileToHTML` / `compileToDOCX` — final Pandoc compilation
|
||||||
|
|
||||||
The key is maintaining clear boundaries and complementary functionality.
|
## Dependencies
|
||||||
|
|
||||||
|
**System**: TeX Live (for PDF), `npm install -g @mermaid-js/mermaid-cli`
|
||||||
|
**Haskell**: Pandoc library, Stack manages GHC automatically
|
||||||
|
|
||||||
|
## Common Gotchas
|
||||||
|
|
||||||
|
- **Text vs String**: Codebase mixes `Data.Text` and `String`. Use `T.pack`/`T.unpack` for conversions.
|
||||||
|
- **PDF needs LaTeX**: BasicTeX/TinyTeX + `tlmgr` for missing packages.
|
||||||
|
- **mmdc in PATH**: `mermaid-cli` must be globally installed and on PATH.
|
||||||
|
|||||||
@ -4,7 +4,7 @@
|
|||||||
module Main (main) where
|
module Main (main) where
|
||||||
|
|
||||||
import Docster.Types (DocsterError(..))
|
import Docster.Types (DocsterError(..))
|
||||||
import Docster.Compiler (compileToPDF, compileToHTML)
|
import Docster.Compiler (compileToPDF, compileToHTML, compileToDOCX)
|
||||||
import System.Environment (getArgs)
|
import System.Environment (getArgs)
|
||||||
import Control.Exception (throwIO)
|
import Control.Exception (throwIO)
|
||||||
|
|
||||||
@ -12,7 +12,8 @@ import Control.Exception (throwIO)
|
|||||||
parseArgs :: [String] -> Either DocsterError (IO ())
|
parseArgs :: [String] -> Either DocsterError (IO ())
|
||||||
parseArgs ["-pdf", path] = Right (compileToPDF path)
|
parseArgs ["-pdf", path] = Right (compileToPDF path)
|
||||||
parseArgs ["-html", path] = Right (compileToHTML path)
|
parseArgs ["-html", path] = Right (compileToHTML path)
|
||||||
parseArgs _ = Left $ InvalidUsage "Usage: docster -pdf|-html <file.md>"
|
parseArgs ["-docx", path] = Right (compileToDOCX path)
|
||||||
|
parseArgs _ = Left $ InvalidUsage "Usage: docster -pdf|-html|-docx <file.md>"
|
||||||
|
|
||||||
-- | Main entry point - parse arguments and execute appropriate action
|
-- | Main entry point - parse arguments and execute appropriate action
|
||||||
main :: IO ()
|
main :: IO ()
|
||||||
|
|||||||
@ -6,12 +6,15 @@ module Docster.Compiler
|
|||||||
( -- * Compilation Functions
|
( -- * Compilation Functions
|
||||||
compileToPDF
|
compileToPDF
|
||||||
, compileToHTML
|
, compileToHTML
|
||||||
|
, compileToDOCX
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import Docster.Types
|
import Docster.Types
|
||||||
( DocsterError(..), OutputFormat(..), SourceDir(..), OutputDir(..), OutputPath(..)
|
( DocsterError(..), OutputFormat(..), SourceDir(..), OutputDir(..), OutputPath(..)
|
||||||
, DiagramConfig(..), computeOutputDir, ensureOutputDir
|
, DiagramConfig(..), computeOutputDir, ensureOutputDir
|
||||||
)
|
)
|
||||||
|
import Text.Pandoc.Writers ()
|
||||||
|
import qualified Data.ByteString.Lazy as BSL
|
||||||
import Docster.Transform (transformDocument)
|
import Docster.Transform (transformDocument)
|
||||||
import Docster.LaTeX (latexTemplate)
|
import Docster.LaTeX (latexTemplate)
|
||||||
import Text.Pandoc
|
import Text.Pandoc
|
||||||
@ -30,6 +33,7 @@ import Control.Monad.Trans.Reader (ReaderT, runReaderT, asks)
|
|||||||
import Control.Monad.Trans.Class (lift)
|
import Control.Monad.Trans.Class (lift)
|
||||||
import Control.Monad.IO.Class (liftIO)
|
import Control.Monad.IO.Class (liftIO)
|
||||||
import Data.Maybe (mapMaybe)
|
import Data.Maybe (mapMaybe)
|
||||||
|
import Data.Char (ord)
|
||||||
|
|
||||||
-- | Success indicator for user feedback
|
-- | Success indicator for user feedback
|
||||||
successEmoji :: Text
|
successEmoji :: Text
|
||||||
@ -38,13 +42,12 @@ successEmoji = "✅"
|
|||||||
-- | Compilation context for pipeline operations
|
-- | Compilation context for pipeline operations
|
||||||
data CompilationContext = CompilationContext
|
data CompilationContext = CompilationContext
|
||||||
{ ccStrategy :: CompilationStrategy
|
{ ccStrategy :: CompilationStrategy
|
||||||
, ccSourceDir :: SourceDir
|
|
||||||
, ccOutputDir :: OutputDir
|
|
||||||
, ccInputPath :: FilePath
|
, ccInputPath :: FilePath
|
||||||
, ccOutputPath :: FilePath
|
, ccOutputPath :: FilePath
|
||||||
, ccDocName :: Text
|
, ccDocName :: Text
|
||||||
, ccReaderOptions :: ReaderOptions
|
, ccReaderOptions :: ReaderOptions
|
||||||
, ccConfig :: DiagramConfig
|
, ccConfig :: DiagramConfig
|
||||||
|
, ccWritesFile :: Bool
|
||||||
}
|
}
|
||||||
|
|
||||||
-- | Monad stack for compilation pipeline
|
-- | Monad stack for compilation pipeline
|
||||||
@ -54,12 +57,14 @@ type CompilationM = ReaderT CompilationContext (ExceptT DocsterError IO)
|
|||||||
data CompilationStrategy = CompilationStrategy
|
data CompilationStrategy = CompilationStrategy
|
||||||
{ -- | Format for diagram configuration
|
{ -- | Format for diagram configuration
|
||||||
csOutputFormat :: OutputFormat
|
csOutputFormat :: OutputFormat
|
||||||
-- | Pandoc writer function
|
-- | Pandoc writer function (returns Text for HTML/PDF, unused for DOCX)
|
||||||
, csWriter :: WriterOptions -> Pandoc -> PandocIO Text
|
, csWriter :: WriterOptions -> Pandoc -> PandocIO Text
|
||||||
-- | Post-processing function for the generated content
|
-- | Post-processing function for the generated content
|
||||||
, csProcessOutput :: String -> Text -> IO (Either DocsterError ())
|
, csProcessOutput :: String -> Text -> IO (Either DocsterError ())
|
||||||
-- | Success message formatter
|
-- | Success message formatter
|
||||||
, csSuccessMessage :: String -> Text
|
, csSuccessMessage :: String -> Text
|
||||||
|
-- | True for formats where writer writes a file directly (DOCX)
|
||||||
|
, csWritesFile :: Bool
|
||||||
}
|
}
|
||||||
|
|
||||||
-- | PDF compilation strategy
|
-- | PDF compilation strategy
|
||||||
@ -69,6 +74,7 @@ pdfStrategy = CompilationStrategy
|
|||||||
, csWriter = writeLaTeX
|
, csWriter = writeLaTeX
|
||||||
, csProcessOutput = processPDFOutput
|
, csProcessOutput = processPDFOutput
|
||||||
, csSuccessMessage = \path -> successEmoji <> " PDF written to " <> T.pack path
|
, csSuccessMessage = \path -> successEmoji <> " PDF written to " <> T.pack path
|
||||||
|
, csWritesFile = False
|
||||||
}
|
}
|
||||||
|
|
||||||
-- | HTML compilation strategy
|
-- | HTML compilation strategy
|
||||||
@ -78,6 +84,17 @@ htmlStrategy = CompilationStrategy
|
|||||||
, csWriter = writeHtml5String
|
, csWriter = writeHtml5String
|
||||||
, csProcessOutput = processHTMLOutput
|
, csProcessOutput = processHTMLOutput
|
||||||
, csSuccessMessage = \path -> successEmoji <> " HTML written to " <> T.pack path
|
, csSuccessMessage = \path -> successEmoji <> " HTML written to " <> T.pack path
|
||||||
|
, csWritesFile = False
|
||||||
|
}
|
||||||
|
|
||||||
|
-- | DOCX compilation strategy (Pandoc writes file directly)
|
||||||
|
docxStrategy :: CompilationStrategy
|
||||||
|
docxStrategy = CompilationStrategy
|
||||||
|
{ csOutputFormat = DOCX
|
||||||
|
, csWriter = \_ _ -> return "" -- unused: writeDocx writes file directly
|
||||||
|
, csProcessOutput = \_ _ -> return $ Right () -- no post-processing needed
|
||||||
|
, csSuccessMessage = \path -> successEmoji <> " DOCX written to " <> T.pack path
|
||||||
|
, csWritesFile = True
|
||||||
}
|
}
|
||||||
|
|
||||||
-- | Parse LaTeX log content to extract meaningful error messages
|
-- | Parse LaTeX log content to extract meaningful error messages
|
||||||
@ -212,11 +229,30 @@ liftEitherM action = do
|
|||||||
Left err -> lift $ throwE err
|
Left err -> lift $ throwE err
|
||||||
Right value -> return value
|
Right value -> return value
|
||||||
|
|
||||||
|
-- | Strip ANSI escape sequences (CSI codes like color/style) from text.
|
||||||
|
-- These appear in copy-pasted terminal output and break LaTeX compilation.
|
||||||
|
stripAnsiCodes :: Text -> Text
|
||||||
|
stripAnsiCodes input = case T.break (== '\x1b') input of
|
||||||
|
(before, rest)
|
||||||
|
| T.null rest -> before
|
||||||
|
| otherwise -> before <> stripAnsiCodes (skipEscape (T.tail rest))
|
||||||
|
where
|
||||||
|
-- Skip an ESC sequence: ESC [ <params> <final byte>
|
||||||
|
skipEscape t
|
||||||
|
| T.null t = t
|
||||||
|
| T.head t == '[' = skipCSIParams (T.tail t)
|
||||||
|
| otherwise = T.tail t -- non-CSI escape: skip one char after ESC
|
||||||
|
-- Skip CSI parameter/intermediate bytes until final byte (0x40-0x7E)
|
||||||
|
skipCSIParams t
|
||||||
|
| T.null t = t
|
||||||
|
| let c = ord (T.head t), c >= 0x40 && c <= 0x7E = T.tail t -- final byte, consume it
|
||||||
|
| otherwise = skipCSIParams (T.tail t)
|
||||||
|
|
||||||
-- | Pipeline step: Read content from input file
|
-- | Pipeline step: Read content from input file
|
||||||
readContent :: CompilationM Text
|
readContent :: CompilationM Text
|
||||||
readContent = do
|
readContent = do
|
||||||
inputPath <- asks ccInputPath
|
inputPath <- asks ccInputPath
|
||||||
liftIO $ TIO.readFile inputPath
|
liftIO $ stripAnsiCodes <$> TIO.readFile inputPath
|
||||||
|
|
||||||
-- | Pipeline step: Parse markdown content into Pandoc AST
|
-- | Pipeline step: Parse markdown content into Pandoc AST
|
||||||
parseDocument :: Text -> CompilationM Pandoc
|
parseDocument :: Text -> CompilationM Pandoc
|
||||||
@ -235,14 +271,24 @@ transformDocumentM pandoc = do
|
|||||||
generateOutputM :: Pandoc -> CompilationM Text
|
generateOutputM :: Pandoc -> CompilationM Text
|
||||||
generateOutputM pandoc = do
|
generateOutputM pandoc = do
|
||||||
strategy <- asks ccStrategy
|
strategy <- asks ccStrategy
|
||||||
liftEitherM $ generateOutput strategy pandoc
|
writesFile <- asks ccWritesFile
|
||||||
|
if writesFile
|
||||||
|
then do
|
||||||
|
outputPath <- asks ccOutputPath
|
||||||
|
_ <- liftIO $ generateOutputFile strategy outputPath pandoc
|
||||||
|
return "" -- placeholder, won't be used
|
||||||
|
else liftEitherM $ generateOutput strategy pandoc
|
||||||
|
|
||||||
-- | Pipeline step: Process output and write to file
|
-- | Pipeline step: Process output and write to file
|
||||||
processOutput :: Text -> CompilationM ()
|
processOutput :: Text -> CompilationM ()
|
||||||
processOutput output = do
|
processOutput output = do
|
||||||
strategy <- asks ccStrategy
|
strategy <- asks ccStrategy
|
||||||
outputPath <- asks ccOutputPath
|
writesFile <- asks ccWritesFile
|
||||||
liftEitherM $ csProcessOutput strategy outputPath output
|
if writesFile
|
||||||
|
then return () -- file already written by writer
|
||||||
|
else do
|
||||||
|
outputPath <- asks ccOutputPath
|
||||||
|
liftEitherM $ csProcessOutput strategy outputPath output
|
||||||
|
|
||||||
-- | Pipeline step: Print success message
|
-- | Pipeline step: Print success message
|
||||||
printSuccess :: CompilationM ()
|
printSuccess :: CompilationM ()
|
||||||
@ -256,7 +302,7 @@ compileWithStrategy :: CompilationStrategy -> SourceDir -> OutputDir -> Text ->
|
|||||||
compileWithStrategy strategy sourceDir outputDir docName (OutputPath inputPath) (OutputPath outputPath) = do
|
compileWithStrategy strategy sourceDir outputDir docName (OutputPath inputPath) (OutputPath outputPath) = do
|
||||||
let readerOptions = def { readerExtensions = getDefaultExtensions "markdown" }
|
let readerOptions = def { readerExtensions = getDefaultExtensions "markdown" }
|
||||||
config = DiagramConfig sourceDir outputDir (csOutputFormat strategy)
|
config = DiagramConfig sourceDir outputDir (csOutputFormat strategy)
|
||||||
context = CompilationContext strategy sourceDir outputDir inputPath outputPath docName readerOptions config
|
context = CompilationContext strategy inputPath outputPath docName readerOptions config (csWritesFile strategy)
|
||||||
pipeline = readContent >>= parseDocument >>= transformDocumentM >>= generateOutputM >>= processOutput >> printSuccess
|
pipeline = readContent >>= parseDocument >>= transformDocumentM >>= generateOutputM >>= processOutput >> printSuccess
|
||||||
|
|
||||||
runExceptT $ runReaderT pipeline context
|
runExceptT $ runReaderT pipeline context
|
||||||
@ -277,8 +323,19 @@ generateOutput strategy transformed = do
|
|||||||
Left err -> Left $ case csOutputFormat strategy of
|
Left err -> Left $ case csOutputFormat strategy of
|
||||||
PDF -> PDFGenerationError $ "LaTeX generation failed: " <> T.pack (show err)
|
PDF -> PDFGenerationError $ "LaTeX generation failed: " <> T.pack (show err)
|
||||||
HTML -> FileError $ "HTML generation failed: " <> T.pack (show err)
|
HTML -> FileError $ "HTML generation failed: " <> T.pack (show err)
|
||||||
|
DOCX -> FileError $ "DOCX generation failed: " <> T.pack (show err)
|
||||||
Right output -> Right output
|
Right output -> Right output
|
||||||
|
|
||||||
|
-- | Generate output file directly (for DOCX which writes to file)
|
||||||
|
generateOutputFile :: CompilationStrategy -> FilePath -> Pandoc -> IO (Either DocsterError ())
|
||||||
|
generateOutputFile _ outputPath pandoc = do
|
||||||
|
result <- runIO $ writeDocx def pandoc
|
||||||
|
case result of
|
||||||
|
Left err -> return $ Left $ FileError $ "DOCX generation failed: " <> T.pack (show err)
|
||||||
|
Right docxBS -> do
|
||||||
|
BSL.writeFile outputPath docxBS
|
||||||
|
return $ Right ()
|
||||||
|
|
||||||
-- | Compile markdown to PDF using XeLaTeX
|
-- | Compile markdown to PDF using XeLaTeX
|
||||||
compileToPDF :: FilePath -> IO ()
|
compileToPDF :: FilePath -> IO ()
|
||||||
compileToPDF = compileWithFormat pdfStrategy "pdf"
|
compileToPDF = compileWithFormat pdfStrategy "pdf"
|
||||||
@ -287,6 +344,10 @@ compileToPDF = compileWithFormat pdfStrategy "pdf"
|
|||||||
compileToHTML :: FilePath -> IO ()
|
compileToHTML :: FilePath -> IO ()
|
||||||
compileToHTML = compileWithFormat htmlStrategy "html"
|
compileToHTML = compileWithFormat htmlStrategy "html"
|
||||||
|
|
||||||
|
-- | Compile markdown to DOCX
|
||||||
|
compileToDOCX :: FilePath -> IO ()
|
||||||
|
compileToDOCX = compileWithFormat docxStrategy "docx"
|
||||||
|
|
||||||
-- | Higher-order function to compile with any format strategy
|
-- | Higher-order function to compile with any format strategy
|
||||||
compileWithFormat :: CompilationStrategy -> String -> FilePath -> IO ()
|
compileWithFormat :: CompilationStrategy -> String -> FilePath -> IO ()
|
||||||
compileWithFormat strategy extension path = do
|
compileWithFormat strategy extension path = do
|
||||||
|
|||||||
@ -9,15 +9,12 @@ module Docster.LaTeX
|
|||||||
import Data.Text (Text)
|
import Data.Text (Text)
|
||||||
import qualified Data.Text as T
|
import qualified Data.Text as T
|
||||||
|
|
||||||
-- | LaTeX template with comprehensive package support for PDF generation
|
-- | LaTeX template with modern corporate styling for PDF generation
|
||||||
latexTemplate :: Text -> Text
|
latexTemplate :: Text -> Text
|
||||||
latexTemplate bodyContent = T.unlines
|
latexTemplate bodyContent = T.unlines
|
||||||
[ "\\documentclass{article}"
|
[ "\\documentclass{article}"
|
||||||
, "\\usepackage[utf8]{inputenc}"
|
-- Packages
|
||||||
, "\\usepackage{fontspec}"
|
, "\\usepackage{fontspec}"
|
||||||
, "\\setmainfont{DejaVu Serif}[Scale=1.0]"
|
|
||||||
, "\\setsansfont{DejaVu Sans}[Scale=1.0]"
|
|
||||||
, "\\setmonofont{DejaVu Sans Mono}[Scale=0.85]"
|
|
||||||
, "\\usepackage{graphicx}"
|
, "\\usepackage{graphicx}"
|
||||||
, "\\usepackage{adjustbox}"
|
, "\\usepackage{adjustbox}"
|
||||||
, "\\usepackage{geometry}"
|
, "\\usepackage{geometry}"
|
||||||
@ -25,16 +22,46 @@ latexTemplate bodyContent = T.unlines
|
|||||||
, "\\usepackage{booktabs}"
|
, "\\usepackage{booktabs}"
|
||||||
, "\\usepackage{array}"
|
, "\\usepackage{array}"
|
||||||
, "\\usepackage{calc}"
|
, "\\usepackage{calc}"
|
||||||
, "\\geometry{margin=1in}"
|
|
||||||
, "\\usepackage{hyperref}"
|
|
||||||
, "\\usepackage{enumitem}"
|
, "\\usepackage{enumitem}"
|
||||||
, "\\usepackage{amsmath}"
|
, "\\usepackage{amsmath}"
|
||||||
, "\\usepackage{amssymb}"
|
, "\\usepackage{amssymb}"
|
||||||
, "\\usepackage{fancyvrb}"
|
, "\\usepackage{fancyvrb}"
|
||||||
, "\\usepackage{color}"
|
, "\\usepackage[dvipsnames,svgnames,x11names]{xcolor}"
|
||||||
|
, "\\usepackage{titlesec}"
|
||||||
|
, "\\usepackage{fancyhdr}"
|
||||||
|
, "\\usepackage{framed}"
|
||||||
|
-- Typography: Helvetica Neue + Menlo, sans-serif default
|
||||||
|
, "\\setmainfont{Helvetica Neue}"
|
||||||
|
, "\\setsansfont{Helvetica Neue}"
|
||||||
|
, "\\setmonofont{Menlo}[Scale=0.85]"
|
||||||
|
, "\\renewcommand{\\familydefault}{\\sfdefault}"
|
||||||
|
-- Layout: wider margins, block paragraphs
|
||||||
|
, "\\geometry{left=0.9in,right=0.9in,top=1in,bottom=1in}"
|
||||||
|
, "\\setlength{\\parindent}{0pt}"
|
||||||
|
, "\\setlength{\\parskip}{0.5em}"
|
||||||
|
-- Color scheme
|
||||||
|
, "\\definecolor{accent}{HTML}{1A365D}"
|
||||||
|
, "\\definecolor{codebg}{HTML}{F5F5F5}"
|
||||||
|
-- Hyperlinks: accent-colored, no boxes
|
||||||
|
, "\\usepackage[colorlinks=true,linkcolor=accent,urlcolor=accent,citecolor=accent]{hyperref}"
|
||||||
|
-- Heading styles
|
||||||
|
, "\\titleformat{\\section}{\\Large\\bfseries\\color{accent}}{\\thesection}{1em}{}[\\vspace{2pt}\\titlerule]"
|
||||||
|
, "\\titleformat{\\subsection}{\\large\\bfseries\\color{accent}}{\\thesubsection}{1em}{}"
|
||||||
|
, "\\titleformat{\\subsubsection}{\\normalsize\\bfseries\\color{accent}}{\\thesubsubsection}{1em}{}"
|
||||||
|
, "\\titlespacing*{\\section}{0pt}{1.5em}{0.8em}"
|
||||||
|
, "\\titlespacing*{\\subsection}{0pt}{1.2em}{0.5em}"
|
||||||
|
, "\\titlespacing*{\\subsubsection}{0pt}{1em}{0.4em}"
|
||||||
|
-- Page header/footer: minimal centered page number
|
||||||
|
, "\\pagestyle{fancy}"
|
||||||
|
, "\\fancyhf{}"
|
||||||
|
, "\\renewcommand{\\headrulewidth}{0pt}"
|
||||||
|
, "\\fancyfoot[C]{\\small\\thepage}"
|
||||||
|
-- Code blocks: light gray background
|
||||||
, "\\DefineVerbatimEnvironment{Highlighting}{Verbatim}{commandchars=\\\\\\{\\}}"
|
, "\\DefineVerbatimEnvironment{Highlighting}{Verbatim}{commandchars=\\\\\\{\\}}"
|
||||||
, "\\newenvironment{Shaded}{}{}"
|
, "\\newenvironment{Shaded}{\\begin{snugshade}}{\\end{snugshade}}"
|
||||||
|
, "\\definecolor{shadecolor}{HTML}{F5F5F5}"
|
||||||
, syntaxHighlightingCommands
|
, syntaxHighlightingCommands
|
||||||
|
-- Pandoc helpers
|
||||||
, "\\providecommand{\\tightlist}{%"
|
, "\\providecommand{\\tightlist}{%"
|
||||||
, " \\setlength{\\itemsep}{0pt}\\setlength{\\parskip}{0pt}}"
|
, " \\setlength{\\itemsep}{0pt}\\setlength{\\parskip}{0pt}}"
|
||||||
, "\\newcommand{\\real}[1]{#1}"
|
, "\\newcommand{\\real}[1]{#1}"
|
||||||
@ -47,6 +74,7 @@ latexTemplate bodyContent = T.unlines
|
|||||||
, "\\def\\maxheight{\\ifdim\\Gin@nat@height>\\textheight\\textheight\\else\\Gin@nat@height\\fi}"
|
, "\\def\\maxheight{\\ifdim\\Gin@nat@height>\\textheight\\textheight\\else\\Gin@nat@height\\fi}"
|
||||||
, "\\makeatother"
|
, "\\makeatother"
|
||||||
, "\\setkeys{Gin}{width=\\maxwidth,height=\\maxheight,keepaspectratio}"
|
, "\\setkeys{Gin}{width=\\maxwidth,height=\\maxheight,keepaspectratio}"
|
||||||
|
, "\\providecommand{\\pandocbounded}[1]{#1}"
|
||||||
, "\\begin{document}"
|
, "\\begin{document}"
|
||||||
, bodyContent
|
, bodyContent
|
||||||
, "\\end{document}"
|
, "\\end{document}"
|
||||||
|
|||||||
BIN
src/Docster/Types.hi
Normal file
BIN
src/Docster/Types.hi
Normal file
Binary file not shown.
@ -45,7 +45,7 @@ data DocsterError
|
|||||||
instance Exception DocsterError
|
instance Exception DocsterError
|
||||||
|
|
||||||
-- | Output format for document generation
|
-- | Output format for document generation
|
||||||
data OutputFormat = PDF | HTML
|
data OutputFormat = PDF | HTML | DOCX
|
||||||
deriving (Show, Eq)
|
deriving (Show, Eq)
|
||||||
|
|
||||||
-- | Type-safe wrapper for source directory paths
|
-- | Type-safe wrapper for source directory paths
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user