1
0
Fork 0
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:
Luca Casonato 2020-03-28 22:35:31 +01:00 committed by GitHub
parent d6cb3892d4
commit ad198b1cf1
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 168 additions and 85 deletions

View file

@ -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);

View file

@ -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,
)
} }

View file

@ -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")
);
} }