diff --git a/chapters.py b/chapters.py index 44d37e4cf8d8f1c56fd919d284b399b709b834b1..5eb22d3aa515ddc886b51b027cba1611bed8e419 100644 --- a/chapters.py +++ b/chapters.py @@ -1,3 +1,4 @@ +import json from server import * @app.route('/internal/newchapter/<int:lectureid>', methods=['POST', 'GET']) @@ -31,4 +32,6 @@ def chapters(lectureid): c['start'] = c['time'] c['end'] = last['start'] if last else 9999 last = c + if 'json' in request.values: + return Response(json.dumps([{'time': c['time'], 'text': c['text']} for c in chapters]), mimetype='application/json') return Response(render_template('chapters.srt',chapters=chapters), 200, {'Content-Type':'text/vtt'}) diff --git a/db.py b/db.py index 9ae0d82f465dce55d9dc852763a281773c084414..3965fae7dda6d8604551b11cfa309f9a16d046ca 100644 --- a/db.py +++ b/db.py @@ -42,6 +42,9 @@ if config['DB_ENGINE'] == 'sqlite': params = [(p.replace(microsecond=0) if isinstance(p, datetime) else p) for p in params] return operation, params + def show(operation, host=None): + return {} + elif config['DB_ENGINE'] == 'mysql': import mysql.connector @@ -57,6 +60,29 @@ elif config['DB_ENGINE'] == 'mysql': params = [(p.replace(microsecond=0) if isinstance(p, datetime) else p) for p in params] return operation, params + def show(operation, host=config.get('MYSQL_HOST', None)): + if host: + db = mysql.connector.connect(user=config['MYSQL_USER'], password=config['MYSQL_PASSWD'], host=host, port=config.get('MYSQL_PORT', 3306)) + else: + db = mysql.connector.connect(user=config['MYSQL_USER'], password=config['MYSQL_PASSWD'], unix_socket=config.get('MYSQL_UNIX', None)) + cur = db.cursor() + cur.execute(operation) + rows = [] + try: + rows = cur.fetchall() + except mysql.connector.errors.InterfaceError as ie: + if ie.msg == 'No result set to fetch from.': + # no problem, we were just at the end of the result set + pass + else: + raise + res = {} + for row in rows: + res[row[0]] = row[1] + cur.close() + db.close() + return res + def query(operation, *params, delim="sep"): operation, params = fix_query(operation, params) cur = get_dbcursor() diff --git a/db_schema.sql b/db_schema.sql index 85036ec2148f7d5e7f6b6884b18ba8d2cb012e0f..44ac919a500b8b2dfb18350d02c58c20aa7efd71 100644 --- a/db_schema.sql +++ b/db_schema.sql @@ -61,7 +61,8 @@ CREATE TABLE IF NOT EXISTS `courses_data` ( `description` text NOT NULL DEFAULT '', `internal` text NOT NULL DEFAULT '', `responsible` text NOT NULL DEFAULT '', - `feed_url` text NOT NULL DEFAULT '' + `feed_url` text NOT NULL DEFAULT '', + `external` INTEGER NOT NULL DEFAULT 0 ); CREATE TABLE IF NOT EXISTS `filesizes` ( `path` varchar(255) NOT NULL PRIMARY KEY, @@ -98,7 +99,8 @@ CREATE TABLE IF NOT EXISTS `lectures_data` ( `time_updated` datetime NOT NULL, `jumplist` text NOT NULL DEFAULT '', `titlefile` varchar(255) NOT NULL DEFAULT '', - `live` INTEGER NOT NULL DEFAULT 0 + `live` INTEGER NOT NULL DEFAULT 0, + `norecording` INTEGER NOT NULL DEFAULT 0 ); CREATE TABLE IF NOT EXISTS `places` ( `place` varchar(20) NOT NULL PRIMARY KEY, diff --git a/edit.py b/edit.py index a00f7a6d6801ddf4ad505c0dc5929f44fa6b0e06..cfd7d1e7debcc85dbd8ef8f29bf259e7def2296d 100644 --- a/edit.py +++ b/edit.py @@ -25,7 +25,8 @@ editable_tables = { 'internal': {'type': 'text'}, 'responsible': {'type': 'shortstring'}, 'deleted': {'type': 'boolean'}, - 'description': {'type': 'text'} }, + 'description': {'type': 'text'}, + 'external': {'type': 'boolean'}}, 'creationtime_fields': ['created_by', 'time_created', 'time_updated'] }, 'lectures': { 'table': 'lectures_data', @@ -41,7 +42,8 @@ editable_tables = { 'duration': {'type': 'duration'}, 'jumplist': {'type': ''}, 'deleted': {'type': 'boolean'}, - 'live': {'type': 'boolean'}}, + 'live': {'type': 'boolean'}, + 'norecording': {'type': 'boolean'}}, 'creationtime_fields': ['course_id', 'time_created', 'time_updated'] }, 'videos': { 'table': 'videos_data', diff --git a/server.py b/server.py index 288b462e67c6524345b8d7c144229e3b101052d4..3cf8f28dc17ff0c58dc8dfecb1193d0a08039d62 100644 --- a/server.py +++ b/server.py @@ -70,7 +70,7 @@ app.jinja_env.globals['gitversion'] = { 'hash': output[1], 'longhash': output[0] if not config.get('SECRET_KEY', None): config['SECRET_KEY'] = os.urandom(24) -from db import query, modify, searchquery, ldapauth, ldapget +from db import query, modify, show, searchquery, ldapauth, ldapget mod_endpoints = [] @@ -325,14 +325,14 @@ def index(): return "Not found", 404 return redirect(url_for('lecture', course=courses[0]['handle'], id=request.args['lectureid']),code=302) - start = date.today() - timedelta(days=1) + start = date.today() end = start + timedelta(days=7) upcomming = query(''' SELECT lectures.*, "course" AS sep, courses.* FROM lectures JOIN courses ON (lectures.course_id = courses.id) - WHERE (time > ?) AND (time < ?) and lectures.visible and courses.visible and courses.listed - ORDER BY time ASC LIMIT 30''',start,end) + WHERE (time > ?) AND (time < ?) AND (? OR (lectures.visible AND courses.visible AND courses.listed)) AND NOT lectures.norecording + ORDER BY time ASC LIMIT 30''', start, end, ismod()) for i in upcomming: i['date'] = i['time'].date() latestvideos=query(''' @@ -509,7 +509,7 @@ def login(): @app.route('/internal/logout', methods=['GET', 'POST']) def logout(): - session.pop('user') + session.pop('user', None) return redirect(request.values.get('ref', url_for('index'))) @app.route('/internal/auth') @@ -617,7 +617,33 @@ def legacy(phpfile=None): return redirect(url_for('feed', handle=request.args.copy().popitem()[0]),code=302) print("Unknown legacy url:",request.url) return redirect(url_for('index'),code=302) - + +import json + +@app.route('/internal/dbstatus') +@register_navbar('DB-Status', icon='ok') +@mod_required +def dbstatus(): + hosts = set() + clusters = {} + status = {} + variables = {} + for host in config.get('MYSQL_DBSTATUS_HOSTS', [])+[config.get('MYSQL_HOST', None)]: + for _host in show('SHOW VARIABLES LIKE "wsrep_cluster_address"', host=host)['wsrep_cluster_address'][len('gcomm://'):].split(','): + hosts.add(_host) + for host in sorted(list(hosts)): + try: + status[host] = show('SHOW GLOBAL STATUS LIKE "wsrep%"', host=host) + variables[host] = show('SHOW GLOBAL VARIABLES LIKE "wsrep%"', host=host) + except: + status[host] = {'wsrep_cluster_state_uuid': '', 'wsrep_local_state_comment': 'Not reachable', 'wsrep_cluster_conf_id': '0', 'wsrep_cluster_status': 'Unknown'} + variables[host] = {'wsrep_node_name': host, 'wsrep_cluster_name': 'unknown'} + cluster = variables[host]['wsrep_cluster_name']+'-'+status[host]['wsrep_cluster_conf_id'] + if cluster not in clusters: + clusters[cluster] = [] + clusters[cluster].append(host) + return render_template('dbstatus.html', clusters=clusters, statuses=status, vars=variables), 200 + import edit import feeds import importer diff --git a/static/videojs/video-js.css b/static/videojs/video-js.css index ef9ff8a37598bf7a1fd12d88f94b57f2378aaa61..821bac9627ed58b0d49383b976f3bcfd7c234dca 100644 --- a/static/videojs/video-js.css +++ b/static/videojs/video-js.css @@ -450,7 +450,7 @@ body.vjs-full-window { line-height: 1.4em; font-size: 1.2em; text-align: center; - text-transform: lowercase; } + /* text-transform: lowercase;}*/ } .vjs-menu li:focus, .vjs-menu li:hover { diff --git a/static/videojs/video.js b/static/videojs/video.js index 9419308b41bce458e90c619ac76ba8c3d21089c9..9cb476d7c5aff8e675d464533d17fe1b74987726 100644 --- a/static/videojs/video.js +++ b/static/videojs/video.js @@ -4404,7 +4404,7 @@ var ChaptersButton = function (_TextTrackButton) { var title = Dom.createEl('li', { className: 'vjs-menu-title', - innerHTML: (0, _toTitleCase2['default'])(this.kind_), + innerHTML: 'Kapitel', tabIndex: -1 }); diff --git a/static/videojs/videojs-markers.js b/static/videojs/videojs-markers.js new file mode 100644 index 0000000000000000000000000000000000000000..d89b4531485cd8fedf83d0a81d389ca5eee55738 --- /dev/null +++ b/static/videojs/videojs-markers.js @@ -0,0 +1,379 @@ +/*! videojs-markers - v0.6.1 - 2016-10-24 +* Copyright (c) 2016 ; Licensed */ +'use strict'; + +(function ($, videojs, undefined) { + // default setting + var defaultSetting = { + markerStyle: { + 'width': '7px', + 'border-radius': '30%', + 'background-color': 'red' + }, + markerTip: { + display: true, + text: function text(marker) { + return "Break: " + marker.text; + }, + time: function time(marker) { + return marker.time; + } + }, + breakOverlay: { + display: false, + displayTime: 3, + text: function text(marker) { + return "Break overlay: " + marker.overlayText; + }, + style: { + 'width': '100%', + 'height': '20%', + 'background-color': 'rgba(0,0,0,0.7)', + 'color': 'white', + 'font-size': '17px' + } + }, + onMarkerClick: function onMarkerClick(marker) {}, + onMarkerReached: function onMarkerReached(marker, index) {}, + markers: [] + }; + + // create a non-colliding random number + function generateUUID() { + var d = new Date().getTime(); + var uuid = 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) { + var r = (d + Math.random() * 16) % 16 | 0; + d = Math.floor(d / 16); + return (c == 'x' ? r : r & 0x3 | 0x8).toString(16); + }); + return uuid; + }; + + var NULL_INDEX = -1; + + function registerVideoJsMarkersPlugin(options) { + /** + * register the markers plugin (dependent on jquery) + */ + + var setting = $.extend(true, {}, defaultSetting, options), + markersMap = {}, + markersList = [], + // list of markers sorted by time + videoWrapper = $(this.el()), + currentMarkerIndex = NULL_INDEX, + player = this, + markerTip = null, + breakOverlay = null, + overlayIndex = NULL_INDEX; + + function sortMarkersList() { + // sort the list by time in asc order + markersList.sort(function (a, b) { + return setting.markerTip.time(a) - setting.markerTip.time(b); + }); + } + + function addMarkers(newMarkers) { + newMarkers.forEach(function (marker) { + marker.key = generateUUID(); + + videoWrapper.find('.vjs-progress-holder').append(createMarkerDiv(marker)); + + // store marker in an internal hash map + markersMap[marker.key] = marker; + markersList.push(marker); + }); + + sortMarkersList(); + } + + function getPosition(marker) { + return setting.markerTip.time(marker) / player.duration() * 100; + } + + function createMarkerDiv(marker) { + var markerDiv = $("<div class='vjs-marker'></div>"); + markerDiv.css(setting.markerStyle).css({ + "margin-left": -parseFloat(markerDiv.css("width")) / 2 + 'px', + "left": getPosition(marker) + '%' + }).attr("data-marker-key", marker.key).attr("data-marker-time", setting.markerTip.time(marker)); + + // add user-defined class to marker + if (marker.class) { + markerDiv.addClass(marker.class); + } + + // bind click event to seek to marker time + markerDiv.on('click', function (e) { + var preventDefault = false; + if (typeof setting.onMarkerClick === "function") { + // if return false, prevent default behavior + preventDefault = setting.onMarkerClick(marker) === false; + } + + if (!preventDefault) { + var key = $(this).data('marker-key'); + player.currentTime(setting.markerTip.time(markersMap[key])); + } + }); + + if (setting.markerTip.display) { + registerMarkerTipHandler(markerDiv); + } + + return markerDiv; + } + + function updateMarkers() { + // update UI for markers whose time changed + markersList.forEach(function (marker) { + var markerDiv = videoWrapper.find(".vjs-marker[data-marker-key='" + marker.key + "']"); + var markerTime = setting.markerTip.time(marker); + + if (markerDiv.data('marker-time') !== markerTime) { + markerDiv.css({ "left": getPosition(marker) + '%' }).attr("data-marker-time", markerTime); + } + }); + sortMarkersList(); + } + + function removeMarkers(indexArray) { + // reset overlay + if (!!breakOverlay) { + overlayIndex = NULL_INDEX; + breakOverlay.css("visibility", "hidden"); + } + currentMarkerIndex = NULL_INDEX; + + var deleteIndexList = []; + indexArray.forEach(function (index) { + var marker = markersList[index]; + if (marker) { + // delete from memory + delete markersMap[marker.key]; + deleteIndexList.push(index); + + // delete from dom + videoWrapper.find(".vjs-marker[data-marker-key='" + marker.key + "']").remove(); + } + }); + + // clean up markers array + deleteIndexList.reverse(); + deleteIndexList.forEach(function (deleteIndex) { + markersList.splice(deleteIndex, 1); + }); + + // sort again + sortMarkersList(); + } + + // attach hover event handler + function registerMarkerTipHandler(markerDiv) { + markerDiv.on('mouseover', function () { + var marker = markersMap[$(markerDiv).data('marker-key')]; + + if (!!markerTip) { + markerTip.find('.vjs-tip-inner').text(setting.markerTip.text(marker)); + + // margin-left needs to minus the padding length to align correctly with the marker + markerTip.css({ + "left": getPosition(marker) + '%', + "margin-left": -parseFloat(markerTip.width()) / 2 - 5 + 'px', + "visibility": "visible" + }); + } + }); + + markerDiv.on('mouseout', function () { + !!markerTip && markerTip.css("visibility", "hidden"); + }); + } + + function initializeMarkerTip() { + markerTip = $("<div class='vjs-tip'><div class='vjs-tip-arrow'></div><div class='vjs-tip-inner'></div></div>"); + videoWrapper.find('.vjs-progress-holder').append(markerTip); + } + + // show or hide break overlays + function updateBreakOverlay() { + if (!setting.breakOverlay.display || currentMarkerIndex < 0) { + return; + } + + var currentTime = player.currentTime(); + var marker = markersList[currentMarkerIndex]; + var markerTime = setting.markerTip.time(marker); + + if (currentTime >= markerTime && currentTime <= markerTime + setting.breakOverlay.displayTime) { + if (overlayIndex !== currentMarkerIndex) { + overlayIndex = currentMarkerIndex; + breakOverlay && breakOverlay.find('.vjs-break-overlay-text').html(setting.breakOverlay.text(marker)); + } + + breakOverlay && breakOverlay.css('visibility', "visible"); + } else { + overlayIndex = NULL_INDEX; + breakOverlay && breakOverlay.css("visibility", "hidden"); + } + } + + // problem when the next marker is within the overlay display time from the previous marker + function initializeOverlay() { + breakOverlay = $("<div class='vjs-break-overlay'><div class='vjs-break-overlay-text'></div></div>").css(setting.breakOverlay.style); + videoWrapper.append(breakOverlay); + overlayIndex = NULL_INDEX; + } + + function onTimeUpdate() { + onUpdateMarker(); + updateBreakOverlay(); + options.onTimeUpdateAfterMarkerUpdate && options.onTimeUpdateAfterMarkerUpdate(); + } + + function onUpdateMarker() { + /* + check marker reached in between markers + the logic here is that it triggers a new marker reached event only if the player + enters a new marker range (e.g. from marker 1 to marker 2). Thus, if player is on marker 1 and user clicked on marker 1 again, no new reached event is triggered) + */ + if (!markersList.length) { + return; + } + + var getNextMarkerTime = function getNextMarkerTime(index) { + if (index < markersList.length - 1) { + return setting.markerTip.time(markersList[index + 1]); + } + // next marker time of last marker would be end of video time + return player.duration(); + }; + var currentTime = player.currentTime(); + var newMarkerIndex = NULL_INDEX; + + if (currentMarkerIndex !== NULL_INDEX) { + // check if staying at same marker + var nextMarkerTime = getNextMarkerTime(currentMarkerIndex); + if (currentTime >= setting.markerTip.time(markersList[currentMarkerIndex]) && currentTime < nextMarkerTime) { + return; + } + + // check for ending (at the end current time equals player duration) + if (currentMarkerIndex === markersList.length - 1 && currentTime === player.duration()) { + return; + } + } + + // check first marker, no marker is selected + if (currentTime < setting.markerTip.time(markersList[0])) { + newMarkerIndex = NULL_INDEX; + } else { + // look for new index + for (var i = 0; i < markersList.length; i++) { + nextMarkerTime = getNextMarkerTime(i); + if (currentTime >= setting.markerTip.time(markersList[i]) && currentTime < nextMarkerTime) { + newMarkerIndex = i; + break; + } + } + } + + // set new marker index + if (newMarkerIndex !== currentMarkerIndex) { + // trigger event if index is not null + if (newMarkerIndex !== NULL_INDEX && options.onMarkerReached) { + options.onMarkerReached(markersList[newMarkerIndex], newMarkerIndex); + } + currentMarkerIndex = newMarkerIndex; + } + } + + // setup the whole thing + function initialize() { + if (setting.markerTip.display) { + initializeMarkerTip(); + } + + // remove existing markers if already initialized + player.markers.removeAll(); + addMarkers(options.markers); + + if (setting.breakOverlay.display) { + initializeOverlay(); + } + onTimeUpdate(); + player.on("timeupdate", onTimeUpdate); + } + + // setup the plugin after we loaded video's meta data + player.on("loadedmetadata", function () { + initialize(); + }); + + // exposed plugin API + player.markers = { + getMarkers: function getMarkers() { + return markersList; + }, + next: function next() { + // go to the next marker from current timestamp + var currentTime = player.currentTime(); + for (var i = 0; i < markersList.length; i++) { + var markerTime = setting.markerTip.time(markersList[i]); + if (markerTime > currentTime) { + player.currentTime(markerTime); + break; + } + } + }, + prev: function prev() { + // go to previous marker + var currentTime = player.currentTime(); + for (var i = markersList.length - 1; i >= 0; i--) { + var markerTime = setting.markerTip.time(markersList[i]); + // add a threshold + if (markerTime + 0.5 < currentTime) { + player.currentTime(markerTime); + return; + } + } + }, + add: function add(newMarkers) { + // add new markers given an array of index + addMarkers(newMarkers); + }, + remove: function remove(indexArray) { + // remove markers given an array of index + removeMarkers(indexArray); + }, + removeAll: function removeAll() { + var indexArray = []; + for (var i = 0; i < markersList.length; i++) { + indexArray.push(i); + } + removeMarkers(indexArray); + }, + updateTime: function updateTime() { + // notify the plugin to update the UI for changes in marker times + updateMarkers(); + }, + reset: function reset(newMarkers) { + // remove all the existing markers and add new ones + player.markers.removeAll(); + addMarkers(newMarkers); + }, + destroy: function destroy() { + // unregister the plugins and clean up even handlers + player.markers.removeAll(); + breakOverlay && breakOverlay.remove(); + markerTip && markerTip.remove(); + player.off("timeupdate", updateBreakOverlay); + delete player.markers; + } + }; + } + + videojs.plugin('markers', registerVideoJsMarkersPlugin); +})(jQuery, window.videojs); +//# sourceMappingURL=videojs-markers.js.map diff --git a/static/videojs/videojs.markers.css b/static/videojs/videojs.markers.css new file mode 100644 index 0000000000000000000000000000000000000000..7655a64f877f1876e46c464435aedb29b5bd99b8 --- /dev/null +++ b/static/videojs/videojs.markers.css @@ -0,0 +1,59 @@ +.vjs-marker { + position: absolute; + left: 0; + bottom: 0em; + opacity: 1; + height: 100%; + transition: opacity .2s ease; + -webkit-transition: opacity .2s ease; + -moz-transition: opacity .2s ease; + z-index: 100; +} +.vjs-marker:hover { + cursor: pointer; + -webkit-transform: scale(1.3, 1.3); + -moz-transform: scale(1.3, 1.3); + -o-transform: scale(1.3, 1.3); + -ms-transform: scale(1.3, 1.3); + transform: scale(1.3, 1.3); +} +.vjs-tip { + visibility: hidden; + display: block; + opacity: 0.8; + padding: 5px; + font-size: 10px; + position: absolute; + bottom: 14px; + z-index: 100000; +} +.vjs-tip .vjs-tip-arrow { + background: url() no-repeat top left; + bottom: 0; + left: 50%; + margin-left: -4px; + background-position: bottom left; + position: absolute; + width: 9px; + height: 5px; +} +.vjs-tip .vjs-tip-inner { + border-radius: 3px; + -moz-border-radius: 3px; + -webkit-border-radius: 3px; + padding: 5px 8px 4px 8px; + background-color: black; + color: white; + max-width: 200px; + text-align: center; +} +.vjs-break-overlay { + visibility: hidden; + position: absolute; + z-index: 100000; + top: 0; +} +.vjs-break-overlay .vjs-break-overlay-text { + padding: 9px; + text-align: center; +} diff --git a/templates/base.html b/templates/base.html index 4ac3bad76935eb1eaaa4257b52d2f7bf24e4f486..31bdd63c5df108ddf0271eab367dd16c295d5128 100644 --- a/templates/base.html +++ b/templates/base.html @@ -18,6 +18,7 @@ <link rel="stylesheet" type="text/css" href="{{url_for('static', filename='style.css')}}"> <link rel="stylesheet" href="{{url_for('static', filename='font-awesome/css/font-awesome.css')}}"> <link rel="stylesheet" type="text/css" href="{{url_for('static', filename='videojs/video-js.css')}}"> + <link rel="stylesheet" type="text/css" href="{{url_for('static', filename='videojs/videojs.markers.css')}}"> <link rel="stylesheet" type="text/css" href="{{url_for('static', filename='videojs/videojs-resolution-switcher.css')}}"> <script src="{{url_for('static', filename='jquery.js')}}"></script> @@ -33,6 +34,7 @@ <script src="{{url_for('static', filename='videojs/videojs-resolution-switcher.js')}}"></script> <script src="{{url_for('static', filename='videojs/videojs-contrib-hls.js')}}"></script> <script src="{{url_for('static', filename='videojs/videojs.hotkeys.js')}}"></script> + <script src="{{url_for('static', filename='videojs/videojs-markers.js')}}"></script> {% endblock %} </head> <body> diff --git a/templates/course.html b/templates/course.html index 9868782fac52a58f22e958483f919f012ad6e4a2..6bbb9e915a58af67c96861574ed0937f2ddd3f3a 100644 --- a/templates/course.html +++ b/templates/course.html @@ -41,6 +41,7 @@ <tbody> <tr><td>Sichtbar:</td><td>{{ moderator_checkbox(['courses',course.id,'visible'], course.visible) }}</td></tr> <tr><td>Gelistet:</td><td>{{ moderator_checkbox(['courses',course.id,'listed'], course.listed) }}</td></tr> + <tr><td>Nicht im Drehplan:</td><td>{{ moderator_checkbox(['courses',course.id,'external'], course.external) }}</td></tr> <tr><td>Videos downloadbar:</td><td>{{ moderator_checkbox(['courses',course.id,'downloadable'], course.downloadable) }}</td></tr> <tr><td>Short:</td><td>{{ moderator_editor(['courses',course.id,'short'], course.short) }}</td></tr> <tr><td>Handle:</td><td>{{ moderator_editor(['courses',course.id,'handle'], course.handle) }}</td></tr> diff --git a/templates/dbstatus.html b/templates/dbstatus.html new file mode 100644 index 0000000000000000000000000000000000000000..384e3a84446cd6003c08ae676d673cffb72281fc --- /dev/null +++ b/templates/dbstatus.html @@ -0,0 +1,53 @@ +{% extends "base.html" %} +{% block content %} +<div class="panel panel-default"> + <div class="panel-heading"><h1 class="panel-title">Cluster</b></h1></div> + <ul class="list-group"> + {% for clustername, clusternodes in clusters.items() %} + <li class="list-group-item"><strong>{{vars[clusternodes|first]['wsrep_cluster_name']}}</strong> ({{statuses[clusternodes|first]['wsrep_cluster_status']}}, Größe={{statuses[clusternodes|first]['wsrep_cluster_size']}}, Konfiguration={{statuses[clusternodes|first]['wsrep_cluster_conf_id']}}) + <ul class="list-group"> + {% for host in clusternodes %} + <li class="list-group-item list-group-item-{{{'1': 'warning', '2': 'warning', '3': 'info', '4': 'success'}.get(statuses[host]['wsrep_local_state'], 'danger')}}"><a href="#{{host|tagid}}">{{vars[host]['wsrep_node_name']}}</a> ({{statuses[host]['wsrep_local_state_comment']}}, Letzte Änderung={{statuses[host]['wsrep_last_committed']}}, Recv.-Avg.={{statuses[host]['wsrep_local_recv_queue_avg']}}, Send-Avg.={{statuses[host]['wsrep_local_send_queue_avg']}})</li> + {% endfor %} + </ul> + </li> + {% endfor %} + </ul> + </table> +</div> +{% for host, status in statuses.items() %} +<div class="panel panel-default" id="{{host|tagid}}"> + <div class="panel-heading"><h1 class="panel-title">{{vars[host]['wsrep_node_name']}}</h1></div> + <div class="row" style="margin: 0px;"> + <div class="col-xs-6 table-responsive"> + <table class="table"> + <tr> + <th>Variable</th> + <th>Wert</th> + </tr> + {% for key, value in vars[host].items() %} + <tr> + <td>{{key}}</td> + <td>{{value}}</td> + </tr> + {% endfor %} + </table> + </div> + <div class="col-xs-6 table-responsive"> + <table class="table"> + <tr> + <th>Status-Variable</th> + <th>Wert</th> + </tr> + {% for key, value in status.items() %} + <tr> + <td>{{key}}</td> + <td>{{value}}</td> + </tr> + {% endfor %} + </table> + </div> + </div> +</div> +{% endfor %} +{% endblock %} diff --git a/templates/index.html b/templates/index.html index fbb2cbc3de210ccef6341a03ef077f6e66965da0..70b72bf0971752f5b2c93093936a067a29df8f05 100644 --- a/templates/index.html +++ b/templates/index.html @@ -111,7 +111,7 @@ <ul class="list-group" style="margin: 0px;"> {% for i in g.list %} <li class="list-group-item list-group-item-condensed"> - {{i.time|time}} <a href="{{url_for('course', id=i.course_id)}}">{{i.course.title}}</a>: <a href="{{url_for('course', id=i.course_id)}}#lecture-{{i.id}}">{{i.title}}</a> {{livelabel(i.live, False)}} + {{i.time|time}} <a href="{{url_for('course', handle=i.course.handle)}}">{{i.course.title}}</a>: <a href="{{url_for('course', handle=i.course.handle)}}#lecture-{{i.id}}">{{i.title}}</a> {{livelabel(i.live, False)}} </li> {% endfor %} </ul> diff --git a/templates/macros.html b/templates/macros.html index c0f27fed22d74b2e0e89451c79d6993a3e20b1c7..4acc84304ea6b645338d74701ee4a63fc127a470 100644 --- a/templates/macros.html +++ b/templates/macros.html @@ -97,6 +97,25 @@ $(function() { videojs("videoplayer").createModal('',{"uncloseable": true }).contentEl().innerHTML='<div class="hidden-print alert alert-danger" role="alert">{{ msg|safe }}</div>'; {% endfor %} + + //markers + $.ajax({method: "GET", url: "{{url_for('chapters',lectureid=lecture.id, json=1)}}", dataType: "json", + success: function (data) { + videojs("videoplayer").markers({ + markerStyle: { + 'width':'5px', + 'border-radius': '40%', + 'background-color': 'black' + }, + markerTip:{ + display: true, + text: function(marker) { + return marker.text; + } + }, + markers: data}); + }}); + }); </script> {% endmacro %} @@ -165,7 +184,7 @@ $('#embedcodebtn').popover( {% endmacro %} {% macro lecture_list_item(lecture,videos,global_permissions) %} -<li class="list-group-item" id="lecture-{{lecture.id}}"> +<li class="list-group-item{% if lecture.norecording %} text-muted{% endif %}" id="lecture-{{lecture.id}}"> <div class="row"> {% if ismod() or (videos|length > 0) %} <div style="background-image: url('{{ config.VIDEOPREFIX }}/thumbnail/l_{{lecture.id}}.jpg')" class="col-sm-2 col-xs-12 thumbnailimg"> @@ -190,6 +209,7 @@ $('#embedcodebtn').popover( <li>{{ moderator_editor(['lectures',lecture.id,'internal'], lecture.internal) }}</li> <li>Sichtbar: {{ moderator_checkbox(['lectures',lecture.id,'visible'], lecture.visible) }}</li> <li>Livestream geplant: {{ moderator_checkbox(['lectures',lecture.id,'live'], lecture.live) }}</li> + <li>Wird nicht aufgenommen: {{ moderator_checkbox(['lectures',lecture.id,'norecording'], lecture.norecording) }}</li> <li>Hörsaal: {{ moderator_editor(['lectures',lecture.id,'place'], lecture.place) }} </li> {% endif %} </ul> diff --git a/templates/sortlog.html b/templates/sortlog.html index 184af50c9510d00fcd17941ceb9c187ddb532d66..21d05c37147e80c5176e31520c95b368506d9e53 100644 --- a/templates/sortlog.html +++ b/templates/sortlog.html @@ -54,7 +54,7 @@ <td><a href="{{ config.VIDEOPREFIX }}/{{i.path}}">{{i.path}}</a></td> <td><a href="{{url_for('course', id=i.course_id)}}">{{i.course_id}}</a></td> <td><a href="{{url_for('course', id=i.course_id)}}#lecture-{{i.lecture_id}}">{{i.lecture_id}}</a></td> - <td>{{i.id}}</td> + <td>{{i.video_id}}</td> </tr> {% endfor %} </table> diff --git a/timetable.py b/timetable.py index 2ccb548870b4a16c87876ed5b1e0b04eec5511e1..ad2a54dfeeb82a585466efdb0b7c4bc0e4d53cb9 100644 --- a/timetable.py +++ b/timetable.py @@ -26,7 +26,7 @@ def timetable(): SELECT lectures.*, courses.short, "course" AS sep, courses.* FROM lectures JOIN courses ON (lectures.course_id = courses.id) - WHERE time < ? and time > ? + WHERE time < ? AND time > ? AND NOT norecording AND NOT external ORDER BY time ASC''', i['date']+timedelta(weeks=2), i['date']-timedelta(weeks=2)): # we can not use the where clause of sql to match against the time, because sqlite and mysql use a different syntax -.- # we still use it to only get the lectures for a 3 week periode