Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: update identity payout address #123

Open
wants to merge 8 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 5 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
23 changes: 22 additions & 1 deletion src/backend_task/core/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,10 @@ use crate::backend_task::BackendTaskSuccessResult;
use crate::config::Config;
use crate::context::AppContext;
use crate::model::wallet::Wallet;
use dash_sdk::dashcore_rpc::RpcApi;
use dash_sdk::dashcore_rpc::dashcore::address::Payload;
use dash_sdk::dashcore_rpc::dashcore::hashes::{hash160, Hash};
use dash_sdk::dashcore_rpc::dashcore::PubkeyHash;
use dash_sdk::dashcore_rpc::{dashcore, RpcApi};
use dash_sdk::dashcore_rpc::{Auth, Client};
use dash_sdk::dpp::dashcore::{Address, ChainLock, Network, OutPoint, Transaction, TxOut};
use std::sync::{Arc, RwLock};
Expand All @@ -16,6 +19,7 @@ pub(crate) enum CoreTask {
GetBestChainLocks,
RefreshWalletInfo(Arc<RwLock<Wallet>>),
StartDashQT(Network, Option<String>, bool),
ProRegUpdateTx(String, Address, Address),
}
impl PartialEq for CoreTask {
fn eq(&self, other: &Self) -> bool {
Expand All @@ -24,6 +28,7 @@ impl PartialEq for CoreTask {
(CoreTask::GetBestChainLocks, CoreTask::GetBestChainLocks) => true,
(CoreTask::RefreshWalletInfo(_), CoreTask::RefreshWalletInfo(_)) => true,
(CoreTask::StartDashQT(_, _, _), CoreTask::StartDashQT(_, _, _)) => true,
(CoreTask::ProRegUpdateTx(_, _, _), CoreTask::ProRegUpdateTx(_, _, _)) => true,
_ => false,
}
}
Expand All @@ -34,6 +39,7 @@ pub(crate) enum CoreItem {
ReceivedAvailableUTXOTransaction(Transaction, Vec<(OutPoint, TxOut, Address)>),
ChainLock(ChainLock, Network),
ChainLocks(Option<ChainLock>, Option<ChainLock>), // Mainnet, Testnet
ProRegUpdateTx(String),
}

impl AppContext {
Expand Down Expand Up @@ -135,6 +141,21 @@ impl AppContext {
.start_dash_qt(network, custom_dash_qt, overwrite_dash_conf)
.map_err(|e| e.to_string())
.map(|_| BackendTaskSuccessResult::None),
CoreTask::ProRegUpdateTx(pro_tx_hash, voting_address, payout_address) => self
.core_client
.get_protx_update_registrar(
pro_tx_hash.as_str(),
"",
payout_address,
voting_address,
None,
)
.map(|pro_tx_hash| {
BackendTaskSuccessResult::CoreItem(CoreItem::ProRegUpdateTx(
pro_tx_hash.to_string(),
))
})
.map_err(|e| e.to_string()),
}
}
}
9 changes: 9 additions & 0 deletions src/ui/identities/identities_screen.rs
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ use crate::ui::components::top_panel::add_top_panel;
use crate::ui::identities::keys::add_key_screen::AddKeyScreen;
use crate::ui::identities::keys::key_info_screen::KeyInfoScreen;
use crate::ui::identities::top_up_identity_screen::TopUpIdentityScreen;
use crate::ui::identities::update_identity_payout_address::UpdateIdentityPayoutScreen;
use crate::ui::transfers::TransferScreen;
use crate::ui::{RootScreenType, Screen, ScreenLike, ScreenType};
use dash_sdk::dpp::identity::accessors::IdentityGettersV0;
Expand Down Expand Up @@ -454,6 +455,14 @@ impl IdentitiesScreen {
),
));
}
if ui.button("Update Payout Address").clicked() {
action = AppAction::AddScreen(Screen::UpdatePayoutAddressScreen(
UpdateIdentityPayoutScreen::new(
qualified_identity.clone(),
&self.app_context,
),
));
}
});
row.col(|ui| {
ui.horizontal(|ui| {
Expand Down
1 change: 1 addition & 0 deletions src/ui/identities/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,4 +5,5 @@ pub mod identities_screen;
pub mod keys;
pub mod register_dpns_name_screen;
pub mod top_up_identity_screen;
pub mod update_identity_payout_address;
pub mod withdraw_from_identity_screen;
239 changes: 239 additions & 0 deletions src/ui/identities/update_identity_payout_address.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,239 @@
use crate::app::AppAction;
use crate::backend_task::core::CoreTask;
use crate::backend_task::BackendTask;
use crate::context::AppContext;
use crate::model::qualified_identity::{IdentityType, QualifiedIdentity};
use crate::model::wallet::Wallet;
use crate::ui::components::top_panel::add_top_panel;
use crate::ui::{MessageType, ScreenLike};
use dash_sdk::dashcore_rpc::dashcore::Address;
use dash_sdk::dpp::identity::accessors::IdentityGettersV0;
use dash_sdk::dpp::platform_value::string_encoding::Encoding;
use eframe::egui::Context;
use egui::{ComboBox, Ui};
use std::sync::atomic::Ordering;
use std::sync::{Arc, RwLock};

pub struct UpdateIdentityPayoutScreen {
pub app_context: Arc<AppContext>,
pub identity: QualifiedIdentity,
selected_wallet: Option<Arc<RwLock<Wallet>>>,
selected_payout_address: Option<Address>,
selected_funding_address: Option<Address>,
error_message: Option<String>,
}

impl UpdateIdentityPayoutScreen {
pub fn new(identity: QualifiedIdentity, app_context: &Arc<AppContext>) -> Self {
let selected_wallet = None;
Self {
app_context: app_context.clone(),
identity,
selected_wallet,
selected_payout_address: None,
selected_funding_address: None,
error_message: None,
}
}

fn render_wallet_selection(&mut self, ui: &mut Ui) {
ui.horizontal(|ui| {
if self.app_context.has_wallet.load(Ordering::Relaxed) {
let wallets = &self.app_context.wallets.read().unwrap();
let wallet_aliases: Vec<String> = wallets
.values()
.map(|wallet| {
wallet
.read()
.unwrap()
.alias
.clone()
.unwrap_or_else(|| "Unnamed Wallet".to_string())
})
.collect();

let selected_wallet_alias = self
.selected_wallet
.as_ref()
.and_then(|wallet| wallet.read().ok()?.alias.clone())
.unwrap_or_else(|| "Select".to_string());

// Display the ComboBox for wallet selection
ComboBox::from_label("")
.selected_text(selected_wallet_alias.clone())
.show_ui(ui, |ui| {
for (idx, wallet) in wallets.values().enumerate() {
let wallet_alias = wallet_aliases[idx].clone();
let is_selected = self
.selected_wallet
.as_ref()
.map_or(false, |selected| Arc::ptr_eq(selected, wallet));
if ui
.selectable_label(is_selected, wallet_alias.clone())
.clicked()
{
// Update the selected wallet
self.selected_wallet = Some(wallet.clone());
}
}
});

ui.add_space(20.0);
} else {
ui.label("No wallets available.");
}
});
}

fn render_selected_wallet_payout_addresses(&mut self, ctx: &Context, ui: &mut Ui) {
if let Some(selected_wallet) = &self.selected_wallet {
// Acquire a read lock
let wallet = selected_wallet.read().unwrap();
ui.add_space(20.0);
ui.heading("Select a Payout Address:");
ui.add_space(5.0);
ui.push_id("payout_combo_id", |ui| {
ComboBox::from_label("")
.selected_text(
self.selected_payout_address
.as_ref() // Get a reference to the Option<Address>
.map(|address| address.to_string()) // Convert Address to String
.unwrap_or_else(|| "".to_string()), // Use default "" if None
)
.show_ui(ui, |ui| {
for (address, _) in &wallet.known_addresses {
if ui
.selectable_value(
&mut self.selected_payout_address,
Some(address.clone()),
address.to_string(),
)
.clicked()
{}
}
});
});
ui.add_space(20.0);
if let Some(selected_address) = &self.selected_payout_address {
if let Some(value) = wallet.address_balances.get(&selected_address) {
ui.label(format!("Selected Address has a balance of {} DASH", value));
} else {
// TODO: Why sometimes balance is not found?
//ui.label("Balance NOT FOUND DASH".to_string());
}
}
}
}

fn render_selected_wallet_funding_addresses(&mut self, ctx: &Context, ui: &mut Ui) {
if let Some(selected_wallet) = &self.selected_wallet {
// Acquire a read lock
let wallet = selected_wallet.read().unwrap();
ui.add_space(20.0);
ui.heading("Select a Funding Address:");
ui.add_space(5.0);
ui.push_id("funding_combo_id", |ui| {
ComboBox::from_label("")
.selected_text(
self.selected_funding_address
.as_ref() // Get a reference to the Option<Address>
.map(|address| address.to_string()) // Convert Address to String
.unwrap_or_else(|| "".to_string()), // Use default "" if None
)
.show_ui(ui, |ui| {
for (address, _) in &wallet.known_addresses {
if ui
.selectable_value(
&mut self.selected_funding_address,
Some(address.clone()),
address.to_string(),
)
.clicked()
{}
}
});
});
ui.add_space(20.0);
if let Some(selected_address) = &self.selected_funding_address {
if let Some(value) = wallet.address_balances.get(&selected_address) {
ui.label(format!("Selected Address has a balance of {} DASH", value));
} else {
// TODO: Why sometimes balance is not found?
//ui.label("Balance NOT FOUND DASH".to_string());
}
}
}
}
}

impl ScreenLike for UpdateIdentityPayoutScreen {
fn display_message(&mut self, message: &str, _message_type: MessageType) {
if _message_type == MessageType::Error {
self.error_message = Some(message.to_string());
}
}

/// Renders the UI components for the withdrawal screen
fn ui(&mut self, ctx: &Context) -> AppAction {
let mut action = add_top_panel(
ctx,
&self.app_context,
vec![
("Identities", AppAction::GoToMainScreen),
("Update Payout Address", AppAction::None),
],
vec![],
);

egui::CentralPanel::default().show(ctx, |mut ui| {
if (self.identity.identity_type == IdentityType::User) {
ui.heading(
"Updating Payout Address for User identities is not allowed.".to_string(),
);
return;
}
if (!self.app_context.has_wallet.load(Ordering::Relaxed)) {
ui.heading("Load a Wallet in order to continue.".to_string());
return;
}
ui.heading("Update Payout Address".to_string());
ui.add_space(20.0);

ui.heading("Load Address from wallet".to_string());
self.render_wallet_selection(&mut ui);

if self.selected_wallet.is_some() {
self.render_selected_wallet_payout_addresses(ctx, &mut ui);
if self.selected_payout_address.is_some() {
self.render_selected_wallet_funding_addresses(ctx, &mut ui);
if self.selected_funding_address.is_some() {
ui.add_space(20.0);
ui.colored_label(
egui::Color32::ORANGE,
"The owner key of the Masternode/Evonode must be known to your wallet.",
);
ui.add_space(20.0);
if ui.button("Update Payout Address").clicked() {
action |= AppAction::BackendTask(BackendTask::CoreTask(
CoreTask::ProRegUpdateTx(
self.identity.identity.id().to_string(Encoding::Hex),
self.selected_payout_address.clone().unwrap(),
self.selected_funding_address.clone().unwrap(),
),
));
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Ensure required fields are selected before proceeding

When the "Update Payout Address" button is clicked, the code assumes that both selected_payout_address and selected_funding_address are Some. To prevent potential unwrap() on None values, check that both fields are selected before proceeding.

Proposed change:

if ui.button("Update Payout Address").clicked() {
+   if let (Some(payout_address), Some(funding_address)) = (
+       self.selected_payout_address.clone(),
+       self.selected_funding_address.clone(),
+   ) {
        action |= AppAction::BackendTask(BackendTask::CoreTask(
            CoreTask::ProRegUpdateTx(
                self.identity.identity.id().to_string(Encoding::Hex),
-               self.selected_payout_address.clone().unwrap(),
-               self.selected_funding_address.clone().unwrap(),
+               payout_address,
+               funding_address,
            ),
        ));
+   } else {
+       self.error_message = Some("Please select both payout and funding addresses.".to_string());
+   }
}
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
if ui.button("Update Payout Address").clicked() {
action |= AppAction::BackendTask(BackendTask::CoreTask(
CoreTask::ProRegUpdateTx(
self.identity.identity.id().to_string(Encoding::Hex),
self.selected_payout_address.clone().unwrap(),
self.selected_funding_address.clone().unwrap(),
),
));
}
if ui.button("Update Payout Address").clicked() {
if let (Some(payout_address), Some(funding_address)) = (
self.selected_payout_address.clone(),
self.selected_funding_address.clone(),
) {
action |= AppAction::BackendTask(BackendTask::CoreTask(
CoreTask::ProRegUpdateTx(
self.identity.identity.id().to_string(Encoding::Hex),
payout_address,
funding_address,
),
));
} else {
self.error_message = Some("Please select both payout and funding addresses.".to_string());
}
}

if self.error_message.is_some() {
ui.add_space(20.0);
ui.colored_label(
egui::Color32::RED,
self.error_message.as_ref().unwrap(),
);
}
Copy link
Contributor

@coderabbitai coderabbitai bot Dec 5, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Reset error message after displaying it

The error message is displayed but not reset, which may cause it to persist longer than intended. Reset self.error_message after displaying it to ensure it's shown appropriately.

Proposed change:

if self.error_message.is_some() {
    ui.add_space(20.0);
    ui.colored_label(
        egui::Color32::RED,
        self.error_message.as_ref().unwrap(),
    );
+   self.error_message = None;
}
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
if self.error_message.is_some() {
ui.add_space(20.0);
ui.colored_label(
egui::Color32::RED,
self.error_message.as_ref().unwrap(),
);
}
if self.error_message.is_some() {
ui.add_space(20.0);
ui.colored_label(
egui::Color32::RED,
self.error_message.as_ref().unwrap(),
);
self.error_message = None;
}

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm pretty sure this will just make the error message flash for a millisecond

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@pauldelucia

You're absolutely right. Resetting the error message immediately after displaying it would cause it to disappear too quickly, preventing the user from reading it. Thank you for pointing that out.


✏️ Learnings added
Learnt from: pauldelucia
PR: dashpay/dash-evo-tool#123
File: src/ui/identities/update_identity_payout_address.rs:225-231
Timestamp: 2024-12-09T10:29:01.778Z
Learning: Avoid resetting error messages immediately after displaying them in the UI, as this may cause them to disappear before the user has time to read them.

Note: Learnings are effective only in the context of similar code segments. To apply general coding guidelines, please configure review instructions. You can manage existing learnings in the app.

}
}
}
});

action
}
}
Loading