Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
17 changes: 17 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
@@ -1 +1,18 @@
# javascript-lotto-precourse
### 🚀 기능 구현 목록
- [x] 로또 구입 금액을 입력받는다.
- 로또 1장의 가격은 1000원이다.
- [x] 구입 금액 만큼의 로또 번호를 생성한다.
- 중복되지 않는 1~45 범위의 숫자를 6개와 보너스 번호를 뽑는다.
- [x] 당첨 번호와 보너스 번호를 입력받는다.
- [x] 당첨 내역을 출력한다.
- 1등: 6개 번호 일치 / 2,000,000,000원
- 2등: 5개 번호 + 보너스 번호 일치 / 30,000,000원
- 3등: 5개 번호 일치 / 1,500,000원
- 4등: 4개 번호 일치 / 50,000원
- 5등: 3개 번호 일치 / 5,000원
- [x] 수익률을 출력한다.
- 예외처리
- [x] 로또 구입 금액은 1000으로 나누어 떨어지는 양의 정수이다.
- [x] 입력받은 당첨번호와 보너스 번호의 범위는 1~45이다.
- [x] 입력받은 당첨번호와 보너스 번호는 겹치지 않는다.
18 changes: 17 additions & 1 deletion src/App.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,21 @@
import { offerLottoSheet } from "./LottoMachine.js";
import { calculateProfit, calculateWinning, getWinningPrice } from "./LottoCalculator.js";
import { getBonusNumber, getLottoNumber, getMoney, printLottoNumbers, printProfit, printResult } from "./View.js";

class App {
async run() {}
async run() {
const lottoQuantity = await getMoney();
const userLotto = await offerLottoSheet(lottoQuantity);
await printLottoNumbers(lottoQuantity, userLotto);
const winningNumbers = await getLottoNumber();
const bonusNumber = await getBonusNumber(winningNumbers);
let winningArray = [winningNumbers, bonusNumber];
const winningResult = await calculateWinning(winningArray, userLotto);
await printResult(winningResult);
const winningPrice = await getWinningPrice(winningResult);
const profit = await calculateProfit(lottoQuantity*1000, winningPrice);
await printProfit(profit);
}
}

export default App;
7 changes: 7 additions & 0 deletions src/Constant.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
export const PRIZE_TABLE = {
3: "5,000",
4: "50,000",
5: "1,500,000",
"5+bonus": "30,000,000",
6: "2,000,000,000"
};
12 changes: 10 additions & 2 deletions src/Lotto.js
Original file line number Diff line number Diff line change
Expand Up @@ -10,9 +10,17 @@ class Lotto {
if (numbers.length !== 6) {
throw new Error("[ERROR] 로또 번호는 6개여야 합니다.");
}
}
numbers.forEach((number) => {
if(number < 1 || number > 45){
throw new Error("[ERROR] 로또 번호는 1부터 45 사이의 숫자여야 합니다.");
}
})

// TODO: 추가 기능 구현
const numberSet = new Set(numbers);
if(numberSet.size !== 6){
throw new Error("[ERROR] 로또 번호는 중복되지 않는 6개의 숫자입니다.");
}
}
}

export default Lotto;
37 changes: 37 additions & 0 deletions src/LottoCalculator.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
import { PRIZE_TABLE } from "./Constant.js";

export const calculateWinning = (winningNumber, userNumbers) => {
const [winningArray, bonus] = winningNumber;

const prizeCounts = { 3: 0, 4: 0, 5: 0, '5+bonus': 0, 6: 0 };

userNumbers.forEach(numbers => {
const matchingCount = numbers.filter(number => winningArray.includes(number)).length;

if (matchingCount === 5 && numbers.includes(bonus)) {
prizeCounts['5+bonus'] += 1;
} else if (matchingCount >= 3) {
prizeCounts[matchingCount] += 1;
}
});

return prizeCounts;
}

export const getWinningPrice = (result) => {
let totalPrize = 0;
for (const [key, count] of Object.entries(result)) {
if (count > 0) {
const prize = parseInt(PRIZE_TABLE[key].replace(/,/g, ''), 10);
totalPrize += prize * count;
}
}

return totalPrize;
}

export const calculateProfit = (price, winningPrice) => {
const profit = winningPrice/price * 100;

return profit.toFixed(1);
}
16 changes: 16 additions & 0 deletions src/LottoMachine.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
import { MissionUtils } from "@woowacourse/mission-utils";

export const createLottoNumbers = () => {
let numberSet = MissionUtils.Random.pickUniqueNumbersInRange(1, 45, 6);
return [...numberSet].sort((a, b) => a - b);
}

export const offerLottoSheet = (quantity) => {
let sheet = [];

for(let i=0; i<quantity; i++){
sheet.push(createLottoNumbers());
}

return sheet;
}
8 changes: 8 additions & 0 deletions src/Validator.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
export const checkValidation = (number, numberSet) => {
if(number < 1 || number > 45){
throw new Error("[ERROR] 로또 번호는 1부터 45 사이의 숫자여야 합니다.");
}
if(numberSet.has(number)){
throw new Error("[ERROR] 보너스 숫자는 로또 번호와 중복되지 않습니다.");
}
}
73 changes: 73 additions & 0 deletions src/View.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,73 @@
import { MissionUtils } from "@woowacourse/mission-utils";
import { checkValidation } from "./Validator.js";
import Lotto from "./Lotto.js";
import { PRIZE_TABLE } from "./Constant.js";

export const getMoney = async () => {
const money = await MissionUtils.Console.readLineAsync("구입금액을 입력해 주세요\n");
if(isNaN(money.trim())){
throw new Error("[ERROR] 구입금액을 숫자로 입력해 주세요.");
}

if(+money < 1000){
throw new Error("[ERROR] 1000원부터 구매가 가능합니다.");
}

if(money%1000 !== 0){
throw new Error("[ERROR] 1000원 단위로 구매가 가능합니다.");
}

return money/1000;
}

export const getLottoNumber = async () => {
try {
const inputNumber = await MissionUtils.Console.readLineAsync("당첨 번호를 입력해 주세요.\n");
const winningNumber = inputNumber.split(',').map((number) => +(number.trim()));

new Lotto(winningNumber);

let lottoArray = winningNumber.sort((a, b) => a - b);

return lottoArray;
} catch (error) {
throw new Error("[ERROR] 정확한 로또 번호를 입력해주세요.");
}
}

export const getBonusNumber = async(numbers) => {
let bonusNumber = await MissionUtils.Console.readLineAsync("보너스 번호를 입력해 주세요.\n");
bonusNumber = +bonusNumber.trim();

checkValidation(bonusNumber, new Set(numbers));
return bonusNumber;
}

export const printLottoNumbers = (number, lottoNumbers) => {
MissionUtils.Console.print(`${number}개를 구매했습니다.`);
lottoNumbers.forEach((lottoNumber) => {
const formattedLottoNumbers = `[${lottoNumber.join(', ')}]`;
MissionUtils.Console.print(formattedLottoNumbers);
})
}

export const printResult = async (result) => {
MissionUtils.Console.print(`
당첨 통계
---
`);
Object.entries(result).forEach(([key, count]) => {
const prize = PRIZE_TABLE[key];
let bonus = '';
if(key === '5+bonus'){
MissionUtils.Console.print(`5개 일치, 보너스 볼 일치 (${prize}원) - ${count}개`);
}
else { MissionUtils.Console.print(`${key}개 일치 (${prize}원) - ${count}개`); }
});

}

export const printProfit = (profit) => {
MissionUtils.Console.print(`총 수익률은 ${profit}%입니다.`)

}