diff --git a/packages/core-data/src/fetch/__experimental-fetch-link-suggestions.ts b/packages/core-data/src/fetch/__experimental-fetch-link-suggestions.ts index 2901219758903..40eccb5e04051 100644 --- a/packages/core-data/src/fetch/__experimental-fetch-link-suggestions.ts +++ b/packages/core-data/src/fetch/__experimental-fetch-link-suggestions.ts @@ -256,9 +256,12 @@ export default async function fetchLinkSuggestions( * a taxonomy title might be more relevant than a post title, but by default taxonomy results will * be ordered after all the (potentially irrelevant) post results. * - * We sort by scoring each result, where the score is the number of tokens in the title that are - * also in the search query, divided by the total number of tokens in the title. This gives us a - * score between 0 and 1, where 1 is a perfect match. + * We sort by scoring each result, where the score is a combination of the number of exact matches + * and sub-matches for tokens in the title that are also in the search query, divided by the total + * number of tokens in the title. This gives us a score between 0 and 1, where 1 is a perfect match. + * + * We then apply various "weights" to the score to influence the sorting order with exact matches + * being more important than sub-matches. We also give a slight bonus to post-type results. * * @param results * @param search @@ -268,6 +271,7 @@ export function sortResults( results: SearchResult[], search: string ) { const scores = {}; for ( const result of results ) { + let score = 0; if ( result.title ) { const titleTokens = tokenize( result.title ); const exactMatchingTokens = titleTokens.filter( ( titleToken ) => @@ -285,17 +289,20 @@ export function sortResults( results: SearchResult[], search: string ) { // The score is a combination of exact matches and sub-matches. // More weight is given to exact matches, as they are more relevant (e.g. "cat" vs "caterpillar"). - // Diving by the total number of tokens in the title normalizes the score and skews - // the results towards shorter titles. const exactMatchScore = ( exactMatchingTokens.length / titleTokens.length ) * 10; const subMatchScore = subMatchingTokens.length / titleTokens.length; - scores[ result.id ] = exactMatchScore + subMatchScore; - } else { - scores[ result.id ] = 0; + score = exactMatchScore + subMatchScore; + } + + // Add a slight bonus for 'post-type' results + if ( result.kind === 'post-type' ) { + score *= 1.1; } + + scores[ result.id ] = score; } return results.sort( ( a, b ) => scores[ b.id ] - scores[ a.id ] ); diff --git a/packages/core-data/src/fetch/test/__experimental-fetch-link-suggestions.js b/packages/core-data/src/fetch/test/__experimental-fetch-link-suggestions.js index ad0014ff86ecb..1409192ed8027 100644 --- a/packages/core-data/src/fetch/test/__experimental-fetch-link-suggestions.js +++ b/packages/core-data/src/fetch/test/__experimental-fetch-link-suggestions.js @@ -430,6 +430,43 @@ describe( 'sortResults', () => { ); expect( order ).toEqual( [ 1, 4, 3, 2 ] ); } ); + + it( 'weight results towards Posts even if less relevant', () => { + const results = [ + { + id: 1, + title: 'Newspapers', + url: 'http://wordpress.local/more-relevant-news/', + type: 'page', + kind: 'taxonomy', + }, + { + id: 2, + title: 'News', + url: 'http://wordpress.local/most-relevant-news/', + type: 'page', + kind: 'media', + }, + { + id: 3, + title: 'Less Relevant News because it has a very long title', + url: 'http://wordpress.local/less-relevant-news/', + type: 'page', + kind: 'post-type', + }, + { + id: 4, + title: 'News', + url: 'http://wordpress.local/same-relevant-news/', + type: 'page', + kind: 'post-type', + }, + ]; + const order = sortResults( results, 'News' ).map( + ( result ) => result.id + ); + expect( order ).toEqual( [ 4, 2, 3, 1 ] ); + } ); } ); describe( 'tokenize', () => {