2024-03-04 14:00:27 +08:00

186 lines
6.8 KiB
Vue

<script>
import {defineComponent, ref, onMounted, onBeforeUnmount, watch, nextTick} from 'vue-demi';
import Spreadsheet from 'x-data-spreadsheet';
import {getData, readExcelData, transferExcelToSpreadSheet} from './excel';
import {renderImage, clearCache} from './media';
import {readOnlyInput} from './hack';
import {debounce} from 'lodash';
import {download as downloadFile} from '../../../utils/url';
const defaultOptions = {
minColLength: 20
};
export default defineComponent({
name: 'VueOfficeExcel',
props: {
src: [String, ArrayBuffer, Blob],
requestOptions: {
type: Object,
default: () => ({})
},
options: {
type: Object,
default: () => ({
...defaultOptions
})
}
},
emits: ['rendered', 'error'],
setup(props, {emit}) {
const wrapperRef = ref(null);
const rootRef = ref(null);
let workbookDataSource = {
_worksheets:[]
};
let mediasSource = [];
let sheetIndex = 0;
let ctx = null;
let xs = null;
let offset = null;
let fileData = null;
function renderExcel(buffer) {
fileData = buffer;
readExcelData(buffer).then(workbook => {
if (!workbook._worksheets || workbook._worksheets.length === 0) {
throw new Error('未获取到数据,可能文件格式不正确或文件已损坏');
}
if(props.options.beforeTransformData && typeof props.options.beforeTransformData === 'function' ){
workbook = props.options.beforeTransformData(workbook);
}
let {workbookData, medias, workbookSource} = transferExcelToSpreadSheet(workbook, {...defaultOptions, ...props.options});
if(props.options.transformData && typeof props.options.transformData === 'function' ){
workbookData = props.options.transformData(workbookData);
}
mediasSource = medias;
workbookDataSource = workbookSource;
offset = null;
sheetIndex = 0;
clearCache();
xs.loadData(workbookData);
renderImage(ctx, mediasSource, workbookDataSource._worksheets[sheetIndex], offset, props.options);
emit('rendered');
//涉及clear和offset
}).catch(e => {
console.warn(e);
mediasSource = [];
workbookDataSource = {
_worksheets:[]
};
clearCache();
xs && xs.loadData({});
emit('error', e);
});
}
const observerCallback = debounce(readOnlyInput, 200).bind(this,rootRef.value);
const observer = new MutationObserver(observerCallback);
const observerConfig = { attributes: true, childList: true, subtree: true };
onMounted(() => {
nextTick(()=>{
observer.observe(rootRef.value, observerConfig);
observerCallback(rootRef);
xs = new Spreadsheet(rootRef.value, {
mode: 'read',
showToolbar: false,
showContextmenu: props.options.showContextmenu || false,
view: {
height: () => wrapperRef.value && wrapperRef.value.clientHeight || 300,
width: () => wrapperRef.value && wrapperRef.value.clientWidth || 1200,
},
row: {
height: 24,
len: 100
},
col: {
len: 26,
width: 80,
indexWidth: 60,
minWidth: 60,
},
autoFocus: false
}).loadData({});
let swapFunc = xs.bottombar.swapFunc;
xs.bottombar.swapFunc = function (index) {
swapFunc.call(xs.bottombar, index);
sheetIndex = index;
setTimeout(()=>{
xs.reRender();
renderImage(ctx, mediasSource, workbookDataSource._worksheets[sheetIndex], offset, props.options);
});
};
let clear = xs.sheet.editor.clear;
xs.sheet.editor.clear = function (...args){
clear.apply(xs.sheet.editor, args);
setTimeout(()=>{
renderImage(ctx, mediasSource, workbookDataSource._worksheets[sheetIndex], offset, props.options);
});
};
let setOffset = xs.sheet.editor.setOffset;
xs.sheet.editor.setOffset = function (...args){
setOffset.apply(xs.sheet.editor, args);
offset = args[0];
renderImage(ctx, mediasSource, workbookDataSource._worksheets[sheetIndex], offset, props.options);
};
const canvas = rootRef.value.querySelector('canvas');
ctx = canvas.getContext('2d');
if (props.src) {
getData(props.src, props.requestOptions).then(renderExcel).catch(e => {
mediasSource = [];
workbookDataSource = {
_worksheets:[]
};
xs.loadData({});
emit('error', e);
});
}
});
});
onBeforeUnmount(()=>{
observer.disconnect();
xs = null;
});
watch(() => props.src, () => {
if (props.src) {
getData(props.src, props.requestOptions).then(renderExcel).catch(e => {
mediasSource = [];
workbookDataSource = {
_worksheets:[]
};
xs.loadData({});
emit('error', e);
});
} else {
mediasSource = [];
workbookDataSource = {
_worksheets:[]
};
xs.loadData({});
emit('error', new Error('src属性不能为空'));
}
});
function save(fileName){
downloadFile(fileName || `vue-office-excel-${new Date().getTime()}.xlsx`,fileData);
}
return {
wrapperRef,
rootRef,
save
};
}
});
</script>
<template>
<div class="vue-office-excel" ref="wrapperRef">
<div class="vue-office-excel-main" ref="rootRef"></div>
</div>
</template>
<style lang="less">
</style>