[프로그래머스] [PCCP 기출문제] 1번 / 동영상 재생기 JavaScript

·2024년 9월 13일

문제

당신은 동영상 재생기를 만들고 있습니다. 당신의 동영상 재생기는 10초 전으로 이동, 10초 후로 이동, 오프닝 건너뛰기 3가지 기능을 지원합니다. 각 기능이 수행하는 작업은 다음과 같습니다.

  • 10초 전으로 이동: 사용자가 "prev" 명령을 입력할 경우 동영상의 재생 위치를 현재 위치에서 10초 전으로 이동합니다. 현재 위치가 10초 미만인 경우 영상의 처음 위치로 이동합니다. 영상의 처음 위치는 0분 0초입니다.
  • 10초 후로 이동: 사용자가 "next" 명령을 입력할 경우 동영상의 재생 위치를 현재 위치에서 10초 후로 이동합니다. 동영상의 남은 시간이 10초 미만일 경우 영상의 마지막 위치로 이동합니다. 영상의 마지막 위치는 동영상의 길이와 같습니다.
  • 오프닝 건너뛰기: 현재 재생 위치가 오프닝 구간(op_start ≤ 현재 재생 위치 ≤ op_end)인 경우 자동으로 오프닝이 끝나는 위치로 이동합니다.

동영상의 길이를 나타내는 문자열 video_len, 기능이 수행되기 직전의 재생위치를 나타내는 문자열 pos, 오프닝 시작 시각을 나타내는 문자열 op_start, 오프닝이 끝나는 시각을 나타내는 문자열 op_end, 사용자의 입력을 나타내는 1차원 문자열 배열 commands가 매개변수로 주어집니다. 이때 사용자의 입력이 모두 끝난 후 동영상의 위치를 "mm:ss" 형식으로 return 하도록 solution 함수를 완성해 주세요.

제한사항

video_len의 길이 = pos의 길이 = op_start의 길이 = op_end의 길이 = 5

  • video_len, pos, op_start, op_end는 "mm:ss" 형식으로 mm분 ss초를 나타냅니다.
  • 0 ≤ mm ≤ 59
  • 0 ≤ ss ≤ 59
  • 분, 초가 한 자리일 경우 0을 붙여 두 자리로 나타냅니다.
  • 비디오의 현재 위치 혹은 오프닝이 끝나는 시각이 동영상의 범위 밖인 경우는 주어지지 않습니다.
  • 오프닝이 시작하는 시각은 항상 오프닝이 끝나는 시각보다 전입니다.

1 ≤ commands의 길이 ≤ 100

  • commands의 원소는 "prev" 혹은 "next"입니다.
  • "prev"는 10초 전으로 이동하는 명령입니다.
  • "next"는 10초 후로 이동하는 명령입니다.

입력

video_len : "34:33"
pos op_start : "13:00"
op_end : "00:55"
commands : "02:55"

출력

"13:00"

내가 했던 풀이 방법

  1. 각각의 시간 정보를 ":"를 기준으로 분할해 Number형으로 저장해준다.
  2. 현재 영상이 오프닝 도중인지를 체크하기 위해 start/end를 초로 변환해준다. (사실 모든 시간 정보를 초로 변환하고 풀이하면 더 쉽게 풀이 가능하지만, 이후 코드를 분/초으로 이미 구현해놔서 start/end 해당 부분만 초로 변환해서 풀이했다.)
  3. isOpening 함수에서는 오프닝 시작 초와 오프닝 엔딩 초 안에 현재 시간이 존재할 경우 true를 반환한다.
  4. commands를 하나씩 체크할 때마다 현재 시간이 오프닝 도중이라면 오프닝 엔딩 시간으로 변환해준다.
  5. 명령어가 next라면 현재 시간 중 "초"를 10초 추가해준다. 만약 59초보다 크다면, 분을 1분 증가시켜주고, 현재 초에 60초를 빼주어 시간 형식에 맞춰지도록 변환해준다. 만약, 모든 처리가 끝난 뒤에 현재 시간의 "분"이 전체 영상 "분"보다 크거나 "분"은 같으나 "초"가 전체 영상 "초"보다 클 경우, 전체 영상 시간으로 현재 시간을 저장해준다.
  6. 명령어가 prev라면 반대로 현재 시간 중 "초"를 10초 빼준다. 만약 0초보다 작다면 분을 1분 감소시켜주고, 현재 초에 60초를 더해주어 시간 형식에 맞춰지도록 변환해준다. 만약, 모든 처리가 끝난 뒤에 현재 시간의 "분"이 음수라면 현재 시간을 0분 0초로 저장해준다.
  7. 해당 명령어를 수행한 이후에도 오프닝 도중이라면 오프닝이 끝나는 시점으로 이동해준다.
  8. 모든 명령어를 처리한 후에 현재 시간을 00:00 형태로 변환해주고 return 한다.

코드

function solution(video_len, pos, op_start, op_end, commands) {
    var answer = '';
    
    let full = video_len.split(":").map(Number);
    let current = pos.split(":").map(Number);
    let start = op_start.split(":").map(Number);
    let end = op_end.split(":").map(Number);
    
    let startSec = start[0]*60+start[1];
    let endSec = end[0]*60+end[1];
    
    function isOpening() {
        let sec = current[0]*60+current[1];
        if(startSec<=sec && sec<=endSec) return true;
        return false;
    }

    for(let i=0; i<commands.length; i++) {
        if(isOpening()) {
                current[0] = end[0];
                current[1] = end[1];
        }
        
        if(commands[i]==="next") { 
            current[1] = current[1] + 10;
            if(current[1]>59) {
                current[0]++;
                current[1] = current[1] - 60;
            }
            if(current[0]>full[0] || (current[0]=== full[0] && current[1]>full[1])) {
                current[0] = full[0];
                current[1] = full[1];
            }
        } else if(commands[i]==="prev") {
            current[1] = current[1] - 10;
            if(current[1]<0) {
                current[0]--;
                current[1] = 60 + current[1];
            }
            if(current[0]<0) {
                current[0] = 0;
                current[1] = 0;
            }
        }
        if(isOpening()) {
                current[0] = end[0];
                current[1] = end[1];
        }
    }
    
    answer = `${(current[0]+"").padStart(2, '0')}:${(current[1]+"").padStart(2, '0')}`;
    return answer;
}

회고

오프닝 도중 시간 체크하기에서 삽질을 정말 많이 했다. 결국 초로 변환해서 해결하는 방법 밖에 없었는데 이미 분/초로 구현을 다 해놔서 효율성 챙기기 위해 모든 코드를 갈아엎기에는 너무 시간 투자를 많이해서.... 오프닝 체크를 위해서만 초로 변환해주었다. 진짜...... 테스트 케이스 하나가 진짜 발목을 너무 잡았는데 아직도 모르겠다...

profile
Frontend🍓

0개의 댓글