1
Fork 0

rustdoc: format where clauses like rust-lang-nursery/fmt-rfcs#38

This commit is contained in:
QuietMisdreavus 2017-03-31 18:04:42 -05:00
parent 8206d0c54e
commit 80bff6b596
3 changed files with 55 additions and 81 deletions

View file

@ -49,14 +49,19 @@ pub struct MutableSpace(pub clean::Mutability);
/// Similar to VisSpace, but used for mutability /// Similar to VisSpace, but used for mutability
#[derive(Copy, Clone)] #[derive(Copy, Clone)]
pub struct RawMutableSpace(pub clean::Mutability); pub struct RawMutableSpace(pub clean::Mutability);
/// Wrapper struct for emitting a where clause from Generics.
pub struct WhereClause<'a>(pub &'a clean::Generics, pub usize);
/// Wrapper struct for emitting type parameter bounds. /// Wrapper struct for emitting type parameter bounds.
pub struct TyParamBounds<'a>(pub &'a [clean::TyParamBound]); pub struct TyParamBounds<'a>(pub &'a [clean::TyParamBound]);
/// Wrapper struct for emitting a comma-separated list of items /// Wrapper struct for emitting a comma-separated list of items
pub struct CommaSep<'a, T: 'a>(pub &'a [T]); pub struct CommaSep<'a, T: 'a>(pub &'a [T]);
pub struct AbiSpace(pub Abi); pub struct AbiSpace(pub Abi);
/// Wrapper struct for emitting a where clause from Generics.
pub struct WhereClause<'a>{
pub gens: &'a clean::Generics,
pub indent: usize,
pub end_newline: bool,
}
pub struct HRef<'a> { pub struct HRef<'a> {
pub did: DefId, pub did: DefId,
pub text: &'a str, pub text: &'a str,
@ -167,24 +172,27 @@ impl fmt::Display for clean::Generics {
impl<'a> fmt::Display for WhereClause<'a> { impl<'a> fmt::Display for WhereClause<'a> {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
let &WhereClause(gens, pad) = self; let &WhereClause{ gens, indent, end_newline } = self;
if gens.where_predicates.is_empty() { if gens.where_predicates.is_empty() {
return Ok(()); return Ok(());
} }
let mut clause = String::new(); let mut clause = String::new();
if f.alternate() { if f.alternate() {
clause.push_str(" where "); clause.push_str(" where");
} else { } else {
clause.push_str(" <span class=\"where fmt-newline\">where "); if end_newline {
clause.push_str("<span class=\"where fmt-newline\">where");
} else {
clause.push_str("<span class=\"where\">where");
}
} }
for (i, pred) in gens.where_predicates.iter().enumerate() { for (i, pred) in gens.where_predicates.iter().enumerate() {
if i > 0 {
if f.alternate() { if f.alternate() {
clause.push_str(", "); clause.push(' ');
} else { } else {
clause.push_str(",<br>"); clause.push_str("<br>");
}
} }
match pred { match pred {
&clean::WherePredicate::BoundPredicate { ref ty, ref bounds } => { &clean::WherePredicate::BoundPredicate { ref ty, ref bounds } => {
let bounds = bounds; let bounds = bounds;
@ -213,21 +221,18 @@ impl<'a> fmt::Display for WhereClause<'a> {
} }
} }
} }
if i < gens.where_predicates.len() - 1 || end_newline {
clause.push(',');
}
} }
if !f.alternate() { if !f.alternate() {
clause.push_str("</span>"); clause.push_str("</span>");
let plain = format!("{:#}", self); let padding = repeat("&nbsp;").take(indent + 4).collect::<String>();
if plain.len() + pad > 80 {
// break it onto its own line regardless, but make sure method impls and trait
// blocks keep their fixed padding (2 and 9, respectively)
let padding = if pad > 10 {
repeat("&nbsp;").take(8).collect::<String>()
} else {
repeat("&nbsp;").take(pad + 6).collect::<String>()
};
clause = clause.replace("<br>", &format!("<br>{}", padding)); clause = clause.replace("<br>", &format!("<br>{}", padding));
} else { clause.insert_str(0, &repeat("&nbsp;").take(indent).collect::<String>());
clause = clause.replace("<br>", " "); if !end_newline {
clause.insert_str(0, "<br>");
} }
} }
write!(f, "{}", clause) write!(f, "{}", clause)
@ -838,43 +843,35 @@ fn fmt_impl(i: &clean::Impl,
f: &mut fmt::Formatter, f: &mut fmt::Formatter,
link_trait: bool, link_trait: bool,
use_absolute: bool) -> fmt::Result { use_absolute: bool) -> fmt::Result {
let mut plain = String::new();
if f.alternate() { if f.alternate() {
write!(f, "impl{:#} ", i.generics)?; write!(f, "impl{:#} ", i.generics)?;
} else { } else {
write!(f, "impl{} ", i.generics)?; write!(f, "impl{} ", i.generics)?;
} }
plain.push_str(&format!("impl{:#} ", i.generics));
if let Some(ref ty) = i.trait_ { if let Some(ref ty) = i.trait_ {
if i.polarity == Some(clean::ImplPolarity::Negative) { if i.polarity == Some(clean::ImplPolarity::Negative) {
write!(f, "!")?; write!(f, "!")?;
plain.push_str("!");
} }
if link_trait { if link_trait {
fmt::Display::fmt(ty, f)?; fmt::Display::fmt(ty, f)?;
plain.push_str(&format!("{:#}", ty));
} else { } else {
match *ty { match *ty {
clean::ResolvedPath { typarams: None, ref path, is_generic: false, .. } => { clean::ResolvedPath { typarams: None, ref path, is_generic: false, .. } => {
let last = path.segments.last().unwrap(); let last = path.segments.last().unwrap();
fmt::Display::fmt(&last.name, f)?; fmt::Display::fmt(&last.name, f)?;
fmt::Display::fmt(&last.params, f)?; fmt::Display::fmt(&last.params, f)?;
plain.push_str(&format!("{:#}{:#}", last.name, last.params));
} }
_ => unreachable!(), _ => unreachable!(),
} }
} }
write!(f, " for ")?; write!(f, " for ")?;
plain.push_str(" for ");
} }
fmt_type(&i.for_, f, use_absolute, true)?; fmt_type(&i.for_, f, use_absolute, true)?;
plain.push_str(&format!("{:#}", i.for_));
fmt::Display::fmt(&WhereClause(&i.generics, plain.len() + 1), f)?; fmt::Display::fmt(&WhereClause{ gens: &i.generics, indent: 0, end_newline: true }, f)?;
Ok(()) Ok(())
} }

View file

@ -2012,7 +2012,7 @@ fn item_function(w: &mut fmt::Formatter, cx: &Context, it: &clean::Item,
abi = AbiSpace(f.abi), abi = AbiSpace(f.abi),
name = it.name.as_ref().unwrap(), name = it.name.as_ref().unwrap(),
generics = f.generics, generics = f.generics,
where_clause = WhereClause(&f.generics, 2), where_clause = WhereClause { gens: &f.generics, indent: 0, end_newline: true },
decl = Method(&f.decl, indent))?; decl = Method(&f.decl, indent))?;
document(w, cx, it) document(w, cx, it)
} }
@ -2047,8 +2047,7 @@ fn item_trait(w: &mut fmt::Formatter, cx: &Context, it: &clean::Item,
it.name.as_ref().unwrap(), it.name.as_ref().unwrap(),
t.generics, t.generics,
bounds, bounds,
// Where clauses in traits are indented nine spaces, per rustdoc.css WhereClause { gens: &t.generics, indent: 0, end_newline: true })?;
WhereClause(&t.generics, 9))?;
let types = t.items.iter().filter(|m| m.is_associated_type()).collect::<Vec<_>>(); let types = t.items.iter().filter(|m| m.is_associated_type()).collect::<Vec<_>>();
let consts = t.items.iter().filter(|m| m.is_associated_const()).collect::<Vec<_>>(); let consts = t.items.iter().filter(|m| m.is_associated_const()).collect::<Vec<_>>();
@ -2087,7 +2086,14 @@ fn item_trait(w: &mut fmt::Formatter, cx: &Context, it: &clean::Item,
for m in &provided { for m in &provided {
write!(w, " ")?; write!(w, " ")?;
render_assoc_item(w, m, AssocItemLink::Anchor(None), ItemType::Trait)?; render_assoc_item(w, m, AssocItemLink::Anchor(None), ItemType::Trait)?;
match m.inner {
clean::MethodItem(ref inner) if !inner.generics.where_predicates.is_empty() => {
write!(w, ",\n {{ ... }}\n")?;
},
_ => {
write!(w, " {{ ... }}\n")?; write!(w, " {{ ... }}\n")?;
},
}
} }
write!(w, "}}")?; write!(w, "}}")?;
} }
@ -2327,14 +2333,14 @@ fn render_assoc_item(w: &mut fmt::Formatter,
name, name,
*g); *g);
let mut indent = prefix.len(); let mut indent = prefix.len();
let where_indent = if parent == ItemType::Trait { let (where_indent, end_newline) = if parent == ItemType::Trait {
indent += 4; indent += 4;
8 (4, false)
} else if parent == ItemType::Impl { } else if parent == ItemType::Impl {
2 (0, true)
} else { } else {
let prefix = prefix + &format!("{:#}", Method(d, indent)); let prefix = prefix + &format!("{:#}", Method(d, indent));
prefix.lines().last().unwrap().len() + 1 (prefix.lines().last().unwrap().len() + 1, true)
}; };
write!(w, "{}{}{}fn <a href='{href}' class='fnname'>{name}</a>\ write!(w, "{}{}{}fn <a href='{href}' class='fnname'>{name}</a>\
{generics}{decl}{where_clause}", {generics}{decl}{where_clause}",
@ -2345,7 +2351,11 @@ fn render_assoc_item(w: &mut fmt::Formatter,
name = name, name = name,
generics = *g, generics = *g,
decl = Method(d, indent), decl = Method(d, indent),
where_clause = WhereClause(g, where_indent)) where_clause = WhereClause {
gens: g,
indent: where_indent,
end_newline: end_newline,
})
} }
match item.inner { match item.inner {
clean::StrippedItem(..) => Ok(()), clean::StrippedItem(..) => Ok(()),
@ -2458,15 +2468,11 @@ fn item_enum(w: &mut fmt::Formatter, cx: &Context, it: &clean::Item,
e: &clean::Enum) -> fmt::Result { e: &clean::Enum) -> fmt::Result {
write!(w, "<pre class='rust enum'>")?; write!(w, "<pre class='rust enum'>")?;
render_attributes(w, it)?; render_attributes(w, it)?;
let padding = format!("{}enum {}{:#} ",
VisSpace(&it.visibility),
it.name.as_ref().unwrap(),
e.generics).len();
write!(w, "{}enum {}{}{}", write!(w, "{}enum {}{}{}",
VisSpace(&it.visibility), VisSpace(&it.visibility),
it.name.as_ref().unwrap(), it.name.as_ref().unwrap(),
e.generics, e.generics,
WhereClause(&e.generics, padding))?; WhereClause { gens: &e.generics, indent: 0, end_newline: true })?;
if e.variants.is_empty() && !e.variants_stripped { if e.variants.is_empty() && !e.variants_stripped {
write!(w, " {{}}")?; write!(w, " {{}}")?;
} else { } else {
@ -2640,23 +2646,17 @@ fn render_struct(w: &mut fmt::Formatter, it: &clean::Item,
fields: &[clean::Item], fields: &[clean::Item],
tab: &str, tab: &str,
structhead: bool) -> fmt::Result { structhead: bool) -> fmt::Result {
let mut plain = String::new();
write!(w, "{}{}{}", write!(w, "{}{}{}",
VisSpace(&it.visibility), VisSpace(&it.visibility),
if structhead {"struct "} else {""}, if structhead {"struct "} else {""},
it.name.as_ref().unwrap())?; it.name.as_ref().unwrap())?;
plain.push_str(&format!("{}{}{}",
VisSpace(&it.visibility),
if structhead {"struct "} else {""},
it.name.as_ref().unwrap()));
if let Some(g) = g { if let Some(g) = g {
plain.push_str(&format!("{:#}", g));
write!(w, "{}", g)? write!(w, "{}", g)?
} }
match ty { match ty {
doctree::Plain => { doctree::Plain => {
if let Some(g) = g { if let Some(g) = g {
write!(w, "{}", WhereClause(g, plain.len() + 1))? write!(w, "{}", WhereClause { gens: g, indent: 0, end_newline: true })?
} }
let mut has_visible_fields = false; let mut has_visible_fields = false;
write!(w, " {{")?; write!(w, " {{")?;
@ -2685,35 +2685,30 @@ fn render_struct(w: &mut fmt::Formatter, it: &clean::Item,
} }
doctree::Tuple => { doctree::Tuple => {
write!(w, "(")?; write!(w, "(")?;
plain.push_str("(");
for (i, field) in fields.iter().enumerate() { for (i, field) in fields.iter().enumerate() {
if i > 0 { if i > 0 {
write!(w, ", ")?; write!(w, ", ")?;
plain.push_str(", ");
} }
match field.inner { match field.inner {
clean::StrippedItem(box clean::StructFieldItem(..)) => { clean::StrippedItem(box clean::StructFieldItem(..)) => {
plain.push_str("_");
write!(w, "_")? write!(w, "_")?
} }
clean::StructFieldItem(ref ty) => { clean::StructFieldItem(ref ty) => {
plain.push_str(&format!("{}{:#}", VisSpace(&field.visibility), *ty));
write!(w, "{}{}", VisSpace(&field.visibility), *ty)? write!(w, "{}{}", VisSpace(&field.visibility), *ty)?
} }
_ => unreachable!() _ => unreachable!()
} }
} }
write!(w, ")")?; write!(w, ")")?;
plain.push_str(")");
if let Some(g) = g { if let Some(g) = g {
write!(w, "{}", WhereClause(g, plain.len() + 1))? write!(w, "{}", WhereClause { gens: g, indent: 0, end_newline: true })?
} }
write!(w, ";")?; write!(w, ";")?;
} }
doctree::Unit => { doctree::Unit => {
// Needed for PhantomData. // Needed for PhantomData.
if let Some(g) = g { if let Some(g) = g {
write!(w, "{}", WhereClause(g, plain.len() + 1))? write!(w, "{}", WhereClause { gens: g, indent: 0, end_newline: true })?
} }
write!(w, ";")?; write!(w, ";")?;
} }
@ -2726,19 +2721,13 @@ fn render_union(w: &mut fmt::Formatter, it: &clean::Item,
fields: &[clean::Item], fields: &[clean::Item],
tab: &str, tab: &str,
structhead: bool) -> fmt::Result { structhead: bool) -> fmt::Result {
let mut plain = String::new();
write!(w, "{}{}{}", write!(w, "{}{}{}",
VisSpace(&it.visibility), VisSpace(&it.visibility),
if structhead {"union "} else {""}, if structhead {"union "} else {""},
it.name.as_ref().unwrap())?; it.name.as_ref().unwrap())?;
plain.push_str(&format!("{}{}{}",
VisSpace(&it.visibility),
if structhead {"union "} else {""},
it.name.as_ref().unwrap()));
if let Some(g) = g { if let Some(g) = g {
write!(w, "{}", g)?; write!(w, "{}", g)?;
plain.push_str(&format!("{:#}", g)); write!(w, "{}", WhereClause { gens: g, indent: 0, end_newline: true })?;
write!(w, "{}", WhereClause(g, plain.len() + 1))?;
} }
write!(w, " {{\n{}", tab)?; write!(w, " {{\n{}", tab)?;
@ -3037,13 +3026,12 @@ fn render_impl(w: &mut fmt::Formatter, cx: &Context, i: &Impl, link: AssocItemLi
fn item_typedef(w: &mut fmt::Formatter, cx: &Context, it: &clean::Item, fn item_typedef(w: &mut fmt::Formatter, cx: &Context, it: &clean::Item,
t: &clean::Typedef) -> fmt::Result { t: &clean::Typedef) -> fmt::Result {
let indent = format!("type {}{:#} ", it.name.as_ref().unwrap(), t.generics).len();
write!(w, "<pre class='rust typedef'>")?; write!(w, "<pre class='rust typedef'>")?;
render_attributes(w, it)?; render_attributes(w, it)?;
write!(w, "type {}{}{where_clause} = {type_};</pre>", write!(w, "type {}{}{where_clause} = {type_};</pre>",
it.name.as_ref().unwrap(), it.name.as_ref().unwrap(),
t.generics, t.generics,
where_clause = WhereClause(&t.generics, indent), where_clause = WhereClause { gens: &t.generics, indent: 0, end_newline: true },
type_ = t.type_)?; type_ = t.type_)?;
document(w, cx, it) document(w, cx, it)

View file

@ -379,12 +379,6 @@ h4 > code, h3 > code, .invisible > code {
.content .where.fmt-newline { .content .where.fmt-newline {
display: block; display: block;
} }
/* Bit of whitespace to indent it */
.content .method .where::before,
.content .fn .where::before,
.content .where.fmt-newline::before {
content: ' ';
}
.content .methods > div { margin-left: 40px; } .content .methods > div { margin-left: 40px; }
@ -399,11 +393,6 @@ h4 > code, h3 > code, .invisible > code {
font-size: 90%; font-size: 90%;
} }
/* Shift where in trait listing down a line */
pre.trait .where::before {
content: '\a ';
}
nav { nav {
border-bottom: 1px solid; border-bottom: 1px solid;
padding-bottom: 10px; padding-bottom: 10px;