]> git.plutz.net Git - shellwiki/commitdiff
define search function as macro
authorPaul Hänsch <paul@plutz.net>
Sun, 7 Apr 2024 13:12:55 +0000 (15:12 +0200)
committerPaul Hänsch <paul@plutz.net>
Sun, 7 Apr 2024 13:12:55 +0000 (15:12 +0200)
handlers/40_search.sh
macros/search [new file with mode: 0755]
pages/[wiki]/search/#page.md [new file with mode: 0644]
pages/[wiki]/search/:de/#page.md [new file with mode: 0644]
themes/default.css
themes/default.sh

index 57117e3834a1bbe0c816f92e785bc38c188bc186..3661f954b7e07c6665f419484bb00c2edb9e7a90 100644 (file)
 [ "$SEARCH_INDEX" != true ] && return 1
 [ "${PATH_INFO%\[search\]}" = "$PATH_INFO" ] && return 1
 
-. "$_EXEC/cgilite/storage.sh"
-. "$_EXEC/cgilite/db23.sh"
-
-I="$_DATA/index"
-tags="$( GET q | awk '
-  BEGIN { # Field separator FS should include punctuation, including Unicode Block U+2000 - U+206F
-          if ( length("¡") == 1 )  # Utf-8 aware AWK
-          FS = "([] \\t\\n\\r!\"'\''()*+,./:;<=>?\\\\^_`{|}~[-]|%[0-9A-Fa-f]{2}|'"$(printf '[\342\200\200-\342\201\257]')"')+";
-          else                     # UTF-8 Hack
-          FS = "([] \\t\\n\\r!\"'\''()*+,./:;<=>?\\\\^_`{|}~[-]|%[0-9A-Fa-f]{2}|'"$(printf '\342\200[\200-\277]|\342\201[\201-\257]')"')+";
-          fi
-        }
-  { for (n = 1; n <= NF; n++) if ($n ~ /#[[:alnum:]_]+/) {
-      sub(/^#/,"",$n)
-      printf "%s       ", toupper($n);
-    }
-  }
-')"
-
-ntags="$( GET q | awk '
-  BEGIN { # Field separator FS should include punctuation, including Unicode Block U+2000 - U+206F
-          if ( length("¡") == 1 )  # Utf-8 aware AWK
-          FS = "([] \\t\\n\\r\"#'\''()*+,./:;<=>?\\\\^_`{|}~[-]|%[0-9A-Fa-f]{2}|'"$(printf '[\342\200\200-\342\201\257]')"')+";
-          else                     # UTF-8 Hack
-          FS = "([] \\t\\n\\r\"#'\''()*+,./:;<=>?\\\\^_`{|}~[-]|%[0-9A-Fa-f]{2}|'"$(printf '\342\200[\200-\277]|\342\201[\201-\257]')"')+";
-          fi
-        }
-  { for (n = 1; n <= NF; n++) if ($n ~ /![[:alnum:]_]+/) {
-      sub(/^!/,"",$n)
-      printf "%s       ", toupper($n);
-    }
-  }
-')"
-
-words="$( GET q | awk '
-  BEGIN { # Field separator FS should include punctuation, including Unicode Block U+2000 - U+206F
-          if ( length("¡") == 1 )  # Utf-8 aware AWK
-          FS = "([] \\t\\n\\r\"#'\''()*+,./:;<=>?\\\\^_`{|}~[-]|%[0-9A-Fa-f]{2}|'"$(printf '[\342\200\200-\342\201\257]')"')+";
-          else                     # UTF-8 Hack
-          FS = "([] \\t\\n\\r\"#'\''()*+,./:;<=>?\\\\^_`{|}~[-]|%[0-9A-Fa-f]{2}|'"$(printf '\342\200[\200-\277]|\342\201[\201-\257]')"')+";
-          fi
-        }
-  { for (n = 1; n <= NF; n++) if ($n !~ /![[:alnum:]_]+/) {
-      sub(/!/," ",$n)
-      printf "%s       ", tolower($n);
-    }
-  }
-')"
-
-searchteaser() {
-  local file="$1" words db3_data
-  local w l nc nl hits mhits cont mcont
-  shift 1; words="$*"
-
-  for w in ${words}; do
-    grep -hiwnF "$w" "$file"
-  done \
-  | sort -t: -k1 -n \
-  | { nc=-1 hits=0 mhits=0
-    while read -r l; do
-      nl="$nc" nc="${l%%:*}"
-      if [ $nc -eq $nl ]; then
-        hits=$((hits + 1))
-      elif [ $nc -eq $((nl + 1 )) ]; then
-        hits=$((hits + 1))
-        cont="${cont}${BR}${l#*:}"
-      elif [ $hits -gt $mhits ]; then
-        mhits="$hits" mcont="$cont"
-        hits=1 cont="${l#*:}"
-      else
-        hits=1 cont="${l#*:}"
-      fi
-    done
-
-    [ $hits -gt $mhits ] \
-    && STRING "$cont" \
-    || STRING "$mcont"
-  }
-}
-
-for w in ${words}; do
-  [ ! -f "$I/$w" ] && continue
-
-  while read date doc freq num total; do
-    P="$_DATA/pages$(UNSTRING "$doc")"
-    d="$(stat -c %Y -- "$P/#index.flag" 2>&-)"
-    [ "$d" -le "$date" -a -f "$P/#page.md" ] 2>&- || continue
-
-    printf '%s %f\n' "$doc" "$freq"
-  done <"$I/$w"
-done \
-| awk '
-      { cnt[$1]++; weight[$1] = weight[$1] ? weight[$1] + $2 : $2; }
-  END { m = 0; for (d in cnt) m = ( m < cnt[d] ) ? cnt[d] : m;
-        for (d in cnt) if ( cnt[d] == m ) printf "%f   %s\n", weight[d], d;
-      }
-' \
-| sort -nr \
-| while read freq doc; do
-  page="$(UNSTRING "$doc")"
-  [ "${page%*/\[*\]/*}" != "$page" ] && continue
-  if [ "$LANGUAGE_DEFAULT" ]; then
-    [ -d "${_DATA}/pages/${page}/:${LANGUAGE}/" ] && continue
-    [ "${page%/:*/}" = "${page%/:${LANGUAGE}/}" ] || continue
-  fi
-  acl_read "$page" || continue
-  has_tags "$page" $tags || continue
-  has_tag "$page" $ntags && continue
-  printf '%s   %s\n' "$doc" "$(searchteaser "$(mdfile "$page")" $words)"
-done \
-| theme_search "${words%       }"
+theme_page "/[wiki]/search/" "$(page_title "/[wiki]/search/")"
+return 0
diff --git a/macros/search b/macros/search
new file mode 100755 (executable)
index 0000000..b3d06e8
--- /dev/null
@@ -0,0 +1,148 @@
+#!/bin/sh
+
+# Copyright 2023, 2024 Paul Hänsch
+# 
+# Permission to use, copy, modify, and/or distribute this software for any
+# purpose with or without fee is hereby granted, provided that the above
+# copyright notice and this permission notice appear in all copies.
+# 
+# THE SOFTWARE IS PROVIDED “AS IS” AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
+# SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR
+# IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+
+I="$_DATA/index"
+
+. "$_EXEC/cgilite/cgilite.sh"
+. "$_EXEC/cgilite/storage.sh"
+. "$_EXEC/cgilite/db23.sh"
+. "$_EXEC/tools.sh"
+. "$_EXEC/acl.sh"
+
+_(){ printf %s\\n "$*"; }
+[ "${LANGUAGE}" -a -r "${_EXEC}/l10n/${LANGUAGE}.sh" ] && . "${_EXEC}/l10n/${LANGUAGE}.sh"
+
+show_form=true show_hits='' action=''
+
+set -- "$@" --
+while [ $# -gt 0 ]; do case $1 in
+  --no-form|--noform)
+    show_form=''; shift 1;;
+  --hits|--results)
+    show_hits='true'; shift 1;;
+  --action)
+    action="$(HTML "$2")"; shift 2;;
+  --action=*|action=*)
+    action="$(HTML "${2#*=}")"; shift 1;;
+  --) shift 1; break;;
+  *) set -- "$@" "$1"; shift 1;;
+esac; done
+
+searchteaser() {
+  local file="$1" words db3_data
+  local w l nc nl hits mhits cont mcont
+  shift 1; words="$*"
+
+  for w in ${words}; do
+    grep -hiwnF "$w" "$file"
+  done \
+  | sort -t: -k1 -n \
+  | { nc=-1 hits=0 mhits=0
+    while read -r l; do
+      nl="$nc" nc="${l%%:*}"
+      if [ $nc -eq $nl ]; then
+        hits=$((hits + 1))
+      elif [ $nc -eq $((nl + 1 )) ]; then
+        hits=$((hits + 1))
+        cont="${cont}${BR}${l#*:}"
+      elif [ $hits -gt $mhits ]; then
+        mhits="$hits" mcont="$cont"
+        hits=1 cont="${l#*:}"
+      else
+        hits=1 cont="${l#*:}"
+      fi
+    done
+
+    [ $hits -gt $mhits ] \
+    && STRING "$cont" \
+    || STRING "$mcont"
+  }
+}
+
+if [ ! "$action" -a "$LANGUAGE_DEFAULT" ]; then
+  action="./:${LANGUAGE}/[search]"
+elif [ ! "$action" ]; then
+  action="./[search]"
+fi
+
+if [ "${show_form}" = true ]; then
+  printf '<form class="macro search" method="GET" action="%s">
+  <input type="search" placeholder="%s" name="q" value="%s"><button type="submit" class="search">%s</button>
+</form>' "$action" "$(_ Search)" "$([ "$show_hits" = true ] && GET q |HTML)" "$(_ Search)"
+fi
+
+if [ "${show_hits}" = true ]; then
+  { read tags; read ntags; read words; } <<-EOF
+       $(GET q | awk '
+       BEGIN { # Field separator FS should include punctuation, including Unicode Block U+2000 - U+206F
+               if ( length("¡") == 1 )  # Utf-8 aware AWK
+               FS = "([] \\t\\n\\r\"'\''()*+,./:;<=>?\\\\^_`{|}~[-]|%[0-9A-Fa-f]{2}|'"$(printf '[\342\200\200-\342\201\257]')"')+";
+               else                     # UTF-8 Hack
+               FS = "([] \\t\\n\\r\"'\''()*+,./:;<=>?\\\\^_`{|}~[-]|%[0-9A-Fa-f]{2}|'"$(printf '\342\200[\200-\277]|\342\201[\201-\257]')"')+";
+               fi
+             }
+       { t=0; for (n = 1; n <= NF; n++) if ($n  ~ /#[[:alnum:]_]+/)  tags[t++] = toupper($n);
+         t=0; for (n = 1; n <= NF; n++) if ($n  ~ /![[:alnum:]_]+/) ntags[t++] = toupper($n);
+         t=0; for (n = 1; n <= NF; n++) if ($n !~ /![[:alnum:]_]+/) words[t++] = tolower($n);
+         for (t in  tags) { sub(/^#/, "",  tags[t]); printf "%s        ",  tags[t]; } print "";
+         for (t in ntags) { sub(/^!/, "", ntags[t]); printf "%s        ", ntags[t]; } print "";
+         for (t in words) { sub(/^[!#]/, "", words[t]); printf "%s     ", words[t]; } print "";
+       }
+       ')
+       EOF
+
+  printf '<ul class="macro search hits">'
+  for w in ${words}; do
+    [ ! -f "$I/$w" ] && continue
+  
+    while read date doc freq num total; do
+      P="$_DATA/pages$(UNSTRING "$doc")"
+      d="$(stat -c %Y -- "$P/#index.flag" 2>&-)"
+      [ "$d" -le "$date" -a -f "$P/#page.md" ] 2>&- || continue
+  
+      printf '%s       %f\n' "$doc" "$freq"
+    done <"$I/$w"
+  done \
+  | awk '
+        { cnt[$1]++; weight[$1] = weight[$1] ? weight[$1] + $2 : $2; }
+    END { m = 0; for (d in cnt) m = ( m < cnt[d] ) ? cnt[d] : m;
+          for (d in cnt) if ( cnt[d] == m ) printf "%f %s\n", weight[d], d;
+        }
+  ' \
+  | sort -nr \
+  | while read freq doc; do
+    page="$(UNSTRING "$doc")"
+    [ "${page%*/\[*\]/*}" != "$page" ] && continue
+    if [ "$LANGUAGE_DEFAULT" ]; then
+      [ -d "${_DATA}/pages/${page}/:${LANGUAGE}/" ] && continue
+      [ "${page%/:*/}" = "${page%/:${LANGUAGE}/}" ] || continue
+    fi
+    acl_read "$page" || continue
+    has_tags "$page" $tags || continue
+    has_tag "$page" $ntags && continue
+    printf '%s %s\n' "$doc" "$(searchteaser "$(mdfile "$page")" $words)"
+  done \
+  | while read -r p t; do
+    path="$(UNSTRING "$p")" pfrag="${path%/}" title=''
+    while [ "$pfrag" ]; do
+      title="$(page_title "$pfrag")/$title"
+      pfrag="${pfrag%/*}"
+    done
+    printf '<li><a href="%s">%s</a><p>%s</p></li>' \
+      "$(URL "$path")" "$(HTML "/$title")" "$(UNSTRING "$t" |HTML)"
+  done
+  printf '</ul>'
+fi
diff --git a/pages/[wiki]/search/#page.md b/pages/[wiki]/search/#page.md
new file mode 100644 (file)
index 0000000..598bbb8
--- /dev/null
@@ -0,0 +1,6 @@
+%nocache
+%title Search
+
+Search Results
+==============
+<<search --action ./[search] --hits>>
diff --git a/pages/[wiki]/search/:de/#page.md b/pages/[wiki]/search/:de/#page.md
new file mode 100644 (file)
index 0000000..0a3ae08
--- /dev/null
@@ -0,0 +1,6 @@
+%nocache
+%title Suche
+
+Suchergebnisse
+==============
+<<search --action ./[search] --hits>>
index e0f7022182b509dfb54737561a760aaf70ba51fd..c7eb6dd0e2c872a1c2e64af30c46557e33cf9d1e 100644 (file)
@@ -173,15 +173,16 @@ input.search, input[type="search"] {
   max-width: 80%;
   max-width: calc(100%  - 2.5em);
 }
-ul.searchresults, ol.searchresults {
+ul.search.hits, ol.search.hits {
   margin-left: auto; margin-right: auto;
   width: 100%; max-width: 540pt;
   text-align: center;
+  list-style: none;
 }
-.searchresults li a {
+.search.hits li a {
   display: block;
 }
-.searchresults li p {
+.search.hits li p {
   display: inline-block;
   margin: 0 auto .5em auto;
   white-space: pre-line;
@@ -240,7 +241,7 @@ table {
     width: 80%;
     width: calc(100% - 2.5em);
   }
-  ul.searchresults, ol.searchresults {
+  ul.search.hits, ol.search.hits {
     min-width: 50%;
   }
 }
index d13e849f72ece086e4fb9a456f2fbe2419ea5f3d..5ebbd0ef6964180a146ea96b10ddf5a2ee5658ec 100755 (executable)
@@ -135,31 +135,6 @@ theme_editor(){
 
 theme_revisions(){ theme_page "$@"; }
 
-theme_search(){
-  local words="$*"
-  # STDIN: [STRING page][TAB][STRING teaser]
-
-  theme_page - "$(_ Search results): ${words}" <<-EOF
-       <article>
-         <h1>$([ "$words" ] && _ "Search results" || _ "Search" )</h1>
-         <form class="search" method="GET">
-           <input type="search" name="q" value="$(GET q |HTML)"><button class="search" type="submit">$(_ Search)</button>
-         </form>
-         <ol class="searchresults">
-           $(while read -r p t; do
-             path="$(UNSTRING "$p")" pfrag="${path%/}" title=''
-              while [ "$pfrag" ]; do
-                title="$(page_title "$pfrag")/$title"
-                pfrag="${pfrag%/*}"
-              done
-             printf '<li><a href="%s">%s</a><p>%s</p></li>' \
-                "$(URL "$path")" "$(HTML "/$title")" "$(UNSTRING "$t" |HTML)"
-           done)
-         </ol>
-       </article>
-       EOF
-}
-
 theme_attachments(){
   local page="$1"