fix:修复七段数码管的显示问题

This commit is contained in:
alivender 2025-07-20 10:33:57 +08:00
parent d30712d0f6
commit 27c8ceb1db
1 changed files with 134 additions and 11 deletions

View File

@ -79,6 +79,7 @@ interface SevenSegmentDisplayProps {
size?: number; size?: number;
color?: string; color?: string;
AFTERGLOW_BUFFER_SIZE?: number; // AFTERGLOW_BUFFER_SIZE?: number; //
AFTERGLOW_DURATION?: number; //
pins?: { pins?: {
pinId: string; pinId: string;
constraint: string; constraint: string;
@ -92,7 +93,7 @@ const props = withDefaults(defineProps<SevenSegmentDisplayProps>(), {
size: 1, size: 1,
color: "red", color: "red",
AFTERGLOW_BUFFER_SIZE: 1, // 100 AFTERGLOW_BUFFER_SIZE: 1, // 100
AFTERGLOW_UPDATE_INTERVAL: 5, // 2 AFTERGLOW_DURATION: 2000, // 500
cathodeType: "common", // cathodeType: "common", //
pins: () => [ pins: () => [
{ pinId: "a", constraint: "", x: 10, y: 170 }, // a { pinId: "a", constraint: "", x: 10, y: 170 }, // a
@ -149,7 +150,7 @@ const afterglowBuffers = ref<Record<string, boolean[]>>({
}); });
// 10 // 10
const STABLE_THRESHOLD = 10; const STABLE_THRESHOLD = 3;
// //
const stableSegmentStates = ref({ const stableSegmentStates = ref({
@ -175,10 +176,54 @@ const segmentStableCounters = ref<Record<string, number>>({
dp: 0, dp: 0,
}); });
// //
const afterglowStates = ref({
a: false,
b: false,
c: false,
d: false,
e: false,
f: false,
g: false,
dp: false,
});
//
const afterglowTimers = ref<Record<string, number | null>>({
a: null,
b: null,
c: null,
d: null,
e: null,
f: null,
g: null,
dp: null,
});
//
const AFTERGLOW_DURATION = computed(() => props.AFTERGLOW_DURATION || 500);
// COM
const currentComActive = ref(false); // false
//
const isInAfterglowMode = ref(false);
//
function isSegmentActive( function isSegmentActive(
segment: "a" | "b" | "c" | "d" | "e" | "f" | "g" | "dp", segment: "a" | "b" | "c" | "d" | "e" | "f" | "g" | "dp",
): boolean { ): boolean {
// 使
if (isInAfterglowMode.value) {
return afterglowStates.value[segment];
}
// COM
if (!currentComActive.value) {
return false;
}
// 使
return stableSegmentStates.value[segment]; return stableSegmentStates.value[segment];
} }
@ -186,21 +231,45 @@ function isSegmentActive(
function updateSegmentStates() { function updateSegmentStates() {
// COM // COM
const comPin = props.pins.find((p) => p.pinId === "COM"); const comPin = props.pins.find((p) => p.pinId === "COM");
let comActive = true; let comActive = false; //
if (comPin && comPin.constraint) { if (comPin && comPin.constraint) {
const comState = getConstraintState(comPin.constraint); const comState = getConstraintState(comPin.constraint);
if (props.cathodeType === "anode") { if (props.cathodeType === "anode") {
// anodeCOM // COM
comActive = comState !== "high"; comActive = comState === "low";
} else {
// COM
comActive = comState === "low";
} }
// } else if (!comPin || !comPin.constraint) {
// COMCOM
comActive = true;
} }
// COM // COM
if (comActive) { const comStateChanged = currentComActive.value !== comActive;
updateAfterglowBuffers(); currentComActive.value = comActive;
// COM
if (comStateChanged && !comActive) {
enterAfterglowMode();
return; //
} }
// COM退
if (comStateChanged && comActive) {
exitAfterglowMode();
}
// COM
if (!comActive || isInAfterglowMode.value) {
return;
}
// COM
updateAfterglowBuffers();
// segmentStates // segmentStates
for (const pin of props.pins) { for (const pin of props.pins) {
if (["a", "b", "c", "d", "e", "f", "g", "dp"].includes(pin.pinId)) { if (["a", "b", "c", "d", "e", "f", "g", "dp"].includes(pin.pinId)) {
@ -215,7 +284,8 @@ function updateSegmentStates() {
} else { } else {
newState = pinState === "low"; newState = pinState === "low";
} }
segmentStates.value[pin.pinId as keyof typeof segmentStates.value] = newState && comActive; // COM
segmentStates.value[pin.pinId as keyof typeof segmentStates.value] = newState;
} }
} }
@ -255,6 +325,49 @@ function updateAfterglowBuffers() {
} }
} }
//
function enterAfterglowMode() {
isInAfterglowMode.value = true;
//
for (const segmentId of ["a", "b", "c", "d", "e", "f", "g", "dp"]) {
const typedSegmentId = segmentId as keyof typeof stableSegmentStates.value;
afterglowStates.value[typedSegmentId] = stableSegmentStates.value[typedSegmentId];
// 退
if (afterglowTimers.value[segmentId]) {
clearTimeout(afterglowTimers.value[segmentId]!);
}
afterglowTimers.value[segmentId] = setTimeout(() => {
afterglowStates.value[typedSegmentId] = false;
//
const allSegmentsOff = Object.values(afterglowStates.value).every(state => !state);
if (allSegmentsOff) {
exitAfterglowMode();
}
}, AFTERGLOW_DURATION.value);
}
}
// 退
function exitAfterglowMode() {
isInAfterglowMode.value = false;
//
for (const segmentId of ["a", "b", "c", "d", "e", "f", "g", "dp"]) {
if (afterglowTimers.value[segmentId]) {
clearTimeout(afterglowTimers.value[segmentId]!);
afterglowTimers.value[segmentId] = null;
}
//
const typedSegmentId = segmentId as keyof typeof afterglowStates.value;
afterglowStates.value[typedSegmentId] = false;
}
}
// //
function onConstraintChange(constraint: string, level: string) { function onConstraintChange(constraint: string, level: string) {
const affectedPin = props.pins.find((pin) => pin.constraint === constraint); const affectedPin = props.pins.find((pin) => pin.constraint === constraint);
@ -276,6 +389,15 @@ onMounted(() => {
onConstraintStateChange(onConstraintChange); onConstraintStateChange(onConstraintChange);
}); });
onUnmounted(() => {
//
for (const segmentId of ["a", "b", "c", "d", "e", "f", "g", "dp"]) {
if (afterglowTimers.value[segmentId]) {
clearTimeout(afterglowTimers.value[segmentId]!);
}
}
});
// //
defineExpose({ defineExpose({
updateSegmentStates, updateSegmentStates,
@ -307,6 +429,7 @@ export function getDefaultProps() {
size: 1, size: 1,
color: "red", color: "red",
cathodeType: "common", cathodeType: "common",
AFTERGLOW_DURATION: 500, // 500
pins: [ pins: [
{ pinId: "a", constraint: "", x: 10, y: 170 }, { pinId: "a", constraint: "", x: 10, y: 170 },
{ pinId: "b", constraint: "", x: 25 - 1, y: 170 }, { pinId: "b", constraint: "", x: 25 - 1, y: 170 },