% \iffalse meta-comment
% Copyright (C) 2021 Jamal Bouajjaj | 2015, 2016, 2017 Mattias Jacobsson and contributors
% This work, cartonaugh, is a fork of Mattias Jacobsson's [karnaugh-map](
https://github.com/2pi/karnaugh-map) package with major changes. As the copyright on the base package is under [CC BY-SA](
https://creativecommons.org/licenses/by-sa/3.0/), so is this fork.
% \fi
%
% \iffalse
%<*package>
%<package>%%
%<package>%% Copyright (C) 2021 Jamal Bouajjaj | 2015, 2016, 2017 Mattias Jacobsson and contributors
%<package>%% This work, cartonaugh, is a fork of Mattias Jacobsson's [karnaugh-map](
https://github.com/2pi/karnaugh-map) package with major changes. As the copyright on the base package is under [CC BY-SA](
https://creativecommons.org/licenses/by-sa/3.0/), so is this fork.
%<package>%%
%<package>
%<package>\NeedsTeXFormat{LaTeX2e}
%<package>\ProvidesPackage{cartonaugh}[2021/07/15 v1.0 Draw Karnaugh Maps]
%
%<*driver>
\documentclass{ltxdoc}
\usepackage{cartonaugh}
\usepackage{multicol}% for documentation
\usepackage{tabularx}% for documentation
\usepackage{float}% for documentation
\usepackage{hyperref}% for documentation
\setlength{\parindent}{0pt}
\setlength{\parskip}{0.6em}
\EnableCrossrefs
\CodelineIndex
\RecordChanges
\OnlyDescription
\begin{document}
\DeleteShortVerb{\|}
\DocInput{cartonaugh.dtx}
\PrintChanges
\end{document}
%</driver>
% \fi
%
% \CharacterTable
% {Upper-case \A\B\C\D\E\F\G\H\I\J\K\L\M\N\O\P\Q\R\S\T\U\V\W\X\Y\Z
% Lower-case \a\b\c\d\e\f\g\h\i\j\k\l\m\n\o\p\q\r\s\t\u\v\w\x\y\z
% Digits \0\1\2\3\4\5\6\7\8\9
% Exclamation \! Double quote \" Hash (number) \#
% Dollar \$ Percent \% Ampersand \&
% Acute accent \' Left paren \( Right paren \)
% Asterisk \* Plus \+ Comma \,
% Minus \- Point \. Solidus \/
% Colon \: Semicolon \; Less than \<
% Equals \= Greater than \> Question mark \?
% Commercial at \@ Left bracket \[ Backslash \\
% Right bracket \] Circumflex \^ Underscore \_
% Grave accent \` Left brace \{ Vertical bar \|
% Right brace \} Tilde \~}
%
%
% \changes{v1.0}{2021/07/15}{Initial Release}
%
% \GetFileInfo{cartonaugh.sty}
%
% \title{The \textsf{cartonaugh} package}
% \author{Jamal Bouajjaj}
% \date{\textsf{cartonaugh}~\fileversion, \filedate}
%
%\maketitle
%
% \begin{abstract}
% This package, a fork of Mattias Jacobsson/2pi's \href{
https://github.com/2pi/karnaugh-map}{karnaugh-map} package, draws karnaugh maps with 2, 3, 4, 5, and 6 variables.
% It also contains commands for filling the karnaugh map with terms semi-automatically or manually.
% Last but not least it contains commands for drawing implicants on top of the map.
% Cartonaugh is a portmanteau of ``cartographer`` and ``karnaugh``.
% Below is an example of a two variable karnaugh map of $X_0 \oplus X_1$.
% \end{abstract}
% \begin{figure}[H]
% \centering
% \begin{cartonaugh}[2][2][1][$X_0$][$X_1$]
% \minterms{1,2}
% \autoterms[0]
% \implicant{1}{1}
% \implicant{2}{2}
% \end{cartonaugh}
% \end{figure}
% \tableofcontents
% \pagebreak
%
% \iffalse
% \begin{macrocode}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
%%
%% Dependencies
%%
%% Check if ran under LuaLaTeX. If not, exit
\RequirePackage{iftex}
\RequireLuaTeX
%% parsing arguments
\RequirePackage{xparse}
%% working with strings
\RequirePackage{xstring}
%% drawing
\RequirePackage{tikz}
\usetikzlibrary{calc,matrix}
%%
%% Helpers
%%
\directlua{require("cartonaugh")}
%% command raises an error if executed outside the cartonaugh environment
\newcommand{\@cartonaugh@func@bailoutsideenvironment@}[0]{%
\ifnum\@cartonaugh@var@isactive@=0
\PackageError{cartonaugh}{%
Command can not be used outside cartonaugh environment%
}{%
Do not use this command outside the cartonaugh environment.%
}
\fi
}
\newcommand{\cartonaughSetColorAtIndex}[2]{\directlua{CARTONAUGH_COLORS[\the\numexpr(#1)\relax] = "\luaescapestring{#2}"}}
%% store whether a kmap has been created
\newcommand{\@cartonaugh@var@isactive@}{0}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### /CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
% \end{macrocode}
% \fi
%
% \section{Usage}
%
% \begin{environment}{cartonaugh}
% The |cartonaugh| environment is the base for this package, and everything related to this package happens inside an instances of this environment.
%
% \textbf{Usage:}
% \MakeShortVerb{\|}
% \begin{tabularx}{\textwidth}{l X}
% \small{|\begin{cartonaugh}|} & \\
% \small{\meta{*}} & \small{One asterisk for black and white implicants, non for colorized implicants} \\
% \small{\oarg{X size}} & \small{Number of X-axis cells. Default: ''4''} \\
% \small{\oarg{Y size}} & \small{Number of Y-axis cells. Default: ''4''} \\
% \small{\oarg{Z size}} & \small{Number of X$\times$Y submaps. Default: ''1''} \\
% \small{\oarg{X label}} & \small{Label for the X-axis. Default: ''$X_1X_0$''} \\
% \small{\oarg{Y label}} & \small{Label for the Y-axis. Default: ''$X_3X_2$''} \\
% \small{\oarg{Z label}} & \small{Label for the submaps. Default: ''$X_5X_4$''} \\
% \small{\oarg{Submap Seperation Type}} & \small{Whether to seperate the submap by space or by a thick line. Default: 0 (so seperation by space)} \\
% \end{tabularx}
% \DeleteShortVerb{\|}
% \textbf{Example:}
%
% Four variable karnaugh map, colorized, with X label $X_1X_0$, and Y label $X_3X_2$.
% \begin{verbatim}
%\begin{cartonaugh}
%\end{cartonaugh}
%
%or
%
%\begin{cartonaugh}[4][4][1][$X_1X_0$][$X_3X_2$]
%\end{cartonaugh}
% \end{verbatim}
% Six variable karnaugh map, black and white, with X label $ba$, Y label $dc$, and Z label $fe$.
% \begin{verbatim}
%\begin{cartonaugh}*[4][4][4][$ba$][$dc$][$fe$]
%\end{cartonaugh}
% \end{verbatim}
%
% Six variable karnaugh map, black and white, with X label $ba$, Y label $dc$, and Z label $fe$, and the submaps are seperated by a thick line in between.
% \begin{verbatim}
%\begin{cartonaugh}*[4][4][4][$ba$][$dc$][$fe$][1]
%\end{cartonaugh}
% \end{verbatim}
%
% \iffalse
% \begin{macrocode}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
\NewDocumentEnvironment{cartonaugh}{s O{4} O{4} O{1} O{$X_1X_0$} O{$X_3X_2$} O{$X_5X_4$} O{0}} {%
\begingroup
% store map size {[START]
\renewcommand{\@cartonaugh@var@isactive@}{1}%
% [END]}
% determinate if markings should be color or black and white
%
% [END]}
% test if a matrix template is found or not(aka "\@cartonaugh@local@matrixtemplate@" equals to '0')
\begin{tikzpicture}
\directlua{init_cartonaught_env(
\the\numexpr(#2)\relax,
\the\numexpr(#3)\relax,
\the\numexpr(#4)\relax,
\the\numexpr(#1)\relax,
"\luaescapestring{\detokenize{#5}}",
"\luaescapestring{\detokenize{#6}}",
"\luaescapestring{\detokenize{#7}}",
\the\numexpr(#8)\relax
)}
}{
\end{tikzpicture}
\endgroup
}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### /CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
% \end{macrocode}
% \fi
% \end{environment}
%
% \newpage
% \MakeShortVerb{\|}
% \subsection{Terms}
% \begin{macro}{\autoterms}
% The |\autoterms| command fills the remaining unfilled cells of the karnaugh map with the contents of the optional argument.
%
% \textbf{Usage:}
%
% \begin{tabularx}{\textwidth}{l X}
% \small{|\autoterms|} & \\
% \small{\oarg{content}} & \small{Content for the remaining unfilled cells. Default: ''-''}
% \end{tabularx}
%
% \textbf{Example:}
%
% Fill all remaining unfilled cells with ''-''.
% \begin{verbatim}
% \DeleteShortVerb{\|}
%\begin{cartonaugh}
% \autoterms[-]
%\end{cartonaugh}
% \end{verbatim}
%
% \iffalse
% \begin{macrocode}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
\DeclareDocumentCommand{\autoterms}{O{-}} {%
\@cartonaugh@func@bailoutsideenvironment@{}
%
\directlua{
autoterms("\luaescapestring{\detokenize{#1}}", '-')
}
}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### /CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
% \end{macrocode}
% \fi
%
% \end{macro}
%
% \begin{macro}{\indeterminants}
% The |\indeterminants| command fills the specified cells with ''-'' if they aren't already filled. Order of the cell numbers does not matter.
%
% \textbf{Usage:}
%
% \begin{tabularx}{\textwidth}{l X}
% \small{|\indeterminants|} & \\
% \small{\marg{cells}} & \small{Comma separated list of cells to fill with ''-''}
% \end{tabularx}
%
% \textbf{Example:}
%
% Fill the top left and right cell with ''-''.
% \begin{verbatim}
%\begin{cartonaugh}
% \indeterminants{0,2}
%\end{cartonaugh}
% \end{verbatim}
%
% \iffalse
% \begin{macrocode}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
\DeclareDocumentCommand{\indeterminants}{m} {%
% bail if outside environment cartonaugh
\@cartonaugh@func@bailoutsideenvironment@{}
%
\directlua{
write_to_cell("\luaescapestring{\detokenize{#1}}", '-')
}
}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### /CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
% \end{macrocode}
% \fi
%
% \end{macro}
%
% \begin{macro}{\manualterms}
% The |\manualterms| command fills the 0th cell with the first element in the argument, the 1st cell with the second element in the argument, and so on. If any of the cells already is filled, it is left as it was.
%
% \textbf{Usage:}
%
% \begin{tabularx}{\textwidth}{l X}
% \small{|\manualterms|} & \\
% \small{\marg{content}} & \small{Comma separated list of cell contents}
% \end{tabularx}
%
% \textbf{Example:}
%
% Fill the first four cells with 0, 1, 0, and 1 respectively.
% \begin{verbatim}
%\begin{cartonaugh}
% \manualterms{0,1,0,1}
%\end{cartonaugh}
% \end{verbatim}
%
% \iffalse
% \begin{macrocode}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
\DeclareDocumentCommand{\manualterms}{m} {%
% bail if outside environment cartonaugh
\@cartonaugh@func@bailoutsideenvironment@{}
%
\directlua{
manualterms("\luaescapestring{\detokenize{#1}}")
}
}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### /CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
% \end{macrocode}
% \fi
%
% \end{macro}
%
% \begin{macro}{\maxterms}
% The |\maxterms| command fills the specified cells with ''0'' if they aren't already filled. Order of the cell numbers does not matter.
%
% \textbf{Usage:}
%
% \begin{tabularx}{\textwidth}{l X}
% \small{|\maxterms|} & \\
% \small{\marg{cells}} & \small{Comma separated list of cells to fill with ''0''} \\
% \end{tabularx}
%
% \textbf{Example:}
%
% Fill the top left and right cell with ''0''.
% \begin{verbatim}
%\begin{cartonaugh}
% \maxterms{0,2}
%\end{cartonaugh}
% \end{verbatim}
%
% \iffalse
% \begin{macrocode}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
\DeclareDocumentCommand{\maxterms}{m} {%
% bail if outside environment cartonaugh
\@cartonaugh@func@bailoutsideenvironment@{}
%
\directlua{
write_to_cell("\luaescapestring{\detokenize{#1}}", '0')
}
}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### /CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
% \end{macrocode}
%\fi
%
% \end{macro}
%
% \begin{macro}{\minterms}
% The |\minterms| command fills the specified cells with ''1'' if they aren't already filled. Order of the cell numbers does not matter.
%
% \textbf{Usage:}
%
% \begin{tabularx}{\textwidth}{l X}
% \small{|\minterms|} & \\
% \small{\marg{cells}} & \small{Comma separated list of cells to fill with ''1''} \\
% \end{tabularx}
%
% \textbf{Example:}
%
% Fill the top left and right cell with ''1''.
% \begin{verbatim}
%\begin{cartonaugh}
% \minterms{0,2}
%\end{cartonaugh}
% \end{verbatim}
%
% \iffalse
% \begin{macrocode}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
\DeclareDocumentCommand{\minterms}{m} {%
% bail if outside environment cartonaugh
\@cartonaugh@func@bailoutsideenvironment@{}
%
\directlua{
write_to_cell("\luaescapestring{\detokenize{#1}}", '1')
}
}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### /CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
% \end{macrocode}
%\fi
%
% \end{macro}
%
% \begin{macro}{\terms}
% The |\terms| command fills the specified cells with the specified content if they aren't already filled. Order of the cell numbers does not matter.
%
% \textbf{Usage:}
%
% \begin{tabularx}{\textwidth}{l X}
% \small{|\terms|} & \\
% \small{\marg{cells}} & \small{Comma separated list of cells to fill with content} \\
% \small{\marg{content}} & \small{Content to fill the cells with} \\
% \end{tabularx}
%
% \textbf{Example:}
%
% Fill the top left and right cell with ''X''.
% \begin{verbatim}
%\begin{cartonaugh}
% \terms{0,2}{X}
%\end{cartonaugh}
% \end{verbatim}
%
% \iffalse
% \begin{macrocode}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
\DeclareDocumentCommand{\terms}{m m} {%
% bail if outside environment cartonaugh
\@cartonaugh@func@bailoutsideenvironment@{}
%
\directlua{
write_to_cell("\luaescapestring{\detokenize{#1}}",
"\luaescapestring{\detokenize{#2}}")
}
}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### /CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
% \end{macrocode}
%\fi
%
% \end{macro}
%
% \newpage
% \MakeShortVerb{\|}
% \subsection{Implicants}
% \begin{macro}{\implicant}
% The |\implicant| command draws quadratic implicants on one or multiple submaps. If the implicant shall be drawn on multiple submaps, \marg{northwest cell} and \marg{southeast cell} must be specified as if the implicant was to be drawn on the 0:th submap. When turned on, colorization is done automatically, following a global sequence of available colors.
%
% \textbf{Usage:}
%
% \begin{tabularx}{\textwidth}{l X}
% \small{|\implicant|} & \\
% \small{\marg{northwest cell}} & \small{The most northwest cell in the implicant} \\
% \small{\marg{southeast cell}} & \small{The most southeast cell in the implicant} \\
% \small{\oarg{submaps}} & \small{Comma separated list of submaps the implicant should be drawn on. Default: ''0''} \\
% \end{tabularx}
%
% \textbf{Example:}
% \DeleteShortVerb{\|}
% \begin{multicols}{2}
% [Implicant around the four most inner cells.]
% \begin{verbatim}
%\begin{cartonaugh}
% \implicant{5}{15}
%\end{cartonaugh}
% \end{verbatim}
% \columnbreak
% \resizebox{\columnwidth}{!}{
% \begin{cartonaugh}
% \implicant{5}{15}
% \end{cartonaugh}
% }
% \end{multicols}
% \begin{multicols}{2}
% [Single cell implicant, 0:th cell, on all four submaps.]
% \begin{verbatim}
%\begin{cartonaugh}[4][4][4]
% [$BA$][$DC$][$FE$][1]
% \implicant{0}{0}[0,1,2,3]
%\end{cartonaugh}
% \end{verbatim}
% \columnbreak
% \resizebox{\columnwidth}{!}{
% \begin{cartonaugh}[4][4][4][$BA$][$DC$][$FE$][1]
% \implicant{0}{0}[0,1,2,3]
% \end{cartonaugh}
% }
% \end{multicols}
%
% \iffalse
% \begin{macrocode}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
\DeclareDocumentCommand{\implicant}{m m O{0}} {%
% bail if outside environment cartonaugh
\@cartonaugh@func@bailoutsideenvironment@{}
%
\directlua{
manual_draw_implicant(
"\luaescapestring{\detokenize{#1}}",
"\luaescapestring{\detokenize{#2}}",
"\luaescapestring{\detokenize{#3}}"
)
}
}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### /CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
% \end{macrocode}
%\fi
%
% \end{macro}
%
% \MakeShortVerb{\|}
% \begin{macro}{\implicantedge}
% The |\implicantedge| command draws quadratic implicants with the middle of the implicant facing the edge of a submap either horizontally or vertically. The function is able to draw the same implicant on one or multiple submaps. However if the implicant shall be drawn on multiple submaps, \marg{northwest part - northwest cell}, \marg{northwest part - southeast cell}, \marg{southeast part - northwest cell}, \marg{southeast part - southeast cell} must be specified as if the implicant was to be drawn on the 0:th submap. When turned on, colorization is done automatically, following a global sequence of available colors.
%
% \textbf{Usage:}
%
% \begin{tabularx}{\textwidth}{l X}
% \small{|\implicantedge|} & \\
% \small{\marg{northwest part - northwest cell}} & \small{The most northwest cell in the northwest part of the implicant} \\
% \small{\marg{northwest part - southeast cell}} & \small{The most southeast cell in the northwest part of the implicant} \\
% \small{\marg{southeast part - northwest cell}} & \small{The most northwest cell in the southeast part of the implicant} \\
% \small{\marg{southeast part - southeast cell}} & \small{The most southeast cell in the southeast part of the implicant} \\
% \small{\oarg{submaps}} & \small{Comma separated list of submaps the implicant should be drawn on. Default: ''0''} \\
% \end{tabularx}
%
% \textbf{Example:}
% \DeleteShortVerb{\|}
%
% \begin{multicols}{2}
% [Horizontal implicant over the submap edge containing the cells 4, 6, 12, and 14.]
% \begin{verbatim}
%\begin{cartonaugh}
% \implicantedge{4}{12}{6}{14}
%\end{cartonaugh}
% \end{verbatim}
% \columnbreak
% \resizebox{\columnwidth}{!}{
% \begin{cartonaugh}
% \implicantedge{4}{12}{6}{14}
% \end{cartonaugh}
% }
% \end{multicols}
%
% \iffalse
% \begin{macrocode}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
\DeclareDocumentCommand{\implicantedge}{m m m m O{0}} {%
% bail if outside environment cartonaugh
\@cartonaugh@func@bailoutsideenvironment@{}
%
\directlua{
manual_draw_edge_implicant(
"\luaescapestring{\detokenize{#1}}",
"\luaescapestring{\detokenize{#2}}",
"\luaescapestring{\detokenize{#3}}",
"\luaescapestring{\detokenize{#4}}",
"\luaescapestring{\detokenize{#5}}"
)
}
}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### /CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
% \end{macrocode}
%\fi
%
% \end{macro}
% \MakeShortVerb{\|}
% \begin{macro}{\implicantcorner}
% The |\implicantcorner| command draws an implicant around only the four corner pieces on one or multiple four variable karnaugh submaps. When turned on, colorization is done automatically, following a global sequence of available colors.
%
% \textbf{Usage:}
%
% \begin{tabularx}{\textwidth}{l X}
% \small{|\implicantcorner|} & \\
% \small{\oarg{submaps}} & \small{Comma separated list of submaps the implicant should be drawn on. Default: ''0''} \\
% \end{tabularx}
%
% \textbf{Example:}
% \DeleteShortVerb{\|}
%
% \begin{multicols}{2}
% [Draw an implicant around all corners on 0th and 2nd submap of a six variable karnaugh map.]
% \begin{verbatim}
%\begin{cartonaugh}[4][4][4]
% \implicantcorner[0,2]
%\end{cartonaugh}
% \end{verbatim}
% \columnbreak
% \resizebox{\columnwidth}{!}{
% \begin{cartonaugh}[4][4][4]
% \implicantcorner[0,2]
% \end{cartonaugh}
% }
% \end{multicols}
%
% \iffalse
% \begin{macrocode}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
\DeclareDocumentCommand{\implicantcorner}{O{0}} {
% bail if outside environment cartonaugh
\@cartonaugh@func@bailoutsideenvironment@{}
%
\directlua{
manual_draw_corner_implicant(
"\luaescapestring{\detokenize{#1}}"
)
}
}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### /CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
% \end{macrocode}
%\fi
%
% \end{macro}
%
% \newpage
% \MakeShortVerb{\|}
% \subsection{Options}
% \begin{macro}{\implicantspread}
% The |\implicantspread| changed the implicant spread
%
% \textbf{Usage:}
%
% \begin{tabularx}{\textwidth}{l X}
% \small{|\implicantspread|} & \\
% \small{\marg{innerspread}} & \small{The inner spread's spread, from 0 to 0.5 (as >0.5 will go out of the implicant square)} \\
% \small{\marg{outerspread}} & \small{The outer spread's spread, from 0.5 and up} \\
% \end{tabularx}
%
% \textbf{Example:}
% \DeleteShortVerb{\|}
%
% \begin{multicols}{2}
% \begin{verbatim}
% \begin{cartonaugh}[4][4][2]
% \implicantedge{4}{12}{6}{14}[0]
% \implicantspread{0.25}{0.7}
% \implicantedge{4}{12}{6}{14}[1]
% \end{cartonaugh}
% \end{verbatim}
% \columnbreak
% \resizebox{\columnwidth}{!}{
% \begin{cartonaugh}[4][4][2]
% \implicantedge{4}{12}{6}{14}[0]
% \implicantspread{0.25}{0.7}
% \implicantedge{4}{12}{6}{14}[1]
% \end{cartonaugh}
% }
% \end{multicols}
%
% \iffalse
% \begin{macrocode}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
\DeclareDocumentCommand{\implicantspread}{m m} {
% bail if outside environment cartonaugh
\@cartonaugh@func@bailoutsideenvironment@{}
%
\directlua{
change_implicant_inner_spread(\luaescapestring{\detokenize{#1}})
change_implicant_outer_spread(\luaescapestring{\detokenize{#2}})
}
}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### /CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
% \end{macrocode}
%\fi
%
% \end{macro}
%
% \MakeShortVerb{\|}
% \begin{macro}{\resetimplicantspread}
% The |\resetimplicantspread| resets the implicant spread if set by |\implicantspread| to default values
% \DeleteShortVerb{\|}
% \iffalse
% \begin{macrocode}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
\DeclareDocumentCommand{\resetimplicantspread}{} {
% bail if outside environment cartonaugh
\@cartonaugh@func@bailoutsideenvironment@{}
%
\directlua{
change_implicant_inner_spread(0.35)
change_implicant_outer_spread(0.55)
}
}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### /CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
% \end{macrocode}
%\fi
%
% \end{macro}
% \MakeShortVerb{\|}
% \begin{macro}{\changecolor}
% The |\changecolor| changes the color for implicants
%
% \textbf{Usage:}
%
% \begin{tabularx}{\textwidth}{l X}
% \small{|\changecolor|} & \\
% \small{\marg{color}} & \small{The new color for implicants, in Tikz format} \\
% \end{tabularx}
% \DeleteShortVerb{\|}
% \textbf{Example:}
%
% \begin{multicols}{2}
% [Change the implicant color to pink, then green]
% \begin{verbatim}
% \begin{cartonaugh}
% \changecolor{pink}
% \implicant{0}{1}
% \changecolor{green}
% \implicant{8}{10}
% \end{cartonaugh}
% \end{verbatim}
% \columnbreak
% \resizebox{\columnwidth}{!}{
% \begin{cartonaugh}
% \changecolor{pink}
% \implicant{0}{1}
% \changecolor{green}
% \implicant{8}{10}
% \end{cartonaugh}
% }
% \end{multicols}
%
%
% \iffalse
% \begin{macrocode}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
\DeclareDocumentCommand{\changecolor}{m} {
% bail if outside environment cartonaugh
\@cartonaugh@func@bailoutsideenvironment@{}
%
\directlua{
customColor("\luaescapestring{\detokenize{#1}}")
}
}
% ^^A ##########################################################################
% ^^A ##########################################################################
% ^^A #### /CODE ####
% ^^A ##########################################################################
% ^^A ##########################################################################
% \end{macrocode}
%\fi
%
% \end{macro}
%
% \newpage
% \section{Examples}
% \begin{multicols}{2}
% [Draw a karnaugh map for \small{$f(a,b,c,d,e,f) =$\\$\Sigma(0,1,2,3,8,13,17,20,22,28,33,32,30,19,40,35,49,42,34,10,60,54,62,51,52)$\\$+d(15,45,47)$}.]
% \begin{verbatim}
%\begin{cartonaugh}[4][4][4][$ba$][$dc$][$fe$][1]
% \minterms{0,1,2,3,8,13,17,20,22,28,
% 33,32,30,19,40,35,49,42,34,10,60,
% 54,62,51,52}
% \indeterminants{15,45,47}
% \autoterms[0]
% \implicantcorner[0,2]
% \implicant{1}{3}[0,1,2,3]
% \implicantedge{4}{12}{6}{14}[1,3]
% \implicant{13}{15}[0,2]
%\end{cartonaugh}
% \end{verbatim}
% \columnbreak
% \resizebox{\columnwidth}{!}{
% \begin{cartonaugh}[4][4][4][$ba$][$dc$][$fe$][1]
% \minterms{0,1,2,3,8,13,17,20,22,28,33,32,30,19,40,35,49,42,34,10,60,54,62,51,52}
% \indeterminants{15,45,47}
% \autoterms[0]
% \implicantcorner[0,2]
% \implicant{1}{3}[0,1,2,3]
% \implicantedge{4}{12}{6}{14}[1,3]
% \implicant{13}{15}[0,2]
% \end{cartonaugh}
% }
% \end{multicols}
%
% \begin{multicols}{2}
% [Draw a karnaugh map for \small{$f(X_0,X_1) = \Pi(0,2,3)$} in black and white.]
% \begin{verbatim}
%\begin{cartonaugh}*[2][2][1][$X_0$][$X_1$]
% \maxterms{0,2,3}
% \autoterms[1]
% \implicant{1}{1}
%\end{cartonaugh}
% \end{verbatim}
% \columnbreak
% \resizebox{\columnwidth}{!}{
% \begin{cartonaugh}*[2][2][1][$X_0$][$X_1$]
% \maxterms{0,2,3}
% \autoterms[1]
% \implicant{1}{1}
% \end{cartonaugh}
% }
% \end{multicols}
%
% \newpage
% \begin{multicols}{2}
% [Draw a variable entered map.]
% \begin{verbatim}
%\begin{cartonaugh}[4][2][1][$ab$][$c$]
% \maxterms{0,2,4,5,6}
% \minterms{3}
% \terms{1}{$d$}
% \terms{7}{$d'$}
% \implicant{1}{3}
% \implicant{3}{7}
%\end{cartonaugh}
% \end{verbatim}
% \columnbreak
% \resizebox{\columnwidth}{!}{
% \begin{cartonaugh}[4][2][1][$ab$][$c$]
% \maxterms{0,2,4,5,6}
% \minterms{3}
% \terms{1}{$d$}
% \terms{7}{$d'$}
% \implicant{1}{3}
% \implicant{3}{7}
% \end{cartonaugh}
% }
% \end{multicols}
%
% \section{Dependencies}
% \begin{itemize}
% \item tikz
% \item xparse
% \item xstring
% \item lualatex
% \end{itemize}
%
% \newpage
% \section{Miscellaneous}
% \MakeShortVerb{\|}
% \subsection*{Resizing}
% The karnaugh maps produced with this package have a prespecified size which can not be changed. However you can resize the karnaugh map to your desired size. Resizing can be done using the |\resizebox| command from the graphicx package. Scaling the karnaugh map to fill the column width while preserving the aspect ratio can be done as follows.
% \begin{verbatim}
%\resizebox{\columnwidth}{!}{
% \DeleteShortVerb{\|}
% \begin{cartonaugh}
% \end{cartonaugh}
%}
% \end{verbatim}
%
% \subsection*{Comma separated lists}
% Anywhere in this package where a comma separated list is used data should only be comma separated. Therefore a comma and space separeated list will for example \textit{not} work properly.
%
% An example of errorious usage related to the \small{\marg{cells}} parameter in the terms related commands can result in multiple zeros, ones and other terms overlapping in the same cell in the outputted karnaugh map.
%
%\iffalse
%</package>
%<*luacode>
%
%
% This is the Luacode that is generated and put into cartonaugh.lua
%
%
%\begin{macrocode}
DEBUG_FLAG = 0
CARTONAUGH_COLORS = {'red', 'green', 'yellow', 'cyan', 'blue', 'magenta'}
CARTONAUGH_DEFAULT_COLOR = 'cyan'
cartonaugh_env_settings = {}
used_cells = {}
-- Function that is used to either print to LaTeX or to the console
-- Used for debugging mode
function localPrint(str)
if DEBUG_FLAG==1 then
print(str)
end
tex.sprint(str)
end
-- Function that takes a string and splits it be delimiter, then returns it back
-- From
https://stackoverflow.com/questions/19262761/lua-need-to-split-at-comma
function split(source, delimiters)
local elements = {}
local pattern = '([^'..delimiters..']+)'
string.gsub(source, pattern, function(value) elements[#elements + 1] = value; end);
return elements
end
-- Function that converts a decimal number to binary
function decimalToBin(num, numb_bits, return_concat)
if return_concat == nil then return_concat=true end
num = tonumber(num)
numb_bits = tonumber(numb_bits)
local t={}
for b=numb_bits,1,-1 do
rest=math.floor(math.fmod(num,2))
t[b]=rest
num=(num-rest)/2
end
if return_concat == true then
return table.concat(t)
else
return t
end
end
-- Function that converts a decimal number to grey code
function decimalToGreyBin(num, numb_bits)
-- Get the binary array
num = num ~ (num >> 1)
local t = decimalToBin(num, numb_bits, false)
return table.concat(t)
end
function greyBinToDecimal(num)
-- num = tonumber(num,2)
local mask = num
while mask > 0 do
mask = mask >> 1
num = num ~ mask
end
return tonumber(num)
end
-- Function to pad a string by a amount with b string
function padString(str, pad_to, pad_with)
local ret = str
for l=1, (pad_to-string.len(str)) do
ret = pad_with .. ret
end
return ret
end
function getColor(index)
if index <= 0 then return CARTONAUGH_DEFAULT_COLOR end
if CARTONAUGH_COLORS[index] ~= nil then
return CARTONAUGH_COLORS[index]
else
return CARTONAUGH_DEFAULT_COLOR
end
end
function customColor(color)
-- Some high number so it's not in the CARTONAUGH_COLORS array
cartonaugh_env_settings.color_index = 100
CARTONAUGH_DEFAULT_COLOR = color
end
-- Function to generate a kmap template
function generateKMap(column, row, grid_numb)
local outside_column_bits = 1
local outside_row_bits = 1
local outside_grid_numb_bits = 2
local return_str = ''
if row >= 4 then
outside_row_bits = 2
end
if column >= 4 then
outside_column_bits = 2
end
for c=1,column,1 do
for r=1,row,1 do
if r == 0 then
if c == 0 then
-- Do Nothing...
elseif c == (column+1) then
return_str = return_str .. ("\\phantom{" .. decimalToBin(0, outside_column_bits) .. "}")
else
return_str = return_str .. (decimalToGreyBin(c-1, outside_column_bits))
end
elseif r==(row+1) then
if c==0 then
return_str = return_str .. ("\\phantom{" .. decimalToBin(0, outside_column_bits) .. "}")
end
else
if c == 0 then
return_str = return_str .. (decimalToGreyBin(r-1, outside_row_bits))
elseif c == (column+1) then
else
return_str = return_str ..("|(" .. padString((decimalToGreyBin(grid_numb, outside_grid_numb_bits) .. decimalToGreyBin(c-1, outside_column_bits) .. decimalToGreyBin(r-1, outside_row_bits)), 6, 0) .. ")|" .. "\\phantom{0}")
--TODO: Look into why reversing c and r from where they should be makes it work
end
end
if r == (row) then
return_str = return_str .. ("\\\\")
else
return_str = return_str .. ("\\&")
end
end
end
return return_str
end
function init_cartonaught_env(numb_cols, numb_row, numb_submaps, is_bw, var12_str, var34_str, var56_str, is_submap_seperated)
-- Change the default texts depending on the number of submaps if not custom name has been given
-- TODO: Get this to actually work
if var12_str == 'X_1X_0' and var34_str == 'X_3X_2' and var56_str == 'X_5X_4' then
if numb_submaps == 1 then
var56_str = 'X_3'
end
if numb_cols == 1 then
var12_str = 'X_0'
end
end
cartonaugh_env_settings = {
cols = numb_cols,
rows = numb_row,
submaps = numb_submaps,
bw = tonumber(is_bw),
var_string = {
v12 = var12_str,
v34 = var34_str,
v56 = var56_str,
},
color_index = 1,
implicant_settings = {
inner_sep = 0.35,
outer_sep = 0.55,
}
}
used_cells = {}
if is_submap_seperated == 0 then
is_submap_seperated = false
else
is_submap_seperated = true
end
draw_pgf_kmap(numb_cols, numb_row, numb_submaps, var12_str, var34_str, var56_str, is_submap_seperated)
end
function change_implicant_inner_spread(new_amount)
cartonaugh_env_settings.implicant_settings.inner_sep = new_amount
end
function change_implicant_outer_spread(new_amount)
cartonaugh_env_settings.implicant_settings.outer_sep = new_amount
end
-- Function to generate the k-maps
-- NOTE: Each variable/cell in the k-map is 1cm. This is so that everything alings with each other just be adding
-- the number of row and column. It's a bit of hack, but for now it will stay this way. Resizing of the matrix
-- will be done with the scale option in the future
function draw_pgf_kmap(column, row, submaps_n, var1, var2, var3, is_submap_seperated)
submaps_n = submaps_n-1
-- TODO: Transform the following settings variables into arguments
local is_multitable_seperated = is_submap_seperated -- Setting to determine if the graphs are drawn with a sperator line or distanced out
local graph_seperator = 1.5 -- Seperation lenght between kmaps if is_multitable_seperated=false
local kmaplevel_seperator_lenght = 0.1 -- Setting to determine the seperator line's thickness if is_multitable_seperated=true
local line_width = 0.015 -- Set the line thickness of things here
local zero_var_line_lenght = 0.75 -- The lenght of the line at the top-left corner of the kmap where the implacants reside
local column_header_numb_bits = ((column-1) // 2)+1
local row_header_numb_bits = ((row-1) // 2)+1
if is_multitable_seperated then graph_seperator = 0 end
for d=0,submaps_n,1 do
-- Find the top-left corner of each grid (seperated by 1 unit)
local grid_x_loc = (d % 2)*(column+graph_seperator)
local grid_y_loc = -(d // 2)*(row+graph_seperator)
-- localPrint(string.format("\\node[above] at (%f,%f) {\\small{%s}};", 0, 0, abimplecant))
if is_multitable_seperated then
if (d % 2) == 1 then
local add_heigh = 0
if d >= 2 then add_heigh = kmaplevel_seperator_lenght end
localPrint(string.format("\\fill[black] (%f,%f) rectangle (%f,%f);", grid_x_loc, grid_y_loc, grid_x_loc+kmaplevel_seperator_lenght, grid_y_loc-row-line_width-add_heigh))
grid_x_loc = grid_x_loc + kmaplevel_seperator_lenght
end
if d >= 2 then
localPrint(string.format("\\fill[black] (%f,%f) rectangle (%f,%f);", grid_x_loc, grid_y_loc, grid_x_loc+column+line_width, grid_y_loc-kmaplevel_seperator_lenght))
grid_y_loc = grid_y_loc - kmaplevel_seperator_lenght
end
end
-- Print out the top-left line corner with the variables
if (is_multitable_seperated == false) or (d==0) then
localPrint(string.format("\\draw[inner sep=0pt, outer sep=0pt] (%f, %f) -- (%f, %f);", grid_x_loc+line_width, grid_y_loc-line_width, grid_x_loc-zero_var_line_lenght, grid_y_loc+zero_var_line_lenght))
localPrint(string.format("\\node[left] at (%f,%f) {\\small{%s}};", grid_x_loc-0.3, grid_y_loc+0.3, var2))
localPrint(string.format("\\node[right] at (%f,%f) {\\small{%s}};", grid_x_loc-0.6, grid_y_loc+0.6, var1))
end
-- Print out the top boolean column header
if (is_multitable_seperated == false) or (d < 2) then
localPrint(string.format("\\matrix[matrix of nodes, ampersand replacement=\\&, column sep={1cm,between origins}, nodes={align=center,text width=1cm,inner sep=0pt}, anchor=south west, inner sep=0pt, outer sep=0pt] at (%f, %f) {",grid_x_loc,grid_y_loc+0.05))
for c=0, column-1, 1 do
localPrint(string.format("%s", decimalToGreyBin(c, column_header_numb_bits)))
if c ~= (column-1) then localPrint("\\&") end
end
localPrint("\\\\};")
end
-- Print out the side boolean row header
if (is_multitable_seperated == false) or (d%2 == 0) then
localPrint(string.format("\\matrix[matrix of nodes, ampersand replacement=\\&, row sep={1cm,between origins}, nodes={minimum height=1cm,inner sep=0pt, text height=2ex, text depth=0.5ex}, anchor=north east, inner sep=0pt, outer sep=0pt] at (%f, %f) {",grid_x_loc-0.05,grid_y_loc))
for r=0, row-1, 1 do
localPrint(string.format("%s \\\\", decimalToGreyBin(r, row_header_numb_bits)))
end
localPrint("};")
end
-- Print out the matrix
localPrint(string.format("\\matrix[matrix of nodes, ampersand replacement=\\&, column sep={1cm,between origins}, row sep={1cm,between origins}, nodes={rectangle,draw,minimum height=1cm,align=center,text width=1cm,inner sep=0pt, text height=2ex, text depth=0.5ex, line width=0.015cm}, anchor=north west, inner sep=0pt, outer sep=0pt] at (%f, %f) {%s};",
grid_x_loc, grid_y_loc, generateKMap(row, column, d)
))
-- Print out the buttom text saying which matrix is which
if (submaps_n > 0) then
if (is_multitable_seperated == false) then
localPrint(string.format("\\node[below] at (%f, %f) {%s = %s};", grid_x_loc+(column//2),grid_y_loc-row,var3, decimalToBin(d, 2)))
elseif (is_multitable_seperated == true) then
if (d < 2) then
localPrint(string.format("\\node[] at (%f, %f) {%s = %s};", grid_x_loc+(column//2), grid_y_loc+1, var3, decimalToBin(d, 2)))
end
if (d % 2 == 0) and (submaps_n > 2) then
localPrint(string.format("\\node[rotate=90] at (%f, %f) {%s = %s};", grid_x_loc-1, grid_y_loc-(row//2), var3, decimalToBin(d, 2)))
end
end
end
end
end
-- Function for drawing an implicant manually (meaning to give the start and endpoint, as well as optional
-- submaps for different ones)
function manual_draw_implicant(st, en, submaps_str)
local color_index = cartonaugh_env_settings.color_index
local max_submaps = cartonaugh_env_settings.submaps
local inner_spread = cartonaugh_env_settings.implicant_settings.inner_sep
st = tonumber(st)
en = tonumber(en)
local submap_arr = split(submaps_str, ',')
-- Check if the implacent selection
for s=1,table.getn(submap_arr),1 do
current_submap = tonumber(submap_arr[s])
if current_submap < max_submaps then
local draw_str = string.format("($(%s.center)+(-%s,%s)$) rectangle ($(%s.center)+(%s,-%s)$)", decimalToGreyBin(current_submap, 2) .. decimalToBin(st,4), inner_spread, inner_spread, decimalToGreyBin(current_submap, 2) .. decimalToBin(en,4), inner_spread, inner_spread)
if cartonaugh_env_settings.bw == 0 then
localPrint(string.format("\\fill[rounded corners=3pt,fill=%s,fill opacity=0.25,] {%s};", getColor(color_index) , draw_str))
end
localPrint(string.format("\\draw[rounded corners=3pt,draw opacity=1.0,] {%s};", draw_str))
else
localPrint(string.format("\\PackageWarning{cartonaugh}{You can only draw on existing sub maps. Ignoring instruction to draw on non existing sub map number %d}", s))
end
end
cartonaugh_env_settings.color_index = cartonaugh_env_settings.color_index+1
end
-- Handler function for drawing edge implacants, figuring out orientation as well
function manual_draw_edge_implicant(corner1, corner2, corner3, corner4, submaps_str)
corner1 = tonumber(corner1)
corner2 = tonumber(corner2)
corner3 = tonumber(corner3)
corner4 = tonumber(corner4)
if corner1-corner2 > corner1-corner3 then
manual_draw_edge_implicant_orientation(corner1, corner2, submaps_str, 'n')
manual_draw_edge_implicant_orientation(corner3, corner4, submaps_str, 's')
else
manual_draw_edge_implicant_orientation(corner1, corner2, submaps_str, 'w')
manual_draw_edge_implicant_orientation(corner3, corner4, submaps_str, 'e')
end
cartonaugh_env_settings.color_index = cartonaugh_env_settings.color_index+1
end
-- Function to draw out a 1 edge implacant given 2 corners, the submaps, and the orientation (n, s, e, or w)
-- TODO: Perhaps find a way to repeat the code between n/s and e/w
-- TODO: For the mirror variable, have that apply to the lua var directly instead of having LaTeX handle it
-- TODO: Maybe add option for squigly lines for the end instead of nothing
-- TODO: Open up the internal inner_spread and outer_spead settings to the user
function manual_draw_edge_implicant_orientation(corner1, corner2, submaps_str, orientation)
local color_index = cartonaugh_env_settings.color_index
local max_submaps = cartonaugh_env_settings.submaps
corner1 = tonumber(corner1)
corner2 = tonumber(corner2)
local submap_arr = split(submaps_str, ',')
local inner_spread = cartonaugh_env_settings.implicant_settings.inner_sep
local outer_spead = cartonaugh_env_settings.implicant_settings.outer_sep
-- Check if the implacent selection
for s=1,table.getn(submap_arr),1 do
current_submap = tonumber(submap_arr[s])
if current_submap < max_submaps then
local draw_string = ""
local mirror = 1
local corner1_bin = decimalToGreyBin(current_submap, 2) .. decimalToBin(corner1,4)
local corner2_bin = decimalToGreyBin(current_submap, 2) .. decimalToBin(corner2,4)
if orientation == 'n' or orientation == 's' then
-- If the orientation is south, just mirror it
if orientation == 's' then mirror = -1 end
draw_string = string.format("($(%s.center)+(-%f,%f*%s)$)", corner1_bin, inner_spread, outer_spead, mirror)
draw_string = draw_string .. string.format("{[rounded corners=3pt] -- ($(%s.center)+(-%f,-%f*%s)$)}", corner1_bin, inner_spread, inner_spread, mirror)
draw_string = draw_string .. string.format("{[rounded corners=3pt] -- ($(%s.center)+(%f,-%f*%s)$)}", corner2_bin, inner_spread, inner_spread, mirror)
draw_string = draw_string .. string.format("-- ($(%s.center)+(%f,%f*%s)$)", corner2_bin, inner_spread, outer_spead, mirror)
else
if orientation == 'e' then mirror = -1 end
draw_string = string.format("($(%s.center)+(-%f*%s,%f)$)", corner1_bin, outer_spead, mirror, inner_spread)
draw_string = draw_string .. string.format("{[rounded corners=3pt] -- ($(%s.center)+(%f*%s,%f)$)}", corner1_bin, inner_spread, mirror, inner_spread)
draw_string = draw_string .. string.format("{[rounded corners=3pt] -- ($(%s.center)+(%f*%s,-%f)$)}", corner2_bin, inner_spread, mirror, inner_spread)
draw_string = draw_string .. string.format("-- ($(%s.center)+(-%f*%s,-%f)$)", corner2_bin, outer_spead, mirror, inner_spread)
end
if cartonaugh_env_settings.bw == 0 then
localPrint(string.format("\\fill[fill=%s,fill opacity=0.25,] {%s};", getColor(color_index), draw_string))
end
localPrint(string.format("\\draw[draw opacity=1.0] {%s};", draw_string))
else
localPrint(string.format("\\PackageWarning{cartonaugh}{You can only draw on existing sub maps. Ignoring instruction to draw on non existing sub map number %d}", s))
end
end
end
-- Function to draw the corner implicants. Only usable on 4x4 matrices
function manual_draw_corner_implicant(submaps_str)
local corner_arr = {0, 2, 8, 10}
local submap_arr = split(submaps_str, ',')
local max_submaps = cartonaugh_env_settings.submaps
local color = cartonaugh_env_settings.color_index
local inner_spread = cartonaugh_env_settings.implicant_settings.inner_sep
local outer_spread = cartonaugh_env_settings.implicant_settings.outer_sep
if cartonaugh_env_settings.cols ~= 4 or cartonaugh_env_settings.rows ~= 4 then
localPrint("\\PackageError{cartonaugh}{Cannot use a corner implicant on anything but a 4x4. Sorry!}")
return
end
for s=1,table.getn(submap_arr),1 do
current_submap = tonumber(submap_arr[s])
if current_submap < max_submaps then
for c=0,3,1 do
local x_mirror = 1-(2*(c%2))
local y_mirror = 1-(2*(c//2))
local corner = decimalToGreyBin(current_submap, 2) .. decimalToBin(corner_arr[c+1],4)
-- Create the string to draw the corners
local draw_string = string.format("($(%s.center)+(-%f*%d,%f*%d)$)--($(%s.center)+(%f*%d,%f*%d)$)", corner, outer_spread, x_mirror, outer_spread, y_mirror, corner, inner_spread, x_mirror, outer_spread, y_mirror)
draw_string = draw_string .. string.format("{ [rounded corners=3pt] --($(%s.center)+(%f*%d,-%f*%d)$) }", corner, inner_spread, x_mirror, inner_spread, y_mirror)
draw_string = draw_string .. string.format("--($(%s.center)+(-%f*%d,-%f*%d)$) -- cycle", corner, outer_spread, x_mirror, inner_spread, y_mirror)
if cartonaugh_env_settings.bw == 0 then
localPrint(string.format("\\fill[fill=%s,fill opacity=0.25,] {%s};", getColor(color), draw_string))
end
localPrint(string.format("\\draw[sharp corners, draw opacity=1.0] {%s};", draw_string))
end
else
localPrint(string.format("\\PackageWarning{cartonaugh}{You can only draw on existing sub maps. Ignoring instruction to draw on non existing sub map number %d}", s))
end
end
cartonaugh_env_settings.color_index = cartonaugh_env_settings.color_index+1
end
-- WORK IN PROGRESS/LONG TERM FUNCTION
-- Goals is to eventually give \implicant{1}{x}{0}{x} for example and have it draw it out for you.
-- May give up on this in favor of other things...don't know
function draw_implicant(var_list)
-- local var_list_arr = split(var_list, ',')
local color_index = cartonaugh_env_settings.color_index
local max_submaps = cartonaugh_env_settings.submaps
localPrint("\\PackageWarning{cartonaugh}{This is a UNSTABLE and WIP function. Procede on your own}")
-- Check argument for submaps greater than 1
-- TODO: before returning print out a package error
if max_submaps > 1 then
for s=0,max_submaps-1,1 do
if var_list[5+s] == '' then
-- TODO: Fix this
localPrint(string.format("\\PackageError{cartonaugh}{Please feed either 1, 0, or x for sub map number %d's variable boolean}", s+1))
return
end
end
end
local low_limit = 0
local high_limit = 0
for b=1,4,1 do
if var_list[b] == 'x' then
print('a',high_limit, greyBinToDecimal(high_limit), decimalToBin(greyBinToDecimal(high_limit), 4))
print('b',high_limit | (1 << (b-1)), greyBinToDecimal(high_limit | (1 << (b-1))), decimalToBin(greyBinToDecimal(high_limit | (1 << (b-1))), 4))
if greyBinToDecimal(high_limit) < greyBinToDecimal(high_limit | (1 << (b-1))) then
high_limit = high_limit | (1 << (b-1))
print(decimalToBin(greyBinToDecimal(high_limit),4), decimalToBin(greyBinToDecimal(high_limit & ~(1 << (b-2))),4))
if greyBinToDecimal(high_limit) < greyBinToDecimal(high_limit & ~(1 << (b-2))) then
high_limit = high_limit & ~(1 << (b-2))
end
end
elseif var_list[b] == '1' then
high_limit = high_limit | (1 << (b-1))
low_limit = low_limit | (1 << (b-1))
end
print('low_limit=', low_limit, 'high_limit=', high_limit)
end
local st = decimalToBin(low_limit,4)
local en = decimalToBin(high_limit,4)
print('low_limit', greyBinToDecimal(low_limit), 'high_limit', greyBinToDecimal(high_limit), 'st=', st, 'en=', en)
for s=1,1,1 do
current_submap = 0
if current_submap < max_submaps then
if cartonaugh_env_settings.bw == 0 then
localPrint(string.format("\\fill[rounded corners=3pt,fill=%s,fill opacity=0.25,] {($(%s.center)+(-0.3,0.3)$) rectangle ($(%s.center)+(0.3,-0.3)$)};", getColor(color_index) , decimalToGreyBin(current_submap, 2) .. st, decimalToGreyBin(current_submap, 2) .. en))
color_index = color_index+1
end
localPrint(string.format("\\draw[rounded corners=3pt,draw opacity=1.0,] {($(%s.center)+(-0.3,0.3)$)rectangle($(%s.center)+(0.3,-0.3)$)};", decimalToGreyBin(current_submap, 2) .. st, decimalToGreyBin(current_submap, 2) .. en))
else
localPrint(string.format("\\PackageWarning{cartonaugh}{You can only draw on existing sub maps. Ignoring instruction to draw on non existing sub map number %d}", s))
end
end
cartonaugh_env_settings.color_index = color_index
end
function autoterms(what_to_write)
local max_cells = cartonaugh_env_settings.submaps * cartonaugh_env_settings.cols * cartonaugh_env_settings.rows
for cell=0,max_cells-1,1 do
if used_cells[cell] == nil then
used_cells[cell] = true
localPrint(string.format("\\path (%s) node {%s};", decimalToBin(cell, 6), what_to_write))
end
end
end
function manualterms(what_to_write)
local what_to_write_arr = split(what_to_write, ',')
for cell=0,table.getn(what_to_write_arr)-1,1 do
if used_cells[cell] == nil then
used_cells[cell] = true
localPrint(string.format("\\path (%s) node {%s};", decimalToBin(cell, 6), what_to_write_arr[cell+1]))
end
end
end
function write_to_cell(cells, what_to_write)
local cells_arr = split(cells, ',')
for c=1,table.getn(cells_arr),1 do
local cell = tonumber(cells_arr[c])
if used_cells[cell] == nil then
used_cells[cell] = true
localPrint(string.format("\\path (%s) node {%s};", decimalToBin(cell, 6), what_to_write))
end
end
end
%\end{macrocode}
%</luacode>
%\fi
%
\endinput