Skip to content

Commit e5bbbca

Browse files
committed
rustdoc: Submit examples to play.rust-lang.org
This grows a new option inside of rustdoc to add the ability to submit examples to an external website. If the `--markdown-playground-url` command line option or crate doc attribute `html_playground_url` is present, then examples will have a button on hover to submit the code to the playground specified. This commit enables submission of example code to play.rust-lang.org. The code submitted is that which is tested by rustdoc, not necessarily the exact code shown in the example. Closes #14654
1 parent cc63d4c commit e5bbbca

File tree

29 files changed

+186
-43
lines changed

29 files changed

+186
-43
lines changed

mk/docs.mk

+3-1
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,9 @@ L10N_LANGS := ja
4343

4444
# The options are passed to the documentation generators.
4545
RUSTDOC_HTML_OPTS_NO_CSS = --markdown-before-content=doc/version_info.html \
46-
--markdown-in-header=doc/favicon.inc --markdown-after-content=doc/footer.inc
46+
--markdown-in-header=doc/favicon.inc \
47+
--markdown-after-content=doc/footer.inc \
48+
--markdown-playground-url='http://play.rust-lang.org/'
4749

4850
RUSTDOC_HTML_OPTS = $(RUSTDOC_HTML_OPTS_NO_CSS) --markdown-css rust.css
4951

src/doc/footer.inc

+2
Original file line numberDiff line numberDiff line change
@@ -5,3 +5,5 @@ or the <a href="http://opensource.org/licenses/MIT">MIT license</a>, at your opt
55
</p><p>
66
This file may not be copied, modified, or distributed except according to those terms.
77
</p></footer>
8+
<script type="text/javascript" src="jquery.js"></script>
9+
<script type="text/javascript" src="playpen.js"></script>

src/doc/rust.css

+13
Original file line numberDiff line numberDiff line change
@@ -313,6 +313,19 @@ table th {
313313
padding: 5px;
314314
}
315315

316+
/* Code snippets */
317+
318+
.rusttest { display: none; }
319+
pre.rust { position: relative; }
320+
pre.rust a { transform: scaleX(-1); }
321+
.test-arrow {
322+
display: inline-block;
323+
position: absolute;
324+
top: 0;
325+
right: 10px;
326+
font-size: 150%;
327+
}
328+
316329
@media (min-width: 1170px) {
317330
pre {
318331
font-size: 15px;

src/libcollections/lib.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,8 @@
1717
#![license = "MIT/ASL2"]
1818
#![doc(html_logo_url = "http://www.rust-lang.org/logos/rust-logo-128x128-blk-v2.png",
1919
html_favicon_url = "http://www.rust-lang.org/favicon.ico",
20-
html_root_url = "http://doc.rust-lang.org/")]
20+
html_root_url = "http://doc.rust-lang.org/",
21+
html_playground_url = "http://play.rust-lang.org/")]
2122

2223
#![feature(macro_rules, managed_boxes, default_type_params, phase, globs)]
2324
#![no_std]

src/libcore/lib.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,8 @@
5050
#![crate_type = "rlib"]
5151
#![doc(html_logo_url = "http://www.rust-lang.org/logos/rust-logo-128x128-blk-v2.png",
5252
html_favicon_url = "http://www.rust-lang.org/favicon.ico",
53-
html_root_url = "http://doc.rust-lang.org/")]
53+
html_root_url = "http://doc.rust-lang.org/",
54+
html_playground_url = "http://play.rust-lang.org/")]
5455

5556
#![no_std]
5657
#![feature(globs, macro_rules, managed_boxes, phase, simd)]

src/libgetopts/lib.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -84,7 +84,8 @@
8484
#![license = "MIT/ASL2"]
8585
#![doc(html_logo_url = "http://www.rust-lang.org/logos/rust-logo-128x128-blk-v2.png",
8686
html_favicon_url = "http://www.rust-lang.org/favicon.ico",
87-
html_root_url = "http://doc.rust-lang.org/")]
87+
html_root_url = "http://doc.rust-lang.org/",
88+
html_playground_url = "http://play.rust-lang.org/")]
8889
#![feature(globs, phase)]
8990
#![deny(missing_doc)]
9091
#![deny(deprecated_owned_vector)]

src/libglob/lib.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -29,8 +29,8 @@
2929
#![license = "MIT/ASL2"]
3030
#![doc(html_logo_url = "http://www.rust-lang.org/logos/rust-logo-128x128-blk-v2.png",
3131
html_favicon_url = "http://www.rust-lang.org/favicon.ico",
32-
html_root_url = "http://doc.rust-lang.org/")]
33-
32+
html_root_url = "http://doc.rust-lang.org/",
33+
html_playground_url = "http://play.rust-lang.org/")]
3434
#![deny(deprecated_owned_vector)]
3535

3636
use std::cell::Cell;

src/libgreen/lib.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -203,7 +203,8 @@
203203
#![crate_type = "dylib"]
204204
#![doc(html_logo_url = "http://www.rust-lang.org/logos/rust-logo-128x128-blk-v2.png",
205205
html_favicon_url = "http://www.rust-lang.org/favicon.ico",
206-
html_root_url = "http://doc.rust-lang.org/")]
206+
html_root_url = "http://doc.rust-lang.org/",
207+
html_playground_url = "http://play.rust-lang.org/")]
207208

208209
// NB this does *not* include globs, please keep it that way.
209210
#![feature(macro_rules, phase)]

src/liblog/lib.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -111,7 +111,8 @@ if logging is disabled, none of the components of the log will be executed.
111111
#![crate_type = "dylib"]
112112
#![doc(html_logo_url = "http://www.rust-lang.org/logos/rust-logo-128x128-blk-v2.png",
113113
html_favicon_url = "http://www.rust-lang.org/favicon.ico",
114-
html_root_url = "http://doc.rust-lang.org/")]
114+
html_root_url = "http://doc.rust-lang.org/",
115+
html_playground_url = "http://play.rust-lang.org/")]
115116

116117
#![feature(macro_rules)]
117118
#![deny(missing_doc, deprecated_owned_vector)]

src/libnum/lib.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,8 @@
5050
#![license = "MIT/ASL2"]
5151
#![doc(html_logo_url = "http://www.rust-lang.org/logos/rust-logo-128x128-blk-v2.png",
5252
html_favicon_url = "http://www.rust-lang.org/favicon.ico",
53-
html_root_url = "http://doc.rust-lang.org/")]
53+
html_root_url = "http://doc.rust-lang.org/",
54+
html_playground_url = "http://play.rust-lang.org/")]
5455

5556
#![deny(deprecated_owned_vector)]
5657

src/librand/lib.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,8 @@
2121
#![crate_type = "rlib"]
2222
#![doc(html_logo_url = "http://www.rust-lang.org/logos/rust-logo-128x128-blk.png",
2323
html_favicon_url = "http://www.rust-lang.org/favicon.ico",
24-
html_root_url = "http://doc.rust-lang.org/")]
24+
html_root_url = "http://doc.rust-lang.org/",
25+
html_playground_url = "http://play.rust-lang.org/")]
2526

2627
#![feature(macro_rules, phase, globs)]
2728
#![no_std]

src/libregex/lib.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -360,7 +360,8 @@
360360
#![license = "MIT/ASL2"]
361361
#![doc(html_logo_url = "http://www.rust-lang.org/logos/rust-logo-128x128-blk-v2.png",
362362
html_favicon_url = "http://www.rust-lang.org/favicon.ico",
363-
html_root_url = "http://doc.rust-lang.org/")]
363+
html_root_url = "http://doc.rust-lang.org/",
364+
html_playground_url = "http://play.rust-lang.org/")]
364365

365366
#![feature(macro_rules, phase)]
366367
#![deny(missing_doc, deprecated_owned_vector)]

src/librustdoc/html/highlight.rs

+10-3
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@ use html::escape::Escape;
2525
use t = syntax::parse::token;
2626

2727
/// Highlights some source code, returning the HTML output.
28-
pub fn highlight(src: &str, class: Option<&str>) -> String {
28+
pub fn highlight(src: &str, class: Option<&str>, id: Option<&str>) -> String {
2929
debug!("highlighting: ================\n{}\n==============", src);
3030
let sess = parse::new_parse_sess();
3131
let fm = parse::string_to_filemap(&sess,
@@ -36,6 +36,7 @@ pub fn highlight(src: &str, class: Option<&str>) -> String {
3636
doit(&sess,
3737
lexer::StringReader::new(&sess.span_diagnostic, fm),
3838
class,
39+
id,
3940
&mut out).unwrap();
4041
str::from_utf8_lossy(out.unwrap().as_slice()).to_string()
4142
}
@@ -47,11 +48,17 @@ pub fn highlight(src: &str, class: Option<&str>) -> String {
4748
/// it's used. All source code emission is done as slices from the source map,
4849
/// not from the tokens themselves, in order to stay true to the original
4950
/// source.
50-
fn doit(sess: &parse::ParseSess, mut lexer: lexer::StringReader, class: Option<&str>,
51+
fn doit(sess: &parse::ParseSess, mut lexer: lexer::StringReader,
52+
class: Option<&str>, id: Option<&str>,
5153
out: &mut Writer) -> io::IoResult<()> {
5254
use syntax::parse::lexer::Reader;
5355

54-
try!(write!(out, "<pre class='rust {}'>\n", class.unwrap_or("")));
56+
try!(write!(out, "<pre "));
57+
match id {
58+
Some(id) => try!(write!(out, "id='{}' ", id)),
59+
None => {}
60+
}
61+
try!(write!(out, "class='rust {}'>\n", class.unwrap_or("")));
5562
let mut last = BytePos(0);
5663
let mut is_attribute = false;
5764
let mut is_macro = false;

src/librustdoc/html/layout.rs

+11-2
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ pub struct Layout {
1616
pub logo: String,
1717
pub favicon: String,
1818
pub krate: String,
19+
pub playground_url: String,
1920
}
2021

2122
pub struct Page<'a> {
@@ -108,11 +109,13 @@ r##"<!DOCTYPE html>
108109
</div>
109110
110111
<script>
111-
var rootPath = "{root_path}";
112-
var currentCrate = "{krate}";
112+
window.rootPath = "{root_path}";
113+
window.currentCrate = "{krate}";
114+
window.playgroundUrl = "{play_url}";
113115
</script>
114116
<script src="{root_path}jquery.js"></script>
115117
<script src="{root_path}main.js"></script>
118+
{play_js}
116119
<script async src="{root_path}search-index.js"></script>
117120
</body>
118121
</html>"##,
@@ -124,6 +127,12 @@ r##"<!DOCTYPE html>
124127
favicon = nonestr(layout.favicon.as_slice()),
125128
sidebar = *sidebar,
126129
krate = layout.krate,
130+
play_url = layout.playground_url,
131+
play_js = if layout.playground_url.len() == 0 {
132+
"".to_string()
133+
} else {
134+
format!(r#"<script src="{}playpen.js"></script>"#, page.root_path)
135+
},
127136
)
128137
}
129138

src/librustdoc/html/markdown.rs

+29-6
Original file line numberDiff line numberDiff line change
@@ -27,14 +27,16 @@
2727
#![allow(non_camel_case_types)]
2828

2929
use libc;
30-
use std::cell::RefCell;
30+
use std::cell::{RefCell, Cell};
3131
use std::fmt;
3232
use std::slice;
3333
use std::str;
3434
use std::collections::HashMap;
3535

3636
use html::toc::TocBuilder;
3737
use html::highlight;
38+
use html::escape::Escape;
39+
use test;
3840

3941
/// A unit struct which has the `fmt::Show` trait implemented. When
4042
/// formatted, this struct will emit the HTML corresponding to the rendered
@@ -139,6 +141,9 @@ fn stripped_filtered_line<'a>(s: &'a str) -> Option<&'a str> {
139141
}
140142

141143
local_data_key!(used_header_map: RefCell<HashMap<String, uint>>)
144+
local_data_key!(test_idx: Cell<uint>)
145+
// None == render an example, but there's no crate name
146+
local_data_key!(pub playground_krate: Option<String>)
142147

143148
pub fn render(w: &mut fmt::Formatter, s: &str, print_toc: bool) -> fmt::Result {
144149
extern fn block(ob: *mut hoedown_buffer, text: *hoedown_buffer,
@@ -149,9 +154,9 @@ pub fn render(w: &mut fmt::Formatter, s: &str, print_toc: bool) -> fmt::Result {
149154
let opaque = opaque as *mut hoedown_html_renderer_state;
150155
let my_opaque: &MyOpaque = &*((*opaque).opaque as *MyOpaque);
151156
slice::raw::buf_as_slice((*text).data, (*text).size as uint, |text| {
152-
let text = str::from_utf8(text).unwrap();
157+
let origtext = str::from_utf8(text).unwrap();
153158
debug!("docblock: ==============\n{}\n=======", text);
154-
let mut lines = text.lines().filter(|l| {
159+
let mut lines = origtext.lines().filter(|l| {
155160
stripped_filtered_line(*l).is_none()
156161
});
157162
let text = lines.collect::<Vec<&str>>().connect("\n");
@@ -180,9 +185,26 @@ pub fn render(w: &mut fmt::Formatter, s: &str, print_toc: bool) -> fmt::Result {
180185
};
181186

182187
if !rendered {
183-
let output = highlight::highlight(text.as_slice(),
184-
None).as_slice()
185-
.to_c_str();
188+
let mut s = String::new();
189+
let id = playground_krate.get().map(|krate| {
190+
let idx = test_idx.get().unwrap();
191+
let i = idx.get();
192+
idx.set(i + 1);
193+
194+
let test = origtext.lines().map(|l| {
195+
stripped_filtered_line(l).unwrap_or(l)
196+
}).collect::<Vec<&str>>().connect("\n");
197+
let krate = krate.as_ref().map(|s| s.as_slice());
198+
let test = test::maketest(test.as_slice(), krate, false);
199+
s.push_str(format!("<span id='rust-example-raw-{}' \
200+
class='rusttest'>{}</span>",
201+
i, Escape(test.as_slice())).as_slice());
202+
format!("rust-example-rendered-{}", i)
203+
});
204+
let id = id.as_ref().map(|a| a.as_slice());
205+
s.push_str(highlight::highlight(text.as_slice(), None, id)
206+
.as_slice());
207+
let output = s.to_c_str();
186208
output.with_ref(|r| {
187209
hoedown_buffer_puts(ob, r)
188210
})
@@ -377,6 +399,7 @@ fn parse_lang_string(string: &str) -> (bool,bool,bool,bool) {
377399
/// previous state (if any).
378400
pub fn reset_headers() {
379401
used_header_map.replace(Some(RefCell::new(HashMap::new())));
402+
test_idx.replace(Some(Cell::new(0)));
380403
}
381404

382405
impl<'a> fmt::Show for Markdown<'a> {

src/librustdoc/html/render.rs

+13-2
Original file line numberDiff line numberDiff line change
@@ -230,6 +230,7 @@ pub fn run(mut krate: clean::Crate, dst: Path) -> io::IoResult<()> {
230230
logo: "".to_string(),
231231
favicon: "".to_string(),
232232
krate: krate.name.clone(),
233+
playground_url: "".to_string(),
233234
},
234235
include_sources: true,
235236
render_redirect_pages: false,
@@ -250,6 +251,14 @@ pub fn run(mut krate: clean::Crate, dst: Path) -> io::IoResult<()> {
250251
if "html_logo_url" == x.as_slice() => {
251252
cx.layout.logo = s.to_string();
252253
}
254+
clean::NameValue(ref x, ref s)
255+
if "html_playground_url" == x.as_slice() => {
256+
cx.layout.playground_url = s.to_string();
257+
let name = krate.name.clone();
258+
if markdown::playground_krate.get().is_none() {
259+
markdown::playground_krate.replace(Some(Some(name)));
260+
}
261+
}
253262
clean::Word(ref x)
254263
if "html_no_source" == x.as_slice() => {
255264
cx.include_sources = false;
@@ -450,6 +459,7 @@ fn write_shared(cx: &Context,
450459
try!(write(cx.dst.join("jquery.js"),
451460
include_bin!("static/jquery-2.1.0.min.js")));
452461
try!(write(cx.dst.join("main.js"), include_bin!("static/main.js")));
462+
try!(write(cx.dst.join("playpen.js"), include_bin!("static/playpen.js")));
453463
try!(write(cx.dst.join("main.css"), include_bin!("static/main.css")));
454464
try!(write(cx.dst.join("normalize.css"),
455465
include_bin!("static/normalize.css")));
@@ -2055,14 +2065,15 @@ impl<'a> fmt::Show for Source<'a> {
20552065
try!(write!(fmt, "<span id='{0:u}'>{0:1$u}</span>\n", i, cols));
20562066
}
20572067
try!(write!(fmt, "</pre>"));
2058-
try!(write!(fmt, "{}", highlight::highlight(s.as_slice(), None)));
2068+
try!(write!(fmt, "{}", highlight::highlight(s.as_slice(), None, None)));
20592069
Ok(())
20602070
}
20612071
}
20622072

20632073
fn item_macro(w: &mut fmt::Formatter, it: &clean::Item,
20642074
t: &clean::Macro) -> fmt::Result {
2065-
try!(w.write(highlight::highlight(t.source.as_slice(), Some("macro")).as_bytes()));
2075+
try!(w.write(highlight::highlight(t.source.as_slice(), Some("macro"),
2076+
None).as_bytes()));
20662077
document(w, it)
20672078
}
20682079

src/librustdoc/html/static/main.css

+12-1
Original file line numberDiff line numberDiff line change
@@ -156,7 +156,7 @@ nav.sub {
156156
padding: 0 10px;
157157
margin-bottom: 10px;
158158
}
159-
.block h2 {
159+
.block h2 {
160160
margin-top: 0;
161161
text-align: center;
162162
}
@@ -396,6 +396,17 @@ pre.rust .doccomment { color: #4D4D4C; }
396396
pre.rust .macro, pre.rust .macro-nonterminal { color: #3E999F; }
397397
pre.rust .lifetime { color: #B76514; }
398398

399+
.rusttest { display: none; }
400+
pre.rust { position: relative; }
401+
pre.rust a { transform: scaleX(-1); }
402+
.test-arrow {
403+
display: inline-block;
404+
position: absolute;
405+
top: 0;
406+
right: 10px;
407+
font-size: 150%;
408+
}
409+
399410
.methods .section-header {
400411
/* Override parent class attributes. */
401412
border-bottom: none !important;

src/librustdoc/html/static/main.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -678,7 +678,7 @@
678678
window.register_implementors(window.pending_implementors);
679679
}
680680

681-
// See documentaiton in html/render.rs for what this is doing.
681+
// See documentation in html/render.rs for what this is doing.
682682
var query = getQueryStringParams();
683683
if (query['gotosrc']) {
684684
window.location = $('#src-' + query['gotosrc']).attr('href');

src/librustdoc/html/static/playpen.js

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
// Copyright 2014 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
/*jslint browser: true, es5: true */
12+
/*globals $: true, rootPath: true */
13+
14+
(function() {
15+
if (window.playgroundUrl) {
16+
$('pre.rust').hover(function() {
17+
var id = '#' + $(this).attr('id').replace('rendered', 'raw');
18+
var a = $('<a>').text('⇱').attr('class', 'test-arrow');
19+
var code = $(id).text();
20+
a.attr('href', window.playgroundUrl + '?code=' +
21+
encodeURIComponent(code));
22+
a.attr('target', '_blank');
23+
$(this).append(a);
24+
}, function() {
25+
$(this).find('a').remove();
26+
});
27+
}
28+
}());
29+

0 commit comments

Comments
 (0)