mirror of
https://github.com/501351981/vue-office.git
synced 2025-06-15 07:00:00 +08:00
149 lines
5.0 KiB
Vue
149 lines
5.0 KiB
Vue
<script>
|
|
import {defineComponent, ref, onMounted, onBeforeUnmount, watch} from 'vue-demi';
|
|
import Spreadsheet from 'x-data-spreadsheet';
|
|
import {getData, readExcelData, transferExcelToSpreadSheet} from './excel';
|
|
import {renderImage, clearCache} from './media';
|
|
import {readOnlyInput} from './hack';
|
|
|
|
export default defineComponent({
|
|
name: 'VueOfficeExcel',
|
|
props: {
|
|
src: [String, ArrayBuffer, Blob],
|
|
requestOptions: {
|
|
type: Object,
|
|
default: () => ({})
|
|
},
|
|
options: {
|
|
type: Object,
|
|
default: () => ({
|
|
minColLength: 20
|
|
})
|
|
}
|
|
},
|
|
emits: ['rendered', 'error'],
|
|
setup(props, {emit}) {
|
|
const wrapperRef = ref(null);
|
|
const rootRef = ref(null);
|
|
let workbookDataSource = {
|
|
_worksheets:[]
|
|
};
|
|
let mediasSource = [];
|
|
let sheetIndex = 1;
|
|
let ctx = null;
|
|
let xs = null;
|
|
let offset = null;
|
|
|
|
function renderExcel(buffer) {
|
|
readExcelData(buffer).then(workbook => {
|
|
if (!workbook._worksheets || workbook._worksheets.length === 0) {
|
|
throw new Error('未获取到数据,可能文件格式不正确或文件已损坏');
|
|
}
|
|
const {workbookData, medias, workbookSource} = transferExcelToSpreadSheet(workbook, props.options);
|
|
mediasSource = medias;
|
|
workbookDataSource = workbookSource;
|
|
offset = null;
|
|
sheetIndex = 1;
|
|
clearCache();
|
|
xs.loadData(workbookData);
|
|
renderImage(ctx, mediasSource, workbookDataSource._worksheets[sheetIndex], offset);
|
|
emit('rendered');
|
|
//涉及clear和offset
|
|
|
|
}).catch(e => {
|
|
console.warn(e);
|
|
mediasSource = [];
|
|
workbookDataSource = {
|
|
_worksheets:[]
|
|
};
|
|
clearCache();
|
|
xs.loadData({});
|
|
emit('error', e);
|
|
});
|
|
}
|
|
const observer = new MutationObserver(readOnlyInput);
|
|
const observerConfig = { attributes: true, childList: true, subtree: true };
|
|
onMounted(() => {
|
|
observer.observe(rootRef.value, observerConfig);
|
|
window.xs = xs = new Spreadsheet(rootRef.value, {
|
|
mode: 'read',
|
|
showToolbar: false,
|
|
view: {
|
|
height: () => wrapperRef.value.clientHeight || 300,
|
|
width: () => wrapperRef.value.clientWidth || 300,
|
|
},
|
|
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 + 1;
|
|
setTimeout(()=>{
|
|
xs.reRender();
|
|
renderImage(ctx, mediasSource, workbookDataSource._worksheets[sheetIndex], offset);
|
|
});
|
|
|
|
};
|
|
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);
|
|
});
|
|
};
|
|
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);
|
|
};
|
|
const canvas = rootRef.value.querySelector('canvas');
|
|
ctx = canvas.getContext('2d');
|
|
if (props.src) {
|
|
getData(props.src, props.requestOptions).then(renderExcel).catch(e => {
|
|
xs.loadData({});
|
|
emit('error', e);
|
|
});
|
|
}
|
|
});
|
|
|
|
onBeforeUnmount(()=>{
|
|
observer.disconnect();
|
|
xs = null;
|
|
});
|
|
watch(() => props.src, () => {
|
|
if (props.src) {
|
|
getData(props.src).then(renderExcel).catch(e => {
|
|
xs.loadData({});
|
|
emit('error', e);
|
|
});
|
|
} else {
|
|
xs.loadData({});
|
|
}
|
|
});
|
|
return {
|
|
wrapperRef,
|
|
rootRef
|
|
};
|
|
}
|
|
});
|
|
</script>
|
|
|
|
<template>
|
|
<div class="vue-office-excel" ref="wrapperRef">
|
|
<div class="vue-office-excel-main" ref="rootRef"></div>
|
|
</div>
|
|
</template>
|
|
<style lang="less">
|
|
|
|
</style> |