mirror of
https://github.com/denoland/deno.git
synced 2024-12-12 02:27:46 -05:00
231 lines
6.6 KiB
Rust
231 lines
6.6 KiB
Rust
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
|
use crate::swc_common::SourceMap;
|
|
use crate::swc_common::Spanned;
|
|
use crate::swc_ecma_ast;
|
|
use serde::Serialize;
|
|
|
|
use super::function::function_to_function_def;
|
|
use super::function::FunctionDef;
|
|
use super::interface::expr_to_name;
|
|
use super::params::assign_pat_to_param_def;
|
|
use super::params::ident_to_param_def;
|
|
use super::params::pat_to_param_def;
|
|
use super::parser::DocParser;
|
|
use super::ts_type::ts_entity_name_to_name;
|
|
use super::ts_type::ts_type_ann_to_def;
|
|
use super::ts_type::TsTypeDef;
|
|
use super::ts_type_param::maybe_type_param_decl_to_type_param_defs;
|
|
use super::ts_type_param::TsTypeParamDef;
|
|
use super::Location;
|
|
use super::ParamDef;
|
|
|
|
#[derive(Debug, Serialize, Clone)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub struct ClassConstructorDef {
|
|
pub js_doc: Option<String>,
|
|
pub accessibility: Option<swc_ecma_ast::Accessibility>,
|
|
pub name: String,
|
|
pub params: Vec<ParamDef>,
|
|
pub location: Location,
|
|
}
|
|
|
|
#[derive(Debug, Serialize, Clone)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub struct ClassPropertyDef {
|
|
pub js_doc: Option<String>,
|
|
pub ts_type: Option<TsTypeDef>,
|
|
pub readonly: bool,
|
|
pub accessibility: Option<swc_ecma_ast::Accessibility>,
|
|
pub optional: bool,
|
|
pub is_abstract: bool,
|
|
pub is_static: bool,
|
|
pub name: String,
|
|
pub location: Location,
|
|
}
|
|
|
|
#[derive(Debug, Serialize, Clone)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub struct ClassMethodDef {
|
|
pub js_doc: Option<String>,
|
|
pub accessibility: Option<swc_ecma_ast::Accessibility>,
|
|
pub optional: bool,
|
|
pub is_abstract: bool,
|
|
pub is_static: bool,
|
|
pub name: String,
|
|
pub kind: swc_ecma_ast::MethodKind,
|
|
pub function_def: FunctionDef,
|
|
pub location: Location,
|
|
}
|
|
|
|
#[derive(Debug, Serialize, Clone)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub struct ClassDef {
|
|
// TODO(bartlomieju): decorators, super_type_params
|
|
pub is_abstract: bool,
|
|
pub constructors: Vec<ClassConstructorDef>,
|
|
pub properties: Vec<ClassPropertyDef>,
|
|
pub methods: Vec<ClassMethodDef>,
|
|
pub extends: Option<String>,
|
|
pub implements: Vec<String>,
|
|
pub type_params: Vec<TsTypeParamDef>,
|
|
}
|
|
|
|
fn prop_name_to_string(
|
|
source_map: &SourceMap,
|
|
prop_name: &swc_ecma_ast::PropName,
|
|
) -> String {
|
|
use crate::swc_ecma_ast::PropName;
|
|
match prop_name {
|
|
PropName::Ident(ident) => ident.sym.to_string(),
|
|
PropName::Str(str_) => str_.value.to_string(),
|
|
PropName::Num(num) => num.value.to_string(),
|
|
PropName::Computed(comp_prop_name) => {
|
|
source_map.span_to_snippet(comp_prop_name.span).unwrap()
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn class_to_class_def(
|
|
doc_parser: &DocParser,
|
|
class: &swc_ecma_ast::Class,
|
|
) -> ClassDef {
|
|
let mut constructors = vec![];
|
|
let mut methods = vec![];
|
|
let mut properties = vec![];
|
|
|
|
let extends: Option<String> = match &class.super_class {
|
|
Some(boxed) => {
|
|
use crate::swc_ecma_ast::Expr;
|
|
let expr: &Expr = &**boxed;
|
|
match expr {
|
|
Expr::Ident(ident) => Some(ident.sym.to_string()),
|
|
_ => None,
|
|
}
|
|
}
|
|
None => None,
|
|
};
|
|
|
|
let implements: Vec<String> = class
|
|
.implements
|
|
.iter()
|
|
.map(|expr| ts_entity_name_to_name(&expr.expr))
|
|
.collect();
|
|
|
|
for member in &class.body {
|
|
use crate::swc_ecma_ast::ClassMember::*;
|
|
|
|
match member {
|
|
Constructor(ctor) => {
|
|
let ctor_js_doc = doc_parser.js_doc_for_span(ctor.span());
|
|
let constructor_name =
|
|
prop_name_to_string(&doc_parser.ast_parser.source_map, &ctor.key);
|
|
|
|
let mut params = vec![];
|
|
|
|
for param in &ctor.params {
|
|
use crate::swc_ecma_ast::PatOrTsParamProp::*;
|
|
|
|
let param_def = match param {
|
|
Pat(pat) => pat_to_param_def(pat),
|
|
TsParamProp(ts_param_prop) => {
|
|
use swc_ecma_ast::TsParamPropParam;
|
|
|
|
match &ts_param_prop.param {
|
|
TsParamPropParam::Ident(ident) => ident_to_param_def(ident),
|
|
TsParamPropParam::Assign(assign_pat) => {
|
|
assign_pat_to_param_def(assign_pat)
|
|
}
|
|
}
|
|
}
|
|
};
|
|
params.push(param_def);
|
|
}
|
|
|
|
let constructor_def = ClassConstructorDef {
|
|
js_doc: ctor_js_doc,
|
|
accessibility: ctor.accessibility,
|
|
name: constructor_name,
|
|
params,
|
|
location: doc_parser.ast_parser.get_span_location(ctor.span).into(),
|
|
};
|
|
constructors.push(constructor_def);
|
|
}
|
|
Method(class_method) => {
|
|
let method_js_doc = doc_parser.js_doc_for_span(class_method.span());
|
|
let method_name = prop_name_to_string(
|
|
&doc_parser.ast_parser.source_map,
|
|
&class_method.key,
|
|
);
|
|
let fn_def = function_to_function_def(&class_method.function);
|
|
let method_def = ClassMethodDef {
|
|
js_doc: method_js_doc,
|
|
accessibility: class_method.accessibility,
|
|
optional: class_method.is_optional,
|
|
is_abstract: class_method.is_abstract,
|
|
is_static: class_method.is_static,
|
|
name: method_name,
|
|
kind: class_method.kind,
|
|
function_def: fn_def,
|
|
location: doc_parser
|
|
.ast_parser
|
|
.get_span_location(class_method.span)
|
|
.into(),
|
|
};
|
|
methods.push(method_def);
|
|
}
|
|
ClassProp(class_prop) => {
|
|
let prop_js_doc = doc_parser.js_doc_for_span(class_prop.span());
|
|
|
|
let ts_type = class_prop
|
|
.type_ann
|
|
.as_ref()
|
|
.map(|rt| ts_type_ann_to_def(rt));
|
|
|
|
let prop_name = expr_to_name(&*class_prop.key);
|
|
|
|
let prop_def = ClassPropertyDef {
|
|
js_doc: prop_js_doc,
|
|
ts_type,
|
|
readonly: class_prop.readonly,
|
|
optional: class_prop.is_optional,
|
|
is_abstract: class_prop.is_abstract,
|
|
is_static: class_prop.is_static,
|
|
accessibility: class_prop.accessibility,
|
|
name: prop_name,
|
|
location: doc_parser
|
|
.ast_parser
|
|
.get_span_location(class_prop.span)
|
|
.into(),
|
|
};
|
|
properties.push(prop_def);
|
|
}
|
|
// TODO(bartlomieju):
|
|
TsIndexSignature(_) => {}
|
|
PrivateMethod(_) => {}
|
|
PrivateProp(_) => {}
|
|
}
|
|
}
|
|
|
|
let type_params =
|
|
maybe_type_param_decl_to_type_param_defs(class.type_params.as_ref());
|
|
|
|
ClassDef {
|
|
is_abstract: class.is_abstract,
|
|
extends,
|
|
implements,
|
|
constructors,
|
|
properties,
|
|
methods,
|
|
type_params,
|
|
}
|
|
}
|
|
|
|
pub fn get_doc_for_class_decl(
|
|
doc_parser: &DocParser,
|
|
class_decl: &swc_ecma_ast::ClassDecl,
|
|
) -> (String, ClassDef) {
|
|
let class_name = class_decl.ident.sym.to_string();
|
|
let class_def = class_to_class_def(doc_parser, &class_decl.class);
|
|
|
|
(class_name, class_def)
|
|
}
|