Compare commits
No commits in common. "47752720b4404d7618176d57954794469bc91bd2" and "05a9669fec107a3668c79e67ab88c8a43672e7f0" have entirely different histories.
47752720b4
...
05a9669fec
@ -26,5 +26,5 @@ pub struct Opt {
|
|||||||
fn main() {
|
fn main() {
|
||||||
//parse parameter
|
//parse parameter
|
||||||
let opt = Opt::from_args();
|
let opt = Opt::from_args();
|
||||||
ui::create_ui(&opt);
|
ui::Ui::run(&opt);
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,8 @@
|
|||||||
|
use std::sync::mpsc;
|
||||||
use std::{io, thread};
|
use std::{io, thread};
|
||||||
|
|
||||||
use crate::Opt;
|
|
||||||
use termion::event::Key;
|
use termion::event::Key;
|
||||||
|
use termion::input::TermRead;
|
||||||
use termion::raw::IntoRawMode;
|
use termion::raw::IntoRawMode;
|
||||||
use tui::backend::TermionBackend;
|
use tui::backend::TermionBackend;
|
||||||
use tui::layout::{Constraint, Direction, Layout};
|
use tui::layout::{Constraint, Direction, Layout};
|
||||||
@ -11,6 +12,7 @@ use crate::widget::info;
|
|||||||
use crate::widget::repo_entry;
|
use crate::widget::repo_entry;
|
||||||
use crate::widget::service_switcher;
|
use crate::widget::service_switcher;
|
||||||
use crate::widget::tag_list;
|
use crate::widget::tag_list;
|
||||||
|
use crate::Opt;
|
||||||
|
|
||||||
pub struct Ui {
|
pub struct Ui {
|
||||||
state: State,
|
state: State,
|
||||||
@ -54,7 +56,7 @@ impl Ui {
|
|||||||
state: State::SelectService,
|
state: State::SelectService,
|
||||||
repo: repo_entry::RepoEntry::new(repo_id),
|
repo: repo_entry::RepoEntry::new(repo_id),
|
||||||
tags: tag_list::TagList::with_status("Tags are empty"),
|
tags: tag_list::TagList::with_status("Tags are empty"),
|
||||||
services: service_switcher::ServiceSwitcher::new(&opt.config).unwrap(),
|
services: service_switcher::ServiceSwitcher::new(&opt.config),
|
||||||
info: info::Info::new("Select image of edit Repository"),
|
info: info::Info::new("Select image of edit Repository"),
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -68,7 +70,7 @@ impl Ui {
|
|||||||
let mut terminal = Terminal::new(backend).unwrap();
|
let mut terminal = Terminal::new(backend).unwrap();
|
||||||
|
|
||||||
//setup input thread
|
//setup input thread
|
||||||
let receiver = super::spawn_stdin_channel();
|
let receiver = ui.spawn_stdin_channel();
|
||||||
|
|
||||||
//core interaction loop
|
//core interaction loop
|
||||||
'core: loop {
|
'core: loop {
|
||||||
@ -88,10 +90,10 @@ impl Ui {
|
|||||||
)
|
)
|
||||||
.split(rect.size());
|
.split(rect.size());
|
||||||
|
|
||||||
let (list, state) = ui.services.render(ui.state == State::SelectService);
|
let (list, state) = ui.services.render(&ui.state);
|
||||||
rect.render_stateful_widget(list, chunks[0], state);
|
rect.render_stateful_widget(list, chunks[0], state);
|
||||||
rect.render_widget(ui.repo.render(ui.state == State::EditRepo), chunks[1]);
|
rect.render_widget(ui.repo.render(&ui.state), chunks[1]);
|
||||||
let (list, state) = ui.tags.render(ui.state == State::SelectTag);
|
let (list, state) = ui.tags.render(&ui.state);
|
||||||
rect.render_stateful_widget(list, chunks[2], state);
|
rect.render_stateful_widget(list, chunks[2], state);
|
||||||
rect.render_widget(ui.info.render(), chunks[3]);
|
rect.render_widget(ui.info.render(), chunks[3]);
|
||||||
})
|
})
|
||||||
@ -143,24 +145,22 @@ impl Ui {
|
|||||||
}
|
}
|
||||||
_ => (),
|
_ => (),
|
||||||
},
|
},
|
||||||
Ok(Key::Char(key)) => match ui.state {
|
Ok(Key::Char(key)) => {
|
||||||
State::SelectService => (),
|
if ui.state == State::EditRepo {
|
||||||
State::EditRepo => {
|
|
||||||
ui.info.set_info("Editing Repository");
|
ui.info.set_info("Editing Repository");
|
||||||
ui.repo.handle_input(Key::Char(key));
|
|
||||||
}
|
}
|
||||||
State::SelectTag => (),
|
ui.repo.handle_input(&ui.state, Key::Char(key));
|
||||||
},
|
ui.tags.handle_input(&ui.state, Key::Char(key));
|
||||||
Ok(Key::Backspace) => match ui.state {
|
}
|
||||||
State::SelectService => (),
|
Ok(Key::Backspace) => {
|
||||||
State::EditRepo => {
|
if ui.state == State::EditRepo {
|
||||||
ui.info.set_info("Editing Repository");
|
ui.info.set_info("Editing Repository");
|
||||||
ui.repo.handle_input(Key::Backspace);
|
|
||||||
}
|
}
|
||||||
State::SelectTag => (),
|
ui.repo.handle_input(&ui.state, Key::Backspace);
|
||||||
},
|
ui.tags.handle_input(&ui.state, Key::Backspace);
|
||||||
Ok(Key::Up) => match ui.state {
|
}
|
||||||
State::SelectService if ui.services.find_previous_match() => {
|
Ok(Key::Up) => {
|
||||||
|
if ui.state == State::SelectService && ui.services.find_previous_match() {
|
||||||
match ui.services.extract_repo() {
|
match ui.services.extract_repo() {
|
||||||
Err(e) => ui.info.set_info(&format!("{}", e)),
|
Err(e) => ui.info.set_info(&format!("{}", e)),
|
||||||
Ok(s) => {
|
Ok(s) => {
|
||||||
@ -176,10 +176,9 @@ impl Ui {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
State::SelectService => (),
|
ui.tags.handle_input(&ui.state, Key::Up);
|
||||||
State::EditRepo => (),
|
ui.repo.handle_input(&ui.state, Key::Up);
|
||||||
State::SelectTag => ui.tags.handle_input(Key::Up),
|
}
|
||||||
},
|
|
||||||
Ok(Key::Down) => match ui.state {
|
Ok(Key::Down) => match ui.state {
|
||||||
State::SelectService if ui.services.find_next_match() => {
|
State::SelectService if ui.services.find_next_match() => {
|
||||||
match ui.services.extract_repo() {
|
match ui.services.extract_repo() {
|
||||||
@ -197,10 +196,15 @@ impl Ui {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
State::SelectService => (),
|
_ => {
|
||||||
State::EditRepo => (),
|
ui.tags.handle_input(&ui.state, Key::Down);
|
||||||
State::SelectTag => ui.tags.handle_input(Key::Down),
|
ui.repo.handle_input(&ui.state, Key::Down);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
Ok(key) => {
|
||||||
|
ui.repo.handle_input(&ui.state, Key::Down);
|
||||||
|
ui.tags.handle_input(&ui.state, key);
|
||||||
|
}
|
||||||
_ => (),
|
_ => (),
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -210,4 +214,18 @@ impl Ui {
|
|||||||
|
|
||||||
terminal.clear().unwrap();
|
terminal.clear().unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// create a thread for catching input and send them to core loop
|
||||||
|
pub fn spawn_stdin_channel(&self) -> mpsc::Receiver<termion::event::Key> {
|
||||||
|
let (tx, rx) = mpsc::channel::<termion::event::Key>();
|
||||||
|
|
||||||
|
thread::spawn(move || loop {
|
||||||
|
let stdin = io::stdin();
|
||||||
|
for c in stdin.keys() {
|
||||||
|
tx.send(c.unwrap()).unwrap();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
thread::sleep(std::time::Duration::from_millis(64));
|
||||||
|
rx
|
||||||
|
}
|
||||||
}
|
}
|
@ -1,32 +0,0 @@
|
|||||||
mod default;
|
|
||||||
mod no_yaml;
|
|
||||||
|
|
||||||
use std::sync::mpsc;
|
|
||||||
use std::{io, thread};
|
|
||||||
|
|
||||||
use crate::Opt;
|
|
||||||
use termion::input::TermRead;
|
|
||||||
|
|
||||||
use crate::widget::service_switcher;
|
|
||||||
|
|
||||||
pub fn create_ui(opt: &Opt) {
|
|
||||||
let service_result = service_switcher::ServiceSwitcher::new(&opt.config);
|
|
||||||
match service_result {
|
|
||||||
None => no_yaml::NoYaml::run(opt),
|
|
||||||
Some(_) => default::Ui::run(opt),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// create a thread for catching input and send them to core loop
|
|
||||||
pub fn spawn_stdin_channel() -> mpsc::Receiver<termion::event::Key> {
|
|
||||||
let (tx, rx) = mpsc::channel::<termion::event::Key>();
|
|
||||||
|
|
||||||
thread::spawn(move || loop {
|
|
||||||
let stdin = io::stdin();
|
|
||||||
for c in stdin.keys() {
|
|
||||||
tx.send(c.unwrap()).unwrap();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
thread::sleep(std::time::Duration::from_millis(64));
|
|
||||||
rx
|
|
||||||
}
|
|
@ -1,138 +0,0 @@
|
|||||||
use std::{io, thread};
|
|
||||||
|
|
||||||
use termion::event::Key;
|
|
||||||
use termion::raw::IntoRawMode;
|
|
||||||
use tui::backend::TermionBackend;
|
|
||||||
use tui::layout::{Constraint, Direction, Layout};
|
|
||||||
use tui::Terminal;
|
|
||||||
|
|
||||||
use crate::widget::info;
|
|
||||||
use crate::widget::repo_entry;
|
|
||||||
use crate::widget::tag_list;
|
|
||||||
use crate::Opt;
|
|
||||||
|
|
||||||
#[derive(PartialEq, Clone)]
|
|
||||||
pub enum State {
|
|
||||||
EditRepo,
|
|
||||||
SelectTag,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl std::iter::Iterator for State {
|
|
||||||
type Item = Self;
|
|
||||||
|
|
||||||
fn next(&mut self) -> Option<Self::Item> {
|
|
||||||
match self {
|
|
||||||
State::EditRepo => *self = State::SelectTag,
|
|
||||||
State::SelectTag => *self = State::EditRepo,
|
|
||||||
}
|
|
||||||
Some(self.clone())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct NoYaml {
|
|
||||||
state: State,
|
|
||||||
repo: repo_entry::RepoEntry,
|
|
||||||
tags: tag_list::TagList,
|
|
||||||
info: info::Info,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl NoYaml {
|
|
||||||
pub fn run(opt: &Opt) {
|
|
||||||
let (repo, load_repo) = match &opt.repo {
|
|
||||||
None => (
|
|
||||||
repo_entry::RepoEntry::new(
|
|
||||||
"enter a repository or select one from docker-compose.yml",
|
|
||||||
),
|
|
||||||
false,
|
|
||||||
),
|
|
||||||
Some(repo_id) => (repo_entry::RepoEntry::new(repo_id), true),
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut ui = NoYaml {
|
|
||||||
state: State::EditRepo,
|
|
||||||
repo,
|
|
||||||
tags: tag_list::TagList::with_status("Tags are empty"),
|
|
||||||
info: info::Info::new("edit the Repository widget to load tags"),
|
|
||||||
};
|
|
||||||
|
|
||||||
// load tags if a repository was given thorugh paramter
|
|
||||||
if load_repo {
|
|
||||||
ui.tags = tag_list::TagList::with_repo(ui.repo.get());
|
|
||||||
}
|
|
||||||
|
|
||||||
//setup tui
|
|
||||||
let stdout = io::stdout().into_raw_mode().unwrap();
|
|
||||||
let backend = TermionBackend::new(stdout);
|
|
||||||
let mut terminal = Terminal::new(backend).unwrap();
|
|
||||||
|
|
||||||
//setup input thread
|
|
||||||
let receiver = super::spawn_stdin_channel();
|
|
||||||
|
|
||||||
//core interaction loop
|
|
||||||
'core: loop {
|
|
||||||
//draw
|
|
||||||
terminal
|
|
||||||
.draw(|rect| {
|
|
||||||
let chunks = Layout::default()
|
|
||||||
.direction(Direction::Vertical)
|
|
||||||
.constraints(
|
|
||||||
[
|
|
||||||
Constraint::Length(3),
|
|
||||||
Constraint::Min(7),
|
|
||||||
Constraint::Length(2),
|
|
||||||
]
|
|
||||||
.as_ref(),
|
|
||||||
)
|
|
||||||
.split(rect.size());
|
|
||||||
|
|
||||||
rect.render_widget(ui.repo.render(ui.state == State::EditRepo), chunks[0]);
|
|
||||||
let (list, state) = ui.tags.render(ui.state == State::SelectTag);
|
|
||||||
rect.render_stateful_widget(list, chunks[1], state);
|
|
||||||
rect.render_widget(ui.info.render(), chunks[2]);
|
|
||||||
})
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
//handle input
|
|
||||||
match receiver.try_recv() {
|
|
||||||
Ok(Key::Ctrl('q')) => break 'core,
|
|
||||||
Ok(Key::Char('\t')) => {
|
|
||||||
ui.state.next();
|
|
||||||
}
|
|
||||||
Ok(Key::Ctrl('r')) => {
|
|
||||||
ui.repo.confirm();
|
|
||||||
ui.tags = tag_list::TagList::with_repo(ui.repo.get());
|
|
||||||
}
|
|
||||||
Ok(Key::Ctrl('n')) => match ui.tags.next_page() {
|
|
||||||
Err(e) => ui.info.set_info(&format!("{}", e)),
|
|
||||||
Ok(_) => (),
|
|
||||||
},
|
|
||||||
Ok(Key::Ctrl('p')) => match ui.tags.prev_page() {
|
|
||||||
Err(e) => ui.info.set_info(&format!("{}", e)),
|
|
||||||
Ok(_) => (),
|
|
||||||
},
|
|
||||||
Ok(Key::Char('\n')) => match ui.state {
|
|
||||||
State::EditRepo => {
|
|
||||||
ui.repo.confirm();
|
|
||||||
ui.tags = tag_list::TagList::with_repo(ui.repo.get());
|
|
||||||
}
|
|
||||||
_ => (),
|
|
||||||
},
|
|
||||||
Ok(Key::Char(key)) => match ui.state {
|
|
||||||
State::EditRepo => {
|
|
||||||
ui.info.set_info("Editing Repository");
|
|
||||||
ui.repo.handle_input(Key::Char(key));
|
|
||||||
}
|
|
||||||
State::SelectTag => {
|
|
||||||
ui.tags.handle_input(Key::Char(key));
|
|
||||||
}
|
|
||||||
},
|
|
||||||
_ => (),
|
|
||||||
}
|
|
||||||
|
|
||||||
//sleep for 32ms (30 fps)
|
|
||||||
thread::sleep(std::time::Duration::from_millis(32));
|
|
||||||
}
|
|
||||||
|
|
||||||
terminal.clear().unwrap();
|
|
||||||
}
|
|
||||||
}
|
|
@ -3,6 +3,8 @@ use tui::layout::Alignment;
|
|||||||
use tui::style::{Color, Style};
|
use tui::style::{Color, Style};
|
||||||
use tui::widgets::{Block, Borders, Paragraph};
|
use tui::widgets::{Block, Borders, Paragraph};
|
||||||
|
|
||||||
|
use crate::ui::State;
|
||||||
|
|
||||||
pub struct RepoEntry {
|
pub struct RepoEntry {
|
||||||
text: String,
|
text: String,
|
||||||
old_text: String,
|
old_text: String,
|
||||||
@ -27,13 +29,13 @@ impl RepoEntry {
|
|||||||
self.old_text = entry;
|
self.old_text = entry;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn render(&self, colored: bool) -> Paragraph {
|
pub fn render(&self, state: &crate::ui::State) -> Paragraph {
|
||||||
let title = match self.changed {
|
let title = match self.changed {
|
||||||
true => "Repository*",
|
true => "Repository*",
|
||||||
false => "Repository",
|
false => "Repository",
|
||||||
};
|
};
|
||||||
|
|
||||||
let border_style = if colored {
|
let border_style = if state == &crate::ui::State::EditRepo {
|
||||||
Style::default().fg(Color::Green)
|
Style::default().fg(Color::Green)
|
||||||
} else {
|
} else {
|
||||||
Style::default().fg(Color::Gray)
|
Style::default().fg(Color::Gray)
|
||||||
@ -50,7 +52,11 @@ impl RepoEntry {
|
|||||||
.alignment(Alignment::Left)
|
.alignment(Alignment::Left)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn handle_input(&mut self, key: termion::event::Key) {
|
pub fn handle_input(&mut self, state: &State, key: termion::event::Key) {
|
||||||
|
if state != &State::EditRepo {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
match key {
|
match key {
|
||||||
// Key::Char('\n') => self.confirm(), //handled in Ui
|
// Key::Char('\n') => self.confirm(), //handled in Ui
|
||||||
Key::Char(c) => {
|
Key::Char(c) => {
|
||||||
|
@ -9,6 +9,7 @@ use tui::style::{Color, Style};
|
|||||||
use tui::widgets::{Block, Borders, List, ListState};
|
use tui::widgets::{Block, Borders, List, ListState};
|
||||||
|
|
||||||
use crate::repo;
|
use crate::repo;
|
||||||
|
use crate::ui::State;
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum Error {
|
pub enum Error {
|
||||||
@ -33,7 +34,7 @@ pub struct ServiceSwitcher {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl ServiceSwitcher {
|
impl ServiceSwitcher {
|
||||||
pub fn new(file: &Option<PathBuf>) -> Option<Self> {
|
pub fn new(file: &Option<PathBuf>) -> Self {
|
||||||
let mut file_list = vec![
|
let mut file_list = vec![
|
||||||
PathBuf::from("docker-compose.yml"),
|
PathBuf::from("docker-compose.yml"),
|
||||||
PathBuf::from("docker-compose.yaml"),
|
PathBuf::from("docker-compose.yaml"),
|
||||||
@ -54,20 +55,25 @@ impl ServiceSwitcher {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
return Some(Self {
|
return Self {
|
||||||
list,
|
list,
|
||||||
state: ListState::default(),
|
state: ListState::default(),
|
||||||
changed: false,
|
changed: false,
|
||||||
opened_file: file,
|
opened_file: file,
|
||||||
});
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
//could not find docker-compose file
|
//could not find docker-compose file
|
||||||
None
|
Self {
|
||||||
|
list: vec![format!("No docker-compose file found")],
|
||||||
|
state: ListState::default(),
|
||||||
|
changed: false,
|
||||||
|
opened_file: PathBuf::new(),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn render(&mut self, colored: bool) -> (List, &mut ListState) {
|
pub fn render(&mut self, state: &State) -> (List, &mut ListState) {
|
||||||
let border_style = if colored {
|
let border_style = if state == &State::SelectService {
|
||||||
Style::default().fg(Color::Green)
|
Style::default().fg(Color::Green)
|
||||||
} else {
|
} else {
|
||||||
Style::default().fg(Color::Gray)
|
Style::default().fg(Color::Gray)
|
||||||
|
@ -5,6 +5,7 @@ use tui::style::{Color, Style};
|
|||||||
use tui::widgets::{Block, Borders, List, ListState};
|
use tui::widgets::{Block, Borders, List, ListState};
|
||||||
|
|
||||||
use crate::tags;
|
use crate::tags;
|
||||||
|
use crate::ui::State;
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum Error {
|
pub enum Error {
|
||||||
@ -108,8 +109,8 @@ impl TagList {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn render(&mut self, colored: bool) -> (List, &mut ListState) {
|
pub fn render(&mut self, state: &State) -> (List, &mut ListState) {
|
||||||
let border_style = if colored {
|
let border_style = if state == &State::SelectTag {
|
||||||
Style::default().fg(Color::Green)
|
Style::default().fg(Color::Green)
|
||||||
} else {
|
} else {
|
||||||
Style::default().fg(Color::Gray)
|
Style::default().fg(Color::Gray)
|
||||||
@ -143,7 +144,11 @@ impl TagList {
|
|||||||
(items, &mut self.state)
|
(items, &mut self.state)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn handle_input(&mut self, key: termion::event::Key) {
|
pub fn handle_input(&mut self, state: &State, key: termion::event::Key) {
|
||||||
|
if state != &State::SelectTag {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
match key {
|
match key {
|
||||||
Key::Down => self.next(),
|
Key::Down => self.next(),
|
||||||
Key::Up => self.previous(),
|
Key::Up => self.previous(),
|
||||||
|
Loading…
Reference in New Issue
Block a user