2015-07-01 23:52:40 -04:00
|
|
|
// Copyright 2013-2015 The Rust Project Developers. See the COPYRIGHT
|
2013-09-18 22:18:38 -07:00
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
2013-10-03 10:24:40 -07:00
|
|
|
//! Rustdoc's HTML Rendering module
|
|
|
|
//!
|
|
|
|
//! This modules contains the bulk of the logic necessary for rendering a
|
|
|
|
//! rustdoc `clean::Crate` instance to a set of static HTML pages. This
|
|
|
|
//! rendering process is largely driven by the `format!` syntax extension to
|
|
|
|
//! perform all I/O into files and streams.
|
|
|
|
//!
|
|
|
|
//! The rendering process is largely driven by the `Context` and `Cache`
|
|
|
|
//! structures. The cache is pre-populated by crawling the crate in question,
|
2015-05-09 00:12:29 +09:00
|
|
|
//! and then it is shared among the various rendering threads. The cache is meant
|
2013-10-03 10:24:40 -07:00
|
|
|
//! to be a fairly large structure not implementing `Clone` (because it's shared
|
2015-05-09 00:12:29 +09:00
|
|
|
//! among threads). The context, however, should be a lightweight structure. This
|
|
|
|
//! is cloned per-thread and contains information about what is currently being
|
2013-10-03 10:24:40 -07:00
|
|
|
//! rendered.
|
|
|
|
//!
|
|
|
|
//! In order to speed up rendering (mostly because of markdown rendering), the
|
|
|
|
//! rendering process has been parallelized. This parallelization is only
|
|
|
|
//! exposed through the `crate` method on the context, and then also from the
|
|
|
|
//! fact that the shared cache is stored in TLS (and must be accessed as such).
|
|
|
|
//!
|
|
|
|
//! In addition to rendering the crate itself, this module is also responsible
|
|
|
|
//! for creating the corresponding search index and source file renderings.
|
2015-05-09 00:12:29 +09:00
|
|
|
//! These threads are not parallelized (they haven't been a bottleneck yet), and
|
2013-10-03 10:24:40 -07:00
|
|
|
//! both occur before the crate is rendered.
|
2014-11-06 00:05:53 -08:00
|
|
|
pub use self::ExternalLocation::*;
|
2013-10-03 10:24:40 -07:00
|
|
|
|
2015-07-08 08:33:13 -07:00
|
|
|
use std::ascii::AsciiExt;
|
2014-11-14 14:20:57 -08:00
|
|
|
use std::cell::RefCell;
|
2014-12-26 10:55:16 +02:00
|
|
|
use std::cmp::Ordering;
|
2016-08-24 10:55:03 +02:00
|
|
|
use std::collections::BTreeMap;
|
2014-11-14 14:20:57 -08:00
|
|
|
use std::default::Default;
|
2015-09-26 23:59:32 +02:00
|
|
|
use std::error;
|
|
|
|
use std::fmt::{self, Display, Formatter};
|
2016-08-03 13:56:18 +12:00
|
|
|
use std::fs::{self, File, OpenOptions};
|
2015-02-26 21:00:43 -08:00
|
|
|
use std::io::prelude::*;
|
|
|
|
use std::io::{self, BufWriter, BufReader};
|
2014-12-10 19:46:38 -08:00
|
|
|
use std::iter::repeat;
|
2015-04-06 15:10:55 -07:00
|
|
|
use std::mem;
|
2016-02-23 07:52:43 +01:00
|
|
|
use std::path::{PathBuf, Path, Component};
|
2014-04-02 16:54:22 -07:00
|
|
|
use std::str;
|
2014-06-07 11:13:26 -07:00
|
|
|
use std::sync::Arc;
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2014-06-28 03:35:25 -07:00
|
|
|
use externalfiles::ExternalHtml;
|
|
|
|
|
2016-02-16 19:48:28 +01:00
|
|
|
use serialize::json::{ToJson, Json, as_json};
|
2015-10-13 06:01:31 +03:00
|
|
|
use syntax::{abi, ast};
|
2016-02-01 13:05:37 -05:00
|
|
|
use syntax::feature_gate::UnstableFeatures;
|
2015-11-25 01:23:22 +02:00
|
|
|
use rustc::middle::cstore::LOCAL_CRATE;
|
2016-03-29 12:54:26 +03:00
|
|
|
use rustc::hir::def_id::{CRATE_DEF_INDEX, DefId};
|
2015-11-19 14:16:35 +03:00
|
|
|
use rustc::middle::privacy::AccessLevels;
|
2015-10-13 06:01:31 +03:00
|
|
|
use rustc::middle::stability;
|
2016-02-01 13:05:37 -05:00
|
|
|
use rustc::session::config::get_unstable_features_setting;
|
2016-03-29 08:50:44 +03:00
|
|
|
use rustc::hir;
|
2016-08-24 10:55:03 +02:00
|
|
|
use rustc::util::nodemap::{FnvHashMap, FnvHashSet};
|
2016-08-12 18:22:02 -04:00
|
|
|
use rustc_data_structures::flock;
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2016-05-08 21:19:29 +03:00
|
|
|
use clean::{self, Attributes, GetDefId};
|
2013-09-18 22:18:38 -07:00
|
|
|
use doctree;
|
|
|
|
use fold::DocFolder;
|
2015-04-13 11:55:00 -07:00
|
|
|
use html::escape::Escape;
|
2015-02-25 22:05:07 +02:00
|
|
|
use html::format::{ConstnessSpace};
|
2015-04-13 11:55:00 -07:00
|
|
|
use html::format::{TyParamBounds, WhereClause, href, AbiSpace};
|
|
|
|
use html::format::{VisSpace, Method, UnsafetySpace, MutableSpace};
|
2016-03-29 01:11:08 +09:00
|
|
|
use html::format::fmt_impl_for_trait_page;
|
2014-12-03 23:57:45 +09:00
|
|
|
use html::item_type::ItemType;
|
2015-04-26 21:03:38 +02:00
|
|
|
use html::markdown::{self, Markdown};
|
|
|
|
use html::{highlight, layout};
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2014-12-23 09:58:38 +08:00
|
|
|
/// A pair of name and its optional document.
|
2015-03-05 16:35:43 +09:00
|
|
|
pub type NameDoc = (String, Option<String>);
|
2014-12-23 09:58:38 +08:00
|
|
|
|
2013-10-03 10:24:40 -07:00
|
|
|
/// Major driving force in all rustdoc rendering. This contains information
|
|
|
|
/// about where in the tree-like hierarchy rendering is occurring and controls
|
|
|
|
/// how the current page is being rendered.
|
|
|
|
///
|
|
|
|
/// It is intended that this context is a lightweight object which can be fairly
|
|
|
|
/// easily cloned because it is cloned per work-job (about once per item in the
|
|
|
|
/// rustdoc tree).
|
2015-01-03 22:54:18 -05:00
|
|
|
#[derive(Clone)]
|
2013-09-18 22:18:38 -07:00
|
|
|
pub struct Context {
|
2013-10-03 10:24:40 -07:00
|
|
|
/// Current hierarchy of components leading down to what's currently being
|
|
|
|
/// rendered
|
2014-05-29 13:50:47 -07:00
|
|
|
pub current: Vec<String>,
|
2013-10-03 10:24:40 -07:00
|
|
|
/// String representation of how to get back to the root path of the 'doc/'
|
|
|
|
/// folder in terms of a relative URL.
|
2014-05-22 16:57:53 -07:00
|
|
|
pub root_path: String,
|
2013-10-03 10:24:40 -07:00
|
|
|
/// The current destination folder of where HTML artifacts should be placed.
|
|
|
|
/// This changes as the context descends into the module hierarchy.
|
2015-02-26 21:00:43 -08:00
|
|
|
pub dst: PathBuf,
|
2016-04-02 09:03:55 +02:00
|
|
|
/// A flag, which when `true`, will render pages which redirect to the
|
|
|
|
/// real location of an item. This is used to allow external links to
|
|
|
|
/// publicly reused items to redirect to the right location.
|
|
|
|
pub render_redirect_pages: bool,
|
|
|
|
pub shared: Arc<SharedContext>,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct SharedContext {
|
|
|
|
/// The path to the crate root source minus the file name.
|
|
|
|
/// Used for simplifying paths to the highlighted source code files.
|
|
|
|
pub src_root: PathBuf,
|
2013-10-03 10:24:40 -07:00
|
|
|
/// This describes the layout of each page, and is not modified after
|
2014-06-28 03:35:25 -07:00
|
|
|
/// creation of the context (contains info like the favicon and added html).
|
2014-03-28 10:27:24 -07:00
|
|
|
pub layout: layout::Layout,
|
2013-10-03 10:24:40 -07:00
|
|
|
/// This flag indicates whether [src] links should be generated or not. If
|
|
|
|
/// the source files are present in the html rendering, then this will be
|
|
|
|
/// `true`.
|
2014-03-28 10:27:24 -07:00
|
|
|
pub include_sources: bool,
|
2016-02-27 07:35:05 +01:00
|
|
|
/// The local file sources we've emitted and their respective url-paths.
|
2016-08-24 10:55:03 +02:00
|
|
|
pub local_sources: FnvHashMap<PathBuf, String>,
|
2014-11-09 19:09:17 -05:00
|
|
|
/// All the passes that were run on this crate.
|
2016-08-24 10:55:03 +02:00
|
|
|
pub passes: FnvHashSet<String>,
|
2015-08-16 15:53:18 +02:00
|
|
|
/// The base-URL of the issue tracker for when an item has been tagged with
|
|
|
|
/// an issue number.
|
|
|
|
pub issue_tracker_base_url: Option<String>,
|
2016-03-13 15:36:01 +01:00
|
|
|
/// The given user css file which allow to customize the generated
|
|
|
|
/// documentation theme.
|
|
|
|
pub css_file_extension: Option<PathBuf>,
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2013-10-03 10:24:40 -07:00
|
|
|
/// Indicates where an external crate can be found.
|
2013-10-02 15:39:32 -07:00
|
|
|
pub enum ExternalLocation {
|
2013-10-03 10:24:40 -07:00
|
|
|
/// Remote URL root of the external crate
|
2014-05-22 16:57:53 -07:00
|
|
|
Remote(String),
|
2013-10-03 10:24:40 -07:00
|
|
|
/// This external crate can be found in the local doc/ folder
|
|
|
|
Local,
|
|
|
|
/// The external crate could not be found.
|
|
|
|
Unknown,
|
2013-10-02 15:39:32 -07:00
|
|
|
}
|
|
|
|
|
2014-05-27 17:52:40 -07:00
|
|
|
/// Metadata about an implementor of a trait.
|
|
|
|
pub struct Implementor {
|
2015-08-16 06:32:28 -04:00
|
|
|
pub def_id: DefId,
|
2014-11-10 15:33:21 -08:00
|
|
|
pub stability: Option<clean::Stability>,
|
2015-07-18 02:02:57 -04:00
|
|
|
pub impl_: clean::Impl,
|
2014-06-26 11:37:39 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Metadata about implementations for a type.
|
2015-01-03 22:54:18 -05:00
|
|
|
#[derive(Clone)]
|
2014-06-26 11:37:39 -07:00
|
|
|
pub struct Impl {
|
2016-05-03 19:15:59 +02:00
|
|
|
pub impl_item: clean::Item,
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2015-04-06 17:56:35 -07:00
|
|
|
impl Impl {
|
2016-05-03 19:15:59 +02:00
|
|
|
fn inner_impl(&self) -> &clean::Impl {
|
|
|
|
match self.impl_item.inner {
|
|
|
|
clean::ImplItem(ref impl_) => impl_,
|
|
|
|
_ => panic!("non-impl item found in impl")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-16 06:32:28 -04:00
|
|
|
fn trait_did(&self) -> Option<DefId> {
|
2016-05-03 19:15:59 +02:00
|
|
|
self.inner_impl().trait_.def_id()
|
2015-04-06 17:56:35 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-26 23:59:32 +02:00
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct Error {
|
|
|
|
file: PathBuf,
|
|
|
|
error: io::Error,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl error::Error for Error {
|
|
|
|
fn description(&self) -> &str {
|
|
|
|
self.error.description()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Display for Error {
|
|
|
|
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
|
|
|
|
write!(f, "\"{}\": {}", self.file.display(), self.error)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Error {
|
|
|
|
pub fn new(e: io::Error, file: &Path) -> Error {
|
|
|
|
Error {
|
|
|
|
file: file.to_path_buf(),
|
|
|
|
error: e,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! try_err {
|
|
|
|
($e:expr, $file:expr) => ({
|
|
|
|
match $e {
|
|
|
|
Ok(e) => e,
|
|
|
|
Err(e) => return Err(Error::new(e, $file)),
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2013-10-03 10:24:40 -07:00
|
|
|
/// This cache is used to store information about the `clean::Crate` being
|
|
|
|
/// rendered in order to provide more useful documentation. This contains
|
|
|
|
/// information like all implementors of a trait, all traits a type implements,
|
|
|
|
/// documentation for all known traits, etc.
|
|
|
|
///
|
|
|
|
/// This structure purposefully does not implement `Clone` because it's intended
|
|
|
|
/// to be a fairly large and expensive structure to clone. Instead this adheres
|
2014-03-22 14:42:32 +01:00
|
|
|
/// to `Send` so it may be stored in a `Arc` instance and shared among the various
|
2015-05-09 00:12:29 +09:00
|
|
|
/// rendering threads.
|
2015-01-03 22:54:18 -05:00
|
|
|
#[derive(Default)]
|
2013-10-05 14:44:37 -07:00
|
|
|
pub struct Cache {
|
2013-10-03 10:24:40 -07:00
|
|
|
/// Mapping of typaram ids to the name of the type parameter. This is used
|
|
|
|
/// when pretty-printing a type (so pretty printing doesn't have to
|
|
|
|
/// painfully maintain a context like this)
|
2016-08-24 10:55:03 +02:00
|
|
|
pub typarams: FnvHashMap<DefId, String>,
|
2013-10-03 10:24:40 -07:00
|
|
|
|
|
|
|
/// Maps a type id to all known implementations for that type. This is only
|
|
|
|
/// recognized for intra-crate `ResolvedPath` types, and is used to print
|
|
|
|
/// out extra documentation on the page of an enum/struct.
|
|
|
|
///
|
|
|
|
/// The values of the map are a list of implementations and documentation
|
|
|
|
/// found on that implementation.
|
2016-08-24 10:55:03 +02:00
|
|
|
pub impls: FnvHashMap<DefId, Vec<Impl>>,
|
2013-10-03 10:24:40 -07:00
|
|
|
|
|
|
|
/// Maintains a mapping of local crate node ids to the fully qualified name
|
|
|
|
/// and "short type description" of that node. This is used when generating
|
|
|
|
/// URLs when a type is being linked to. External paths are not located in
|
|
|
|
/// this map because the `External` type itself has all the information
|
|
|
|
/// necessary.
|
2016-08-24 10:55:03 +02:00
|
|
|
pub paths: FnvHashMap<DefId, (Vec<String>, ItemType)>,
|
2013-10-03 10:24:40 -07:00
|
|
|
|
2014-05-23 20:17:27 -07:00
|
|
|
/// Similar to `paths`, but only holds external paths. This is only used for
|
|
|
|
/// generating explicit hyperlinks to other crates.
|
2016-08-24 10:55:03 +02:00
|
|
|
pub external_paths: FnvHashMap<DefId, (Vec<String>, ItemType)>,
|
2014-05-23 20:17:27 -07:00
|
|
|
|
2013-10-03 10:24:40 -07:00
|
|
|
/// This map contains information about all known traits of this crate.
|
|
|
|
/// Implementations of a crate should inherit the documentation of the
|
2013-10-21 11:33:04 -07:00
|
|
|
/// parent trait if no extra documentation is specified, and default methods
|
|
|
|
/// should show up in documentation about trait implementations.
|
2016-08-24 10:55:03 +02:00
|
|
|
pub traits: FnvHashMap<DefId, clean::Trait>,
|
2013-10-03 10:24:40 -07:00
|
|
|
|
|
|
|
/// When rendering traits, it's often useful to be able to list all
|
|
|
|
/// implementors of the trait, and this mapping is exactly, that: a mapping
|
|
|
|
/// of trait ids to the list of known implementors of the trait
|
2016-08-24 10:55:03 +02:00
|
|
|
pub implementors: FnvHashMap<DefId, Vec<Implementor>>,
|
2013-10-03 10:24:40 -07:00
|
|
|
|
|
|
|
/// Cache of where external crate documentation can be found.
|
2016-08-24 10:55:03 +02:00
|
|
|
pub extern_locations: FnvHashMap<ast::CrateNum, (String, ExternalLocation)>,
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2014-05-28 19:53:37 -07:00
|
|
|
/// Cache of where documentation for primitives can be found.
|
2016-08-24 10:55:03 +02:00
|
|
|
pub primitive_locations: FnvHashMap<clean::PrimitiveType, ast::CrateNum>,
|
2014-05-28 19:53:37 -07:00
|
|
|
|
2016-04-17 08:54:48 +02:00
|
|
|
// Note that external items for which `doc(hidden)` applies to are shown as
|
|
|
|
// non-reachable while local items aren't. This is because we're reusing
|
|
|
|
// the access levels from crateanalysis.
|
|
|
|
pub access_levels: Arc<AccessLevels<DefId>>,
|
|
|
|
|
2013-10-03 10:24:40 -07:00
|
|
|
// Private fields only used when initially crawling a crate to build a cache
|
|
|
|
|
2014-05-22 22:00:18 -07:00
|
|
|
stack: Vec<String>,
|
2015-08-16 06:32:28 -04:00
|
|
|
parent_stack: Vec<DefId>,
|
2016-02-23 09:52:44 +01:00
|
|
|
parent_is_trait_impl: bool,
|
2014-05-22 22:00:18 -07:00
|
|
|
search_index: Vec<IndexItem>,
|
2016-08-24 10:55:03 +02:00
|
|
|
seen_modules: FnvHashSet<DefId>,
|
2016-04-22 17:53:15 +02:00
|
|
|
seen_mod: bool,
|
2016-03-31 18:15:54 +02:00
|
|
|
stripped_mod: bool,
|
2015-08-16 06:32:28 -04:00
|
|
|
deref_trait_did: Option<DefId>,
|
2014-03-07 17:26:06 +08:00
|
|
|
|
|
|
|
// In rare case where a structure is defined in one module but implemented
|
|
|
|
// in another, if the implementing module is parsed before defining module,
|
|
|
|
// then the fully qualified name of the structure isn't presented in `paths`
|
|
|
|
// yet when its implementation methods are being indexed. Caches such methods
|
|
|
|
// and their parent id here and indexes them at the end of crate parsing.
|
2015-09-02 16:11:32 -04:00
|
|
|
orphan_methods: Vec<(DefId, clean::Item)>,
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2016-04-07 05:59:02 +02:00
|
|
|
/// Temporary storage for data obtained during `RustdocVisitor::clean()`.
|
|
|
|
/// Later on moved into `CACHE_KEY`.
|
|
|
|
#[derive(Default)]
|
|
|
|
pub struct RenderInfo {
|
2016-08-24 10:55:03 +02:00
|
|
|
pub inlined: FnvHashSet<DefId>,
|
2016-04-07 05:59:02 +02:00
|
|
|
pub external_paths: ::core::ExternalPaths,
|
2016-08-24 10:55:03 +02:00
|
|
|
pub external_typarams: FnvHashMap<DefId, String>,
|
2016-04-07 05:59:02 +02:00
|
|
|
pub deref_trait_did: Option<DefId>,
|
|
|
|
}
|
|
|
|
|
2013-10-03 10:24:40 -07:00
|
|
|
/// Helper struct to render all source code to HTML pages
|
2013-12-09 23:16:18 -08:00
|
|
|
struct SourceCollector<'a> {
|
2016-04-02 09:03:55 +02:00
|
|
|
scx: &'a mut SharedContext,
|
2013-10-03 10:24:40 -07:00
|
|
|
|
|
|
|
/// Root destination to place all HTML output into
|
2015-02-26 21:00:43 -08:00
|
|
|
dst: PathBuf,
|
2013-09-27 15:12:23 -07:00
|
|
|
}
|
|
|
|
|
2013-10-03 10:24:40 -07:00
|
|
|
/// Wrapper struct to render the source code of a file. This will do things like
|
|
|
|
/// adding line numbers to the left-hand side.
|
2013-12-09 23:16:18 -08:00
|
|
|
struct Source<'a>(&'a str);
|
2013-10-03 10:24:40 -07:00
|
|
|
|
|
|
|
// Helper structs for rendering items/sidebars and carrying along contextual
|
|
|
|
// information
|
|
|
|
|
2015-03-30 09:40:52 -04:00
|
|
|
#[derive(Copy, Clone)]
|
librustc: Make `Copy` opt-in.
This change makes the compiler no longer infer whether types (structures
and enumerations) implement the `Copy` trait (and thus are implicitly
copyable). Rather, you must implement `Copy` yourself via `impl Copy for
MyType {}`.
A new warning has been added, `missing_copy_implementations`, to warn
you if a non-generic public type has been added that could have
implemented `Copy` but didn't.
For convenience, you may *temporarily* opt out of this behavior by using
`#![feature(opt_out_copy)]`. Note though that this feature gate will never be
accepted and will be removed by the time that 1.0 is released, so you should
transition your code away from using it.
This breaks code like:
#[deriving(Show)]
struct Point2D {
x: int,
y: int,
}
fn main() {
let mypoint = Point2D {
x: 1,
y: 1,
};
let otherpoint = mypoint;
println!("{}{}", mypoint, otherpoint);
}
Change this code to:
#[deriving(Show)]
struct Point2D {
x: int,
y: int,
}
impl Copy for Point2D {}
fn main() {
let mypoint = Point2D {
x: 1,
y: 1,
};
let otherpoint = mypoint;
println!("{}{}", mypoint, otherpoint);
}
This is the backwards-incompatible part of #13231.
Part of RFC #3.
[breaking-change]
2014-12-05 17:01:33 -08:00
|
|
|
struct Item<'a> {
|
|
|
|
cx: &'a Context,
|
|
|
|
item: &'a clean::Item,
|
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
struct Sidebar<'a> { cx: &'a Context, item: &'a clean::Item, }
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2013-10-03 10:24:40 -07:00
|
|
|
/// Struct representing one entry in the JS search index. These are all emitted
|
|
|
|
/// by hand to a large JS file at the end of cache-creation.
|
2013-09-18 22:18:38 -07:00
|
|
|
struct IndexItem {
|
2014-04-09 16:49:31 +09:00
|
|
|
ty: ItemType,
|
2014-05-22 16:57:53 -07:00
|
|
|
name: String,
|
|
|
|
path: String,
|
|
|
|
desc: String,
|
2015-08-16 06:32:28 -04:00
|
|
|
parent: Option<DefId>,
|
2016-02-16 19:48:28 +01:00
|
|
|
parent_idx: Option<usize>,
|
2015-02-26 01:03:06 +02:00
|
|
|
search_type: Option<IndexItemFunctionType>,
|
|
|
|
}
|
|
|
|
|
2016-02-16 19:48:28 +01:00
|
|
|
impl ToJson for IndexItem {
|
|
|
|
fn to_json(&self) -> Json {
|
|
|
|
assert_eq!(self.parent.is_some(), self.parent_idx.is_some());
|
|
|
|
|
|
|
|
let mut data = Vec::with_capacity(6);
|
|
|
|
data.push((self.ty as usize).to_json());
|
|
|
|
data.push(self.name.to_json());
|
|
|
|
data.push(self.path.to_json());
|
|
|
|
data.push(self.desc.to_json());
|
|
|
|
data.push(self.parent_idx.to_json());
|
|
|
|
data.push(self.search_type.to_json());
|
|
|
|
|
|
|
|
Json::Array(data)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-26 01:03:06 +02:00
|
|
|
/// A type used for the search index.
|
|
|
|
struct Type {
|
|
|
|
name: Option<String>,
|
|
|
|
}
|
|
|
|
|
2016-02-16 19:48:28 +01:00
|
|
|
impl ToJson for Type {
|
|
|
|
fn to_json(&self) -> Json {
|
2015-02-26 01:03:06 +02:00
|
|
|
match self.name {
|
2016-02-16 19:48:28 +01:00
|
|
|
Some(ref name) => {
|
|
|
|
let mut data = BTreeMap::new();
|
|
|
|
data.insert("name".to_owned(), name.to_json());
|
|
|
|
Json::Object(data)
|
|
|
|
},
|
|
|
|
None => Json::Null
|
2015-02-26 01:03:06 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Full type of functions/methods in the search index.
|
|
|
|
struct IndexItemFunctionType {
|
|
|
|
inputs: Vec<Type>,
|
|
|
|
output: Option<Type>
|
|
|
|
}
|
|
|
|
|
2016-02-16 19:48:28 +01:00
|
|
|
impl ToJson for IndexItemFunctionType {
|
|
|
|
fn to_json(&self) -> Json {
|
2015-02-26 01:03:06 +02:00
|
|
|
// If we couldn't figure out a type, just write `null`.
|
2016-02-16 19:48:28 +01:00
|
|
|
if self.inputs.iter().chain(self.output.iter()).any(|ref i| i.name.is_none()) {
|
|
|
|
Json::Null
|
|
|
|
} else {
|
|
|
|
let mut data = BTreeMap::new();
|
|
|
|
data.insert("inputs".to_owned(), self.inputs.to_json());
|
|
|
|
data.insert("output".to_owned(), self.output.to_json());
|
|
|
|
Json::Object(data)
|
2015-02-26 01:03:06 +02:00
|
|
|
}
|
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2013-10-03 10:24:40 -07:00
|
|
|
// TLS keys used to carry information around during rendering.
|
2013-09-27 15:12:23 -07:00
|
|
|
|
2014-11-14 09:18:10 -08:00
|
|
|
thread_local!(static CACHE_KEY: RefCell<Arc<Cache>> = Default::default());
|
2014-11-14 14:20:57 -08:00
|
|
|
thread_local!(pub static CURRENT_LOCATION_KEY: RefCell<Vec<String>> =
|
2014-11-14 09:18:10 -08:00
|
|
|
RefCell::new(Vec::new()));
|
2016-08-24 10:55:03 +02:00
|
|
|
thread_local!(static USED_ID_MAP: RefCell<FnvHashMap<String, usize>> =
|
2015-12-03 02:19:23 +01:00
|
|
|
RefCell::new(init_ids()));
|
|
|
|
|
2016-08-24 10:55:03 +02:00
|
|
|
fn init_ids() -> FnvHashMap<String, usize> {
|
2015-12-03 02:19:23 +01:00
|
|
|
[
|
|
|
|
"main",
|
|
|
|
"search",
|
|
|
|
"help",
|
|
|
|
"TOC",
|
|
|
|
"render-detail",
|
|
|
|
"associated-types",
|
|
|
|
"associated-const",
|
|
|
|
"required-methods",
|
|
|
|
"provided-methods",
|
|
|
|
"implementors",
|
|
|
|
"implementors-list",
|
|
|
|
"methods",
|
|
|
|
"deref-methods",
|
|
|
|
"implementations",
|
2016-02-28 16:38:51 +01:00
|
|
|
].into_iter().map(|id| (String::from(*id), 1)).collect()
|
2015-12-03 02:19:23 +01:00
|
|
|
}
|
2015-12-03 00:06:26 +01:00
|
|
|
|
|
|
|
/// This method resets the local table of used ID attributes. This is typically
|
|
|
|
/// used at the beginning of rendering an entire HTML page to reset from the
|
|
|
|
/// previous state (if any).
|
2016-03-26 16:42:38 +01:00
|
|
|
pub fn reset_ids(embedded: bool) {
|
|
|
|
USED_ID_MAP.with(|s| {
|
|
|
|
*s.borrow_mut() = if embedded {
|
|
|
|
init_ids()
|
|
|
|
} else {
|
2016-08-24 10:55:03 +02:00
|
|
|
FnvHashMap()
|
2016-03-26 16:42:38 +01:00
|
|
|
};
|
|
|
|
});
|
2015-12-03 00:06:26 +01:00
|
|
|
}
|
|
|
|
|
2015-12-04 00:48:59 +01:00
|
|
|
pub fn derive_id(candidate: String) -> String {
|
2015-12-03 00:06:26 +01:00
|
|
|
USED_ID_MAP.with(|map| {
|
2015-12-04 00:48:59 +01:00
|
|
|
let id = match map.borrow_mut().get_mut(&candidate) {
|
|
|
|
None => candidate,
|
2015-12-03 00:06:26 +01:00
|
|
|
Some(a) => {
|
|
|
|
let id = format!("{}-{}", candidate, *a);
|
|
|
|
*a += 1;
|
2015-12-04 00:48:59 +01:00
|
|
|
id
|
2015-12-03 00:06:26 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2015-12-04 00:48:59 +01:00
|
|
|
map.borrow_mut().insert(id.clone(), 1);
|
|
|
|
id
|
2015-12-03 00:06:26 +01:00
|
|
|
})
|
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
|
|
|
|
/// Generates the documentation for `crate` into the directory `dst`
|
2014-11-09 19:09:17 -05:00
|
|
|
pub fn run(mut krate: clean::Crate,
|
|
|
|
external_html: &ExternalHtml,
|
2015-02-26 21:00:43 -08:00
|
|
|
dst: PathBuf,
|
2016-08-24 10:55:03 +02:00
|
|
|
passes: FnvHashSet<String>,
|
2016-04-07 05:59:02 +02:00
|
|
|
css_file_extension: Option<PathBuf>,
|
|
|
|
renderinfo: RenderInfo) -> Result<(), Error> {
|
2015-02-26 21:00:43 -08:00
|
|
|
let src_root = match krate.src.parent() {
|
|
|
|
Some(p) => p.to_path_buf(),
|
2015-03-18 09:14:54 -07:00
|
|
|
None => PathBuf::new(),
|
2015-02-26 21:00:43 -08:00
|
|
|
};
|
2016-04-02 09:03:55 +02:00
|
|
|
let mut scx = SharedContext {
|
2015-02-26 21:00:43 -08:00
|
|
|
src_root: src_root,
|
2014-11-09 19:09:17 -05:00
|
|
|
passes: passes,
|
2016-04-02 09:03:55 +02:00
|
|
|
include_sources: true,
|
2016-08-24 10:55:03 +02:00
|
|
|
local_sources: FnvHashMap(),
|
2016-04-02 09:03:55 +02:00
|
|
|
issue_tracker_base_url: None,
|
2013-09-18 22:18:38 -07:00
|
|
|
layout: layout::Layout {
|
2014-05-25 03:17:19 -07:00
|
|
|
logo: "".to_string(),
|
|
|
|
favicon: "".to_string(),
|
2014-06-28 03:35:25 -07:00
|
|
|
external_html: external_html.clone(),
|
2014-02-05 22:15:24 +01:00
|
|
|
krate: krate.name.clone(),
|
2014-06-06 09:12:18 -07:00
|
|
|
playground_url: "".to_string(),
|
2013-09-18 22:18:38 -07:00
|
|
|
},
|
2016-04-05 05:12:25 +02:00
|
|
|
css_file_extension: css_file_extension.clone(),
|
2013-09-18 22:18:38 -07:00
|
|
|
};
|
2014-06-28 03:35:25 -07:00
|
|
|
|
2014-05-27 17:12:48 -07:00
|
|
|
// Crawl the crate attributes looking for attributes which control how we're
|
|
|
|
// going to emit HTML
|
2016-03-22 20:26:33 +01:00
|
|
|
if let Some(attrs) = krate.module.as_ref().map(|m| m.attrs.list("doc")) {
|
2016-02-28 12:11:13 +01:00
|
|
|
for attr in attrs {
|
|
|
|
match *attr {
|
|
|
|
clean::NameValue(ref x, ref s)
|
|
|
|
if "html_favicon_url" == *x => {
|
2016-04-02 09:03:55 +02:00
|
|
|
scx.layout.favicon = s.to_string();
|
2016-02-28 12:11:13 +01:00
|
|
|
}
|
|
|
|
clean::NameValue(ref x, ref s)
|
|
|
|
if "html_logo_url" == *x => {
|
2016-04-02 09:03:55 +02:00
|
|
|
scx.layout.logo = s.to_string();
|
2016-02-28 12:11:13 +01:00
|
|
|
}
|
|
|
|
clean::NameValue(ref x, ref s)
|
|
|
|
if "html_playground_url" == *x => {
|
2016-04-02 09:03:55 +02:00
|
|
|
scx.layout.playground_url = s.to_string();
|
2016-02-28 12:11:13 +01:00
|
|
|
markdown::PLAYGROUND_KRATE.with(|slot| {
|
|
|
|
if slot.borrow().is_none() {
|
|
|
|
let name = krate.name.clone();
|
|
|
|
*slot.borrow_mut() = Some(Some(name));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
clean::NameValue(ref x, ref s)
|
|
|
|
if "issue_tracker_base_url" == *x => {
|
2016-04-02 09:03:55 +02:00
|
|
|
scx.issue_tracker_base_url = Some(s.to_string());
|
2016-02-28 12:11:13 +01:00
|
|
|
}
|
|
|
|
clean::Word(ref x)
|
|
|
|
if "html_no_source" == *x => {
|
2016-04-02 09:03:55 +02:00
|
|
|
scx.include_sources = false;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2016-02-28 12:11:13 +01:00
|
|
|
_ => {}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-04-02 09:03:55 +02:00
|
|
|
try_err!(mkdir(&dst), &dst);
|
|
|
|
krate = render_sources(&dst, &mut scx, krate)?;
|
|
|
|
let cx = Context {
|
|
|
|
current: Vec::new(),
|
|
|
|
root_path: String::new(),
|
|
|
|
dst: dst,
|
|
|
|
render_redirect_pages: false,
|
|
|
|
shared: Arc::new(scx),
|
|
|
|
};
|
2013-09-18 22:18:38 -07:00
|
|
|
|
|
|
|
// Crawl the crate to build various caches used for the output
|
2016-04-07 05:59:02 +02:00
|
|
|
let RenderInfo {
|
2016-07-09 14:07:37 +01:00
|
|
|
inlined: _,
|
2016-04-07 05:59:02 +02:00
|
|
|
external_paths,
|
|
|
|
external_typarams,
|
|
|
|
deref_trait_did,
|
|
|
|
} = renderinfo;
|
|
|
|
|
2016-07-09 14:07:37 +01:00
|
|
|
let external_paths = external_paths.into_iter()
|
2016-08-20 15:55:43 -04:00
|
|
|
.map(|(k, (v, t))| (k, (v, ItemType::from(t))))
|
2016-07-09 14:07:37 +01:00
|
|
|
.collect();
|
2016-04-07 05:59:02 +02:00
|
|
|
|
2014-04-28 20:36:08 -07:00
|
|
|
let mut cache = Cache {
|
2016-08-24 10:55:03 +02:00
|
|
|
impls: FnvHashMap(),
|
2016-07-09 14:07:37 +01:00
|
|
|
external_paths: external_paths,
|
2016-08-24 10:55:03 +02:00
|
|
|
paths: FnvHashMap(),
|
|
|
|
implementors: FnvHashMap(),
|
2014-04-28 20:36:08 -07:00
|
|
|
stack: Vec::new(),
|
|
|
|
parent_stack: Vec::new(),
|
|
|
|
search_index: Vec::new(),
|
2016-02-23 09:52:44 +01:00
|
|
|
parent_is_trait_impl: false,
|
2016-08-24 10:55:03 +02:00
|
|
|
extern_locations: FnvHashMap(),
|
|
|
|
primitive_locations: FnvHashMap(),
|
|
|
|
seen_modules: FnvHashSet(),
|
2016-04-22 17:53:15 +02:00
|
|
|
seen_mod: false,
|
2016-03-31 18:15:54 +02:00
|
|
|
stripped_mod: false,
|
2016-04-07 05:59:02 +02:00
|
|
|
access_levels: krate.access_levels.clone(),
|
2014-04-28 20:36:08 -07:00
|
|
|
orphan_methods: Vec::new(),
|
2016-08-24 10:55:03 +02:00
|
|
|
traits: mem::replace(&mut krate.external_traits, FnvHashMap()),
|
2016-04-07 05:59:02 +02:00
|
|
|
deref_trait_did: deref_trait_did,
|
|
|
|
typarams: external_typarams,
|
2014-04-28 20:36:08 -07:00
|
|
|
};
|
2014-04-09 15:52:31 +09:00
|
|
|
|
2014-05-28 19:53:37 -07:00
|
|
|
// Cache where all our extern crates are located
|
2015-01-31 12:20:46 -05:00
|
|
|
for &(n, ref e) in &krate.externs {
|
2015-04-13 15:25:40 -07:00
|
|
|
cache.extern_locations.insert(n, (e.name.clone(),
|
|
|
|
extern_location(e, &cx.dst)));
|
2015-09-17 14:29:59 -04:00
|
|
|
let did = DefId { krate: n, index: CRATE_DEF_INDEX };
|
2016-07-09 14:07:37 +01:00
|
|
|
cache.external_paths.insert(did, (vec![e.name.to_string()], ItemType::Module));
|
2014-05-27 17:12:48 -07:00
|
|
|
}
|
|
|
|
|
2014-05-28 19:53:37 -07:00
|
|
|
// Cache where all known primitives have their documentation located.
|
|
|
|
//
|
|
|
|
// Favor linking to as local extern as possible, so iterate all crates in
|
|
|
|
// reverse topological order.
|
|
|
|
for &(n, ref e) in krate.externs.iter().rev() {
|
2015-01-31 12:20:46 -05:00
|
|
|
for &prim in &e.primitives {
|
2014-05-28 19:53:37 -07:00
|
|
|
cache.primitive_locations.insert(prim, n);
|
|
|
|
}
|
|
|
|
}
|
2015-01-31 12:20:46 -05:00
|
|
|
for &prim in &krate.primitives {
|
2015-08-16 06:32:28 -04:00
|
|
|
cache.primitive_locations.insert(prim, LOCAL_CRATE);
|
2014-05-28 19:53:37 -07:00
|
|
|
}
|
|
|
|
|
2015-04-13 16:23:32 -07:00
|
|
|
cache.stack.push(krate.name.clone());
|
|
|
|
krate = cache.fold_crate(krate);
|
|
|
|
|
2014-05-28 19:53:37 -07:00
|
|
|
// Build our search index
|
2015-09-26 23:59:32 +02:00
|
|
|
let index = build_index(&krate, &mut cache);
|
2014-05-27 17:15:10 -07:00
|
|
|
|
|
|
|
// Freeze the cache now that the index has been built. Put an Arc into TLS
|
|
|
|
// for future parallelization opportunities
|
|
|
|
let cache = Arc::new(cache);
|
2014-11-14 14:20:57 -08:00
|
|
|
CACHE_KEY.with(|v| *v.borrow_mut() = cache.clone());
|
|
|
|
CURRENT_LOCATION_KEY.with(|s| s.borrow_mut().clear());
|
2014-05-27 17:15:10 -07:00
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
write_shared(&cx, &krate, &*cache, index)?;
|
2014-05-27 17:12:48 -07:00
|
|
|
|
|
|
|
// And finally render the whole crate's documentation
|
2015-04-13 11:55:00 -07:00
|
|
|
cx.krate(krate)
|
2014-05-27 17:12:48 -07:00
|
|
|
}
|
|
|
|
|
2016-02-16 20:00:57 +01:00
|
|
|
/// Build the search index from the collected metadata
|
2015-09-26 23:59:32 +02:00
|
|
|
fn build_index(krate: &clean::Crate, cache: &mut Cache) -> String {
|
2016-08-24 10:55:03 +02:00
|
|
|
let mut nodeid_to_pathid = FnvHashMap();
|
2016-02-16 19:48:28 +01:00
|
|
|
let mut crate_items = Vec::with_capacity(cache.search_index.len());
|
|
|
|
let mut crate_paths = Vec::<Json>::new();
|
|
|
|
|
|
|
|
let Cache { ref mut search_index,
|
|
|
|
ref orphan_methods,
|
|
|
|
ref mut paths, .. } = *cache;
|
|
|
|
|
|
|
|
// Attach all orphan methods to the type's definition if the type
|
|
|
|
// has since been learned.
|
|
|
|
for &(did, ref item) in orphan_methods {
|
2016-07-03 14:38:37 -07:00
|
|
|
if let Some(&(ref fqp, _)) = paths.get(&did) {
|
|
|
|
search_index.push(IndexItem {
|
2016-08-03 13:14:59 +12:00
|
|
|
ty: item_type(item),
|
2016-07-03 14:38:37 -07:00
|
|
|
name: item.name.clone().unwrap(),
|
|
|
|
path: fqp[..fqp.len() - 1].join("::"),
|
|
|
|
desc: Escape(&shorter(item.doc_value())).to_string(),
|
|
|
|
parent: Some(did),
|
|
|
|
parent_idx: None,
|
|
|
|
search_type: get_index_search_type(&item),
|
|
|
|
});
|
2014-04-09 03:25:54 +09:00
|
|
|
}
|
2014-03-07 17:26:06 +08:00
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2016-02-16 19:48:28 +01:00
|
|
|
// Reduce `NodeId` in paths into smaller sequential numbers,
|
|
|
|
// and prune the paths that do not appear in the index.
|
|
|
|
let mut lastpath = String::new();
|
|
|
|
let mut lastpathid = 0usize;
|
2016-02-16 20:00:57 +01:00
|
|
|
|
2016-02-16 19:48:28 +01:00
|
|
|
for item in search_index {
|
|
|
|
item.parent_idx = item.parent.map(|nodeid| {
|
|
|
|
if nodeid_to_pathid.contains_key(&nodeid) {
|
|
|
|
*nodeid_to_pathid.get(&nodeid).unwrap()
|
|
|
|
} else {
|
|
|
|
let pathid = lastpathid;
|
|
|
|
nodeid_to_pathid.insert(nodeid, pathid);
|
|
|
|
lastpathid += 1;
|
2014-05-27 17:12:48 -07:00
|
|
|
|
2016-02-16 19:48:28 +01:00
|
|
|
let &(ref fqp, short) = paths.get(&nodeid).unwrap();
|
|
|
|
crate_paths.push(((short as usize), fqp.last().unwrap().clone()).to_json());
|
|
|
|
pathid
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// Omit the parent path if it is same to that of the prior item.
|
2014-11-27 14:19:27 -05:00
|
|
|
if lastpath == item.path {
|
2016-02-16 19:48:28 +01:00
|
|
|
item.path.clear();
|
2014-05-27 17:12:48 -07:00
|
|
|
} else {
|
2016-02-16 19:48:28 +01:00
|
|
|
lastpath = item.path.clone();
|
2014-05-27 17:12:48 -07:00
|
|
|
}
|
2016-02-16 19:48:28 +01:00
|
|
|
crate_items.push(item.to_json());
|
2014-05-27 17:12:48 -07:00
|
|
|
}
|
2014-04-10 02:24:00 +09:00
|
|
|
|
2016-02-16 19:48:28 +01:00
|
|
|
let crate_doc = krate.module.as_ref().map(|module| {
|
|
|
|
Escape(&shorter(module.doc_value())).to_string()
|
|
|
|
}).unwrap_or(String::new());
|
2014-04-10 02:24:00 +09:00
|
|
|
|
2016-02-16 19:48:28 +01:00
|
|
|
let mut crate_data = BTreeMap::new();
|
|
|
|
crate_data.insert("doc".to_owned(), Json::String(crate_doc));
|
|
|
|
crate_data.insert("items".to_owned(), Json::Array(crate_items));
|
|
|
|
crate_data.insert("paths".to_owned(), Json::Array(crate_paths));
|
2014-03-16 01:08:56 -07:00
|
|
|
|
2016-02-16 19:48:28 +01:00
|
|
|
// Collect the index into a string
|
|
|
|
format!("searchIndex[{}] = {};",
|
|
|
|
as_json(&krate.name),
|
|
|
|
Json::Object(crate_data))
|
2014-05-27 17:12:48 -07:00
|
|
|
}
|
2014-03-16 01:08:56 -07:00
|
|
|
|
2014-05-27 17:12:48 -07:00
|
|
|
fn write_shared(cx: &Context,
|
|
|
|
krate: &clean::Crate,
|
|
|
|
cache: &Cache,
|
2015-09-26 23:59:32 +02:00
|
|
|
search_index: String) -> Result<(), Error> {
|
2014-03-16 01:08:56 -07:00
|
|
|
// Write out the shared files. Note that these are shared among all rustdoc
|
|
|
|
// docs placed in the output directory, so this needs to be a synchronized
|
|
|
|
// operation with respect to all other rustdocs running around.
|
2015-09-26 23:59:32 +02:00
|
|
|
try_err!(mkdir(&cx.dst), &cx.dst);
|
2016-08-15 13:52:38 -04:00
|
|
|
let _lock = flock::Lock::panicking_new(&cx.dst.join(".lock"), true, true, true);
|
2014-05-27 17:12:48 -07:00
|
|
|
|
|
|
|
// Add all the static files. These may already exist, but we just
|
|
|
|
// overwrite them anyway to make sure that they're fresh and up-to-date.
|
2016-03-13 15:36:01 +01:00
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("jquery.js"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/jquery-2.1.4.min.js"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("main.js"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/main.js"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("playpen.js"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/playpen.js"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("rustdoc.css"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/rustdoc.css"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("main.css"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/styles/main.css"))?;
|
2016-04-05 05:12:25 +02:00
|
|
|
if let Some(ref css) = cx.shared.css_file_extension {
|
2016-03-13 15:36:01 +01:00
|
|
|
let mut content = String::new();
|
|
|
|
let css = css.as_path();
|
|
|
|
let mut f = try_err!(File::open(css), css);
|
|
|
|
|
|
|
|
try_err!(f.read_to_string(&mut content), css);
|
|
|
|
let css = cx.dst.join("theme.css");
|
|
|
|
let css = css.as_path();
|
|
|
|
let mut f = try_err!(File::create(css), css);
|
|
|
|
try_err!(write!(f, "{}", &content), css);
|
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("normalize.css"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/normalize.css"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("FiraSans-Regular.woff"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/FiraSans-Regular.woff"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("FiraSans-Medium.woff"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/FiraSans-Medium.woff"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("FiraSans-LICENSE.txt"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/FiraSans-LICENSE.txt"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("Heuristica-Italic.woff"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/Heuristica-Italic.woff"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("Heuristica-LICENSE.txt"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/Heuristica-LICENSE.txt"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("SourceSerifPro-Regular.woff"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/SourceSerifPro-Regular.woff"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("SourceSerifPro-Bold.woff"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/SourceSerifPro-Bold.woff"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("SourceSerifPro-LICENSE.txt"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/SourceSerifPro-LICENSE.txt"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("SourceCodePro-Regular.woff"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/SourceCodePro-Regular.woff"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("SourceCodePro-Semibold.woff"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/SourceCodePro-Semibold.woff"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("SourceCodePro-LICENSE.txt"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/SourceCodePro-LICENSE.txt"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("LICENSE-MIT.txt"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/LICENSE-MIT.txt"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("LICENSE-APACHE.txt"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/LICENSE-APACHE.txt"))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write(cx.dst.join("COPYRIGHT.txt"),
|
2016-03-22 17:58:45 -05:00
|
|
|
include_bytes!("static/COPYRIGHT.txt"))?;
|
2014-05-27 17:12:48 -07:00
|
|
|
|
|
|
|
fn collect(path: &Path, krate: &str,
|
2015-02-26 21:00:43 -08:00
|
|
|
key: &str) -> io::Result<Vec<String>> {
|
2014-05-27 17:12:48 -07:00
|
|
|
let mut ret = Vec::new();
|
|
|
|
if path.exists() {
|
2016-03-22 22:01:37 -05:00
|
|
|
for line in BufReader::new(File::open(path)?).lines() {
|
|
|
|
let line = line?;
|
2014-11-27 14:19:27 -05:00
|
|
|
if !line.starts_with(key) {
|
2016-08-03 13:14:59 +12:00
|
|
|
continue;
|
2014-03-16 01:08:56 -07:00
|
|
|
}
|
2016-02-16 20:00:57 +01:00
|
|
|
if line.starts_with(&format!(r#"{}["{}"]"#, key, krate)) {
|
2016-08-03 13:14:59 +12:00
|
|
|
continue;
|
2014-05-27 17:12:48 -07:00
|
|
|
}
|
|
|
|
ret.push(line.to_string());
|
2014-01-30 11:30:21 -08:00
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2014-05-27 17:12:48 -07:00
|
|
|
return Ok(ret);
|
|
|
|
}
|
2014-05-21 16:41:58 -07:00
|
|
|
|
2014-05-27 17:12:48 -07:00
|
|
|
// Update the search index
|
|
|
|
let dst = cx.dst.join("search-index.js");
|
2015-09-26 23:59:32 +02:00
|
|
|
let all_indexes = try_err!(collect(&dst, &krate.name, "searchIndex"), &dst);
|
|
|
|
let mut w = try_err!(File::create(&dst), &dst);
|
|
|
|
try_err!(writeln!(&mut w, "var searchIndex = {{}};"), &dst);
|
|
|
|
try_err!(writeln!(&mut w, "{}", search_index), &dst);
|
2015-01-31 12:20:46 -05:00
|
|
|
for index in &all_indexes {
|
2015-09-26 23:59:32 +02:00
|
|
|
try_err!(writeln!(&mut w, "{}", *index), &dst);
|
2014-05-27 17:12:48 -07:00
|
|
|
}
|
2015-09-26 23:59:32 +02:00
|
|
|
try_err!(writeln!(&mut w, "initSearch(searchIndex);"), &dst);
|
2014-05-27 17:12:48 -07:00
|
|
|
|
|
|
|
// Update the list of all implementors for traits
|
|
|
|
let dst = cx.dst.join("implementors");
|
2015-09-26 23:59:32 +02:00
|
|
|
try_err!(mkdir(&dst), &dst);
|
2015-01-31 12:20:46 -05:00
|
|
|
for (&did, imps) in &cache.implementors {
|
2014-05-29 13:50:47 -07:00
|
|
|
// Private modules can leak through to this phase of rustdoc, which
|
|
|
|
// could contain implementations for otherwise private types. In some
|
|
|
|
// rare cases we could find an implementation for an item which wasn't
|
|
|
|
// indexed, so we just skip this step in that case.
|
|
|
|
//
|
|
|
|
// FIXME: this is a vague explanation for why this can't be a `get`, in
|
|
|
|
// theory it should be...
|
2014-11-06 12:25:16 -05:00
|
|
|
let &(ref remote_path, remote_item_type) = match cache.paths.get(&did) {
|
2014-05-29 13:50:47 -07:00
|
|
|
Some(p) => p,
|
2016-07-09 14:07:37 +01:00
|
|
|
None => match cache.external_paths.get(&did) {
|
|
|
|
Some(p) => p,
|
|
|
|
None => continue,
|
|
|
|
}
|
2014-05-29 13:50:47 -07:00
|
|
|
};
|
2014-05-27 17:12:48 -07:00
|
|
|
|
|
|
|
let mut mydst = dst.clone();
|
2015-01-31 12:20:46 -05:00
|
|
|
for part in &remote_path[..remote_path.len() - 1] {
|
2015-02-01 21:53:25 -05:00
|
|
|
mydst.push(part);
|
2015-09-26 23:59:32 +02:00
|
|
|
try_err!(mkdir(&mydst), &mydst);
|
2014-05-27 17:12:48 -07:00
|
|
|
}
|
2015-02-26 21:00:43 -08:00
|
|
|
mydst.push(&format!("{}.{}.js",
|
2016-08-03 13:14:59 +12:00
|
|
|
remote_item_type.css_class(),
|
2015-02-26 21:00:43 -08:00
|
|
|
remote_path[remote_path.len() - 1]));
|
2015-09-26 23:59:32 +02:00
|
|
|
let all_implementors = try_err!(collect(&mydst, &krate.name,
|
|
|
|
"implementors"),
|
|
|
|
&mydst);
|
2014-05-27 17:12:48 -07:00
|
|
|
|
2015-09-26 23:59:32 +02:00
|
|
|
try_err!(mkdir(mydst.parent().unwrap()),
|
|
|
|
&mydst.parent().unwrap().to_path_buf());
|
|
|
|
let mut f = BufWriter::new(try_err!(File::create(&mydst), &mydst));
|
|
|
|
try_err!(writeln!(&mut f, "(function() {{var implementors = {{}};"), &mydst);
|
2014-05-27 17:12:48 -07:00
|
|
|
|
2015-01-31 12:20:46 -05:00
|
|
|
for implementor in &all_implementors {
|
2015-09-26 23:59:32 +02:00
|
|
|
try_err!(write!(&mut f, "{}", *implementor), &mydst);
|
2014-03-16 01:08:56 -07:00
|
|
|
}
|
2014-05-21 16:41:58 -07:00
|
|
|
|
2016-05-09 10:18:06 +02:00
|
|
|
try_err!(write!(&mut f, r#"implementors["{}"] = ["#, krate.name), &mydst);
|
2015-01-31 12:20:46 -05:00
|
|
|
for imp in imps {
|
2014-06-01 11:30:53 -07:00
|
|
|
// If the trait and implementation are in the same crate, then
|
|
|
|
// there's no need to emit information about it (there's inlining
|
|
|
|
// going on). If they're in different crates then the crate defining
|
|
|
|
// the trait will be interested in our implementation.
|
|
|
|
if imp.def_id.krate == did.krate { continue }
|
2015-09-26 23:59:32 +02:00
|
|
|
try_err!(write!(&mut f, r#""{}","#, imp.impl_), &mydst);
|
2014-05-21 16:41:58 -07:00
|
|
|
}
|
2015-09-26 23:59:32 +02:00
|
|
|
try_err!(writeln!(&mut f, r"];"), &mydst);
|
|
|
|
try_err!(writeln!(&mut f, "{}", r"
|
2014-05-27 17:12:48 -07:00
|
|
|
if (window.register_implementors) {
|
|
|
|
window.register_implementors(implementors);
|
|
|
|
} else {
|
|
|
|
window.pending_implementors = implementors;
|
|
|
|
}
|
2015-09-26 23:59:32 +02:00
|
|
|
"), &mydst);
|
|
|
|
try_err!(writeln!(&mut f, r"}})()"), &mydst);
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2014-05-27 17:12:48 -07:00
|
|
|
Ok(())
|
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2016-04-02 09:03:55 +02:00
|
|
|
fn render_sources(dst: &Path, scx: &mut SharedContext,
|
2015-09-26 23:59:32 +02:00
|
|
|
krate: clean::Crate) -> Result<clean::Crate, Error> {
|
2014-05-27 17:12:48 -07:00
|
|
|
info!("emitting source files");
|
2016-04-02 09:03:55 +02:00
|
|
|
let dst = dst.join("src");
|
2015-09-26 23:59:32 +02:00
|
|
|
try_err!(mkdir(&dst), &dst);
|
2015-02-01 21:53:25 -05:00
|
|
|
let dst = dst.join(&krate.name);
|
2015-09-26 23:59:32 +02:00
|
|
|
try_err!(mkdir(&dst), &dst);
|
2014-05-27 17:12:48 -07:00
|
|
|
let mut folder = SourceCollector {
|
|
|
|
dst: dst,
|
2016-04-02 09:03:55 +02:00
|
|
|
scx: scx,
|
2014-05-27 17:12:48 -07:00
|
|
|
};
|
|
|
|
Ok(folder.fold_crate(krate))
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2013-10-03 10:24:40 -07:00
|
|
|
/// Writes the entire contents of a string to a destination, not attempting to
|
|
|
|
/// catch any errors.
|
2015-09-26 23:59:32 +02:00
|
|
|
fn write(dst: PathBuf, contents: &[u8]) -> Result<(), Error> {
|
|
|
|
Ok(try_err!(try_err!(File::create(&dst), &dst).write_all(contents), &dst))
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2016-04-24 21:37:22 -07:00
|
|
|
/// Makes a directory on the filesystem, failing the thread if an error occurs
|
|
|
|
/// and skipping if the directory already exists.
|
|
|
|
///
|
|
|
|
/// Note that this also handles races as rustdoc is likely to be run
|
|
|
|
/// concurrently against another invocation.
|
2015-02-26 21:00:43 -08:00
|
|
|
fn mkdir(path: &Path) -> io::Result<()> {
|
2016-04-24 21:37:22 -07:00
|
|
|
match fs::create_dir(path) {
|
|
|
|
Ok(()) => Ok(()),
|
|
|
|
Err(ref e) if e.kind() == io::ErrorKind::AlreadyExists => Ok(()),
|
|
|
|
Err(e) => Err(e)
|
2014-01-30 11:30:21 -08:00
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2014-12-03 23:57:45 +09:00
|
|
|
/// Returns a documentation-level item type from the item.
|
2016-08-03 13:14:59 +12:00
|
|
|
fn item_type(item: &clean::Item) -> ItemType {
|
2016-08-20 14:22:16 -04:00
|
|
|
ItemType::from(item)
|
2014-12-03 23:57:45 +09:00
|
|
|
}
|
|
|
|
|
2013-10-03 10:24:40 -07:00
|
|
|
/// Takes a path to a source file and cleans the path to it. This canonicalizes
|
2013-09-26 17:21:59 -07:00
|
|
|
/// things like ".." to components which preserve the "top down" hierarchy of a
|
2015-03-16 19:44:13 -07:00
|
|
|
/// static HTML tree. Each component in the cleaned path will be passed as an
|
|
|
|
/// argument to `f`. The very last component of the path (ie the file name) will
|
|
|
|
/// be passed to `f` if `keep_filename` is true, and ignored otherwise.
|
2013-09-26 17:21:59 -07:00
|
|
|
// FIXME (#9639): The closure should deal with &[u8] instead of &str
|
2014-12-01 20:46:51 +09:00
|
|
|
// FIXME (#9639): This is too conservative, rejecting non-UTF-8 paths
|
2015-03-16 19:44:13 -07:00
|
|
|
fn clean_srcpath<F>(src_root: &Path, p: &Path, keep_filename: bool, mut f: F) where
|
2014-12-09 15:22:19 -05:00
|
|
|
F: FnMut(&str),
|
|
|
|
{
|
2014-12-01 20:46:51 +09:00
|
|
|
// make it relative, if possible
|
2016-01-15 10:07:52 -08:00
|
|
|
let p = p.strip_prefix(src_root).unwrap_or(p);
|
2014-12-01 20:46:51 +09:00
|
|
|
|
2016-02-23 07:52:43 +01:00
|
|
|
let mut iter = p.components().peekable();
|
|
|
|
|
2015-03-16 19:44:13 -07:00
|
|
|
while let Some(c) = iter.next() {
|
|
|
|
if !keep_filename && iter.peek().is_none() {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-02-23 07:52:43 +01:00
|
|
|
match c {
|
|
|
|
Component::ParentDir => f("up"),
|
|
|
|
Component::Normal(c) => f(c.to_str().unwrap()),
|
|
|
|
_ => continue,
|
2013-09-27 15:12:23 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-03 10:24:40 -07:00
|
|
|
/// Attempts to find where an external crate is located, given that we're
|
|
|
|
/// rendering in to the specified source destination.
|
2013-10-02 15:39:32 -07:00
|
|
|
fn extern_location(e: &clean::ExternalCrate, dst: &Path) -> ExternalLocation {
|
|
|
|
// See if there's documentation generated into the local directory
|
2015-02-01 21:53:25 -05:00
|
|
|
let local_location = dst.join(&e.name);
|
2013-10-02 15:39:32 -07:00
|
|
|
if local_location.is_dir() {
|
|
|
|
return Local;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Failing that, see if there's an attribute specifying where to find this
|
|
|
|
// external crate
|
2016-03-22 20:26:33 +01:00
|
|
|
e.attrs.list("doc").value("html_root_url").map(|url| {
|
2016-02-28 10:12:41 +01:00
|
|
|
let mut url = url.to_owned();
|
|
|
|
if !url.ends_with("/") {
|
|
|
|
url.push('/')
|
2013-10-02 15:39:32 -07:00
|
|
|
}
|
2016-02-28 10:12:41 +01:00
|
|
|
Remote(url)
|
|
|
|
}).unwrap_or(Unknown) // Well, at least we tried.
|
2013-10-02 15:39:32 -07:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a> DocFolder for SourceCollector<'a> {
|
2013-09-27 15:12:23 -07:00
|
|
|
fn fold_item(&mut self, item: clean::Item) -> Option<clean::Item> {
|
2013-10-03 10:24:40 -07:00
|
|
|
// If we're including source files, and we haven't seen this file yet,
|
|
|
|
// then we need to render it out to the filesystem
|
2016-04-02 09:03:55 +02:00
|
|
|
if self.scx.include_sources
|
2016-02-27 07:35:05 +01:00
|
|
|
// skip all invalid spans
|
|
|
|
&& item.source.filename != ""
|
|
|
|
// macros from other libraries get special filenames which we can
|
|
|
|
// safely ignore
|
|
|
|
&& !(item.source.filename.starts_with("<")
|
|
|
|
&& item.source.filename.ends_with("macros>")) {
|
2013-10-03 10:24:40 -07:00
|
|
|
|
2013-09-30 12:58:18 -07:00
|
|
|
// If it turns out that we couldn't read this file, then we probably
|
|
|
|
// can't read any of the files (generating html output from json or
|
|
|
|
// something like that), so just don't include sources for the
|
|
|
|
// entire crate. The other option is maintaining this mapping on a
|
|
|
|
// per-file basis, but that's probably not worth it...
|
2016-04-02 09:03:55 +02:00
|
|
|
self.scx
|
2016-02-23 07:52:43 +01:00
|
|
|
.include_sources = match self.emit_source(&item.source.filename) {
|
2014-01-30 11:30:21 -08:00
|
|
|
Ok(()) => true,
|
|
|
|
Err(e) => {
|
|
|
|
println!("warning: source code was requested to be rendered, \
|
|
|
|
but processing `{}` had an error: {}",
|
|
|
|
item.source.filename, e);
|
|
|
|
println!(" skipping rendering of source code");
|
|
|
|
false
|
|
|
|
}
|
|
|
|
};
|
2013-09-27 15:12:23 -07:00
|
|
|
}
|
|
|
|
self.fold_item_recur(item)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a> SourceCollector<'a> {
|
2013-10-03 10:24:40 -07:00
|
|
|
/// Renders the given filename into its corresponding HTML source file.
|
2015-02-26 21:00:43 -08:00
|
|
|
fn emit_source(&mut self, filename: &str) -> io::Result<()> {
|
2015-03-18 09:14:54 -07:00
|
|
|
let p = PathBuf::from(filename);
|
2016-04-02 09:03:55 +02:00
|
|
|
if self.scx.local_sources.contains_key(&p) {
|
2016-02-27 07:35:05 +01:00
|
|
|
// We've already emitted this source
|
|
|
|
return Ok(());
|
|
|
|
}
|
2013-09-27 15:12:23 -07:00
|
|
|
|
2015-02-26 21:00:43 -08:00
|
|
|
let mut contents = Vec::new();
|
2016-03-22 22:01:37 -05:00
|
|
|
File::open(&p).and_then(|mut f| f.read_to_end(&mut contents))?;
|
2016-02-27 07:35:05 +01:00
|
|
|
|
2015-02-01 21:53:25 -05:00
|
|
|
let contents = str::from_utf8(&contents).unwrap();
|
2013-09-27 15:12:23 -07:00
|
|
|
|
2014-03-18 08:59:44 +08:00
|
|
|
// Remove the utf-8 BOM if any
|
2014-12-09 14:08:10 -08:00
|
|
|
let contents = if contents.starts_with("\u{feff}") {
|
2015-01-17 16:15:52 -08:00
|
|
|
&contents[3..]
|
2014-03-18 08:59:44 +08:00
|
|
|
} else {
|
2014-07-04 22:38:13 +02:00
|
|
|
contents
|
2014-03-18 08:59:44 +08:00
|
|
|
};
|
|
|
|
|
2013-09-27 15:12:23 -07:00
|
|
|
// Create the intermediate directories
|
|
|
|
let mut cur = self.dst.clone();
|
2015-06-08 16:55:35 +02:00
|
|
|
let mut root_path = String::from("../../");
|
2016-02-27 07:35:05 +01:00
|
|
|
let mut href = String::new();
|
2016-04-02 09:03:55 +02:00
|
|
|
clean_srcpath(&self.scx.src_root, &p, false, |component| {
|
2013-10-05 19:49:32 -07:00
|
|
|
cur.push(component);
|
2014-01-30 11:30:21 -08:00
|
|
|
mkdir(&cur).unwrap();
|
2013-09-27 15:12:23 -07:00
|
|
|
root_path.push_str("../");
|
2016-02-27 07:35:05 +01:00
|
|
|
href.push_str(component);
|
|
|
|
href.push('/');
|
2013-11-21 15:42:55 -08:00
|
|
|
});
|
2015-02-26 21:00:43 -08:00
|
|
|
let mut fname = p.file_name().expect("source has no filename")
|
|
|
|
.to_os_string();
|
2015-03-02 10:46:05 -08:00
|
|
|
fname.push(".html");
|
2016-08-03 13:14:59 +12:00
|
|
|
cur.push(&fname);
|
2016-02-27 07:35:05 +01:00
|
|
|
href.push_str(&fname.to_string_lossy());
|
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
let mut w = BufWriter::new(File::create(&cur)?);
|
2015-02-26 21:00:43 -08:00
|
|
|
let title = format!("{} -- source", cur.file_name().unwrap()
|
|
|
|
.to_string_lossy());
|
2014-08-04 13:53:33 -07:00
|
|
|
let desc = format!("Source to the Rust file `{}`.", filename);
|
2013-09-27 15:12:23 -07:00
|
|
|
let page = layout::Page {
|
2015-02-01 21:53:25 -05:00
|
|
|
title: &title,
|
2016-08-03 13:14:59 +12:00
|
|
|
css_class: "source",
|
2015-02-01 21:53:25 -05:00
|
|
|
root_path: &root_path,
|
|
|
|
description: &desc,
|
2016-02-28 14:01:43 +01:00
|
|
|
keywords: BASIC_KEYWORDS,
|
2013-09-27 15:12:23 -07:00
|
|
|
};
|
2016-04-02 09:03:55 +02:00
|
|
|
layout::render(&mut w, &self.scx.layout,
|
2016-03-13 15:36:01 +01:00
|
|
|
&page, &(""), &Source(contents),
|
2016-04-05 05:12:25 +02:00
|
|
|
self.scx.css_file_extension.is_some())?;
|
2016-03-22 22:01:37 -05:00
|
|
|
w.flush()?;
|
2016-04-02 09:03:55 +02:00
|
|
|
self.scx.local_sources.insert(p, href);
|
2016-02-27 07:35:05 +01:00
|
|
|
Ok(())
|
2013-09-27 15:12:23 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl DocFolder for Cache {
|
2013-09-18 22:18:38 -07:00
|
|
|
fn fold_item(&mut self, item: clean::Item) -> Option<clean::Item> {
|
2016-03-31 18:15:54 +02:00
|
|
|
// If this is a stripped module,
|
|
|
|
// we don't want it or its children in the search index.
|
|
|
|
let orig_stripped_mod = match item.inner {
|
|
|
|
clean::StrippedItem(box clean::ModuleItem(..)) => {
|
2016-04-22 17:53:15 +02:00
|
|
|
mem::replace(&mut self.stripped_mod, true)
|
2014-01-10 14:54:11 -08:00
|
|
|
}
|
2016-03-31 18:15:54 +02:00
|
|
|
_ => self.stripped_mod,
|
2014-01-10 14:54:11 -08:00
|
|
|
};
|
|
|
|
|
2016-04-22 17:53:15 +02:00
|
|
|
// Inlining can cause us to visit the same item multiple times.
|
|
|
|
// (i.e. relevant for gathering impls and implementors)
|
|
|
|
let orig_seen_mod = if item.is_mod() {
|
|
|
|
let seen_this = self.seen_mod || !self.seen_modules.insert(item.def_id);
|
|
|
|
mem::replace(&mut self.seen_mod, seen_this)
|
|
|
|
} else {
|
|
|
|
self.seen_mod
|
|
|
|
};
|
|
|
|
|
2013-09-18 22:18:38 -07:00
|
|
|
// Register any generics to their corresponding string. This is used
|
|
|
|
// when pretty-printing types
|
2016-08-20 13:36:52 -04:00
|
|
|
if let Some(generics) = item.inner.generics() {
|
|
|
|
self.generics(generics);
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2016-04-22 17:53:15 +02:00
|
|
|
if !self.seen_mod {
|
|
|
|
// Propagate a trait methods' documentation to all implementors of the
|
|
|
|
// trait
|
|
|
|
if let clean::TraitItem(ref t) = item.inner {
|
|
|
|
self.traits.insert(item.def_id, t.clone());
|
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2016-04-22 17:53:15 +02:00
|
|
|
// Collect all the implementors of traits.
|
|
|
|
if let clean::ImplItem(ref i) = item.inner {
|
|
|
|
if let Some(did) = i.trait_.def_id() {
|
|
|
|
self.implementors.entry(did).or_insert(vec![]).push(Implementor {
|
|
|
|
def_id: item.def_id,
|
|
|
|
stability: item.stability.clone(),
|
|
|
|
impl_: i.clone(),
|
|
|
|
});
|
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Index this method for searching later on
|
2014-11-29 16:41:21 -05:00
|
|
|
if let Some(ref s) = item.name {
|
|
|
|
let (parent, is_method) = match item.inner {
|
2016-03-31 18:15:54 +02:00
|
|
|
clean::StrippedItem(..) => ((None, None), false),
|
2016-02-23 10:24:53 +01:00
|
|
|
clean::AssociatedConstItem(..) |
|
|
|
|
clean::TypedefItem(_, true) if self.parent_is_trait_impl => {
|
|
|
|
// skip associated items in trait impls
|
2016-02-23 09:52:44 +01:00
|
|
|
((None, None), false)
|
|
|
|
}
|
2015-05-09 00:03:42 +02:00
|
|
|
clean::AssociatedTypeItem(..) |
|
|
|
|
clean::AssociatedConstItem(..) |
|
2014-11-29 16:41:21 -05:00
|
|
|
clean::TyMethodItem(..) |
|
|
|
|
clean::StructFieldItem(..) |
|
|
|
|
clean::VariantItem(..) => {
|
|
|
|
((Some(*self.parent_stack.last().unwrap()),
|
2015-01-19 11:07:13 -05:00
|
|
|
Some(&self.stack[..self.stack.len() - 1])),
|
2014-11-29 16:41:21 -05:00
|
|
|
false)
|
|
|
|
}
|
|
|
|
clean::MethodItem(..) => {
|
2015-03-24 16:53:34 -07:00
|
|
|
if self.parent_stack.is_empty() {
|
2014-11-29 16:41:21 -05:00
|
|
|
((None, None), false)
|
|
|
|
} else {
|
|
|
|
let last = self.parent_stack.last().unwrap();
|
|
|
|
let did = *last;
|
|
|
|
let path = match self.paths.get(&did) {
|
2015-04-06 19:21:52 -07:00
|
|
|
// The current stack not necessarily has correlation
|
|
|
|
// for where the type was defined. On the other
|
|
|
|
// hand, `paths` always has the right
|
|
|
|
// information if present.
|
2016-07-09 14:07:37 +01:00
|
|
|
Some(&(ref fqp, ItemType::Trait)) |
|
2014-12-03 23:57:45 +09:00
|
|
|
Some(&(ref fqp, ItemType::Struct)) |
|
2016-08-10 21:00:17 +03:00
|
|
|
Some(&(ref fqp, ItemType::Union)) |
|
2014-12-03 23:57:45 +09:00
|
|
|
Some(&(ref fqp, ItemType::Enum)) =>
|
2015-01-19 11:07:13 -05:00
|
|
|
Some(&fqp[..fqp.len() - 1]),
|
2015-02-01 21:53:25 -05:00
|
|
|
Some(..) => Some(&*self.stack),
|
2014-11-29 16:41:21 -05:00
|
|
|
None => None
|
|
|
|
};
|
|
|
|
((Some(*last), path), true)
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2014-11-29 16:41:21 -05:00
|
|
|
}
|
2015-02-01 21:53:25 -05:00
|
|
|
_ => ((None, Some(&*self.stack)), false)
|
2014-11-29 16:41:21 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
match parent {
|
2016-04-02 08:17:59 +02:00
|
|
|
(parent, Some(path)) if is_method || (!self.stripped_mod) => {
|
2016-03-31 18:15:54 +02:00
|
|
|
debug_assert!(!item.is_stripped());
|
2015-02-26 01:03:06 +02:00
|
|
|
|
2016-02-28 12:11:13 +01:00
|
|
|
// A crate has a module at its root, containing all items,
|
|
|
|
// which should not be indexed. The crate-item itself is
|
|
|
|
// inserted later on when serializing the search-index.
|
2015-12-21 03:21:56 +01:00
|
|
|
if item.def_id.index != CRATE_DEF_INDEX {
|
|
|
|
self.search_index.push(IndexItem {
|
2016-08-03 13:14:59 +12:00
|
|
|
ty: item_type(&item),
|
2015-12-21 03:21:56 +01:00
|
|
|
name: s.to_string(),
|
|
|
|
path: path.join("::").to_string(),
|
2016-02-13 11:27:53 +01:00
|
|
|
desc: Escape(&shorter(item.doc_value())).to_string(),
|
2015-12-21 03:21:56 +01:00
|
|
|
parent: parent,
|
2016-02-16 19:48:28 +01:00
|
|
|
parent_idx: None,
|
2016-05-08 21:19:29 +03:00
|
|
|
search_type: get_index_search_type(&item),
|
2015-12-21 03:21:56 +01:00
|
|
|
});
|
|
|
|
}
|
2014-11-29 16:41:21 -05:00
|
|
|
}
|
2016-07-09 14:07:37 +01:00
|
|
|
(Some(parent), None) if is_method => {
|
|
|
|
// We have a parent, but we don't know where they're
|
|
|
|
// defined yet. Wait for later to index this item.
|
|
|
|
self.orphan_methods.push((parent, item.clone()));
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2014-11-29 16:41:21 -05:00
|
|
|
_ => {}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Keep track of the fully qualified path for this item.
|
2016-02-28 12:11:13 +01:00
|
|
|
let pushed = match item.name {
|
|
|
|
Some(ref n) if !n.is_empty() => {
|
2014-05-25 03:17:19 -07:00
|
|
|
self.stack.push(n.to_string());
|
2013-09-18 22:18:38 -07:00
|
|
|
true
|
2016-02-28 12:11:13 +01:00
|
|
|
}
|
|
|
|
_ => false,
|
|
|
|
};
|
|
|
|
|
2013-09-18 22:18:38 -07:00
|
|
|
match item.inner {
|
2013-11-28 12:22:53 -08:00
|
|
|
clean::StructItem(..) | clean::EnumItem(..) |
|
|
|
|
clean::TypedefItem(..) | clean::TraitItem(..) |
|
|
|
|
clean::FunctionItem(..) | clean::ModuleItem(..) |
|
2016-06-03 22:59:45 +01:00
|
|
|
clean::ForeignFunctionItem(..) | clean::ForeignStaticItem(..) |
|
2016-08-10 21:00:17 +03:00
|
|
|
clean::ConstantItem(..) | clean::StaticItem(..) |
|
|
|
|
clean::UnionItem(..)
|
2016-06-03 22:59:45 +01:00
|
|
|
if !self.stripped_mod => {
|
2014-05-22 22:00:18 -07:00
|
|
|
// Reexported items mean that the same id can show up twice
|
|
|
|
// in the rustdoc ast that we're looking at. We know,
|
|
|
|
// however, that a reexported item doesn't show up in the
|
|
|
|
// `public_items` map, so we can skip inserting into the
|
|
|
|
// paths map if there was already an entry present and we're
|
|
|
|
// not a public item.
|
2015-09-04 13:52:28 -04:00
|
|
|
if
|
|
|
|
!self.paths.contains_key(&item.def_id) ||
|
2015-11-19 14:16:35 +03:00
|
|
|
self.access_levels.is_public(item.def_id)
|
2015-09-04 13:52:28 -04:00
|
|
|
{
|
2014-05-22 22:00:18 -07:00
|
|
|
self.paths.insert(item.def_id,
|
2016-08-03 13:14:59 +12:00
|
|
|
(self.stack.clone(), item_type(&item)));
|
2014-02-16 23:11:09 -08:00
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2014-02-16 14:35:13 -08:00
|
|
|
// link variants to their parent enum because pages aren't emitted
|
|
|
|
// for each variant
|
2016-03-31 18:15:54 +02:00
|
|
|
clean::VariantItem(..) if !self.stripped_mod => {
|
2014-02-16 14:35:13 -08:00
|
|
|
let mut stack = self.stack.clone();
|
|
|
|
stack.pop();
|
2014-12-03 23:57:45 +09:00
|
|
|
self.paths.insert(item.def_id, (stack, ItemType::Enum));
|
2014-02-16 14:35:13 -08:00
|
|
|
}
|
2014-06-03 17:55:30 -07:00
|
|
|
|
|
|
|
clean::PrimitiveItem(..) if item.visibility.is_some() => {
|
|
|
|
self.paths.insert(item.def_id, (self.stack.clone(),
|
2016-08-03 13:14:59 +12:00
|
|
|
item_type(&item)));
|
2014-06-03 17:55:30 -07:00
|
|
|
}
|
|
|
|
|
2013-09-18 22:18:38 -07:00
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Maintain the parent stack
|
2016-02-23 09:52:44 +01:00
|
|
|
let orig_parent_is_trait_impl = self.parent_is_trait_impl;
|
2013-09-18 22:18:38 -07:00
|
|
|
let parent_pushed = match item.inner {
|
2016-08-10 21:00:17 +03:00
|
|
|
clean::TraitItem(..) | clean::EnumItem(..) |
|
|
|
|
clean::StructItem(..) | clean::UnionItem(..) => {
|
2014-05-22 22:00:18 -07:00
|
|
|
self.parent_stack.push(item.def_id);
|
2016-02-23 09:52:44 +01:00
|
|
|
self.parent_is_trait_impl = false;
|
2014-05-03 02:08:58 -07:00
|
|
|
true
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
clean::ImplItem(ref i) => {
|
2016-02-23 09:52:44 +01:00
|
|
|
self.parent_is_trait_impl = i.trait_.is_some();
|
2013-09-18 22:18:38 -07:00
|
|
|
match i.for_ {
|
2014-05-09 13:52:17 -07:00
|
|
|
clean::ResolvedPath{ did, .. } => {
|
2014-05-22 22:00:18 -07:00
|
|
|
self.parent_stack.push(did);
|
|
|
|
true
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2015-04-07 17:35:23 -07:00
|
|
|
ref t => {
|
|
|
|
match t.primitive_type() {
|
|
|
|
Some(prim) => {
|
2015-09-17 14:29:59 -04:00
|
|
|
let did = DefId::local(prim.to_def_index());
|
2015-04-07 17:35:23 -07:00
|
|
|
self.parent_stack.push(did);
|
|
|
|
true
|
|
|
|
}
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => false
|
|
|
|
};
|
|
|
|
|
|
|
|
// Once we've recursively found all the generics, then hoard off all the
|
|
|
|
// implementations elsewhere
|
2016-02-28 12:11:13 +01:00
|
|
|
let ret = self.fold_item_recur(item).and_then(|item| {
|
2016-05-03 19:15:59 +02:00
|
|
|
if let clean::Item { inner: clean::ImplItem(_), .. } = item {
|
2016-02-28 12:11:13 +01:00
|
|
|
// Figure out the id of this impl. This may map to a
|
|
|
|
// primitive rather than always to a struct/enum.
|
2016-05-03 19:15:59 +02:00
|
|
|
// Note: matching twice to restrict the lifetime of the `i` borrow.
|
|
|
|
let did = if let clean::Item { inner: clean::ImplItem(ref i), .. } = item {
|
|
|
|
match i.for_ {
|
|
|
|
clean::ResolvedPath { did, .. } |
|
|
|
|
clean::BorrowedRef {
|
|
|
|
type_: box clean::ResolvedPath { did, .. }, ..
|
|
|
|
} => {
|
|
|
|
Some(did)
|
|
|
|
}
|
|
|
|
ref t => {
|
|
|
|
t.primitive_type().and_then(|t| {
|
|
|
|
self.primitive_locations.get(&t).map(|n| {
|
|
|
|
let id = t.to_def_index();
|
|
|
|
DefId { krate: *n, index: id }
|
|
|
|
})
|
2016-02-28 12:11:13 +01:00
|
|
|
})
|
2016-05-03 19:15:59 +02:00
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2016-05-03 19:15:59 +02:00
|
|
|
} else {
|
|
|
|
unreachable!()
|
2016-02-28 12:11:13 +01:00
|
|
|
};
|
2016-04-22 17:53:15 +02:00
|
|
|
if !self.seen_mod {
|
|
|
|
if let Some(did) = did {
|
|
|
|
self.impls.entry(did).or_insert(vec![]).push(Impl {
|
2016-05-03 19:15:59 +02:00
|
|
|
impl_item: item,
|
2016-04-22 17:53:15 +02:00
|
|
|
});
|
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2016-02-28 12:11:13 +01:00
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(item)
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2016-02-28 12:11:13 +01:00
|
|
|
});
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2013-12-23 16:20:52 +01:00
|
|
|
if pushed { self.stack.pop().unwrap(); }
|
|
|
|
if parent_pushed { self.parent_stack.pop().unwrap(); }
|
2016-04-22 17:53:15 +02:00
|
|
|
self.seen_mod = orig_seen_mod;
|
2016-03-31 18:15:54 +02:00
|
|
|
self.stripped_mod = orig_stripped_mod;
|
2016-02-23 09:52:44 +01:00
|
|
|
self.parent_is_trait_impl = orig_parent_is_trait_impl;
|
2013-09-18 22:18:38 -07:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a> Cache {
|
2013-09-18 22:18:38 -07:00
|
|
|
fn generics(&mut self, generics: &clean::Generics) {
|
2015-01-31 12:20:46 -05:00
|
|
|
for typ in &generics.type_params {
|
2014-05-03 02:08:58 -07:00
|
|
|
self.typarams.insert(typ.did, typ.name.clone());
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Context {
|
2013-10-03 10:24:40 -07:00
|
|
|
/// Recurse in the directory structure and change the "root path" to make
|
|
|
|
/// sure it always points to the top (relatively)
|
2014-12-09 15:22:19 -05:00
|
|
|
fn recurse<T, F>(&mut self, s: String, f: F) -> T where
|
|
|
|
F: FnOnce(&mut Context) -> T,
|
|
|
|
{
|
2015-03-24 16:53:34 -07:00
|
|
|
if s.is_empty() {
|
2014-12-20 00:09:35 -08:00
|
|
|
panic!("Unexpected empty destination: {:?}", self.current);
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2013-09-26 17:21:59 -07:00
|
|
|
let prev = self.dst.clone();
|
2015-02-01 21:53:25 -05:00
|
|
|
self.dst.push(&s);
|
2013-09-18 22:18:38 -07:00
|
|
|
self.root_path.push_str("../");
|
|
|
|
self.current.push(s);
|
|
|
|
|
2013-12-17 11:19:14 -05:00
|
|
|
info!("Recursing into {}", self.dst.display());
|
|
|
|
|
2013-09-18 22:18:38 -07:00
|
|
|
let ret = f(self);
|
|
|
|
|
2013-12-17 11:19:14 -05:00
|
|
|
info!("Recursed; leaving {}", self.dst.display());
|
|
|
|
|
2013-09-18 22:18:38 -07:00
|
|
|
// Go back to where we were at
|
|
|
|
self.dst = prev;
|
|
|
|
let len = self.root_path.len();
|
|
|
|
self.root_path.truncate(len - 3);
|
2013-12-23 16:20:52 +01:00
|
|
|
self.current.pop().unwrap();
|
2013-09-18 22:18:38 -07:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-12-05 18:19:06 -08:00
|
|
|
/// Main method for rendering a crate.
|
|
|
|
///
|
|
|
|
/// This currently isn't parallelized, but it'd be pretty easy to add
|
|
|
|
/// parallelization to this function.
|
2015-09-26 23:59:32 +02:00
|
|
|
fn krate(self, mut krate: clean::Crate) -> Result<(), Error> {
|
2014-02-05 22:15:24 +01:00
|
|
|
let mut item = match krate.module.take() {
|
2013-09-18 22:18:38 -07:00
|
|
|
Some(i) => i,
|
2014-01-30 11:30:21 -08:00
|
|
|
None => return Ok(())
|
2013-09-18 22:18:38 -07:00
|
|
|
};
|
2014-02-05 22:15:24 +01:00
|
|
|
item.name = Some(krate.name);
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2014-07-04 00:51:46 -07:00
|
|
|
// render the crate documentation
|
2014-03-05 15:28:08 -08:00
|
|
|
let mut work = vec!((self, item));
|
2014-07-04 00:51:46 -07:00
|
|
|
|
2016-03-31 18:15:54 +02:00
|
|
|
while let Some((mut cx, item)) = work.pop() {
|
|
|
|
cx.item(item, |cx, item| {
|
|
|
|
work.push((cx.clone(), item))
|
|
|
|
})?
|
|
|
|
}
|
2014-01-30 11:30:21 -08:00
|
|
|
Ok(())
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2016-08-03 13:14:59 +12:00
|
|
|
fn render_item(&self,
|
|
|
|
writer: &mut io::Write,
|
|
|
|
it: &clean::Item,
|
|
|
|
pushname: bool)
|
|
|
|
-> io::Result<()> {
|
|
|
|
// A little unfortunate that this is done like this, but it sure
|
|
|
|
// does make formatting *a lot* nicer.
|
|
|
|
CURRENT_LOCATION_KEY.with(|slot| {
|
|
|
|
*slot.borrow_mut() = self.current.clone();
|
|
|
|
});
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2016-08-03 13:14:59 +12:00
|
|
|
let mut title = if it.is_primitive() {
|
|
|
|
// No need to include the namespace for primitive types
|
|
|
|
String::new()
|
|
|
|
} else {
|
|
|
|
self.current.join("::")
|
|
|
|
};
|
|
|
|
if pushname {
|
|
|
|
if !title.is_empty() {
|
|
|
|
title.push_str("::");
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2016-08-03 13:14:59 +12:00
|
|
|
title.push_str(it.name.as_ref().unwrap());
|
|
|
|
}
|
|
|
|
title.push_str(" - Rust");
|
|
|
|
let tyname = item_type(it).css_class();
|
|
|
|
let desc = if it.is_crate() {
|
|
|
|
format!("API documentation for the Rust `{}` crate.",
|
|
|
|
self.shared.layout.krate)
|
|
|
|
} else {
|
|
|
|
format!("API documentation for the Rust `{}` {} in crate `{}`.",
|
|
|
|
it.name.as_ref().unwrap(), tyname, self.shared.layout.krate)
|
|
|
|
};
|
|
|
|
let keywords = make_item_keywords(it);
|
|
|
|
let page = layout::Page {
|
|
|
|
css_class: tyname,
|
|
|
|
root_path: &self.root_path,
|
|
|
|
title: &title,
|
|
|
|
description: &desc,
|
|
|
|
keywords: &keywords,
|
|
|
|
};
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2016-08-03 13:14:59 +12:00
|
|
|
reset_ids(true);
|
2014-03-04 11:24:20 -08:00
|
|
|
|
2016-08-03 13:14:59 +12:00
|
|
|
if !self.render_redirect_pages {
|
|
|
|
layout::render(writer, &self.shared.layout, &page,
|
|
|
|
&Sidebar{ cx: self, item: it },
|
|
|
|
&Item{ cx: self, item: it },
|
|
|
|
self.shared.css_file_extension.is_some())?;
|
|
|
|
} else {
|
|
|
|
let mut url = repeat("../").take(self.current.len())
|
|
|
|
.collect::<String>();
|
|
|
|
if let Some(&(ref names, ty)) = cache().paths.get(&it.def_id) {
|
|
|
|
for name in &names[..names.len() - 1] {
|
|
|
|
url.push_str(name);
|
|
|
|
url.push_str("/");
|
2014-05-29 13:50:47 -07:00
|
|
|
}
|
2016-08-03 13:14:59 +12:00
|
|
|
url.push_str(&item_path(ty, names.last().unwrap()));
|
|
|
|
layout::redirect(writer, &url)?;
|
2014-05-29 13:50:47 -07:00
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2016-08-03 13:14:59 +12:00
|
|
|
Ok(())
|
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2016-08-03 13:14:59 +12:00
|
|
|
/// Non-parallelized version of rendering an item. This will take the input
|
|
|
|
/// item, render its contents, and then invoke the specified closure with
|
|
|
|
/// all sub-items which need to be rendered.
|
|
|
|
///
|
|
|
|
/// The rendering driver uses this closure to queue up more work.
|
|
|
|
fn item<F>(&mut self, item: clean::Item, mut f: F) -> Result<(), Error> where
|
|
|
|
F: FnMut(&mut Context, clean::Item),
|
|
|
|
{
|
2016-03-31 18:15:54 +02:00
|
|
|
// Stripped modules survive the rustdoc passes (i.e. `strip-private`)
|
|
|
|
// if they contain impls for public types. These modules can also
|
2014-06-03 17:55:30 -07:00
|
|
|
// contain items such as publicly reexported structures.
|
|
|
|
//
|
|
|
|
// External crates will provide links to these structures, so
|
2016-03-31 18:15:54 +02:00
|
|
|
// these modules are recursed into, but not rendered normally
|
|
|
|
// (a flag on the context).
|
2014-06-03 17:55:30 -07:00
|
|
|
if !self.render_redirect_pages {
|
2016-08-20 14:22:08 -04:00
|
|
|
self.render_redirect_pages = maybe_ignore_item(&item);
|
2014-06-03 17:55:30 -07:00
|
|
|
}
|
|
|
|
|
2016-03-31 18:15:54 +02:00
|
|
|
if item.is_mod() {
|
2013-09-24 13:56:52 -07:00
|
|
|
// modules are special because they add a namespace. We also need to
|
|
|
|
// recurse into the items of the module as well.
|
2016-03-31 18:15:54 +02:00
|
|
|
let name = item.name.as_ref().unwrap().to_string();
|
|
|
|
let mut item = Some(item);
|
|
|
|
self.recurse(name, |this| {
|
|
|
|
let item = item.take().unwrap();
|
2016-06-02 16:16:21 +01:00
|
|
|
|
|
|
|
let mut buf = Vec::new();
|
2016-08-03 13:14:59 +12:00
|
|
|
this.render_item(&mut buf, &item, false).unwrap();
|
2016-06-02 16:16:21 +01:00
|
|
|
// buf will be empty if the module is stripped and there is no redirect for it
|
|
|
|
if !buf.is_empty() {
|
|
|
|
let joint_dst = this.dst.join("index.html");
|
|
|
|
try_err!(fs::create_dir_all(&this.dst), &this.dst);
|
|
|
|
let mut dst = try_err!(File::create(&joint_dst), &joint_dst);
|
|
|
|
try_err!(dst.write_all(&buf), &joint_dst);
|
|
|
|
}
|
2015-03-05 16:35:43 +09:00
|
|
|
|
2016-03-31 18:15:54 +02:00
|
|
|
let m = match item.inner {
|
|
|
|
clean::StrippedItem(box clean::ModuleItem(m)) |
|
|
|
|
clean::ModuleItem(m) => m,
|
|
|
|
_ => unreachable!()
|
|
|
|
};
|
2013-09-24 13:56:52 -07:00
|
|
|
|
2016-08-03 13:14:59 +12:00
|
|
|
// Render sidebar-items.js used throughout this module.
|
2016-06-02 16:16:21 +01:00
|
|
|
if !this.render_redirect_pages {
|
2016-03-31 18:15:54 +02:00
|
|
|
let items = this.build_sidebar_items(&m);
|
|
|
|
let js_dst = this.dst.join("sidebar-items.js");
|
|
|
|
let mut js_out = BufWriter::new(try_err!(File::create(&js_dst), &js_dst));
|
|
|
|
try_err!(write!(&mut js_out, "initSidebarItems({});",
|
|
|
|
as_json(&items)), &js_dst);
|
|
|
|
}
|
2015-09-26 23:59:32 +02:00
|
|
|
|
2016-03-31 18:15:54 +02:00
|
|
|
for item in m.items {
|
|
|
|
f(this,item);
|
|
|
|
}
|
2016-08-03 13:14:59 +12:00
|
|
|
|
2015-09-26 23:59:32 +02:00
|
|
|
Ok(())
|
2016-08-03 13:14:59 +12:00
|
|
|
})?;
|
2016-03-31 18:15:54 +02:00
|
|
|
} else if item.name.is_some() {
|
2016-06-02 16:16:21 +01:00
|
|
|
let mut buf = Vec::new();
|
2016-08-03 13:14:59 +12:00
|
|
|
self.render_item(&mut buf, &item, true).unwrap();
|
2016-06-02 16:16:21 +01:00
|
|
|
// buf will be empty if the item is stripped and there is no redirect for it
|
|
|
|
if !buf.is_empty() {
|
2016-08-03 13:56:18 +12:00
|
|
|
let name = item.name.as_ref().unwrap();
|
|
|
|
let item_type = item_type(&item);
|
|
|
|
let file_name = &item_path(item_type, name);
|
|
|
|
let joint_dst = self.dst.join(file_name);
|
2016-06-02 16:16:21 +01:00
|
|
|
try_err!(fs::create_dir_all(&self.dst), &self.dst);
|
|
|
|
let mut dst = try_err!(File::create(&joint_dst), &joint_dst);
|
|
|
|
try_err!(dst.write_all(&buf), &joint_dst);
|
2016-08-03 13:56:18 +12:00
|
|
|
|
|
|
|
// Redirect from a sane URL using the namespace to Rustdoc's
|
|
|
|
// URL for the page.
|
|
|
|
let redir_name = format!("{}.{}.html", name, item_type.name_space());
|
|
|
|
let redir_dst = self.dst.join(redir_name);
|
|
|
|
if let Ok(mut redirect_out) = OpenOptions::new().create_new(true)
|
|
|
|
.write(true)
|
|
|
|
.open(&redir_dst) {
|
|
|
|
try_err!(layout::redirect(&mut redirect_out, file_name), &redir_dst);
|
|
|
|
}
|
2016-08-16 14:25:12 +12:00
|
|
|
|
|
|
|
// If the item is a macro, redirect from the old macro URL (with !)
|
|
|
|
// to the new one (without).
|
|
|
|
// FIXME(#35705) remove this redirect.
|
|
|
|
if item_type == ItemType::Macro {
|
|
|
|
let redir_name = format!("{}.{}!.html", item_type, name);
|
|
|
|
let redir_dst = self.dst.join(redir_name);
|
|
|
|
let mut redirect_out = try_err!(File::create(&redir_dst), &redir_dst);
|
|
|
|
try_err!(layout::redirect(&mut redirect_out, file_name), &redir_dst);
|
|
|
|
}
|
2016-06-02 16:16:21 +01:00
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2016-08-03 13:14:59 +12:00
|
|
|
Ok(())
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2014-11-09 19:09:17 -05:00
|
|
|
|
2015-04-16 12:14:45 +02:00
|
|
|
fn build_sidebar_items(&self, m: &clean::Module) -> BTreeMap<String, Vec<NameDoc>> {
|
|
|
|
// BTreeMap instead of HashMap to get a sorted output
|
|
|
|
let mut map = BTreeMap::new();
|
2015-01-31 12:20:46 -05:00
|
|
|
for item in &m.items {
|
2016-08-20 14:22:08 -04:00
|
|
|
if maybe_ignore_item(item) { continue }
|
2014-11-09 19:09:17 -05:00
|
|
|
|
2016-08-03 13:14:59 +12:00
|
|
|
let short = item_type(item).css_class();
|
2014-11-09 19:09:17 -05:00
|
|
|
let myname = match item.name {
|
|
|
|
None => continue,
|
|
|
|
Some(ref s) => s.to_string(),
|
|
|
|
};
|
2015-01-04 14:07:32 -05:00
|
|
|
let short = short.to_string();
|
2015-03-20 13:43:01 -04:00
|
|
|
map.entry(short).or_insert(vec![])
|
2015-03-01 09:42:11 -05:00
|
|
|
.push((myname, Some(plain_summary_line(item.doc_value()))));
|
2014-11-09 19:09:17 -05:00
|
|
|
}
|
|
|
|
|
2015-01-31 20:02:00 -05:00
|
|
|
for (_, items) in &mut map {
|
2014-11-27 16:49:29 -05:00
|
|
|
items.sort();
|
2014-11-09 19:09:17 -05:00
|
|
|
}
|
|
|
|
return map;
|
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a> Item<'a> {
|
2014-05-24 11:56:38 -07:00
|
|
|
/// Generate a url appropriate for an `href` attribute back to the source of
|
|
|
|
/// this item.
|
|
|
|
///
|
|
|
|
/// The url generated, when clicked, will redirect the browser back to the
|
|
|
|
/// original source code.
|
|
|
|
///
|
|
|
|
/// If `None` is returned, then a source link couldn't be generated. This
|
|
|
|
/// may happen, for example, with externally inlined items where the source
|
|
|
|
/// of their crate documentation isn't known.
|
2016-02-27 07:35:05 +01:00
|
|
|
fn href(&self) -> Option<String> {
|
2015-04-13 15:25:40 -07:00
|
|
|
let href = if self.item.source.loline == self.item.source.hiline {
|
|
|
|
format!("{}", self.item.source.loline)
|
|
|
|
} else {
|
|
|
|
format!("{}-{}", self.item.source.loline, self.item.source.hiline)
|
|
|
|
};
|
|
|
|
|
|
|
|
// First check to see if this is an imported macro source. In this case
|
|
|
|
// we need to handle it specially as cross-crate inlined macros have...
|
|
|
|
// odd locations!
|
|
|
|
let imported_macro_from = match self.item.inner {
|
|
|
|
clean::MacroItem(ref m) => m.imported_from.as_ref(),
|
|
|
|
_ => None,
|
|
|
|
};
|
|
|
|
if let Some(krate) = imported_macro_from {
|
|
|
|
let cache = cache();
|
|
|
|
let root = cache.extern_locations.values().find(|&&(ref n, _)| {
|
|
|
|
*krate == *n
|
|
|
|
}).map(|l| &l.1);
|
|
|
|
let root = match root {
|
|
|
|
Some(&Remote(ref s)) => s.to_string(),
|
|
|
|
Some(&Local) => self.cx.root_path.clone(),
|
|
|
|
None | Some(&Unknown) => return None,
|
|
|
|
};
|
|
|
|
Some(format!("{root}/{krate}/macro.{name}.html?gotomacrosrc=1",
|
|
|
|
root = root,
|
|
|
|
krate = krate,
|
|
|
|
name = self.item.name.as_ref().unwrap()))
|
|
|
|
|
2014-05-24 11:56:38 -07:00
|
|
|
// If this item is part of the local crate, then we're guaranteed to
|
|
|
|
// know the span, so we plow forward and generate a proper url. The url
|
|
|
|
// has anchors for the line numbers that we're linking to.
|
2015-08-16 06:32:28 -04:00
|
|
|
} else if self.item.def_id.is_local() {
|
2016-04-02 09:03:55 +02:00
|
|
|
let path = PathBuf::from(&self.item.source.filename);
|
|
|
|
self.cx.shared.local_sources.get(&path).map(|path| {
|
2016-03-08 08:07:57 +01:00
|
|
|
format!("{root}src/{krate}/{path}#{href}",
|
2016-02-27 07:35:05 +01:00
|
|
|
root = self.cx.root_path,
|
2016-04-02 09:03:55 +02:00
|
|
|
krate = self.cx.shared.layout.krate,
|
2016-02-27 07:35:05 +01:00
|
|
|
path = path,
|
|
|
|
href = href)
|
|
|
|
})
|
2014-05-24 11:56:38 -07:00
|
|
|
// If this item is not part of the local crate, then things get a little
|
|
|
|
// trickier. We don't actually know the span of the external item, but
|
|
|
|
// we know that the documentation on the other end knows the span!
|
|
|
|
//
|
|
|
|
// In this case, we generate a link to the *documentation* for this type
|
|
|
|
// in the original crate. There's an extra URL parameter which says that
|
|
|
|
// we want to go somewhere else, and the JS on the destination page will
|
|
|
|
// pick it up and instantly redirect the browser to the source code.
|
|
|
|
//
|
|
|
|
// If we don't know where the external documentation for this crate is
|
|
|
|
// located, then we return `None`.
|
2014-05-02 17:08:11 -07:00
|
|
|
} else {
|
2014-11-14 14:20:57 -08:00
|
|
|
let cache = cache();
|
2016-06-18 18:41:13 +01:00
|
|
|
let external_path = match cache.external_paths.get(&self.item.def_id) {
|
2016-07-09 14:07:37 +01:00
|
|
|
Some(&(ref path, _)) => path,
|
2016-05-03 19:15:59 +02:00
|
|
|
None => return None,
|
|
|
|
};
|
2016-06-18 18:41:13 +01:00
|
|
|
let mut path = match cache.extern_locations.get(&self.item.def_id.krate) {
|
2016-05-03 19:15:59 +02:00
|
|
|
Some(&(_, Remote(ref s))) => s.to_string(),
|
|
|
|
Some(&(_, Local)) => self.cx.root_path.clone(),
|
|
|
|
Some(&(_, Unknown)) => return None,
|
|
|
|
None => return None,
|
2014-05-23 20:17:27 -07:00
|
|
|
};
|
2016-06-18 18:41:13 +01:00
|
|
|
for item in &external_path[..external_path.len() - 1] {
|
|
|
|
path.push_str(item);
|
|
|
|
path.push_str("/");
|
|
|
|
}
|
|
|
|
Some(format!("{path}{file}?gotosrc={goto}",
|
|
|
|
path = path,
|
2016-08-03 13:14:59 +12:00
|
|
|
file = item_path(item_type(self.item), external_path.last().unwrap()),
|
2015-09-17 14:29:59 -04:00
|
|
|
goto = self.item.def_id.index.as_usize()))
|
2014-05-23 20:17:27 -07:00
|
|
|
}
|
2014-05-02 17:08:11 -07:00
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2015-01-20 15:45:07 -08:00
|
|
|
impl<'a> fmt::Display for Item<'a> {
|
2014-02-05 23:55:13 +11:00
|
|
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
2016-03-31 18:15:54 +02:00
|
|
|
debug_assert!(!self.item.is_stripped());
|
2013-09-18 22:18:38 -07:00
|
|
|
// Write the breadcrumb trail header for the top
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "\n<h1 class='fqn'><span class='in-band'>")?;
|
2014-02-05 23:55:13 +11:00
|
|
|
match self.item.inner {
|
2014-02-28 22:33:45 +01:00
|
|
|
clean::ModuleItem(ref m) => if m.is_crate {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "Crate ")?;
|
2014-02-28 22:33:45 +01:00
|
|
|
} else {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "Module ")?;
|
2014-02-28 22:33:45 +01:00
|
|
|
},
|
2016-03-22 22:01:37 -05:00
|
|
|
clean::FunctionItem(..) => write!(fmt, "Function ")?,
|
|
|
|
clean::TraitItem(..) => write!(fmt, "Trait ")?,
|
|
|
|
clean::StructItem(..) => write!(fmt, "Struct ")?,
|
2016-08-10 21:00:17 +03:00
|
|
|
clean::UnionItem(..) => write!(fmt, "Union ")?,
|
2016-03-22 22:01:37 -05:00
|
|
|
clean::EnumItem(..) => write!(fmt, "Enum ")?,
|
|
|
|
clean::PrimitiveItem(..) => write!(fmt, "Primitive Type ")?,
|
2013-09-18 22:18:38 -07:00
|
|
|
_ => {}
|
|
|
|
}
|
2016-06-30 23:16:44 +01:00
|
|
|
if !self.item.is_primitive() {
|
2015-02-01 21:53:25 -05:00
|
|
|
let cur = &self.cx.current;
|
2016-03-31 18:15:54 +02:00
|
|
|
let amt = if self.item.is_mod() { cur.len() - 1 } else { cur.len() };
|
2014-05-28 19:53:37 -07:00
|
|
|
for (i, component) in cur.iter().enumerate().take(amt) {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "<a href='{}index.html'>{}</a>::<wbr>",
|
2016-03-22 17:58:45 -05:00
|
|
|
repeat("../").take(cur.len() - i - 1)
|
|
|
|
.collect::<String>(),
|
|
|
|
component)?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "<a class='{}' href=''>{}</a>",
|
2016-08-03 13:14:59 +12:00
|
|
|
item_type(self.item), self.item.name.as_ref().unwrap())?;
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "</span>")?; // in-band
|
|
|
|
write!(fmt, "<span class='out-of-band'>")?;
|
2016-05-10 21:01:10 +02:00
|
|
|
if let Some(version) = self.item.stable_since() {
|
2016-05-18 03:54:52 +02:00
|
|
|
write!(fmt, "<span class='since' title='Stable since Rust version {0}'>{0}</span>",
|
|
|
|
version)?;
|
2016-05-10 21:01:10 +02:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt,
|
2016-03-22 17:58:45 -05:00
|
|
|
r##"<span id='render-detail'>
|
|
|
|
<a id="toggle-all-docs" href="javascript:void(0)" title="collapse all docs">
|
|
|
|
[<span class='inner'>−</span>]
|
|
|
|
</a>
|
|
|
|
</span>"##)?;
|
2014-07-30 21:31:34 -04:00
|
|
|
|
2014-04-25 17:34:32 +09:00
|
|
|
// Write `src` tag
|
2014-05-24 11:56:38 -07:00
|
|
|
//
|
|
|
|
// When this item is part of a `pub use` in a downstream crate, the
|
|
|
|
// [src] link in the downstream documentation will actually come back to
|
|
|
|
// this page, and this link will be auto-clicked. The `id` attribute is
|
|
|
|
// used to find the link to auto-click.
|
2016-06-30 23:16:44 +01:00
|
|
|
if self.cx.shared.include_sources && !self.item.is_primitive() {
|
2016-02-28 12:11:13 +01:00
|
|
|
if let Some(l) = self.href() {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "<a id='src-{}' class='srclink' \
|
2016-03-22 17:58:45 -05:00
|
|
|
href='{}' title='{}'>[src]</a>",
|
|
|
|
self.item.def_id.index.as_usize(), l, "goto source code")?;
|
2014-05-23 20:17:27 -07:00
|
|
|
}
|
2014-04-12 21:39:12 +02:00
|
|
|
}
|
2014-07-04 00:51:46 -07:00
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "</span>")?; // out-of-band
|
2014-07-04 00:51:46 -07:00
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "</h1>\n")?;
|
2014-04-12 21:39:12 +02:00
|
|
|
|
2014-02-05 23:55:13 +11:00
|
|
|
match self.item.inner {
|
2014-03-05 15:28:08 -08:00
|
|
|
clean::ModuleItem(ref m) => {
|
2015-02-01 21:53:25 -05:00
|
|
|
item_module(fmt, self.cx, self.item, &m.items)
|
2014-03-05 15:28:08 -08:00
|
|
|
}
|
2013-09-26 11:57:25 -07:00
|
|
|
clean::FunctionItem(ref f) | clean::ForeignFunctionItem(ref f) =>
|
2015-08-16 17:05:18 +02:00
|
|
|
item_function(fmt, self.cx, self.item, f),
|
2014-05-21 16:41:58 -07:00
|
|
|
clean::TraitItem(ref t) => item_trait(fmt, self.cx, self.item, t),
|
2015-08-16 17:05:18 +02:00
|
|
|
clean::StructItem(ref s) => item_struct(fmt, self.cx, self.item, s),
|
2016-08-10 21:00:17 +03:00
|
|
|
clean::UnionItem(ref s) => item_union(fmt, self.cx, self.item, s),
|
2015-08-16 17:05:18 +02:00
|
|
|
clean::EnumItem(ref e) => item_enum(fmt, self.cx, self.item, e),
|
|
|
|
clean::TypedefItem(ref t, _) => item_typedef(fmt, self.cx, self.item, t),
|
|
|
|
clean::MacroItem(ref m) => item_macro(fmt, self.cx, self.item, m),
|
|
|
|
clean::PrimitiveItem(ref p) => item_primitive(fmt, self.cx, self.item, p),
|
2014-12-04 00:56:45 +09:00
|
|
|
clean::StaticItem(ref i) | clean::ForeignStaticItem(ref i) =>
|
2015-08-16 17:05:18 +02:00
|
|
|
item_static(fmt, self.cx, self.item, i),
|
|
|
|
clean::ConstantItem(ref c) => item_constant(fmt, self.cx, self.item, c),
|
2014-01-30 11:30:21 -08:00
|
|
|
_ => Ok(())
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-12 18:27:17 +01:00
|
|
|
fn item_path(ty: ItemType, name: &str) -> String {
|
|
|
|
match ty {
|
|
|
|
ItemType::Module => format!("{}/index.html", name),
|
2016-08-03 13:14:59 +12:00
|
|
|
_ => format!("{}.{}.html", ty.css_class(), name),
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-22 16:57:53 -07:00
|
|
|
fn full_path(cx: &Context, item: &clean::Item) -> String {
|
2015-07-10 08:19:21 -04:00
|
|
|
let mut s = cx.current.join("::");
|
2013-09-18 22:18:38 -07:00
|
|
|
s.push_str("::");
|
2015-02-01 21:53:25 -05:00
|
|
|
s.push_str(item.name.as_ref().unwrap());
|
2014-05-12 13:44:59 -07:00
|
|
|
return s
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2015-03-14 00:45:39 +02:00
|
|
|
fn shorter<'a>(s: Option<&'a str>) -> String {
|
2013-09-18 22:18:38 -07:00
|
|
|
match s {
|
2015-03-14 00:45:39 +02:00
|
|
|
Some(s) => s.lines().take_while(|line|{
|
|
|
|
(*line).chars().any(|chr|{
|
|
|
|
!chr.is_whitespace()
|
|
|
|
})
|
2015-07-10 08:19:21 -04:00
|
|
|
}).collect::<Vec<_>>().join("\n"),
|
2015-03-14 00:45:39 +02:00
|
|
|
None => "".to_string()
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-23 09:58:38 +08:00
|
|
|
#[inline]
|
2015-03-10 20:55:09 +08:00
|
|
|
fn plain_summary_line(s: Option<&str>) -> String {
|
2016-02-26 17:39:37 +01:00
|
|
|
let line = shorter(s).replace("\n", " ");
|
|
|
|
markdown::plain_summary_line(&line[..])
|
2014-12-23 09:58:38 +08:00
|
|
|
}
|
|
|
|
|
2015-08-16 17:05:18 +02:00
|
|
|
fn document(w: &mut fmt::Formatter, cx: &Context, item: &clean::Item) -> fmt::Result {
|
2016-06-15 23:55:11 +01:00
|
|
|
document_stability(w, cx, item)?;
|
|
|
|
document_full(w, item)?;
|
2014-01-30 11:30:21 -08:00
|
|
|
Ok(())
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2016-05-17 06:50:39 +05:30
|
|
|
fn document_short(w: &mut fmt::Formatter, item: &clean::Item, link: AssocItemLink) -> fmt::Result {
|
|
|
|
if let Some(s) = item.doc_value() {
|
2016-05-18 23:57:13 +05:30
|
|
|
let markdown = if s.contains('\n') {
|
2016-05-19 00:07:58 +05:30
|
|
|
format!("{} [Read more]({})",
|
|
|
|
&plain_summary_line(Some(s)), naive_assoc_href(item, link))
|
2016-05-18 23:57:13 +05:30
|
|
|
} else {
|
|
|
|
format!("{}", &plain_summary_line(Some(s)))
|
|
|
|
};
|
|
|
|
write!(w, "<div class='docblock'>{}</div>", Markdown(&markdown))?;
|
2016-05-17 06:50:39 +05:30
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2016-06-15 23:55:11 +01:00
|
|
|
fn document_full(w: &mut fmt::Formatter, item: &clean::Item) -> fmt::Result {
|
|
|
|
if let Some(s) = item.doc_value() {
|
|
|
|
write!(w, "<div class='docblock'>{}</div>", Markdown(s))?;
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn document_stability(w: &mut fmt::Formatter, cx: &Context, item: &clean::Item) -> fmt::Result {
|
|
|
|
for stability in short_stability(item, cx, true) {
|
|
|
|
write!(w, "<div class='stability'>{}</div>", stability)?;
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2014-05-10 14:05:06 -07:00
|
|
|
fn item_module(w: &mut fmt::Formatter, cx: &Context,
|
2014-01-30 11:30:21 -08:00
|
|
|
item: &clean::Item, items: &[clean::Item]) -> fmt::Result {
|
2016-03-22 22:01:37 -05:00
|
|
|
document(w, cx, item)?;
|
2014-07-04 00:51:46 -07:00
|
|
|
|
2015-01-26 16:05:07 -05:00
|
|
|
let mut indices = (0..items.len()).filter(|i| {
|
2016-06-28 22:50:44 +01:00
|
|
|
if let clean::DefaultImplItem(..) = items[*i].inner {
|
|
|
|
return false;
|
|
|
|
}
|
2016-08-20 14:22:08 -04:00
|
|
|
!maybe_ignore_item(&items[*i])
|
2015-03-25 17:06:52 -07:00
|
|
|
}).collect::<Vec<usize>>();
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2014-12-03 23:57:45 +09:00
|
|
|
// the order of item types in the listing
|
|
|
|
fn reorder(ty: ItemType) -> u8 {
|
|
|
|
match ty {
|
2014-12-26 10:55:16 +02:00
|
|
|
ItemType::ExternCrate => 0,
|
|
|
|
ItemType::Import => 1,
|
|
|
|
ItemType::Primitive => 2,
|
|
|
|
ItemType::Module => 3,
|
|
|
|
ItemType::Macro => 4,
|
|
|
|
ItemType::Struct => 5,
|
|
|
|
ItemType::Enum => 6,
|
|
|
|
ItemType::Constant => 7,
|
|
|
|
ItemType::Static => 8,
|
|
|
|
ItemType::Trait => 9,
|
|
|
|
ItemType::Function => 10,
|
|
|
|
ItemType::Typedef => 12,
|
2016-08-10 21:00:17 +03:00
|
|
|
ItemType::Union => 13,
|
|
|
|
_ => 14 + ty as u8,
|
2014-12-03 23:57:45 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-25 17:06:52 -07:00
|
|
|
fn cmp(i1: &clean::Item, i2: &clean::Item, idx1: usize, idx2: usize) -> Ordering {
|
2016-08-03 13:14:59 +12:00
|
|
|
let ty1 = item_type(i1);
|
|
|
|
let ty2 = item_type(i2);
|
2015-04-13 11:55:00 -07:00
|
|
|
if ty1 != ty2 {
|
|
|
|
return (reorder(ty1), idx1).cmp(&(reorder(ty2), idx2))
|
|
|
|
}
|
|
|
|
let s1 = i1.stability.as_ref().map(|s| s.level);
|
|
|
|
let s2 = i2.stability.as_ref().map(|s| s.level);
|
|
|
|
match (s1, s2) {
|
2015-10-13 06:01:31 +03:00
|
|
|
(Some(stability::Unstable), Some(stability::Stable)) => return Ordering::Greater,
|
|
|
|
(Some(stability::Stable), Some(stability::Unstable)) => return Ordering::Less,
|
2015-04-13 11:55:00 -07:00
|
|
|
_ => {}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2015-04-13 11:55:00 -07:00
|
|
|
i1.name.cmp(&i2.name)
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2013-12-20 14:42:00 +11:00
|
|
|
indices.sort_by(|&i1, &i2| cmp(&items[i1], &items[i2], i1, i2));
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2014-12-20 00:09:35 -08:00
|
|
|
debug!("{:?}", indices);
|
2014-04-09 16:49:31 +09:00
|
|
|
let mut curty = None;
|
2015-01-31 12:20:46 -05:00
|
|
|
for &idx in &indices {
|
2013-09-18 22:18:38 -07:00
|
|
|
let myitem = &items[idx];
|
2016-03-31 18:15:54 +02:00
|
|
|
if myitem.is_stripped() {
|
|
|
|
continue;
|
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2016-08-03 13:14:59 +12:00
|
|
|
let myty = Some(item_type(myitem));
|
2014-12-26 10:55:16 +02:00
|
|
|
if curty == Some(ItemType::ExternCrate) && myty == Some(ItemType::Import) {
|
|
|
|
// Put `extern crate` and `use` re-exports in the same section.
|
|
|
|
curty = myty;
|
|
|
|
} else if myty != curty {
|
2014-04-09 16:49:31 +09:00
|
|
|
if curty.is_some() {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "</table>")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
curty = myty;
|
2014-12-03 23:57:45 +09:00
|
|
|
let (short, name) = match myty.unwrap() {
|
2014-12-26 10:55:16 +02:00
|
|
|
ItemType::ExternCrate |
|
|
|
|
ItemType::Import => ("reexports", "Reexports"),
|
2014-12-03 23:57:45 +09:00
|
|
|
ItemType::Module => ("modules", "Modules"),
|
|
|
|
ItemType::Struct => ("structs", "Structs"),
|
2016-08-10 21:00:17 +03:00
|
|
|
ItemType::Union => ("unions", "Unions"),
|
2014-12-03 23:57:45 +09:00
|
|
|
ItemType::Enum => ("enums", "Enums"),
|
|
|
|
ItemType::Function => ("functions", "Functions"),
|
|
|
|
ItemType::Typedef => ("types", "Type Definitions"),
|
|
|
|
ItemType::Static => ("statics", "Statics"),
|
|
|
|
ItemType::Constant => ("constants", "Constants"),
|
|
|
|
ItemType::Trait => ("traits", "Traits"),
|
|
|
|
ItemType::Impl => ("impls", "Implementations"),
|
|
|
|
ItemType::TyMethod => ("tymethods", "Type Methods"),
|
|
|
|
ItemType::Method => ("methods", "Methods"),
|
|
|
|
ItemType::StructField => ("fields", "Struct Fields"),
|
|
|
|
ItemType::Variant => ("variants", "Variants"),
|
|
|
|
ItemType::Macro => ("macros", "Macros"),
|
|
|
|
ItemType::Primitive => ("primitives", "Primitive Types"),
|
|
|
|
ItemType::AssociatedType => ("associated-types", "Associated Types"),
|
2015-03-14 12:05:00 -06:00
|
|
|
ItemType::AssociatedConst => ("associated-consts", "Associated Constants"),
|
2014-03-04 11:24:20 -08:00
|
|
|
};
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<h2 id='{id}' class='section-header'>\
|
2016-03-22 17:58:45 -05:00
|
|
|
<a href=\"#{id}\">{name}</a></h2>\n<table>",
|
|
|
|
id = derive_id(short.to_owned()), name = name)?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2014-10-06 17:41:15 -07:00
|
|
|
match myitem.inner {
|
2014-12-26 10:55:16 +02:00
|
|
|
clean::ExternCrateItem(ref name, ref src) => {
|
2016-04-25 08:24:50 +02:00
|
|
|
use html::format::HRef;
|
|
|
|
|
2014-12-26 10:55:16 +02:00
|
|
|
match *src {
|
|
|
|
Some(ref src) => {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<tr><td><code>{}extern crate {} as {};",
|
2016-03-25 06:08:11 +00:00
|
|
|
VisSpace(&myitem.visibility),
|
2016-04-25 08:24:50 +02:00
|
|
|
HRef::new(myitem.def_id, src),
|
2016-03-22 17:58:45 -05:00
|
|
|
name)?
|
2013-09-24 13:56:52 -07:00
|
|
|
}
|
2014-12-26 10:55:16 +02:00
|
|
|
None => {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<tr><td><code>{}extern crate {};",
|
2016-04-25 08:24:50 +02:00
|
|
|
VisSpace(&myitem.visibility),
|
|
|
|
HRef::new(myitem.def_id, name))?
|
2013-09-24 13:56:52 -07:00
|
|
|
}
|
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "</code></td></tr>")?;
|
2014-12-26 10:55:16 +02:00
|
|
|
}
|
2013-09-24 13:56:52 -07:00
|
|
|
|
2014-12-26 10:55:16 +02:00
|
|
|
clean::ImportItem(ref import) => {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<tr><td><code>{}{}</code></td></tr>",
|
2016-03-25 06:08:11 +00:00
|
|
|
VisSpace(&myitem.visibility), *import)?;
|
2013-09-24 13:56:52 -07:00
|
|
|
}
|
|
|
|
|
2013-09-18 22:18:38 -07:00
|
|
|
_ => {
|
2013-10-01 14:31:03 -07:00
|
|
|
if myitem.name.is_none() { continue }
|
2016-04-22 23:26:08 -04:00
|
|
|
|
|
|
|
let stabilities = short_stability(myitem, cx, false);
|
|
|
|
|
|
|
|
let stab_docs = if !stabilities.is_empty() {
|
|
|
|
stabilities.iter()
|
|
|
|
.map(|s| format!("[{}]", s))
|
|
|
|
.collect::<Vec<_>>()
|
|
|
|
.as_slice()
|
|
|
|
.join(" ")
|
2015-04-13 11:55:00 -07:00
|
|
|
} else {
|
|
|
|
String::new()
|
|
|
|
};
|
2016-02-12 13:45:07 +01:00
|
|
|
let doc_value = myitem.doc_value().unwrap_or("");
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "
|
2016-03-22 17:58:45 -05:00
|
|
|
<tr class='{stab} module-item'>
|
|
|
|
<td><a class='{class}' href='{href}'
|
|
|
|
title='{title}'>{name}</a></td>
|
|
|
|
<td class='docblock short'>
|
|
|
|
{stab_docs} {docs}
|
|
|
|
</td>
|
|
|
|
</tr>",
|
|
|
|
name = *myitem.name.as_ref().unwrap(),
|
|
|
|
stab_docs = stab_docs,
|
|
|
|
docs = shorter(Some(&Markdown(doc_value).to_string())),
|
2016-08-03 13:14:59 +12:00
|
|
|
class = item_type(myitem),
|
2016-03-22 17:58:45 -05:00
|
|
|
stab = myitem.stability_class(),
|
2016-08-03 13:14:59 +12:00
|
|
|
href = item_path(item_type(myitem), myitem.name.as_ref().unwrap()),
|
2016-03-22 17:58:45 -05:00
|
|
|
title = full_path(cx, myitem))?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-07-04 00:51:46 -07:00
|
|
|
|
2016-06-06 08:00:34 +00:00
|
|
|
if curty.is_some() {
|
|
|
|
write!(w, "</table>")?;
|
|
|
|
}
|
|
|
|
Ok(())
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2016-08-20 14:22:08 -04:00
|
|
|
fn maybe_ignore_item(it: &clean::Item) -> bool {
|
|
|
|
match it.inner {
|
|
|
|
clean::StrippedItem(..) => true,
|
|
|
|
clean::ModuleItem(ref m) => {
|
|
|
|
it.doc_value().is_none() && m.items.is_empty()
|
|
|
|
&& it.visibility != Some(clean::Public)
|
|
|
|
},
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-22 23:26:08 -04:00
|
|
|
fn short_stability(item: &clean::Item, cx: &Context, show_reason: bool) -> Vec<String> {
|
|
|
|
let mut stability = vec![];
|
|
|
|
|
|
|
|
if let Some(stab) = item.stability.as_ref() {
|
2015-04-13 16:23:32 -07:00
|
|
|
let reason = if show_reason && !stab.reason.is_empty() {
|
2015-04-13 11:55:00 -07:00
|
|
|
format!(": {}", stab.reason)
|
|
|
|
} else {
|
|
|
|
String::new()
|
|
|
|
};
|
2016-04-22 23:26:08 -04:00
|
|
|
if !stab.deprecated_since.is_empty() {
|
2015-04-13 11:55:00 -07:00
|
|
|
let since = if show_reason {
|
|
|
|
format!(" since {}", Escape(&stab.deprecated_since))
|
|
|
|
} else {
|
|
|
|
String::new()
|
|
|
|
};
|
2016-04-22 23:26:08 -04:00
|
|
|
let text = format!("Deprecated{}{}", since, Markdown(&reason));
|
|
|
|
stability.push(format!("<em class='stab deprecated'>{}</em>", text))
|
|
|
|
};
|
|
|
|
|
|
|
|
if stab.level == stability::Unstable {
|
2015-08-16 17:20:37 +02:00
|
|
|
let unstable_extra = if show_reason {
|
2016-04-02 09:03:55 +02:00
|
|
|
match (!stab.feature.is_empty(), &cx.shared.issue_tracker_base_url, stab.issue) {
|
2016-01-11 14:38:40 -05:00
|
|
|
(true, &Some(ref tracker_url), Some(issue_no)) if issue_no > 0 =>
|
2015-08-16 23:06:57 +02:00
|
|
|
format!(" (<code>{}</code> <a href=\"{}{}\">#{}</a>)",
|
|
|
|
Escape(&stab.feature), tracker_url, issue_no, issue_no),
|
2016-01-11 14:38:40 -05:00
|
|
|
(false, &Some(ref tracker_url), Some(issue_no)) if issue_no > 0 =>
|
2015-08-16 17:20:37 +02:00
|
|
|
format!(" (<a href=\"{}{}\">#{}</a>)", Escape(&tracker_url), issue_no,
|
|
|
|
issue_no),
|
|
|
|
(true, _, _) =>
|
|
|
|
format!(" (<code>{}</code>)", Escape(&stab.feature)),
|
|
|
|
_ => String::new(),
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
String::new()
|
|
|
|
};
|
2016-04-22 23:26:08 -04:00
|
|
|
let text = format!("Unstable{}{}", unstable_extra, Markdown(&reason));
|
|
|
|
stability.push(format!("<em class='stab unstable'>{}</em>", text))
|
|
|
|
};
|
|
|
|
} else if let Some(depr) = item.deprecation.as_ref() {
|
|
|
|
let note = if show_reason && !depr.note.is_empty() {
|
|
|
|
format!(": {}", depr.note)
|
2015-04-13 11:55:00 -07:00
|
|
|
} else {
|
2016-04-22 23:26:08 -04:00
|
|
|
String::new()
|
|
|
|
};
|
|
|
|
let since = if show_reason && !depr.since.is_empty() {
|
|
|
|
format!(" since {}", Escape(&depr.since))
|
|
|
|
} else {
|
|
|
|
String::new()
|
2015-04-13 11:55:00 -07:00
|
|
|
};
|
2015-12-12 23:01:27 +03:00
|
|
|
|
2016-04-22 23:26:08 -04:00
|
|
|
let text = format!("Deprecated{}{}", since, Markdown(¬e));
|
|
|
|
stability.push(format!("<em class='stab deprecated'>{}</em>", text))
|
|
|
|
}
|
|
|
|
|
|
|
|
stability
|
2015-04-13 11:55:00 -07:00
|
|
|
}
|
|
|
|
|
2014-11-22 20:48:55 +13:00
|
|
|
struct Initializer<'a>(&'a str);
|
2014-12-20 00:09:35 -08:00
|
|
|
|
2015-01-20 15:45:07 -08:00
|
|
|
impl<'a> fmt::Display for Initializer<'a> {
|
2014-11-22 20:48:55 +13:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
let Initializer(s) = *self;
|
2015-03-24 16:53:34 -07:00
|
|
|
if s.is_empty() { return Ok(()); }
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(f, "<code> = </code>")?;
|
2016-05-03 13:31:12 +02:00
|
|
|
write!(f, "<code>{}</code>", Escape(s))
|
2014-11-22 20:48:55 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-16 17:05:18 +02:00
|
|
|
fn item_constant(w: &mut fmt::Formatter, cx: &Context, it: &clean::Item,
|
2014-11-22 20:48:55 +13:00
|
|
|
c: &clean::Constant) -> fmt::Result {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<pre class='rust const'>{vis}const \
|
2016-03-22 17:58:45 -05:00
|
|
|
{name}: {typ}{init}</pre>",
|
2016-03-25 06:08:11 +00:00
|
|
|
vis = VisSpace(&it.visibility),
|
2015-02-01 21:53:25 -05:00
|
|
|
name = it.name.as_ref().unwrap(),
|
2014-11-22 20:48:55 +13:00
|
|
|
typ = c.type_,
|
2016-03-22 22:01:37 -05:00
|
|
|
init = Initializer(&c.expr))?;
|
2015-08-16 17:05:18 +02:00
|
|
|
document(w, cx, it)
|
2014-11-22 20:48:55 +13:00
|
|
|
}
|
|
|
|
|
2015-08-16 17:05:18 +02:00
|
|
|
fn item_static(w: &mut fmt::Formatter, cx: &Context, it: &clean::Item,
|
2014-11-22 20:48:55 +13:00
|
|
|
s: &clean::Static) -> fmt::Result {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<pre class='rust static'>{vis}static {mutability}\
|
2016-03-22 17:58:45 -05:00
|
|
|
{name}: {typ}{init}</pre>",
|
2016-03-25 06:08:11 +00:00
|
|
|
vis = VisSpace(&it.visibility),
|
2014-11-22 20:48:55 +13:00
|
|
|
mutability = MutableSpace(s.mutability),
|
2015-02-01 21:53:25 -05:00
|
|
|
name = it.name.as_ref().unwrap(),
|
2014-11-22 20:48:55 +13:00
|
|
|
typ = s.type_,
|
2016-03-22 22:01:37 -05:00
|
|
|
init = Initializer(&s.expr))?;
|
2015-08-16 17:05:18 +02:00
|
|
|
document(w, cx, it)
|
2014-11-22 20:48:55 +13:00
|
|
|
}
|
|
|
|
|
2015-08-16 17:05:18 +02:00
|
|
|
fn item_function(w: &mut fmt::Formatter, cx: &Context, it: &clean::Item,
|
2014-01-30 11:30:21 -08:00
|
|
|
f: &clean::Function) -> fmt::Result {
|
2016-03-22 20:26:33 +01:00
|
|
|
// FIXME(#24111): remove when `const_fn` is stabilized
|
2016-02-01 13:05:37 -05:00
|
|
|
let vis_constness = match get_unstable_features_setting() {
|
|
|
|
UnstableFeatures::Allow => f.constness,
|
|
|
|
_ => hir::Constness::NotConst
|
|
|
|
};
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<pre class='rust fn'>{vis}{constness}{unsafety}{abi}fn \
|
2016-03-22 17:58:45 -05:00
|
|
|
{name}{generics}{decl}{where_clause}</pre>",
|
2016-03-25 06:08:11 +00:00
|
|
|
vis = VisSpace(&it.visibility),
|
2016-02-01 13:05:37 -05:00
|
|
|
constness = ConstnessSpace(vis_constness),
|
2014-12-09 10:36:46 -05:00
|
|
|
unsafety = UnsafetySpace(f.unsafety),
|
2015-04-07 14:22:55 -07:00
|
|
|
abi = AbiSpace(f.abi),
|
2015-02-01 21:53:25 -05:00
|
|
|
name = it.name.as_ref().unwrap(),
|
2013-09-18 22:18:38 -07:00
|
|
|
generics = f.generics,
|
2014-09-25 02:01:42 -07:00
|
|
|
where_clause = WhereClause(&f.generics),
|
2016-03-22 22:01:37 -05:00
|
|
|
decl = f.decl)?;
|
2015-08-16 17:05:18 +02:00
|
|
|
document(w, cx, it)
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2014-05-21 16:41:58 -07:00
|
|
|
fn item_trait(w: &mut fmt::Formatter, cx: &Context, it: &clean::Item,
|
2014-01-30 11:30:21 -08:00
|
|
|
t: &clean::Trait) -> fmt::Result {
|
2014-08-27 21:46:52 -04:00
|
|
|
let mut bounds = String::new();
|
2015-03-24 16:54:09 -07:00
|
|
|
if !t.bounds.is_empty() {
|
|
|
|
if !bounds.is_empty() {
|
2014-11-24 10:14:46 -08:00
|
|
|
bounds.push(' ');
|
|
|
|
}
|
2014-08-27 21:46:52 -04:00
|
|
|
bounds.push_str(": ");
|
|
|
|
for (i, p) in t.bounds.iter().enumerate() {
|
|
|
|
if i > 0 { bounds.push_str(" + "); }
|
2015-02-01 21:53:25 -05:00
|
|
|
bounds.push_str(&format!("{}", *p));
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Output the trait definition
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<pre class='rust trait'>{}{}trait {}{}{}{} ",
|
2016-03-25 06:08:11 +00:00
|
|
|
VisSpace(&it.visibility),
|
2016-03-22 17:58:45 -05:00
|
|
|
UnsafetySpace(t.unsafety),
|
|
|
|
it.name.as_ref().unwrap(),
|
|
|
|
t.generics,
|
|
|
|
bounds,
|
|
|
|
WhereClause(&t.generics))?;
|
2014-11-20 11:22:09 -08:00
|
|
|
|
2016-02-28 12:23:07 +01:00
|
|
|
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 required = t.items.iter().filter(|m| m.is_ty_method()).collect::<Vec<_>>();
|
|
|
|
let provided = t.items.iter().filter(|m| m.is_method()).collect::<Vec<_>>();
|
2014-08-04 13:56:56 -07:00
|
|
|
|
2015-03-24 16:53:34 -07:00
|
|
|
if t.items.is_empty() {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "{{ }}")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
} else {
|
2016-04-16 11:46:52 -04:00
|
|
|
// FIXME: we should be using a derived_id for the Anchors here
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "{{\n")?;
|
2015-01-31 12:20:46 -05:00
|
|
|
for t in &types {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, " ")?;
|
2016-04-16 11:46:52 -04:00
|
|
|
render_assoc_item(w, t, AssocItemLink::Anchor(None))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, ";\n")?;
|
2014-11-20 11:22:09 -08:00
|
|
|
}
|
2015-04-30 09:37:13 -07:00
|
|
|
if !types.is_empty() && !consts.is_empty() {
|
2016-03-22 22:01:37 -05:00
|
|
|
w.write_str("\n")?;
|
2015-04-30 09:37:13 -07:00
|
|
|
}
|
|
|
|
for t in &consts {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, " ")?;
|
2016-04-16 11:46:52 -04:00
|
|
|
render_assoc_item(w, t, AssocItemLink::Anchor(None))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, ";\n")?;
|
2015-04-30 09:37:13 -07:00
|
|
|
}
|
|
|
|
if !consts.is_empty() && !required.is_empty() {
|
2016-03-22 22:01:37 -05:00
|
|
|
w.write_str("\n")?;
|
2014-11-20 11:22:09 -08:00
|
|
|
}
|
2015-01-31 12:20:46 -05:00
|
|
|
for m in &required {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, " ")?;
|
2016-04-16 11:46:52 -04:00
|
|
|
render_assoc_item(w, m, AssocItemLink::Anchor(None))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, ";\n")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2015-03-24 16:54:09 -07:00
|
|
|
if !required.is_empty() && !provided.is_empty() {
|
2016-03-22 22:01:37 -05:00
|
|
|
w.write_str("\n")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2015-01-31 12:20:46 -05:00
|
|
|
for m in &provided {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, " ")?;
|
2016-04-16 11:46:52 -04:00
|
|
|
render_assoc_item(w, m, AssocItemLink::Anchor(None))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, " {{ ... }}\n")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "}}")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "</pre>")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
|
|
|
|
// Trait documentation
|
2016-03-22 22:01:37 -05:00
|
|
|
document(w, cx, it)?;
|
2014-01-30 11:30:21 -08:00
|
|
|
|
2016-02-09 21:15:29 -05:00
|
|
|
fn trait_item(w: &mut fmt::Formatter, cx: &Context, m: &clean::Item, t: &clean::Item)
|
2014-08-04 13:56:56 -07:00
|
|
|
-> fmt::Result {
|
2015-12-02 23:49:18 +01:00
|
|
|
let name = m.name.as_ref().unwrap();
|
2016-08-16 10:27:07 +12:00
|
|
|
let item_type = item_type(m);
|
|
|
|
let id = derive_id(format!("{}.{}", item_type, name));
|
|
|
|
let ns_id = derive_id(format!("{}.{}", name, item_type.name_space()));
|
|
|
|
write!(w, "<h3 id='{id}' class='method stab {stab}'>\
|
|
|
|
<span id='{ns_id}' class='invisible'><code>",
|
2016-03-22 17:58:45 -05:00
|
|
|
id = id,
|
2016-08-16 10:27:07 +12:00
|
|
|
stab = m.stability_class(),
|
|
|
|
ns_id = ns_id)?;
|
2016-04-16 11:46:52 -04:00
|
|
|
render_assoc_item(w, m, AssocItemLink::Anchor(Some(&id)))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "</code>")?;
|
|
|
|
render_stability_since(w, m, t)?;
|
2016-08-16 10:27:07 +12:00
|
|
|
write!(w, "</span></h3>")?;
|
2016-03-22 22:01:37 -05:00
|
|
|
document(w, cx, m)?;
|
2014-01-30 11:30:21 -08:00
|
|
|
Ok(())
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2015-03-24 16:54:09 -07:00
|
|
|
if !types.is_empty() {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "
|
2014-11-20 11:22:09 -08:00
|
|
|
<h2 id='associated-types'>Associated Types</h2>
|
|
|
|
<div class='methods'>
|
2016-03-22 22:01:37 -05:00
|
|
|
")?;
|
2015-01-31 12:20:46 -05:00
|
|
|
for t in &types {
|
2016-03-22 22:01:37 -05:00
|
|
|
trait_item(w, cx, *t, it)?;
|
2014-11-20 11:22:09 -08:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "</div>")?;
|
2014-11-20 11:22:09 -08:00
|
|
|
}
|
|
|
|
|
2015-05-09 00:16:20 +02:00
|
|
|
if !consts.is_empty() {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "
|
2015-05-09 00:16:20 +02:00
|
|
|
<h2 id='associated-const'>Associated Constants</h2>
|
|
|
|
<div class='methods'>
|
2016-03-22 22:01:37 -05:00
|
|
|
")?;
|
2015-05-09 00:16:20 +02:00
|
|
|
for t in &consts {
|
2016-03-22 22:01:37 -05:00
|
|
|
trait_item(w, cx, *t, it)?;
|
2015-05-09 00:16:20 +02:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "</div>")?;
|
2015-05-09 00:16:20 +02:00
|
|
|
}
|
|
|
|
|
2013-09-18 22:18:38 -07:00
|
|
|
// Output the documentation for each function individually
|
2015-03-24 16:54:09 -07:00
|
|
|
if !required.is_empty() {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "
|
2013-09-18 22:18:38 -07:00
|
|
|
<h2 id='required-methods'>Required Methods</h2>
|
|
|
|
<div class='methods'>
|
2016-03-22 22:01:37 -05:00
|
|
|
")?;
|
2015-01-31 12:20:46 -05:00
|
|
|
for m in &required {
|
2016-03-22 22:01:37 -05:00
|
|
|
trait_item(w, cx, *m, it)?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "</div>")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2015-03-24 16:54:09 -07:00
|
|
|
if !provided.is_empty() {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "
|
2013-09-18 22:18:38 -07:00
|
|
|
<h2 id='provided-methods'>Provided Methods</h2>
|
|
|
|
<div class='methods'>
|
2016-03-22 22:01:37 -05:00
|
|
|
")?;
|
2015-01-31 12:20:46 -05:00
|
|
|
for m in &provided {
|
2016-03-22 22:01:37 -05:00
|
|
|
trait_item(w, cx, *m, it)?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "</div>")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2015-04-06 19:21:52 -07:00
|
|
|
// If there are methods directly on this trait object, render them here.
|
2016-03-22 22:01:37 -05:00
|
|
|
render_assoc_items(w, cx, it, it.def_id, AssocItemRender::All)?;
|
2015-04-06 19:21:52 -07:00
|
|
|
|
2014-11-14 14:20:57 -08:00
|
|
|
let cache = cache();
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "
|
2014-05-27 17:52:40 -07:00
|
|
|
<h2 id='implementors'>Implementors</h2>
|
|
|
|
<ul class='item-list' id='implementors-list'>
|
2016-03-22 22:01:37 -05:00
|
|
|
")?;
|
2016-07-03 14:38:37 -07:00
|
|
|
if let Some(implementors) = cache.implementors.get(&it.def_id) {
|
|
|
|
for i in implementors {
|
|
|
|
write!(w, "<li><code>")?;
|
|
|
|
fmt_impl_for_trait_page(&i.impl_, w)?;
|
|
|
|
writeln!(w, "</code></li>")?;
|
2013-12-05 18:19:06 -08:00
|
|
|
}
|
2014-04-28 20:36:08 -07:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "</ul>")?;
|
|
|
|
write!(w, r#"<script type="text/javascript" async
|
2016-03-22 17:58:45 -05:00
|
|
|
src="{root_path}/implementors/{path}/{ty}.{name}.js">
|
|
|
|
</script>"#,
|
|
|
|
root_path = vec![".."; cx.current.len()].join("/"),
|
|
|
|
path = if it.def_id.is_local() {
|
|
|
|
cx.current.join("/")
|
|
|
|
} else {
|
2016-07-09 14:07:37 +01:00
|
|
|
let (ref path, _) = cache.external_paths[&it.def_id];
|
2016-03-22 17:58:45 -05:00
|
|
|
path[..path.len() - 1].join("/")
|
|
|
|
},
|
2016-08-03 13:14:59 +12:00
|
|
|
ty = item_type(it).css_class(),
|
2016-03-22 17:58:45 -05:00
|
|
|
name = *it.name.as_ref().unwrap())?;
|
2014-04-28 20:36:08 -07:00
|
|
|
Ok(())
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2016-03-25 00:10:15 +01:00
|
|
|
fn naive_assoc_href(it: &clean::Item, link: AssocItemLink) -> String {
|
|
|
|
use html::item_type::ItemType::*;
|
|
|
|
|
|
|
|
let name = it.name.as_ref().unwrap();
|
2016-08-03 13:14:59 +12:00
|
|
|
let ty = match item_type(it) {
|
2016-03-25 00:10:15 +01:00
|
|
|
Typedef | AssociatedType => AssociatedType,
|
|
|
|
s@_ => s,
|
|
|
|
};
|
|
|
|
|
|
|
|
let anchor = format!("#{}.{}", ty, name);
|
|
|
|
match link {
|
2016-04-16 11:46:52 -04:00
|
|
|
AssocItemLink::Anchor(Some(ref id)) => format!("#{}", id),
|
|
|
|
AssocItemLink::Anchor(None) => anchor,
|
2016-03-25 00:10:15 +01:00
|
|
|
AssocItemLink::GotoSource(did, _) => {
|
|
|
|
href(did).map(|p| format!("{}{}", p.0, anchor)).unwrap_or(anchor)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn assoc_const(w: &mut fmt::Formatter,
|
|
|
|
it: &clean::Item,
|
|
|
|
ty: &clean::Type,
|
|
|
|
default: Option<&String>,
|
|
|
|
link: AssocItemLink) -> fmt::Result {
|
|
|
|
write!(w, "const <a href='{}' class='constant'>{}</a>",
|
|
|
|
naive_assoc_href(it, link),
|
|
|
|
it.name.as_ref().unwrap())?;
|
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, ": {}", ty)?;
|
2015-04-30 09:37:13 -07:00
|
|
|
if let Some(default) = default {
|
2016-05-03 13:31:12 +02:00
|
|
|
write!(w, " = {}", Escape(default))?;
|
2015-03-15 19:35:25 -06:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2015-01-02 08:26:55 -05:00
|
|
|
fn assoc_type(w: &mut fmt::Formatter, it: &clean::Item,
|
2015-03-10 12:28:44 +02:00
|
|
|
bounds: &Vec<clean::TyParamBound>,
|
2016-03-25 00:10:15 +01:00
|
|
|
default: Option<&clean::Type>,
|
|
|
|
link: AssocItemLink) -> fmt::Result {
|
|
|
|
write!(w, "type <a href='{}' class='type'>{}</a>",
|
|
|
|
naive_assoc_href(it, link),
|
|
|
|
it.name.as_ref().unwrap())?;
|
2015-03-24 16:54:09 -07:00
|
|
|
if !bounds.is_empty() {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, ": {}", TyParamBounds(bounds))?
|
2015-01-02 08:26:55 -05:00
|
|
|
}
|
2016-03-25 00:10:15 +01:00
|
|
|
if let Some(default) = default {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, " = {}", default)?;
|
2015-01-02 08:26:55 -05:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2016-02-09 21:15:29 -05:00
|
|
|
fn render_stability_since_raw<'a>(w: &mut fmt::Formatter,
|
|
|
|
ver: Option<&'a str>,
|
|
|
|
containing_ver: Option<&'a str>) -> fmt::Result {
|
2016-02-28 12:11:13 +01:00
|
|
|
if let Some(v) = ver {
|
|
|
|
if containing_ver != ver && v.len() > 0 {
|
2016-05-18 03:54:52 +02:00
|
|
|
write!(w, "<div class='since' title='Stable since Rust version {0}'>{0}</div>",
|
2016-03-22 17:58:45 -05:00
|
|
|
v)?
|
2016-02-09 21:15:29 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn render_stability_since(w: &mut fmt::Formatter,
|
|
|
|
item: &clean::Item,
|
|
|
|
containing_item: &clean::Item) -> fmt::Result {
|
|
|
|
render_stability_since_raw(w, item.stable_since(), containing_item.stable_since())
|
|
|
|
}
|
|
|
|
|
2016-03-24 06:10:52 +01:00
|
|
|
fn render_assoc_item(w: &mut fmt::Formatter,
|
|
|
|
item: &clean::Item,
|
2015-03-14 12:05:00 -06:00
|
|
|
link: AssocItemLink) -> fmt::Result {
|
2015-02-25 22:05:07 +02:00
|
|
|
fn method(w: &mut fmt::Formatter,
|
2016-03-24 06:10:52 +01:00
|
|
|
meth: &clean::Item,
|
2015-07-31 00:04:06 -07:00
|
|
|
unsafety: hir::Unsafety,
|
|
|
|
constness: hir::Constness,
|
2015-02-25 22:05:07 +02:00
|
|
|
abi: abi::Abi,
|
|
|
|
g: &clean::Generics,
|
|
|
|
d: &clean::FnDecl,
|
|
|
|
link: AssocItemLink)
|
|
|
|
-> fmt::Result {
|
2016-03-24 06:10:52 +01:00
|
|
|
let name = meth.name.as_ref().unwrap();
|
2016-08-03 13:14:59 +12:00
|
|
|
let anchor = format!("#{}.{}", item_type(meth), name);
|
2015-04-06 17:56:35 -07:00
|
|
|
let href = match link {
|
2016-04-16 11:46:52 -04:00
|
|
|
AssocItemLink::Anchor(Some(ref id)) => format!("#{}", id),
|
|
|
|
AssocItemLink::Anchor(None) => anchor,
|
2016-03-24 06:10:52 +01:00
|
|
|
AssocItemLink::GotoSource(did, provided_methods) => {
|
|
|
|
// We're creating a link from an impl-item to the corresponding
|
|
|
|
// trait-item and need to map the anchored type accordingly.
|
|
|
|
let ty = if provided_methods.contains(name) {
|
|
|
|
ItemType::Method
|
|
|
|
} else {
|
|
|
|
ItemType::TyMethod
|
|
|
|
};
|
|
|
|
|
|
|
|
href(did).map(|p| format!("{}#{}.{}", p.0, ty, name)).unwrap_or(anchor)
|
2015-04-06 17:56:35 -07:00
|
|
|
}
|
|
|
|
};
|
2016-03-25 00:10:15 +01:00
|
|
|
// FIXME(#24111): remove when `const_fn` is stabilized
|
2016-02-01 13:05:37 -05:00
|
|
|
let vis_constness = match get_unstable_features_setting() {
|
|
|
|
UnstableFeatures::Allow => constness,
|
|
|
|
_ => hir::Constness::NotConst
|
|
|
|
};
|
2015-02-25 22:05:07 +02:00
|
|
|
write!(w, "{}{}{}fn <a href='{href}' class='fnname'>{name}</a>\
|
2014-09-25 02:01:42 -07:00
|
|
|
{generics}{decl}{where_clause}",
|
2016-02-01 13:05:37 -05:00
|
|
|
ConstnessSpace(vis_constness),
|
2015-11-19 21:08:50 +01:00
|
|
|
UnsafetySpace(unsafety),
|
2016-04-22 11:48:46 +01:00
|
|
|
AbiSpace(abi),
|
2015-04-06 17:56:35 -07:00
|
|
|
href = href,
|
|
|
|
name = name,
|
2013-09-18 22:18:38 -07:00
|
|
|
generics = *g,
|
2016-05-08 21:19:29 +03:00
|
|
|
decl = Method(d),
|
2014-09-25 02:01:42 -07:00
|
|
|
where_clause = WhereClause(g))
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2016-03-24 06:10:52 +01:00
|
|
|
match item.inner {
|
2016-03-31 18:15:54 +02:00
|
|
|
clean::StrippedItem(..) => Ok(()),
|
2013-09-18 22:18:38 -07:00
|
|
|
clean::TyMethodItem(ref m) => {
|
2016-03-24 06:10:52 +01:00
|
|
|
method(w, item, m.unsafety, hir::Constness::NotConst,
|
2016-05-08 21:19:29 +03:00
|
|
|
m.abi, &m.generics, &m.decl, link)
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
clean::MethodItem(ref m) => {
|
2016-03-24 06:10:52 +01:00
|
|
|
method(w, item, m.unsafety, m.constness,
|
2016-05-08 21:19:29 +03:00
|
|
|
m.abi, &m.generics, &m.decl,
|
2015-04-06 17:56:35 -07:00
|
|
|
link)
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2015-03-15 19:35:25 -06:00
|
|
|
clean::AssociatedConstItem(ref ty, ref default) => {
|
2016-03-25 00:10:15 +01:00
|
|
|
assoc_const(w, item, ty, default.as_ref(), link)
|
2015-03-15 19:35:25 -06:00
|
|
|
}
|
2015-03-10 12:28:44 +02:00
|
|
|
clean::AssociatedTypeItem(ref bounds, ref default) => {
|
2016-03-25 00:10:15 +01:00
|
|
|
assoc_type(w, item, bounds, default.as_ref(), link)
|
2014-11-20 11:22:09 -08:00
|
|
|
}
|
2015-03-14 12:05:00 -06:00
|
|
|
_ => panic!("render_assoc_item called on non-associated-item")
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-16 17:05:18 +02:00
|
|
|
fn item_struct(w: &mut fmt::Formatter, cx: &Context, it: &clean::Item,
|
2014-01-30 11:30:21 -08:00
|
|
|
s: &clean::Struct) -> fmt::Result {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<pre class='rust struct'>")?;
|
|
|
|
render_attributes(w, it)?;
|
|
|
|
render_struct(w,
|
2016-03-22 17:58:45 -05:00
|
|
|
it,
|
|
|
|
Some(&s.generics),
|
|
|
|
s.struct_type,
|
|
|
|
&s.fields,
|
|
|
|
"",
|
|
|
|
true)?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "</pre>")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
document(w, cx, it)?;
|
2016-05-27 12:04:56 +02:00
|
|
|
let mut fields = s.fields.iter().filter_map(|f| {
|
2014-04-19 22:24:52 -07:00
|
|
|
match f.inner {
|
2016-05-27 12:04:56 +02:00
|
|
|
clean::StructFieldItem(ref ty) => Some((f, ty)),
|
|
|
|
_ => None,
|
2014-04-19 22:24:52 -07:00
|
|
|
}
|
|
|
|
}).peekable();
|
2014-11-29 16:41:21 -05:00
|
|
|
if let doctree::Plain = s.struct_type {
|
|
|
|
if fields.peek().is_some() {
|
2016-05-27 12:04:56 +02:00
|
|
|
write!(w, "<h2 class='fields'>Fields</h2>")?;
|
|
|
|
for (field, ty) in fields {
|
2016-08-16 10:27:07 +12:00
|
|
|
let id = derive_id(format!("{}.{}",
|
|
|
|
ItemType::StructField,
|
|
|
|
field.name.as_ref().unwrap()));
|
|
|
|
let ns_id = derive_id(format!("{}.{}",
|
|
|
|
field.name.as_ref().unwrap(),
|
|
|
|
ItemType::StructField.name_space()));
|
|
|
|
write!(w, "<span id='{id}' class='{item_type}'>
|
|
|
|
<span id='{ns_id}' class='invisible'>
|
2016-08-03 13:56:18 +12:00
|
|
|
<code>{name}: {ty}</code>
|
2016-08-16 10:27:07 +12:00
|
|
|
</span></span><span class='stab {stab}'></span>",
|
2016-08-03 13:14:59 +12:00
|
|
|
item_type = ItemType::StructField,
|
2016-08-16 10:27:07 +12:00
|
|
|
id = id,
|
|
|
|
ns_id = ns_id,
|
2016-03-22 17:58:45 -05:00
|
|
|
stab = field.stability_class(),
|
2016-05-27 12:04:56 +02:00
|
|
|
name = field.name.as_ref().unwrap(),
|
|
|
|
ty = ty)?;
|
2016-03-22 22:01:37 -05:00
|
|
|
document(w, cx, field)?;
|
2013-09-30 16:31:35 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-02-09 21:15:29 -05:00
|
|
|
render_assoc_items(w, cx, it, it.def_id, AssocItemRender::All)
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2016-08-10 21:00:17 +03:00
|
|
|
fn item_union(w: &mut fmt::Formatter, cx: &Context, it: &clean::Item,
|
|
|
|
s: &clean::Union) -> fmt::Result {
|
|
|
|
write!(w, "<pre class='rust union'>")?;
|
|
|
|
render_attributes(w, it)?;
|
|
|
|
render_union(w,
|
|
|
|
it,
|
|
|
|
Some(&s.generics),
|
|
|
|
&s.fields,
|
|
|
|
"",
|
|
|
|
true)?;
|
|
|
|
write!(w, "</pre>")?;
|
|
|
|
|
|
|
|
document(w, cx, it)?;
|
|
|
|
let mut fields = s.fields.iter().filter_map(|f| {
|
|
|
|
match f.inner {
|
|
|
|
clean::StructFieldItem(ref ty) => Some((f, ty)),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}).peekable();
|
|
|
|
if fields.peek().is_some() {
|
|
|
|
write!(w, "<h2 class='fields'>Fields</h2>")?;
|
|
|
|
for (field, ty) in fields {
|
|
|
|
write!(w, "<span id='{shortty}.{name}' class='{shortty}'><code>{name}: {ty}</code>
|
|
|
|
</span><span class='stab {stab}'></span>",
|
|
|
|
shortty = ItemType::StructField,
|
|
|
|
stab = field.stability_class(),
|
|
|
|
name = field.name.as_ref().unwrap(),
|
|
|
|
ty = ty)?;
|
|
|
|
document(w, cx, field)?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
render_assoc_items(w, cx, it, it.def_id, AssocItemRender::All)
|
|
|
|
}
|
|
|
|
|
2015-08-16 17:05:18 +02:00
|
|
|
fn item_enum(w: &mut fmt::Formatter, cx: &Context, it: &clean::Item,
|
2014-05-10 14:05:06 -07:00
|
|
|
e: &clean::Enum) -> fmt::Result {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<pre class='rust enum'>")?;
|
|
|
|
render_attributes(w, it)?;
|
|
|
|
write!(w, "{}enum {}{}{}",
|
2016-03-25 06:08:11 +00:00
|
|
|
VisSpace(&it.visibility),
|
2016-03-22 17:58:45 -05:00
|
|
|
it.name.as_ref().unwrap(),
|
|
|
|
e.generics,
|
|
|
|
WhereClause(&e.generics))?;
|
2015-03-24 16:53:34 -07:00
|
|
|
if e.variants.is_empty() && !e.variants_stripped {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, " {{}}")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
} else {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, " {{\n")?;
|
2015-01-31 12:20:46 -05:00
|
|
|
for v in &e.variants {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, " ")?;
|
2015-02-01 21:53:25 -05:00
|
|
|
let name = v.name.as_ref().unwrap();
|
2013-09-18 22:18:38 -07:00
|
|
|
match v.inner {
|
|
|
|
clean::VariantItem(ref var) => {
|
|
|
|
match var.kind {
|
2016-03-22 22:01:37 -05:00
|
|
|
clean::CLikeVariant => write!(w, "{}", name)?,
|
2013-09-18 22:18:38 -07:00
|
|
|
clean::TupleVariant(ref tys) => {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "{}(", name)?;
|
2013-09-18 22:18:38 -07:00
|
|
|
for (i, ty) in tys.iter().enumerate() {
|
2014-01-30 11:30:21 -08:00
|
|
|
if i > 0 {
|
2016-05-27 12:04:56 +02:00
|
|
|
write!(w, ", ")?
|
2014-01-30 11:30:21 -08:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "{}", *ty)?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, ")")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
clean::StructVariant(ref s) => {
|
2016-03-22 22:01:37 -05:00
|
|
|
render_struct(w,
|
2016-03-22 17:58:45 -05:00
|
|
|
v,
|
|
|
|
None,
|
|
|
|
s.struct_type,
|
|
|
|
&s.fields,
|
|
|
|
" ",
|
|
|
|
false)?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => unreachable!()
|
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, ",\n")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2013-10-13 20:37:43 -07:00
|
|
|
|
|
|
|
if e.variants_stripped {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, " // some variants omitted\n")?;
|
2013-10-13 20:37:43 -07:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "}}")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "</pre>")?;
|
|
|
|
render_stability_since_raw(w, it.stable_since(), None)?;
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
document(w, cx, it)?;
|
2015-03-24 16:54:09 -07:00
|
|
|
if !e.variants.is_empty() {
|
2016-05-27 12:04:56 +02:00
|
|
|
write!(w, "<h2 class='variants'>Variants</h2>\n")?;
|
2015-01-31 12:20:46 -05:00
|
|
|
for variant in &e.variants {
|
2016-08-16 10:27:07 +12:00
|
|
|
let id = derive_id(format!("{}.{}",
|
|
|
|
ItemType::Variant,
|
|
|
|
variant.name.as_ref().unwrap()));
|
|
|
|
let ns_id = derive_id(format!("{}.{}",
|
|
|
|
variant.name.as_ref().unwrap(),
|
|
|
|
ItemType::Variant.name_space()));
|
|
|
|
write!(w, "<span id='{id}' class='variant'>\
|
|
|
|
<span id='{ns_id}' class='invisible'><code>{name}",
|
|
|
|
id = id,
|
|
|
|
ns_id = ns_id,
|
2016-03-22 17:58:45 -05:00
|
|
|
name = variant.name.as_ref().unwrap())?;
|
2016-05-27 12:04:56 +02:00
|
|
|
if let clean::VariantItem(ref var) = variant.inner {
|
|
|
|
if let clean::TupleVariant(ref tys) = var.kind {
|
|
|
|
write!(w, "(")?;
|
|
|
|
for (i, ty) in tys.iter().enumerate() {
|
|
|
|
if i > 0 {
|
|
|
|
write!(w, ", ")?;
|
|
|
|
}
|
|
|
|
write!(w, "{}", *ty)?;
|
|
|
|
}
|
|
|
|
write!(w, ")")?;
|
|
|
|
}
|
|
|
|
}
|
2016-08-16 10:27:07 +12:00
|
|
|
write!(w, "</code></span></span>")?;
|
2016-03-22 22:01:37 -05:00
|
|
|
document(w, cx, variant)?;
|
2016-02-28 12:11:13 +01:00
|
|
|
|
|
|
|
use clean::{Variant, StructVariant};
|
|
|
|
if let clean::VariantItem( Variant { kind: StructVariant(ref s) } ) = variant.inner {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<h3 class='fields'>Fields</h3>\n
|
2016-03-22 17:58:45 -05:00
|
|
|
<table>")?;
|
2016-05-27 12:04:56 +02:00
|
|
|
for field in &s.fields {
|
|
|
|
use clean::StructFieldItem;
|
|
|
|
if let StructFieldItem(ref ty) = field.inner {
|
2016-08-16 10:27:07 +12:00
|
|
|
let id = derive_id(format!("variant.{}.field.{}",
|
|
|
|
variant.name.as_ref().unwrap(),
|
|
|
|
field.name.as_ref().unwrap()));
|
|
|
|
let ns_id = derive_id(format!("{}.{}.{}.{}",
|
|
|
|
variant.name.as_ref().unwrap(),
|
|
|
|
ItemType::Variant.name_space(),
|
|
|
|
field.name.as_ref().unwrap(),
|
|
|
|
ItemType::StructField.name_space()));
|
2016-05-27 12:04:56 +02:00
|
|
|
write!(w, "<tr><td \
|
2016-08-16 10:27:07 +12:00
|
|
|
id='{id}'>\
|
|
|
|
<span id='{ns_id}' class='invisible'>\
|
|
|
|
<code>{f}: {t}</code></span></td><td>",
|
|
|
|
id = id,
|
|
|
|
ns_id = ns_id,
|
2016-05-27 12:04:56 +02:00
|
|
|
f = field.name.as_ref().unwrap(),
|
|
|
|
t = *ty)?;
|
|
|
|
document(w, cx, field)?;
|
|
|
|
write!(w, "</td></tr>")?;
|
|
|
|
}
|
2013-10-18 22:00:08 -07:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "</table>")?;
|
2013-10-18 22:00:08 -07:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
render_stability_since(w, variant, it)?;
|
2013-09-30 16:31:35 -07:00
|
|
|
}
|
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
render_assoc_items(w, cx, it, it.def_id, AssocItemRender::All)?;
|
2014-01-30 11:30:21 -08:00
|
|
|
Ok(())
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2015-02-13 00:47:03 +09:00
|
|
|
fn render_attributes(w: &mut fmt::Formatter, it: &clean::Item) -> fmt::Result {
|
|
|
|
for attr in &it.attrs {
|
|
|
|
match *attr {
|
|
|
|
clean::Word(ref s) if *s == "must_use" => {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "#[{}]\n", s)?;
|
2015-02-13 00:47:03 +09:00
|
|
|
}
|
|
|
|
clean::NameValue(ref k, ref v) if *k == "must_use" => {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "#[{} = \"{}\"]\n", k, v)?;
|
2015-02-13 00:47:03 +09:00
|
|
|
}
|
|
|
|
_ => ()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2014-05-10 14:05:06 -07:00
|
|
|
fn render_struct(w: &mut fmt::Formatter, it: &clean::Item,
|
2013-09-18 22:18:38 -07:00
|
|
|
g: Option<&clean::Generics>,
|
|
|
|
ty: doctree::StructType,
|
|
|
|
fields: &[clean::Item],
|
2013-09-30 11:44:25 -07:00
|
|
|
tab: &str,
|
2014-01-30 11:30:21 -08:00
|
|
|
structhead: bool) -> fmt::Result {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "{}{}{}",
|
2016-03-25 06:08:11 +00:00
|
|
|
VisSpace(&it.visibility),
|
2016-03-22 17:58:45 -05:00
|
|
|
if structhead {"struct "} else {""},
|
|
|
|
it.name.as_ref().unwrap())?;
|
2016-02-28 12:11:13 +01:00
|
|
|
if let Some(g) = g {
|
2016-07-25 18:19:11 +01:00
|
|
|
write!(w, "{}", g)?
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
match ty {
|
|
|
|
doctree::Plain => {
|
2016-07-25 18:19:11 +01:00
|
|
|
if let Some(g) = g {
|
|
|
|
write!(w, "{}", WhereClause(g))?
|
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, " {{\n{}", tab)?;
|
2015-01-31 12:20:46 -05:00
|
|
|
for field in fields {
|
2016-04-02 08:17:59 +02:00
|
|
|
if let clean::StructFieldItem(ref ty) = field.inner {
|
|
|
|
write!(w, " {}{}: {},\n{}",
|
2016-03-25 06:08:11 +00:00
|
|
|
VisSpace(&field.visibility),
|
2016-04-02 08:17:59 +02:00
|
|
|
field.name.as_ref().unwrap(),
|
|
|
|
*ty,
|
|
|
|
tab)?;
|
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2013-10-13 20:37:43 -07:00
|
|
|
|
2016-04-02 08:17:59 +02:00
|
|
|
if it.has_stripped_fields().unwrap() {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, " // some fields omitted\n{}", tab)?;
|
2013-10-13 20:37:43 -07:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "}}")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
doctree::Tuple | doctree::Newtype => {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "(")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
for (i, field) in fields.iter().enumerate() {
|
2014-01-30 11:30:21 -08:00
|
|
|
if i > 0 {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, ", ")?;
|
2014-01-30 11:30:21 -08:00
|
|
|
}
|
2013-09-18 22:18:38 -07:00
|
|
|
match field.inner {
|
2016-04-02 08:17:59 +02:00
|
|
|
clean::StrippedItem(box clean::StructFieldItem(..)) => {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "_")?
|
2014-04-19 22:24:52 -07:00
|
|
|
}
|
2016-04-02 08:17:59 +02:00
|
|
|
clean::StructFieldItem(ref ty) => {
|
2016-03-25 06:08:11 +00:00
|
|
|
write!(w, "{}{}", VisSpace(&field.visibility), *ty)?
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
_ => unreachable!()
|
|
|
|
}
|
|
|
|
}
|
2016-07-25 18:19:11 +01:00
|
|
|
write!(w, ")")?;
|
|
|
|
if let Some(g) = g {
|
|
|
|
write!(w, "{}", WhereClause(g))?
|
|
|
|
}
|
|
|
|
write!(w, ";")?;
|
2014-01-30 11:30:21 -08:00
|
|
|
}
|
|
|
|
doctree::Unit => {
|
2016-07-25 18:19:11 +01:00
|
|
|
// Needed for PhantomData.
|
|
|
|
if let Some(g) = g {
|
|
|
|
write!(w, "{}", WhereClause(g))?
|
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, ";")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|
2014-01-30 11:30:21 -08:00
|
|
|
Ok(())
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2016-08-10 21:00:17 +03:00
|
|
|
fn render_union(w: &mut fmt::Formatter, it: &clean::Item,
|
|
|
|
g: Option<&clean::Generics>,
|
|
|
|
fields: &[clean::Item],
|
|
|
|
tab: &str,
|
|
|
|
structhead: bool) -> fmt::Result {
|
|
|
|
write!(w, "{}{}{}",
|
|
|
|
VisSpace(&it.visibility),
|
|
|
|
if structhead {"union "} else {""},
|
|
|
|
it.name.as_ref().unwrap())?;
|
|
|
|
if let Some(g) = g {
|
2016-08-18 15:43:24 +03:00
|
|
|
write!(w, "{}", g)?;
|
|
|
|
write!(w, "{}", WhereClause(g))?;
|
2016-08-10 21:00:17 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
write!(w, " {{\n{}", tab)?;
|
|
|
|
for field in fields {
|
|
|
|
if let clean::StructFieldItem(ref ty) = field.inner {
|
|
|
|
write!(w, " {}{}: {},\n{}",
|
|
|
|
VisSpace(&field.visibility),
|
|
|
|
field.name.as_ref().unwrap(),
|
|
|
|
*ty,
|
|
|
|
tab)?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if it.has_stripped_fields().unwrap() {
|
|
|
|
write!(w, " // some fields omitted\n{}", tab)?;
|
|
|
|
}
|
|
|
|
write!(w, "}}")?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2015-04-06 17:56:35 -07:00
|
|
|
#[derive(Copy, Clone)]
|
2016-03-24 06:10:52 +01:00
|
|
|
enum AssocItemLink<'a> {
|
2016-04-16 11:46:52 -04:00
|
|
|
Anchor(Option<&'a str>),
|
2016-08-24 10:55:03 +02:00
|
|
|
GotoSource(DefId, &'a FnvHashSet<String>),
|
2015-04-06 17:56:35 -07:00
|
|
|
}
|
|
|
|
|
2016-04-16 11:46:52 -04:00
|
|
|
impl<'a> AssocItemLink<'a> {
|
|
|
|
fn anchor(&self, id: &'a String) -> Self {
|
|
|
|
match *self {
|
|
|
|
AssocItemLink::Anchor(_) => { AssocItemLink::Anchor(Some(&id)) },
|
|
|
|
ref other => *other,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-14 12:05:00 -06:00
|
|
|
enum AssocItemRender<'a> {
|
2015-04-13 16:23:32 -07:00
|
|
|
All,
|
|
|
|
DerefFor { trait_: &'a clean::Type, type_: &'a clean::Type },
|
|
|
|
}
|
|
|
|
|
2015-03-14 12:05:00 -06:00
|
|
|
fn render_assoc_items(w: &mut fmt::Formatter,
|
2015-08-16 17:05:18 +02:00
|
|
|
cx: &Context,
|
2016-02-09 21:15:29 -05:00
|
|
|
containing_item: &clean::Item,
|
2015-08-16 06:32:28 -04:00
|
|
|
it: DefId,
|
2015-03-14 12:05:00 -06:00
|
|
|
what: AssocItemRender) -> fmt::Result {
|
2015-04-13 16:23:32 -07:00
|
|
|
let c = cache();
|
|
|
|
let v = match c.impls.get(&it) {
|
|
|
|
Some(v) => v,
|
2015-04-06 19:21:52 -07:00
|
|
|
None => return Ok(()),
|
|
|
|
};
|
2015-04-13 16:23:32 -07:00
|
|
|
let (non_trait, traits): (Vec<_>, _) = v.iter().partition(|i| {
|
2016-05-03 19:15:59 +02:00
|
|
|
i.inner_impl().trait_.is_none()
|
2015-04-13 16:23:32 -07:00
|
|
|
});
|
2015-03-24 16:54:09 -07:00
|
|
|
if !non_trait.is_empty() {
|
2015-04-13 16:23:32 -07:00
|
|
|
let render_header = match what {
|
2015-03-14 12:05:00 -06:00
|
|
|
AssocItemRender::All => {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<h2 id='methods'>Methods</h2>")?;
|
2015-04-13 16:23:32 -07:00
|
|
|
true
|
|
|
|
}
|
2015-03-14 12:05:00 -06:00
|
|
|
AssocItemRender::DerefFor { trait_, type_ } => {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<h2 id='deref-methods'>Methods from \
|
2016-03-22 17:58:45 -05:00
|
|
|
{}<Target={}></h2>", trait_, type_)?;
|
2015-04-13 16:23:32 -07:00
|
|
|
false
|
|
|
|
}
|
|
|
|
};
|
2015-04-06 19:21:52 -07:00
|
|
|
for i in &non_trait {
|
2016-04-16 11:46:52 -04:00
|
|
|
render_impl(w, cx, i, AssocItemLink::Anchor(None), render_header,
|
2016-03-22 17:58:45 -05:00
|
|
|
containing_item.stable_since())?;
|
2015-04-06 19:21:52 -07:00
|
|
|
}
|
|
|
|
}
|
2015-03-14 12:05:00 -06:00
|
|
|
if let AssocItemRender::DerefFor { .. } = what {
|
2016-02-28 12:11:13 +01:00
|
|
|
return Ok(());
|
2015-04-13 16:23:32 -07:00
|
|
|
}
|
2015-03-24 16:54:09 -07:00
|
|
|
if !traits.is_empty() {
|
2015-04-13 16:23:32 -07:00
|
|
|
let deref_impl = traits.iter().find(|t| {
|
2016-05-03 19:15:59 +02:00
|
|
|
t.inner_impl().trait_.def_id() == c.deref_trait_did
|
2015-04-13 16:23:32 -07:00
|
|
|
});
|
|
|
|
if let Some(impl_) = deref_impl {
|
2016-03-22 22:01:37 -05:00
|
|
|
render_deref_methods(w, cx, impl_, containing_item)?;
|
2015-04-13 16:23:32 -07:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<h2 id='implementations'>Trait \
|
2016-03-22 17:58:45 -05:00
|
|
|
Implementations</h2>")?;
|
2016-06-05 23:50:18 +01:00
|
|
|
for i in &traits {
|
2015-04-06 19:21:52 -07:00
|
|
|
let did = i.trait_did().unwrap();
|
2016-05-03 19:15:59 +02:00
|
|
|
let assoc_link = AssocItemLink::GotoSource(did, &i.inner_impl().provided_trait_methods);
|
2016-03-24 06:10:52 +01:00
|
|
|
render_impl(w, cx, i, assoc_link, true, containing_item.stable_since())?;
|
2015-04-06 19:21:52 -07:00
|
|
|
}
|
2014-04-28 20:36:08 -07:00
|
|
|
}
|
|
|
|
Ok(())
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2016-02-09 21:15:29 -05:00
|
|
|
fn render_deref_methods(w: &mut fmt::Formatter, cx: &Context, impl_: &Impl,
|
|
|
|
container_item: &clean::Item) -> fmt::Result {
|
2016-05-03 19:15:59 +02:00
|
|
|
let deref_type = impl_.inner_impl().trait_.as_ref().unwrap();
|
|
|
|
let target = impl_.inner_impl().items.iter().filter_map(|item| {
|
2015-04-13 16:23:32 -07:00
|
|
|
match item.inner {
|
2015-05-21 14:17:37 +02:00
|
|
|
clean::TypedefItem(ref t, true) => Some(&t.type_),
|
2015-04-13 16:23:32 -07:00
|
|
|
_ => None,
|
|
|
|
}
|
2015-05-21 14:17:37 +02:00
|
|
|
}).next().expect("Expected associated type binding");
|
2015-03-14 12:05:00 -06:00
|
|
|
let what = AssocItemRender::DerefFor { trait_: deref_type, type_: target };
|
2016-03-24 06:10:52 +01:00
|
|
|
if let Some(did) = target.def_id() {
|
|
|
|
render_assoc_items(w, cx, container_item, did, what)
|
|
|
|
} else {
|
|
|
|
if let Some(prim) = target.primitive_type() {
|
|
|
|
if let Some(c) = cache().primitive_locations.get(&prim) {
|
|
|
|
let did = DefId { krate: *c, index: prim.to_def_index() };
|
|
|
|
render_assoc_items(w, cx, container_item, did, what)?;
|
2015-04-13 16:23:32 -07:00
|
|
|
}
|
|
|
|
}
|
2016-03-24 06:10:52 +01:00
|
|
|
Ok(())
|
2014-05-03 02:08:58 -07:00
|
|
|
}
|
2015-04-13 16:23:32 -07:00
|
|
|
}
|
|
|
|
|
2015-04-26 21:03:38 +02:00
|
|
|
// Render_header is false when we are rendering a `Deref` impl and true
|
|
|
|
// otherwise. If render_header is false, we will avoid rendering static
|
|
|
|
// methods, since they are not accessible for the type implementing `Deref`
|
2015-08-16 17:05:18 +02:00
|
|
|
fn render_impl(w: &mut fmt::Formatter, cx: &Context, i: &Impl, link: AssocItemLink,
|
2016-02-09 21:15:29 -05:00
|
|
|
render_header: bool, outer_version: Option<&str>) -> fmt::Result {
|
2015-04-13 16:23:32 -07:00
|
|
|
if render_header {
|
2016-05-03 19:15:59 +02:00
|
|
|
write!(w, "<h3 class='impl'><span class='in-band'><code>{}</code>", i.inner_impl())?;
|
|
|
|
write!(w, "</span><span class='out-of-band'>")?;
|
2016-05-06 18:47:12 +02:00
|
|
|
let since = i.impl_item.stability.as_ref().map(|s| &s.since[..]);
|
2016-05-03 19:15:59 +02:00
|
|
|
if let Some(l) = (Item { item: &i.impl_item, cx: cx }).href() {
|
2016-05-06 18:47:12 +02:00
|
|
|
write!(w, "<div class='ghost'></div>")?;
|
|
|
|
render_stability_since_raw(w, since, outer_version)?;
|
2016-05-03 19:15:59 +02:00
|
|
|
write!(w, "<a id='src-{}' class='srclink' \
|
|
|
|
href='{}' title='{}'>[src]</a>",
|
|
|
|
i.impl_item.def_id.index.as_usize(), l, "goto source code")?;
|
2016-05-06 18:47:12 +02:00
|
|
|
} else {
|
|
|
|
render_stability_since_raw(w, since, outer_version)?;
|
2016-05-03 19:15:59 +02:00
|
|
|
}
|
|
|
|
write!(w, "</span>")?;
|
|
|
|
write!(w, "</h3>\n")?;
|
|
|
|
if let Some(ref dox) = i.impl_item.attrs.value("doc") {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<div class='docblock'>{}</div>", Markdown(dox))?;
|
2015-04-13 16:23:32 -07:00
|
|
|
}
|
2013-09-30 17:04:14 -07:00
|
|
|
}
|
2013-10-21 11:33:04 -07:00
|
|
|
|
2016-08-16 10:27:07 +12:00
|
|
|
fn doc_impl_item(w: &mut fmt::Formatter, cx: &Context, item: &clean::Item,
|
|
|
|
link: AssocItemLink, render_static: bool,
|
|
|
|
is_default_item: bool, outer_version: Option<&str>,
|
|
|
|
trait_: Option<&clean::Trait>) -> fmt::Result {
|
2016-08-03 13:14:59 +12:00
|
|
|
let item_type = item_type(item);
|
2015-12-02 23:49:18 +01:00
|
|
|
let name = item.name.as_ref().unwrap();
|
2016-02-28 12:11:13 +01:00
|
|
|
|
|
|
|
let is_static = match item.inner {
|
2016-05-08 21:19:29 +03:00
|
|
|
clean::MethodItem(ref method) => !method.decl.has_self(),
|
|
|
|
clean::TyMethodItem(ref method) => !method.decl.has_self(),
|
2016-02-28 12:11:13 +01:00
|
|
|
_ => false
|
|
|
|
};
|
|
|
|
|
2014-11-20 11:22:09 -08:00
|
|
|
match item.inner {
|
|
|
|
clean::MethodItem(..) | clean::TyMethodItem(..) => {
|
2015-04-26 21:03:38 +02:00
|
|
|
// Only render when the method is not static or we allow static methods
|
2016-02-28 12:11:13 +01:00
|
|
|
if !is_static || render_static {
|
2016-08-03 13:14:59 +12:00
|
|
|
let id = derive_id(format!("{}.{}", item_type, name));
|
2016-08-16 10:27:07 +12:00
|
|
|
let ns_id = derive_id(format!("{}.{}", name, item_type.name_space()));
|
2016-08-03 13:14:59 +12:00
|
|
|
write!(w, "<h4 id='{}' class='{}'>", id, item_type)?;
|
2016-08-16 10:27:07 +12:00
|
|
|
write!(w, "<span id='{}' class='invisible'>", ns_id)?;
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<code>")?;
|
2016-04-16 11:46:52 -04:00
|
|
|
render_assoc_item(w, item, link.anchor(&id))?;
|
2016-04-22 10:16:13 +02:00
|
|
|
write!(w, "</code>")?;
|
|
|
|
render_stability_since_raw(w, item.stable_since(), outer_version)?;
|
2016-08-16 10:27:07 +12:00
|
|
|
write!(w, "</span></h4>\n")?;
|
2015-04-26 21:03:38 +02:00
|
|
|
}
|
2014-11-20 11:22:09 -08:00
|
|
|
}
|
2015-05-21 14:17:37 +02:00
|
|
|
clean::TypedefItem(ref tydef, _) => {
|
2016-03-24 06:16:23 +01:00
|
|
|
let id = derive_id(format!("{}.{}", ItemType::AssociatedType, name));
|
2016-08-16 10:27:07 +12:00
|
|
|
let ns_id = derive_id(format!("{}.{}", name, item_type.name_space()));
|
2016-08-03 13:56:18 +12:00
|
|
|
write!(w, "<h4 id='{}' class='{}'>", id, item_type)?;
|
2016-08-16 10:27:07 +12:00
|
|
|
write!(w, "<span id='{}' class='invisible'><code>", ns_id)?;
|
2016-04-16 11:46:52 -04:00
|
|
|
assoc_type(w, item, &Vec::new(), Some(&tydef.type_), link.anchor(&id))?;
|
2016-08-16 10:27:07 +12:00
|
|
|
write!(w, "</code></span></h4>\n")?;
|
2014-11-20 11:22:09 -08:00
|
|
|
}
|
2015-03-15 19:35:25 -06:00
|
|
|
clean::AssociatedConstItem(ref ty, ref default) => {
|
2016-08-03 13:14:59 +12:00
|
|
|
let id = derive_id(format!("{}.{}", item_type, name));
|
2016-08-16 10:27:07 +12:00
|
|
|
let ns_id = derive_id(format!("{}.{}", name, item_type.name_space()));
|
2016-08-03 13:56:18 +12:00
|
|
|
write!(w, "<h4 id='{}' class='{}'>", id, item_type)?;
|
2016-08-16 10:27:07 +12:00
|
|
|
write!(w, "<span id='{}' class='invisible'><code>", ns_id)?;
|
2016-04-16 11:46:52 -04:00
|
|
|
assoc_const(w, item, ty, default.as_ref(), link.anchor(&id))?;
|
2016-08-16 10:27:07 +12:00
|
|
|
write!(w, "</code></span></h4>\n")?;
|
2015-04-30 09:37:13 -07:00
|
|
|
}
|
|
|
|
clean::ConstantItem(ref c) => {
|
2016-08-03 13:14:59 +12:00
|
|
|
let id = derive_id(format!("{}.{}", item_type, name));
|
2016-08-16 10:27:07 +12:00
|
|
|
let ns_id = derive_id(format!("{}.{}", name, item_type.name_space()));
|
2016-08-03 13:56:18 +12:00
|
|
|
write!(w, "<h4 id='{}' class='{}'>", id, item_type)?;
|
2016-08-16 10:27:07 +12:00
|
|
|
write!(w, "<span id='{}' class='invisible'><code>", ns_id)?;
|
2016-04-16 11:46:52 -04:00
|
|
|
assoc_const(w, item, &c.type_, Some(&c.expr), link.anchor(&id))?;
|
2016-08-16 10:27:07 +12:00
|
|
|
write!(w, "</code></span></h4>\n")?;
|
2015-03-15 19:35:25 -06:00
|
|
|
}
|
2015-03-10 12:28:44 +02:00
|
|
|
clean::AssociatedTypeItem(ref bounds, ref default) => {
|
2016-08-03 13:14:59 +12:00
|
|
|
let id = derive_id(format!("{}.{}", item_type, name));
|
2016-08-16 10:27:07 +12:00
|
|
|
let ns_id = derive_id(format!("{}.{}", name, item_type.name_space()));
|
2016-08-03 13:56:18 +12:00
|
|
|
write!(w, "<h4 id='{}' class='{}'>", id, item_type)?;
|
2016-08-16 10:27:07 +12:00
|
|
|
write!(w, "<span id='{}' class='invisible'><code>", ns_id)?;
|
2016-04-16 11:46:52 -04:00
|
|
|
assoc_type(w, item, bounds, default.as_ref(), link.anchor(&id))?;
|
2016-08-16 10:27:07 +12:00
|
|
|
write!(w, "</code></span></h4>\n")?;
|
2015-01-02 08:26:55 -05:00
|
|
|
}
|
2016-03-31 18:15:54 +02:00
|
|
|
clean::StrippedItem(..) => return Ok(()),
|
2014-12-20 00:09:35 -08:00
|
|
|
_ => panic!("can't make docs for trait item with name {:?}", item.name)
|
2014-11-20 11:22:09 -08:00
|
|
|
}
|
2015-04-26 21:03:38 +02:00
|
|
|
|
2016-05-19 00:07:58 +05:30
|
|
|
if !is_static || render_static {
|
|
|
|
if !is_default_item {
|
2016-06-15 23:55:11 +01:00
|
|
|
if let Some(t) = trait_ {
|
2016-06-23 23:17:35 +01:00
|
|
|
// The trait item may have been stripped so we might not
|
|
|
|
// find any documentation or stability for it.
|
|
|
|
if let Some(it) = t.items.iter().find(|i| i.name == item.name) {
|
|
|
|
// We need the stability of the item from the trait
|
|
|
|
// because impls can't have a stability.
|
|
|
|
document_stability(w, cx, it)?;
|
|
|
|
if item.doc_value().is_some() {
|
|
|
|
document_full(w, item)?;
|
|
|
|
} else {
|
|
|
|
// In case the item isn't documented,
|
|
|
|
// provide short documentation from the trait.
|
|
|
|
document_short(w, it, link)?;
|
|
|
|
}
|
2016-05-17 07:31:16 +05:30
|
|
|
}
|
2016-06-15 23:55:11 +01:00
|
|
|
} else {
|
|
|
|
document(w, cx, item)?;
|
2016-05-17 07:31:16 +05:30
|
|
|
}
|
2016-05-19 00:07:58 +05:30
|
|
|
} else {
|
2016-06-15 23:55:11 +01:00
|
|
|
document_stability(w, cx, item)?;
|
2016-05-19 00:07:58 +05:30
|
|
|
document_short(w, item, link)?;
|
2016-05-17 07:31:16 +05:30
|
|
|
}
|
2013-10-21 11:33:04 -07:00
|
|
|
}
|
2016-05-19 00:07:58 +05:30
|
|
|
Ok(())
|
2013-10-21 11:33:04 -07:00
|
|
|
}
|
|
|
|
|
2016-05-17 07:31:16 +05:30
|
|
|
let traits = &cache().traits;
|
|
|
|
let trait_ = i.trait_did().and_then(|did| traits.get(&did));
|
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<div class='impl-items'>")?;
|
2016-05-03 19:15:59 +02:00
|
|
|
for trait_item in &i.inner_impl().items {
|
2016-08-16 10:27:07 +12:00
|
|
|
doc_impl_item(w, cx, trait_item, link, render_header,
|
|
|
|
false, outer_version, trait_)?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2013-10-21 11:33:04 -07:00
|
|
|
|
2015-03-14 12:05:00 -06:00
|
|
|
fn render_default_items(w: &mut fmt::Formatter,
|
2015-08-16 17:05:18 +02:00
|
|
|
cx: &Context,
|
2015-03-14 12:05:00 -06:00
|
|
|
t: &clean::Trait,
|
2016-03-24 06:10:52 +01:00
|
|
|
i: &clean::Impl,
|
|
|
|
render_static: bool,
|
|
|
|
outer_version: Option<&str>) -> fmt::Result {
|
2015-01-31 12:20:46 -05:00
|
|
|
for trait_item in &t.items {
|
2015-03-10 12:28:44 +02:00
|
|
|
let n = trait_item.name.clone();
|
2016-03-24 06:10:52 +01:00
|
|
|
if i.items.iter().find(|m| m.name == n).is_some() {
|
2016-02-28 12:11:13 +01:00
|
|
|
continue;
|
2014-05-03 02:08:58 -07:00
|
|
|
}
|
2016-03-24 06:10:52 +01:00
|
|
|
let did = i.trait_.as_ref().unwrap().def_id().unwrap();
|
|
|
|
let assoc_link = AssocItemLink::GotoSource(did, &i.provided_trait_methods);
|
2014-05-03 02:08:58 -07:00
|
|
|
|
2016-08-16 10:27:07 +12:00
|
|
|
doc_impl_item(w, cx, trait_item, assoc_link, render_static, true,
|
|
|
|
outer_version, None)?;
|
2014-05-03 02:08:58 -07:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2013-10-21 11:33:04 -07:00
|
|
|
// If we've implemented a trait, then also emit documentation for all
|
2016-03-25 00:10:15 +01:00
|
|
|
// default items which weren't overridden in the implementation block.
|
2016-05-17 07:31:16 +05:30
|
|
|
if let Some(t) = trait_ {
|
|
|
|
render_default_items(w, cx, t, &i.inner_impl(), render_header, outer_version)?;
|
2013-10-21 11:33:04 -07:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "</div>")?;
|
2014-01-30 11:30:21 -08:00
|
|
|
Ok(())
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2015-08-16 17:05:18 +02:00
|
|
|
fn item_typedef(w: &mut fmt::Formatter, cx: &Context, it: &clean::Item,
|
2014-01-30 11:30:21 -08:00
|
|
|
t: &clean::Typedef) -> fmt::Result {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(w, "<pre class='rust typedef'>type {}{}{where_clause} = {type_};</pre>",
|
2016-03-22 17:58:45 -05:00
|
|
|
it.name.as_ref().unwrap(),
|
|
|
|
t.generics,
|
|
|
|
where_clause = WhereClause(&t.generics),
|
|
|
|
type_ = t.type_)?;
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2015-08-16 17:05:18 +02:00
|
|
|
document(w, cx, it)
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2015-01-20 15:45:07 -08:00
|
|
|
impl<'a> fmt::Display for Sidebar<'a> {
|
2014-02-05 23:55:13 +11:00
|
|
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
let cx = self.cx;
|
|
|
|
let it = self.item;
|
2015-03-05 16:35:43 +09:00
|
|
|
let parentlen = cx.current.len() - if it.is_mod() {1} else {0};
|
|
|
|
|
2015-03-05 23:10:15 +09:00
|
|
|
// the sidebar is designed to display sibling functions, modules and
|
2016-07-03 10:00:52 +02:00
|
|
|
// other miscellaneous information. since there are lots of sibling
|
2015-03-05 23:10:15 +09:00
|
|
|
// items (and that causes quadratic growth in large modules),
|
|
|
|
// we refactor common parts into a shared JavaScript file per module.
|
|
|
|
// still, we don't move everything into JS because we want to preserve
|
|
|
|
// as much HTML as possible in order to allow non-JS-enabled browsers
|
|
|
|
// to navigate the documentation (though slightly inefficiently).
|
|
|
|
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "<p class='location'>")?;
|
2015-03-05 16:35:43 +09:00
|
|
|
for (i, name) in cx.current.iter().take(parentlen).enumerate() {
|
2014-01-30 11:30:21 -08:00
|
|
|
if i > 0 {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "::<wbr>")?;
|
2014-01-30 11:30:21 -08:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "<a href='{}index.html'>{}</a>",
|
2016-03-22 17:58:45 -05:00
|
|
|
&cx.root_path[..(cx.current.len() - i - 1) * 3],
|
|
|
|
*name)?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "</p>")?;
|
2013-09-18 22:18:38 -07:00
|
|
|
|
2015-03-05 16:35:43 +09:00
|
|
|
// sidebar refers to the enclosing module, not this module
|
2016-02-28 12:23:07 +01:00
|
|
|
let relpath = if it.is_mod() { "../" } else { "" };
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt,
|
2016-03-22 17:58:45 -05:00
|
|
|
"<script>window.sidebarCurrent = {{\
|
|
|
|
name: '{name}', \
|
|
|
|
ty: '{ty}', \
|
|
|
|
relpath: '{path}'\
|
|
|
|
}};</script>",
|
|
|
|
name = it.name.as_ref().map(|x| &x[..]).unwrap_or(""),
|
2016-08-03 13:14:59 +12:00
|
|
|
ty = item_type(it).css_class(),
|
2016-03-22 17:58:45 -05:00
|
|
|
path = relpath)?;
|
2015-03-05 16:35:43 +09:00
|
|
|
if parentlen == 0 {
|
|
|
|
// there is no sidebar-items.js beyond the crate root path
|
|
|
|
// FIXME maybe dynamic crate loading can be merged here
|
|
|
|
} else {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "<script defer src=\"{path}sidebar-items.js\"></script>",
|
2016-03-22 17:58:45 -05:00
|
|
|
path = relpath)?;
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
|
2014-01-30 11:30:21 -08:00
|
|
|
Ok(())
|
2013-09-18 22:18:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-20 15:45:07 -08:00
|
|
|
impl<'a> fmt::Display for Source<'a> {
|
2014-02-05 23:55:13 +11:00
|
|
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
let Source(s) = *self;
|
2014-06-05 23:18:51 -07:00
|
|
|
let lines = s.lines().count();
|
2013-09-27 15:12:23 -07:00
|
|
|
let mut cols = 0;
|
|
|
|
let mut tmp = lines;
|
|
|
|
while tmp > 0 {
|
|
|
|
cols += 1;
|
|
|
|
tmp /= 10;
|
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "<pre class=\"line-numbers\">")?;
|
2015-01-26 15:46:12 -05:00
|
|
|
for i in 1..lines + 1 {
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "<span id=\"{0}\">{0:1$}</span>\n", i, cols)?;
|
2013-09-27 15:12:23 -07:00
|
|
|
}
|
2016-03-22 22:01:37 -05:00
|
|
|
write!(fmt, "</pre>")?;
|
2016-04-04 11:07:41 +12:00
|
|
|
write!(fmt, "{}", highlight::render_with_highlighting(s, None, None))?;
|
2014-01-30 11:30:21 -08:00
|
|
|
Ok(())
|
2013-09-27 15:12:23 -07:00
|
|
|
}
|
2013-09-26 11:09:47 -07:00
|
|
|
}
|
2014-02-16 21:40:26 -08:00
|
|
|
|
2015-08-16 17:05:18 +02:00
|
|
|
fn item_macro(w: &mut fmt::Formatter, cx: &Context, it: &clean::Item,
|
2014-02-16 21:40:26 -08:00
|
|
|
t: &clean::Macro) -> fmt::Result {
|
2016-04-04 11:07:41 +12:00
|
|
|
w.write_str(&highlight::render_with_highlighting(&t.source,
|
|
|
|
Some("macro"),
|
|
|
|
None))?;
|
2016-03-22 22:01:37 -05:00
|
|
|
render_stability_since_raw(w, it.stable_since(), None)?;
|
2015-08-16 17:05:18 +02:00
|
|
|
document(w, cx, it)
|
2014-02-16 21:40:26 -08:00
|
|
|
}
|
2014-05-28 19:53:37 -07:00
|
|
|
|
2015-08-16 17:05:18 +02:00
|
|
|
fn item_primitive(w: &mut fmt::Formatter, cx: &Context,
|
2014-05-28 19:53:37 -07:00
|
|
|
it: &clean::Item,
|
2014-09-11 17:07:49 +12:00
|
|
|
_p: &clean::PrimitiveType) -> fmt::Result {
|
2016-03-22 22:01:37 -05:00
|
|
|
document(w, cx, it)?;
|
2016-02-09 21:15:29 -05:00
|
|
|
render_assoc_items(w, cx, it, it.def_id, AssocItemRender::All)
|
2014-05-28 19:53:37 -07:00
|
|
|
}
|
2014-05-29 13:50:47 -07:00
|
|
|
|
2016-02-28 14:01:43 +01:00
|
|
|
const BASIC_KEYWORDS: &'static str = "rust, rustlang, rust-lang";
|
2014-08-04 14:30:06 -07:00
|
|
|
|
|
|
|
fn make_item_keywords(it: &clean::Item) -> String {
|
2016-02-28 14:01:43 +01:00
|
|
|
format!("{}, {}", BASIC_KEYWORDS, it.name.as_ref().unwrap())
|
2014-08-04 14:30:06 -07:00
|
|
|
}
|
2014-11-14 14:20:57 -08:00
|
|
|
|
2016-05-08 21:19:29 +03:00
|
|
|
fn get_index_search_type(item: &clean::Item) -> Option<IndexItemFunctionType> {
|
|
|
|
let decl = match item.inner {
|
|
|
|
clean::FunctionItem(ref f) => &f.decl,
|
|
|
|
clean::MethodItem(ref m) => &m.decl,
|
|
|
|
clean::TyMethodItem(ref m) => &m.decl,
|
2015-02-26 01:03:06 +02:00
|
|
|
_ => return None
|
|
|
|
};
|
|
|
|
|
2016-05-08 21:19:29 +03:00
|
|
|
let inputs = decl.inputs.values.iter().map(|arg| get_index_type(&arg.type_)).collect();
|
2015-02-26 01:03:06 +02:00
|
|
|
let output = match decl.output {
|
|
|
|
clean::FunctionRetTy::Return(ref return_type) => Some(get_index_type(return_type)),
|
|
|
|
_ => None
|
|
|
|
};
|
|
|
|
|
|
|
|
Some(IndexItemFunctionType { inputs: inputs, output: output })
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_index_type(clean_type: &clean::Type) -> Type {
|
2015-07-08 08:33:13 -07:00
|
|
|
Type { name: get_index_type_name(clean_type).map(|s| s.to_ascii_lowercase()) }
|
2015-02-26 01:03:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fn get_index_type_name(clean_type: &clean::Type) -> Option<String> {
|
|
|
|
match *clean_type {
|
|
|
|
clean::ResolvedPath { ref path, .. } => {
|
|
|
|
let segments = &path.segments;
|
|
|
|
Some(segments[segments.len() - 1].name.clone())
|
|
|
|
},
|
|
|
|
clean::Generic(ref s) => Some(s.clone()),
|
|
|
|
clean::Primitive(ref p) => Some(format!("{:?}", p)),
|
|
|
|
clean::BorrowedRef { ref type_, .. } => get_index_type_name(type_),
|
|
|
|
// FIXME: add all from clean::Type.
|
|
|
|
_ => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-14 14:20:57 -08:00
|
|
|
pub fn cache() -> Arc<Cache> {
|
|
|
|
CACHE_KEY.with(|c| c.borrow().clone())
|
|
|
|
}
|
2015-12-05 23:09:20 +01:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
#[test]
|
|
|
|
fn test_unique_id() {
|
|
|
|
let input = ["foo", "examples", "examples", "method.into_iter","examples",
|
|
|
|
"method.into_iter", "foo", "main", "search", "methods",
|
|
|
|
"examples", "method.into_iter", "assoc_type.Item", "assoc_type.Item"];
|
|
|
|
let expected = ["foo", "examples", "examples-1", "method.into_iter", "examples-2",
|
|
|
|
"method.into_iter-1", "foo-1", "main-1", "search-1", "methods-1",
|
|
|
|
"examples-3", "method.into_iter-2", "assoc_type.Item", "assoc_type.Item-1"];
|
|
|
|
|
|
|
|
let test = || {
|
|
|
|
let actual: Vec<String> = input.iter().map(|s| derive_id(s.to_string())).collect();
|
|
|
|
assert_eq!(&actual[..], expected);
|
|
|
|
};
|
|
|
|
test();
|
2016-03-26 16:42:38 +01:00
|
|
|
reset_ids(true);
|
2015-12-05 23:09:20 +01:00
|
|
|
test();
|
|
|
|
}
|