diff --git a/src-tauri/src/commands/mod.rs b/src-tauri/src/commands/mod.rs index 4169960f..8d1be68e 100644 --- a/src-tauri/src/commands/mod.rs +++ b/src-tauri/src/commands/mod.rs @@ -1,10 +1,11 @@ pub(crate) mod app_version; -pub(crate) mod mining; +// pub(crate) mod mining; pub(crate) mod networks; pub(crate) mod preferences; pub(crate) mod query; pub(crate) mod tx; pub(crate) mod wallets; pub(crate) mod web_logs; +pub(crate) mod user_preferences; //pub use app_version::*; diff --git a/src-tauri/src/commands/tx.rs b/src-tauri/src/commands/tx.rs index cce919f7..917b6d3c 100644 --- a/src-tauri/src/commands/tx.rs +++ b/src-tauri/src/commands/tx.rs @@ -30,8 +30,8 @@ pub async fn coin_transfer( }; let mut config = get_cfg()?; - inject_private_key_to_cfg(&mut config)?; - let mut sender = Sender::from_app_cfg(&config, None, legacy).await?; + inject_private_key_to_cfg(&mut config, _sender)?; + let mut sender = Sender::from_app_cfg(&config, Some(_sender.to_string()), legacy).await?; sender .transfer(receiver_account, amount as f64, false) .await?; diff --git a/src-tauri/src/commands/user_preferences.rs b/src-tauri/src/commands/user_preferences.rs new file mode 100644 index 00000000..1002e39a --- /dev/null +++ b/src-tauri/src/commands/user_preferences.rs @@ -0,0 +1,61 @@ +use crate::carpe_error::CarpeError; +use serde::{Deserialize, Serialize}; +use std::fs::{File, OpenOptions}; +use std::io::{Read, Write}; +use std::path::{PathBuf}; +use crate::configs::default_config_path; + +#[derive(Serialize, Deserialize)] +pub struct UserPreferences { + accounts_list_sort_column: Option, + accounts_list_sort_order: Option, +} + +// Utility function to retrieve the full path to the preferences file +fn get_preferences_path() -> Result { + let app_dir_path = default_config_path(); // Assuming this returns a PathBuf or Path + + // Check if the path exists, if not, return an error + if !app_dir_path.exists() { + return Err(CarpeError::misc("App directory not found")); + } + + Ok(app_dir_path.join("user_preferences.json")) +} + +#[tauri::command(async)] +pub async fn get_user_preferences() -> Result { + let file_path = get_preferences_path()?; + match File::open(&file_path) { + Ok(mut file) => { + let mut contents = String::new(); + if let Err(e) = file.read_to_string(&mut contents) { + return Err(CarpeError::misc(&format!("Failed to read from preferences file: {}", e))); + } + serde_json::from_str(&contents).map_err(|e| CarpeError::misc(&format!("Failed to parse preferences: {}", e))) + }, + Err(e) => Err(CarpeError::misc(&format!("Failed to open preferences file: {}", e))), + } +} + +#[tauri::command(async)] +pub async fn set_accounts_list_preference(sort_column: String, sort_order: String) -> Result<(), CarpeError> { + let file_path = get_preferences_path()?; + let mut file = OpenOptions::new() + .write(true) + .create(true) + .truncate(true) + .open(&file_path) + .map_err(|e| CarpeError::misc(&format!("Failed to open preferences file for writing: {}", e)))?; + + let preferences = UserPreferences { + accounts_list_sort_column: Some(sort_column), + accounts_list_sort_order: Some(sort_order), + }; + + let serialized_data = serde_json::to_string_pretty(&preferences) + .map_err(|e| CarpeError::misc(&format!("Failed to serialize preferences: {}", e)))?; + + file.write_all(serialized_data.as_bytes()) + .map_err(|e| CarpeError::misc(&format!("Failed to write preferences file: {}", e))) +} diff --git a/src-tauri/src/commands/wallets.rs b/src-tauri/src/commands/wallets.rs index e8f5b61f..e26981cb 100644 --- a/src-tauri/src/commands/wallets.rs +++ b/src-tauri/src/commands/wallets.rs @@ -1,7 +1,7 @@ use crate::{ carpe_error::CarpeError, commands::query, - configs::{self, default_legacy_account_path, get_cfg, get_client}, + configs::{self, default_legacy_account_path, get_cfg, get_client, default_config_path}, configs_profile, key_manager::{self, get_private_key, inject_private_key_to_cfg}, }; @@ -17,7 +17,10 @@ use libra_types::{ use libra_wallet::account_keys::{self, KeyChain}; use serde::{Deserialize, Serialize}; use std::fs::{self, File}; -use std::io::prelude::*; +use std::io::{Write, prelude::*}; +use std::fs::OpenOptions; +use std::path::{PathBuf, Path}; +use serde_json; #[derive(serde::Deserialize, serde::Serialize, Debug)] pub struct NewKeygen { @@ -49,6 +52,7 @@ pub struct CarpeProfile { on_chain: bool, balance: SlowWalletBalance, locale: Option, // TODO: refactor, tauri now offers locale of the OS + note: Option, } impl From<&Profile> for CarpeProfile { @@ -63,6 +67,7 @@ impl From<&Profile> for CarpeProfile { total: core_profile.balance.total, }, // TODO: refactor upstream to have Clone locale: core_profile.locale.clone(), + note: None, } } } @@ -123,6 +128,12 @@ pub async fn init_from_private_key( Ok(core_profile.into()) } +#[derive(Serialize, Deserialize)] +struct Note { + account: String, + note: String, +} + /// read all accounts from profile #[tauri::command(async)] pub fn get_all_accounts() -> Result, CarpeError> { @@ -131,6 +142,93 @@ pub fn get_all_accounts() -> Result, CarpeError> { Ok(mapped) } +/// read all accounts from profile plus notes +#[tauri::command] +pub fn get_all_accounts_with_notes() -> Result, CarpeError> { + let mut accounts = get_all_accounts()?; + let _ = assign_notes_to_accounts(&mut accounts); + Ok(accounts) +} + +fn notes_file_path() -> PathBuf { + let app_dir_path = default_config_path(); // Assuming this returns a PathBuf or Path + return app_dir_path.join("account_notes.json"); +} + + +fn read_notes() -> Result, CarpeError> { + let file_path = notes_file_path(); + + // Check if the file exists before attempting to open it + if !Path::new(&file_path).exists() { + return Ok(vec![]); // Return an empty vector if the file does not exist + } + + let mut file = match File::open(&file_path) { + Ok(f) => f, + Err(_e) => return Err(CarpeError::misc("Failed to open notes file")), + }; + + let mut contents = String::new(); + if let Err(_e) = file.read_to_string(&mut contents) { + return Err(CarpeError::misc("Failed to read from notes file")); + } + + match serde_json::from_str(&contents) { + Ok(notes) => Ok(notes), + Err(_e) => Err(CarpeError::misc("Failed to parse notes JSON")), + } +} + +fn assign_notes_to_accounts(accounts: &mut Vec) -> Result<(), CarpeError> { + let notes = read_notes()?; + for account in accounts.iter_mut() { + let note_option = notes.iter() + .find(|note| note.account == account.account.to_string().to_uppercase()) + .map(|note| note.note.clone()); + account.note = note_option; + } + Ok(()) +} +#[tauri::command] +pub fn associate_note_with_account(account: String, note: String) -> Result<(), CarpeError> { + let file_path = notes_file_path(); + let mut notes = read_notes().map_err(|_e| CarpeError::misc("Failed to read notes"))?; + let address = account.to_uppercase(); + + // Check if the account already exists and update the note if it does + let mut found = false; + for account_note in notes.iter_mut() { + if account_note.account == address { + account_note.note = note.clone(); + found = true; + break; + } + } + + // If the account does not exist, add a new note + if !found { + notes.push(Note { account: address, note }); + } + + let notes_json = serde_json::to_string(¬es) + .map_err(|_e| CarpeError::misc("Failed to serialize notes"))?; + + let mut file = OpenOptions::new() + .write(true) + .truncate(true) + .create(true) + .open(&file_path) + .map_err(|_e| CarpeError::misc("Failed to open file for writing"))?; + + file.write_all(notes_json.as_bytes()) + .map_err(|_e| CarpeError::misc("Failed to write to file"))?; + + Ok(()) +} + + + /// read all accounts from profile #[tauri::command(async)] pub fn get_default_profile() -> Result { @@ -150,7 +248,8 @@ pub async fn refresh_accounts() -> Result, CarpeError> { map_get_balance(&mut app_cfg.user_profiles).await?; app_cfg.save_file()?; - let mapped: Vec = app_cfg.user_profiles.iter().map(|p| p.into()).collect(); + let mut mapped: Vec = app_cfg.user_profiles.iter().map(|p| p.into()).collect(); + let _ = assign_notes_to_accounts(&mut mapped); Ok(mapped) } @@ -158,7 +257,6 @@ pub async fn refresh_accounts() -> Result, CarpeError> { /// check if this account is a slow wallet pub async fn is_slow(account: AccountAddress) -> anyhow::Result { let c = get_client()?; - println!("is slow"); let b = c .view_ext( "0x1::slow_wallet::is_slow", @@ -166,7 +264,6 @@ pub async fn is_slow(account: AccountAddress) -> anyhow::Result Ok(b), None => Ok(false), @@ -210,7 +307,7 @@ pub async fn get_originating_address( } } -/// Switch tx profiles, change 0L.toml to use selected account +/// Switch tx profiles, change libra.yaml to use selected account #[tauri::command(async)] // IMPORTANT: don't return the profile, since it has keys pub async fn switch_profile(account: AccountAddress) -> Result { @@ -218,9 +315,15 @@ pub async fn switch_profile(account: AccountAddress) -> Result = vec![profile.into()]; + assign_notes_to_accounts(&mut profiles)?; + + Ok(profiles.into_iter().next().unwrap().into()) } // remove all accounts which are being tracked. @@ -266,11 +369,11 @@ async fn test_fetch_originating() { } #[tauri::command(async)] -pub async fn set_slow_wallet(legacy: bool) -> Result<(), CarpeError> { +pub async fn set_slow_wallet(legacy: bool, _sender: AccountAddress) -> Result<(), CarpeError> { // NOTE: unsure Serde was catching all cases check serialization let mut config = get_cfg()?; - inject_private_key_to_cfg(&mut config)?; - let mut sender = Sender::from_app_cfg(&config, None, legacy).await?; + inject_private_key_to_cfg(&mut config, _sender)?; + let mut sender = Sender::from_app_cfg(&config, Some(_sender.to_string()), legacy).await?; let t = SetSlowTx {}; t.run(&mut sender).await?; diff --git a/src-tauri/src/key_manager.rs b/src-tauri/src/key_manager.rs index ce2c5bd0..0e6a9b24 100644 --- a/src-tauri/src/key_manager.rs +++ b/src-tauri/src/key_manager.rs @@ -75,9 +75,12 @@ pub fn get_keypair( /// insert the public key into the AppCfg temporarily so that we don't need /// to prompt user for mnemonic. // NOTE to future devs: DANGER: make sure this is never called in a flow that uses save_file(). The upstream prevents the key from serializing, but it should be guarded here as well. -pub fn inject_private_key_to_cfg(app_cfg_mut: &mut AppCfg) -> anyhow::Result<(), CarpeError> { +pub fn inject_private_key_to_cfg( + app_cfg_mut: &mut AppCfg, + account: AccountAddress, +) -> anyhow::Result<(), CarpeError> { // gets the default profile - let profile = app_cfg_mut.get_profile_mut(None)?; + let profile = app_cfg_mut.get_profile_mut(Some(account.to_string()))?; let pri_key = get_private_key(&profile.account).map_err(|e| CarpeError { category: ErrorCat::Configs, uid: E_KEY_NOT_REGISTERED, diff --git a/src-tauri/src/main.rs b/src-tauri/src/main.rs index b349ba8f..37f15f1b 100644 --- a/src-tauri/src/main.rs +++ b/src-tauri/src/main.rs @@ -90,6 +90,8 @@ async fn main() { commands::wallets::get_default_profile, commands::wallets::refresh_accounts, commands::wallets::get_all_accounts, + commands::wallets::get_all_accounts_with_notes, + commands::wallets::associate_note_with_account, commands::wallets::keygen, commands::wallets::init_from_mnem, commands::wallets::init_from_private_key, @@ -114,14 +116,14 @@ async fn main() { commands::tx::coin_transfer, // claim_make_whole, //////// Tower //////// - commands::query::get_onchain_tower_state, - commands::mining::miner_once, - commands::mining::start_backlog_sender_listener, - commands::mining::get_local_height, - commands::mining::get_epoch_rules, - commands::mining::submit_backlog, - commands::mining::get_last_local_proof, - commands::mining::debug_highest_proof_path, + // commands::query::get_onchain_tower_state, + // commands::mining::miner_once, + // commands::mining::start_backlog_sender_listener, + // commands::mining::get_local_height, + // commands::mining::get_epoch_rules, + // commands::mining::submit_backlog, + // commands::mining::get_last_local_proof, + // commands::mining::debug_highest_proof_path, // submit_proof_zero, //////// Preferences //////// @@ -135,6 +137,10 @@ async fn main() { commands::preferences::set_preferences_locale, commands::preferences::get_miner_txs_cost, commands::preferences::set_miner_txs_cost, + + //////// User Preferences //////// + commands::user_preferences::get_user_preferences, + commands::user_preferences::set_accounts_list_preference, ///////// Debug //////// commands::app_version::get_app_version, commands::web_logs::log_this, diff --git a/src/components/Nav.svelte b/src/components/Nav.svelte index 1a295ea4..112efd98 100644 --- a/src/components/Nav.svelte +++ b/src/components/Nav.svelte @@ -5,7 +5,7 @@ // views import AccountSwitcher from './wallet/AccountSwitcher.svelte' - import { signingAccount } from '../modules/accounts' + import { signingAccount } from '../modules/accounts' // import MakeWholeLink from "./make-whole/MakeWholeLink.svelte"; // init_preferences() @@ -20,7 +20,6 @@ ] const location_store = useLocation() -
@@ -32,23 +31,23 @@ {/if} {#if $signingAccount} -
-
    -
  • - {$_('nav.wallet')} -
  • - -
  • - {$_('nav.transactions')} -
  • - - - - -
-
+
  • + {$_('nav.transactions')} +
  • + + + + + + {/if}
    diff --git a/src/components/settings/UpdateNetwork.svelte b/src/components/settings/UpdateNetwork.svelte index 63cc8c84..a54e6459 100644 --- a/src/components/settings/UpdateNetwork.svelte +++ b/src/components/settings/UpdateNetwork.svelte @@ -48,6 +48,7 @@ .then((res: NetworkPlaylist) => { network_profile.set(res); notify_success("Network Settings Updated"); + check_sync(); }) .catch((error) => { raise_error(error as CarpeError, false, "forceUpstream"); diff --git a/src/components/txs/SetWalletType.svelte b/src/components/txs/SetWalletType.svelte index d1be88fd..f0cf417d 100644 --- a/src/components/txs/SetWalletType.svelte +++ b/src/components/txs/SetWalletType.svelte @@ -1,9 +1,11 @@
    -
      -
    • - - {$_('txs.set_wallet_type.actions_title')} - - -
      -

      - {$_('txs.set_wallet_type.set_slow_title')} -

      -

      {$_('txs.set_wallet_type.subtitle')}

      + {#await is_slow_wallet($signingAccount.account) then res} + {#if res} +

      + {$_('wallet.settings.slow_label')} +

      +

      {$_('wallet.settings.slow_info')}

      + {:else if !$signingAccount.watch_only && $signingAccount.on_chain} +
        +
      • + + {$_('txs.set_wallet_type.actions_title')} + + +
        +

        + {$_('txs.set_wallet_type.set_slow_title')} +

        +

        {$_('txs.set_wallet_type.subtitle')}

        -
        -
        - +
        +
        + +
        + {#if loading} +
        + +
        + {/if} +
        - +
      • +
      - {#if loading} -
      - -
      - {/if} + +
      +
      +

      + {$_('txs.set_wallet_type.warning')} +

      +
      -
    • -
    - - -
    -
    -

    - {$_('txs.set_wallet_type.warning')} -

    - -
    -
    + {/if} + {/await}
    diff --git a/src/components/txs/Transactions.svelte b/src/components/txs/Transactions.svelte index da712e06..d88708fe 100644 --- a/src/components/txs/Transactions.svelte +++ b/src/components/txs/Transactions.svelte @@ -10,6 +10,7 @@ import type { CarpeProfile } from '../../modules/accounts' import { raise_error } from '../../modules/carpeError' import CantStart from '../miner/cards/CantStart.svelte' + import { refreshAccounts } from '../../modules/accountActions' const errorDic = { '120127': $_('txs.transfer.error_slow_wallet'), @@ -46,8 +47,8 @@ account && amount > unscaledCoins(account.balance) ? $_('txs.transfer.error_amount_greater_than_balance') : receiver && receiver.toUpperCase() == account.account.toUpperCase() - ? $_('txs.transfer.error_receiver_equals_sender') - : '' + ? $_('txs.transfer.error_receiver_equals_sender') + : '' onDestroy(async () => { unsubs && unsubs() @@ -68,6 +69,7 @@ waitingTxs = false amount = 0 receiver = null + refreshAccounts() }) .catch((error) => { responses.set(JSON.stringify(error)) diff --git a/src/components/wallet/AccountNote.svelte b/src/components/wallet/AccountNote.svelte new file mode 100644 index 00000000..b94855e3 --- /dev/null +++ b/src/components/wallet/AccountNote.svelte @@ -0,0 +1,49 @@ + + + + + diff --git a/src/components/wallet/AccountsList.svelte b/src/components/wallet/AccountsList.svelte index 4beac967..c1f97b72 100644 --- a/src/components/wallet/AccountsList.svelte +++ b/src/components/wallet/AccountsList.svelte @@ -4,69 +4,92 @@ import Icons from 'uikit/dist/js/uikit-icons' import { allAccounts, formatAccount, signingAccount } from '../../modules/accounts' import { printCoins, unscaledCoins } from '../../modules/coinHelpers' - import { minerLoopEnabled } from '../../modules/miner' import { connected } from '../../modules/networks' import { setAccount } from '../../modules/accountActions' import Actions from './Actions.svelte' import Copy from '../../components/layout/Copy.svelte' + import { preferences, setAccountsListPreference } from '../../modules/user_preferences' UIkit.use(Icons) let showOptions = false + let showNoteColumn = false + $: showNoteColumn = $allAccounts.some( + (account) => account.note !== null && account.note.trim() !== '', + ) + const toggleOptions = () => { - showOptions ? (showOptions = false) : (showOptions = true) + showOptions = !showOptions } - signingAccount.subscribe(() => { + function selectAddress(address) { showOptions = false - }) + setAccount(address) + } - let sortOrder = 'asc' // Initial sort order ('asc' for ascending, 'desc' for descending) - let sortColumn = null // Default column for initial sorting + // Subscribe to the preferences store + let sortOrder = 'asc' + let sortColumn = null + $: if ($preferences.accounts_list_sort_column && $preferences.accounts_list_sort_order) { + sortColumn = $preferences.accounts_list_sort_column + sortOrder = $preferences.accounts_list_sort_order + } - // Function to sort accounts based on the column header clicked + // Update preferences when user clicks to sort function sortAccounts(column) { - if (sortColumn === column) { - sortOrder = sortOrder === 'asc' ? 'desc' : 'asc' - } else { - sortColumn = column - sortOrder = 'desc' - } + const newSortOrder = sortColumn === column && sortOrder === 'asc' ? 'desc' : 'asc' + + // Set sortColumn and sortOrder immediately for local update + sortColumn = column + sortOrder = newSortOrder privateSortAccounts(column) + setAccountsListPreference(column, sortOrder) } + // Private function to sort accounts function privateSortAccounts(column) { $allAccounts = $allAccounts.slice().sort((a, b) => { - // Access nested properties for 'unlocked' and 'balance' let valA = column === 'balance' ? a.balance.total : column === 'unlocked' ? a.balance.unlocked - : formatAccount(a.account) + : column === 'note' + ? a.note || '' + : formatAccount(a.account) let valB = column === 'balance' ? b.balance.total : column === 'unlocked' ? b.balance.unlocked - : formatAccount(b.account) + : column === 'note' + ? b.note || '' + : formatAccount(b.account) - // Handle case-insensitive sorting for string types if (typeof valA === 'string') valA = valA.toLowerCase() if (typeof valB === 'string') valB = valB.toLowerCase() - if (sortOrder === 'asc') { - return valA > valB ? 1 : -1 - } else { - return valA < valB ? 1 : -1 + if (valA === valB) { + // Desempate usando formatAccount + let secondaryValA = formatAccount(a.account).toLowerCase() + let secondaryValB = formatAccount(b.account).toLowerCase() + return sortOrder === 'asc' + ? secondaryValA > secondaryValB + ? 1 + : -1 + : secondaryValA < secondaryValB + ? 1 + : -1 } + + // Ordenação principal + return sortOrder === 'asc' ? (valA > valB ? 1 : -1) : valA < valB ? 1 : -1 }) } - $: if (sortColumn && $allAccounts.length > 0) { - privateSortAccounts(sortColumn) - } + // Re-sort when preferences change + $: sortColumn && $allAccounts.length > 0 && privateSortAccounts(sortColumn)
    @@ -74,9 +97,23 @@ - + {/if} - {#each $allAccounts as a} - - + + {#if showNoteColumn} + + {/if} - -
    + + {#if showNoteColumn} + sortAccounts('note')} + > + {$_('wallet.account_list.note')} + {#if sortColumn === 'note'} + + {/if} + sortAccounts('account')} > {$_('wallet.account_list.address')} @@ -88,7 +125,7 @@ {/if} sortAccounts('unlocked')} > {$_('wallet.account_list.unlocked')} @@ -99,7 +136,10 @@ /> {/if} sortAccounts('balance')}> + sortAccounts('balance')} + > {$_('wallet.account_list.balance')} {#if sortColumn === 'balance'}
    {#if a.account == $signingAccount.account} {a.note ? a.note : ''} setAccount(a.account)} + on:click={() => selectAddress(a.account)} class="uk-transition-toggle uk-table-shrink" - style="cursor:grab" + style="cursor: grab" >
    - {formatAccount(a.account)}{formatAccount(a.account)}
    {printCoins(a.balance.unlocked)} + {#if a.on_chain != null && a.on_chain == false} {$_('wallet.account_list.account_on_chain')} {:else if a.on_chain}
    {#if unscaledCoins(a.balance) < 1} - - +
    {$_('wallet.account_list.message')}
    {/if} - {printCoins(a.balance.total)}
    {:else if a.balance == null} @@ -172,10 +211,9 @@
    {/if} - - {#if showOptions} - - {/if} +
    + +
    diff --git a/src/components/wallet/Actions.svelte b/src/components/wallet/Actions.svelte index 15855399..bd17ad16 100644 --- a/src/components/wallet/Actions.svelte +++ b/src/components/wallet/Actions.svelte @@ -1,27 +1,18 @@
    - - -
    - - {#await is_slow_wallet($signingAccount.account) then res} - {#if res} -

    {$_('wallet.settings.slow_label')}

    -

    {$_('wallet.settings.slow_info')}

    - {:else if !$signingAccount.watch_only && $signingAccount.on_chain } - - {/if} - {/await} + + +
    diff --git a/src/components/wallet/Info.svelte b/src/components/wallet/Info.svelte index 5d13d50b..862f9063 100644 --- a/src/components/wallet/Info.svelte +++ b/src/components/wallet/Info.svelte @@ -1,8 +1,8 @@
    @@ -12,11 +12,11 @@

    {$_('wallet.account_list.address')}: {$signingAccount.account} - - +

    -

    {$_('wallet.account_list.authkey')}: - {$signingAccount.auth_key} - +

    + {$_('wallet.account_list.authkey')}: + {$signingAccount.auth_key} +

    diff --git a/src/components/wallet/Keygen.svelte b/src/components/wallet/Keygen.svelte index 3ed39b7d..94f7be5f 100644 --- a/src/components/wallet/Keygen.svelte +++ b/src/components/wallet/Keygen.svelte @@ -3,7 +3,6 @@ import { invoke } from '@tauri-apps/api/tauri' import { tempCreateAccount } from '../../modules/accounts' import type { CarpeProfile } from '../../modules/accounts' - // import { minerLoopEnabled } from '../../modules/miner' import { raise_error } from '../../modules/carpeError' import { responses } from '../../modules/debug' diff --git a/src/components/wallet/Wallet.svelte b/src/components/wallet/Wallet.svelte index afe35c67..913d70ae 100644 --- a/src/components/wallet/Wallet.svelte +++ b/src/components/wallet/Wallet.svelte @@ -27,14 +27,18 @@ {$_('wallet.wallet')}
    -
    - {$_('wallet.account_list.balance')}: {printCoins($totalBalance.total)} 1} +
    - {$_('wallet.account_list.unlocked')}: {printCoins($totalBalance.unlocked)} -
    + + {$_('wallet.account_list.balance')}: {printCoins($totalBalance.total)} + + + {$_('wallet.account_list.unlocked')}: {printCoins($totalBalance.unlocked)} + +
    + {/if} diff --git a/src/lang/locales/ar.json b/src/lang/locales/ar.json index 76b52c01..09ed7cfd 100644 --- a/src/lang/locales/ar.json +++ b/src/lang/locales/ar.json @@ -44,6 +44,7 @@ "onboard_key": "ONBOARD KEY" }, "account_list": { + "note": "Note", "nickname": "Nickname", "address": "Address", "authkey": "Authkey", @@ -55,6 +56,12 @@ "message": " Your balance will go down for every transaction you send, including mining.", "info_title": "Account Info" }, + "account_note": { + "title": "Account Note", + "description": "A note to help you identify your account:", + "placeholder": "Enter your note here...", + "btn_save": "Save Note" + }, "keygen": { "title": "Create New Account", "description": "After you generate an account and secret phrase, you'll need someone to send one 0L coin to that account for it to be created on chain.", diff --git a/src/lang/locales/de.json b/src/lang/locales/de.json index 4fbdd970..58e7b282 100644 --- a/src/lang/locales/de.json +++ b/src/lang/locales/de.json @@ -44,6 +44,7 @@ "onboard_key": "ONBOARD KEY" }, "account_list": { + "note": "Notiz", "nickname": "Spitzname", "address": "Adresse", "authkey": "Authkey", @@ -55,6 +56,12 @@ "message": " Dein Guthaben verringert sich bei jeder Transaktion, die du sendest, einschließlich Mining.", "info_title": "Konto-Informationen" }, + "account_note": { + "title": "Kontonotiz", + "description": "Eine Notiz, um Ihnen zu helfen, Ihr Konto zu identifizieren:", + "placeholder": "Geben Sie hier Ihre Notiz ein...", + "btn_save": "Notiz speichern" + }, "keygen": { "title": "Neues Konto anlegen", "description": "Nachdem du ein Konto und eine geheime Phrase erstellt hast, muss jemand einen 0L-Coin an dieses Konto senden, damit es auf der Chain erstellt wird.", diff --git a/src/lang/locales/en.json b/src/lang/locales/en.json index 9130d937..8267f22b 100644 --- a/src/lang/locales/en.json +++ b/src/lang/locales/en.json @@ -44,6 +44,7 @@ "onboard_key": "ADDRESS" }, "account_list": { + "note": "Note", "nickname": "Nickname", "address": "Address", "authkey": "Authkey", @@ -55,6 +56,12 @@ "message": "Your balance will go down for every transaction you send, including mining.", "info_title": "Account Info" }, + "account_note": { + "title": "Account Note", + "description": "A note to help you identify your account:", + "placeholder": "Enter your note here...", + "btn_save": "Save Note" + }, "keygen": { "title": "Create New Account", "description": "After you generate an account and secret phrase, you'll need someone to send one 0L coin to that account for it to be created on chain.", diff --git a/src/lang/locales/es.json b/src/lang/locales/es.json index b722e847..18f1fb1b 100644 --- a/src/lang/locales/es.json +++ b/src/lang/locales/es.json @@ -44,6 +44,7 @@ "onboard_key": "ONBOARD KEY" }, "account_list": { + "note": "Nota", "nickname": "Nickname", "address": "Address", "authkey": "Authkey", @@ -55,6 +56,12 @@ "message": " Your balance will go down for every transaction you send, including mining.", "info_title": "Account Info" }, + "account_note": { + "title": "Nota de cuenta", + "description": "Una nota para ayudarte a identificar tu cuenta:", + "placeholder": "Ingresa tu nota aquí...", + "btn_save": "Guardar nota" + }, "keygen": { "title": "Create New Account", "description": "After you generate an account and secret phrase, you'll need someone to send one 0L coin to that account for it to be created on chain.", diff --git a/src/lang/locales/fr.json b/src/lang/locales/fr.json index 9b54b4c0..fe0c0162 100644 --- a/src/lang/locales/fr.json +++ b/src/lang/locales/fr.json @@ -44,6 +44,7 @@ "onboard_key": "ONBOARD KEY" }, "account_list": { + "note": "Note", "nickname": "Pseudo", "address": "Adresse", "authkey": "Authkey", @@ -55,6 +56,12 @@ "message": " Votre solde diminuera pour toutes transactions effectuées, y compris lorsque vous minez.", "info_title": "Informations sur le compte" }, + "account_note": { + "title": "Note du compte", + "description": "Une note pour vous aider à identifier votre compte :", + "placeholder": "Entrez votre note ici...", + "btn_save": "Sauvegarder la note" + }, "keygen": { "title": "Creer un nouveau compte", "description": "Aprés avoir généré votre compte et phrase secrète, vous aurez besoin que quelqu'un envoie un Coin 0L à ce compte pour qu'il soit mis sur la chaîne.", diff --git a/src/lang/locales/it.json b/src/lang/locales/it.json index 56d48839..f9f2902d 100644 --- a/src/lang/locales/it.json +++ b/src/lang/locales/it.json @@ -44,6 +44,7 @@ "onboard_key": "ONBOARD KEY" }, "account_list": { + "note": "Nota", "nickname": "Nickname", "address": "Address", "authkey": "Authkey", @@ -55,6 +56,12 @@ "message": " Your balance will go down for every transaction you send, including mining.", "info_title": "Account Info" }, + "account_note": { + "title": "Nota dell'account", + "description": "Una nota per aiutarti a identificare il tuo account:", + "placeholder": "Inserisci qui la tua nota...", + "btn_save": "Salva nota" + }, "keygen": { "title": "Create New Account", "description": "After you generate an account and secret phrase, you'll need someone to send one 0L coin to that account for it to be created on chain.", diff --git a/src/lang/locales/pt.json b/src/lang/locales/pt.json index a6cc6340..c98f8b64 100644 --- a/src/lang/locales/pt.json +++ b/src/lang/locales/pt.json @@ -44,6 +44,7 @@ "onboard_key": "CHAVE DE INTEGRAÇÃO" }, "account_list": { + "note": "Nota", "nickname": "Apelido", "address": "Conta", "authkey": "Chave de autenticação", @@ -55,6 +56,12 @@ "message": "Seu saldo diminuirá para cada transação que você enviar, incluindo as de mineração.", "info_title": "Informações sobre a conta" }, + "account_note": { + "title": "Nota da Conta", + "description": "Uma nota para ajudá-lo a identificar sua conta:", + "placeholder": "Digite sua nota aqui...", + "btn_save": "Salvar Nota" + }, "keygen": { "title": "Criar nova conta", "description": "Depois de gerar uma conta e a frase secreta, você precisará que alguém envie uma moeda do 0L para esta conta para que ela seja criada na rede.", diff --git a/src/lang/locales/zh_cn.json b/src/lang/locales/zh_cn.json index 18f00d6d..11896c7d 100644 --- a/src/lang/locales/zh_cn.json +++ b/src/lang/locales/zh_cn.json @@ -44,6 +44,7 @@ "onboard_key": "激活码(AuthKey)" }, "account_list": { + "note": "笔记", "nickname": "昵称", "address": "地址", "authkey": "激活码", @@ -55,6 +56,12 @@ "message": " 所有发送的交易(包括:挖矿)都会消耗一些余额作为手续费。", "info_title": "账户信息" }, + "account_note": { + "title": "账户备注", + "description": "一条备注,帮助您识别您的账户:", + "placeholder": "在此输入您的备注...", + "btn_save": "保存备注" + }, "keygen": { "title": "创建新账号", "description": "创建账户后,您的账户需要收到来自链上的任意一个钱包至少1个0L代币,才能够被激活上链。", diff --git a/src/modules/accountActions.ts b/src/modules/accountActions.ts index 411644c1..e28111b3 100644 --- a/src/modules/accountActions.ts +++ b/src/modules/accountActions.ts @@ -54,7 +54,7 @@ export const getDefaultProfile = async () => { export const getAccounts = async () => { // first make sure we don't have empty accounts - invoke('get_all_accounts') + invoke('get_all_accounts_with_notes') .then((result: CarpeProfile[]) => { const watchList = get(watchAccounts) result.map((item) => { @@ -64,7 +64,7 @@ export const getAccounts = async () => { }) allAccounts.set(result) }) - .catch((e) => raise_error(e, true, 'get_all_accounts')) + .catch((e) => raise_error(e, true, 'get_all_accounts_with_notes')) } export const refreshAccounts = async () => { @@ -187,12 +187,6 @@ export function findOneAccount(account: string): CarpeProfile | undefined { export const setAccount = async (account: string, notifySucess = true) => { if (get(signingAccount).account == account) return - // cannot switch profile with miner running - if (get(minerLoopEnabled)) { - notify_error('To switch accounts you need to turn miner off first.') - return - } - invoke('switch_profile', { account }) .then((res: CarpeProfile) => { res.account = res.account.toLocaleUpperCase() @@ -440,11 +434,9 @@ async function onAccountAdd(res: CarpeProfile) { // set as init so we don't get sent back to Newbie account creation. isInit.set(true) responses.set(JSON.stringify(res)) - // cannot switch profile with miner running - if (!get(minerLoopEnabled)) { - res.account = res.account.toLocaleUpperCase() - signingAccount.set(res) - } + res.account = res.account.toLocaleUpperCase() + signingAccount.set(res) + await initNetwork() if (!get(isCarpeTickRunning)) { // start the carpe tick for every 30 secs, this is async @@ -456,3 +448,26 @@ async function onAccountAdd(res: CarpeProfile) { await refreshAccounts().finally(() => navigate('wallet')) } + +export const associateNoteWithAccount = async (account, note) => { + // update allAccounts set account with the new note + const list = get(allAccounts) + const updatedList = list.map((item) => { + if (item.account === account) { + item.note = note + } + return item + }) + allAccounts.set(updatedList) + + // update the backend + try { + const result = await invoke('associate_note_with_account', { account, note }) + refreshAccounts() + notify_success('Note successfully associated with account') + return result + } catch (e) { + raise_error(e, true, 'associateNoteWithAccount') + notify_error('Failed to associate note with account') + } +} diff --git a/src/modules/accountTransactions.ts b/src/modules/accountTransactions.ts index 12d32b20..5f706b93 100644 --- a/src/modules/accountTransactions.ts +++ b/src/modules/accountTransactions.ts @@ -15,6 +15,7 @@ export const setWalletType = async (wtype: WalletType) => { const account = get(signingAccount) invoke('set_slow_wallet', { legacy: account.account.startsWith('0'.repeat(32)), + sender: account.account, }) .then((res: string) => { notify_success(`The account is set to: SlowWallet`) diff --git a/src/modules/accounts.ts b/src/modules/accounts.ts index a5b29668..bfd4e765 100644 --- a/src/modules/accounts.ts +++ b/src/modules/accounts.ts @@ -8,6 +8,7 @@ export interface CarpeProfile { balance?: SlowWalletBalance locale?: string // TODO: refactor, tauri now offers locale of the OS watch_only?: boolean + note: string } export interface SlowWalletBalance { unlocked: number @@ -22,6 +23,7 @@ export const new_account = (account: string, authkey: string, nickname: string): on_chain: false, balance: { unlocked: 0, total: 0 }, watch_only: false, + note: null, } } diff --git a/src/modules/user_preferences.ts b/src/modules/user_preferences.ts new file mode 100644 index 00000000..73d8dd2b --- /dev/null +++ b/src/modules/user_preferences.ts @@ -0,0 +1,42 @@ +import { writable } from 'svelte/store' +import { invoke } from '@tauri-apps/api/tauri' + +// Define the interface for UserPreferences +interface UserPreferences { + accounts_list_sort_column?: string + accounts_list_sort_order?: string +} + +// Create a writable store to hold the preferences +export const preferences = writable({}) + +// Function to load preferences from the backend +export async function loadUserPreferences() { + try { + const prefs = await invoke('get_user_preferences') + preferences.set(prefs) + } catch (error) { + console.error('Failed to load user preferences:', error) + // Optionally set defaults or handle errors in a specific way + preferences.set({}) + } +} + +// Function to save preferences to the backend +export async function setAccountsListPreference(sortColumn: string, sortOrder: string) { + try { + await invoke('set_accounts_list_preference', { sortColumn, sortOrder }) + // Update the local store after successful backend update + preferences.update((prefs) => ({ + ...prefs, + accounts_list_sort_column: sortColumn, + accounts_list_sort_order: sortOrder, + })) + } catch (error) { + console.error('Failed to set accounts list preference:', error) + // Handle errors, possibly revert to old values or show user feedback + } +} + +// Initial load of preferences +loadUserPreferences()