Compare commits
No commits in common. "61d9acc7c60336577727eb4f57c0f8c72c96fb9e" and "c98a0655870ed7e8810cc7c92849d3a6d91d6618" have entirely different histories.
61d9acc7c6
...
c98a065587
@ -25,7 +25,7 @@ tauri-build = { version = "1.0.4", features = [] }
|
|||||||
[dependencies]
|
[dependencies]
|
||||||
serde_json = "1.0"
|
serde_json = "1.0"
|
||||||
serde = { version = "1.0", features = ["derive"] }
|
serde = { version = "1.0", features = ["derive"] }
|
||||||
tauri = { version = "1.2", features = ["dialog", "dialog-open", "global-shortcut", "os-all", "system-tray"] }
|
tauri = { version = "1.2", features = ["dialog", "dialog-open", "os-all", "system-tray"] }
|
||||||
tauri-plugin-single-instance = { git = "https://github.com/tauri-apps/plugins-workspace", branch = "dev" }
|
tauri-plugin-single-instance = { git = "https://github.com/tauri-apps/plugins-workspace", branch = "dev" }
|
||||||
sodiumoxide = "0.2.7"
|
sodiumoxide = "0.2.7"
|
||||||
tokio = { version = ">=1.19", features = ["full"] }
|
tokio = { version = ">=1.19", features = ["full"] }
|
||||||
|
@ -83,7 +83,6 @@ async fn setup(app: &mut App) -> Result<(), Box<dyn Error>> {
|
|||||||
let srv = Server::new(conf.listen_addr, conf.listen_port, app.handle()).await?;
|
let srv = Server::new(conf.listen_addr, conf.listen_port, app.handle()).await?;
|
||||||
|
|
||||||
config::set_auto_launch(conf.start_on_login)?;
|
config::set_auto_launch(conf.start_on_login)?;
|
||||||
config::register_hotkeys(&conf.hotkeys)?;
|
|
||||||
// if session is empty, this is probably the first launch, so don't autohide
|
// if session is empty, this is probably the first launch, so don't autohide
|
||||||
if !conf.start_minimized || is_first_launch {
|
if !conf.start_minimized || is_first_launch {
|
||||||
app.get_window("main")
|
app.get_window("main")
|
||||||
|
@ -5,11 +5,6 @@ use auto_launch::AutoLaunchBuilder;
|
|||||||
use is_terminal::IsTerminal;
|
use is_terminal::IsTerminal;
|
||||||
use serde::{Serialize, Deserialize};
|
use serde::{Serialize, Deserialize};
|
||||||
use sqlx::SqlitePool;
|
use sqlx::SqlitePool;
|
||||||
use tauri::{
|
|
||||||
Manager,
|
|
||||||
GlobalShortcutManager,
|
|
||||||
async_runtime as rt,
|
|
||||||
};
|
|
||||||
|
|
||||||
use crate::errors::*;
|
use crate::errors::*;
|
||||||
|
|
||||||
@ -25,21 +20,6 @@ pub struct TermConfig {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[derive(Clone, Debug, Serialize, Deserialize, Eq, PartialEq)]
|
|
||||||
pub struct Hotkey {
|
|
||||||
pub keys: String,
|
|
||||||
pub enabled: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#[derive(Clone, Debug, Serialize, Deserialize, Eq, PartialEq)]
|
|
||||||
pub struct HotkeysConfig {
|
|
||||||
// tauri uses strings to represent keybinds, so we will as well
|
|
||||||
pub show_window: Hotkey,
|
|
||||||
pub launch_terminal: Hotkey,
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#[derive(Clone, Debug, Serialize, Deserialize)]
|
#[derive(Clone, Debug, Serialize, Deserialize)]
|
||||||
pub struct AppConfig {
|
pub struct AppConfig {
|
||||||
#[serde(default = "default_listen_addr")]
|
#[serde(default = "default_listen_addr")]
|
||||||
@ -54,8 +34,6 @@ pub struct AppConfig {
|
|||||||
pub start_on_login: bool,
|
pub start_on_login: bool,
|
||||||
#[serde(default = "default_term_config")]
|
#[serde(default = "default_term_config")]
|
||||||
pub terminal: TermConfig,
|
pub terminal: TermConfig,
|
||||||
#[serde(default = "default_hotkey_config")]
|
|
||||||
pub hotkeys: HotkeysConfig,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -68,7 +46,6 @@ impl Default for AppConfig {
|
|||||||
start_minimized: default_start_minimized(),
|
start_minimized: default_start_minimized(),
|
||||||
start_on_login: default_start_on_login(),
|
start_on_login: default_start_on_login(),
|
||||||
terminal: default_term_config(),
|
terminal: default_term_config(),
|
||||||
hotkeys: default_hotkey_config(),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -193,51 +170,6 @@ fn default_term_config() -> TermConfig {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
fn default_hotkey_config() -> HotkeysConfig {
|
|
||||||
HotkeysConfig {
|
|
||||||
show_window: Hotkey {keys: "alt+shift+C".into(), enabled: true},
|
|
||||||
launch_terminal: Hotkey {keys: "alt+shift+T".into(), enabled: true},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// note: will panic if called before APP is set
|
|
||||||
pub fn register_hotkeys(hotkeys: &HotkeysConfig) -> tauri::Result<()> {
|
|
||||||
let app = crate::app::APP.get().unwrap();
|
|
||||||
|
|
||||||
let mut manager = app.global_shortcut_manager();
|
|
||||||
manager.unregister_all()?;
|
|
||||||
|
|
||||||
if hotkeys.show_window.enabled {
|
|
||||||
let handle = app.app_handle();
|
|
||||||
manager.register(
|
|
||||||
&hotkeys.show_window.keys,
|
|
||||||
move || {
|
|
||||||
handle.get_window("main")
|
|
||||||
.map(|w| w.show().error_popup("Failed to show"))
|
|
||||||
.ok_or(HandlerError::NoMainWindow)
|
|
||||||
.error_popup("No main window");
|
|
||||||
},
|
|
||||||
)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
if hotkeys.launch_terminal.enabled {
|
|
||||||
// register() doesn't take an async fn, so we have to use spawn
|
|
||||||
manager.register(
|
|
||||||
&hotkeys.launch_terminal.keys,
|
|
||||||
|| {
|
|
||||||
rt::spawn(async {
|
|
||||||
crate::terminal::launch(false)
|
|
||||||
.await
|
|
||||||
.error_popup("Failed to launch");
|
|
||||||
});
|
|
||||||
}
|
|
||||||
)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
fn default_listen_addr() -> Ipv4Addr { Ipv4Addr::LOCALHOST }
|
fn default_listen_addr() -> Ipv4Addr { Ipv4Addr::LOCALHOST }
|
||||||
fn default_rehide_ms() -> u64 { 1000 }
|
fn default_rehide_ms() -> u64 { 1000 }
|
||||||
// start minimized and on login only in production mode
|
// start minimized and on login only in production mode
|
||||||
|
@ -95,8 +95,6 @@ pub enum SetupError {
|
|||||||
ServerSetupError(#[from] std::io::Error),
|
ServerSetupError(#[from] std::io::Error),
|
||||||
#[error("Failed to resolve data directory: {0}")]
|
#[error("Failed to resolve data directory: {0}")]
|
||||||
DataDir(#[from] DataDirError),
|
DataDir(#[from] DataDirError),
|
||||||
#[error("Failed to register hotkeys: {0}")]
|
|
||||||
RegisterHotkeys(#[from] tauri::Error),
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -244,19 +242,6 @@ pub enum ExecError {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[derive(Debug, ThisError, AsRefStr)]
|
|
||||||
pub enum LaunchTerminalError {
|
|
||||||
#[error("Could not discover main window")]
|
|
||||||
NoMainWindow,
|
|
||||||
#[error("Failed to communicate with main Creddy window")]
|
|
||||||
IpcFailed(#[from] tauri::Error),
|
|
||||||
#[error("Failed to launch terminal: {0}")]
|
|
||||||
Exec(#[from] ExecError),
|
|
||||||
#[error(transparent)]
|
|
||||||
GetCredentials(#[from] GetCredentialsError),
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// =========================
|
// =========================
|
||||||
// Serialize implementations
|
// Serialize implementations
|
||||||
// =========================
|
// =========================
|
||||||
@ -362,18 +347,3 @@ impl Serialize for ExecError {
|
|||||||
map.end()
|
map.end()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
impl Serialize for LaunchTerminalError {
|
|
||||||
fn serialize<S: Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
|
|
||||||
let mut map = serializer.serialize_map(None)?;
|
|
||||||
map.serialize_entry("code", self.as_ref())?;
|
|
||||||
map.serialize_entry("msg", &format!("{self}"))?;
|
|
||||||
|
|
||||||
match self {
|
|
||||||
LaunchTerminalError::Exec(src) => map.serialize_entry("source", &src)?,
|
|
||||||
_ => serialize_upstream_err(self, &mut map)?,
|
|
||||||
}
|
|
||||||
map.end()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -82,6 +82,6 @@ pub async fn save_config(config: AppConfig, app_state: State<'_, AppState>) -> R
|
|||||||
|
|
||||||
|
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
pub async fn launch_terminal(base: bool) -> Result<(), LaunchTerminalError> {
|
pub async fn launch_terminal(base: bool) -> Result<(), ExecError> {
|
||||||
terminal::launch(base).await
|
terminal::launch(base).await
|
||||||
}
|
}
|
||||||
|
@ -29,7 +29,6 @@ pub struct AppState {
|
|||||||
pub session: RwLock<Session>,
|
pub session: RwLock<Session>,
|
||||||
pub request_count: RwLock<u64>,
|
pub request_count: RwLock<u64>,
|
||||||
pub open_requests: RwLock<HashMap<u64, Sender<ipc::Approval>>>,
|
pub open_requests: RwLock<HashMap<u64, Sender<ipc::Approval>>>,
|
||||||
pub pending_terminal_request: RwLock<bool>,
|
|
||||||
pub bans: RwLock<std::collections::HashSet<Option<Client>>>,
|
pub bans: RwLock<std::collections::HashSet<Option<Client>>>,
|
||||||
server: RwLock<Server>,
|
server: RwLock<Server>,
|
||||||
pool: sqlx::SqlitePool,
|
pool: sqlx::SqlitePool,
|
||||||
@ -42,7 +41,6 @@ impl AppState {
|
|||||||
session: RwLock::new(session),
|
session: RwLock::new(session),
|
||||||
request_count: RwLock::new(0),
|
request_count: RwLock::new(0),
|
||||||
open_requests: RwLock::new(HashMap::new()),
|
open_requests: RwLock::new(HashMap::new()),
|
||||||
pending_terminal_request: RwLock::new(false),
|
|
||||||
bans: RwLock::new(HashSet::new()),
|
bans: RwLock::new(HashSet::new()),
|
||||||
server: RwLock::new(server),
|
server: RwLock::new(server),
|
||||||
pool,
|
pool,
|
||||||
@ -61,23 +59,15 @@ impl AppState {
|
|||||||
pub async fn update_config(&self, new_config: AppConfig) -> Result<(), SetupError> {
|
pub async fn update_config(&self, new_config: AppConfig) -> Result<(), SetupError> {
|
||||||
let mut live_config = self.config.write().await;
|
let mut live_config = self.config.write().await;
|
||||||
|
|
||||||
// update autostart if necessary
|
|
||||||
if new_config.start_on_login != live_config.start_on_login {
|
if new_config.start_on_login != live_config.start_on_login {
|
||||||
config::set_auto_launch(new_config.start_on_login)?;
|
config::set_auto_launch(new_config.start_on_login)?;
|
||||||
}
|
}
|
||||||
// rebind socket if necessary
|
|
||||||
if new_config.listen_addr != live_config.listen_addr
|
if new_config.listen_addr != live_config.listen_addr
|
||||||
|| new_config.listen_port != live_config.listen_port
|
|| new_config.listen_port != live_config.listen_port
|
||||||
{
|
{
|
||||||
let mut sv = self.server.write().await;
|
let mut sv = self.server.write().await;
|
||||||
sv.rebind(new_config.listen_addr, new_config.listen_port).await?;
|
sv.rebind(new_config.listen_addr, new_config.listen_port).await?;
|
||||||
}
|
}
|
||||||
// re-register hotkeys if necessary
|
|
||||||
if new_config.hotkeys.show_window != live_config.hotkeys.show_window
|
|
||||||
|| new_config.hotkeys.launch_terminal != live_config.hotkeys.launch_terminal
|
|
||||||
{
|
|
||||||
config::register_hotkeys(&new_config.hotkeys)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
new_config.save(&self.pool).await?;
|
new_config.save(&self.pool).await?;
|
||||||
*live_config = new_config;
|
*live_config = new_config;
|
||||||
@ -151,11 +141,6 @@ impl AppState {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn is_unlocked(&self) -> bool {
|
|
||||||
let session = self.session.read().await;
|
|
||||||
matches!(*session, Session::Unlocked{..})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn serialize_base_creds(&self) -> Result<String, GetCredentialsError> {
|
pub async fn serialize_base_creds(&self) -> Result<String, GetCredentialsError> {
|
||||||
let app_session = self.session.read().await;
|
let app_session = self.session.read().await;
|
||||||
let (base, _session) = app_session.try_get()?;
|
let (base, _session) = app_session.try_get()?;
|
||||||
@ -174,21 +159,4 @@ impl AppState {
|
|||||||
*app_session = Session::Unlocked {base, session};
|
*app_session = Session::Unlocked {base, session};
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn register_terminal_request(&self) -> Result<(), ()> {
|
|
||||||
let mut req = self.pending_terminal_request.write().await;
|
|
||||||
if *req {
|
|
||||||
// if a request is already pending, we can't register a new one
|
|
||||||
Err(())
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
*req = true;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn unregister_terminal_request(&self) {
|
|
||||||
let mut req = self.pending_terminal_request.write().await;
|
|
||||||
*req = false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -7,15 +7,9 @@ use crate::errors::*;
|
|||||||
use crate::state::AppState;
|
use crate::state::AppState;
|
||||||
|
|
||||||
|
|
||||||
pub async fn launch(use_base: bool) -> Result<(), LaunchTerminalError> {
|
pub async fn launch(use_base: bool) -> Result<(), ExecError> {
|
||||||
let app = APP.get().unwrap();
|
let state = APP.get().unwrap().state::<AppState>();
|
||||||
let state = app.state::<AppState>();
|
// do all this in a block so we don't hold the lock any longer than necessary
|
||||||
|
|
||||||
// register_terminal_request() returns Err if there is another request pending
|
|
||||||
if state.register_terminal_request().await.is_err() {
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut cmd = {
|
let mut cmd = {
|
||||||
let config = state.config.read().await;
|
let config = state.config.read().await;
|
||||||
let mut cmd = Command::new(&config.terminal.exec);
|
let mut cmd = Command::new(&config.terminal.exec);
|
||||||
@ -23,38 +17,10 @@ pub async fn launch(use_base: bool) -> Result<(), LaunchTerminalError> {
|
|||||||
cmd
|
cmd
|
||||||
};
|
};
|
||||||
|
|
||||||
// if session is unlocked or empty, wait for credentials from frontend
|
// similarly
|
||||||
if !state.is_unlocked().await {
|
|
||||||
app.emit_all("launch-terminal-request", ())?;
|
|
||||||
let window = app.get_window("main")
|
|
||||||
.ok_or(LaunchTerminalError::NoMainWindow)?;
|
|
||||||
if !window.is_visible()? {
|
|
||||||
window.unminimize()?;
|
|
||||||
window.show()?;
|
|
||||||
}
|
|
||||||
window.set_focus()?;
|
|
||||||
|
|
||||||
let (tx, rx) = tokio::sync::oneshot::channel();
|
|
||||||
app.once_global("credentials-event", move |e| {
|
|
||||||
let success = match e.payload() {
|
|
||||||
Some("\"unlocked\"") | Some("\"entered\"") => true,
|
|
||||||
_ => false,
|
|
||||||
};
|
|
||||||
let _ = tx.send(success);
|
|
||||||
});
|
|
||||||
|
|
||||||
if !rx.await.unwrap_or(false) {
|
|
||||||
state.unregister_terminal_request().await;
|
|
||||||
return Ok(()); // request was canceled by user
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// more lock-management
|
|
||||||
{
|
{
|
||||||
|
let state = APP.get().unwrap().state::<AppState>();
|
||||||
let app_session = state.session.read().await;
|
let app_session = state.session.read().await;
|
||||||
// session should really be unlocked at this point, but if the frontend misbehaves
|
|
||||||
// (i.e. lies about unlocking) we could end up here with a locked session
|
|
||||||
// this will result in an error popup to the user (see main hotkey handler)
|
|
||||||
let (base_creds, session_creds) = app_session.try_get()?;
|
let (base_creds, session_creds) = app_session.try_get()?;
|
||||||
if use_base {
|
if use_base {
|
||||||
cmd.env("AWS_ACCESS_KEY_ID", &base_creds.access_key_id);
|
cmd.env("AWS_ACCESS_KEY_ID", &base_creds.access_key_id);
|
||||||
@ -67,16 +33,11 @@ pub async fn launch(use_base: bool) -> Result<(), LaunchTerminalError> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let res = match cmd.spawn() {
|
match cmd.spawn() {
|
||||||
Ok(_) => Ok(()),
|
Ok(_) => Ok(()),
|
||||||
Err(e) if std::io::ErrorKind::NotFound == e.kind() => {
|
Err(e) if std::io::ErrorKind::NotFound == e.kind() => {
|
||||||
Err(ExecError::NotFound(cmd.get_program().to_owned()))
|
Err(ExecError::NotFound(cmd.get_program().to_owned()))
|
||||||
},
|
},
|
||||||
Err(e) => Err(ExecError::ExecutionFailed(e)),
|
Err(e) => Err(e.into()),
|
||||||
};
|
}
|
||||||
|
|
||||||
state.unregister_terminal_request().await;
|
|
||||||
|
|
||||||
res?; // ? auto-conversion is more liberal than .into()
|
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
|
@ -16,20 +16,6 @@ listen('credentials-request', (tauriEvent) => {
|
|||||||
$appState.pendingRequests.put(tauriEvent.payload);
|
$appState.pendingRequests.put(tauriEvent.payload);
|
||||||
});
|
});
|
||||||
|
|
||||||
listen('launch-terminal-request', async (tauriEvent) => {
|
|
||||||
if ($appState.currentRequest === null) {
|
|
||||||
let status = await invoke('get_session_status');
|
|
||||||
if (status === 'locked') {
|
|
||||||
navigate('Unlock');
|
|
||||||
}
|
|
||||||
else if (status === 'empty') {
|
|
||||||
navigate('EnterCredentials');
|
|
||||||
}
|
|
||||||
// else, session is unlocked, so do nothing
|
|
||||||
// (although we shouldn't even get the event in that case)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
acceptRequest();
|
acceptRequest();
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
@ -9,10 +9,6 @@ export default function() {
|
|||||||
|
|
||||||
resolvers: [],
|
resolvers: [],
|
||||||
|
|
||||||
size() {
|
|
||||||
return this.items.length;
|
|
||||||
},
|
|
||||||
|
|
||||||
put(item) {
|
put(item) {
|
||||||
this.items.push(item);
|
this.items.push(item);
|
||||||
let resolver = this.resolvers.shift();
|
let resolver = this.resolvers.shift();
|
||||||
|
@ -1,13 +0,0 @@
|
|||||||
<script>
|
|
||||||
export let keys;
|
|
||||||
</script>
|
|
||||||
|
|
||||||
|
|
||||||
<div class="flex gap-x-[0.2em] items-center">
|
|
||||||
{#each keys as key, i}
|
|
||||||
{#if i > 0}
|
|
||||||
<span class="mt-[-0.1em]">+</span>
|
|
||||||
{/if}
|
|
||||||
<kbd class="normal-case px-1 py-0.5 rounded border border-neutral">{key}</kbd>
|
|
||||||
{/each}
|
|
||||||
</div>
|
|
@ -4,13 +4,14 @@
|
|||||||
import Setting from './Setting.svelte';
|
import Setting from './Setting.svelte';
|
||||||
|
|
||||||
export let title;
|
export let title;
|
||||||
|
export let divider = true;
|
||||||
export let value;
|
export let value;
|
||||||
|
|
||||||
const dispatch = createEventDispatcher();
|
const dispatch = createEventDispatcher();
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|
||||||
<Setting {title}>
|
<Setting {title} {divider}>
|
||||||
<div slot="input">
|
<div slot="input">
|
||||||
<input
|
<input
|
||||||
type="text"
|
type="text"
|
||||||
|
@ -1,61 +0,0 @@
|
|||||||
<script>
|
|
||||||
import { createEventDispatcher } from 'svelte';
|
|
||||||
import KeyCombo from '../KeyCombo.svelte';
|
|
||||||
|
|
||||||
export let description;
|
|
||||||
export let value;
|
|
||||||
|
|
||||||
const id = Math.random().toString().slice(2);
|
|
||||||
const dispatch = createEventDispatcher();
|
|
||||||
let listening = false;
|
|
||||||
|
|
||||||
function listen() {
|
|
||||||
// don't re-listen if we already are
|
|
||||||
if (listening) return;
|
|
||||||
|
|
||||||
listening = true;
|
|
||||||
window.addEventListener('keyup', setKeybind, {once: true});
|
|
||||||
// setTimeout avoids reacting to the click event that we are currently processing
|
|
||||||
setTimeout(() => window.addEventListener('click', cancel, {once: true}), 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
function setKeybind(event) {
|
|
||||||
console.log(event);
|
|
||||||
let keys = [];
|
|
||||||
if (event.ctrlKey) keys.push('ctrl');
|
|
||||||
if (event.altKey) keys.push('alt');
|
|
||||||
if (event.metaKey) keys.push('meta');
|
|
||||||
if (event.shiftKey) keys.push('shift');
|
|
||||||
keys.push(event.key);
|
|
||||||
|
|
||||||
value.keys = keys.join('+');
|
|
||||||
dispatch('update', {value});
|
|
||||||
listening = false;
|
|
||||||
window.removeEventListener('click', cancel, {once: true});
|
|
||||||
event.preventDefault();
|
|
||||||
event.stopPropagation();
|
|
||||||
}
|
|
||||||
|
|
||||||
function cancel() {
|
|
||||||
listening = false;
|
|
||||||
window.removeEventListener('keyup', setKeybind, {once: true});
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
|
|
||||||
<input
|
|
||||||
{id}
|
|
||||||
type="checkbox"
|
|
||||||
class="checkbox checkbox-primary"
|
|
||||||
bind:checked={value.enabled}
|
|
||||||
on:change={() => dispatch('update', {value})}
|
|
||||||
>
|
|
||||||
<label for={id} class="cursor-pointer ml-4 text-lg">{description}</label>
|
|
||||||
|
|
||||||
<button class="h-12 p-2 rounded border border-neutral cursor-pointer text-center" on:click={listen}>
|
|
||||||
{#if listening}
|
|
||||||
Click to cancel
|
|
||||||
{:else}
|
|
||||||
<KeyCombo keys={value.keys.split('+')} />
|
|
||||||
{/if}
|
|
||||||
</button>
|
|
@ -4,8 +4,8 @@
|
|||||||
import Setting from './Setting.svelte';
|
import Setting from './Setting.svelte';
|
||||||
|
|
||||||
export let title;
|
export let title;
|
||||||
|
export let divider = true;
|
||||||
export let value;
|
export let value;
|
||||||
|
|
||||||
export let unit = '';
|
export let unit = '';
|
||||||
export let min = null;
|
export let min = null;
|
||||||
export let max = null;
|
export let max = null;
|
||||||
@ -60,7 +60,7 @@
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|
||||||
<Setting {title}>
|
<Setting {title} {divider}>
|
||||||
<div slot="input">
|
<div slot="input">
|
||||||
{#if unit}
|
{#if unit}
|
||||||
<span class="mr-2">{unit}:</span>
|
<span class="mr-2">{unit}:</span>
|
||||||
|
@ -3,15 +3,16 @@
|
|||||||
import ErrorAlert from '../ErrorAlert.svelte';
|
import ErrorAlert from '../ErrorAlert.svelte';
|
||||||
|
|
||||||
export let title;
|
export let title;
|
||||||
|
export let divider = true;
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|
||||||
<div>
|
{#if divider}
|
||||||
|
<div class="divider"></div>
|
||||||
|
{/if}
|
||||||
<div class="flex flex-wrap justify-between gap-y-4">
|
<div class="flex flex-wrap justify-between gap-y-4">
|
||||||
<h3 class="text-lg font-bold shrink-0">{title}</h3>
|
<h3 class="text-lg font-bold shrink-0">{title}</h3>
|
||||||
{#if $$slots.input}
|
|
||||||
<slot name="input"></slot>
|
<slot name="input"></slot>
|
||||||
{/if}
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{#if $$slots.description}
|
{#if $$slots.description}
|
||||||
@ -19,4 +20,3 @@
|
|||||||
<slot name="description"></slot>
|
<slot name="description"></slot>
|
||||||
</p>
|
</p>
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
|
||||||
|
@ -1,14 +0,0 @@
|
|||||||
<script>
|
|
||||||
export let name;
|
|
||||||
</script>
|
|
||||||
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<div class="divider mt-0 mb-8">
|
|
||||||
<h2 class="text-xl font-bold">{name}</h2>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="space-y-12">
|
|
||||||
<slot></slot>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
@ -3,13 +3,14 @@
|
|||||||
import Setting from './Setting.svelte';
|
import Setting from './Setting.svelte';
|
||||||
|
|
||||||
export let title;
|
export let title;
|
||||||
|
export let divider = true;
|
||||||
export let value;
|
export let value;
|
||||||
|
|
||||||
const dispatch = createEventDispatcher();
|
const dispatch = createEventDispatcher();
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|
||||||
<Setting {title}>
|
<Setting {title} {divider}>
|
||||||
<div slot="input">
|
<div slot="input">
|
||||||
<input
|
<input
|
||||||
type="text"
|
type="text"
|
||||||
|
@ -4,13 +4,14 @@
|
|||||||
import Setting from './Setting.svelte';
|
import Setting from './Setting.svelte';
|
||||||
|
|
||||||
export let title;
|
export let title;
|
||||||
|
export let divider = true; // passed through to Setting
|
||||||
export let value;
|
export let value;
|
||||||
|
|
||||||
const dispatch = createEventDispatcher();
|
const dispatch = createEventDispatcher();
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|
||||||
<Setting {title}>
|
<Setting {title} {divider}>
|
||||||
<input
|
<input
|
||||||
slot="input"
|
slot="input"
|
||||||
type="checkbox"
|
type="checkbox"
|
||||||
|
@ -6,7 +6,6 @@
|
|||||||
import { appState, completeRequest } from '../lib/state.js';
|
import { appState, completeRequest } from '../lib/state.js';
|
||||||
import ErrorAlert from '../ui/ErrorAlert.svelte';
|
import ErrorAlert from '../ui/ErrorAlert.svelte';
|
||||||
import Link from '../ui/Link.svelte';
|
import Link from '../ui/Link.svelte';
|
||||||
import KeyCombo from '../ui/KeyCombo.svelte';
|
|
||||||
|
|
||||||
|
|
||||||
// Send response to backend, display error if applicable
|
// Send response to backend, display error if applicable
|
||||||
@ -109,15 +108,17 @@
|
|||||||
<div class="w-full flex justify-between">
|
<div class="w-full flex justify-between">
|
||||||
<Link target={deny} hotkey="Escape">
|
<Link target={deny} hotkey="Escape">
|
||||||
<button class="btn btn-error justify-self-start">
|
<button class="btn btn-error justify-self-start">
|
||||||
<span class="mr-2">Deny</span>
|
Deny
|
||||||
<KeyCombo keys={['Esc']} />
|
<kbd class="ml-2 normal-case px-1 py-0.5 rounded border border-neutral">Esc</kbd>
|
||||||
</button>
|
</button>
|
||||||
</Link>
|
</Link>
|
||||||
|
|
||||||
<Link target={approve} hotkey="Enter" shift="{true}">
|
<Link target={approve} hotkey="Enter" shift="{true}">
|
||||||
<button class="btn btn-success justify-self-end">
|
<button class="btn btn-success justify-self-end">
|
||||||
<span class="mr-2">Approve</span>
|
Approve
|
||||||
<KeyCombo keys={['Shift', 'Enter']} />
|
<kbd class="ml-2 normal-case px-1 py-0.5 rounded border border-neutral">Shift</kbd>
|
||||||
|
<span class="mx-0.5">+</span>
|
||||||
|
<kbd class="normal-case px-1 py-0.5 rounded border border-neutral">Enter</kbd>
|
||||||
</button>
|
</button>
|
||||||
</Link>
|
</Link>
|
||||||
</div>
|
</div>
|
||||||
|
@ -31,7 +31,6 @@
|
|||||||
try {
|
try {
|
||||||
saving = true;
|
saving = true;
|
||||||
await invoke('save_credentials', {credentials, passphrase});
|
await invoke('save_credentials', {credentials, passphrase});
|
||||||
emit('credentials-event', 'entered');
|
|
||||||
if ($appState.currentRequest) {
|
if ($appState.currentRequest) {
|
||||||
navigate('Approve');
|
navigate('Approve');
|
||||||
}
|
}
|
||||||
@ -57,11 +56,6 @@
|
|||||||
saving = false;
|
saving = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function cancel() {
|
|
||||||
emit('credentials-event', 'enter-canceled');
|
|
||||||
navigate('Home');
|
|
||||||
}
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|
||||||
@ -85,7 +79,7 @@
|
|||||||
Submit
|
Submit
|
||||||
{/if}
|
{/if}
|
||||||
</button>
|
</button>
|
||||||
<Link target={cancel} hotkey="Escape">
|
<Link target="Home" hotkey="Escape">
|
||||||
<button class="btn btn-sm btn-outline w-full">Cancel</button>
|
<button class="btn btn-sm btn-outline w-full">Cancel</button>
|
||||||
</Link>
|
</Link>
|
||||||
</form>
|
</form>
|
||||||
|
@ -1,18 +1,11 @@
|
|||||||
<script context="module">
|
|
||||||
import { type } from '@tauri-apps/api/os';
|
|
||||||
const osType = await type();
|
|
||||||
</script>
|
|
||||||
|
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { invoke } from '@tauri-apps/api/tauri';
|
import { invoke } from '@tauri-apps/api/tauri';
|
||||||
|
import { type } from '@tauri-apps/api/os';
|
||||||
|
|
||||||
import { appState } from '../lib/state.js';
|
import { appState } from '../lib/state.js';
|
||||||
import Nav from '../ui/Nav.svelte';
|
import Nav from '../ui/Nav.svelte';
|
||||||
import Link from '../ui/Link.svelte';
|
import Link from '../ui/Link.svelte';
|
||||||
import ErrorAlert from '../ui/ErrorAlert.svelte';
|
import ErrorAlert from '../ui/ErrorAlert.svelte';
|
||||||
import SettingsGroup from '../ui/settings/SettingsGroup.svelte';
|
|
||||||
import Keybind from '../ui/settings/Keybind.svelte';
|
|
||||||
import { Setting, ToggleSetting, NumericSetting, FileSetting, TextSetting } from '../ui/settings';
|
import { Setting, ToggleSetting, NumericSetting, FileSetting, TextSetting } from '../ui/settings';
|
||||||
|
|
||||||
import { fly } from 'svelte/transition';
|
import { fly } from 'svelte/transition';
|
||||||
@ -21,7 +14,6 @@
|
|||||||
|
|
||||||
let error = null;
|
let error = null;
|
||||||
async function save() {
|
async function save() {
|
||||||
console.log('updating config');
|
|
||||||
try {
|
try {
|
||||||
await invoke('save_config', {config: $appState.config});
|
await invoke('save_config', {config: $appState.config});
|
||||||
}
|
}
|
||||||
@ -30,6 +22,12 @@
|
|||||||
$appState.config = await invoke('get_config');
|
$appState.config = await invoke('get_config');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let osType = '';
|
||||||
|
type().then(t => osType = t);
|
||||||
|
|
||||||
|
console.log($appState.config.terminal);
|
||||||
|
window.term = $appState.config.terminal;
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|
||||||
@ -38,9 +36,14 @@
|
|||||||
</Nav>
|
</Nav>
|
||||||
|
|
||||||
{#await invoke('get_config') then config}
|
{#await invoke('get_config') then config}
|
||||||
<div class="max-w-lg mx-auto mt-1.5 p-4 space-y-16">
|
<div class="max-w-lg mx-auto mt-1.5 p-4">
|
||||||
<SettingsGroup name="General">
|
<!-- <h2 class="text-2xl font-bold text-center">Settings</h2> -->
|
||||||
<ToggleSetting title="Start on login" bind:value={$appState.config.start_on_login} on:update={save}>
|
|
||||||
|
<div class="divider mt-0 mb-8">
|
||||||
|
<h2 class="text-xl font-bold">General</h2>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<ToggleSetting title="Start on login" divider={false} bind:value={$appState.config.start_on_login} on:update={save}>
|
||||||
<svelte:fragment slot="description">
|
<svelte:fragment slot="description">
|
||||||
Start Creddy when you log in to your computer.
|
Start Creddy when you log in to your computer.
|
||||||
</svelte:fragment>
|
</svelte:fragment>
|
||||||
@ -81,28 +84,20 @@
|
|||||||
</svelte:fragment>
|
</svelte:fragment>
|
||||||
</Setting>
|
</Setting>
|
||||||
|
|
||||||
|
<div class="divider mt-10 mb-8">
|
||||||
|
<h2 class="text-xl font-bold">Terminal</h2>
|
||||||
|
</div>
|
||||||
|
|
||||||
<FileSetting
|
<FileSetting
|
||||||
title="Terminal emulator"
|
title="Emulator"
|
||||||
|
divider={false}
|
||||||
bind:value={$appState.config.terminal.exec}
|
bind:value={$appState.config.terminal.exec}
|
||||||
on:update={save}
|
on:update={save}
|
||||||
>
|
>
|
||||||
<svelte:fragment slot="description">
|
<svelte:fragment slot="description">
|
||||||
Choose your preferred terminal emulator (e.g. <code>gnome-terminal</code> or <code>wt.exe</code>.) May be an absolute path or an executable discoverable on <code>$PATH</code>.
|
Choose your preferred terminal emulator (e.g. <code>gnome-terminal</code>, <code>wt.exe</code>.) May be an absolute path or an executable discoverable on <code>$PATH</code>.
|
||||||
</svelte:fragment>
|
</svelte:fragment>
|
||||||
</FileSetting>
|
</FileSetting>
|
||||||
</SettingsGroup>
|
|
||||||
|
|
||||||
<SettingsGroup name="Hotkeys">
|
|
||||||
<div class="space-y-4">
|
|
||||||
<p>Click on a keybinding to modify it. Use the checkbox to enable or disable a keybinding entirely.</p>
|
|
||||||
|
|
||||||
<div class="grid grid-cols-[auto_1fr_auto] gap-y-3 items-center">
|
|
||||||
<Keybind description="Show Creddy" value={$appState.config.hotkeys.show_window} on:update={save} />
|
|
||||||
<Keybind description="Launch terminal" value={$appState.config.hotkeys.launch_terminal} on:update={save} />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</SettingsGroup>
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
{/await}
|
{/await}
|
||||||
|
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
<script>
|
<script>
|
||||||
import { invoke } from '@tauri-apps/api/tauri';
|
import { invoke } from '@tauri-apps/api/tauri';
|
||||||
import { emit } from '@tauri-apps/api/event';
|
|
||||||
import { onMount } from 'svelte';
|
import { onMount } from 'svelte';
|
||||||
|
|
||||||
import { appState } from '../lib/state.js';
|
import { appState } from '../lib/state.js';
|
||||||
@ -27,7 +26,6 @@
|
|||||||
saving = true;
|
saving = true;
|
||||||
let r = await invoke('unlock', {passphrase});
|
let r = await invoke('unlock', {passphrase});
|
||||||
$appState.credentialStatus = 'unlocked';
|
$appState.credentialStatus = 'unlocked';
|
||||||
emit('credentials-event', 'unlocked');
|
|
||||||
if ($appState.currentRequest) {
|
if ($appState.currentRequest) {
|
||||||
navigate('Approve');
|
navigate('Approve');
|
||||||
}
|
}
|
||||||
@ -53,11 +51,6 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function cancel() {
|
|
||||||
emit('credentials-event', 'unlock-canceled');
|
|
||||||
navigate('Home');
|
|
||||||
}
|
|
||||||
|
|
||||||
onMount(() => {
|
onMount(() => {
|
||||||
loadTime = Date.now();
|
loadTime = Date.now();
|
||||||
})
|
})
|
||||||
@ -82,7 +75,7 @@
|
|||||||
{/if}
|
{/if}
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<Link target={cancel} hotkey="Escape">
|
<Link target="Home" hotkey="Escape">
|
||||||
<button class="btn btn-sm btn-outline w-full">Cancel</button>
|
<button class="btn btn-outline btn-sm w-full">Cancel</button>
|
||||||
</Link>
|
</Link>
|
||||||
</form>
|
</form>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user