blob: 41a38dd7a63bfb2e200550451e21fc2e319be36d [file] [log] [blame]
use crate::syntax::qualified::QualifiedName;
use crate::syntax::Api;
#[cfg(test)]
use proc_macro2::Span;
use quote::IdentFragment;
use std::fmt::{self, Display};
use std::slice::Iter;
use syn::parse::{Parse, ParseStream, Result};
use syn::{Ident, Token};
mod kw {
syn::custom_keyword!(namespace);
}
#[derive(Clone)]
pub struct Namespace {
segments: Vec<Ident>,
}
impl Namespace {
pub fn none() -> Self {
Namespace {
segments: Vec::new(),
}
}
pub fn iter(&self) -> Iter<Ident> {
self.segments.iter()
}
pub fn parse_bridge_attr_namespace(input: ParseStream) -> Result<Namespace> {
if input.is_empty() {
return Ok(Namespace::none());
}
input.parse::<kw::namespace>()?;
input.parse::<Token![=]>()?;
let ns = input.parse::<Namespace>()?;
input.parse::<Option<Token![,]>>()?;
Ok(ns)
}
#[cfg(test)]
pub fn from_str(ns: &str) -> Self {
Namespace {
segments: ns
.split("::")
.map(|x| Ident::new(x, Span::call_site()))
.collect(),
}
}
}
impl Parse for Namespace {
fn parse(input: ParseStream) -> Result<Self> {
let segments = QualifiedName::parse_quoted_or_unquoted(input)?.segments;
Ok(Namespace { segments })
}
}
impl Display for Namespace {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
for segment in self {
write!(f, "{}$", segment)?;
}
Ok(())
}
}
impl IdentFragment for Namespace {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
Display::fmt(self, f)
}
}
impl<'a> IntoIterator for &'a Namespace {
type Item = &'a Ident;
type IntoIter = Iter<'a, Ident>;
fn into_iter(self) -> Self::IntoIter {
self.iter()
}
}
impl Api {
pub fn get_namespace(&self) -> Option<&Namespace> {
match self {
Api::CxxFunction(cfn) => Some(&cfn.ident.cxx.ns),
Api::CxxType(cty) => Some(&cty.ident.cxx.ns),
Api::Enum(enm) => Some(&enm.ident.cxx.ns),
Api::Struct(strct) => Some(&strct.ident.cxx.ns),
Api::RustType(rty) => Some(&rty.ident.cxx.ns),
Api::RustFunction(rfn) => Some(&rfn.ident.cxx.ns),
Api::Impl(_) | Api::Include(_) | Api::TypeAlias(_) => None,
}
}
}