From 297648f1e634900ce4e22498f9c061f670c7321f Mon Sep 17 00:00:00 2001
From: 梁灏 <admin@aresn.com>
Date: Tue, 19 Sep 2017 14:26:46 +0800
Subject: [PATCH] fixed #1063

---
 examples/routers/spin.vue                | 25 +++++++++++++++++++++++++
 src/components/modal/mixins-scrollbar.js | 34 ++++++++++++++++++++++++++++++++++
 src/components/modal/modal.vue           | 32 ++------------------------------
 src/components/spin/index.js             | 33 ++++++++++++++++++++++++++++++++-
 src/components/spin/spin.js              | 55 +++++++++++++++++++++++++++++++++++++++++++++++++++++++
 src/components/spin/spin.vue             | 29 +++++++++++++++++++++++++++--
 src/index.js                             |  1 +
 src/styles/components/spin.less          |  3 +++
 src/styles/custom.less                   |  1 +
 9 files changed, 180 insertions(+), 33 deletions(-)
 create mode 100644 src/components/modal/mixins-scrollbar.js
 create mode 100644 src/components/spin/spin.js

diff --git a/examples/routers/spin.vue b/examples/routers/spin.vue
index cc878a4..5500eb3 100644
--- a/examples/routers/spin.vue
+++ b/examples/routers/spin.vue
@@ -181,6 +181,8 @@
         </div>
         <br>
         切换显示状态:<i-switch @on-change="spinShow = !spinShow"></i-switch>
+        <Button @click="show">show</Button>
+        <Button @click="hide">hide</Button>
     </div>
 </template>
 <script>
@@ -189,6 +191,29 @@
             return {
                 spinShow: true
             }
+        },
+        methods: {
+            show () {
+                this.$Spin.show({
+                    render: (h) => {
+                        return h('div', [
+                            h('Icon', {
+                                props: {
+                                    type: 'load-c',
+                                    size: 24
+                                }
+                            }),
+                            h('div', 'Loading')
+                        ])
+                    }
+                });
+                setTimeout(() => {
+                    this.$Spin.hide();
+                }, 3000)
+            },
+            hide () {
+                this.$Spin.hide();
+            }
         }
     }
 </script>
diff --git a/src/components/modal/mixins-scrollbar.js b/src/components/modal/mixins-scrollbar.js
new file mode 100644
index 0000000..1570fb7
--- /dev/null
+++ b/src/components/modal/mixins-scrollbar.js
@@ -0,0 +1,34 @@
+// used for Modal & $Spin
+import { getScrollBarSize } from '../../utils/assist';
+export default {
+    methods: {
+        checkScrollBar () {
+            let fullWindowWidth = window.innerWidth;
+            if (!fullWindowWidth) { // workaround for missing window.innerWidth in IE8
+                const documentElementRect = document.documentElement.getBoundingClientRect();
+                fullWindowWidth = documentElementRect.right - Math.abs(documentElementRect.left);
+            }
+            this.bodyIsOverflowing = document.body.clientWidth < fullWindowWidth;
+            if (this.bodyIsOverflowing) {
+                this.scrollBarWidth = getScrollBarSize();
+            }
+        },
+        setScrollBar () {
+            if (this.bodyIsOverflowing && this.scrollBarWidth !== undefined) {
+                document.body.style.paddingRight = `${this.scrollBarWidth}px`;
+            }
+        },
+        resetScrollBar () {
+            document.body.style.paddingRight = '';
+        },
+        addScrollEffect () {
+            this.checkScrollBar();
+            this.setScrollBar();
+            document.body.style.overflow = 'hidden';
+        },
+        removeScrollEffect() {
+            document.body.style.overflow = '';
+            this.resetScrollBar();
+        }
+    }
+};
\ No newline at end of file
diff --git a/src/components/modal/modal.vue b/src/components/modal/modal.vue
index db4ae39..1eaa6fa 100644
--- a/src/components/modal/modal.vue
+++ b/src/components/modal/modal.vue
@@ -30,15 +30,15 @@
     import Icon from '../icon';
     import iButton from '../button/button.vue';
     import TransferDom from '../../directives/transfer-dom';
-    import { getScrollBarSize } from '../../utils/assist';
     import Locale from '../../mixins/locale';
     import Emitter from '../../mixins/emitter';
+    import ScrollbarMixins from './mixins-scrollbar';
 
     const prefixCls = 'ivu-modal';
 
     export default {
         name: 'Modal',
-        mixins: [ Locale, Emitter ],
+        mixins: [ Locale, Emitter, ScrollbarMixins ],
         components: { Icon, iButton },
         directives: { TransferDom },
         props: {
@@ -186,34 +186,6 @@
                     }
                 }
             },
-            checkScrollBar () {
-                let fullWindowWidth = window.innerWidth;
-                if (!fullWindowWidth) { // workaround for missing window.innerWidth in IE8
-                    const documentElementRect = document.documentElement.getBoundingClientRect();
-                    fullWindowWidth = documentElementRect.right - Math.abs(documentElementRect.left);
-                }
-                this.bodyIsOverflowing = document.body.clientWidth < fullWindowWidth;
-                if (this.bodyIsOverflowing) {
-                    this.scrollBarWidth = getScrollBarSize();
-                }
-            },
-            setScrollBar () {
-                if (this.bodyIsOverflowing && this.scrollBarWidth !== undefined) {
-                    document.body.style.paddingRight = `${this.scrollBarWidth}px`;
-                }
-            },
-            resetScrollBar () {
-                document.body.style.paddingRight = '';
-            },
-            addScrollEffect () {
-                this.checkScrollBar();
-                this.setScrollBar();
-                document.body.style.overflow = 'hidden';
-            },
-            removeScrollEffect() {
-                document.body.style.overflow = '';
-                this.resetScrollBar();
-            },
             animationFinish() {
                 this.$emit('on-hidden');
             }
diff --git a/src/components/spin/index.js b/src/components/spin/index.js
index 59e6d66..ffba3c7 100644
--- a/src/components/spin/index.js
+++ b/src/components/spin/index.js
@@ -1,2 +1,33 @@
-import Spin from './spin.vue';
+import Spin from './spin.js';
+
+let spinInstance;
+
+function getSpinInstance (render = undefined) {
+    spinInstance = spinInstance || Spin.newInstance({
+        render: render
+    });
+
+    return spinInstance;
+}
+
+function loading (options) {
+    const render = ('render' in options) ? options.render : undefined;
+    let instance  = getSpinInstance(render);
+
+    instance.show(options);
+}
+
+Spin.show = function (props = {}) {
+    return loading(props);
+};
+Spin.hide = function () {
+    if (!spinInstance) return false;
+
+    const instance = getSpinInstance();
+
+    instance.remove(() => {
+        spinInstance = null;
+    });
+};
+
 export default Spin;
\ No newline at end of file
diff --git a/src/components/spin/spin.js b/src/components/spin/spin.js
new file mode 100644
index 0000000..619a326
--- /dev/null
+++ b/src/components/spin/spin.js
@@ -0,0 +1,55 @@
+import Vue from 'vue';
+import Spin from './spin.vue';
+
+Spin.newInstance = properties => {
+    const _props = properties || {};
+
+    const Instance = new Vue({
+        data: Object.assign({}, _props, {
+
+        }),
+        render (h) {
+            let vnode = '';
+            if (this.render) {
+                vnode = h(Spin, {
+                    props: {
+                        fix: true,
+                        fullscreen: true
+                    }
+                }, [this.render(h)]);
+            } else {
+                vnode = h(Spin, {
+                    props: {
+                        size: 'large',
+                        fix: true,
+                        fullscreen: true
+                    }
+                });
+            }
+            return h('div', {
+                'class': 'ivu-spin-fullscreen'
+            }, [vnode]);
+        }
+    });
+
+    const component = Instance.$mount();
+    document.body.appendChild(component.$el);
+    const spin = Instance.$children[0];
+
+    return {
+        show () {
+            spin.visible = true;
+        },
+        remove (cb) {
+            spin.visible = false;
+            setTimeout(function() {
+                spin.$parent.$destroy();
+                document.body.removeChild(document.getElementsByClassName('ivu-spin-fullscreen')[0]);
+                cb();
+            }, 500);
+        },
+        component: spin
+    };
+};
+
+export default Spin;
\ No newline at end of file
diff --git a/src/components/spin/spin.vue b/src/components/spin/spin.vue
index 25a619d..9cb683c 100644
--- a/src/components/spin/spin.vue
+++ b/src/components/spin/spin.vue
@@ -1,6 +1,6 @@
 <template>
     <transition name="fade">
-        <div :class="classes">
+        <div :class="classes" v-if="fullscreenVisible">
             <div :class="mainClasses">
                 <span :class="dotClasses"></span>
                 <div :class="textClasses"><slot></slot></div>
@@ -10,11 +10,13 @@
 </template>
 <script>
     import { oneOf } from '../../utils/assist';
+    import ScrollbarMixins from '../modal/mixins-scrollbar';
 
     const prefixCls = 'ivu-spin';
 
     export default {
         name: 'Spin',
+        mixins: [ ScrollbarMixins ],
         props: {
             size: {
                 validator (value) {
@@ -24,11 +26,17 @@
             fix: {
                 type: Boolean,
                 default: false
+            },
+            fullscreen: {
+                type: Boolean,
+                default: false
             }
         },
         data () {
             return {
-                showText: false
+                showText: false,
+                // used for $Spin
+                visible: false
             };
         },
         computed: {
@@ -39,6 +47,7 @@
                         [`${prefixCls}-${this.size}`]: !!this.size,
                         [`${prefixCls}-fix`]: this.fix,
                         [`${prefixCls}-show-text`]: this.showText,
+                        [`${prefixCls}-fullscreen`]: this.fullscreen
                     }
                 ];
             },
@@ -50,6 +59,22 @@
             },
             textClasses () {
                 return `${prefixCls}-text`;
+            },
+            fullscreenVisible () {
+                if (this.fullscreen) {
+                    return this.visible;
+                } else {
+                    return true;
+                }
+            }
+        },
+        watch: {
+            visible (val) {
+                if (val) {
+                    this.addScrollEffect();
+                } else {
+                    this.removeScrollEffect();
+                }
             }
         },
         mounted () {
diff --git a/src/index.js b/src/index.js
index cd1b054..0b19973 100644
--- a/src/index.js
+++ b/src/index.js
@@ -139,6 +139,7 @@ const install = function (Vue, opts = {}) {
     Vue.prototype.$Message = Message;
     Vue.prototype.$Modal = Modal;
     Vue.prototype.$Notice = Notice;
+    Vue.prototype.$Spin = Spin;
 };
 
 // auto install
diff --git a/src/styles/components/spin.less b/src/styles/components/spin.less
index c5b35bb..9207cf1 100644
--- a/src/styles/components/spin.less
+++ b/src/styles/components/spin.less
@@ -33,6 +33,9 @@
         .square(100%);
         background-color: rgba(255,255,255,.9);
     }
+    &-fullscreen{
+        z-index: @zindex-spin-fullscreen;
+    }
 
     &-fix &-main {
         position: absolute;
diff --git a/src/styles/custom.less b/src/styles/custom.less
index 55c13bd..3ef5e28 100644
--- a/src/styles/custom.less
+++ b/src/styles/custom.less
@@ -149,6 +149,7 @@
 @zindex-tooltip         : 1060;
 @zindex-transfer        : 1060;
 @zindex-loading-bar     : 2000;
+@zindex-spin-fullscreen : 2010;
 
 // Animation
 @animation-time         : .3s;
--
libgit2 0.21.4