新增在线练题示例
This commit is contained in:
parent
c3b455b05d
commit
e553404b0a
@ -1,13 +1,13 @@
|
|||||||
@font-face {
|
@font-face {
|
||||||
font-family: 'Netron';
|
font-family: 'Netron';
|
||||||
src: url('../../../public/fonts/Netron.ttf') format('truetype');
|
src: url('/public/fonts/Netron.ttf') format('truetype');
|
||||||
font-weight: normal;
|
font-weight: normal;
|
||||||
font-style: normal;
|
font-style: normal;
|
||||||
font-display: swap;
|
font-display: swap;
|
||||||
}
|
}
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: 'huangkaihua';
|
font-family: 'huangkaihua';
|
||||||
src: url('../../../public/fonts/huangkaihuaLawyerfont-2.ttf') format('truetype');
|
src: url('/public/fonts/huangkaihuaLawyerfont-2.ttf') format('truetype');
|
||||||
font-weight: normal;
|
font-weight: normal;
|
||||||
font-style: normal;
|
font-style: normal;
|
||||||
font-display: swap;
|
font-display: swap;
|
||||||
|
@ -21,15 +21,22 @@ const formattedLatex = ref('');
|
|||||||
|
|
||||||
// 监听外部内容的变化,并同步更新内部内容
|
// 监听外部内容的变化,并同步更新内部内容
|
||||||
watch(() => props.contentInput, (newValue) => {
|
watch(() => props.contentInput, (newValue) => {
|
||||||
localInput.value = newValue.replaceAll('<code>', '$')
|
localInput.value = newValue;
|
||||||
|
});
|
||||||
|
|
||||||
|
function processFormat(text) {
|
||||||
|
return String(text).replaceAll('<code>', '$')
|
||||||
.replaceAll('</code>', '$')
|
.replaceAll('</code>', '$')
|
||||||
.replaceAll('/lt', '<')
|
.replaceAll('/lt', '<')
|
||||||
.replaceAll('/gt', '>');
|
.replaceAll('/gt', '>')
|
||||||
});
|
.replaceAll(/(?<!<)\/(\w+)/g, '\\$1')
|
||||||
|
.replaceAll('m\\s', 'm/s')
|
||||||
|
;
|
||||||
|
}
|
||||||
|
|
||||||
// 监听 localInput 的变化,实时更新渲染内容
|
// 监听 localInput 的变化,实时更新渲染内容
|
||||||
watch(localInput, async (newValue) => {
|
watch(localInput, async (newValue) => {
|
||||||
formattedLatex.value = marked.parse(newValue);
|
formattedLatex.value = marked.parse(processFormat(newValue));
|
||||||
await nextTick(); // 等待 DOM 更新完成
|
await nextTick(); // 等待 DOM 更新完成
|
||||||
if (window.MathJax) {
|
if (window.MathJax) {
|
||||||
MathJax.typeset(); // 触发 MathJax 重新渲染公式
|
MathJax.typeset(); // 触发 MathJax 重新渲染公式
|
||||||
@ -45,7 +52,7 @@ const checkWindowSize = () => {
|
|||||||
onMounted(async () => {
|
onMounted(async () => {
|
||||||
checkWindowSize();
|
checkWindowSize();
|
||||||
window.addEventListener('resize', checkWindowSize);
|
window.addEventListener('resize', checkWindowSize);
|
||||||
formattedLatex.value = marked.parse(localInput.value);
|
formattedLatex.value = marked.parse(processFormat(localInput.value));
|
||||||
await nextTick();
|
await nextTick();
|
||||||
if (window.MathJax) {
|
if (window.MathJax) {
|
||||||
MathJax.typeset();
|
MathJax.typeset();
|
||||||
|
@ -53,8 +53,8 @@ const navItems = [
|
|||||||
{name: '主页', link: '/'},
|
{name: '主页', link: '/'},
|
||||||
'divider',
|
'divider',
|
||||||
{name: '博客', link: '/blog'},
|
{name: '博客', link: '/blog'},
|
||||||
{name: '项目', link: '/projects'},
|
// {name: '项目', link: '/projects'},
|
||||||
// {name: '实例', link: '/demos'},
|
{name: '实例', link: '/demos'},
|
||||||
{name: '小工具', link: '/tools'},
|
{name: '小工具', link: '/tools'},
|
||||||
{name: '留言板', link: '/demos/board'},
|
{name: '留言板', link: '/demos/board'},
|
||||||
'divider',
|
'divider',
|
||||||
|
@ -1,14 +1,21 @@
|
|||||||
<script setup>
|
<script setup>
|
||||||
import { ref } from 'vue'
|
import { ref } from 'vue'
|
||||||
import Demos_box from '../components/Demos_box.vue'
|
import Demos_box from '../components/Demos_box.vue'
|
||||||
|
import {getDomain} from "../utils/getDomain.js";
|
||||||
const searchQuery = ref('')
|
const searchQuery = ref('')
|
||||||
|
|
||||||
const demos = ref([
|
const demos = ref([
|
||||||
{
|
{
|
||||||
id: 1,
|
id: 'board',
|
||||||
name: '留言板',
|
name: '留言板',
|
||||||
author: ["麦克荣", "路易斯周"],
|
author: ["Mike Rong", "Louis Zhou"],
|
||||||
tags: ['功能']
|
tags: ['功能']
|
||||||
|
},{
|
||||||
|
id: 'pod',
|
||||||
|
name: '在线练题',
|
||||||
|
author: ["Louis Zhou"],
|
||||||
|
tags: ['课内'],
|
||||||
|
image: 'https://' + getDomain() + '/data/file/pod.png'
|
||||||
},
|
},
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
74
src/pages/demoPages/podExercise/Pod_page.vue
Normal file
74
src/pages/demoPages/podExercise/Pod_page.vue
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
<script setup>
|
||||||
|
|
||||||
|
import {ref} from "vue";
|
||||||
|
|
||||||
|
const podList = ref([
|
||||||
|
{
|
||||||
|
title: 'PHY MCQ 1',
|
||||||
|
file: 'phy250226',
|
||||||
|
describe: '',
|
||||||
|
date: '2025-2-26'
|
||||||
|
},{
|
||||||
|
title: 'PHY MCQ 2',
|
||||||
|
file: 'phy250227',
|
||||||
|
describe: '',
|
||||||
|
date: '2025-2-27'
|
||||||
|
},
|
||||||
|
])
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div class="container" v-if="/^\/demos\/pod\/?$/.test($route.path)">
|
||||||
|
<h1>无限制做题大赛</h1>
|
||||||
|
<div class="pod-container">
|
||||||
|
<router-link class="pods" v-for="pod in podList" :to="`/demos/pod/quiz?id=${pod.file}`">
|
||||||
|
<div class="title">{{ pod.title }}</div>
|
||||||
|
<div class="date">{{ pod.date }}</div>
|
||||||
|
</router-link>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<router-view v-else />
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.container {
|
||||||
|
width: 100%;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
justify-content: flex-start;
|
||||||
|
align-items: center;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 20px;
|
||||||
|
}
|
||||||
|
.pod-container {
|
||||||
|
width: 100%;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: flex-start;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 20px;
|
||||||
|
}
|
||||||
|
.pods {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
width: 150px;
|
||||||
|
height: 150px;
|
||||||
|
background: #2a2a2a;
|
||||||
|
max-width: 800px;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
.theme-light .pods {
|
||||||
|
background: #efefef;
|
||||||
|
}
|
||||||
|
.title {
|
||||||
|
font-size: 20px;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
.date {
|
||||||
|
font-size: 15px;
|
||||||
|
opacity: 0.5;
|
||||||
|
}
|
||||||
|
</style>
|
@ -1,6 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<general-renderer :content-input="processedText" class="question-text"></general-renderer>
|
<general-renderer :content-input="processedText" class="question-text"></general-renderer>
|
||||||
<!-- <div v-html="" class="question-text"></div>-->
|
<!-- <div v-html="processedText" class="question-text"></div>-->
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
@ -26,7 +26,7 @@ function replaceMedia(text, medias) {
|
|||||||
if (medias[mediaKey]) {
|
if (medias[mediaKey]) {
|
||||||
const [type, url, width, height] = medias[mediaKey];
|
const [type, url, width, height] = medias[mediaKey];
|
||||||
if (type === 'image') {
|
if (type === 'image') {
|
||||||
return `<img src="${url}" width="${width}" height="${height}" alt="Question Image" />`;
|
return `<img src="${url}" width="100%" height="auto" alt="Question Image" />`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return '';
|
return '';
|
||||||
@ -37,12 +37,8 @@ const processedText = computed(() => replaceMedia(props.text, props.medias));
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
.question-text {
|
img {
|
||||||
line-height: 1.6;
|
|
||||||
}
|
|
||||||
|
|
||||||
.question-text img {
|
|
||||||
max-width: 100%;
|
|
||||||
height: auto;
|
height: auto;
|
||||||
margin: 10px 0;
|
margin: 10px 0;
|
||||||
}
|
}
|
||||||
|
@ -50,11 +50,11 @@
|
|||||||
|
|
||||||
<!-- 进度条 -->
|
<!-- 进度条 -->
|
||||||
<div v-if="!isLoading" class="progress-bar">
|
<div v-if="!isLoading" class="progress-bar">
|
||||||
<div
|
<div v-if="progressPercentage"
|
||||||
class="progress"
|
class="progress"
|
||||||
:style="{
|
:style="{
|
||||||
width: `${progressPercentage}%`,
|
width: `${progressPercentage}%`,
|
||||||
background: `linear-gradient(to right, red, yellow, green ${progressPercentage}%, transparent ${progressPercentage}%)`
|
background: getColor(progressPercentage)
|
||||||
}"
|
}"
|
||||||
>
|
>
|
||||||
{{ progressPercentage.toFixed(1) }}%
|
{{ progressPercentage.toFixed(1) }}%
|
||||||
@ -64,10 +64,14 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import { reactive, ref, onMounted, computed } from 'vue';
|
import {reactive, ref, onMounted, computed, onUpdated} from 'vue';
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import QuestionText from './QuestionText.vue';
|
import QuestionText from './QuestionText.vue';
|
||||||
import GeneralRenderer from "../../../components/GeneralRenderer.vue";
|
import GeneralRenderer from "../../../components/GeneralRenderer.vue";
|
||||||
|
import { useRoute } from 'vue-router'
|
||||||
|
import swal from "../../../utils/sweetalert.js";
|
||||||
|
|
||||||
|
const route = useRoute()
|
||||||
|
|
||||||
// 响应式数据存储 JSON
|
// 响应式数据存储 JSON
|
||||||
const data = reactive({
|
const data = reactive({
|
||||||
@ -103,21 +107,35 @@ const progressPercentage = computed(() => {
|
|||||||
return (correctCount / Math.min(recentResults.value.length, totalQuestions.value)) * 100;
|
return (correctCount / Math.min(recentResults.value.length, totalQuestions.value)) * 100;
|
||||||
});
|
});
|
||||||
|
|
||||||
// 加载数据
|
const initial = async () => {
|
||||||
onMounted(async () => {
|
const file = route.query.id || 'phy250226';
|
||||||
try {
|
try {
|
||||||
const response = await axios.get('https://mva-cyber.club/data/file/phy250226.json');
|
const response = await axios.get(`https://mva-cyber.club/data/file/${file}.json`);
|
||||||
const jsonData = response.data;
|
const jsonData = response.data;
|
||||||
Object.assign(data, jsonData);
|
Object.assign(data, jsonData);
|
||||||
generateNewExam(); // 初始生成一份试卷
|
generateNewExam(); // 初始生成一份试卷
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('加载数据失败:', error);
|
console.error('加载数据失败:', error);
|
||||||
alert('数据加载失败,请稍后重试。');
|
swal.tip('error', '数据加载失败,请稍后重试。')
|
||||||
} finally {
|
} finally {
|
||||||
isLoading.value = false;
|
isLoading.value = false;
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
|
|
||||||
|
// 加载数据
|
||||||
|
onMounted(initial);
|
||||||
|
|
||||||
|
function getColor(progressPercentage) {
|
||||||
|
let r, g, b;
|
||||||
|
|
||||||
|
// 0% 到 100% 渐变,从红色到绿色
|
||||||
|
r = Math.floor(200 * (1 - progressPercentage / 100)); // 红色逐渐减少
|
||||||
|
g = Math.floor(200 * (progressPercentage / 100)); // 绿色逐渐增加
|
||||||
|
b = 0; // 蓝色始终为0
|
||||||
|
|
||||||
|
// 返回rgb颜色格式
|
||||||
|
return `rgb(${r}, ${g}, ${b})`;
|
||||||
|
}
|
||||||
// 随机选择一道题
|
// 随机选择一道题
|
||||||
function selectRandomQuestion() {
|
function selectRandomQuestion() {
|
||||||
if (data.texts.length <= 1) return;
|
if (data.texts.length <= 1) return;
|
||||||
@ -143,7 +161,7 @@ function processMedia(text, medias) {
|
|||||||
if (medias[mediaKey]) {
|
if (medias[mediaKey]) {
|
||||||
const [type, url, width, height] = medias[mediaKey];
|
const [type, url, width, height] = medias[mediaKey];
|
||||||
if (type === 'image') {
|
if (type === 'image') {
|
||||||
return `<img src="${url}" width="${width}" height="${height}" alt="Question Image" />`;
|
return `<img src="${url}" width="100%" height="auto" alt="Question Image" />`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return '';
|
return '';
|
||||||
@ -232,7 +250,6 @@ function findSimilarQuestions() {
|
|||||||
const currentText = currentQuestion.value.text;
|
const currentText = currentQuestion.value.text;
|
||||||
const similar = [];
|
const similar = [];
|
||||||
for (let i = 1; i < data.itemtypes.length; i++) {
|
for (let i = 1; i < data.itemtypes.length; i++) {
|
||||||
console.log(calculateStringSimilarity(data.texts[i], currentText), {1: data.texts[i]})
|
|
||||||
if (
|
if (
|
||||||
calculateStringSimilarity(data.texts[i], currentText) > 50 && // 相似度
|
calculateStringSimilarity(data.texts[i], currentText) > 50 && // 相似度
|
||||||
data.texts[i] !== currentQuestion.value.text// 不是当前题目
|
data.texts[i] !== currentQuestion.value.text// 不是当前题目
|
||||||
@ -258,7 +275,7 @@ function selectCurrentQuestion() {
|
|||||||
// 提交答案
|
// 提交答案
|
||||||
function submitAnswer() {
|
function submitAnswer() {
|
||||||
if (userAnswer.value === null) {
|
if (userAnswer.value === null) {
|
||||||
alert('请选择一个答案!');
|
swal.tip('info', '必须选一个')
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
correctIndex.value = currentQuestion.value.answerkeys.findIndex((key, idx) => idx > 0 && key === 1) - 1;
|
correctIndex.value = currentQuestion.value.answerkeys.findIndex((key, idx) => idx > 0 && key === 1) - 1;
|
||||||
@ -282,7 +299,7 @@ function nextQuestion() {
|
|||||||
questionCount.value++;
|
questionCount.value++;
|
||||||
selectCurrentQuestion();
|
selectCurrentQuestion();
|
||||||
} else {
|
} else {
|
||||||
alert('试卷已完成!将生成新试卷。');
|
swal.tip('success', '所有题目已完成!将生成新试卷。')
|
||||||
generateNewExam(); // 完成当前试卷后生成新试卷
|
generateNewExam(); // 完成当前试卷后生成新试卷
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -290,9 +307,14 @@ function nextQuestion() {
|
|||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
.quiz-container {
|
.quiz-container {
|
||||||
max-width: 800px;
|
display: flex;
|
||||||
margin: 20px auto;
|
flex-direction: column;
|
||||||
height: calc(100vh - 110px);
|
align-items: center;
|
||||||
|
justify-content: flex-start;
|
||||||
|
width: calc(100% - 40px);
|
||||||
|
//max-width: 800px;
|
||||||
|
margin: auto;
|
||||||
|
height: calc(100vh - 120px);
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
padding: 20px;
|
padding: 20px;
|
||||||
font-family: Arial, sans-serif;
|
font-family: Arial, sans-serif;
|
||||||
@ -306,16 +328,18 @@ function nextQuestion() {
|
|||||||
|
|
||||||
.question-container {
|
.question-container {
|
||||||
margin-bottom: 20px;
|
margin-bottom: 20px;
|
||||||
|
max-width: 800px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.progress-bar {
|
.progress-bar {
|
||||||
width: 100%;
|
width: calc(100% - 100px);
|
||||||
height: 20px;
|
height: 20px;
|
||||||
background-color: #f0f0f0;
|
background-color: rgba(126, 126, 126, 0.23);
|
||||||
margin-top: 20px;
|
margin-top: 20px;
|
||||||
position: relative;
|
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
position: fixed;
|
||||||
|
bottom: 20px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.progress {
|
.progress {
|
||||||
@ -334,10 +358,12 @@ function nextQuestion() {
|
|||||||
|
|
||||||
.option {
|
.option {
|
||||||
margin: 10px 0;
|
margin: 10px 0;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.option label {
|
.option label {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
padding: 10px 0;
|
padding: 10px 0;
|
||||||
|
@ -15,7 +15,8 @@ import Account_admin_userManage from "../pages/accountPages/Account_admin_userMa
|
|||||||
import Projects from "../pages/Projects_home.vue";
|
import Projects from "../pages/Projects_home.vue";
|
||||||
import Demos_home from "../pages/Demos_home.vue";
|
import Demos_home from "../pages/Demos_home.vue";
|
||||||
import Board_page from "../pages/demoPages/messageBoard/Board_page.vue";
|
import Board_page from "../pages/demoPages/messageBoard/Board_page.vue";
|
||||||
import Pod_page from "../pages/demoPages/podExercise/Quiz.vue";
|
import Pod_page from "../pages/demoPages/podExercise/Pod_page.vue";
|
||||||
|
import Pod_quiz from "../pages/demoPages/podExercise/Quiz.vue";
|
||||||
import Tools_home from "../pages/Tools_home.vue";
|
import Tools_home from "../pages/Tools_home.vue";
|
||||||
import GpaCalculator_page from "../pages/toolPages/gpaCalculator/gpaCalculator_page.vue";
|
import GpaCalculator_page from "../pages/toolPages/gpaCalculator/gpaCalculator_page.vue";
|
||||||
import About from "../pages/About.vue";
|
import About from "../pages/About.vue";
|
||||||
@ -43,9 +44,14 @@ const routes = [
|
|||||||
name: 'Demos',
|
name: 'Demos',
|
||||||
component: Demos_home,
|
component: Demos_home,
|
||||||
children: [
|
children: [
|
||||||
{path: "1", component: Board_page},
|
|
||||||
{path: "board", component: Board_page},
|
{path: "board", component: Board_page},
|
||||||
{path: "pod", component: Pod_page}
|
{
|
||||||
|
path: "pod",
|
||||||
|
component: Pod_page,
|
||||||
|
children: [
|
||||||
|
{path: "quiz", component: Pod_quiz}
|
||||||
|
]
|
||||||
|
}
|
||||||
]
|
]
|
||||||
}, {
|
}, {
|
||||||
path: '/tools',
|
path: '/tools',
|
||||||
|
Loading…
x
Reference in New Issue
Block a user