Skip to content

Commit 0417f7b

Browse files
authored
feat(docs): algolia powered search (#2952)
* feat(docs): add algolia search * Update search.vue * Update search.vue * add algolia API key + search improvemnts * Update styles.scss * Update styles.scss * Update search.vue * Update _slug.vue * Update _slug.vue * Update index.js * Update play.vue * Update index.js * Update styles.scss * Update componentdoc.vue * Update importdoc.vue * Update nuxt.config.js * Update importdoc.vue * Update importdoc.vue * Update componentdoc.vue * Update nuxt.config.js * Update styles.scss * Update importdoc.vue * Update styles.scss * Update styles.scss * Update nuxt.config.js * Update styles.scss * Update styles.scss * Update componentdoc.vue * Update importdoc.vue * Update styles.scss * Update styles.scss * Update styles.scss * Update styles.scss * Update styles.scss * Update search.vue
1 parent 6964f7b commit 0417f7b

File tree

10 files changed

+195
-117
lines changed

10 files changed

+195
-117
lines changed

docs/assets/scss/styles.scss

+37
Original file line numberDiff line numberDiff line change
@@ -145,9 +145,46 @@ pre.editable.error:after {
145145
background-color: rgba(255, 0, 0, 0.1);
146146
}
147147

148+
/* Additional styling for (responsive) markdown tables */
149+
.bv-docs-table {
150+
font-size: 90%;
151+
152+
thead > tr > th {
153+
min-width: 80px;
154+
}
155+
156+
code {
157+
white-space: nowrap !important;
158+
}
159+
}
160+
148161
/* CSS for table transition example */
149162
table#table-transition-example {
150163
.flip-list-move {
151164
transition: transform 1s;
152165
}
153166
}
167+
168+
/*
169+
* Docsearch overrides
170+
* See: https://github.com/twbs/bootstrap/blob/master/site/static/docs/4.3/assets/scss/_algolia.scss
171+
*/
172+
.algolia-autocomplete {
173+
.ds-dropdown-menu {
174+
background: #fff !important;
175+
border: 1px solid rgba(0, 0, 0, 0.1) !important;
176+
box-shadow: 0 0.5rem 1rem rgba(0, 0, 0, 0.175) !important;
177+
178+
.ds-suggestions {
179+
@media (min-width: 768px) {
180+
max-height: calc(100vh - 12rem);
181+
overflow-y: auto;
182+
}
183+
184+
.algolia-docsearch-suggestion--subcategory-inline,
185+
.algolia-docsearch-suggestion--title {
186+
display: inline-block !important;
187+
}
188+
}
189+
}
190+
}

docs/components/componentdoc.vue

+12-2
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,8 @@
3030
<b-table
3131
:items="propsItems"
3232
:fields="propsFields"
33+
class="bv-docs-table"
34+
responsive="sm"
3335
small
3436
head-variant="default"
3537
striped
@@ -60,6 +62,8 @@
6062
<b-table
6163
:items="[componentVModel]"
6264
:fields="['prop', 'event']"
65+
class="bv-docs-table"
66+
responsive="sm"
6367
small
6468
head-variant="default"
6569
striped
@@ -81,6 +85,8 @@
8185
<b-table
8286
:items="slots"
8387
:fields="slotsFields"
88+
class="bv-docs-table"
89+
responsive="sm"
8490
small
8591
head-variant="default"
8692
striped
@@ -98,6 +104,8 @@
98104
<b-table
99105
:items="events"
100106
:fields="eventsFields"
107+
class="bv-docs-table"
108+
responsive="sm"
101109
small
102110
head-variant="default"
103111
striped
@@ -110,7 +118,7 @@
110118
v-for="arg in value"
111119
:key="`event-${item.event}-${arg.arg ? arg.arg : 'none'}`"
112120
>
113-
<template v-if="arg.arg"><code>{{ arg.arg }}</code> - </template>
121+
<template v-if="arg.arg"><code class="text-nowrap">{{ arg.arg }}</code> - </template>
114122
<span>{{ arg.description }}</span>
115123
</div>
116124
</template>
@@ -128,6 +136,8 @@
128136
<b-table
129137
:items="rootEventListeners"
130138
:fields="rootEventListenersFields"
139+
class="bv-docs-table"
140+
responsive="sm"
131141
small
132142
head-variant="default"
133143
striped
@@ -141,7 +151,7 @@
141151
:key="`event-${item.event}-${arg.arg ? arg.arg : 'none'}`"
142152
>
143153
<template v-if="arg.arg">
144-
<code>{{ arg.arg }}</code>
154+
<code class="text-nowrap">{{ arg.arg }}</code>
145155
<span v-if="arg.description"> - {{ arg.description }}</span>
146156
</template>
147157
</div>

docs/components/importdoc.vue

+17-4
Original file line numberDiff line numberDiff line change
@@ -9,12 +9,18 @@
99
Importing individual {{ pluginTitle }} Components
1010
</anchored-heading>
1111

12-
<b-table :items="componentImports" small head-variant="default" striped>
12+
<b-table
13+
:items="componentImports"
14+
class="bv-docs-table"
15+
small
16+
striped
17+
head-variant="default"
18+
>
1319
<template slot="component" slot-scope="{ value }">
1420
<code class="text-nowrap">{{ value }}</code>
1521
</template>
1622
<template slot="importPath" slot-scope="{ value }">
17-
<code>{{ value }}</code>
23+
<code class="text-nowrap">{{ value }}</code>
1824
</template>
1925
</b-table>
2026

@@ -30,12 +36,19 @@
3036
Importing individual {{ pluginTitle }} Directives
3137
</anchored-heading>
3238

33-
<b-table :items="directiveImports" small head-variant="default" striped>
39+
<b-table
40+
:items="directiveImports"
41+
class="bv-docs-table"
42+
small
43+
striped
44+
responsive="sm"
45+
head-variant="default"
46+
>
3447
<template slot="directive" slot-scope="{ value }">
3548
<code class="text-nowrap">{{ value }}</code>
3649
</template>
3750
<template slot="importPath" slot-scope="{ value }">
38-
<code>{{ value }}</code>
51+
<code class="text-nowrap">{{ value }}</code>
3952
</template>
4053
</b-table>
4154

docs/components/search.vue

+62-101
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,14 @@
11
<template>
2-
<div class="bd-search d-flex align-items-center">
2+
<form
3+
class="bd-search d-flex align-items-center"
4+
@submit.stop.prevent
5+
>
36
<b-form-input
47
id="bd-search-input"
5-
v-model="search"
68
autocomplete="off"
79
type="search"
8-
placeholder="Search keywords..."
9-
aria-label="Search site"
10+
placeholder="Search..."
11+
aria-label="Search docs"
1012
></b-form-input>
1113
<button
1214
v-b-toggle.bd-docs-nav
@@ -32,119 +34,78 @@
3234
/>
3335
</svg>
3436
</button>
35-
<b-popover target="bd-search-input" placement="bottom" triggers="focus">
36-
<span v-if="search.length && Object.keys(results).length === 0">No results found</span>
37-
<span v-else-if="search.length"></span>
38-
<span v-else>Type something to start search</span>
39-
40-
<div
41-
v-for="(results, section, idx) in results"
42-
:key="section"
43-
:class="idx > 0 ? 'mt-2' : ''"
44-
role="group"
45-
>
46-
<h6 class="bd-text-purple my-1" v-html="section"></h6>
47-
<div v-for="t in results" :key="t.href" class="my-1">
48-
<b-link :to="t.href" @click="search = ''" v-html="t.title"></b-link>
49-
</div>
50-
</div>
51-
</b-popover>
52-
</div>
37+
</form>
5338
</template>
5439

5540
<script>
56-
import groupBy from 'lodash/groupBy'
57-
import intersectionBy from 'lodash/intersectionBy'
58-
import { makeTOC } from '~/utils'
59-
import { components, directives, reference, misc } from '~/content'
60-
61-
const SEARCH = []
62-
63-
const process = (readme, section, page) => {
64-
const baseURL = '/' + ['docs', section, page].filter(v => !!v).join('/')
65-
const { title, toc } = makeTOC(readme)
66-
;[...toc].forEach(({ label, href }) => {
67-
SEARCH.push({
68-
section: title,
69-
title: label,
70-
href: `${baseURL}${href}`.replace('/#', '#')
71-
})
72-
})
73-
}
41+
import { relativeUrl } from '../utils'
7442
75-
// Async build the search database
76-
import('~/markdown/intro/README.md' /* webpackChunkName: "docs/intro" */).then(readme =>
77-
process(readme.default, '', '')
78-
)
79-
Object.keys(components).forEach(page => {
80-
import(`~/../src/components/${page}/README.md` /* webpackChunkName: "docs/components" */).then(
81-
readme => process(readme.default, 'components', page)
82-
)
83-
})
84-
Object.keys(directives).forEach(page => {
85-
import(`~/../src/directives/${page}/README.md` /* webpackChunkName: "docs/directives" */).then(
86-
readme => process(readme.default, 'directives', page)
87-
)
88-
})
89-
Object.keys(reference).forEach(page => {
90-
import(`~/markdown/reference/${page}/README.md` /* webpackChunkName: "docs/reference" */).then(
91-
readme => process(readme.default, 'reference', page)
92-
)
93-
})
94-
Object.keys(misc).forEach(page => {
95-
import(`~/markdown/misc/${page}/README.md` /* webpackChunkName: "docs/misc" */).then(readme =>
96-
process(readme.default, 'misc', page)
97-
)
98-
})
43+
let scriptsInjected = false
9944
10045
export default {
10146
data() {
10247
return {
103-
search: ''
48+
docsearch: null
10449
}
10550
},
106-
computed: {
107-
results() {
108-
if (!this.search.length) {
109-
return {}
51+
mounted() {
52+
this.loadDocsearch().then(this.initDocsearch)
53+
},
54+
methods: {
55+
async loadDocsearch() {
56+
if (scriptsInjected) {
57+
return
11058
}
11159
112-
// Break the searh into individual terms
113-
const terms = this.search
114-
.replace(/\s+/g, ' ')
115-
.split(/\s+/)
116-
.filter(t => t)
117-
if (terms.length === 0) {
118-
return {}
119-
}
60+
// Search indexing config stored at:
61+
// https://github.com/algolia/docsearch-configs/blob/master/configs/bootstrap-vue.json
12062
121-
// Find results for each term
122-
let results = terms.map(term => this.resultsFor(term))
63+
const cdnBaseUrl = '//cdn.jsdelivr.net/docsearch.js/2/'
64+
const $body = document.body
12365
124-
// If no results return emptiness
125-
if (results.length === 0) {
126-
return {}
127-
}
66+
// Load JS
67+
const loadJs = new Promise(resolve => {
68+
let $script = document.createElement('script')
69+
$script.setAttribute('type', 'text/javascript')
70+
$script.setAttribute('src', `${cdnBaseUrl}docsearch.min.js`)
71+
$body.appendChild($script)
72+
$script.onload = resolve
73+
})
12874
129-
// Add our intersectionBy 'iteratee' key as the last array entry
130-
results.push('href')
131-
// Find the intersection (common) of all individual term results (all retults ANDed)
132-
results = intersectionBy(...results)
75+
// Load CSS
76+
const loadCss = new Promise(resolve => {
77+
let $link = document.createElement('link')
78+
$link.setAttribute('rel', 'stylesheet')
79+
$link.setAttribute('type', 'text/css')
80+
$link.setAttribute('href', `${cdnBaseUrl}docsearch.min.css`)
81+
$body.appendChild($link)
82+
$link.onload = resolve
83+
})
13384
134-
// Return the first 6 results or an empty array
135-
return groupBy(results.slice(0, 6), 'section')
136-
}
137-
},
138-
methods: {
139-
resultsFor(term) {
140-
// Return the search entries for a particular search term
141-
const regex = new RegExp(`\\b${term}`, 'i')
142-
return SEARCH.reduce((results, item) => {
143-
if (regex.test(item.title) || regex.test(item.section)) {
144-
results.push(item)
145-
}
146-
return results
147-
}, [])
85+
await Promise.all([loadJs, loadCss])
86+
87+
scriptsInjected = true
88+
},
89+
initDocsearch() {
90+
if (this.docsearch) {
91+
return
92+
}
93+
// Initialize docsearch
94+
this.docsearch = window.docsearch({
95+
apiKey: 'c816d3054b015320f0cfb40042f7e2bc',
96+
indexName: 'bootstrap-vue',
97+
inputSelector: '#bd-search-input',
98+
transformData(hits) {
99+
return hits.map(function(hit) {
100+
// Transform URL to a relative URL
101+
hit.url = relativeUrl(hit.url)
102+
103+
return hit
104+
})
105+
},
106+
// Set debug to `true` if you want to inspect the dropdown
107+
debug: false
108+
})
148109
}
149110
}
150111
}

docs/nuxt.config.js

+4-3
Original file line numberDiff line numberDiff line change
@@ -45,10 +45,11 @@ renderer.heading = function(text, level, raw, slugger) {
4545
// BS4 table support for markdown renderer
4646
const originalTable = renderer.table
4747
renderer.table = function(header, body) {
48-
let r = originalTable.apply(this, arguments)
49-
return r
50-
.replace('<table>', '<table class="table b-table table-striped">')
48+
let table = originalTable.apply(this, arguments)
49+
table = table
50+
.replace('<table>', '<table class="table b-table table-striped table-sm bv-docs-table">')
5151
.replace('<thead>', '<thead class="thead-default">')
52+
return `<div class="table-responsive-sm">${table}</div>`
5253
}
5354

5455
module.exports = {

docs/pages/docs/index.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ export default {
1313
defaultConfig
1414
}
1515
},
16-
template: `<div class="container bd-content">${readme}</div>`,
16+
// We use a string template here so that the docs README can do interpolation
17+
template: `<main class="container"><div class="bd-content">${readme}</div></main>`,
1718
layout: 'docs'
1819
}

docs/pages/docs/misc/_slug.vue

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
<template>
2-
<div class="container">
2+
<main class="container">
33
<div class="bd-content" v-html="readme"></div>
4-
</div>
4+
</main>
55
</template>
66

77
<script>

docs/pages/docs/reference/_slug.vue

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
<template>
2-
<div class="container">
2+
<main class="container">
33
<div v-play class="bd-content" v-html="readme"></div>
4-
</div>
4+
</main>
55
</template>
66

77
<script>

0 commit comments

Comments
 (0)