input 계산기 수정 #317

Merged
ysCha merged 1 commits from dev into dev-deploy 2025-08-29 17:11:07 +09:00
2 changed files with 70 additions and 106 deletions

View File

@ -4,34 +4,37 @@ import '@/styles/calc.scss'
export const CalculatorInput = ({ value, onChange, label, options = {}, id, className = 'calculator-input', readOnly = false }) => { export const CalculatorInput = ({ value, onChange, label, options = {}, id, className = 'calculator-input', readOnly = false }) => {
const [showKeypad, setShowKeypad] = useState(false) const [showKeypad, setShowKeypad] = useState(false)
const [displayValue, setDisplayValue] = useState(value || '0')
const [hasOperation, setHasOperation] = useState(false)
const calculatorRef = useRef(createCalculator(options)) const calculatorRef = useRef(createCalculator(options))
const containerRef = useRef(null) const containerRef = useRef(null)
const inputRef = useRef(null) // input ref const inputRef = useRef(null)
// Sync displayValue with value prop
useEffect(() => {
setDisplayValue(value || '0')
}, [value])
// //
useEffect(() => { useEffect(() => {
const handleClickOutside = (event) => { const handleClickOutside = (event) => {
if (containerRef.current && !containerRef.current.contains(event.target)) { if (containerRef.current && !containerRef.current.contains(event.target)) {
setShowKeypad(false) setShowKeypad(false)
if (/[+\-×÷]/.test(value)) { if (hasOperation) {
const newValue = calculatorRef.current.clear() // If there's an operation in progress, compute the result when losing focus
onChange(newValue) handleCompute()
} }
} }
} }
document.addEventListener('mousedown', handleClickOutside) document.addEventListener('mousedown', handleClickOutside)
return () => document.removeEventListener('mousedown', handleClickOutside) return () => document.removeEventListener('mousedown', handleClickOutside)
}, [value, onChange]) }, [value, onChange, hasOperation])
// state
const [displayValue, setDisplayValue] = useState(value || '0')
const [hasOperation, setHasOperation] = useState(false)
// //
const handleNumber = (num) => { const handleNumber = (num) => {
const calculator = calculatorRef.current const calculator = calculatorRef.current
let newValue = '' let newDisplayValue = ''
if (hasOperation) { if (hasOperation) {
// //
@ -41,30 +44,34 @@ export const CalculatorInput = ({ value, onChange, label, options = {}, id, clas
} else { } else {
calculator.currentOperand = (calculator.currentOperand || '') + num calculator.currentOperand = (calculator.currentOperand || '') + num
} }
newValue = calculator.previousOperand + calculator.operation + calculator.currentOperand newDisplayValue = calculator.previousOperand + calculator.operation + calculator.currentOperand
setDisplayValue(newValue) setDisplayValue(newDisplayValue)
onChange(calculator.currentOperand)
} else { } else {
// //
if (value === '0' || calculator.shouldResetDisplay) { if (displayValue === '0' || calculator.shouldResetDisplay) {
calculator.currentOperand = num.toString() calculator.currentOperand = num.toString()
calculator.shouldResetDisplay = false calculator.shouldResetDisplay = false
newDisplayValue = calculator.currentOperand
setDisplayValue(newDisplayValue)
if (!hasOperation) {
onChange(calculator.currentOperand)
}
} else { } else {
calculator.currentOperand = (calculator.currentOperand || '') + num calculator.currentOperand = (calculator.currentOperand || '') + num
newDisplayValue = calculator.currentOperand
setDisplayValue(newDisplayValue)
if (!hasOperation) {
onChange(newDisplayValue)
}
} }
newValue = calculator.currentOperand
setDisplayValue(newValue)
onChange(newValue)
} }
// // ( )
requestAnimationFrame(() => { requestAnimationFrame(() => {
if (inputRef.current) { if (inputRef.current) {
inputRef.current.focus() inputRef.current.focus()
const len = newValue.length const len = newDisplayValue.length
inputRef.current.setSelectionRange(len, len) inputRef.current.setSelectionRange(len, len)
// Ensure focus is maintained
inputRef.current.focus()
} }
}) })
} }
@ -72,67 +79,42 @@ export const CalculatorInput = ({ value, onChange, label, options = {}, id, clas
// //
const handleOperation = (operation) => { const handleOperation = (operation) => {
const calculator = calculatorRef.current const calculator = calculatorRef.current
let newDisplayValue = ''
// ( ) // ( )
if (!calculator.currentOperand && calculator.previousOperand) { if (!calculator.currentOperand && calculator.previousOperand) {
calculator.operation = operation calculator.operation = operation
const newValue = calculator.previousOperand + operation newDisplayValue = calculator.previousOperand + operation
setDisplayValue(newValue) setDisplayValue(newDisplayValue)
setHasOperation(true) setHasOperation(true)
// } else if (hasOperation) {
requestAnimationFrame(() => {
if (inputRef.current) {
inputRef.current.focus()
const len = newValue.length
inputRef.current.setSelectionRange(len, len)
inputRef.current.focus()
}
})
return
}
if (hasOperation) {
// , // ,
const result = calculator.compute() const result = calculator.compute()
if (result !== undefined) { if (result !== undefined) {
calculator.previousOperand = result.toString() calculator.previousOperand = result.toString()
calculator.operation = operation calculator.operation = operation
calculator.currentOperand = '' calculator.currentOperand = ''
const newValue = result.toString() + operation newDisplayValue = calculator.previousOperand + operation
setDisplayValue(newValue) setDisplayValue(newDisplayValue)
setHasOperation(true)
onChange(result.toString())
//
requestAnimationFrame(() => {
if (inputRef.current) {
inputRef.current.focus()
const len = newValue.length
inputRef.current.setSelectionRange(len, len)
inputRef.current.focus()
}
})
} }
} else { } else {
// //
calculator.previousOperand = displayValue calculator.previousOperand = calculator.currentOperand || '0'
calculator.operation = operation calculator.operation = operation
calculator.currentOperand = '' calculator.currentOperand = ''
const newValue = displayValue + operation
setDisplayValue(newValue)
setHasOperation(true) setHasOperation(true)
onChange(displayValue) newDisplayValue = calculator.previousOperand + operation
setDisplayValue(newDisplayValue)
//
requestAnimationFrame(() => {
if (inputRef.current) {
inputRef.current.focus()
const len = newValue.length
inputRef.current.setSelectionRange(len, len)
inputRef.current.focus()
}
})
} }
// ( )
requestAnimationFrame(() => {
if (inputRef.current) {
inputRef.current.focus()
const len = newDisplayValue.length
inputRef.current.setSelectionRange(len, len)
}
})
} }
// AC // AC
@ -157,44 +139,26 @@ export const CalculatorInput = ({ value, onChange, label, options = {}, id, clas
// //
const handleCompute = () => { const handleCompute = () => {
if (!hasOperation) return
const calculator = calculatorRef.current const calculator = calculatorRef.current
if (!hasOperation || !calculator.currentOperand) return
// 0
if (!calculator.currentOperand) {
calculator.currentOperand = '0'
}
//
const result = calculator.compute() const result = calculator.compute()
if (result !== undefined) {
// const resultStr = result.toString()
if (result === undefined || result === null) { setDisplayValue(resultStr)
console.error('계산 결과가 유효하지 않습니다.') setHasOperation(false)
return // Only call onChange with the final result
onChange(resultStr)
// ( )
requestAnimationFrame(() => {
if (inputRef.current) {
inputRef.current.focus()
const len = resultStr.length
inputRef.current.setSelectionRange(len, len)
}
})
} }
//
const resultStr = result.toString()
setDisplayValue(resultStr)
setHasOperation(false)
onChange(resultStr)
// ( )
calculator.clear()
calculator.previousOperand = resultStr
//
requestAnimationFrame(() => {
if (inputRef.current) {
inputRef.current.focus()
const len = resultStr.length
inputRef.current.setSelectionRange(len, len)
// Ensure focus is maintained
inputRef.current.focus()
}
})
} }
// DEL // DEL

View File

@ -353,23 +353,23 @@ export default function PlacementShapeSetting({ id, pos = { x: 50, y: 180 }, pla
onChange={(value) => { onChange={(value) => {
if (index === 0) { if (index === 0) {
const num = value === '' ? '' : Number(value) const num = value === '' ? '' : Number(value)
setCurrentRoof({ setCurrentRoof(prev => ({
...currentRoof, ...prev,
pitch: num === '' ? '' : num, pitch: num === '' ? '' : num,
angle: num === '' ? '' : getDegreeByChon(num), angle: num === '' ? '' : getDegreeByChon(num),
}) }))
} else { } else {
const num = value === '' ? '' : Number(value) const num = value === '' ? '' : Number(value)
setCurrentRoof({ setCurrentRoof( prev => ({
...currentRoof, ...prev,
pitch: num === '' ? '' : getChonByDegree(num), pitch: num === '' ? '' : getChonByDegree(num),
angle: num === '' ? '' : num, angle: num === '' ? '' : num,
}) }))
} }
}} }}
options={{ options={{
allowNegative: false, allowNegative: false,
allowDecimal: (index !== 0), allowDecimal: false //(index !== 0),
}} }}
/> />
</div> </div>