##// END OF EJS Templates
Merge pull request #5957 from damianavila/fragments...
Min RK -
r17105:e181fce2 merge
parent child Browse files
Show More
@@ -1,112 +1,112 b''
1 1 """PostProcessor for serving reveal.js HTML slideshows."""
2 2 from __future__ import print_function
3 3 #-----------------------------------------------------------------------------
4 4 #Copyright (c) 2013, the IPython Development Team.
5 5 #
6 6 #Distributed under the terms of the Modified BSD License.
7 7 #
8 8 #The full license is in the file COPYING.txt, distributed with this software.
9 9 #-----------------------------------------------------------------------------
10 10
11 11 #-----------------------------------------------------------------------------
12 12 # Imports
13 13 #-----------------------------------------------------------------------------
14 14
15 15 import os
16 16 import webbrowser
17 17
18 18 from tornado import web, ioloop, httpserver
19 19 from tornado.httpclient import AsyncHTTPClient
20 20
21 21 from IPython.utils.traitlets import Bool, Unicode, Int
22 22
23 23 from .base import PostProcessorBase
24 24
25 25 #-----------------------------------------------------------------------------
26 26 # Classes
27 27 #-----------------------------------------------------------------------------
28 28
29 29 class ProxyHandler(web.RequestHandler):
30 30 """handler the proxies requests from a local prefix to a CDN"""
31 31 @web.asynchronous
32 32 def get(self, prefix, url):
33 33 """proxy a request to a CDN"""
34 34 proxy_url = "/".join([self.settings['cdn'], url])
35 35 client = self.settings['client']
36 36 client.fetch(proxy_url, callback=self.finish_get)
37 37
38 38 def finish_get(self, response):
39 39 """finish the request"""
40 40 # copy potentially relevant headers
41 41 for header in ["Content-Type", "Cache-Control", "Date", "Last-Modified", "Expires"]:
42 42 if header in response.headers:
43 43 self.set_header(header, response.headers[header])
44 44 self.finish(response.body)
45 45
46 46 class ServePostProcessor(PostProcessorBase):
47 47 """Post processor designed to serve files
48 48
49 49 Proxies reveal.js requests to a CDN if no local reveal.js is present
50 50 """
51 51
52 52
53 53 open_in_browser = Bool(True, config=True,
54 54 help="""Should the browser be opened automatically?"""
55 55 )
56 reveal_cdn = Unicode("https://cdn.jsdelivr.net/reveal.js/2.5.0", config=True,
56 reveal_cdn = Unicode("https://cdn.jsdelivr.net/reveal.js/2.6.2", config=True,
57 57 help="""URL for reveal.js CDN."""
58 58 )
59 59 reveal_prefix = Unicode("reveal.js", config=True, help="URL prefix for reveal.js")
60 60 ip = Unicode("127.0.0.1", config=True, help="The IP address to listen on.")
61 61 port = Int(8000, config=True, help="port for the server to listen on.")
62 62
63 63 def postprocess(self, input):
64 64 """Serve the build directory with a webserver."""
65 65 dirname, filename = os.path.split(input)
66 66 handlers = [
67 67 (r"/(.+)", web.StaticFileHandler, {'path' : dirname}),
68 68 (r"/", web.RedirectHandler, {"url": "/%s" % filename})
69 69 ]
70 70
71 71 if ('://' in self.reveal_prefix or self.reveal_prefix.startswith("//")):
72 72 # reveal specifically from CDN, nothing to do
73 73 pass
74 74 elif os.path.isdir(os.path.join(dirname, self.reveal_prefix)):
75 75 # reveal prefix exists
76 76 self.log.info("Serving local %s", self.reveal_prefix)
77 77 else:
78 78 self.log.info("Redirecting %s requests to %s", self.reveal_prefix, self.reveal_cdn)
79 79 handlers.insert(0, (r"/(%s)/(.*)" % self.reveal_prefix, ProxyHandler))
80 80
81 81 app = web.Application(handlers,
82 82 cdn=self.reveal_cdn,
83 83 client=AsyncHTTPClient(),
84 84 )
85 85 # hook up tornado logging to our logger
86 86 try:
87 87 from tornado import log
88 88 log.app_log = self.log
89 89 except ImportError:
90 90 # old tornado (<= 3), ignore
91 91 pass
92 92
93 93 http_server = httpserver.HTTPServer(app)
94 94 http_server.listen(self.port, address=self.ip)
95 95 url = "http://%s:%i/%s" % (self.ip, self.port, filename)
96 96 print("Serving your slides at %s" % url)
97 97 print("Use Control-C to stop this server")
98 98 if self.open_in_browser:
99 99 webbrowser.open(url, new=2)
100 100 try:
101 101 ioloop.IOLoop.instance().start()
102 102 except KeyboardInterrupt:
103 103 print("\nInterrupted")
104 104
105 105 def main(path):
106 106 """allow running this module to serve the slides"""
107 107 server = ServePostProcessor()
108 108 server(path)
109 109
110 110 if __name__ == '__main__':
111 111 import sys
112 112 main(sys.argv[1])
@@ -1,63 +1,75 b''
1 1 """Module that pre-processes the notebook for export via Reveal.
2 2 """
3 3 #-----------------------------------------------------------------------------
4 4 # Copyright (c) 2013, the IPython Development Team.
5 5 #
6 6 # Distributed under the terms of the Modified BSD License.
7 7 #
8 8 # The full license is in the file COPYING.txt, distributed with this software.
9 9 #-----------------------------------------------------------------------------
10 10
11 11 #-----------------------------------------------------------------------------
12 12 # Imports
13 13 #-----------------------------------------------------------------------------
14 14
15 15 from .base import Preprocessor
16 16 from IPython.utils.traitlets import Unicode
17 17
18 18 #-----------------------------------------------------------------------------
19 19 # Classes and functions
20 20 #-----------------------------------------------------------------------------
21 21
22 22 class RevealHelpPreprocessor(Preprocessor):
23 23
24 24 url_prefix = Unicode('reveal.js', config=True,
25 25 help="""The URL prefix for reveal.js.
26 26 This can be a a relative URL for a local copy of reveal.js,
27 27 or point to a CDN.
28 28
29 29 For speaker notes to work, a local reveal.js prefix must be used.
30 30 """
31 31 )
32 32
33 33 def preprocess(self, nb, resources):
34 34 """
35 35 Called once to 'preprocess' contents of the notebook.
36 36
37 37 Parameters
38 38 ----------
39 39 nb : NotebookNode
40 40 Notebook being converted
41 41 resources : dictionary
42 42 Additional resources used in the conversion process. Allows
43 43 preprocessors to pass variables into the Jinja engine.
44 44 """
45 45
46 46 for worksheet in nb.worksheets:
47 47 for index, cell in enumerate(worksheet.cells):
48 48
49 49 #Make sure the cell has slideshow metadata.
50 50 cell.metadata.slide_type = cell.get('metadata', {}).get('slideshow', {}).get('slide_type', '-')
51 51
52 #Get the slide type. If type is start of subslide or slide,
52 #Get the slide type. If type is start of subslide or slide,
53 53 #end the last subslide/slide.
54 54 if cell.metadata.slide_type in ['slide']:
55 55 worksheet.cells[index - 1].metadata.slide_helper = 'slide_end'
56 56 if cell.metadata.slide_type in ['subslide']:
57 57 worksheet.cells[index - 1].metadata.slide_helper = 'subslide_end'
58
58 #Prevent the rendering of "do nothing" cells before fragments
59 #Group fragments passing frag_number to the data-fragment-index
60 if cell.metadata.slide_type in ['fragment']:
61 worksheet.cells[index].metadata.frag_number = index
62 i = 1
63 while i < len(worksheet.cells) - index:
64 worksheet.cells[index + i].metadata.frag_helper = 'fragment_end'
65 worksheet.cells[index + i].metadata.frag_number = index
66 i += 1
67 #Restart the slide_helper when the cell status is changed
68 #to other types.
69 if cell.metadata.slide_type in ['-', 'skip', 'notes', 'fragment']:
70 worksheet.cells[index - 1].metadata.slide_helper = '-'
59 71
60 72 if not isinstance(resources['reveal'], dict):
61 73 resources['reveal'] = {}
62 74 resources['reveal']['url_prefix'] = self.url_prefix
63 75 return nb, resources
@@ -1,93 +1,94 b''
1 1 """
2 2 Module with tests for the revealhelp preprocessor
3 3 """
4 4
5 5 #-----------------------------------------------------------------------------
6 6 # Copyright (c) 2013, the IPython Development Team.
7 7 #
8 8 # Distributed under the terms of the Modified BSD License.
9 9 #
10 10 # The full license is in the file COPYING.txt, distributed with this software.
11 11 #-----------------------------------------------------------------------------
12 12
13 13 #-----------------------------------------------------------------------------
14 14 # Imports
15 15 #-----------------------------------------------------------------------------
16 16
17 17 from IPython.nbformat import current as nbformat
18 18
19 19 from .base import PreprocessorTestsBase
20 20 from ..revealhelp import RevealHelpPreprocessor
21 21
22 22
23 23 #-----------------------------------------------------------------------------
24 24 # Class
25 25 #-----------------------------------------------------------------------------
26 26
27 27 class Testrevealhelp(PreprocessorTestsBase):
28 28 """Contains test functions for revealhelp.py"""
29 29
30 30 def build_notebook(self):
31 """Build a reveal slides notebook in memory for use with tests.
31 """Build a reveal slides notebook in memory for use with tests.
32 32 Overrides base in PreprocessorTestsBase"""
33 33
34 34 outputs = [nbformat.new_output(output_type="stream", stream="stdout", output_text="a")]
35
35
36 36 slide_metadata = {'slideshow' : {'slide_type': 'slide'}}
37 37 subslide_metadata = {'slideshow' : {'slide_type': 'subslide'}}
38 38
39 39 cells=[nbformat.new_code_cell(input="", prompt_number=1, outputs=outputs),
40 40 nbformat.new_text_cell('markdown', source="", metadata=slide_metadata),
41 41 nbformat.new_code_cell(input="", prompt_number=2, outputs=outputs),
42 42 nbformat.new_text_cell('markdown', source="", metadata=slide_metadata),
43 43 nbformat.new_text_cell('markdown', source="", metadata=subslide_metadata)]
44 44 worksheets = [nbformat.new_worksheet(name="worksheet1", cells=cells)]
45 45
46 46 return nbformat.new_notebook(name="notebook1", worksheets=worksheets)
47 47
48 48
49 49 def build_preprocessor(self):
50 50 """Make an instance of a preprocessor"""
51 51 preprocessor = RevealHelpPreprocessor()
52 52 preprocessor.enabled = True
53 53 return preprocessor
54 54
55 55
56 56 def test_constructor(self):
57 57 """Can a RevealHelpPreprocessor be constructed?"""
58 58 self.build_preprocessor()
59
59
60 60
61 61 def test_reveal_attribute(self):
62 62 """Make sure the reveal url_prefix resources is set"""
63 63 nb = self.build_notebook()
64 64 res = self.build_resources()
65 65 preprocessor = self.build_preprocessor()
66 66 nb, res = preprocessor(nb, res)
67 67 assert 'reveal' in res
68 68 assert 'url_prefix' in res['reveal']
69 69
70 70
71 71 def test_reveal_output(self):
72 72 """Make sure that the reveal preprocessor """
73 73 nb = self.build_notebook()
74 74 res = self.build_resources()
75 75 preprocessor = self.build_preprocessor()
76 76 nb, res = preprocessor(nb, res)
77 77 cells = nb.worksheets[0].cells
78 78
79 79 # Make sure correct metadata tags are available on every cell.
80 80 for cell in cells:
81 81 assert 'slide_type' in cell.metadata
82 82
83 83 # Make sure slide end is only applied to the cells preceeding slide
84 84 # cells.
85 assert 'slide_helper' not in cells[1].metadata
85 assert 'slide_helper' in cells[1].metadata
86 self.assertEqual(cells[1].metadata['slide_helper'], '-')
86 87
87 88 # Verify 'slide-end'
88 89 assert 'slide_helper' in cells[0].metadata
89 90 self.assertEqual(cells[0].metadata['slide_helper'], 'slide_end')
90 91 assert 'slide_helper' in cells[2].metadata
91 92 self.assertEqual(cells[2].metadata['slide_helper'], 'slide_end')
92 93 assert 'slide_helper' in cells[3].metadata
93 94 self.assertEqual(cells[3].metadata['slide_helper'], 'subslide_end')
@@ -1,197 +1,203 b''
1 1 {%- extends 'basic.tpl' -%}
2 2 {% from 'mathjax.tpl' import mathjax %}
3 3
4 4 {%- block any_cell scoped -%}
5 5 {%- if cell.metadata.slide_type in ['slide'] -%}
6 6 <section>
7 7 <section>
8 8 {{ super() }}
9 9 {%- elif cell.metadata.slide_type in ['subslide'] -%}
10 10 <section>
11 11 {{ super() }}
12 12 {%- elif cell.metadata.slide_type in ['-'] -%}
13 {{ super() }}
13 {%- if cell.metadata.frag_helper in ['fragment_end'] -%}
14 <div class="fragment" data-fragment-index="{{ cell.metadata.frag_number }}">
15 {{ super() }}
16 </div>
17 {%- else -%}
18 {{ super() }}
19 {%- endif -%}
14 20 {%- elif cell.metadata.slide_type in ['skip'] -%}
15 21 <div style=display:none>
16 22 {{ super() }}
17 23 </div>
18 24 {%- elif cell.metadata.slide_type in ['notes'] -%}
19 25 <aside class="notes">
20 26 {{ super() }}
21 27 </aside>
22 28 {%- elif cell.metadata.slide_type in ['fragment'] -%}
23 <div class="fragment">
29 <div class="fragment" data-fragment-index="{{ cell.metadata.frag_number }}">
24 30 {{ super() }}
25 31 </div>
26 32 {%- endif -%}
27 33 {%- if cell.metadata.slide_helper in ['subslide_end'] -%}
28 34 </section>
29 35 {%- elif cell.metadata.slide_helper in ['slide_end'] -%}
30 36 </section>
31 37 </section>
32 38 {%- endif -%}
33 39 {%- endblock any_cell -%}
34 40
35 41 {% block header %}
36 42 <!DOCTYPE html>
37 43 <html>
38 44 <head>
39 45
40 46 <meta charset="utf-8" />
41 47 <meta http-equiv="X-UA-Compatible" content="chrome=1" />
42 48
43 49 <meta name="apple-mobile-web-app-capable" content="yes" />
44 50 <meta name="apple-mobile-web-app-status-bar-style" content="black-translucent" />
45 51
46 52 <title>{{resources['metadata']['name']}} slides</title>
47 53
48 54 <script src="https://cdnjs.cloudflare.com/ajax/libs/require.js/2.1.10/require.min.js"></script>
49 55 <script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/2.0.3/jquery.min.js"></script>
50 56
51 57 <!-- General and theme style sheets -->
52 58 <link rel="stylesheet" href="{{resources.reveal.url_prefix}}/css/reveal.css">
53 59 <link rel="stylesheet" href="{{resources.reveal.url_prefix}}/css/theme/simple.css" id="theme">
54 60
55 61 <!-- For syntax highlighting -->
56 62 <link rel="stylesheet" href="{{resources.reveal.url_prefix}}/lib/css/zenburn.css">
57 63
58 64 <!-- If the query includes 'print-pdf', include the PDF print sheet -->
59 65 <script>
60 66 if( window.location.search.match( /print-pdf/gi ) ) {
61 67 var link = document.createElement( 'link' );
62 68 link.rel = 'stylesheet';
63 69 link.type = 'text/css';
64 70 link.href = '{{resources.reveal.url_prefix}}/css/print/pdf.css';
65 71 document.getElementsByTagName( 'head' )[0].appendChild( link );
66 72 }
67 73
68 74 </script>
69 75
70 76 <!--[if lt IE 9]>
71 77 <script src="{{resources.reveal.url_prefix}}/lib/js/html5shiv.js"></script>
72 78 <![endif]-->
73 79
74 80 <!-- Get Font-awesome from cdn -->
75 81 <link rel="stylesheet" href="//netdna.bootstrapcdn.com/font-awesome/3.2.1/css/font-awesome.css">
76 82
77 83 {% for css in resources.inlining.css -%}
78 84 <style type="text/css">
79 85 {{ css }}
80 86 </style>
81 87 {% endfor %}
82 88
83 89 <style type="text/css">
84 90 /* Overrides of notebook CSS for static HTML export */
85 91 html {
86 92 overflow-y: auto;
87 93 }
88 94 .reveal {
89 95 font-size: 160%;
90 96 }
91 97 .reveal pre {
92 98 width: inherit;
93 99 padding: 0.4em;
94 100 margin: 0px;
95 101 font-family: monospace, sans-serif;
96 102 font-size: 80%;
97 103 box-shadow: 0px 0px 0px rgba(0, 0, 0, 0);
98 104 }
99 105 .reveal section img {
100 106 border: 0px solid black;
101 107 box-shadow: 0 0 10px rgba(0, 0, 0, 0);
102 108 }
103 109 .reveal i {
104 110 font-style: normal;
105 111 font-family: FontAwesome;
106 112 font-size: 2em;
107 113 }
108 114 .reveal .slides {
109 115 text-align: left;
110 116 }
111 117 .reveal.fade {
112 118 opacity: 1;
113 119 }
114 120 .reveal .progress {
115 121 position: static;
116 122 }
117 123 div.input_area {
118 124 padding: 0.06em;
119 125 }
120 126 div.code_cell {
121 127 background-color: transparent;
122 128 }
123 129 div.prompt {
124 130 width: 11ex;
125 131 padding: 0.4em;
126 132 margin: 0px;
127 133 font-family: monospace, sans-serif;
128 134 font-size: 80%;
129 135 text-align: right;
130 136 }
131 137 div.output_area pre {
132 138 font-family: monospace, sans-serif;
133 139 font-size: 80%;
134 140 }
135 141 div.output_prompt {
136 142 /* 5px right shift to account for margin in parent container */
137 143 margin: 5px 5px 0 0;
138 144 }
139 145 .rendered_html p {
140 146 text-align: inherit;
141 147 }
142 148 </style>
143 149
144 150 <!-- Custom stylesheet, it must be in the same directory as the html file -->
145 151 <link rel="stylesheet" href="custom.css">
146 152
147 153 </head>
148 154 {% endblock header%}
149 155
150 156
151 157 {% block body %}
152 158 <body>
153 159 <div class="reveal">
154 160 <div class="slides">
155 161 {{ super() }}
156 162 </div>
157 163 </div>
158 164
159 165 <script src="{{resources.reveal.url_prefix}}/lib/js/head.min.js"></script>
160 166
161 167 <script src="{{resources.reveal.url_prefix}}/js/reveal.js"></script>
162 168
163 169 <script>
164 170
165 171 // Full list of configuration options available here: https://github.com/hakimel/reveal.js#configuration
166 172 Reveal.initialize({
167 173 controls: true,
168 174 progress: true,
169 175 history: true,
170 176
171 177 theme: Reveal.getQueryHash().theme, // available themes are in /css/theme
172 178 transition: Reveal.getQueryHash().transition || 'linear', // default/cube/page/concave/zoom/linear/none
173 179
174 180 // Optional libraries used to extend on reveal.js
175 181 dependencies: [
176 182 { src: "{{resources.reveal.url_prefix}}/lib/js/classList.js", condition: function() { return !document.body.classList; } },
177 183 { src: "{{resources.reveal.url_prefix}}/plugin/notes/notes.js", async: true, condition: function() { return !!document.body.classList; } }
178 184 ]
179 185 });
180 186 </script>
181 187
182 188 <!-- Loading mathjax macro -->
183 189 {{ mathjax() }}
184 190
185 191 <script>
186 192 Reveal.addEventListener( 'slidechanged', function( event ) {
187 193 window.scrollTo(0,0);
188 194 MathJax.Hub.Rerender(event.currentSlide);
189 195 });
190 196 </script>
191 197
192 198 </body>
193 199 {% endblock body %}
194 200
195 201 {% block footer %}
196 202 </html>
197 203 {% endblock footer %} No newline at end of file
General Comments 0
You need to be logged in to leave comments. Login now