|
|
- #!/usr/bin/env sh
- # surf_linkselect.sh:
- # Usage: curl somesite.com | surf_linkselect [SURFWINDOWID] [PROMPT]
- # Deps: xmllint, dmenu
- # Info:
- # Designed to be used w/ surf externalpipe patch. Enables keyboard-only
- # link selection via dmenu. Given HTML stdin, extracts links one per line
- # Selected link is normalized based on current URI and printed to STDOUT.
- # Pipe the result to a new surf or xprop _SURF_URI accordingly.
- SURF_WINDOW="${1:-$(xprop -root | sed -n '/^_NET_ACTIVE_WINDOW/ s/.* //p')}"
- DMENU_PROMPT="${2:-Link}"
-
- function dump_links_with_titles() {
- awk '{
- input = $0;
-
- $0 = input;
- gsub("<[^>]*>", "");
- gsub(/[ ]+/, " ");
- gsub("&", "\\&");
- gsub("<", "<");
- gsub(">", ">");
- $1 = $1;
- title = ($0 == "" ? "None" : $0);
-
- $0 = input;
- match($0, /\<[ ]*[aA][^>]* [hH][rR][eE][fF]=["]([^"]+)["]/, linkextract);
- $0 = linkextract[1];
- gsub(/^[ \t]+/,"");
- gsub(/[ \t]+$/,"");
- gsub("[ ]", "%20");
- link = $0;
-
- if (link != "") {
- print title ": " link;
- }
- }'
- }
-
- function link_normalize() {
- URI=$1
- awk -v uri=$URI '{
- gsub("&", "\\&");
-
- if ($0 ~ /^https?:\/\// || $0 ~ /^\/\/.+$/) {
- print $0;
- } else if ($0 ~/^#/) {
- gsub(/[#?][^#?]+/, "", uri);
- print uri $0;
- } else if ($0 ~/^\//) {
- split(uri, uri_parts, "/");
- print uri_parts[3] $0;
- } else {
- gsub(/[#][^#]+/, "", uri);
- uri_parts_size = split(uri, uri_parts, "/");
- delete uri_parts[uri_parts_size];
- for (v in uri_parts) {
- uri_pagestripped = uri_pagestripped uri_parts[v] "/"
- }
- print uri_pagestripped $0;
- }
- }'
- }
-
- function link_select() {
- tr '\n\r' ' ' |
- xmllint --html --xpath "//a" - |
- dump_links_with_titles |
- awk '!x[$0]++' |
- # sort | uniq
- dmenu -p "$DMENU_PROMPT" -l 10 -i -w $SURF_WINDOW |
- awk -F' ' '{print $NF}' |
- link_normalize $(xprop -id $SURF_WINDOW _SURF_URI | cut -d '"' -f 2)
- }
-
- link_select
|