diff --git a/eh17/__init__.py b/eh17/__init__.py
new file mode 100644
index 0000000..b2d3600
--- /dev/null
+++ b/eh17/__init__.py
@@ -0,0 +1,341 @@
+#!/usr/bin/python3
+
+import subprocess
+import os.path
+from renderlib import *
+from easing import *
+import svg.path
+from twisted.words.protocols.irc import lowDequote
+import random
+from itertools import permutations
+
+
+# URL to Schedule-XML
+scheduleUrl = 'https://programm.froscon.de/2016/schedule.xml'
+
+# For (really) too long titles
+titlemap = {
+ #
+}
+
+
+class animate(object):
+
+ def __init__(self, low, high, xml):
+ self.low = low * fps
+ self.high = high * fps
+ self.frames = self.high - self.low
+ self.xml = xml
+
+ def get(self, frame):
+
+ if self.low <= frame <= self.high:
+ return self.frame(frame)
+
+ def relframe(self,frame):
+ return frame - self.low
+
+
+class background(animate):
+
+ def __init__(self, low, high, xml):
+ animate.__init__(self, low, high, xml)
+
+ self.pathstr = xml.find(".//*[@id='animatePath']").get('d')
+ self.path = svg.path.parse_path(self.pathstr)
+ self.init = self.path.point(0)
+
+
+ def frame(self,frame):
+ p = self.path.point(frame / self.frames) - self.init
+ return (
+ ('bgtext', 'attr', 'transform', 'translate(%.4f, %.4f)' % (p.real, p.imag)), )
+
+
+class urldate(animate):
+
+ def frame(self,frame):
+ return(
+ ('url', 'style', 'opacity', easeOutQuad(self.relframe(frame), 1, -1, self.frames)),
+ ('date', 'style', 'opacity', easeOutQuad(self.relframe(frame), 0, 1, self.frames)),
+ )
+
+
+class hasenfarbe(animate):
+
+ def __init__(self, low, high, xml):
+ animate.__init__(self, low, high, xml)
+ colors=['#9e00a0','#ffe72d','#ff8600','#0bc401','#d40010','#0049da']
+ self.hasen=[]
+ for p in permutations(colors):
+ self.hasen.append(p)
+ random.shuffle(self.hasen)
+
+
+ def frame(self,frame):
+ if frame % 3 is 0:
+ return(
+ ('hase001', 'style', 'fill', self.hasen[frame][0]),
+ ('hase002', 'style', 'fill', self.hasen[frame][0]),
+ ('hase003', 'style', 'fill', self.hasen[frame][0]),
+ ('hase004', 'style', 'fill', self.hasen[frame][0]),
+ ('hase005', 'style', 'fill', self.hasen[frame][1]),
+ ('hase006', 'style', 'fill', self.hasen[frame][1]),
+ ('hase007', 'style', 'fill', self.hasen[frame][1]),
+ ('hase008', 'style', 'fill', self.hasen[frame][2]),
+ ('hase009', 'style', 'fill', self.hasen[frame][2]),
+ ('hase010', 'style', 'fill', self.hasen[frame][2]),
+ ('hase011', 'style', 'fill', self.hasen[frame][3]),
+ ('hase012', 'style', 'fill', self.hasen[frame][3]),
+ ('hase013', 'style', 'fill', self.hasen[frame][3]),
+ ('hase014', 'style', 'fill', self.hasen[frame][4]),
+ ('hase015', 'style', 'fill', self.hasen[frame][4]),
+ ('hase016', 'style', 'fill', self.hasen[frame][4]),
+ ('hase017', 'style', 'fill', self.hasen[frame][4]),
+ ('hase018', 'style', 'fill', self.hasen[frame][5]),
+ ('hase019', 'style', 'fill', self.hasen[frame][5]),
+ ('hase020', 'style', 'fill', self.hasen[frame][5]),
+ )
+
+
+def introFrames(args):
+ xml = etree.parse('eh17/artwork/intro.svg').getroot()
+
+ animations = [
+ background(0,6,xml),
+ urldate(0.5,1,xml),
+ hasenfarbe(1,5,xml)]
+
+
+ frames = int(6 * fps)
+ for frame in range(0,frames):
+
+ frameactions = ()
+ for a in animations:
+ action = a.get(frame)
+ if action:
+ frameactions += action
+
+ print (frameactions)
+ yield frameactions
+
+
+
+def unused():
+
+
+ bg_i = 0
+
+ frames = int(0.5 * fps)
+ for i in range(0, frames):
+ p = path.point(i / frames) - init
+ yield (
+ ('bgtext', 'attr', 'transform', 'translate(%.4f, %.4f)' % (p.real, p.imag)),
+ ('animatePath', 'style', 'opacity', 0),
+ ('date', 'style', 'opacity', 0),
+ )
+ bg_i += 1
+
+ frames = 3 * fps
+ for i in range(0, frames):
+ p = path.point(i / frames) - init
+ yield (
+ ('bgtext', 'attr', 'transform', 'translate(%.4f, %.4f)' % (p.real, p.imag)),
+ )
+
+ frames = int(0.5 * fps)
+ for i in range(0, frames):
+ yield tuple()
+
+ frames = 1 * fps
+ for i in range(0, frames):
+ yield (
+ ('url', 'style', 'opacity', easeOutQuad(i, 1, -1, frames)),
+ ('date', 'style', 'opacity', easeOutQuad(i, 0, 1, frames)),
+ )
+
+ frames = int(1.5 * fps)
+ for i in range(0, frames):
+ yield (
+ ('url', 'style', 'opacity', 0),
+ ('date', 'style', 'opacity', 1),
+ ('bar', 'style', 'opacity', easeLinear(i, 1, -1, frames)),
+ ('title', 'style', 'opacity', easeLinear(i, 1, -1, frames)),
+ )
+
+
+ # frames = 1*fps
+ # for i in range(0, frames):
+ # yield (
+ # )
+
+ frames = int(0.5 * fps) + 1
+ for i in range(0, frames):
+ yield (
+ ('bar', 'style', 'opacity', 0),
+ ('title', 'style', 'opacity', 0),
+ )
+
+def outroFrames(args):
+ xml = etree.parse('eh17/artwork/outro.svg').getroot()
+ pathstr = xml.find(".//*[@id='animatePath']").get('d')
+ frog = xml.find(".//*[@id='animatePath']").get('d')
+ path = svg.path.parse_path(pathstr)
+
+ init = path.point(0)
+
+ frames = int(0.5 * fps)
+ for i in range(0, frames):
+ p = path.point(i / frames) - init
+ yield (
+ ('animatePath', 'style', 'opacity', 0),
+ ('license', 'style', 'opacity', 0),
+ )
+
+ frames = 3 * fps
+ for i in range(0, frames):
+ p = path.point(i / frames) - init
+ yield (
+ ('frog', 'attr', 'transform', 'translate(%.4f, %.4f)' % (p.real, p.imag)),
+ )
+
+ frames = int(0.5 * fps) + 1
+ for i in range(0, frames):
+ yield tuple()
+
+ frames = 1 * fps
+ for i in range(0, frames):
+ yield (
+ ('logo', 'style', 'opacity', easeLinear(i, 1, -1, frames)),
+ )
+
+ frames = 1 * fps
+ for i in range(0, frames):
+ yield (
+ ('logo', 'style', 'opacity', 0),
+ ('license', 'style', 'opacity', easeLinear(i, 0, 1, frames)),
+ )
+
+ frames = 2 * fps
+ for i in range(0, frames):
+ yield (
+ ('logo', 'style', 'opacity', 0),
+ ('license', 'style', 'opacity', 1),
+ )
+
+ frames = 1 * fps
+ for i in range(0, frames):
+ yield (
+ ('logo', 'style', 'opacity', 0),
+ ('license', 'style', 'opacity', easeLinear(i, 1, -1, frames)),
+ )
+
+ frames = 1 * fps
+ for i in range(0, frames):
+ yield (
+ ('logo', 'style', 'opacity', 0),
+ ('license', 'style', 'opacity', 0),
+ )
+
+def pauseFrames(args):
+ frames = 2 * fps
+ for i in range(0, frames):
+ yield (
+ ('text1', 'style', 'opacity', 1),
+ ('text2', 'style', 'opacity', 0),
+ )
+
+ frames = 1 * fps
+ for i in range(0, frames):
+ yield (
+ ('text1', 'style', 'opacity', easeLinear(i, 1, -1, frames)),
+ ('text2', 'style', 'opacity', 0),
+ )
+
+ frames = 1 * fps
+ for i in range(0, frames):
+ yield (
+ ('text1', 'style', 'opacity', 0),
+ ('text2', 'style', 'opacity', easeLinear(i, 0, 1, frames)),
+ )
+
+ frames = 2 * fps
+ for i in range(0, frames):
+ yield (
+ ('text1', 'style', 'opacity', 0),
+ ('text2', 'style', 'opacity', 1),
+ )
+
+ frames = 1 * fps
+ for i in range(0, frames):
+ yield (
+ ('text1', 'style', 'opacity', 0),
+ ('text2', 'style', 'opacity', easeLinear(i, 1, -1, frames)),
+ )
+
+ frames = 1 * fps
+ for i in range(0, frames):
+ yield (
+ ('text1', 'style', 'opacity', easeLinear(i, 0, 1, frames)),
+ ('text2', 'style', 'opacity', 0),
+ )
+
+def debug():
+ render('intro.svg',
+ '../intro.ts',
+ introFrames,
+ {
+ '$id': 1302,
+ '$title': 'VlizedLab - Eine Open Source-Virtualisierungslösung für PC-Räume',
+ '$subtitle': 'IT Automatisierung und zentrales Management mit SALT',
+ '$personnames': 'Thorsten Kramm',
+ '$url':'blubb',
+ '$date':'huhu'
+
+ }
+ )
+
+# # render('outro.svg',
+# '../outro.ts',
+# outroFrames
+# )
+#
+# render('pause.svg',
+# '../pause.ts',
+# pauseFrames
+# )
+
+
+def tasks(queue, args):
+ # iterate over all events extracted from the schedule xml-export
+ for event in events(scheduleUrl):
+ if event['room'] not in ('Saal 1', 'Saal 3', 'Saal 4', 'Saal 5', 'Saal 6', 'Saal 7', 'Saal 8'):
+ print("skipping room %s (%s)" % (event['room'], event['title']))
+ continue
+
+ # generate a task description and put them into the queue
+ queue.put(Rendertask(
+ infile='intro.svg',
+ outfile=str(event['id']) + ".ts",
+ sequence=introFrames,
+ parameters={
+ '$id': event['id'],
+ '$title': event['title'],
+ '$subtitle': event['subtitle'],
+ '$personnames': event['personnames']
+ }
+ ))
+
+ # place a task for the outro into the queue
+ queue.put(Rendertask(
+ infile='outro.svg',
+ outfile='outro.ts',
+ sequence=outroFrames
+ ))
+
+ # place the pause-sequence into the queue
+ queue.put(Rendertask(
+ infile='pause.svg',
+ outfile='pause.ts',
+ sequence=pauseFrames
+ ))
diff --git a/eh17/artwork/background.png b/eh17/artwork/background.png
new file mode 100644
index 0000000..3cd0879
Binary files /dev/null and b/eh17/artwork/background.png differ
diff --git a/eh17/artwork/background.xcf b/eh17/artwork/background.xcf
new file mode 100644
index 0000000..d9a487c
Binary files /dev/null and b/eh17/artwork/background.xcf differ
diff --git a/eh17/artwork/intro.svg b/eh17/artwork/intro.svg
new file mode 100644
index 0000000..b7eaac7
--- /dev/null
+++ b/eh17/artwork/intro.svg
@@ -0,0 +1,914 @@
+
+
+
+
diff --git a/eh17/artwork/logo-bw.svg b/eh17/artwork/logo-bw.svg
new file mode 100644
index 0000000..f30babb
--- /dev/null
+++ b/eh17/artwork/logo-bw.svg
@@ -0,0 +1,208 @@
+
+
+
+
diff --git a/eh17/artwork/logo.svg b/eh17/artwork/logo.svg
new file mode 100644
index 0000000..a57e99c
--- /dev/null
+++ b/eh17/artwork/logo.svg
@@ -0,0 +1,222 @@
+
+
+
+
diff --git a/eh17/artwork/outro.svg b/eh17/artwork/outro.svg
new file mode 100644
index 0000000..ef2f5eb
--- /dev/null
+++ b/eh17/artwork/outro.svg
@@ -0,0 +1,496 @@
+
+
+
+
diff --git a/eh17/artwork/overlay.svg b/eh17/artwork/overlay.svg
new file mode 100644
index 0000000..76f5ffe
--- /dev/null
+++ b/eh17/artwork/overlay.svg
@@ -0,0 +1,179 @@
+
+
+
+
diff --git a/eh17/artwork/pause.svg b/eh17/artwork/pause.svg
new file mode 100644
index 0000000..7fc2b1a
--- /dev/null
+++ b/eh17/artwork/pause.svg
@@ -0,0 +1,479 @@
+
+
+
+