flask_httpauth
flask
werkzeug
+jinja2
--- /dev/null
+{
+ "shoot": {
+ "date": "30th March 2025",
+ "location": "MediaCity UoS, Room 3.08"
+ },
+ "crew":[
+ {
+ "name": "William Greenwood",
+ "role": "Producer",
+ "phone": "075 9476 8180",
+ "email": "W.Greenwood@edu.salford.ac.uk",
+ "absent": false
+ },
+ {
+ "name": "Finn Downton",
+ "role": "Narrative enginneer",
+ "phone": "079 1003 1116",
+ "email": "E.Downton@edu.salford.ac.uk",
+ "absent": false
+ },
+ {
+ "name": "Jack Christian-Sims",
+ "role": "Director",
+ "phone": "075 4297 4087",
+ "email": "J.Christian-Sims@edu.salford.ac.uk",
+ "absent": false
+ },
+ {
+ "name": "Eleanor Haughton",
+ "role": "Vision Mixer",
+ "phone": "074 9127 2723",
+ "email": "E.V.Haughton@edu.salford.ac.uk",
+ "absent": false
+ },
+ {
+ "name": "Heather Digwood",
+ "role": "Floor manager",
+ "phone": "077 8817 3853",
+ "email": "H.Digwood@edu.salford.ac.uk",
+ "absent": false
+ },
+ {
+ "name": "Olivia Gillett",
+ "role": "Floor manager",
+ "phone": "073 6852 4302",
+ "email": "O.Gillett@edu.salford.ac.uk",
+ "absent": false
+ },
+ {
+ "name": "Leo Garside-Holdich",
+ "role": "Camera opperator",
+ "phone": "078 7474 0891",
+ "email": "L.N.Garside-Holdich@edu.salford.ac.uk",
+ "absent": false
+ },
+ {
+ "name": "Rebecca Dixon",
+ "role": "Sound opperator",
+ "phone": "074 3400 7701",
+ "email": "R.Dixon2@edu.salford.ac.uk",
+ "absent": false
+ },
+ {
+ "name": "Alex Roberts",
+ "role": "Lighting opperator",
+ "phone": "074 9660 1832",
+ "email": "A.Roberts17@edu.salford.ac.uk",
+ "absent": true
+ },
+ {
+ "name": "Tegan Blake-Barnard",
+ "role": "Set-design lead",
+ "phone": "075 4893 0872",
+ "email": "T.blake-barnard@edu.Salford.ac.uk",
+ "absent": false
+ },
+ {
+ "name": "Lydia Wilkinson",
+ "role": "Set-design / GFX opperator",
+ "phone": "078 0269 0037",
+ "email": "L.wilkinson14@edu.salford.ac.uk",
+ "absent": false
+ },
+ {
+ "name": "Leigha Blanchard",
+ "role": "Set-design / GFX opperator",
+ "phone": "074 7700 0501",
+ "email": "L.blanchard@salford.ac.uk",
+ "absent": false
+ },
+ {
+ "name": "Daisy Devoe",
+ "role": "Set-design / GFX opperator",
+ "phone": "079 7056 4500",
+ "email": "D.Devoe@edu.salford.ac.uk",
+ "absent": false
+ },
+ {
+ "name": "Brynn Yates",
+ "role": "GFX opperator",
+ "phone": "073 7766 9812",
+ "email": "M.Yates5@edu.salford.ac.uk",
+ "absent": false
+ }
+ ],
+ "cast":[
+ {
+ "name": "David Smith",
+ "role": "Anchor",
+ "phone": "078 8839 8003",
+ "email": "D.Smith51@edu.salford.ac.uk",
+ "absent": false
+ }
+ ],
+ "schedule":[
+ {
+ "time": "0000",
+ "title": "This is an action",
+ "extra": [
+ "Test"
+ ]
+ },
+ {
+ "time": "0000",
+ "title": "This is an action",
+ "extra": [
+ "Test"
+ ]
+ },
+ {
+ "time": "0000",
+ "title": "This is an action",
+ "extra": [
+ "Test"
+ ]
+ }
+ ],
+ "notes": ["These are some notes on what we'll be doing"]
+}
"THE THREAT",
"I used to be so unimaginably tall!",
"Doomsday",
+ "Lets eat grandma",
"It's here!"
],
"banner":[
"rating":"★★★★★",
"subtext":"Starting soon!",
"description":"The show will be starting soon!",
+ "currency":"£",
+ "prefix":true,
"origional_price":1,
+ "gallery_price":1,
+ "cost_price":1,
"stock_count":1,
- "notes":"<p>[autocue] Showing: test product.<br>Commencing shortly.</p>",
+ "notes":"[autocue] Showing: test product.<br>Commencing shortly.",
"crew_notes":"Test crew notes"
},
{
"rating":"★★★★★",
"subtext":"Hyperclean™! Teethpaste",
"description":"Hyperclean™! Professional Dentist-grade Teethpaste with a dark shine and a matte-finish. Dentists want it off the air!",
+ "currency":"£",
+ "prefix":true,
"origional_price":14.99,
+ "gallery_price":0.7,
+ "cost_price":0.9,
"stock_count":230,
"notes":"<ul><li>Dentists have got the manufacturers shut down</li><li>Leaves a matte finish and a strange smell</li><li>Limited supply, last time on air</li><li>Larger capacity tube 10% more paste</li><li>Strange texture feel through tube</li><li>Only 230 in stock (tooth joke)</li></ul>",
"crew_notes":"Test crew notes"
"rating":"★★★★★",
"subtext":"Is that a hairline fracture? A crack?",
"description":"Don't look too close! Detailed painting from a long time ago! Decades of a careful hand in observation of the bright and round moon.",
+ "currency":"£",
+ "prefix":true,
"origional_price":4500,
+ "gallery_price":0.6,
+ "cost_price":0.8,
"stock_count":5,
"notes":"<ul><li>Paining of the moon thats difficult to look at</li><li>Look away tell cameraman to look away</li><li>Cover it with your jacket</li><li>Represents the moon hatching into an egg</li><li>Something the grandchildren would love</li></ul>",
"crew_notes":"Test crew notes"
"rating":"★★★★★",
"subtext":"Abundance Declared Cigarettes",
"description":"Smoke breaks too short? borrow the time! Try royal abundance.",
+ "currency":"£",
+ "prefix":true,
"origional_price":35,
+ "gallery_price":1,
+ "cost_price":0.5,
"stock_count":111,
"notes":"<ul><li>Play up the whole not meant to be selling them thing</li><li>Each one takes an entire tobacco plant to make</li><li>each cigarette is 2ft long</li></ul>",
"crew_notes":"Test crew notes"
"rating":"★★★★★",
"subtext":"Coffee. we killed the sheep!",
"description":"Bad dreams? Never sleep again. <em>New</em> formula, 100x strength",
+ "currency":"£",
+ "prefix":true,
"origional_price":12,
+ "gallery_price":0.6,
+ "cost_price":0.65,
"stock_count":100,
"notes":"<ul><li>Stops you sleeping possibly permanently</li><li>Enhanced with special chemical that comes from adrenal glands</li><li>Describe vividly fever dreams</li></ul>",
"crew_notes":"Test crew notes"
"rating":"★★★★★",
"subtext":"The replacement mirror",
"description":"Ready to move on? Look over your shoulder to the person replacing you when you go.",
+ "currency":"£",
+ "prefix":true,
"origional_price":2000,
+ "gallery_price":0.35,
+ "cost_price":0.4,
"stock_count":40,
"notes":"<ul><li>Something about the generation your handing the baton to</li><li>See nothing in the mirror but play up that you do because you're scared of nothing coming after you</li></ul>",
"crew_notes":"Test crew notes"
"code":"777777",
"rating":"★★★★★",
"subtext":"The number previously known as 7",
- "description":"Ready to move on? Look over your shoulder to the person replacing you when you go.",
+ "description":"The Magical Number Seven, Plus or Minus Two. This is the 7 things.",
+ "currency":"£",
+ "prefix":true,
"origional_price":823543,
+ "gallery_price":0.7,
+ "cost_price":0.77,
"stock_count":7,
"notes":"<ul><li>Do the 7 ate 9 joke</li><li>Discount by 7s (70%, 7%)</li><li>List plenty of things that begin with 7</li></ul>",
"crew_notes":"Test crew notes"
"code":"424F58",
"rating":"★★★★★",
"subtext":"Surprise jack-in-the-box, shock your friends!",
- "description":"Surprise-Box™ Limited edition, only 100 EVER made!. Beautiful hand-painted sides.",
+ "description":"Surprise-Box™ Limited edition, only 100 EVER made! Beautiful hand-painted sides.",
+ "currency":"£",
+ "prefix":true,
"origional_price":45,
+ "gallery_price":1,
+ "cost_price":0.8,
"stock_count":99,
"notes":"<ul><li>JACK IN THE BOX IS FULL OF BLOOD</li><li>Dont tell the audience its full of blood</li><li>Dont turn the crank unless instructed</li></ul>",
"crew_notes":"Test crew notes"
"rating":"★★★★★",
"subtext":"Black and White Blood Chocolate Sauce",
"description":"The perfect topping for Noir detectives in ice-cream bars! Sweet and thick chocolate, made from all natural ingredients.",
+ "currency":"£",
+ "prefix":true,
"origional_price":8,
+ "gallery_price":0.5,
+ "cost_price":0.87,
"stock_count":300,
"notes":"<ul><li>Novelty chocolate sauce that looks like blood in black and white</li><li>Describe how refreshing a nice bowl of ice cream would be</li><li>Treat yourself</li></ul>",
"crew_notes":"Test crew notes"
#!.venv/bin/python
+from flask import Flask, Response, request, render_template, jsonify, send_from_directory
from werkzeug.security import generate_password_hash, check_password_hash
-from flask import Flask, Response, request, render_template, jsonify
+from jinja2 import Environment, FileSystemLoader
from flask_httpauth import HTTPBasicAuth
from datetime import datetime, timezone
+from os import path, environ, system
from math import radians, cos, sin
from ast import literal_eval
-from os import path, environ
import sqlite3
import json
app = Flask(__name__)
auth = HTTPBasicAuth()
-app.root_path = environ["SHOPPING_PATH"]
+try:
+ app.root_path = environ["SHOPPING_PATH"]
+except KeyError:
+ pass
# get the secrets ready
with open(path.join(app.root_path, "..", "secrets.json"), "r") as f:
with open(path.join(app.root_path, "static", "static.json"), "r") as f:
static_data = json.loads(f.read())
+with open(path.join(app.root_path, "static", "info.json"), "r") as f:
+ static_info = json.loads(f.read())
+
def gfx_main():
return Response(render_template("gfx.html"), mimetype="text/html")
-@app.route("/<string:page>")
+@app.route("/autocue")
def gfx_page(page):
- if page in ["autocue"]:
- return Response(render_template(f"{page}.html"), mimetype="text/html")
- else:
- return "", 404
+ return Response(render_template("autocue.html"), mimetype="text/html")
+
+@app.route("/docs")
+@auth.login_required
+def web_docs():
+ return Response(render_template("docs.html", data=static_data, info=static_info), mimetype="text/html")
+
+@app.route("/docs/<string:filename>")
+@auth.login_required
+def paper_docs(filename):
+ return send_from_directory("docs", filename, as_attachment=True)
@app.route("/admin")
@auth.login_required
+def generate_docs():
+ latex_enviroment = Environment(
+ loader=FileSystemLoader(path.join(app.root_path, "templates")),
+ block_start_string = "|%",
+ block_end_string = "%|",
+ variable_start_string = "|~",
+ variable_end_string = "~|",
+ comment_start_string = "|#",
+ comment_end_string = "#|",
+ trim_blocks = True,
+ lstrip_blocks = True
+ )
+
+ for document in ["call-sheet.tex", "manifest-unsafe.tex", "manifest-safe.tex", "gfx-text.tex"]:
+ template = latex_enviroment.get_template(document)
+ docs_path = path.join(app.root_path, "docs")
+
+ with open(path.join(docs_path, document), "w") as f:
+ f.write(template.render(data=static_data, info=static_info))
+
+ system(f"pdflatex -interaction='nonstopmode' -output-directory='{docs_path}' '{path.join(docs_path, document)}' > /dev/zero")
+ system(f"pdflatex -interaction='nonstopmode' -output-directory='{docs_path}' '{path.join(docs_path, document)}' > /dev/zero")
+
+ print(f" - Generated {document}")
+
if __name__ == "__main__":
# sanity check on the db
with open(path.join(app.root_path, "schema"), "r") as f:
cursor.execute(load)
connection.commit()
- app.run(host='127.0.0.1', port=8000, debug=True)
+ print("Generating static documentation...")
+ generate_docs()
+
+ app.run(host='192.168.0.144', port=8000, debug=True)
--- /dev/null
+\documentclass{article}
+
+\usepackage{tabularx}
+\usepackage[
+ top=1.5cm,
+ bottom=1.5cm,
+ left=1cm,
+ right=1cm
+ ]{geometry}
+
+\renewcommand{\arraystretch}{1.2}
+
+\begin{document}
+
+\thispagestyle{empty}
+
+\noindent{\Large{}\textbf{XMDV Teleshopping\hspace*{\fill}|~ info.shoot.date ~|}}\\
+Location: |~ info.shoot.location ~|
+
+\vspace{4mm}
+
+\noindent{\large{}\textbf{Production Contacts}}
+
+\vspace{2mm}
+
+\noindent\begin{tabularx}{\textwidth}{r l l X}
+ \hline
+\textbf{Name} & \textbf{Role} & \textbf{Contact} &\\
+ \hline
+ \hline
+\textbf{Crew} &&&\\
+ |% for person in info.crew %|
+ \hline
+ |~ person.name ~| & |~ person.role ~| & |~ person.phone ~| & |~ person.email ~| \\
+ |% endfor %|
+ \hline
+ \hline
+\textbf{Cast} &&&\\
+ |% for person in info.cast %|
+ \hline
+ |~ person.name ~| & |~ person.role ~| & |~ person.phone ~| & |~ person.email ~| \\
+ |% endfor %|
+ \hline
+\end{tabularx}
+
+\vspace{2mm}
+
+\noindent\textit{First Aider: William Greenwood 075 9476 8180}
+
+\vspace{4mm}
+
+\noindent{\large{}\textbf{Schedule}}
+
+\vspace{2mm}
+
+\noindent\begin{tabularx}{\textwidth}{r X}
+ \hline
+\textbf{Time} & \\
+ \hline
+ |% for action in info.schedule %|
+ \hline
+ \textbf{|~ action.time ~|} & |~ action.title ~|\\
+ |% if action.extra|length > 0 %|
+ |% for line in action.extra %|
+ & \textit{|~ line ~|}\\
+ |% endfor %|
+ |% endif %|
+ |% endfor %|
+ \hline
+\end{tabularx}
+
+\end{document}
<!DOCTYPE html>
-<html lang="">
- <head>
- <meta charset="utf-8">
- <title></title>
- </head>
- <body>
- <header></header>
- <main></main>
- <footer></footer>
- </body>
+<html lang="en">
+ <head>
+ <meta charset="utf-8">
+ <title>XMDV</title>
+ <style>
+
+table {
+ border: 1px solid black;
+ border-collapse: collapse;
+ width: 100%;
+}
+th, td {
+ border-top: 1px solid black;
+ text-align: left;
+ padding: 2px 5px 2px 5px;
+ vertical-align: top;
+ }
+th:first-child, td:first-child {
+ text-align: right;
+}
+details {
+ color: red;
+}
+ </style>
+ </head>
+ <body>
+ <h1>XMDV Teleshopping documentation</h1>
+ <h2 style="color: red">Contains forbidden information</h2>
+ <h2>{{info.shoot.date}}</h2>
+ <p><b>Location:</b> {{info.shoot.location}}</p>
+
+ <h2>Call-sheet</h2>
+ <table>
+ <tr>
+ <th>Name</th>
+ <th>Role</th>
+ <th>Contact</th>
+ </tr>
+ <tr>
+ <td colspan="3" style="text-align: center; font-weight: bold">Crew</th>
+ </tr>
+ {% for person in info.crew %}
+ {% if not person.absent %}
+ <tr>
+ <td>{{person.name}}</td>
+ <td>{{person.role}}</td>
+ <td>{{person.phone}}<br>{{person.email}}</td>
+ </tr>
+ {% endif %}
+ {% endfor %}
+ <tr>
+ <td colspan="3" style="text-align: center; font-weight: bold">Cast</th>
+ </tr>
+ {% for person in info.cast %}
+ <tr>
+ <td>{{person.name}}</td>
+ <td>{{person.role}}</td>
+ <td>{{person.phone}}<br>{{person.email}}</td>
+ </tr>
+ {% endfor %}
+ </table>
+ <p><em>First Aider: William Greenwood 075 9476 8180</em></p>
+
+
+ <h2>Schedule</h2>
+ <table>
+ <colspan>
+ <col width="50em">
+ </colspan>
+ <tr>
+ <th>Time</th>
+ <th>Action</th>
+ </tr>
+ {% for action in info.schedule %}
+ <tr>
+ <td>{{action.time}}</td>
+ <td>{{action.title}}</td>
+ </tr>
+ {% if action.extra|length > 0 %}
+ {% for line in action.extra %}
+ <tr>
+ <td></td>
+ <td><em>{{line}}</em></td>
+ </tr>
+ {% endfor %}
+ {% endif %}
+ {% endfor %}
+ </table>
+
+ <h2>Shoot notes</h2>
+ <ul>
+ {% for line in info.notes %}
+ <li>{{line}}</li>
+ {% endfor %}
+ </ul>
+
+ {% macro price(origional, currency, prefix, p=0) %}
+ {% set new_price = (origional * (1-p))|round(2) %}
+ {% if prefix %}
+ {{currency ~ new_price}}
+ {% else %}
+ {{new_price ~ currency}}
+ {% endif %}
+ {% if p != 0 %}
+ (~{{(p * 100)|round|int}}% off)
+ {% endif %}
+ {% endmacro %}
+
+ <h2>Item manifest</h2>
+ <table>
+ {% for item in data['items'] %}
+ {% if loop.index0 > 0 %}
+ <tr><td colspan="2"></td><tr>
+ <tr><td colspan="2"></td><tr>
+ {% endif %}
+ <tr>
+ <th>Item code</th>
+ <td>{{item.code}}</td>
+ </tr>
+ <tr>
+ <th>Name</th>
+ <td>{{item.subtext|safe}}</td>
+ </tr>
+ <tr>
+ <td colspan="2" style="text-align: left">
+ <b>Initial</b> {{price(item.origional_price, item.currency, item.prefix)}}
+ <b style="padding-left: 20px">Gallery price</b> {{price(item.origional_price, item.currency, item.prefix, p=item.gallery_price)}}
+ <b style="padding-left: 20px">Purchase price</b> {{price(item.origional_price, item.currency, item.prefix, p=item.cost_price)}}
+ </td>
+ </tr>
+ <tr>
+ <th>Description</th>
+ <td>{{item.description|safe}}</td>
+ </tr>
+ <tr>
+ <th>Notes</th>
+ <td>{{item.notes|safe}}</td>
+ </tr>
+ <tr style="color: red">
+ <th>Forbidden</th>
+ <td>
+ <details>
+ <summary>Reveal</summary>
+ {{item.crew_notes}}
+ </details>
+ </td>
+ </tr>
+ {% endfor %}
+ </table>
+
+ <h2>Text options</h2>
+ <table>
+ <tr>
+ <th style='text-align: center;'>Crawler top text</th>
+ </tr>
+ {% for line in data.text.crawler_top %}
+ <tr>
+ <td style='text-align: left;'>{{line}}</th>
+ </tr>
+ {% endfor %}
+ <tr>
+ <th style='text-align: center;'>Crawler bottom text</th>
+ </tr>
+ {% for line in data.text.crawler_bottom %}
+ <tr>
+ <td style='text-align: left;'>{{line}}</th>
+ </tr>
+ {% endfor %}
+ <tr>
+ <th style='text-align: center;'>Banner text</th>
+ </tr>
+ {% for line in data.text.banner %}
+ <tr>
+ <td style='text-align: left;'>{{line}}</th>
+ </tr>
+ {% endfor %}
+ </table>
+ </body>
+
+ <h2>PDF Documents</h2>
+ <ul>
+ <li><a href="/docs/call-sheet.pdf">Call-sheet</a></li>
+ <li><a href="/docs/manifest-safe.pdf">Item manifest (safe)</a></li>
+ <li><a href="/docs/manifest-unsafe.pdf">Item manifest (unsafe)</a></li>
+ <li><a href="/docs/gfx-text.pdf">GFX text sheet</a></li>
+ </ul>
</html>
--- /dev/null
+\documentclass{article}
+
+\usepackage{tabularx}
+\usepackage[
+ top=1.5cm,
+ bottom=1.5cm,
+ left=1cm,
+ right=1cm
+ ]{geometry}
+
+\renewcommand{\arraystretch}{1.2}
+
+\begin{document}
+
+\thispagestyle{empty}
+
+\noindent{\Large{}\textbf{XMDV Teleshopping\hspace*{\fill}|~ info.shoot.date ~|}}
+
+\vspace{4mm}
+
+\subsection*{Crawler top text}
+|% for line in data.text.crawler_top %||~ line ~|\\|% endfor %|
+
+\subsection*{Crawler bottom text}
+|% for line in data.text.crawler_bottom %||~ line ~|\\|% endfor %|
+
+\subsection*{Banner text}
+|% for line in data.text.banner %||~ line ~|\\|% endfor %|
+
+\end{document}
--- /dev/null
+\documentclass{article}
+
+\usepackage{calc}
+\usepackage{newtx}
+\usepackage[
+ top=1.5cm,
+ bottom=1.5cm,
+ inner=1cm,
+ outer=1cm
+ ]{geometry}
+
+\usepackage{tabularx}
+\usepackage{ltablex}
+
+\begin{document}
+
+ \noindent{\Large{}\textbf{XMDV Teleshopping\hspace*{\fill}|~ info.shoot.date ~|}}
+
+ \begin{tabularx}{\textwidth}{| l || X || l | l | l | l | }
+ % HEADER SETUP
+ \hline
+ \textbf{Code} & \textbf{Description} & \textbf{Qty} & \textbf{Start price} & \textbf{Gallery price} & \textbf{Cost price} \\
+ \hline
+ \hline
+ \endhead
+
+ % FOOTER SETUP
+ \hline
+ \textbf{Code} & \textbf{Description} & \textbf{Qty} & \textbf{Start price} & \textbf{Gallery price} & \textbf{Cost price} \\
+ \hline
+ \endfoot
+
+ % TABLE BODY
+ |% macro price(origional, currency, prefix, p=0) %|
+ |% set new_price = (origional * (1-p))|round(2) %|
+ |% if prefix %|
+ |~ currency ~ new_price ~|
+ |% else %|
+ |~ new_price ~ currency ~|
+ |% endif %|
+ |% if p != 0 %|
+ (~ |~ (p * 100)|round|int ~|\% off)
+ |% endif %|
+ |% endmacro %|
+
+ |% macro format(string) %|
+ |~ string|replace("#", "~")|replace("&", "~")|replace("%", "\%") ~|
+ |% endmacro %|
+
+ |% macro html(string) %|
+ |~ string
+ |replace("<ul>", " ")
+ |replace("</ul>", " ")
+ |replace("<li>", " ")
+ |replace("<br>", " ")
+ |replace("</li>", ". ")
+ |replace("<em>", " ")
+ |replace("</em>", " ")
+ ~|
+ |% endmacro %|
+
+ |% for item in data['items'] %|
+ |~ item.code ~| & |~ format(item.subtext) ~| & |~ item.stock_count ~| &
+ |~ price(item.origional_price, item.currency, item.prefix) ~| &
+ |~ price(item.origional_price, item.currency, item.prefix, p=item.gallery_price) ~| &
+ |~ price(item.origional_price, item.currency, item.prefix, p=item.cost_price) ~| \\
+ \hline
+ \multicolumn{6}{|p{\textwidth - 13pt}|}{|~ html(format(item.description)) ~|}\\
+ \hline
+ \multicolumn{6}{|p{\textwidth - 13pt}|}{|~ html(format(item.notes)) ~|}\\
+ \hline
+ \hline
+ |% endfor %|
+ \end{tabularx}
+\end{document}
--- /dev/null
+\documentclass{article}
+
+\usepackage{calc}
+\usepackage{xcolor}
+\usepackage{newtx}
+\usepackage[text=FORBIDDEN, scale=0.5]{draftwatermark}
+\usepackage[
+ top=1.5cm,
+ bottom=1.5cm,
+ inner=1cm,
+ outer=1cm
+ ]{geometry}
+
+\usepackage{tabularx}
+\usepackage{ltablex}
+
+\begin{document}
+
+ \noindent{\Large{}\textbf{XMDV Teleshopping\hspace*{\fill}|~ info.shoot.date ~|}}\\
+ \textcolor{red}{\textbf{Contains forbidden information}}
+
+ \begin{tabularx}{\textwidth}{| l || X || l | l | l | l | }
+ % HEADER SETUP
+ \hline
+ \textbf{Code} & \textbf{Description} & \textbf{Qty} & \textbf{Start price} & \textbf{Gallery price} & \textbf{Cost price} \\
+ \hline
+ \hline
+ \endhead
+
+ % FOOTER SETUP
+ \hline
+ \textbf{Code} & \textbf{Description} & \textbf{Qty} & \textbf{Start price} & \textbf{Gallery price} & \textbf{Cost price} \\
+ \hline
+ \endfoot
+
+ % TABLE BODY
+ |% macro price(origional, currency, prefix, p=0) %|
+ |% set new_price = (origional * (1-p))|round(2) %|
+ |% if prefix %|
+ |~ currency ~ new_price ~|
+ |% else %|
+ |~ new_price ~ currency ~|
+ |% endif %|
+ |% if p != 0 %|
+ (~ |~ (p * 100)|round|int ~|\% off)
+ |% endif %|
+ |% endmacro %|
+
+ |% macro format(string) %|
+ |~ string|replace("#", "~")|replace("&", "~")|replace("%", "\%") ~|
+ |% endmacro %|
+
+ |% macro html(string) %|
+ |~ string
+ |replace("<ul>", " ")
+ |replace("</ul>", " ")
+ |replace("<li>", " ")
+ |replace("<br>", " ")
+ |replace("</li>", ". ")
+ |replace("<em>", " ")
+ |replace("</em>", " ")
+ ~|
+ |% endmacro %|
+
+ |% for item in data['items'] %|
+ |~ item.code ~| & |~ format(item.subtext) ~| & |~ item.stock_count ~| &
+ |~ price(item.origional_price, item.currency, item.prefix) ~| &
+ |~ price(item.origional_price, item.currency, item.prefix, p=item.gallery_price) ~| &
+ |~ price(item.origional_price, item.currency, item.prefix, p=item.cost_price) ~| \\
+ \hline
+ \multicolumn{6}{|p{\textwidth - 13pt}|}{|~ html(format(item.description)) ~|}\\
+ \hline
+ \multicolumn{6}{|p{\textwidth - 13pt}|}{|~ html(format(item.notes)) ~|}\\
+ \hline
+ \multicolumn{6}{|p{\textwidth - 13pt}|}{\textcolor{red}{\textbf{|~ format(item.crew_notes) ~|}}}\\
+ \hline
+ \hline
+ |% endfor %|
+ \end{tabularx}
+\end{document}