Skip to content
This repository was archived by the owner on Sep 20, 2024. It is now read-only.

Commit 93b82d5

Browse files
committed
feat(icon): added icon component
feat(icon): can use internal icons API feat(icon): can extend internal icons with custom icons feat(icon): can register icons from font-awesome
1 parent b6f9d11 commit 93b82d5

File tree

12 files changed

+116
-74
lines changed

12 files changed

+116
-74
lines changed

babel.config.js

-3
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,6 @@ module.exports = {
44
'@babel/preset-env'
55
],
66
env: {
7-
development: {
8-
plugins: ['transform-es2015-modules-commonjs']
9-
},
107
test: {
118
plugins: ['transform-es2015-modules-commonjs']
129
}

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616
},
1717
"dependencies": {
1818
"@babel/preset-env": "^7.6.3",
19+
"@fortawesome/free-solid-svg-icons": "^5.11.2",
1920
"@styled-system/should-forward-prop": "^5.1.2",
2021
"@vue/composition-api": "^0.3.2",
2122
"color": "^3.1.2",

src/App.vue

+16-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
<template>
2-
<theme-provider :theme="theme">
2+
<theme-provider :theme="theme" :icons="$kiwi.icons">
33
<div class="root">
44
<Icon
55
name="star"
@@ -10,13 +10,22 @@
1010
name="email"
1111
color="teal.600"
1212
size="8"
13-
mr="3"
1413
/>
1514
<Icon
1615
name="phone"
1716
color="indigo.400"
1817
size="10"
1918
/>
19+
<Icon
20+
name="ambulance"
21+
color="indigo.400"
22+
size="10"
23+
/>
24+
<Icon
25+
name="not-allowed"
26+
color="teal.600"
27+
size="12"
28+
/>
2029
</div>
2130
</theme-provider>
2231
</template>
@@ -47,6 +56,11 @@ export default {
4756
alert () {
4857
console.log('button clicked')
4958
}
59+
},
60+
computed: {
61+
iconPack () {
62+
return this.$kiwi.iconPack
63+
}
5064
}
5165
}
5266
</script>

src/components/Icon/Search.vue

-6
This file was deleted.

src/components/Icon/icon.utils.js

-9
Original file line numberDiff line numberDiff line change
@@ -24,12 +24,3 @@ export const iconStyles = (props) => {
2424
...sizeProps(props)
2525
}
2626
}
27-
28-
/**
29-
* @description Get reference to an Icon component if Font Awesome or MDI is used
30-
* @param {String} name - Name of the registered icon.
31-
* @returns {Object} Icon Component
32-
*/
33-
export const getIconComponent = (name) => {
34-
35-
}

src/components/Icon/index.js

+2-24
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,3 @@
1-
// import styled from 'vue-styled-components'
21
import { Box } from '../../lib/core/'
32
import iconPaths from '../../lib/plugin/iconsPaths'
43
import { forwardProps } from '../../lib/utils'
@@ -7,31 +6,12 @@ import { baseProps } from '../../lib/config/props'
76

87
const fallbackIcon = iconPaths['question-outline']
98

10-
/**
11-
* Features:
12-
* 1) Register custom icons
13-
* 2) Support Font Awesome icons.
14-
* 3) Support MD icons.
15-
*
16-
* CUSTOM ICONS:
17-
* - All icons exist as object.
18-
* - Load them in at project build time
19-
* - Should be extendable by users.
20-
* - If "name" prop is provided, search icons config for icon.
21-
*
22-
* SUPPORT FONTAWESOME/MDI
23-
* - Provide Object/Array of Fontawesome components
24-
* - Iterate over array of components to globally register sprites as components
25-
* - Accept "use" prop to calculate reference to globally registered component and feed it to Box component
26-
* - ^^ Use "as" prop to render component passed.
27-
*/
28-
299
/**
3010
* The Icon component renders SVGs for visual aid
3111
*/
3212
export default {
3313
name: 'Icon',
34-
inject: ['$theme', '$colorMode'],
14+
inject: ['$theme', '$icons'],
3515
props: {
3616
name: {
3717
type: [String, Array]
@@ -59,9 +39,7 @@ export default {
5939
render (h) {
6040
let icon, viewBox
6141
if (this.name) {
62-
icon = iconPaths[this.name]
63-
} else if (this.use) {
64-
// icon =
42+
icon = this.$icons[this.name]
6543
} else {
6644
console.warn(`[KiwiIcon]: You need to provide the "name" or "use" prop to for the Icon component`)
6745
}

src/components/ThemeProvider/index.js

+6-1
Original file line numberDiff line numberDiff line change
@@ -11,12 +11,17 @@ const ThemeProvider = {
1111
colorMode: {
1212
type: String,
1313
default: 'light'
14+
},
15+
icons: {
16+
type: Object,
17+
required: false
1418
}
1519
},
1620
provide () {
1721
return {
1822
$theme: () => this.theme,
19-
$colorMode: this.colorMode
23+
$colorMode: this.colorMode,
24+
$icons: this.icons
2025
}
2126
},
2227
render: function () {

src/lib/plugin/index.js

+19-17
Original file line numberDiff line numberDiff line change
@@ -1,27 +1,29 @@
1-
import Theme from '../../../kiwi.config'
1+
import { parsePackIcons } from '../utils/icons'
2+
import internalIcons from '../../lib/plugin/iconsPaths'
23

4+
/**
5+
* Kiwi Component library plugin
6+
*/
37
const Kiwi = {
48
install (Vue, options = {}) {
5-
// Create Kiwi instance and initialize theme
6-
const theme = {
7-
...Theme,
8-
...options.theme
9-
}
9+
let packIcons = {}
10+
const extendedIcons = options.icons.extend || {}
1011

11-
Vue.prototype.$kiwi = {
12-
theme
12+
if (options.icons.iconPack) {
13+
packIcons = parsePackIcons(options.icons.iconPack, options.icons.iconSet)
1314
}
1415

15-
// TODO:
16-
// - Make theme observable and use styled components to set theme dynamically
16+
const icons = {
17+
...internalIcons,
18+
...packIcons,
19+
...extendedIcons
20+
}
1721

18-
// Provide Theme via global mixin.
19-
// Vue.mixin({
20-
// provide: {
21-
// theme,
22-
// colorMode: 'light'
23-
// }
24-
// })
22+
// Bind theme and icons to prototype
23+
Vue.prototype.$kiwi = {
24+
theme: options.theme,
25+
icons
26+
}
2527
}
2628
}
2729

src/lib/utils/icons.js

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
import merge from 'lodash-es/merge'
2+
3+
const parseFAIcons = (iconSet) => {
4+
const parseFAIcon = (iconObject) => {
5+
const { icon } = iconObject
6+
return {
7+
[`${iconObject.iconName}`]: {
8+
path: `<path d="${icon[4]}" fa-key="${3}" fill="currentColor" />`,
9+
viewBox: `0 0 ${icon[0]} ${icon[1]}`
10+
}
11+
}
12+
}
13+
14+
const result = Object.values(iconSet)
15+
.map(value => parseFAIcon(value))
16+
.reduce((target, source) => merge(target, source), {})
17+
18+
console.log({ result })
19+
return result
20+
}
21+
22+
/**
23+
* @description Parse Icon packs
24+
* @param {String} pack Icon pack name
25+
* @param {Object} iconSet Registered Icon set
26+
* @returns {Object} Parsed pack icons object
27+
*/
28+
export const parsePackIcons = (pack, iconSet) => {
29+
const packIcons = pack === 'fa' ? parseFAIcons(iconSet) : {}
30+
return packIcons
31+
}

src/lib/utils/index.js

+1
Original file line numberDiff line numberDiff line change
@@ -3,3 +3,4 @@ export { provideTheme } from './provide-theme'
33
export { transformAlias as tx } from './transform'
44
export { pickProperty as forwardProps, filterPseudo } from './object'
55
export { addOpacity } from './color'
6+
export { parsePackIcons } from './icons'

src/main.js

+28-12
Original file line numberDiff line numberDiff line change
@@ -4,26 +4,42 @@ import App from './App.vue'
44
import './registerServiceWorker'
55
import Kiwi from './lib/plugin'
66

7-
Vue.config.productionTip = false
7+
// Import FA Icons
8+
import { faCoffee,
9+
faAmbulance,
10+
faCalendar,
11+
faCar,
12+
faBraille,
13+
faCaretLeft } from '@fortawesome/free-solid-svg-icons'
814

15+
Vue.config.productionTip = false
916
Vue.use(VueCompositionApi)
1017

1118
// Install Kiwi plugin
1219
Vue.use(Kiwi, {
13-
theme: {
14-
primary: 'hsl(209, 100%, 50%)',
15-
secondary: 'hsl(278, 100%, 69%)',
16-
success: 'hsl(160, 100%, 43%)',
17-
warning: 'hsl(40, 100%, 50%)',
18-
danger: 'hsl(350, 100%, 56%)',
19-
light: 'hsl(208, 100%, 94%)',
20-
dark: 'hsl(221, 15%, 29%)'
20+
icons: {
21+
iconPack: 'fa',
22+
iconSet: {
23+
faCalendar,
24+
faCar,
25+
faCoffee,
26+
faBraille,
27+
faAmbulance,
28+
faCaretLeft
29+
},
30+
extend: {
31+
'not-allowed': {
32+
path: `
33+
<path
34+
fill="currentColor"
35+
d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 18c-4.42 0-8-3.58-8-8 0-1.85.63-3.55 1.69-4.9L16.9 18.31C15.55 19.37 13.85 20 12 20zm6.31-3.1L7.1 5.69C8.45 4.63 10.15 4 12 4c4.42 0 8 3.58 8 8 0 1.85-.63 3.55-1.69 4.9z"
36+
/>
37+
`
38+
}
39+
}
2140
}
2241
})
2342

2443
new Vue({
25-
// Alternative way to provide theme would be to import { provideTheme } from utils and provide it with render function
26-
// Provide theme to the root of the application
27-
// render: h => provideTheme(h, App)
2844
render: h => h(App)
2945
}).$mount('#app')

yarn.lock

+12
Original file line numberDiff line numberDiff line change
@@ -1142,6 +1142,18 @@
11421142
resolved "https://registry.yarnpkg.com/@emotion/weak-memoize/-/weak-memoize-0.2.4.tgz#622a72bebd1e3f48d921563b4b60a762295a81fc"
11431143
integrity sha512-6PYY5DVdAY1ifaQW6XYTnOMihmBVT27elqSjEoodchsGjzYlEsTQMcEhSud99kVawatyTZRTiVkJ/c6lwbQ7nA==
11441144

1145+
"@fortawesome/fontawesome-common-types@^0.2.25":
1146+
version "0.2.25"
1147+
resolved "https://npm.fontawesome.com/@fortawesome/fontawesome-common-types/-/fontawesome-common-types-0.2.25.tgz#6df015905081f2762e5cfddeb7a20d2e9b16c786"
1148+
integrity sha512-3RuZPDuuPELd7RXtUqTCfed14fcny9UiPOkdr2i+cYxBoTOfQgxcDoq77fHiiHcgWuo1LoBUpvGxFF1H/y7s3Q==
1149+
1150+
"@fortawesome/free-solid-svg-icons@^5.11.2":
1151+
version "5.11.2"
1152+
resolved "https://npm.fontawesome.com/@fortawesome/free-solid-svg-icons/-/free-solid-svg-icons-5.11.2.tgz#2f2f1459743a27902b76655a0d0bc5ec4d945631"
1153+
integrity sha512-zBue4i0PAZJUXOmLBBvM7L0O7wmsDC8dFv9IhpW5QL4kT9xhhVUsYg/LX1+5KaukWq4/cbDcKT+RT1aRe543sg==
1154+
dependencies:
1155+
"@fortawesome/fontawesome-common-types" "^0.2.25"
1156+
11451157
11461158
version "2.1.1"
11471159
resolved "https://registry.yarnpkg.com/@hapi/address/-/address-2.1.1.tgz#61395b5ed94c4cb19c2dc4c85969cff3d40d583f"

0 commit comments

Comments
 (0)