Skip to content
Snippets Groups Projects
password.rs 7.11 KiB
Newer Older
Caleb Yates's avatar
Caleb Yates committed
use crate::{
    cosmic_edit::DefaultAttrs, focus::FocusSet, placeholder::Placeholder, prelude::*,
    render::RenderSet,
};
use cosmic_text::{Cursor, Edit, Selection, Shaping};
use unicode_segmentation::UnicodeSegmentation;
/// System set for password blocking systems. Runs in [`PostUpdate`]
#[derive(SystemSet, Debug, Clone, PartialEq, Eq, Hash)]
Caleb Yates's avatar
Caleb Yates committed
pub(crate) struct PasswordSet;
sam edelsten's avatar
sam edelsten committed
pub(crate) fn plugin(app: &mut App) {
    app.add_systems(
        PreUpdate,
        (hide_password_text.before(crate::input::InputSet),),
    )
    .add_systems(
        Update,
        (restore_password_text
            .before(crate::input::keyboard::kb_input_text)
            .after(crate::input::keyboard::kb_move_cursor),),
    )
    .add_systems(
        PostUpdate,
        (
            hide_password_text.before(RenderSet).in_set(PasswordSet),
            restore_password_text.before(FocusSet).after(RenderSet),
        ),
    );
Caleb Yates's avatar
Caleb Yates committed
/// Component to be added to an entity with a [`CosmicEditBuffer`] to block contents with a
/// password blocker glyph
///
/// ```
/// # use bevy::prelude::*;
/// # use bevy_cosmic_edit::prelude::*;
/// use bevy_cosmic_edit::password::Password;
///
/// #
/// # fn setup(mut commands: Commands) {
/// // Create a new cosmic bundle
Caleb Yates's avatar
Caleb Yates committed
/// commands.spawn((
Caleb Yates's avatar
Caleb Yates committed
///     CosmicEditBuffer::default(),
///     Sprite {
///         custom_size: Some(Vec2::new(300.0, 40.0)),
///         ..default()
///     },
///     Password::default()
/// ));
/// # }
/// #
/// # fn main() {
/// #     App::new()
/// #         .add_plugins(MinimalPlugins)
/// #         .add_plugins(CosmicEditPlugin::default())
/// #         .add_systems(Startup, setup);
/// # }
sam edelsten's avatar
sam edelsten committed
#[derive(Component)]
pub struct Password {
    real_text: String,
    glyph: char,
}

impl Default for Password {
    fn default() -> Self {
        Self {
            real_text: Default::default(),
impl Password {
    /// New password component with custom blocker glyph
    pub fn new(glyph: char) -> Self {
        Self { glyph, ..default() }
    }
}

Caleb Yates's avatar
Caleb Yates committed
/// Stores [`CosmicEditBuffer`] contents into [`Password.real_text`]
sam edelsten's avatar
sam edelsten committed
fn hide_password_text(
    mut q: Query<(
        &mut Password,
sam edelsten's avatar
sam edelsten committed
        &DefaultAttrs,
        Option<&Placeholder>,
sam edelsten's avatar
sam edelsten committed
    )>,
    mut font_system: ResMut<CosmicFontSystem>,
) {
    for (mut password, mut editor, attrs, placeholder_opt) in q.iter_mut() {
        if let Some(placeholder) = placeholder_opt {
            if placeholder.is_active() {
Caleb Yates's avatar
Caleb Yates committed
                // doesn't override placeholder
        match editor.editor() {
            Some(editor) => {
                let mut cursor = editor.cursor();
                let mut selection = editor.selection();

                editor.with_buffer_mut(|buffer| {
                    let text = buffer.get_text();

                    // Translate cursor to correct position for blocker glyphs
                    let translate_cursor = |c: &mut Cursor| {
                        let (pre, _post) = text.split_at(c.index);
                        let graphemes = pre.graphemes(true).count();
                        c.index = graphemes * password.glyph.len_utf8();
                    };

                    translate_cursor(&mut cursor);

                    // Translate selection cursor
                    match selection {
                        Selection::None => {}
                        Selection::Line(ref mut c) => {
                            translate_cursor(c);
                        }
                        Selection::Word(ref mut c) => {
                            translate_cursor(c);
                        }
                        Selection::Normal(ref mut c) => {
                            translate_cursor(c);
                        }
                    // Update text to blockers
                    buffer.set_text(
                        &mut font_system,
                        password
                            .glyph
                            .to_string()
                            .repeat(text.graphemes(true).count())
                            .as_str(),
                        attrs.as_attrs(),
                        Shaping::Advanced,
                    );

                    password.real_text = text;
                });

                editor.set_cursor(cursor);
                editor.set_selection(selection);
            }
            None => {
                let text = editor.get_text();

                editor.set_text(
sam edelsten's avatar
sam edelsten committed
                    &mut font_system,
                    password.glyph.to_string().repeat(text.len()).as_str(),
sam edelsten's avatar
sam edelsten committed
                    attrs.as_attrs(),
                );
                password.real_text = text;
Caleb Yates's avatar
Caleb Yates committed
/// Replaces [`CosmicEditBuffer`] contents with [`Password.real_text`]
sam edelsten's avatar
sam edelsten committed
fn restore_password_text(
    mut q: Query<(
        &Password,
Caleb Yates's avatar
Caleb Yates committed
        &mut CosmicEditBuffer,
sam edelsten's avatar
sam edelsten committed
        &DefaultAttrs,
        Option<&mut CosmicEditor>,
        Option<&Placeholder>,
sam edelsten's avatar
sam edelsten committed
    )>,
    mut font_system: ResMut<CosmicFontSystem>,
) {
    for (password, mut buffer, attrs, editor_opt, placeholder_opt) in q.iter_mut() {
        if let Some(placeholder) = placeholder_opt {
            if placeholder.is_active() {
                continue;
            }
        }
sam edelsten's avatar
sam edelsten committed
        if let Some(mut editor) = editor_opt {
            let mut cursor = editor.cursor();
            let mut selection = editor.selection();
sam edelsten's avatar
sam edelsten committed
            editor.with_buffer_mut(|buffer| {
                let text = buffer.get_text();

                // Find cursor position and translate back to correct position in real text
                let restore_cursor = |c: &mut Cursor| {
                    let (pre, _post) = text.split_at(c.index);
                    let graphemes = pre.graphemes(true).count();
                    let mut n_i = 0;
                    if let Some((i, _)) = password.real_text.grapheme_indices(true).nth(graphemes) {
                        n_i = i;
                    } else if c.index > 0 {
                        n_i = password.real_text.len();
                restore_cursor(&mut cursor);
                // Translate selection cursor
                match selection {
                    Selection::None => {}
                    Selection::Line(ref mut c) => {
                        restore_cursor(c);
                    }
                    Selection::Word(ref mut c) => {
                        restore_cursor(c);
                    }
                    Selection::Normal(ref mut c) => {
                        restore_cursor(c);
                    }
sam edelsten's avatar
sam edelsten committed
                buffer.set_text(
                    &mut font_system,
                    password.real_text.as_str(),
                    attrs.as_attrs(),
                    Shaping::Advanced,
sam edelsten's avatar
sam edelsten committed
            });

            editor.set_cursor(cursor);
            editor.set_selection(selection);
sam edelsten's avatar
sam edelsten committed
            continue;
        }

        buffer.set_text(
            &mut font_system,
            password.real_text.as_str(),
            attrs.as_attrs(),
        );
    }
}