곰픽 [4-3][4-4] 채점시 레이어 이름을 검색할 경우 유사도 옵션 추가
This commit is contained in:
@@ -498,8 +498,8 @@
|
|||||||
},
|
},
|
||||||
"4": {
|
"4": {
|
||||||
"type": "layer.Effects",
|
"type": "layer.Effects",
|
||||||
"ele": "//Layer[Name[@value='{layer}']]/Effects/Item",
|
"ele": "//Layer[Name[@value='{search}']]/Effects/Item",
|
||||||
"layer": "Flower",
|
"search": "Flower",
|
||||||
"value": {
|
"value": {
|
||||||
"name": "세피아",
|
"name": "세피아",
|
||||||
"option": {
|
"option": {
|
||||||
@@ -557,19 +557,21 @@
|
|||||||
"point": 6
|
"point": 6
|
||||||
},
|
},
|
||||||
"11": {
|
"11": {
|
||||||
"ele": "none",
|
"type": "none",
|
||||||
|
"ele": "",
|
||||||
"point": 0,
|
"point": 0,
|
||||||
"desc": "기본설정"
|
"desc": "기본설정"
|
||||||
},
|
},
|
||||||
"12": {
|
"12": {
|
||||||
"ele": "none",
|
"type": "none",
|
||||||
|
"ele": "",
|
||||||
"point": 0,
|
"point": 0,
|
||||||
"desc": "파일명 확인"
|
"desc": "파일명 확인"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"5": {
|
"5": {
|
||||||
"1": {
|
"1": {
|
||||||
"type": "multi",
|
"type": "canvas.Size",
|
||||||
"ele": "//Document/Width/@value | //Document/Height/@value",
|
"ele": "//Document/Width/@value | //Document/Height/@value",
|
||||||
"value": [
|
"value": [
|
||||||
"650",
|
"650",
|
||||||
@@ -579,7 +581,8 @@
|
|||||||
"desc": "캔버스 사이즈 650*450"
|
"desc": "캔버스 사이즈 650*450"
|
||||||
},
|
},
|
||||||
"2": {
|
"2": {
|
||||||
"ele": "none",
|
"type": "none",
|
||||||
|
"ele": "",
|
||||||
"point": 5,
|
"point": 5,
|
||||||
"desc": "배경색 문항은 채점 불가"
|
"desc": "배경색 문항은 채점 불가"
|
||||||
},
|
},
|
||||||
@@ -587,10 +590,12 @@
|
|||||||
"type": "exists",
|
"type": "exists",
|
||||||
"ele": "//Layer/MaskOpType/@value",
|
"ele": "//Layer/MaskOpType/@value",
|
||||||
"value": "Layering",
|
"value": "Layering",
|
||||||
"point": 6
|
"point": 6,
|
||||||
|
"desc": "레이어 마스크 설정 확인"
|
||||||
},
|
},
|
||||||
"4": {
|
"4": {
|
||||||
"ele": "none",
|
"type": "none",
|
||||||
|
"ele": "",
|
||||||
"point": 6,
|
"point": 6,
|
||||||
"desc": "가로방향 흐릿하게 문항은 채점 불가"
|
"desc": "가로방향 흐릿하게 문항은 채점 불가"
|
||||||
},
|
},
|
||||||
@@ -601,7 +606,7 @@
|
|||||||
"point": 3
|
"point": 3
|
||||||
},
|
},
|
||||||
"6": {
|
"6": {
|
||||||
"type": "size",
|
"type": "shape.size",
|
||||||
"ele": "//Layer//op_points",
|
"ele": "//Layer//op_points",
|
||||||
"value": {
|
"value": {
|
||||||
"width": 400,
|
"width": 400,
|
||||||
@@ -611,13 +616,13 @@
|
|||||||
"desc": "레이어 쉐이프 X, Y 좌표를 가지고 너비, 높이 계산하여 정답 채점"
|
"desc": "레이어 쉐이프 X, Y 좌표를 가지고 너비, 높이 계산하여 정답 채점"
|
||||||
},
|
},
|
||||||
"7": {
|
"7": {
|
||||||
"type": "gradient",
|
"type": "gradient.color",
|
||||||
"ele": "//Layer/Shapes/Shape",
|
"ele": "//Layer/Shapes/Shape",
|
||||||
"startColor": "gradient_start_color/@value",
|
"startColor": "gradient_start_color/@value",
|
||||||
"endColor": "gradient_end_color/@value",
|
"endColor": "gradient_end_color/@value",
|
||||||
"value": {
|
"value": {
|
||||||
"startColor": "ffe000",
|
"startColor": "ffe000",
|
||||||
"endColor": "34a159"
|
"endColor": "34A159"
|
||||||
},
|
},
|
||||||
"point": 6
|
"point": 6
|
||||||
},
|
},
|
||||||
@@ -647,11 +652,10 @@
|
|||||||
"point": 3
|
"point": 3
|
||||||
},
|
},
|
||||||
"12": {
|
"12": {
|
||||||
"type": "color",
|
"type": "text.color",
|
||||||
"ele": "//Layer//Shape[shape_type/@value='TEXT'][contains(draw_type/@value, 'Interior')]/secondary_color/@value",
|
"ele": "//Layer//Shape[shape_type/@value='TEXT'][contains(draw_type/@value, 'Interior')]/secondary_color/@value",
|
||||||
"value": "b46ef8",
|
"value": "b46Ef8",
|
||||||
"point": 3,
|
"point": 3
|
||||||
"desc": "색상 코드 비교 시 소문자로 입력할 것"
|
|
||||||
},
|
},
|
||||||
"13": {
|
"13": {
|
||||||
"type": "exists",
|
"type": "exists",
|
||||||
@@ -660,14 +664,15 @@
|
|||||||
"point": 3
|
"point": 3
|
||||||
},
|
},
|
||||||
"14": {
|
"14": {
|
||||||
"type": "color",
|
"type": "text.color",
|
||||||
"ele": "//Layer//Shape[shape_type/@value='TEXT'][contains(draw_type/@value, 'Outline')]/primary_color/@value",
|
"ele": "//Layer//Shape[shape_type/@value='TEXT'][contains(draw_type/@value, 'Outline')]/primary_color/@value",
|
||||||
"value": "ffffff",
|
"value": "ffffff",
|
||||||
"point": 3,
|
"point": 3
|
||||||
"desc": "색상 코드 비교 시 소문자로 입력할 것"
|
|
||||||
},
|
},
|
||||||
"15": {
|
"15": {
|
||||||
"ele": "//Layer[MaskOpType/@value='Clipping'][last()]",
|
"type": "exists",
|
||||||
|
"ele": "//Layer/MaskOpType/@value",
|
||||||
|
"value": "Clipping",
|
||||||
"point": 6,
|
"point": 6,
|
||||||
"desc": "클리핑 마스크 항목은 별도 레이어로 추가되고 해당 속성을 추가해놓은 레이어가 있는지 여부 체크 함"
|
"desc": "클리핑 마스크 항목은 별도 레이어로 추가되고 해당 속성을 추가해놓은 레이어가 있는지 여부 체크 함"
|
||||||
},
|
},
|
||||||
@@ -675,11 +680,15 @@
|
|||||||
"type": "exists",
|
"type": "exists",
|
||||||
"ele": "//Layer/Shapes/Shape/shape_type/@value",
|
"ele": "//Layer/Shapes/Shape/shape_type/@value",
|
||||||
"value": "RECTANGLE",
|
"value": "RECTANGLE",
|
||||||
"point": 3
|
"point": 3,
|
||||||
|
"desc": {
|
||||||
|
"사각형": "RECTANGLE"
|
||||||
|
}
|
||||||
},
|
},
|
||||||
"17": {
|
"17": {
|
||||||
"type": "size",
|
"type": "clipping.size",
|
||||||
"ele": "//Layer//op_points",
|
"ele": "//Layer//Shape[shape_type/@value='{option}']//op_points",
|
||||||
|
"option": "RECTANGLE",
|
||||||
"value": {
|
"value": {
|
||||||
"width": 150,
|
"width": 150,
|
||||||
"height": 150
|
"height": 150
|
||||||
@@ -689,27 +698,32 @@
|
|||||||
},
|
},
|
||||||
"18": {
|
"18": {
|
||||||
"type": "exists",
|
"type": "exists",
|
||||||
"ele": "//Layer//outline_peninfo/Width/@value",
|
"ele": "//Layer//Shape[shape_type/@value='{option}']/outline_peninfo/Width/@value",
|
||||||
|
"option": "RECTANGLE",
|
||||||
"value": "7",
|
"value": "7",
|
||||||
"point": 3
|
"point": 3
|
||||||
},
|
},
|
||||||
"19": {
|
"19": {
|
||||||
"type": "color",
|
"type": "clipping.color",
|
||||||
"ele": "//Layer//Shape[contains(draw_type/@value, 'Outline')]/primary_color/@value",
|
"ele": "//Layer//Shape[shape_type/@value='{option}' and contains(draw_type/@value, 'Outline')]/primary_color/@value",
|
||||||
|
"option": "RECTANGLE",
|
||||||
"value": "e8e88e",
|
"value": "e8e88e",
|
||||||
"point": 3,
|
"point": 3,
|
||||||
"desc": "색상 코드 비교 시 소문자로 입력할 것(채우기:secondary_color, 외곽선:primary_color)"
|
"desc": "채우기:secondary_color, 외곽선:primary_color"
|
||||||
},
|
},
|
||||||
"20": {
|
"20": {
|
||||||
"type": "shadow",
|
"type": "shadow",
|
||||||
"ele": {
|
"ele": "//Layer//Shape[shape_type/@value='{option}']",
|
||||||
|
"ele2": {
|
||||||
"shadow": "//Layer//Shape[contains(draw_type/@value, 'Shadow')]",
|
"shadow": "//Layer//Shape[contains(draw_type/@value, 'Shadow')]",
|
||||||
"width": "//Layer//Shape[contains(draw_type/@value, 'Shadow')]/shadow_width/@value",
|
"width": "//Layer//Shape[contains(draw_type/@value, 'Shadow')]/shadow_width/@value",
|
||||||
"distance": "//Layer//Shape[contains(draw_type/@value, 'Shadow')]/shadow_distance/@value",
|
"distance": "//Layer//Shape[contains(draw_type/@value, 'Shadow')]/shadow_distance/@value",
|
||||||
"blur": "//Layer//Shape[contains(draw_type/@value, 'Shadow')]/shadow_blur/@value",
|
"blur": "//Layer//Shape[contains(draw_type/@value, 'Shadow')]/shadow_blur/@value",
|
||||||
"angle": "//Layer//Shape[contains(draw_type/@value, 'Shadow')]/shadow_angle/@value"
|
"angle": "//Layer//Shape[contains(draw_type/@value, 'Shadow')]/shadow_angle/@value"
|
||||||
},
|
},
|
||||||
|
"option": "RECTANGLE",
|
||||||
"value": {
|
"value": {
|
||||||
|
"shadow": true,
|
||||||
"width": "3",
|
"width": "3",
|
||||||
"distance": "5",
|
"distance": "5",
|
||||||
"blur": "1",
|
"blur": "1",
|
||||||
@@ -719,12 +733,14 @@
|
|||||||
"desc": "그림자 속성이 있는 경우 그림자 속성의 너비, 거리, 흐림 정도, 각도를 비교하여 정답 채점"
|
"desc": "그림자 속성이 있는 경우 그림자 속성의 너비, 거리, 흐림 정도, 각도를 비교하여 정답 채점"
|
||||||
},
|
},
|
||||||
"21": {
|
"21": {
|
||||||
"ele": "none",
|
"type": "none",
|
||||||
|
"ele": "",
|
||||||
"point": 0,
|
"point": 0,
|
||||||
"desc": "기본설정"
|
"desc": "기본설정"
|
||||||
},
|
},
|
||||||
"22": {
|
"22": {
|
||||||
"ele": "none",
|
"type": "none",
|
||||||
|
"ele": "",
|
||||||
"point": 0,
|
"point": 0,
|
||||||
"desc": "파일명 확인"
|
"desc": "파일명 확인"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,16 +11,24 @@ const stringSimilarity = require("string-similarity");
|
|||||||
*/
|
*/
|
||||||
function findSimilarString(xmlDoc, targetString, threshold = 0.8) {
|
function findSimilarString(xmlDoc, targetString, threshold = 0.8) {
|
||||||
// XML 내부의 비교 대상 텍스트 리스트 찾기
|
// XML 내부의 비교 대상 텍스트 리스트 찾기
|
||||||
function getTextNodes(xmlDoc, stringList = []) {
|
function getTextNodes(xmlDoc, paths = ["//CRCUnitArr/@Name"]) {
|
||||||
const stringNodes = xpath.select("//CRCUnitArr/@Name", xmlDoc);
|
const stringList = [];
|
||||||
stringNodes.forEach(stringNode => {
|
|
||||||
stringList.push(stringNode.value);
|
paths.forEach(path => {
|
||||||
|
const nodes = xpath.select(path, xmlDoc);
|
||||||
|
nodes.forEach(node => {
|
||||||
|
stringList.push(node.value);
|
||||||
});
|
});
|
||||||
|
});
|
||||||
|
|
||||||
return stringList;
|
return stringList;
|
||||||
}
|
}
|
||||||
|
|
||||||
// XML에서 모든 텍스트 추출
|
// XML에서 모든 텍스트 추출
|
||||||
const stringList = getTextNodes(xmlDoc);
|
const stringList = getTextNodes(xmlDoc, [
|
||||||
|
"//CRCUnitArr/@Name",
|
||||||
|
"//Layer/Name/@value",
|
||||||
|
]);
|
||||||
|
|
||||||
// 유사도 비교하여 가장 유사한 문자열 찾기
|
// 유사도 비교하여 가장 유사한 문자열 찾기
|
||||||
let bestMatch = null;
|
let bestMatch = null;
|
||||||
|
|||||||
264
gpdpScoring.js
264
gpdpScoring.js
@@ -1,5 +1,8 @@
|
|||||||
const xpath = require('xpath');
|
const xpath = require('xpath');
|
||||||
const { DOMParser } = require('xmldom');
|
const { DOMParser } = require('xmldom');
|
||||||
|
const _ = require('lodash');
|
||||||
|
|
||||||
|
const findSimilarString = require('./findSimilarString');
|
||||||
|
|
||||||
function parseColorToHex(colorString) {
|
function parseColorToHex(colorString) {
|
||||||
// 정규식을 사용하여 B, G, R, A 값 추출
|
// 정규식을 사용하여 B, G, R, A 값 추출
|
||||||
@@ -49,47 +52,69 @@ function getGpdpScore(gpdpData, scoringJson, index) {
|
|||||||
const {
|
const {
|
||||||
tolerance = 0, // 숫자 비교 시 허용 오차
|
tolerance = 0, // 숫자 비교 시 허용 오차
|
||||||
type = 'auto', // 'auto' | 'number[]' | 'string[]' | 'object' | 'primitive'
|
type = 'auto', // 'auto' | 'number[]' | 'string[]' | 'object' | 'primitive'
|
||||||
|
partial = false,
|
||||||
} = options;
|
} = options;
|
||||||
|
|
||||||
if (type === 'force-correct') {
|
if (type === 'force-correct') {
|
||||||
isEqual = true;
|
score = point;
|
||||||
}
|
}
|
||||||
else if (Array.isArray(right) && Array.isArray(user)) {
|
// 객체 비교
|
||||||
if (right.length === user.length) {
|
else if (
|
||||||
if
|
(type === 'object' || (type === 'auto' && typeof right === 'object' && typeof user === 'object'))
|
||||||
(type === 'number[]' ||
|
&& !Array.isArray(right)
|
||||||
(type === 'auto' && typeof right[0] === 'number')) {
|
) {
|
||||||
isEqual = right.every((val, idx) => Math.abs(val - user[idx]) <= tolerance);
|
if (partial) {
|
||||||
}
|
const keys = Object.keys(right);
|
||||||
else if
|
const partialPoint = point / keys.length;
|
||||||
(type === 'string[]' ||
|
for (const k of keys) {
|
||||||
(type === 'auto' && typeof right[0] === 'string')) {
|
if (_.isEqual(user[k], right[k])) {
|
||||||
isEqual = right.every((val, idx) => val === user[idx]);
|
score += partialPoint;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
if (_.isEqual(user, right)) {
|
||||||
|
score = point;
|
||||||
}
|
}
|
||||||
else if
|
|
||||||
(type === 'object' ||
|
|
||||||
(type === 'auto' && typeof right === 'object' && typeof user === 'object')) {
|
|
||||||
isEqual = JSON.stringify(user) === JSON.stringify(right);
|
|
||||||
}
|
}
|
||||||
else {
|
|
||||||
isEqual = user == right; // primitive 비교
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isEqual) {
|
// 배열 비교
|
||||||
|
else if (Array.isArray(right) && Array.isArray(user)) {
|
||||||
|
if (right.length === user.length) {
|
||||||
|
if (type === 'number[]' || (type === 'auto' && typeof right[0] === 'number')) {
|
||||||
|
if (right.every((val, idx) => val === user[idx])) {
|
||||||
score = point;
|
score = point;
|
||||||
console.log('작성답안: ', user);
|
}
|
||||||
console.log('>⭕ 정답: ', right);
|
} else if (type === 'string[]' || (type === 'auto' && typeof right[0] === 'string')) {
|
||||||
} else {
|
if (right.every((val, idx) => val === user[idx])) {
|
||||||
console.log('작성답안: ', user);
|
score = point;
|
||||||
console.log('>❌ 오답: ', right);
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 원시값 비교
|
||||||
|
else {
|
||||||
|
if (user == right) {
|
||||||
|
score = point;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(`▶ 작성답안:`, user);
|
||||||
|
if (score > 0) {
|
||||||
|
if (partial)
|
||||||
|
console.log('⭕부분정답:', right, `(${score}/${point})`);
|
||||||
|
else
|
||||||
|
console.log('⭕ 정답 : ', right, `(${score}/${point})`);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
console.log(`❌ 오답:`, right);
|
||||||
}
|
}
|
||||||
|
|
||||||
scoringResult[key] = score;
|
scoringResult[key] = score;
|
||||||
// totalScore += score;
|
|
||||||
return score;
|
return score;
|
||||||
}
|
}
|
||||||
|
|
||||||
const gpdpXmlDoc = gpdpData;
|
const gpdpXmlDoc = gpdpData;
|
||||||
const scoringResult = {};
|
const scoringResult = {};
|
||||||
|
|
||||||
@@ -109,9 +134,10 @@ function getGpdpScore(gpdpData, scoringJson, index) {
|
|||||||
|
|
||||||
// 채점기준표 문항별 분류
|
// 채점기준표 문항별 분류
|
||||||
for (const key in scoringData) {
|
for (const key in scoringData) {
|
||||||
|
console.log(`[❔]문제번호 : [${index}-${key}]`)
|
||||||
|
|
||||||
let ele = scoringData[key].ele;
|
let ele = scoringData[key].ele;
|
||||||
let ele2 = scoringData[key].ele2;
|
let ele2 = scoringData[key].ele2;
|
||||||
let existEle = scoringData[key].existEle;
|
|
||||||
const rightAnswer = scoringData[key].value;
|
const rightAnswer = scoringData[key].value;
|
||||||
const point = scoringData[key].point;
|
const point = scoringData[key].point;
|
||||||
const type = scoringData[key].type;
|
const type = scoringData[key].type;
|
||||||
@@ -127,19 +153,26 @@ function getGpdpScore(gpdpData, scoringJson, index) {
|
|||||||
ele = typeof ele === 'string' ? ele.replace(/{option}/g, option) : ele;
|
ele = typeof ele === 'string' ? ele.replace(/{option}/g, option) : ele;
|
||||||
ele = typeof ele === 'string' ? ele.replace(/{style}/g, style) : ele;
|
ele = typeof ele === 'string' ? ele.replace(/{style}/g, style) : ele;
|
||||||
|
|
||||||
|
// search 값이 undefined 아니면 ele의 {search}부분을 search로 치환
|
||||||
|
/**
|
||||||
|
* JSON파일 곰믹스 5번문항/22번 문항
|
||||||
|
* type : "video" 인 항목들
|
||||||
|
* GPString태그 VID7속성 찾는 xpath구문
|
||||||
|
* CRCUnitArr태그 Name속성 찾는 구문으로 변환
|
||||||
|
* > 멀티라인 텍스트 유사도 판별하기 어려움
|
||||||
|
*/
|
||||||
if (search !== undefined) {
|
if (search !== undefined) {
|
||||||
let result = findSimilarString(gpdpXmlDoc, search, 0.8)
|
let result = findSimilarString(gpdpXmlDoc, search, 0.8);
|
||||||
// xpath 내부 "(큰따옴표) 필터링
|
|
||||||
if (result !== null) {
|
if (result !== null) {
|
||||||
result = result.replace(/"/g, "'");
|
result = result.replace(/"/g, "'");
|
||||||
|
search = result;
|
||||||
|
ele = ele?.replace(/{search}/g, search);
|
||||||
}
|
}
|
||||||
ele = ele.replace(/{search}/g, result);
|
else {
|
||||||
if (existEle !== undefined) {
|
ele = ele?.includes('{search}') ? null : ele;
|
||||||
existEle = existEle.replace(/{search}/g, result);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log(`❔문제번호 : [4-${key}]`)
|
|
||||||
|
|
||||||
if (type === "none") {
|
if (type === "none") {
|
||||||
console.log("❌ 채점하지 않음");
|
console.log("❌ 채점하지 않음");
|
||||||
@@ -162,17 +195,34 @@ function getGpdpScore(gpdpData, scoringJson, index) {
|
|||||||
else if (type === "layer.Exists") {
|
else if (type === "layer.Exists") {
|
||||||
const layerNameList = xpath.select(ele, gpdpXmlDoc);
|
const layerNameList = xpath.select(ele, gpdpXmlDoc);
|
||||||
const layerNames = layerNameList.map(layer => layer.value);
|
const layerNames = layerNameList.map(layer => layer.value);
|
||||||
console.log("🚀 ~ getGpdpScore ~ layerNames:", layerNames);
|
let isMatched = false
|
||||||
|
|
||||||
// userAnswer = layerNames.find(name => name === rightAnswer);
|
// for (const layerName of layerNames) {
|
||||||
for (const layerName of layerNames) {
|
// if (layerName.trim().toLowerCase() === rightAnswer.trim().toLowerCase()) {
|
||||||
if (layerName.trim().toLowerCase() === rightAnswer.trim().toLowerCase()) {
|
// userAnswer = layerName;
|
||||||
userAnswer = layerName;
|
// isMatched = true;
|
||||||
break;
|
// break;
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
// if (isMatched) {
|
||||||
|
// totalScore += compareAndScore(userAnswer, rightAnswer, point, key, scoringResult);
|
||||||
|
// }
|
||||||
|
|
||||||
|
let result = findSimilarString(gpdpXmlDoc, rightAnswer, 0.8);
|
||||||
|
if (result !== null) {
|
||||||
|
userAnswer = result;
|
||||||
|
isMatched = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (isMatched) {
|
||||||
|
totalScore += compareAndScore(userAnswer, rightAnswer, point, key, scoringResult, {
|
||||||
|
type: 'force-correct'
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
totalScore += compareAndScore(userAnswer, rightAnswer, point, key, scoringResult);
|
totalScore += compareAndScore(userAnswer, rightAnswer, point, key, scoringResult);
|
||||||
continue;
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// [1-4] 사진1 > 조정
|
// [1-4] 사진1 > 조정
|
||||||
@@ -245,7 +295,7 @@ function getGpdpScore(gpdpData, scoringJson, index) {
|
|||||||
const vibranceValue = xpath.select1('VibranceValue/@value', effectData)?.value;
|
const vibranceValue = xpath.select1('VibranceValue/@value', effectData)?.value;
|
||||||
// 생동감 옵션값이 프로그램에서 적용한 값에 오차가 발생하는 경우가 있음
|
// 생동감 옵션값이 프로그램에서 적용한 값에 오차가 발생하는 경우가 있음
|
||||||
// 곰픽>XML / 30>29 / 40>39
|
// 곰픽>XML / 30>29 / 40>39
|
||||||
// 설정한 값 그대로 적용되는 경우도 있어서 오차범위를 설정
|
// 설정한 값 그대로 적용되는 경우도 있어서 오차범위 2로 설정
|
||||||
const userValue = parseInt(vibranceValue, 10);
|
const userValue = parseInt(vibranceValue, 10);
|
||||||
const rightValue = parseInt(rightAnswer.option['생동감'], 10);
|
const rightValue = parseInt(rightAnswer.option['생동감'], 10);
|
||||||
|
|
||||||
@@ -277,29 +327,40 @@ function getGpdpScore(gpdpData, scoringJson, index) {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//
|
||||||
// [1-6][1-7]
|
|
||||||
else if (type === "exists") {
|
else if (type === "exists") {
|
||||||
const existsValues = xpath.select(ele, gpdpXmlDoc);
|
const existsValues = xpath.select(ele, gpdpXmlDoc);
|
||||||
|
let isMatched = false;
|
||||||
|
|
||||||
for (const v of existsValues) {
|
for (const v of existsValues) {
|
||||||
|
// 하나라도 일치하면 정답
|
||||||
if (v.value === rightAnswer) {
|
if (v.value === rightAnswer) {
|
||||||
userAnswer = v.value;
|
userAnswer = v.value;
|
||||||
|
isMatched = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// if (isMatched) {
|
||||||
|
|
||||||
|
// }
|
||||||
|
// else {
|
||||||
|
|
||||||
|
// }
|
||||||
totalScore = compareAndScore(userAnswer, rightAnswer, point, key, scoringResult);
|
totalScore = compareAndScore(userAnswer, rightAnswer, point, key, scoringResult);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
else if (type === "shape.size") {
|
// else if (type === "shape.size") {
|
||||||
|
else if (type.includes("size")) {
|
||||||
const items = xpath.select(ele, gpdpXmlDoc);
|
const items = xpath.select(ele, gpdpXmlDoc);
|
||||||
|
let isMatched = false;
|
||||||
|
|
||||||
// 각 Item 요소별 x,y 좌표 시작점과 끝점의 거리를 계산해 정답과 비교
|
// 각 Item 요소별 x,y 좌표 시작점과 끝점의 거리를 계산해 정답과 비교
|
||||||
for (const item of items) {
|
for (const item of items) {
|
||||||
const x1 = Number(xpath.select1('Item[1]/X/@value', item)?.value);
|
const x1 = Number(xpath.select1('Item[1]/X/@value', item)?.value);
|
||||||
const x2 = Number(xpath.select1('Item[last()]/X/@value', item)?.value);
|
|
||||||
const y1 = Number(xpath.select1('Item[1]/Y/@value', item)?.value);
|
const y1 = Number(xpath.select1('Item[1]/Y/@value', item)?.value);
|
||||||
|
|
||||||
|
const x2 = Number(xpath.select1('Item[last()]/X/@value', item)?.value);
|
||||||
const y2 = Number(xpath.select1('Item[last()]/Y/@value', item)?.value);
|
const y2 = Number(xpath.select1('Item[last()]/Y/@value', item)?.value);
|
||||||
|
|
||||||
const width = Math.round(Math.abs(x2 - x1));
|
const width = Math.round(Math.abs(x2 - x1));
|
||||||
@@ -309,26 +370,72 @@ function getGpdpScore(gpdpData, scoringJson, index) {
|
|||||||
width: width,
|
width: width,
|
||||||
height: height,
|
height: height,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// 하나라도 일치하면 정답
|
||||||
|
if (JSON.stringify(userAnswer) == JSON.stringify(rightAnswer)) {
|
||||||
|
isMatched = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
totalScore = compareAndScore(userAnswer, rightAnswer, point, key, scoringResult);
|
totalScore = compareAndScore(userAnswer, rightAnswer, point, key, scoringResult);
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
// [1-8]
|
// [1-8]
|
||||||
else if (type === "shape.color") {
|
else if (type.includes("color")) {
|
||||||
const items = xpath.select(ele, gpdpXmlDoc);
|
const items = xpath.select(ele, gpdpXmlDoc);
|
||||||
|
let normalizedAnswer = null;
|
||||||
|
let isMatched = false;
|
||||||
|
|
||||||
for (const item of items) {
|
for (const item of items) {
|
||||||
const color = parseColorToHex(item.value);
|
if (type.includes('gradient')) {
|
||||||
userAnswer = color;
|
const startColorXpath = scoringData[key].startColor;
|
||||||
|
const endColorXpath = scoringData[key].endColor;
|
||||||
|
|
||||||
|
const startColorRGB = xpath.select1(startColorXpath, item).value;
|
||||||
|
const endColorRGB = xpath.select1(endColorXpath, item).value;
|
||||||
|
|
||||||
|
const startColor = parseColorToHex(startColorRGB);
|
||||||
|
const endColor = parseColorToHex(endColorRGB);
|
||||||
|
|
||||||
|
userAnswer = {
|
||||||
|
startColor: startColor,
|
||||||
|
endColor: endColor,
|
||||||
}
|
}
|
||||||
|
|
||||||
const normalizedRA = rightAnswer.toLowerCase?.();
|
// JSON파일에서 대문자로 입력된 경우 소문자로 변환
|
||||||
totalScore = compareAndScore(userAnswer, normalizedRA, point, key, scoringResult);
|
normalizedAnswer = {
|
||||||
|
startColor: rightAnswer.startColor.toLowerCase(),
|
||||||
|
endColor: rightAnswer.endColor.toLowerCase(),
|
||||||
|
}
|
||||||
|
|
||||||
|
// 하나라도 일치하면 정답
|
||||||
|
if (JSON.stringify(userAnswer) == JSON.stringify(normalizedAnswer)) {
|
||||||
|
isMatched = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// else if (type.includes('shape') || type.includes('text') || type.includes('clipping')) {
|
||||||
|
else {
|
||||||
|
const color = parseColorToHex(item.value);
|
||||||
|
userAnswer = color;
|
||||||
|
normalizedAnswer = rightAnswer.toLowerCase?.();
|
||||||
|
|
||||||
|
// 하나라도 일치하면 정답
|
||||||
|
if (userAnswer === normalizedAnswer) {
|
||||||
|
isMatched = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
totalScore = compareAndScore(userAnswer, normalizedAnswer, point, key, scoringResult);
|
||||||
}
|
}
|
||||||
|
|
||||||
else if (type === 'layer.blend.opacity') {
|
else if (type === 'layer.blend.opacity') {
|
||||||
const layers = xpath.select(ele, gpdpXmlDoc);
|
const layers = xpath.select(ele, gpdpXmlDoc);
|
||||||
|
let isMatched = false;
|
||||||
|
|
||||||
for (const layer of layers) {
|
for (const layer of layers) {
|
||||||
const blendop = xpath.select1('BlendOp/@value', layer).value;
|
const blendop = xpath.select1('BlendOp/@value', layer).value;
|
||||||
@@ -338,10 +445,63 @@ function getGpdpScore(gpdpData, scoringJson, index) {
|
|||||||
BlendOp: blendop,
|
BlendOp: blendop,
|
||||||
Opacity: opacity,
|
Opacity: opacity,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 하나라도 일치하면 정답
|
||||||
|
if (JSON.stringify(userAnswer) == JSON.stringify(rightAnswer)) {
|
||||||
|
isMatched = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
totalScore = compareAndScore(userAnswer, rightAnswer, point, key, scoringResult);
|
totalScore = compareAndScore(userAnswer, rightAnswer, point, key, scoringResult);
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// [5-20]
|
||||||
|
else if (type === 'shadow') {
|
||||||
|
const shapes = xpath.select(ele, gpdpXmlDoc);
|
||||||
|
|
||||||
|
for (const shape of shapes) {
|
||||||
|
// 그림자 설정 여부
|
||||||
|
const shadowExists = xpath.select1('contains(draw_type/@value, "Shadow")', shape);
|
||||||
|
// Shadow 옵션이 있다면
|
||||||
|
if (shadowExists) {
|
||||||
|
// 두께
|
||||||
|
const width = xpath.select1('shadow_width/@value', shape).value;
|
||||||
|
// 거리
|
||||||
|
const distance = xpath.select1('shadow_distance/@value', shape).value;
|
||||||
|
// 분산도
|
||||||
|
const blur = xpath.select1('shadow_blur/@value', shape).value;
|
||||||
|
// 각도
|
||||||
|
const angle = xpath.select1('shadow_angle/@value', shape).value;
|
||||||
|
|
||||||
|
userAnswer = {
|
||||||
|
shadow: shadowExists,
|
||||||
|
width: width,
|
||||||
|
distance: distance,
|
||||||
|
blur: blur,
|
||||||
|
angle: angle,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
userAnswer = {
|
||||||
|
shadow: shadowExists,
|
||||||
|
width: null,
|
||||||
|
distance: null,
|
||||||
|
blur: null,
|
||||||
|
angle: null,
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
console.log("🚀 ~ userAnswer:", userAnswer);
|
||||||
|
console.log("🚀 ~ rightAnswer : ", rightAnswer)
|
||||||
|
totalScore += compareAndScore(userAnswer, rightAnswer, point, key, scoringResult, {
|
||||||
|
partial: true
|
||||||
|
})
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
else if (type == "boolean") {
|
else if (type == "boolean") {
|
||||||
const items = xpath.select(ele, gpdpXmlDoc);
|
const items = xpath.select(ele, gpdpXmlDoc);
|
||||||
|
|
||||||
@@ -562,7 +722,7 @@ function getGpdpScore(gpdpData, scoringJson, index) {
|
|||||||
totalScore += result.length > 0 ? point : 0;
|
totalScore += result.length > 0 ? point : 0;
|
||||||
scoringResult[key] = result.length > 0 ? point : 0;
|
scoringResult[key] = result.length > 0 ? point : 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
scoringResult['총점'] = totalScore;
|
scoringResult['총점'] = totalScore;
|
||||||
return scoringResult;
|
return scoringResult;
|
||||||
}
|
}
|
||||||
|
|||||||
8
package-lock.json
generated
8
package-lock.json
generated
@@ -10,6 +10,7 @@
|
|||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"jsonpath": "^1.1.1",
|
"jsonpath": "^1.1.1",
|
||||||
|
"loadsh": "^0.0.4",
|
||||||
"psd": "^3.4.0",
|
"psd": "^3.4.0",
|
||||||
"string-similarity": "^4.0.4",
|
"string-similarity": "^4.0.4",
|
||||||
"xlsx": "^0.18.5",
|
"xlsx": "^0.18.5",
|
||||||
@@ -209,6 +210,13 @@
|
|||||||
"node": ">= 0.8.0"
|
"node": ">= 0.8.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/loadsh": {
|
||||||
|
"version": "0.0.4",
|
||||||
|
"resolved": "https://registry.npmjs.org/loadsh/-/loadsh-0.0.4.tgz",
|
||||||
|
"integrity": "sha512-U+wLL8InpfRalWrr+0SuhWgGt10M4OyAk6G8xCYo2rwpiHtxZkWiFpjei0vO463ghW8LPCdhqQxXlMy2qicAEw==",
|
||||||
|
"deprecated": "This is a typosquat on the popular Lodash package. This is not maintained nor is the original Lodash package.",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/lodash": {
|
"node_modules/lodash": {
|
||||||
"version": "4.17.21",
|
"version": "4.17.21",
|
||||||
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
|
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
|
||||||
|
|||||||
@@ -12,6 +12,7 @@
|
|||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"jsonpath": "^1.1.1",
|
"jsonpath": "^1.1.1",
|
||||||
|
"loadsh": "^0.0.4",
|
||||||
"psd": "^3.4.0",
|
"psd": "^3.4.0",
|
||||||
"string-similarity": "^4.0.4",
|
"string-similarity": "^4.0.4",
|
||||||
"xlsx": "^0.18.5",
|
"xlsx": "^0.18.5",
|
||||||
|
|||||||
@@ -9,8 +9,8 @@ const { DOMParser } = require('xmldom');
|
|||||||
const findSimilarString = require('./findSimilarString');
|
const findSimilarString = require('./findSimilarString');
|
||||||
const getGpdpScore = require('./gpdpScoring.js');
|
const getGpdpScore = require('./gpdpScoring.js');
|
||||||
const getToday = require('./getToday.js');
|
const getToday = require('./getToday.js');
|
||||||
const { userInfo } = require('os');
|
// const { userInfo } = require('os');
|
||||||
const { get } = require('http');
|
// const { get } = require('http');
|
||||||
const todayDate = getToday();
|
const todayDate = getToday();
|
||||||
|
|
||||||
const examRound = '2504';
|
const examRound = '2504';
|
||||||
|
|||||||
2
z.xbook
2
z.xbook
@@ -1 +1 @@
|
|||||||
[{"kind":2,"language":"xpath","value":"//Layer[Name[@value='Tracking']]/Effects/Item/Name/@value"},{"kind":2,"language":"xpath","value":"sum(//CRTrackList[@Name='텍스트' or @Name='비디오2']/CRTrackClip[@ClipIndex=0]/preceding-sibling::CRTrackClip/@Length)"},{"kind":2,"language":"xpath","value":"//CROwneUnit/CRCUnitArr[@Name=\"아름다운 꽃 축제 (Happy Flower Festival)\"]/@Name"},{"kind":2,"language":"xpath","value":"//Layer[Name[@value='{layer}']]/Effects/Item[EffectData/{option}]/Name/@value | //Layer[Name[@value='{layer}']]/Effects/Item/EffectData/{option}/@value"},{"kind":2,"language":"xpath","value":"//CRTrackList[@Name='비디오1']/CRTrackClip[@ClipIndex='3']/@Length"},{"kind":2,"language":"xpath","value":"//CRTransFilter[@ClipIndex=count(//CRTrackList[@Name='비디오1']/CRTrackClip[@ClipIndex=count(//CRClip[@Path='이미지3.jpg']/preceding-sibling::CRClip | //CRClip[@Type='11']/CRCUnitArr[@Path='이미지3.jpg']/../preceding-sibling::CRClip)][1]/preceding-sibling::CRTrackClip)][@Type='2']/@*[name()='ID' or name()='Range' or name()='Type']"},{"kind":2,"language":"xpath","value":"//Document/Width/@value | //Document/Height/@value"},{"kind":2,"language":"xpath","value":"//Layer[Name[@value='Flower']]/Effects/Item[EffectData/VibranceValue]/(Name/@value | EffectData/VibranceValue/@value)"},{"kind":2,"language":"xpath","value":"//Layer[Name[@value='Flower']]/Effects/Item[EffectData/VibranceValue]/Name/@value | //Layer[Name[@value='Flower']]/Effects/Item/EffectData/VibranceValue/@value"}]
|
[{"kind":2,"language":"xpath","value":"//Layer[Name[@value='Tracking']]/Effects/Item/Name/@value"},{"kind":2,"language":"xpath","value":"sum(//CRTrackList[@Name='텍스트' or @Name='비디오2']/CRTrackClip[@ClipIndex=0]/preceding-sibling::CRTrackClip/@Length)"},{"kind":2,"language":"xpath","value":"//CROwneUnit/CRCUnitArr[@Name=\"아름다운 꽃 축제 (Happy Flower Festival)\"]/@Name"},{"kind":2,"language":"xpath","value":"//Layer[Name[@value='{layer}']]/Effects/Item[EffectData/{option}]/Name/@value | //Layer[Name[@value='{layer}']]/Effects/Item/EffectData/{option}/@value"},{"kind":2,"language":"xpath","value":"//CRTrackList[@Name='비디오1']/CRTrackClip[@ClipIndex='3']/@Length"},{"kind":2,"language":"xpath","value":"//CRTransFilter[@ClipIndex=count(//CRTrackList[@Name='비디오1']/CRTrackClip[@ClipIndex=count(//CRClip[@Path='이미지3.jpg']/preceding-sibling::CRClip | //CRClip[@Type='11']/CRCUnitArr[@Path='이미지3.jpg']/../preceding-sibling::CRClip)][1]/preceding-sibling::CRTrackClip)][@Type='2']/@*[name()='ID' or name()='Range' or name()='Type']"},{"kind":2,"language":"xpath","value":"//Layer[MaskOpType/@value='Clipping']"},{"kind":2,"language":"xpath","value":"//Layer//Shape[shape_type/@value='RECTANGLE' and contains(draw_type/@value, 'Outline')]/primary_color/@value"},{"kind":2,"language":"xpath","value":"//Layer[Name[@value='Flower']]/Effects/Item[EffectData/VibranceValue]/Name/@value | //Layer[Name[@value='Flower']]/Effects/Item/EffectData/VibranceValue/@value"}]
|
||||||
Binary file not shown.
Reference in New Issue
Block a user