mirror of
https://github.com/denoland/deno.git
synced 2025-01-15 18:38:53 -05:00
feat: Added colors to doc output (#4518)
This commit is contained in:
parent
d6cb3892d4
commit
ad198b1cf1
3 changed files with 168 additions and 85 deletions
|
@ -3,7 +3,7 @@ use regex::Regex;
|
||||||
use std::env;
|
use std::env;
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
use std::io::Write;
|
use std::io::Write;
|
||||||
use termcolor::Color::{Ansi256, Black, Red, White};
|
use termcolor::Color::{Ansi256, Black, Magenta, Red, White};
|
||||||
use termcolor::{Ansi, ColorSpec, WriteColor};
|
use termcolor::{Ansi, ColorSpec, WriteColor};
|
||||||
|
|
||||||
#[cfg(windows)]
|
#[cfg(windows)]
|
||||||
|
@ -88,6 +88,12 @@ pub fn green(s: String) -> impl fmt::Display {
|
||||||
style(&s, style_spec)
|
style(&s, style_spec)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn magenta(s: String) -> impl fmt::Display {
|
||||||
|
let mut style_spec = ColorSpec::new();
|
||||||
|
style_spec.set_fg(Some(Magenta));
|
||||||
|
style(&s, style_spec)
|
||||||
|
}
|
||||||
|
|
||||||
pub fn bold(s: String) -> impl fmt::Display {
|
pub fn bold(s: String) -> impl fmt::Display {
|
||||||
let mut style_spec = ColorSpec::new();
|
let mut style_spec = ColorSpec::new();
|
||||||
style_spec.set_bold(true);
|
style_spec.set_bold(true);
|
||||||
|
|
|
@ -10,6 +10,7 @@
|
||||||
// unnecessary and can result in unnecessary copying. Instead they should take
|
// unnecessary and can result in unnecessary copying. Instead they should take
|
||||||
// references.
|
// references.
|
||||||
|
|
||||||
|
use crate::colors;
|
||||||
use crate::doc;
|
use crate::doc;
|
||||||
use crate::doc::ts_type::TsTypeDefKind;
|
use crate::doc::ts_type::TsTypeDefKind;
|
||||||
use crate::doc::DocNodeKind;
|
use crate::doc::DocNodeKind;
|
||||||
|
@ -22,8 +23,11 @@ pub fn format_details(node: doc::DocNode) -> String {
|
||||||
let mut details = String::new();
|
let mut details = String::new();
|
||||||
|
|
||||||
details.push_str(&format!(
|
details.push_str(&format!(
|
||||||
"Defined in {}:{}:{}.\n",
|
"{}",
|
||||||
|
colors::gray(format!(
|
||||||
|
"Defined in {}:{}:{} \n\n",
|
||||||
node.location.filename, node.location.line, node.location.col
|
node.location.filename, node.location.line, node.location.col
|
||||||
|
))
|
||||||
));
|
));
|
||||||
|
|
||||||
details.push_str(&format_signature(&node, 0));
|
details.push_str(&format_signature(&node, 0));
|
||||||
|
@ -275,11 +279,12 @@ fn render_ts_type(ts_type: doc::ts_type::TsTypeDef) -> String {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn format_indent(indent: i64) -> String {
|
fn add_indent(string: String, indent: i64) -> String {
|
||||||
let mut indent_str = String::new();
|
let mut indent_str = String::new();
|
||||||
for _ in 0..indent {
|
for _ in 0..(indent * 2) {
|
||||||
indent_str.push_str(" ");
|
indent_str += " ";
|
||||||
}
|
}
|
||||||
|
indent_str += string.as_str();
|
||||||
indent_str
|
indent_str
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -291,15 +296,13 @@ fn format_jsdoc(jsdoc: String, truncated: bool, indent: i64) -> String {
|
||||||
|
|
||||||
if truncated {
|
if truncated {
|
||||||
let first_line = lines.next().unwrap_or_else(|| "".to_string());
|
let first_line = lines.next().unwrap_or_else(|| "".to_string());
|
||||||
js_doc.push_str(&format_indent(indent + 1));
|
js_doc.push_str(&add_indent(format!("{}\n", first_line), indent + 1));
|
||||||
js_doc.push_str(&format!("{}\n", first_line));
|
|
||||||
} else {
|
} else {
|
||||||
for line in lines {
|
for line in lines {
|
||||||
js_doc.push_str(&format_indent(indent + 1));
|
js_doc.push_str(&add_indent(format!("{}\n", line), indent + 1));
|
||||||
js_doc.push_str(&format!("{}\n", line));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
js_doc
|
format!("{}", colors::gray(js_doc))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn format_class_details(node: doc::DocNode) -> String {
|
fn format_class_details(node: doc::DocNode) -> String {
|
||||||
|
@ -307,10 +310,14 @@ fn format_class_details(node: doc::DocNode) -> String {
|
||||||
|
|
||||||
let class_def = node.class_def.unwrap();
|
let class_def = node.class_def.unwrap();
|
||||||
for node in class_def.constructors {
|
for node in class_def.constructors {
|
||||||
details.push_str(&format!(
|
details.push_str(&add_indent(
|
||||||
"constructor {}({})\n",
|
format!(
|
||||||
node.name,
|
"{} {}({})\n",
|
||||||
|
colors::magenta("constructor".to_string()),
|
||||||
|
colors::bold(node.name),
|
||||||
render_params(node.params),
|
render_params(node.params),
|
||||||
|
),
|
||||||
|
1,
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
for node in class_def.properties.iter().filter(|node| {
|
for node in class_def.properties.iter().filter(|node| {
|
||||||
|
@ -319,18 +326,22 @@ fn format_class_details(node: doc::DocNode) -> String {
|
||||||
.unwrap_or(swc_ecma_ast::Accessibility::Public)
|
.unwrap_or(swc_ecma_ast::Accessibility::Public)
|
||||||
!= swc_ecma_ast::Accessibility::Private
|
!= swc_ecma_ast::Accessibility::Private
|
||||||
}) {
|
}) {
|
||||||
details.push_str(&format!(
|
details.push_str(&add_indent(
|
||||||
"{} {}: {}\n",
|
format!(
|
||||||
|
"{}{}: {}\n",
|
||||||
|
colors::magenta(
|
||||||
match node
|
match node
|
||||||
.accessibility
|
.accessibility
|
||||||
.unwrap_or(swc_ecma_ast::Accessibility::Public)
|
.unwrap_or(swc_ecma_ast::Accessibility::Public)
|
||||||
{
|
{
|
||||||
swc_ecma_ast::Accessibility::Protected => "protected".to_string(),
|
swc_ecma_ast::Accessibility::Protected => "protected ".to_string(),
|
||||||
swc_ecma_ast::Accessibility::Public => "public".to_string(),
|
|
||||||
_ => "".to_string(),
|
_ => "".to_string(),
|
||||||
},
|
}
|
||||||
node.name,
|
),
|
||||||
|
colors::bold(node.name.clone()),
|
||||||
render_ts_type(node.ts_type.clone().unwrap())
|
render_ts_type(node.ts_type.clone().unwrap())
|
||||||
|
),
|
||||||
|
1,
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
for node in class_def.methods.iter().filter(|node| {
|
for node in class_def.methods.iter().filter(|node| {
|
||||||
|
@ -340,24 +351,28 @@ fn format_class_details(node: doc::DocNode) -> String {
|
||||||
!= swc_ecma_ast::Accessibility::Private
|
!= swc_ecma_ast::Accessibility::Private
|
||||||
}) {
|
}) {
|
||||||
let function_def = node.function_def.clone();
|
let function_def = node.function_def.clone();
|
||||||
details.push_str(&format!(
|
details.push_str(&add_indent(
|
||||||
"{} {}{}({}): {}\n",
|
format!(
|
||||||
|
"{}{}{}({}): {}\n",
|
||||||
|
colors::magenta(
|
||||||
match node
|
match node
|
||||||
.accessibility
|
.accessibility
|
||||||
.unwrap_or(swc_ecma_ast::Accessibility::Public)
|
.unwrap_or(swc_ecma_ast::Accessibility::Public)
|
||||||
{
|
{
|
||||||
swc_ecma_ast::Accessibility::Protected => "protected".to_string(),
|
swc_ecma_ast::Accessibility::Protected => "protected ".to_string(),
|
||||||
swc_ecma_ast::Accessibility::Public => "public".to_string(),
|
|
||||||
_ => "".to_string(),
|
_ => "".to_string(),
|
||||||
},
|
}
|
||||||
match node.kind {
|
),
|
||||||
|
colors::magenta(match node.kind {
|
||||||
swc_ecma_ast::MethodKind::Getter => "get ".to_string(),
|
swc_ecma_ast::MethodKind::Getter => "get ".to_string(),
|
||||||
swc_ecma_ast::MethodKind::Setter => "set ".to_string(),
|
swc_ecma_ast::MethodKind::Setter => "set ".to_string(),
|
||||||
_ => "".to_string(),
|
_ => "".to_string(),
|
||||||
},
|
}),
|
||||||
node.name,
|
colors::bold(node.name.clone()),
|
||||||
render_params(function_def.params),
|
render_params(function_def.params),
|
||||||
render_ts_type(function_def.return_type.unwrap())
|
render_ts_type(function_def.return_type.unwrap())
|
||||||
|
),
|
||||||
|
1,
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
details.push_str("\n");
|
details.push_str("\n");
|
||||||
|
@ -369,64 +384,102 @@ fn format_namespace_details(node: doc::DocNode) -> String {
|
||||||
|
|
||||||
let elements = node.namespace_def.unwrap().elements;
|
let elements = node.namespace_def.unwrap().elements;
|
||||||
for node in elements {
|
for node in elements {
|
||||||
ns.push_str(&format_signature(&node, 0));
|
ns.push_str(&format_signature(&node, 1));
|
||||||
}
|
}
|
||||||
ns.push_str("\n");
|
ns.push_str("\n");
|
||||||
ns
|
ns
|
||||||
}
|
}
|
||||||
|
|
||||||
fn format_function_signature(node: &doc::DocNode, indent: i64) -> String {
|
fn format_function_signature(node: &doc::DocNode, indent: i64) -> String {
|
||||||
format_indent(indent);
|
|
||||||
let function_def = node.function_def.clone().unwrap();
|
let function_def = node.function_def.clone().unwrap();
|
||||||
let return_type = function_def.return_type.unwrap();
|
let return_type = function_def.return_type.unwrap();
|
||||||
|
add_indent(
|
||||||
format!(
|
format!(
|
||||||
"function {}({}): {}\n",
|
"{} {}{}\n",
|
||||||
node.name,
|
colors::magenta("function".to_string()),
|
||||||
|
colors::bold(node.name.clone()),
|
||||||
|
format!(
|
||||||
|
"({}): {}",
|
||||||
render_params(function_def.params),
|
render_params(function_def.params),
|
||||||
render_ts_type(return_type).as_str()
|
render_ts_type(return_type).as_str()
|
||||||
)
|
)
|
||||||
|
),
|
||||||
|
indent,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn format_class_signature(node: &doc::DocNode, indent: i64) -> String {
|
fn format_class_signature(node: &doc::DocNode, indent: i64) -> String {
|
||||||
format_indent(indent);
|
add_indent(
|
||||||
format!("class {}\n", node.name)
|
format!(
|
||||||
|
"{} {}\n",
|
||||||
|
colors::magenta("class".to_string()),
|
||||||
|
colors::bold(node.name.clone())
|
||||||
|
),
|
||||||
|
indent,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn format_variable_signature(node: &doc::DocNode, indent: i64) -> String {
|
fn format_variable_signature(node: &doc::DocNode, indent: i64) -> String {
|
||||||
format_indent(indent);
|
|
||||||
let variable_def = node.variable_def.clone().unwrap();
|
let variable_def = node.variable_def.clone().unwrap();
|
||||||
|
add_indent(
|
||||||
format!(
|
format!(
|
||||||
"{} {}{}\n",
|
"{} {}{}\n",
|
||||||
match variable_def.kind {
|
colors::magenta(match variable_def.kind {
|
||||||
swc_ecma_ast::VarDeclKind::Const => "const".to_string(),
|
swc_ecma_ast::VarDeclKind::Const => "const".to_string(),
|
||||||
swc_ecma_ast::VarDeclKind::Let => "let".to_string(),
|
swc_ecma_ast::VarDeclKind::Let => "let".to_string(),
|
||||||
swc_ecma_ast::VarDeclKind::Var => "var".to_string(),
|
swc_ecma_ast::VarDeclKind::Var => "var".to_string(),
|
||||||
},
|
}),
|
||||||
node.name,
|
colors::bold(node.name.clone()),
|
||||||
if variable_def.ts_type.is_some() {
|
if variable_def.ts_type.is_some() {
|
||||||
format!(": {}", render_ts_type(variable_def.ts_type.unwrap()))
|
format!(": {}", render_ts_type(variable_def.ts_type.unwrap()))
|
||||||
} else {
|
} else {
|
||||||
"".to_string()
|
"".to_string()
|
||||||
}
|
}
|
||||||
|
),
|
||||||
|
indent,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn format_enum_signature(node: &doc::DocNode, indent: i64) -> String {
|
fn format_enum_signature(node: &doc::DocNode, indent: i64) -> String {
|
||||||
format_indent(indent);
|
add_indent(
|
||||||
format!("enum {}\n", node.name)
|
format!(
|
||||||
|
"{} {}\n",
|
||||||
|
colors::magenta("enum".to_string()),
|
||||||
|
colors::bold(node.name.clone())
|
||||||
|
),
|
||||||
|
indent,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn format_interface_signature(node: &doc::DocNode, indent: i64) -> String {
|
fn format_interface_signature(node: &doc::DocNode, indent: i64) -> String {
|
||||||
format_indent(indent);
|
add_indent(
|
||||||
format!("interface {}\n", node.name)
|
format!(
|
||||||
|
"{} {}\n",
|
||||||
|
colors::magenta("interface".to_string()),
|
||||||
|
colors::bold(node.name.clone())
|
||||||
|
),
|
||||||
|
indent,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn format_type_alias_signature(node: &doc::DocNode, indent: i64) -> String {
|
fn format_type_alias_signature(node: &doc::DocNode, indent: i64) -> String {
|
||||||
format_indent(indent);
|
add_indent(
|
||||||
format!("type {}\n", node.name)
|
format!(
|
||||||
|
"{} {}\n",
|
||||||
|
colors::magenta("type".to_string()),
|
||||||
|
colors::bold(node.name.clone())
|
||||||
|
),
|
||||||
|
indent,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn format_namespace_signature(node: &doc::DocNode, indent: i64) -> String {
|
fn format_namespace_signature(node: &doc::DocNode, indent: i64) -> String {
|
||||||
format_indent(indent);
|
add_indent(
|
||||||
format!("namespace {}\n", node.name)
|
format!(
|
||||||
|
"{} {}\n",
|
||||||
|
colors::magenta("namespace".to_string()),
|
||||||
|
colors::bold(node.name.clone())
|
||||||
|
),
|
||||||
|
indent,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
||||||
use super::DocParser;
|
use super::DocParser;
|
||||||
|
use crate::colors;
|
||||||
use serde_json;
|
use serde_json;
|
||||||
use serde_json::json;
|
use serde_json::json;
|
||||||
|
|
||||||
|
@ -61,7 +62,10 @@ export function foo(a: string, b: number): void {
|
||||||
let actual = serde_json::to_value(entry).unwrap();
|
let actual = serde_json::to_value(entry).unwrap();
|
||||||
assert_eq!(actual, expected_json);
|
assert_eq!(actual, expected_json);
|
||||||
|
|
||||||
assert!(super::printer::format(entries).contains("Hello there"));
|
assert!(
|
||||||
|
colors::strip_ansi_codes(super::printer::format(entries).as_str())
|
||||||
|
.contains("Hello there")
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -90,7 +94,10 @@ fn export_const() {
|
||||||
let actual = serde_json::to_value(entry).unwrap();
|
let actual = serde_json::to_value(entry).unwrap();
|
||||||
assert_eq!(actual, expected_json);
|
assert_eq!(actual, expected_json);
|
||||||
|
|
||||||
assert!(super::printer::format(entries).contains("Something about fizzBuzz"));
|
assert!(
|
||||||
|
colors::strip_ansi_codes(super::printer::format(entries).as_str())
|
||||||
|
.contains("Something about fizzBuzz")
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -299,7 +306,10 @@ export class Foobar extends Fizz implements Buzz {
|
||||||
let actual = serde_json::to_value(entry).unwrap();
|
let actual = serde_json::to_value(entry).unwrap();
|
||||||
assert_eq!(actual, expected_json);
|
assert_eq!(actual, expected_json);
|
||||||
|
|
||||||
assert!(super::printer::format(entries).contains("class Foobar"));
|
assert!(
|
||||||
|
colors::strip_ansi_codes(super::printer::format(entries).as_str())
|
||||||
|
.contains("class Foobar")
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -381,7 +391,10 @@ export interface Reader {
|
||||||
let actual = serde_json::to_value(entry).unwrap();
|
let actual = serde_json::to_value(entry).unwrap();
|
||||||
assert_eq!(actual, expected_json);
|
assert_eq!(actual, expected_json);
|
||||||
|
|
||||||
assert!(super::printer::format(entries).contains("interface Reader"));
|
assert!(
|
||||||
|
colors::strip_ansi_codes(super::printer::format(entries).as_str())
|
||||||
|
.contains("interface Reader")
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -424,7 +437,10 @@ export type NumberArray = Array<number>;
|
||||||
let actual = serde_json::to_value(entry).unwrap();
|
let actual = serde_json::to_value(entry).unwrap();
|
||||||
assert_eq!(actual, expected_json);
|
assert_eq!(actual, expected_json);
|
||||||
|
|
||||||
assert!(super::printer::format(entries).contains("Array holding numbers"));
|
assert!(
|
||||||
|
colors::strip_ansi_codes(super::printer::format(entries).as_str())
|
||||||
|
.contains("Array holding numbers")
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -470,9 +486,14 @@ export enum Hello {
|
||||||
let actual = serde_json::to_value(entry).unwrap();
|
let actual = serde_json::to_value(entry).unwrap();
|
||||||
assert_eq!(actual, expected_json);
|
assert_eq!(actual, expected_json);
|
||||||
|
|
||||||
assert!(super::printer::format(entries.clone())
|
assert!(colors::strip_ansi_codes(
|
||||||
|
super::printer::format(entries.clone()).as_str()
|
||||||
|
)
|
||||||
.contains("Some enum for good measure"));
|
.contains("Some enum for good measure"));
|
||||||
assert!(super::printer::format(entries).contains("enum Hello"));
|
assert!(
|
||||||
|
colors::strip_ansi_codes(super::printer::format(entries).as_str())
|
||||||
|
.contains("enum Hello")
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -564,5 +585,8 @@ export namespace RootNs {
|
||||||
});
|
});
|
||||||
let actual = serde_json::to_value(entry).unwrap();
|
let actual = serde_json::to_value(entry).unwrap();
|
||||||
assert_eq!(actual, expected_json);
|
assert_eq!(actual, expected_json);
|
||||||
assert!(super::printer::format(entries).contains("namespace RootNs"));
|
assert!(
|
||||||
|
colors::strip_ansi_codes(super::printer::format(entries).as_str())
|
||||||
|
.contains("namespace RootNs")
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue