// Copyright 2023 The Grim Developers
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

use std::sync::Arc;
use std::thread;
use egui::{Id, Margin, RichText, ScrollArea};
use egui::scroll_area::ScrollBarVisibility;
use grin_core::core::{amount_from_hr_string, amount_to_hr_string};
use grin_wallet_libwallet::{Error, Slate, SlateState};
use log::error;
use parking_lot::RwLock;

use crate::gui::Colors;
use crate::gui::icons::{BROOM, CLIPBOARD_TEXT, COPY, DOWNLOAD_SIMPLE, PROHIBIT, QR_CODE, SCAN, UPLOAD_SIMPLE};
use crate::gui::platform::PlatformCallbacks;
use crate::gui::views::{CameraContent, Modal, QrCodeContent, Root, View};
use crate::gui::views::types::{ModalPosition, QrScanResult, TextEditOptions};
use crate::gui::views::wallets::wallet::types::{SLATEPACK_MESSAGE_HINT, WalletTab, WalletTabType};
use crate::gui::views::wallets::wallet::WalletContent;
use crate::wallet::types::WalletTransaction;
use crate::wallet::Wallet;

#[derive(Clone, Eq, PartialEq, Debug, thiserror::Error)]
enum MessageError {
    #[error("{0}")]
    Response(String),
    #[error("{0}")]
    Parse(String),
    #[error("{0}")]
    Finalize(String),
    #[error("{0}")]
    Other(String),
}

impl MessageError {
    pub fn text(&self) -> &String {
        match self {
            MessageError::Response(text) => text,
            MessageError::Parse(text) => text,
            MessageError::Finalize(text) => text,
            MessageError::Other(text) => text
        }
    }
}

/// Slatepacks messages interaction tab content.
pub struct WalletMessages {
    /// Slatepack message to create response message.
    message_edit: String,
    /// Parsed Slatepack message.
    message_slate: Option<Slate>,
    /// Flag to check if message request is loading.
    message_loading: bool,
    /// Message request result.
    receive_pay_result: Arc<RwLock<Option<(Slate, Result<String, Error>)>>>,
    /// Message finalize or post result.
    final_post_result: Arc<RwLock<Option<Result<(), Error>>>>,
    /// Slatepack error on finalization, parse and response creation.
    message_error: Option<MessageError>,
    /// Generated Slatepack response message.
    response_edit: String,
    /// Flag to check if Dandelion is needed to finalize transaction.
    dandelion: bool,

    /// Flag to check if invoice or sending request was opened for [`Modal`].
    request_invoice: bool,
    /// Amount to send or receive at [`Modal`].
    request_amount_edit: String,
    /// Generated Slatepack message as request to send or receive funds at [`Modal`].
    request_edit: String,
    /// Flag to check if there is an error happened on request creation at [`Modal`].
    request_error: Option<MessageError>,
    /// Flag to check if response Slatepack message is showing as QR code image at [`Modal`].
    request_qr: bool,
    /// Request Slatepack message QR code image [`Modal`] content.
    request_qr_content: QrCodeContent,
    /// Flag to check if request is loading at [`Modal`].
    request_loading: bool,
    /// Request result if there is no error at [`Modal`].
    request_result: Arc<RwLock<Option<Result<(Slate, String), Error>>>>,

    /// Camera content for Slatepack message QR code scanning [`Modal`].
    message_camera_content: CameraContent,
    /// Flag to check if there is an error on scanning Slatepack message QR code at [`Modal`].
    message_scan_error: bool,

    /// QR code Slatepacks message text to show at [`Modal`].
    qr_message_text: Option<String>,
    /// QR code Slatepack message image [`Modal`] content.
    qr_message_content: QrCodeContent,
}

/// Identifier for amount input [`Modal`] to create invoice or sending request.
const REQUEST_MODAL: &'static str = "messages_request_modal";

/// Identifier for QR code Slatepack message scan [`Modal`].
const QR_SLATEPACK_MESSAGE_SCAN_MODAL: &'static str = "qr_slatepack_message_scan_modal";

/// Identifier for [`Modal`] to show QR code Slatepack message image.
const QR_SLATEPACK_MESSAGE_MODAL: &'static str = "qr_slatepack_message_modal";

impl WalletTab for WalletMessages {
    fn get_type(&self) -> WalletTabType {
        WalletTabType::Messages
    }

    fn ui(&mut self,
          ui: &mut egui::Ui,
          _: &mut eframe::Frame,
          wallet: &mut Wallet,
          cb: &dyn PlatformCallbacks) {
        if WalletContent::sync_ui(ui, wallet) {
            return;
        }

        // Show modal content for this ui container.
        self.modal_content_ui(ui, wallet, cb);

        // Show manual wallet content panel.
        egui::CentralPanel::default()
            .frame(egui::Frame {
                stroke: View::ITEM_STROKE,
                fill: Colors::WHITE,
                inner_margin: Margin {
                    left: View::far_left_inset_margin(ui) + 4.0,
                    right: View::get_right_inset() + 4.0,
                    top: 3.0,
                    bottom: 4.0,
                },
                ..Default::default()
            })
            .show_inside(ui, |ui| {
                ScrollArea::vertical()
                    .scroll_bar_visibility(ScrollBarVisibility::AlwaysHidden)
                    .id_source(Id::from("wallet_messages").with(wallet.get_config().id))
                    .auto_shrink([false; 2])
                    .show(ui, |ui| {
                        ui.vertical_centered(|ui| {
                            View::max_width_ui(ui, Root::SIDE_PANEL_WIDTH * 1.3, |ui| {
                                self.ui(ui, wallet, cb);
                            });
                        });
                    });
            });
    }
}

impl WalletMessages {
    /// Create new content instance, put message into input if provided.
    pub fn new(dandelion: bool, message: Option<String>) -> Self {
        Self {
            request_invoice: false,
            message_edit: message.unwrap_or("".to_string()),
            message_slate: None,
            message_loading: false,
            receive_pay_result: Arc::new(RwLock::new(None)),
            final_post_result: Arc::new(RwLock::new(None)),
            message_error: None,
            response_edit: "".to_string(),
            dandelion,
            request_amount_edit: "".to_string(),
            request_edit: "".to_string(),
            request_error: None,
            request_qr: false,
            request_qr_content: QrCodeContent::new("".to_string(), true),
            request_loading: false,
            request_result: Arc::new(RwLock::new(None)),
            message_camera_content: CameraContent::default(),
            message_scan_error: false,
            qr_message_text: None,
            qr_message_content: QrCodeContent::new("".to_string(), true),
        }
    }

    /// Draw manual wallet transaction interaction content.
    pub fn ui(&mut self,
              ui: &mut egui::Ui,
              wallet: &mut Wallet,
              cb: &dyn PlatformCallbacks) {
        ui.add_space(3.0);

        // Show creation of request to send or receive funds.
        self.request_ui(ui, wallet, cb);

        ui.add_space(12.0);
        View::horizontal_line(ui, Colors::ITEM_STROKE);
        ui.add_space(6.0);

        // Show Slatepack message input field.
        self.input_slatepack_ui(ui, wallet, cb);

        ui.add_space(6.0);
    }

    /// Draw [`Modal`] content for this ui container.
    fn modal_content_ui(&mut self,
                        ui: &mut egui::Ui,
                        wallet: &mut Wallet,
                        cb: &dyn PlatformCallbacks) {
        match Modal::opened() {
            None => {}
            Some(id) => {
                match id {
                    REQUEST_MODAL => {
                        Modal::ui(ui.ctx(), |ui, modal| {
                            self.request_modal_ui(ui, wallet, modal, cb);
                        });
                    }
                    QR_SLATEPACK_MESSAGE_SCAN_MODAL => {
                        Modal::ui(ui.ctx(), |ui, modal| {
                            self.qr_message_scan_modal_ui(ui, modal, wallet, cb);
                        });
                    }
                    QR_SLATEPACK_MESSAGE_MODAL => {
                        Modal::ui(ui.ctx(), |ui, modal| {
                            self.qr_message_modal_ui(ui, modal, cb);
                        });
                    }
                    _ => {}
                }
            }
        }
    }

    /// Draw creation of request to send or receive funds.
    fn request_ui(&mut self,
                  ui: &mut egui::Ui,
                  wallet: &Wallet,
                  cb: &dyn PlatformCallbacks) {
        ui.label(RichText::new(t!("wallets.create_request_desc"))
            .size(16.0)
            .color(Colors::INACTIVE_TEXT));
        ui.add_space(7.0);

        // Show send button only if balance is not empty.
        let data = wallet.get_data().unwrap();
        if data.info.amount_currently_spendable > 0 {
            // Setup spacing between buttons.
            ui.spacing_mut().item_spacing = egui::Vec2::new(6.0, 0.0);

            ui.columns(2, |columns| {
                columns[0].vertical_centered_justified(|ui| {
                    // Draw sending request creation button.
                    let send_text = format!("{} {}", UPLOAD_SIMPLE, t!("wallets.send"));
                    View::colored_text_button(ui, send_text, Colors::RED, Colors::BUTTON, || {
                        self.show_request_modal(false, cb);
                    });
                });
                columns[1].vertical_centered_justified(|ui| {
                    // Draw invoice request creation button.
                    self.receive_button_ui(ui, cb);
                });
            });
        } else {
            // Draw invoice creation button.
            self.receive_button_ui(ui, cb);
        }
    }

    /// Draw invoice request creation button.
    fn receive_button_ui(&mut self, ui: &mut egui::Ui, cb: &dyn PlatformCallbacks) {
        let receive_text = format!("{} {}", DOWNLOAD_SIMPLE, t!("wallets.receive"));
        View::colored_text_button(ui, receive_text, Colors::GREEN, Colors::BUTTON, || {
            self.show_request_modal(true, cb);
        });
    }

    /// Show [`Modal`] to create invoice or sending request.
    fn show_request_modal(&mut self, invoice: bool, cb: &dyn PlatformCallbacks) {
        // Setup modal values.
        self.request_invoice = invoice;
        self.request_qr = false;
        self.request_edit = "".to_string();
        self.request_amount_edit = "".to_string();
        self.request_error = None;
        {
            let mut w_result = self.request_result.write();
            *w_result = None;
        }
        // Show receive amount modal.
        let title = if self.request_invoice {
            t!("wallets.receive")
        } else {
            t!("wallets.send")
        };
        Modal::new(REQUEST_MODAL).position(ModalPosition::CenterTop).title(title).show();
        cb.show_keyboard();
    }

    /// Draw invoice or sending request creation [`Modal`] content.
    fn request_modal_ui(&mut self,
                        ui: &mut egui::Ui,
                        wallet: &mut Wallet,
                        modal: &Modal,
                        cb: &dyn PlatformCallbacks) {
        ui.add_space(6.0);

        if self.request_loading {
            ui.add_space(34.0);
            ui.vertical_centered(|ui| {
                View::big_loading_spinner(ui);
            });
            ui.add_space(50.0);

            // Check if there is request result error.
            if self.request_error.is_some() {
                modal.enable_closing();
                self.request_loading = false;
                return;
            }

            // Update data on request result.
            let r_request = self.request_result.read();
            if r_request.is_some() {
                let message = r_request.as_ref().unwrap();
                match message {
                    Ok((_, message)) => {
                        self.request_edit = message.clone();
                    }
                    Err(err) => {
                        match err {
                            Error::NotEnoughFunds { .. } => {
                                let m = t!(
                                    "wallets.pay_balance_error",
                                    "amount" => self.request_amount_edit
                                );
                                self.request_error = Some(MessageError::Other(m));
                            }
                            _ => {
                                let m = t!("wallets.invoice_slatepack_err");
                                self.request_error = Some(MessageError::Other(m));
                            }
                        }
                    }
                }
                modal.enable_closing();
                self.request_loading = false;
            }
        } else if self.request_edit.is_empty() {
            ui.vertical_centered(|ui| {
                let enter_text = if self.request_invoice {
                    t!("wallets.enter_amount_receive")
                } else {
                    let data = wallet.get_data().unwrap();
                    let amount = amount_to_hr_string(data.info.amount_currently_spendable, true);
                    t!("wallets.enter_amount_send","amount" => amount)
                };
                ui.label(RichText::new(enter_text)
                    .size(17.0)
                    .color(Colors::GRAY));
            });
            ui.add_space(8.0);

            // Draw request amount text input.
            let amount_edit_id = Id::from(modal.id).with(wallet.get_config().id);
            let mut amount_edit_opts = TextEditOptions::new(amount_edit_id).h_center();
            let amount_edit_before = self.request_amount_edit.clone();
            View::text_edit(ui, cb, &mut self.request_amount_edit, &mut amount_edit_opts);

            // Check value if input was changed.
            if amount_edit_before != self.request_amount_edit {
                self.request_error = None;
                if !self.request_amount_edit.is_empty() {
                    match amount_from_hr_string(self.request_amount_edit.as_str()) {
                        Ok(a) => {
                            if !self.request_amount_edit.contains(".") {
                                // To avoid input of several "0".
                                if a == 0 {
                                    self.request_amount_edit = "0".to_string();
                                    return;
                                }
                            } else {
                                // Check input after ".".
                                let parts = self.request_amount_edit
                                    .split(".")
                                    .collect::<Vec<&str>>();
                                if parts.len() == 2 && parts[1].len() > 9 {
                                    self.request_amount_edit = amount_edit_before;
                                    return;
                                }
                            }

                            // Do not input amount more than balance in sending.
                            if !self.request_invoice {
                                let b = wallet.get_data().unwrap().info.amount_currently_spendable;
                                if b < a {
                                    self.request_amount_edit = amount_edit_before;
                                }
                            }
                        }
                        Err(_) => {
                            self.request_amount_edit = amount_edit_before;
                        }
                    }
                }
            }

            // Show request creation error.
            if self.request_error.is_some() {
                ui.add_space(12.0);
                ui.vertical_centered(|ui| {
                    ui.label(RichText::new(self.request_error.clone().unwrap().text())
                        .size(17.0)
                        .color(Colors::RED));
                });
            }

            ui.add_space(12.0);

            // Setup spacing between buttons.
            ui.spacing_mut().item_spacing = egui::Vec2::new(6.0, 0.0);

            ui.columns(2, |columns| {
                columns[0].vertical_centered_justified(|ui| {
                    View::button(ui, t!("modal.cancel"), Colors::WHITE, || {
                        self.request_amount_edit = "".to_string();
                        self.request_error = None;
                        cb.hide_keyboard();
                        modal.close();
                    });
                });
                columns[1].vertical_centered_justified(|ui| {
                    // Button to create Slatepack message request.
                    View::button(ui, t!("continue"), Colors::WHITE, || {
                        if self.request_amount_edit.is_empty() {
                            return;
                        }
                        if let Ok(a) = amount_from_hr_string(self.request_amount_edit.as_str()) {
                            cb.hide_keyboard();
                            // Setup data for request.
                            let wallet = wallet.clone();
                            let invoice = self.request_invoice.clone();
                            let result = self.request_result.clone();
                            // Send request at another thread.
                            self.request_loading = true;
                            modal.disable_closing();
                            thread::spawn(move || {
                                let message = if invoice {
                                    wallet.issue_invoice(a)
                                } else {
                                    wallet.send(a)
                                };
                                let mut w_result = result.write();
                                *w_result = Some(message);
                            });
                        } else {
                            self.request_error = Some(
                                MessageError::Other(t!("wallets.invoice_slatepack_err"))
                            );
                        }
                    });
                });
            });
            ui.add_space(6.0);
        } else {
            ui.vertical_centered(|ui| {
                let amount = amount_from_hr_string(self.request_amount_edit.as_str()).unwrap();
                let amount_format = amount_to_hr_string(amount, true);
                let desc_text = if self.request_invoice {
                    t!("wallets.invoice_desc","amount" => amount_format)
                } else {
                    t!("wallets.send_request_desc","amount" => amount_format)
                };
                ui.label(RichText::new(desc_text).size(16.0).color(Colors::GRAY));
            });
            ui.add_space(6.0);

            // Draw QR code content if requested.
            if self.request_qr {
                // Draw QR code content.
                let text = self.request_edit.clone();
                if text.is_empty() {
                    self.request_qr = false;
                }
                self.request_qr_content.ui(ui, text.clone(), cb);
                ui.add_space(6.0);

                // Show button to close modal.
                ui.vertical_centered_justified(|ui| {
                    View::button(ui, t!("close"), Colors::WHITE, || {
                        self.request_qr_content.clear_state();
                        self.request_qr = false;
                        modal.close();
                    });
                });
                ui.add_space(6.0);
                return;
            }

            View::horizontal_line(ui, Colors::ITEM_STROKE);
            ui.add_space(3.0);

            // Draw request Slatepack message text.
            let scroll_id = if self.request_invoice {
                Id::from("receive_request").with(wallet.get_config().id)
            } else {
                Id::from("send_request").with(wallet.get_config().id)
            };
            ScrollArea::vertical()
                .id_source(scroll_id)
                .scroll_bar_visibility(ScrollBarVisibility::AlwaysHidden)
                .max_height(128.0)
                .auto_shrink([false; 2])
                .show(ui, |ui| {
                    ui.add_space(7.0);
                    let input_id =  Id::from(scroll_id).with("_input");
                    egui::TextEdit::multiline(&mut self.request_edit)
                        .id(input_id)
                        .font(egui::TextStyle::Small)
                        .desired_rows(5)
                        .interactive(false)
                        .hint_text(SLATEPACK_MESSAGE_HINT)
                        .desired_width(f32::INFINITY)
                        .show(ui);
                    ui.add_space(6.0);
                });
            ui.add_space(2.0);
            View::horizontal_line(ui, Colors::ITEM_STROKE);

            ui.add_space(10.0);

            ui.scope(|ui| {
                // Setup spacing between buttons.
                ui.spacing_mut().item_spacing = egui::Vec2::new(6.0, 0.0);

                ui.columns(2, |columns| {
                    columns[0].vertical_centered_justified(|ui| {
                        // Draw button to show request as QR code.
                        let qr_text = format!("{} {}", QR_CODE, t!("qr_code"));
                        View::button(ui, qr_text, Colors::BUTTON, || {
                            self.request_qr = true;
                        });
                    });
                    columns[1].vertical_centered_justified(|ui| {
                        // Draw button to copy request to clipboard.
                        let copy_text = format!("{} {}", COPY, t!("copy"));
                        View::button(ui, copy_text, Colors::BUTTON, || {
                            cb.copy_string_to_buffer(self.request_edit.clone());
                            self.request_amount_edit = "".to_string();
                            self.request_edit = "".to_string();
                            modal.close();
                        });
                    });
                });

                ui.add_space(10.0);

                ui.columns(2, |columns| {
                    columns[0].vertical_centered_justified(|ui| {
                        // Draw button to cancel transaction.
                        let cancel = t!("modal.cancel");
                        View::colored_text_button(ui, cancel, Colors::RED, Colors::BUTTON, || {
                            if let Ok(slate) = wallet.parse_slatepack(&self.request_edit) {
                                if let Some(tx) = wallet.tx_by_slate(&slate) {
                                    wallet.cancel(tx.data.id);
                                }
                            }
                            self.request_amount_edit = "".to_string();
                            self.request_edit = "".to_string();
                            cb.hide_keyboard();
                            modal.close();
                        });
                    });
                    columns[1].vertical_centered_justified(|ui| {
                        // Draw button to close modal.
                        View::button(ui, t!("close"), Colors::WHITE, || {
                            self.request_amount_edit = "".to_string();
                            self.request_edit = "".to_string();
                            modal.close();
                        });
                    });
                });
            });
            ui.add_space(6.0);
        }
    }

    /// Draw Slatepack message input content.
    fn input_slatepack_ui(&mut self,
                      ui: &mut egui::Ui,
                      wallet: &mut Wallet,
                      cb: &dyn PlatformCallbacks) {
        // Setup description text.
        let empty_fields = self.message_edit.is_empty() && self.request_edit.is_empty();
        let response_empty = self.response_edit.is_empty();
        if let Some(err) = &self.message_error {
            ui.label(RichText::new(err.text()).size(16.0).color(Colors::RED));
        } else {
            let desc_text = if self.message_slate.is_none() || empty_fields {
                t!("wallets.input_slatepack_desc")
            } else {
                let slate = self.message_slate.clone().unwrap();
                let amount = amount_to_hr_string(slate.amount, true);
                match slate.state {
                    SlateState::Standard1 => {
                        t!("wallets.parse_s1_slatepack_desc","amount" => amount)
                    }
                    SlateState::Standard2 => {
                        t!("wallets.parse_s2_slatepack_desc","amount" => amount)
                    }
                    SlateState::Standard3 => {
                        t!("wallets.parse_s3_slatepack_desc","amount" => amount)
                    }
                    SlateState::Invoice1 => {
                        t!("wallets.parse_i1_slatepack_desc","amount" => amount)
                    }
                    SlateState::Invoice2 => {
                        t!("wallets.parse_i2_slatepack_desc","amount" => amount)
                    }
                    SlateState::Invoice3 => {
                        t!("wallets.parse_i3_slatepack_desc","amount" => amount)
                    }
                    _ => {
                        t!("wallets.input_slatepack_desc")
                    }
                }
            };
            ui.label(RichText::new(desc_text).size(16.0).color(Colors::INACTIVE_TEXT));
        }
        ui.add_space(6.0);

        // Setup Slatepack message text input.
        let message = if response_empty {
            &mut self.message_edit
        } else {
            &mut self.response_edit
        };

        // Save message to check for changes.
        let message_before = message.clone();

        View::horizontal_line(ui, Colors::ITEM_STROKE);
        ui.add_space(3.0);
        let scroll_id = Id::from(
            if response_empty {
                "message_input"
            } else {
                "response_input"
            }).with(wallet.get_config().id);
        ScrollArea::vertical()
            .id_source(scroll_id)
            .scroll_bar_visibility(ScrollBarVisibility::AlwaysHidden)
            .max_height(128.0)
            .auto_shrink([false; 2])
            .show(ui, |ui| {
                ui.add_space(7.0);
                let input_id = scroll_id.with("_input");
                let resp = egui::TextEdit::multiline(message)
                    .id(input_id)
                    .font(egui::TextStyle::Small)
                    .desired_rows(5)
                    .interactive(response_empty && !self.message_loading)
                    .hint_text(SLATEPACK_MESSAGE_HINT)
                    .desired_width(f32::INFINITY)
                    .show(ui)
                    .response;
                // Show soft keyboard on click.
                if response_empty && resp.clicked() {
                    cb.show_keyboard();
                }
                if response_empty && resp.has_focus() {
                    // Apply text from input on Android as temporary fix for egui.
                    View::on_soft_input(ui, input_id, message);
                }
                ui.add_space(6.0);
            });
        ui.add_space(2.0);
        View::horizontal_line(ui, Colors::ITEM_STROKE);
        ui.add_space(10.0);

        // Parse Slatepack message if input field was changed, resetting message error.
        if &message_before != message {
            self.parse_message(wallet);
        }

        // Draw buttons to clear/copy/paste.
        let columns_num = if self.message_loading { 1 } else { 2 };
        let mut show_dandelion = false;
        ui.scope(|ui| {
            // Setup spacing between buttons.
            ui.spacing_mut().item_spacing = egui::Vec2::new(6.0, 0.0);

            ui.columns(columns_num, |columns| {
                let first_column_content = |ui: &mut egui::Ui| {
                    if self.message_slate.is_some() && !empty_fields {
                        if self.response_edit.is_empty() {
                            // Draw button to clear message input.
                            let clear_text = format!("{} {}", BROOM, t!("clear"));
                            View::button(ui, clear_text, Colors::BUTTON, || {
                                self.message_edit.clear();
                                self.response_edit.clear();
                                self.message_error = None;
                                self.message_slate = None;
                            });
                        } else {
                            // Draw button to show Slatepack message as QR code.
                            let qr_text = format!("{} {}", QR_CODE, t!("qr_code"));
                            View::button(ui, qr_text, Colors::BUTTON, || {
                                let text = self.response_edit.clone();
                                self.message_edit.clear();
                                self.response_edit.clear();
                                self.show_qr_message_modal(text);
                            });
                        }
                    } else {
                        if self.message_loading {
                            View::small_loading_spinner(ui);
                            // Check loading result.
                            self.check_message_loading_result(wallet);
                        } else {
                            // Draw button to scan Slatepack message QR code.
                            let scan_text = format!("{} {}", SCAN, t!("scan"));
                            View::button(ui, scan_text, Colors::BUTTON, || {
                                self.message_edit.clear();
                                self.message_error = None;
                                self.show_qr_message_scan_modal(cb);
                            });
                        }
                    }
                };
                if columns_num == 1 {
                    columns[0].vertical_centered(first_column_content);
                } else {
                    columns[0].vertical_centered_justified(first_column_content);
                    columns[1].vertical_centered_justified(|ui| {
                        if self.message_slate.is_some() && !empty_fields {
                            if !self.response_edit.is_empty() {
                                // Draw button to copy response to clipboard.
                                let copy_text = format!("{} {}", COPY, t!("copy"));
                                View::button(ui, copy_text, Colors::BUTTON, || {
                                    cb.copy_string_to_buffer(self.response_edit.clone());
                                    self.message_edit.clear();
                                    self.response_edit.clear();
                                    self.message_slate = None;
                                });
                            } else {
                                show_dandelion = true;
                                // Draw button to finalize or repost transaction.
                                View::button(ui, t!("wallets.finalize"), Colors::GOLD, || {
                                    let slate = self.message_slate.clone().unwrap();
                                    self.message_slate = None;
                                    let dandelion = self.dandelion;
                                    let message_edit = self.message_edit.clone();
                                    let wallet = wallet.clone();
                                    let result = self.final_post_result.clone();

                                    // Finalize or post transaction at separate thread.
                                    self.message_loading = true;
                                    thread::spawn(move || {
                                        let res = if slate.state == SlateState::Invoice3 ||
                                            slate.state == SlateState::Standard3 {
                                            wallet.post(&slate, dandelion)
                                        } else {
                                            match wallet.finalize(&message_edit, dandelion) {
                                                Ok(_) => {
                                                    Ok(())
                                                }
                                                Err(e) => {
                                                    Err(e)
                                                }
                                            }
                                        };
                                        let mut w_res = result.write();
                                        *w_res = Some(res);
                                    });
                                });
                            }
                        } else {
                            // Draw button to paste text from clipboard.
                            let paste = format!("{} {}", CLIPBOARD_TEXT, t!("paste"));
                            View::button(ui, paste, Colors::BUTTON, || {
                                let buf = cb.get_string_from_buffer();
                                let previous = self.message_edit.clone();
                                self.message_edit = buf.clone().trim().to_string();
                                // Parse Slatepack message resetting message error.
                                if buf != previous {
                                    self.parse_message(wallet);
                                }
                            });
                        }
                    });
                }
            });

            ui.add_space(10.0);

            // Draw clear button on message input or cancel and clear buttons on response.
            if !self.message_loading {
                if self.message_slate.is_none() && !self.message_edit.is_empty() {
                    // Draw button to clear message input.
                    let clear_text = format!("{} {}", BROOM, t!("clear"));
                    View::button(ui, clear_text, Colors::BUTTON, || {
                        self.message_edit.clear();
                        self.response_edit.clear();
                        self.message_error = None;
                        self.message_slate = None;
                    });
                } else if !self.response_edit.is_empty() && self.message_slate.is_some() {
                    // Draw cancel button.
                    let cancel = format!("{} {}", PROHIBIT, t!("modal.cancel"));
                    View::colored_text_button(ui, cancel, Colors::RED, Colors::BUTTON, || {
                        let slate = self.message_slate.clone().unwrap();
                        if let Some(tx) = wallet.tx_by_slate(&slate) {
                            wallet.cancel(tx.data.id);
                            self.message_edit.clear();
                            self.response_edit.clear();
                            self.message_slate = None;
                        }
                    });
                }
            }
        });

        // Draw setup of ability to post transaction with Dandelion.
        if show_dandelion {
            let dandelion_before = self.dandelion;
            View::checkbox(ui, dandelion_before, t!("wallets.use_dandelion"), || {
                self.dandelion = !dandelion_before;
                wallet.update_use_dandelion(self.dandelion);
            });
        }
    }

    /// Show QR code Slatepack message [`Modal`].
    pub fn show_qr_message_modal(&mut self, text: String) {
        self.qr_message_text = Some(text);
        self.qr_message_content.clear_state();
        let slate = self.message_slate.clone().unwrap();
        let title = if slate.state == SlateState::Standard1 {
            t!("wallets.receive")
        } else {
            t!("wallets.send")
        };
        Modal::new(QR_SLATEPACK_MESSAGE_MODAL)
            .position(ModalPosition::CenterTop)
            .title(title)
            .show();
    }

    /// Draw QR code Slatepack message image [`Modal`] content.
    fn qr_message_modal_ui(&mut self, ui: &mut egui::Ui, m: &Modal, cb: &dyn PlatformCallbacks) {
        ui.add_space(6.0);

        // Setup title for Slatepack message.
        ui.vertical_centered(|ui| {
            let slate = self.message_slate.clone().unwrap();
            let amount = amount_to_hr_string(slate.amount, true);
            let title = if slate.state == SlateState::Standard1 {
                t!("wallets.parse_s1_slatepack_desc","amount" => amount)
            } else {
                t!("wallets.parse_i1_slatepack_desc","amount" => amount)
            };
            ui.label(RichText::new(title).size(16.0).color(Colors::INACTIVE_TEXT));
        });
        ui.add_space(6.0);

        // Draw QR code content.
        let text = self.qr_message_text.clone().unwrap();
        self.qr_message_content.ui(ui, text.clone(), cb);
        ui.add_space(6.0);

        ui.vertical_centered_justified(|ui| {
            View::button(ui, t!("close"), Colors::WHITE, || {
                self.qr_message_text = None;
                self.qr_message_content.clear_state();
                self.response_edit.clear();
                self.message_slate = None;
                m.close();
            });
        });
        ui.add_space(6.0);
    }

    /// Show QR code Slatepack message scanner [`Modal`].
    pub fn show_qr_message_scan_modal(&mut self, cb: &dyn PlatformCallbacks) {
        self.message_scan_error = false;
        // Show QR code scan modal.
        Modal::new(QR_SLATEPACK_MESSAGE_SCAN_MODAL)
            .position(ModalPosition::CenterTop)
            .title(t!("scan_qr"))
            .closeable(false)
            .show();
        cb.start_camera();
    }

    /// Draw QR code scanner [`Modal`] content.
    fn qr_message_scan_modal_ui(&mut self,
                                ui: &mut egui::Ui,
                                modal: &Modal,
                                wallet: &Wallet,
                                cb: &dyn PlatformCallbacks) {
        if self.message_scan_error {
            ui.add_space(6.0);
            ui.vertical_centered(|ui| {
                let err_text = format!("{}", t!("wallets.parse_slatepack_err")).replace(":", ".");
                ui.label(RichText::new(err_text)
                    .size(17.0)
                    .color(Colors::RED));
            });
            ui.add_space(12.0);

            // Setup spacing between buttons.
            ui.spacing_mut().item_spacing = egui::Vec2::new(6.0, 0.0);

            ui.columns(2, |columns| {
                columns[0].vertical_centered_justified(|ui| {
                    View::button(ui, t!("close"), Colors::WHITE, || {
                        self.message_scan_error = false;
                        modal.close();
                    });
                });
                columns[1].vertical_centered_justified(|ui| {
                    View::button(ui, t!("repeat"), Colors::WHITE, || {
                        Modal::set_title(t!("scan_qr"));
                        self.message_scan_error = false;
                        cb.start_camera();
                    });
                });
            });
            ui.add_space(6.0);
            return;
        } else if let Some(result) = self.message_camera_content.qr_scan_result() {
            cb.stop_camera();
            self.message_camera_content.clear_state();
            match &result {
                QrScanResult::Slatepack(text) => {
                    self.message_edit = text.to_string();
                    self.parse_message(wallet);
                    modal.close();
                }
                _ => {
                    self.message_scan_error = true;
                }
            }
        } else {
            ui.add_space(6.0);
            self.message_camera_content.ui(ui, cb);
            ui.add_space(8.0);
        }

        ui.vertical_centered_justified(|ui| {
            View::button(ui, t!("modal.cancel"), Colors::WHITE, || {
                cb.stop_camera();
                modal.close();
            });
        });
        ui.add_space(6.0);
    }

    /// Check Slatepack message request loading result.
    fn check_message_loading_result(&mut self, wallet: &Wallet) {
        // Check finalize post pay result.
        let has_finalize_post_result = {
            let r_res = self.final_post_result.read();
            r_res.is_some()
        };
        if has_finalize_post_result {
            let resp = {
                let r_res = self.final_post_result.read();
                r_res.as_ref().unwrap().clone()
            };
            if resp.is_ok() {
                self.message_edit.clear();
                self.message_slate = None;
            } else {
                self.message_error = Some(
                    MessageError::Finalize(
                        t!("wallets.finalize_slatepack_err")
                    )
                );
            }
            self.message_loading = false;
        }

        // Check receive pay result.
        let has_receive_pay_result = {
            let r_res = self.receive_pay_result.read();
            r_res.is_some()
        };
        if has_receive_pay_result {
            let (slate, resp) = {
                let r_res = self.receive_pay_result.read();
                r_res.as_ref().unwrap().clone()
            };
            if resp.is_ok() {
                self.response_edit = resp.as_ref().unwrap().clone();
            } else {
                let err = resp.as_ref().err().unwrap();
                match err {
                    // Set already canceled transaction error message.
                    Error::TransactionWasCancelled {..}
                    => {
                        self.message_error = Some(
                            MessageError::Response(
                                t!("wallets.resp_canceled_err")
                            )
                        );
                    }
                    // Set an error when there is not enough funds to pay.
                    Error::NotEnoughFunds {..} => {
                        let m = t!(
                                                "wallets.pay_balance_error",
                                                "amount" => amount_to_hr_string(slate.amount, true)
                                            );
                        self.message_error = Some(MessageError::Response(m));
                    }
                    // Set default error message.
                    _ => {
                        self.message_error = Some(
                            MessageError::Response(
                                t!("wallets.resp_slatepack_err")
                            )
                        );
                    }
                }
                // Check if tx with same slate id already exists.
                if self.message_error.is_none() {
                    let exists_tx = wallet.tx_by_slate(&slate).is_some();
                    if exists_tx {
                        let mut sl = slate.clone();
                        sl.state = if sl.state == SlateState::Standard1 {
                            SlateState::Standard2
                        } else {
                            SlateState::Invoice2
                        };
                        match wallet.read_slatepack(&sl) {
                            None => {
                                self.message_error = Some(
                                    MessageError::Response(
                                        t!("wallets.resp_slatepack_err")
                                    )
                                );
                            }
                            Some(sp) => {
                                self.response_edit = sp;
                            }
                        }
                    }
                }
            }
            // Setup message slate.
            if self.message_error.is_none() {
                self.message_slate = Some(slate);
            }
            // Clear message loading result and status.
            {
                let mut w_res = self.receive_pay_result.write();
                *w_res = None;
            }
            self.message_loading = false;
        }
    }

    /// Parse message input into [`Slate`] updating slate and response input.
    pub fn parse_message(&mut self, wallet: &Wallet) {
        self.message_slate = None;
        self.message_error = None;
        if self.message_edit.is_empty() {
           return;
        }
        // Trim message.
        self.message_edit = self.message_edit.trim().to_string();

        // Parse message.
        if let Ok(mut slate) = wallet.parse_slatepack(&self.message_edit) {
            // Try to setup empty amount from transaction by id.
            if slate.amount == 0 {
                let _ = wallet.get_data().unwrap().txs.clone().iter().map(|tx| {
                    if tx.data.tx_slate_id == Some(slate.id) {
                        if slate.amount == 0 {
                            slate.amount = tx.amount;
                        }
                    }
                    tx
                }).collect::<Vec<&WalletTransaction>>();
            }

            if slate.amount == 0 {
                self.message_error = Some(
                    MessageError::Response(t!("wallets.resp_slatepack_err"))
                );
                return;
            }

            // Make operation based on incoming state status.
            match slate.state {
                SlateState::Standard1 | SlateState::Invoice1 => {
                    let slate = slate.clone();
                    let message = self.message_edit.clone();
                    let message_result = self.receive_pay_result.clone();
                    let wallet = wallet.clone();
                    // Create response to sender or receiver at separate thread.
                    self.message_loading = true;
                    thread::spawn(move || {
                        let resp = if slate.state == SlateState::Standard1 {
                            wallet.receive(&message)
                        } else {
                            wallet.pay(&message)
                        };
                        let mut w_res = message_result.write();
                        *w_res = Some((slate, resp));
                    });
                    return;
                }
                SlateState::Standard2 | SlateState::Invoice2 => {
                    // Check if slatepack with same id and state already exists.
                    let mut sl = slate.clone();
                    sl.state = if sl.state == SlateState::Standard2 {
                        SlateState::Standard1
                    } else {
                        SlateState::Invoice1
                    };
                    match wallet.read_slatepack(&sl) {
                        None => {
                            match wallet.read_slatepack(&slate) {
                                None => {
                                    self.message_error = Some(
                                        MessageError::Response(t!("wallets.resp_slatepack_err"))
                                    );
                                }
                                Some(sp) => {
                                    self.message_slate = Some(sl);
                                    self.response_edit = sp;
                                    return;
                                }
                            }
                        }
                        Some(_) => {
                            self.message_slate = Some(slate.clone());
                            return;
                        }
                    }
                }
                _ => {
                    self.response_edit = "".to_string();
                }
            }
            self.message_slate = Some(slate);
        } else {
            self.message_slate = None;
            self.message_error = Some(MessageError::Parse(t!("wallets.resp_slatepack_err")));
        }
    }
}