1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
// Copyright (c) 2024 Jonas Schäfer <jonas@zombofant.net>
//
// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this
// file, You can obtain one at http://mozilla.org/MPL/2.0/.

//! This module concerns the processing of untyped `minidom::Element`
//! children.
//!
//! In particular, it provides the `#[xml(element)]` implementation.

use proc_macro2::{Span, TokenStream};
use quote::quote;
use syn::*;

use crate::error_message::ParentRef;
use crate::scope::{AsItemsScope, FromEventsScope};
use crate::types::{
    default_fn, element_ty, from_xml_builder_ty, into_iterator_into_iter_fn, into_iterator_iter_ty,
    item_iter_ty, option_ty, ref_ty,
};

use super::{Field, FieldBuilderPart, FieldIteratorPart, FieldTempInit, NestedMatcher};

pub(super) struct ElementField;

impl Field for ElementField {
    fn make_builder_part(
        &self,
        scope: &FromEventsScope,
        _container_name: &ParentRef,
        member: &Member,
        ty: &Type,
    ) -> Result<FieldBuilderPart> {
        let FromEventsScope {
            ref substate_result,
            ..
        } = scope;
        let field_access = scope.access_field(member);

        let element_ty = element_ty(Span::call_site());
        let default_fn = default_fn(ty.clone());
        let builder = from_xml_builder_ty(element_ty.clone());

        Ok(FieldBuilderPart::Nested {
            extra_defs: TokenStream::default(),
            value: FieldTempInit {
                init: quote! { #default_fn() },
                ty: ty.clone(),
            },
            matcher: NestedMatcher::Fallback(quote! {
                #builder::new(name, attrs)
            }),
            builder,
            collect: quote! {
                <#ty as ::core::iter::Extend::<#element_ty>>::extend(&mut #field_access, [#substate_result]);
            },
            finalize: quote! {
                #field_access
            },
        })
    }

    fn make_iterator_part(
        &self,
        scope: &AsItemsScope,
        _container_name: &ParentRef,
        bound_name: &Ident,
        _member: &Member,
        ty: &Type,
    ) -> Result<FieldIteratorPart> {
        let AsItemsScope { ref lifetime, .. } = scope;

        let element_ty = element_ty(Span::call_site());
        let iter_ty = item_iter_ty(element_ty.clone(), lifetime.clone());
        let element_iter = into_iterator_iter_ty(ref_ty(ty.clone(), lifetime.clone()));
        let into_iter = into_iterator_into_iter_fn(ref_ty(ty.clone(), lifetime.clone()));

        let state_ty = Type::Tuple(TypeTuple {
            paren_token: token::Paren::default(),
            elems: [element_iter, option_ty(iter_ty)].into_iter().collect(),
        });

        Ok(FieldIteratorPart::Content {
            extra_defs: TokenStream::default(),
            value: FieldTempInit {
                init: quote! {
                    (#into_iter(#bound_name), ::core::option::Option::None)
                },
                ty: state_ty,
            },
            generator: quote! {
                loop {
                    if let ::core::option::Option::Some(current) = #bound_name.1.as_mut() {
                        if let ::core::option::Option::Some(item) = current.next() {
                            break ::core::option::Option::Some(item).transpose();
                        }
                    }
                    if let ::core::option::Option::Some(item) = #bound_name.0.next() {
                        #bound_name.1 = ::core::option::Option::Some(
                            <#element_ty as ::xso::AsXml>::as_xml_iter(item)?
                        );
                    } else {
                        break ::core::result::Result::Ok(::core::option::Option::None)
                    }
                }
            },
        })
    }
}