diff --git a/algo/ca/TrackingChain.h b/algo/ca/TrackingChain.h
index 93185cc7af96e3e36595f6a456449f8ae2e68516..9adc9528daf7b5959b6083f5cf3ee4e4c6427e98 100644
--- a/algo/ca/TrackingChain.h
+++ b/algo/ca/TrackingChain.h
@@ -117,7 +117,7 @@ namespace cbm::algo
     /// \note  Indexing: [globalHitID], value: (DetID, partitionID, hitID)
     ca::Vector<std::tuple<ca::EDetectorID, uint32_t, uint32_t>> faHitExternalIndices{"faHitExternalIndices"};
 
-    static constexpr bool kDEBUG = true;  ///< Debug mode
+    static constexpr bool kDEBUG = false;  ///< Debug mode
   };
 
 
diff --git a/core/qa/report/CbmQaReportBeamerEngine.cxx b/core/qa/report/CbmQaReportBeamerEngine.cxx
index 02dde8dd227f3574255eb9571fd42f0f30393468..c829e29a96daea14f71a23786b3578b2d1b2848c 100644
--- a/core/qa/report/CbmQaReportBeamerEngine.cxx
+++ b/core/qa/report/CbmQaReportBeamerEngine.cxx
@@ -11,20 +11,25 @@
 
 #include "CbmQaReportFigure.h"
 #include "CbmQaReportHeader.h"
+#include "CbmQaReportLatexFormat.h"
 #include "CbmQaReportSection.h"
 #include "CbmQaReportTable.h"
 #include "CbmQaReportTail.h"
 #include "Logger.h"
 
+#include <boost/algorithm/string/join.hpp>
+
 #include <chrono>
 #include <ctime>
 #include <iomanip>
+#include <numeric>
 #include <regex>
 #include <sstream>
 
 using cbm::qa::report::BeamerEngine;
 using cbm::qa::report::Figure;
 using cbm::qa::report::Header;
+using cbm::qa::report::LatexFormat;
 using cbm::qa::report::Section;
 using cbm::qa::report::Table;
 using cbm::qa::report::Tail;
@@ -33,7 +38,86 @@ using cbm::qa::report::Tail;
 //
 std::string BeamerEngine::FigureBody(const Figure& figure) const
 {
+  size_t nPlots = figure.GetPlots().size();
+  if (nPlots == 0) {
+    LOG(warn) << "No plots provided for figure " << figure.GetLabel() << ". Ignoring.";
+    return "";
+  }
+
   std::stringstream out;
+  out << "\\begin{frame}{" << LatexFormat::Apply(figure.GetMother()->GetTitle()) << "}\n";
+  out << "  \\begin{figure}\n";
+  out << "    \\def\\hgt{\\textheight-2\\baselineskip}\n";
+  out << "    \\centering\n";
+  if (nPlots == 1) {
+    out << "    \\begin{adjustbox}{width=\\textwidth, totalheight=0.9\\hgt, keepaspectratio}\n";
+    out << "      \\includegraphics[height=0.75\\paperheight]{" << figure.GetPlots()[0].fsPath << "}\n";
+    out << "    \\end{adjustbox}\n";
+  }
+  else {
+    std::vector<size_t> vPlotGrid;
+    bool bDefineDefaultGrid = figure.GetPlotGrid().empty();
+    if (!bDefineDefaultGrid) {
+      size_t nPlotsByGrid = std::accumulate(figure.GetPlotGrid().begin(), figure.GetPlotGrid().end(), 0);
+      if (nPlotsByGrid < nPlots) {
+        LOG(warn) << "Figure " << figure.GetLabel()
+                  << ": provided grid does not fit the subfigures, define the default one";
+        bDefineDefaultGrid = true;
+      }
+    }
+
+    if (bDefineDefaultGrid) {
+      if (nPlots < 4) {
+        vPlotGrid.resize(1, nPlots);
+      }
+      else {
+        int nY = nPlots < 13 ? 2 : 3;
+        vPlotGrid.resize(nY, size_t(std::ceil(double(nPlots) / nY)));
+      }
+    }
+    else {
+      vPlotGrid = figure.GetPlotGrid();
+    }
+
+    size_t iPlot = 0;
+    double hSize = 0.9 / vPlotGrid.size();
+    for (size_t iY = 0; iY < vPlotGrid.size(); ++iY) {
+      if (iPlot >= nPlots) {
+        break;
+      }
+      size_t nX = vPlotGrid[iY];
+      if (nX == 0) {
+        continue;
+      }
+      out << "    \\begin{adjustbox}{width=\\textwidth, totalheight=" << hSize << "\\hgt, keepaspectratio}\n";
+      for (size_t iX = 0; iX < nX; ++iX) {
+        if (iPlot >= nPlots) {
+          break;
+        }
+        const auto& plot = figure.GetPlots()[iPlot];
+        out << "      \\stackunder[1pt]{\\includegraphics[height=0.75\\paperheight]{" << plot.fsPath << "}}{";
+        out << plot.fsCaption << "}\n";
+        if (iX == nX - 1) {
+          out << '\n';
+        }
+        else {
+          out << "      \\quad\n";
+        }
+        ++iPlot;
+      }
+      out << "    \\end{adjustbox}\n\n";
+      if (iY != vPlotGrid.size() - 1) {
+        out << '\n';
+      }
+    }
+  }
+
+  if (!figure.GetCaption().empty()) {
+    out << "    \\caption{" << LatexFormat::Apply(figure.GetCaption()) << "}\n";
+  }
+  out << "    \\label{" << figure.GetLabel() << "}\n";
+  out << "  \\end{figure}\n";
+  out << "\\end{frame}\n";
   return out.str();
 }
 
@@ -42,14 +126,127 @@ std::string BeamerEngine::FigureBody(const Figure& figure) const
 std::string BeamerEngine::HeaderBody(const Header& header) const
 {
   std::stringstream out;
+
+  // TODO: Move these definitions to text-file (?)
+  // Settings
+
+  out << "\\documentclass[aspectratio=169,xcolor=dvipsnames]{beamer}\n";
+  out << "\\usetheme{Madrid}\n";
+  out << "\\useinnertheme{circles}\n";
+  out << "\\setbeamerfont{structure}{family=\\sffamily,series=\\mdseries}\n";
+  out << "\\setbeamerfont{title}{size=\\LARGE,parent=structure}\n";
+  out << "\\setbeamerfont{subtitle}{size=\\normalsize,parent=title}\n";
+  out << "\\setbeamerfont{date}{size=\\scriptsize,series=\\mdseries,parent=structure}\n";
+  out << "\\setbeamerfont{author}{size=\\Large,series=\\mdseries,parent=structure}\n";
+  out << "\\setbeamerfont{institute}{size=\\scriptsize,series=\\mdseries,parent=structure}\n";
+  out << "\\setbeamerfont{section in toc}{size=\\Large,parent=structure}\n";
+  out << "\\setbeamerfont{section in head/foot}{size=\\tiny,parent=structure}\n";
+  out << "\\setbeamerfont{subsection in toc}{size=\\large,parent={section in toc}}\n";
+  out << "\\setbeamerfont{frametitle}{parent=structure,size=\\LARGE}\n";
+  out << "\\setbeamerfont{framesubtitle}{parent=frametitle,size=\\Large}\n";
+  out << "\\setbeamerfont{caption}{size=\\footnotesize}\n";
+  out << "\\setbeamerfont{item}{parent=structure,series=\\mdseries}\n";
+  out << "\\setbeamerfont{block title}{size=\\large,series=\\mdseries,parent={structure,block body}}\n";
+  out << "\\definecolor{InvisibleRed}{rgb}{0.92, 0.9, 0.9}\n";
+  out << "\\definecolor{InvisibleGreen}{rgb}{0.9, 0.92, 0.9}\n";
+  out << "\\definecolor{InvisibleBlue}{rgb}{0.9, 0.9, 0.92}\n";
+  out << "\\definecolor{LightBlue}{rgb}{0.4, 0.55, 0.65}\n";
+  out << "\\definecolor{LightBlue}{rgb}{0.4, 0.55, 0.65}\n";
+  out << "\\definecolor{MediumRed}{rgb}{0.92549, 0.34509, 0.34509}\n";
+  out << "\\definecolor{MediumGreen}{rgb}{0.36862, 0.66666, 0.65882}\n";
+  out << "\\definecolor{MediumBlue}{rgb}{0.01176, 0.31372, 0.43529}\n";
+  out << "\\definecolor{DarkBlue}{rgb}{0.05, 0.15, 0.3} \n";
+  out << "\\usecolortheme[named=DarkBlue]{structure}\n";
+  out << "\\setbeamercolor{alerted text}{fg=LightBlue}\n";
+  out << "\\setbeamercolor{palette primary}{bg=DarkBlue,fg=white}\n";
+  out << "\\setbeamercolor{palette secondary}{bg=MediumBlue,fg=white}\n";
+  out << "\\setbeamercolor{palette tertiary}{bg=LightBlue,fg=white}\n";
+  out << "\\setbeamercolor{block title}{bg=MediumBlue}\n";
+  out << "\\setbeamercolor{block body}{bg=InvisibleBlue}\n";
+  out << "\\setbeamercolor{block title example}{bg=MediumGreen}\n";
+  out << "\\setbeamercolor{block body example}{bg=InvisibleGreen}\n";
+  out << "\\setbeamercolor{block title alerted}{bg=MediumRed}\n";
+  out << "\\setbeamercolor{block body alerted}{bg=InvisibleRed}\n";
+  out << "\\setbeamertemplate{footline}[page number]\n";
+  out << "\\setbeamertemplate{navigation symbols}{}\n";
+  out << "\\setbeamertemplate{blocks}[rounded][shadow=true]\n";
+  out << "\\setbeamertemplate{enumerate items}[default]\n";
+  out << "\\setbeamertemplate{section in toc}[sections numbered]\n";
+  out << "\\setbeamertemplate{subsection in toc}[default]\n";
+
+  out << "\\usepackage{hyperref}\n";
+  out << "\\usepackage{graphicx}\n";
+  out << "\\usepackage{booktabs}\n";
+  out << "\\usepackage{listings}\n";
+  out << "\\usepackage{amsmath}\n";
+  out << "\\usepackage{listings}\n";
+  out << "\\usepackage{xcolor}\n";
+  //out << "\\usepackage{adjustbox}\n";
+  out << "\\usepackage{subfig}\n";
+  out << "\\usepackage{hyperref}\n";
+  out << "\\usepackage{stackengine}\n";
+  out << "\\usepackage{longtable}\n";
+  out << "\\usepackage[export]{adjustbox}\n";
+
+  out << "\\setbeamertemplate{caption}[numbered]\n";
+
+  //out << "\\usepackage{color}\n";
+  //out << "\\definecolor{hrefcolor}{rgb}{0.2, 0.2, 0.6}\n";
+  //out << "\\hypersetup{color}{colorlinks=true, urlcolor=hrefcolor,linkcolor=black,citecolor=hrefcolor}\n";
+  if (!header.GetPageHeader().empty()) {
+    //out << "\\usepackage{fancyhdr}\n\\pagestyle{fancy}\n\\lhead{"
+    //    << LatexFormat::Apply(header.GetPageHeader()) << "}\n";
+  }
+  out << "\n";
+  // Title page settings
+  out << "\\title{" << LatexFormat::Apply(header.GetTitle()) << "}\n";
+  out << "\\subtitle{" << LatexFormat::Apply(header.GetSubtitle()) << "}\n";
+  auto timeNow = std::chrono::system_clock::to_time_t(std::chrono::system_clock::now());
+  out << "\\date{Generated on " << std::put_time(std::localtime(&timeNow), "%a %d.%m.%Y, %X") << "}\n";
+
+
+  // Title page
+  out << "\\begin{document}\n\n";
+  out << "\\begin{frame}\n";
+  out << "  \\titlepage\n";
+  out << "  \\begin{tabular}{r@{ }l}\n";
+  out << "     author: & " << header.GetAuthor() << " \\\\\n";
+  out << "     setup: & " << header.GetSetup() << "\\\\\n";
+  out << "     tags: & " << boost::algorithm::join(header.GetTags(), ", ") << '\n';
+  out << "  \\end{tabular}\n";
+  out << "\\end{frame}\n";
+
+  // Table of contents
+  out << "\\begin{frame}{Outline}\n";
+  out << "  \\tableofcontents\n";
+  out << "\\end{frame}\n";
+
+  out << '\n';
   return out.str();
 }
 
+
 // ---------------------------------------------------------------------------------------------------------------------
 //
 std::string BeamerEngine::SectionBody(const Section& section) const
 {
   std::stringstream out;
+  std::string sSectionType = "";
+  if (section.GetLevel() == 0) {
+    sSectionType = "section";
+  }
+  else if (section.GetLevel() == 1) {
+    sSectionType = "subsection";
+  }
+  else {
+    sSectionType = "subsubsection";
+  }
+
+  out << "\\" << sSectionType << "{" << section.GetTitle() << "}\n\n";
+  for (const auto& pElement : section.GetDaughterElements()) {
+    out << pElement->GetBody(*this) << '\n';
+  }
+  out << '\n';
   return out.str();
 }
 
@@ -58,6 +255,37 @@ std::string BeamerEngine::SectionBody(const Section& section) const
 std::string BeamerEngine::TableBody(const Table& table) const
 {
   std::stringstream out;
+  int nRows = table.GetNofRows();
+  int nCols = table.GetNofCols();
+
+  out << "\\begin{frame}{" << table.GetMother()->GetTitle() << "}\n";
+  out << "  \\begin{table}\n";
+  out << "  \\footnotesize\n";
+  // tabular header
+  if (!table.GetCaption().empty()) {
+    out << "    \\caption{" << LatexFormat::Apply(table.GetCaption()) << "}\n";
+  }
+  out << "    \\begin{longtable}{" << std::string(nCols, 'c') << "}\n";
+  // table header
+  out << "      \\hline\n";
+  out << "    ";
+  for (int iCol = 0; iCol < nCols; ++iCol) {
+    out << table.GetColumnTitle(iCol) << ((iCol < nCols - 1) ? " & " : " \\\\\n");
+  }
+  out << "      \\hline\n";
+  // table body
+  for (int iRow = 0; iRow < nRows; ++iRow) {
+    out << "    ";
+    for (int iCol = 0; iCol < nCols; ++iCol) {
+      out << table(iRow, iCol) << ((iCol < nCols - 1) ? " & " : " \\\\\n");
+    }
+  }
+  out << "      \\hline\n";
+
+  out << "    \\end{longtable}\n";
+  out << "    \\label{" << table.GetLabel() << "}\n";
+  out << "  \\end{table}\n";
+  out << "\\end{frame}\n";
   return out.str();
 }
 
@@ -66,9 +294,44 @@ std::string BeamerEngine::TableBody(const Table& table) const
 std::string BeamerEngine::TailBody(const Tail& figure) const
 {
   std::stringstream out;
+  out << "\\end{document}\n";
   return out.str();
 }
 
 // ---------------------------------------------------------------------------------------------------------------------
 //
-void BeamerEngine::Compile(const std::string& source) const {}
+void BeamerEngine::Compile(const std::string& source) const
+{
+  // Search for compiler program in path
+  std::string compiler = fsLatexCompiler.substr(0, fsLatexCompiler.find_first_of(' '));  // compiler name without opts
+  int bCompilerFonud   = false;
+  {
+    std::stringstream stream(getenv("PATH"));
+    std::string dir;
+    while (!bCompilerFonud && std::getline(stream, dir, ':')) {
+      bCompilerFonud = fs::exists(fs::path(dir) / compiler);
+    }
+  }
+
+  if (!bCompilerFonud) {
+    LOG(error) << "cbm::qa::report::BeamerEngine::Compile(): compiler \"" << fsLatexCompiler << "\" not found in PATH. "
+               << "Please use another compiler or compile the sourec in external program";
+    return;
+  }
+
+  // Execute compiler
+  fs::path sourceDir = fs::path(source).parent_path();
+  fs::path currPath  = fs::current_path();
+  fs::current_path(sourceDir);
+
+  std::string logFile = compiler + ".log";
+  std::string command = fmt::format("{} {} > {}", fsLatexCompiler, source, logFile);
+  command             = command + ";" + command;  // compile two times to get contents
+
+  LOG(info) << "cbm::qa::report::BeamerEngine::Compile(): executing command: \n\t" << command;
+  int res = std::system(command.c_str());
+  fs::current_path(currPath);
+
+  LOG(info) << "cbm::qa::report::BemaerEngine::Compile(): compillation " << (res == 0 ? "succeed" : "failed")
+            << "(compiler log: \"" << logFile << "\")";
+}
diff --git a/core/qa/report/CbmQaReportElement.h b/core/qa/report/CbmQaReportElement.h
index 81c99468f7701bdc76d1e34ee0f19f514c196e94..e469f10425492e7658938840ff79873088e74cdc 100644
--- a/core/qa/report/CbmQaReportElement.h
+++ b/core/qa/report/CbmQaReportElement.h
@@ -23,6 +23,8 @@ namespace cbm::qa::report
   /// \brief Interface for the report element
   ///
   class Element {
+    friend class CollapsibleElement;
+
    public:
     /// \brief Constructor
     /// \param label  Element label
@@ -51,10 +53,16 @@ namespace cbm::qa::report
     /// \brief Sets name
     void SetTitle(std::string_view title) { fsTitle = title; }
 
+    /// \brief Gets mother element
+    const Element* const GetMother() const { return fpMother; }
+
    private:
+    void AssignMother(const Element* pMother) { fpMother = pMother; }
+
     // TODO: Add check for multiple label definition
-    std::string fsLabel = "";  ///< Label for referencing
-    std::string fsTitle = "";  ///< Title of the element
+    const Element* fpMother = nullptr;  ///< Mother element
+    std::string fsLabel     = "";       ///< Label for referencing
+    std::string fsTitle     = "";       ///< Title of the element
   };
 
   /// \class CollapsibleElement
@@ -69,7 +77,11 @@ namespace cbm::qa::report
 
     /// \brief Adds element
     // TODO: Check for label
-    virtual void Add(std::shared_ptr<Element> pElement) { fvDaughterElements.push_back(pElement); }
+    virtual void Add(std::shared_ptr<Element> pElement)
+    {
+      pElement->AssignMother(this);
+      fvDaughterElements.push_back(pElement);
+    }
 
     /// \brief Get daughter elements
     const std::vector<std::shared_ptr<Element>> GetDaughterElements() const { return fvDaughterElements; }
diff --git a/core/qa/report/CbmQaReportHeader.h b/core/qa/report/CbmQaReportHeader.h
index 7b457f7ee2e38d1f5662fd747e11b4ce02622913..622e67689967e018bddc47ca6295fd8b7194694a 100644
--- a/core/qa/report/CbmQaReportHeader.h
+++ b/core/qa/report/CbmQaReportHeader.h
@@ -23,6 +23,9 @@ namespace cbm::qa::report
     /// \brief Destructor
     virtual ~Header() = default;
 
+    /// \brief Add tag
+    void AddTag(std::string_view tag) { fvsTags.emplace_back(tag); }
+
     /// \brief Gets body of the element
     /// \param engine  A concrete implementation of the Engine to get the element body
     std::string GetBody(const Engine& engine) const { return engine.HeaderBody(*this); }
@@ -39,6 +42,9 @@ namespace cbm::qa::report
     /// \brief Gets subtitle
     const std::string& GetSubtitle() const { return fsSubtitle; }
 
+    /// \brief Gets tags
+    const std::vector<std::string>& GetTags() const { return fvsTags; }
+
     /// \brief Gets title
     const std::string& GetTitle() const { return fsTitle; }
 
@@ -58,6 +64,8 @@ namespace cbm::qa::report
     void SetTitle(std::string_view title) { fsTitle = title; }
 
    private:
+    std::vector<std::string> fvsTags{};  ///< Different Tags
+
     std::string fsAuthor     = "";
     std::string fsSetup      = "";
     std::string fsSubtitle   = "";
diff --git a/core/qa/report/CbmQaReportLatexFormat.h b/core/qa/report/CbmQaReportLatexFormat.h
index afdfc157cc82164142459c213e01571b11c16998..733cba42b873a4c04833eaa2a4519dc251d8f419 100644
--- a/core/qa/report/CbmQaReportLatexFormat.h
+++ b/core/qa/report/CbmQaReportLatexFormat.h
@@ -21,4 +21,4 @@ namespace cbm::qa::report
     /// \brief Applies a LaTeX-friendly formatting
     static std::string Apply(std::string_view input);
   };
-}  // namespace cbm::qa::report
\ No newline at end of file
+}  // namespace cbm::qa::report
diff --git a/macro/qa/qa_report_ca_offline.C b/macro/qa/qa_report_ca_offline.C
index 353ac5ba11426baae50bd09b7f415470653f378f..e314c941c2ac64c07496ec7b3767f87984dd53ae 100644
--- a/macro/qa/qa_report_ca_offline.C
+++ b/macro/qa/qa_report_ca_offline.C
@@ -206,6 +206,6 @@ void qa_report_ca_offline(
     }
   }
 
-  cbm::qa::report::LatexEngine latex;
+  cbm::qa::report::BeamerEngine latex;
   pReport->CreateScript(latex);
 }