├── .babelrc ├── .gitignore ├── App.vue ├── README-CN.md ├── README.md ├── assets └── logo.png ├── dist ├── vue-jstree.js └── vue-jstree.js.map ├── index.html ├── main.js ├── package-lock.json ├── package.json ├── pic.png ├── src ├── index.js ├── less │ ├── base.less │ ├── main.less │ ├── mixins.less │ └── style.less ├── tree-item.vue └── tree.vue └── webpack.config.js /.babelrc: -------------------------------------------------------------------------------- 1 | { 2 | "presets": [ 3 | ["env", { "modules": false }] 4 | ] 5 | } 6 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | node_modules/ 3 | npm-debug.log 4 | yarn-error.log 5 | 6 | # Editor directories and files 7 | .idea 8 | *.suo 9 | *.ntvs* 10 | *.njsproj 11 | *.sln 12 | 13 | gh-pages/ -------------------------------------------------------------------------------- /App.vue: -------------------------------------------------------------------------------- 1 | 183 | 184 | 504 | 505 | 567 | -------------------------------------------------------------------------------- /README-CN.md: -------------------------------------------------------------------------------- 1 | # vue-jstree 2 | 3 | [![npm](https://img.shields.io/npm/dt/vue-jstree.svg?style=flat-square)](https://github.com/zdy1988/vue-jstree) 4 | 5 | 6 | [English](./README.md)/[中文](./README-CN.md) 7 | 8 | ## 介绍 9 | 10 | A tree plugin for vue2 11 | 12 | 13 | 14 | ## 例子 15 | 16 | [http://zdy1988.github.io/vue-jstree](http://zdy1988.github.io/vue-jstree) 17 | 18 | ## NPM 19 | 20 | ```html 21 | npm install vue-jstree 22 | ``` 23 | 24 | ## ES6 调用 25 | 26 | ```html 27 | import VJstree from 'vue-jstree' 28 | 29 | new Vue({ 30 | components: { 31 | VJstree 32 | } 33 | }) 34 | ``` 35 | 36 | ## 安装 37 | 38 | ```html 39 | npm install 40 | npm run dev 41 | ``` 42 | 43 | ## 简单示例 44 | 45 | ```html 46 | 47 | 48 | new Vue({ 49 | data: { 50 | data: [ 51 | { 52 | "text": "Same but with checkboxes", 53 | "children": [ 54 | { 55 | "text": "initially selected", 56 | "selected": true 57 | }, 58 | { 59 | "text": "custom icon", 60 | "icon": "fa fa-warning icon-state-danger" 61 | }, 62 | { 63 | "text": "initially open", 64 | "icon": "fa fa-folder icon-state-default", 65 | "opened": true, 66 | "children": [ 67 | { 68 | "text": "Another node" 69 | } 70 | ] 71 | }, 72 | { 73 | "text": "custom icon", 74 | "icon": "fa fa-warning icon-state-warning" 75 | }, 76 | { 77 | "text": "disabled node", 78 | "icon": "fa fa-check icon-state-success", 79 | "disabled": true 80 | } 81 | ] 82 | }, 83 | { 84 | "text": "Same but with checkboxes", 85 | "opened": true, 86 | "children": [ 87 | { 88 | "text": "initially selected", 89 | "selected": true 90 | }, 91 | { 92 | "text": "custom icon", 93 | "icon": "fa fa-warning icon-state-danger" 94 | }, 95 | { 96 | "text": "initially open", 97 | "icon": "fa fa-folder icon-state-default", 98 | "opened": true, 99 | "children": [ 100 | { 101 | "text": "Another node" 102 | } 103 | ] 104 | }, 105 | { 106 | "text": "custom icon", 107 | "icon": "fa fa-warning icon-state-warning" 108 | }, 109 | { 110 | "text": "disabled node", 111 | "icon": "fa fa-check icon-state-success", 112 | "disabled": true 113 | } 114 | ] 115 | }, 116 | { 117 | "text": "And wholerow selection" 118 | } 119 | ] 120 | }, 121 | methods: { 122 | itemClick (node) { 123 | console.log(node.model.text + ' clicked !') 124 | } 125 | } 126 | }) 127 | ``` 128 | 129 | ## API 130 | 131 | | Props | Type | Default | Describe | 132 | | ------------- |:-------------:|:-----:|:--------------------------------------------------------| 133 | | data | Array | | 设置树的数据源 | 134 | | size | String | | 设置树节点的大小, 可选值 : 'large' or '' or ''small' | 135 | | show-checkbox | Boolean | false | 设置是否显示选择框 | 136 | | allow-transition | Boolean | true | 设置是否允许使用过渡效果 | 137 | | whole-row | Boolean | false | 设置是否整个树节点高亮 | 138 | | no-dots | Boolean | false | 设置是否显示树节点前的虚线 | 139 | | collapse | Boolean | false | 设置节点全部展开或合并的初始值,不设置按节点自身的 opened 属性控制 | 140 | | multiple | Boolean | false | 设置是否可以多选 | 141 | | allow-batch | Boolean | false | 设置允许批量选择子节点 | 142 | | text-field-name | String | 'text' | 设置 **文字** 的字段名称,默认为 **text** | 143 | | value-field-name | String | 'value' | 设置 **值** 的字段名称,默认为 **value** | 144 | | children-field-name | String | 'children' | 设置 **子节点** 的字段名称,默认为 **children** | 145 | | item-events | Object | {} | 注册任意事件到每个数节点上, [例子](https://github.com/zdy1988/vue-jstree/blob/master/App.vue) | 146 | | async | Function | | 异步加载数据的回调函数 , 如果当前节点没有子项 ,设置树节点中的 'isLeaf: true' 可不现实 + 号 | 147 | | loading-text | String | 'Loading' | 设置 Loading 文字 | 148 | | draggable | Boolean | false | 设置是否启用拖拽 , 默认全部节点可拖拽, 如自定义个别节点禁用拖拽或禁用拖放可设置 'dragDisabled: true' 和 'dropDisabled: true'| 149 | | drag-over-background-color | String | '#C9FDC9' | 设置拖拽进入节点时的背景色 | 150 | | klass | String | | 设置追加 class | 151 | 152 | ## node.model 中的方法 153 | 154 | | Method | Params | 155 | | ------------- |:-------------:| 156 | | addChild | (object) newDataItem | 157 | | addAfter | (object) newDataItem, (object) selectedNode | 158 | | addBefore | (object) newDataItem, (object) selectedNode | 159 | | openChildren | | 160 | | closeChildren | | 161 | 162 | ## 可选择事件 163 | 164 | **@item-click(node, item, e)** 165 | 166 | **@item-toggle(node, item, e)** 167 | 168 | **@item-drag-start(node, item, e)** 169 | 170 | **@item-drag-end(node, item, e)** 171 | 172 | **@item-drop-before(node, item, draggedItem, e)** 173 | 174 | **@item-drop(node, item, draggedItem, e)** 175 | 176 | **node** : 当前节点的 vue 对象 177 | 178 | **item** : 当前节点的数据对象 179 | 180 | **e** : 事件参数 181 | 182 | ## 节点的数据参数 183 | 184 | | Name | Type | Default | Describe | 185 | | ------------- |:-------------:| -----:|:----------------------------------------------| 186 | | icon | String | | 自定义图标样式 class | 187 | | opened | Boolean | false | 设置节点展开或合并 | 188 | | selected | Boolean | false | 设置节点被选择 | 189 | | disabled | Boolean | false | 设置禁用节点 | 190 | | isLeaf | Boolean | false | 如果节点没有子项 , 设置为 true 可以隐藏 '+' | 191 | | dragDisabled | Boolean | false | 设置当前节点禁止拖拽 | 192 | | dropDisabled | Boolean | false | 设置当前节点禁止拖放 | 193 | 194 | ## 自定义树节点的例子 195 | 196 | ``` 197 | 198 | 205 | 206 | ``` 207 | 208 | 更优雅的操作方式: 209 | 210 | ``` 211 | 212 | 218 | 219 | ``` 220 | 221 | **scope** 在 **vue@2.5.0+** 中被替换, **vue@2.5.0+** 以上的版本,请使用 **slot-scope** 222 | 223 | ## License 224 | 225 | Licensed under the [MIT license](https://opensource.org/licenses/mit-license.php). 226 | 227 | 感谢 [jstree](https://github.com/vakata/jstree) 的 样式支持 -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # vue-jstree 2 | 3 | [![npm](https://img.shields.io/npm/dt/vue-jstree.svg?style=flat-square)](https://github.com/zdy1988/vue-jstree) 4 | 5 | [English](./README.md)/[中文](./README-CN.md) 6 | 7 | ## Introduction 8 | 9 | A tree plugin for vue2 10 | 11 | 12 | 13 | ## DEMO 14 | 15 | [http://zdy1988.github.io/vue-jstree](http://zdy1988.github.io/vue-jstree) 16 | 17 | ## NPM 18 | 19 | ```html 20 | npm install vue-jstree 21 | ``` 22 | 23 | ## ES6 24 | 25 | ```html 26 | import VJstree from 'vue-jstree' 27 | 28 | new Vue({ 29 | components: { 30 | VJstree 31 | } 32 | }) 33 | ``` 34 | 35 | ## Setup 36 | 37 | ```html 38 | npm install 39 | npm run dev 40 | ``` 41 | 42 | ## Usage 43 | 44 | ```html 45 | 46 | 47 | new Vue({ 48 | data: { 49 | data: [ 50 | { 51 | "text": "Same but with checkboxes", 52 | "children": [ 53 | { 54 | "text": "initially selected", 55 | "selected": true 56 | }, 57 | { 58 | "text": "custom icon", 59 | "icon": "fa fa-warning icon-state-danger" 60 | }, 61 | { 62 | "text": "initially open", 63 | "icon": "fa fa-folder icon-state-default", 64 | "opened": true, 65 | "children": [ 66 | { 67 | "text": "Another node" 68 | } 69 | ] 70 | }, 71 | { 72 | "text": "custom icon", 73 | "icon": "fa fa-warning icon-state-warning" 74 | }, 75 | { 76 | "text": "disabled node", 77 | "icon": "fa fa-check icon-state-success", 78 | "disabled": true 79 | } 80 | ] 81 | }, 82 | { 83 | "text": "Same but with checkboxes", 84 | "opened": true, 85 | "children": [ 86 | { 87 | "text": "initially selected", 88 | "selected": true 89 | }, 90 | { 91 | "text": "custom icon", 92 | "icon": "fa fa-warning icon-state-danger" 93 | }, 94 | { 95 | "text": "initially open", 96 | "icon": "fa fa-folder icon-state-default", 97 | "opened": true, 98 | "children": [ 99 | { 100 | "text": "Another node" 101 | } 102 | ] 103 | }, 104 | { 105 | "text": "custom icon", 106 | "icon": "fa fa-warning icon-state-warning" 107 | }, 108 | { 109 | "text": "disabled node", 110 | "icon": "fa fa-check icon-state-success", 111 | "disabled": true 112 | } 113 | ] 114 | }, 115 | { 116 | "text": "And wholerow selection" 117 | } 118 | ] 119 | }, 120 | methods: { 121 | itemClick (node) { 122 | console.log(node.model.text + ' clicked !') 123 | } 124 | } 125 | }) 126 | ``` 127 | 128 | ## API 129 | 130 | | Props | Type | Default | Describe | 131 | | ------------- |:-------------:|:-----:|:--------------------------------------------------------| 132 | | data | Array | | set tree data | 133 | | size | String | | set tree item size , value : 'large' or '' or ''small' | 134 | | show-checkbox | Boolean | false | set it show checkbox | 135 | | allow-transition | Boolean | true | allow use transition animation | 136 | | whole-row | Boolean | false | use whole row state | 137 | | no-dots | Boolean | false | show or hide dots | 138 | | collapse | Boolean | false | set all tree item collapse state | 139 | | multiple | Boolean | false | set multiple selected tree item | 140 | | allow-batch | Boolean | false | in multiple choices. allow batch select | 141 | | text-field-name | String | 'text' | set tree item display field | 142 | | value-field-name | String | 'value' | set tree item value field | 143 | | children-field-name | String | 'children' | set tree item children field | 144 | | item-events | Object | {} | register any event to tree item, [example](https://github.com/zdy1988/vue-jstree/blob/master/App.vue) | 145 | | async | Function | | async load callback function , if node is a leaf ,you can set 'isLeaf: true' in data | 146 | | loading-text | String | 'Loading' | set loading text | 147 | | draggable | Boolean | false | set tree item can be dragged , selective drag and drop can set 'dragDisabled: true' and 'dropDisabled: true' , all default value is 'false' | 148 | | drag-over-background-color | String | '#C9FDC9' | set drag over background color | 149 | | klass | String | | set append tree class | 150 | 151 | ## Methods in node.model 152 | 153 | | Method | Params | 154 | | ------------- |:-------------:| 155 | | addChild | (object) newDataItem | 156 | | addAfter | (object) newDataItem, (object) selectedNode | 157 | | addBefore | (object) newDataItem, (object) selectedNode | 158 | | openChildren | | 159 | | closeChildren | | 160 | 161 | ## Event 162 | 163 | **@item-click(node, item, e)** 164 | 165 | **@item-toggle(node, item, e)** 166 | 167 | **@item-drag-start(node, item, e)** 168 | 169 | **@item-drag-end(node, item, e)** 170 | 171 | **@item-drop-before(node, item, draggedItem, e)** 172 | 173 | **@item-drop(node, item, draggedItem, e)** 174 | 175 | **node** : current node vue object 176 | 177 | **item** : current node data item object 178 | 179 | **e** : event 180 | 181 | ## Data Item Optional Properties 182 | 183 | | Name | Type | Default | Describe | 184 | | ------------- |:-------------:| -----:|:----------------------------------------------| 185 | | icon | String | | custom icon css class | 186 | | opened | Boolean | false | set leaf opened | 187 | | selected | Boolean | false | set node selected | 188 | | disabled | Boolean | false | set node disabled | 189 | | isLeaf | Boolean | false | if node is a leaf , set true can hide '+' | 190 | | dragDisabled | Boolean | false | selective drag | 191 | | dropDisabled | Boolean | false | selective drop | 192 | 193 | ## Custom Item Example 194 | 195 | ``` 196 | 197 | 204 | 205 | 206 | ``` 207 | 208 | more elegant: 209 | 210 | ``` 211 | 212 | 218 | 219 | ``` 220 | 221 | **scope** be replaced in the **vue@2.5.0+** , over **vue@2.5.0+** use **slot-scope** 222 | 223 | ## License 224 | 225 | Licensed under the [MIT license](https://opensource.org/licenses/mit-license.php). 226 | 227 | Thanks For [jstree](https://github.com/vakata/jstree)'s UI -------------------------------------------------------------------------------- /assets/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/zdy1988/vue-jstree/40bf85e612ab96804a57e823ffec07441b984120/assets/logo.png -------------------------------------------------------------------------------- /dist/vue-jstree.js: -------------------------------------------------------------------------------- 1 | !function(e,A){"object"==typeof exports&&"object"==typeof module?module.exports=A():"function"==typeof define&&define.amd?define("vue-jstree",[],A):"object"==typeof exports?exports["vue-jstree"]=A():e["vue-jstree"]=A()}(this,function(){return function(e){function A(r){if(t[r])return t[r].exports;var n=t[r]={i:r,l:!1,exports:{}};return e[r].call(n.exports,n,n.exports,A),n.l=!0,n.exports}var t={};return A.m=e,A.c=t,A.i=function(e){return e},A.d=function(e,t,r){A.o(e,t)||Object.defineProperty(e,t,{configurable:!1,enumerable:!0,get:r})},A.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return A.d(t,"a",t),t},A.o=function(e,A){return Object.prototype.hasOwnProperty.call(e,A)},A.p="dist/",A(A.s=4)}([function(e,A){e.exports=function(e,A,t,r,n){var o,a=e=e||{},l=typeof e.default;"object"!==l&&"function"!==l||(o=e,a=e.default);var i="function"==typeof a?a.options:a;A&&(i.render=A.render,i.staticRenderFns=A.staticRenderFns),r&&(i._scopeId=r);var d;if(n?(d=function(e){e=e||this.$vnode&&this.$vnode.ssrContext||this.parent&&this.parent.$vnode&&this.parent.$vnode.ssrContext,e||"undefined"==typeof __VUE_SSR_CONTEXT__||(e=__VUE_SSR_CONTEXT__),t&&t.call(this,e),e&&e._registeredComponents&&e._registeredComponents.add(n)},i._ssrRegister=d):t&&(d=t),d){var g=i.functional,s=g?i.render:i.beforeCreate;g?i.render=function(e,A){return d.call(A),s(e,A)}:i.beforeCreate=s?[].concat(s,d):[d]}return{esModule:o,exports:a,options:i}}},function(e,A,t){function r(e){t(10)}var n=t(0)(t(3),t(9),r,null,null);e.exports=n.exports},function(e,A,t){"use strict";function r(e,A,t){return A in e?Object.defineProperty(e,A,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[A]=t,e}Object.defineProperty(A,"__esModule",{value:!0}),A.default={name:"TreeItem",props:{data:{type:Object,required:!0},textFieldName:{type:String},valueFieldName:{type:String},childrenFieldName:{type:String},itemEvents:{type:Object},wholeRow:{type:Boolean,default:!1},showCheckbox:{type:Boolean,default:!1},allowTransition:{type:Boolean,default:!0},height:{type:Number,default:24},parentItem:{type:Array},draggable:{type:Boolean,default:!1},dragOverBackgroundColor:{type:String},onItemClick:{type:Function,default:function(){return!1}},onItemToggle:{type:Function,default:function(){return!1}},onItemDragStart:{type:Function,default:function(){return!1}},onItemDragEnd:{type:Function,default:function(){return!1}},onItemDrop:{type:Function,default:function(){return!1}},klass:String},data:function(){return{isHover:!1,isDragEnter:!1,model:this.data,maxHeight:0,events:{}}},watch:{isDragEnter:function(e){this.$el.style.backgroundColor=e?this.dragOverBackgroundColor:"inherit"},data:function(e){this.model=e},"model.opened":{handler:function(e,A){this.onItemToggle(this,this.model),this.handleGroupMaxHeight()},deep:!0}},computed:{isFolder:function(){return this.model[this.childrenFieldName]&&this.model[this.childrenFieldName].length},classes:function(){return[{"tree-node":!0},{"tree-open":this.model.opened},{"tree-closed":!this.model.opened},{"tree-leaf":!this.isFolder},{"tree-loading":!!this.model.loading},{"tree-drag-enter":this.isDragEnter},r({},this.klass,!!this.klass)]},anchorClasses:function(){return[{"tree-anchor":!0},{"tree-disabled":this.model.disabled},{"tree-selected":this.model.selected},{"tree-hovered":this.isHover}]},wholeRowClasses:function(){return[{"tree-wholerow":!0},{"tree-wholerow-clicked":this.model.selected},{"tree-wholerow-hovered":this.isHover}]},themeIconClasses:function(){return[{"tree-icon":!0},{"tree-themeicon":!0},r({},this.model.icon,!!this.model.icon),{"tree-themeicon-custom":!!this.model.icon}]},isWholeRow:function(){if(this.wholeRow)return void 0===this.$parent.model||!0===this.$parent.model.opened},groupStyle:function(){return{position:this.model.opened?"":"relative","max-height":this.allowTransition?this.maxHeight+"px":"","transition-duration":this.allowTransition?300*Math.ceil(this.model[this.childrenFieldName].length/100)+"ms":"","transition-property":this.allowTransition?"max-height":"",display:this.allowTransition?"block":this.model.opened?"block":"none"}}},methods:{handleItemToggle:function(e){this.isFolder&&(this.model.opened=!this.model.opened,this.onItemToggle(this,this.model))},handleGroupMaxHeight:function(){if(this.allowTransition){var e=0,A=0;if(this.model.opened){e=this.$children.length;var t=!0,r=!1,n=void 0;try{for(var o,a=this.$children[Symbol.iterator]();!(t=(o=a.next()).done);t=!0){A+=o.value.maxHeight}}catch(e){r=!0,n=e}finally{try{!t&&a.return&&a.return()}finally{if(r)throw n}}}this.maxHeight=e*this.height+A,"tree-item"===this.$parent.$options._componentTag&&this.$parent.handleGroupMaxHeight()}},handleItemClick:function(e){this.model.disabled||(this.model.selected=!this.model.selected,this.onItemClick(this,this.model,e))},handleItemMouseOver:function(){this.isHover=!0},handleItemMouseOut:function(){this.isHover=!1},handleItemDrop:function(e,A,t){this.$el.style.backgroundColor="inherit",this.onItemDrop(e,A,t)}},created:function(){var e=this,A=this,t={click:this.handleItemClick,mouseover:this.handleItemMouseOver,mouseout:this.handleItemMouseOut};for(var r in this.itemEvents)!function(r){var n=e.itemEvents[r];if(t.hasOwnProperty(r)){var o=t[r];t[r]=function(e){o(A,A.model,e),n(A,A.model,e)}}else t[r]=function(e){n(A,A.model,e)}}(r);this.events=t},mounted:function(){this.handleGroupMaxHeight()}}},function(e,A,t){"use strict";function r(e,A,t){return A in e?Object.defineProperty(e,A,{value:t,enumerable:!0,configurable:!0,writable:!0}):e[A]=t,e}Object.defineProperty(A,"__esModule",{value:!0});var n=t(7),o=t.n(n),a="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e},l=0;A.default={name:"VJstree",props:{data:{type:Array},size:{type:String,validator:function(e){return["large","small"].indexOf(e)>-1}},showCheckbox:{type:Boolean,default:!1},wholeRow:{type:Boolean,default:!1},noDots:{type:Boolean,default:!1},collapse:{type:Boolean,default:!1},multiple:{type:Boolean,default:!1},allowBatch:{type:Boolean,default:!1},allowTransition:{type:Boolean,default:!0},textFieldName:{type:String,default:"text"},valueFieldName:{type:String,default:"value"},childrenFieldName:{type:String,default:"children"},itemEvents:{type:Object,default:function(){return{}}},async:{type:Function},loadingText:{type:String,default:"Loading..."},draggable:{type:Boolean,default:!1},dragOverBackgroundColor:{type:String,default:"#C9FDC9"},klass:String},data:function(){return{draggedItem:void 0,draggedElm:void 0}},computed:{classes:function(){return[{tree:!0},{"tree-default":!this.size},r({},"tree-default-"+this.size,!!this.size),{"tree-checkbox-selection":!!this.showCheckbox},r({},this.klass,!!this.klass)]},containerClasses:function(){return[{"tree-container-ul":!0},{"tree-children":!0},{"tree-wholerow-ul":!!this.wholeRow},{"tree-no-dots":!!this.noDots}]},sizeHeight:function(){switch(this.size){case"large":return 32;case"small":return 18;default:return 24}}},methods:{initializeData:function(e){if(e&&e.length>0)for(var A in e){var t=this.initializeDataItem(e[A]);e[A]=t,this.initializeData(e[A][this.childrenFieldName])}},initializeDataItem:function(e){function A(e,A,t,r,n){this.id=e.id||l++,this[A]=e[A]||"",this[t]=e[t]||e[A],this.icon=e.icon||"",this.opened=e.opened||n,this.selected=e.selected||!1,this.disabled=e.disabled||!1,this.loading=e.loading||!1,this[r]=e[r]||[]}var t=Object.assign(new A(e,this.textFieldName,this.valueFieldName,this.childrenFieldName,this.collapse),e),r=this;return t.addBefore=function(e,A){var n=r.initializeDataItem(e),o=A.parentItem.findIndex(function(e){return e.id===t.id});A.parentItem.splice(o,0,n)},t.addAfter=function(e,A){var n=r.initializeDataItem(e),o=A.parentItem.findIndex(function(e){return e.id===t.id})+1;A.parentItem.splice(o,0,n)},t.addChild=function(e){var A=r.initializeDataItem(e);t.opened=!0,t[r.childrenFieldName].push(A)},t.openChildren=function(){t.opened=!0,r.handleRecursionNodeChildren(t,function(e){e.opened=!0})},t.closeChildren=function(){t.opened=!1,r.handleRecursionNodeChildren(t,function(e){e.opened=!1})},t},initializeLoading:function(){var e={};return e[this.textFieldName]=this.loadingText,e.disabled=!0,e.loading=!0,this.initializeDataItem(e)},handleRecursionNodeChilds:function(e,A){if(!1!==A(e)&&e.$children&&e.$children.length>0){var t=!0,r=!1,n=void 0;try{for(var o,a=e.$children[Symbol.iterator]();!(t=(o=a.next()).done);t=!0){var l=o.value;l.disabled||this.handleRecursionNodeChilds(l,A)}}catch(e){r=!0,n=e}finally{try{!t&&a.return&&a.return()}finally{if(r)throw n}}}},handleRecursionNodeChildren:function(e,A){if(!1!==A(e)&&e[this.childrenFieldName]&&e[this.childrenFieldName].length>0){var t=!0,r=!1,n=void 0;try{for(var o,a=e[this.childrenFieldName][Symbol.iterator]();!(t=(o=a.next()).done);t=!0){var l=o.value;this.handleRecursionNodeChildren(l,A)}}catch(e){r=!0,n=e}finally{try{!t&&a.return&&a.return()}finally{if(r)throw n}}}},onItemClick:function(e,A,t){this.multiple?this.allowBatch&&this.handleBatchSelectItems(e,A):this.handleSingleSelectItems(e,A),this.$emit("item-click",e,A,t)},handleSingleSelectItems:function(e,A){this.handleRecursionNodeChilds(this,function(e){e.model&&(e.model.selected=!1)}),e.model.selected=!0},handleBatchSelectItems:function(e,A){this.handleRecursionNodeChilds(e,function(A){A.model.disabled||(A.model.selected=e.model.selected)})},onItemToggle:function(e,A,t){e.model.opened&&this.handleAsyncLoad(e.model[this.childrenFieldName],e,A),this.$emit("item-toggle",e,A,t)},handleAsyncLoad:function(e,A,t){var r=this;this.async&&e[0].loading&&this.async(A,function(t){if(t.length>0)for(var n in t){t[n].isLeaf||"object"!==a(t[n][r.childrenFieldName])&&(t[n][r.childrenFieldName]=[r.initializeLoading()]);var o=r.initializeDataItem(t[n]);r.$set(e,n,o)}else A.model[r.childrenFieldName]=[]})},onItemDragStart:function(e,A,t){if(!this.draggable||t.dragDisabled)return!1;e.dataTransfer.effectAllowed="move",e.dataTransfer.setData("text",null),this.draggedElm=e.target,this.draggedItem={item:t,parentItem:A.parentItem,index:A.parentItem.findIndex(function(e){return e.id===t.id})},this.$emit("item-drag-start",A,t,e)},onItemDragEnd:function(e,A,t){this.draggedItem=void 0,this.draggedElm=void 0,this.$emit("item-drag-end",A,t,e)},onItemDrop:function(e,A,t){var r=this;if(!this.draggable||t.dropDisabled)return!1;if(this.$emit("item-drop-before",A,t,this.draggedItem?this.draggedItem.item:void 0,e),this.draggedElm&&this.draggedElm!==e.target&&!this.draggedElm.contains(e.target)&&this.draggedItem){if(this.draggedItem.parentItem===t[this.childrenFieldName]||this.draggedItem.item===t||t[this.childrenFieldName]&&-1!==t[this.childrenFieldName].findIndex(function(e){return e.id===r.draggedItem.item.id}))return;t[this.childrenFieldName]?t[this.childrenFieldName].push(this.draggedItem.item):t[this.childrenFieldName]=[this.draggedItem.item],t.opened=!0;var n=this.draggedItem;this.$nextTick(function(){n.parentItem.splice(n.index,1)}),this.$emit("item-drop",A,t,n.item,e)}}},created:function(){this.initializeData(this.data)},mounted:function(){this.async&&(this.$set(this.data,0,this.initializeLoading()),this.handleAsyncLoad(this.data,this))},components:{TreeItem:o.a}}},function(e,A,t){"use strict";Object.defineProperty(A,"__esModule",{value:!0});var r=t(1),n=t.n(r);n.a.install=function(e){e.component(n.a.name,n.a)},"undefined"!=typeof window&&window.Vue&&window.Vue.use(n.a),A.default=n.a},function(e,A,t){A=e.exports=t(6)(),A.push([e.i,'.tree-children,.tree-container-ul,.tree-node{display:block;margin:0;padding:0;list-style-type:none;list-style-image:none}.tree-children{overflow:hidden}.tree-anchor,.tree-node{white-space:nowrap}.tree-anchor{display:inline-block;color:#000;padding:0 4px 0 1px;margin:0;vertical-align:top;font-size:14px;cursor:pointer}.tree-anchor:focus{outline:0}.tree-anchor,.tree-anchor:active,.tree-anchor:hover,.tree-anchor:link,.tree-anchor:visited{text-decoration:none;color:inherit}.tree-icon,.tree-icon:empty{display:inline-block;text-decoration:none;margin:0;padding:0;vertical-align:top;text-align:center}.tree-ocl{cursor:pointer}.tree-leaf>.tree-ocl{cursor:default}.tree-anchor>.tree-themeicon{margin-right:2px}.tree-anchor>.tree-themeicon-hidden,.tree-hidden,.tree-no-icons .tree-themeicon,.tree-node.tree-hidden{display:none}.tree-rtl .tree-anchor{padding:0 1px 0 4px}.tree-rtl .tree-anchor>.tree-themeicon{margin-left:2px;margin-right:0}.tree-rtl .tree-node{margin-left:0}.tree-rtl .tree-container-ul>.tree-node{margin-right:0}.tree-wholerow-ul{position:relative;display:inline-block;min-width:100%}.tree-wholerow-ul .tree-leaf>.tree-ocl{cursor:pointer}.tree-wholerow-ul .tree-anchor,.tree-wholerow-ul .tree-icon{position:relative}.tree-wholerow-ul .tree-wholerow{width:100%;cursor:pointer;z-index:-1;position:absolute;left:0;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.tree{text-align:left}.tree-default .tree-icon,.tree-default .tree-node{background-repeat:no-repeat;background-color:transparent}.tree-default .tree-anchor,.tree-default .tree-animated,.tree-default .tree-wholerow{transition:background-color .15s,box-shadow .15s}.tree-default .tree-context,.tree-default .tree-hovered{background:#eee;border:0;box-shadow:none}.tree-default .tree-selected{background:#e1e1e1;border:0;box-shadow:none}.tree-default .tree-no-icons .tree-anchor>.tree-themeicon{display:none}.tree-default .tree-disabled{color:#666}.tree-default .tree-disabled.tree-hovered{box-shadow:none}.tree-default .tree-disabled>.tree-icon{opacity:.8;filter:url("data:image/svg+xml;utf8,#tree-grayscale");filter:gray;-webkit-filter:grayscale(100%)}.tree-default .tree-search{font-style:italic;color:#8b0000;font-weight:700}.tree-default .tree-no-checkboxes .tree-checkbox{display:none!important}.tree-default.tree-checkbox-no-clicked .tree-selected{background:transparent;box-shadow:none}.tree-default.tree-checkbox-no-clicked .tree-selected.tree-hovered{background:#eee}.tree-default.tree-checkbox-no-clicked>.tree-wholerow-ul .tree-wholerow-clicked{background:transparent}.tree-default.tree-checkbox-no-clicked>.tree-wholerow-ul .tree-wholerow-clicked.tree-wholerow-hovered{background:#eee}.tree-default>.tree-striped{min-width:100%;display:inline-block;background:url("") 0 0 repeat}.tree-default>.tree-wholerow-ul .tree-hovered,.tree-default>.tree-wholerow-ul .tree-selected{background:transparent;box-shadow:none;border-radius:0}.tree-default .tree-wholerow{box-sizing:border-box}.tree-default .tree-wholerow-hovered{background:#eee}.tree-default .tree-wholerow-clicked{background:#e1e1e1}.tree-default .tree-node{min-height:24px;line-height:24px;margin-left:30px;min-width:24px}.tree-default .tree-anchor,.tree-default .tree-icon{line-height:24px;height:24px}.tree-default .tree-icon{width:24px}.tree-default .tree-icon:empty{width:24px;height:24px;line-height:24px}.tree-default.tree-rtl .tree-node{margin-right:24px}.tree-default .tree-wholerow{height:24px}.tree-default .tree-icon,.tree-default .tree-node{background-image:url("")}.tree-default .tree-node{background-position:-292px -4px;background-repeat:repeat-y}.tree-default .tree-last{background:transparent}.tree-default .tree-open>.tree-ocl{background-position:-132px -4px}.tree-default .tree-closed>.tree-ocl{background-position:-100px -4px}.tree-default .tree-leaf>.tree-ocl{background-position:-68px -4px}.tree-default .tree-themeicon{background-position:-260px -4px}.tree-default>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default>.tree-no-dots .tree-node{background:transparent}.tree-default>.tree-no-dots .tree-open>.tree-ocl{background-position:-36px -4px}.tree-default>.tree-no-dots .tree-closed>.tree-ocl{background-position:-4px -4px}.tree-default .tree-disabled,.tree-default .tree-disabled.tree-hovered{background:transparent}.tree-default .tree-disabled.tree-selected{background:#efefef}.tree-default .tree-checkbox{background-position:-164px -4px}.tree-default .tree-checkbox:hover{background-position:-164px -36px}.tree-default.tree-checkbox-selection .tree-selected>.tree-checkbox,.tree-default .tree-checked>.tree-checkbox{background-position:-228px -4px}.tree-default.tree-checkbox-selection .tree-selected>.tree-checkbox:hover,.tree-default .tree-checked>.tree-checkbox:hover{background-position:-228px -36px}.tree-default .tree-anchor>.tree-undetermined{background-position:-196px -4px}.tree-default .tree-anchor>.tree-undetermined:hover{background-position:-196px -36px}.tree-default .tree-checkbox-disabled{opacity:.8;filter:url("data:image/svg+xml;utf8,#tree-grayscale");filter:gray;-webkit-filter:grayscale(100%)}.tree-default>.tree-striped{background-size:auto 48px}.tree-default.tree-rtl .tree-node{background-position:100% 1px;background-repeat:repeat-y}.tree-default.tree-rtl .tree-open>.tree-ocl{background-position:-132px -36px}.tree-default.tree-rtl .tree-closed>.tree-ocl{background-position:-100px -36px}.tree-default.tree-rtl .tree-leaf>.tree-ocl{background-position:-68px -36px}.tree-default.tree-rtl>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default.tree-rtl>.tree-no-dots .tree-node{background:transparent}.tree-default.tree-rtl>.tree-no-dots .tree-open>.tree-ocl{background-position:-36px -36px}.tree-default.tree-rtl>.tree-no-dots .tree-closed>.tree-ocl{background-position:-4px -36px}.tree-default .tree-themeicon-custom{background-color:transparent;background-image:none;background-position:0 0}.tree-default .tree-node.tree-loading{background:none}.tree-default>.tree-container-ul .tree-loading>.tree-ocl{background:url("") 50% no-repeat}.tree-default .tree-file{background:url("") -100px -68px no-repeat}.tree-default .tree-folder{background:url("") -260px -4px no-repeat}.tree-default>.tree-container-ul>.tree-node{margin-left:0;margin-right:0}.tree-default .tree-ellipsis{overflow:hidden}.tree-default .tree-ellipsis .tree-anchor{width:calc(100% - 29px);text-overflow:ellipsis;overflow:hidden}.tree-default .tree-ellipsis.tree-no-icons .tree-anchor{width:calc(100% - 5px)}.tree-default.tree-rtl .tree-node{background-image:url("")}.tree-default.tree-rtl .tree-last{background:transparent}.tree-default-small .tree-node{min-height:18px;line-height:18px;margin-left:24px;min-width:18px}.tree-default-small .tree-anchor{line-height:18px;height:18px}.tree-default-small .tree-icon,.tree-default-small .tree-icon:empty{width:18px;height:18px;line-height:18px}.tree-default-small.tree-rtl .tree-node{margin-right:18px}.tree-default-small .tree-wholerow{height:18px}.tree-default-small .tree-icon,.tree-default-small .tree-node{background-image:url("")}.tree-default-small .tree-node{background-position:-295px -7px;background-repeat:repeat-y}.tree-default-small .tree-last{background:transparent}.tree-default-small .tree-open>.tree-ocl{background-position:-135px -7px}.tree-default-small .tree-closed>.tree-ocl{background-position:-103px -7px}.tree-default-small .tree-leaf>.tree-ocl{background-position:-71px -7px}.tree-default-small .tree-themeicon{background-position:-263px -7px}.tree-default-small>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-small>.tree-no-dots .tree-node{background:transparent}.tree-default-small>.tree-no-dots .tree-open>.tree-ocl{background-position:-39px -7px}.tree-default-small>.tree-no-dots .tree-closed>.tree-ocl{background-position:-7px -7px}.tree-default-small .tree-disabled,.tree-default-small .tree-disabled.tree-hovered{background:transparent}.tree-default-small .tree-disabled.tree-selected{background:#efefef}.tree-default-small .tree-checkbox{background-position:-167px -7px}.tree-default-small .tree-checkbox:hover{background-position:-167px -39px}.tree-default-small.tree-checkbox-selection .tree-selected>.tree-checkbox,.tree-default-small .tree-checked>.tree-checkbox{background-position:-231px -7px}.tree-default-small.tree-checkbox-selection .tree-selected>.tree-checkbox:hover,.tree-default-small .tree-checked>.tree-checkbox:hover{background-position:-231px -39px}.tree-default-small .tree-anchor>.tree-undetermined{background-position:-199px -7px}.tree-default-small .tree-anchor>.tree-undetermined:hover{background-position:-199px -39px}.tree-default-small .tree-checkbox-disabled{opacity:.8;filter:url("data:image/svg+xml;utf8,#tree-grayscale");filter:gray;-webkit-filter:grayscale(100%)}.tree-default-small>.tree-striped{background-size:auto 36px}.tree-default-small.tree-rtl .tree-node{background-image:url("");background-position:100% 1px;background-repeat:repeat-y}.tree-default-small.tree-rtl .tree-open>.tree-ocl{background-position:-135px -39px}.tree-default-small.tree-rtl .tree-closed>.tree-ocl{background-position:-103px -39px}.tree-default-small.tree-rtl .tree-leaf>.tree-ocl{background-position:-71px -39px}.tree-default-small.tree-rtl>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-small.tree-rtl>.tree-no-dots .tree-node{background:transparent}.tree-default-small.tree-rtl>.tree-no-dots .tree-open>.tree-ocl{background-position:-39px -39px}.tree-default-small.tree-rtl>.tree-no-dots .tree-closed>.tree-ocl{background-position:-7px -39px}.tree-default-small .tree-themeicon-custom{background-color:transparent;background-image:none;background-position:0 0}.tree-default-small .tree-node.tree-loading{background:none}.tree-default-small>.tree-container-ul .tree-loading>.tree-ocl{background:url("") 50% no-repeat}.tree-default-small .tree-file{background:url("") -103px -71px no-repeat}.tree-default-small .tree-folder{background:url("") -263px -7px no-repeat}.tree-default-small>.tree-container-ul>.tree-node{margin-left:0;margin-right:0}.tree-default-small .tree-ellipsis{overflow:hidden}.tree-default-small .tree-ellipsis .tree-anchor{width:calc(100% - 23px);text-overflow:ellipsis;overflow:hidden}.tree-default-small .tree-ellipsis.tree-no-icons .tree-anchor{width:calc(100% - 5px)}.tree-default-small.tree-rtl .tree-node{background-image:url("")}.tree-default-small.tree-rtl .tree-last{background:transparent}.tree-default-large .tree-node{min-height:32px;line-height:32px;margin-left:38px;min-width:32px}.tree-default-large .tree-anchor{line-height:32px;height:32px}.tree-default-large .tree-icon,.tree-default-large .tree-icon:empty{width:32px;height:32px;line-height:32px}.tree-default-large.tree-rtl .tree-node{margin-right:32px}.tree-default-large .tree-wholerow{height:32px}.tree-default-large .tree-icon,.tree-default-large .tree-node{background-image:url("")}.tree-default-large .tree-node{background-position:-288px 0;background-repeat:repeat-y}.tree-default-large .tree-last{background:transparent}.tree-default-large .tree-open>.tree-ocl{background-position:-128px 0}.tree-default-large .tree-closed>.tree-ocl{background-position:-96px 0}.tree-default-large .tree-leaf>.tree-ocl{background-position:-64px 0}.tree-default-large .tree-themeicon{background-position:-256px 0}.tree-default-large>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-large>.tree-no-dots .tree-node{background:transparent}.tree-default-large>.tree-no-dots .tree-open>.tree-ocl{background-position:-32px 0}.tree-default-large>.tree-no-dots .tree-closed>.tree-ocl{background-position:0 0}.tree-default-large .tree-disabled,.tree-default-large .tree-disabled.tree-hovered{background:transparent}.tree-default-large .tree-disabled.tree-selected{background:#efefef}.tree-default-large .tree-checkbox{background-position:-160px 0}.tree-default-large .tree-checkbox:hover{background-position:-160px -32px}.tree-default-large.tree-checkbox-selection .tree-selected>.tree-checkbox,.tree-default-large .tree-checked>.tree-checkbox{background-position:-224px 0}.tree-default-large.tree-checkbox-selection .tree-selected>.tree-checkbox:hover,.tree-default-large .tree-checked>.tree-checkbox:hover{background-position:-224px -32px}.tree-default-large .tree-anchor>.tree-undetermined{background-position:-192px 0}.tree-default-large .tree-anchor>.tree-undetermined:hover{background-position:-192px -32px}.tree-default-large .tree-checkbox-disabled{opacity:.8;filter:url("data:image/svg+xml;utf8,#tree-grayscale");filter:gray;-webkit-filter:grayscale(100%)}.tree-default-large>.tree-striped{background-size:auto 64px}.tree-default-large.tree-rtl .tree-node{background-image:url("");background-position:100% 1px;background-repeat:repeat-y}.tree-default-large.tree-rtl .tree-open>.tree-ocl{background-position:-128px -32px}.tree-default-large.tree-rtl .tree-closed>.tree-ocl{background-position:-96px -32px}.tree-default-large.tree-rtl .tree-leaf>.tree-ocl{background-position:-64px -32px}.tree-default-large.tree-rtl>.tree-no-dots .tree-leaf>.tree-ocl,.tree-default-large.tree-rtl>.tree-no-dots .tree-node{background:transparent}.tree-default-large.tree-rtl>.tree-no-dots .tree-open>.tree-ocl{background-position:-32px -32px}.tree-default-large.tree-rtl>.tree-no-dots .tree-closed>.tree-ocl{background-position:0 -32px}.tree-default-large .tree-themeicon-custom{background-color:transparent;background-image:none;background-position:0 0}.tree-default-large .tree-node.tree-loading{background:none}.tree-default-large>.tree-container-ul .tree-loading>.tree-ocl{background:url("") 50% no-repeat}.tree-default-large .tree-file{background:url("") -96px -64px no-repeat}.tree-default-large .tree-folder{background:url("") -256px 0 no-repeat}.tree-default-large>.tree-container-ul>.tree-node{margin-left:0;margin-right:0}.tree-default-large .tree-ellipsis{overflow:hidden}.tree-default-large .tree-ellipsis .tree-anchor{width:calc(100% - 37px);text-overflow:ellipsis;overflow:hidden}.tree-default-large .tree-ellipsis.tree-no-icons .tree-anchor{width:calc(100% - 5px)}.tree-default-large.tree-rtl .tree-node{background-image:url("")}.tree-default-large.tree-rtl .tree-last{background:transparent}',""])},function(e,A){e.exports=function(){var e=[];return e.toString=function(){for(var e=[],A=0;At.parts.length&&(r.parts.length=t.parts.length)}else{for(var a=[],n=0;n 2 | 3 | 4 | 5 | vue-jstree 6 | 7 | 8 | 9 |
10 | 11 | 12 | 13 | -------------------------------------------------------------------------------- /main.js: -------------------------------------------------------------------------------- 1 | import Vue from 'vue' 2 | 3 | import VJstree from './src/index.js' 4 | Vue.use(VJstree) 5 | 6 | import App from './App.vue' 7 | 8 | new Vue({ 9 | el: '#app', 10 | render: h => h(App) 11 | }) 12 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "vue-jstree", 3 | "version": "2.1.6", 4 | "description": "a tree plugin for vue2", 5 | "keywords": [ 6 | "vue2", 7 | "tree" 8 | ], 9 | "author": "zdy1988 ", 10 | "main": "dist/vue-jstree.js", 11 | "private": false, 12 | "license": "MIT", 13 | "repository": { 14 | "type": "git", 15 | "url": "https://github.com/zdy1988/vue-jstree.git" 16 | }, 17 | "scripts": { 18 | "dev": "cross-env NODE_ENV=development webpack-dev-server --open --hot", 19 | "build": "cross-env NODE_ENV=production webpack --progress --hide-modules" 20 | }, 21 | "dependencies": {}, 22 | "devDependencies": { 23 | "babel-core": "^6.0.0", 24 | "babel-loader": "^6.0.0", 25 | "babel-preset-env": "^1.5.1", 26 | "cross-env": "^3.0.0", 27 | "css-loader": "^0.25.0", 28 | "file-loader": "^0.11.2", 29 | "less": "^2.7.2", 30 | "less-loader": "^4.0.5", 31 | "style-loader": "^0.18.2", 32 | "vue": "^2.3.3", 33 | "vue-loader": "^12.1.0", 34 | "vue-template-compiler": "^2.3.3", 35 | "webpack": "^2.6.1", 36 | "webpack-dev-server": "^2.4.5" 37 | } 38 | } 39 | -------------------------------------------------------------------------------- /pic.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/zdy1988/vue-jstree/40bf85e612ab96804a57e823ffec07441b984120/pic.png -------------------------------------------------------------------------------- /src/index.js: -------------------------------------------------------------------------------- 1 | /** 2 | * Created by virus_zhh on 2017/9/29. 3 | */ 4 | import VJstree from './tree.vue' 5 | 6 | VJstree.install = function(Vue){ 7 | Vue.component(VJstree.name, VJstree) 8 | } 9 | 10 | if (typeof window !== 'undefined' && window.Vue) { 11 | window.Vue.use(VJstree); 12 | } 13 | 14 | export default VJstree -------------------------------------------------------------------------------- /src/less/base.less: -------------------------------------------------------------------------------- 1 | // base tree 2 | .tree-node, .tree-children, .tree-container-ul { display:block; margin:0; padding:0; list-style-type:none; list-style-image:none; } 3 | .tree-children {overflow: hidden; } 4 | .tree-node { white-space:nowrap; } 5 | .tree-anchor { display:inline-block; color:black; white-space:nowrap; padding:0 4px 0 1px; margin:0; vertical-align:top; font-size: 14px; cursor: pointer;} 6 | .tree-anchor:focus { outline:0; } 7 | .tree-anchor, .tree-anchor:link, .tree-anchor:visited, .tree-anchor:hover, .tree-anchor:active { text-decoration:none; color:inherit; } 8 | .tree-icon { display:inline-block; text-decoration:none; margin:0; padding:0; vertical-align:top; text-align:center; } 9 | .tree-icon:empty { display:inline-block; text-decoration:none; margin:0; padding:0; vertical-align:top; text-align:center; } 10 | .tree-ocl { cursor:pointer; } 11 | .tree-leaf > .tree-ocl { cursor:default; } 12 | .tree-anchor > .tree-themeicon { margin-right:2px; } 13 | .tree-no-icons .tree-themeicon, 14 | .tree-anchor > .tree-themeicon-hidden { display:none; } 15 | .tree-hidden, .tree-node.tree-hidden { display:none; } 16 | 17 | // base tree rtl 18 | .tree-rtl { 19 | .tree-anchor { padding:0 1px 0 4px; } 20 | .tree-anchor > .tree-themeicon { margin-left:2px; margin-right:0; } 21 | .tree-node { margin-left:0; } 22 | .tree-container-ul > .tree-node { margin-right:0; } 23 | } 24 | 25 | // base tree wholerow 26 | .tree-wholerow-ul { 27 | position:relative; 28 | display:inline-block; 29 | min-width:100%; 30 | .tree-leaf > .tree-ocl { cursor:pointer; } 31 | .tree-anchor, .tree-icon { position:relative; } 32 | .tree-wholerow { width:100%; cursor:pointer; z-index: -1; position:absolute; left:0; -webkit-user-select:none; -moz-user-select:none; -ms-user-select:none; user-select:none; } 33 | } 34 | 35 | 36 | -------------------------------------------------------------------------------- /src/less/main.less: -------------------------------------------------------------------------------- 1 | .tree { 2 | text-align: left; 3 | } 4 | .tree-@{theme-name} { 5 | .tree-node, 6 | .tree-icon { background-repeat:no-repeat; background-color:transparent; } 7 | .tree-anchor, 8 | .tree-animated, 9 | .tree-wholerow { transition:background-color 0.15s, box-shadow 0.15s; } 10 | .tree-hovered { background:@hovered-bg-color; border: 0px; box-shadow:none; } 11 | .tree-context { background:@hovered-bg-color; border: 0px; box-shadow:none; } 12 | .tree-selected { background:@clicked-bg-color; border: 0px; box-shadow:none; } 13 | .tree-no-icons .tree-anchor > .tree-themeicon { display:none; } 14 | .tree-disabled { 15 | background:transparent; color:@disabled-color; 16 | &.tree-hovered { background:transparent; box-shadow:none; } 17 | &.tree-selected { background:@disabled-bg-color; } 18 | > .tree-icon { opacity:0.8; filter: url("data:image/svg+xml;utf8,#tree-grayscale"); /* Firefox 10+ */ filter: gray; /* IE6-9 */ -webkit-filter: grayscale(100%); /* Chrome 19+ & Safari 6+ */ } 19 | } 20 | // search 21 | .tree-search { font-style:italic; color:@search-result-color; font-weight:bold; } 22 | // checkboxes 23 | .tree-no-checkboxes .tree-checkbox { display:none !important; } 24 | &.tree-checkbox-no-clicked { 25 | .tree-selected { 26 | background:transparent; 27 | box-shadow:none; 28 | &.tree-hovered { background:@hovered-bg-color; } 29 | } 30 | > .tree-wholerow-ul .tree-wholerow-clicked { 31 | background:transparent; 32 | &.tree-wholerow-hovered { background:@hovered-bg-color; } 33 | } 34 | } 35 | // stripes 36 | > .tree-striped { min-width:100%; display:inline-block; background:url("") left top repeat; } 37 | // wholerow 38 | > .tree-wholerow-ul .tree-hovered, 39 | > .tree-wholerow-ul .tree-selected { background:transparent; box-shadow:none; border-radius:0; } 40 | .tree-wholerow { -moz-box-sizing:border-box; -webkit-box-sizing:border-box; box-sizing:border-box; } 41 | .tree-wholerow-hovered { background:@hovered-bg-color; } 42 | .tree-wholerow-clicked { background:@clicked-bg-color; } 43 | } 44 | 45 | // theme variants 46 | .tree-@{theme-name} { 47 | .tree-theme(24px, "", 32px); 48 | &.tree-rtl .tree-node { background-image:url(""); } 49 | &.tree-rtl .tree-last { background:transparent; } 50 | } 51 | .tree-@{theme-name}-small { 52 | .tree-theme(18px, "", 32px); 53 | &.tree-rtl .tree-node { background-image:url(""); } 54 | &.tree-rtl .tree-last { background:transparent; } 55 | } 56 | .tree-@{theme-name}-large { 57 | .tree-theme(32px, "", 32px); 58 | &.tree-rtl .tree-node { background-image:url(""); } 59 | &.tree-rtl .tree-last { background:transparent; } 60 | } 61 | -------------------------------------------------------------------------------- /src/less/mixins.less: -------------------------------------------------------------------------------- 1 | .tree-theme (@base-height, @image, @image-height) { 2 | @correction: (@image-height - @base-height) / 2; 3 | 4 | .tree-node { min-height:@base-height; line-height:@base-height; margin-left:@base-height + 6; min-width:@base-height; } 5 | .tree-anchor { line-height:@base-height; height:@base-height; } 6 | .tree-icon { width:@base-height; height:@base-height; line-height:@base-height; } 7 | .tree-icon:empty { width:@base-height; height:@base-height; line-height:@base-height; } 8 | &.tree-rtl .tree-node { margin-right:@base-height; } 9 | .tree-wholerow { height:@base-height; } 10 | 11 | .tree-node, 12 | .tree-icon { background-image:url("@{image}"); } 13 | .tree-node { background-position:-(@image-height * 9 + @correction) -@correction; background-repeat:repeat-y; } 14 | .tree-last { background:transparent; } 15 | 16 | .tree-open > .tree-ocl { background-position:-(@image-height * 4 + @correction) -@correction; } 17 | .tree-closed > .tree-ocl { background-position:-(@image-height * 3 + @correction) -@correction; } 18 | .tree-leaf > .tree-ocl { background-position:-(@image-height * 2 + @correction) -@correction; } 19 | 20 | .tree-themeicon { background-position:-(@image-height * 8 + @correction) -@correction; } 21 | 22 | > .tree-no-dots { 23 | .tree-node, 24 | .tree-leaf > .tree-ocl { background:transparent; } 25 | .tree-open > .tree-ocl { background-position:-(@image-height * 1 + @correction) -@correction; } 26 | .tree-closed > .tree-ocl { background-position:-@correction -@correction; } 27 | } 28 | 29 | .tree-disabled { 30 | background:transparent; 31 | &.tree-hovered { 32 | background:transparent; 33 | } 34 | &.tree-selected { 35 | background:#efefef; 36 | } 37 | } 38 | 39 | .tree-checkbox { 40 | background-position:-(@image-height * 5 + @correction) -@correction; 41 | &:hover { background-position:-(@image-height * 5 + @correction) -(@image-height * 1 + @correction); } 42 | } 43 | 44 | &.tree-checkbox-selection .tree-selected, .tree-checked { 45 | > .tree-checkbox { 46 | background-position:-(@image-height * 7 + @correction) -@correction; 47 | &:hover { background-position:-(@image-height * 7 + @correction) -(@image-height * 1 + @correction); } 48 | } 49 | } 50 | .tree-anchor { 51 | > .tree-undetermined { 52 | background-position:-(@image-height * 6 + @correction) -@correction; 53 | &:hover { 54 | background-position:-(@image-height * 6 + @correction) -(@image-height * 1 + @correction); 55 | } 56 | } 57 | } 58 | .tree-checkbox-disabled { opacity:0.8; filter: url("data:image/svg+xml;utf8,#tree-grayscale"); /* Firefox 10+ */ filter: gray; /* IE6-9 */ -webkit-filter: grayscale(100%); /* Chrome 19+ & Safari 6+ */ } 59 | 60 | > .tree-striped { background-size:auto (@base-height * 2); } 61 | 62 | &.tree-rtl { 63 | .tree-node { background-image:url(""); background-position: 100% 1px; background-repeat:repeat-y; } 64 | .tree-last { background:transparent; } 65 | .tree-open > .tree-ocl { background-position:-(@image-height * 4 + @correction) -(@image-height * 1 + @correction); } 66 | .tree-closed > .tree-ocl { background-position:-(@image-height * 3 + @correction) -(@image-height * 1 + @correction); } 67 | .tree-leaf > .tree-ocl { background-position:-(@image-height * 2 + @correction) -(@image-height * 1 + @correction); } 68 | > .tree-no-dots { 69 | .tree-node, 70 | .tree-leaf > .tree-ocl { background:transparent; } 71 | .tree-open > .tree-ocl { background-position:-(@image-height * 1 + @correction) -(@image-height * 1 + @correction); } 72 | .tree-closed > .tree-ocl { background-position:-@correction -(@image-height * 1 + @correction); } 73 | } 74 | } 75 | .tree-themeicon-custom { background-color:transparent; background-image:none; background-position:0 0; } 76 | 77 | .tree-node.tree-loading{background: none;} 78 | 79 | > .tree-container-ul .tree-loading > .tree-ocl { background:url("") center center no-repeat; } 80 | 81 | .tree-file { background:url("@{image}") -(@image-height * 3 + @correction) -(@image-height * 2 + @correction) no-repeat; } 82 | .tree-folder { background:url("@{image}") -(@image-height * 8 + @correction) -(@correction) no-repeat; } 83 | 84 | > .tree-container-ul > .tree-node { margin-left:0; margin-right:0; } 85 | 86 | // ellipsis 87 | .tree-ellipsis { overflow: hidden; } 88 | // base height + PADDINGS! 89 | .tree-ellipsis .tree-anchor { width: calc(100% ~"-" (@base-height + 5px)); text-overflow: ellipsis; overflow: hidden; } 90 | .tree-ellipsis.tree-no-icons .tree-anchor { width: calc(100% ~"-" 5px); } 91 | } 92 | -------------------------------------------------------------------------------- /src/less/style.less: -------------------------------------------------------------------------------- 1 | /* tree default theme */ 2 | @theme-name: default; 3 | @hovered-bg-color: #eee; 4 | @hovered-shadow-color: #cccccc; 5 | @disabled-color: #666666; 6 | @disabled-bg-color: #efefef; 7 | @clicked-bg-color: #e1e1e1; 8 | @clicked-shadow-color: #999999; 9 | @search-result-color: #8b0000; 10 | @mobile-wholerow-bg-color: #ebebeb; 11 | @mobile-wholerow-shadow: #666666; 12 | @mobile-wholerow-bordert: rgba(255,255,255,0.7); 13 | @mobile-wholerow-borderb: rgba(64,64,64,0.2); 14 | @responsive: true; 15 | @image-path: "./"; 16 | @base-height: 40px; 17 | 18 | @import "./mixins.less"; 19 | @import "./base.less"; 20 | @import "./main.less"; -------------------------------------------------------------------------------- /src/tree-item.vue: -------------------------------------------------------------------------------- 1 | 51 | 238 | -------------------------------------------------------------------------------- /src/tree.vue: -------------------------------------------------------------------------------- 1 | 34 | 294 | 297 | -------------------------------------------------------------------------------- /webpack.config.js: -------------------------------------------------------------------------------- 1 | var path = require('path') 2 | var webpack = require('webpack') 3 | 4 | module.exports = { 5 | //entry: './main.js', 6 | entry: './src/index.js', 7 | output: { 8 | path: path.resolve(__dirname, './dist'), 9 | //publicPath: '/dist/', 10 | publicPath: 'dist/', 11 | filename: 'vue-jstree.js', 12 | library: 'vue-jstree', 13 | libraryTarget: 'umd', 14 | umdNamedDefine: true 15 | }, 16 | module: { 17 | rules: [ 18 | { 19 | test: /\.vue$/, 20 | loader: 'vue-loader', 21 | options: { 22 | loaders: { 23 | } 24 | // other vue-loader options go here 25 | } 26 | }, 27 | { 28 | test: /\.js$/, 29 | loader: 'babel-loader', 30 | exclude: /node_modules/ 31 | }, 32 | { 33 | test: /\.(png|jpg|gif|svg)$/, 34 | loader: 'file-loader', 35 | options: { 36 | name: '[name].[ext]' 37 | } 38 | }, 39 | { 40 | test: /\.less$/, 41 | use: [{ 42 | loader: "style-loader" // creates style nodes from JS strings 43 | }, { 44 | loader: "css-loader" // translates CSS into CommonJS 45 | }, { 46 | loader: "less-loader" // compiles Less to CSS 47 | }] 48 | } 49 | ] 50 | }, 51 | resolve: { 52 | alias: { 53 | 'vue$': 'vue/dist/vue.esm.js' 54 | } 55 | }, 56 | devServer: { 57 | historyApiFallback: true, 58 | noInfo: true 59 | }, 60 | performance: { 61 | hints: false 62 | }, 63 | devtool: '#eval-source-map' 64 | } 65 | 66 | if (process.env.NODE_ENV === 'production') { 67 | module.exports.devtool = '#source-map' 68 | // http://vue-loader.vuejs.org/en/workflow/production.html 69 | module.exports.plugins = (module.exports.plugins || []).concat([ 70 | new webpack.DefinePlugin({ 71 | 'process.env': { 72 | NODE_ENV: '"production"' 73 | } 74 | }), 75 | new webpack.optimize.UglifyJsPlugin({ 76 | sourceMap: true, 77 | compress: { 78 | warnings: false 79 | } 80 | }), 81 | new webpack.LoaderOptionsPlugin({ 82 | minimize: true 83 | }) 84 | ]) 85 | } 86 | --------------------------------------------------------------------------------