diff --git a/README.md b/README.md
index 797600109..339b54360 100644
--- a/README.md
+++ b/README.md
@@ -13,7 +13,7 @@ Detailed user documentation is available by running `help` within fish, and also
## Building
-fish can be built using autotools or Xcode:
+fish can be built using autotools or Xcode.
### Autotools Build
diff --git a/color.cpp b/color.cpp
index 3b7baa898..dce115aa2 100644
--- a/color.cpp
+++ b/color.cpp
@@ -164,11 +164,9 @@ static unsigned char term8_color_for_rgb(const unsigned char rgb[3]) {
0x000000, //Black
0xFF0000, //Red
0x00FF00, //Green
- 0x725000, //Brown
0xFFFF00, //Yellow
0x0000FF, //Blue
0xFF00FF, //Magenta
- 0xFF00FF, //Purple
0x00FFFF, //Cyan
0xFFFFFF, //White
};
diff --git a/share/tools/web_config/index.html b/share/tools/web_config/index.html
index 41c426f09..8b9c6132b 100644
--- a/share/tools/web_config/index.html
+++ b/share/tools/web_config/index.html
@@ -38,7 +38,7 @@ body {
padding-top: 15px;
font-size: 17pt;
text-align: center;
- width: 25%;
+ width: 20%;
background-color: #292929;
cursor: pointer;
}
@@ -93,8 +93,9 @@ body {
text-align: right;
min-width: 200px;
font-size: 16pt;
- padding-bottom: 5px;
- margin-top: -7px;
+ padding-bottom: 20px;
+ padding-top: 35px;
+ vertical-align: top;
}
#detail {
@@ -316,6 +317,57 @@ img.delete_icon {
color: #C8C8C8;
}
+.prompt_demo {
+ /* This is the div that holds what the prompt looks like */
+ width: 100%;
+ background-color: black;
+ border-radius: 5px;
+ display: table;
+}
+
+.prompt_save_button {
+ border-radius: 5px;
+ border: solid #474747 1px;
+ text-shadow: 0px 1px #000;
+ padding: 5px 8px;
+ font-size: 12pt;
+ display: inline-block;
+ margin-top: 12px;
+ background-color: #282828;
+ color: #DDD;
+ cursor: pointer;
+}
+
+.prompt_save_button:hover {
+ background-color: #333;
+ border: solid #525252 1px;
+ color: #EEE;
+}
+
+.prompt_demo_text {
+ white-space: pre;
+ line-height: 170%;
+ padding: 4px 12px;
+ font-size: 14pt;
+ top: 0px;
+ bottom: 0px;
+ vertical-align: middle;
+ display: table-cell;
+ height: 72px; /* this is really the min height */
+}
+
+.prompt_function {
+ /* This is the div that holds the prompt function's definition */
+ width: 100%;
+ color: #BBB;
+ font-size: 10pt;
+}
+
+.prompt_function_text {
+ white-space: pre-wrap;
+ padding: 25px 3px;
+}
+
@@ -466,6 +518,15 @@ function master_color_for_color(color_str) {
})
}
+/* Update prompt_demo_text with the given text, adjusting the font */
+function set_prompt_demo_text(txt, font_size) {
+ /* Set the font size and the text */
+ var prompt_demo_text = $('.prompt_demo_text')
+ prompt_demo_text.css('font-size', font_size)
+ prompt_demo_text.html(txt)
+}
+
+
function switch_tab(new_tab) {
/* Switch selected tab */
$(".selected_tab").removeClass("selected_tab")
@@ -480,6 +541,7 @@ function switch_tab(new_tab) {
/* Hide some things */
$('#master_detail_table').hide()
$('#detail_colorpicker').hide()
+ $('#detail_prompt').hide()
$('#detail_function').hide()
$('#data_table').hide()
$('#table_filter_container').hide()
@@ -508,11 +570,30 @@ function switch_tab(new_tab) {
$('#detail_colorpicker').show()
$('#master_detail_table').show()
wants_data_table = false
+ } else if (new_tab == 'tab_prompt') {
+ /* Get rid of all sample prompts */
+ sample_prompts.length = 0
+ /* Color the first one blue */
+ var first = true;
+ run_get_request('/sample_prompts/', function(sample_prompt){
+ var name = sample_prompt['name']
+ sample_prompts[name] = sample_prompt
+ var color = first ? '66F' : 'AAA'
+ var func = first ? select_current_prompt_master_element : select_sample_prompt_master_element;
+ var elem = create_master_element(name, false/* description */, color, '13pt', func)
+ if (first) {
+ elem.children('.master_element_text').css('border-bottom-color', color)
+ select_current_prompt_master_element(elem)
+ }
+ first = false;
+ })
+ $('#detail_prompt').show()
+ $('#master_detail_table').show()
} else if (new_tab == 'tab_functions') {
/* Keep track of whether this is the first element */
var first = true
run_get_request('/functions/', function(contents){
- elem = create_master_element(contents, false/* description */, 'AAAAAA', '11pt', select_function_master_element)
+ var elem = create_master_element(contents, false/* description */, 'AAAAAA', '11pt', select_function_master_element)
if (first) {
/* It's the first element, so select it, so something gets selected */
select_function_master_element(elem)
@@ -576,7 +657,7 @@ function current_master_element_name() {
}
elem = elems[0]
if (elem.id.indexOf('master_') != 0) {
- show_error('Unknown color variable')
+ show_error('Unknown master variable')
return ''
}
return elem.id.substring(7)
@@ -641,6 +722,24 @@ function reflect_style() {
}
}
+function cleanup_fish_function(contents) {
+ /* Replace leading tabs and groups of four spaces at the beginning of a line with two spaces. */
+ lines = contents.split('\n')
+ rx = /^[\t ]+/
+ for (var i=0; i < lines.length; i++) {
+ line = lines[i]
+ /* Get leading tabs and spaces */
+ whitespace_arr = rx.exec(line)
+ if (whitespace_arr) {
+ /* Replace four spaces with two spaces, and tabs with two spaces */
+ var whitespace = whitespace_arr[0]
+ new_whitespace = whitespace.replace(/( )|(\t)/g, ' ')
+ lines[i] = new_whitespace + line.slice(whitespace.length)
+ }
+ }
+ return lines.join('\n')
+}
+
function select_master_element(elem) {
$('.selected_master_elem').removeClass('selected_master_elem')
$(elem).addClass('selected_master_elem')
@@ -660,25 +759,57 @@ function select_function_master_element(elem) {
what: current_master_element_name()
}, function(contents){
/* Replace leading tabs and groups of four spaces at the beginning of a line with two spaces. */
- lines = contents.split('\n')
- rx = /^[\t ]+/
- for (var i=0; i < lines.length; i++) {
- line = lines[i]
- /* Get leading tabs and spaces */
- whitespace_arr = rx.exec(line)
- if (whitespace_arr) {
- /* Replace four spaces with two spaces, and tabs with two spaces */
- var whitespace = whitespace_arr[0]
- new_whitespace = whitespace.replace(/( )|(\t)/g, ' ')
- lines[i] = new_whitespace + line.slice(whitespace.length)
- }
- }
- munged_contents = lines.join('\n')
-
+ munged_contents = cleanup_fish_function(contents)
$('#detail_function').text(munged_contents)
});
}
+var sample_prompts = new Array();
+
+function select_sample_prompt_master_element(elem) {
+ $('.prompt_save_button').show()
+ select_master_element(elem)
+ var name = current_master_element_name()
+ sample_prompt = sample_prompts[name]
+ run_post_request('/get_sample_prompt/', {
+ what: sample_prompt['function']
+ }, function(keys_and_values){
+ var prompt_func = keys_and_values['function']
+ var prompt_demo = keys_and_values['demo']
+ var prompt_font_size = keys_and_values['font_size']
+ set_prompt_demo_text(prompt_demo, prompt_font_size)
+ //$('.prompt_demo_text').html(prompt_demo)
+ $('.prompt_function_text').text(cleanup_fish_function(prompt_func))
+ })
+}
+
+function select_current_prompt_master_element(elem) {
+ $('.prompt_save_button').hide()
+ select_master_element(elem)
+ run_get_request_with_bulk_handler('/current_prompt/', function(keys_and_values){
+ var prompt_func = keys_and_values['function']
+ var prompt_demo = keys_and_values['demo']
+ var prompt_font_size = keys_and_values['font_size']
+ set_prompt_demo_text(prompt_demo, prompt_font_size)
+ $('.prompt_function_text').text(cleanup_fish_function(prompt_func))
+ })
+}
+
+/* Applies the current prompt */
+function save_current_prompt() {
+ var name = current_master_element_name()
+ var sample_prompt = sample_prompts[name]
+ run_post_request('/set_prompt/', {
+ what: sample_prompt['function']
+ }, function(contents){
+ if (contents == "OK") {
+ select_current_prompt_master_element($('#master_Current'))
+ } else {
+ show_error(contents)
+ }
+ })
+}
+
function post_style_to_server() {
style = current_style()
if (! style)
@@ -1220,6 +1351,9 @@ $(document).ready(function() {
case '#history':
tab_name = 'tab_history'
break
+ case '#prompt':
+ tab_name = 'tab_prompt'
+ break
case '#colors':
default:
tab_name = 'tab_colors'
@@ -1237,6 +1371,7 @@ $(document).ready(function() {
colors
+
prompt
functions
variables
history
@@ -1256,6 +1391,21 @@ $(document).ready(function() {
+
+
+
+
+ use prompt
+
+
+
+
diff --git a/share/tools/web_config/sample_prompts/classic.fish b/share/tools/web_config/sample_prompts/classic.fish
new file mode 100644
index 000000000..9464b778f
--- /dev/null
+++ b/share/tools/web_config/sample_prompts/classic.fish
@@ -0,0 +1,36 @@
+# name: Classic
+function fish_prompt --description "Write out the prompt"
+
+ # Just calculate these once, to save a few cycles when displaying the prompt
+ if not set -q __fish_prompt_hostname
+ set -g __fish_prompt_hostname (hostname|cut -d . -f 1)
+ end
+
+ if not set -q __fish_prompt_normal
+ set -g __fish_prompt_normal (set_color normal)
+ end
+
+ switch $USER
+
+ case root
+
+ if not set -q __fish_prompt_cwd
+ if set -q fish_color_cwd_root
+ set -g __fish_prompt_cwd (set_color $fish_color_cwd_root)
+ else
+ set -g __fish_prompt_cwd (set_color $fish_color_cwd)
+ end
+ end
+
+ echo -n -s "$USER" @ "$__fish_prompt_hostname" ' ' "$__fish_prompt_cwd" (prompt_pwd) "$__fish_prompt_normal" '# '
+
+ case '*'
+
+ if not set -q __fish_prompt_cwd
+ set -g __fish_prompt_cwd (set_color $fish_color_cwd)
+ end
+
+ echo -n -s "$USER" @ "$__fish_prompt_hostname" ' ' "$__fish_prompt_cwd" (prompt_pwd) "$__fish_prompt_normal" '> '
+
+ end
+end
diff --git a/share/tools/web_config/sample_prompts/classic_git.fish b/share/tools/web_config/sample_prompts/classic_git.fish
new file mode 100644
index 000000000..b09fcef97
--- /dev/null
+++ b/share/tools/web_config/sample_prompts/classic_git.fish
@@ -0,0 +1,86 @@
+# name: Classic + Git
+# author: Kevin Ballard
+
+function fish_prompt --description 'Write out the prompt'
+
+ set -l last_status $status
+
+ # Just calculate these once, to save a few cycles when displaying the prompt
+ if not set -q __fish_prompt_hostname
+ set -g __fish_prompt_hostname (hostname|cut -d . -f 1)
+ end
+
+ if not set -q __fish_prompt_normal
+ set -g __fish_prompt_normal (set_color normal)
+ end
+
+ set -l delim '>'
+
+ switch $USER
+
+ case root
+
+ if not set -q __fish_prompt_cwd
+ if set -q fish_color_cwd_root
+ set -g __fish_prompt_cwd (set_color $fish_color_cwd_root)
+ else
+ set -g __fish_prompt_cwd (set_color $fish_color_cwd)
+ end
+ end
+
+ case '*'
+
+ if not set -q __fish_prompt_cwd
+ set -g __fish_prompt_cwd (set_color $fish_color_cwd)
+ end
+
+ end
+
+ set -l prompt_status
+ if test $last_status -ne 0
+ if not set -q __fish_prompt_status
+ set -g __fish_prompt_status (set_color $fish_color_status)
+ end
+ set prompt_status "$__fish_prompt_status [$last_status]$__fish_prompt_normal"
+ end
+
+ if not set -q __fish_prompt_user
+ set -g __fish_prompt_user (set_color $fish_color_user)
+ end
+ if not set -q __fish_prompt_host
+ set -g __fish_prompt_host (set_color $fish_color_host)
+ end
+
+ echo -n -s "$__fish_prompt_user" "$USER" "$__fish_prompt_normal" @ "$__fish_prompt_host" "$__fish_prompt_hostname" "$__fish_prompt_normal" ' ' "$__fish_prompt_cwd" (prompt_pwd) (__fish_git_prompt) "$__fish_prompt_normal" "$prompt_status" "$delim" ' '
+end
+
+function __fish_repaint_user --on-variable fish_color_user --description "Event handler, repaint when fish_color_user changes"
+ if status --is-interactive
+ set -e __fish_prompt_user
+ commandline -f repaint ^/dev/null
+ end
+end
+
+function __fish_repaint_host --on-variable fish_color_host --description "Event handler, repaint when fish_color_host changes"
+ if status --is-interactive
+ set -e __fish_prompt_host
+ commandline -f repaint ^/dev/null
+ end
+end
+
+function __fish_repaint_status --on-variable fish_color_status --description "Event handler; repaint when fish_color_status changes"
+ if status --is-interactive
+ set -e __fish_prompt_status
+ commandline -f repaint ^/dev/null
+ end
+end
+
+# initialize our new variables
+# in theory this would be in a fish_prompt event, but this file isn't sourced
+# until the fish_prompt function is called anyway.
+if not set -q __prompt_initialized_2
+ set -U fish_color_user -o green
+ set -U fish_color_host -o cyan
+ set -U fish_color_status red
+ set -U __prompt_initialized_2
+end
diff --git a/share/tools/web_config/sample_prompts/classic_status.fish b/share/tools/web_config/sample_prompts/classic_status.fish
new file mode 100644
index 000000000..a93ff9f22
--- /dev/null
+++ b/share/tools/web_config/sample_prompts/classic_status.fish
@@ -0,0 +1,47 @@
+# name: Classic + Status
+# author: David Frascone
+
+function fish_prompt --description 'Write out the prompt'
+ # Save our status
+ set -l last_status $status
+
+ set -l last_status_string ""
+ if [ $last_status -ne 0 ]
+ printf "%s(%d)%s " (set_color red --bold) $last_status (set_color normal)
+ end
+
+ # Just calculate these once, to save a few cycles when displaying the prompt
+ if not set -q __fish_prompt_hostname
+ set -g __fish_prompt_hostname (hostname|cut -d . -f 1)
+ end
+
+ if not set -q __fish_prompt_normal
+ set -g __fish_prompt_normal (set_color normal)
+ end
+
+ set -l user_prompt '>'
+ switch $USER
+ # Set our root colors, if we're root :)
+ case root
+ set user_prompt '#'
+ if not set -q __fish_prompt_cwd
+ if set -q fish_color_cwd_root
+ set -g __fish_prompt_cwd (set_color $fish_color_cwd_root)
+ else
+ set -g __fish_prompt_cwd (set_color $fish_color_cwd)
+ end
+ end
+ case '*'
+ if not set -q __fish_prompt_cwd
+ set -g __fish_prompt_cwd (set_color $fish_color_cwd)
+ end
+ end
+ #printf '%s@%s %s%s%s# ' $USER $__fish_prompt_hostname "$__fish_prompt_cwd" (prompt_pwd) "$__fish_prompt_normal"
+ #printf "LAST STATUS STRING: $last_status_string \n"
+ printf '%s@%s %s%s%s%s%s ' $USER $__fish_prompt_hostname "$__fish_prompt_cwd" (prompt_pwd) "$__fish_prompt_normal" $user_prompt
+
+end
+
+function fish_title
+ echo $_ ' ' (prompt_pwd)
+end
diff --git a/share/tools/web_config/sample_prompts/hyperminimalist.fish b/share/tools/web_config/sample_prompts/hyperminimalist.fish
new file mode 100644
index 000000000..0aca5508f
--- /dev/null
+++ b/share/tools/web_config/sample_prompts/hyperminimalist.fish
@@ -0,0 +1,4 @@
+# name: Hyper Minimalist
+function fish_prompt
+ echo -n '$ '
+end
diff --git a/share/tools/web_config/sample_prompts/informative.fish b/share/tools/web_config/sample_prompts/informative.fish
new file mode 100644
index 000000000..9095b740f
--- /dev/null
+++ b/share/tools/web_config/sample_prompts/informative.fish
@@ -0,0 +1,51 @@
+# name: Informative
+# http://michal.karzynski.pl/blog/2009/11/19/my-informative-shell-prompt/
+
+
+function fish_prompt --description 'Write out the prompt'
+ #Save the return status of the previous command
+ set stat $status
+
+# Just calculate these once, to save a few cycles when displaying the prompt
+ if not set -q __fish_prompt_hostname
+ set -g __fish_prompt_hostname (hostname|cut -d . -f 1)
+ end
+
+if not set -q __fish_prompt_normal
+ set -g __fish_prompt_normal (set_color normal)
+ end
+
+if not set -q __fish_color_blue
+ set -g __fish_color_blue (set_color -o blue)
+ end
+
+#Set the color for the status depending on the value
+ set __fish_color_status (set_color -o green)
+ if test $stat -gt 0
+ set __fish_color_status (set_color -o red)
+ end
+
+switch $USER
+
+case root
+
+if not set -q __fish_prompt_cwd
+ if set -q fish_color_cwd_root
+ set -g __fish_prompt_cwd (set_color $fish_color_cwd_root)
+ else
+ set -g __fish_prompt_cwd (set_color $fish_color_cwd)
+ end
+ end
+
+printf '%s@%s %s%s%s# ' $USER $__fish_prompt_hostname "$__fish_prompt_cwd" (prompt_pwd) "$__fish_prompt_normal"
+
+case '*'
+
+if not set -q __fish_prompt_cwd
+ set -g __fish_prompt_cwd (set_color $fish_color_cwd)
+ end
+
+printf '[%s] %s%s@%s %s%s %s(%s)%s \f\r> ' (date "+%H:%M:%S") "$__fish_color_blue" $USER $__fish_prompt_hostname "$__fish_prompt_cwd" (pwd) "$__fish_color_status" "$stat" "$__fish_prompt_normal"
+
+end
+end
\ No newline at end of file
diff --git a/share/tools/web_config/sample_prompts/lonetwin.fish b/share/tools/web_config/sample_prompts/lonetwin.fish
new file mode 100644
index 000000000..674785941
--- /dev/null
+++ b/share/tools/web_config/sample_prompts/lonetwin.fish
@@ -0,0 +1,20 @@
+# name: lonetwin
+# author: Steve
+
+function fish_prompt --description 'Write out the prompt'
+ # Just calculate these once, to save a few cycles when displaying the prompt
+ if not set -q __fish_prompt_hostname
+ set -g __fish_prompt_hostname (hostname -s)
+ end
+
+ if not set -q __fish_prompt_normal
+ set -g __fish_prompt_normal (set_color normal)
+ end
+
+ if not set -q __fish_prompt_cwd
+ set -g __fish_prompt_cwd (set_color $fish_color_cwd)
+ end
+
+ echo -n -s "$USER" @ "$__fish_prompt_hostname" ' ' "$__fish_prompt_cwd" (prompt_pwd) (__fish_git_prompt) "$__fish_prompt_normal" '> '
+
+end
diff --git a/share/tools/web_config/sample_prompts/minimalist.fish b/share/tools/web_config/sample_prompts/minimalist.fish
new file mode 100644
index 000000000..14cc04458
--- /dev/null
+++ b/share/tools/web_config/sample_prompts/minimalist.fish
@@ -0,0 +1,9 @@
+# name: Minimalist
+# author: ridiculous_fish
+
+function fish_prompt
+ set_color $fish_color_cwd
+ echo -n (basename $PWD)
+ set_color normal
+ echo -n ' ) '
+end
diff --git a/share/tools/web_config/sample_prompts/nim.fish b/share/tools/web_config/sample_prompts/nim.fish
new file mode 100644
index 000000000..6eab39e68
--- /dev/null
+++ b/share/tools/web_config/sample_prompts/nim.fish
@@ -0,0 +1,90 @@
+# name: Nim
+# author: Guilhem "Nim" Saurel − https://github.com/nim65s/dotfiles/
+
+function fish_prompt
+ and set retc green; or set retc red
+ tty|grep -q tty; and set tty tty; or set tty pts
+
+ set_color $retc
+ if [ $tty = tty ]
+ echo -n .-
+ else
+ echo -n '┬─'
+ end
+ set_color -o green
+ echo -n [
+ if [ $USER = root ]
+ set_color -o red
+ else
+ set_color -o yellow
+ end
+ echo -n $USER
+ set_color -o white
+ echo -n @
+ if [ -z "$SSH_CLIENT" ]
+ set_color -o blue
+ else
+ set_color -o cyan
+ end
+ echo -n (hostname)
+ set_color -o white
+ #echo -n :(prompt_pwd)
+ echo -n :(pwd|sed "s=$HOME=~=")
+ set_color -o green
+ echo -n ']'
+ set_color normal
+ set_color $retc
+ if [ $tty = tty ]
+ echo -n '-'
+ else
+ echo -n '─'
+ end
+ set_color -o green
+ echo -n '['
+ set_color normal
+ set_color $retc
+ echo -n (date +%X)
+ set_color -o green
+ echo -n ]
+
+ # Check if acpi exists
+ if not set -q __fish_nim_prompt_has_acpi
+ if type acpi > /dev/null
+ set -g __fish_nim_prompt_has_acpi ''
+ else
+ set -g __fish_nim_prompt_has_acpi '' # empty string
+ end
+ end
+
+ if test "$__fish_nim_prompt_has_acpi"
+ if [ (acpi -a 2> /dev/null | grep off) ]
+ echo -n '─['
+ set_color -o red
+ echo -n (acpi -b|cut -d' ' -f 4-)
+ set_color -o green
+ echo -n ']'
+ end
+ end
+ echo
+ set_color normal
+ for job in (jobs)
+ set_color $retc
+ if [ $tty = tty ]
+ echo -n '; '
+ else
+ echo -n '│ '
+ end
+ set_color brown
+ echo $job
+ end
+ set_color normal
+ set_color $retc
+ if [ $tty = tty ]
+ echo -n "'->"
+ else
+ echo -n '╰─>'
+ end
+ set_color -o red
+ echo -n '$ '
+ set_color normal
+end
diff --git a/share/tools/web_config/sample_prompts/pythonista.fish b/share/tools/web_config/sample_prompts/pythonista.fish
new file mode 100644
index 000000000..5c98e56b6
--- /dev/null
+++ b/share/tools/web_config/sample_prompts/pythonista.fish
@@ -0,0 +1,33 @@
+# name: Simple Pythonista
+# author: davbo
+
+set normal (set_color normal)
+set magenta (set_color magenta)
+set yellow (set_color yellow)
+set green (set_color green)
+set gray (set_color -o black)
+
+
+function fish_prompt
+ set_color yellow
+ printf '%s' (whoami)
+ set_color normal
+ printf ' at '
+
+ set_color magenta
+ printf '%s' (hostname|cut -d . -f 1)
+ set_color normal
+ printf ' in '
+
+ set_color $fish_color_cwd
+ printf '%s' (prompt_pwd)
+ set_color normal
+
+ # Line 2
+ echo
+ if test $VIRTUAL_ENV
+ printf "(%s) " (set_color blue)(basename $VIRTUAL_ENV)(set_color normal)
+ end
+ printf '↪ '
+ set_color normal
+end
diff --git a/share/tools/web_config/sample_prompts/screen_savvy.fish b/share/tools/web_config/sample_prompts/screen_savvy.fish
new file mode 100644
index 000000000..83c7db641
--- /dev/null
+++ b/share/tools/web_config/sample_prompts/screen_savvy.fish
@@ -0,0 +1,9 @@
+# name: Screen Savvy
+# author: Matthias
+function fish_prompt -d "Write out the prompt"
+ if test -z $WINDOW
+ printf '%s%s@%s%s%s%s%s> ' (set_color yellow) (whoami) (set_color purple) (hostname|cut -d . -f 1) (set_color $fish_color_cwd) (prompt_pwd) (set_color normal)
+ else
+ printf '%s%s@%s%s%s(%s)%s%s%s> ' (set_color yellow) (whoami) (set_color purple) (hostname|cut -d . -f 1) (set_color white) (echo $WINDOW) (set_color $fish_color_cwd) (prompt_pwd) (set_color normal)
+ end
+end
diff --git a/share/tools/web_config/sample_prompts/user_host_path.fish b/share/tools/web_config/sample_prompts/user_host_path.fish
new file mode 100644
index 000000000..c125eea0c
--- /dev/null
+++ b/share/tools/web_config/sample_prompts/user_host_path.fish
@@ -0,0 +1,13 @@
+# name: User, Host, Path
+# author: Jon Clayden
+
+function fish_prompt -d "Write out the prompt"
+ set -l home_escaped (echo -n $HOME | sed 's/\//\\\\\//g')
+ set -l pwd (echo -n $PWD | sed "s/^$home_escaped/~/" | sed 's/ /%20/g')
+ set -l prompt_symbol ''
+ switch $USER
+ case root; set prompt_symbol '#'
+ case '*'; set prompt_symbol '$'
+ end
+ printf "[%s@%s %s%s%s]%s " $USER (hostname -s) (set_color $fish_color_cwd) $pwd (set_color normal) $prompt_symbol
+end
diff --git a/share/tools/web_config/webconfig.py b/share/tools/web_config/webconfig.py
index b25337537..6b124ef63 100755
--- a/share/tools/web_config/webconfig.py
+++ b/share/tools/web_config/webconfig.py
@@ -14,7 +14,7 @@ else:
from urllib.parse import parse_qs
import webbrowser
import subprocess
-import re, json, socket, os, sys, cgi, select, time
+import re, json, socket, os, sys, cgi, select, time, glob
def run_fish_cmd(text):
from subprocess import PIPE
@@ -97,6 +97,122 @@ def parse_bool(val):
if val.startswith('t') or val.startswith('1'): return True
return bool(val)
+def html_color_for_ansi_color_index(val):
+ arr = ['black', '#AA0000', '#00AA00', '#AA5500', '#0000AA', '#AA00AA', '#00AAAA', '#AAAAAA', '#555555', '#FF5555', '#55FF55', '#FFFF55', '#5555FF', '#FF55FF', '#55FFFF', 'white', '#000000', '#00005f', '#000087', '#0000af', '#0000d7', '#0000ff', '#005f00', '#005f5f', '#005f87', '#005faf', '#005fd7', '#005fff', '#008700', '#00875f', '#008787', '#0087af', '#0087d7', '#0087ff', '#00af00', '#00af5f', '#00af87', '#00afaf', '#00afd7', '#00afff', '#00d700', '#00d75f', '#00d787', '#00d7af', '#00d7d7', '#00d7ff', '#00ff00', '#00ff5f', '#00ff87', '#00ffaf', '#00ffd7', '#00ffff', '#5f0000', '#5f005f', '#5f0087', '#5f00af', '#5f00d7', '#5f00ff', '#5f5f00', '#5f5f5f', '#5f5f87', '#5f5faf', '#5f5fd7', '#5f5fff', '#5f8700', '#5f875f', '#5f8787', '#5f87af', '#5f87d7', '#5f87ff', '#5faf00', '#5faf5f', '#5faf87', '#5fafaf', '#5fafd7', '#5fafff', '#5fd700', '#5fd75f', '#5fd787', '#5fd7af', '#5fd7d7', '#5fd7ff', '#5fff00', '#5fff5f', '#5fff87', '#5fffaf', '#5fffd7', '#5fffff', '#870000', '#87005f', '#870087', '#8700af', '#8700d7', '#8700ff', '#875f00', '#875f5f', '#875f87', '#875faf', '#875fd7', '#875fff', '#878700', '#87875f', '#878787', '#8787af', '#8787d7', '#8787ff', '#87af00', '#87af5f', '#87af87', '#87afaf', '#87afd7', '#87afff', '#87d700', '#87d75f', '#87d787', '#87d7af', '#87d7d7', '#87d7ff', '#87ff00', '#87ff5f', '#87ff87', '#87ffaf', '#87ffd7', '#87ffff', '#af0000', '#af005f', '#af0087', '#af00af', '#af00d7', '#af00ff', '#af5f00', '#af5f5f', '#af5f87', '#af5faf', '#af5fd7', '#af5fff', '#af8700', '#af875f', '#af8787', '#af87af', '#af87d7', '#af87ff', '#afaf00', '#afaf5f', '#afaf87', '#afafaf', '#afafd7', '#afafff', '#afd700', '#afd75f', '#afd787', '#afd7af', '#afd7d7', '#afd7ff', '#afff00', '#afff5f', '#afff87', '#afffaf', '#afffd7', '#afffff', '#d70000', '#d7005f', '#d70087', '#d700af', '#d700d7', '#d700ff', '#d75f00', '#d75f5f', '#d75f87', '#d75faf', '#d75fd7', '#d75fff', '#d78700', '#d7875f', '#d78787', '#d787af', '#d787d7', '#d787ff', '#d7af00', '#d7af5f', '#d7af87', '#d7afaf', '#d7afd7', '#d7afff', '#d7d700', '#d7d75f', '#d7d787', '#d7d7af', '#d7d7d7', '#d7d7ff', '#d7ff00', '#d7ff5f', '#d7ff87', '#d7ffaf', '#d7ffd7', '#d7ffff', '#ff0000', '#ff005f', '#ff0087', '#ff00af', '#ff00d7', '#ff00ff', '#ff5f00', '#ff5f5f', '#ff5f87', '#ff5faf', '#ff5fd7', '#ff5fff', '#ff8700', '#ff875f', '#ff8787', '#ff87af', '#ff87d7', '#ff87ff', '#ffaf00', '#ffaf5f', '#ffaf87', '#ffafaf', '#ffafd7', '#ffafff', '#ffd700', '#ffd75f', '#ffd787', '#ffd7af', '#ffd7d7', '#ffd7ff', '#ffff00', '#ffff5f', '#ffff87', '#ffffaf', '#ffffd7', '#ffffff', '#080808', '#121212', '#1c1c1c', '#262626', '#303030', '#3a3a3a', '#444444', '#4e4e4e', '#585858', '#626262', '#6c6c6c', '#767676', '#808080', '#8a8a8a', '#949494', '#9e9e9e', '#a8a8a8', '#b2b2b2', '#bcbcbc', '#c6c6c6', '#d0d0d0', '#dadada', '#e4e4e4', '#eeeeee']
+ if val < 0 or val >= len(arr):
+ return ''
+ else:
+ return arr[val]
+
+# Function to return special ANSI escapes like exit_attribute_mode
+g_special_escapes_dict = None
+def get_special_ansi_escapes():
+ global g_special_escapes_dict
+ if g_special_escapes_dict is None:
+ import curses
+ g_special_escapes_dict = {}
+ curses.setupterm()
+
+ # Helper function to get a value for a tparm
+ def get_tparm(key):
+ val = None
+ key = curses.tigetstr("sgr0")
+ if key: val = curses.tparm(key)
+ return val
+
+ # Just a few for now
+ g_special_escapes_dict['exit_attribute_mode'] = get_tparm('sgr0')
+ g_special_escapes_dict['bold'] = get_tparm('bold')
+ g_special_escapes_dict['underline'] = get_tparm('smul')
+
+ return g_special_escapes_dict
+
+# Given a known ANSI escape sequence, convert it to HTML and append to the list
+# Returns whether we have an open
+def append_html_for_ansi_escape(full_val, result, span_open):
+
+ # Strip off the initial \x1b[ and terminating m
+ val = full_val[2:-1]
+
+ # Helper function to close a span if it's open
+ def close_span():
+ if span_open:
+ result.append('')
+
+ # term256 foreground color
+ match = re.match('38;5;(\d+)', val)
+ if match is not None:
+ close_span()
+ html_color = html_color_for_ansi_color_index(int(match.group(1)))
+ result.append('')
+ return True # span now open
+
+ # term8 foreground color
+ if val in [str(x) for x in range(30, 38)]:
+ close_span()
+ html_color = html_color_for_ansi_color_index(int(val) - 30)
+ result.append('')
+ return True # span now open
+
+ # Try special escapes
+ special_escapes = get_special_ansi_escapes()
+ if full_val == special_escapes['exit_attribute_mode']:
+ close_span()
+ return False
+
+ # We don't handle bold or underline yet
+
+ # Do nothing on failure
+ return span_open
+
+def strip_ansi(val):
+ # Make a half-assed effort to strip ANSI control sequences
+ # We assume that all such sequences start with 0x1b and end with m,
+ # which catches most cases
+ return re.sub("\x1b[^m]*m", '', val)
+
+def ansi_prompt_line_width(val):
+ # Given an ANSI prompt, return the length of its longest line, as in the number of characters it takes up
+ # Start by stripping off ANSI
+ stripped_val = strip_ansi(val)
+
+ # Now count the longest line
+ return max([len(x) for x in stripped_val.split('\n')])
+
+
+def ansi_to_html(val):
+ # Split us up by ANSI escape sequences
+ # We want to catch not only the standard color codes, but also things like sgr0
+ # Hence this lame check
+ separated = re.split("""
+ ( # Capture
+ \x1b # Escpae
+ [^m]+ # One or more non-'m's
+ m # Literal m terminates the sequence
+ ) # End capture
+ """, val, 0, re.VERBOSE)
+
+ # We have to HTML escape the text and convert ANSI escapes into HTML
+ # Collect it all into this array
+ result = []
+
+ span_open = False
+
+ # Text is at even indexes, escape sequences at odd indexes
+ for i in xrange(len(separated)):
+ component = separated[i]
+ if i % 2 == 0:
+ # It's text, possibly empty
+ # Clean up other ANSI junk
+ result.append(cgi.escape(strip_ansi(component)))
+ else:
+ # It's an escape sequence. Close the previous escape.
+ span_open = append_html_for_ansi_escape(component, result, span_open)
+
+ # Close final escape
+ if span_open: result.append('')
+ return ''.join(result)
+
class FishVar:
""" A class that represents a variable """
def __init__(self, name, value):
@@ -253,6 +369,90 @@ class FishConfigHTTPRequestHandler(SimpleHTTPServer.SimpleHTTPRequestHandler):
# It's really lame that we always return success here
out, err = run_fish_cmd('builtin history --save --delete -- ' + escape_fish_cmd(history_item_text))
return True
+
+ def do_set_prompt_function(self, prompt_func):
+ cmd = prompt_func + '\n' + 'funcsave fish_prompt'
+ out, err = run_fish_cmd(cmd)
+ return len(err) == 0
+
+ def do_get_prompt(self, command_to_run, prompt_function_text):
+ # Return the prompt output by the given command
+ prompt_demo_ansi, err = run_fish_cmd(command_to_run)
+ prompt_demo_html = ansi_to_html(prompt_demo_ansi)
+ prompt_demo_font_size = self.font_size_for_ansi_prompt(prompt_demo_ansi)
+ return {'function': prompt_function_text, 'demo': prompt_demo_html, 'font_size': prompt_demo_font_size }
+
+ def do_get_current_prompt(self):
+ # Return the current prompt
+ prompt_func, err = run_fish_cmd('functions fish_prompt')
+ return self.do_get_prompt('cd "' + initial_wd + '" ; fish_prompt', prompt_func.strip())
+
+ def do_get_sample_prompt(self, text):
+ # Return the prompt you get from the given text
+ cmd = text + "\n cd \"" + initial_wd + "\" \n fish_prompt\n"
+ return self.do_get_prompt(cmd, text.strip())
+
+ def parse_one_sample_prompt_hash(self, line, result_dict):
+ # Allow us to skip whitespace, etc.
+ if not line: return True
+ if line.isspace(): return True
+
+ # Parse a comment hash like '# name: Classic'
+ match = re.match(r"#\s*(\w+?): (.+)", line, re.IGNORECASE)
+ if match:
+ key = match.group(1).strip()
+ value = match.group(2).strip()
+ result_dict[key] = value
+ return True
+ # Skip other hash comments
+ return line.startswith('#')
+
+
+ def read_one_sample_prompt(self, fd):
+ # Read one sample prompt from fd
+ function_lines = []
+ result = {}
+ parsing_hashes = True
+ for line in fd:
+ # Parse hashes until parse_one_sample_prompt_hash return False
+ if parsing_hashes:
+ parsing_hashes = self.parse_one_sample_prompt_hash(line, result)
+ # Maybe not we're not parsing hashes, or maybe we already were not
+ if not parsing_hashes:
+ function_lines.append(line)
+ result['function'] = ''.join(function_lines).strip()
+ return result
+
+ def do_get_sample_prompts_list(self):
+ result = []
+ # Start with the "Current" meta-sample
+ result.append({'name': 'Current'})
+
+ # Read all of the prompts in sample_prompts
+ paths = glob.iglob('sample_prompts/*.fish')
+ for path in paths:
+ try:
+ fd = open(path)
+ result.append(self.read_one_sample_prompt(fd))
+ fd.close()
+ except IOError:
+ # Ignore unreadable files, etc
+ pass
+ return result
+
+ def font_size_for_ansi_prompt(self, prompt_demo_ansi):
+ width = ansi_prompt_line_width(prompt_demo_ansi)
+ # Pick a font size
+ if width >= 70: font_size = '8pt'
+ if width >= 60: font_size = '10pt'
+ elif width >= 50: font_size = '11pt'
+ elif width >= 40: font_size = '13pt'
+ elif width >= 30: font_size = '15pt'
+ elif width >= 25: font_size = '16pt'
+ elif width >= 20: font_size = '17pt'
+ else: font_size = '18pt'
+ return font_size
+
def do_GET(self):
p = self.path
@@ -267,6 +467,10 @@ class FishConfigHTTPRequestHandler(SimpleHTTPServer.SimpleHTTPRequestHandler):
output = self.do_get_history()
# end = time.time()
# print "History: ", end - start
+ elif p == '/current_prompt/':
+ output = self.do_get_current_prompt()
+ elif p == '/sample_prompts/':
+ output = self.do_get_sample_prompts_list()
elif re.match(r"/color/(\w+)/", p):
name = re.match(r"/color/(\w+)/", p).group(1)
output = self.do_get_color_for_variable(name)
@@ -317,12 +521,21 @@ class FishConfigHTTPRequestHandler(SimpleHTTPServer.SimpleHTTPRequestHandler):
elif p == '/get_function/':
what = postvars.get('what')
output = [self.do_get_function(what[0])]
+ elif p == '/get_sample_prompt/':
+ what = postvars.get('what')
+ output = [self.do_get_sample_prompt(what[0])]
elif p == '/delete_history_item/':
what = postvars.get('what')
if self.do_delete_history_item(what[0]):
output = ["OK"]
else:
output = ["Unable to delete history item"]
+ elif p == '/set_prompt/':
+ what = postvars.get('what')
+ if self.do_set_prompt_function(what[0]):
+ output = ["OK"]
+ else:
+ output = ["Unable to set prompt"]
else:
return SimpleHTTPServer.SimpleHTTPRequestHandler.do_POST(self)
@@ -338,6 +551,10 @@ class FishConfigHTTPRequestHandler(SimpleHTTPServer.SimpleHTTPRequestHandler):
""" Disable request logging """
pass
+# We want to show the demo prompts in the directory from which this was invoked,
+# so get the current working directory
+initial_wd = os.getcwd()
+
# Make sure that the working directory is the one that contains the script server file,
# because the document root is the working directory
where = os.path.dirname(sys.argv[0])
@@ -367,7 +584,7 @@ if PORT > 9000:
# Just look at the first letter
initial_tab = ''
if len(sys.argv) > 1:
- for tab in ['functions', 'colors', 'variables', 'history']:
+ for tab in ['functions', 'prompt', 'colors', 'variables', 'history']:
if tab.startswith(sys.argv[1]):
initial_tab = '#' + tab
break