I wanted to set up a kiosk system that displayed some static and some constantly-updated information. I already had a kiosk that used Apple Keynote in full-screen presentation mode to show a slideshow, but all of the slides themselves were static.
In pondering the idea of keeping the slideshow fresh with more dynamic and up-to-date information, I wondered if I could start with an RSS or ATOM feed as the source. After all, the Web is full of dynamically-updated content, thanks to blogging and content management (CMS) software. These updates are often streamed out in RSS or ATOM format. If that software is already present and the workflow is (or can be) a part of a group’s Web presence, then repurposing those same dynamic updates is an efficient use of peoples’ time. That the software is often reasonable for use by trained staff or dedicated volunteers is another benefit.
With that in mind, I constructed a proof of concept that let me add new slides to a Keynote presentation.
For this task, I wanted to avoid controlling Keynote with AppleScript because I’d had an interest in trying my hand at Python Appscript for a while. I found Appscript to be simple to use when controlling Keynote — and it made as much if not more sense to me than AppleScript did.
Unfortunately, Appscript is now deprecated, so while my proof of concept (below) works for now, it may not continue to do so with inevitable march of software progress.
For this sample project, we need to have the Universal Feed Parser and Appscript installed.
Unfortunately, with the disappearance of Mark Pilgrim and his projects from the Web, the original source site for the Universal Feed Parser no longer exists. However, UFP is still open source software and people had the source and are interested in further developing it. When installing UFP, Python “easy_install” will fetch it from its home on Google Code.
At this point, with two key Python modules in strange circumstances, you are probably wondering what I was thinking. My only defense is that I did all of this some time ago and neither the AppScript nor UFP situations was apparent at the time.
Install Python Appscript. This requires that you are logged in with a “sudo”-ready account, or have used “su” to switch to such a user.
Password:
Searching for feedparser
Reading <a href="http://pypi.python.org/simple/feedparser/
Reading" title="http://pypi.python.org/simple/feedparser/
Reading">http://pypi.python.org/simple/feedparser/
Reading</a> <a href="https://code.google.com/p/feedparser/
Reading" title="https://code.google.com/p/feedparser/
Reading">https://code.google.com/p/feedparser/
Reading</a> <a href="http://code.google.com/p/feedparser/
Best" title="http://code.google.com/p/feedparser/
Best">http://code.google.com/p/feedparser/
Best</a> match: feedparser 5.1
Downloading <a href="https://feedparser.googlecode.com/files/feedparser-5.1.zip
Processing" title="https://feedparser.googlecode.com/files/feedparser-5.1.zip
Processing">https://feedparser.googlecode.com/files/feedparser-5.1.zip
Processing</a> feedparser-5.1.zip
Running feedparser-5.1/setup.py -q bdist_egg —dist-dir /tmp/easy_install-tOBghK/feedparser-5.1/egg-dist-tmp-s77JZ2
zip_safe flag not set; analyzing archive contents…
Adding feedparser 5.1 to easy-install.pth file
Installed /Library/Python/2.7/site-packages/feedparser-5.1-py2.7.egg
Processing dependencies for feedparser
Finished processing dependencies for feedparser
Install Python Appscript.
Password:
Searching for appscript
Reading <a href="http://pypi.python.org/simple/appscript/
Reading" title="http://pypi.python.org/simple/appscript/
Reading">http://pypi.python.org/simple/appscript/
Reading</a> <a href="http://appscript.sourceforge.net
Best" title="http://appscript.sourceforge.net
Best">http://appscript.sourceforge.net
Best</a> match: appscript 1.0.0
Downloading <a href="http://pypi.python.org/packages/source/a/appscript/appscript-1.0.0.tar.gz#md5=6619b637037ea0f391f45870c13ae38a
Processing" title="http://pypi.python.org/packages/source/a/appscript/appscript-1.0.0.tar.gz#md5=6619b637037ea0f391f45870c13ae38a
Processing">http://pypi.python.org/packages/source/a/appscript/appscript-1.0.0.tar.g…</a> appscript-1.0.0.tar.gz
Running appscript-1.0.0/setup.py -q bdist_egg —dist-dir /tmp/easy_install-RuEa2e/appscript-1.0.0/egg-dist-tmp-_nOlL1
zip_safe flag not set; analyzing archive contents…
Adding appscript 1.0.0 to easy-install.pth file
Installed /Library/Python/2.7/site-packages/appscript-1.0.0-py2.7-macosx-10.7-intel.egg
Processing dependencies for appscript
Finished processing dependencies for appscript
Launch Keynote and create a new presentation from the template selector. We will enter the “title” and “summary” RSS/ATOM feed data into the first Keynote presentation, so you need to have a presentation open first.
In either a Python script or the Python interpreter in Terminal, use the following sample code to download the feed data from the Apple Hot News RSS feed and enter it into the Keynote document.
I picked the Apple Hot News RSS feed because it is commonly seen on Macs that use the RSS screen saver and its defaults. Because of that, I also thought the feed was likely to be around long enough into the future to serve as a good example for others that came across this article. I am not endorsing this particular feed, don’t follow it, and don’t know its contents at any given time. You are free to substitute the RSS/ATOM feed URL of your own choosing — well, as long as the Universal Feed Parser can handle it — with the following example.
#All rights reserved.
#Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
# * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
# * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
# Import the feedparser module
import feedparser
# Define a feed to download and parse
d = feedparser.parse(‘http://images.apple.com/main/rss/hotnews/hotnews.rss’)
# Import appscript
from appscript import *
# Define a variable to refer to the Keynote application
kn = app(‘Keynote’)
# Process the items in the RSS feed represented by the variable
for entry in d.entries:
# For each item in the RSS feed, create a new slide
kn.slideshows[0].make(new=k.slide)
# Add the title of the RSS feed item as the slide title
kn.slideshows[0].current_slide.title.set(entry[‘title’])
# Add the summary of the RSS feed item as the slide body
kn.slideshows[0].current_slide.body.set(entry[‘summary’])
The Keynote document will be populated with new slides. Each new slide will feature the “title” of the RSS/ATOM feed item as the slide title, and the “summary” as the slide body. That’s it!
I’ll leave it as an exercise for another day to consider how to work a dynamic batch of RSS/ATOM-based slides into an existing static presentation.