diff --git a/files.go b/files.go index 931daaa..fb849cd 100755 --- a/files.go +++ b/files.go @@ -41,8 +41,8 @@ func initializeTorrentSites() { func handleFileSearch(w http.ResponseWriter, settings UserSettings, query string, page int) { startTime := time.Now() - cacheKey := CacheKey{Query: query, Page: page, Safe: settings.SafeSearch == "active", Lang: settings.Language, Type: "file"} - combinedResults := getFileResultsFromCacheOrFetch(cacheKey, query, settings.SafeSearch, settings.Language, page) + cacheKey := CacheKey{Query: query, Page: page, Safe: settings.SafeSearch == "active", Lang: settings.SearchLanguage, Type: "file"} + combinedResults := getFileResultsFromCacheOrFetch(cacheKey, query, settings.SafeSearch, settings.SearchLanguage, page) sort.Slice(combinedResults, func(i, j int) bool { return combinedResults[i].Seeders > combinedResults[j].Seeders }) @@ -82,7 +82,7 @@ func handleFileSearch(w http.ResponseWriter, settings UserSettings, query string HasPrevPage: page > 1, HasNextPage: len(combinedResults) > 0, LanguageOptions: languageOptions, - CurrentLang: settings.Language, + CurrentLang: settings.SearchLanguage, Theme: settings.Theme, Safe: settings.SafeSearch, IsThemeDark: settings.IsThemeDark, diff --git a/forums.go b/forums.go index 4c15490..e9fa982 100755 --- a/forums.go +++ b/forums.go @@ -102,7 +102,7 @@ func handleForumsSearch(w http.ResponseWriter, settings UserSettings, query stri results, err := PerformRedditSearch(query, settings.SafeSearch, page) if err != nil || len(results) == 0 { // 0 == 0 to force search by other node log.Printf("No results from primary search, trying other nodes") - results = tryOtherNodesForForumSearch(query, settings.SafeSearch, settings.Language, page) + results = tryOtherNodesForForumSearch(query, settings.SafeSearch, settings.SearchLanguage, page) } data := struct { @@ -123,7 +123,7 @@ func handleForumsSearch(w http.ResponseWriter, settings UserSettings, query stri HasPrevPage: page > 1, HasNextPage: len(results) == 25, LanguageOptions: languageOptions, - CurrentLang: settings.Language, + CurrentLang: settings.SearchLanguage, Theme: settings.Theme, Safe: settings.SafeSearch, IsThemeDark: settings.IsThemeDark, diff --git a/go.mod b/go.mod index f6755b1..a747de0 100644 --- a/go.mod +++ b/go.mod @@ -23,6 +23,9 @@ require ( ) require ( + github.com/chai2010/webp v1.1.1 // indirect + github.com/disintegration/imaging v1.6.2 // indirect github.com/fsnotify/fsnotify v1.7.0 // indirect + golang.org/x/image v0.20.0 // indirect gopkg.in/ini.v1 v1.67.0 // indirect ) diff --git a/go.sum b/go.sum index 7fab23f..4120d74 100644 --- a/go.sum +++ b/go.sum @@ -2,12 +2,16 @@ github.com/PuerkitoBio/goquery v1.9.1 h1:mTL6XjbJTZdpfL+Gwl5U2h1l9yEkJjhmlTeV9VP github.com/PuerkitoBio/goquery v1.9.1/go.mod h1:cW1n6TmIMDoORQU5IU/P1T3tGFunOeXEpGP2WHRwkbY= github.com/andybalholm/cascadia v1.3.2 h1:3Xi6Dw5lHF15JtdcmAHD3i1+T8plmv7BQ/nsViSLyss= github.com/andybalholm/cascadia v1.3.2/go.mod h1:7gtRlve5FxPPgIgX36uWBX58OdBsSS6lUvCFb+h7KvU= +github.com/chai2010/webp v1.1.1 h1:jTRmEccAJ4MGrhFOrPMpNGIJ/eybIgwKpcACsrTEapk= +github.com/chai2010/webp v1.1.1/go.mod h1:0XVwvZWdjjdxpUEIf7b9g9VkHFnInUSYujwqTLEuldU= github.com/chromedp/cdproto v0.0.0-20240202021202-6d0b6a386732 h1:XYUCaZrW8ckGWlCRJKCSoh/iFwlpX316a8yY9IFEzv8= github.com/chromedp/cdproto v0.0.0-20240202021202-6d0b6a386732/go.mod h1:GKljq0VrfU4D5yc+2qA6OVr8pmO/MBbPEWqWQ/oqGEs= github.com/chromedp/chromedp v0.9.5 h1:viASzruPJOiThk7c5bueOUY91jGLJVximoEMGoH93rg= github.com/chromedp/chromedp v0.9.5/go.mod h1:D4I2qONslauw/C7INoCir1BJkSwBYMyZgx8X276z3+Y= github.com/chromedp/sysutil v1.0.0 h1:+ZxhTpfpZlmchB58ih/LBHX52ky7w2VhQVKQMucy3Ic= github.com/chromedp/sysutil v1.0.0/go.mod h1:kgWmDdq8fTzXYcKIBqIYvRRTnYb9aNS9moAV0xufSww= +github.com/disintegration/imaging v1.6.2 h1:w1LecBlG2Lnp8B3jk5zSuNqd7b4DXhcjwek1ei82L+c= +github.com/disintegration/imaging v1.6.2/go.mod h1:44/5580QXChDfwIclfc/PCwrr44amcmDAg8hxG0Ewe4= github.com/fsnotify/fsnotify v1.7.0 h1:8JEhPFa5W2WU7YfeZzPNqzMP6Lwt7L2715Ggo0nosvA= github.com/fsnotify/fsnotify v1.7.0/go.mod h1:40Bi/Hjc2AVfZrqy+aj+yEI+/bRxZnMJyTJwOpGvigM= github.com/go-ole/go-ole v1.2.6 h1:/Fpf6oFPoeFik9ty7siob0G6Ke8QvQEuVcuChpwXzpY= @@ -31,6 +35,10 @@ github.com/yusufpapurcu/wmi v1.2.4 h1:zFUKzehAFReQwLys1b/iSMl+JQGSCSjtVqQn9bBrPo github.com/yusufpapurcu/wmi v1.2.4/go.mod h1:SBZ9tNy3G9/m5Oi98Zks0QjeHVDvuK0qfxQmPyzfmi0= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= +golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8 h1:hVwzHzIUGRjiF7EcUjqNxk3NCfkPxbDKRdnNE1Rpg0U= +golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= +golang.org/x/image v0.20.0 h1:7cVCUjQwfL18gyBJOmYvptfSHS8Fb3YUDtfLIZ7Nbpw= +golang.org/x/image v0.20.0/go.mod h1:0a88To4CYVBAHp5FXJm8o7QbUl37Vd85ply1vyD8auM= golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= diff --git a/imageproxy.go b/imageproxy.go index 74a1cbb..4dd7478 100644 --- a/imageproxy.go +++ b/imageproxy.go @@ -1,7 +1,6 @@ package main import ( - "fmt" "io" "net/http" ) @@ -14,20 +13,12 @@ func handleImageProxy(w http.ResponseWriter, r *http.Request) { return } - // Try to fetch the image from Bing first - bingURL := fmt.Sprintf("https://tse.mm.bing.net/th?q=%s", imageURL) - resp, err := http.Get(bingURL) - if err != nil || resp.StatusCode != http.StatusOK { - // If fetching from Bing fails, attempt to fetch from the original image URL - printWarn("Error fetching image from Bing, trying original URL.") - - // Attempt to fetch the image directly - resp, err = http.Get(imageURL) - if err != nil { - printWarn("Error fetching image: %v", err) - http.Error(w, "Internal Server Error", http.StatusInternalServerError) - return - } + // Fetch the image from the external URL + resp, err := http.Get(imageURL) + if err != nil { + printWarn("Error fetching image: %v", err) + http.Error(w, "Internal Server Error", http.StatusInternalServerError) + return } defer resp.Body.Close() diff --git a/images-bing.go b/images-bing.go index d6e37c5..48ba32d 100644 --- a/images-bing.go +++ b/images-bing.go @@ -12,6 +12,7 @@ import ( "github.com/PuerkitoBio/goquery" ) +// PerformBingImageSearch performs a Bing image search and returns the results. func PerformBingImageSearch(query, safe, lang string, page int) ([]ImageSearchResult, time.Duration, error) { startTime := time.Now() @@ -66,14 +67,16 @@ func PerformBingImageSearch(query, safe, lang string, page int) ([]ImageSearchRe if err := json.Unmarshal([]byte(metadata), &data); err == nil { mediaURL, ok := data["murl"].(string) if ok { + // Apply the image proxy + proxiedURL := "/imgproxy?url=" + mediaURL results = append(results, ImageSearchResult{ Thumbnail: imgSrc, Title: strings.TrimSpace(title), Media: mediaURL, + Source: mediaURL, + ThumbProxy: proxiedURL, // Use the proxied URL Width: width, Height: height, - Source: mediaURL, - ThumbProxy: imgSrc, }) } } @@ -89,6 +92,7 @@ func PerformBingImageSearch(query, safe, lang string, page int) ([]ImageSearchRe return results, duration, nil } +// buildBingSearchURL constructs the search URL for Bing Image Search func buildBingSearchURL(query string, page int) string { baseURL := "https://www.bing.com/images/search" params := url.Values{} @@ -99,6 +103,7 @@ func buildBingSearchURL(query string, page int) string { return baseURL + "?" + params.Encode() } +// Example usage in main (commented out for clarity) // func main() { // results, duration, err := PerformBingImageSearch("kittens", "false", "en", 1) // if err != nil { diff --git a/images-deviantart.go b/images-deviantart.go index cbeeea4..3b12def 100644 --- a/images-deviantart.go +++ b/images-deviantart.go @@ -156,7 +156,7 @@ func PerformDeviantArtImageSearch(query, safe, lang string, page int) ([]ImageSe Width: 0, Height: 0, Source: resultURL, - ThumbProxy: imgSrc, + ThumbProxy: "/imgproxy?url=" + imgSrc, } } }(imgSrc, resultURL, title) diff --git a/images-quant.go b/images-quant.go index 0dccdec..8690239 100644 --- a/images-quant.go +++ b/images-quant.go @@ -5,6 +5,7 @@ import ( "fmt" "net/http" "net/url" + "sync" "time" ) @@ -141,19 +142,37 @@ func PerformQwantImageSearch(query, safe, lang string, page int) ([]ImageSearchR return nil, 0, fmt.Errorf("decoding response: %v", err) } - var results []ImageSearchResult - for _, item := range apiResp.Data.Result.Items { - results = append(results, ImageSearchResult{ - Thumbnail: item.Thumbnail, - Title: item.Title, - Media: item.Media, - Source: item.Url, - ThumbProxy: "/img_proxy?url=" + url.QueryEscape(item.Media), // New proxy not exactly working as intended - Width: item.Width, - Height: item.Height, - }) + var wg sync.WaitGroup + results := make([]ImageSearchResult, len(apiResp.Data.Result.Items)) + + for i, item := range apiResp.Data.Result.Items { + wg.Add(1) + go func(i int, item struct { + Media string `json:"media"` + Thumbnail string `json:"thumbnail"` + Title string `json:"title"` + Url string `json:"url"` + Width int `json:"width"` + Height int `json:"height"` + }) { + defer wg.Done() + + // Populate the result + results[i] = ImageSearchResult{ + Thumbnail: item.Thumbnail, + Title: item.Title, + Media: item.Media, + Source: item.Url, + ThumbProxy: "/imgproxy?url=" + item.Media, + Width: item.Width, + Height: item.Height, + } + }(i, item) } + // Wait for all goroutines to complete + wg.Wait() + duration := time.Since(startTime) // Calculate the duration return results, duration, nil diff --git a/images.go b/images.go index 4c6b957..7bd55a4 100755 --- a/images.go +++ b/images.go @@ -13,17 +13,17 @@ var imageSearchEngines []SearchEngine func init() { imageSearchEngines = []SearchEngine{ {Name: "Qwant", Func: wrapImageSearchFunc(PerformQwantImageSearch), Weight: 1}, - {Name: "DeviantArt", Func: wrapImageSearchFunc(PerformDeviantArtImageSearch), Weight: 2}, - {Name: "Bing", Func: wrapImageSearchFunc(PerformBingImageSearch), Weight: 2}, // Bing sometimes returns with low amount of images, this leads to danamica page loading not working - {Name: "Imgur", Func: wrapImageSearchFunc(PerformImgurImageSearch), Weight: 3}, + {Name: "Bing", Func: wrapImageSearchFunc(PerformBingImageSearch), Weight: 2}, + {Name: "DeviantArt", Func: wrapImageSearchFunc(PerformDeviantArtImageSearch), Weight: 3}, + //{Name: "Imgur", Func: wrapImageSearchFunc(PerformImgurImageSearch), Weight: 4}, // Image proxy not working } } func handleImageSearch(w http.ResponseWriter, settings UserSettings, query string, page int) { startTime := time.Now() - cacheKey := CacheKey{Query: query, Page: page, Safe: settings.SafeSearch == "active", Lang: settings.Language, Type: "image"} - combinedResults := getImageResultsFromCacheOrFetch(cacheKey, query, settings.SafeSearch, settings.Language, page) + cacheKey := CacheKey{Query: query, Page: page, Safe: settings.SafeSearch == "active", Lang: settings.SearchLanguage, Type: "image"} + combinedResults := getImageResultsFromCacheOrFetch(cacheKey, query, settings.SafeSearch, settings.SearchLanguage, page) elapsedTime := time.Since(startTime) tmpl, err := template.New("images.html").Funcs(funcs).ParseFiles("templates/images.html") @@ -55,7 +55,7 @@ func handleImageSearch(w http.ResponseWriter, settings UserSettings, query strin HasNextPage: len(combinedResults) >= 50, NoResults: len(combinedResults) == 0, LanguageOptions: languageOptions, - CurrentLang: settings.Language, + CurrentLang: settings.SearchLanguage, Theme: settings.Theme, Safe: settings.SafeSearch, IsThemeDark: settings.IsThemeDark, diff --git a/main.go b/main.go index 63b60ca..e3b102e 100755 --- a/main.go +++ b/main.go @@ -17,7 +17,7 @@ type LanguageOption struct { var settings UserSettings var languageOptions = []LanguageOption{ - {Code: "", Name: "Auto-detect"}, + {Code: "", Name: "Any Language"}, {Code: "en", Name: "English"}, {Code: "af", Name: "Afrikaans"}, {Code: "ar", Name: "العربية (Arabic)"}, @@ -69,35 +69,37 @@ var languageOptions = []LanguageOption{ func handleSearch(w http.ResponseWriter, r *http.Request) { query, safe, lang, searchType, page := parseSearchParams(r) - // Load user settings settings = loadUserSettings(w, r) - // Update theme if provided, or use existing settings theme := r.URL.Query().Get("theme") if theme != "" { settings.Theme = theme - saveUserSettings(w, settings) // Save if theme is updated + saveUserSettings(w, settings) } else if settings.Theme == "" { - settings.Theme = "dark" // Default theme + settings.Theme = "dark" } - // Update safe search if provided, or use existing settings if safe != "" && safe != settings.SafeSearch { settings.SafeSearch = safe - saveUserSettings(w, settings) // Save if safe search is updated + saveUserSettings(w, settings) } - // Update language if provided, or use existing settings - if lang != "" && lang != settings.Language { - settings.Language = lang - saveUserSettings(w, settings) // Save if language is updated - } else if settings.Language == "" { - // If no language set, auto-detect from browser or default to "en" - settings.Language = normalizeLangCode(r.Header.Get("Accept-Language")) - saveUserSettings(w, settings) // Save if language is auto-detected + // Update site language if provided, or use existing settings + if lang != "" && lang != settings.SiteLanguage { + settings.SiteLanguage = lang + saveUserSettings(w, settings) + } else if settings.SiteLanguage == "" { + settings.SiteLanguage = normalizeLangCode(r.Header.Get("Accept-Language")) + saveUserSettings(w, settings) + } + + // Update search language (can be empty) + searchLang := r.URL.Query().Get("search_lang") + if searchLang != settings.SearchLanguage { + settings.SearchLanguage = searchLang + saveUserSettings(w, settings) } - // This will do for now (to handle Dark Reader addon) switch settings.Theme { case "dark", "black", "night", "latte": settings.IsThemeDark = true @@ -105,21 +107,21 @@ func handleSearch(w http.ResponseWriter, r *http.Request) { settings.IsThemeDark = false } - // Check if there is a search query if query == "" { - // If no query is provided, render the search page template data := struct { - LanguageOptions []LanguageOption - CurrentLang string - Theme string - Safe string - IsThemeDark bool + LanguageOptions []LanguageOption + CurrentLang string + CurrentSearchLang string + Theme string + Safe string + IsThemeDark bool }{ - LanguageOptions: languageOptions, - CurrentLang: settings.Language, - Theme: settings.Theme, - Safe: settings.SafeSearch, - IsThemeDark: settings.IsThemeDark, + LanguageOptions: languageOptions, + CurrentLang: settings.SiteLanguage, + CurrentSearchLang: settings.SearchLanguage, + Theme: settings.Theme, + Safe: settings.SafeSearch, + IsThemeDark: settings.IsThemeDark, } tmpl := template.Must(template.ParseFiles("templates/search.html")) @@ -127,7 +129,6 @@ func handleSearch(w http.ResponseWriter, r *http.Request) { return } - // Handle search based on the type switch searchType { case "image": handleImageSearch(w, settings, query, page) @@ -183,7 +184,7 @@ func runServer() { http.HandleFunc("/", handleSearch) http.HandleFunc("/search", handleSearch) http.HandleFunc("/suggestions", handleSuggestions) - http.HandleFunc("/img_proxy", handleImageProxy) + http.HandleFunc("/imgproxy", handleImageProxy) http.HandleFunc("/node", handleNodeRequest) http.HandleFunc("/settings", handleSettings) http.HandleFunc("/save-settings", handleSaveSettings) diff --git a/static/css/style-search.css b/static/css/style-search.css new file mode 100644 index 0000000..adda3d2 --- /dev/null +++ b/static/css/style-search.css @@ -0,0 +1,61 @@ + + +.search-page-content { + display: flex; + flex-direction: column; + align-items: center; +} + +.search-page-content h1 { + text-align: center; + margin-bottom: 20px; +} + +#search-input { + width: 100%; + padding: 12px; + font-size: 16px; +} + +.search-type-icons { + display: flex; + justify-content: center; + flex-wrap: wrap; + gap: 30px; + margin-top: 30px; +} + +.icon-button { + display: flex; + flex-direction: column; + align-items: center; + background: none; + border: none; + cursor: pointer; + text-align: center; +} + +.icon-button .material-icons-round { + font-size: 48px; + color: var(--sub-search-wrapper-ico); +} + +.icon-button p { + margin-top: 8px; + font-size: 14px; + color: var(--sub-search-wrapper-ico); +} + +.icon-button:hover .material-icons-round { + transition: all .3s ease; + color: var(--blue); +} + +.icon-button:hover p { + transition: all .3s ease; + color: var(--blue); +} + +.icon-button button:focus { + outline: none; +} \ No newline at end of file diff --git a/static/css/style-settings.css b/static/css/style-settings.css index 0dda333..81a40b6 100644 --- a/static/css/style-settings.css +++ b/static/css/style-settings.css @@ -53,10 +53,48 @@ gap: 10px; } -@media (max-width: 600px) { - .theme-link { - width: 100%; - } +#searchLanguageSelect, +#safeSearchSelect, +#siteLanguageSelect { + border-radius: 4px; + padding: 6px; + font-size: 15px; + border: 1px solid var(--border); + color: var(--font-fg); + width: 160px; + background: var(--button); + float: right; + cursor: pointer; + transition: all 0.3s ease; + text-align: center; + box-sizing: border-box; /* Ensures consistent width with padding */ +} + +#searchLanguageSelect:hover, +#safeSearchSelect:hover, +#siteLanguageSelect:hover { + border: 1px solid #5f6368; + /* background-color: var(--button-hover); */ +} + +.save.save-settings-page { + padding: 6px; + width: 160px; + height: 40px; +} + +/* Ensure correct aligment */ +.settings-row { + display: flex; + align-items: center; + justify-content: space-between; +} + +.settings-row select, +.settings-row button { + width: 160px; + height: 40px; + box-sizing: border-box; } /* --- */ diff --git a/static/css/style.css b/static/css/style.css index 782163e..e7a394c 100644 --- a/static/css/style.css +++ b/static/css/style.css @@ -67,13 +67,10 @@ /* Support for all WebKit browsers. */ -webkit-font-feature-settings: 'liga'; -webkit-font-smoothing: antialiased; - /* Support for Safari and Chrome. */ text-rendering: optimizeLegibility; - /* Support for Firefox. */ -moz-osx-font-smoothing: grayscale; - /* Support for IE. */ font-feature-settings: 'liga'; } @@ -549,8 +546,9 @@ hr { margin: 0 auto; background: var(--search-bg-input); border-radius: 22px; - position: absolute; - width: 520px; + position: relative; + width: 100%; + max-width: 600px; overflow: hidden; margin-left: auto; margin-right: auto; @@ -654,7 +652,7 @@ hr { .settings-nav { max-width: 100%; - height: 40px; + height: 50px; background-color: var(--search-bg); border-bottom: 1px solid var(--border); padding: 10px; @@ -674,6 +672,7 @@ hr { .settings-row { display: flex; align-items: center; + justify-content: flex-end; padding: 2px; padding-left: 10px; padding-right: 10px; @@ -682,7 +681,9 @@ hr { .settings-row select, .settings-row button { - margin-left: auto; + width: 160px; + height: 40px; + margin: 0; } .kno_wiki { @@ -773,6 +774,8 @@ form.torrent-sort { #settingsButton { transition: all .3s ease; + /* width: 283px+6px; + height: 31px; */ } .settings-icon-link { @@ -824,6 +827,12 @@ form.torrent-sort { cursor: pointer; } +.search-menu select:hover { + border: 1px solid #5f6368; + cursor: pointer; + transition: all .3s ease; +} + .settings-content { display: flex; flex-direction: column; @@ -867,7 +876,7 @@ form.torrent-sort { .theme-settings { margin-top: 10px; - width: 90%; + width: 100%-4px; border: 1px solid var(--snip-border); background: var(--snip-background); color: var(--fg); @@ -881,6 +890,10 @@ form.torrent-sort { margin-left: 3%; } +.theme-mini-settings { + width: 90%; +} + .settings-search-div:hover p { color: #8ab4f8; } @@ -1158,7 +1171,7 @@ p { position: absolute; margin-top: 0px; top: 20px; - left: 38px; + left: 28px; } .sub-search-button-wrapper button { @@ -1540,6 +1553,20 @@ body, h1, p, a, input, button { } } +/* +.material-icons-round { + font-family: 'Material Icons Round'; + font-weight: normal; + font-style: normal; + line-height: 1; + letter-spacing: normal; + text-transform: none; + display: inline-block; + white-space: nowrap; + word-wrap: normal; + direction: ltr; +} */ + @media only screen and (max-width: 1220px) { .snip { @@ -1864,12 +1891,32 @@ body, h1, p, a, input, button { display: none; } + .icon-button { + margin-top: 30px; + } + + .icon-button button { + margin-top: 30px; + } + + .icon-button p { + margin-top: 30px; + } + #clearSearch { top: 6px; } } + /* This is really bad */ +@media only screen and (max-width: 400px) { + + .icon-button { + padding: 5% + } +} + /* Ensuring dark theme compliance */ @media (prefers-color-scheme: dark) { .leaflet-control-locate, @@ -1891,6 +1938,7 @@ body, h1, p, a, input, button { color: var(--link) !important; } } + /* :root { --background-color: #ffffff; diff --git a/static/js/autocomplete.js b/static/js/autocomplete.js index 3e1cb58..354f254 100644 --- a/static/js/autocomplete.js +++ b/static/js/autocomplete.js @@ -52,6 +52,7 @@ async function getSuggestions(query) { let currentIndex = -1; // Keep track of the currently selected suggestion +// Handle click events on the type buttons let results = []; searchInput.addEventListener('input', async () => { let input = searchInput.value; @@ -128,15 +129,6 @@ function renderResults(results) { resultsWrapper.innerHTML = ``; } -// Function to handle search input -searchInput.addEventListener('input', async () => { - let input = searchInput.value; - if (input.length) { - const results = await getSuggestions(input); - renderResults(results); - } -}); - // Handle click events on the type buttons const typeButtons = document.querySelectorAll('[name="t"]'); typeButtons.forEach(button => { @@ -178,11 +170,11 @@ document.addEventListener('click', (event) => { } }); -// Update visual feedback for selected type on page load -document.addEventListener("DOMContentLoaded", () => { - const activeButton = document.querySelector(`[name="t"][value="${selectedType}"]`); - if (activeButton) { - typeButtons.forEach(btn => btn.classList.remove('search-active')); - activeButton.classList.add('search-active'); - } -}); +// // Update visual feedback for selected type on page load +// document.addEventListener("DOMContentLoaded", () => { +// const activeButton = document.querySelector(`[name="t"][value="${selectedType}"]`); +// if (activeButton) { +// typeButtons.forEach(btn => btn.classList.remove('search-active')); +// activeButton.classList.add('search-active'); +// } +// }); diff --git a/suggestions.go b/suggestions.go index 73e9b02..089ca8b 100644 --- a/suggestions.go +++ b/suggestions.go @@ -6,8 +6,67 @@ import ( "io" "net/http" "net/url" + "sort" + "sync" + "time" ) +// SuggestionSource represents a search suggestion source along with its latency. +type SuggestionSource struct { + Name string + FetchFunc func(string) []string + Latency time.Duration + mu sync.Mutex +} + +// Initialize suggestion sources with default latency values. +var suggestionSources = []SuggestionSource{ + { + Name: "DuckDuckGo", + FetchFunc: fetchDuckDuckGoSuggestions, + Latency: 50 * time.Millisecond, + }, + { + Name: "Edge", + FetchFunc: fetchEdgeSuggestions, + Latency: 50 * time.Millisecond, + }, + { + Name: "Brave", + FetchFunc: fetchBraveSuggestions, + Latency: 50 * time.Millisecond, + }, + { + Name: "Ecosia", + FetchFunc: fetchEcosiaSuggestions, + Latency: 50 * time.Millisecond, + }, + // { // Not working with fetchSuggestionsFromURL func + // Name: "Qwant", + // FetchFunc: fetchQwantSuggestions, + // Latency: 50 * time.Millisecond, + // }, + { + Name: "Startpage", + FetchFunc: fetchStartpageSuggestions, + Latency: 50 * time.Millisecond, + }, + { + Name: "Yahoo", + FetchFunc: fetchYahooSuggestions, + Latency: 50 * time.Millisecond, + }, + // I advise against it, but you can use it if you want to + // { + // Name: "Google", + // FetchFunc: fetchGoogleSuggestions, + // Latency: 500 * time.Millisecond, + // }, +} + +// Mutex to protect the suggestionSources during sorting. +var suggestionsMU sync.Mutex + func handleSuggestions(w http.ResponseWriter, r *http.Request) { query := r.URL.Query().Get("q") if query == "" { @@ -16,37 +75,58 @@ func handleSuggestions(w http.ResponseWriter, r *http.Request) { return } - // Define the fallback sequence with Google lower in the hierarchy - suggestionSources := []func(string) []string{ - fetchDuckDuckGoSuggestions, - fetchEdgeSuggestions, - fetchBraveSuggestions, - fetchEcosiaSuggestions, - fetchQwantSuggestions, - fetchStartpageSuggestions, - // fetchGoogleSuggestions, // I advise against it, but you can use it if you want to - } + // Sort the suggestion sources based on their latency. + suggestionsMU.Lock() + sort.Slice(suggestionSources, func(i, j int) bool { + return suggestionSources[i].Latency < suggestionSources[j].Latency + }) + suggestionsMU.Unlock() var suggestions []string - for _, fetchFunc := range suggestionSources { - suggestions = fetchFunc(query) + for i := range suggestionSources { + source := &suggestionSources[i] + start := time.Now() + suggestions = source.FetchFunc(query) + elapsed := time.Since(start) + + updateLatency(source, elapsed) + if len(suggestions) > 0 { - printDebug("Suggestions found using %T", fetchFunc) + printDebug("Suggestions found using %s", source.Name) break } else { - printWarn("%T did not return any suggestions or failed.", fetchFunc) + printWarn("%s did not return any suggestions or failed.", source.Name) } } + // Trim the suggestions to a maximum of 8 items + suggestions = trimSuggestions(suggestions) + if len(suggestions) == 0 { printErr("All suggestion services failed. Returning empty response.") } - // Return the final suggestions as JSON + // Return the final suggestions as JSON. w.Header().Set("Content-Type", "application/json") fmt.Fprintf(w, `["",%s]`, toJSONStringArray(suggestions)) } +// trimSuggestions trims the suggestion list to a maximum of 8 suggestions. +func trimSuggestions(suggestions []string) []string { + if len(suggestions) > 8 { + return suggestions[:8] + } + return suggestions +} + +// updateLatency updates the latency of a suggestion source using an exponential moving average. +func updateLatency(source *SuggestionSource, newLatency time.Duration) { + source.mu.Lock() + defer source.mu.Unlock() + const alpha = 0.5 // Smoothing factor. + source.Latency = time.Duration(float64(source.Latency)*(1-alpha) + float64(newLatency)*alpha) +} + func fetchGoogleSuggestions(query string) []string { encodedQuery := url.QueryEscape(query) url := fmt.Sprintf("http://suggestqueries.google.com/complete/search?client=firefox&q=%s", encodedQuery) @@ -82,6 +162,7 @@ func fetchEcosiaSuggestions(query string) []string { return fetchSuggestionsFromURL(url) } +// Is this working? func fetchQwantSuggestions(query string) []string { encodedQuery := url.QueryEscape(query) url := fmt.Sprintf("https://api.qwant.com/v3/suggest?q=%s", encodedQuery) @@ -96,6 +177,20 @@ func fetchStartpageSuggestions(query string) []string { return fetchSuggestionsFromURL(url) } +func fetchYahooSuggestions(query string) []string { + encodedQuery := url.QueryEscape(query) + url := fmt.Sprintf("https://search.yahoo.com/sugg/gossip/gossip-us-ura/?output=fxjson&command=%s", encodedQuery) + printDebug("Fetching suggestions from Yahoo: %s", url) + return fetchSuggestionsFromURL(url) +} + +// func fetchBaiduSuggestions(query string) []string { +// encodedQuery := url.QueryEscape(query) +// url := fmt.Sprintf("https://suggestion.baidu.com/su?wd=%s", encodedQuery) +// printDebug("Fetching suggestions from Baidu: %s", url) +// return fetchSuggestionsFromURL(url) +// } + func fetchSuggestionsFromURL(url string) []string { resp, err := http.Get(url) if err != nil { @@ -110,17 +205,20 @@ func fetchSuggestionsFromURL(url string) []string { return []string{} } - // Log the Content-Type for debugging + // Print the raw HTTP response for debugging + fmt.Printf("Raw response from %s:\n%s\n", url, string(body)) + + // Log the Content-Type for debugging. contentType := resp.Header.Get("Content-Type") printDebug("Response Content-Type from %s: %s", url, contentType) - // Check if the body is non-empty + // Check if the body is non-empty. if len(body) == 0 { printWarn("Received empty response body from %s", url) return []string{} } - // Attempt to parse the response as JSON regardless of Content-Type + // Attempt to parse the response as JSON regardless of Content-Type. var parsedResponse []interface{} if err := json.Unmarshal(body, &parsedResponse); err != nil { printErr("Error parsing JSON from %s: %v", url, err) @@ -128,7 +226,7 @@ func fetchSuggestionsFromURL(url string) []string { return []string{} } - // Ensure the response structure is as expected + // Ensure the response structure is as expected. if len(parsedResponse) < 2 { printWarn("Unexpected response format from %v: %v", url, string(body)) return []string{} @@ -148,6 +246,7 @@ func fetchSuggestionsFromURL(url string) []string { return suggestions } +// toJSONStringArray converts a slice of strings to a JSON array string. func toJSONStringArray(strings []string) string { result := "" for i, str := range strings { @@ -158,3 +257,30 @@ func toJSONStringArray(strings []string) string { } return "[" + result + "]" } + +// func testSuggestionSources(query string) { +// for _, source := range suggestionSources { +// fmt.Printf("Testing %s...\n", source.Name) + +// // Fetch suggestions +// suggestions := source.FetchFunc(query) + +// // If we get results, print them +// if len(suggestions) > 0 { +// fmt.Printf("Suggestions from %s:\n", source.Name) +// for i, suggestion := range suggestions { +// fmt.Printf("%d: %s\n", i+1, suggestion) +// } +// } else { +// fmt.Printf("No suggestions from %s.\n", source.Name) +// } + +// // Small separator for clarity +// fmt.Println("--------------------------") +// } +// } + +// func main() { +// query := "test query" +// testSuggestionSources(query) +// } diff --git a/templates/files.html b/templates/files.html index e119424..4b1b246 100755 --- a/templates/files.html +++ b/templates/files.html @@ -15,7 +15,7 @@

Ocásek

- +
    diff --git a/templates/forums.html b/templates/forums.html index e833d4f..3fbdea7 100755 --- a/templates/forums.html +++ b/templates/forums.html @@ -15,7 +15,7 @@

    Ocásek

    - +
      diff --git a/templates/images.html b/templates/images.html index ccf08ae..8f82a2b 100755 --- a/templates/images.html +++ b/templates/images.html @@ -15,7 +15,7 @@

      Ocásek

      - +
        diff --git a/templates/map.html b/templates/map.html index 5ce8c62..d0d9fa3 100644 --- a/templates/map.html +++ b/templates/map.html @@ -30,7 +30,7 @@

        Ocásek

        - +
          diff --git a/templates/search.html b/templates/search.html index 61fbb0d..4f75c28 100755 --- a/templates/search.html +++ b/templates/search.html @@ -8,6 +8,7 @@ {{ end }} Search with Ocásek + @@ -52,9 +53,17 @@ }); // Event listener for Language Selection - document.getElementById('languageSelect').addEventListener('change', function () { - updateSettings('lang', this.value); - }); + if (siteLanguageSelect) { + siteLanguageSelect.addEventListener('change', function () { + updateSettings('site_lang', this.value); + }); + } + + // if (searchLanguageSelect) { + // searchLanguageSelect.addEventListener('change', function () { + // updateSettings('search_lang', this.value); + // }); + // } });