Делал я недавно доклад (на Percona Live). И как-то ночью (jet-lag все-таки) подумал, что неплохо бы как-нибудь видеть свои комментарии к слайдам — подсказки — и переключать слайды не подходя к ноутбуку. Чудо-устройства, чтоб и слайды переключало и комментарии показывало, у меня не было. Но был смартфон с Андроидом. Гугл-маркет разочаровал. Большинство подобных приложений работало только с PowerPoint, для LibreOffice я нашел только одно, платное, и выглядело оно как-то не так.
Но есть простое решение — браузер. Можно поднять http-сервер на ноутбуке, зайти с телефона, и по ссылкам переключать слайды. И будет работать не только с Андроидом, а и с яблочными, и с виндофонами. Да вообще смартфон не нужен, можно хоть через WAP заходить, или с другого ноутбука.
Получилось довольно просто. Около 70 строк на питоне, никаких внешних зависимостей (почти).
Комментарии читаются из файла (заморачиваться с вытаскиванием их из презентации я не стал). Поднимаем BaseHTTPServer на каком-то порту, я пробую все от 8000 до 9000 в случайном порядке, пока не получится. По инструкции со StackOverflow определяем свой IP. И вдобавок генерируем большое случайное число — оно будет частью URL-а, чтобы никто из аудитории не переключал нам слайды. Как мелкое удобство открываю браузер, чтобы показал QR-код для доступа к серверу. Если доступа в Интернет во время конференции не будет — не беда, придется ввести URL руками, и все. А для работы достаточно внутреннего вайфая.
По инструкции к BaseHTTPServer, чтобы обрабатывать GET запросы надо создать do_GET метод. Создаем. Первым делом надо проверить, чтоб было наше случайное число. Если есть, и URL заканчивается на next или prev, то имитируем нажатие пробела или, соответственно, backspace, и возвращаем страничку с комментариями для следующей или предыдущей страницы.
У меня стоит fvwm, так что с имитацией нажатий проблем не возникло. Для других менеджеров окон можно использовать отдельную программку или даже специальный модуль для питона, такие есть — я проверял.
Ну, собственно, и все. Только шрифт в настройках мобильного браузера нужно делать покрупнее. И комментарии писать покороче, иначе их долго читать — получаются неприятные паузы во время доклада. Да, и таймаут на отключение экрана убрать на смартфоне и в ноутбуке.
Испытание в боевых условиях прошло успешно.
Но есть простое решение — браузер. Можно поднять http-сервер на ноутбуке, зайти с телефона, и по ссылкам переключать слайды. И будет работать не только с Андроидом, а и с яблочными, и с виндофонами. Да вообще смартфон не нужен, можно хоть через WAP заходить, или с другого ноутбука.
Получилось довольно просто. Около 70 строк на питоне, никаких внешних зависимостей (почти).
Комментарии читаются из файла (заморачиваться с вытаскиванием их из презентации я не стал). Поднимаем BaseHTTPServer на каком-то порту, я пробую все от 8000 до 9000 в случайном порядке, пока не получится. По инструкции со StackOverflow определяем свой IP. И вдобавок генерируем большое случайное число — оно будет частью URL-а, чтобы никто из аудитории не переключал нам слайды. Как мелкое удобство открываю браузер, чтобы показал QR-код для доступа к серверу. Если доступа в Интернет во время конференции не будет — не беда, придется ввести URL руками, и все. А для работы достаточно внутреннего вайфая.
По инструкции к BaseHTTPServer, чтобы обрабатывать GET запросы надо создать do_GET метод. Создаем. Первым делом надо проверить, чтоб было наше случайное число. Если есть, и URL заканчивается на next или prev, то имитируем нажатие пробела или, соответственно, backspace, и возвращаем страничку с комментариями для следующей или предыдущей страницы.
У меня стоит fvwm, так что с имитацией нажатий проблем не возникло. Для других менеджеров окон можно использовать отдельную программку или даже специальный модуль для питона, такие есть — я проверял.
Ну, собственно, и все. Только шрифт в настройках мобильного браузера нужно делать покрупнее. И комментарии писать покороче, иначе их долго читать — получаются неприятные паузы во время доклада. Да, и таймаут на отключение экрана убрать на смартфоне и в ноутбуке.
Испытание в боевых условиях прошло успешно.
#!/usr/bin/python import sys import os import random import socket import BaseHTTPServer import re handler_class=BaseHTTPServer.BaseHTTPRequestHandler token="/" + str(random.randint(0, sys.maxint)) + "/" page = 1 validate = re.compile(token + "(prev|next)$") def send_key(key): os.system("FvwmCommand 'All (VCLSalFrame) FakeKeypress press {}'".format(key)) class Handler(BaseHTTPServer.BaseHTTPRequestHandler): def link(self,s,n): if n < 0 or n >= len(data): return " " return '<a href="{}{}">{}</a>'.format(token,s,s) def do_GET(self): cmd = validate.match(self.path) if cmd is None: self.send_response(404) self.send_header("Content-type", 'text/html') self.end_headers() self.wfile.write('<h1>OK</h1>') return global page if cmd.group(1) == 'prev' and page > 0: page = page - 1 send_key('BackSpace') elif cmd.group(1) == 'next' and page < len(data)-1: page = page + 1 send_key('space') self.send_response(200) self.send_header("Content-type", 'text/html; charset=utf-8') self.end_headers() self.wfile.write(""" {pagenum} - {next} <hr> {data} <hr> {prev} - {pagenum} """.format(prev = self.link('prev', page - 1), pagenum = page + 1, next = self.link('next', page + 1), data = data[page])) def read_file(s): with open(s) as f: return re.split('--- *\n?', f.read()) def ip(): s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) s.connect(('192.168.1.1', 0)) # fake, but nobody cares return s.getsockname()[0] def make_url(port): url="http://{}:{}{}prev".format(ip(), port, token) print url os.system("firefox 'http://chart.apis.google.com/chart?cht=qr&chs=200x200&chl={}' &".format(url)) def run(): for port in random.sample(xrange(8000, 9000),1000): try: httpd = BaseHTTPServer.HTTPServer(('',port), Handler) httpd.server_activate() make_url(port) httpd.serve_forever() except socket.error, e: if e.errno not in (98,): print e exit() except KeyboardInterrupt: print "\nExiting" exit() data = read_file('notes.txt') print "Got {} slides".format(len(data)) run()
