活动介绍

display: flex;position: sticky;超出一屏后会滚上去

时间: 2025-04-20 13:34:07 浏览: 28
### 解决 CSS Flex 布局与 Sticky 定位组合使用的滚动问题 当 `display: flex` 与 `position: sticky` 组合使用时,可能会遇到一些意料之外的行为,尤其是在处理滚动条显示方面。为了确保在不同浏览器环境下的一致性和预期行为,可以采取以下方法: #### 方法一:调整父容器的溢出属性 通过设置父级容器的 `overflow` 属性为 `auto` 而不是 `scroll-y` 可以有效防止不必要的滚动条出现[^1]。 ```css .parent-container { display: flex; overflow: auto; /* 使用此代替 scroll-y */ } ``` #### 方法二:确保子元素具有适当的高度约束 对于需要应用粘性定位 (`position: sticky`) 的子项来说,应该为其设定明确的最大高度或最小高度限制,从而避免因内容过少而提前触发滚动效果。 ```css .child-item { max-height: calc(100vh - 80px); /* 根据实际需求调整数值 */ min-height: fit-content; position: sticky; top: 0; } ``` #### 方法三:利用伪类优化视觉体验 为了让用户体验更加流畅自然,在定义 `.child-item` 类的同时还可以加入额外的状态样式,比如悬停状态下的背景颜色变化等,增强交互感而不影响布局逻辑。 ```css .child-item:hover { background-color: rgba(255, 255, 255, 0.9); } /* 或者更具体的例子 */ .make-me-sticky { position: relative; position: -webkit-sticky; position: -moz-sticky; position: -ms-sticky; position: -o-sticky; position: sticky; top: 0; } ``` 以上措施能够帮助改善 `flexbox` 结构下 `sticky` 元素可能出现的各种滚动异常现象,提高跨平台兼容性并提供更好的用户体验。
阅读全文

相关推荐

在原有的页面上添加 ,底部 已加载完毕 分割线 避免被购物车挡住 原有页面如下 : <template> <view class="shop-container"> <view class="fixed-header"> <view class="search-bar"> <uni-search-bar placeholder="搜索菜品" radius="100" @confirm="searchDish" @input="searchInput" cancelButton="none" /> </view> <scroll-view class="category-scroll" scroll-x="true"> <view v-for="(category, index) in categories" :key="index" class="category-item" :class="{ active: currentCategory === index }" @click="changeCategory(index)" > {{ category.name }} </view> </scroll-view> </view> <scroll-view class="dish-list" scroll-y="true"> <view v-for="(dish, index) in filteredDishes" :key="dish.id" class="dish-item"> <image class="dish-image" :src="dish.image" mode="aspectFill" /> <view class="dish-info"> <text class="dish-name">{{ dish.name }}</text> <text class="dish-price">¥{{ dish.price.toFixed(2) }}</text> <text class="dish-stock" :class="{ 'low-stock': dish.stock < 5 }"> 剩余: {{ dish.stock }}份 </text> </view> <view class="quantity-selector" v-if="cartItemQuantity(dish.id) > 0"> <button class="quantity-btn" @click="decreaseQuantity(dish)"> <uni-icons type="minus" size="16" color="#ff6b6b"></uni-icons> </button> <text class="quantity-text">{{ cartItemQuantity(dish.id) }}</text> <button class="quantity-btn" :disabled="cartItemQuantity(dish.id) >= dish.stock" @click="increaseQuantity(dish)" > <uni-icons type="plus" size="16" color="#ff6b6b"></uni-icons> </button> </view> <view class="add-btn-container" v-else> <button v-if="dish.stock > 0" class="add-btn" :disabled="dish.stock <= 0" @click="addToCart(dish)"> <uni-icons type="plusempty" size="20" color="#fff"></uni-icons> </button> <text v-if="dish.stock <= 0" class="sold-out-text">已售罄</text> </view> </view> </scroll-view> <view class="cart-bar" @click="showCartDetail"> <view class="cart-icon-container"> <uni-icons type="cart-filled" size="30" color="#fff"></uni-icons> <view v-if="cartItemCount > 0" class="cart-badge">{{ cartItemCount }}</view> </view> <text class="total-price">¥{{ totalPrice.toFixed(2) }}</text> <button class="checkout-btn" :disabled="cartItemCount === 0">去结算</button> </view> <uni-popup ref="cartPopup" type="bottom" :maskClick="false"> <view class="cart-detail"> <view class="cart-header"> <text class="cart-title">已选菜品</text> <uni-icons type="close" size="24" color="#999" @click="closeCartDetail"></uni-icons> </view> <scroll-view class="cart-list" scroll-y="true"> <view v-for="(item, index) in cart" :key="item.id" class="cart-item"> <image class="cart-item-image" :src="item.image" mode="aspectFill" /> <view class="cart-item-info"> <text class="cart-item-name">{{ item.name }}</text> <text class="cart-item-price">¥{{ item.price.toFixed(2) }}</text> </view> <view class="cart-item-quantity"> <button class="cart-quantity-btn" @click="decreaseCartQuantity(item)"> <uni-icons type="minus" size="16" color="#ff6b6b"></uni-icons> </button> <text class="cart-quantity-text">{{ item.quantity }}</text> <button class="cart-quantity-btn" :disabled="item.quantity >= item.stock" @click="increaseCartQuantity(item)" > <uni-icons type="plus" size="16" color="#ff6b6b"></uni-icons> </button> </view> </view> </scroll-view> <view class="cart-footer"> <text class="cart-total">总计: ¥{{ totalPrice.toFixed(2) }}</text> <button class="cart-checkout-btn" @click="checkout">去结算</button> </view> </view> </uni-popup> </view> </template> <script> export default { data() { return { currentCategory: 0, // 当前选中的分类索引 searchKeyword: '', // 搜索关键词 cart: [], // 购物车 // 菜品分类 categories: [ { id: 0, name: '全部' }, { id: 1, name: '热销' }, { id: 2, name: '主食' }, { id: 3, name: '汤类' }, { id: 4, name: '小吃' }, { id: 5, name: '饮品' } ], // 菜品数据 dishes: [ { id: 1, name: '鱼香肉丝', price: 38, stock: 15, image: '/static/images/dishes/1.jpeg', category: 1 }, { id: 2, name: '宫保鸡丁', price: 42, stock: 8, image: '/static/images/dishes/2.jpeg', category: 1 }, { id: 3, name: '麻婆豆腐', price: 28, stock: 0, image: '/static/images/dishes/3.jpg', category: 1 }, { id: 4, name: '扬州炒饭', price: 25, stock: 12, image: '/static/images/dishes/4.jpeg', category: 2 }, { id: 5, name: '番茄鸡蛋面', price: 22, stock: 20, image: '/static/images/dishes/4.jpeg', category: 2 }, { id: 6, name: '酸辣汤', price: 18, stock: 5, image: '/static/images/dishes/4.jpeg', category: 3 }, { id: 7, name: '春卷', price: 15, stock: 3, image: '/static/images/dishes/4.jpeg', category: 4 }, { id: 8, name: '冰镇酸梅汤', price: 12, stock: 25, image: '/static/images/dishes/dish08.jpg', category: 5 } ] } }, computed: { // 过滤后的菜品列表 filteredDishes() { let result = [...this.dishes]; // 按分类过滤 if (this.currentCategory > 0) { result = result.filter(dish => dish.category === this.currentCategory); } // 按关键词搜索 if (this.searchKeyword) { const keyword = this.searchKeyword.toLowerCase(); result = result.filter(dish => dish.name.toLowerCase().includes(keyword) ); } return result; }, // 购物车商品数量 cartItemCount() { return this.cart.reduce((total, item) => total + item.quantity, 0); }, // 总金额 totalPrice() { return this.cart.reduce((total, item) => total + (item.price * item.quantity), 0); } }, methods: { // 切换分类 changeCategory(index) { this.currentCategory = index; }, // 搜索输入 searchInput(e) { this.searchKeyword = e.value; }, // 搜索确认 searchDish(e) { this.searchKeyword = e.value; }, // 获取菜品在购物车中的数量 cartItemQuantity(dishId) { const item = this.cart.find(item => item.id === dishId); return item ? item.quantity : 0; }, // 添加到购物车 addToCart(dish) { if (dish.stock <= 0) return; // 检查是否已在购物车 const existingItem = this.cart.find(item => item.id === dish.id); if (existingItem) { // 如果已在购物车,增加数量 this.increaseQuantity(dish); } else { // 添加到购物车 this.cart.push({ ...dish, quantity: 1 }); } }, // 增加菜品数量 increaseQuantity(dish) { const item = this.cart.find(item => item.id === dish.id); if (item && item.quantity < dish.stock) { item.quantity++; } }, // 减少菜品数量 decreaseQuantity(dish) { const item = this.cart.find(item => item.id === dish.id); if (item) { if (item.quantity > 1) { item.quantity--; } else { // 数量为1时减少则移除 this.cart = this.cart.filter(cartItem => cartItem.id !== dish.id); } } }, // 增加购物车中菜品数量 increaseCartQuantity(item) { const dish = this.dishes.find(d => d.id === item.id); if (dish && item.quantity < dish.stock) { item.quantity++; } }, // 减少购物车中菜品数量 decreaseCartQuantity(item) { if (item.quantity > 1) { item.quantity--; } else { // 数量为1时减少则移除 this.cart = this.cart.filter(cartItem => cartItem.id !== item.id); } }, // 显示购物车详情 showCartDetail() { if (this.cartItemCount > 0) { this.$refs.cartPopup.open(); } }, // 关闭购物车详情 closeCartDetail() { this.$refs.cartPopup.close(); }, // 结算 checkout() { uni.showToast({ title: '结算成功', icon: 'success' }); // 实际开发中这里会有支付流程 this.closeCartDetail(); } } } </script> <style lang="scss" scoped> .shop-container { display: flex; flex-direction: column; height: 100vh; background-color: #f5f5f5; padding-bottom: 100rpx; } /* 新增顶部固定容器 */ .fixed-header { position: sticky; top: 0; z-index: 1000; /* 确保在顶部 */ background: white; /* 背景色,避免滚动时内容透出 */ box-shadow: 0 2rpx 10rpx rgba(0, 0, 0, 0.1); /* 添加阴影增强层次感 */ } .search-bar { padding: 20rpx; background-color: #fff; } .category-scroll { white-space: nowrap; background-color: #fff; padding: 20rpx 0; border-bottom: 1rpx solid #eee; } .category-item { display: inline-block; padding: 10rpx 30rpx; margin: 0 10rpx; border-radius: 30rpx; background-color: #f5f5f5; font-size: 28rpx; &.active { background-color: #ff6b6b; color: #fff; } } .dish-list { flex: 1; padding: 20rpx; /* 添加顶部内边距,避免内容被固定栏遮挡 */ // padding-top: 0; } .dish-item { display: flex; align-items: center; background-color: #fff; border-radius: 16rpx; margin-bottom: 15rpx; padding: 20rpx; box-shadow: 0 2rpx 12rpx rgba(0, 0, 0, 0.05); } .dish-image { width: 180rpx; height: 180rpx; border-radius: 8rpx; margin-right: 20rpx; } .dish-info { flex: 1; display: flex; flex-direction: column; justify-content: space-between; } .dish-name { font-size: 28rpx; font-weight: bold; color: #333; } .dish-price { font-size: 30rpx; color: #ff6b6b; margin-top: 10rpx; } .dish-stock { font-size: 24rpx; color: #999; margin-top: 10rpx; &.low-stock { color: #ff6b6b; } } .add-btn-container { display: flex; align-items: center; } .add-btn { width: 70rpx; height: 70rpx; border-radius: 50%; background-color: #ff6b6b; display: flex; justify-content: center; align-items: center; color: #fff; font-size: 24rpx; padding: 0; margin: 0; &.disabled { background-color: #ccc; } } .sold-out-text { font-size: 28rpx; color: #999; padding: 10rpx 20rpx; background-color: #f5f5f5; border-radius: 8rpx; } .quantity-selector { display: flex; align-items: center; background-color: #fff; border-radius: 40rpx; padding: 6rpx; box-shadow: 0 2rpx 8rpx rgba(0, 0, 0, 0.1); } .quantity-btn { width: 50rpx; height: 50rpx; border-radius: 50%; display: flex; justify-content: center; align-items: center; padding: 0; margin: 0; background-color: #fff; border: 1rpx solid #eee; } .quantity-text { font-size: 28rpx; min-width: 60rpx; text-align: center; font-weight: bold; } .cart-bar { position: fixed; bottom: 0; left: 0; right: 0; height: 100rpx; background-color: #333; display: flex; align-items: center; padding: 0 30rpx; color: #fff; z-index: 999; } .cart-icon-container { position: relative; margin-right: 20rpx; } .cart-badge { position: absolute; top: -10rpx; right: -10rpx; background-color: #ff6b6b; color: #fff; border-radius: 50%; width: 40rpx; height: 40rpx; display: flex; justify-content: center; align-items: center; font-size: 24rpx; } .total-price { flex: 1; font-size: 36rpx; font-weight: bold; } .checkout-btn { background-color: #ff6b6b; color: #fff; border-radius: 50rpx; padding: 0 40rpx; height: 70rpx; line-height: 70rpx; font-size: 30rpx; &[disabled] { background-color: #999; opacity: 0.6; } } /* 购物车详情弹窗样式 */ .cart-detail { background-color: #fff; border-top-left-radius: 20rpx; border-top-right-radius: 20rpx; padding: 30rpx; max-height: 80vh; display: flex; flex-direction: column; } .cart-header { display: flex; justify-content: space-between; align-items: center; padding-bottom: 20rpx; border-bottom: 1rpx solid #eee; margin-bottom: 20rpx; } .cart-title { font-size: 36rpx; font-weight: bold; color: #333; } .cart-list { flex: 1; max-height: 60vh; } .cart-item { display: flex; align-items: center; padding: 20rpx 0; border-bottom: 1rpx solid #f5f5f5; } .cart-item-image { width: 120rpx; height: 120rpx; border-radius: 8rpx; margin-right: 20rpx; } .cart-item-info { flex: 1; } .cart-item-name { font-size: 30rpx; color: #333; margin-bottom: 10rpx; } .cart-item-price { font-size: 28rpx; color: #ff6b6b; } .cart-item-quantity { display: flex; align-items: center; } .cart-quantity-btn { width: 50rpx; height: 50rpx; border-radius: 50%; display: flex; justify-content: center; align-items: center; padding: 0; margin: 0 10rpx; background-color: #fff; border: 1rpx solid #eee; } .cart-quantity-text { font-size: 28rpx; min-width: 60rpx; text-align: center; font-weight: bold; } .cart-footer { display: flex; justify-content: space-between; align-items: center; margin-top: 20rpx; padding-top: 20rpx; border-top: 1rpx solid #eee; } .cart-total { font-size: 36rpx; font-weight: bold; color: #ff6b6b; } .cart-checkout-btn { background-color: #ff6b6b; color: #fff; border-radius: 50rpx; padding: 0 40rpx; height: 80rpx; line-height: 80rpx; font-size: 32rpx; } </style>

以下代碼載入醫案后,當轉到其他頁面后再點擊醫案編輯器時,能保持原先載入醫案的狀態: <template> <WangEditor v-model="content" @response="(msg) => content = msg" /> <mreditor1 ref="mreditor1" />
<button @click="toggleDisplayFormat">{{ displayButtonText }}</button> <button @click="resetAll" class="reset-btn">輸入新醫案</button> <input v-model="fetchId" placeholder="輸入醫案ID" type="number" min="1" /> <button @click="fetchById">載入醫案</button> <button @click="updateContent" class="update-btn">更新醫案</button> <button @click="deleteContent" class="delete-btn">刪除醫案</button> <button @click="submitContent" class="submit-btn">提交醫案</button> 醫案 ID: {{ submittedId }} 醫案編輯器
</template> <script> import WangEditor from './WangEditor.vue'; import mreditor1 from './mreditor1.vue'; export default { components: { WangEditor, mreditor1 }, data() { return { content: '', // 移除 isVisible submittedId: null, fetchId: null, dnTags: [], // 存储从API获取的标签数据 // 新增:显示格式状态 showRawHtml: false // false显示渲染格式,true显示原始HTML标签 }; }, computed: { // 计算高亮后的内容 highlightedContent() { if (!this.dnTags.length || !this.content) return this.content; const tempEl = document.createElement('div'); tempEl.innerHTML = this.content; const walker = document.createTreeWalker( tempEl, NodeFilter.SHOW_TEXT ); const nodes = []; while (walker.nextNode()) { nodes.push(walker.currentNode); } nodes.forEach(node => { let text = node.nodeValue; let newHtml = text; this.dnTags .slice() .sort((a, b) => b.length - a.length) .forEach(tag => { const regex = new RegExp( escapeRegExp(tag), 'g' ); newHtml = newHtml.replace( regex, ${tag} ); }); if (newHtml !== text) { const span = document.createElement('span'); span.innerHTML = newHtml; node.parentNode.replaceChild(span, node); } }); return tempEl.innerHTML; }, // 新增:根据状态返回显示内容 displayContent() { if (this.showRawHtml) { // 显示原始HTML标签 return this.escapeHtml(this.content); } // 显示渲染格式(带高亮) return this.highlightedContent; }, // 新增:动态按钮文字 displayButtonText() { return this.showRawHtml ? '顯示渲染格式' : '顯示原始標籤'; } }, async mounted() { await this.fetchDNTags(); }, methods: { // HTML转义方法 escapeHtml(unsafe) { if (!unsafe) return ''; return unsafe .replace(/&/g, "&") .replace(/</g, "<") .replace(/>/g, ">") .replace(/"/g, """) .replace(/'/g, "'"); }, // 修改:切换显示格式 toggleDisplayFormat() { this.showRawHtml = !this.showRawHtml; }, // 其余方法保持不变 async fetchDNTags() { try { const response = await fetch('DNTag/?format=json'); const data = await response.json(); this.dnTags = data .map(item => item.dnname) .filter(name => name && name.trim().length > 0); } catch (error) { console.error('获取标签失败:', error); alert('标签数据加载失败,高亮功能不可用'); } }, async fetchById() { if (!this.fetchId) { alert('請輸入有效的醫案ID'); return; } try { const response = await fetch(MRInfo/${this.fetchId}/?format=json); if (response.ok) { const data = await response.json(); this.$refs.mreditor1.formData.mrname = data.mrname || ''; this.$refs.mreditor1.formData.mrposter = data.mrposter || ''; this.content = data.mrcase || ''; this.submittedId = data.id; this.fetchId = null; await this.fetchDNTags(); alert('醫案數據加載成功!'); } else if (response.status === 404) { alert('未找到該ID的醫案'); } else { throw new Error('獲取醫案失敗'); } } catch (error) { console.error('Error:', error); alert(獲取醫案失敗: ${error.message}); } }, async submitContent() { const formData = this.$refs.mreditor1.getFormData(); const postData = { mrcase: this.content, ...formData }; try { const response = await fetch('MRInfo/?format=json', { method: 'POST', headers: { 'Content-Type': 'application/json', }, body: JSON.stringify(postData), }); if(response.ok) { const data = await response.json(); this.submittedId = data.id; alert('醫案提交成功!'); } else { throw new Error('提交失败'); } } catch (error) { console.error('Error:', error); alert(提交失败: ${error.message}); } }, async updateContent() { if (!this.submittedId) return; const formData = this.$refs.mreditor1.getFormData(); const postData = { mrcase: this.content, ...formData }; try { const response = await fetch(MRInfo/${this.submittedId}/?format=json, { method: 'PUT', headers: { 'Content-Type': 'application/json', }, body: JSON.stringify(postData), }); if (response.ok) { alert('醫案更新成功!'); } else { throw new Error('更新失败'); } } catch (error) { console.error('Error:', error); alert(更新失败: ${error.message}); } }, async deleteContent() { if (!this.submittedId) return; if (!confirm('確定要刪除這個醫案嗎?')) return; try { const response = await fetch(MRInfo/${this.submittedId}/?format=json, { method: 'DELETE', headers: { 'Content-Type': 'application/json', } }); if (response.ok) { this.resetAll(); alert('醫案刪除成功!'); } else { throw new Error('刪除失败'); } } catch (error) { console.error('Error:', error); alert(刪除失败: ${error.message}); } }, resetAll() { this.content = ''; this.submittedId = null; this.fetchId = null; this.showRawHtml = false; // 重置显示格式 this.$refs.mreditor1.resetForm(); } } }; function escapeRegExp(string) { return string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'); } </script> <style scoped> /* 样式保持不变 */ .wangeditor { flex: 1; padding: 10px; overflow-y: auto; } .right-panel { position: fixed; top: 56px; bottom: 45px; right: 0; width: 30%; background: white; padding: 10px; z-index: 100; overflow-y: auto; } .content-display { position: fixed; top: 490px; left: 0; width: 70%; bottom: 45px; z-index: 999; background-color: white; overflow-y: auto; padding: 10px; border: 1px solid #eee; white-space: pre-wrap; /* 保留原始格式 */ } .sticky-footer { display: flex; justify-content: flex-end; align-items: center; position: fixed; bottom: 0; left: 0; width: 100%; background-color: #ffd800ff; z-index: 999; padding: 10px 20px; box-sizing: border-box; flex-wrap: wrap; } .sticky-footer > span { margin-left: 5px; display: flex; align-items: center; } .submitted-id { padding: 2px; background-color: #e2f0fd; color: #004085; border-radius: 4px; } .reset-btn { margin-left: 10px; padding: 2px; background-color: #dc3545; color: white; border: none; border-radius: 4px; cursor: pointer; } .reset-btn:hover { background-color: #c82333; } .id-input { display: flex; align-items: center; } .id-input input { width: 100px; padding: 2px 5px; margin-right: 5px; border: 1px solid #ccc; border-radius: 4px; } .submit-btn { background-color: #28a745; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .update-btn { background-color: #007bff; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .delete-btn { background-color: #dc3545; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .submit-btn:hover { background-color: #218838; } .update-btn:hover { background-color: #0069d9; } .delete-btn:hover { background-color: #c82333; } </style>

以下頁面代碼當轉到其他頁面后再回到此頁面時時,能保持原先頁面的狀態,請完整展示修改后的代碼: <WangEditor v-model="content" @response="(msg) => content = msg" /> <mreditor1 ref="mreditor1" />
<button @click="toggleDisplayFormat">{{ displayButtonText }}</button> <button @click="resetAll" class="reset-btn">輸入新醫案</button> <input v-model="fetchId" placeholder="輸入醫案ID" type="number" min="1" /> <button @click="fetchById">載入醫案</button> <button @click="updateContent" class="update-btn">更新醫案</button> <button @click="deleteContent" class="delete-btn">刪除醫案</button> <button @click="submitContent" class="submit-btn">提交醫案</button> 醫案 ID: {{ submittedId }} 醫案編輯器
</template> <script> import WangEditor from './WangEditor.vue'; import mreditor1 from './mreditor1.vue'; export default { components: { WangEditor, mreditor1 }, data() { return { content: '', // 移除 isVisible submittedId: null, fetchId: null, dnTags: [], // 存储从API获取的标签数据 // 新增:显示格式状态 showRawHtml: false // false显示渲染格式,true显示原始HTML标签 }; }, computed: { // 计算高亮后的内容 highlightedContent() { if (!this.dnTags.length || !this.content) return this.content; const tempEl = document.createElement('div'); tempEl.innerHTML = this.content; const walker = document.createTreeWalker( tempEl, NodeFilter.SHOW_TEXT ); const nodes = []; while (walker.nextNode()) { nodes.push(walker.currentNode); } nodes.forEach(node => { let text = node.nodeValue; let newHtml = text; this.dnTags .slice() .sort((a, b) => b.length - a.length) .forEach(tag => { const regex = new RegExp( escapeRegExp(tag), 'g' ); newHtml = newHtml.replace( regex, ${tag} ); }); if (newHtml !== text) { const span = document.createElement('span'); span.innerHTML = newHtml; node.parentNode.replaceChild(span, node); } }); return tempEl.innerHTML; }, // 新增:根据状态返回显示内容 displayContent() { if (this.showRawHtml) { // 显示原始HTML标签 return this.escapeHtml(this.content); } // 显示渲染格式(带高亮) return this.highlightedContent; }, // 新增:动态按钮文字 displayButtonText() { return this.showRawHtml ? '顯示渲染格式' : '顯示原始標籤'; } }, async mounted() { await this.fetchDNTags(); }, methods: { // HTML转义方法 escapeHtml(unsafe) { if (!unsafe) return ''; return unsafe .replace(/&/g, "&") .replace(/</g, "<") .replace(/>/g, ">") .replace(/"/g, """) .replace(/'/g, "'"); }, // 修改:切换显示格式 toggleDisplayFormat() { this.showRawHtml = !this.showRawHtml; }, // 其余方法保持不变 async fetchDNTags() { try { const response = await fetch('DNTag/?format=json'); const data = await response.json(); this.dnTags = data .map(item => item.dnname) .filter(name => name && name.trim().length > 0); } catch (error) { console.error('获取标签失败:', error); alert('标签数据加载失败,高亮功能不可用'); } }, async fetchById() { if (!this.fetchId) { alert('請輸入有效的醫案ID'); return; } try { const response = await fetch(MRInfo/${this.fetchId}/?format=json); if (response.ok) { const data = await response.json(); this.$ refs.mreditor1.formData.mrname = data.mrname || ''; this.$ refs.mreditor1.formData.mrposter = data.mrposter || ''; this.content = data.mrcase || ''; this.submittedId = data.id; this.fetchId = null; await this.fetchDNTags(); alert('醫案數據加載成功!'); } else if (response.status === 404) { alert('未找到該ID的醫案'); } else { throw new Error('獲取醫案失敗'); } } catch (error) { console.error('Error:', error); alert(獲取醫案失敗: ${error.message}); } }, async submitContent() { const formData = this.$ refs.mreditor1.getFormData(); const postData = { mrcase: this.content, ...formData }; try { const response = await fetch('MRInfo/?format=json', { method: 'POST', headers: { 'Content-Type': 'application/json', }, body: JSON.stringify(postData), }); if(response.ok) { const data = await response.json(); this.submittedId = data.id; alert('醫案提交成功!'); } else { throw new Error('提交失败'); } } catch (error) { console.error('Error:', error); alert(提交失败: ${error.message}); } }, async updateContent() { if (!this.submittedId) return; const formData = this.$ refs.mreditor1.getFormData(); const postData = { mrcase: this.content, ...formData }; try { const response = await fetch(MRInfo/${this.submittedId}/?format=json, { method: 'PUT', headers: { 'Content-Type': 'application/json', }, body: JSON.stringify(postData), }); if (response.ok) { alert('醫案更新成功!'); } else { throw new Error('更新失败'); } } catch (error) { console.error('Error:', error); alert(更新失败: ${error.message}); } }, async deleteContent() { if (!this.submittedId) return; if (!confirm('確定要刪除這個醫案嗎?')) return; try { const response = await fetch(MRInfo/${this.submittedId}/?format=json, { method: 'DELETE', headers: { 'Content-Type': 'application/json', } }); if (response.ok) { this.resetAll(); alert('醫案刪除成功!'); } else { throw new Error('刪除失败'); } } catch (error) { console.error('Error:', error); alert(刪除失败: ${error.message}); } }, resetAll() { this.content = ''; this.submittedId = null; this.fetchId = null; this.showRawHtml = false; // 重置显示格式 this.$ refs.mreditor1.resetForm(); } } }; function escapeRegExp(string) { return string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'); } </script> <style scoped> /* 样式保持不变 */ .wangeditor { flex: 1; padding: 10px; overflow-y: auto; } .right-panel { position: fixed; top: 56px; bottom: 45px; right: 0; width: 30%; background: white; padding: 10px; z-index: 100; overflow-y: auto; } .content-display { position: fixed; top: 490px; left: 0; width: 70%; bottom: 45px; z-index: 999; background-color: white; overflow-y: auto; padding: 10px; border: 1px solid #eee; white-space: pre-wrap; /* 保留原始格式 */ } .sticky-footer { display: flex; justify-content: flex-end; align-items: center; position: fixed; bottom: 0; left: 0; width: 100%; background-color: #ffd800ff; z-index: 999; padding: 10px 20px; box-sizing: border-box; flex-wrap: wrap; } .sticky-footer > span { margin-left: 5px; display: flex; align-items: center; } .submitted-id { padding: 2px; background-color: #e2f0fd; color: #004085; border-radius: 4px; } .reset-btn { margin-left: 10px; padding: 2px; background-color: #dc3545; color: white; border: none; border-radius: 4px; cursor: pointer; } .reset-btn:hover { background-color: #c82333; } .id-input { display: flex; align-items: center; } .id-input input { width: 100px; padding: 2px 5px; margin-right: 5px; border: 1px solid #ccc; border-radius: 4px; } .submit-btn { background-color: #28a745; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .update-btn { background-color: #007bff; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .delete-btn { background-color: #dc3545; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .submit-btn:hover { background-color: #218838; } .update-btn:hover { background-color: #0069d9; } .delete-btn:hover { background-color: #c82333; } </style>

以下代碼在content-display的部份在footer增加一個切換按鈕,除了原本顯示格式外,可切換成帶有html標籤的格式: <template> <WangEditor v-model="content" @response="(msg) => content = msg" /> <mreditor1 ref="mreditor1" />
<button @click="toggleContent">顯示/隱藏標籤</button> <button @click="resetAll" class="reset-btn">輸入新醫案</button> <input v-model="fetchId" placeholder="輸入醫案ID" type="number" min="1" /> <button @click="fetchById">獲取醫案</button> <button @click="updateContent" class="update-btn">更新醫案</button> <button @click="deleteContent" class="delete-btn">刪除醫案</button> <button @click="submitContent" class="submit-btn">提交醫案</button> 醫案 ID: {{ submittedId }} 醫案編輯器
</template> <script> import WangEditor from './WangEditor.vue'; import mreditor1 from './mreditor1.vue'; export default { components: { WangEditor, mreditor1 }, data() { return { content: '', isVisible: true, submittedId: null, fetchId: null, dnTags: [] // 存储从API获取的标签数据 }; }, computed: { // 计算高亮后的内容 highlightedContent() { if (!this.dnTags.length || !this.content) return this.content; // 创建临时DOM元素处理高亮 const tempEl = document.createElement('div'); tempEl.innerHTML = this.content; // 遍历所有文本节点 const walker = document.createTreeWalker( tempEl, NodeFilter.SHOW_TEXT ); const nodes = []; while (walker.nextNode()) { nodes.push(walker.currentNode); } // 对每个文本节点进行处理 nodes.forEach(node => { let text = node.nodeValue; let newHtml = text; // 对每个标签进行高亮处理(按长度降序排序,避免短标签优先匹配) this.dnTags .slice() .sort((a, b) => b.length - a.length) .forEach(tag => { const regex = new RegExp( escapeRegExp(tag), 'g' ); newHtml = newHtml.replace( regex, ${tag} ); }); // 如果内容有变化则替换节点 if (newHtml !== text) { const span = document.createElement('span'); span.innerHTML = newHtml; node.parentNode.replaceChild(span, node); } }); return tempEl.innerHTML; } }, async mounted() { // 组件挂载时获取标签数据 await this.fetchDNTags(); }, methods: { // 获取标签数据 async fetchDNTags() { try { const response = await fetch('DNTag/?format=json'); const data = await response.json(); this.dnTags = data .map(item => item.dnname) .filter(name => name && name.trim().length > 0); } catch (error) { console.error('获取标签失败:', error); alert('标签数据加载失败,高亮功能不可用'); } }, // 通过ID获取医案数据 async fetchById() { if (!this.fetchId) { alert('請輸入有效的醫案ID'); return; } try { const response = await fetch(MRInfo/${this.fetchId}/?format=json); if (response.ok) { const data = await response.json(); // 填充表单数据 this.$refs.mreditor1.formData.mrname = data.mrname || ''; this.$refs.mreditor1.formData.mrposter = data.mrposter || ''; this.content = data.mrcase || ''; this.submittedId = data.id; this.fetchId = null; // 清空输入框 // 刷新标签数据 await this.fetchDNTags(); alert('醫案數據加載成功!'); } else if (response.status === 404) { alert('未找到該ID的醫案'); } else { throw new Error('獲取醫案失敗'); } } catch (error) { console.error('Error:', error); alert(獲取醫案失敗: ${error.message}); } }, // 提交医案 async submitContent() { const formData = this.$refs.mreditor1.getFormData(); const postData = { mrcase: this.content, ...formData }; try { const response = await fetch('MRInfo/?format=json', { method: 'POST', headers: { 'Content-Type': 'application/json', }, body: JSON.stringify(postData), }); if(response.ok) { const data = await response.json(); this.submittedId = data.id; alert('醫案提交成功!'); } else { throw new Error('提交失败'); } } catch (error) { console.error('Error:', error); alert(提交失败: ${error.message}); } }, // 更新医案 async updateContent() { if (!this.submittedId) return; const formData = this.$refs.mreditor1.getFormData(); const postData = { mrcase: this.content, ...formData }; try { const response = await fetch(MRInfo/${this.submittedId}/?format=json, { method: 'PUT', headers: { 'Content-Type': 'application/json', }, body: JSON.stringify(postData), }); if (response.ok) { alert('醫案更新成功!'); } else { throw new Error('更新失败'); } } catch (error) { console.error('Error:', error); alert(更新失败: ${error.message}); } }, // 删除医案 async deleteContent() { if (!this.submittedId) return; if (!confirm('確定要刪除這個醫案嗎?')) return; try { const response = await fetch(MRInfo/${this.submittedId}/?format=json, { method: 'DELETE', headers: { 'Content-Type': 'application/json', } }); if (response.ok) { this.resetAll(); alert('醫案刪除成功!'); } else { throw new Error('刪除失败'); } } catch (error) { console.error('Error:', error); alert(刪除失败: ${error.message}); } }, // 重置表单 resetAll() { this.content = ''; this.submittedId = null; this.fetchId = null; this.$refs.mreditor1.resetForm(); }, // 切换内容显示 toggleContent() { this.isVisible = !this.isVisible; } } }; // 辅助函数:转义正则特殊字符 function escapeRegExp(string) { return string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'); } </script> <style scoped> /* 原有样式保持不变 */ .wangeditor { flex: 1; padding: 10px; overflow-y: auto; } .right-panel { position: fixed; top: 56px; bottom: 45px; right: 0; width: 30%; background: white; padding: 10px; z-index: 100; overflow-y: auto; } .content-display { position: fixed; top: 570px; left: 0; width: 70%; bottom: 45px; z-index: 999; background-color: white; overflow-y: auto; padding: 10px; border: 1px solid #eee; } .sticky-footer { display: flex; justify-content: flex-end; align-items: center; position: fixed; bottom: 0; left: 0; width: 100%; background-color: #ffd800ff; z-index: 999; padding: 10px 20px; box-sizing: border-box; flex-wrap: wrap; } .sticky-footer > span { margin-left: 5px; display: flex; align-items: center; } .submitted-id { padding: 2px; background-color: #e2f0fd; color: #004085; border-radius: 4px; } .reset-btn { margin-left: 10px; padding: 2px; background-color: #dc3545; color: white; border: none; border-radius: 4px; cursor: pointer; } .reset-btn:hover { background-color: #c82333; } /* 新增ID输入框样式 */ .id-input { display: flex; align-items: center; } .id-input input { width: 100px; padding: 2px 5px; margin-right: 5px; border: 1px solid #ccc; border-radius: 4px; } /* 操作按钮样式 */ .submit-btn { background-color: #28a745; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .update-btn { background-color: #007bff; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .delete-btn { background-color: #dc3545; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .submit-btn:hover { background-color: #218838; } .update-btn:hover { background-color: #0069d9; } .delete-btn:hover { background-color: #c82333; } </style>

以下代碼將顯示/隱藏按鈕改成切換按鈕,點按后將content-display原內容改成顯示html標籤格式,再點按時可切換成與原來相同的顯示格式,以完整代碼展示: <template> <WangEditor v-model="content" @response="(msg) => content = msg" /> <mreditor1 ref="mreditor1" />
<button @click="toggleContent">顯示/隱藏標籤</button> <button @click="resetAll" class="reset-btn">輸入新醫案</button> <input v-model="fetchId" placeholder="輸入醫案ID" type="number" min="1" /> <button @click="fetchById">獲取醫案</button> <button @click="updateContent" class="update-btn">更新醫案</button> <button @click="deleteContent" class="delete-btn">刪除醫案</button> <button @click="submitContent" class="submit-btn">提交醫案</button> 醫案 ID: {{ submittedId }} 醫案編輯器
</template> <script> import WangEditor from './WangEditor.vue'; import mreditor1 from './mreditor1.vue'; export default { components: { WangEditor, mreditor1 }, data() { return { content: '', isVisible: true, submittedId: null, fetchId: null, dnTags: [] // 存储从API获取的标签数据 }; }, computed: { // 计算高亮后的内容 highlightedContent() { if (!this.dnTags.length || !this.content) return this.content; // 创建临时DOM元素处理高亮 const tempEl = document.createElement('div'); tempEl.innerHTML = this.content; // 遍历所有文本节点 const walker = document.createTreeWalker( tempEl, NodeFilter.SHOW_TEXT ); const nodes = []; while (walker.nextNode()) { nodes.push(walker.currentNode); } // 对每个文本节点进行处理 nodes.forEach(node => { let text = node.nodeValue; let newHtml = text; // 对每个标签进行高亮处理(按长度降序排序,避免短标签优先匹配) this.dnTags .slice() .sort((a, b) => b.length - a.length) .forEach(tag => { const regex = new RegExp( escapeRegExp(tag), 'g' ); newHtml = newHtml.replace( regex, ${tag} ); }); // 如果内容有变化则替换节点 if (newHtml !== text) { const span = document.createElement('span'); span.innerHTML = newHtml; node.parentNode.replaceChild(span, node); } }); return tempEl.innerHTML; } }, async mounted() { // 组件挂载时获取标签数据 await this.fetchDNTags(); }, methods: { // 获取标签数据 async fetchDNTags() { try { const response = await fetch('DNTag/?format=json'); const data = await response.json(); this.dnTags = data .map(item => item.dnname) .filter(name => name && name.trim().length > 0); } catch (error) { console.error('获取标签失败:', error); alert('标签数据加载失败,高亮功能不可用'); } }, // 通过ID获取医案数据 async fetchById() { if (!this.fetchId) { alert('請輸入有效的醫案ID'); return; } try { const response = await fetch(MRInfo/${this.fetchId}/?format=json); if (response.ok) { const data = await response.json(); // 填充表单数据 this.$refs.mreditor1.formData.mrname = data.mrname || ''; this.$refs.mreditor1.formData.mrposter = data.mrposter || ''; this.content = data.mrcase || ''; this.submittedId = data.id; this.fetchId = null; // 清空输入框 // 刷新标签数据 await this.fetchDNTags(); alert('醫案數據加載成功!'); } else if (response.status === 404) { alert('未找到該ID的醫案'); } else { throw new Error('獲取醫案失敗'); } } catch (error) { console.error('Error:', error); alert(獲取醫案失敗: ${error.message}); } }, // 提交医案 async submitContent() { const formData = this.$refs.mreditor1.getFormData(); const postData = { mrcase: this.content, ...formData }; try { const response = await fetch('MRInfo/?format=json', { method: 'POST', headers: { 'Content-Type': 'application/json', }, body: JSON.stringify(postData), }); if(response.ok) { const data = await response.json(); this.submittedId = data.id; alert('醫案提交成功!'); } else { throw new Error('提交失败'); } } catch (error) { console.error('Error:', error); alert(提交失败: ${error.message}); } }, // 更新医案 async updateContent() { if (!this.submittedId) return; const formData = this.$refs.mreditor1.getFormData(); const postData = { mrcase: this.content, ...formData }; try { const response = await fetch(MRInfo/${this.submittedId}/?format=json, { method: 'PUT', headers: { 'Content-Type': 'application/json', }, body: JSON.stringify(postData), }); if (response.ok) { alert('醫案更新成功!'); } else { throw new Error('更新失败'); } } catch (error) { console.error('Error:', error); alert(更新失败: ${error.message}); } }, // 删除医案 async deleteContent() { if (!this.submittedId) return; if (!confirm('確定要刪除這個醫案嗎?')) return; try { const response = await fetch(MRInfo/${this.submittedId}/?format=json, { method: 'DELETE', headers: { 'Content-Type': 'application/json', } }); if (response.ok) { this.resetAll(); alert('醫案刪除成功!'); } else { throw new Error('刪除失败'); } } catch (error) { console.error('Error:', error); alert(刪除失败: ${error.message}); } }, // 重置表单 resetAll() { this.content = ''; this.submittedId = null; this.fetchId = null; this.$refs.mreditor1.resetForm(); }, // 切换内容显示 toggleContent() { this.isVisible = !this.isVisible; } } }; // 辅助函数:转义正则特殊字符 function escapeRegExp(string) { return string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'); } </script> <style scoped> /* 原有样式保持不变 */ .wangeditor { flex: 1; padding: 10px; overflow-y: auto; } .right-panel { position: fixed; top: 56px; bottom: 45px; right: 0; width: 30%; background: white; padding: 10px; z-index: 100; overflow-y: auto; } .content-display { position: fixed; top: 490px; left: 0; width: 70%; bottom: 45px; z-index: 999; background-color: white; overflow-y: auto; padding: 10px; border: 1px solid #eee; } .sticky-footer { display: flex; justify-content: flex-end; align-items: center; position: fixed; bottom: 0; left: 0; width: 100%; background-color: #ffd800ff; z-index: 999; padding: 10px 20px; box-sizing: border-box; flex-wrap: wrap; } .sticky-footer > span { margin-left: 5px; display: flex; align-items: center; } .submitted-id { padding: 2px; background-color: #e2f0fd; color: #004085; border-radius: 4px; } .reset-btn { margin-left: 10px; padding: 2px; background-color: #dc3545; color: white; border: none; border-radius: 4px; cursor: pointer; } .reset-btn:hover { background-color: #c82333; } /* 新增ID输入框样式 */ .id-input { display: flex; align-items: center; } .id-input input { width: 100px; padding: 2px 5px; margin-right: 5px; border: 1px solid #ccc; border-radius: 4px; } /* 操作按钮样式 */ .submit-btn { background-color: #28a745; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .update-btn { background-color: #007bff; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .delete-btn { background-color: #dc3545; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .submit-btn:hover { background-color: #218838; } .update-btn:hover { background-color: #0069d9; } .delete-btn:hover { background-color: #c82333; } </style>

以下修改mreditor.vue后展示完整代碼,依mrviewer.vue代碼為範例,在right-panel區塊醫案提交者的下方,顯示病態名稱的功能及樣式: mrviewer.vue範例代碼:<template> <button @click="fetchData" class="refresh-btn">刷新數據</button> <input v-model="searchQuery" placeholder="搜索..." class="search-input" /> 每頁顯示: <select v-model.number="pageSize" class="page-size-select"> <option value="1">1筆</option> <option value="4">4筆</option> <option value="10">10筆</option> </select> <button @click="prevPage" :disabled="currentPage === 1">上一页</button> <input type="number" v-model.number="inputPage" min="1" :max="totalPages" class="page-input" @input="handlePageInput"> 頁 / 共 {{ totalPages }} 頁 <button @click="nextPage" :disabled="currentPage === totalPages">下一頁</button> 醫案閱讀器 0"> 醫案 #{{ (currentPage - 1) * pageSize + index + 1 }} {{ key }}: {{ formatDntagValue(value) }} 沒有找到匹配的數據 </template> <script> export default { name: 'mrviewer', // 必须设置组件名称用于keep-alive data() { return { api1Data: [], api2Data: [], mergedData: [], currentPage: 1, pageSize: 1, searchQuery: '', sortKey: '', sortOrders: {}, inputPage: 1, // 页码输入框绑定的值 // 字段名称映射表 fieldNames: { 'mrcase': '醫案全文', 'mrname': '醫案命名', 'mrposter': '醫案提交者', 'mrlasttime': '最後編輯時間', 'mreditnumber': '編輯次數', 'mrreadnumber': '閱讀次數', 'mrpriority': '重要性', 'dntag': '病態名稱' }, inputTimeout: null, // 用于输入防抖 dnNames: [], // 存储所有病态名称 stateVersion: '1.0' // 状态版本控制 }; }, computed: { filteredData() { const query = this.searchQuery.trim(); // 判斷是否為數字(即搜尋 ID) if (query && /^\d+$/.test(query)) { const idToSearch = parseInt(query, 10); return this.mergedData.filter(item => item.id === idToSearch); } // 一般搜索邏輯 if (!query) return this.mergedData; const lowerQuery = query.toLowerCase(); return this.mergedData.filter(item => { return Object.values(item).some(value => { if (value === null || value === undefined) return false; // 处理数组类型的值 if (Array.isArray(value)) { return value.some(subValue => { if (typeof subValue === 'object' && subValue !== null) { return JSON.stringify(subValue).toLowerCase().includes(lowerQuery); } return String(subValue).toLowerCase().includes(lowerQuery); }); } // 处理对象类型的值 if (typeof value === 'object' && value !== null) { return JSON.stringify(value).toLowerCase().includes(lowerQuery); } return String(value).toLowerCase().includes(lowerQuery); }); }); }, sortedData() { if (!this.sortKey) return this.filteredData; const order = this.sortOrders[this.sortKey] || 1; return [...this.filteredData].sort((a, b) => { const getValue = (obj) => { const val = obj[this.sortKey]; if (Array.isArray(val)) { return JSON.stringify(val); } return val; }; const aValue = getValue(a); const bValue = getValue(b); if (aValue === bValue) return 0; return aValue > bValue ? order : -order; }); }, paginatedData() { const start = (this.currentPage - 1) * Number(this.pageSize); const end = start + Number(this.pageSize); return this.sortedData.slice(start, end); }, totalPages() { return Math.ceil(this.filteredData.length / this.pageSize) || 1; } }, watch: { // 监控分页大小变化 pageSize() { // 重置到第一页 this.currentPage = 1; this.inputPage = 1; // 同步输入框的值 this.saveState(); // 保存状态 }, // 监控当前页码变化 currentPage(newVal) { // 同步输入框的值 this.inputPage = newVal; this.saveState(); // 保存状态 }, // 监控过滤数据变化 filteredData() { // 确保当前页码有效 if (this.currentPage > this.totalPages) { this.currentPage = Math.max(1, this.totalPages); } // 同步输入框的值 this.inputPage = this.currentPage; }, // 监控搜索查询变化 searchQuery() { this.saveState(); // 保存状态 } }, methods: { // 保存当前状态 saveState() { const state = { version: this.stateVersion, currentPage: this.currentPage, pageSize: this.pageSize, searchQuery: this.searchQuery, timestamp: new Date().getTime() }; sessionStorage.setItem('mrviewerState', JSON.stringify(state)); }, // 恢复状态 restoreState() { const savedState = sessionStorage.getItem('mrviewerState'); if (!savedState) return; try { const state = JSON.parse(savedState); // 检查状态版本 if (state.version !== this.stateVersion) { console.warn('状态版本不匹配,跳过恢复'); return; } // 恢复状态 this.currentPage = state.currentPage || 1; this.pageSize = state.pageSize || 1; this.searchQuery = state.searchQuery || ''; this.inputPage = this.currentPage; console.log('医案阅读器状态已恢复'); } catch (e) { console.error('状态恢复失败', e); sessionStorage.removeItem('mrviewerState'); } }, // 清除保存的状态 clearState() { sessionStorage.removeItem('mrviewerState'); }, async fetchData() { try { const api1Response = await fetch("MRInfo/?format=json"); this.api1Data = await api1Response.json(); const api2Response = await fetch("DNTag/?format=json"); this.api2Data = await api2Response.json(); // 提取所有dnname this.dnNames = this.api2Data.map(item => item.dnname).filter(name => name && name.trim()); // 按长度降序排序(确保长字符串优先匹配) this.dnNames.sort((a, b) => b.length - a.length); this.mergeData(); this.currentPage = 1; this.inputPage = 1; // 重置输入框 this.saveState(); // 保存状态 } catch (error) { console.error("獲取數據失敗:", error); alert("數據加載失敗,請稍後重試"); } }, mergeData() { this.mergedData = this.api1Data.map((item) => { const newItem = { ...item }; if (newItem.dntag && Array.isArray(newItem.dntag)) { newItem.dntag = newItem.dntag.map((tagId) => { const matchedItem = this.api2Data.find( (api2Item) => api2Item.id === tagId ); return matchedItem || { id: tagId, dnname: "未找到匹配的數據" }; }); } return newItem; }); this.sortOrders = {}; if (this.mergedData.length > 0) { Object.keys(this.mergedData[0]).forEach(key => { this.sortOrders[key] = 1; }); } }, // 处理字段名称映射 processFieldNames(item) { const result = {}; for (const key in item) { // 使用映射表转换字段名,如果没有映射则使用原字段名 const newKey = this.fieldNames[key] || key; result[newKey] = item[key]; } return result; }, // 高亮匹配文本的方法 highlightMatches(text) { if (!text || typeof text !== 'string' || this.dnNames.length === 0) { return text; } // 创建正则表达式(转义特殊字符) const pattern = new RegExp( this.dnNames .map(name => name.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')) .join('|'), 'gi' ); // 替换匹配文本 return text.replace(pattern, match => ${match} ); }, formatValue(value, fieldName) { if (value === null || value === undefined) return ''; // 医案全文字段特殊处理 if (fieldName === '醫案全文' && typeof value === 'string') { return this.highlightMatches(value); } // 其他字段保持原逻辑 if (typeof value === 'string' && value.startsWith('http')) { return ${value}; } return value; }, // 专门处理dntag字段的显示格式 formatDntagValue(dntagArray) { return dntagArray.map(tagObj => { // 只显示name属性,隐藏id等其他属性 return tagObj.dnname || tagObj.name || '未命名標籤'; }).join(';'); // 使用大写分号分隔 }, sortBy(key) { // 需要从中文名称映射回原始字段名进行排序 const originalKey = Object.keys(this.fieldNames).find( origKey => this.fieldNames[origKey] === key ) || key; this.sortKey = originalKey; this.sortOrders[originalKey] = this.sortOrders[originalKey] * -1; this.saveState(); // 保存状态 }, prevPage() { if (this.currentPage > 1) { this.currentPage--; this.saveState(); // 保存状态 } }, nextPage() { if (this.currentPage < this.totalPages) { this.currentPage++; this.saveState(); // 保存状态 } }, // 实时处理页码输入 handlePageInput() { // 清除之前的定时器 clearTimeout(this.inputTimeout); // 设置新的定时器(防抖处理) this.inputTimeout = setTimeout(() => { this.goToPage(); this.saveState(); // 保存状态 }, 300); // 300毫秒后执行 }, // 跳转到指定页码 goToPage() { // 处理空值情况 if (this.inputPage === null || this.inputPage === undefined || this.inputPage === '') { this.inputPage = this.currentPage; return; } // 转换为整数 const page = parseInt(this.inputPage); // 处理非数字情况 if (isNaN(page)) { this.inputPage = this.currentPage; return; } // 确保页码在有效范围内 if (page < 1) { this.currentPage = 1; } else if (page > this.totalPages) { this.currentPage = this.totalPages; } else { this.currentPage = page; } // 同步输入框显示 this.inputPage = this.currentPage; } }, mounted() { this.restoreState(); // 恢复状态 this.fetchData(); }, // 添加keep-alive生命周期钩子 activated() { console.log('醫案閱讀器被激活'); this.restoreState(); // 恢复状态 }, deactivated() { console.log('醫案閱讀器被停用'); this.saveState(); // 保存状态 } }; </script> <style scoped> .container { max-width: 1200px; margin: 0px; padding: 0px; } .control-panel { margin-bottom: 0px; display: flex; flex-wrap: wrap; gap: 10px; justify-content: flex-end; align-items: center; position: fixed; bottom: 0; left: 0; width: 100%; background-color: #ffd800ff; z-index: 999; padding: 10px 20px; box-sizing: border-box; } .content-area { position: fixed; top: 56px; bottom: 45px; /* 位于底部按钮上方 */ left: 0; width: 100%; background: white; padding: 1px; z-index: 100; overflow-y: auto; } .refresh-btn { padding: 4px; background-color: #4CAF50; color: white; border: none; border-radius: 4px; cursor: pointer; } .refresh-btn:hover { background-color: #45a049; } .search-input { padding: 8px; border: 1px solid #ddd; border-radius: 4px; flex-grow: 1; max-width: 300px; } .pagination-controls { display: flex; align-items: center; gap: 5px; } .page-size-select { padding: 4px; border-radius: 4px; width: 70px; } /* 页码输入框样式 */ .page-input { width: 50px; padding: 4px; border: 1px solid #ddd; border-radius: 4px; text-align: center; } /* 水平记录样式 */ .horizontal-records { display: flex; flex-direction: column; gap: 20px; } .record-card { border: 1px solid #ddd; border-radius: 4px; overflow: hidden; box-shadow: 0 2px 4px rgba(0,0,0,0.1); } .record-header { padding: 12px 16px; background-color: #f5f5f5; border-bottom: 1px solid #ddd; } .record-header h3 { margin: 0; font-size: 1.1em; } .record-body { padding: 16px; } .record-field { display: flex; margin-bottom: 12px; line-height: 1.5; } .record-field:last-child { margin-bottom: 0; } .field-name { font-weight: bold; min-width: 120px; color: #555; } .field-value { flex-grow: 1; display: flex; flex-wrap: wrap; gap: 8px; } .dntag-value { display: flex; flex-wrap: wrap; gap: 8px; } .array-value { display: flex; flex-wrap: wrap; gap: 8px; } .no-data { padding: 20px; text-align: center; color: #666; font-style: italic; } button:disabled { opacity: 0.5; cursor: not-allowed; } </style> mreditor.vue代碼: <template> <WangEditor v-model="content" @response="(msg) => content = msg" /> <mreditor1 ref="mreditor1" />
<button @click="toggleDisplayFormat">{{ displayButtonText }}</button> <button @click="resetAll" class="reset-btn">輸入新醫案</button> <input v-model="fetchId" placeholder="輸入醫案ID" type="number" min="1" /> <button @click="fetchById">載入醫案</button> <button @click="updateContent" class="update-btn">更新醫案</button> <button @click="deleteContent" class="delete-btn">刪除醫案</button> <button @click="submitContent" class="submit-btn">提交醫案</button> 醫案 ID: {{ submittedId }} 醫案編輯器
</template> <script> import WangEditor from './WangEditor.vue'; import mreditor1 from './mreditor1.vue'; import LZString from 'lz-string'; export default { name: 'mreditor', // 必须设置组件名称用于keep-alive components: { WangEditor, mreditor1 }, data() { return { content: '', submittedId: null, fetchId: null, dnTags: [], showRawHtml: false, stateVersion: '1.0', autoSaveTimer: null, autoSaveInterval: 30000 // 30秒自动保存 }; }, computed: { highlightedContent() { if (!this.dnTags.length || !this.content) return this.content; const tempEl = document.createElement('div'); tempEl.innerHTML = this.content; const walker = document.createTreeWalker( tempEl, NodeFilter.SHOW_TEXT ); const nodes = []; while (walker.nextNode()) { nodes.push(walker.currentNode); } nodes.forEach(node => { let text = node.nodeValue; let newHtml = text; this.dnTags .slice() .sort((a, b) => b.length - a.length) .forEach(tag => { const regex = new RegExp( this.escapeRegExp(tag), 'g' ); newHtml = newHtml.replace( regex, ${tag} ); }); if (newHtml !== text) { const span = document.createElement('span'); span.innerHTML = newHtml; node.parentNode.replaceChild(span, node); } }); return tempEl.innerHTML; }, displayContent() { if (this.showRawHtml) { return this.escapeHtml(this.content); } return this.highlightedContent; }, displayButtonText() { return this.showRawHtml ? '顯示渲染格式' : '顯示原始標籤'; } }, watch: { // 监听内容变化实现自动保存 content: { handler() { this.debouncedSaveState(); }, deep: true } }, async mounted() { await this.fetchDNTags(); this.restoreState(); this.setupAutoSave(); }, // 添加keep-alive生命周期钩子 activated() { console.log('醫案編輯器被激活'); this.restoreState(); this.setupAutoSave(); }, deactivated() { console.log('醫案編輯器被停用'); this.saveState(); // 离开时保存状态 this.clearAutoSave(); }, beforeDestroy() { this.clearAutoSave(); }, methods: { // HTML转义 escapeHtml(unsafe) { if (!unsafe) return ''; return unsafe .replace(/&/g, "&") .replace(/</g, "<") .replace(/>/g, ">") .replace(/"/g, """) .replace(/'/g, "'"); }, // 正则表达式转义 escapeRegExp(string) { return string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'); }, // 设置自动保存 setupAutoSave() { this.clearAutoSave(); this.autoSaveTimer = setInterval(() => { if (this.content || this.submittedId) { console.log('自动保存状态...'); this.saveState(); } }, this.autoSaveInterval); }, // 清除自动保存 clearAutoSave() { if (this.autoSaveTimer) { clearInterval(this.autoSaveTimer); this.autoSaveTimer = null; } }, // 防抖保存状态 debouncedSaveState() { this.clearAutoSave(); this.autoSaveTimer = setTimeout(() => { this.saveState(); this.setupAutoSave(); }, 2000); }, // 保存当前状态 saveState() { // 获取子组件状态 const formData = this.$refs.mreditor1 ? this.$refs.mreditor1.getFormData() : { }; const state = { version: this.stateVersion, content: this.content, submittedId: this.submittedId, formData: formData, showRawHtml: this.showRawHtml, timestamp: new Date().getTime() }; // 对大内容进行压缩 if (this.content.length > 2000) { try { state.compressed = true; state.content = LZString.compressToUTF16(this.content); } catch (e) { console.error('压缩失败,使用原始数据', e); state.compressed = false; } } sessionStorage.setItem('medicalRecordState', JSON.stringify(state)); }, // 恢复状态 restoreState() { const savedState = sessionStorage.getItem('medicalRecordState'); if (!savedState) return; try { const state = JSON.parse(savedState); if (state.version !== this.stateVersion) { console.warn('状态版本不匹配,跳过恢复'); return; } let content = state.content; if (state.compressed) { try { content = LZString.decompressFromUTF16(content); } catch (e) { console.error('解压失败,使用原始数据', e); } } this.content = content; this.submittedId = state.submittedId; this.showRawHtml = state.showRawHtml; if (this.$refs.mreditor1 && state.formData) { this.$refs.mreditor1.setFormData(state.formData); } console.log('医案状态已恢复'); } catch (e) { console.error('状态恢复失败', e); sessionStorage.removeItem('medicalRecordState'); } }, // 清除保存的状态 clearState() { sessionStorage.removeItem('medicalRecordState'); }, // 切换显示格式 toggleDisplayFormat() { this.showRawHtml = !this.showRawHtml; this.saveState(); }, async fetchDNTags() { try { const response = await fetch('DNTag/?format=json'); const data = await response.json(); this.dnTags = data .map(item => item.dnname) .filter(name => name && name.trim().length > 0); } catch (error) { console.error('获取标签失败:', error); alert('标签数据加载失败,高亮功能不可用'); } }, async fetchById() { if (!this.fetchId) { alert('請輸入有效的醫案ID'); return; } try { const response = await fetch(MRInfo/${this.fetchId}/?format=json); if (response.ok) { const data = await response.json(); if (this.$refs.mreditor1) { this.$refs.mreditor1.formData.mrname = data.mrname || ''; this.$refs.mreditor1.formData.mrposter = data.mrposter || ''; } this.content = data.mrcase || ''; this.submittedId = data.id; this.fetchId = null; await this.fetchDNTags(); this.saveState(); alert('醫案數據加載成功!'); } else if (response.status === 404) { alert('未找到該ID的醫案'); } else { throw new Error('獲取醫案失敗'); } } catch (error) { console.error('Error:', error); alert(獲取醫案失敗: ${error.message}); } }, async submitContent() { if (!this.$refs.mreditor1) return; const formData = this.$refs.mreditor1.getFormData(); const postData = { mrcase: this.content, ...formData }; try { const response = await fetch('MRInfo/?format=json', { method: 'POST', headers: { 'Content-Type': 'application/json', }, body: JSON.stringify(postData), }); if (response.ok) { const data = await response.json(); this.submittedId = data.id; this.saveState(); alert('醫案提交成功!'); } else { throw new Error('提交失败'); } } catch (error) { console.error('Error:', error); alert(提交失败: ${error.message}); } }, async updateContent() { if (!this.submittedId || !this.$refs.mreditor1) return; const formData = this.$refs.mreditor1.getFormData(); const postData = { mrcase: this.content, ...formData }; try { const response = await fetch(MRInfo/${this.submittedId}/?format=json, { method: 'PUT', headers: { 'Content-Type': 'application/json', }, body: JSON.stringify(postData), }); if (response.ok) { this.saveState(); alert('醫案更新成功!'); } else { throw new Error('更新失败'); } } catch (error) { console.error('Error:', error); alert(更新失败: ${error.message}); } }, async deleteContent() { if (!this.submittedId) return; if (!confirm('確定要刪除這個醫案嗎?')) return; try { const response = await fetch(MRInfo/${this.submittedId}/?format=json, { method: 'DELETE', headers: { 'Content-Type': 'application/json', } }); if (response.ok) { this.clearState(); this.resetAll(); alert('醫案刪除成功!'); } else { throw new Error('刪除失败'); } } catch (error) { console.error('Error:', error); alert(刪除失败: ${error.message}); } }, resetAll() { this.content = ''; this.submittedId = null; this.fetchId = null; this.showRawHtml = false; if (this.$refs.mreditor1) { this.$refs.mreditor1.resetForm(); } this.clearState(); } } }; </script> <style scoped> .wangeditor { flex: 1; padding: 10px; overflow-y: auto; } .right-panel { position: fixed; top: 55px; bottom: 480px; right: 0; width: 30%; background: lightblue; padding: 1px; z-index: 100; overflow-y: auto; } .tag-panel { position: fixed; top: 260px; bottom: 300px; right: 0; width: 30%; background: lightyellow; padding: 10px; z-index: 100; overflow-y: auto; } .content-display { position: fixed; top: 490px; left: 0; width: 70%; bottom: 45px; z-index: 999; background-color: white; overflow-y: auto; padding: 10px; border: 1px solid #eee; white-space: pre-wrap; } .sticky-footer { display: flex; justify-content: flex-end; align-items: center; position: fixed; bottom: 0; left: 0; width: 100%; background-color: #ffd800ff; z-index: 999; padding: 10px 20px; box-sizing: border-box; flex-wrap: wrap; } .sticky-footer > span { margin-left: 5px; display: flex; align-items: center; } .submitted-id { padding: 2px; background-color: #e2f0fd; color: #004085; border-radius: 4px; } .reset-btn { margin-left: 10px; padding: 2px; background-color: #dc3545; color: white; border: none; border-radius: 4px; cursor: pointer; } .reset-btn:hover { background-color: #c82333; } .id-input { display: flex; align-items: center; } .id-input input { width: 100px; padding: 2px 5px; margin-right: 5px; border: 1px solid #ccc; border-radius: 4px; } .submit-btn { background-color: #28a745; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .update-btn { background-color: #007bff; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .delete-btn { background-color: #dc3545; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .submit-btn:hover { background-color: #218838; } .update-btn:hover { background-color: #0069d9; } .delete-btn:hover { background-color: #c82333; } </style>

以下代碼頁面獲得json資料時,將字段中mrcase內的字串與,另一個位於DNTag/?format=json的drf api中字段為dnname內的字串比對,mrcase欄位中與dnname欄位相同的字用rgb(212, 107, 8)之顏色標示,然後在頁面content中展示: <template> <WangEditor v-model="content" @response="(msg) => content = msg" /> <mreditor1 ref="mreditor1" /> {{ content }}
<button @click="toggleContent">顯示/隱藏標籤</button> <button @click="resetAll" class="reset-btn">輸入新醫案</button> <input v-model="fetchId" placeholder="輸入醫案ID" type="number" min="1" /> <button @click="fetchById">獲取醫案</button> <button @click="updateContent" class="update-btn">更新醫案</button> <button @click="deleteContent" class="delete-btn">刪除醫案</button> <button @click="submitContent" class="submit-btn">提交醫案</button> 醫案 ID: {{ submittedId }} 醫案編輯器
</template> <script> import WangEditor from './WangEditor.vue'; import mreditor1 from './mreditor1.vue'; export default { components: { WangEditor, mreditor1 }, data() { return { content: '', isVisible: true, submittedId: null, fetchId: null // 新增:用于存储要获取的ID }; }, methods: { async submitContent() { const formData = this.$refs.mreditor1.getFormData(); const postData = { mrcase: this.content, ...formData }; try { const response = await fetch('MRInfo/?format=json', { method: 'POST', headers: { 'Content-Type': 'application/json', }, body: JSON.stringify(postData), }); if(response.ok) { const data = await response.json(); this.submittedId = data.id; alert('醫案提交成功!'); } else { throw new Error('提交失败'); } } catch (error) { console.error('Error:', error); alert(提交失败: ${error.message}); } }, async updateContent() { if (!this.submittedId) return; const formData = this.$refs.mreditor1.getFormData(); const postData = { mrcase: this.content, ...formData }; try { const response = await fetch(MRInfo/${this.submittedId}/?format=json, { method: 'PUT', headers: { 'Content-Type': 'application/json', }, body: JSON.stringify(postData), }); if (response.ok) { alert('醫案更新成功!'); } else { throw new Error('更新失败'); } } catch (error) { console.error('Error:', error); alert(更新失败: ${error.message}); } }, async deleteContent() { if (!this.submittedId) return; if (!confirm('確定要刪除這個醫案嗎?')) return; try { const response = await fetch(MRInfo/${this.submittedId}/?format=json, { method: 'DELETE', headers: { 'Content-Type': 'application/json', } }); if (response.ok) { this.resetAll(); alert('醫案刪除成功!'); } else { throw new Error('刪除失败'); } } catch (error) { console.error('Error:', error); alert(刪除失败: ${error.message}); } }, // 新增:通过ID获取医案数据 async fetchById() { if (!this.fetchId) { alert('請輸入有效的醫案ID'); return; } try { const response = await fetch(MRInfo/${this.fetchId}/?format=json); if (response.ok) { const data = await response.json(); // 填充表单数据 this.$refs.mreditor1.formData.mrname = data.mrname || ''; this.$refs.mreditor1.formData.mrposter = data.mrposter || ''; this.content = data.mrcase || ''; this.submittedId = data.id; this.fetchId = null; // 清空输入框 alert('醫案數據加載成功!'); } else if (response.status === 404) { alert('未找到該ID的醫案'); } else { throw new Error('獲取醫案失敗'); } } catch (error) { console.error('Error:', error); alert(獲取醫案失敗: ${error.message}); } }, resetAll() { this.content = ''; this.submittedId = null; this.fetchId = null; this.$refs.mreditor1.resetForm(); }, toggleContent() { this.isVisible = !this.isVisible; } } }; </script> <style scoped> /* 原有样式保持不变 */ .wangeditor { flex: 1; padding: 10px; overflow-y: auto; } .right-panel { position: fixed; top: 56px; bottom: 45px; right: 0; width: 30%; background: white; padding: 10px; z-index: 100; overflow-y: auto; } .content-display { position: fixed; top: 570px; left: 0; width: 70%; bottom: 45px; z-index: 999; background-color: white; overflow-y: auto; } .sticky-footer { display: flex; justify-content: flex-end; align-items: center; position: fixed; bottom: 0; left: 0; width: 100%; background-color: #ffd800ff; z-index: 999; padding: 10px 20px; box-sizing: border-box; flex-wrap: wrap; } .sticky-footer > span { margin-left: 5px; display: flex; align-items: center; } .submitted-id { padding: 2px; background-color: #e2f0fd; color: #004085; border-radius: 4px; } .reset-btn { margin-left: 10px; padding: 2px; background-color: #dc3545; color: white; border: none; border-radius: 4px; cursor: pointer; } .reset-btn:hover { background-color: #c82333; } /* 新增ID输入框样式 */ .id-input { display: flex; align-items: center; } .id-input input { width: 100px; padding: 2px 5px; margin-right: 5px; border: 1px solid #ccc; border-radius: 4px; } /* 操作按钮样式 */ .submit-btn { background-color: #28a745; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .update-btn { background-color: #007bff; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .delete-btn { background-color: #dc3545; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .submit-btn:hover { background-color: #218838; } .update-btn:hover { background-color: #0069d9; } .delete-btn:hover { background-color: #c82333; } </style>

以上子組件如下,請以完整代碼展示更改: <template> <WangEditor v-model="content" @response="(msg) => content = msg" /> <mreditor1 ref="mreditor1" />
<button @click="toggleDisplayFormat">{{ displayButtonText }}</button> <button @click="resetAll" class="reset-btn">輸入新醫案</button> <input v-model="fetchId" placeholder="輸入醫案ID" type="number" min="1" /> <button @click="fetchById">載入醫案</button> <button @click="updateContent" class="update-btn">更新醫案</button> <button @click="deleteContent" class="delete-btn">刪除醫案</button> <button @click="submitContent" class="submit-btn">提交醫案</button> 醫案 ID: {{ submittedId }} 醫案編輯器
</template> <script> import WangEditor from './WangEditor.vue'; import mreditor1 from './mreditor1.vue'; import LZString from 'lz-string'; // 引入压缩库 export default { components: { WangEditor, mreditor1 }, data() { return { content: '', submittedId: null, fetchId: null, dnTags: [], showRawHtml: false, // 状态版本控制 stateVersion: '1.0' }; }, computed: { // 计算高亮后的内容 highlightedContent() { if (!this.dnTags.length || !this.content) return this.content; const tempEl = document.createElement('div'); tempEl.innerHTML = this.content; const walker = document.createTreeWalker( tempEl, NodeFilter.SHOW_TEXT ); const nodes = []; while (walker.nextNode()) { nodes.push(walker.currentNode); } nodes.forEach(node => { let text = node.nodeValue; let newHtml = text; this.dnTags .slice() .sort((a, b) => b.length - a.length) .forEach(tag => { const regex = new RegExp( this.escapeRegExp(tag), 'g' ); newHtml = newHtml.replace( regex, ${tag} ); }); if (newHtml !== text) { const span = document.createElement('span'); span.innerHTML = newHtml; node.parentNode.replaceChild(span, node); } }); return tempEl.innerHTML; }, // 根据状态返回显示内容 displayContent() { if (this.showRawHtml) { // 显示原始HTML标签 return this.escapeHtml(this.content); } // 显示渲染格式(带高亮) return this.highlightedContent; }, // 动态按钮文字 displayButtonText() { return this.showRawHtml ? '顯示渲染格式' : '顯示原始標籤'; } }, async mounted() { await this.fetchDNTags(); this.restoreState(); // 组件挂载时恢复状态 }, methods: { // HTML转义方法 escapeHtml(unsafe) { if (!unsafe) return ''; return unsafe .replace(/&/g, "&") .replace(/</g, "<") .replace(/>/g, ">") .replace(/"/g, """) .replace(/'/g, "'"); }, // 正则表达式转义 escapeRegExp(string) { return string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'); }, // 保存当前状态到sessionStorage saveState() { const state = { version: this.stateVersion, content: this.content, submittedId: this.submittedId, formData: this.$refs.mreditor1.getFormData(), showRawHtml: this.showRawHtml, timestamp: new Date().getTime() }; // 对大内容进行压缩 if (this.content.length > 2000) { try { state.compressed = true; state.content = LZString.compressToUTF16(this.content); } catch (e) { console.error('压缩失败,使用原始数据', e); state.compressed = false; } } sessionStorage.setItem('medicalRecordState', JSON.stringify(state)); }, // 从sessionStorage恢复状态 restoreState() { const savedState = sessionStorage.getItem('medicalRecordState'); if (!savedState) return; try { const state = JSON.parse(savedState); // 检查状态版本 if (state.version !== this.stateVersion) { console.warn('状态版本不匹配,跳过恢复'); return; } // 解压内容 let content = state.content; if (state.compressed) { try { content = LZString.decompressFromUTF16(content); } catch (e) { console.error('解压失败,使用原始数据', e); } } // 恢复状态 this.content = content; this.submittedId = state.submittedId; this.showRawHtml = state.showRawHtml; // 恢复子组件状态 if (this.$refs.mreditor1 && state.formData) { this.$refs.mreditor1.setFormData(state.formData); } console.log('医案状态已恢复'); } catch (e) { console.error('状态恢复失败', e); sessionStorage.removeItem('medicalRecordState'); } }, // 清除保存的状态 clearState() { sessionStorage.removeItem('medicalRecordState'); }, // 切换显示格式 toggleDisplayFormat() { this.showRawHtml = !this.showRawHtml; this.saveState(); // 保存状态变化 }, async fetchDNTags() { try { const response = await fetch('DNTag/?format=json'); const data = await response.json(); this.dnTags = data .map(item => item.dnname) .filter(name => name && name.trim().length > 0); } catch (error) { console.error('获取标签失败:', error); alert('标签数据加载失败,高亮功能不可用'); } }, async fetchById() { if (!this.fetchId) { alert('請輸入有效的醫案ID'); return; } try { const response = await fetch(MRInfo/${this.fetchId}/?format=json); if (response.ok) { const data = await response.json(); this.$refs.mreditor1.formData.mrname = data.mrname || ''; this.$refs.mreditor1.formData.mrposter = data.mrposter || ''; this.content = data.mrcase || ''; this.submittedId = data.id; this.fetchId = null; await this.fetchDNTags(); this.saveState(); // 保存状态 alert('醫案數據加載成功!'); } else if (response.status === 404) { alert('未找到該ID的醫案'); } else { throw new Error('獲取醫案失敗'); } } catch (error) { console.error('Error:', error); alert(獲取醫案失敗: ${error.message}); } }, async submitContent() { const formData = this.$refs.mreditor1.getFormData(); const postData = { mrcase: this.content, ...formData }; try { const response = await fetch('MRInfo/?format=json', { method: 'POST', headers: { 'Content-Type': 'application/json', }, body: JSON.stringify(postData), }); if(response.ok) { const data = await response.json(); this.submittedId = data.id; this.saveState(); // 保存状态 alert('醫案提交成功!'); } else { throw new Error('提交失败'); } } catch (error) { console.error('Error:', error); alert(提交失败: ${error.message}); } }, async updateContent() { if (!this.submittedId) return; const formData = this.$refs.mreditor1.getFormData(); const postData = { mrcase: this.content, ...formData }; try { const response = await fetch(MRInfo/${this.submittedId}/?format=json, { method: 'PUT', headers: { 'Content-Type': 'application/json', }, body: JSON.stringify(postData), }); if (response.ok) { this.saveState(); // 保存状态 alert('醫案更新成功!'); } else { throw new Error('更新失败'); } } catch (error) { console.error('Error:', error); alert(更新失败: ${error.message}); } }, async deleteContent() { if (!this.submittedId) return; if (!confirm('確定要刪除這個醫案嗎?')) return; try { const response = await fetch(MRInfo/${this.submittedId}/?format=json, { method: 'DELETE', headers: { 'Content-Type': 'application/json', } }); if (response.ok) { this.clearState(); // 清除保存的状态 this.resetAll(); alert('醫案刪除成功!'); } else { throw new Error('刪除失败'); } } catch (error) { console.error('Error:', error); alert(刪除失败: ${error.message}); } }, resetAll() { this.content = ''; this.submittedId = null; this.fetchId = null; this.showRawHtml = false; this.$refs.mreditor1.resetForm(); this.clearState(); // 清除保存的状态 } } }; </script> <style scoped> .wangeditor { flex: 1; padding: 10px; overflow-y: auto; } .right-panel { position: fixed; top: 56px; bottom: 45px; right: 0; width: 30%; background: white; padding: 10px; z-index: 100; overflow-y: auto; } .content-display { position: fixed; top: 490px; left: 0; width: 70%; bottom: 45px; z-index: 999; background-color: white; overflow-y: auto; padding: 10px; border: 1px solid #eee; white-space: pre-wrap; } .sticky-footer { display: flex; justify-content: flex-end; align-items: center; position: fixed; bottom: 0; left: 0; width: 100%; background-color: #ffd800ff; z-index: 999; padding: 10px 20px; box-sizing: border-box; flex-wrap: wrap; } .sticky-footer > span { margin-left: 5px; display: flex; align-items: center; } .submitted-id { padding: 2px; background-color: #e2f0fd; color: #004085; border-radius: 4px; } .reset-btn { margin-left: 10px; padding: 2px; background-color: #dc3545; color: white; border: none; border-radius: 4px; cursor: pointer; } .reset-btn:hover { background-color: #c82333; } .id-input { display: flex; align-items: center; } .id-input input { width: 100px; padding: 2px 5px; margin-right: 5px; border: 1px solid #ccc; border-radius: 4px; } .submit-btn { background-color: #28a745; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .update-btn { background-color: #007bff; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .delete-btn { background-color: #dc3545; color: white; border: none; padding: 2px 8px; border-radius: 4px; cursor: pointer; } .submit-btn:hover { background-color: #218838; } .update-btn:hover { background-color: #0069d9; } .delete-btn:hover { background-color: #c82333; } </style>

最新推荐

recommend-type

Qt开发:XML文件读取、滚动区域控件布局与多Sheet Excel保存的界面设计实例

内容概要:本文介绍了基于Qt框架的界面设计例程,重点讲解了三个主要功能模块:一是利用XML文件进行配置信息的读取并初始化界面组件;二是实现了滚动区域内的灵活控件布局,在空间不足时自动生成滚动条以扩展显示范围;三是提供了将界面上的数据导出到带有多个工作表的Excel文件的功能。文中还提及了所用IDE的具体版本(Qt Creator 4.8.0 和 Qt 5.12.0),并且强调了这些技术的实际应用场景及其重要性。 适合人群:对Qt有初步了解,希望深入学习Qt界面设计技巧的开发者。 使用场景及目标:适用于需要快速构建复杂用户界面的应用程序开发,特别是那些涉及大量数据展示和交互的设计任务。通过学习本文提供的案例,可以提高对于Qt框架的理解,掌握更多实用技能。 其他说明:为了帮助读者更好地理解和实践,作者推荐前往B站观看高清的教学视频,以便于更直观地感受整个项目的开发流程和技术细节。
recommend-type

锂电池保护板方案:中颖SH367309原理图与PCB源代码详解及应用技巧

基于中颖SH367309芯片的锂电池保护板设计方案,涵盖原理图解析、PCB布局优化、硬件选型要点以及软件编程技巧。重点讨论了电流检测精度、过压保护阈值设定、通信协议处理和温度传感器布置等方面的实际开发经验和技术难点。文中还分享了一些实用的小贴士,如采用星型接地减少干扰、利用过孔阵列降低温升、为MOS管增加RC缓冲避免高频振荡等。 适合人群:从事锂电池管理系统(BMS)开发的技术人员,尤其是有一定硬件设计基础并希望深入了解具体实现细节的工程师。 使用场景及目标:帮助开发者掌握锂电池保护板的关键技术和常见问题解决方案,确保产品在各种工况下都能安全可靠运行,同时提高系统性能指标如效率、响应速度和稳定性。 阅读建议:由于涉及较多底层硬件知识和实战案例,建议读者结合自身项目背景进行针对性学习,在遇到类似问题时能够快速定位原因并找到有效对策。此外,对于初学者来说,可以从简单的电路搭建开始逐步深入研究复杂的功能模块。
recommend-type

Web前端开发:CSS与HTML设计模式深入解析

《Pro CSS and HTML Design Patterns》是一本专注于Web前端设计模式的书籍,特别针对CSS(层叠样式表)和HTML(超文本标记语言)的高级应用进行了深入探讨。这本书籍属于Pro系列,旨在为专业Web开发人员提供实用的设计模式和实践指南,帮助他们构建高效、美观且可维护的网站和应用程序。 在介绍这本书的知识点之前,我们首先需要了解CSS和HTML的基础知识,以及它们在Web开发中的重要性。 HTML是用于创建网页和Web应用程序的标准标记语言。它允许开发者通过一系列的标签来定义网页的结构和内容,如段落、标题、链接、图片等。HTML5作为最新版本,不仅增强了网页的表现力,还引入了更多新的特性,例如视频和音频的内置支持、绘图API、离线存储等。 CSS是用于描述HTML文档的表现(即布局、颜色、字体等样式)的样式表语言。它能够让开发者将内容的表现从结构中分离出来,使得网页设计更加模块化和易于维护。随着Web技术的发展,CSS也经历了多个版本的更新,引入了如Flexbox、Grid布局、过渡、动画以及Sass和Less等预处理器技术。 现在让我们来详细探讨《Pro CSS and HTML Design Patterns》中可能包含的知识点: 1. CSS基础和选择器: 书中可能会涵盖CSS基本概念,如盒模型、边距、填充、边框、背景和定位等。同时还会介绍CSS选择器的高级用法,例如属性选择器、伪类选择器、伪元素选择器以及选择器的组合使用。 2. CSS布局技术: 布局是网页设计中的核心部分。本书可能会详细讲解各种CSS布局技术,包括传统的浮动(Floats)布局、定位(Positioning)布局,以及最新的布局模式如Flexbox和CSS Grid。此外,也会介绍响应式设计的媒体查询、视口(Viewport)单位等。 3. 高级CSS技巧: 这些技巧可能包括动画和过渡效果,以及如何优化性能和兼容性。例如,CSS3动画、关键帧动画、转换(Transforms)、滤镜(Filters)和混合模式(Blend Modes)。 4. HTML5特性: 书中可能会深入探讨HTML5的新标签和语义化元素,如`<article>`、`<section>`、`<nav>`等,以及如何使用它们来构建更加标准化和语义化的页面结构。还会涉及到Web表单的新特性,比如表单验证、新的输入类型等。 5. 可访问性(Accessibility): Web可访问性越来越受到重视。本书可能会介绍如何通过HTML和CSS来提升网站的无障碍访问性,比如使用ARIA标签(Accessible Rich Internet Applications)来增强屏幕阅读器的使用体验。 6. 前端性能优化: 性能优化是任何Web项目成功的关键。本书可能会涵盖如何通过优化CSS和HTML来提升网站的加载速度和运行效率。内容可能包括代码压缩、合并、避免重绘和回流、使用Web字体的最佳实践等。 7. JavaScript与CSS/HTML的交互: 在现代Web开发中,JavaScript与CSS及HTML的交云并用是不可或缺的。书中可能会讲解如何通过JavaScript动态地修改样式、操作DOM元素以及使用事件监听和响应用户交互。 8. Web框架和预处理器: 这本书可能会提到流行的Web开发框架和预处理器,比如Bootstrap、Foundation、Sass和Less等,它们是如何简化和加速开发流程的。 9. 测试和维护: 书中也可能包含关于如何测试网页以及如何持续优化和维护CSS和HTML代码的章节。例如,使用断言测试、自动化测试、性能分析工具等。 最后,鉴于文件名称列表中的“压缩包子文件”的表述,这可能是对“压缩包”文件的一种误译或误用,此处“压缩包”应该指的是包含该书籍PDF文件的压缩文件格式,如ZIP或RAR。而“Pro CSS and HTML Design Patterns.pdf”指的就是该书籍的PDF格式电子版文件。 以上所述,构成了《Pro CSS and HTML Design Patterns》一书可能包含的核心知识点。通过学习这些内容,Web前端开发者可以掌握更为高效和优雅的设计模式,从而在日常工作中更高效地解决实际问题。
recommend-type

Zotero 7数据同步:Attanger插件安装&设置,打造文献管理利器

/i.s3.glbimg.com/v1/AUTH_08fbf48bc0524877943fe86e43087e7a/internal_photos/bs/2018/L/j/ipKA8TRPyt4ptaG1DGww/t-m-01.png) # 1. Zotero 7数据同步简介 在数字时代,管理研究数据变得日益重要,而Zot
recommend-type

卷积神经网络的基础理论200字

<think>我们正在回答用户关于卷积神经网络(CNN)基础理论的询问。根据提供的引用资料,我们可以概述CNN的基本结构和工作原理。引用[1]提到CNN是一种前馈神经网络,信息从输入单向流动到输出。引用[2]指出CNN在图像处理等领域应用广泛,通过层次结构和参数调整实现模式识别。引用[3]说明CNN包含卷积层,使用卷积核提取局部特征,减少参数量,提高效率。引用[4]则提到了训练过程,包括前向传播、误差计算和反向传播(梯度下降)。因此,我们将从以下几个方面概述CNN:1.CNN的基本结构(卷积层、池化层、全连接层)2.卷积操作的工作原理(局部感受野、权重共享)3.训练过程(前向传播、损失函数、反
recommend-type

轻便实用的Java库类查询工具介绍

标题 "java2库类查询" 和描述表明,所提及的工具是一个专门用于查询Java库类的应用程序。此软件旨在帮助开发者快速地查找和引用Java的标准开发工具包(SDK)中包含的所有应用程序编程接口(API)类。通过这样的工具,开发者可以节省大量在官方文档或搜索引擎上寻找类定义和使用方法的时间。它被描述为轻巧且方便,这表明其占用的系统资源相对较少,同时提供直观的用户界面,使得查询过程简洁高效。 从描述中可以得出几个关键知识点: 1. Java SDK:Java的软件开发工具包(SDK)是Java平台的一部分,提供了一套用于开发Java应用软件的软件包和库。这些软件包通常被称为API,为开发者提供了编程界面,使他们能够使用Java语言编写各种类型的应用程序。 2. 库类查询:这个功能对于开发者来说非常关键,因为它提供了一个快速查找特定库类及其相关方法、属性和使用示例的途径。良好的库类查询工具可以帮助开发者提高工作效率,减少因查找文档而中断编程思路的时间。 3. 轻巧性:软件的轻巧性通常意味着它对计算机资源的要求较低。这样的特性对于资源受限的系统尤为重要,比如老旧的计算机、嵌入式设备或是当开发者希望最小化其开发环境占用空间时。 4. 方便性:软件的方便性通常关联于其用户界面设计,一个直观、易用的界面可以让用户快速上手,并减少在使用过程中遇到的障碍。 5. 包含所有API:一个优秀的Java库类查询软件应当能够覆盖Java所有标准API,这包括Java.lang、Java.util、Java.io等核心包,以及Java SE平台的所有其他标准扩展包。 从标签 "java 库 查询 类" 可知,这个软件紧密关联于Java编程语言的核心功能——库类的管理和查询。这些标签可以关联到以下知识点: - Java:一种广泛用于企业级应用、移动应用(如Android应用)、网站后端、大型系统和许多其他平台的编程语言。 - 库:在Java中,库是一组预打包的类和接口,它们可以被应用程序重复使用。Java提供了庞大的标准库,以支持各种常见的任务和功能。 - 查询:查询指的是利用软件工具搜索、定位和检索信息的过程。对于Java库类查询工具来说,这意味着可以通过类名、方法签名或其他标识符来查找特定的API条目。 最后,压缩包文件列表包含了两个文件:“java.dit”和“Java.exe”。其中“Java.exe”很可能是程序的可执行文件,而“java.dit”可能是一个数据文件,用于存储Java类的索引或数据。由于文件名后缀通常与文件类型相关联,但“dit”并不是一个常见的文件扩展名。这可能是一个特定于软件的自定义格式,或是一个打字错误。 总结来说,"java2库类查询" 是一个针对Java开发者的实用工具,它提供了一个轻量级、易用的平台来查询和定位Java标准库中的所有类和API。此工具对优化开发流程,减少查找Java类文档的时间大有裨益,尤其适合需要频繁查阅Java API的开发者使用。
recommend-type

【Zotero 7终极指南】:新手必备!Attanger插件全攻略与数据同步神技

# 1. Zotero 7与Attanger插件的介绍 在当今的学术研究和知识管理领域,高效的文献管理工具至关重要。Zotero 7作为一个流行的参考文献管理软件,因其强大的功能和用户友好的界面而受到专业人士的青睐。而Attanger插件则为Zotero 7带来了更多定制化和高级功能,极大地增强
recommend-type

MATLAB整段注释快捷键

<think>首先,用户的问题是:“MATLAB整段注释快捷键”,意思是“MATLAB整段注释快捷键”。我需要用中文回答。 根据规则: - 回答必须用中文。 - 回答完问题后,必须在末尾输出“§§相关问题§§”,然后提出3-5个相关问题。 - “§§相关问题§§”必须完整输出在第一行,禁止换行分散。 - 相关问题必须基于用户的问题和我的回答。 现在,回忆MATLAB的快捷键:在MATLAB中,整段注释通常使用快捷键Ctrl + R来注释选中的代码行,使用Ctrl + T来取消注释。用户问的是“整段注释”,所以我应该确认快捷键。 在MATLAB中: - 注释选中的行:Ctrl + R -
recommend-type

Eclipse Jad反编译插件:提升.class文件查看便捷性

反编译插件for Eclipse是一个专门设计用于在Eclipse集成开发环境中进行Java反编译的工具。通过此类插件,开发者可以在不直接访问源代码的情况下查看Java编译后的.class文件的源代码,这在开发、维护和学习使用Java技术的过程中具有重要的作用。 首先,我们需要了解Eclipse是一个跨平台的开源集成开发环境,主要用来开发Java应用程序,但也支持其他诸如C、C++、PHP等多种语言的开发。Eclipse通过安装不同的插件来扩展其功能。这些插件可以由社区开发或者官方提供,而jadclipse就是这样一个社区开发的插件,它利用jad.exe这个第三方命令行工具来实现反编译功能。 jad.exe是一个反编译Java字节码的命令行工具,它可以将Java编译后的.class文件还原成一个接近原始Java源代码的格式。这个工具非常受欢迎,原因在于其反编译速度快,并且能够生成相对清晰的Java代码。由于它是一个独立的命令行工具,直接使用命令行可以提供较强的灵活性,但是对于一些不熟悉命令行操作的用户来说,集成到Eclipse开发环境中将会极大提高开发效率。 使用jadclipse插件可以很方便地在Eclipse中打开任何.class文件,并且将反编译的结果显示在编辑器中。用户可以在查看反编译的源代码的同时,进行阅读、调试和学习。这样不仅可以帮助开发者快速理解第三方库的工作机制,还能在遇到.class文件丢失源代码时进行紧急修复工作。 对于Eclipse用户来说,安装jadclipse插件相当简单。一般步骤包括: 1. 下载并解压jadclipse插件的压缩包。 2. 在Eclipse中打开“Help”菜单,选择“Install New Software”。 3. 点击“Add”按钮,输入插件更新地址(通常是jadclipse的更新站点URL)。 4. 选择相应的插件(通常名为“JadClipse”),然后进行安装。 5. 安装完成后重启Eclipse,插件开始工作。 一旦插件安装好之后,用户只需在Eclipse中双击.class文件,或者右键点击文件并选择“Open With Jadclipse”,就能看到对应的Java源代码。如果出现反编译不准确或失败的情况,用户还可以直接在Eclipse中配置jad.exe的路径,或者调整jadclipse的高级设置来优化反编译效果。 需要指出的是,使用反编译工具虽然方便,但要注意反编译行为可能涉及到版权问题。在大多数国家和地区,反编译软件代码属于合法行为,但仅限于学习、研究、安全测试或兼容性开发等目的。如果用户意图通过反编译获取商业机密或进行非法复制,则可能违反相关法律法规。 总的来说,反编译插件for Eclipse是一个强大的工具,它极大地简化了Java反编译流程,提高了开发效率,使得开发者在没有源代码的情况下也能有效地维护和学习Java程序。但开发者在使用此类工具时应遵守法律与道德规范,避免不当使用。
recommend-type

【进阶Python绘图】:掌握matplotlib坐标轴刻度间隔的高级技巧,让你的图表脱颖而出

# 摘要 本文系统地探讨了matplotlib库中坐标轴刻度间隔的定制与优化技术。首先概述了matplotlib坐标轴刻度间隔的基本概念及其在图表中的重要性,接