86 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
			
		
		
	
	
			86 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
| import shallowEqual from 'shallowequal'
 | |
| import omit from 'omit.js'
 | |
| import { getOptionProps } from '../props-util'
 | |
| import PropTypes from '../vue-types'
 | |
| 
 | |
| function getDisplayName (WrappedComponent) {
 | |
|   return WrappedComponent.name || 'Component'
 | |
| }
 | |
| 
 | |
| const defaultMapStateToProps = () => ({})
 | |
| export default function connect (mapStateToProps) {
 | |
|   const shouldSubscribe = !!mapStateToProps
 | |
|   const finnalMapStateToProps = mapStateToProps || defaultMapStateToProps
 | |
|   return function wrapWithConnect (WrappedComponent) {
 | |
|     const tempProps = omit(WrappedComponent.props || {}, ['store'])
 | |
|     const props = {}
 | |
|     Object.keys(tempProps).forEach(k => { props[k] = PropTypes.any })
 | |
|     const Connect = {
 | |
|       name: `Connect_${getDisplayName(WrappedComponent)}`,
 | |
|       props,
 | |
|       inject: {
 | |
|         _store: { default: {}},
 | |
|       },
 | |
|       data () {
 | |
|         this.store = this._store.store
 | |
|         return {
 | |
|           subscribed: finnalMapStateToProps(this.store.getState(), this.$props),
 | |
|         }
 | |
|       },
 | |
|       mounted () {
 | |
|         this.trySubscribe()
 | |
|       },
 | |
| 
 | |
|       beforeDestroy () {
 | |
|         this.tryUnsubscribe()
 | |
|       },
 | |
|       methods: {
 | |
|         handleChange () {
 | |
|           if (!this.unsubscribe) {
 | |
|             return
 | |
|           }
 | |
| 
 | |
|           const nextState = finnalMapStateToProps(this.store.getState(), this.$props)
 | |
|           if (!shallowEqual(this.nextState, nextState)) {
 | |
|             this.nextState = nextState
 | |
|             this.subscribed = nextState
 | |
|           }
 | |
|         },
 | |
| 
 | |
|         trySubscribe () {
 | |
|           if (shouldSubscribe) {
 | |
|             this.unsubscribe = this.store.subscribe(this.handleChange)
 | |
|             this.handleChange()
 | |
|           }
 | |
|         },
 | |
| 
 | |
|         tryUnsubscribe () {
 | |
|           if (this.unsubscribe) {
 | |
|             this.unsubscribe()
 | |
|             this.unsubscribe = null
 | |
|           }
 | |
|         },
 | |
|       },
 | |
|       render () {
 | |
|         const { $listeners, $slots, $attrs, $scopedSlots, subscribed, store } = this
 | |
|         const props = getOptionProps(this)
 | |
|         const wrapProps = {
 | |
|           props: {
 | |
|             ...props,
 | |
|             ...subscribed,
 | |
|             store,
 | |
|           },
 | |
|           on: $listeners,
 | |
|           attrs: $attrs,
 | |
|           slots: $slots,
 | |
|           scopedSlots: $scopedSlots,
 | |
|         }
 | |
|         return (
 | |
|           <WrappedComponent {...wrapProps}/>
 | |
|         )
 | |
|       },
 | |
|     }
 | |
|     return Connect
 | |
|   }
 | |
| }
 |