JS.Class - 2. Methods & Types

本文探讨了JavaScript中面向对象编程的特点及技巧,包括this指向问题、使用method()解决回调中的this问题、Singleton模式的应用、类型验证方法isA及klass属性、子类查找与实例化、以及接口确保对象符合特定契约。
说实话,在 JavaScript 里玩 object-oriented 感觉像坐过山车,晕晕乎乎的。那个可以被称为 "上帝金手指" 的 this 会随时玩出一些魔法来,要永远记住它指向的是 "调用者",而不一定是对象实例本身。 var MyClass = new JS.Class( { test : function(o) { alert(o == this); } }); var o = new MyClass(); var func = o.test; o.test(o); // true func(o); // false 你可能想当然认为 func 是 o.test 的指针…… 嗯,这么说也没错。只不过一旦执行 func(o),其方法体内部的 this 就不再代表实例引用 o,而是 func() 的调用者 BOM。因此 o == BOM 的结果自然是否定的。正由于这种 "奇怪" 的特性,使得 this 的使用往往变成一个误区和灾区,作为最常用的 function callback,某些时候只有老天才知道为啥出现奇怪的结果。 很显然,JS.Class 的作者是个细心的人,他了解像我这种习惯 "严谨" 的古董有很多,因此他为对象提供了一个 method() 方法来改善这个问题。OK,我们修改一下上面的调用方法。 var o = new MyClass(); var func = o.method("test"); o.test(o); // true func(o); // true 世界正常了。当然它对静态方法同样有效。 var MyClass = new JS.Class( { extend : { test : function() { alert(this == MyClass); } } }); MyClass.test(); // true var func1 = MyClass.test; func1(); // false var func2 = MyClass.method("test"); func2(); // true 另外一个实用的方法就是 Singleton 了,最常用的模式之一,不是吗?要知道 JavaScript 可没有 private 访问修饰符,我们无法弄一个 private initialize() 出来。 var User = new JS.Singleton( { name : "User", print : function() { document.write(this.name + "
"); } }); //var o = new User(); User.print(); 这样用来,你无法用 new User() 来创建对象实例,它会导致一个错误。 Microsoft JScript 运行时错误: 对象不支持此操作 当然,我们也可以让继承类变成 Singleton。 var User = new JS.Class( { name : "User", print : function() { document.write(this.name + "
"); } }); var Manager = new JS.Singleton(User, {}); var o = new User(); Manager.print(); 既然有了类和继承,那么 Type 的相关操作就是必须的了。要知道 JavaScript 的变量是无类型的,某些时候我们必须验证一些危险分子的身份。 var User = new JS.Class({}); var Manager = new JS.Class(User, {}); var o = new Manager("Tom"); alert(o.isA(Manager)); alert(o.isA(User)); 很典型的 C# is 关键字的用法,isA 很好地支持了继承体系。 var User = new JS.Class({}); var Manager = new JS.Class(User, {}); var o = new Manager("Tom"); alert(o.klass == Manager); // true alert(o.klass.superclass == User); // true klass 其实就是 class (class 是 JavaScript 保留字),它指向类型对象,有点类似 C# GetType()。我们还可以用 superclass 找到其基类。 利用 klass 我们可以做更多的事情,比如说调用静态成员。很显然,某些时候(比如类库或重构)比直接用 User.test() 这种硬编码要好些。 var User = new JS.Class( { extend : { test : function() { alert("Test!"); } } }); var o = new User(); o.klass.test(); 继续玩点有深度的,找出所有的子类,并创建实例。 var User = new JS.Class( { name : "User", print : function() { document.write(this.name + "
"); } }); var Manager = new JS.Class(User, { name : "Manager" }); var System = new JS.Class(User, { name : "System" }); var o = new User(); for(var i in o.klass.subclasses) { var k = o.klass.subclasses[i]; new k().print(); } JS.Class 甚至允许我们创建 Interfaces,这很奇怪不是吗?不过它并不是我们通常所理解的那种接口,它只是一个约束,用来检查某个对象是否包含接口所指定的成员。 var IUser = new JS.Interface(["print", "test"]); var User = new JS.Class( { name : "User", print : function() { }, test : function() {} }); var o = new User(); JS.Interface.ensure(o, IUser); 有点遗憾的是它只能测试实例方法,而不管静态方法和字段成员。当然,只有在测试没通过的时候,你才看到 JS.Interface.ensure 的效果。 Microsoft JScript 运行时错误: object does not implement name() 可以测试对象是否实现了多个接口。 JS.Interface.ensure(o, IUser, IOther...);
<template> <view class="container"> <view class="list-item" v-for="item in types" :key="item.id" @click="toDetail"> <view class="item-content"> <view class="left-content"> <text class="type">{{ item.application_type }}</text> <text class="name">{{ item.applicant }}</text> <text class="remark">备注:{{ item.remark }}</text> </view> <view class="right-content"> <text class="time">{{ item.submitTime }}</text> <text class="status">{{ item.status }}</text> </view> </view> </view> </view> </template> <script> export default { data() { return { types: [{ id: 1, application_type: '加班申请', applicant: '张三', submitTime: '2023-01-01', status: '待审批', remark: '紧急任务' }, { id: 2, application_type: '请假申请', applicant: '李四', submitTime: '2023-01-02', status: '待审批', remark: '病假' } ] }; }, methods: { toDetail() { uni.navigateTo({ url: '/pages/work/detail' }); } } }; </script> <style> .container { padding: 10px; } .list-item { background-color: #fafafa; border-bottom: 1px solid #eee; padding: 10px; margin-bottom: 10px; } .item-content { display: flex; justify-content: space-between; } .left-content { display: flex; flex-direction: column; } .right-content { display: flex; flex-direction: column; align-items: flex-end; } .item-content text { margin-bottom: 5px; font-size: 14px; } .type { font-weight: bold; } .remark { margin-top: 10px; } </style> {errMsg: 'navigateTo:fail page `/pages/work/detail` is not found'}
07-12
<template> <Teleport to="body"> <div v-if="visible" class="pdf-mask" @mousedown="startDrag"> <div ref="modalRef" class="pdf-modal" :style="modalStyle"> <!-- PDF 头部:标题 + 操作按钮 --> <div class="pdf-header"> <span class="title">报关资料</span> <div class="tools"> <el-button size="small" @click="zoomIn">放大</el-button> <el-button size="small" @click="zoomOut">缩小</el-button> <el-button size="small" @click="close">关闭</el-button> </div> </div> <!-- PDF 标签页 + 内容区域 --> <el-tabs v-model="activeTab" class="pdf-tabs" @tab-click="handleTabClick"> <el-tab-pane v-for="tab in tabs" :key="tab.id" :label="tab.name" :name="tab.id.toString()" > <div class="pdf-body" :ref="(el) => el && scrollRefMap.set(tab.id, el as HTMLDivElement)" @wheel="(e) => onWheel(e, tab.id)" @mousedown="(e) => onPdfMouseDown(e, tab.id)" > <div :ref="(el) => el && setPdfContainer(tab.id, el as HTMLDivElement)" /> </div> </el-tab-pane> </el-tabs> </div> </div> </Teleport> </template> <script setup lang="ts"> import { ref, reactive, watch, nextTick, onUnmounted } from 'vue' import type { CSSProperties } from 'vue' import * as pdfjsLib from 'pdfjs-dist' import pdfjsWorker from 'pdfjs-dist/build/pdf.worker?worker&url' // 修复类型导入 import type { PDFDocumentLoadingTask, PDFDocumentProxy, PDFPageProxy, RenderParameters } from 'pdfjs-dist/types/src/display/api' import type { PageViewport } from 'pdfjs-dist' import { ElButton, ElTabs, ElTabPane } from 'element-plus' import type { TabsPaneContext } from 'element-plus' // PDF.js 初始化:设置 Worker 路径 pdfjsLib.GlobalWorkerOptions.workerSrc = pdfjsWorker /* ---------------- Props & Emits 定义 ---------------- */ const props = defineProps<{ visible: boolean url?: string highlight?: { x0: number; y0: number; x1: number; y1: number } | null }>() const emit = defineEmits<{ (e: 'update:visible', value: boolean): void (e: 'tabschange', tabId: string | number): void }>() /* ---------------- 状态管理 ---------------- */ const tabs = ref<Array<{ id: string | number; name: string; url: string }>>([]) const activeTab = ref<string>('') const pdfContainerMap = new Map<string | number, HTMLDivElement>() const scrollRefMap = new Map<string | number, HTMLDivElement>() const tabScale = new Map<string | number, number>() const tabBaseViewport = new Map<string | number, PageViewport>() const pdfRealSize = ref({ width: 0, height: 0 }) const modalRef = ref<HTMLDivElement | null>(null) let abortController: AbortController | null = null let currentScrollContainer: HTMLDivElement | null = null /* ---------------- 弹窗位置 & 样式 ---------------- */ const modalStyle = reactive<{ left: string top: string width: string height: string cursor: CSSProperties['cursor'] }>({ left: '0px', top: '0px', width: '800px', height: '600px', cursor: 'default' }) let dragStart: { x: number; y: number } | null = null /* ---------------- 标签页初始化 & 渲染 ---------------- */ async function openWithTabs( list: Array<{ code: string; name: string; url: string }> ) { tabs.value = [] pdfContainerMap.clear() scrollRefMap.clear() tabScale.clear() tabBaseViewport.clear() if (abortController) { abortController.abort() abortController = null } tabs.value = list .filter(item => item.name !== 'all_files') .map(item => ({ id: item.code, name: item.name, url: item.url })) const firstTabId = tabs.value[0]?.id.toString() ?? '' activeTab.value = firstTabId emit('update:visible', true) await nextTick() const firstTab = tabs.value[0] if (firstTab) { const container = pdfContainerMap.get(firstTab.id) if (container) { await renderSinglePDF(firstTab.url, container, firstTab.id) } } // 弹窗居中 const windowWidth = window.innerWidth const windowHeight = window.innerHeight modalStyle.left = `${(windowWidth - 800) / 2}px` modalStyle.top = `${(windowHeight - 600) / 2}px` } function setPdfContainer(tabId: string | number, el: HTMLDivElement) { pdfContainerMap.set(tabId, el) } async function renderSinglePDF( pdfUrl: string, container: HTMLDivElement, currentTabId: string | number ) { if (activeTab.value !== currentTabId.toString()) return if (!(container instanceof HTMLDivElement)) return if (abortController) { abortController.abort() } abortController = new AbortController() container.innerHTML = '' const scrollContainer = scrollRefMap.get(currentTabId) if (!scrollContainer) return try { // 修复类型:使用 PDFDocumentLoadingTask const pdfLoadingTask: PDFDocumentLoadingTask = pdfjsLib.getDocument({ url: pdfUrl, signal: abortController.signal } as any) as PDFDocumentLoadingTask const pdfDoc: PDFDocumentProxy = await pdfLoadingTask.promise const pdfPage: PDFPageProxy = await pdfDoc.getPage(1) if (activeTab.value !== currentTabId.toString()) return const baseViewport = pdfPage.getViewport({ scale: 1 }) tabBaseViewport.set(currentTabId, baseViewport) const currentScale = tabScale.get(currentTabId) ?? 1 const renderScale = currentScale * (800 / baseViewport.width) const renderViewport = pdfPage.getViewport({ scale: renderScale }) const canvas = document.createElement('canvas') const ctx = canvas.getContext('2d') if (!ctx) { container.innerHTML = '<div class="pdf-error">Canvas 初始化失败</div>' return } canvas.width = renderViewport.width canvas.height = renderViewport.height canvas.dataset.scale = String(currentScale) container.appendChild(canvas) // 修复渲染参数类型 const renderParams: RenderParameters = { canvasContext: ctx, viewport: renderViewport, signal: abortController } as any await pdfPage.render(renderParams).promise if (activeTab.value !== currentTabId.toString()) { container.innerHTML = '' return } scrollContainer.style.width = `${renderViewport.width}px` scrollContainer.style.height = `${renderViewport.height}px` pdfRealSize.value = { width: renderViewport.width, height: renderViewport.height } if (props.highlight) { drawHighlight(props.highlight, currentTabId) } } catch (error) { if ((error as Error).name !== 'AbortError') { console.error(`PDF 加载失败:`, error) container.innerHTML = '<div class="pdf-error">PDF 加载失败,请重试</div>' } } } /* ---------------- 标签页切换 & 缩放 ---------------- */ function handleTabClick(tab: TabsPaneContext) { emit('tabschange', tab.paneName) } watch(activeTab, (newTabIdStr) => { const newTabId = tabs.value.find(tab => tab.id.toString() === newTabIdStr)?.id if (!newTabId) return const targetTab = tabs.value.find(tab => tab.id === newTabId) const targetContainer = pdfContainerMap.get(newTabId) if (targetTab && targetContainer) { renderSinglePDF(targetTab.url, targetContainer, newTabId) } }, { immediate: false }) const zoomIn = () => { const currentTabIdStr = activeTab.value const currentTabId = tabs.value.find(tab => tab.id.toString() === currentTabIdStr)?.id if (!currentTabId) return const newScale = Math.min((tabScale.get(currentTabId) ?? 1) + 0.2, 3) tabScale.set(currentTabId, newScale) rerenderActiveTab() } const zoomOut = () => { const currentTabIdStr = activeTab.value const currentTabId = tabs.value.find(tab => tab.id.toString() === currentTabIdStr)?.id if (!currentTabId) return const newScale = Math.max((tabScale.get(currentTabId) ?? 1) - 0.2, 0.5) tabScale.set(currentTabId, newScale) rerenderActiveTab() } async function rerenderActiveTab() { const currentTabIdStr = activeTab.value const currentTabId = tabs.value.find(tab => tab.id.toString() === currentTabIdStr)?.id if (!currentTabId) return const targetTab = tabs.value.find(tab => tab.id === currentTabId) const targetContainer = pdfContainerMap.get(currentTabId) if (targetTab && targetContainer) { await renderSinglePDF(targetTab.url, targetContainer, currentTabId) } } /* ---------------- 弹窗拖拽功能 ---------------- */ function startDrag(e: MouseEvent) { const target = e.target as HTMLElement if (!target.classList.contains('pdf-mask') && !target.closest('.pdf-header')) { return } if (!modalRef.value) return dragStart = { x: e.clientX, y: e.clientY } modalStyle.cursor = 'grabbing' document.addEventListener('mousemove', onDrag) document.addEventListener('mouseup', stopDrag) e.preventDefault() } function onDrag(e: MouseEvent) { if (!dragStart) return const dx = e.clientX - dragStart.x const dy = e.clientY - dragStart.y modalStyle.left = `${parseFloat(modalStyle.left) + dx}px` modalStyle.top = `${parseFloat(modalStyle.top) + dy}px` dragStart = { x: e.clientX, y: e.clientY } } function stopDrag() { dragStart = null modalStyle.cursor = 'default' document.removeEventListener('mousemove', onDrag) document.removeEventListener('mouseup', stopDrag) } /* ---------------- PDF 内容拖拽 ---------------- */ let isPdfDragging = false let pdfDragStart = { x: 0, y: 0 } let scrollStart = { left: 0, top: 0 } function onPdfMouseDown(e: MouseEvent, tabId: string | number) { const scrollContainer = scrollRefMap.get(tabId) if (!scrollContainer) return const currentScale = tabScale.get(tabId) ?? 1 if (currentScale <= 1) return isPdfDragging = true pdfDragStart = { x: e.clientX, y: e.clientY } scrollStart = { left: scrollContainer.scrollLeft, top: scrollContainer.scrollTop } currentScrollContainer = scrollContainer document.addEventListener('mousemove', onPdfMouseMove) document.addEventListener('mouseup', onPdfMouseUp) e.preventDefault() } function onPdfMouseMove(e: MouseEvent) { if (!isPdfDragging || !currentScrollContainer) return const dx = e.clientX - pdfDragStart.x const dy = e.clientY - pdfDragStart.y currentScrollContainer.scrollTo({ left: scrollStart.left - dx, top: scrollStart.top - dy, behavior: 'auto' }) } function onPdfMouseUp() { isPdfDragging = false currentScrollContainer = null document.removeEventListener('mousemove', onPdfMouseMove) document.removeEventListener('mouseup', onPdfMouseUp) } /* ---------------- 滚轮事件 ---------------- */ function onWheel(e: WheelEvent, tabId: string | number) { e.stopPropagation() const scrollContainer = scrollRefMap.get(tabId) if (scrollContainer) { // 考虑缩放比例的滚动速度 const scale = tabScale.get(tabId) ?? 1 scrollContainer.scrollLeft -= e.deltaX * scale scrollContainer.scrollTop -= e.deltaY * scale e.preventDefault() } } /* ---------------- 高亮区域绘制 ---------------- */ function drawHighlight( rect: { x0: number; y0: number; x1: number; y1: number }, tabId?: string | number ) { const targetTabId = tabId ?? ( tabs.value.find(tab => tab.id.toString() === activeTab.value)?.id ?? '' ) const pdfContainer = pdfContainerMap.get(targetTabId) const scrollContainer = scrollRefMap.get(targetTabId) if (!pdfContainer || !scrollContainer) return pdfContainer.querySelectorAll('.pdf-highlight').forEach(el => el.remove()) const canvas = pdfContainer.querySelector('canvas') const baseViewport = tabBaseViewport.get(targetTabId) if (!canvas || !baseViewport) return const currentScale = parseFloat(canvas.dataset.scale || '1') const scaleX = canvas.width / baseViewport.width const scaleY = canvas.height / baseViewport.height // 修复:考虑容器偏移 const containerRect = pdfContainer.getBoundingClientRect() const highlightStyle = { left: `${rect.x0 * scaleX}px`, top: `${rect.y0 * scaleY}px`, width: `${(rect.x1 - rect.x0) * scaleX}px`, height: `${(rect.y1 - rect.y0) * scaleY}px` } const highlightEl = document.createElement('div') highlightEl.className = 'pdf-highlight' Object.assign(highlightEl.style, { ...highlightStyle, position: 'absolute', background: 'rgba(0, 255, 0, 0.4)', border: '2px dashed #ff4d4f', pointerEvents: 'none', zIndex: 9999 }) pdfContainer.style.position = 'relative' pdfContainer.appendChild(highlightEl) nextTick(() => { const containerWidth = scrollContainer.clientWidth const containerHeight = scrollContainer.clientHeight scrollContainer.scrollTo({ left: Math.max(0, parseFloat(highlightStyle.left) - containerWidth/3), top: Math.max(0, parseFloat(highlightStyle.top) - containerHeight/3), behavior: 'smooth' }) }) } function drawHighlightByCoords(rect: { x0: number; y0: number; x1: number; y1: number }) { drawHighlight(rect) } /* ---------------- 监听 Props 变化 ---------------- */ watch( () => props.visible, async (isVisible) => { if (!isVisible) { tabScale.clear() if (abortController) { abortController.abort() abortController = null } currentScrollContainer = null return } await nextTick() const windowWidth = window.innerWidth const windowHeight = window.innerHeight modalStyle.left = `${(windowWidth - 800) / 2}px` modalStyle.top = `${(windowHeight - 600) / 2}px` }, { immediate: true } ) watch( () => props.highlight, (newHighlight) => { if (newHighlight && activeTab.value) { drawHighlight(newHighlight) } }, { immediate: false, deep: true } ) /* ---------------- 关闭弹窗 ---------------- */ function close() { if (abortController) { abortController.abort() abortController = null } currentScrollContainer = null pdfRealSize.value = { width: 0, height: 0 } emit('update:visible', false) } /* ---------------- 组件卸载清理 ---------------- */ onUnmounted(() => { if (abortController) { abortController.abort() abortController = null } document.removeEventListener('mousemove', onDrag) document.removeEventListener('mouseup', stopDrag) document.removeEventListener('mousemove', onPdfMouseMove) document.removeEventListener('mouseup', onPdfMouseUp) pdfContainerMap.clear() scrollRefMap.clear() tabScale.clear() tabBaseViewport.clear() currentScrollContainer = null }) /* ---------------- 暴露方法 ---------------- */ defineExpose({ openWithTabs, drawHighlight: drawHighlightByCoords }) </script> <style scoped> .pdf-mask { position: fixed; inset: 0; z-index: 9999; background: rgba(0, 0, 0, 0.3); } .pdf-modal { position: absolute; background: #ffffff; border: 1px solid #e4e7ed; border-radius: 8px; box-shadow: 0 8px 24px rgba(0, 0, 0, 0.15); display: flex; flex-direction: column; min-width: 400px; min-height: 300px; resize: both; overflow: hidden; } .pdf-header { display: flex; justify-content: space-between; align-items: center; padding: 12px 16px; background: #f5f7fa; cursor: grab; border-bottom: 1px solid #e4e7ed; user-select: none; } .pdf-header:active { cursor: grabbing; } .pdf-header .title { font-size: 16px; font-weight: 500; color: #1f2937; } .pdf-header .tools { display: flex; gap: 8px; } .pdf-tabs { flex: 1; display: flex; flex-direction: column; overflow: hidden; } .pdf-body { flex: 1; overflow: auto; position: relative; } .pdf-body canvas { pointer-events: none; display: block; } .pdf-highlight { box-sizing: border-box; } .pdf-tabs .el-tabs__content { flex: 1; overflow: hidden; padding: 0 !important; } .pdf-tabs .el-tab-pane { height: 100%; display: flex; flex-direction: column; } /* 错误提示样式 */ .pdf-error { padding: 20px; text-align: center; color: #f56c6c; } </style> 这个代码的功能是可以的,但是有一个问题,就是我放大很多很多倍之后,就会只显示左上角,拖拽只能上下拖拽,右边的数据根本看不到,根本拖拽不了
08-28
为什么匹配不到这个字符串 dubbo调用异常 com.alibaba.dubbo.rpc.RpcException: Failfast invoke providers dubbo://168.15.17.29:21883/com.htsc.pos.api.prdt.dis.PosPrdtNavApi?anyhost=true&application=pos_ms_server&check=false&cluster=failfast&default.check=false&default.delay=-1&default.reference.filter=traceFilter,authrpc,sentineldubboconsumer&default.service.filter=traceFilter,dubboLogFilter,sentineldubboprovider&delay=-1&dubbo=1.4.9-SNAPSHOT&generic=false&interface=com.htsc.pos.api.prdt.dis.PosPrdtNavApi&methods=getPosPrdtNavInfoList,getHistoryPrdtNavListKafka,getSplitPosPrdtNavListKafka,getPosPrdtNavKafka,getPosPrdtNavListKafka,getHistoryPrdtNavList,getListPrdt,getAllHistoryPrdtNavListKafkaNew,getPosPrdtNavList,getHistoryPrdtNavListFromLeap,getAllHistoryPrdtNavList,getAllHistoryPrdtNavListKafka,getPosPrdtNav,getPosPrdtNavInfoListKafka,getSplitPosPrdtNavList&payload=62914560&pid=187664&remote.timestamp=1763371483109&revision=20251117-SNAPSHOT&side=consumer&timeout=60000×tamp=1763114972569 RandomLoadBalance select from all providers [com.alibaba.dubbo.registry.integration.RegistryDirectory$InvokerDelegete@f0ec1c1, com.alibaba.dubbo.registry.integration.RegistryDirectory$InvokerDelegete@7e84fc8f, com.alibaba.dubbo.registry.integration.RegistryDirectory$InvokerDelegete@77d6e2e5, com.alibaba.dubbo.registry.integration.RegistryDirectory$InvokerDelegete@1714cb14] for service com.htsc.pos.api.prdt.dis.PosPrdtNavApi method getAllHistoryPrdtNavListKafka on consumer 168.9.10.68 use dubbo version 1.4.9-SNAPSHOT, but no luck to perform the invocation. Last error is: Invoke remote method timeout. method: getAllHistoryPrdtNavListKafka, provider: dubbo://168.15.17.29:21883/com.htsc.pos.api.prdt.dis.PosPrdtNavApi?anyhost=true&application=pos_ms_server&check=false&cluster=failfast&default.check=false&default.delay=-1&default.reference.filter=traceFilter,authrpc,sentineldubboconsumer&default.service.filter=traceFilter,dubboLogFilter,sentineldubboprovider&delay=-1&dubbo=1.4.9-SNAPSHOT&generic=false&interface=com.htsc.pos.api.prdt.dis.PosPrdtNavApi&methods=getPosPrdtNavInfoList,getHistoryPrdtNavListKafka,getSplitPosPrdtNavListKafka,getPosPrdtNavKafka,getPosPrdtNavListKafka,getHistoryPrdtNavList,getListPrdt,getAllHistoryPrdtNavListKafkaNew,getPosPrdtNavList,getHistoryPrdtNavListFromLeap,getAllHistoryPrdtNavList,getAllHistoryPrdtNavListKafka,getPosPrdtNav,getPosPrdtNavInfoListKafka,getSplitPosPrdtNavList&payload=62914560&pid=187664&remote.timestamp=1763371483109&revision=20251117-SNAPSHOT&side=consumer&timeout=60000×tamp=1763114972569, cause: Waiting server-side response timeout by scan timer. start time: 2025-11-18 16:19:24.894, end time: 2025-11-18 16:20:24.911, client elapsed: 0 ms, server elapsed: 60017 ms, timeout: 60000 ms, request: Request [id=5964454, version=2.0.0, twoway=true, event=false, broken=false, data=RpcInvocation [methodName=getAllHistoryPrdtNavListKafka, parameterTypes=[class com.htsc.pos.api.prdt.dis.req.HistoryPosPrdtNavListReq], arguments=[HistoryPosPrdtNavListReq(beginDate=1000-01-01, endDate=2025-11-18, mangerId=668191, userId=26133358, prdtCode=null, prdtConvertCode=null, prdtInfo=null, prdtAllInfo=null, key=64a74233-508e-43ae-85e6-adf219d8c66e, queryfreq=null, dateType=null, haveChildFlag=null, prdtStatusLt=null, prdtStatusGt=null, busType=null, leapHaveChildFlag=null, haveChildFlagList=null, busTypeEqual=null, prdtCodeList=null, apiUserId=null, prdtConvertCodeList=null, navDateSortType=null, lockState=null, clsFlag=null, prdtIdList=null)], attachments={async=false, path=com.htsc.pos.api.prdt.dis.PosPrdtNavApi, TraceId=f1d7c42d-f481-46ac-b2e7-e8588f829d2f, htdubbox_probe=htdubbox, interface=com.htsc.pos.api.prdt.dis.PosPrdtNavApi, ChainId=0.0.0, version=0.0.0, timeout=60000, X-Sampled=1}]], channel: /168.9.10.68:48252 -> /168.15.17.29:21883
11-19
<template> <div class="custom-table-container"> <el-table ref="customTable" :data="tableData" :border="border" :stripe="stripe" :size="size" :header-cell-style="headerStyle" :row-style="rowStyle" :cell-style="cellStyle" @selection-change="handleSelectionChange" @row-click="handleRowClick" highlight-current-row v-bind="$attrs" v-on="$listeners" > <!-- 多选列 --> <el-table-column v-if="selection" type="selection" align="center" :reserve-selection="reserveSelection" ></el-table-column> <!-- 索引列 --> <el-table-column v-if="index" type="index" :label="indexLabel" align="center" ></el-table-column> <!-- 动态列 --> <el-table-column v-for="(column, index) in columns" :key="column.prop" :prop="column.prop" :label="column.label" :width="column.width" :min-width="column.minWidth" :align="column.align || 'center'" :header-align="column.headerAlign || column.align || 'center'" :sortable="column.sortable || false" :fixed="column.fixed" :formatter="column.formatter" :show-overflow-tooltip="true" > <template slot-scope="scope"> <!-- 如果是数组,渲染 Tag --> <template v-if="Array.isArray(scope.row[column.prop])"> <el-tag v-for="(item, idx) in scope.row[column.prop]" :key="idx" :type=" typeof column.tagType === 'function' ? column.tagType(item) : column.tagType || 'info' " :style=" typeof column.tagStyle === 'function' ? column.tagStyle(item) : column.tagStyle || {} " > {{ item }} </el-tag> </template> <!-- 如果不是数组 --> <template v-else> <!-- 如果是 Tag 类型 --> <template v-if="column.tagType"> <el-tag :type=" typeof column.tagType === 'function' ? column.tagType(scope.row) : column.tagType " :style="column.tagStyle || {}" > {{ scope.row[column.prop] }} </el-tag> </template> <!-- 普通文本(支持百分比) --> <template v-else> {{ column.percentage ? `${scope.row[column.prop]}%` : scope.row[column.prop] }} </template> </template> </template> </el-table-column> </el-table> <!-- 分页 --> <!-- <div class="pagination-container" v-if="pagination"> <el-pagination @size-change="handleSizeChange" @current-change="handleCurrentChange" :current-page="currentPage" :page-sizes="pageSizes" :page-size="pageSize" :layout="paginationLayout" :total="total" :background="paginationBackground" ></el-pagination> </div> --> </div> </template> <script> export default { name: "CustomTable", props: { // 表格数据 tableData: { type: Array, default: () => [], }, // 列配置 columns: { type: Array, required: true, validator: (value) => { return value.every((item) => item.prop && item.label); }, }, // 是否显示边框 border: { type: Boolean, default: false, }, // 是否为斑马纹 stripe: { type: Boolean, default: true, }, // 表格尺寸 size: { type: String, default: "medium", validator: (value) => ["medium", "small", "mini"].includes(value), }, // 表头样式 headerStyle: { type: Object, default: () => ({ // backgroundColor: '#f5f7fa', // color: '#606266', // fontWeight: 'bold' }), }, // 行样式 rowStyle: { type: Function, default: ({ rowIndex }) => { // return rowIndex % 2 === 1 // ? { // background: 'linear-gradient(90deg, #562711 0%, rgba(86, 46, 17, 0.27) 100%)' // } // : {}; }, }, // 是否显示多选框 selection: { type: Boolean, default: false, }, // 是否显示索引列 index: { type: Boolean, default: false, }, // 索引列标题 indexLabel: { type: String, default: "序号", }, // 是否显示分页 pagination: { type: Boolean, default: false, }, // 当前页码 currentPage: { type: Number, default: 1, }, // 每页显示条目个数 pageSize: { type: Number, default: 10, }, // 每页显示个数选择器的选项设置 pageSizes: { type: Array, default: () => [10, 20, 30, 50], }, // 总条目数 total: { type: Number, default: 0, }, // 分页布局 paginationLayout: { type: String, default: "total, sizes, prev, pager, next, jumper", }, // 分页背景色 paginationBackground: { type: Boolean, default: true, }, rowBorderColor: { type: String, default: "", }, //tag 类型配置 tagTypes: { type: Object, default: () => ({}), }, // 是否保留选中状态(跨页保持) reserveSelection: { type: Boolean, default: false, }, // 是否默认选中第一行 defaultSelectFirstRow: { type: Boolean, default: false, }, }, watch: { tableData: { immediate: true, handler(newVal) { if (newVal && newVal.length > 0 && this.defaultSelectFirstRow) { this.$nextTick(() => { this.setDefaultSelection(); }); } }, }, }, methods: { handleSelectionChange(val) { this.$emit("selection-change", val); }, handleSizeChange(val) { this.$emit("size-change", val); }, handleCurrentChange(val) { this.$emit("current-change", val); }, // 点击行事件 handleRowClick(row, column, event) { // this.$refs.customTable.toggleRowSelection(row); this.$emit("row-click", row, column, event); }, // 设置默认选中第一行 setDefaultSelection() { if (this.tableData && this.tableData.length > 0) { const firstRow = this.tableData[0]; // this.$refs.customTable.toggleRowSelection(firstRow, true); this.$emit("selection-change", [firstRow]); } }, // 清除所有选中 clearSelection() { this.$refs.customTable.clearSelection(); }, getTagType(row, column) { if (typeof column.tagType === "function") { return column.tagType(row); } const value = row[column.prop]; if (this.tagTypes[value] && this.tagTypes[value].type) { return this.tagTypes[value].type; } return ""; }, // 格式化单元格值 formatCellValue(row, column) { if (column.formatter) { return column.formatter(row, column); } return row[column.prop]; }, getTagStyle(column) { // 优先使用列配置的样式 if (column.tagStyle) return column.tagStyle; return { backgroundColor: "#F4F4F5", color: "#909399", border: "1px solid #D3D4D6", }; }, // 用于控制下边框 cellStyle({ row, column, rowIndex }) { const style = {}; if (this.rowBorderColor) { style.borderBottom = `1px solid ${this.rowBorderColor}`; } return style; }, }, }; </script> <style scoped lang="scss"> .custom-table-container { width: 100%; // height: 100%; // display: flex; // flex-direction: column; } .pagination-container { margin-top: 20px; text-align: right; } /deep/ .el-table, /deep/ .el-table th, /deep/ .el-table tr, /deep/ .el-table td { background-color: transparent !important; color: #fff; } /deep/ .el-table, /deep/td.el-table__cell, /deep/.el-table, /deep/th.el-table__cell, /deep/.is-leaf { border-bottom: 0 !important; } /deep/ .el-table .el-table__cell { padding: 2px !important; } ::v-deep .el-table__cell { padding-left: 0 !important; padding-right: 0 !important; } /deep/.el-table--border::after, /deep/.el-table--group::after, /deep/.el-table::before { background-color: transparent !important; /* 移除底部白线 */ } /deep/.el-table .cell { padding-left: 0 !important; padding-right: 0 !important; } /* 新增:确保 tag 在单元格中居中显示 */ /deep/ .el-table .cell .el-tag { display: inline-flex; align-items: center; justify-content: center; min-width: 60px; } /* 添加行点击高亮效果 */ /deep/ .el-table--enable-row-hover .el-table__body tr:hover > td { background-color: rgba(255, 255, 255, 0.1) !important; cursor: pointer; } /deep/ .el-table__body tr.current-row > td { background-color: rgba(255, 255, 255, 0.2) !important; } </style> 我公用组件这样写的 我父组件怎么配置
最新发布
12-07
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值