281 lines
6.6 KiB
Vue
281 lines
6.6 KiB
Vue
|
|
<template>
|
|||
|
|
<el-card class="step-card">
|
|||
|
|
<template #header>
|
|||
|
|
<div class="card-header">
|
|||
|
|
<div class="header-icon">
|
|||
|
|
<el-icon class="text-blue-600">
|
|||
|
|
<DocumentTextIcon />
|
|||
|
|
</el-icon>
|
|||
|
|
</div>
|
|||
|
|
<div class="header-content">
|
|||
|
|
<h2 class="header-title">合同预览</h2>
|
|||
|
|
<p class="header-subtitle">请仔细阅读合同内容,确认无误后再进行签署</p>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
</template>
|
|||
|
|
<div class="contract-preview-section">
|
|||
|
|
<div v-if="contractUrl" class="pdf-viewer-wrapper">
|
|||
|
|
<vue-pdf-embed
|
|||
|
|
ref="pdfRef"
|
|||
|
|
:source="contractUrl"
|
|||
|
|
:page="page"
|
|||
|
|
:scale="pdfScale"
|
|||
|
|
@loaded="handleLoaded"
|
|||
|
|
class="pdf-embed"
|
|||
|
|
/>
|
|||
|
|
<div class="pdf-controls">
|
|||
|
|
<el-button size="large" :disabled="page <= 1" @click="page--"> 上一页 </el-button>
|
|||
|
|
<span class="pdf-page-info">第 {{ page }} / {{ pageCount }} 页</span>
|
|||
|
|
<el-button size="large" :disabled="page >= pageCount" @click="page++"> 下一页 </el-button>
|
|||
|
|
<el-divider direction="vertical" />
|
|||
|
|
<el-button size="large" @click="fullscreen = true"> 全屏预览 </el-button>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
<el-button type="primary" @click="emit('start-sign')" :loading="props.loading">
|
|||
|
|
<el-icon class="mr-2"><DocumentTextIcon /></el-icon>
|
|||
|
|
开始签署
|
|||
|
|
</el-button>
|
|||
|
|
</div>
|
|||
|
|
<!-- 极简全屏预览弹窗,无顶部栏 -->
|
|||
|
|
<el-dialog
|
|||
|
|
v-model="fullscreen"
|
|||
|
|
fullscreen
|
|||
|
|
append-to-body
|
|||
|
|
:close-on-click-modal="true"
|
|||
|
|
class="fullscreen-dialog"
|
|||
|
|
:show-close="false"
|
|||
|
|
lock-scroll
|
|||
|
|
:header="false"
|
|||
|
|
>
|
|||
|
|
<div class="fullscreen-pdf-simple">
|
|||
|
|
<vue-pdf-embed
|
|||
|
|
:source="contractUrl"
|
|||
|
|
:page="page"
|
|||
|
|
:scale="fullscreenScale"
|
|||
|
|
class="fullscreen-pdf-embed"
|
|||
|
|
/>
|
|||
|
|
<div class="fullscreen-pdf-controls">
|
|||
|
|
<el-button size="large" :disabled="page <= 1" @click="page--">上一页</el-button>
|
|||
|
|
<span class="pdf-page-info">第 {{ page }} / {{ pageCount }} 页</span>
|
|||
|
|
<el-button size="large" :disabled="page >= pageCount" @click="page++">下一页</el-button>
|
|||
|
|
<el-button type="primary" size="large" @click="fullscreen = false"> 关闭预览 </el-button>
|
|||
|
|
</div>
|
|||
|
|
</div>
|
|||
|
|
</el-dialog>
|
|||
|
|
</el-card>
|
|||
|
|
</template>
|
|||
|
|
|
|||
|
|
<script setup>
|
|||
|
|
|
|||
|
|
import { DocumentTextIcon } from '@heroicons/vue/24/outline'
|
|||
|
|
import VuePdfEmbed from 'vue-pdf-embed'
|
|||
|
|
|
|||
|
|
const props = defineProps({
|
|||
|
|
contractUrl: String,
|
|||
|
|
loading: Boolean
|
|||
|
|
})
|
|||
|
|
const emit = defineEmits(['start-sign'])
|
|||
|
|
|
|||
|
|
const pdfRef = ref(null)
|
|||
|
|
const page = ref(1)
|
|||
|
|
const pageCount = ref(0)
|
|||
|
|
const pdfScale = ref(1.3)
|
|||
|
|
const fullscreen = ref(false)
|
|||
|
|
const fullscreenScale = ref(1.7)
|
|||
|
|
|
|||
|
|
function handleLoaded(pdf) {
|
|||
|
|
pageCount.value = pdf.numPages
|
|||
|
|
}
|
|||
|
|
</script>
|
|||
|
|
|
|||
|
|
<style scoped>
|
|||
|
|
.step-card {
|
|||
|
|
border-radius: 20px;
|
|||
|
|
box-shadow: 0 20px 60px rgba(0, 0, 0, 0.1);
|
|||
|
|
border: none;
|
|||
|
|
background: rgba(255, 255, 255, 0.95);
|
|||
|
|
backdrop-filter: blur(10px);
|
|||
|
|
border: 1px solid rgba(255, 255, 255, 0.2);
|
|||
|
|
}
|
|||
|
|
.card-header {
|
|||
|
|
display: flex;
|
|||
|
|
align-items: center;
|
|||
|
|
gap: 16px;
|
|||
|
|
padding: 8px 0;
|
|||
|
|
}
|
|||
|
|
.header-icon {
|
|||
|
|
width: 48px;
|
|||
|
|
height: 48px;
|
|||
|
|
border-radius: 12px;
|
|||
|
|
background: linear-gradient(135deg, #dbeafe 0%, #bfdbfe 100%);
|
|||
|
|
display: flex;
|
|||
|
|
align-items: center;
|
|||
|
|
justify-content: center;
|
|||
|
|
font-size: 24px;
|
|||
|
|
}
|
|||
|
|
.header-content {
|
|||
|
|
flex: 1;
|
|||
|
|
}
|
|||
|
|
.header-title {
|
|||
|
|
font-size: 24px;
|
|||
|
|
font-weight: 700;
|
|||
|
|
color: #1e293b;
|
|||
|
|
margin: 0 0 4px 0;
|
|||
|
|
}
|
|||
|
|
.header-subtitle {
|
|||
|
|
font-size: 14px;
|
|||
|
|
color: #64748b;
|
|||
|
|
margin: 0;
|
|||
|
|
font-weight: 500;
|
|||
|
|
}
|
|||
|
|
.contract-preview-section {
|
|||
|
|
padding: 32px 0;
|
|||
|
|
text-align: center;
|
|||
|
|
}
|
|||
|
|
.pdf-viewer-wrapper {
|
|||
|
|
display: flex;
|
|||
|
|
flex-direction: column;
|
|||
|
|
align-items: center;
|
|||
|
|
margin-bottom: 16px;
|
|||
|
|
}
|
|||
|
|
.pdf-embed {
|
|||
|
|
width: 100%;
|
|||
|
|
max-width: 600px;
|
|||
|
|
min-height: 600px;
|
|||
|
|
max-height: 600px;
|
|||
|
|
border: 1px solid #e5e7eb;
|
|||
|
|
border-radius: 16px;
|
|||
|
|
box-shadow: 0 4px 24px rgba(0, 0, 0, 0.08);
|
|||
|
|
background: #fafbfc;
|
|||
|
|
overflow-y: auto;
|
|||
|
|
overflow-x: auto;
|
|||
|
|
scrollbar-width: thin;
|
|||
|
|
scrollbar-color: #cbd5e1 #f1f5f9;
|
|||
|
|
}
|
|||
|
|
.pdf-embed::-webkit-scrollbar {
|
|||
|
|
width: 8px;
|
|||
|
|
}
|
|||
|
|
.pdf-embed::-webkit-scrollbar-thumb {
|
|||
|
|
background: #cbd5e1;
|
|||
|
|
border-radius: 4px;
|
|||
|
|
}
|
|||
|
|
.pdf-embed::-webkit-scrollbar-track {
|
|||
|
|
background: #f1f5f9;
|
|||
|
|
border-radius: 4px;
|
|||
|
|
}
|
|||
|
|
.pdf-controls {
|
|||
|
|
margin: 24px 0 0 0;
|
|||
|
|
display: flex;
|
|||
|
|
align-items: center;
|
|||
|
|
gap: 24px;
|
|||
|
|
justify-content: center;
|
|||
|
|
font-size: 16px;
|
|||
|
|
}
|
|||
|
|
.pdf-btn {
|
|||
|
|
border-radius: 12px;
|
|||
|
|
background: linear-gradient(135deg, #3b82f6 0%, #1d4ed8 100%);
|
|||
|
|
color: #fff;
|
|||
|
|
font-weight: 600;
|
|||
|
|
border: none;
|
|||
|
|
box-shadow: 0 2px 8px rgba(59, 130, 246, 0.1);
|
|||
|
|
transition:
|
|||
|
|
background 0.2s,
|
|||
|
|
color 0.2s,
|
|||
|
|
box-shadow 0.2s;
|
|||
|
|
min-width: 110px;
|
|||
|
|
height: 44px;
|
|||
|
|
font-size: 16px;
|
|||
|
|
}
|
|||
|
|
.pdf-btn:disabled {
|
|||
|
|
background: #e2e8f0;
|
|||
|
|
color: #94a3b8;
|
|||
|
|
box-shadow: none;
|
|||
|
|
}
|
|||
|
|
.pdf-btn:hover:not(:disabled) {
|
|||
|
|
background: #2563eb;
|
|||
|
|
color: #fff;
|
|||
|
|
box-shadow: 0 4px 16px rgba(59, 130, 246, 0.18);
|
|||
|
|
}
|
|||
|
|
.pdf-page-info {
|
|||
|
|
color: #64748b;
|
|||
|
|
font-size: 16px;
|
|||
|
|
font-weight: 500;
|
|||
|
|
}
|
|||
|
|
.contract-sign-btn {
|
|||
|
|
min-width: 220px;
|
|||
|
|
height: 52px;
|
|||
|
|
font-size: 18px;
|
|||
|
|
font-weight: 700;
|
|||
|
|
border-radius: 14px;
|
|||
|
|
background: linear-gradient(135deg, #3b82f6 0%, #1d4ed8 100%);
|
|||
|
|
border: none;
|
|||
|
|
box-shadow: 0 8px 24px rgba(59, 130, 246, 0.18);
|
|||
|
|
transition:
|
|||
|
|
background 0.2s,
|
|||
|
|
color 0.2s,
|
|||
|
|
box-shadow 0.2s;
|
|||
|
|
margin-top: 32px;
|
|||
|
|
}
|
|||
|
|
.contract-sign-btn:hover:not(:disabled) {
|
|||
|
|
background: #2563eb;
|
|||
|
|
color: #fff;
|
|||
|
|
box-shadow: 0 12px 32px rgba(59, 130, 246, 0.22);
|
|||
|
|
}
|
|||
|
|
.contract-sign-btn:disabled {
|
|||
|
|
background: #e2e8f0;
|
|||
|
|
color: #94a3b8;
|
|||
|
|
box-shadow: none;
|
|||
|
|
}
|
|||
|
|
.fullscreen-dialog >>> .el-dialog__body {
|
|||
|
|
padding: 0;
|
|||
|
|
}
|
|||
|
|
.fullscreen-pdf-simple {
|
|||
|
|
display: flex;
|
|||
|
|
flex-direction: column;
|
|||
|
|
align-items: center;
|
|||
|
|
justify-content: center;
|
|||
|
|
background: #fafbfc;
|
|||
|
|
overflow: auto;
|
|||
|
|
}
|
|||
|
|
.fullscreen-pdf-embed {
|
|||
|
|
width: 100vw;
|
|||
|
|
max-width: 1000px;
|
|||
|
|
height: calc(100vh - 116px);
|
|||
|
|
border: none;
|
|||
|
|
border-radius: 0;
|
|||
|
|
background: #fff;
|
|||
|
|
overflow: auto;
|
|||
|
|
box-shadow: none;
|
|||
|
|
}
|
|||
|
|
.fullscreen-pdf-controls {
|
|||
|
|
width: 100%;
|
|||
|
|
height: 68px;
|
|||
|
|
background-color: #e2e8f0;
|
|||
|
|
display: flex;
|
|||
|
|
align-items: center;
|
|||
|
|
gap: 24px;
|
|||
|
|
justify-content: center;
|
|||
|
|
font-size: 18px;
|
|||
|
|
}
|
|||
|
|
.fullscreen-close-btn {
|
|||
|
|
min-width: 120px;
|
|||
|
|
height: 44px;
|
|||
|
|
font-size: 16px;
|
|||
|
|
font-weight: 700;
|
|||
|
|
border-radius: 12px;
|
|||
|
|
background: linear-gradient(135deg, #3b82f6 0%, #1d4ed8 100%);
|
|||
|
|
border: none;
|
|||
|
|
box-shadow: 0 4px 16px rgba(59, 130, 246, 0.18);
|
|||
|
|
margin-left: 32px;
|
|||
|
|
transition:
|
|||
|
|
background 0.2s,
|
|||
|
|
color 0.2s,
|
|||
|
|
box-shadow 0.2s;
|
|||
|
|
}
|
|||
|
|
.fullscreen-close-btn:hover {
|
|||
|
|
background: #2563eb;
|
|||
|
|
color: #fff;
|
|||
|
|
box-shadow: 0 8px 24px rgba(59, 130, 246, 0.22);
|
|||
|
|
}
|
|||
|
|
</style>
|