diff --git a/src/librustdoc/html/static/css/rustdoc.css b/src/librustdoc/html/static/css/rustdoc.css index e2b4cc50dd580..58898247a9f40 100644 --- a/src/librustdoc/html/static/css/rustdoc.css +++ b/src/librustdoc/html/static/css/rustdoc.css @@ -328,10 +328,11 @@ details:not(.toggle) summary { margin-bottom: .6em; } -code, pre, a.test-arrow, .code-header { +code, pre, a.test-arrow, .code-header, .search-results .type-signature { font-family: "Source Code Pro", monospace; } -.docblock code, .docblock-short code { +.docblock code, .docblock-short code, +.search-results .type-signature strong { border-radius: 3px; padding: 0 0.125em; } @@ -681,7 +682,8 @@ ul.block, .block li { } .docblock code, .docblock-short code, -pre, .rustdoc.src .example-wrap { +pre, .rustdoc.src .example-wrap, +.search-results .type-signature strong { background-color: var(--code-block-background-color); } diff --git a/src/librustdoc/html/static/js/search.js b/src/librustdoc/html/static/js/search.js index 48c9a53a28310..91a79524b7ed6 100644 --- a/src/librustdoc/html/static/js/search.js +++ b/src/librustdoc/html/static/js/search.js @@ -262,6 +262,14 @@ function initSearch(rawSearchIndex) { * Special type name IDs for searching by both array and slice (`[]` syntax). */ let typeNameIdOfArrayOrSlice; + /** + * Special type name IDs for searching never (the `!` type). + */ + let typeNameIdOfNever; + /** + * Special type name IDs for searching tuple (the `()` type). + */ + let typeNameIdOfTuple; /** * Add an item to the type Name->ID map, or, if one already exists, use it. @@ -467,12 +475,13 @@ function initSearch(rawSearchIndex) { * @param {ParsedQuery} query * @param {ParserState} parserState * @param {string} name - Name of the query element. + * @param {string} normalizedName - Name of the query element will be used for searching. * @param {Array} generics - List of generics of this query element. * * @return {QueryElement} - The newly created `QueryElement`. */ - function createQueryElement(query, parserState, name, generics, isInGenerics) { - const path = name.trim(); + function createQueryElement(query, parserState, name, normalizedName, generics, isInGenerics) { + const path = normalizedName.trim(); if (path.length === 0 && generics.length === 0) { throw ["Unexpected ", parserState.userQuery[parserState.pos]]; } else if (path === "*") { @@ -483,7 +492,7 @@ function initSearch(rawSearchIndex) { } const typeFilter = parserState.typeFilter; parserState.typeFilter = null; - if (name === "!") { + if (normalizedName === "!") { if (typeFilter !== null && typeFilter !== "primitive") { throw [ "Invalid search type: primitive never type ", @@ -502,6 +511,7 @@ function initSearch(rawSearchIndex) { } return { name: "never", + normalizedName: "never", id: null, fullPath: ["never"], pathWithoutLast: [], @@ -544,6 +554,7 @@ function initSearch(rawSearchIndex) { } return { name: name.trim(), + normalizedName: normalizedName.trim(), id: null, fullPath: pathSegments, pathWithoutLast: pathSegments.slice(0, pathSegments.length - 1), @@ -673,6 +684,7 @@ function initSearch(rawSearchIndex) { } elems.push({ name: "[]", + normalizedName: "[]", id: null, fullPath: ["[]"], pathWithoutLast: [], @@ -709,6 +721,7 @@ function initSearch(rawSearchIndex) { createQueryElement( query, parserState, + parserState.original.slice(start, end), parserState.userQuery.slice(start, end), generics, isInGenerics @@ -771,7 +784,7 @@ function initSearch(rawSearchIndex) { // The type filter doesn't count as an element since it's a modifier. const typeFilterElem = elems.pop(); checkExtraTypeFilterCharacters(start, parserState); - parserState.typeFilter = typeFilterElem.name; + parserState.typeFilter = typeFilterElem.normalizedName; parserState.pos += 1; parserState.totalElems -= 1; query.literalSearch = false; @@ -894,7 +907,7 @@ function initSearch(rawSearchIndex) { // The type filter doesn't count as an element since it's a modifier. const typeFilterElem = query.elems.pop(); checkExtraTypeFilterCharacters(start, parserState); - parserState.typeFilter = typeFilterElem.name; + parserState.typeFilter = typeFilterElem.normalizedName; parserState.pos += 1; parserState.totalElems -= 1; query.literalSearch = false; @@ -1063,6 +1076,7 @@ function initSearch(rawSearchIndex) { totalElems: 0, genericsElems: 0, typeFilter: null, + original: userQuery, userQuery: userQuery.toLowerCase(), }; let query = newParsedQuery(userQuery); @@ -1129,29 +1143,35 @@ function initSearch(rawSearchIndex) { * marked for removal. * * @param {[ResultObject]} results + * @param {boolean} isType - True will include function signature string * @returns {[ResultObject]} */ - function transformResults(results) { + function transformResults(results, isType) { const duplicates = new Set(); const out = []; for (const result of results) { if (result.id !== -1) { const obj = searchIndex[result.id]; - obj.dist = result.dist; const res = buildHrefAndPath(obj); - obj.displayPath = pathSplitter(res[0]); - obj.fullPath = obj.displayPath + obj.name; + const displayPath = pathSplitter(res[0]); + let fullPath = displayPath + obj.name; // To be sure than it some items aren't considered as duplicate. - obj.fullPath += "|" + obj.ty; + fullPath += "|" + obj.ty; - if (duplicates.has(obj.fullPath)) { + if (!isType && duplicates.has(fullPath)) { continue; } - duplicates.add(obj.fullPath); - - obj.href = res[1]; - out.push(obj); + duplicates.add(fullPath); + + out.push(Object.assign({ + dist: result.dist, + index: result.index, + displayTypeSignature: result.displayTypeSignature, + displayPath, + fullPath, + href: res[1], + }, obj)); if (out.length >= MAX_RESULTS) { break; } @@ -1160,6 +1180,119 @@ function initSearch(rawSearchIndex) { return out; } + /** + * Convert a type signature to a readable string. + * + * @param {FunctionSearchType?} results + * + * @returns string? + */ + function formatResultTypeSignature(type) { + if (type === null) { + return null; + } + + let result = []; + + function pushNotHighlighted(text) { + if (result.length % 2 === 0) { + result.push(""); + } + result[result.length - 1] = `${result[result.length - 1]}${text}`; + } + function pushHighlighted(text) { + if (result.length === 0) { + result.push(""); + } + if (result.length % 2 !== 0) { + result.push(""); + } + result[result.length - 1] = `${result[result.length - 1]}${text}`; + } + function push(text, highlighted) { + if (highlighted) { + pushHighlighted(text); + } else { + pushNotHighlighted(text); + } + } + + function isTransitivelyHighlighted(fnType) { + // Caching the flag is necessary to avoid quadratic complexity + // in this function. + if (Object.prototype.hasOwnProperty.call(fnType, "isTransitivelyHighlighted")) { + return fnType.isTransitivelyHighlighted; + } + const isHighlighted = fnType.highlighted || + (fnType.generics && fnType.generics.length > 0 && + fnType.generics.some(isTransitivelyHighlighted)) || + (type.where_clause && type.where_clause[(-fnType.id) - 1] && + type.where_clause[(-fnType.id) - 1].some(isTransitivelyHighlighted)); + fnType.isTransitivelyHighlighted = isHighlighted; + return isHighlighted; + } + + function formatTypeList(fnTypes, delim) { + if (!fnTypes) { + return; + } + let first = true; + for (const fnType of fnTypes) { + if (first) { + first = false; + } else { + pushNotHighlighted(delim); + } + if (fnType.ty === TY_PRIMITIVE && + (fnType.id === typeNameIdOfSlice || fnType.id === typeNameIdOfArray) + ) { + push("[", fnType.highlighted); + if (fnType.generics && fnType.generics.some(isTransitivelyHighlighted)) { + formatTypeList(fnType.generics, ", "); + } + push("]", fnType.highlighted); + } else if (fnType.ty === TY_PRIMITIVE && fnType.id === typeNameIdOfNever) { + push("!", fnType.highlighted); + } else if (fnType.name === null) { + if (fnType.generics && fnType.generics.some(t => t.name !== null)) { + push("impl ", fnType.highlighted); + formatTypeList(fnType.generics, " + "); + } else if (type.where_clause && + type.where_clause[(-fnType.id) - 1].some(isTransitivelyHighlighted) + ) { + push("impl ", fnType.highlighted); + formatTypeList(type.where_clause[(-fnType.id) - 1], " + "); + } else { + push("_", fnType.highlighted); + } + } else { + push(fnType.name, fnType.highlighted); + if (fnType.generics && fnType.generics.some(isTransitivelyHighlighted)) { + pushNotHighlighted("<"); + formatTypeList(fnType.generics, ", "); + pushNotHighlighted(">"); + } + } + } + } + + formatTypeList(type.inputs, ", "); + if (result.every(f => f === "")) { + result = []; + } + const returnsUnit = type.output.length === 1 && + type.output[0].id === typeNameIdOfTuple && + type.output[0].generics.length === 0; + if (type.output.length !== 0 && !returnsUnit) { + pushNotHighlighted(result.length === 0 ? "-> " : " -> "); + formatTypeList(type.output, ", "); + } + if (result.every(f => f === "")) { + return null; + } + return result; + } + /** * This function takes a result map, and sorts it by various criteria, including edit * distance, substring match, and the crate it comes from. @@ -1305,39 +1438,37 @@ function initSearch(rawSearchIndex) { result.id = -1; } } - return transformResults(result_list); + return transformResults(result_list, isType); } /** - * This function checks generics in search query `queryElem` can all be found in the - * search index (`fnType`), - * - * This function returns `true` if it matches, and also writes the results to mgensInout. - * It returns `false` if no match is found, and leaves mgensInout untouched. + * The type tree on each function is annotated with information about which + * node to highlight. This is then used to generate the formatted type + * signature for the result. * - * @param {FunctionType} fnType - The object to check. - * @param {QueryElement} queryElem - The element from the parsed query. - * @param {[FunctionType]} whereClause - Trait bounds for generic items. - * @param {Map|null} mgensInout - Map functions generics to query generics. - * - * @return {boolean} - Returns true if a match, false otherwise. + * Between each result, it gets cleared out. This function clears it. */ - function checkGenerics(fnType, queryElem, whereClause, mgensInout) { - return unifyFunctionTypes( - fnType.generics, - queryElem.generics, - whereClause, - mgensInout, - mgens => { - if (mgensInout) { - for (const [fid, qid] of mgens.entries()) { - mgensInout.set(fid, qid); - } - } - return true; + function clearHighlighted(fnType) { + function doClearHighlighted(fnTypeList) { + if (!fnTypeList) { + return; } - ); + for (const fnType of fnTypeList) { + fnType.highlighted = false; + delete fnType.isTransitivelyHighlighted; + delete fnType.matchedGenerics; + doClearHighlighted(fnType.generics); + } + } + doClearHighlighted(fnType.inputs); + doClearHighlighted(fnType.output); + if (fnType.where_clause) { + for (const where of fnType.where_clause) { + doClearHighlighted(where); + } + } } + /** * This function checks if a list of search query `queryElems` can all be found in the * search index (`fnTypes`). @@ -1353,9 +1484,10 @@ function initSearch(rawSearchIndex) { * @param {[FunctionType]} whereClause - Trait bounds for generic items. * @param {Map|null} mgensIn * - Map functions generics to query generics (never modified). - * @param {null|Map -> bool} solutionCb - Called for each `mgens` solution. + * @param {(null|Map, Array) -> bool} solutionCb + * - Called for each `mgens` solution. * - * @return {boolean} - Returns true if a match, false otherwise. + * @return {Array|false} - Returns array of matched parts, false otherwise. */ function unifyFunctionTypes(fnTypesIn, queryElems, whereClause, mgensIn, solutionCb) { /** @@ -1363,7 +1495,7 @@ function initSearch(rawSearchIndex) { */ let mgens = new Map(mgensIn); if (queryElems.length === 0) { - return !solutionCb || solutionCb(mgens); + return !solutionCb || solutionCb(mgens, []); } if (!fnTypesIn || fnTypesIn.length === 0) { return false; @@ -1411,6 +1543,7 @@ function initSearch(rawSearchIndex) { queryElemsOffset, fnTypesOffset, unbox, + matchedGenerics, } = backtracking.pop(); mgens = new Map(mgensScratch); const fnType = fnTypesScratch[fnTypesOffset]; @@ -1440,6 +1573,7 @@ function initSearch(rawSearchIndex) { fl = fnTypes.length; const tmp = fnTypes[queryElemsOffset]; fnTypes[queryElemsOffset] = fnTypes[fnTypesOffset]; + fnTypes[queryElemsOffset].matchedGenerics = matchedGenerics; fnTypes[fnTypesOffset] = tmp; // this is known as a good match; go to the next one i = queryElemsOffset; @@ -1470,10 +1604,11 @@ function initSearch(rawSearchIndex) { queryElem.generics, whereClause, mgens, - mgensScratch => { + (mgensScratch, matchedGenerics) => { matchCandidates.push({ fnTypesScratch, mgensScratch, + matchedGenerics, queryElemsOffset: i, fnTypesOffset: j, unbox: false, @@ -1492,6 +1627,7 @@ function initSearch(rawSearchIndex) { backtracking.push({ fnTypesScratch, mgensScratch, + matchedGenerics: [], queryElemsOffset: i, fnTypesOffset: j, unbox: true, @@ -1506,7 +1642,8 @@ function initSearch(rawSearchIndex) { } } // use the current candidate - const {fnTypesOffset: candidate, mgensScratch: mgensNew} = matchCandidates.pop(); + const {fnTypesOffset: candidate, mgensScratch: mgensNew, matchedGenerics} + = matchCandidates.shift(); if (fnTypes[candidate].id < 0 && queryElems[i].id < 0) { mgens.set(fnTypes[candidate].id, queryElems[i].id); } @@ -1523,9 +1660,22 @@ function initSearch(rawSearchIndex) { for (const otherCandidate of matchCandidates) { backtracking.push(otherCandidate); } + fnTypes[i].matchedGenerics = matchedGenerics; // If we're on the last item, check the solution with the callback // backtrack if the callback says its unsuitable - while (i === (ql - 1) && solutionCb && !solutionCb(mgens)) { + while (i === (ql - 1) && solutionCb) { + // The above loop partitions the list into "matched" and "unmatched" parts + // by swapping things into place when they match. + let highlightable = fnTypes.slice(0, ql); + for (let j = 0; j < highlightable.length; ++j) { + if (highlightable[j].matchedGenerics.length === 0) { + continue; + } + highlightable = [...highlightable, ...highlightable[j].matchedGenerics]; + } + if (solutionCb(mgens, highlightable)) { + break; + } if (!backtrack()) { return false; } @@ -1649,38 +1799,15 @@ function initSearch(rawSearchIndex) { * @return {boolean} - Returns true if the type matches, false otherwise. */ function checkType(row, elem, whereClause) { - if (row.id === null) { - // This is a pure "generic" search, no need to run other checks. - return row.generics.length > 0 - ? checkIfInList(row.generics, elem, whereClause) - : false; - } - - if (row.id < 0 && elem.id >= 0) { - const gid = (-row.id) - 1; - return checkIfInList(whereClause[gid], elem, whereClause); - } - - if (row.id < 0 && elem.id < 0) { - return true; - } - - const matchesExact = row.id === elem.id; - const matchesArrayOrSlice = elem.id === typeNameIdOfArrayOrSlice && - (row.id === typeNameIdOfSlice || row.id === typeNameIdOfArray); - - if ((matchesExact || matchesArrayOrSlice) && - typePassesFilter(elem.typeFilter, row.ty)) { - if (elem.generics.length > 0) { - return checkGenerics(row, elem, whereClause, new Map()); - } - return true; + if (row.id > 0 && elem.id > 0 && elem.pathWithoutLast.length === 0 && + row.generics.length === 0 && elem.generics.length === 0 && + // special case + elem.id !== typeNameIdOfArrayOrSlice + ) { + return row.id === elem.id && typePassesFilter(elem.typeFilter, row.ty); + } else { + return unifyFunctionTypes([row], [elem], whereClause); } - - // If the current item does not match, try [unboxing] the generic. - // [unboxing]: - // https://ndmitchell.com/downloads/slides-hoogle_fast_type_searching-09_aug_2008.pdf - return checkIfInList(row.generics, elem, whereClause); } function checkPath(contains, ty, maxEditDistance) { @@ -1831,8 +1958,19 @@ function initSearch(rawSearchIndex) { * @param {integer} index * @param {integer} dist * @param {integer} path_dist + * @param {integer} maxEditDistance + * @param {boolean} isType */ - function addIntoResults(results, fullId, id, index, dist, path_dist, maxEditDistance) { + function addIntoResults( + results, + fullId, + id, + index, + dist, + path_dist, + maxEditDistance, + isType + ) { const inBounds = dist <= maxEditDistance || index !== -1; if (dist === 0 || (!parsedQuery.literalSearch && inBounds)) { if (results.has(fullId)) { @@ -1841,12 +1979,16 @@ function initSearch(rawSearchIndex) { return; } } + const displayTypeSignature = isType ? + formatResultTypeSignature(searchIndex[id].type) : + null; results.set(fullId, { id: id, index: index, dontValidate: parsedQuery.literalSearch, dist: dist, path_dist: path_dist, + displayTypeSignature, }); } } @@ -1885,14 +2027,46 @@ function initSearch(rawSearchIndex) { const in_args = row.type && row.type.inputs && checkIfInList(row.type.inputs, elem, row.type.where_clause); if (in_args) { + // Generate type signature information for search result display. + clearHighlighted(row.type); + unifyFunctionTypes( + row.type.inputs, + parsedQuery.elems, + row.type.where_clause, + null, + (mgens, parts) => { + writeNamesToTypeParameters(mgens, row.type, parsedQuery); + for (const part of parts) { + part.highlighted = true; + } + return true; + } + ); // path_dist is 0 because no parent path information is currently stored // in the search index - addIntoResults(results_in_args, fullId, pos, -1, 0, 0, maxEditDistance); + addIntoResults(results_in_args, fullId, pos, -1, 0, 0, maxEditDistance, true); } const returned = row.type && row.type.output && checkIfInList(row.type.output, elem, row.type.where_clause); if (returned) { - addIntoResults(results_returned, fullId, pos, -1, 0, 0, maxEditDistance); + // Generate type signature information for search result display. + if (!in_args) { + clearHighlighted(row.type); + unifyFunctionTypes( + row.type.output, + parsedQuery.elems, + row.type.where_clause, + null, + (mgens, parts) => { + writeNamesToTypeParameters(mgens, row.type, parsedQuery); + for (const part of parts) { + part.highlighted = true; + } + return true; + } + ); + } + addIntoResults(results_returned, fullId, pos, -1, 0, 0, maxEditDistance, true); } if (!typePassesFilter(elem.typeFilter, row.ty)) { @@ -1922,8 +2096,17 @@ function initSearch(rawSearchIndex) { } if (parsedQuery.literalSearch) { - if (searchWord === elem.name) { - addIntoResults(results_others, fullId, pos, index, 0, path_dist); + if (searchWord === elem.normalizedName) { + addIntoResults( + results_others, + fullId, + pos, + index, + 0, + path_dist, + -1, + false + ); } return; } @@ -1934,7 +2117,16 @@ function initSearch(rawSearchIndex) { return; } - addIntoResults(results_others, fullId, pos, index, dist, path_dist, maxEditDistance); + addIntoResults( + results_others, + fullId, + pos, + index, + dist, + path_dist, + maxEditDistance, + false + ); } /** @@ -1951,25 +2143,80 @@ function initSearch(rawSearchIndex) { return; } - // If the result is too "bad", we return false and it ends this search. if (!unifyFunctionTypes( row.type.inputs, parsedQuery.elems, row.type.where_clause, null, - mgens => { + (mgens, matchedPartsParams) => { return unifyFunctionTypes( row.type.output, parsedQuery.returned, row.type.where_clause, - mgens + mgens, + (completeMgens, matchedPartsReturn) => { + writeNamesToTypeParameters(completeMgens, row.type, parsedQuery); + clearHighlighted(row.type); + for (const part of matchedPartsParams) { + part.highlighted = true; + } + for (const part of matchedPartsReturn) { + part.highlighted = true; + } + return true; + } ); } )) { return; } - addIntoResults(results, row.id, pos, 0, 0, 0, Number.MAX_VALUE); + addIntoResults(results, row.id, pos, 0, 0, 0, Number.MAX_VALUE, true); + } + + function writeNamesToTypeParameters(mgens, rowType, parsedQuery) { + const queryTypeToName = new Map(); + function getQueryTypeFromList(list) { + if (!list) { + return; + } + for (const queryElem of list) { + if (queryElem.generics && queryElem.generics.length > 0) { + getQueryTypeFromList(queryElem.generics); + } + if (!queryTypeToName.has(queryElem.id)) { + queryTypeToName.set(queryElem.id, queryElem.name); + } + } + } + getQueryTypeFromList(parsedQuery.elems); + getQueryTypeFromList(parsedQuery.returned); + function assignFunctionNameFromList(list) { + if (!list) { + return; + } + for (const fnType of list) { + if (fnType.id < 0) { + assignFunctionNameFromList(rowType.where_clause[(-fnType.id) - 1]); + if (mgens.has(fnType.id)) { + const queryId = mgens.get(fnType.id); + if (queryTypeToName.has(queryId) && queryId < 0) { + const queryName = queryTypeToName.get(queryId); + fnType.name = queryName; + } else { + fnType.name = null; + } + } else { + fnType.name = null; + } + } + if (fnType.generics && fnType.generics.length > 0) { + assignFunctionNameFromList(fnType.generics); + } + } + } + assignFunctionNameFromList(rowType.inputs); + assignFunctionNameFromList(rowType.output); } function innerRunQuery() { @@ -1977,10 +2224,10 @@ function initSearch(rawSearchIndex) { let queryLen = 0; for (const elem of parsedQuery.elems) { - queryLen += elem.name.length; + queryLen += elem.normalizedName.length; } for (const elem of parsedQuery.returned) { - queryLen += elem.name.length; + queryLen += elem.normalizedName.length; } const maxEditDistance = Math.floor(queryLen / 3); @@ -2027,21 +2274,21 @@ function initSearch(rawSearchIndex) { if ((elem.id === null && parsedQuery.totalElems > 1 && elem.typeFilter === -1 && elem.generics.length === 0) || elem.typeFilter === TY_GENERIC) { - if (genericSymbols.has(elem.name)) { - elem.id = genericSymbols.get(elem.name); + if (genericSymbols.has(elem.normalizedName)) { + elem.id = genericSymbols.get(elem.normalizedName); } else { elem.id = -(genericSymbols.size + 1); - genericSymbols.set(elem.name, elem.id); + genericSymbols.set(elem.normalizedName, elem.id); } - if (elem.typeFilter === -1 && elem.name.length >= 3) { + if (elem.typeFilter === -1 && elem.normalizedName.length >= 3) { // Silly heuristic to catch if the user probably meant // to not write a generic parameter. We don't use it, // just bring it up. - const maxPartDistance = Math.floor(elem.name.length / 3); + const maxPartDistance = Math.floor(elem.normalizedName.length / 3); let matchDist = maxPartDistance + 1; let matchName = ""; for (const name of typeNameIdMap.keys()) { - const dist = editDistance(name, elem.name, maxPartDistance); + const dist = editDistance(name, elem.normalizedName, maxPartDistance); if (dist <= matchDist && dist <= maxPartDistance) { if (dist === matchDist && matchName > name) { continue; @@ -2051,7 +2298,7 @@ function initSearch(rawSearchIndex) { } } if (matchName !== "") { - parsedQuery.proposeCorrectionFrom = elem.name; + parsedQuery.proposeCorrectionFrom = elem.normalizedName; parsedQuery.proposeCorrectionTo = matchName; } } @@ -2098,10 +2345,22 @@ function initSearch(rawSearchIndex) { elem = parsedQuery.returned[0]; for (i = 0, nSearchWords = searchWords.length; i < nSearchWords; ++i) { row = searchIndex[i]; - in_returned = row.type && unifyFunctionTypes( + if (!row.type) { + continue; + } + in_returned = unifyFunctionTypes( row.type.output, parsedQuery.returned, - row.type.where_clause + row.type.where_clause, + null, + (mgens, parts) => { + writeNamesToTypeParameters(mgens, row.type, parsedQuery); + clearHighlighted(row.type); + for (const part of parts) { + part.highlighted = true; + } + return true; + } ); if (in_returned) { addIntoResults( @@ -2110,7 +2369,9 @@ function initSearch(rawSearchIndex) { i, -1, 0, - Number.MAX_VALUE + Number.MAX_VALUE, + undefined, + true ); } } @@ -2126,10 +2387,11 @@ function initSearch(rawSearchIndex) { innerRunQuery(); } + const isOthersTypeSearch = parsedQuery.foundElems > 1 || parsedQuery.returned.length > 0; const ret = createQueryResults( sortResults(results_in_args, true, currentCrate), sortResults(results_returned, true, currentCrate), - sortResults(results_others, false, currentCrate), + sortResults(results_others, isOthersTypeSearch, currentCrate), parsedQuery); handleAliases(ret, parsedQuery.original.replace(/"/g, ""), filterCrates, currentCrate); if (parsedQuery.error !== null && ret.others.length !== 0) { @@ -2264,11 +2526,12 @@ function initSearch(rawSearchIndex) { /** * Render a set of search results for a single tab. - * @param {Array} array - The search results for this tab + * @param {Array} array - The search results for this tab * @param {ParsedQuery} query * @param {boolean} display - True if this is the active tab + * @param {boolean} isType - True shows the function signature */ - function addTab(array, query, display) { + function addTab(array, query, display, isType) { let extraClass = ""; if (display === true) { extraClass = " active"; @@ -2313,6 +2576,20 @@ ${item.displayPath}${name}\ const description = document.createElement("div"); description.className = "desc"; + if (isType) { + const displayTypeSignature = document.createElement("div"); + item.displayTypeSignature.forEach((value, index) => { + if (index % 2 !== 0) { + const highlight = document.createElement("strong"); + highlight.appendChild(document.createTextNode(value)); + displayTypeSignature.appendChild(highlight); + } else { + displayTypeSignature.appendChild(document.createTextNode(value)); + } + }); + displayTypeSignature.className = "type-signature"; + description.appendChild(displayTypeSignature); + } description.insertAdjacentHTML("beforeend", item.desc); link.appendChild(description); @@ -2392,9 +2669,24 @@ ${item.displayPath}${name}\ currentResults = results.query.userQuery; - const ret_others = addTab(results.others, results.query, true); - const ret_in_args = addTab(results.in_args, results.query, false); - const ret_returned = addTab(results.returned, results.query, false); + const ret_others = addTab( + results.others, + results.query, + true, + results.query.foundElems > 1 || results.query.returned.length > 0 + ); + const ret_in_args = addTab( + results.in_args, + results.query, + false, + true + ); + const ret_returned = addTab( + results.returned, + results.query, + false, + true + ); // Navigate to the relevant tab if the current tab is empty, like in case users search // for "-> String". If they had selected another tab previously, they have to click on @@ -2426,6 +2718,9 @@ ${item.displayPath}${name}\ let output = `

Results${crates}

`; if (results.query.error !== null) { const error = results.query.error; + if (!error.forEach) { + throw error; + } error.forEach((value, index) => { value = value.split("<").join("<").split(">").join(">"); if (index % 2 !== 0) { @@ -2569,19 +2864,22 @@ ${item.displayPath}${name}\ * * @param {null|Array} types * @param {Array<{name: string, ty: number}>} lowercasePaths + * @param {Array<{name: string, ty: number}>} paths * * @return {Array} */ - function buildItemSearchTypeAll(types, lowercasePaths) { - return types.map(type => buildItemSearchType(type, lowercasePaths)); + function buildItemSearchTypeAll(types, lowercasePaths, paths) { + return types.map(type => buildItemSearchType(type, lowercasePaths, paths)); } /** * Converts a single type. * * @param {RawFunctionType} type + * @param {Array<{name: string, ty: number}>} lowercasePaths + * @param {Array<{name: string, ty: number}>} paths */ - function buildItemSearchType(type, lowercasePaths) { + function buildItemSearchType(type, lowercasePaths, paths) { const PATH_INDEX_DATA = 0; const GENERICS_DATA = 1; let pathIndex, generics; @@ -2592,7 +2890,8 @@ ${item.displayPath}${name}\ pathIndex = type[PATH_INDEX_DATA]; generics = buildItemSearchTypeAll( type[GENERICS_DATA], - lowercasePaths + lowercasePaths, + paths ); } if (pathIndex < 0) { @@ -2600,6 +2899,7 @@ ${item.displayPath}${name}\ // the actual names of generic parameters aren't stored, since they aren't API return { id: pathIndex, + name: null, ty: TY_GENERIC, path: null, generics, @@ -2609,6 +2909,7 @@ ${item.displayPath}${name}\ // `0` is used as a sentinel because it's fewer bytes than `null` return { id: null, + name: null, ty: null, path: null, generics, @@ -2617,6 +2918,7 @@ ${item.displayPath}${name}\ const item = lowercasePaths[pathIndex - 1]; return { id: buildTypeMapIndex(item.name), + name: paths[pathIndex - 1].name, ty: item.ty, path: item.path, generics, @@ -2635,11 +2937,11 @@ ${item.displayPath}${name}\ * * @param {RawFunctionSearchType} functionSearchType * @param {Array<{name: string, ty: number}>} lowercasePaths - * @param {Map} + * @param {Array<{name: string, ty: number}>} paths * * @return {null|FunctionSearchType} */ - function buildFunctionSearchType(functionSearchType, lowercasePaths) { + function buildFunctionSearchType(functionSearchType, lowercasePaths, paths) { const INPUTS_DATA = 0; const OUTPUT_DATA = 1; // `0` is used as a sentinel because it's fewer bytes than `null` @@ -2648,20 +2950,26 @@ ${item.displayPath}${name}\ } let inputs, output; if (typeof functionSearchType[INPUTS_DATA] === "number") { - inputs = [buildItemSearchType(functionSearchType[INPUTS_DATA], lowercasePaths)]; + inputs = [buildItemSearchType(functionSearchType[INPUTS_DATA], lowercasePaths, paths)]; } else { inputs = buildItemSearchTypeAll( functionSearchType[INPUTS_DATA], - lowercasePaths + lowercasePaths, + paths ); } if (functionSearchType.length > 1) { if (typeof functionSearchType[OUTPUT_DATA] === "number") { - output = [buildItemSearchType(functionSearchType[OUTPUT_DATA], lowercasePaths)]; + output = [buildItemSearchType( + functionSearchType[OUTPUT_DATA], + lowercasePaths, + paths + )]; } else { output = buildItemSearchTypeAll( functionSearchType[OUTPUT_DATA], - lowercasePaths + lowercasePaths, + paths ); } } else { @@ -2671,8 +2979,8 @@ ${item.displayPath}${name}\ const l = functionSearchType.length; for (let i = 2; i < l; ++i) { where_clause.push(typeof functionSearchType[i] === "number" - ? [buildItemSearchType(functionSearchType[i], lowercasePaths)] - : buildItemSearchTypeAll(functionSearchType[i], lowercasePaths)); + ? [buildItemSearchType(functionSearchType[i], lowercasePaths, paths)] + : buildItemSearchTypeAll(functionSearchType[i], lowercasePaths, paths)); } return { inputs, output, where_clause, @@ -2697,6 +3005,8 @@ ${item.displayPath}${name}\ typeNameIdOfArray = buildTypeMapIndex("array"); typeNameIdOfSlice = buildTypeMapIndex("slice"); typeNameIdOfArrayOrSlice = buildTypeMapIndex("[]"); + typeNameIdOfNever = buildTypeMapIndex("never"); + typeNameIdOfTuple = buildTypeMapIndex("tuple"); for (const crate in rawSearchIndex) { if (!hasOwnPropertyRustdoc(rawSearchIndex, crate)) { @@ -2856,7 +3166,8 @@ ${item.displayPath}${name}\ parent: itemParentIdxs[i] > 0 ? paths[itemParentIdxs[i] - 1] : undefined, type: buildFunctionSearchType( itemFunctionSearchTypes[i], - lowercasePaths + lowercasePaths, + paths ), id: id, normalizedName: word.indexOf("_") === -1 ? word : word.replace(/_/g, ""), diff --git a/src/tools/rustdoc-js/tester.js b/src/tools/rustdoc-js/tester.js index c7e6dd3615e94..9d979067d6a63 100644 --- a/src/tools/rustdoc-js/tester.js +++ b/src/tools/rustdoc-js/tester.js @@ -373,8 +373,35 @@ function loadSearchJS(doc_folder, resource_suffix) { return { doSearch: function(queryStr, filterCrate, currentCrate) { - return searchModule.execQuery(searchModule.parseQuery(queryStr), searchWords, + const results = searchModule.execQuery(searchModule.parseQuery(queryStr), searchWords, filterCrate, currentCrate); + for (const key in results) { + if (results[key]) { + for (const resultKey in results[key]) { + if (!Object.prototype.hasOwnProperty.call(results[key], resultKey)) { + continue; + } + const entry = results[key][resultKey]; + if (!entry) { + continue; + } + if (Object.prototype.hasOwnProperty.call(entry, "displayTypeSignature") && + entry.displayTypeSignature !== null && + entry.displayTypeSignature instanceof Array + ) { + entry.displayTypeSignature.forEach((value, index) => { + if (index % 2 === 1) { + entry.displayTypeSignature[index] = "*" + value + "*"; + } else { + entry.displayTypeSignature[index] = value; + } + }); + entry.displayTypeSignature = entry.displayTypeSignature.join(""); + } + } + } + } + return results; }, getCorrections: function(queryStr, filterCrate, currentCrate) { const parsedQuery = searchModule.parseQuery(queryStr); diff --git a/tests/rustdoc-gui/search-corrections.goml b/tests/rustdoc-gui/search-corrections.goml index aeb3c9b31a3fd..a8a3f3eda0e97 100644 --- a/tests/rustdoc-gui/search-corrections.goml +++ b/tests/rustdoc-gui/search-corrections.goml @@ -24,7 +24,7 @@ assert-css: (".search-corrections", { }) assert-text: ( ".search-corrections", - "Type \"notablestructwithlongnamr\" not found. Showing results for closest type name \"notablestructwithlongname\" instead." + "Type \"NotableStructWithLongNamr\" not found. Showing results for closest type name \"notablestructwithlongname\" instead." ) // Corrections do get shown on the "In Return Type" tab. @@ -35,7 +35,7 @@ assert-css: (".search-corrections", { }) assert-text: ( ".search-corrections", - "Type \"notablestructwithlongnamr\" not found. Showing results for closest type name \"notablestructwithlongname\" instead." + "Type \"NotableStructWithLongNamr\" not found. Showing results for closest type name \"notablestructwithlongname\" instead." ) // Now, explicit return values @@ -52,7 +52,7 @@ assert-css: (".search-corrections", { }) assert-text: ( ".search-corrections", - "Type \"notablestructwithlongnamr\" not found. Showing results for closest type name \"notablestructwithlongname\" instead." + "Type \"NotableStructWithLongNamr\" not found. Showing results for closest type name \"notablestructwithlongname\" instead." ) // Now, generic correction @@ -69,7 +69,7 @@ assert-css: (".search-corrections", { }) assert-text: ( ".search-corrections", - "Type \"notablestructwithlongnamr\" not found and used as generic parameter. Consider searching for \"notablestructwithlongname\" instead." + "Type \"NotableStructWithLongNamr\" not found and used as generic parameter. Consider searching for \"notablestructwithlongname\" instead." ) // Now, generic correction plus error @@ -86,7 +86,7 @@ assert-css: (".search-corrections", { }) assert-text: ( ".search-corrections", - "Type \"notablestructwithlongnamr\" not found and used as generic parameter. Consider searching for \"notablestructwithlongname\" instead." + "Type \"NotableStructWithLongNamr\" not found and used as generic parameter. Consider searching for \"notablestructwithlongname\" instead." ) go-to: "file://" + |DOC_PATH| + "/test_docs/index.html" @@ -102,5 +102,5 @@ assert-css: (".error", { }) assert-text: ( ".error", - "Query parser error: \"Generic type parameter notablestructwithlongnamr does not accept generic parameters\"." + "Query parser error: \"Generic type parameter NotableStructWithLongNamr does not accept generic parameters\"." ) diff --git a/tests/rustdoc-js/generics-impl.js b/tests/rustdoc-js/generics-impl.js index 5e33e224876fe..1e9d57dcf6e85 100644 --- a/tests/rustdoc-js/generics-impl.js +++ b/tests/rustdoc-js/generics-impl.js @@ -4,33 +4,61 @@ const EXPECTED = [ { 'query': 'Aaaaaaa -> u32', 'others': [ - { 'path': 'generics_impl::Aaaaaaa', 'name': 'bbbbbbb' }, + { + 'path': 'generics_impl::Aaaaaaa', + 'name': 'bbbbbbb', + 'displayTypeSignature': '*Aaaaaaa* -> *u32*' + }, ], }, { 'query': 'Aaaaaaa -> bool', 'others': [ - { 'path': 'generics_impl::Aaaaaaa', 'name': 'ccccccc' }, + { + 'path': 'generics_impl::Aaaaaaa', + 'name': 'ccccccc', + 'displayTypeSignature': '*Aaaaaaa* -> *bool*' + }, ], }, { 'query': 'Aaaaaaa -> usize', 'others': [ - { 'path': 'generics_impl::Aaaaaaa', 'name': 'read' }, + { + 'path': 'generics_impl::Aaaaaaa', + 'name': 'read', + 'displayTypeSignature': '*Aaaaaaa*, [] -> Result<*usize*>' + }, ], }, { 'query': 'Read -> u64', 'others': [ - { 'path': 'generics_impl::Ddddddd', 'name': 'eeeeeee' }, - { 'path': 'generics_impl::Ddddddd', 'name': 'ggggggg' }, + { + 'path': 'generics_impl::Ddddddd', + 'name': 'eeeeeee', + 'displayTypeSignature': 'impl *Read* -> *u64*' + }, + { + 'path': 'generics_impl::Ddddddd', + 'name': 'ggggggg', + 'displayTypeSignature': 'Ddddddd -> *u64*' + }, ], }, { 'query': 'trait:Read -> u64', 'others': [ - { 'path': 'generics_impl::Ddddddd', 'name': 'eeeeeee' }, - { 'path': 'generics_impl::Ddddddd', 'name': 'ggggggg' }, + { + 'path': 'generics_impl::Ddddddd', + 'name': 'eeeeeee', + 'displayTypeSignature': 'impl *Read* -> *u64*' + }, + { + 'path': 'generics_impl::Ddddddd', + 'name': 'ggggggg', + 'displayTypeSignature': 'Ddddddd -> *u64*' + }, ], }, { @@ -40,19 +68,31 @@ const EXPECTED = [ { 'query': 'bool -> u64', 'others': [ - { 'path': 'generics_impl::Ddddddd', 'name': 'fffffff' }, + { + 'path': 'generics_impl::Ddddddd', + 'name': 'fffffff', + 'displayTypeSignature': '*bool* -> *u64*' + }, ], }, { 'query': 'Ddddddd -> u64', 'others': [ - { 'path': 'generics_impl::Ddddddd', 'name': 'ggggggg' }, + { + 'path': 'generics_impl::Ddddddd', + 'name': 'ggggggg', + 'displayTypeSignature': '*Ddddddd* -> *u64*' + }, ], }, { 'query': '-> Ddddddd', 'others': [ - { 'path': 'generics_impl::Ddddddd', 'name': 'hhhhhhh' }, + { + 'path': 'generics_impl::Ddddddd', + 'name': 'hhhhhhh', + 'displayTypeSignature': '-> *Ddddddd*' + }, ], }, ]; diff --git a/tests/rustdoc-js/generics-match-ambiguity.js b/tests/rustdoc-js/generics-match-ambiguity.js index edce4268c5ac7..247e66697d5d9 100644 --- a/tests/rustdoc-js/generics-match-ambiguity.js +++ b/tests/rustdoc-js/generics-match-ambiguity.js @@ -8,15 +8,31 @@ const EXPECTED = [ { 'query': 'Wrap', 'in_args': [ - { 'path': 'generics_match_ambiguity', 'name': 'bar' }, - { 'path': 'generics_match_ambiguity', 'name': 'foo' }, + { + 'path': 'generics_match_ambiguity', + 'name': 'bar', + 'displayTypeSignature': '*Wrap*, Wrap' + }, + { + 'path': 'generics_match_ambiguity', + 'name': 'foo', + 'displayTypeSignature': '*Wrap*, Wrap' + }, ], }, { 'query': 'Wrap', 'in_args': [ - { 'path': 'generics_match_ambiguity', 'name': 'bar' }, - { 'path': 'generics_match_ambiguity', 'name': 'foo' }, + { + 'path': 'generics_match_ambiguity', + 'name': 'bar', + 'displayTypeSignature': '*Wrap*<*i32*, u32>, Wrap' + }, + { + 'path': 'generics_match_ambiguity', + 'name': 'foo', + 'displayTypeSignature': '*Wrap*<*i32*>, Wrap' + }, ], }, { diff --git a/tests/rustdoc-js/type-parameters.js b/tests/rustdoc-js/type-parameters.js index e695f189bb672..e637cb4fb254e 100644 --- a/tests/rustdoc-js/type-parameters.js +++ b/tests/rustdoc-js/type-parameters.js @@ -5,79 +5,79 @@ const EXPECTED = [ { query: '-> trait:Some', others: [ - { path: 'foo', name: 'alef' }, - { path: 'foo', name: 'alpha' }, + { path: 'foo', name: 'alef', displayTypeSignature: '-> impl *Some*' }, + { path: 'foo', name: 'alpha', displayTypeSignature: '-> impl *Some*' }, ], }, { query: '-> generic:T', others: [ - { path: 'foo', name: 'bet' }, - { path: 'foo', name: 'alef' }, - { path: 'foo', name: 'beta' }, + { path: 'foo', name: 'bet', displayTypeSignature: '_ -> *T*' }, + { path: 'foo', name: 'alef', displayTypeSignature: '-> *T*' }, + { path: 'foo', name: 'beta', displayTypeSignature: 'T -> *T*' }, ], }, { query: 'A -> B', others: [ - { path: 'foo', name: 'bet' }, + { path: 'foo', name: 'bet', displayTypeSignature: '*A* -> *B*' }, ], }, { query: 'A -> A', others: [ - { path: 'foo', name: 'beta' }, + { path: 'foo', name: 'beta', displayTypeSignature: '*A* -> *A*' }, ], }, { query: 'A, A', others: [ - { path: 'foo', name: 'alternate' }, + { path: 'foo', name: 'alternate', displayTypeSignature: '*A*, *A*' }, ], }, { query: 'A, B', others: [ - { path: 'foo', name: 'other' }, + { path: 'foo', name: 'other', displayTypeSignature: '*A*, *B*' }, ], }, { query: 'Other, Other', others: [ - { path: 'foo', name: 'other' }, - { path: 'foo', name: 'alternate' }, + { path: 'foo', name: 'other', displayTypeSignature: 'impl *Other*, impl *Other*' }, + { path: 'foo', name: 'alternate', displayTypeSignature: 'impl *Other*, impl *Other*' }, ], }, { query: 'generic:T', in_args: [ - { path: 'foo', name: 'bet' }, - { path: 'foo', name: 'beta' }, - { path: 'foo', name: 'other' }, - { path: 'foo', name: 'alternate' }, + { path: 'foo', name: 'bet', displayTypeSignature: '*T* -> _' }, + { path: 'foo', name: 'beta', displayTypeSignature: '*T* -> T' }, + { path: 'foo', name: 'other', displayTypeSignature: '*T*, _' }, + { path: 'foo', name: 'alternate', displayTypeSignature: '*T*, T' }, ], }, { query: 'generic:Other', in_args: [ - { path: 'foo', name: 'bet' }, - { path: 'foo', name: 'beta' }, - { path: 'foo', name: 'other' }, - { path: 'foo', name: 'alternate' }, + { path: 'foo', name: 'bet', displayTypeSignature: '*Other* -> _' }, + { path: 'foo', name: 'beta', displayTypeSignature: '*Other* -> Other' }, + { path: 'foo', name: 'other', displayTypeSignature: '*Other*, _' }, + { path: 'foo', name: 'alternate', displayTypeSignature: '*Other*, Other' }, ], }, { query: 'trait:Other', in_args: [ - { path: 'foo', name: 'other' }, - { path: 'foo', name: 'alternate' }, + { path: 'foo', name: 'other', displayTypeSignature: '_, impl *Other*' }, + { path: 'foo', name: 'alternate', displayTypeSignature: 'impl *Other*, impl *Other*' }, ], }, { query: 'Other', in_args: [ - { path: 'foo', name: 'other' }, - { path: 'foo', name: 'alternate' }, + { path: 'foo', name: 'other', displayTypeSignature: '_, impl *Other*' }, + { path: 'foo', name: 'alternate', displayTypeSignature: 'impl *Other*, impl *Other*' }, ], }, {