mod commands; mod default_context; mod explore; mod nu_common; mod pager; mod registry; mod views; pub use default_context::add_explore_context; pub use explore::Explore; use std::io; use commands::{ExpandCmd, HelpCmd, HelpManual, NuCmd, QuitCmd, TableCmd, TryCmd}; use nu_common::{collect_pipeline, has_simple_value, CtrlC}; use nu_protocol::{ engine::{EngineState, Stack}, PipelineData, Value, }; use pager::{Page, Pager}; use registry::{Command, CommandRegistry}; use terminal_size::{Height, Width}; use views::{BinaryView, InformationView, Orientation, Preview, RecordView}; use pager::{PagerConfig, StyleConfig}; mod util { pub use super::nu_common::{create_lscolors, create_map, map_into_value}; } fn run_pager( engine_state: &EngineState, stack: &mut Stack, ctrlc: CtrlC, input: PipelineData, config: PagerConfig, ) -> io::Result> { let mut p = Pager::new(config.clone()); let commands = create_command_registry(); let is_record = matches!(input, PipelineData::Value(Value::Record { .. }, ..)); let is_binary = matches!(input, PipelineData::Value(Value::Binary { .. }, ..)); if is_binary { p.show_message("For help type :help"); let view = binary_view(input); return p.run(engine_state, stack, ctrlc, view, commands); } let (columns, data) = collect_pipeline(input); let has_no_input = columns.is_empty() && data.is_empty(); if has_no_input { return p.run(engine_state, stack, ctrlc, information_view(), commands); } p.show_message("For help type :help"); if let Some(value) = has_simple_value(&data) { let text = value.to_abbreviated_string(config.nu_config); let view = Some(Page::new(Preview::new(&text), true)); return p.run(engine_state, stack, ctrlc, view, commands); } let view = create_record_view(columns, data, is_record, config); p.run(engine_state, stack, ctrlc, view, commands) } fn create_record_view( columns: Vec, data: Vec>, is_record: bool, config: PagerConfig, ) -> Option { let mut view = RecordView::new(columns, data); if is_record { view.set_orientation_current(Orientation::Left); } if config.reverse { if let Some((Width(w), Height(h))) = terminal_size::terminal_size() { view.reverse(w, h); } } Some(Page::new(view, false)) } fn information_view() -> Option { Some(Page::new(InformationView, true)) } fn binary_view(input: PipelineData) -> Option { let data = match input { PipelineData::Value(Value::Binary { val, .. }, _) => val, _ => unreachable!("checked beforehand"), }; let view = BinaryView::new(data); Some(Page::new(view, false)) } fn create_command_registry() -> CommandRegistry { let mut registry = CommandRegistry::new(); create_commands(&mut registry); create_aliases(&mut registry); // reregister help && config commands let commands = registry.get_commands().cloned().collect::>(); let aliases = registry.get_aliases().collect::>(); let help_cmd = create_help_command(&commands, &aliases); registry.register_command_view(help_cmd, true); registry } fn create_commands(registry: &mut CommandRegistry) { registry.register_command_view(NuCmd::new(), false); registry.register_command_view(TableCmd::new(), false); registry.register_command_view(ExpandCmd::new(), true); registry.register_command_view(TryCmd::new(), true); registry.register_command_view(HelpCmd::default(), true); registry.register_command_reactive(QuitCmd); } fn create_aliases(registry: &mut CommandRegistry) { registry.create_aliases("h", HelpCmd::NAME); registry.create_aliases("e", ExpandCmd::NAME); registry.create_aliases("q", QuitCmd::NAME); registry.create_aliases("q!", QuitCmd::NAME); } fn create_help_command(commands: &[Command], aliases: &[(&str, &str)]) -> HelpCmd { let help_manuals = create_help_manuals(commands); HelpCmd::new(help_manuals, aliases) } fn create_help_manuals(cmd_list: &[Command]) -> Vec { cmd_list.iter().map(create_help_manual).collect() } fn create_help_manual(cmd: &Command) -> HelpManual { let name = match cmd { Command::Reactive(cmd) => cmd.name(), Command::View { cmd, .. } => cmd.name(), }; let manual = match cmd { Command::Reactive(cmd) => cmd.help(), Command::View { cmd, .. } => cmd.help(), }; __create_help_manual(manual, name) } fn __create_help_manual(manual: Option, name: &'static str) -> HelpManual { manual.unwrap_or(HelpManual { name, ..HelpManual::default() }) }