mirror of
				https://github.com/scinote-eln/scinote-web.git
				synced 2025-10-25 21:47:03 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			88 lines
		
	
	
	
		
			2.7 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			88 lines
		
	
	
	
		
			2.7 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
| <template>
 | |
|   <div ref="modal" @keydown.esc="close" class="modal sci-reorderable-items" tabindex="-1" role="dialog">
 | |
|     <div class="modal-dialog" role="document">
 | |
|       <div class="modal-content">
 | |
|         <div class="modal-header">
 | |
|           <button @click="close" type="button" class="close" data-dismiss="modal" aria-label="Close"><i class="sn-icon sn-icon-close"></i></button>
 | |
|           <h4 class="modal-title">
 | |
|             {{ title }}
 | |
|           </h4>
 | |
|         </div>
 | |
|         <div class="modal-body">
 | |
|           <Draggable
 | |
|             v-model="reorderedItems"
 | |
|             :ghostClass="'step-checklist-item-ghost'"
 | |
|             :dragClass="'step-checklist-item-drag'"
 | |
|             :chosenClass="'step-checklist-item-chosen'"
 | |
|             :handle="'.step-element-grip'"
 | |
|             item-key="id"
 | |
|           >
 | |
|             <template #item="{element, index}">
 | |
|               <div class="step-element-header flex items-center">
 | |
|                 <div class="step-element-grip step-element-grip--draggable">
 | |
|                   <i class="sn-icon sn-icon-drag"></i>
 | |
|                 </div>
 | |
|                 <div class="step-element-name text-center flex items-center gap-2">
 | |
|                   <strong v-if="includeNumbers" class="step-element-number">{{ index + 1 }}</strong>
 | |
|                   <i v-if="element.attributes.icon" class="fas" :class="element.attributes.icon"></i>
 | |
|                   <span :title="nameWithFallbacks(element)" v-if="nameWithFallbacks(element)">{{ nameWithFallbacks(element) }}</span>
 | |
|                   <span :title="element.attributes.placeholder" v-else class="step-element-name-placeholder">{{ element.attributes.placeholder }}</span>
 | |
|                 </div>
 | |
|               </div>
 | |
|             </template>
 | |
|           </Draggable>
 | |
|         </div>
 | |
|       </div>
 | |
|     </div>
 | |
|   </div>
 | |
| </template>
 | |
| <script>
 | |
| import Draggable from 'vuedraggable';
 | |
| 
 | |
| export default {
 | |
|   name: 'ReorderableItems',
 | |
|   components: {
 | |
|     Draggable
 | |
|   },
 | |
|   props: {
 | |
|     title: {
 | |
|       type: String,
 | |
|       required: true
 | |
|     },
 | |
|     items: {
 | |
|       type: Array,
 | |
|       required: true
 | |
|     },
 | |
|     includeNumbers: {
 | |
|       type: Boolean,
 | |
|       default: false
 | |
|     }
 | |
|   },
 | |
|   data() {
 | |
|     return {
 | |
|       reorderedItems: [...this.items]
 | |
|     };
 | |
|   },
 | |
|   mounted() {
 | |
|     window.$(this.$refs.modal).modal('show');
 | |
|     window.$(this.$refs.modal).on('hidden.bs.modal', () => {
 | |
|       this.close();
 | |
|     });
 | |
|   },
 | |
|   methods: {
 | |
|     close() {
 | |
|       this.$emit('reorder', this.reorderedItems);
 | |
|       this.$emit('close');
 | |
|     },
 | |
|     getTitle(item) {
 | |
|       const $item_html = $(item.attributes.text);
 | |
|       $item_html.remove('table, img');
 | |
|       const str = $item_html.text().trim();
 | |
|       return str.length > 56 ? `${str.slice(0, 56)}...` : str;
 | |
|     },
 | |
|     nameWithFallbacks(item) {
 | |
|       return this.getTitle(item) || item.attributes.name;
 | |
|     }
 | |
|   }
 | |
| };
 | |
| </script>
 |