├── .gitignore
├── .travis.yml
├── LICENSE
├── README.md
├── mikrowelle.py
├── requirements.txt
├── settings.default.json
├── templates
└── base-template
│ ├── archive.tpl
│ ├── head.tpl
│ ├── index.tpl
│ ├── meta_block.tpl
│ ├── player.tpl
│ ├── post.tpl
│ ├── res
│ └── pwp
│ │ ├── css
│ │ ├── pwp-blue-orange.css
│ │ ├── pwp-blue-orange.min.css
│ │ ├── pwp-creamy.css
│ │ ├── pwp-creamy.min.css
│ │ ├── pwp-dark-blaze.css
│ │ ├── pwp-dark-blaze.min.css
│ │ ├── pwp-dark-gray.css
│ │ ├── pwp-dark-gray.min.css
│ │ ├── pwp-dark-green.css
│ │ ├── pwp-dark-green.min.css
│ │ ├── pwp-dark.css
│ │ ├── pwp-dark.min.css
│ │ ├── pwp-jungle-green.css
│ │ ├── pwp-jungle-green.min.css
│ │ ├── pwp-light-gray.css
│ │ ├── pwp-light-gray.min.css
│ │ ├── pwp-light.css
│ │ ├── pwp-light.min.css
│ │ ├── pwp-midnightblue.css
│ │ ├── pwp-midnightblue.min.css
│ │ ├── pwp-silver-blaze.css
│ │ ├── pwp-silver-blaze.min.css
│ │ ├── pwp-silver.css
│ │ ├── pwp-silver.min.css
│ │ └── vendor
│ │ │ ├── progress-polyfill.css
│ │ │ └── style.css
│ │ ├── examples
│ │ ├── coverimage-blue.png
│ │ ├── coverimage-green.png
│ │ ├── coverimage-red.png
│ │ ├── coverimage.png
│ │ ├── cre-206
│ │ │ └── index.html
│ │ ├── deeplinking
│ │ │ └── index.html
│ │ ├── examples.css
│ │ ├── folder-30c3.png
│ │ ├── index.html
│ │ ├── injected
│ │ │ └── index.html
│ │ ├── minimal
│ │ │ └── index.html
│ │ ├── picture-stream
│ │ │ ├── Kapitel-Demo.m4a
│ │ │ └── index.html
│ │ ├── themes
│ │ │ ├── index.html
│ │ │ └── themed.html
│ │ ├── video
│ │ │ └── index.html
│ │ └── which-format
│ │ │ ├── index.html
│ │ │ ├── podlove-test-track.mp3
│ │ │ ├── podlove-test-track.mp4
│ │ │ ├── podlove-test-track.ogg
│ │ │ └── podlove-test-track.opus
│ │ ├── font
│ │ ├── podlove.eot
│ │ ├── podlove.svg
│ │ ├── podlove.ttf
│ │ └── podlove.woff
│ │ ├── img
│ │ ├── arrow-down-black.png
│ │ ├── arrow-down-white.png
│ │ ├── download-overlay.png
│ │ └── icon-podlove-subscribe-600.png
│ │ ├── index.html
│ │ ├── js
│ │ ├── podlove-web-moderator.js
│ │ ├── podlove-web-moderator.min.js
│ │ ├── podlove-web-player.js
│ │ ├── podlove-web-player.min.js
│ │ └── vendor
│ │ │ ├── html5shiv-printshiv.js
│ │ │ ├── html5shiv-printshiv.min.js
│ │ │ ├── html5shiv.js
│ │ │ ├── html5shiv.min.js
│ │ │ ├── jquery.js
│ │ │ ├── jquery.min.js
│ │ │ ├── jquery.min.map
│ │ │ ├── progress-polyfill.js
│ │ │ ├── progress-polyfill.min.js
│ │ │ ├── tests.js
│ │ │ └── user-interaction.js
│ │ └── static.html
│ └── single.tpl
├── test
├── audio
│ ├── 001.json
│ ├── 001.m4a
│ ├── 001.mp3
│ ├── 001.opus
│ ├── 002.json
│ ├── 002.m4a
│ ├── 002.mp3
│ └── 002.opus
└── test_workflow.py
└── util
├── __init__.py
├── install.sh
├── mime_types.py
├── rssgen.py
└── search.py
/.gitignore:
--------------------------------------------------------------------------------
1 | *.py[cod]
2 |
3 | # C extensions
4 | *.so
5 |
6 | # Packages
7 | *.egg
8 | *.egg-info
9 | dist
10 | build
11 | eggs
12 | parts
13 | bin
14 | var
15 | sdist
16 | develop-eggs
17 | .installed.cfg
18 | lib
19 | lib64
20 |
21 | # Installer logs
22 | pip-log.txt
23 |
24 | # Unit test / coverage reports
25 | .coverage
26 | .tox
27 | nosetests.xml
28 |
29 | # Translations
30 | *.mo
31 |
32 | # Mr Developer
33 | .mr.developer.cfg
34 | .project
35 | .pydevproject
36 |
37 | # Additional
38 | .DS_Store
39 |
40 | pub/
41 | audio/
42 | test/tmp/
43 | posts/
44 |
--------------------------------------------------------------------------------
/.travis.yml:
--------------------------------------------------------------------------------
1 | language: python
2 | python:
3 | - "2.7"
4 | - "3.3"
5 | - "3.4"
6 | - "pypy"
7 | install: "pip install -r requirements.txt"
8 | script: py.test
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | BSD 2-Clause License
2 |
3 | Copyright (c) 2013-2016, Thomas Skowron and others
4 | All rights reserved.
5 |
6 | Redistribution and use in source and binary forms, with or without modification,
7 | are permitted provided that the following conditions are met:
8 |
9 | Redistributions of source code must retain the above copyright notice, this list
10 | of conditions and the following disclaimer.
11 | Redistributions in binary form must reproduce the above copyright notice, this
12 | list of conditions and the following disclaimer in the documentation and/or
13 | other materials provided with the distribution.
14 | THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
15 | ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
16 | WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
17 | DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
18 | ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
19 | (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
20 | LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
21 | ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 | (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
23 | SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # Mikrowelle OS [](http://travis-ci.org/thomersch/Mikrowelle-OS)
2 |
3 | Mikrowelle OS is a static page generator for podcast websites and feeds.
4 |
5 | Tested on Python 2.7, 3.3, 3.4 and PyPy 2.5. Will not work under Python 2.6, please update in this case. If possible, prefer Python 3.
6 |
7 | ## Feature List
8 |
9 | * Generating podcast feeds
10 | * iTunes compatible
11 | * Payment tag (Flattr)
12 | * with TTL-support (reduces traffic by setting a sane default interval for feed-refreshing in podcast clients)
13 | * Full text search in the browser
14 | * Flattr support
15 | * New item for every episode, no manual generation needed.
16 | * Modern player
17 | * Podlove Web Player included
18 | * Chapter mark support
19 | * Auphonic support
20 | * Podlove Simple Chapters
21 | * Suitable for heavy load situations
22 | * no CGI/PHP/runtime bloat
23 |
24 | ## What does "static page generator" mean?
25 |
26 | You won't need a special webserver to serve the application. You can simply throw in all podcast data into this application and it will generate all necessary files. Those files can be uploaded to any hosting without any need for CGI, PHP or so.
27 |
28 | ## Is it working?
29 |
30 | Yep, we are using it actively in our podcast: [Mikrowelle](http://mikrowelle.me/). Though, there may be some issues. If you need assistance, feel free to contact me ([Twitter](http://twitter.com/thomersch)) or create an issue.
31 |
32 | ## Dependencies
33 |
34 | * Python
35 | * lxml (for feed generation)
36 | * jinja2 (templating engine)
37 | * markdown (easy markdown language)
38 | * JavaScript _(those are already included in the repository)_
39 | * jQuery
40 | * Podlove Web Player
41 |
42 | ## Installation Instructions
43 |
44 | * Clone git repository.
45 | * `git clone https://github.com/thomersch/Mikrowelle-OS.git`
46 | * Install Python dependencies.
47 | * on Debian based systems: `sudo apt-get install python3-lxml`
48 | * `pip install -r requirements.txt`
49 | * Customize your podcast settings.
50 | * Rename `settings.default.json` to `settings.json` and change the values the way you need it.
51 | * For the best expierence, use Auphonic. For instructions see section *"Working with Auphonic"*.
52 | * Run `mikrowelle.py`.
53 | * Publish the `audio` folder and the `pub` folder on the interwebz.
54 |
55 |
56 | ## Working with [Auphonic](http://auphonic.com/)
57 |
58 | Auphonic is a fully automatic audio post-production service, that converts all your podcast data into well sounding audio files.
59 |
60 | This application is build for best use with Auphonic: All meta tags should be put into the webinterface of Auphonic. Description is automatically converted with markdown as well as the summary. Put into the "track" file the number of the podcast episode.
61 |
62 | As output files you should select all of the desired audio data types. Plus you must add the "Production description" as json file.
63 |
64 | You may as well add chapter marks, they will be automatically processed and put into the web player.
65 |
66 | The output of auphonic has to be put where the settings.json `filefolder` is configured. The generator will automatically pick up the right files and do the rest.
67 |
68 | ## Working without Auphonic
69 |
70 | You can run _Mikrowelle OS_ without Auphonic as well. In order to do that, you will need to create a json file per episode and put it into your `posts` folder.
71 |
72 | The file name doesn't really matter, but your episode order will depend on it.
73 |
74 | ### Example: Post file
75 |
76 | {
77 | "chapters": [
78 | {
79 | "image": null,
80 | "start": "00:22:46.190",
81 | "title": "Chapter 1",
82 | "url": "http://example.com/1/"
83 | },
84 | {
85 | "image": null,
86 | "start": "01:21:33.010",
87 | "title": "Chapter 2",
88 | "url": "http://example.com/2/"
89 | },
90 | {
91 | "image": null,
92 | "start": "01:17:00.130",
93 | "title": "Chapter 3",
94 | "url": "http://example.com/3/"
95 | }
96 | ],
97 | "content": "
Summary
",
98 | "date": "2013-06-09T22:55:48.500Z",
99 | "duration": "01:35:29.914",
100 | "episode": "001",
101 | "filename": "001",
102 | "humandate": "09.06.2013",
103 | "subtitle": "It is just the beginning.
",
104 | "title": "001 Title"
105 | }
106 |
107 | ## Templating
108 | ### Changing the looks of your podcast page.
109 |
110 | If you want to start a new style, I recommend to duplicate the `templates/base-template` folder and start editing in the duplicate. Remember to change the `tplfolder` path in `settings.json`.
111 |
112 | Adding a stylesheet is nice as well, give it a try. The directory `res` in the template path will be linked in the pub-path, so you can place all your assets in `$templatename/res` and those will be available at `res` on your main publishing path, e.g. `podcast.example.com/res`.
113 |
114 | ## Feature requests
115 |
116 | Give 'em to me. Create a GitHub issue or contact me.
117 |
118 | ## Other Questions?
119 |
120 | Ask me.
121 |
122 | ## License
123 |
124 | BSD License. No bullshit. See `LICENSE` file.
125 |
126 | ## Contributors
127 |
128 | * Thomas Skowron
129 | * Thomas Sänger (HorayNarea)
130 | * Andreas Hubel (saerdnaer)
131 |
--------------------------------------------------------------------------------
/mikrowelle.py:
--------------------------------------------------------------------------------
1 | # -*- coding: utf-8 -*-
2 |
3 | __version__ = (2, 0, 0)
4 | __author__ = "Thomas Skowron"
5 |
6 | import util.rssgen as rssgen
7 | import util.search as search
8 | from util.mime_types import mimetypes
9 |
10 | import sys
11 | import os
12 | import json
13 | import shutil
14 | import math
15 | from datetime import datetime
16 | import distutils.dir_util as du
17 |
18 | import markdown
19 | from jinja2 import FileSystemLoader, Environment
20 |
21 | if sys.version_info < (3, 0, 0):
22 | from codecs import open
23 |
24 | POST_PATH = "posts"
25 | TMP_PATH = "tmp_output"
26 |
27 | def get_settings():
28 | with open("settings.json", "r", encoding="utf-8") as f:
29 | settings = json.loads(f.read())
30 | return settings
31 |
32 |
33 | def get_posts():
34 | posts = []
35 | if not os.path.exists(POST_PATH):
36 | print("[INFO] No posts found.")
37 | os.mkdir(POST_PATH)
38 |
39 | for filename in sorted(os.listdir(POST_PATH), reverse=True):
40 | with open(os.path.join(POST_PATH, filename), "r", encoding="utf-8") as f:
41 | # read data from json
42 | try:
43 | p = json.loads(f.read())
44 | posts.append(p)
45 | except ValueError:
46 | print("[INFO] Ignored non-json file in posts-folder:", filename)
47 | return posts
48 |
49 |
50 | def write_post(post, settings, single_template, player_template):
51 | format_mimes = {}
52 | for fmt in settings["feeds"]:
53 | format_mimes[fmt] = mimetypes[fmt]
54 |
55 | # write individual page for post
56 | with open(os.path.join(TMP_PATH, "%s.html" % post["episode"]),
57 | "a+", encoding="utf-8") as w:
58 | w.write(single_template.render(post=post, settings=settings, formats=format_mimes, index_page=False))
59 |
60 | # write player page
61 | with open(os.path.join(TMP_PATH, "player_%s.html" % post["episode"]), "a+", encoding="utf-8") as w:
62 | w.write(player_template.render(post=post, settings=settings, formats=format_mimes))
63 |
64 |
65 | def _write_json_data(filefolder, fn):
66 | with open("{}{}".format(filefolder, fn), encoding="utf-8") as f:
67 | h = json.loads(f.read())
68 |
69 | # get all the metadata!
70 | o = {
71 | "episode": h["metadata"]["track"],
72 | "title": h["metadata"]["title"],
73 | "content": markdown.markdown(h["metadata"]["summary"]),
74 | "subtitle": markdown.markdown(h["metadata"]["subtitle"]),
75 | "date": h["change_time"],
76 | #TODO: make date format string configurable
77 | "humandate": datetime.strptime(h["change_time"], "%Y-%m-%dT%H:%M:%S.%fZ").strftime("%d.%m.%Y"),
78 | "humanduration": datetime.strptime(h["length_timestring"], "%H:%M:%S.%f").strftime("%H:%M"),
79 | "filename": h["output_basename"],
80 | "duration": h["length_timestring"],
81 | "chapters": h["chapters"]
82 | }
83 | j = json.dumps(o, sort_keys=True, indent=4, separators=(',', ': '))
84 |
85 | # write converted file to post storage folder
86 | if not os.path.exists(POST_PATH):
87 | os.mkdir(POST_PATH)
88 | episode_file_name = os.path.join(POST_PATH, "%s" % format(fn))
89 | with open(episode_file_name, "a+", encoding="utf-8") as e:
90 | e.write(j)
91 |
92 |
93 | def _get_json_file_list(filefolder):
94 | def filterFile(fn):
95 | if fn.endswith(".json") and not os.path.exists(os.path.join(POST_PATH, fn)):
96 | return True
97 |
98 | return [fn for fn in os.listdir(filefolder) if filterFile(fn)]
99 |
100 |
101 | def _get_elements(posts, settings, format):
102 | filefolder = settings["filefolder"]
103 | baseurl = settings["baseurl"]
104 |
105 | elements = []
106 | for post in posts:
107 | filesize = os.path.getsize("{}{}.{}".format(filefolder, post["filename"], format))
108 | mime = mimetypes[format]
109 | episode_link = "{}{}.html".format(baseurl, post["episode"])
110 | guid = "{}{}.html.{}".format(baseurl, post["episode"], post["date"])
111 |
112 | elements.append(
113 | {
114 | "title": post["title"],
115 | "link": episode_link,
116 | "description": post["content"],
117 | "guid": guid,
118 | "pubdate": datetime.strptime(post["date"], "%Y-%m-%dT%H:%M:%S.%fZ"),
119 | "enclosure": {
120 | "url": "{}{}.{}".format(settings["audio_base_url"], post["filename"], format),
121 | "length": filesize,
122 | "type": mime
123 | },
124 | "chapters": post["chapters"]
125 | }
126 | )
127 |
128 | return elements
129 |
130 |
131 | def _generate_index_pages(posts, settings, formats, index_template):
132 | format_mimes = {}
133 | for fmt in settings["feeds"]:
134 | format_mimes[fmt] = mimetypes[fmt]
135 |
136 | posts_on_page = settings.get("posts_on_page", None)
137 | cur_page = 0
138 | if posts_on_page is not None:
139 | while cur_page*posts_on_page < len(posts):
140 | page_posts = posts[posts_on_page*cur_page:posts_on_page*(cur_page+1)]
141 | if cur_page == 0:
142 | filename = os.path.join(TMP_PATH, "index.html")
143 | else:
144 | filename = os.path.join(TMP_PATH, "index.%i.html" % cur_page)
145 |
146 | with open(filename, "a+", encoding="utf-8") as f:
147 | if (cur_page + 1)*posts_on_page >= len(posts):
148 | next = None
149 | else:
150 | next = cur_page+1
151 | if cur_page > 0:
152 | prev = cur_page-1
153 | else:
154 | prev = None
155 | f.write(index_template.render(posts=page_posts, settings=settings, formats=format_mimes, index_page=True, prev=prev, next=next))
156 | cur_page += 1
157 |
158 | else:
159 | # write index.html with all posts
160 | with open(os.path.join(TMP_PATH, "index.html"), "a+", encoding="utf-8") as f:
161 | f.write(index_template.render(posts=posts, settings=settings, formats=format_mimes, index_page=True))
162 |
163 | def _generate_archive_page(posts, settings, formats, archive_template):
164 | format_mimes = {}
165 | for fmt in settings["feeds"]:
166 | format_mimes[fmt] = mimetypes[fmt]
167 |
168 | with open(os.path.join(TMP_PATH, "archive.html"), "a+", encoding="utf-8") as f:
169 | f.write(archive_template.render(posts=posts, settings=settings, formats=format_mimes))
170 |
171 | def json_transform(settings):
172 | """
173 | json_transform() transforms auphonic input files into post files
174 | that can be read by the generate() method
175 | """
176 |
177 | filefolder = settings["filefolder"]
178 |
179 | if not os.path.exists(filefolder):
180 | print("[INFO] No media in %s found." % filefolder)
181 | os.mkdir(filefolder)
182 |
183 | for fn in _get_json_file_list(filefolder):
184 | _write_json_data(filefolder, fn)
185 |
186 |
187 | def generate(settings):
188 | # settings mapping
189 | tplfolder = settings["tplfolder"]
190 | publish = settings["publish"]
191 | formats = settings["feeds"]
192 |
193 | # create output folder
194 | if os.path.exists(TMP_PATH):
195 | shutil.rmtree(TMP_PATH)
196 | os.mkdir(TMP_PATH)
197 |
198 | # load template files
199 | jenv = Environment(loader=FileSystemLoader(tplfolder))
200 | index_template = jenv.get_template("index.tpl")
201 | single_template = jenv.get_template("single.tpl")
202 | player_template = jenv.get_template("player.tpl")
203 | archive_template = jenv.get_template("archive.tpl")
204 | if os.path.exists(os.path.join(tplfolder, "style.css")):
205 | shutil.copy(os.path.join(tplfolder, "style.css"), os.path.join(TMP_PATH, "style.css"))
206 |
207 | # get posts from files
208 | posts = get_posts()
209 |
210 | # write posts with templates
211 | for post in posts:
212 | write_post(post, settings, single_template, player_template)
213 |
214 | _generate_index_pages(posts, settings, formats, index_template)
215 | _generate_archive_page(posts, settings, formats, archive_template)
216 |
217 | # generate feed_description
218 | for fmt in formats:
219 | elements = _get_elements(posts, settings, fmt)
220 |
221 | channel = {
222 | "title": settings["feed_title"].format(fmt),
223 | "link": settings["baseurl"],
224 | "feedinterval": settings["feedinterval"],
225 | "description": settings["feed_description"],
226 | "author": settings["author"]
227 | }
228 |
229 | if "artwork_url" in settings:
230 | channel["artwork"] = settings["artwork_url"]
231 |
232 | if "explicit" in settings:
233 | channel["explicit"] = settings["explicit"]
234 |
235 | with open(os.path.join(TMP_PATH, "{}.xml".format(fmt)), "wb") as f:
236 | f.write(rssgen.generate(channel=channel, elements=elements, settings=settings))
237 |
238 | # build search
239 | search.build_index(posts, os.path.join(TMP_PATH, "episode_index.json"))
240 |
241 | # copy from temp to production and remove tmp
242 | du.copy_tree(TMP_PATH, publish)
243 | shutil.rmtree(TMP_PATH)
244 |
245 | # link resources to pub
246 | src = os.path.join(publish, "res")
247 | dst = os.path.join(tplfolder, "res")
248 | if not os.path.exists(src):
249 | os.symlink(os.path.abspath(dst), src)
250 |
251 |
252 | def run():
253 | settings = get_settings()
254 | json_transform(settings)
255 | generate(settings)
256 |
257 |
258 | if __name__ == "__main__":
259 | if sys.version_info >= (2, 7, 0):
260 | print("Running Mikrowelle OS {}.{}.{}".format(__version__[0], __version__[1], __version__[2]))
261 | run()
262 | else:
263 | print("[ERROR] Your python interpreter version is too old. Required: 2.7")
264 |
--------------------------------------------------------------------------------
/requirements.txt:
--------------------------------------------------------------------------------
1 | Jinja2>=2.6
2 | Markdown>=2.2.1
3 | lxml>=3.2.3
--------------------------------------------------------------------------------
/settings.default.json:
--------------------------------------------------------------------------------
1 | {
2 | "tplfolder": "./templates/base-template/",
3 | "baseurl": "http://example.com/",
4 | "filefolder": "./audio/",
5 | "publish": "./pub/",
6 | "feeds": [
7 | "mp3",
8 | "m4a",
9 | "opus"
10 | ],
11 | "feedinterval": "1440",
12 | "web_title": "Podcastname",
13 | "feed_title": "Podcastname ({} Feed)",
14 | "feed_description": "Feedbeschreibung/Podcastbeschreibung",
15 | "author": "Podcastautor",
16 | "artwork_url": "http://example.com/logo.png",
17 | "audio_base_url": "http://audio.example.com/",
18 | "flattrlink": "https://flattr.com/submit/auto?url={}&title={}&user_id=FLATTRNAME&language=de_DE&category=audio",
19 | "category": "Technology & Education",
20 | "language": "de-de",
21 | "posts_on_page": 5,
22 | "explicit": false
23 | }
24 |
--------------------------------------------------------------------------------
/templates/base-template/archive.tpl:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | {{ settings.web_title }}
6 |
7 | {% include 'head.tpl' %}
8 |
9 |
10 |
11 |
12 | {{ settings.web_title }}
13 |
14 |
15 |
16 | {% include 'meta_block.tpl' %}
17 |
18 | {% for post in posts %}
19 |
20 | {{ post.title }}
21 | {{ post.subtitle }}
22 | Duration: {{ post.humanduration }} h
23 |
24 | Published on: {{ post.humandate }},
25 |
Direct Episode Link
26 | Download:
27 | {% for extension, mime in formats.items() %}
28 |
{{ extension }}
29 | {% endfor %}
30 |
31 |
32 | {% endfor %}
33 |
34 |
35 |
36 |
37 |
--------------------------------------------------------------------------------
/templates/base-template/head.tpl:
--------------------------------------------------------------------------------
1 | {% for feed in feeds %}
2 | {% endfor %}
3 |
4 |
5 |
10 |
--------------------------------------------------------------------------------
/templates/base-template/index.tpl:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | {{ settings.web_title }}
6 |
7 | {% include 'head.tpl' %}
8 |
9 |
10 |
11 |
12 | {{ settings.web_title }}
13 |
14 |
15 |
16 | {% include 'meta_block.tpl' %}
17 |
18 | {% for post in posts %}
19 | {% include 'post.tpl' %}
20 | {% endfor %}
21 |
22 | {% if prev == 0 %}
23 |
26 | {% elif prev and prev > 0 %}
27 |
30 | {% endif %}
31 | {% if next %}
32 |
35 | {% endif %}
36 |
37 |
38 |
39 |
40 |
--------------------------------------------------------------------------------
/templates/base-template/meta_block.tpl:
--------------------------------------------------------------------------------
1 |
2 | Feeds: {% for feed in settings.feeds %}{{ feed }} | {% endfor %}Bitlove
3 | All episodes: Archive
4 | Powered by Mikrowelle OS
5 |
6 |
--------------------------------------------------------------------------------
/templates/base-template/player.tpl:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 | {% for extension, mime in formats.items() %}
9 |
10 | {% endfor %}
11 |
12 |
13 |
14 |
15 |
28 |
29 |
--------------------------------------------------------------------------------
/templates/base-template/post.tpl:
--------------------------------------------------------------------------------
1 |
2 | {{ post.title }}
3 |
4 |
5 | {% for extension, mime in formats.items() %}
6 |
7 | {% endfor %}
8 |
9 |
10 | {{ post.content }}
11 |
12 |
13 | Published on: {{ post.humandate }},
14 |
Direct Episode Link
15 | Download:
16 | {% for extension, mime in formats.items() %}
17 |
{{ extension }}
18 | {% endfor %}
19 |
20 |
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/css/vendor/progress-polyfill.css:
--------------------------------------------------------------------------------
1 | progress[role] {
2 | display: inline-block;
3 | position: relative;
4 | width: 10em;
5 | height: 1em;
6 | vertical-align: -.2em;
7 | background-image: url('');
8 |
9 | -moz-box-sizing: border-box;
10 | box-sizing: border-box;
11 | }
12 |
13 | progress[role],
14 | progress[aria-valuenow]:before {
15 | background-color: #5af;
16 | }
17 |
18 | progress[role],
19 | progress[role]:after {
20 | background-repeat:repeat-x;
21 | background-position: 0 0;
22 | -moz-background-size: auto 100%;
23 | -webkit-background-size: auto 100%;
24 | background-size: auto 100%;
25 | }
26 |
27 | /* Determinate only overrides */
28 | progress[aria-valuenow] {
29 | background: #eee;
30 | }
31 |
32 | progress[aria-valuenow]:before {
33 | content: "";
34 | display: block;
35 | height: 100%;
36 | }
37 |
38 | /* Overlay */
39 | progress[role]:after {
40 | content: "";
41 | position: absolute;
42 | top: 0;
43 | right: 0;
44 | bottom: 0;
45 | left: 0;
46 | background-image: url('');
47 | }
48 |
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/css/vendor/style.css:
--------------------------------------------------------------------------------
1 | @import url(http://fonts.googleapis.com/css?family=Sniglet:800&v2);
2 |
3 | body {
4 | max-width: 800px;
5 | margin: 10px auto;
6 | font:16px/1.4 sans-serif;
7 | }
8 |
9 | a {
10 | color: inherit;
11 | text-decoration: none;
12 | }
13 |
14 | h1, h2, h3 {
15 | font-family: 'Sniglet', sans-serif;
16 | }
17 |
18 | h1 {
19 | font-size: 340%;
20 | margin-bottom: 0;
21 | }
22 |
23 | h2 {
24 | font-size: 240%;
25 | color: deeppink;
26 | margin: 1em 0 0;
27 | }
28 |
29 | h1+h2 {
30 | margin-top: 0;
31 | color: gray;
32 | }
33 |
34 | hgroup {
35 | text-align: center;
36 | }
37 |
38 | h3 {
39 | font-size: 150%;
40 | color: gray;
41 | }
42 |
43 | ul {
44 | margin: 0;
45 | padding-left:0;
46 | }
47 |
48 | h2+p, h3+p {
49 | margin-top: 0;
50 | }
51 |
52 | /* Unit tests styling */
53 | .unit-tests {
54 | padding: 0;
55 | margin: 0;
56 | }
57 |
58 | .unit-tests > li {
59 | margin: 5px 0;
60 | padding: 5px 10px;
61 | list-style: none;
62 | color: white;
63 | text-shadow: 1px 1px 2px rgba(0,0,0,.4);
64 | }
65 |
66 | .unit-tests > li.pass {
67 | background: green;
68 | }
69 |
70 | .unit-tests > li.fail {
71 | background: red;
72 | }
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/coverimage-blue.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/examples/coverimage-blue.png
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/coverimage-green.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/examples/coverimage-green.png
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/coverimage-red.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/examples/coverimage-red.png
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/coverimage.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/examples/coverimage.png
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/cre-206/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 | CRE 206 - media is loaded from source server
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
124 |
125 |
126 |
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/deeplinking/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 | Audio Tag Decoration Test Page
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 | Start Player at position 00:01.000
22 | Player Info for clients without Javascript enabled.
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 | Start Player at position 00:01.000 and stop at 00:02.000
34 | Player Info for clients without Javascript enabled.
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/examples.css:
--------------------------------------------------------------------------------
1 | body {
2 | background: #ffffff;
3 | font-family: 'Roboto', 'Helvetica Neue', 'Helvetica', sans-serif;
4 | }
5 |
6 | /* player */
7 |
8 | .podlove-player-wrapper {
9 | overflow: auto;
10 | -webkit-overflow-scrolling: touch;
11 | }
12 |
13 | .podlove-player-wrapper iframe {
14 | width: 1px;
15 | min-width: 100%;
16 | }
17 |
18 | h1 {
19 | color: #0E116C;
20 | font-size: 29px;
21 | width: 85%;
22 | }
23 |
24 | a {
25 | text-decoration: none;
26 | }
27 |
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/folder-30c3.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/examples/folder-30c3.png
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 | Audio Tag Decoration Test Page
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 | 1st embedded player
27 | First Player Info for clients without Javascript enabled.
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 | 2nd embedded player
39 | Second Player Info for clients without Javascript enabled.
40 |
41 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
56 |
57 |
58 |
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/injected/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 | Inject data test page
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 | Injected Data
23 | Media and Metadata are injected into the iframe
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
138 |
139 |
140 |
141 |
142 |
143 |
144 |
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/minimal/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 | MINIMAL
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 | Minimal Example
23 | The absolute minimal data set is handed over
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
91 |
92 |
93 |
94 |
95 |
96 |
97 |
98 |
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/picture-stream/Kapitel-Demo.m4a:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/examples/picture-stream/Kapitel-Demo.m4a
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/picture-stream/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 | Picture Stream
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
92 |
93 |
94 |
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/themes/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 | Audio Tag Decoration Test Page
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 | Player Info for clients without Javascript enabled.
28 |
31 |
32 |
33 |
34 |
35 | Player Info for clients without Javascript enabled.
36 |
39 |
40 |
41 |
42 |
43 | Player Info for clients without Javascript enabled.
44 |
47 |
48 |
49 |
50 |
51 | Player Info for clients without Javascript enabled.
52 |
55 |
56 |
57 |
58 |
59 | Player Info for clients without Javascript enabled.
60 |
63 |
64 |
65 |
66 |
67 | Player Info for clients without Javascript enabled.
68 |
71 |
72 |
73 |
74 |
75 | Player Info for clients without Javascript enabled.
76 |
77 |
78 |
79 |
80 |
81 |
82 | Player Info for clients without Javascript enabled.
83 |
84 |
85 |
86 |
87 |
88 |
89 | Player Info for clients without Javascript enabled.
90 |
91 |
92 |
93 |
94 |
95 |
96 | Player Info for clients without Javascript enabled.
97 |
98 |
99 |
100 |
101 |
102 |
103 | Player Info for clients without Javascript enabled.
104 |
105 |
106 |
107 |
108 |
109 |
110 | Player Info for clients without Javascript enabled.
111 |
112 |
113 |
114 |
115 |
116 |
117 |
118 |
119 |
120 |
121 |
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/themes/themed.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
Silver Blaze Theme Test Page
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
136 |
137 |
138 |
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/video/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
Video
8 |
9 |
10 |
11 |
12 |
13 |
14 |
Video Test Page
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
63 |
64 |
65 |
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/which-format/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
Which format plays?
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
114 |
115 |
116 |
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/which-format/podlove-test-track.mp3:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/examples/which-format/podlove-test-track.mp3
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/which-format/podlove-test-track.mp4:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/examples/which-format/podlove-test-track.mp4
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/which-format/podlove-test-track.ogg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/examples/which-format/podlove-test-track.ogg
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/examples/which-format/podlove-test-track.opus:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/examples/which-format/podlove-test-track.opus
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/font/podlove.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/font/podlove.eot
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/font/podlove.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/font/podlove.ttf
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/font/podlove.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/font/podlove.woff
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/img/arrow-down-black.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/img/arrow-down-black.png
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/img/arrow-down-white.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/img/arrow-down-white.png
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/img/download-overlay.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/img/download-overlay.png
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/img/icon-podlove-subscribe-600.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/thomersch/Mikrowelle-OS/b68617c005f22dce49624336928ad1619635da7a/templates/base-template/res/pwp/img/icon-podlove-subscribe-600.png
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
Podlove Web Player Documentation
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 | Podlove Webplayer
27 |
28 |
29 |
42 |
43 |
44 | This is an embedded player
45 | Player Info for clients without Javascript enabled.
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
55 |
56 |
57 |
58 |
59 |
60 |
--------------------------------------------------------------------------------
/templates/base-template/res/pwp/js/podlove-web-moderator.min.js:
--------------------------------------------------------------------------------
1 | !function e(t,n,r){function o(a,u){if(!n[a]){if(!t[a]){var s="function"==typeof require&&require;if(!u&&s)return s(a,!0);if(i)return i(a,!0);throw new Error("Cannot find module '"+a+"'")}var f=n[a]={exports:{}};t[a][0].call(f.exports,function(e){var n=t[a][1][e];return o(n?n:e)},f,f.exports,e,t,n,r)}return n[a].exports}for(var i="function"==typeof require&&require,a=0;a
u;u++)a[u]=o.isBuffer(e)?e.readUInt8(u):e[u];else if("string"===r)a.write(e,0,t);else if("number"===r&&!o._useTypedArrays&&!n)for(u=0;i>u;u++)a[u]=0;return a}function i(e,t,n,r){n=Number(n)||0;var i=e.length-n;r?(r=Number(r),r>i&&(r=i)):r=i;var a=t.length;R(a%2===0,"Invalid hex string"),r>a/2&&(r=a/2);for(var u=0;r>u;u++){var s=parseInt(t.substr(2*u,2),16);R(!isNaN(s),"Invalid hex string"),e[n+u]=s}return o._charsWritten=2*u,u}function a(e,t,n,r){var i=o._charsWritten=W(F(t),e,n,r);return i}function u(e,t,n,r){var i=o._charsWritten=W(j(t),e,n,r);return i}function s(e,t,n,r){return u(e,t,n,r)}function f(e,t,n,r){var i=o._charsWritten=W(O(t),e,n,r);return i}function l(e,t,n,r){var i=o._charsWritten=W(D(t),e,n,r);return i}function d(e,t,n){return 0===t&&n===e.length?X.fromByteArray(e):X.fromByteArray(e.slice(t,n))}function c(e,t,n){var r="",o="";n=Math.min(e.length,n);for(var i=t;n>i;i++)e[i]<=127?(r+=P(o)+String.fromCharCode(e[i]),o=""):o+="%"+e[i].toString(16);return r+P(o)}function h(e,t,n){var r="";n=Math.min(e.length,n);for(var o=t;n>o;o++)r+=String.fromCharCode(e[o]);return r}function g(e,t,n){return h(e,t,n)}function p(e,t,n){var r=e.length;(!t||0>t)&&(t=0),(!n||0>n||n>r)&&(n=r);for(var o="",i=t;n>i;i++)o+=N(e[i]);return o}function w(e,t,n){for(var r=e.slice(t,n),o="",i=0;i=o)){var i;return n?(i=e[t],o>t+1&&(i|=e[t+1]<<8)):(i=e[t]<<8,o>t+1&&(i|=e[t+1])),i}}function m(e,t,n,r){r||(R("boolean"==typeof n,"missing or invalid endian"),R(void 0!==t&&null!==t,"missing offset"),R(t+3=o)){var i;return n?(o>t+2&&(i=e[t+2]<<16),o>t+1&&(i|=e[t+1]<<8),i|=e[t],o>t+3&&(i+=e[t+3]<<24>>>0)):(o>t+1&&(i=e[t+1]<<16),o>t+2&&(i|=e[t+2]<<8),o>t+3&&(i|=e[t+3]),i+=e[t]<<24>>>0),i}}function b(e,t,n,r){r||(R("boolean"==typeof n,"missing or invalid endian"),R(void 0!==t&&null!==t,"missing offset"),R(t+1=o)){var i=y(e,t,n,!0),a=32768&i;return a?-1*(65535-i+1):i}}function v(e,t,n,r){r||(R("boolean"==typeof n,"missing or invalid endian"),R(void 0!==t&&null!==t,"missing offset"),R(t+3=o)){var i=m(e,t,n,!0),a=2147483648&i;return a?-1*(4294967295-i+1):i}}function E(e,t,n,r){return r||(R("boolean"==typeof n,"missing or invalid endian"),R(t+3=i))for(var a=0,u=Math.min(i-n,2);u>a;a++)e[n+a]=(t&255<<8*(r?a:1-a))>>>8*(r?a:1-a)}function B(e,t,n,r,o){o||(R(void 0!==t&&null!==t,"missing value"),R("boolean"==typeof r,"missing or invalid endian"),R(void 0!==n&&null!==n,"missing offset"),R(n+3=i))for(var a=0,u=Math.min(i-n,4);u>a;a++)e[n+a]=t>>>8*(r?a:3-a)&255}function _(e,t,n,r,o){o||(R(void 0!==t&&null!==t,"missing value"),R("boolean"==typeof r,"missing or invalid endian"),R(void 0!==n&&null!==n,"missing offset"),R(n+1=i||(t>=0?A(e,t,n,r,o):A(e,65535+t+1,n,r,o))}function L(e,t,n,r,o){o||(R(void 0!==t&&null!==t,"missing value"),R("boolean"==typeof r,"missing or invalid endian"),R(void 0!==n&&null!==n,"missing offset"),R(n+3=i||(t>=0?B(e,t,n,r,o):B(e,4294967295+t+1,n,r,o))}function M(e,t,n,r,o){o||(R(void 0!==t&&null!==t,"missing value"),R("boolean"==typeof r,"missing or invalid endian"),R(void 0!==n&&null!==n,"missing offset"),R(n+3=i||Y.write(e,t,n,r,23,4)}function U(e,t,n,r,o){o||(R(void 0!==t&&null!==t,"missing value"),R("boolean"==typeof r,"missing or invalid endian"),R(void 0!==n&&null!==n,"missing offset"),R(n+7=i||Y.write(e,t,n,r,52,8)}function C(e){return e.trim?e.trim():e.replace(/^\s+|\s+$/g,"")}function S(e,t,n){return"number"!=typeof e?n:(e=~~e,e>=t?t:e>=0?e:(e+=t,e>=0?e:0))}function k(e){return e=~~Math.ceil(+e),0>e?0:e}function x(e){return(Array.isArray||function(e){return"[object Array]"===Object.prototype.toString.call(e)})(e)}function T(e){return x(e)||o.isBuffer(e)||e&&"object"==typeof e&&"number"==typeof e.length}function N(e){return 16>e?"0"+e.toString(16):e.toString(16)}function F(e){for(var t=[],n=0;n=r)t.push(e.charCodeAt(n));else{var o=n;r>=55296&&57343>=r&&n++;for(var i=encodeURIComponent(e.slice(o,n+1)).substr(1).split("%"),a=0;a>8,r=t%256,o.push(r),o.push(n);return o}function O(e){return X.toByteArray(e)}function W(e,t,n,r){for(var o=0;r>o&&!(o+n>=t.length||o>=e.length);o++)t[o+n]=e[o];return o}function P(e){try{return decodeURIComponent(e)}catch(t){return String.fromCharCode(65533)}}function q(e,t){R("number"==typeof e,"cannot write a non-number as a number"),R(e>=0,"specified a negative value for writing an unsigned value"),R(t>=e,"value is larger than maximum value for type"),R(Math.floor(e)===e,"value has a fractional component")}function z(e,t,n){R("number"==typeof e,"cannot write a non-number as a number"),R(t>=e,"value larger than maximum allowed value"),R(e>=n,"value smaller than minimum allowed value"),R(Math.floor(e)===e,"value has a fractional component")}function J(e,t,n){R("number"==typeof e,"cannot write a non-number as a number"),R(t>=e,"value larger than maximum allowed value"),R(e>=n,"value smaller than minimum allowed value")}function R(e,t){if(!e)throw new Error(t||"Failed assertion")}var X=e("base64-js"),Y=e("ieee754");n.Buffer=o,n.SlowBuffer=o,n.INSPECT_MAX_BYTES=50,o.poolSize=8192,o._useTypedArrays=function(){try{var e=new ArrayBuffer(0),t=new Uint8Array(e);return t.foo=function(){return 42},42===t.foo()&&"function"==typeof t.subarray}catch(n){return!1}}(),o.isEncoding=function(e){switch(String(e).toLowerCase()){case"hex":case"utf8":case"utf-8":case"ascii":case"binary":case"base64":case"raw":case"ucs2":case"ucs-2":case"utf16le":case"utf-16le":return!0;default:return!1}},o.isBuffer=function(e){return!(null===e||void 0===e||!e._isBuffer)},o.byteLength=function(e,t){var n;switch(e+="",t||"utf8"){case"hex":n=e.length/2;break;case"utf8":case"utf-8":n=F(e).length;break;case"ascii":case"binary":case"raw":n=e.length;break;case"base64":n=O(e).length;break;case"ucs2":case"ucs-2":case"utf16le":case"utf-16le":n=2*e.length;break;default:throw new Error("Unknown encoding")}return n},o.concat=function(e,t){if(R(x(e),"Usage: Buffer.concat(list, [totalLength])\nlist should be an Array."),0===e.length)return new o(0);if(1===e.length)return e[0];var n;if("number"!=typeof t)for(t=0,n=0;nd&&(n=d)):n=d,r=String(r||"utf8").toLowerCase();var c;switch(r){case"hex":c=i(this,e,t,n);break;case"utf8":case"utf-8":c=a(this,e,t,n);break;case"ascii":c=u(this,e,t,n);break;case"binary":c=s(this,e,t,n);break;case"base64":c=f(this,e,t,n);break;case"ucs2":case"ucs-2":case"utf16le":case"utf-16le":c=l(this,e,t,n);break;default:throw new Error("Unknown encoding")}return c},o.prototype.toString=function(e,t,n){var r=this;if(e=String(e||"utf8").toLowerCase(),t=Number(t)||0,n=void 0!==n?Number(n):n=r.length,n===t)return"";var o;switch(e){case"hex":o=p(r,t,n);break;case"utf8":case"utf-8":o=c(r,t,n);break;case"ascii":o=h(r,t,n);break;case"binary":o=g(r,t,n);break;case"base64":o=d(r,t,n);break;case"ucs2":case"ucs-2":case"utf16le":case"utf-16le":o=w(r,t,n);break;default:throw new Error("Unknown encoding")}return o},o.prototype.toJSON=function(){return{type:"Buffer",data:Array.prototype.slice.call(this._arr||this,0)}},o.prototype.copy=function(e,t,n,r){var i=this;if(n||(n=0),r||0===r||(r=this.length),t||(t=0),r!==n&&0!==e.length&&0!==i.length){R(r>=n,"sourceEnd < sourceStart"),R(t>=0&&t=0&&n=0&&r<=i.length,"sourceEnd out of bounds"),r>this.length&&(r=this.length),e.length-ta||!o._useTypedArrays)for(var u=0;a>u;u++)e[u+t]=this[u+n];else e._set(this.subarray(n,n+a),t)}},o.prototype.slice=function(e,t){var n=this.length;if(e=S(e,n,0),t=S(t,n,n),o._useTypedArrays)return o._augment(this.subarray(e,t));for(var r=t-e,i=new o(r,void 0,!0),a=0;r>a;a++)i[a]=this[a+e];return i},o.prototype.get=function(e){return console.log(".get() is deprecated. Access using array indexes instead."),this.readUInt8(e)},o.prototype.set=function(e,t){return console.log(".set() is deprecated. Access using array indexes instead."),this.writeUInt8(e,t)},o.prototype.readUInt8=function(e,t){return t||(R(void 0!==e&&null!==e,"missing offset"),R(e=this.length?void 0:this[e]},o.prototype.readUInt16LE=function(e,t){return y(this,e,!0,t)},o.prototype.readUInt16BE=function(e,t){return y(this,e,!1,t)},o.prototype.readUInt32LE=function(e,t){return m(this,e,!0,t)},o.prototype.readUInt32BE=function(e,t){return m(this,e,!1,t)},o.prototype.readInt8=function(e,t){if(t||(R(void 0!==e&&null!==e,"missing offset"),R(e=this.length)){var n=128&this[e];return n?-1*(255-this[e]+1):this[e]}},o.prototype.readInt16LE=function(e,t){return b(this,e,!0,t)},o.prototype.readInt16BE=function(e,t){return b(this,e,!1,t)},o.prototype.readInt32LE=function(e,t){return v(this,e,!0,t)},o.prototype.readInt32BE=function(e,t){return v(this,e,!1,t)},o.prototype.readFloatLE=function(e,t){return E(this,e,!0,t)},o.prototype.readFloatBE=function(e,t){return E(this,e,!1,t)},o.prototype.readDoubleLE=function(e,t){return I(this,e,!0,t)},o.prototype.readDoubleBE=function(e,t){return I(this,e,!1,t)},o.prototype.writeUInt8=function(e,t,n){n||(R(void 0!==e&&null!==e,"missing value"),R(void 0!==t&&null!==t,"missing offset"),R(t=this.length||(this[t]=e)},o.prototype.writeUInt16LE=function(e,t,n){A(this,e,t,!0,n)},o.prototype.writeUInt16BE=function(e,t,n){A(this,e,t,!1,n)},o.prototype.writeUInt32LE=function(e,t,n){B(this,e,t,!0,n)},o.prototype.writeUInt32BE=function(e,t,n){B(this,e,t,!1,n)},o.prototype.writeInt8=function(e,t,n){n||(R(void 0!==e&&null!==e,"missing value"),R(void 0!==t&&null!==t,"missing offset"),R(t=this.length||(e>=0?this.writeUInt8(e,t,n):this.writeUInt8(255+e+1,t,n))},o.prototype.writeInt16LE=function(e,t,n){_(this,e,t,!0,n)},o.prototype.writeInt16BE=function(e,t,n){_(this,e,t,!1,n)},o.prototype.writeInt32LE=function(e,t,n){L(this,e,t,!0,n)},o.prototype.writeInt32BE=function(e,t,n){L(this,e,t,!1,n)},o.prototype.writeFloatLE=function(e,t,n){M(this,e,t,!0,n)},o.prototype.writeFloatBE=function(e,t,n){M(this,e,t,!1,n)},o.prototype.writeDoubleLE=function(e,t,n){U(this,e,t,!0,n)},o.prototype.writeDoubleBE=function(e,t,n){U(this,e,t,!1,n)},o.prototype.fill=function(e,t,n){if(e||(e=0),t||(t=0),n||(n=this.length),"string"==typeof e&&(e=e.charCodeAt(0)),R("number"==typeof e&&!isNaN(e),"value is not a number"),R(n>=t,"end < start"),n!==t&&0!==this.length){R(t>=0&&t=0&&n<=this.length,"end out of bounds");for(var r=t;n>r;r++)this[r]=e}},o.prototype.inspect=function(){for(var e=[],t=this.length,r=0;t>r;r++)if(e[r]=N(this[r]),r===n.INSPECT_MAX_BYTES){e[r+1]="...";break}return""},o.prototype.toArrayBuffer=function(){if("undefined"!=typeof Uint8Array){if(o._useTypedArrays)return new o(this).buffer;for(var e=new Uint8Array(this.length),t=0,n=e.length;n>t;t+=1)e[t]=this[t];return e.buffer}throw new Error("Buffer.toArrayBuffer not supported in this browser")};var Q=o.prototype;o._augment=function(e){return e._isBuffer=!0,e._get=e.get,e._set=e.set,e.get=Q.get,e.set=Q.set,e.write=Q.write,e.toString=Q.toString,e.toLocaleString=Q.toString,e.toJSON=Q.toJSON,e.copy=Q.copy,e.slice=Q.slice,e.readUInt8=Q.readUInt8,e.readUInt16LE=Q.readUInt16LE,e.readUInt16BE=Q.readUInt16BE,e.readUInt32LE=Q.readUInt32LE,e.readUInt32BE=Q.readUInt32BE,e.readInt8=Q.readInt8,e.readInt16LE=Q.readInt16LE,e.readInt16BE=Q.readInt16BE,e.readInt32LE=Q.readInt32LE,e.readInt32BE=Q.readInt32BE,e.readFloatLE=Q.readFloatLE,e.readFloatBE=Q.readFloatBE,e.readDoubleLE=Q.readDoubleLE,e.readDoubleBE=Q.readDoubleBE,e.writeUInt8=Q.writeUInt8,e.writeUInt16LE=Q.writeUInt16LE,e.writeUInt16BE=Q.writeUInt16BE,e.writeUInt32LE=Q.writeUInt32LE,e.writeUInt32BE=Q.writeUInt32BE,e.writeInt8=Q.writeInt8,e.writeInt16LE=Q.writeInt16LE,e.writeInt16BE=Q.writeInt16BE,e.writeInt32LE=Q.writeInt32LE,e.writeInt32BE=Q.writeInt32BE,e.writeFloatLE=Q.writeFloatLE,e.writeFloatBE=Q.writeFloatBE,e.writeDoubleLE=Q.writeDoubleLE,e.writeDoubleBE=Q.writeDoubleBE,e.fill=Q.fill,e.inspect=Q.inspect,e.toArrayBuffer=Q.toArrayBuffer,e}}).call(this,e("oMfpAn"),"undefined"!=typeof self?self:"undefined"!=typeof window?window:{},e("buffer").Buffer,arguments[3],arguments[4],arguments[5],arguments[6],"/../../node_modules/gulp-browserify/node_modules/browserify/node_modules/buffer/index.js","/../../node_modules/gulp-browserify/node_modules/browserify/node_modules/buffer")},{"base64-js":2,buffer:1,ieee754:3,oMfpAn:4}],2:[function(e,t,n){(function(){var e="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";!function(t){"use strict";function n(e){var t=e.charCodeAt(0);return t===a||t===d?62:t===u||t===c?63:s>t?-1:s+10>t?t-s+26+26:l+26>t?t-l:f+26>t?t-f+26:void 0}function r(e){function t(e){f[d++]=e}var r,o,a,u,s,f;if(e.length%4>0)throw new Error("Invalid string. Length must be a multiple of 4");var l=e.length;s="="===e.charAt(l-2)?2:"="===e.charAt(l-1)?1:0,f=new i(3*e.length/4-s),a=s>0?e.length-4:e.length;var d=0;for(r=0,o=0;a>r;r+=4,o+=3)u=n(e.charAt(r))<<18|n(e.charAt(r+1))<<12|n(e.charAt(r+2))<<6|n(e.charAt(r+3)),t((16711680&u)>>16),t((65280&u)>>8),t(255&u);return 2===s?(u=n(e.charAt(r))<<2|n(e.charAt(r+1))>>4,t(255&u)):1===s&&(u=n(e.charAt(r))<<10|n(e.charAt(r+1))<<4|n(e.charAt(r+2))>>2,t(u>>8&255),t(255&u)),f}function o(t){function n(t){return e.charAt(t)}function r(e){return n(e>>18&63)+n(e>>12&63)+n(e>>6&63)+n(63&e)}var o,i,a,u=t.length%3,s="";for(o=0,a=t.length-u;a>o;o+=3)i=(t[o]<<16)+(t[o+1]<<8)+t[o+2],s+=r(i);switch(u){case 1:i=t[t.length-1],s+=n(i>>2),s+=n(i<<4&63),s+="==";break;case 2:i=(t[t.length-2]<<8)+t[t.length-1],s+=n(i>>10),s+=n(i>>4&63),s+=n(i<<2&63),s+="="}return s}var i="undefined"!=typeof Uint8Array?Uint8Array:Array,a="+".charCodeAt(0),u="/".charCodeAt(0),s="0".charCodeAt(0),f="a".charCodeAt(0),l="A".charCodeAt(0),d="-".charCodeAt(0),c="_".charCodeAt(0);t.toByteArray=r,t.fromByteArray=o}("undefined"==typeof n?this.base64js={}:n)}).call(this,e("oMfpAn"),"undefined"!=typeof self?self:"undefined"!=typeof window?window:{},e("buffer").Buffer,arguments[3],arguments[4],arguments[5],arguments[6],"/../../node_modules/gulp-browserify/node_modules/browserify/node_modules/buffer/node_modules/base64-js/lib/b64.js","/../../node_modules/gulp-browserify/node_modules/browserify/node_modules/buffer/node_modules/base64-js/lib")},{buffer:1,oMfpAn:4}],3:[function(e,t,n){(function(){n.read=function(e,t,n,r,o){var i,a,u=8*o-r-1,s=(1<>1,l=-7,d=n?o-1:0,c=n?-1:1,h=e[t+d];for(d+=c,i=h&(1<<-l)-1,h>>=-l,l+=u;l>0;i=256*i+e[t+d],d+=c,l-=8);for(a=i&(1<<-l)-1,i>>=-l,l+=r;l>0;a=256*a+e[t+d],d+=c,l-=8);if(0===i)i=1-f;else{if(i===s)return a?0/0:1/0*(h?-1:1);a+=Math.pow(2,r),i-=f}return(h?-1:1)*a*Math.pow(2,i-r)},n.write=function(e,t,n,r,o,i){var a,u,s,f=8*i-o-1,l=(1<>1,c=23===o?Math.pow(2,-24)-Math.pow(2,-77):0,h=r?0:i-1,g=r?1:-1,p=0>t||0===t&&0>1/t?1:0;for(t=Math.abs(t),isNaN(t)||1/0===t?(u=isNaN(t)?1:0,a=l):(a=Math.floor(Math.log(t)/Math.LN2),t*(s=Math.pow(2,-a))<1&&(a--,s*=2),t+=a+d>=1?c/s:c*Math.pow(2,1-d),t*s>=2&&(a++,s/=2),a+d>=l?(u=0,a=l):a+d>=1?(u=(t*s-1)*Math.pow(2,o),a+=d):(u=t*Math.pow(2,d-1)*Math.pow(2,o),a=0));o>=8;e[n+h]=255&u,h+=g,u/=256,o-=8);for(a=a<0;e[n+h]=255&a,h+=g,a/=256,f-=8);e[n+h-g]|=128*p}}).call(this,e("oMfpAn"),"undefined"!=typeof self?self:"undefined"!=typeof window?window:{},e("buffer").Buffer,arguments[3],arguments[4],arguments[5],arguments[6],"/../../node_modules/gulp-browserify/node_modules/browserify/node_modules/buffer/node_modules/ieee754/index.js","/../../node_modules/gulp-browserify/node_modules/browserify/node_modules/buffer/node_modules/ieee754")},{buffer:1,oMfpAn:4}],4:[function(e,t){(function(e){function n(){}var e=t.exports={};e.nextTick=function(){var e="undefined"!=typeof window&&window.setImmediate,t="undefined"!=typeof window&&window.postMessage&&window.addEventListener;if(e)return function(e){return window.setImmediate(e)};if(t){var n=[];return window.addEventListener("message",function(e){var t=e.source;if((t===window||null===t)&&"process-tick"===e.data&&(e.stopPropagation(),n.length>0)){var r=n.shift();r()}},!0),function(e){n.push(e),window.postMessage("process-tick","*")}}return function(e){setTimeout(e,0)}}(),e.title="browser",e.browser=!0,e.env={},e.argv=[],e.on=n,e.addListener=n,e.once=n,e.off=n,e.removeListener=n,e.removeAllListeners=n,e.emit=n,e.binding=function(){throw new Error("process.binding is not supported")},e.cwd=function(){return"/"},e.chdir=function(){throw new Error("process.chdir is not supported")}}).call(this,e("oMfpAn"),"undefined"!=typeof self?self:"undefined"!=typeof window?window:{},e("buffer").Buffer,arguments[3],arguments[4],arguments[5],arguments[6],"/../../node_modules/gulp-browserify/node_modules/browserify/node_modules/process/browser.js","/../../node_modules/gulp-browserify/node_modules/browserify/node_modules/process")},{buffer:1,oMfpAn:4}],5:[function(e){(function(){"use strict";function t(e,t,n){return t>e||e>n}function n(e){return!e||isNaN(e)?(console.warn("Set frame height to default"),h):(t(e,g,p)||console.warn("Frame height",e,"out of bounds."),d(e,g,p))}function r(){return"100%"}function o(e){var t=e.href,n=t.indexOf("#"),r=t.length;return n>=0&&(r=n),t.substring(0,r)}function i(e){if(!f.staticEmbedPage)throw new Error('"staticEmbedPage" parameter missing.');return f.staticEmbedPage+"?"+e}function a(){var e,t,o,a=c(this),u=a.data("podlove-web-player-source");if(!u){var s=a.get(0).id;if(!s)throw new Error("Element without source set needs an ID");if(u=i(s),o=window[m][s],!o)throw new Error('No data found for "'+s+'"')}return y&&b[0]&&(y=!1,u+="#t="+l.getFragment("t")),e=c("