# 1.系統函數 由系統提供,直接拿來用或是導入模塊後使用 ``` a = 1.12386 result = round(a,2) print(result) > 1.12 ``` # 2.自定義函數 * 函數是結構化編程的核心 * 使用關鍵詞`def`來定義函數 ``` #函數定義 def fun ...
倉庫連接: https://github.com/GaN601/egui-demo-download-util
這是我第一個rust gui demo, 學習rust有挺長時間了, 但是一直沒有落實到實踐中, 本著對桌面應用的興趣, 考察了slint、egui兩種框架, 最後還是選擇了egui.
這篇博客同時包含我當前的一些理解, 但是自身技術有限, 可能有不少錯誤的地方. 有意者請在評論區指正.
這個demo的效果就是通過主視窗的按鈕, 呼出子視窗的輸入框, 點擊下載按鈕後就可以下載文件, 因為只是demo, 下載功能不詳細, 只是用reqwest請求下載了而已.
egui要求我們創建一個自己的結構體來進行狀態保存, 因此我們需要以下結構體:
點擊查看結構體
#[derive(Default)]
struct MainWindow {
window_download_url: DownloadUrl,
}
#[derive(Default, Clone)]
pub struct DownloadUrl {
pub is_show: bool,
pub is_start: bool,
pub url: String,
pub local_path: String,
}
impl DownloadUrl {
pub fn show_window(&mut self, ctx: &Context) {
let _ = Window::new("Download Url")
.open(&mut self.is_show.clone())
.show(ctx, |ui| {
// 這裡在為下載視窗添加一些ui元素
ui.heading("Download Url");
ui.text_edit_singleline(&mut self.url);
ui.text_edit_singleline(&mut self.local_path);
// 這裡將用戶輸入的數據保存在MainWindow中, 這樣當我們點擊下載按鈕時就會開始下載文件
if ui.button("Select Folder").clicked() {
if let Some(path) = rfd::FileDialog::new().pick_folder() {
self.local_path = path.display().to_string();
}
}
// 關閉當前視窗. 因為是即時模式, 因此下一幀這個視窗不會出現. 我們只需要修改布爾值即可.
if ui.button("Download").clicked() {
self.is_show = false;
self.is_start = true;
}
});
}
}
點擊查看項目依賴
egui = "0.19.0"
eframe = "0.19.0"
reqwest = "0.11.12"
tokio = { version = "1.21.2" , features=["full"]}
rfd = "0.10.0"
點擊查看項目代碼
use eframe::{run_native, App, Frame, NativeOptions};
use egui::{CentralPanel, Context};
use std::fs::File;
use std::io::Write;
use std::path::Path;
//首先使用tokio的main方法
#[tokio::main]
async fn main() {
println!("Hello, world!");
let option = NativeOptions {
// 定義視窗大小
initial_window_size: Some(egui::vec2(640.0, 480.0)),
..Default::default()
};
// 啟動egui的主視窗, MainWindow就是我們保持狀態的結構體
run_native(
"egui download util",
option,
Box::new(|_c| Box::<MainWindow>::default()),
);
}
// 實現App Trait, 因為egui是即時模式, 因此狀態數據只能從self(MainWindow)拿
impl App for MainWindow {
fn update(&mut self, ctx: &Context, frame: &mut Frame) {
// 這裡是創建了一個面板, 並且面板里有一個下載的按鈕, 當點擊按鈕後, 會展示一個子視窗
CentralPanel::default().show(ctx, |ui| {
if ui.button("Download").clicked() {
self.window_download_url.is_show = true;
}
});
if self.window_download_url.is_show {
self.window_download_url.show_window(ctx);
}
// 在這裡開始執行下載文件的邏輯, 因為所有權問題, 因此我直接clone了這個結構
let url = &mut self.window_download_url;
let target = url.clone();
if !(target.url.is_empty() || target.local_path.is_empty()) && url.is_start {
url.is_start = false;
tokio::spawn(async move {
// 執行下載文件的邏輯, 失敗的處理感覺沒啥必要, 其實可以考慮出個dialog
download_file_to_local_path(&target)
.await
.expect("TODO: panic message");
});
}
}
}
async fn download_file_to_local_path(
target: &DownloadUrl,
) -> Result<(), Box<dyn std::error::Error>> {
// 獲取文件夾路徑選擇器的路徑, 因為不打算太精細, 就直接生成了當前時間戳的文件名, 連文件尾碼都不給.
let file_path = Path::new(&target.local_path).join(
SystemTime::now()
.duration_since(UNIX_EPOCH)
.unwrap()
.as_millis()
.to_string(),
);
let mut file = File::create(file_path)?;
let response = reqwest::get(&target.url).await?;
// 寫入文件, 下載文件邏輯完成
file.write_all(&response.bytes().await?)?;
Ok(())
}