Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/HauntedThemes/ghost-search

A simple but powerful search library for Ghost Blogging Platform.
https://github.com/HauntedThemes/ghost-search

ghost search

Last synced: about 2 months ago
JSON representation

A simple but powerful search library for Ghost Blogging Platform.

Awesome Lists containing this project

README

        

[![Min Ghost Version](https://img.shields.io/badge/Min%20Ghost%20v.-%3E%3D%203.0.0-blue.svg)](https://github.com/TryGhost/Ghost)
[![npm version](https://badge.fury.io/js/ghost-search.svg)](https://www.npmjs.com/package/ghost-search)

# ghost-search

A simple but powerful search library for [Ghost Blogging Platform](https://ghost.org/).

## Setup

### Step 1 - Setup Content API Client Library

Get the [latest version](https://unpkg.com/@tryghost/content-api) of Ghost Content API Client Library from unpkg.com.

```js

```

Add the script before the `{{ghost_foot}}` tag. This will, most likely, be in `default.hbs`.

### Step 2 - Setup ghost-search

Open your theme directory and navigate to `assets` subdirectory. \
Create a directory called `js`, if there isn't already one in there, and add the minified version of `ghost-search` in it. \
Open `default.hbs` that is located at the root of your theme. \
At the bottom of this file you should see `{{ghost_foot}}`. \
Add the following code above it (after the content-api script at Step 1) and save it:

```

```

Add the following code, in a `.hbs` file, where you want to show the search input:

```html

```

Add the following code, in a `.hbs` file, where you want to show the search results:

```html


```

You will need to initialize `ghost-search` to make the search functional. \
Add the following js code after you included `ghost-search.min.js`:

```javascript

let ghostSearch = new GhostSearch({
key: '22444f78447824223cefc48062', // This is just a demo key. Replace the key with a real one. See Step 3.
url: 'https://demo.ghost.io', // This is just a demo host. Replace the demo url with a real one. See Step 3.
})

```

### Step 3 - Setup a Custom integration

Go in your Ghost's dashboard -> Integrations -> Add custom integration \
Set a name: Haunted Themes Search \
Get the Content API Key and replace the demo key with this one. [More details](https://ghost.org/docs/api/v3/content/#key). \
Get the admin domain. This will be different in some cases. [More details](https://ghost.org/docs/api/v3/content/#url).

## Use ghost-search from CDN

```html

```

## npm

```javascript
npm install ghost-search
```

## Live Examples

[Set a basic search](https://www.hauntedthemes.com/ghost-search-our-first-open-source-library-for-ghost/#set-a-basic-search) \
[Display a message if there are no posts found](https://www.hauntedthemes.com/ghost-search-our-first-open-source-library-for-ghost/#display-a-message-if-there-are-no-posts-found) \
[Search only through tags](https://www.hauntedthemes.com/ghost-search-our-first-open-source-library-for-ghost/#search-only-through-tags) \
[Search posts from a custom collection](https://www.hauntedthemes.com/ghost-search-our-first-open-source-library-for-ghost/#search-posts-from-a-custom-collection) \
[Search posts that are published after 01 Jan 2018](https://www.hauntedthemes.com/ghost-search-our-first-open-source-library-for-ghost/#search-posts-that-are-published-after-01-jan-2018) \
[Search through the title and content of posts](https://www.hauntedthemes.com/ghost-search-our-first-open-source-library-for-ghost/#search-through-the-title-and-content-of-posts) \
[Get the results when a button is clicked](https://www.hauntedthemes.com/ghost-search-our-first-open-source-library-for-ghost/#get-the-results-when-a-button-is-clicked) \
[Get proper results when your Ghost is on a sub-path](https://www.hauntedthemes.com/ghost-search-our-first-open-source-library-for-ghost/#get-proper-results-when-your-ghost-is-on-a-sub-path) \
[Set multiple instances of ghost-search on the same page](https://www.hauntedthemes.com/ghost-search-our-first-open-source-library-for-ghost/#set-multiple-instances-of-ghost-search-on-the-same-page) \
[Add a loading icon when you have a lot of posts](https://www.hauntedthemes.com/ghost-search-our-first-open-source-library-for-ghost/#add-a-loading-icon-when-you-have-a-lot-of-posts) \
[Limit the results displayed](https://www.hauntedthemes.com/ghost-search-our-first-open-source-library-for-ghost/#limit-the-results-displayed)

## Default Options

```js
{
host: '',
key: '',
version: 'v2',
input: '#ghost-search-field',
results: '#ghost-search-results',
button: '',
development: false,
defaultValue: '',
template: function(result) {
let url = [location.protocol, '//', location.host].join('');
return '' + result.title + '';
},
trigger: 'focus',
options: {
keys: [
'title'
],
limit: 100,
threshold: -3500,
allowTypo: false
},
api: {
resource: 'posts',
parameters: {
limit: 'all',
fields: ['title', 'slug'],
filter: '',
include: '',
order: '',
formats: '',
},
},
on: {
beforeDisplay: function(){},
afterDisplay: function(results){},
beforeFetch: function(){},
afterFetch: function(){}
}
}
```

## Options

### url

The url that needs to be set in order for Content API to properly authenticate. [More details](https://docs.ghost.org/api/content/#url).

### key

The key that needs to be set in order for Content API to properly authenticate. [More details](https://docs.ghost.org/api/content/#key).

### version

The version that needs to be set in order for Content API to properly authenticate. [More details](https://ghost.org/docs/api/v3/content/#path--version).

Default value: `'v3'`

### input

The ID of the input field that will be transformed into search filter. \
You can set your own id if you want like this:

```html

```

```html

let ghostSearch = new GhostSearch({
input: '#my-custom-input'
})

```

Default value: `'#ghost-search-field'`

### results

The ID of the element that will be transformed into search results. \
You can set your own id if you want like this:

```html


```

```html

let ghostSearch = new GhostSearch({
results: '#my-custom-results'
})

```

Default value: `'#ghost-search-results'`

### button

The ID of the element that will trigger the search results after it's clicked. \
By default, the button parameter is empty because `ghost-search` displays the results when you write in the input. \
To make this work you need to add the input and the button in a `form` element:

```html



```

```html

let ghostSearch = new GhostSearch({
input: '#my-custom-input',
results: '#my-custom-results',
button: '#my-custom-button'
})

```

Default value: `''`

### defaultValue

A parameter that will set a default value for the input and performs the search.

```html

let ghostSearch = new GhostSearch({
defaultValue: 'ghost'
})

```

Default value: `''`

### template

The template that will be used to render individual items in the search result. \
The method has a parameter `result` that stores all the data that you can use inside the method.

#### Examples:

Make the results a list and wrap each result with `

  • `:

    ```html



      ```

      ```html

      let ghostSearch = new GhostSearch({
      input: '#my-custom-input',
      results: '#my-custom-results',
      template: function(result) {
      let url = [location.protocol, '//', location.host].join('');
      return '<li><a href="' + url + '/' + result.slug + '">' + result.title + '</a></li>';
      }
      })

      ```

      Set url with sub-path:

      ```html

      let ghostSearch = new GhostSearch({
      input: '#my-custom-input',
      results: '#my-custom-results',
      template: function(result) {
      let url = [location.protocol, '//', location.host].join('') + '/sub-path/';
      return '<a href="' + url + '/' + result.slug + '">' + result.title + '</a>';
      }
      })

      ```

      Default value:

      ```js
      function(result) {
      let url = [location.protocol, '//', location.host].join('');
      return '' + result.title + '';
      }
      ```

      ### trigger

      Tells the script when to fetch the collection of data. The default value is `focus`, that means when a user clicks the input, all the data is fetched. \
      You can also use `load`. This will fetch the data when the page loads.

      `load` might **create a DDOS effect** because it loads all the data every time a page loads. Use carefully.

      Default value: `'focus'`

      ### options

      `ghost-search` is using `fuzzysort` as an algorithm for search. The `option` parameter supports all the options from [fuzzysort](https://github.com/farzher/fuzzysort#options).
      By default, `ghost-search` is showing the first 10 results and searches only based on title.

      Let's try another example that will show the first 3 results and searches both title and the content of a collection:

      ```html

      let ghostSearch = new GhostSearch({
      options: {
      keys: [
      'title',
      'plaintext'
      ],
      limit: 3,
      },
      api: {
      resource: 'posts',
      parameters: {
      fields: ['title', 'slug', 'plaintext'],
      formats: 'plaintext',
      },
      },
      })

      ```

      Default value:

      ```js
      {
      keys: [
      'title'
      ],
      limit: 10,
      threshold: -3500,
      allowTypo: false
      }
      ```

      ### api

      The api parameter is an object that supports most of the [resources](https://docs.ghost.org/api/content/#resources) and [parameters](https://docs.ghost.org/api/content/#parameters) by [Content API](https://docs.ghost.org/api/content/).

      Resources: [posts](https://docs.ghost.org/api/content/#posts), [tags](https://docs.ghost.org/api/content/#tags), [authors](https://docs.ghost.org/api/content/#authors) \
      Parameters: [fields](https://docs.ghost.org/api/content/#fields), [filter](https://docs.ghost.org/api/content/#filter), [include](https://docs.ghost.org/api/content/#include), [order](https://docs.ghost.org/api/content/#order), [formats](https://docs.ghost.org/api/content/#formats), [limit](https://docs.ghost.org/api/content/#limit)

      Examples:

      Search through tags:

      ```html

      let ghostSearch = new GhostSearch({
      options: {
      keys: [
      'name',
      ],
      },
      api: {
      resource: 'tags',
      parameters: {
      fields: ['name', 'slug'],
      },
      },
      template: function(result) {
      let url = [location.protocol, '//', location.host].join('') + '/tag';
      return '<a href="' + url + '/' + result.slug + '/">' + result.name + '</a>';
      },
      })

      ```

      Search through a custom collection:

      Let's say we have a `routes.yaml` like this:

      ```yaml
      routes:

      collections:
      /themes/:
      permalink: /themes/{slug}/
      filter: tag:themes
      data: tag.themes
      /:
      permalink: /{slug}/
      filter: tag:-themes
      template:
      - index

      taxonomies:
      tag: /tag/{slug}/
      author: /author/{slug}/
      ```

      `/themes/` is a collection that will show posts with tag `themes`. A post like this will have the url `example.com/themes/post-slug`.

      Our `ghost-search` will become:

      ```js
      let ghostSearch = new GhostSearch({
      options: {
      keys: [
      'title',
      ],
      },
      api: {
      resource: 'posts',
      parameters: {
      fields: ['title', 'slug'],
      filter: 'tags:[themes]',
      include: 'tags'
      },
      },
      template: function(result) {
      let collection = 'themes';
      let url = [location.protocol, '//', location.host].join('') + '/' + collection;
      return '' + result.title + '';
      },
      })
      ```

      ### on

      This parameter has 4 methods in it: `beforeDisplay`, `afterDisplay`, `beforeFetch`, `afterFetch`. \
      `afterDisplay` and `afterFetch` have a parameter `results` that contains the results fetched. \
      They are useful to do things before results are visible to users.

      Example:

      ```
      let ghostSearch = new GhostSearch({
      on: {
      beforeFetch: function(){
      // Create a div that has a spinning icon
      console.log('Loading appears');
      },
      afterFetch: function(results){
      // Remove the spinning icon
      console.log('Loading disappears');
      }
      }
      })
      ```

      ## Contributing

      All changes should be committed to `src/` files only.

      ## Known Issues
      * DDOS effect when `trigger` is set to `load`.
      If you have a lot of posts and set `trigger` to `load` you might get a DDOS effect because you are loading all the post everything a page loads. It would be better to just set `trigger` to `focus`.

      ## Changelog

      ### 1.1.0 - 15 Nov 2019
      * Migrated to Ghost v3

      ### 1.0.1 - 28 Jan 2019
      * Editable limit parameter. [5](https://github.com/HauntedThemes/ghost-search/issues/5)

      ### 1.0.0 - 21 Jan 2019
      * Public API (deprecated) will not work anymore. The library is accessing Content API.

      ### 0.1.1 - 29 Nov 2018
      * Added `defaultValue` parameter.

      ### 0.1.0 - 17 Sep 2018
      * Initial release

      ## Thank You
      ghost-search is using as a search algorithm [fuzzysort](https://github.com/farzher/fuzzysort). \
      Thank you [farzher](https://github.com/farzher/) for creating fuzzysort, a simple and usable search library.

      ## Copyright & License

      Copyright (c) 2019 Haunted Themes - Released under the [MIT license](LICENSE). \
      [Ghost is a trademark of The Ghost Foundation](https://ghost.org/)