wdpe 0.4.2

WebDynpro Parse Engine
Documentation
use std::borrow::Cow;

use scraper::Selector;

use crate::error::{BodyError, WebDynproError};

use super::Element;

/// 컴파일 타임에서도 생성할 수 있는 [`Element`] 정의
pub trait ElementDefinition<'body>: Sized {
    /// 해당 정의가 생성할 수 있는 엘리먼트
    type Element: Element<'body>;

    /// 런타임에서 엘리먼트 정의를 생성합니다. 엘리먼트의 Id 등을 컴파일 타임에서 미리 알 수 없는 경우 유용합니다.
    /// ### 예시
    /// ```
    /// # use rusaint::element::{ action::ButtonDef, definition::ElementDefinition };
    /// # fn get_dynamic_button() -> String { return "TEST.BUTTON1".to_string() }
    /// let runtime_string: String = get_dynamic_button();
    /// let button_def: ButtonDef = ButtonDef::new_dynamic(runtime_string);
    /// ```
    fn new_dynamic(id: String) -> Self;

    /// [`scraper::ElementRef`]에서 엘리먼트 정의를 생성합니다.
    fn from_ref(element_ref: scraper::ElementRef<'_>) -> Result<Self, WebDynproError>;

    /// 엘리먼트의 Id를 반환합니다.
    fn id(&self) -> &str;

    /// [`Cow`]형태의 Id가 필요한 경우 사용합니다.
    fn id_cow(&self) -> Cow<'static, str>;

    /// `scraper`에서 이 엘리먼트를 선택할 수 있는 CSS Selector를 반환합니다.
    /// ### 예시
    /// ```ignore
    /// let parser = ElementParser(body);
    /// const BUTTON: ElementDef<'_, Button<'_>> = ElementDef::new("TEST.BUTTON1");
    /// let selector = BUTTON.selector().unwrap();
    /// let btn_elem = parser.document().select(&selector).next().unwrap();
    /// let btn = ElementWrapper::from_ref(btn_elem).unwrap();
    /// if let ElementWrapper::Button(btn) = btn {
    ///     println!("It's button!");
    /// }
    /// ```
    fn selector(&self) -> Result<Selector, WebDynproError> {
        Ok(
            Selector::parse(format!(r#"[id="{}"]"#, self.id()).as_str()).map_err(|err| {
                tracing::warn!(?err, "failed to parse selector");
                BodyError::InvalidSelector
            })?,
        )
    }
}