From e7cf62951569a1a8b4f93035a78364d019c08d36 Mon Sep 17 00:00:00 2001 From: Songwonseok Date: Fri, 11 Dec 2020 12:16:17 +0900 Subject: [PATCH] =?UTF-8?q?deploy:=20Week3=20=EB=B0=B0=ED=8F=AC=20(#133)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * Update README.md * feat: [FE] 오디오 정지, 구간 건너뛰기 기능 구현 (#98) * feat : [FE] Audio Stop 구현 - 정지 버튼을 누르면 오디오 재생이 정지됨 - 마커의 위치가 0으로 이동 - 재생시간이 0으로 초기화 - controller에 setMarkerWidthToZero 메서드 추가 - store에 setMarkerWidthToZero 메서드 추가 - storetype으로 current_position_zero_channel 추가 - Marker는 해당 채널을 subscribe해 마커 위치를 0으로 초기화함 * chore : [FE] 공백 삭제 - enter가 두 번 들어간 부분 삭제 * feat : [FE] 건너뛴 시간이 음수일 때 처리 - markerTime이 음수가 됐을 때 0으로 변경 - playTime이 음수가 넘어왔을 때 재계산 * feat : [FE] 마커의 위치가 음수일 때 처리 - 마커가 움직인 위치가 음수가 넘어와 현재 위치와 계산 시 음수가 됐을 때 0으로 값 적용 * feat : [FE] 구간 건너뛰기 기능 구현 - 구간 건너뛰기 기능 구현 - quantum은 3으로 설정 - 앞으로 건너뛰기, 뒤로 건너뛰기 가능 - markerTime 변경을 audioContext.currentTime을 사용하는 것이 아닌 setInterval에서 주기적으로 변하게 하는 것으로 변경 - setInterval이 건너뛰기에서도 필요하여 따로 함수로 분리해 사용 * chore : [FE] 주석 삭제 - 주석을 몇 개 삭제했습니다. - 이해가 헷갈리는 부분들의 주석은 아직 삭제하지 않았습니다. * feat : [FE] 맨 앞으로 건너뛰기 - 기존의 stop, play 기능을 호출하여 맨 앞으로 건너뛸 수 있도록 구현 - stop을 했을 때 delay가 발생하기 때문에 setTimeout 내에서 play를 호출하도록 함 * feat : [FE] marker position 클릭으로 변경 시 재생 위한 위치 선정 - MarkerEventUtil의 clickMarkerListener에서 play 함수를 호충하면 됨. - 아직 play 함수를 다른 곳에서 호출할 수 없어서 위치를 잡아두기만 함. * fix : [FE] FastRewind 시의 시작 오류 수정 - fastRewind가 0이하로 됐을 때 waitTime 시간이 제대로 계산이 안되어 delay가 생김 - markerTime - QUANTUM이 음수가 됐을 때 waitTime을 0으로 설정. * refactor : [FE] 변수 이름 변경과 주석 삭제 - when, offset, duration 변수 이름을 waitTime, audioStartTime, playDuration으로 변경 - 주석 삭제 close #23, close #25 * feat: [FE] 구간 편집기능(잘라내기, 붙여넣기, 구간 자르기) 구현 (#101) * refactor: [FE] auto increment 초기값 1로 수정 - clipBoard에 저장되는 TrackSection과 구분하기 위하여 auto increment의 초기값을 1로 수정 * refactor: [FE] DeleteCommand CopyUtil 적용 - CopyUtil로 deleteList에 저장하도록 수정 * refactor: [FE] CopyUtil 파라미터 수정 - 파라미터로 id 정보를 받도록 수정했습니다. - 기존에는 clipBoard에 저장하는 용도로 만들었지만 DelelteCommand에서는 id 정보가 필요해서 파라미터로 추가했습니다. * feat: [FE] controller cutCommand() 구현 - 기존에 구현한 copy와 delete 기능을 합쳐서 cutCommand 함수를 구현했습니다. - 잘라내기는 섹션을 1개 선택했을 때만 가능하기 때문에 setClipBoard()의 return값을 boolean으로 수정하여 유효성 체크를 했습니다. * feat: [FE] EditTools 잘라내기 이벤트 등록 - 선택한 구간 잘라내기 기능을 위한 이벤트를 등록했습니다. * feat: [FE] 잘라내기 단축키 구현 - 1개의 구간을 선택하고 ctrl + X 를 누르면 cutCommand()가 실행되도록 구현했습니다. * feat: [FE] icons에 iconInfo 추가 - IconButton에 대한 hover 기능을 위해 아이콘 정보를 알려주는 iconInfo를 구현했습니다. - iconInfo는 iconType에 대한 상세정보를 알려줍니다. * style: [FE] hover시 icon 상세정보가 나오도록 style 적용 - visibility 속성을 이용하여 hover시에 아이콘에 대한 정보가 나오도록 style 적용했습니다. - transition-delay로 0.3초 후에 정보가 나오도록 했습니다. * feat: [FE] 시작위치, 종료위치 이동 단축키 추가 - 시작위치로 이동은 '[', 종료위치로 이동은 ']' 키로 동작하도록 keyType과 keyDown 이벤트를 추가했습니다. * refactor: [FE] publish 빈 인자값 명시적으로 표현 - 빈 인자를 넘기는 경우 null로 표현 * refacot: [FE] Command 클래스 index에서 export하도록 변경 - import하기 쉽게 index.js에서 export 관리 * feat: [FE] CopyUtil copyTrack 함수 구현 - Track을 deep copy하기 위한 함수 구현 * refactor: [FE] store setTrack, setTrackSection 수정 - setTrack은 파라미터의 따라 다르게 동작하도록 변경 - newTrack의 id값이 이미 TrackList에 존재하는 경우에는 값만 변경 - 존재하지 않는 경우에는 TrackList에 새로 추가 - setTrackSection - clipBoard에 복사된 trackSection은 0의 값을 가지므로 새로 id값을 변경 - undo에서 그전에 저장된 상태 그대로 변경해야 되기 때문에 trackSection id값을 그대로 사용 * feat: [FE] controller pasteCommand 함수 구현 - 붙여넣기는 현재 clipBoard에 복사된 구간의 정보를 사용합니다. - 현재는 1개의 구간을 선택한 경우에만 붙여넣기를 할 수 있습니다. - 추후 마커위치 관련해서 붙여넣기 기능 리팩토링 필요 - pasteCommand()에 대한 설명 - 1) focusList의 길이가 1인지 확인합니다. - 2) 붙여넣기의 대상이 되는 track과 clipBoard를 deep copy합니다. - 3) copySection의 trackStartTime을 현재 focus한 trackSection의 endTime으로 변경합니다. - 4) PasteCommand의 인자로 넘겨줍니다. * feat: [FE] PasteCommand 클래스 구현 - beforeTrack은 execute가 실행되기 전 저장하는 상태값입니다. - execute() - 1) track과 section의 정보를 deep copy합니다. - 2) track에서 newSection이 종료되는 시점인 endTime을 구합니다. - 3) newSection의 종료시간 전에 시작하는 section이 있는지 findIndex를 통해 index를 구합니다. - 4-1) firstDelayIndex가 -1이 아니라면 (존재) - endTime에서 해당 section의 시작시간을 빼서 얼마나 delayTime을 구합니다. - firstDelayIndex와 그 뒤에 존재하는 section들의 trackStartTime에 delayTime을 더해줍니다. - 4-2) firstDelayIndex가 -1인 경우 (존재x) - 바로 트랙에 newSection을 추가합니다. - undo() - 저장된 beforeTrack을 deep copy하여 setTrack으로 덮어씌우고 publish합니다. * feat: [FE] EditTools 붙여넣기 이벤트 등록 - 붙여넣기를 위한 click 이벤트 등록 * feat: [FE] 붙여넣기 단축키 기능 구현 - 클립보드에 복사된 상태에서 ctrl + v 를 누르면 붙여넣기 기능이 실행되도록 구현 * style: [FE] PlayBarUtil 주석 추가 - 나중에 리팩토링을 위한 주석추가 - PR merge하면서 지우겠습니다 * feat: [FE] SplitCommand 구현 - 구간 자르기 기능을 위한 SplitCommand 구현 - 클릭한 곳의 시간과 클릭한 trackSection의 trackStartTime으로 splitTime을 구했습니다. - splitTime을 기준으로 왼쪽 오른쪽 섹션을 생성하여 기존의 trackSection을 지우고 새로 track에 추가했습니다. - 기준이 되는 element는 audi-main-audio-track-container div로 설정했습니다. * feat: [FE] AudioTrackSection 컴포넌트 click 이벤트 분기처리 - cursorMode에 따라 focus되거나 split 하도록 조건문을 추가했습니다 * feat: [FE] controller splitCommand() 구현 - SplitCommand를 실행하는 함수를 구현했습니다. - getTrack은 자주 쓰일 것 같아서 구현했습니다. * feat: [FE] AudioTrack 자르기 모드 커트라인 구현 - Track 별로 cut-line 관리하도록 구현했습니다. - 평소에는 class="hide"로 되어있다가 cursorMode가 cut일 때, trackSection의 mousemove 이벤트로 hide가 해체됩니다. * feat: [FE] EventType mouseout 추가 - 마우스가 canvas 밖으로 나갔을 때 cut-line이 다시 hide 되도록 mouseout 이벤트를 추가했습니다. * feat: [FE] AudioTrackSection mousemove, mouseout 이벤트 구현 - TrackSection위에서 cursorMode가 자르기 모드일 경우 마우스 커서 위치에 따라 cut-line이 나오도록 구현했습니다. - mouseout은 커서가 TrackSection 밖 나갔을 경우 다시 hide하기 위한 기능입니다. * style: [FE] image cursor 위치 조정 - 커트라인과 마우스 커서의 위치의 싱크가 잘 맞지 않아서 수정했습니다. close #65, close #68, close #73, close #99, close #100 * feat: [FE] 재생바 마커 드래그로 이동하기 구현 (#102) * feat: [FE] PlayBarMarkerBlurZone 컴포넌트 구현 - 재생바 마커를 움직일시 바깥 영역의 색을 바꾸는 컴포넌트를 구현했습니다. * style: [FE] PlayBarMarkerBlurZone style 구현 - PlayBarMarkerBlurZone style을 적용했습니다. * style: [FE] PlayBarMarker style 수정 - cursor: pointer를 추가했습니다 - z-index: 4를 추가했습니다. * feat: [FE] playbarMarker 드래그 이벤트 추가 - 재생바 마커가 드래그로 움직이도록 했습니다. - 드래그시 바깥 영역에 색이 들어가도록 구현했습니다. * feat: [FE] playbar-event-zone 추가 - 재생바 이벤트가 적용되는 영역에 대한 css 설정 * feat: [FE] eventType 추가 - PLAYBAR_EVENT_ZONE_DROP eventkey 추가 * feat: [FE] 재생바 더블클릭시 바깥영역 색 지정 - 더블클릭 했을때도 바깥 영역의 색이 변경되도록 수정했습니다. Co-authored-by: Woojin Shin * refactor: [FE] AudioTrack Scroll 기능 구현 및 코드리팩토링 (#103) * feat: [FE] store 트랙 최대 너비 사이즈 추가 - 트랙의 최대 너비 사이즈로 모든 트랙의 너비사이즈를 조정하여 전체 트랙을 스크롤하기 위해 store에 최대 트랙 너비사이즈 추가 - 컨트롤러를 통해 최대 트랙 너비사이즈를 업데이트 할 수 있도록 인터페이스 추가 * feat: [FE] eventType에 mouseup, mousemove 이벤트 추가 - Zoombar 마우스 이벤트를 위한 이벤트 추가 * feat: [FE] ZoomBar 컴포넌트 제작 - 트랙 사이즈에 따라 동적으로 사이즈가 변하도록 제작 - 트랙 사이즈에 따라 스크롤하는 px값이 변하도록 제작 * feat: [FE] Main 컴포넌트에 트랙 스크롤 영역 추가 - Zoombar 컴포넌트로 스크롤 하기위한 영역 추가 * refactor: [FE] AudioTrack, AudioTrackSection 스크롤 가능하도록 수정 - 트랙의 너비사이즈를 최대 트랙의 너비 사이즈로 모두 변경하도록 수정 - 트랙영역을 overflow하면 스크롤이 가능하도록 수정 - 트랙의 너비사이즈가 변하면 store에 동기화 하도록 수정 * refactor: [FE] 코드 리팩토링 및 버그수정 - 트랙에 내용이 없는 경우에 Drag&Drop 메세지가 나타나도록 수정 - Zoombar EventType 추가 * style: [FE] 메인 페이지 레이아웃 수정 - 메인 페이지 노트북, 데스크탑 화면 레이아웃 깨지는 스타일 수정 * refactor: [FE] 루트 컴포넌트 이벤트 delegation 처리 추가 - 루트에서 이벤트 실행시 delegation을 적용한 부모 태그를 찾아서 이벤트를 실행하도록 코드 수정 * feat: [FE] TrackOptionType 정의 - TrackOption 컴포넌트 OptionType 정의 * feat: [FE] AudioTrackOptionClick EventKey 정의 - AudioTrackOption 이벤트 처리를 위한 이벤트 키 정의 * feat: [FE] AudioTrackOption 컴포넌트 제작 - AudioTrack에 따라 컴포넌트가 메인페이지가 추가되도록 제작 - Event Delegation으로 event-key를 찾을 수 있도록 수정 * feat: [FE] AudioTrackMenu 컴포넌트 제작 * style: [FE] 메인페이지 레이아웃 조정 - 헤더 레이아웃 데스크탑, 노트북에 따라 조정 close #10, close #12 * feat: [FE] compressor 전체 트랙 압축 기능 구현 (#116) * fix: [FE] PasteCommand trackStartTime 버그 해결 - paste된 TrackSection의 trackStartTime의 값이 이상하게 저장되는 버그 발견 - delayTime을 구할 때 trackStartTime이 아니라 audioStartTime으로 구해서 버그 발생했었음 - audioStartTime -> trackStartTime으로 변경하여 해결 * refactor: [FE] 모든 트랙 merge하여 save하도록 변경 - 파일 생성 단계 - 1) getTrackArrayBuffer()로 해당 트랙을 하나의 ArrayBuffer로 생성합니다. - 2) for문으로 모든 트랙의 ArrayBuffer를 배열에 담아줍니다. - 3) 모든 트랙의 ArrayBuffer가 담긴 arrayBufferList를 mergeTrackArrayBuffer()로 merge시켜줍니다. - 4) merge된 buffer를 ChannelDataToWave()로 wavBuffer로 만들어줍니다. - 5-1) wav로 저장 - wavBuffer를 file로 생성하고 버튼의 href를 변경 - 5-2) mp3로 저장 - wavBuffer를 lamejs 라이브러리로 mp3로 encoding 한 후에 file로 생성하여 href 변경 * fix: [FE] Paste 이후 선택구간 테두리 초기화 버그 - setSection으로 리렌더링 되면서 focus된 구간의 border style을 먹인 class가 사라지는 문제 발생 - trackSection 컴포넌트에서 initState()로 focusList의 section과 동일한 id를 가지고 있다면 focusInfo의 element를 갱신하고, classList.add('focused-section')를 해줬습니다. * feat: [FE] EditorMenu 컴포넌트 init 구현 - 모든 트랙에 trackSection이 하나도 없으면 저장하기 버튼을 비활성화 시켰습니다. - init에서 모든 track의 trackSectionList의 길이를 확인했습니다. * feat: [FE] store publish 추가 - EditorMenu의 리렌더링을 위한 publish 추가 * feat: [FE] controller getSourceList() 구현 - 모든 sourceList를 조회할 수 있는 함수 구현 * refactor: [FE] SourceDowload saveFile 인자 변경 - 기존에 test용으로 사용하던 audio파일을 제거하고 인자를 compressorObject만 넘겨주도록 수정 * refactor: [FE] compressor emptyTime 삭제 - 오늘 스크럼때 설명하면서 깨달았는데 emptyTime이 필요가 없어서 삭제했습니다. - 구현 초기에 사용하고 로직이 변경되어 emptyTime이 필요가 없어서 삭제 - start()의 첫번째 인자로 trackStartTime을 넘겨주면서 필요가 없어졌습니다. * feat: [FE] 변환 완료된 파일명 변경 기능 구현 - 파일을 변환 완료한 상태에서 기존에는 확장자를 바꾸거나 모달창을 닫으면 초기화됩니다. - 하지만 변환 완료된 상태에서 파일명을 바꿀 때도 초기화 되면 너무 비효율적이라고 생각해서 download의 저장된 파일명만 변경되도록 구현했습니다. * refactor: [FE] compressor getTrackArrayBuffer 속도 개선 - 기존에는 for문으로 모든 섹션의 채널데이터를 매번 index 0부터 계속 더하는 식으로 구현했습니다. - offset 변수를 둬서 그전에 저장했던 index는 건너뛸 수 있도록 변경했습니다. * style: [FE] compressor 공백 제거 - 불필요한 공백을 제거했습니다. * refactor: [FE] EffectState 컴포넌트 구현 & PlayBarUtil / controller/ store 함수명, 변수명 리팩토링 (#118) * feat: [FE] EffectState 컴포넌트 제작 - 구간이 선택되면 해당 구간에 적용된 effect 목록을 보여주는 컴포넌트를 구현했습니다. * style: [FE] SectionEffectList style 적용 - SectionEffectList 컴포넌트의 style을 지정했습니다. * feat: [FE] SectionEffectList와 관련된 Util 생성 - hideEffectList는 section 선택을 하지 않았을때 EffectList를 숨기고 SourceList를 보여주는 함수입니다. - showEffectList는 section 선택을 했을 때 EffectList를 보여주고 SourceList를 숨기는 함수입니다. * feat: [FE] SourceList 숨김/표시 함수 추가 - section에 포커스가 있을 경우 sourcelist가 보이지 않도록 하는 함수를 만들었습니다. - section에 포커스가 없을 경우 sourcelist가 보이도록 하는 함수를 만들었습니다. * feat: [FE] SectionEffectList 컴포넌트 Sidebar에 추가 - Sidebar 컴포넌트에 SectionEffectList 컴포넌트를 넣었습니다. * feat: [FE] focus를 줄때 SectionEffectList 숨김/표시 설정 - addFocus시 SectionEffectList가 보이도록 로직 작성 - removeFocus, resetFocus시 SectionEffectList가 숨겨지도록 로직 작성 * feat: [FE] EffectListType 설정 - EffectListType을 설정했습니다. * feat: [FE] eventType 추가 - EffectList Modal 창을 여는 이벤트타입 설정했습니다. * refactor: [FE] setMarkerWidthToZero 관련 함수 삭제 - setMarkerWidth에 함수 인자 0을 넣어주면 마커의 위치가 초기화 되도록했습니다. - 따라서 같은 역할을 하는 setMarkerWidthToZero 함수는 필요가 없다고 생각되어 삭제했습니다. * refactor: [FE] store state 변수명 변경 - 시간을 string으로 나타내는 변수는 ~~StringTime으로 표기했습니다. - 시간을 초단위로 나타내는 변수는 ~~NumberTime으로 표기했습니다. * refactor: [FE] TimeUtil 파일 생성 - PlayBarUtil에서 시간과 관련된 함수들을 분리했습니다. - getSplitTime은 초단위로 정의된 시간을 분, 초, 밀리초로 분해해주는 함수입니다. - getStringPlayBarTime은 재생바 눈금을 나타내주는 함수입니다. - getNumberTime은 마우스 위치를 초단위 시간으로 나타내주는 함수입니다. - getStringTime은 초단위 시간을 문자로 나타내주는 함수입니다. ex) '05:11:236' * refactor: [FE] setCursorStringTime 함수 내용 변경 - 함수 이름과 맞는 로직으로 수정했습니다. 함수 인자로 string 형식의 시간을 받아오고 바로 새로운 stringTime을 state에 넣어줬습니다. - 그 외 잘못 변경된 state 변수명들을 수정하고 그에 맞는 함수들도 이름을 변경했습니다. * refactor: [FE] controller 함수명 변경 - resetPlayTime 함수를 changeMarkerPlayStringTime으로 변경했습니다. - 해당 함수는 재생바를 클릭했을 경우 마커의 시간을 변경해주는 함수입니다. * refactor: [FE] PlayBarUtil 파일명 변경 - WidthUtil로 이름 변경했습니다. - 너비를 구하는 함수들이 모여있습니다. * refactor: [FE] 함수명 변경 - 리팩토링한 함수로 교체했습니다. * refactor: [FE] 왼쪽 재생바 마커 위치 수정 - 100%이던 위치를 99.3%로 바꿨습니다. - 아마 프로젝트가 거의 다 완성되면 또 다시 바뀔 것 같습니다. * refactor: [FE] getNumberTime 명확한 변수명으로 수정 - calculateTimeOfCursorPosition 함수명으로 이름을 바꾸어 무슨 역할을 하는지 명확하게 작성했습니다. Co-authored-by: Woojin Shin close #117 * refactor: [FE] 구간 나누기 버그 수정 및 코드 리팩토링 (#115) * refactor: [FE] SplitCommand 코드 리팩토링 - excute 함수 작은 함수단위로 분리 - PlayBarUtil은 PlayBar와 관련되어 보이는데 섹션을 나누는 커맨드에서 사용되어 코드의 가독성이 떨어지고, PlayBarUtil은 리팩토링 과정이 필요하므로 해당 부분을 삭제하고, 필요한 로직부분만 추가하도록 수정 (추후 공통 유틸이 만들어지면 수정필요) - 트랙 섹션을 나눌 때, 파형이 이상하게 나누어지는 버그 수정. trackStartTime, channelStartTime 설정로직 수정하여 해결 * refactor: [FE] AudioTrackSection 리팩토링 - 타입 관리를 위한 SectionDataType 분리 - CursorMode를 subscribe하여 자동으로 업데이트 되도록 수정 - 함수 네이밍 명확하게 변경 및 작은 함수 단위로 분리 * refactor: [FE] AudioTrack 리팩토링 - TrackSection 모델 parsedChannel 정보 삭제 - 소스리스트에서 드래그앤드랍으로 새로운 트랙을 추가할 때 비즈니스 로직을 컨트롤러로 이동 * refactor: [FE] EditTools 리팩토링 - 함수 네이밍 명확하게 변경 * refactor: [FE] Controller 리팩토링 - getSectionData 함수 가독성 향상 및 유지보수를 위해 함수형으로 기능단위로 분리 * fix: [FE] 머지과정에서 발생한 버그 수정 (#119) - PlaybackTools 리렌더링을 통한 프로퍼티 참조 문제 해결 - Controller 함수 getSourceBySourceId 추가, SpliCommand 수정 사항 반영 * feat: [FE] AudioMeter 컴포넌트 구현 (#120) * feat: [FE] AudioUtil getDecibel() 구현 - analyser의 getFloatTimeDomainData를 Decibel로 변환하는 함수 구현 * feat: [FE] AudioMeter 컴포넌트 구현 - 현재 재생중인 volume에 따라 반응하는 AudioMeter 컴포넌트 구현 - 최대 -72까지를 기준으로 구현했고, 백분율로 다른 gradient 컬러를 먹였습니다. - 스크롤 기능이 다 구현되면 나중에 canvas로 반응형으로 구현해도 될 것 같습니다. * style: [FE] AudioMeter style 적용 - decibel-Marekr는 decibel이 100%를 넘어가는 경우를 위해 -10px 했습니다. * feat: [FE] components/index.js, Main.js에 AudioMeter 적용 - index.js와 Main 컴포넌트에 AudioMeter 컴포넌트를 적용했습니다. * feat: [FE] PlayBackTools의 setInterval 수정 - setInterval로 재생중인 오디오 정보로 AudioMeter의 style을 지속적으로 변경하도록 구현 - 나중에 AudioContext를 따로 store에서 관리하도록 수정해야 될 것 같습니다. - 현재 건너뛰기로 audioContext가 변경되면 AudioMeter에 적용되지 않는 문제가 있어서 store 리팩토링이 끝나면 해결하겠습니다. Co-authored-by: Woojin Shin close #17 * feat: [FE] 구간 이동하기 (드래그 앤 드롭) 기능 구현 (#122) * feat: [FE] MoveCommand 생성 - 드래그하여 trackSection을 이동할 수 있는 MoveCommand를 생성했습니다. - dragstart 할 때의 커서 시간에서 trackStartTime을 빼준 차이를 구합니다. - 그런 후 drop시의 커서 시간에서 빼준 차이를 다시 빼준게 newTrackStartTime입니다. - checkEnterTrack은 아직 미구현 함수입니다! * feat: [FE] audioTrack 관련 eventType 생성 - AUDIO_TRACK_MULTIPLE과 AUDIO_TRACK_AREA_MULTIPLE 타입을 만들었습니다. * refactor: [FE] 사용되지 않는 변수 삭제 - 더이상 사용되지 않는 변수를 삭제했습니다. * feat: [FE] 너비의 차가 항상 양수가 되도록 수정 - 너비의 차가 항상 양수가 되도록 Math.abs 함수를 사용했습니다. * feat: [FE] drop 이벤트 생성 - dropAudioTrackListener - 현재 커서의 위치를 구합니다. - 커서 위치의 시간값을 구합니다. - 필요한 인자들을 moveCommand에 넘겨줍니다. - audio-track-area에 data-track-id를 추가해줬습니다. * feat: [FE] dragstart 이벤트 함수 생성 - 드래그가 시작할 때 해당 함수가 작동하도록 했습니다. - 드래그 시작시 커서 위치를 구하고 그때의 시간값을 구했습니다. - 그런뒤 필요한 변수들을 dataTransfer를 통해 전달해줬습니다. * feat: [FE] moveCommand 함수 추가 - moveCommand를 실행해주는 함수를 생성했습니다. - getSource와 getTrackSection 함수를 만들었습니다. - 각각의 데이터가 필요할 때가 있어 분리하여 만들었습니다. * fix: [FE] 스크롤 기능 추가로 발생한 버그 수정 (#121) * feat: [FE] 트랙 파형 동적인 사이즈 변경 적용 - 트랙 파형을 브라우저의 사이즈 변경에 따라 동적으로 변하도록 구현 * feat: [FE] store maxTrackPlayTime 정의 - 재생바의 시간을 동적으로 변경하기 위해 멀티트랙에서 최대 재생시간을 store에 정의 * feat: [FE] Controller maxTrackPlayTime 변경 함수 추가 - 최대 트랙 재생시간 업데이트하기위한 changeMaxTrackPlayTime 함수 추가 * refactor: [FE] 트랙 영역 반응형 사이즈 적용 - AudioTrack 컴포넌트의 트랙 영역을 최대 트랙 사이즈에 따라 반응형 으로 리사이징 적용 * refactor: [FE] 메인컴포넌트 트랙 스크롤 영역 수정 - 트랙 스크롤 영역을 재생바도 포함하도록 수정 * refactor: [FE] 재생바 동적 사이즈 변경 적용 및 코드리팩토링 - 트랙의 최대사이즈 및 최대재생시간에 따라 재생바가 동적으로 렌더링되도록 수정 - 변수 및 함수 네이밍을 명확하게 변경 * style: [FE] TimeInfo 반응형 사이즈 변경 * feat: [FE] store currentScrollAmount 정의 - 스크롤에 따른 재생바 커서 시간 싱크를 맞추기위해 스크롤한 양(pixel) 상태유지 추가 * refactor: [FE] ZoomBar 현재 스크롤양을 상태유지 하도록 수정 - 현재 스크롤양을 상태유지하기 위해 ZoomBar에서 Controller를 활용하여 스크롤시 스크롤양을 store에 업데이트 * refactor: [FE] TimeUtil / WidthUtil 리팩토링 - 트랙의 재생시간 변경에 따라 커서 시간계산을 위해 재생시간을 300초 고정이 아닌 동적으로 변경될 수 있도록 수정 - 변수 및 함수 네이밍 명확하게 수정 - Util 코드 수정으로 인한 Util 함수를 사용하는 각 컴포넌트 코드 수정 * refactor: [FE] 재생바 playTime 반응형 사이즈 적용 - 브라우저 사이즈에 따라 재생바의 playTime 반응형으로 사이즈 조절 - playTime이 반응형으로 변경되면서 커서 타임 계산도 변경된 사이즈에 맞게 보여주도록 수정 * fix: [FE] 트랙섹션 cutLine 스크롤에 따라 위치 조정 - 스크롤시 트랙섹션의 cutLine이 위치가 조정되지 않는 버그 수정 * refactor: [FE] SplitCommand 나누기 로직 수정 - splitCommand 나누기 로직 동작 버그 수정중 * fix: [FE] 오디오 미터 출력 버그 수정 (#123) - 음악 재생시 오디오미터 출력 버그 수정 - 함수 이름 변경으로 인한 오류 해결 * feat: [FE] 단축키 기능, 재생 중 커서 클릭 시 원하는 위치에서 시작, solo/mute 기능 적용 (#124) * chore : [FE] audioStartTime 삭제와 channelStartTime 사용 - trackSection에서 audioStartTime을 삭제했습니다. - channelStartTime을 사용하도록 변경했습니다. * chore : [FE] 주석 삭제 - subscribe 부분의 주석을 삭제했습니다. * feat : [FE] 채널 삭제 및 추가와 solo, mute 관련 - CURRENT_POSITION_ZERO_CHANNEL 삭제 - play, pause 아이콘 변경을 위한 PLAY_OR_PAUSE_CHANNEL 추가 - solo 상태를 관리하기 위한 SOLO_CHANNEL 추가 - solo, mute 클릭 시 색상 변경 * feat : [FE] play/pause 아이콘 변경, solo 위한 메소드 추가 - changePlayPauseIcon으로 상태에 따라 아이콘 변경 - soloPlay로 전체 solo 선택 상태 변경 * feat : [FE] 단축키 함수 정의 - Controller에 단축키에서 사용할 함수 정의. - playbackTool의 함수를 호출한다. * feat : [FE] 재생 중 커서로 클릭한 곳에서 바로 재생 - 재생 중에 커서로 클릭한 마커 위치에서 다시 재생할 수 있는 audioCursorPlay 함수 호출. - play 함수가 호출됨. * chore : [FE] updateMarkerPosition 고려사항 - 함수가 받아온 값이 0일 때 0으로 값을 바꾸고 0이 아닐 때는 이전 값에 더해나가는 형식. - 오른쪽 끝으로 옮길 때도 더해나가는 방식이 되어 수정이 필요. 현재 마지막 위치 관련해서 스크롤과 관련해 수정이 필요해보여 메모만 해 둠. * feat : [FE] PlaybackTools 기능관련 부분 이동 - PlaybackToolClass로 옮겨 controller에서 사용할 수 있도록 함. - 정지와 반복을 제외한 모든 키는 단축키로 사용할 수 있음. - mute, solo에 대한 정보를 지역적으로 가지고 있음. * feat : [FE] 반복버튼 클릭시 색상 변화 - 반복 버튼 클릭 시 색상 변화 - 한 번 클릭 시 초록색, 다시 클릭 시 흰색으로 돌아옴. - store에 isRepeat 추가 - playbackTools에서 IS_REPEAT_CHANNEL을 구독 중. - isRepeat 값을 변경할 때 publish 되어 상태 전달. * feat : [FE] 반복 재생 버튼 클릭시 시작 위치 변경 - 반복재생 버튼 클릭 시 시작 위치가 변경되도록 했습니다. - 재생, 일시정지 버튼 클릭 시 반복재생 버튼의 색상이 초기화되는 문제가 있습니다. * style : [FE] 반복 재생 버튼 green color - 클릭 시 green color로 변경되도록 scss에 class를 추가. * chore : [FE] 주석 삭제 - PlaybackTools에서 고민하던 흔적이 담긴 주석을 지웠습니다. * refactor : [FE] Handler Listener로 변경 - AudioTrackOption에서 Handler로 쓰인 부분을 Listener로 변경했습니다. Co-authored-by: Woojin Shin close #109, close #108, close #106, close #104 * fix: [FE] Marker 이동 및 AudioMeter 버그 해결 (#125) - 변경된 함수 적용(네이밍, 인자) - PlaybackToolsClass의 모듈화로 AudioMeter 관련 삭제된 로직 추가 * Update README.md * refactor: [FE] Solo 다중 선택할 수 있도록 수정, 오디오 미터 멈춤 현상 수정 (#126) * refactor : [FE] 변수 명 수정 - ret으로 사용해서 명확하지 않았던 변수 명을 수정했습니다. * feat : [FE] Solo 다중 선택 기능 - Solo 기능을 다중선택 할 수 있도록 수정. - Solo 버튼이 눌린 경우 재생됨. 눌리지 않은 트랙은 재생 안됨. * fix : [FE] 오디오 미터 멈춤 문제 해결 - createAndConnectAnalyser 메소드를 만들어 다시 재생할 때마다 호출. - 새롭게 analyser를 만들고 connect 해 줌. * refactor : [FE] !=를 !==로 변경 - 명확하게 !== 으로 변경했습니다. * refactor : [FE] repeat 버튼 클릭 시 색 변경로직 수정 - channel을 사용하지 않아 삭제. - 관련했던 여러 함수 삭제. - store의 isRepeat 값을 확인해 render 시에 그려주는 형식으로 변경. * feat: [FE] 오디오 구간 이동 구현, 소스리스트 파형 추가 리팩토링 (#128) * feat: [FE] SectionDragStartData Model 구현 - DragStart 이벤트에서 transferData 대신 store에서 상태관리를 위한 SectionDragStartData Model 구현 - dragover에서 getData를 사용할 수 없어서 추가 - SectionDragStartData에는 TrackSection, dragstart 당시 커서 위치, x좌표 값 계산을 위한 offsetLeft가 담겨 있습니다. * refactor: [FE] MoveCommand 유효성검사 조건 변경 및 모듈화 - 기존의 조건문에 문제를 발견해서 수정했습니다. - 구간 이동하기에서 이동할 수 있는 위치인지 판단하는 checkEnterTrack() 함수를 ValidUtil로 모듈화 시켰습니다. * refactor: [FE] TRACK_CHANEL 채널 publish 삭제 - 필요없는 publish 코드를 삭제했습니다. * feat: [FE] SectionDragStartData storeType 추가 * refactor: [FE] 단축키 설명 수정 - redo와 undo의 단축키 정보가 뒤바껴서 수정했습니다.... * feat: [FE] ValidUtil export 추가, eventType 추가 - 구간 이동할때 잔상을 위한 AUDIO_TRACK_AFTERIMAGE_DROP :ㅈEventType 추가 * style: [FE] AudioMeter console.log 제거 * style: [FE] AudioTrack 잔상효과 style 적용 - 부모요소의 padding값에 맞춰서 top과 height를 조절했습니다. - padding값이 변경되면 height와 top도 변경되야 합니다. * feat: [FE] AudioTrack 섹션의 시간에 맞는 위치 적용 - initPosition() 함수로 TrackSection들의 위치를 시간에 맞게 변경 - prevEndOffset은 바로 전 section의 마지막 위치를 나타내는데 초기값은 0 - preEndOffset과 현재 section의 trackStartTime의 차이 만큼 초당 px 크기를 계산해서 곱해서 margin-left에 적용했습니다. - afterimage - 오디오 구간을 AudioTrack에 drag했을 때 옮길 수 있는 위치면 잔상이 뜨도록 구현했습니다. - dragover, dragenter, dragleave로 잔상의 left값과 width를 변경하거나 display속성을 다르게 적용했습니다. * style: [FE] AudioTrackSetion style 수정 - 잘린 구간끼리 구분이 되지 않아서 border를 추가했습니다. * feat: [FE] AudioTrackSection drag 이벤트 적용 - 기존에 dragstart 했을 때 dataTransfer로 정보를 저장했는데 dragover 이벤트에서는 getData를 할 수 없어서 store에서 관련 정보를 저장하도록 수정했습니다. - afterimage를 위한 dragover 이벤트 추가 - drag에서 자신 위에 drop이 안되던 문제로 drop 이벤트도 추가했습니다. * style: [FE] IconButton 컴포넌트에 cursor:pointer 스타일 적용 - IconButton 컴포넌트는 기본으로 cursor:pointer로 되도록 수정했습니다. * feat: [FE] SectionDragStartData 생성자 파라미터 변경 - 소스리스트에서 드래그한 경우에는 prevCursorTime이 없어서 null 타입 추가 * refactor: [FE] DragUtil 구현 - 구간 이동할 때 dragover, drop에서 중복되는 코드 모듈화 시켰습니다. * refactor: [FE] SectionEffectList render() 수정 - IconButton 커스텀 엘리먼트 사용 * refactor: [FE] SourceList 드래그앤 드롭 관련 함수 추가 - addTrackSectionFromSource()를 삭제하고 현재 방식에 맞게 새로운 함수들을 추가했습니다. - moveCommand()의 파라미터를 sectionId -> trackSection으로 수정했습니다. - 현재 스크롤 시간과 위치를 구할 수 있는 getCurrentScrollAmount, getCurrentScrollTime 함수 구현 * refactor: [FE] SourceList MoveCommand 형식으로 변경 - SouceList에서도 구간 이동 기능처럼 동작하도록 로직을 수정했습니다. * refactor: [FE] PasteCommand, SplitCommand TRACK_CHANNEL 채널 추가 - 재생관련된 코드라고 들어서 지웠던 TRACK_CHANNEL 채널을 다시 추가했습니다. * refactor: [FE] MoveCommand SourceList 관련 조건문 추가 - SourceList에서 drag된 SectionDragStartData는 trackSection의 id값이 0으로 저장되어 있어서 조건문으로 removeSection을 실행하도록 수정했습니다. close #70 * feat: [FE] 트랙 추가하기 커맨드 구현 및 메인페이지 리팩토링 (#127) * fix: [FE] 스크롤시 구간 나누기 버그 수정 - 스크롤시 구간 나누기하면 스크롤 위치가 반영되지 않아 오작동 하는 버그 수정 * style: [FE] 메인페이지 레이아웃 및 반응형 스타일 수정 - ZoomBar, AudioMeter 위치 조정 - 메인페이지 레이아웃 반응형 사이즈 조절 * refactor: [FE] iconButton 이벤트 위임 처리하도록 수정 - 기존에 이벤트 키를 하위 컴포넌트에 모두 전달해야하는 방식에서 이벤트 위임 방식으로 아이콘 버튼 수정 * refactor: [FE] command 리팩토링 - command 모듈 내보내기 관련 에러 발생 이슈 해결 - 모듈을 내보낼 때 export default로 내보내서 참조하는 곳에서 destructure해서 사용할 수 있도록 수정 - Command 인터페이스 네이밍 변경 * refactor: [FE] Main 컴포넌트 리팩토링 - 코드 유지보수 및 렌더링 최적화를 위해 Main 컴포넌트를 하위 컴포넌트들의 조합으로 분리 * style: [FE] Main 컴포넌트 분리로 인한 스타일 수정 - 본문 내용 * feat: [FE] 이벤트 타입 정의 추가 - TrackAddMenu click 이벤트 키 추가 - TrackScroll List scroll 이벤트 키 추가 및 scroll 이벤트 추가 * refactor: [FE] Main 컴포넌트에서 TrackList, TrackOptionList 분리 - 트랙 추가로 인한 전체 렌더링을 방지하기 위해 리렌더링이 필요한 컴포넌트를 분리 - 컴포넌트 분리로 인한 스타일 수정 * feat: [FE] StoreType 추가 정의 - trackList 업데이트로 정보를 위한 TRACK_LIST_CHANNEL 추가 * feat: [FE] AddTrackCommand 정의 - 트랙 추가하기 커맨드 정의 - excute, undo시 트랙을 추가하거나 제거하고, TRACK_LIST_CHANNEL에 publish하여 trackList 변경내용을 렌더링 하도록 구현 * feat: [FE] CommandController 정의 - Command 관련 작업을 수행하는 CommandController 정의 - 기존 Controller의 역할이 증가함에 따라 Command 관련 책임을 CommandController로 분리 * feat: [FE] AudioTrackMenu 트랙 추가하기 기능 구현 - 트랙 추가하기 버튼을 클릭하면 AddTrackCommand를 실행하도록 구현 - 트랙 추가하기로 인한 트랙 리렌더링시 트랙 리사이즈 로직 변경 * refactor: [FE] Controller 리팩토링 - Controller에서 Command 관련 함수 CommandController로 이동 * fix: [FE] 트랙 나누기 redo시 브라우저가 멈추는 버그 해결 - 생성자함수 내에서 element를 한번 참조하고, 이후에 리렌더링을 통해 참조를 잃어버려서 발생하는 문제 해결 * fix: [FE] 트랙 이동하기 및 재생버튼 미동작 버그 수정 (#129) - 최초 소스에서 트랙으로 드래그앤드랍할 때, 위치값을 찾지 못하는 버그를 initPosition위치 및 trackWidth 재계산을 통해 버그 수정 - playBackTools에서 아이콘 버튼 delegation 적용하여 버그 수정 * feat: [FE] 트랙 추가하기 커맨드 수정 및 트랙 삭제하기 커맨드 구현 (#132) * refactor: [FE] 트랙 추가하기 커맨드 trackIndex를 고려하여 로직 변경 - store에 트랙 리스트, 트랙 인덱스 설정 함수 정의 - controller에 트랙 리스트에서 pop과 동시에 트랙 인덱스도 초기화 해주는 함수 정의 - AddTrackCommand 인덱스를 고려하여 excute, redo 로직 변경 * refactor: [FE] CommandController 오타 수정 - CommandController 오타 수정 및 deleteTrackCommand 실행 함수 추가 * feat: [FE] Controller 트랙리스트 push, pop함수 정의 - 트랙 추가하기, 삭제하기 시 인덱스를 고려하기 위해 트랙리스트 pop, push 함수 정의 * feat: [FE] 트랙 삭제하기 커맨드 정의 - 트랙을 삭제하는 커맨드 정의 * feat: [FE] 트랙 옵션 메뉴 삭제하기 버튼 기능 활성화 - 삭제하기 버튼 클릭 시 deleteTrackCommand 실행하도록 리스너 추가 * feat: [FE] Controller 트랙 제거, 트랙 삽입 정의 - 트랙 리스트에서 트랙 ID로 식별되는 트랙을 제거하는 함수 정의 - 트랙 리스트에서 삽입하고자 하는 인덱스 값을 통해 해당 위치에 삽입해주는 함수 정의 * refactor: [FE] 트랙 삭제하기 커맨드 로직 변경 - 트랙 삭제하기 커맨드 로직을 delete option 클릭 시 해당 트랙을 삭제하도록 로직 변경 * fix: [FE] 트랙 삭제하기 커맨드 undo 오작동 버그 해결 - Controller insertTrack 로직을 인덱스를 이용하여 제 위치에 삽입되도록 코드 수정 * refactor: [FE] 트랙 삭제하기 alert 메시지 추가 - 트랙이 최소 3개 이상 존재하도록 트랙 삭제시 alert 메시지 추가 close #110 * feat: [FE] 마커 종료시점 이동 및 오디오 반복 재생 구현, 앞으로 건너뛰기 버그 수정 (#131) * feat : [FE] 원하는 시간으로 마커 옮기기 위해 setMarkerWidth 수정 - 맨 마지막 건너뛰기, loop시 마커를 원하는 시간으로 이동하기 위함 - number 배열을 넘겼을 때 두 번째 값이 1이면 첫번째 값으로 할당하도록 수정. * feat : [FE] 맨 뒤로 건너뛰기시 마커이동, loop 구현 - 맨 뒤로 건너뛰면 가장 마지막에 있는 마커 시간으로 이동. - 구간을 구하기 위해 AudioTrack에 있던 calculateTrackWidth 함수를 복사해 가져와서 사용했는데 후에 따로 빼거나 해야할 것. - loop는 isRepeat의 상태와 markerTime의 위치를 계속해서 판단해 동작하도록 함. - loop가 돼있지 않을 때는 마지막 시간에서 멈춤. - loop일 때는 마지막 시간에서 다시 처음으로 돌아감. 후에 loop의 start와 endTime을 지정할 수 있게 되면 수정필요. close #24, close #29, close #111 * fix: [FE] 트랙 선택기능 구현, 좌우 단축키 버그 수정, backend 환경 삭제 (#130) * refactor: [FE] 재생중에 command 실행 안되도록 수정 - isPause를 확인하여 재생중에 command가 실행되지 않도록 변경 - CommandControllers에 executeMoveCommand() 추가 - excute -> execute로 변경 * refactor: [FE] moveCommand -> CommandController로 변경 * style: [FE] excute -> execute로 변경 - excute -> execute로 변경 * refactor: [FE] left, right 단축키 preventDefault() 설정 - left, right keydown에 스크롤이 움직이는 문제 해결 * feat: [FE] 트랙 선택기능 구현 - 트랙에 아무 구간도 존재하지 않을 경우 트랙을 선택해서 붙여넣기를 할 수 있는 기능 구현 - AudioTrack 컴포넌트에 track-select-line 엘리먼트를 추가해서 현재 store의 SelectTrackData를 확인하여 left, display 속성을 변경 - SelectTrackData는 어떤 트랙에 존재하는지 확인하기 위한 trackId, paste command 할 때 필요한 trackStartTime인 selectedTime이 저장되어 있습니다. - 트랙 선택은 focus 기능을 사용할 땐 초기화 되도록 구현했고, 트랙영역이 아닌 다른 focusList reset 구역을 클릭할 때도 초기화됩니다. * refactor: [FE] PasteCommand에 트랙선택 관련 조건문 추가 - 선택된 구간이 1개이거나 트랙을 선택했을 경우에 따라 다르게 동작하도록 구현했습니다. * feat: [FE] selectTrackData관련 controller, store 수정 - 트랙선택 관련 SelectTrackData 관련 함수들을 추가했습니다. - focus 선택 기능 관련하여 조건에 따라 selectTrackData가 reset되도록 수정했습니다. * chore: [BE] backend 환경 삭제 - 프로젝트 설계 변경으로 인한 backend 환경 삭제 Co-authored-by: Woojin Shin close #7 * Update README.md * Update README.md Co-authored-by: Woojin Shin Co-authored-by: pieisland <35261724+pieisland@users.noreply.github.com> Co-authored-by: Jeongeun-Choi --- README.md | 24 +- backend/.babelrc.js | 10 - backend/.eslintrc.js | 30 - backend/.prettierrc.js | 14 - backend/package-lock.json | 10524 ---------------- backend/package.json | 66 - backend/src/__test__/TransactionWrapper.js | 10 - backend/src/__test__/router/api.test.js | 58 - backend/src/__test__/router/auth.test.js | 76 - .../src/__test__/service/user-service.test.js | 79 - .../src/application/application-factory.js | 11 - backend/src/application/application.js | 79 - backend/src/application/index.js | 2 - .../database/connection-option-generator.js | 38 - .../common/config/database/database-type.js | 11 - backend/src/common/env/database-env.js | 72 - backend/src/common/env/env-type.js | 10 - backend/src/common/error/business-error.js | 8 - .../src/common/error/entity-already-exist.js | 10 - .../common/error/entity-not-found-error.js | 10 - backend/src/common/error/error-code.js | 10 - .../src/common/middleware/error-handler.js | 25 - backend/src/common/middleware/index.js | 3 - backend/src/common/middleware/request-type.js | 7 - backend/src/common/middleware/transformer.js | 10 - backend/src/common/middleware/validator.js | 8 - backend/src/controller/index.js | 3 - backend/src/controller/user-controller.js | 33 - backend/src/dto/index.js | 1 - backend/src/dto/signup-request-body.js | 20 - backend/src/main.js | 8 - backend/src/model/user.js | 43 - backend/src/router/api.js | 8 - backend/src/router/auth.js | 11 - backend/src/router/index.js | 12 - backend/src/router/oauth.js | 8 - backend/src/service/index.js | 1 - backend/src/service/user-service.js | 46 - backend/tsconfig.json | 29 - frontend/env/.env.sample | 2 - frontend/package-lock.json | 6 - frontend/package.json | 5 +- .../src/common/command/AddTrackCommand.ts | 46 + frontend/src/common/command/Command.ts | 6 + frontend/src/common/command/CommandManager.ts | 20 +- frontend/src/common/command/DeleteCommand.ts | 18 +- .../src/common/command/DeleteTrackCommand.ts | 60 + frontend/src/common/command/ICommand.ts | 17 - frontend/src/common/command/MoveCommand.ts | 72 + frontend/src/common/command/PasteCommand.ts | 52 + frontend/src/common/command/SplitCommand.ts | 75 + frontend/src/common/command/index.js | 8 + frontend/src/common/types/eventTypes.ts | 36 +- frontend/src/common/types/index.js | 4 +- frontend/src/common/types/keyType.ts | 4 +- frontend/src/common/types/storeTypes.ts | 34 +- frontend/src/common/types/trackOptionTypes.ts | 9 + .../src/common/types/trackSectionTypes.ts | 8 + frontend/src/common/util/AudioUtil.ts | 70 +- frontend/src/common/util/CopyUtil.ts | 20 +- frontend/src/common/util/DragUtil.ts | 70 + frontend/src/common/util/MarkerEventUtil.ts | 30 +- frontend/src/common/util/PlayBarUtil.ts | 59 - .../src/common/util/SectionEffectListUtil.ts | 19 + frontend/src/common/util/TimeUtil.ts | 54 + frontend/src/common/util/ValidUtil.ts | 10 + frontend/src/common/util/WidthUtil.ts | 29 + frontend/src/common/util/compressor.ts | 143 +- frontend/src/common/util/index.js | 6 +- frontend/src/components/App/App.ts | 50 +- .../src/components/AudioMeter/AudioMeter.scss | 46 + .../src/components/AudioMeter/AudioMeter.ts | 47 + .../src/components/AudioTrack/AudioTrack.scss | 70 +- .../src/components/AudioTrack/AudioTrack.ts | 246 +- .../AudioTrackMenu/AudioTrackMenu.scss | 31 + .../AudioTrackMenu/AudioTrackMenu.ts | 74 + .../AudioTrackOption/AudioTrackOption.scss | 50 + .../AudioTrackOption/AudioTrackOption.ts | 117 + .../AudioTrackSection/AudioTrackSection.scss | 23 +- .../AudioTrackSection/AudioTrackSection.ts | 242 +- frontend/src/components/AudioTrack/index.js | 2 + .../src/components/EditTools/EditTools.scss | 2 - .../src/components/EditTools/EditTools.ts | 146 +- .../src/components/EditorMenu/EditorMenu.scss | 16 +- .../src/components/EditorMenu/EditorMenu.ts | 39 +- .../src/components/IconButton/IconButton.scss | 31 + .../src/components/IconButton/IconButton.ts | 30 +- frontend/src/components/IconButton/icons.ts | 44 +- frontend/src/components/Logo/Logo.scss | 8 + frontend/src/components/Logo/Logo.ts | 2 + frontend/src/components/Main/Main.scss | 32 +- frontend/src/components/Main/Main.ts | 78 +- .../MainAsideContent/MainAsideContent.scss | 9 + .../Main/MainAsideContent/MainAsideContent.ts | 25 + .../MainMiddleContent/MainMiddleContent.scss | 18 + .../MainMiddleContent/MainMiddleContent.ts | 27 + .../MainTrackOptionListArea.scss | 18 + .../MainTrackOptionListArea.ts | 51 + .../MainRightContent/MainRightContent.scss | 44 + .../Main/MainRightContent/MainRightContent.ts | 116 + .../MainTrackListArea/MainTrackListArea.scss | 15 + .../MainTrackListArea/MainTrackListArea.ts | 53 + frontend/src/components/Main/index.js | 6 + frontend/src/components/Marker/Marker.ts | 14 +- .../SourceDownload/SourceDownload.ts | 13 +- frontend/src/components/PlayBar/PlayBar.scss | 20 +- frontend/src/components/PlayBar/PlayBar.ts | 286 +- .../PlayBar/PlayBarMarker/PlayBarMarker.scss | 4 + .../PlayBarMarkerBlurZone.scss | 12 + .../PlayBarMarkerBlurZone.ts | 43 + .../PlaybackTools/PlaybackToolClass.ts | 555 + .../PlaybackTools/PlaybackTools.scss | 12 +- .../components/PlaybackTools/PlaybackTools.ts | 178 +- .../SectionEffectList/SectionEffectList.scss | 58 + .../SectionEffectList/SectionEffectList.ts | 84 + frontend/src/components/Sidebar/Sidebar.scss | 3 +- frontend/src/components/Sidebar/Sidebar.ts | 4 +- .../src/components/SourceList/SourceList.ts | 76 +- .../src/components/TimeInfo/TimeInfo.scss | 36 + frontend/src/components/ZoomBar/ZoomBar.scss | 21 + frontend/src/components/ZoomBar/ZoomBar.ts | 151 + frontend/src/components/index.js | 6 +- frontend/src/components/sidebar/Sidebar.ts | 27 - frontend/src/components/sidebar/sidebar.scss | 28 - frontend/src/controllers/CommandController.ts | 126 + frontend/src/controllers/controller.ts | 465 +- frontend/src/controllers/index.js | 1 + frontend/src/model/SectionDragStartData.ts | 19 + frontend/src/model/SelectTrackData.ts | 14 + frontend/src/model/TrackSection.ts | 34 +- frontend/src/model/index.js | 10 +- frontend/src/store/store.ts | 163 +- frontend/src/store/storeChannel.ts | 4 +- frontend/webpack.config.common.js | 22 +- 134 files changed, 4252 insertions(+), 12465 deletions(-) delete mode 100644 backend/.babelrc.js delete mode 100644 backend/.eslintrc.js delete mode 100644 backend/.prettierrc.js delete mode 100644 backend/package-lock.json delete mode 100644 backend/package.json delete mode 100644 backend/src/__test__/TransactionWrapper.js delete mode 100644 backend/src/__test__/router/api.test.js delete mode 100644 backend/src/__test__/router/auth.test.js delete mode 100644 backend/src/__test__/service/user-service.test.js delete mode 100644 backend/src/application/application-factory.js delete mode 100644 backend/src/application/application.js delete mode 100644 backend/src/application/index.js delete mode 100644 backend/src/common/config/database/connection-option-generator.js delete mode 100644 backend/src/common/config/database/database-type.js delete mode 100644 backend/src/common/env/database-env.js delete mode 100644 backend/src/common/env/env-type.js delete mode 100644 backend/src/common/error/business-error.js delete mode 100644 backend/src/common/error/entity-already-exist.js delete mode 100644 backend/src/common/error/entity-not-found-error.js delete mode 100644 backend/src/common/error/error-code.js delete mode 100644 backend/src/common/middleware/error-handler.js delete mode 100644 backend/src/common/middleware/index.js delete mode 100644 backend/src/common/middleware/request-type.js delete mode 100644 backend/src/common/middleware/transformer.js delete mode 100644 backend/src/common/middleware/validator.js delete mode 100644 backend/src/controller/index.js delete mode 100644 backend/src/controller/user-controller.js delete mode 100644 backend/src/dto/index.js delete mode 100644 backend/src/dto/signup-request-body.js delete mode 100644 backend/src/main.js delete mode 100644 backend/src/model/user.js delete mode 100644 backend/src/router/api.js delete mode 100644 backend/src/router/auth.js delete mode 100644 backend/src/router/index.js delete mode 100644 backend/src/router/oauth.js delete mode 100644 backend/src/service/index.js delete mode 100644 backend/src/service/user-service.js delete mode 100644 backend/tsconfig.json delete mode 100644 frontend/env/.env.sample create mode 100644 frontend/src/common/command/AddTrackCommand.ts create mode 100644 frontend/src/common/command/Command.ts create mode 100644 frontend/src/common/command/DeleteTrackCommand.ts delete mode 100644 frontend/src/common/command/ICommand.ts create mode 100644 frontend/src/common/command/MoveCommand.ts create mode 100644 frontend/src/common/command/PasteCommand.ts create mode 100644 frontend/src/common/command/SplitCommand.ts create mode 100644 frontend/src/common/command/index.js create mode 100644 frontend/src/common/types/trackOptionTypes.ts create mode 100644 frontend/src/common/types/trackSectionTypes.ts create mode 100644 frontend/src/common/util/DragUtil.ts delete mode 100644 frontend/src/common/util/PlayBarUtil.ts create mode 100644 frontend/src/common/util/SectionEffectListUtil.ts create mode 100644 frontend/src/common/util/TimeUtil.ts create mode 100644 frontend/src/common/util/ValidUtil.ts create mode 100644 frontend/src/common/util/WidthUtil.ts create mode 100644 frontend/src/components/AudioMeter/AudioMeter.scss create mode 100644 frontend/src/components/AudioMeter/AudioMeter.ts create mode 100644 frontend/src/components/AudioTrack/AudioTrackMenu/AudioTrackMenu.scss create mode 100644 frontend/src/components/AudioTrack/AudioTrackMenu/AudioTrackMenu.ts create mode 100644 frontend/src/components/AudioTrack/AudioTrackOption/AudioTrackOption.scss create mode 100644 frontend/src/components/AudioTrack/AudioTrackOption/AudioTrackOption.ts create mode 100644 frontend/src/components/Logo/Logo.scss create mode 100644 frontend/src/components/Main/MainAsideContent/MainAsideContent.scss create mode 100644 frontend/src/components/Main/MainAsideContent/MainAsideContent.ts create mode 100644 frontend/src/components/Main/MainMiddleContent/MainMiddleContent.scss create mode 100644 frontend/src/components/Main/MainMiddleContent/MainMiddleContent.ts create mode 100644 frontend/src/components/Main/MainMiddleContent/MainTrackOptionListArea/MainTrackOptionListArea.scss create mode 100644 frontend/src/components/Main/MainMiddleContent/MainTrackOptionListArea/MainTrackOptionListArea.ts create mode 100644 frontend/src/components/Main/MainRightContent/MainRightContent.scss create mode 100644 frontend/src/components/Main/MainRightContent/MainRightContent.ts create mode 100644 frontend/src/components/Main/MainRightContent/MainTrackListArea/MainTrackListArea.scss create mode 100644 frontend/src/components/Main/MainRightContent/MainTrackListArea/MainTrackListArea.ts create mode 100644 frontend/src/components/Main/index.js create mode 100644 frontend/src/components/PlayBar/PlayBarMarkerBlurZone/PlayBarMarkerBlurZone.scss create mode 100644 frontend/src/components/PlayBar/PlayBarMarkerBlurZone/PlayBarMarkerBlurZone.ts create mode 100644 frontend/src/components/PlaybackTools/PlaybackToolClass.ts create mode 100644 frontend/src/components/SectionEffectList/SectionEffectList.scss create mode 100644 frontend/src/components/SectionEffectList/SectionEffectList.ts create mode 100644 frontend/src/components/ZoomBar/ZoomBar.scss create mode 100644 frontend/src/components/ZoomBar/ZoomBar.ts delete mode 100644 frontend/src/components/sidebar/Sidebar.ts delete mode 100644 frontend/src/components/sidebar/sidebar.scss create mode 100644 frontend/src/controllers/CommandController.ts create mode 100644 frontend/src/model/SectionDragStartData.ts create mode 100644 frontend/src/model/SelectTrackData.ts diff --git a/README.md b/README.md index fc52df5e..8ec16e23 100644 --- a/README.md +++ b/README.md @@ -7,8 +7,6 @@ - -

@@ -20,24 +18,14 @@ ## [🎨UI Prototype](https://ovenapp.io/view/q5bGJWiAvVedUkqksVhGGjf8NpimvrDb/) -## [🎼Project URL](http://49.50.163.212/) +## [🎼Project URL](http://www.audi-editor.ml/) ## [🥝WIKI](https://github.com/boostcamp-2020/Project14-A-Web-Audio-Editor/wiki) ## 💻실행방법 +- 프로젝트 배포 링크 : [audi-editor.ml](audi-editor.ml) -frontend/env 폴더에 .env.development에 밑의 코드를 적은 파일을 생성합니다. -``` -ENVIRONMENT=development -``` -그 후 다음과 같은 명령어들을 실행합니다. -``` -cd backend -npm install - -cd ../frontend -npm install -npm run dev - -localhost:5500으로 접속 -``` +- Frontend + - 패키지 설치 : npm i + - 개발모드 웹팩 개발 서버 실행 : npm run dev + - 프로덕션모드 빌드 실행 : npm run build diff --git a/backend/.babelrc.js b/backend/.babelrc.js deleted file mode 100644 index 02c13592..00000000 --- a/backend/.babelrc.js +++ /dev/null @@ -1,10 +0,0 @@ -const presets = ["@babel/env"]; -const plugins = [ - "@babel/plugin-transform-runtime", - ["@babel/plugin-proposal-decorators", { legacy: true }], - ["@babel/plugin-proposal-class-properties", { loose: true }], - ["@babel/plugin-proposal-private-methods", { loose: true }], - "babel-plugin-parameter-decorator" -]; - -module.exports = { presets, plugins }; diff --git a/backend/.eslintrc.js b/backend/.eslintrc.js deleted file mode 100644 index d0cbc2de..00000000 --- a/backend/.eslintrc.js +++ /dev/null @@ -1,30 +0,0 @@ -module.exports = { - env: { - es2021: true, - node: true - }, - extends: ['airbnb-base', 'prettier/@typescript-eslint'], - parser: '@typescript-eslint/parser', - parserOptions: { - ecmaVersion: 12, - sourceType: 'module' - }, - plugins: ['prettier', '@typescript-eslint'], - settings: { - 'import/resolver': { - node: { - extensions: ['.js', '.ts'] - } - } - }, - rules: { - 'import/extensions': 'off', - 'no-undef': 'off', - 'no-param-reassign': 'off', - 'import/no-unresolved': 'off', - 'no-unused-vars': 'off', - 'comma-dangle': 'off', - 'import/prefer-default-export': 'off', - 'linebreak-style': 'off' - } -}; diff --git a/backend/.prettierrc.js b/backend/.prettierrc.js deleted file mode 100644 index f1d5a4bf..00000000 --- a/backend/.prettierrc.js +++ /dev/null @@ -1,14 +0,0 @@ -module.exports = { - printWidth: 150, - tabWidth: 2, - singleQuote: true, - trailingComma: "none", - bracketSpacing: true, - semi: true, - useTabs: false, - arrowParens: "always", - endOfLine: "lf", - orderedImports: true, - jsxBracketSameLine: true, - jsxSingleQuote: false -}; diff --git a/backend/package-lock.json b/backend/package-lock.json deleted file mode 100644 index 04a395d3..00000000 --- a/backend/package-lock.json +++ /dev/null @@ -1,10524 +0,0 @@ -{ - "name": "backend", - "version": "1.0.0", - "lockfileVersion": 1, - "requires": true, - "dependencies": { - "@babel/cli": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/cli/-/cli-7.12.1.tgz", - "integrity": "sha512-eRJREyrfAJ2r42Iaxe8h3v6yyj1wu9OyosaUHW6UImjGf9ahGL9nsFNh7OCopvtcPL8WnEo7tp78wrZaZ6vG9g==", - "dev": true, - "requires": { - "@nicolo-ribaudo/chokidar-2": "^2.1.8", - "chokidar": "^3.4.0", - "commander": "^4.0.1", - "convert-source-map": "^1.1.0", - "fs-readdir-recursive": "^1.1.0", - "glob": "^7.0.0", - "lodash": "^4.17.19", - "make-dir": "^2.1.0", - "slash": "^2.0.0", - "source-map": "^0.5.0" - } - }, - "@babel/code-frame": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.10.4.tgz", - "integrity": "sha512-vG6SvB6oYEhvgisZNFRmRCUkLz11c7rp+tbNTynGqc6mS1d5ATd/sGyV6W0KZZnXRKMTzZDRgQT3Ou9jhpAfUg==", - "dev": true, - "requires": { - "@babel/highlight": "^7.10.4" - } - }, - "@babel/compat-data": { - "version": "7.12.5", - "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.12.5.tgz", - "integrity": "sha512-DTsS7cxrsH3by8nqQSpFSyjSfSYl57D6Cf4q8dW3LK83tBKBDCkfcay1nYkXq1nIHXnpX8WMMb/O25HOy3h1zg==", - "dev": true - }, - "@babel/core": { - "version": "7.12.3", - "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.12.3.tgz", - "integrity": "sha512-0qXcZYKZp3/6N2jKYVxZv0aNCsxTSVCiK72DTiTYZAu7sjg73W0/aynWjMbiGd87EQL4WyA8reiJVh92AVla9g==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.10.4", - "@babel/generator": "^7.12.1", - "@babel/helper-module-transforms": "^7.12.1", - "@babel/helpers": "^7.12.1", - "@babel/parser": "^7.12.3", - "@babel/template": "^7.10.4", - "@babel/traverse": "^7.12.1", - "@babel/types": "^7.12.1", - "convert-source-map": "^1.7.0", - "debug": "^4.1.0", - "gensync": "^1.0.0-beta.1", - "json5": "^2.1.2", - "lodash": "^4.17.19", - "resolve": "^1.3.2", - "semver": "^5.4.1", - "source-map": "^0.5.0" - }, - "dependencies": { - "debug": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", - "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", - "dev": true, - "requires": { - "ms": "2.1.2" - } - } - } - }, - "@babel/eslint-parser": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/eslint-parser/-/eslint-parser-7.12.1.tgz", - "integrity": "sha512-cc7WQHnHQY3++/bghgbDtPx+5bf6xTsokyGzV6Qzh65NLz/unv+mPQuACkQ9GFhIhcTFv6yqwNaEcfX7EkOEsg==", - "dev": true, - "requires": { - "eslint-scope": "5.1.0", - "eslint-visitor-keys": "^1.3.0", - "semver": "^6.3.0" - }, - "dependencies": { - "semver": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", - "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", - "dev": true - } - } - }, - "@babel/generator": { - "version": "7.12.5", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.12.5.tgz", - "integrity": "sha512-m16TQQJ8hPt7E+OS/XVQg/7U184MLXtvuGbCdA7na61vha+ImkyyNM/9DDA0unYCVZn3ZOhng+qz48/KBOT96A==", - "dev": true, - "requires": { - "@babel/types": "^7.12.5", - "jsesc": "^2.5.1", - "source-map": "^0.5.0" - } - }, - "@babel/helper-annotate-as-pure": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.10.4.tgz", - "integrity": "sha512-XQlqKQP4vXFB7BN8fEEerrmYvHp3fK/rBkRFz9jaJbzK0B1DSfej9Kc7ZzE8Z/OnId1jpJdNAZ3BFQjWG68rcA==", - "dev": true, - "requires": { - "@babel/types": "^7.10.4" - } - }, - "@babel/helper-builder-binary-assignment-operator-visitor": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-builder-binary-assignment-operator-visitor/-/helper-builder-binary-assignment-operator-visitor-7.10.4.tgz", - "integrity": "sha512-L0zGlFrGWZK4PbT8AszSfLTM5sDU1+Az/En9VrdT8/LmEiJt4zXt+Jve9DCAnQcbqDhCI+29y/L93mrDzddCcg==", - "dev": true, - "requires": { - "@babel/helper-explode-assignable-expression": "^7.10.4", - "@babel/types": "^7.10.4" - } - }, - "@babel/helper-compilation-targets": { - "version": "7.12.5", - "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.12.5.tgz", - "integrity": "sha512-+qH6NrscMolUlzOYngSBMIOQpKUGPPsc61Bu5W10mg84LxZ7cmvnBHzARKbDoFxVvqqAbj6Tg6N7bSrWSPXMyw==", - "dev": true, - "requires": { - "@babel/compat-data": "^7.12.5", - "@babel/helper-validator-option": "^7.12.1", - "browserslist": "^4.14.5", - "semver": "^5.5.0" - } - }, - "@babel/helper-create-class-features-plugin": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.12.1.tgz", - "integrity": "sha512-hkL++rWeta/OVOBTRJc9a5Azh5mt5WgZUGAKMD8JM141YsE08K//bp1unBBieO6rUKkIPyUE0USQ30jAy3Sk1w==", - "dev": true, - "requires": { - "@babel/helper-function-name": "^7.10.4", - "@babel/helper-member-expression-to-functions": "^7.12.1", - "@babel/helper-optimise-call-expression": "^7.10.4", - "@babel/helper-replace-supers": "^7.12.1", - "@babel/helper-split-export-declaration": "^7.10.4" - } - }, - "@babel/helper-create-regexp-features-plugin": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.12.1.tgz", - "integrity": "sha512-rsZ4LGvFTZnzdNZR5HZdmJVuXK8834R5QkF3WvcnBhrlVtF0HSIUC6zbreL9MgjTywhKokn8RIYRiq99+DLAxA==", - "dev": true, - "requires": { - "@babel/helper-annotate-as-pure": "^7.10.4", - "@babel/helper-regex": "^7.10.4", - "regexpu-core": "^4.7.1" - } - }, - "@babel/helper-define-map": { - "version": "7.10.5", - "resolved": "https://registry.npmjs.org/@babel/helper-define-map/-/helper-define-map-7.10.5.tgz", - "integrity": "sha512-fMw4kgFB720aQFXSVaXr79pjjcW5puTCM16+rECJ/plGS+zByelE8l9nCpV1GibxTnFVmUuYG9U8wYfQHdzOEQ==", - "dev": true, - "requires": { - "@babel/helper-function-name": "^7.10.4", - "@babel/types": "^7.10.5", - "lodash": "^4.17.19" - } - }, - "@babel/helper-explode-assignable-expression": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-explode-assignable-expression/-/helper-explode-assignable-expression-7.12.1.tgz", - "integrity": "sha512-dmUwH8XmlrUpVqgtZ737tK88v07l840z9j3OEhCLwKTkjlvKpfqXVIZ0wpK3aeOxspwGrf/5AP5qLx4rO3w5rA==", - "dev": true, - "requires": { - "@babel/types": "^7.12.1" - } - }, - "@babel/helper-function-name": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.10.4.tgz", - "integrity": "sha512-YdaSyz1n8gY44EmN7x44zBn9zQ1Ry2Y+3GTA+3vH6Mizke1Vw0aWDM66FOYEPw8//qKkmqOckrGgTYa+6sceqQ==", - "dev": true, - "requires": { - "@babel/helper-get-function-arity": "^7.10.4", - "@babel/template": "^7.10.4", - "@babel/types": "^7.10.4" - } - }, - "@babel/helper-get-function-arity": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-get-function-arity/-/helper-get-function-arity-7.10.4.tgz", - "integrity": "sha512-EkN3YDB+SRDgiIUnNgcmiD361ti+AVbL3f3Henf6dqqUyr5dMsorno0lJWJuLhDhkI5sYEpgj6y9kB8AOU1I2A==", - "dev": true, - "requires": { - "@babel/types": "^7.10.4" - } - }, - "@babel/helper-hoist-variables": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.10.4.tgz", - "integrity": "sha512-wljroF5PgCk2juF69kanHVs6vrLwIPNp6DLD+Lrl3hoQ3PpPPikaDRNFA+0t81NOoMt2DL6WW/mdU8k4k6ZzuA==", - "dev": true, - "requires": { - "@babel/types": "^7.10.4" - } - }, - "@babel/helper-member-expression-to-functions": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.12.1.tgz", - "integrity": "sha512-k0CIe3tXUKTRSoEx1LQEPFU9vRQfqHtl+kf8eNnDqb4AUJEy5pz6aIiog+YWtVm2jpggjS1laH68bPsR+KWWPQ==", - "dev": true, - "requires": { - "@babel/types": "^7.12.1" - } - }, - "@babel/helper-module-imports": { - "version": "7.12.5", - "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.12.5.tgz", - "integrity": "sha512-SR713Ogqg6++uexFRORf/+nPXMmWIn80TALu0uaFb+iQIUoR7bOC7zBWyzBs5b3tBBJXuyD0cRu1F15GyzjOWA==", - "dev": true, - "requires": { - "@babel/types": "^7.12.5" - } - }, - "@babel/helper-module-transforms": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.12.1.tgz", - "integrity": "sha512-QQzehgFAZ2bbISiCpmVGfiGux8YVFXQ0abBic2Envhej22DVXV9nCFaS5hIQbkyo1AdGb+gNME2TSh3hYJVV/w==", - "dev": true, - "requires": { - "@babel/helper-module-imports": "^7.12.1", - "@babel/helper-replace-supers": "^7.12.1", - "@babel/helper-simple-access": "^7.12.1", - "@babel/helper-split-export-declaration": "^7.11.0", - "@babel/helper-validator-identifier": "^7.10.4", - "@babel/template": "^7.10.4", - "@babel/traverse": "^7.12.1", - "@babel/types": "^7.12.1", - "lodash": "^4.17.19" - } - }, - "@babel/helper-optimise-call-expression": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.10.4.tgz", - "integrity": "sha512-n3UGKY4VXwXThEiKrgRAoVPBMqeoPgHVqiHZOanAJCG9nQUL2pLRQirUzl0ioKclHGpGqRgIOkgcIJaIWLpygg==", - "dev": true, - "requires": { - "@babel/types": "^7.10.4" - } - }, - "@babel/helper-plugin-utils": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.10.4.tgz", - "integrity": "sha512-O4KCvQA6lLiMU9l2eawBPMf1xPP8xPfB3iEQw150hOVTqj/rfXz0ThTb4HEzqQfs2Bmo5Ay8BzxfzVtBrr9dVg==", - "dev": true - }, - "@babel/helper-regex": { - "version": "7.10.5", - "resolved": "https://registry.npmjs.org/@babel/helper-regex/-/helper-regex-7.10.5.tgz", - "integrity": "sha512-68kdUAzDrljqBrio7DYAEgCoJHxppJOERHOgOrDN7WjOzP0ZQ1LsSDRXcemzVZaLvjaJsJEESb6qt+znNuENDg==", - "dev": true, - "requires": { - "lodash": "^4.17.19" - } - }, - "@babel/helper-remap-async-to-generator": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.12.1.tgz", - "integrity": "sha512-9d0KQCRM8clMPcDwo8SevNs+/9a8yWVVmaE80FGJcEP8N1qToREmWEGnBn8BUlJhYRFz6fqxeRL1sl5Ogsed7A==", - "dev": true, - "requires": { - "@babel/helper-annotate-as-pure": "^7.10.4", - "@babel/helper-wrap-function": "^7.10.4", - "@babel/types": "^7.12.1" - } - }, - "@babel/helper-replace-supers": { - "version": "7.12.5", - "resolved": "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.12.5.tgz", - "integrity": "sha512-5YILoed0ZyIpF4gKcpZitEnXEJ9UoDRki1Ey6xz46rxOzfNMAhVIJMoune1hmPVxh40LRv1+oafz7UsWX+vyWA==", - "dev": true, - "requires": { - "@babel/helper-member-expression-to-functions": "^7.12.1", - "@babel/helper-optimise-call-expression": "^7.10.4", - "@babel/traverse": "^7.12.5", - "@babel/types": "^7.12.5" - } - }, - "@babel/helper-simple-access": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.12.1.tgz", - "integrity": "sha512-OxBp7pMrjVewSSC8fXDFrHrBcJATOOFssZwv16F3/6Xtc138GHybBfPbm9kfiqQHKhYQrlamWILwlDCeyMFEaA==", - "dev": true, - "requires": { - "@babel/types": "^7.12.1" - } - }, - "@babel/helper-skip-transparent-expression-wrappers": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-skip-transparent-expression-wrappers/-/helper-skip-transparent-expression-wrappers-7.12.1.tgz", - "integrity": "sha512-Mf5AUuhG1/OCChOJ/HcADmvcHM42WJockombn8ATJG3OnyiSxBK/Mm5x78BQWvmtXZKHgbjdGL2kin/HOLlZGA==", - "dev": true, - "requires": { - "@babel/types": "^7.12.1" - } - }, - "@babel/helper-split-export-declaration": { - "version": "7.11.0", - "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.11.0.tgz", - "integrity": "sha512-74Vejvp6mHkGE+m+k5vHY93FX2cAtrw1zXrZXRlG4l410Nm9PxfEiVTn1PjDPV5SnmieiueY4AFg2xqhNFuuZg==", - "dev": true, - "requires": { - "@babel/types": "^7.11.0" - } - }, - "@babel/helper-validator-identifier": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.10.4.tgz", - "integrity": "sha512-3U9y+43hz7ZM+rzG24Qe2mufW5KhvFg/NhnNph+i9mgCtdTCtMJuI1TMkrIUiK7Ix4PYlRF9I5dhqaLYA/ADXw==", - "dev": true - }, - "@babel/helper-validator-option": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.12.1.tgz", - "integrity": "sha512-YpJabsXlJVWP0USHjnC/AQDTLlZERbON577YUVO/wLpqyj6HAtVYnWaQaN0iUN+1/tWn3c+uKKXjRut5115Y2A==", - "dev": true - }, - "@babel/helper-wrap-function": { - "version": "7.12.3", - "resolved": "https://registry.npmjs.org/@babel/helper-wrap-function/-/helper-wrap-function-7.12.3.tgz", - "integrity": "sha512-Cvb8IuJDln3rs6tzjW3Y8UeelAOdnpB8xtQ4sme2MSZ9wOxrbThporC0y/EtE16VAtoyEfLM404Xr1e0OOp+ow==", - "dev": true, - "requires": { - "@babel/helper-function-name": "^7.10.4", - "@babel/template": "^7.10.4", - "@babel/traverse": "^7.10.4", - "@babel/types": "^7.10.4" - } - }, - "@babel/helpers": { - "version": "7.12.5", - "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.12.5.tgz", - "integrity": "sha512-lgKGMQlKqA8meJqKsW6rUnc4MdUk35Ln0ATDqdM1a/UpARODdI4j5Y5lVfUScnSNkJcdCRAaWkspykNoFg9sJA==", - "dev": true, - "requires": { - "@babel/template": "^7.10.4", - "@babel/traverse": "^7.12.5", - "@babel/types": "^7.12.5" - } - }, - "@babel/highlight": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.10.4.tgz", - "integrity": "sha512-i6rgnR/YgPEQzZZnbTHHuZdlE8qyoBNalD6F+q4vAFlcMEcqmkoG+mPqJYJCo63qPf74+Y1UZsl3l6f7/RIkmA==", - "dev": true, - "requires": { - "@babel/helper-validator-identifier": "^7.10.4", - "chalk": "^2.0.0", - "js-tokens": "^4.0.0" - }, - "dependencies": { - "ansi-styles": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", - "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", - "dev": true, - "requires": { - "color-convert": "^1.9.0" - } - }, - "chalk": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", - "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", - "dev": true, - "requires": { - "ansi-styles": "^3.2.1", - "escape-string-regexp": "^1.0.5", - "supports-color": "^5.3.0" - } - }, - "color-convert": { - "version": "1.9.3", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", - "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", - "dev": true, - "requires": { - "color-name": "1.1.3" - } - }, - "color-name": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", - "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", - "dev": true - }, - "has-flag": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", - "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", - "dev": true - }, - "supports-color": { - "version": "5.5.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", - "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", - "dev": true, - "requires": { - "has-flag": "^3.0.0" - } - } - } - }, - "@babel/node": { - "version": "7.12.6", - "resolved": "https://registry.npmjs.org/@babel/node/-/node-7.12.6.tgz", - "integrity": "sha512-A1TpW2X05ZkI5+WV7Aa24QX4LyGwrGUQPflG1CyBdr84jUuH0mhkE2BQWSQAlfRnp4bMLjeveMJIhS20JaOfVQ==", - "dev": true, - "requires": { - "@babel/register": "^7.12.1", - "commander": "^4.0.1", - "core-js": "^3.2.1", - "lodash": "^4.17.19", - "node-environment-flags": "^1.0.5", - "regenerator-runtime": "^0.13.4", - "resolve": "^1.13.1", - "v8flags": "^3.1.1" - } - }, - "@babel/parser": { - "version": "7.12.5", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.12.5.tgz", - "integrity": "sha512-FVM6RZQ0mn2KCf1VUED7KepYeUWoVShczewOCfm3nzoBybaih51h+sYVVGthW9M6lPByEPTQf+xm27PBdlpwmQ==", - "dev": true - }, - "@babel/plugin-proposal-async-generator-functions": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-async-generator-functions/-/plugin-proposal-async-generator-functions-7.12.1.tgz", - "integrity": "sha512-d+/o30tJxFxrA1lhzJqiUcEJdI6jKlNregCv5bASeGf2Q4MXmnwH7viDo7nhx1/ohf09oaH8j1GVYG/e3Yqk6A==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-remap-async-to-generator": "^7.12.1", - "@babel/plugin-syntax-async-generators": "^7.8.0" - } - }, - "@babel/plugin-proposal-class-properties": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-class-properties/-/plugin-proposal-class-properties-7.12.1.tgz", - "integrity": "sha512-cKp3dlQsFsEs5CWKnN7BnSHOd0EOW8EKpEjkoz1pO2E5KzIDNV9Ros1b0CnmbVgAGXJubOYVBOGCT1OmJwOI7w==", - "dev": true, - "requires": { - "@babel/helper-create-class-features-plugin": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-proposal-decorators": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-decorators/-/plugin-proposal-decorators-7.12.1.tgz", - "integrity": "sha512-knNIuusychgYN8fGJHONL0RbFxLGawhXOJNLBk75TniTsZZeA+wdkDuv6wp4lGwzQEKjZi6/WYtnb3udNPmQmQ==", - "dev": true, - "requires": { - "@babel/helper-create-class-features-plugin": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/plugin-syntax-decorators": "^7.12.1" - } - }, - "@babel/plugin-proposal-dynamic-import": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-dynamic-import/-/plugin-proposal-dynamic-import-7.12.1.tgz", - "integrity": "sha512-a4rhUSZFuq5W8/OO8H7BL5zspjnc1FLd9hlOxIK/f7qG4a0qsqk8uvF/ywgBA8/OmjsapjpvaEOYItfGG1qIvQ==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/plugin-syntax-dynamic-import": "^7.8.0" - } - }, - "@babel/plugin-proposal-export-namespace-from": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-export-namespace-from/-/plugin-proposal-export-namespace-from-7.12.1.tgz", - "integrity": "sha512-6CThGf0irEkzujYS5LQcjBx8j/4aQGiVv7J9+2f7pGfxqyKh3WnmVJYW3hdrQjyksErMGBPQrCnHfOtna+WLbw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/plugin-syntax-export-namespace-from": "^7.8.3" - } - }, - "@babel/plugin-proposal-json-strings": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-json-strings/-/plugin-proposal-json-strings-7.12.1.tgz", - "integrity": "sha512-GoLDUi6U9ZLzlSda2Df++VSqDJg3CG+dR0+iWsv6XRw1rEq+zwt4DirM9yrxW6XWaTpmai1cWJLMfM8qQJf+yw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/plugin-syntax-json-strings": "^7.8.0" - } - }, - "@babel/plugin-proposal-logical-assignment-operators": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-logical-assignment-operators/-/plugin-proposal-logical-assignment-operators-7.12.1.tgz", - "integrity": "sha512-k8ZmVv0JU+4gcUGeCDZOGd0lCIamU/sMtIiX3UWnUc5yzgq6YUGyEolNYD+MLYKfSzgECPcqetVcJP9Afe/aCA==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/plugin-syntax-logical-assignment-operators": "^7.10.4" - } - }, - "@babel/plugin-proposal-nullish-coalescing-operator": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-nullish-coalescing-operator/-/plugin-proposal-nullish-coalescing-operator-7.12.1.tgz", - "integrity": "sha512-nZY0ESiaQDI1y96+jk6VxMOaL4LPo/QDHBqL+SF3/vl6dHkTwHlOI8L4ZwuRBHgakRBw5zsVylel7QPbbGuYgg==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.0" - } - }, - "@babel/plugin-proposal-numeric-separator": { - "version": "7.12.5", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-numeric-separator/-/plugin-proposal-numeric-separator-7.12.5.tgz", - "integrity": "sha512-UiAnkKuOrCyjZ3sYNHlRlfuZJbBHknMQ9VMwVeX97Ofwx7RpD6gS2HfqTCh8KNUQgcOm8IKt103oR4KIjh7Q8g==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/plugin-syntax-numeric-separator": "^7.10.4" - } - }, - "@babel/plugin-proposal-object-rest-spread": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-object-rest-spread/-/plugin-proposal-object-rest-spread-7.12.1.tgz", - "integrity": "sha512-s6SowJIjzlhx8o7lsFx5zmY4At6CTtDvgNQDdPzkBQucle58A6b/TTeEBYtyDgmcXjUTM+vE8YOGHZzzbc/ioA==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/plugin-syntax-object-rest-spread": "^7.8.0", - "@babel/plugin-transform-parameters": "^7.12.1" - } - }, - "@babel/plugin-proposal-optional-catch-binding": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-catch-binding/-/plugin-proposal-optional-catch-binding-7.12.1.tgz", - "integrity": "sha512-hFvIjgprh9mMw5v42sJWLI1lzU5L2sznP805zeT6rySVRA0Y18StRhDqhSxlap0oVgItRsB6WSROp4YnJTJz0g==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/plugin-syntax-optional-catch-binding": "^7.8.0" - } - }, - "@babel/plugin-proposal-optional-chaining": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-chaining/-/plugin-proposal-optional-chaining-7.12.1.tgz", - "integrity": "sha512-c2uRpY6WzaVDzynVY9liyykS+kVU+WRZPMPYpkelXH8KBt1oXoI89kPbZKKG/jDT5UK92FTW2fZkZaJhdiBabw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-skip-transparent-expression-wrappers": "^7.12.1", - "@babel/plugin-syntax-optional-chaining": "^7.8.0" - } - }, - "@babel/plugin-proposal-private-methods": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-private-methods/-/plugin-proposal-private-methods-7.12.1.tgz", - "integrity": "sha512-mwZ1phvH7/NHK6Kf8LP7MYDogGV+DKB1mryFOEwx5EBNQrosvIczzZFTUmWaeujd5xT6G1ELYWUz3CutMhjE1w==", - "dev": true, - "requires": { - "@babel/helper-create-class-features-plugin": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-proposal-unicode-property-regex": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-unicode-property-regex/-/plugin-proposal-unicode-property-regex-7.12.1.tgz", - "integrity": "sha512-MYq+l+PvHuw/rKUz1at/vb6nCnQ2gmJBNaM62z0OgH7B2W1D9pvkpYtlti9bGtizNIU1K3zm4bZF9F91efVY0w==", - "dev": true, - "requires": { - "@babel/helper-create-regexp-features-plugin": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-syntax-async-generators": { - "version": "7.8.4", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz", - "integrity": "sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.0" - } - }, - "@babel/plugin-syntax-bigint": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-bigint/-/plugin-syntax-bigint-7.8.3.tgz", - "integrity": "sha512-wnTnFlG+YxQm3vDxpGE57Pj0srRU4sHE/mDkt1qv2YJJSeUAec2ma4WLUnUPeKjyrfntVwe/N6dCXpU+zL3Npg==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.0" - } - }, - "@babel/plugin-syntax-class-properties": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-properties/-/plugin-syntax-class-properties-7.12.1.tgz", - "integrity": "sha512-U40A76x5gTwmESz+qiqssqmeEsKvcSyvtgktrm0uzcARAmM9I1jR221f6Oq+GmHrcD+LvZDag1UTOTe2fL3TeA==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-syntax-decorators": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-decorators/-/plugin-syntax-decorators-7.12.1.tgz", - "integrity": "sha512-ir9YW5daRrTYiy9UJ2TzdNIJEZu8KclVzDcfSt4iEmOtwQ4llPtWInNKJyKnVXp1vE4bbVd5S31M/im3mYMO1w==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-syntax-dynamic-import": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-dynamic-import/-/plugin-syntax-dynamic-import-7.8.3.tgz", - "integrity": "sha512-5gdGbFon+PszYzqs83S3E5mpi7/y/8M9eC90MRTZfduQOYW76ig6SOSPNe41IG5LoP3FGBn2N0RjVDSQiS94kQ==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.0" - } - }, - "@babel/plugin-syntax-export-namespace-from": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-export-namespace-from/-/plugin-syntax-export-namespace-from-7.8.3.tgz", - "integrity": "sha512-MXf5laXo6c1IbEbegDmzGPwGNTsHZmEy6QGznu5Sh2UCWvueywb2ee+CCE4zQiZstxU9BMoQO9i6zUFSY0Kj0Q==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.3" - } - }, - "@babel/plugin-syntax-import-meta": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-meta/-/plugin-syntax-import-meta-7.10.4.tgz", - "integrity": "sha512-Yqfm+XDx0+Prh3VSeEQCPU81yC+JWZ2pDPFSS4ZdpfZhp4MkFMaDC1UqseovEKwSUpnIL7+vK+Clp7bfh0iD7g==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-syntax-json-strings": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.8.3.tgz", - "integrity": "sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.0" - } - }, - "@babel/plugin-syntax-logical-assignment-operators": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-logical-assignment-operators/-/plugin-syntax-logical-assignment-operators-7.10.4.tgz", - "integrity": "sha512-d8waShlpFDinQ5MtvGU9xDAOzKH47+FFoney2baFIoMr952hKOLp1HR7VszoZvOsV/4+RRszNY7D17ba0te0ig==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-syntax-nullish-coalescing-operator": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-nullish-coalescing-operator/-/plugin-syntax-nullish-coalescing-operator-7.8.3.tgz", - "integrity": "sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.0" - } - }, - "@babel/plugin-syntax-numeric-separator": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-numeric-separator/-/plugin-syntax-numeric-separator-7.10.4.tgz", - "integrity": "sha512-9H6YdfkcK/uOnY/K7/aA2xpzaAgkQn37yzWUMRK7OaPOqOpGS1+n0H5hxT9AUw9EsSjPW8SVyMJwYRtWs3X3ug==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-syntax-object-rest-spread": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.8.3.tgz", - "integrity": "sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.0" - } - }, - "@babel/plugin-syntax-optional-catch-binding": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.8.3.tgz", - "integrity": "sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.0" - } - }, - "@babel/plugin-syntax-optional-chaining": { - "version": "7.8.3", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-chaining/-/plugin-syntax-optional-chaining-7.8.3.tgz", - "integrity": "sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.8.0" - } - }, - "@babel/plugin-syntax-top-level-await": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.12.1.tgz", - "integrity": "sha512-i7ooMZFS+a/Om0crxZodrTzNEPJHZrlMVGMTEpFAj6rYY/bKCddB0Dk/YxfPuYXOopuhKk/e1jV6h+WUU9XN3A==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-arrow-functions": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-arrow-functions/-/plugin-transform-arrow-functions-7.12.1.tgz", - "integrity": "sha512-5QB50qyN44fzzz4/qxDPQMBCTHgxg3n0xRBLJUmBlLoU/sFvxVWGZF/ZUfMVDQuJUKXaBhbupxIzIfZ6Fwk/0A==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-async-to-generator": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-async-to-generator/-/plugin-transform-async-to-generator-7.12.1.tgz", - "integrity": "sha512-SDtqoEcarK1DFlRJ1hHRY5HvJUj5kX4qmtpMAm2QnhOlyuMC4TMdCRgW6WXpv93rZeYNeLP22y8Aq2dbcDRM1A==", - "dev": true, - "requires": { - "@babel/helper-module-imports": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-remap-async-to-generator": "^7.12.1" - } - }, - "@babel/plugin-transform-block-scoped-functions": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoped-functions/-/plugin-transform-block-scoped-functions-7.12.1.tgz", - "integrity": "sha512-5OpxfuYnSgPalRpo8EWGPzIYf0lHBWORCkj5M0oLBwHdlux9Ri36QqGW3/LR13RSVOAoUUMzoPI/jpE4ABcHoA==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-block-scoping": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoping/-/plugin-transform-block-scoping-7.12.1.tgz", - "integrity": "sha512-zJyAC9sZdE60r1nVQHblcfCj29Dh2Y0DOvlMkcqSo0ckqjiCwNiUezUKw+RjOCwGfpLRwnAeQ2XlLpsnGkvv9w==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-classes": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-classes/-/plugin-transform-classes-7.12.1.tgz", - "integrity": "sha512-/74xkA7bVdzQTBeSUhLLJgYIcxw/dpEpCdRDiHgPJ3Mv6uC11UhjpOhl72CgqbBCmt1qtssCyB2xnJm1+PFjog==", - "dev": true, - "requires": { - "@babel/helper-annotate-as-pure": "^7.10.4", - "@babel/helper-define-map": "^7.10.4", - "@babel/helper-function-name": "^7.10.4", - "@babel/helper-optimise-call-expression": "^7.10.4", - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-replace-supers": "^7.12.1", - "@babel/helper-split-export-declaration": "^7.10.4", - "globals": "^11.1.0" - } - }, - "@babel/plugin-transform-computed-properties": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-computed-properties/-/plugin-transform-computed-properties-7.12.1.tgz", - "integrity": "sha512-vVUOYpPWB7BkgUWPo4C44mUQHpTZXakEqFjbv8rQMg7TC6S6ZhGZ3otQcRH6u7+adSlE5i0sp63eMC/XGffrzg==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-destructuring": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.12.1.tgz", - "integrity": "sha512-fRMYFKuzi/rSiYb2uRLiUENJOKq4Gnl+6qOv5f8z0TZXg3llUwUhsNNwrwaT/6dUhJTzNpBr+CUvEWBtfNY1cw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-dotall-regex": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.12.1.tgz", - "integrity": "sha512-B2pXeRKoLszfEW7J4Hg9LoFaWEbr/kzo3teWHmtFCszjRNa/b40f9mfeqZsIDLLt/FjwQ6pz/Gdlwy85xNckBA==", - "dev": true, - "requires": { - "@babel/helper-create-regexp-features-plugin": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-duplicate-keys": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-duplicate-keys/-/plugin-transform-duplicate-keys-7.12.1.tgz", - "integrity": "sha512-iRght0T0HztAb/CazveUpUQrZY+aGKKaWXMJ4uf9YJtqxSUe09j3wteztCUDRHs+SRAL7yMuFqUsLoAKKzgXjw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-exponentiation-operator": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.12.1.tgz", - "integrity": "sha512-7tqwy2bv48q+c1EHbXK0Zx3KXd2RVQp6OC7PbwFNt/dPTAV3Lu5sWtWuAj8owr5wqtWnqHfl2/mJlUmqkChKug==", - "dev": true, - "requires": { - "@babel/helper-builder-binary-assignment-operator-visitor": "^7.10.4", - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-for-of": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-for-of/-/plugin-transform-for-of-7.12.1.tgz", - "integrity": "sha512-Zaeq10naAsuHo7heQvyV0ptj4dlZJwZgNAtBYBnu5nNKJoW62m0zKcIEyVECrUKErkUkg6ajMy4ZfnVZciSBhg==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-function-name": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-function-name/-/plugin-transform-function-name-7.12.1.tgz", - "integrity": "sha512-JF3UgJUILoFrFMEnOJLJkRHSk6LUSXLmEFsA23aR2O5CSLUxbeUX1IZ1YQ7Sn0aXb601Ncwjx73a+FVqgcljVw==", - "dev": true, - "requires": { - "@babel/helper-function-name": "^7.10.4", - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-literals": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-literals/-/plugin-transform-literals-7.12.1.tgz", - "integrity": "sha512-+PxVGA+2Ag6uGgL0A5f+9rklOnnMccwEBzwYFL3EUaKuiyVnUipyXncFcfjSkbimLrODoqki1U9XxZzTvfN7IQ==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-member-expression-literals": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-member-expression-literals/-/plugin-transform-member-expression-literals-7.12.1.tgz", - "integrity": "sha512-1sxePl6z9ad0gFMB9KqmYofk34flq62aqMt9NqliS/7hPEpURUCMbyHXrMPlo282iY7nAvUB1aQd5mg79UD9Jg==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-modules-amd": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-amd/-/plugin-transform-modules-amd-7.12.1.tgz", - "integrity": "sha512-tDW8hMkzad5oDtzsB70HIQQRBiTKrhfgwC/KkJeGsaNFTdWhKNt/BiE8c5yj19XiGyrxpbkOfH87qkNg1YGlOQ==", - "dev": true, - "requires": { - "@babel/helper-module-transforms": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4", - "babel-plugin-dynamic-import-node": "^2.3.3" - } - }, - "@babel/plugin-transform-modules-commonjs": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.12.1.tgz", - "integrity": "sha512-dY789wq6l0uLY8py9c1B48V8mVL5gZh/+PQ5ZPrylPYsnAvnEMjqsUXkuoDVPeVK+0VyGar+D08107LzDQ6pag==", - "dev": true, - "requires": { - "@babel/helper-module-transforms": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-simple-access": "^7.12.1", - "babel-plugin-dynamic-import-node": "^2.3.3" - } - }, - "@babel/plugin-transform-modules-systemjs": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-systemjs/-/plugin-transform-modules-systemjs-7.12.1.tgz", - "integrity": "sha512-Hn7cVvOavVh8yvW6fLwveFqSnd7rbQN3zJvoPNyNaQSvgfKmDBO9U1YL9+PCXGRlZD9tNdWTy5ACKqMuzyn32Q==", - "dev": true, - "requires": { - "@babel/helper-hoist-variables": "^7.10.4", - "@babel/helper-module-transforms": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-validator-identifier": "^7.10.4", - "babel-plugin-dynamic-import-node": "^2.3.3" - } - }, - "@babel/plugin-transform-modules-umd": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-umd/-/plugin-transform-modules-umd-7.12.1.tgz", - "integrity": "sha512-aEIubCS0KHKM0zUos5fIoQm+AZUMt1ZvMpqz0/H5qAQ7vWylr9+PLYurT+Ic7ID/bKLd4q8hDovaG3Zch2uz5Q==", - "dev": true, - "requires": { - "@babel/helper-module-transforms": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-named-capturing-groups-regex": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.12.1.tgz", - "integrity": "sha512-tB43uQ62RHcoDp9v2Nsf+dSM8sbNodbEicbQNA53zHz8pWUhsgHSJCGpt7daXxRydjb0KnfmB+ChXOv3oADp1Q==", - "dev": true, - "requires": { - "@babel/helper-create-regexp-features-plugin": "^7.12.1" - } - }, - "@babel/plugin-transform-new-target": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-new-target/-/plugin-transform-new-target-7.12.1.tgz", - "integrity": "sha512-+eW/VLcUL5L9IvJH7rT1sT0CzkdUTvPrXC2PXTn/7z7tXLBuKvezYbGdxD5WMRoyvyaujOq2fWoKl869heKjhw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-object-super": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-object-super/-/plugin-transform-object-super-7.12.1.tgz", - "integrity": "sha512-AvypiGJH9hsquNUn+RXVcBdeE3KHPZexWRdimhuV59cSoOt5kFBmqlByorAeUlGG2CJWd0U+4ZtNKga/TB0cAw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-replace-supers": "^7.12.1" - } - }, - "@babel/plugin-transform-parameters": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-parameters/-/plugin-transform-parameters-7.12.1.tgz", - "integrity": "sha512-xq9C5EQhdPK23ZeCdMxl8bbRnAgHFrw5EOC3KJUsSylZqdkCaFEXxGSBuTSObOpiiHHNyb82es8M1QYgfQGfNg==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-property-literals": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-property-literals/-/plugin-transform-property-literals-7.12.1.tgz", - "integrity": "sha512-6MTCR/mZ1MQS+AwZLplX4cEySjCpnIF26ToWo942nqn8hXSm7McaHQNeGx/pt7suI1TWOWMfa/NgBhiqSnX0cQ==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-regenerator": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.12.1.tgz", - "integrity": "sha512-gYrHqs5itw6i4PflFX3OdBPMQdPbF4bj2REIUxlMRUFk0/ZOAIpDFuViuxPjUL7YC8UPnf+XG7/utJvqXdPKng==", - "dev": true, - "requires": { - "regenerator-transform": "^0.14.2" - } - }, - "@babel/plugin-transform-reserved-words": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-reserved-words/-/plugin-transform-reserved-words-7.12.1.tgz", - "integrity": "sha512-pOnUfhyPKvZpVyBHhSBoX8vfA09b7r00Pmm1sH+29ae2hMTKVmSp4Ztsr8KBKjLjx17H0eJqaRC3bR2iThM54A==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-runtime": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-runtime/-/plugin-transform-runtime-7.12.1.tgz", - "integrity": "sha512-Ac/H6G9FEIkS2tXsZjL4RAdS3L3WHxci0usAnz7laPWUmFiGtj7tIASChqKZMHTSQTQY6xDbOq+V1/vIq3QrWg==", - "dev": true, - "requires": { - "@babel/helper-module-imports": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4", - "resolve": "^1.8.1", - "semver": "^5.5.1" - } - }, - "@babel/plugin-transform-shorthand-properties": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.12.1.tgz", - "integrity": "sha512-GFZS3c/MhX1OusqB1MZ1ct2xRzX5ppQh2JU1h2Pnfk88HtFTM+TWQqJNfwkmxtPQtb/s1tk87oENfXJlx7rSDw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-spread": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-spread/-/plugin-transform-spread-7.12.1.tgz", - "integrity": "sha512-vuLp8CP0BE18zVYjsEBZ5xoCecMK6LBMMxYzJnh01rxQRvhNhH1csMMmBfNo5tGpGO+NhdSNW2mzIvBu3K1fng==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-skip-transparent-expression-wrappers": "^7.12.1" - } - }, - "@babel/plugin-transform-sticky-regex": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-sticky-regex/-/plugin-transform-sticky-regex-7.12.1.tgz", - "integrity": "sha512-CiUgKQ3AGVk7kveIaPEET1jNDhZZEl1RPMWdTBE1799bdz++SwqDHStmxfCtDfBhQgCl38YRiSnrMuUMZIWSUQ==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-regex": "^7.10.4" - } - }, - "@babel/plugin-transform-template-literals": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-template-literals/-/plugin-transform-template-literals-7.12.1.tgz", - "integrity": "sha512-b4Zx3KHi+taXB1dVRBhVJtEPi9h1THCeKmae2qP0YdUHIFhVjtpqqNfxeVAa1xeHVhAy4SbHxEwx5cltAu5apw==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-typeof-symbol": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typeof-symbol/-/plugin-transform-typeof-symbol-7.12.1.tgz", - "integrity": "sha512-EPGgpGy+O5Kg5pJFNDKuxt9RdmTgj5sgrus2XVeMp/ZIbOESadgILUbm50SNpghOh3/6yrbsH+NB5+WJTmsA7Q==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-unicode-escapes": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-escapes/-/plugin-transform-unicode-escapes-7.12.1.tgz", - "integrity": "sha512-I8gNHJLIc7GdApm7wkVnStWssPNbSRMPtgHdmH3sRM1zopz09UWPS4x5V4n1yz/MIWTVnJ9sp6IkuXdWM4w+2Q==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/plugin-transform-unicode-regex": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-regex/-/plugin-transform-unicode-regex-7.12.1.tgz", - "integrity": "sha512-SqH4ClNngh/zGwHZOOQMTD+e8FGWexILV+ePMyiDJttAWRh5dhDL8rcl5lSgU3Huiq6Zn6pWTMvdPAb21Dwdyg==", - "dev": true, - "requires": { - "@babel/helper-create-regexp-features-plugin": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4" - } - }, - "@babel/preset-env": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/preset-env/-/preset-env-7.12.1.tgz", - "integrity": "sha512-H8kxXmtPaAGT7TyBvSSkoSTUK6RHh61So05SyEbpmr0MCZrsNYn7mGMzzeYoOUCdHzww61k8XBft2TaES+xPLg==", - "dev": true, - "requires": { - "@babel/compat-data": "^7.12.1", - "@babel/helper-compilation-targets": "^7.12.1", - "@babel/helper-module-imports": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-validator-option": "^7.12.1", - "@babel/plugin-proposal-async-generator-functions": "^7.12.1", - "@babel/plugin-proposal-class-properties": "^7.12.1", - "@babel/plugin-proposal-dynamic-import": "^7.12.1", - "@babel/plugin-proposal-export-namespace-from": "^7.12.1", - "@babel/plugin-proposal-json-strings": "^7.12.1", - "@babel/plugin-proposal-logical-assignment-operators": "^7.12.1", - "@babel/plugin-proposal-nullish-coalescing-operator": "^7.12.1", - "@babel/plugin-proposal-numeric-separator": "^7.12.1", - "@babel/plugin-proposal-object-rest-spread": "^7.12.1", - "@babel/plugin-proposal-optional-catch-binding": "^7.12.1", - "@babel/plugin-proposal-optional-chaining": "^7.12.1", - "@babel/plugin-proposal-private-methods": "^7.12.1", - "@babel/plugin-proposal-unicode-property-regex": "^7.12.1", - "@babel/plugin-syntax-async-generators": "^7.8.0", - "@babel/plugin-syntax-class-properties": "^7.12.1", - "@babel/plugin-syntax-dynamic-import": "^7.8.0", - "@babel/plugin-syntax-export-namespace-from": "^7.8.3", - "@babel/plugin-syntax-json-strings": "^7.8.0", - "@babel/plugin-syntax-logical-assignment-operators": "^7.10.4", - "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.0", - "@babel/plugin-syntax-numeric-separator": "^7.10.4", - "@babel/plugin-syntax-object-rest-spread": "^7.8.0", - "@babel/plugin-syntax-optional-catch-binding": "^7.8.0", - "@babel/plugin-syntax-optional-chaining": "^7.8.0", - "@babel/plugin-syntax-top-level-await": "^7.12.1", - "@babel/plugin-transform-arrow-functions": "^7.12.1", - "@babel/plugin-transform-async-to-generator": "^7.12.1", - "@babel/plugin-transform-block-scoped-functions": "^7.12.1", - "@babel/plugin-transform-block-scoping": "^7.12.1", - "@babel/plugin-transform-classes": "^7.12.1", - "@babel/plugin-transform-computed-properties": "^7.12.1", - "@babel/plugin-transform-destructuring": "^7.12.1", - "@babel/plugin-transform-dotall-regex": "^7.12.1", - "@babel/plugin-transform-duplicate-keys": "^7.12.1", - "@babel/plugin-transform-exponentiation-operator": "^7.12.1", - "@babel/plugin-transform-for-of": "^7.12.1", - "@babel/plugin-transform-function-name": "^7.12.1", - "@babel/plugin-transform-literals": "^7.12.1", - "@babel/plugin-transform-member-expression-literals": "^7.12.1", - "@babel/plugin-transform-modules-amd": "^7.12.1", - "@babel/plugin-transform-modules-commonjs": "^7.12.1", - "@babel/plugin-transform-modules-systemjs": "^7.12.1", - "@babel/plugin-transform-modules-umd": "^7.12.1", - "@babel/plugin-transform-named-capturing-groups-regex": "^7.12.1", - "@babel/plugin-transform-new-target": "^7.12.1", - "@babel/plugin-transform-object-super": "^7.12.1", - "@babel/plugin-transform-parameters": "^7.12.1", - "@babel/plugin-transform-property-literals": "^7.12.1", - "@babel/plugin-transform-regenerator": "^7.12.1", - "@babel/plugin-transform-reserved-words": "^7.12.1", - "@babel/plugin-transform-shorthand-properties": "^7.12.1", - "@babel/plugin-transform-spread": "^7.12.1", - "@babel/plugin-transform-sticky-regex": "^7.12.1", - "@babel/plugin-transform-template-literals": "^7.12.1", - "@babel/plugin-transform-typeof-symbol": "^7.12.1", - "@babel/plugin-transform-unicode-escapes": "^7.12.1", - "@babel/plugin-transform-unicode-regex": "^7.12.1", - "@babel/preset-modules": "^0.1.3", - "@babel/types": "^7.12.1", - "core-js-compat": "^3.6.2", - "semver": "^5.5.0" - } - }, - "@babel/preset-modules": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/@babel/preset-modules/-/preset-modules-0.1.4.tgz", - "integrity": "sha512-J36NhwnfdzpmH41M1DrnkkgAqhZaqr/NBdPfQ677mLzlaXo+oDiv1deyCDtgAhz8p328otdob0Du7+xgHGZbKg==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.0.0", - "@babel/plugin-proposal-unicode-property-regex": "^7.4.4", - "@babel/plugin-transform-dotall-regex": "^7.4.4", - "@babel/types": "^7.4.4", - "esutils": "^2.0.2" - } - }, - "@babel/register": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/register/-/register-7.12.1.tgz", - "integrity": "sha512-XWcmseMIncOjoydKZnWvWi0/5CUCD+ZYKhRwgYlWOrA8fGZ/FjuLRpqtIhLOVD/fvR1b9DQHtZPn68VvhpYf+Q==", - "dev": true, - "requires": { - "find-cache-dir": "^2.0.0", - "lodash": "^4.17.19", - "make-dir": "^2.1.0", - "pirates": "^4.0.0", - "source-map-support": "^0.5.16" - } - }, - "@babel/runtime": { - "version": "7.12.5", - "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.12.5.tgz", - "integrity": "sha512-plcc+hbExy3McchJCEQG3knOsuh3HH+Prx1P6cLIkET/0dLuQDEnrT+s27Axgc9bqfsmNUNHfscgMUdBpC9xfg==", - "dev": true, - "requires": { - "regenerator-runtime": "^0.13.4" - } - }, - "@babel/template": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.10.4.tgz", - "integrity": "sha512-ZCjD27cGJFUB6nmCB1Enki3r+L5kJveX9pq1SvAUKoICy6CZ9yD8xO086YXdYhvNjBdnekm4ZnaP5yC8Cs/1tA==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.10.4", - "@babel/parser": "^7.10.4", - "@babel/types": "^7.10.4" - } - }, - "@babel/traverse": { - "version": "7.12.5", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.12.5.tgz", - "integrity": "sha512-xa15FbQnias7z9a62LwYAA5SZZPkHIXpd42C6uW68o8uTuua96FHZy1y61Va5P/i83FAAcMpW8+A/QayntzuqA==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.10.4", - "@babel/generator": "^7.12.5", - "@babel/helper-function-name": "^7.10.4", - "@babel/helper-split-export-declaration": "^7.11.0", - "@babel/parser": "^7.12.5", - "@babel/types": "^7.12.5", - "debug": "^4.1.0", - "globals": "^11.1.0", - "lodash": "^4.17.19" - }, - "dependencies": { - "debug": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", - "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", - "dev": true, - "requires": { - "ms": "2.1.2" - } - } - } - }, - "@babel/types": { - "version": "7.12.6", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.12.6.tgz", - "integrity": "sha512-hwyjw6GvjBLiyy3W0YQf0Z5Zf4NpYejUnKFcfcUhZCSffoBBp30w6wP2Wn6pk31jMYZvcOrB/1b7cGXvEoKogA==", - "dev": true, - "requires": { - "@babel/helper-validator-identifier": "^7.10.4", - "lodash": "^4.17.19", - "to-fast-properties": "^2.0.0" - } - }, - "@bcoe/v8-coverage": { - "version": "0.2.3", - "resolved": "https://registry.npmjs.org/@bcoe/v8-coverage/-/v8-coverage-0.2.3.tgz", - "integrity": "sha512-0hYQ8SB4Db5zvZB4axdMHGwEaQjkZzFjQiN9LVYvIFB2nSUHW9tYpxWriPrWDASIxiaXax83REcLxuSdnGPZtw==", - "dev": true - }, - "@cnakazawa/watch": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/@cnakazawa/watch/-/watch-1.0.4.tgz", - "integrity": "sha512-v9kIhKwjeZThiWrLmj0y17CWoyddASLj9O2yvbZkbvw/N3rWOYy9zkV66ursAoVr0mV15bL8g0c4QZUE6cdDoQ==", - "dev": true, - "requires": { - "exec-sh": "^0.3.2", - "minimist": "^1.2.0" - } - }, - "@eslint/eslintrc": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-0.2.1.tgz", - "integrity": "sha512-XRUeBZ5zBWLYgSANMpThFddrZZkEbGHgUdt5UJjZfnlN9BGCiUBrf+nvbRupSjMvqzwnQN0qwCmOxITt1cfywA==", - "dev": true, - "requires": { - "ajv": "^6.12.4", - "debug": "^4.1.1", - "espree": "^7.3.0", - "globals": "^12.1.0", - "ignore": "^4.0.6", - "import-fresh": "^3.2.1", - "js-yaml": "^3.13.1", - "lodash": "^4.17.19", - "minimatch": "^3.0.4", - "strip-json-comments": "^3.1.1" - }, - "dependencies": { - "debug": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", - "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", - "dev": true, - "requires": { - "ms": "2.1.2" - } - }, - "globals": { - "version": "12.4.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-12.4.0.tgz", - "integrity": "sha512-BWICuzzDvDoH54NHKCseDanAhE3CeDorgDL5MT6LMXXj2WCnd9UC2szdk4AWLfjdgNBCXLUanXYcpBBKOSWGwg==", - "dev": true, - "requires": { - "type-fest": "^0.8.1" - } - }, - "ignore": { - "version": "4.0.6", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", - "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", - "dev": true - }, - "strip-json-comments": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", - "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", - "dev": true - } - } - }, - "@istanbuljs/load-nyc-config": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@istanbuljs/load-nyc-config/-/load-nyc-config-1.1.0.tgz", - "integrity": "sha512-VjeHSlIzpv/NyD3N0YuHfXOPDIixcA1q2ZV98wsMqcYlPmv2n3Yb2lYP9XMElnaFVXg5A7YLTeLu6V84uQDjmQ==", - "dev": true, - "requires": { - "camelcase": "^5.3.1", - "find-up": "^4.1.0", - "get-package-type": "^0.1.0", - "js-yaml": "^3.13.1", - "resolve-from": "^5.0.0" - } - }, - "@istanbuljs/schema": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.2.tgz", - "integrity": "sha512-tsAQNx32a8CoFhjhijUIhI4kccIAgmGhy8LZMZgGfmXcpMbPRUqn5LWmgRttILi6yeGmBJd2xsPkFMs0PzgPCw==", - "dev": true - }, - "@jest/console": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/@jest/console/-/console-26.6.2.tgz", - "integrity": "sha512-IY1R2i2aLsLr7Id3S6p2BA82GNWryt4oSvEXLAKc+L2zdi89dSkE8xC1C+0kpATG4JhBJREnQOH7/zmccM2B0g==", - "dev": true, - "requires": { - "@jest/types": "^26.6.2", - "@types/node": "*", - "chalk": "^4.0.0", - "jest-message-util": "^26.6.2", - "jest-util": "^26.6.2", - "slash": "^3.0.0" - }, - "dependencies": { - "slash": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", - "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", - "dev": true - } - } - }, - "@jest/core": { - "version": "26.6.3", - "resolved": "https://registry.npmjs.org/@jest/core/-/core-26.6.3.tgz", - "integrity": "sha512-xvV1kKbhfUqFVuZ8Cyo+JPpipAHHAV3kcDBftiduK8EICXmTFddryy3P7NfZt8Pv37rA9nEJBKCCkglCPt/Xjw==", - "dev": true, - "requires": { - "@jest/console": "^26.6.2", - "@jest/reporters": "^26.6.2", - "@jest/test-result": "^26.6.2", - "@jest/transform": "^26.6.2", - "@jest/types": "^26.6.2", - "@types/node": "*", - "ansi-escapes": "^4.2.1", - "chalk": "^4.0.0", - "exit": "^0.1.2", - "graceful-fs": "^4.2.4", - "jest-changed-files": "^26.6.2", - "jest-config": "^26.6.3", - "jest-haste-map": "^26.6.2", - "jest-message-util": "^26.6.2", - "jest-regex-util": "^26.0.0", - "jest-resolve": "^26.6.2", - "jest-resolve-dependencies": "^26.6.3", - "jest-runner": "^26.6.3", - "jest-runtime": "^26.6.3", - "jest-snapshot": "^26.6.2", - "jest-util": "^26.6.2", - "jest-validate": "^26.6.2", - "jest-watcher": "^26.6.2", - "micromatch": "^4.0.2", - "p-each-series": "^2.1.0", - "rimraf": "^3.0.0", - "slash": "^3.0.0", - "strip-ansi": "^6.0.0" - }, - "dependencies": { - "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", - "dev": true - }, - "braces": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", - "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", - "dev": true, - "requires": { - "fill-range": "^7.0.1" - } - }, - "fill-range": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", - "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", - "dev": true, - "requires": { - "to-regex-range": "^5.0.1" - } - }, - "is-number": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "dev": true - }, - "micromatch": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", - "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", - "dev": true, - "requires": { - "braces": "^3.0.1", - "picomatch": "^2.0.5" - } - }, - "slash": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", - "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", - "dev": true - }, - "strip-ansi": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", - "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", - "dev": true, - "requires": { - "ansi-regex": "^5.0.0" - } - }, - "to-regex-range": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", - "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "dev": true, - "requires": { - "is-number": "^7.0.0" - } - } - } - }, - "@jest/environment": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/@jest/environment/-/environment-26.6.2.tgz", - "integrity": "sha512-nFy+fHl28zUrRsCeMB61VDThV1pVTtlEokBRgqPrcT1JNq4yRNIyTHfyht6PqtUvY9IsuLGTrbG8kPXjSZIZwA==", - "dev": true, - "requires": { - "@jest/fake-timers": "^26.6.2", - "@jest/types": "^26.6.2", - "@types/node": "*", - "jest-mock": "^26.6.2" - } - }, - "@jest/fake-timers": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/@jest/fake-timers/-/fake-timers-26.6.2.tgz", - "integrity": "sha512-14Uleatt7jdzefLPYM3KLcnUl1ZNikaKq34enpb5XG9i81JpppDb5muZvonvKyrl7ftEHkKS5L5/eB/kxJ+bvA==", - "dev": true, - "requires": { - "@jest/types": "^26.6.2", - "@sinonjs/fake-timers": "^6.0.1", - "@types/node": "*", - "jest-message-util": "^26.6.2", - "jest-mock": "^26.6.2", - "jest-util": "^26.6.2" - } - }, - "@jest/globals": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/@jest/globals/-/globals-26.6.2.tgz", - "integrity": "sha512-85Ltnm7HlB/KesBUuALwQ68YTU72w9H2xW9FjZ1eL1U3lhtefjjl5c2MiUbpXt/i6LaPRvoOFJ22yCBSfQ0JIA==", - "dev": true, - "requires": { - "@jest/environment": "^26.6.2", - "@jest/types": "^26.6.2", - "expect": "^26.6.2" - } - }, - "@jest/reporters": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/@jest/reporters/-/reporters-26.6.2.tgz", - "integrity": "sha512-h2bW53APG4HvkOnVMo8q3QXa6pcaNt1HkwVsOPMBV6LD/q9oSpxNSYZQYkAnjdMjrJ86UuYeLo+aEZClV6opnw==", - "dev": true, - "requires": { - "@bcoe/v8-coverage": "^0.2.3", - "@jest/console": "^26.6.2", - "@jest/test-result": "^26.6.2", - "@jest/transform": "^26.6.2", - "@jest/types": "^26.6.2", - "chalk": "^4.0.0", - "collect-v8-coverage": "^1.0.0", - "exit": "^0.1.2", - "glob": "^7.1.2", - "graceful-fs": "^4.2.4", - "istanbul-lib-coverage": "^3.0.0", - "istanbul-lib-instrument": "^4.0.3", - "istanbul-lib-report": "^3.0.0", - "istanbul-lib-source-maps": "^4.0.0", - "istanbul-reports": "^3.0.2", - "jest-haste-map": "^26.6.2", - "jest-resolve": "^26.6.2", - "jest-util": "^26.6.2", - "jest-worker": "^26.6.2", - "node-notifier": "^8.0.0", - "slash": "^3.0.0", - "source-map": "^0.6.0", - "string-length": "^4.0.1", - "terminal-link": "^2.0.0", - "v8-to-istanbul": "^7.0.0" - }, - "dependencies": { - "slash": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", - "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", - "dev": true - }, - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } - } - }, - "@jest/source-map": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/@jest/source-map/-/source-map-26.6.2.tgz", - "integrity": "sha512-YwYcCwAnNmOVsZ8mr3GfnzdXDAl4LaenZP5z+G0c8bzC9/dugL8zRmxZzdoTl4IaS3CryS1uWnROLPFmb6lVvA==", - "dev": true, - "requires": { - "callsites": "^3.0.0", - "graceful-fs": "^4.2.4", - "source-map": "^0.6.0" - }, - "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } - } - }, - "@jest/test-result": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/@jest/test-result/-/test-result-26.6.2.tgz", - "integrity": "sha512-5O7H5c/7YlojphYNrK02LlDIV2GNPYisKwHm2QTKjNZeEzezCbwYs9swJySv2UfPMyZ0VdsmMv7jIlD/IKYQpQ==", - "dev": true, - "requires": { - "@jest/console": "^26.6.2", - "@jest/types": "^26.6.2", - "@types/istanbul-lib-coverage": "^2.0.0", - "collect-v8-coverage": "^1.0.0" - } - }, - "@jest/test-sequencer": { - "version": "26.6.3", - "resolved": "https://registry.npmjs.org/@jest/test-sequencer/-/test-sequencer-26.6.3.tgz", - "integrity": "sha512-YHlVIjP5nfEyjlrSr8t/YdNfU/1XEt7c5b4OxcXCjyRhjzLYu/rO69/WHPuYcbCWkz8kAeZVZp2N2+IOLLEPGw==", - "dev": true, - "requires": { - "@jest/test-result": "^26.6.2", - "graceful-fs": "^4.2.4", - "jest-haste-map": "^26.6.2", - "jest-runner": "^26.6.3", - "jest-runtime": "^26.6.3" - } - }, - "@jest/transform": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/@jest/transform/-/transform-26.6.2.tgz", - "integrity": "sha512-E9JjhUgNzvuQ+vVAL21vlyfy12gP0GhazGgJC4h6qUt1jSdUXGWJ1wfu/X7Sd8etSgxV4ovT1pb9v5D6QW4XgA==", - "dev": true, - "requires": { - "@babel/core": "^7.1.0", - "@jest/types": "^26.6.2", - "babel-plugin-istanbul": "^6.0.0", - "chalk": "^4.0.0", - "convert-source-map": "^1.4.0", - "fast-json-stable-stringify": "^2.0.0", - "graceful-fs": "^4.2.4", - "jest-haste-map": "^26.6.2", - "jest-regex-util": "^26.0.0", - "jest-util": "^26.6.2", - "micromatch": "^4.0.2", - "pirates": "^4.0.1", - "slash": "^3.0.0", - "source-map": "^0.6.1", - "write-file-atomic": "^3.0.0" - }, - "dependencies": { - "braces": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", - "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", - "dev": true, - "requires": { - "fill-range": "^7.0.1" - } - }, - "fill-range": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", - "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", - "dev": true, - "requires": { - "to-regex-range": "^5.0.1" - } - }, - "is-number": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "dev": true - }, - "micromatch": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", - "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", - "dev": true, - "requires": { - "braces": "^3.0.1", - "picomatch": "^2.0.5" - } - }, - "slash": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", - "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", - "dev": true - }, - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - }, - "to-regex-range": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", - "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "dev": true, - "requires": { - "is-number": "^7.0.0" - } - } - } - }, - "@jest/types": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/@jest/types/-/types-26.6.2.tgz", - "integrity": "sha512-fC6QCp7Sc5sX6g8Tvbmj4XUTbyrik0akgRy03yjXbQaBWWNWGE7SGtJk98m0N8nzegD/7SggrUlivxo5ax4KWQ==", - "dev": true, - "requires": { - "@types/istanbul-lib-coverage": "^2.0.0", - "@types/istanbul-reports": "^3.0.0", - "@types/node": "*", - "@types/yargs": "^15.0.0", - "chalk": "^4.0.0" - } - }, - "@nicolo-ribaudo/chokidar-2": { - "version": "2.1.8", - "resolved": "https://registry.npmjs.org/@nicolo-ribaudo/chokidar-2/-/chokidar-2-2.1.8.tgz", - "integrity": "sha512-FohwULwAebCUKi/akMFyGi7jfc7JXTeMHzKxuP3umRd9mK/2Y7/SMBSI2jX+YLopPXi+PF9l307NmpfxTdCegA==", - "dev": true, - "optional": true, - "requires": { - "chokidar": "2.1.8" - }, - "dependencies": { - "chokidar": { - "version": "2.1.8", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.1.8.tgz", - "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", - "dev": true, - "optional": true, - "requires": { - "anymatch": "^2.0.0", - "async-each": "^1.0.1", - "braces": "^2.3.2", - "fsevents": "^1.2.7", - "glob-parent": "^3.1.0", - "inherits": "^2.0.3", - "is-binary-path": "^1.0.0", - "is-glob": "^4.0.0", - "normalize-path": "^3.0.0", - "path-is-absolute": "^1.0.0", - "readdirp": "^2.2.1", - "upath": "^1.1.1" - } - } - } - }, - "@nodelib/fs.scandir": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.3.tgz", - "integrity": "sha512-eGmwYQn3gxo4r7jdQnkrrN6bY478C3P+a/y72IJukF8LjB6ZHeB3c+Ehacj3sYeSmUXGlnA67/PmbM9CVwL7Dw==", - "dev": true, - "requires": { - "@nodelib/fs.stat": "2.0.3", - "run-parallel": "^1.1.9" - } - }, - "@nodelib/fs.stat": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.3.tgz", - "integrity": "sha512-bQBFruR2TAwoevBEd/NWMoAAtNGzTRgdrqnYCc7dhzfoNvqPzLyqlEQnzZ3kVnNrSp25iyxE00/3h2fqGAGArA==", - "dev": true - }, - "@nodelib/fs.walk": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.4.tgz", - "integrity": "sha512-1V9XOY4rDW0rehzbrcqAmHnz8e7SKvX27gh8Gt2WgB0+pdzdiLV83p72kZPU+jvMbS1qU5mauP2iOvO8rhmurQ==", - "dev": true, - "requires": { - "@nodelib/fs.scandir": "2.1.3", - "fastq": "^1.6.0" - } - }, - "@sindresorhus/is": { - "version": "0.14.0", - "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-0.14.0.tgz", - "integrity": "sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ==", - "dev": true - }, - "@sinonjs/commons": { - "version": "1.8.1", - "resolved": "https://registry.npmjs.org/@sinonjs/commons/-/commons-1.8.1.tgz", - "integrity": "sha512-892K+kWUUi3cl+LlqEWIDrhvLgdL79tECi8JZUyq6IviKy/DNhuzCRlbHUjxK89f4ypPMMaFnFuR9Ie6DoIMsw==", - "dev": true, - "requires": { - "type-detect": "4.0.8" - } - }, - "@sinonjs/fake-timers": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/@sinonjs/fake-timers/-/fake-timers-6.0.1.tgz", - "integrity": "sha512-MZPUxrmFubI36XS1DI3qmI0YdN1gks62JtFZvxR67ljjSNCeK6U08Zx4msEWOXuofgqUt6zPHSi1H9fbjR/NRA==", - "dev": true, - "requires": { - "@sinonjs/commons": "^1.7.0" - } - }, - "@sqltools/formatter": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/@sqltools/formatter/-/formatter-1.2.2.tgz", - "integrity": "sha512-/5O7Fq6Vnv8L6ucmPjaWbVG1XkP4FO+w5glqfkIsq3Xw4oyNAdJddbnYodNDAfjVUvo/rrSCTom4kAND7T1o5Q==" - }, - "@szmarczak/http-timer": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-1.1.2.tgz", - "integrity": "sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA==", - "dev": true, - "requires": { - "defer-to-connect": "^1.0.1" - } - }, - "@types/babel__core": { - "version": "7.1.12", - "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.1.12.tgz", - "integrity": "sha512-wMTHiiTiBAAPebqaPiPDLFA4LYPKr6Ph0Xq/6rq1Ur3v66HXyG+clfR9CNETkD7MQS8ZHvpQOtA53DLws5WAEQ==", - "dev": true, - "requires": { - "@babel/parser": "^7.1.0", - "@babel/types": "^7.0.0", - "@types/babel__generator": "*", - "@types/babel__template": "*", - "@types/babel__traverse": "*" - } - }, - "@types/babel__generator": { - "version": "7.6.2", - "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.6.2.tgz", - "integrity": "sha512-MdSJnBjl+bdwkLskZ3NGFp9YcXGx5ggLpQQPqtgakVhsWK0hTtNYhjpZLlWQTviGTvF8at+Bvli3jV7faPdgeQ==", - "dev": true, - "requires": { - "@babel/types": "^7.0.0" - } - }, - "@types/babel__template": { - "version": "7.4.0", - "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.0.tgz", - "integrity": "sha512-NTPErx4/FiPCGScH7foPyr+/1Dkzkni+rHiYHHoTjvwou7AQzJkNeD60A9CXRy+ZEN2B1bggmkTMCDb+Mv5k+A==", - "dev": true, - "requires": { - "@babel/parser": "^7.1.0", - "@babel/types": "^7.0.0" - } - }, - "@types/babel__traverse": { - "version": "7.0.15", - "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.0.15.tgz", - "integrity": "sha512-Pzh9O3sTK8V6I1olsXpCfj2k/ygO2q1X0vhhnDrEQyYLHZesWz+zMZMVcwXLCYf0U36EtmyYaFGPfXlTtDHe3A==", - "dev": true, - "requires": { - "@babel/types": "^7.3.0" - } - }, - "@types/body-parser": { - "version": "1.19.0", - "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.0.tgz", - "integrity": "sha512-W98JrE0j2K78swW4ukqMleo8R7h/pFETjM2DQ90MF6XK2i4LO4W3gQ71Lt4w3bfm2EvVSyWHplECvB5sK22yFQ==", - "dev": true, - "requires": { - "@types/connect": "*", - "@types/node": "*" - } - }, - "@types/cls-hooked": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/@types/cls-hooked/-/cls-hooked-4.3.1.tgz", - "integrity": "sha512-nMjNjQAk9vAYnDEXRUxGSACarjPDRKVaZ8xrwKzRy1BmzG5tN3JUkuvdVwE8P2GBkSRokxVw+QUpuvFzvOsKpA==", - "requires": { - "@types/node": "*" - } - }, - "@types/connect": { - "version": "3.4.33", - "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.33.tgz", - "integrity": "sha512-2+FrkXY4zllzTNfJth7jOqEHC+enpLeGslEhpnTAkg21GkRrWV4SsAtqchtT4YS9/nODBU2/ZfsBY2X4J/dX7A==", - "dev": true, - "requires": { - "@types/node": "*" - } - }, - "@types/dotenv": { - "version": "8.2.0", - "resolved": "https://registry.npmjs.org/@types/dotenv/-/dotenv-8.2.0.tgz", - "integrity": "sha512-ylSC9GhfRH7m1EUXBXofhgx4lUWmFeQDINW5oLuS+gxWdfUeW4zJdeVTYVkexEW+e2VUvlZR2kGnGGipAWR7kw==", - "dev": true, - "requires": { - "dotenv": "*" - } - }, - "@types/express": { - "version": "4.17.9", - "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.9.tgz", - "integrity": "sha512-SDzEIZInC4sivGIFY4Sz1GG6J9UObPwCInYJjko2jzOf/Imx/dlpume6Xxwj1ORL82tBbmN4cPDIDkLbWHk9hw==", - "dev": true, - "requires": { - "@types/body-parser": "*", - "@types/express-serve-static-core": "*", - "@types/qs": "*", - "@types/serve-static": "*" - } - }, - "@types/express-serve-static-core": { - "version": "4.17.13", - "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.17.13.tgz", - "integrity": "sha512-RgDi5a4nuzam073lRGKTUIaL3eF2+H7LJvJ8eUnCI0wA6SNjXc44DCmWNiTLs/AZ7QlsFWZiw/gTG3nSQGL0fA==", - "dev": true, - "requires": { - "@types/node": "*", - "@types/qs": "*", - "@types/range-parser": "*" - } - }, - "@types/graceful-fs": { - "version": "4.1.4", - "resolved": "https://registry.npmjs.org/@types/graceful-fs/-/graceful-fs-4.1.4.tgz", - "integrity": "sha512-mWA/4zFQhfvOA8zWkXobwJvBD7vzcxgrOQ0J5CH1votGqdq9m7+FwtGaqyCZqC3NyyBkc9z4m+iry4LlqcMWJg==", - "dev": true, - "requires": { - "@types/node": "*" - } - }, - "@types/istanbul-lib-coverage": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.3.tgz", - "integrity": "sha512-sz7iLqvVUg1gIedBOvlkxPlc8/uVzyS5OwGz1cKjXzkl3FpL3al0crU8YGU1WoHkxn0Wxbw5tyi6hvzJKNzFsw==", - "dev": true - }, - "@types/istanbul-lib-report": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/@types/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz", - "integrity": "sha512-plGgXAPfVKFoYfa9NpYDAkseG+g6Jr294RqeqcqDixSbU34MZVJRi/P+7Y8GDpzkEwLaGZZOpKIEmeVZNtKsrg==", - "dev": true, - "requires": { - "@types/istanbul-lib-coverage": "*" - } - }, - "@types/istanbul-reports": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.0.tgz", - "integrity": "sha512-nwKNbvnwJ2/mndE9ItP/zc2TCzw6uuodnF4EHYWD+gCQDVBuRQL5UzbZD0/ezy1iKsFU2ZQiDqg4M9dN4+wZgA==", - "dev": true, - "requires": { - "@types/istanbul-lib-report": "*" - } - }, - "@types/jest": { - "version": "26.0.15", - "resolved": "https://registry.npmjs.org/@types/jest/-/jest-26.0.15.tgz", - "integrity": "sha512-s2VMReFXRg9XXxV+CW9e5Nz8fH2K1aEhwgjUqPPbQd7g95T0laAcvLv032EhFHIa5GHsZ8W7iJEQVaJq6k3Gog==", - "dev": true, - "requires": { - "jest-diff": "^26.0.0", - "pretty-format": "^26.0.0" - } - }, - "@types/json-schema": { - "version": "7.0.6", - "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.6.tgz", - "integrity": "sha512-3c+yGKvVP5Y9TYBEibGNR+kLtijnj7mYrXRg+WpFb2X9xm04g/DXYkfg4hmzJQosc9snFNUPkbYIhu+KAm6jJw==", - "dev": true - }, - "@types/json5": { - "version": "0.0.29", - "resolved": "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz", - "integrity": "sha1-7ihweulOEdK4J7y+UnC86n8+ce4=", - "dev": true - }, - "@types/jsonwebtoken": { - "version": "8.5.0", - "resolved": "https://registry.npmjs.org/@types/jsonwebtoken/-/jsonwebtoken-8.5.0.tgz", - "integrity": "sha512-9bVao7LvyorRGZCw0VmH/dr7Og+NdjYSsKAxB43OQoComFbBgsEpoR9JW6+qSq/ogwVBg8GI2MfAlk4SYI4OLg==", - "dev": true, - "requires": { - "@types/node": "*" - } - }, - "@types/mime": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/@types/mime/-/mime-2.0.3.tgz", - "integrity": "sha512-Jus9s4CDbqwocc5pOAnh8ShfrnMcPHuJYzVcSUU7lrh8Ni5HuIqX3oilL86p3dlTrk0LzHRCgA/GQ7uNCw6l2Q==", - "dev": true - }, - "@types/mysql": { - "version": "github:types/mysql#c26b1bc2bac17010081455e3127a90fb2eafcec9", - "from": "github:types/mysql", - "dev": true - }, - "@types/mysql2": { - "version": "github:types/mysql2#d4ef3b2292f328049f7e4c545f6adab7d6a350a9", - "from": "github:types/mysql2", - "dev": true, - "requires": { - "@types/mysql": "github:types/mysql" - } - }, - "@types/node": { - "version": "14.14.8", - "resolved": "https://registry.npmjs.org/@types/node/-/node-14.14.8.tgz", - "integrity": "sha512-z/5Yd59dCKI5kbxauAJgw6dLPzW+TNOItNE00PkpzNwUIEwdj/Lsqwq94H5DdYBX7C13aRA0CY32BK76+neEUA==" - }, - "@types/normalize-package-data": { - "version": "2.4.0", - "resolved": "https://registry.npmjs.org/@types/normalize-package-data/-/normalize-package-data-2.4.0.tgz", - "integrity": "sha512-f5j5b/Gf71L+dbqxIpQ4Z2WlmI/mPJ0fOkGGmFgtb6sAu97EPczzbS3/tJKxmcYDj55OX6ssqwDAWOHIYDRDGA==", - "dev": true - }, - "@types/prettier": { - "version": "2.1.5", - "resolved": "https://registry.npmjs.org/@types/prettier/-/prettier-2.1.5.tgz", - "integrity": "sha512-UEyp8LwZ4Dg30kVU2Q3amHHyTn1jEdhCIE59ANed76GaT1Vp76DD3ZWSAxgCrw6wJ0TqeoBpqmfUHiUDPs//HQ==", - "dev": true - }, - "@types/qs": { - "version": "6.9.5", - "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.5.tgz", - "integrity": "sha512-/JHkVHtx/REVG0VVToGRGH2+23hsYLHdyG+GrvoUGlGAd0ErauXDyvHtRI/7H7mzLm+tBCKA7pfcpkQ1lf58iQ==", - "dev": true - }, - "@types/range-parser": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.3.tgz", - "integrity": "sha512-ewFXqrQHlFsgc09MK5jP5iR7vumV/BYayNC6PgJO2LPe8vrnNFyjQjSppfEngITi0qvfKtzFvgKymGheFM9UOA==", - "dev": true - }, - "@types/serve-static": { - "version": "1.13.8", - "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.13.8.tgz", - "integrity": "sha512-MoJhSQreaVoL+/hurAZzIm8wafFR6ajiTM1m4A0kv6AGeVBl4r4pOV8bGFrjjq1sGxDTnCoF8i22o0/aE5XCyA==", - "dev": true, - "requires": { - "@types/mime": "*", - "@types/node": "*" - } - }, - "@types/stack-utils": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-2.0.0.tgz", - "integrity": "sha512-RJJrrySY7A8havqpGObOB4W92QXKJo63/jFLLgpvOtsGUqbQZ9Sbgl35KMm1DjC6j7AvmmU2bIno+3IyEaemaw==", - "dev": true - }, - "@types/validator": { - "version": "13.0.0", - "resolved": "https://registry.npmjs.org/@types/validator/-/validator-13.0.0.tgz", - "integrity": "sha512-WAy5txG7aFX8Vw3sloEKp5p/t/Xt8jD3GRD9DacnFv6Vo8ubudAsRTXgxpQwU0mpzY/H8U4db3roDuCMjShBmw==" - }, - "@types/yargs": { - "version": "15.0.10", - "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-15.0.10.tgz", - "integrity": "sha512-z8PNtlhrj7eJNLmrAivM7rjBESG6JwC5xP3RVk12i/8HVP7Xnx/sEmERnRImyEuUaJfO942X0qMOYsoupaJbZQ==", - "dev": true, - "requires": { - "@types/yargs-parser": "*" - } - }, - "@types/yargs-parser": { - "version": "15.0.0", - "resolved": "https://registry.npmjs.org/@types/yargs-parser/-/yargs-parser-15.0.0.tgz", - "integrity": "sha512-FA/BWv8t8ZWJ+gEOnLLd8ygxH/2UFbAvgEonyfN6yWGLKc7zVjbpl2Y4CTjid9h2RfgPP6SEt6uHwEOply00yw==", - "dev": true - }, - "@typescript-eslint/eslint-plugin": { - "version": "4.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-4.8.1.tgz", - "integrity": "sha512-d7LeQ7dbUrIv5YVFNzGgaW3IQKMmnmKFneRWagRlGYOSfLJVaRbj/FrBNOBC1a3tVO+TgNq1GbHvRtg1kwL0FQ==", - "dev": true, - "requires": { - "@typescript-eslint/experimental-utils": "4.8.1", - "@typescript-eslint/scope-manager": "4.8.1", - "debug": "^4.1.1", - "functional-red-black-tree": "^1.0.1", - "regexpp": "^3.0.0", - "semver": "^7.3.2", - "tsutils": "^3.17.1" - }, - "dependencies": { - "debug": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", - "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", - "dev": true, - "requires": { - "ms": "2.1.2" - } - }, - "semver": { - "version": "7.3.2", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", - "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", - "dev": true - } - } - }, - "@typescript-eslint/experimental-utils": { - "version": "4.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/experimental-utils/-/experimental-utils-4.8.1.tgz", - "integrity": "sha512-WigyLn144R3+lGATXW4nNcDJ9JlTkG8YdBWHkDlN0lC3gUGtDi7Pe3h5GPvFKMcRz8KbZpm9FJV9NTW8CpRHpg==", - "dev": true, - "requires": { - "@types/json-schema": "^7.0.3", - "@typescript-eslint/scope-manager": "4.8.1", - "@typescript-eslint/types": "4.8.1", - "@typescript-eslint/typescript-estree": "4.8.1", - "eslint-scope": "^5.0.0", - "eslint-utils": "^2.0.0" - } - }, - "@typescript-eslint/parser": { - "version": "4.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-4.8.1.tgz", - "integrity": "sha512-QND8XSVetATHK9y2Ltc/XBl5Ro7Y62YuZKnPEwnNPB8E379fDsvzJ1dMJ46fg/VOmk0hXhatc+GXs5MaXuL5Uw==", - "dev": true, - "requires": { - "@typescript-eslint/scope-manager": "4.8.1", - "@typescript-eslint/types": "4.8.1", - "@typescript-eslint/typescript-estree": "4.8.1", - "debug": "^4.1.1" - }, - "dependencies": { - "debug": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", - "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", - "dev": true, - "requires": { - "ms": "2.1.2" - } - } - } - }, - "@typescript-eslint/scope-manager": { - "version": "4.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-4.8.1.tgz", - "integrity": "sha512-r0iUOc41KFFbZdPAdCS4K1mXivnSZqXS5D9oW+iykQsRlTbQRfuFRSW20xKDdYiaCoH+SkSLeIF484g3kWzwOQ==", - "dev": true, - "requires": { - "@typescript-eslint/types": "4.8.1", - "@typescript-eslint/visitor-keys": "4.8.1" - } - }, - "@typescript-eslint/types": { - "version": "4.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-4.8.1.tgz", - "integrity": "sha512-ave2a18x2Y25q5K05K/U3JQIe2Av4+TNi/2YuzyaXLAsDx6UZkz1boZ7nR/N6Wwae2PpudTZmHFXqu7faXfHmA==", - "dev": true - }, - "@typescript-eslint/typescript-estree": { - "version": "4.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-4.8.1.tgz", - "integrity": "sha512-bJ6Fn/6tW2g7WIkCWh3QRlaSU7CdUUK52shx36/J7T5oTQzANvi6raoTsbwGM11+7eBbeem8hCCKbyvAc0X3sQ==", - "dev": true, - "requires": { - "@typescript-eslint/types": "4.8.1", - "@typescript-eslint/visitor-keys": "4.8.1", - "debug": "^4.1.1", - "globby": "^11.0.1", - "is-glob": "^4.0.1", - "lodash": "^4.17.15", - "semver": "^7.3.2", - "tsutils": "^3.17.1" - }, - "dependencies": { - "debug": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", - "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", - "dev": true, - "requires": { - "ms": "2.1.2" - } - }, - "semver": { - "version": "7.3.2", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", - "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", - "dev": true - } - } - }, - "@typescript-eslint/visitor-keys": { - "version": "4.8.1", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-4.8.1.tgz", - "integrity": "sha512-3nrwXFdEYALQh/zW8rFwP4QltqsanCDz4CwWMPiIZmwlk9GlvBeueEIbq05SEq4ganqM0g9nh02xXgv5XI3PeQ==", - "dev": true, - "requires": { - "@typescript-eslint/types": "4.8.1", - "eslint-visitor-keys": "^2.0.0" - }, - "dependencies": { - "eslint-visitor-keys": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.0.0.tgz", - "integrity": "sha512-QudtT6av5WXels9WjIM7qz1XD1cWGvX4gGXvp/zBn9nXG02D0utdU3Em2m/QjTnrsk6bBjmCygl3rmj118msQQ==", - "dev": true - } - } - }, - "abab": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/abab/-/abab-2.0.5.tgz", - "integrity": "sha512-9IK9EadsbHo6jLWIpxpR6pL0sazTXV6+SQv25ZB+F7Bj9mJNaOc4nCRabwd5M/JwmUa8idz6Eci6eKfJryPs6Q==", - "dev": true - }, - "abbrev": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", - "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==" - }, - "accepts": { - "version": "1.3.7", - "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", - "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", - "requires": { - "mime-types": "~2.1.24", - "negotiator": "0.6.2" - } - }, - "acorn": { - "version": "7.4.1", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.4.1.tgz", - "integrity": "sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A==", - "dev": true - }, - "acorn-globals": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/acorn-globals/-/acorn-globals-6.0.0.tgz", - "integrity": "sha512-ZQl7LOWaF5ePqqcX4hLuv/bLXYQNfNWw2c0/yX/TsPRKamzHcTGQnlCjHT3TsmkOUVEPS3crCxiPfdzE/Trlhg==", - "dev": true, - "requires": { - "acorn": "^7.1.1", - "acorn-walk": "^7.1.1" - } - }, - "acorn-jsx": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.1.tgz", - "integrity": "sha512-K0Ptm/47OKfQRpNQ2J/oIN/3QYiK6FwW+eJbILhsdxh2WTLdl+30o8aGdTbm5JbffpFFAg/g+zi1E+jvJha5ng==", - "dev": true - }, - "acorn-walk": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-7.2.0.tgz", - "integrity": "sha512-OPdCF6GsMIP+Az+aWfAAOEt2/+iVDKE7oy6lJ098aoe59oAmK76qV6Gw60SbZ8jHuG2wH058GF4pLFbYamYrVA==", - "dev": true - }, - "ajv": { - "version": "6.12.6", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", - "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", - "dev": true, - "requires": { - "fast-deep-equal": "^3.1.1", - "fast-json-stable-stringify": "^2.0.0", - "json-schema-traverse": "^0.4.1", - "uri-js": "^4.2.2" - } - }, - "ansi-align": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/ansi-align/-/ansi-align-3.0.0.tgz", - "integrity": "sha512-ZpClVKqXN3RGBmKibdfWzqCY4lnjEuoNzU5T0oEFpfd/z5qJHVarukridD4juLO2FXMiwUQxr9WqQtaYa8XRYw==", - "dev": true, - "requires": { - "string-width": "^3.0.0" - }, - "dependencies": { - "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", - "dev": true - }, - "emoji-regex": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", - "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", - "dev": true - }, - "is-fullwidth-code-point": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", - "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", - "dev": true - }, - "string-width": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", - "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", - "dev": true, - "requires": { - "emoji-regex": "^7.0.1", - "is-fullwidth-code-point": "^2.0.0", - "strip-ansi": "^5.1.0" - } - }, - "strip-ansi": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", - "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", - "dev": true, - "requires": { - "ansi-regex": "^4.1.0" - } - } - } - }, - "ansi-colors": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz", - "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==", - "dev": true - }, - "ansi-escapes": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.1.tgz", - "integrity": "sha512-JWF7ocqNrp8u9oqpgV+wH5ftbt+cfvv+PTjOvKLT3AdYly/LmORARfEVT1iyjwN+4MqE5UmVKoAdIBqeoCHgLA==", - "dev": true, - "requires": { - "type-fest": "^0.11.0" - }, - "dependencies": { - "type-fest": { - "version": "0.11.0", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.11.0.tgz", - "integrity": "sha512-OdjXJxnCN1AvyLSzeKIgXTXxV+99ZuXl3Hpo9XpJAv9MBcHrrJOQ5kV7ypXOuQie+AmWG25hLbiKdwYTifzcfQ==", - "dev": true - } - } - }, - "ansi-regex": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", - "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=" - }, - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", - "requires": { - "color-convert": "^2.0.1" - } - }, - "any-promise": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/any-promise/-/any-promise-1.3.0.tgz", - "integrity": "sha1-q8av7tzqUugJzcA3au0845Y10X8=" - }, - "anymatch": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-2.0.0.tgz", - "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", - "dev": true, - "requires": { - "micromatch": "^3.1.4", - "normalize-path": "^2.1.1" - }, - "dependencies": { - "normalize-path": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz", - "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=", - "dev": true, - "requires": { - "remove-trailing-separator": "^1.0.1" - } - } - } - }, - "app-root-path": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/app-root-path/-/app-root-path-3.0.0.tgz", - "integrity": "sha512-qMcx+Gy2UZynHjOHOIXPNvpf+9cjvk3cWrBBK7zg4gH9+clobJRb9NGzcT7mQTcV/6Gm/1WelUtqxVXnNlrwcw==" - }, - "aproba": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/aproba/-/aproba-1.2.0.tgz", - "integrity": "sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw==" - }, - "are-we-there-yet": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/are-we-there-yet/-/are-we-there-yet-1.1.5.tgz", - "integrity": "sha512-5hYdAkZlcG8tOLujVDTgCT+uPX0VnpAH28gWsLfzpXYm7wP6mp5Q/gYyR7YQ0cKVJcXJnl3j2kpBan13PtQf6w==", - "requires": { - "delegates": "^1.0.0", - "readable-stream": "^2.0.6" - } - }, - "argparse": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", - "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", - "requires": { - "sprintf-js": "~1.0.2" - } - }, - "arr-diff": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", - "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=", - "dev": true - }, - "arr-flatten": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/arr-flatten/-/arr-flatten-1.1.0.tgz", - "integrity": "sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==", - "dev": true - }, - "arr-union": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/arr-union/-/arr-union-3.1.0.tgz", - "integrity": "sha1-45sJrqne+Gao8gbiiK9jkZuuOcQ=", - "dev": true - }, - "array-flatten": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", - "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" - }, - "array-includes": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.1.tgz", - "integrity": "sha512-c2VXaCHl7zPsvpkFsw4nxvFie4fh1ur9bpcgsVkIjqn0H/Xwdg+7fv3n2r/isyS8EBj5b06M9kHyZuIr4El6WQ==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.0", - "is-string": "^1.0.5" - } - }, - "array-union": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz", - "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", - "dev": true - }, - "array-unique": { - "version": "0.3.2", - "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", - "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", - "dev": true - }, - "array.prototype.flat": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.2.4.tgz", - "integrity": "sha512-4470Xi3GAPAjZqFcljX2xzckv1qeKPizoNkiS0+O4IoPR2ZNpcjE0pkhdihlDouK+x6QOast26B4Q/O9DJnwSg==", - "dev": true, - "requires": { - "call-bind": "^1.0.0", - "define-properties": "^1.1.3", - "es-abstract": "^1.18.0-next.1" - }, - "dependencies": { - "es-abstract": { - "version": "1.18.0-next.1", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.18.0-next.1.tgz", - "integrity": "sha512-I4UGspA0wpZXWENrdA0uHbnhte683t3qT/1VFH9aX2dA5PPSf6QW5HHXf5HImaqPmjXaVeVk4RGWnaylmV7uAA==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-negative-zero": "^2.0.0", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - } - } - }, - "asn1": { - "version": "0.2.4", - "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.4.tgz", - "integrity": "sha512-jxwzQpLQjSmWXgwaCZE9Nz+glAG01yF1QnWgbhGwHI5A6FRIEY6IVqtHhIepHqI7/kyEyQEagBC5mBEFlIYvdg==", - "dev": true, - "requires": { - "safer-buffer": "~2.1.0" - } - }, - "assert-plus": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", - "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=", - "dev": true - }, - "assign-symbols": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/assign-symbols/-/assign-symbols-1.0.0.tgz", - "integrity": "sha1-WWZ/QfrdTyDMvCu5a41Pf3jsA2c=", - "dev": true - }, - "astral-regex": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-1.0.0.tgz", - "integrity": "sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==", - "dev": true - }, - "async-each": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/async-each/-/async-each-1.0.3.tgz", - "integrity": "sha512-z/WhQ5FPySLdvREByI2vZiTWwCnF0moMJ1hK9YQwDTHKh6I7/uSckMetoRGb5UBZPC1z0jlw+n/XCgjeH7y1AQ==", - "dev": true, - "optional": true - }, - "async-hook-jl": { - "version": "1.7.6", - "resolved": "https://registry.npmjs.org/async-hook-jl/-/async-hook-jl-1.7.6.tgz", - "integrity": "sha512-gFaHkFfSxTjvoxDMYqDuGHlcRyUuamF8s+ZTtJdDzqjws4mCt7v0vuV79/E2Wr2/riMQgtG4/yUtXWs1gZ7JMg==", - "requires": { - "stack-chain": "^1.3.7" - } - }, - "asynckit": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", - "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=", - "dev": true - }, - "atob": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/atob/-/atob-2.1.2.tgz", - "integrity": "sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==", - "dev": true - }, - "aws-sign2": { - "version": "0.7.0", - "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", - "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=", - "dev": true - }, - "aws4": { - "version": "1.11.0", - "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.11.0.tgz", - "integrity": "sha512-xh1Rl34h6Fi1DC2WWKfxUTVqRsNnr6LsKz2+hfwDxQJWmrx8+c7ylaqBMcHfl1U1r2dsifOvKX3LQuLNZ+XSvA==", - "dev": true - }, - "babel-jest": { - "version": "26.6.3", - "resolved": "https://registry.npmjs.org/babel-jest/-/babel-jest-26.6.3.tgz", - "integrity": "sha512-pl4Q+GAVOHwvjrck6jKjvmGhnO3jHX/xuB9d27f+EJZ/6k+6nMuPjorrYp7s++bKKdANwzElBWnLWaObvTnaZA==", - "dev": true, - "requires": { - "@jest/transform": "^26.6.2", - "@jest/types": "^26.6.2", - "@types/babel__core": "^7.1.7", - "babel-plugin-istanbul": "^6.0.0", - "babel-preset-jest": "^26.6.2", - "chalk": "^4.0.0", - "graceful-fs": "^4.2.4", - "slash": "^3.0.0" - }, - "dependencies": { - "slash": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", - "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", - "dev": true - } - } - }, - "babel-plugin-dynamic-import-node": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/babel-plugin-dynamic-import-node/-/babel-plugin-dynamic-import-node-2.3.3.tgz", - "integrity": "sha512-jZVI+s9Zg3IqA/kdi0i6UDCybUI3aSBLnglhYbSSjKlV7yF1F/5LWv8MakQmvYpnbJDS6fcBL2KzHSxNCMtWSQ==", - "dev": true, - "requires": { - "object.assign": "^4.1.0" - } - }, - "babel-plugin-istanbul": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/babel-plugin-istanbul/-/babel-plugin-istanbul-6.0.0.tgz", - "integrity": "sha512-AF55rZXpe7trmEylbaE1Gv54wn6rwU03aptvRoVIGP8YykoSxqdVLV1TfwflBCE/QtHmqtP8SWlTENqbK8GCSQ==", - "dev": true, - "requires": { - "@babel/helper-plugin-utils": "^7.0.0", - "@istanbuljs/load-nyc-config": "^1.0.0", - "@istanbuljs/schema": "^0.1.2", - "istanbul-lib-instrument": "^4.0.0", - "test-exclude": "^6.0.0" - } - }, - "babel-plugin-jest-hoist": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-26.6.2.tgz", - "integrity": "sha512-PO9t0697lNTmcEHH69mdtYiOIkkOlj9fySqfO3K1eCcdISevLAE0xY59VLLUj0SoiPiTX/JU2CYFpILydUa5Lw==", - "dev": true, - "requires": { - "@babel/template": "^7.3.3", - "@babel/types": "^7.3.3", - "@types/babel__core": "^7.0.0", - "@types/babel__traverse": "^7.0.6" - } - }, - "babel-plugin-parameter-decorator": { - "version": "1.0.16", - "resolved": "https://registry.npmjs.org/babel-plugin-parameter-decorator/-/babel-plugin-parameter-decorator-1.0.16.tgz", - "integrity": "sha512-yUT2WPTUg1JaPmRGRSF557m1HJ9vdFQInRWOkiOyO5a9HhqlXffJu+fQ2xd5+qU/35ICMrrk9eWKsHCairKA9w==", - "dev": true - }, - "babel-preset-current-node-syntax": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/babel-preset-current-node-syntax/-/babel-preset-current-node-syntax-1.0.0.tgz", - "integrity": "sha512-mGkvkpocWJes1CmMKtgGUwCeeq0pOhALyymozzDWYomHTbDLwueDYG6p4TK1YOeYHCzBzYPsWkgTto10JubI1Q==", - "dev": true, - "requires": { - "@babel/plugin-syntax-async-generators": "^7.8.4", - "@babel/plugin-syntax-bigint": "^7.8.3", - "@babel/plugin-syntax-class-properties": "^7.8.3", - "@babel/plugin-syntax-import-meta": "^7.8.3", - "@babel/plugin-syntax-json-strings": "^7.8.3", - "@babel/plugin-syntax-logical-assignment-operators": "^7.8.3", - "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3", - "@babel/plugin-syntax-numeric-separator": "^7.8.3", - "@babel/plugin-syntax-object-rest-spread": "^7.8.3", - "@babel/plugin-syntax-optional-catch-binding": "^7.8.3", - "@babel/plugin-syntax-optional-chaining": "^7.8.3", - "@babel/plugin-syntax-top-level-await": "^7.8.3" - } - }, - "babel-preset-jest": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/babel-preset-jest/-/babel-preset-jest-26.6.2.tgz", - "integrity": "sha512-YvdtlVm9t3k777c5NPQIv6cxFFFapys25HiUmuSgHwIZhfifweR5c5Sf5nwE3MAbfu327CYSvps8Yx6ANLyleQ==", - "dev": true, - "requires": { - "babel-plugin-jest-hoist": "^26.6.2", - "babel-preset-current-node-syntax": "^1.0.0" - } - }, - "balanced-match": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", - "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=" - }, - "base": { - "version": "0.11.2", - "resolved": "https://registry.npmjs.org/base/-/base-0.11.2.tgz", - "integrity": "sha512-5T6P4xPgpp0YDFvSWwEZ4NoE3aM4QBQXDzmVbraCkFj8zHM+mba8SyqB5DbZWyR7mYHo6Y7BdQo3MoA4m0TeQg==", - "dev": true, - "requires": { - "cache-base": "^1.0.1", - "class-utils": "^0.3.5", - "component-emitter": "^1.2.1", - "define-property": "^1.0.0", - "isobject": "^3.0.1", - "mixin-deep": "^1.2.0", - "pascalcase": "^0.1.1" - }, - "dependencies": { - "define-property": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", - "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", - "dev": true, - "requires": { - "is-descriptor": "^1.0.0" - } - }, - "is-accessor-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", - "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-data-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", - "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-descriptor": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", - "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, - "requires": { - "is-accessor-descriptor": "^1.0.0", - "is-data-descriptor": "^1.0.0", - "kind-of": "^6.0.2" - } - } - } - }, - "base64-js": { - "version": "1.5.1", - "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", - "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==" - }, - "bcrypt": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/bcrypt/-/bcrypt-5.0.0.tgz", - "integrity": "sha512-jB0yCBl4W/kVHM2whjfyqnxTmOHkCX4kHEa5nYKSoGeYe8YrjTYTc87/6bwt1g8cmV0QrbhKriETg9jWtcREhg==", - "requires": { - "node-addon-api": "^3.0.0", - "node-pre-gyp": "0.15.0" - } - }, - "bcrypt-pbkdf": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", - "integrity": "sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=", - "dev": true, - "requires": { - "tweetnacl": "^0.14.3" - } - }, - "binary-extensions": { - "version": "1.13.1", - "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-1.13.1.tgz", - "integrity": "sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==", - "dev": true, - "optional": true - }, - "bindings": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/bindings/-/bindings-1.5.0.tgz", - "integrity": "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==", - "dev": true, - "optional": true, - "requires": { - "file-uri-to-path": "1.0.0" - } - }, - "block-stream": { - "version": "0.0.9", - "resolved": "https://registry.npmjs.org/block-stream/-/block-stream-0.0.9.tgz", - "integrity": "sha1-E+v+d4oDIFz+A3UUgeu0szAMEmo=", - "dev": true, - "optional": true, - "requires": { - "inherits": "~2.0.0" - } - }, - "body-parser": { - "version": "1.19.0", - "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", - "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", - "requires": { - "bytes": "3.1.0", - "content-type": "~1.0.4", - "debug": "2.6.9", - "depd": "~1.1.2", - "http-errors": "1.7.2", - "iconv-lite": "0.4.24", - "on-finished": "~2.3.0", - "qs": "6.7.0", - "raw-body": "2.4.0", - "type-is": "~1.6.17" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "requires": { - "ms": "2.0.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" - } - } - }, - "boxen": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/boxen/-/boxen-4.2.0.tgz", - "integrity": "sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ==", - "dev": true, - "requires": { - "ansi-align": "^3.0.0", - "camelcase": "^5.3.1", - "chalk": "^3.0.0", - "cli-boxes": "^2.2.0", - "string-width": "^4.1.0", - "term-size": "^2.1.0", - "type-fest": "^0.8.1", - "widest-line": "^3.1.0" - }, - "dependencies": { - "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", - "dev": true - }, - "chalk": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", - "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", - "dev": true, - "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" - } - }, - "is-fullwidth-code-point": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", - "dev": true - }, - "string-width": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", - "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", - "dev": true, - "requires": { - "emoji-regex": "^8.0.0", - "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.0" - } - }, - "strip-ansi": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", - "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", - "dev": true, - "requires": { - "ansi-regex": "^5.0.0" - } - } - } - }, - "brace-expansion": { - "version": "1.1.11", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", - "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", - "requires": { - "balanced-match": "^1.0.0", - "concat-map": "0.0.1" - } - }, - "braces": { - "version": "2.3.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", - "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", - "dev": true, - "requires": { - "arr-flatten": "^1.1.0", - "array-unique": "^0.3.2", - "extend-shallow": "^2.0.1", - "fill-range": "^4.0.0", - "isobject": "^3.0.1", - "repeat-element": "^1.1.2", - "snapdragon": "^0.8.1", - "snapdragon-node": "^2.0.1", - "split-string": "^3.0.2", - "to-regex": "^3.0.1" - }, - "dependencies": { - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - } - } - }, - "browser-process-hrtime": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/browser-process-hrtime/-/browser-process-hrtime-1.0.0.tgz", - "integrity": "sha512-9o5UecI3GhkpM6DrXr69PblIuWxPKk9Y0jHBRhdocZ2y7YECBFCsHm79Pr3OyR2AvjhDkabFJaDJMYRazHgsow==", - "dev": true - }, - "browserslist": { - "version": "4.14.7", - "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.14.7.tgz", - "integrity": "sha512-BSVRLCeG3Xt/j/1cCGj1019Wbty0H+Yvu2AOuZSuoaUWn3RatbL33Cxk+Q4jRMRAbOm0p7SLravLjpnT6s0vzQ==", - "dev": true, - "requires": { - "caniuse-lite": "^1.0.30001157", - "colorette": "^1.2.1", - "electron-to-chromium": "^1.3.591", - "escalade": "^3.1.1", - "node-releases": "^1.1.66" - } - }, - "bs-logger": { - "version": "0.2.6", - "resolved": "https://registry.npmjs.org/bs-logger/-/bs-logger-0.2.6.tgz", - "integrity": "sha512-pd8DCoxmbgc7hyPKOvxtqNcjYoOsABPQdcCUjGp3d42VR2CX1ORhk2A87oqqu5R1kk+76nsxZupkmyd+MVtCog==", - "dev": true, - "requires": { - "fast-json-stable-stringify": "2.x" - } - }, - "bser": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/bser/-/bser-2.1.1.tgz", - "integrity": "sha512-gQxTNE/GAfIIrmHLUE3oJyp5FO6HRBfhjnw4/wMmA63ZGDJnWBmgY/lyQBpnDUkGmAhbSe39tx2d/iTOAfglwQ==", - "dev": true, - "requires": { - "node-int64": "^0.4.0" - } - }, - "buffer": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", - "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", - "requires": { - "base64-js": "^1.3.1", - "ieee754": "^1.1.13" - } - }, - "buffer-equal-constant-time": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", - "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=" - }, - "buffer-from": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", - "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==", - "dev": true - }, - "bytes": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", - "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==" - }, - "cache-base": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/cache-base/-/cache-base-1.0.1.tgz", - "integrity": "sha512-AKcdTnFSWATd5/GCPRxr2ChwIJ85CeyrEyjRHlKxQ56d4XJMGym0uAiKn0xbLOGOl3+yRpOTi484dVCEc5AUzQ==", - "dev": true, - "requires": { - "collection-visit": "^1.0.0", - "component-emitter": "^1.2.1", - "get-value": "^2.0.6", - "has-value": "^1.0.0", - "isobject": "^3.0.1", - "set-value": "^2.0.0", - "to-object-path": "^0.3.0", - "union-value": "^1.0.0", - "unset-value": "^1.0.0" - } - }, - "cacheable-request": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-6.1.0.tgz", - "integrity": "sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg==", - "dev": true, - "requires": { - "clone-response": "^1.0.2", - "get-stream": "^5.1.0", - "http-cache-semantics": "^4.0.0", - "keyv": "^3.0.0", - "lowercase-keys": "^2.0.0", - "normalize-url": "^4.1.0", - "responselike": "^1.0.2" - }, - "dependencies": { - "get-stream": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", - "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", - "dev": true, - "requires": { - "pump": "^3.0.0" - } - }, - "lowercase-keys": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", - "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==", - "dev": true - } - } - }, - "call-bind": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.0.tgz", - "integrity": "sha512-AEXsYIyyDY3MCzbwdhzG3Jx1R0J2wetQyUynn6dYHAO+bg8l1k7jwZtRv4ryryFs7EP+NDlikJlVe59jr0cM2w==", - "dev": true, - "requires": { - "function-bind": "^1.1.1", - "get-intrinsic": "^1.0.0" - } - }, - "callsites": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", - "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", - "dev": true - }, - "camelcase": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", - "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==" - }, - "caniuse-lite": { - "version": "1.0.30001159", - "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001159.tgz", - "integrity": "sha512-w9Ph56jOsS8RL20K9cLND3u/+5WASWdhC/PPrf+V3/HsM3uHOavWOR1Xzakbv4Puo/srmPHudkmCRWM7Aq+/UA==", - "dev": true - }, - "capture-exit": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/capture-exit/-/capture-exit-2.0.0.tgz", - "integrity": "sha512-PiT/hQmTonHhl/HFGN+Lx3JJUznrVYJ3+AQsnthneZbvW7x+f08Tk7yLJTLEOUvBTbduLeeBkxEaYXUOUrRq6g==", - "dev": true, - "requires": { - "rsvp": "^4.8.4" - } - }, - "caseless": { - "version": "0.12.0", - "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", - "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=", - "dev": true - }, - "chalk": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", - "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", - "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" - } - }, - "char-regex": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/char-regex/-/char-regex-1.0.2.tgz", - "integrity": "sha512-kWWXztvZ5SBQV+eRgKFeh8q5sLuZY2+8WUIzlxWVTg+oGwY14qylx1KbKzHd8P6ZYkAg0xyIDU9JMHhyJMZ1jw==", - "dev": true - }, - "chokidar": { - "version": "3.4.3", - "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.4.3.tgz", - "integrity": "sha512-DtM3g7juCXQxFVSNPNByEC2+NImtBuxQQvWlHunpJIS5Ocr0lG306cC7FCi7cEA0fzmybPUIl4txBIobk1gGOQ==", - "dev": true, - "requires": { - "anymatch": "~3.1.1", - "braces": "~3.0.2", - "fsevents": "~2.1.2", - "glob-parent": "~5.1.0", - "is-binary-path": "~2.1.0", - "is-glob": "~4.0.1", - "normalize-path": "~3.0.0", - "readdirp": "~3.5.0" - }, - "dependencies": { - "anymatch": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", - "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", - "dev": true, - "requires": { - "normalize-path": "^3.0.0", - "picomatch": "^2.0.4" - } - }, - "binary-extensions": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.1.0.tgz", - "integrity": "sha512-1Yj8h9Q+QDF5FzhMs/c9+6UntbD5MkRfRwac8DoEm9ZfUBZ7tZ55YcGVAzEe4bXsdQHEk+s9S5wsOKVdZrw0tQ==", - "dev": true - }, - "braces": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", - "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", - "dev": true, - "requires": { - "fill-range": "^7.0.1" - } - }, - "fill-range": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", - "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", - "dev": true, - "requires": { - "to-regex-range": "^5.0.1" - } - }, - "fsevents": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz", - "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==", - "dev": true, - "optional": true - }, - "glob-parent": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz", - "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==", - "dev": true, - "requires": { - "is-glob": "^4.0.1" - } - }, - "is-binary-path": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", - "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", - "dev": true, - "requires": { - "binary-extensions": "^2.0.0" - } - }, - "is-number": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "dev": true - }, - "readdirp": { - "version": "3.5.0", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.5.0.tgz", - "integrity": "sha512-cMhu7c/8rdhkHXWsY+osBhfSy0JikwpHK/5+imo+LpeasTF8ouErHrlYkwT0++njiyuDvc7OFY5T3ukvZ8qmFQ==", - "dev": true, - "requires": { - "picomatch": "^2.2.1" - } - }, - "to-regex-range": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", - "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "dev": true, - "requires": { - "is-number": "^7.0.0" - } - } - } - }, - "chownr": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", - "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==" - }, - "ci-info": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-2.0.0.tgz", - "integrity": "sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ==", - "dev": true - }, - "cjs-module-lexer": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-0.6.0.tgz", - "integrity": "sha512-uc2Vix1frTfnuzxxu1Hp4ktSvM3QaI4oXl4ZUqL1wjTu/BGki9TrCWoqLTg/drR1KwAEarXuRFCG2Svr1GxPFw==", - "dev": true - }, - "class-transformer": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/class-transformer/-/class-transformer-0.3.1.tgz", - "integrity": "sha512-cKFwohpJbuMovS8xVLmn8N2AUbAuc8pVo4zEfsUVo8qgECOogns1WVk/FkOZoxhOPTyTYFckuoH+13FO+MQ8GA==" - }, - "class-utils": { - "version": "0.3.6", - "resolved": "https://registry.npmjs.org/class-utils/-/class-utils-0.3.6.tgz", - "integrity": "sha512-qOhPa/Fj7s6TY8H8esGu5QNpMMQxz79h+urzrNYN6mn+9BnxlDGf5QZ+XeCDsxSjPqsSR56XOZOJmpeurnLMeg==", - "dev": true, - "requires": { - "arr-union": "^3.1.0", - "define-property": "^0.2.5", - "isobject": "^3.0.0", - "static-extend": "^0.1.1" - }, - "dependencies": { - "define-property": { - "version": "0.2.5", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, - "requires": { - "is-descriptor": "^0.1.0" - } - } - } - }, - "class-validator": { - "version": "0.12.2", - "resolved": "https://registry.npmjs.org/class-validator/-/class-validator-0.12.2.tgz", - "integrity": "sha512-TDzPzp8BmpsbPhQpccB3jMUE/3pK0TyqamrK0kcx+ZeFytMA+O6q87JZZGObHHnoo9GM8vl/JppIyKWeEA/EVw==", - "requires": { - "@types/validator": "13.0.0", - "google-libphonenumber": "^3.2.8", - "tslib": ">=1.9.0", - "validator": "13.0.0" - } - }, - "cli-boxes": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-2.2.1.tgz", - "integrity": "sha512-y4coMcylgSCdVinjiDBuR8PCC2bLjyGTwEmPb9NHR/QaNU6EUOXcTY/s6VjGMD6ENSEaeQYHCY0GNGS5jfMwPw==", - "dev": true - }, - "cli-highlight": { - "version": "2.1.4", - "resolved": "https://registry.npmjs.org/cli-highlight/-/cli-highlight-2.1.4.tgz", - "integrity": "sha512-s7Zofobm20qriqDoU9sXptQx0t2R9PEgac92mENNm7xaEe1hn71IIMsXMK+6encA6WRCWWxIGQbipr3q998tlQ==", - "requires": { - "chalk": "^3.0.0", - "highlight.js": "^9.6.0", - "mz": "^2.4.0", - "parse5": "^5.1.1", - "parse5-htmlparser2-tree-adapter": "^5.1.1", - "yargs": "^15.0.0" - }, - "dependencies": { - "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==" - }, - "chalk": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", - "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", - "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" - } - }, - "is-fullwidth-code-point": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==" - }, - "string-width": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", - "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", - "requires": { - "emoji-regex": "^8.0.0", - "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.0" - } - }, - "strip-ansi": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", - "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", - "requires": { - "ansi-regex": "^5.0.0" - } - }, - "yargs": { - "version": "15.4.1", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.4.1.tgz", - "integrity": "sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A==", - "requires": { - "cliui": "^6.0.0", - "decamelize": "^1.2.0", - "find-up": "^4.1.0", - "get-caller-file": "^2.0.1", - "require-directory": "^2.1.1", - "require-main-filename": "^2.0.0", - "set-blocking": "^2.0.0", - "string-width": "^4.2.0", - "which-module": "^2.0.0", - "y18n": "^4.0.0", - "yargs-parser": "^18.1.2" - } - } - } - }, - "cliui": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-6.0.0.tgz", - "integrity": "sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ==", - "requires": { - "string-width": "^4.2.0", - "strip-ansi": "^6.0.0", - "wrap-ansi": "^6.2.0" - }, - "dependencies": { - "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==" - }, - "is-fullwidth-code-point": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==" - }, - "string-width": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", - "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", - "requires": { - "emoji-regex": "^8.0.0", - "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.0" - } - }, - "strip-ansi": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", - "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", - "requires": { - "ansi-regex": "^5.0.0" - } - } - } - }, - "clone-response": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz", - "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=", - "dev": true, - "requires": { - "mimic-response": "^1.0.0" - } - }, - "cls-hooked": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/cls-hooked/-/cls-hooked-4.2.2.tgz", - "integrity": "sha512-J4Xj5f5wq/4jAvcdgoGsL3G103BtWpZrMo8NEinRltN+xpTZdI+M38pyQqhuFU/P792xkMFvnKSf+Lm81U1bxw==", - "requires": { - "async-hook-jl": "^1.7.6", - "emitter-listener": "^1.0.1", - "semver": "^5.4.1" - } - }, - "co": { - "version": "4.6.0", - "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", - "integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=", - "dev": true - }, - "code-point-at": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz", - "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c=" - }, - "collect-v8-coverage": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/collect-v8-coverage/-/collect-v8-coverage-1.0.1.tgz", - "integrity": "sha512-iBPtljfCNcTKNAto0KEtDfZ3qzjJvqE3aTGZsbhjSBlorqpXJlaWWtPO35D+ZImoC3KWejX64o+yPGxhWSTzfg==", - "dev": true - }, - "collection-visit": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/collection-visit/-/collection-visit-1.0.0.tgz", - "integrity": "sha1-S8A3PBZLwykbTTaMgpzxqApZ3KA=", - "dev": true, - "requires": { - "map-visit": "^1.0.0", - "object-visit": "^1.0.0" - } - }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", - "requires": { - "color-name": "~1.1.4" - } - }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" - }, - "colorette": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/colorette/-/colorette-1.2.1.tgz", - "integrity": "sha512-puCDz0CzydiSYOrnXpz/PKd69zRrribezjtE9yd4zvytoRc8+RY/KJPvtPFKZS3E3wP6neGyMe0vOTlHO5L3Pw==", - "dev": true - }, - "combined-stream": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", - "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", - "dev": true, - "requires": { - "delayed-stream": "~1.0.0" - } - }, - "commander": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/commander/-/commander-4.1.1.tgz", - "integrity": "sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==", - "dev": true - }, - "commondir": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz", - "integrity": "sha1-3dgA2gxmEnOTzKWVDqloo6rxJTs=", - "dev": true - }, - "component-emitter": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz", - "integrity": "sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg==", - "dev": true - }, - "concat-map": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", - "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" - }, - "configstore": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/configstore/-/configstore-5.0.1.tgz", - "integrity": "sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA==", - "dev": true, - "requires": { - "dot-prop": "^5.2.0", - "graceful-fs": "^4.1.2", - "make-dir": "^3.0.0", - "unique-string": "^2.0.0", - "write-file-atomic": "^3.0.0", - "xdg-basedir": "^4.0.0" - }, - "dependencies": { - "make-dir": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", - "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", - "dev": true, - "requires": { - "semver": "^6.0.0" - } - }, - "semver": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", - "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", - "dev": true - } - } - }, - "confusing-browser-globals": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/confusing-browser-globals/-/confusing-browser-globals-1.0.10.tgz", - "integrity": "sha512-gNld/3lySHwuhaVluJUKLePYirM3QNCKzVxqAdhJII9/WXKVX5PURzMVJspS1jTslSqjeuG4KMVTSouit5YPHA==", - "dev": true - }, - "console-control-strings": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/console-control-strings/-/console-control-strings-1.1.0.tgz", - "integrity": "sha1-PXz0Rk22RG6mRL9LOVB/mFEAjo4=" - }, - "contains-path": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/contains-path/-/contains-path-0.1.0.tgz", - "integrity": "sha1-/ozxhP9mcLa67wGp1IYaXL7EEgo=", - "dev": true - }, - "content-disposition": { - "version": "0.5.3", - "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", - "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", - "requires": { - "safe-buffer": "5.1.2" - } - }, - "content-type": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", - "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" - }, - "convert-source-map": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.7.0.tgz", - "integrity": "sha512-4FJkXzKXEDB1snCFZlLP4gpC3JILicCpGbzG9f9G7tGqGCzETQ2hWPrcinA9oU4wtf2biUaEH5065UnMeR33oA==", - "dev": true, - "requires": { - "safe-buffer": "~5.1.1" - } - }, - "cookie": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", - "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" - }, - "cookie-parser": { - "version": "1.4.5", - "resolved": "https://registry.npmjs.org/cookie-parser/-/cookie-parser-1.4.5.tgz", - "integrity": "sha512-f13bPUj/gG/5mDr+xLmSxxDsB9DQiTIfhJS/sqjrmfAWiAN+x2O4i/XguTL9yDZ+/IFDanJ+5x7hC4CXT9Tdzw==", - "requires": { - "cookie": "0.4.0", - "cookie-signature": "1.0.6" - } - }, - "cookie-signature": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", - "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" - }, - "cookiejar": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/cookiejar/-/cookiejar-2.1.2.tgz", - "integrity": "sha512-Mw+adcfzPxcPeI+0WlvRrr/3lGVO0bD75SxX6811cxSh1Wbxx7xZBGK1eVtDf6si8rg2lhnUjsVLMFMfbRIuwA==", - "dev": true - }, - "copy-descriptor": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/copy-descriptor/-/copy-descriptor-0.1.1.tgz", - "integrity": "sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=", - "dev": true - }, - "core-js": { - "version": "3.7.0", - "resolved": "https://registry.npmjs.org/core-js/-/core-js-3.7.0.tgz", - "integrity": "sha512-NwS7fI5M5B85EwpWuIwJN4i/fbisQUwLwiSNUWeXlkAZ0sbBjLEvLvFLf1uzAUV66PcEPt4xCGCmOZSxVf3xzA==", - "dev": true - }, - "core-js-compat": { - "version": "3.7.0", - "resolved": "https://registry.npmjs.org/core-js-compat/-/core-js-compat-3.7.0.tgz", - "integrity": "sha512-V8yBI3+ZLDVomoWICO6kq/CD28Y4r1M7CWeO4AGpMdMfseu8bkSubBmUPySMGKRTS+su4XQ07zUkAsiu9FCWTg==", - "dev": true, - "requires": { - "browserslist": "^4.14.6", - "semver": "7.0.0" - }, - "dependencies": { - "semver": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.0.0.tgz", - "integrity": "sha512-+GB6zVA9LWh6zovYQLALHwv5rb2PHGlJi3lfiqIHxR0uuwCgefcOJc59v9fv1w8GbStwxuuqqAjI9NMAOOgq1A==", - "dev": true - } - } - }, - "core-util-is": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", - "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" - }, - "cors": { - "version": "2.8.5", - "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", - "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", - "requires": { - "object-assign": "^4", - "vary": "^1" - } - }, - "cross-spawn": { - "version": "6.0.5", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", - "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", - "dev": true, - "requires": { - "nice-try": "^1.0.4", - "path-key": "^2.0.1", - "semver": "^5.5.0", - "shebang-command": "^1.2.0", - "which": "^1.2.9" - } - }, - "crypto-random-string": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz", - "integrity": "sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA==", - "dev": true - }, - "cssom": { - "version": "0.4.4", - "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.4.4.tgz", - "integrity": "sha512-p3pvU7r1MyyqbTk+WbNJIgJjG2VmTIaB10rI93LzVPrmDJKkzKYMtxxyAvQXR/NS6otuzveI7+7BBq3SjBS2mw==", - "dev": true - }, - "cssstyle": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/cssstyle/-/cssstyle-2.3.0.tgz", - "integrity": "sha512-AZL67abkUzIuvcHqk7c09cezpGNcxUxU4Ioi/05xHk4DQeTkWmGYftIE6ctU6AEt+Gn4n1lDStOtj7FKycP71A==", - "dev": true, - "requires": { - "cssom": "~0.3.6" - }, - "dependencies": { - "cssom": { - "version": "0.3.8", - "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.3.8.tgz", - "integrity": "sha512-b0tGHbfegbhPJpxpiBPU2sCkigAqtM9O121le6bbOlgyV+NyGyCmVfJ6QW9eRjz8CpNfWEOYBIMIGRYkLwsIYg==", - "dev": true - } - } - }, - "dashdash": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", - "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", - "dev": true, - "requires": { - "assert-plus": "^1.0.0" - } - }, - "data-urls": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-2.0.0.tgz", - "integrity": "sha512-X5eWTSXO/BJmpdIKCRuKUgSCgAN0OwliVK3yPKbwIWU1Tdw5BRajxlzMidvh+gwko9AfQ9zIj52pzF91Q3YAvQ==", - "dev": true, - "requires": { - "abab": "^2.0.3", - "whatwg-mimetype": "^2.3.0", - "whatwg-url": "^8.0.0" - } - }, - "debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "requires": { - "ms": "^2.1.1" - } - }, - "decamelize": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", - "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=" - }, - "decimal.js": { - "version": "10.2.1", - "resolved": "https://registry.npmjs.org/decimal.js/-/decimal.js-10.2.1.tgz", - "integrity": "sha512-KaL7+6Fw6i5A2XSnsbhm/6B+NuEA7TZ4vqxnd5tXz9sbKtrN9Srj8ab4vKVdK8YAqZO9P1kg45Y6YLoduPf+kw==", - "dev": true - }, - "decode-uri-component": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/decode-uri-component/-/decode-uri-component-0.2.0.tgz", - "integrity": "sha1-6zkTMzRYd1y4TNGh+uBiEGu4dUU=", - "dev": true - }, - "decompress-response": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz", - "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=", - "dev": true, - "requires": { - "mimic-response": "^1.0.0" - } - }, - "deep-extend": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", - "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==" - }, - "deep-is": { - "version": "0.1.3", - "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz", - "integrity": "sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=", - "dev": true - }, - "deepmerge": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.2.2.tgz", - "integrity": "sha512-FJ3UgI4gIl+PHZm53knsuSFpE+nESMr7M4v9QcgB7S63Kj/6WqMiFQJpBBYz1Pt+66bZpP3Q7Lye0Oo9MPKEdg==", - "dev": true - }, - "defer-to-connect": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-1.1.3.tgz", - "integrity": "sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ==", - "dev": true - }, - "define-properties": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", - "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", - "dev": true, - "requires": { - "object-keys": "^1.0.12" - } - }, - "define-property": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-2.0.2.tgz", - "integrity": "sha512-jwK2UV4cnPpbcG7+VRARKTZPUWowwXA8bzH5NP6ud0oeAxyYPuGZUAC7hMugpCdz4BeSZl2Dl9k66CHJ/46ZYQ==", - "dev": true, - "requires": { - "is-descriptor": "^1.0.2", - "isobject": "^3.0.1" - }, - "dependencies": { - "is-accessor-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", - "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-data-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", - "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-descriptor": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", - "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, - "requires": { - "is-accessor-descriptor": "^1.0.0", - "is-data-descriptor": "^1.0.0", - "kind-of": "^6.0.2" - } - } - } - }, - "delayed-stream": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", - "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", - "dev": true - }, - "delegates": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/delegates/-/delegates-1.0.0.tgz", - "integrity": "sha1-hMbhWbgZBP3KWaDvRM2HDTElD5o=" - }, - "denque": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/denque/-/denque-1.4.1.tgz", - "integrity": "sha512-OfzPuSZKGcgr96rf1oODnfjqBFmr1DVoc/TrItj3Ohe0Ah1C5WX5Baquw/9U9KovnQ88EqmJbD66rKYUQYN1tQ==" - }, - "depd": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", - "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" - }, - "destroy": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", - "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" - }, - "detect-libc": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-1.0.3.tgz", - "integrity": "sha1-+hN8S9aY7fVc1c0CrFWfkaTEups=" - }, - "detect-newline": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/detect-newline/-/detect-newline-3.1.0.tgz", - "integrity": "sha512-TLz+x/vEXm/Y7P7wn1EJFNLxYpUD4TgMosxY6fAVJUnJMbupHBOncxyWUG9OpTaH9EBD7uFI5LfEgmMOc54DsA==", - "dev": true - }, - "diff-sequences": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-26.6.2.tgz", - "integrity": "sha512-Mv/TDa3nZ9sbc5soK+OoA74BsS3mL37yixCvUAQkiuA4Wz6YtwP/K47n2rv2ovzHZvoiQeA5FTQOschKkEwB0Q==", - "dev": true - }, - "dir-glob": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", - "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", - "dev": true, - "requires": { - "path-type": "^4.0.0" - } - }, - "doctrine": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", - "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", - "dev": true, - "requires": { - "esutils": "^2.0.2" - } - }, - "domexception": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/domexception/-/domexception-2.0.1.tgz", - "integrity": "sha512-yxJ2mFy/sibVQlu5qHjOkf9J3K6zgmCxgJ94u2EdvDOV09H+32LtRswEcUsmUWN72pVLOEnTSRaIVVzVQgS0dg==", - "dev": true, - "requires": { - "webidl-conversions": "^5.0.0" - }, - "dependencies": { - "webidl-conversions": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-5.0.0.tgz", - "integrity": "sha512-VlZwKPCkYKxQgeSbH5EyngOmRp7Ww7I9rQLERETtf5ofd9pGeswWiOtogpEO850jziPRarreGxn5QIiTqpb2wA==", - "dev": true - } - } - }, - "dot-prop": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.3.0.tgz", - "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==", - "dev": true, - "requires": { - "is-obj": "^2.0.0" - } - }, - "dotenv": { - "version": "8.2.0", - "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-8.2.0.tgz", - "integrity": "sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw==" - }, - "duplexer3": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", - "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", - "dev": true - }, - "ecc-jsbn": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", - "integrity": "sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=", - "dev": true, - "requires": { - "jsbn": "~0.1.0", - "safer-buffer": "^2.1.0" - } - }, - "ecdsa-sig-formatter": { - "version": "1.0.11", - "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", - "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", - "requires": { - "safe-buffer": "^5.0.1" - } - }, - "ee-first": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", - "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" - }, - "electron-to-chromium": { - "version": "1.3.601", - "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.601.tgz", - "integrity": "sha512-ctRyXD9y0mZu8pgeNwBUhLP3Guyr5YuqkfLKYmpTwYx7o9JtCEJme9JVX4xBXPr5ZNvr/iBXUvHLFEVJQThATg==", - "dev": true - }, - "emitter-listener": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/emitter-listener/-/emitter-listener-1.1.2.tgz", - "integrity": "sha512-Bt1sBAGFHY9DKY+4/2cV6izcKJUf5T7/gkdmkxzX/qv9CcGH8xSwVRW5mtX03SWJtRTWSOpzCuWN9rBFYZepZQ==", - "requires": { - "shimmer": "^1.2.0" - } - }, - "emittery": { - "version": "0.7.2", - "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.7.2.tgz", - "integrity": "sha512-A8OG5SR/ij3SsJdWDJdkkSYUjQdCUx6APQXem0SaEePBSRg4eymGYwBkKo1Y6DU+af/Jn2dBQqDBvjnr9Vi8nQ==", - "dev": true - }, - "emoji-regex": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", - "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" - }, - "encodeurl": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", - "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" - }, - "end-of-stream": { - "version": "1.4.4", - "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", - "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", - "dev": true, - "requires": { - "once": "^1.4.0" - } - }, - "enquirer": { - "version": "2.3.6", - "resolved": "https://registry.npmjs.org/enquirer/-/enquirer-2.3.6.tgz", - "integrity": "sha512-yjNnPr315/FjS4zIsUxYguYUPP2e1NK4d7E7ZOLiyYCcbFBiTMyID+2wvm2w6+pZ/odMA7cRkjhsPbltwBOrLg==", - "dev": true, - "requires": { - "ansi-colors": "^4.1.1" - } - }, - "error-ex": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", - "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", - "dev": true, - "requires": { - "is-arrayish": "^0.2.1" - } - }, - "es-abstract": { - "version": "1.17.7", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.7.tgz", - "integrity": "sha512-VBl/gnfcJ7OercKA9MVaegWsBHFjV492syMudcnQZvt/Dw8ezpcOHYZXa/J96O8vx+g4x65YKhxOwDUh63aS5g==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - }, - "es-to-primitive": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", - "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", - "dev": true, - "requires": { - "is-callable": "^1.1.4", - "is-date-object": "^1.0.1", - "is-symbol": "^1.0.2" - } - }, - "escalade": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", - "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==" - }, - "escape-goat": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/escape-goat/-/escape-goat-2.1.1.tgz", - "integrity": "sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q==", - "dev": true - }, - "escape-html": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", - "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" - }, - "escape-string-regexp": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", - "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=" - }, - "escodegen": { - "version": "1.14.3", - "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.14.3.tgz", - "integrity": "sha512-qFcX0XJkdg+PB3xjZZG/wKSuT1PnQWx57+TVSjIMmILd2yC/6ByYElPwJnslDsuWuSAp4AwJGumarAAmJch5Kw==", - "dev": true, - "requires": { - "esprima": "^4.0.1", - "estraverse": "^4.2.0", - "esutils": "^2.0.2", - "optionator": "^0.8.1", - "source-map": "~0.6.1" - }, - "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true, - "optional": true - } - } - }, - "eslint": { - "version": "7.13.0", - "resolved": "https://registry.npmjs.org/eslint/-/eslint-7.13.0.tgz", - "integrity": "sha512-uCORMuOO8tUzJmsdRtrvcGq5qposf7Rw0LwkTJkoDbOycVQtQjmnhZSuLQnozLE4TmAzlMVV45eCHmQ1OpDKUQ==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.0.0", - "@eslint/eslintrc": "^0.2.1", - "ajv": "^6.10.0", - "chalk": "^4.0.0", - "cross-spawn": "^7.0.2", - "debug": "^4.0.1", - "doctrine": "^3.0.0", - "enquirer": "^2.3.5", - "eslint-scope": "^5.1.1", - "eslint-utils": "^2.1.0", - "eslint-visitor-keys": "^2.0.0", - "espree": "^7.3.0", - "esquery": "^1.2.0", - "esutils": "^2.0.2", - "file-entry-cache": "^5.0.1", - "functional-red-black-tree": "^1.0.1", - "glob-parent": "^5.0.0", - "globals": "^12.1.0", - "ignore": "^4.0.6", - "import-fresh": "^3.0.0", - "imurmurhash": "^0.1.4", - "is-glob": "^4.0.0", - "js-yaml": "^3.13.1", - "json-stable-stringify-without-jsonify": "^1.0.1", - "levn": "^0.4.1", - "lodash": "^4.17.19", - "minimatch": "^3.0.4", - "natural-compare": "^1.4.0", - "optionator": "^0.9.1", - "progress": "^2.0.0", - "regexpp": "^3.1.0", - "semver": "^7.2.1", - "strip-ansi": "^6.0.0", - "strip-json-comments": "^3.1.0", - "table": "^5.2.3", - "text-table": "^0.2.0", - "v8-compile-cache": "^2.0.3" - }, - "dependencies": { - "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", - "dev": true - }, - "cross-spawn": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", - "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", - "dev": true, - "requires": { - "path-key": "^3.1.0", - "shebang-command": "^2.0.0", - "which": "^2.0.1" - } - }, - "debug": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", - "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", - "dev": true, - "requires": { - "ms": "2.1.2" - } - }, - "eslint-scope": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz", - "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", - "dev": true, - "requires": { - "esrecurse": "^4.3.0", - "estraverse": "^4.1.1" - } - }, - "eslint-visitor-keys": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.0.0.tgz", - "integrity": "sha512-QudtT6av5WXels9WjIM7qz1XD1cWGvX4gGXvp/zBn9nXG02D0utdU3Em2m/QjTnrsk6bBjmCygl3rmj118msQQ==", - "dev": true - }, - "glob-parent": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz", - "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==", - "dev": true, - "requires": { - "is-glob": "^4.0.1" - } - }, - "globals": { - "version": "12.4.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-12.4.0.tgz", - "integrity": "sha512-BWICuzzDvDoH54NHKCseDanAhE3CeDorgDL5MT6LMXXj2WCnd9UC2szdk4AWLfjdgNBCXLUanXYcpBBKOSWGwg==", - "dev": true, - "requires": { - "type-fest": "^0.8.1" - } - }, - "ignore": { - "version": "4.0.6", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", - "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", - "dev": true - }, - "levn": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", - "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", - "dev": true, - "requires": { - "prelude-ls": "^1.2.1", - "type-check": "~0.4.0" - } - }, - "optionator": { - "version": "0.9.1", - "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.1.tgz", - "integrity": "sha512-74RlY5FCnhq4jRxVUPKDaRwrVNXMqsGsiW6AJw4XK8hmtm10wC0ypZBLw5IIp85NZMr91+qd1RvvENwg7jjRFw==", - "dev": true, - "requires": { - "deep-is": "^0.1.3", - "fast-levenshtein": "^2.0.6", - "levn": "^0.4.1", - "prelude-ls": "^1.2.1", - "type-check": "^0.4.0", - "word-wrap": "^1.2.3" - } - }, - "path-key": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", - "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", - "dev": true - }, - "prelude-ls": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", - "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", - "dev": true - }, - "semver": { - "version": "7.3.2", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", - "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", - "dev": true - }, - "shebang-command": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", - "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", - "dev": true, - "requires": { - "shebang-regex": "^3.0.0" - } - }, - "shebang-regex": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", - "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", - "dev": true - }, - "strip-ansi": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", - "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", - "dev": true, - "requires": { - "ansi-regex": "^5.0.0" - } - }, - "strip-json-comments": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", - "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", - "dev": true - }, - "type-check": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", - "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", - "dev": true, - "requires": { - "prelude-ls": "^1.2.1" - } - }, - "which": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", - "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", - "dev": true, - "requires": { - "isexe": "^2.0.0" - } - } - } - }, - "eslint-config-airbnb-base": { - "version": "14.2.1", - "resolved": "https://registry.npmjs.org/eslint-config-airbnb-base/-/eslint-config-airbnb-base-14.2.1.tgz", - "integrity": "sha512-GOrQyDtVEc1Xy20U7vsB2yAoB4nBlfH5HZJeatRXHleO+OS5Ot+MWij4Dpltw4/DyIkqUfqz1epfhVR5XWWQPA==", - "dev": true, - "requires": { - "confusing-browser-globals": "^1.0.10", - "object.assign": "^4.1.2", - "object.entries": "^1.1.2" - } - }, - "eslint-config-prettier": { - "version": "6.15.0", - "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-6.15.0.tgz", - "integrity": "sha512-a1+kOYLR8wMGustcgAjdydMsQ2A/2ipRPwRKUmfYaSxc9ZPcrku080Ctl6zrZzZNs/U82MjSv+qKREkoq3bJaw==", - "dev": true, - "requires": { - "get-stdin": "^6.0.0" - } - }, - "eslint-import-resolver-node": { - "version": "0.3.4", - "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.4.tgz", - "integrity": "sha512-ogtf+5AB/O+nM6DIeBUNr2fuT7ot9Qg/1harBfBtaP13ekEWFQEEMP94BCB7zaNW3gyY+8SHYF00rnqYwXKWOA==", - "dev": true, - "requires": { - "debug": "^2.6.9", - "resolve": "^1.13.1" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "dev": true, - "requires": { - "ms": "2.0.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - } - } - }, - "eslint-module-utils": { - "version": "2.6.0", - "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.6.0.tgz", - "integrity": "sha512-6j9xxegbqe8/kZY8cYpcp0xhbK0EgJlg3g9mib3/miLaExuuwc3n5UEfSnU6hWMbT0FAYVvDbL9RrRgpUeQIvA==", - "dev": true, - "requires": { - "debug": "^2.6.9", - "pkg-dir": "^2.0.0" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "dev": true, - "requires": { - "ms": "2.0.0" - } - }, - "find-up": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", - "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", - "dev": true, - "requires": { - "locate-path": "^2.0.0" - } - }, - "locate-path": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz", - "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", - "dev": true, - "requires": { - "p-locate": "^2.0.0", - "path-exists": "^3.0.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - }, - "p-limit": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz", - "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", - "dev": true, - "requires": { - "p-try": "^1.0.0" - } - }, - "p-locate": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", - "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", - "dev": true, - "requires": { - "p-limit": "^1.1.0" - } - }, - "p-try": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz", - "integrity": "sha1-y8ec26+P1CKOE/Yh8rGiN8GyB7M=", - "dev": true - }, - "path-exists": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", - "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", - "dev": true - }, - "pkg-dir": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-2.0.0.tgz", - "integrity": "sha1-9tXREJ4Z1j7fQo4L1X4Sd3YVM0s=", - "dev": true, - "requires": { - "find-up": "^2.1.0" - } - } - } - }, - "eslint-plugin-import": { - "version": "2.22.1", - "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.22.1.tgz", - "integrity": "sha512-8K7JjINHOpH64ozkAhpT3sd+FswIZTfMZTjdx052pnWrgRCVfp8op9tbjpAk3DdUeI/Ba4C8OjdC0r90erHEOw==", - "dev": true, - "requires": { - "array-includes": "^3.1.1", - "array.prototype.flat": "^1.2.3", - "contains-path": "^0.1.0", - "debug": "^2.6.9", - "doctrine": "1.5.0", - "eslint-import-resolver-node": "^0.3.4", - "eslint-module-utils": "^2.6.0", - "has": "^1.0.3", - "minimatch": "^3.0.4", - "object.values": "^1.1.1", - "read-pkg-up": "^2.0.0", - "resolve": "^1.17.0", - "tsconfig-paths": "^3.9.0" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "dev": true, - "requires": { - "ms": "2.0.0" - } - }, - "doctrine": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-1.5.0.tgz", - "integrity": "sha1-N53Ocw9hZvds76TmcHoVmwLFpvo=", - "dev": true, - "requires": { - "esutils": "^2.0.2", - "isarray": "^1.0.0" - } - }, - "find-up": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", - "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", - "dev": true, - "requires": { - "locate-path": "^2.0.0" - } - }, - "locate-path": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz", - "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", - "dev": true, - "requires": { - "p-locate": "^2.0.0", - "path-exists": "^3.0.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - }, - "p-limit": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz", - "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", - "dev": true, - "requires": { - "p-try": "^1.0.0" - } - }, - "p-locate": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", - "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", - "dev": true, - "requires": { - "p-limit": "^1.1.0" - } - }, - "p-try": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz", - "integrity": "sha1-y8ec26+P1CKOE/Yh8rGiN8GyB7M=", - "dev": true - }, - "path-exists": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", - "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", - "dev": true - }, - "path-type": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/path-type/-/path-type-2.0.0.tgz", - "integrity": "sha1-8BLMuEFbcJb8LaoQVMPXI4lZTHM=", - "dev": true, - "requires": { - "pify": "^2.0.0" - } - }, - "pify": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", - "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", - "dev": true - }, - "read-pkg": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-2.0.0.tgz", - "integrity": "sha1-jvHAYjxqbbDcZxPEv6xGMysjaPg=", - "dev": true, - "requires": { - "load-json-file": "^2.0.0", - "normalize-package-data": "^2.3.2", - "path-type": "^2.0.0" - } - }, - "read-pkg-up": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-2.0.0.tgz", - "integrity": "sha1-a3KoBImE4MQeeVEP1en6mbO1Sb4=", - "dev": true, - "requires": { - "find-up": "^2.0.0", - "read-pkg": "^2.0.0" - } - } - } - }, - "eslint-plugin-prettier": { - "version": "3.1.4", - "resolved": "https://registry.npmjs.org/eslint-plugin-prettier/-/eslint-plugin-prettier-3.1.4.tgz", - "integrity": "sha512-jZDa8z76klRqo+TdGDTFJSavwbnWK2ZpqGKNZ+VvweMW516pDUMmQ2koXvxEE4JhzNvTv+radye/bWGBmA6jmg==", - "dev": true, - "requires": { - "prettier-linter-helpers": "^1.0.0" - } - }, - "eslint-scope": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.0.tgz", - "integrity": "sha512-iiGRvtxWqgtx5m8EyQUJihBloE4EnYeGE/bz1wSPwJE6tZuJUtHlhqDM4Xj2ukE8Dyy1+HCZ4hE0fzIVMzb58w==", - "dev": true, - "requires": { - "esrecurse": "^4.1.0", - "estraverse": "^4.1.1" - } - }, - "eslint-utils": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.1.0.tgz", - "integrity": "sha512-w94dQYoauyvlDc43XnGB8lU3Zt713vNChgt4EWwhXAP2XkBvndfxF0AgIqKOOasjPIPzj9JqgwkwbCYD0/V3Zg==", - "dev": true, - "requires": { - "eslint-visitor-keys": "^1.1.0" - } - }, - "eslint-visitor-keys": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", - "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", - "dev": true - }, - "espree": { - "version": "7.3.0", - "resolved": "https://registry.npmjs.org/espree/-/espree-7.3.0.tgz", - "integrity": "sha512-dksIWsvKCixn1yrEXO8UosNSxaDoSYpq9reEjZSbHLpT5hpaCAKTLBwq0RHtLrIr+c0ByiYzWT8KTMRzoRCNlw==", - "dev": true, - "requires": { - "acorn": "^7.4.0", - "acorn-jsx": "^5.2.0", - "eslint-visitor-keys": "^1.3.0" - } - }, - "esprima": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", - "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==" - }, - "esquery": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.3.1.tgz", - "integrity": "sha512-olpvt9QG0vniUBZspVRN6lwB7hOZoTRtT+jzR+tS4ffYx2mzbw+z0XCOk44aaLYKApNX5nMm+E+P6o25ip/DHQ==", - "dev": true, - "requires": { - "estraverse": "^5.1.0" - }, - "dependencies": { - "estraverse": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.2.0.tgz", - "integrity": "sha512-BxbNGGNm0RyRYvUdHpIwv9IWzeM9XClbOxwoATuFdOE7ZE6wHL+HQ5T8hoPM+zHvmKzzsEqhgy0GrQ5X13afiQ==", - "dev": true - } - } - }, - "esrecurse": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", - "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", - "dev": true, - "requires": { - "estraverse": "^5.2.0" - }, - "dependencies": { - "estraverse": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.2.0.tgz", - "integrity": "sha512-BxbNGGNm0RyRYvUdHpIwv9IWzeM9XClbOxwoATuFdOE7ZE6wHL+HQ5T8hoPM+zHvmKzzsEqhgy0GrQ5X13afiQ==", - "dev": true - } - } - }, - "estraverse": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", - "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", - "dev": true - }, - "esutils": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", - "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", - "dev": true - }, - "etag": { - "version": "1.8.1", - "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", - "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" - }, - "exec-sh": { - "version": "0.3.4", - "resolved": "https://registry.npmjs.org/exec-sh/-/exec-sh-0.3.4.tgz", - "integrity": "sha512-sEFIkc61v75sWeOe72qyrqg2Qg0OuLESziUDk/O/z2qgS15y2gWVFrI6f2Qn/qw/0/NCfCEsmNA4zOjkwEZT1A==", - "dev": true - }, - "execa": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/execa/-/execa-1.0.0.tgz", - "integrity": "sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA==", - "dev": true, - "requires": { - "cross-spawn": "^6.0.0", - "get-stream": "^4.0.0", - "is-stream": "^1.1.0", - "npm-run-path": "^2.0.0", - "p-finally": "^1.0.0", - "signal-exit": "^3.0.0", - "strip-eof": "^1.0.0" - } - }, - "exit": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", - "integrity": "sha1-BjJjj42HfMghB9MKD/8aF8uhzQw=", - "dev": true - }, - "expand-brackets": { - "version": "2.1.4", - "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-2.1.4.tgz", - "integrity": "sha1-t3c14xXOMPa27/D4OwQVGiJEliI=", - "dev": true, - "requires": { - "debug": "^2.3.3", - "define-property": "^0.2.5", - "extend-shallow": "^2.0.1", - "posix-character-classes": "^0.1.0", - "regex-not": "^1.0.0", - "snapdragon": "^0.8.1", - "to-regex": "^3.0.1" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "dev": true, - "requires": { - "ms": "2.0.0" - } - }, - "define-property": { - "version": "0.2.5", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, - "requires": { - "is-descriptor": "^0.1.0" - } - }, - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - } - } - }, - "expect": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/expect/-/expect-26.6.2.tgz", - "integrity": "sha512-9/hlOBkQl2l/PLHJx6JjoDF6xPKcJEsUlWKb23rKE7KzeDqUZKXKNMW27KIue5JMdBV9HgmoJPcc8HtO85t9IA==", - "dev": true, - "requires": { - "@jest/types": "^26.6.2", - "ansi-styles": "^4.0.0", - "jest-get-type": "^26.3.0", - "jest-matcher-utils": "^26.6.2", - "jest-message-util": "^26.6.2", - "jest-regex-util": "^26.0.0" - } - }, - "express": { - "version": "4.17.1", - "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", - "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", - "requires": { - "accepts": "~1.3.7", - "array-flatten": "1.1.1", - "body-parser": "1.19.0", - "content-disposition": "0.5.3", - "content-type": "~1.0.4", - "cookie": "0.4.0", - "cookie-signature": "1.0.6", - "debug": "2.6.9", - "depd": "~1.1.2", - "encodeurl": "~1.0.2", - "escape-html": "~1.0.3", - "etag": "~1.8.1", - "finalhandler": "~1.1.2", - "fresh": "0.5.2", - "merge-descriptors": "1.0.1", - "methods": "~1.1.2", - "on-finished": "~2.3.0", - "parseurl": "~1.3.3", - "path-to-regexp": "0.1.7", - "proxy-addr": "~2.0.5", - "qs": "6.7.0", - "range-parser": "~1.2.1", - "safe-buffer": "5.1.2", - "send": "0.17.1", - "serve-static": "1.14.1", - "setprototypeof": "1.1.1", - "statuses": "~1.5.0", - "type-is": "~1.6.18", - "utils-merge": "1.0.1", - "vary": "~1.1.2" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "requires": { - "ms": "2.0.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" - } - } - }, - "express-async-errors": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/express-async-errors/-/express-async-errors-3.1.1.tgz", - "integrity": "sha512-h6aK1da4tpqWSbyCa3FxB/V6Ehd4EEB15zyQq9qe75OZBp0krinNKuH4rAY+S/U/2I36vdLAUFSjQJ+TFmODng==" - }, - "extend": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", - "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", - "dev": true - }, - "extend-shallow": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-3.0.2.tgz", - "integrity": "sha1-Jqcarwc7OfshJxcnRhMcJwQCjbg=", - "dev": true, - "requires": { - "assign-symbols": "^1.0.0", - "is-extendable": "^1.0.1" - }, - "dependencies": { - "is-extendable": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", - "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", - "dev": true, - "requires": { - "is-plain-object": "^2.0.4" - } - } - } - }, - "extglob": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/extglob/-/extglob-2.0.4.tgz", - "integrity": "sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw==", - "dev": true, - "requires": { - "array-unique": "^0.3.2", - "define-property": "^1.0.0", - "expand-brackets": "^2.1.4", - "extend-shallow": "^2.0.1", - "fragment-cache": "^0.2.1", - "regex-not": "^1.0.0", - "snapdragon": "^0.8.1", - "to-regex": "^3.0.1" - }, - "dependencies": { - "define-property": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", - "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", - "dev": true, - "requires": { - "is-descriptor": "^1.0.0" - } - }, - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - }, - "is-accessor-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", - "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-data-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", - "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-descriptor": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", - "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, - "requires": { - "is-accessor-descriptor": "^1.0.0", - "is-data-descriptor": "^1.0.0", - "kind-of": "^6.0.2" - } - } - } - }, - "extsprintf": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", - "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=", - "dev": true - }, - "fast-deep-equal": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", - "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", - "dev": true - }, - "fast-diff": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/fast-diff/-/fast-diff-1.2.0.tgz", - "integrity": "sha512-xJuoT5+L99XlZ8twedaRf6Ax2TgQVxvgZOYoPKqZufmJib0tL2tegPBOZb1pVNgIhlqDlA0eO0c3wBvQcmzx4w==", - "dev": true - }, - "fast-glob": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.2.4.tgz", - "integrity": "sha512-kr/Oo6PX51265qeuCYsyGypiO5uJFgBS0jksyG7FUeCyQzNwYnzrNIMR1NXfkZXsMYXYLRAHgISHBz8gQcxKHQ==", - "dev": true, - "requires": { - "@nodelib/fs.stat": "^2.0.2", - "@nodelib/fs.walk": "^1.2.3", - "glob-parent": "^5.1.0", - "merge2": "^1.3.0", - "micromatch": "^4.0.2", - "picomatch": "^2.2.1" - }, - "dependencies": { - "braces": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", - "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", - "dev": true, - "requires": { - "fill-range": "^7.0.1" - } - }, - "fill-range": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", - "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", - "dev": true, - "requires": { - "to-regex-range": "^5.0.1" - } - }, - "glob-parent": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz", - "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==", - "dev": true, - "requires": { - "is-glob": "^4.0.1" - } - }, - "is-number": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "dev": true - }, - "micromatch": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", - "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", - "dev": true, - "requires": { - "braces": "^3.0.1", - "picomatch": "^2.0.5" - } - }, - "to-regex-range": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", - "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "dev": true, - "requires": { - "is-number": "^7.0.0" - } - } - } - }, - "fast-json-stable-stringify": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", - "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", - "dev": true - }, - "fast-levenshtein": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", - "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", - "dev": true - }, - "fast-safe-stringify": { - "version": "2.0.7", - "resolved": "https://registry.npmjs.org/fast-safe-stringify/-/fast-safe-stringify-2.0.7.tgz", - "integrity": "sha512-Utm6CdzT+6xsDk2m8S6uL8VHxNwI6Jub+e9NYTcAms28T84pTa25GJQV9j0CY0N1rM8hK4x6grpF2BQf+2qwVA==", - "dev": true - }, - "fastq": { - "version": "1.9.0", - "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.9.0.tgz", - "integrity": "sha512-i7FVWL8HhVY+CTkwFxkN2mk3h+787ixS5S63eb78diVRc1MCssarHq3W5cj0av7YDSwmaV928RNag+U1etRQ7w==", - "dev": true, - "requires": { - "reusify": "^1.0.4" - } - }, - "fb-watchman": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/fb-watchman/-/fb-watchman-2.0.1.tgz", - "integrity": "sha512-DkPJKQeY6kKwmuMretBhr7G6Vodr7bFwDYTXIkfG1gjvNpaxBTQV3PbXg6bR1c1UP4jPOX0jHUbbHANL9vRjVg==", - "dev": true, - "requires": { - "bser": "2.1.1" - } - }, - "figlet": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/figlet/-/figlet-1.5.0.tgz", - "integrity": "sha512-ZQJM4aifMpz6H19AW1VqvZ7l4pOE9p7i/3LyxgO2kp+PO/VcDYNqIHEMtkccqIhTXMKci4kjueJr/iCQEaT/Ww==" - }, - "file-entry-cache": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-5.0.1.tgz", - "integrity": "sha512-bCg29ictuBaKUwwArK4ouCaqDgLZcysCFLmM/Yn/FDoqndh/9vNuQfXRDvTuXKLxfD/JtZQGKFT8MGcJBK644g==", - "dev": true, - "requires": { - "flat-cache": "^2.0.1" - } - }, - "file-uri-to-path": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz", - "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==", - "dev": true, - "optional": true - }, - "fill-range": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", - "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", - "dev": true, - "requires": { - "extend-shallow": "^2.0.1", - "is-number": "^3.0.0", - "repeat-string": "^1.6.1", - "to-regex-range": "^2.1.0" - }, - "dependencies": { - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - } - } - }, - "finalhandler": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", - "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", - "requires": { - "debug": "2.6.9", - "encodeurl": "~1.0.2", - "escape-html": "~1.0.3", - "on-finished": "~2.3.0", - "parseurl": "~1.3.3", - "statuses": "~1.5.0", - "unpipe": "~1.0.0" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "requires": { - "ms": "2.0.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" - } - } - }, - "find-cache-dir": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-2.1.0.tgz", - "integrity": "sha512-Tq6PixE0w/VMFfCgbONnkiQIVol/JJL7nRMi20fqzA4NRs9AfeqMGeRdPi3wIhYkxjeBaWh2rxwapn5Tu3IqOQ==", - "dev": true, - "requires": { - "commondir": "^1.0.1", - "make-dir": "^2.0.0", - "pkg-dir": "^3.0.0" - } - }, - "find-up": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", - "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", - "requires": { - "locate-path": "^5.0.0", - "path-exists": "^4.0.0" - } - }, - "flat-cache": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-2.0.1.tgz", - "integrity": "sha512-LoQe6yDuUMDzQAEH8sgmh4Md6oZnc/7PjtwjNFSzveXqSHt6ka9fPBuso7IGf9Rz4uqnSnWiFH2B/zj24a5ReA==", - "dev": true, - "requires": { - "flatted": "^2.0.0", - "rimraf": "2.6.3", - "write": "1.0.3" - }, - "dependencies": { - "rimraf": { - "version": "2.6.3", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", - "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - } - } - }, - "flatted": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/flatted/-/flatted-2.0.2.tgz", - "integrity": "sha512-r5wGx7YeOwNWNlCA0wQ86zKyDLMQr+/RB8xy74M4hTphfmjlijTSSXGuH8rnvKZnfT9i+75zmd8jcKdMR4O6jA==", - "dev": true - }, - "for-in": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz", - "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=", - "dev": true - }, - "forever-agent": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", - "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=", - "dev": true - }, - "form-data": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz", - "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==", - "dev": true, - "requires": { - "asynckit": "^0.4.0", - "combined-stream": "^1.0.6", - "mime-types": "^2.1.12" - } - }, - "formidable": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/formidable/-/formidable-1.2.2.tgz", - "integrity": "sha512-V8gLm+41I/8kguQ4/o1D3RIHRmhYFG4pnNyonvua+40rqcEmT4+V71yaZ3B457xbbgCsCfjSPi65u/W6vK1U5Q==", - "dev": true - }, - "forwarded": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", - "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" - }, - "fragment-cache": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/fragment-cache/-/fragment-cache-0.2.1.tgz", - "integrity": "sha1-QpD60n8T6Jvn8zeZxrxaCr//DRk=", - "dev": true, - "requires": { - "map-cache": "^0.2.2" - } - }, - "fresh": { - "version": "0.5.2", - "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", - "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" - }, - "fs-minipass": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-1.2.7.tgz", - "integrity": "sha512-GWSSJGFy4e9GUeCcbIkED+bgAoFyj7XF1mV8rma3QW4NIqX9Kyx79N/PF61H5udOV3aY1IaMLs6pGbH71nlCTA==", - "requires": { - "minipass": "^2.6.0" - } - }, - "fs-readdir-recursive": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/fs-readdir-recursive/-/fs-readdir-recursive-1.1.0.tgz", - "integrity": "sha512-GNanXlVr2pf02+sPN40XN8HG+ePaNcvM0q5mZBd668Obwb0yD5GiUbZOFgwn8kGMY6I3mdyDJzieUy3PTYyTRA==", - "dev": true - }, - "fs.realpath": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", - "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=" - }, - "fsevents": { - "version": "1.2.13", - "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.13.tgz", - "integrity": "sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==", - "dev": true, - "optional": true, - "requires": { - "bindings": "^1.5.0", - "nan": "^2.12.1" - } - }, - "fstream": { - "version": "1.0.12", - "resolved": "https://registry.npmjs.org/fstream/-/fstream-1.0.12.tgz", - "integrity": "sha512-WvJ193OHa0GHPEL+AycEJgxvBEwyfRkN1vhjca23OaPVMCaLCXTd5qAu82AjTcgP1UJmytkOKb63Ypde7raDIg==", - "dev": true, - "optional": true, - "requires": { - "graceful-fs": "^4.1.2", - "inherits": "~2.0.0", - "mkdirp": ">=0.5 0", - "rimraf": "2" - }, - "dependencies": { - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "optional": true, - "requires": { - "glob": "^7.1.3" - } - } - } - }, - "function-bind": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", - "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", - "dev": true - }, - "functional-red-black-tree": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", - "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=", - "dev": true - }, - "gauge": { - "version": "2.7.4", - "resolved": "https://registry.npmjs.org/gauge/-/gauge-2.7.4.tgz", - "integrity": "sha1-LANAXHU4w51+s3sxcCLjJfsBi/c=", - "requires": { - "aproba": "^1.0.3", - "console-control-strings": "^1.0.0", - "has-unicode": "^2.0.0", - "object-assign": "^4.1.0", - "signal-exit": "^3.0.0", - "string-width": "^1.0.1", - "strip-ansi": "^3.0.1", - "wide-align": "^1.1.0" - } - }, - "generate-function": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/generate-function/-/generate-function-2.3.1.tgz", - "integrity": "sha512-eeB5GfMNeevm/GRYq20ShmsaGcmI81kIX2K9XQx5miC8KdHaC6Jm0qQ8ZNeGOi7wYB8OsdxKs+Y2oVuTFuVwKQ==", - "requires": { - "is-property": "^1.0.2" - } - }, - "gensync": { - "version": "1.0.0-beta.2", - "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", - "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", - "dev": true - }, - "get-caller-file": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", - "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==" - }, - "get-intrinsic": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.0.1.tgz", - "integrity": "sha512-ZnWP+AmS1VUaLgTRy47+zKtjTxz+0xMpx3I52i+aalBK1QP19ggLF3Db89KJX7kjfOfP2eoa01qc++GwPgufPg==", - "dev": true, - "requires": { - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1" - } - }, - "get-package-type": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/get-package-type/-/get-package-type-0.1.0.tgz", - "integrity": "sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==", - "dev": true - }, - "get-stdin": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/get-stdin/-/get-stdin-6.0.0.tgz", - "integrity": "sha512-jp4tHawyV7+fkkSKyvjuLZswblUtz+SQKzSWnBbii16BuZksJlU1wuBYXY75r+duh/llF1ur6oNwi+2ZzjKZ7g==", - "dev": true - }, - "get-stream": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", - "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", - "dev": true, - "requires": { - "pump": "^3.0.0" - } - }, - "get-value": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/get-value/-/get-value-2.0.6.tgz", - "integrity": "sha1-3BXKHGcjh8p2vTesCjlbogQqLCg=", - "dev": true - }, - "getpass": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", - "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", - "dev": true, - "requires": { - "assert-plus": "^1.0.0" - } - }, - "glob": { - "version": "7.1.6", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", - "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", - "requires": { - "fs.realpath": "^1.0.0", - "inflight": "^1.0.4", - "inherits": "2", - "minimatch": "^3.0.4", - "once": "^1.3.0", - "path-is-absolute": "^1.0.0" - } - }, - "glob-parent": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", - "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", - "dev": true, - "optional": true, - "requires": { - "is-glob": "^3.1.0", - "path-dirname": "^1.0.0" - }, - "dependencies": { - "is-glob": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", - "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", - "dev": true, - "optional": true, - "requires": { - "is-extglob": "^2.1.0" - } - } - } - }, - "global-dirs": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/global-dirs/-/global-dirs-2.0.1.tgz", - "integrity": "sha512-5HqUqdhkEovj2Of/ms3IeS/EekcO54ytHRLV4PEY2rhRwrHXLQjeVEES0Lhka0xwNDtGYn58wyC4s5+MHsOO6A==", - "dev": true, - "requires": { - "ini": "^1.3.5" - } - }, - "globals": { - "version": "11.12.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", - "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", - "dev": true - }, - "globby": { - "version": "11.0.1", - "resolved": "https://registry.npmjs.org/globby/-/globby-11.0.1.tgz", - "integrity": "sha512-iH9RmgwCmUJHi2z5o2l3eTtGBtXek1OYlHrbcxOYugyHLmAsZrPj43OtHThd62Buh/Vv6VyCBD2bdyWcGNQqoQ==", - "dev": true, - "requires": { - "array-union": "^2.1.0", - "dir-glob": "^3.0.1", - "fast-glob": "^3.1.1", - "ignore": "^5.1.4", - "merge2": "^1.3.0", - "slash": "^3.0.0" - }, - "dependencies": { - "slash": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", - "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", - "dev": true - } - } - }, - "google-libphonenumber": { - "version": "3.2.15", - "resolved": "https://registry.npmjs.org/google-libphonenumber/-/google-libphonenumber-3.2.15.tgz", - "integrity": "sha512-tbCIuzMoH34RdrbFRw5kijAZn/p6JMQvsgtr1glg2ugbwqrMPlOL8pHNK8cyGo9B6SXpcMm4hdyDqwomR+HPRg==" - }, - "got": { - "version": "9.6.0", - "resolved": "https://registry.npmjs.org/got/-/got-9.6.0.tgz", - "integrity": "sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q==", - "dev": true, - "requires": { - "@sindresorhus/is": "^0.14.0", - "@szmarczak/http-timer": "^1.1.2", - "cacheable-request": "^6.0.0", - "decompress-response": "^3.3.0", - "duplexer3": "^0.1.4", - "get-stream": "^4.1.0", - "lowercase-keys": "^1.0.1", - "mimic-response": "^1.0.1", - "p-cancelable": "^1.0.0", - "to-readable-stream": "^1.0.0", - "url-parse-lax": "^3.0.0" - } - }, - "graceful-fs": { - "version": "4.2.4", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz", - "integrity": "sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==", - "dev": true - }, - "growly": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/growly/-/growly-1.3.0.tgz", - "integrity": "sha1-8QdIy+dq+WS3yWyTxrzCivEgwIE=", - "dev": true, - "optional": true - }, - "har-schema": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz", - "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=", - "dev": true - }, - "har-validator": { - "version": "5.1.5", - "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.5.tgz", - "integrity": "sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w==", - "dev": true, - "requires": { - "ajv": "^6.12.3", - "har-schema": "^2.0.0" - } - }, - "has": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", - "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", - "dev": true, - "requires": { - "function-bind": "^1.1.1" - } - }, - "has-ansi": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", - "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=", - "requires": { - "ansi-regex": "^2.0.0" - } - }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==" - }, - "has-symbols": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.1.tgz", - "integrity": "sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg==", - "dev": true - }, - "has-unicode": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/has-unicode/-/has-unicode-2.0.1.tgz", - "integrity": "sha1-4Ob+aijPUROIVeCG0Wkedx3iqLk=" - }, - "has-value": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/has-value/-/has-value-1.0.0.tgz", - "integrity": "sha1-GLKB2lhbHFxR3vJMkw7SmgvmsXc=", - "dev": true, - "requires": { - "get-value": "^2.0.6", - "has-values": "^1.0.0", - "isobject": "^3.0.0" - } - }, - "has-values": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/has-values/-/has-values-1.0.0.tgz", - "integrity": "sha1-lbC2P+whRmGab+V/51Yo1aOe/k8=", - "dev": true, - "requires": { - "is-number": "^3.0.0", - "kind-of": "^4.0.0" - }, - "dependencies": { - "kind-of": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-4.0.0.tgz", - "integrity": "sha1-IIE989cSkosgc3hpGkUGb65y3Vc=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } - } - }, - "has-yarn": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/has-yarn/-/has-yarn-2.1.0.tgz", - "integrity": "sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw==", - "dev": true - }, - "highlight.js": { - "version": "9.18.4", - "resolved": "https://registry.npmjs.org/highlight.js/-/highlight.js-9.18.4.tgz", - "integrity": "sha512-Bb4K6pAKRr3WCRI4dZaak0lPgerK/bMDuaDhKpomWdeKnpcunMtSWi4KFqeoLrHlITf88TISu2ZX7dZe6fSJfQ==" - }, - "homedir-polyfill": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/homedir-polyfill/-/homedir-polyfill-1.0.3.tgz", - "integrity": "sha512-eSmmWE5bZTK2Nou4g0AI3zZ9rswp7GRKoKXS1BLUkvPviOqs4YTN1djQIqrXy9k5gEtdLPy86JjRwsNM9tnDcA==", - "dev": true, - "requires": { - "parse-passwd": "^1.0.0" - } - }, - "hosted-git-info": { - "version": "2.8.8", - "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.8.tgz", - "integrity": "sha512-f/wzC2QaWBs7t9IYqB4T3sR1xviIViXJRJTWBlx2Gf3g0Xi5vI7Yy4koXQ1c9OYDGHN9sBy1DQ2AB8fqZBWhUg==", - "dev": true - }, - "html-encoding-sniffer": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-2.0.1.tgz", - "integrity": "sha512-D5JbOMBIR/TVZkubHT+OyT2705QvogUW4IBn6nHd756OwieSF9aDYFj4dv6HHEVGYbHaLETa3WggZYWWMyy3ZQ==", - "dev": true, - "requires": { - "whatwg-encoding": "^1.0.5" - } - }, - "html-escaper": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", - "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", - "dev": true - }, - "http-cache-semantics": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz", - "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==", - "dev": true - }, - "http-errors": { - "version": "1.7.2", - "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", - "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", - "requires": { - "depd": "~1.1.2", - "inherits": "2.0.3", - "setprototypeof": "1.1.1", - "statuses": ">= 1.5.0 < 2", - "toidentifier": "1.0.0" - }, - "dependencies": { - "inherits": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", - "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" - } - } - }, - "http-signature": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", - "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", - "dev": true, - "requires": { - "assert-plus": "^1.0.0", - "jsprim": "^1.2.2", - "sshpk": "^1.7.0" - } - }, - "human-signals": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-1.1.1.tgz", - "integrity": "sha512-SEQu7vl8KjNL2eoGBLF3+wAjpsNfA9XMlXAYj/3EdaNfAlxKthD1xjEQfGOUhllCGGJVNY34bRr6lPINhNjyZw==", - "dev": true - }, - "iconv-lite": { - "version": "0.4.24", - "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", - "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", - "requires": { - "safer-buffer": ">= 2.1.2 < 3" - } - }, - "ieee754": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", - "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==" - }, - "ignore": { - "version": "5.1.8", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.1.8.tgz", - "integrity": "sha512-BMpfD7PpiETpBl/A6S498BaIJ6Y/ABT93ETbby2fP00v4EbvPBXWEoaR1UBPKs3iR53pJY7EtZk5KACI57i1Uw==", - "dev": true - }, - "ignore-by-default": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/ignore-by-default/-/ignore-by-default-1.0.1.tgz", - "integrity": "sha1-SMptcvbGo68Aqa1K5odr44ieKwk=", - "dev": true - }, - "ignore-walk": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/ignore-walk/-/ignore-walk-3.0.3.tgz", - "integrity": "sha512-m7o6xuOaT1aqheYHKf8W6J5pYH85ZI9w077erOzLje3JsB1gkafkAhHHY19dqjulgIZHFm32Cp5uNZgcQqdJKw==", - "requires": { - "minimatch": "^3.0.4" - } - }, - "import-fresh": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.2.2.tgz", - "integrity": "sha512-cTPNrlvJT6twpYy+YmKUKrTSjWFs3bjYjAhCwm+z4EOCubZxAuO+hHpRN64TqjEaYSHs7tJAE0w1CKMGmsG/lw==", - "dev": true, - "requires": { - "parent-module": "^1.0.0", - "resolve-from": "^4.0.0" - }, - "dependencies": { - "resolve-from": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", - "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", - "dev": true - } - } - }, - "import-lazy": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/import-lazy/-/import-lazy-2.1.0.tgz", - "integrity": "sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM=", - "dev": true - }, - "import-local": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.0.2.tgz", - "integrity": "sha512-vjL3+w0oulAVZ0hBHnxa/Nm5TAurf9YLQJDhqRZyqb+VKGOB6LU8t9H1Nr5CIo16vh9XfJTOoHwU0B71S557gA==", - "dev": true, - "requires": { - "pkg-dir": "^4.2.0", - "resolve-cwd": "^3.0.0" - }, - "dependencies": { - "pkg-dir": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", - "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", - "dev": true, - "requires": { - "find-up": "^4.0.0" - } - } - } - }, - "imurmurhash": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", - "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", - "dev": true - }, - "inflight": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", - "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", - "requires": { - "once": "^1.3.0", - "wrappy": "1" - } - }, - "inherits": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", - "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" - }, - "ini": { - "version": "1.3.5", - "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz", - "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw==" - }, - "ip-regex": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/ip-regex/-/ip-regex-2.1.0.tgz", - "integrity": "sha1-+ni/XS5pE8kRzp+BnuUUa7bYROk=", - "dev": true - }, - "ipaddr.js": { - "version": "1.9.1", - "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", - "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" - }, - "is-accessor-descriptor": { - "version": "0.1.6", - "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz", - "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=", - "dev": true, - "requires": { - "kind-of": "^3.0.2" - }, - "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } - } - }, - "is-arrayish": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", - "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=", - "dev": true - }, - "is-binary-path": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-1.0.1.tgz", - "integrity": "sha1-dfFmQrSA8YenEcgUFh/TpKdlWJg=", - "dev": true, - "optional": true, - "requires": { - "binary-extensions": "^1.0.0" - } - }, - "is-buffer": { - "version": "1.1.6", - "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", - "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==", - "dev": true - }, - "is-callable": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.2.tgz", - "integrity": "sha512-dnMqspv5nU3LoewK2N/y7KLtxtakvTuaCsU9FU50/QDmdbHNy/4/JuRtMHqRU22o3q+W89YQndQEeCVwK+3qrA==", - "dev": true - }, - "is-ci": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-ci/-/is-ci-2.0.0.tgz", - "integrity": "sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w==", - "dev": true, - "requires": { - "ci-info": "^2.0.0" - } - }, - "is-core-module": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.1.0.tgz", - "integrity": "sha512-YcV7BgVMRFRua2FqQzKtTDMz8iCuLEyGKjr70q8Zm1yy2qKcurbFEd79PAdHV77oL3NrAaOVQIbMmiHQCHB7ZA==", - "dev": true, - "requires": { - "has": "^1.0.3" - } - }, - "is-data-descriptor": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", - "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", - "dev": true, - "requires": { - "kind-of": "^3.0.2" - }, - "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } - } - }, - "is-date-object": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.2.tgz", - "integrity": "sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g==", - "dev": true - }, - "is-descriptor": { - "version": "0.1.6", - "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz", - "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", - "dev": true, - "requires": { - "is-accessor-descriptor": "^0.1.6", - "is-data-descriptor": "^0.1.4", - "kind-of": "^5.0.0" - }, - "dependencies": { - "kind-of": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", - "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", - "dev": true - } - } - }, - "is-docker": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-2.1.1.tgz", - "integrity": "sha512-ZOoqiXfEwtGknTiuDEy8pN2CfE3TxMHprvNer1mXiqwkOT77Rw3YVrUQ52EqAOU3QAWDQ+bQdx7HJzrv7LS2Hw==", - "dev": true, - "optional": true - }, - "is-extendable": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", - "integrity": "sha1-YrEQ4omkcUGOPsNqYX1HLjAd/Ik=", - "dev": true - }, - "is-extglob": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", - "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", - "dev": true - }, - "is-fullwidth-code-point": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", - "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", - "requires": { - "number-is-nan": "^1.0.0" - } - }, - "is-generator-fn": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/is-generator-fn/-/is-generator-fn-2.1.0.tgz", - "integrity": "sha512-cTIB4yPYL/Grw0EaSzASzg6bBy9gqCofvWN8okThAYIxKJZC+udlRAmGbM0XLeniEJSs8uEgHPGuHSe1XsOLSQ==", - "dev": true - }, - "is-glob": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", - "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", - "dev": true, - "requires": { - "is-extglob": "^2.1.1" - } - }, - "is-installed-globally": { - "version": "0.3.2", - "resolved": "https://registry.npmjs.org/is-installed-globally/-/is-installed-globally-0.3.2.tgz", - "integrity": "sha512-wZ8x1js7Ia0kecP/CHM/3ABkAmujX7WPvQk6uu3Fly/Mk44pySulQpnHG46OMjHGXApINnV4QhY3SWnECO2z5g==", - "dev": true, - "requires": { - "global-dirs": "^2.0.1", - "is-path-inside": "^3.0.1" - } - }, - "is-negative-zero": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.0.tgz", - "integrity": "sha1-lVOxIbD6wohp2p7UWeIMdUN4hGE=", - "dev": true - }, - "is-npm": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/is-npm/-/is-npm-4.0.0.tgz", - "integrity": "sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig==", - "dev": true - }, - "is-number": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", - "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", - "dev": true, - "requires": { - "kind-of": "^3.0.2" - }, - "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } - } - }, - "is-obj": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz", - "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==", - "dev": true - }, - "is-path-inside": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-3.0.2.tgz", - "integrity": "sha512-/2UGPSgmtqwo1ktx8NDHjuPwZWmHhO+gj0f93EkhLB5RgW9RZevWYYlIkS6zePc6U2WpOdQYIwHe9YC4DWEBVg==", - "dev": true - }, - "is-plain-object": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz", - "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==", - "dev": true, - "requires": { - "isobject": "^3.0.1" - } - }, - "is-potential-custom-element-name": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-potential-custom-element-name/-/is-potential-custom-element-name-1.0.0.tgz", - "integrity": "sha1-DFLlS8yjkbssSUsh6GJtczbG45c=", - "dev": true - }, - "is-property": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-property/-/is-property-1.0.2.tgz", - "integrity": "sha1-V/4cTkhHTt1lsJkR8msc1Ald2oQ=" - }, - "is-regex": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.1.tgz", - "integrity": "sha512-1+QkEcxiLlB7VEyFtyBg94e08OAsvq7FUBgApTq/w2ymCLyKJgDPsybBENVtA7XCQEgEXxKPonG+mvYRxh/LIg==", - "dev": true, - "requires": { - "has-symbols": "^1.0.1" - } - }, - "is-stream": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz", - "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=", - "dev": true - }, - "is-string": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.5.tgz", - "integrity": "sha512-buY6VNRjhQMiF1qWDouloZlQbRhDPCebwxSjxMjxgemYT46YMd2NR0/H+fBhEfWX4A/w9TBJ+ol+okqJKFE6vQ==", - "dev": true - }, - "is-symbol": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.3.tgz", - "integrity": "sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==", - "dev": true, - "requires": { - "has-symbols": "^1.0.1" - } - }, - "is-typedarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", - "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", - "dev": true - }, - "is-windows": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", - "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", - "dev": true - }, - "is-wsl": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-2.2.0.tgz", - "integrity": "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==", - "dev": true, - "optional": true, - "requires": { - "is-docker": "^2.0.0" - } - }, - "is-yarn-global": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz", - "integrity": "sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw==", - "dev": true - }, - "isarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", - "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=" - }, - "isexe": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", - "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", - "dev": true - }, - "isobject": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", - "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", - "dev": true - }, - "isstream": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", - "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=", - "dev": true - }, - "istanbul-lib-coverage": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.0.0.tgz", - "integrity": "sha512-UiUIqxMgRDET6eR+o5HbfRYP1l0hqkWOs7vNxC/mggutCMUIhWMm8gAHb8tHlyfD3/l6rlgNA5cKdDzEAf6hEg==", - "dev": true - }, - "istanbul-lib-instrument": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-4.0.3.tgz", - "integrity": "sha512-BXgQl9kf4WTCPCCpmFGoJkz/+uhvm7h7PFKUYxh7qarQd3ER33vHG//qaE8eN25l07YqZPpHXU9I09l/RD5aGQ==", - "dev": true, - "requires": { - "@babel/core": "^7.7.5", - "@istanbuljs/schema": "^0.1.2", - "istanbul-lib-coverage": "^3.0.0", - "semver": "^6.3.0" - }, - "dependencies": { - "semver": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", - "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", - "dev": true - } - } - }, - "istanbul-lib-report": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz", - "integrity": "sha512-wcdi+uAKzfiGT2abPpKZ0hSU1rGQjUQnLvtY5MpQ7QCTahD3VODhcu4wcfY1YtkGaDD5yuydOLINXsfbus9ROw==", - "dev": true, - "requires": { - "istanbul-lib-coverage": "^3.0.0", - "make-dir": "^3.0.0", - "supports-color": "^7.1.0" - }, - "dependencies": { - "make-dir": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", - "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", - "dev": true, - "requires": { - "semver": "^6.0.0" - } - }, - "semver": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", - "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", - "dev": true - } - } - }, - "istanbul-lib-source-maps": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.0.tgz", - "integrity": "sha512-c16LpFRkR8vQXyHZ5nLpY35JZtzj1PQY1iZmesUbf1FZHbIupcWfjgOXBY9YHkLEQ6puz1u4Dgj6qmU/DisrZg==", - "dev": true, - "requires": { - "debug": "^4.1.1", - "istanbul-lib-coverage": "^3.0.0", - "source-map": "^0.6.1" - }, - "dependencies": { - "debug": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", - "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", - "dev": true, - "requires": { - "ms": "2.1.2" - } - }, - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } - } - }, - "istanbul-reports": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.0.2.tgz", - "integrity": "sha512-9tZvz7AiR3PEDNGiV9vIouQ/EAcqMXFmkcA1CDFTwOB98OZVDL0PH9glHotf5Ugp6GCOTypfzGWI/OqjWNCRUw==", - "dev": true, - "requires": { - "html-escaper": "^2.0.0", - "istanbul-lib-report": "^3.0.0" - } - }, - "jest": { - "version": "26.6.3", - "resolved": "https://registry.npmjs.org/jest/-/jest-26.6.3.tgz", - "integrity": "sha512-lGS5PXGAzR4RF7V5+XObhqz2KZIDUA1yD0DG6pBVmy10eh0ZIXQImRuzocsI/N2XZ1GrLFwTS27In2i2jlpq1Q==", - "dev": true, - "requires": { - "@jest/core": "^26.6.3", - "import-local": "^3.0.2", - "jest-cli": "^26.6.3" - }, - "dependencies": { - "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", - "dev": true - }, - "is-fullwidth-code-point": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", - "dev": true - }, - "jest-cli": { - "version": "26.6.3", - "resolved": "https://registry.npmjs.org/jest-cli/-/jest-cli-26.6.3.tgz", - "integrity": "sha512-GF9noBSa9t08pSyl3CY4frMrqp+aQXFGFkf5hEPbh/pIUFYWMK6ZLTfbmadxJVcJrdRoChlWQsA2VkJcDFK8hg==", - "dev": true, - "requires": { - "@jest/core": "^26.6.3", - "@jest/test-result": "^26.6.2", - "@jest/types": "^26.6.2", - "chalk": "^4.0.0", - "exit": "^0.1.2", - "graceful-fs": "^4.2.4", - "import-local": "^3.0.2", - "is-ci": "^2.0.0", - "jest-config": "^26.6.3", - "jest-util": "^26.6.2", - "jest-validate": "^26.6.2", - "prompts": "^2.0.1", - "yargs": "^15.4.1" - } - }, - "string-width": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", - "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", - "dev": true, - "requires": { - "emoji-regex": "^8.0.0", - "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.0" - } - }, - "strip-ansi": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", - "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", - "dev": true, - "requires": { - "ansi-regex": "^5.0.0" - } - }, - "yargs": { - "version": "15.4.1", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.4.1.tgz", - "integrity": "sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A==", - "dev": true, - "requires": { - "cliui": "^6.0.0", - "decamelize": "^1.2.0", - "find-up": "^4.1.0", - "get-caller-file": "^2.0.1", - "require-directory": "^2.1.1", - "require-main-filename": "^2.0.0", - "set-blocking": "^2.0.0", - "string-width": "^4.2.0", - "which-module": "^2.0.0", - "y18n": "^4.0.0", - "yargs-parser": "^18.1.2" - } - } - } - }, - "jest-changed-files": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-changed-files/-/jest-changed-files-26.6.2.tgz", - "integrity": "sha512-fDS7szLcY9sCtIip8Fjry9oGf3I2ht/QT21bAHm5Dmf0mD4X3ReNUf17y+bO6fR8WgbIZTlbyG1ak/53cbRzKQ==", - "dev": true, - "requires": { - "@jest/types": "^26.6.2", - "execa": "^4.0.0", - "throat": "^5.0.0" - }, - "dependencies": { - "cross-spawn": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", - "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", - "dev": true, - "requires": { - "path-key": "^3.1.0", - "shebang-command": "^2.0.0", - "which": "^2.0.1" - } - }, - "execa": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/execa/-/execa-4.1.0.tgz", - "integrity": "sha512-j5W0//W7f8UxAn8hXVnwG8tLwdiUy4FJLcSupCg6maBYZDpyBvTApK7KyuI4bKj8KOh1r2YH+6ucuYtJv1bTZA==", - "dev": true, - "requires": { - "cross-spawn": "^7.0.0", - "get-stream": "^5.0.0", - "human-signals": "^1.1.1", - "is-stream": "^2.0.0", - "merge-stream": "^2.0.0", - "npm-run-path": "^4.0.0", - "onetime": "^5.1.0", - "signal-exit": "^3.0.2", - "strip-final-newline": "^2.0.0" - } - }, - "get-stream": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", - "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", - "dev": true, - "requires": { - "pump": "^3.0.0" - } - }, - "is-stream": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.0.tgz", - "integrity": "sha512-XCoy+WlUr7d1+Z8GgSuXmpuUFC9fOhRXglJMx+dwLKTkL44Cjd4W1Z5P+BQZpr+cR93aGP4S/s7Ftw6Nd/kiEw==", - "dev": true - }, - "npm-run-path": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-4.0.1.tgz", - "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==", - "dev": true, - "requires": { - "path-key": "^3.0.0" - } - }, - "path-key": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", - "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", - "dev": true - }, - "shebang-command": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", - "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", - "dev": true, - "requires": { - "shebang-regex": "^3.0.0" - } - }, - "shebang-regex": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", - "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", - "dev": true - }, - "which": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", - "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", - "dev": true, - "requires": { - "isexe": "^2.0.0" - } - } - } - }, - "jest-config": { - "version": "26.6.3", - "resolved": "https://registry.npmjs.org/jest-config/-/jest-config-26.6.3.tgz", - "integrity": "sha512-t5qdIj/bCj2j7NFVHb2nFB4aUdfucDn3JRKgrZnplb8nieAirAzRSHP8uDEd+qV6ygzg9Pz4YG7UTJf94LPSyg==", - "dev": true, - "requires": { - "@babel/core": "^7.1.0", - "@jest/test-sequencer": "^26.6.3", - "@jest/types": "^26.6.2", - "babel-jest": "^26.6.3", - "chalk": "^4.0.0", - "deepmerge": "^4.2.2", - "glob": "^7.1.1", - "graceful-fs": "^4.2.4", - "jest-environment-jsdom": "^26.6.2", - "jest-environment-node": "^26.6.2", - "jest-get-type": "^26.3.0", - "jest-jasmine2": "^26.6.3", - "jest-regex-util": "^26.0.0", - "jest-resolve": "^26.6.2", - "jest-util": "^26.6.2", - "jest-validate": "^26.6.2", - "micromatch": "^4.0.2", - "pretty-format": "^26.6.2" - }, - "dependencies": { - "braces": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", - "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", - "dev": true, - "requires": { - "fill-range": "^7.0.1" - } - }, - "fill-range": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", - "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", - "dev": true, - "requires": { - "to-regex-range": "^5.0.1" - } - }, - "is-number": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "dev": true - }, - "micromatch": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", - "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", - "dev": true, - "requires": { - "braces": "^3.0.1", - "picomatch": "^2.0.5" - } - }, - "to-regex-range": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", - "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "dev": true, - "requires": { - "is-number": "^7.0.0" - } - } - } - }, - "jest-diff": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-26.6.2.tgz", - "integrity": "sha512-6m+9Z3Gv9wN0WFVasqjCL/06+EFCMTqDEUl/b87HYK2rAPTyfz4ZIuSlPhY51PIQRWx5TaxeF1qmXKe9gfN3sA==", - "dev": true, - "requires": { - "chalk": "^4.0.0", - "diff-sequences": "^26.6.2", - "jest-get-type": "^26.3.0", - "pretty-format": "^26.6.2" - } - }, - "jest-docblock": { - "version": "26.0.0", - "resolved": "https://registry.npmjs.org/jest-docblock/-/jest-docblock-26.0.0.tgz", - "integrity": "sha512-RDZ4Iz3QbtRWycd8bUEPxQsTlYazfYn/h5R65Fc6gOfwozFhoImx+affzky/FFBuqISPTqjXomoIGJVKBWoo0w==", - "dev": true, - "requires": { - "detect-newline": "^3.0.0" - } - }, - "jest-each": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-each/-/jest-each-26.6.2.tgz", - "integrity": "sha512-Mer/f0KaATbjl8MCJ+0GEpNdqmnVmDYqCTJYTvoo7rqmRiDllmp2AYN+06F93nXcY3ur9ShIjS+CO/uD+BbH4A==", - "dev": true, - "requires": { - "@jest/types": "^26.6.2", - "chalk": "^4.0.0", - "jest-get-type": "^26.3.0", - "jest-util": "^26.6.2", - "pretty-format": "^26.6.2" - } - }, - "jest-environment-jsdom": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-environment-jsdom/-/jest-environment-jsdom-26.6.2.tgz", - "integrity": "sha512-jgPqCruTlt3Kwqg5/WVFyHIOJHsiAvhcp2qiR2QQstuG9yWox5+iHpU3ZrcBxW14T4fe5Z68jAfLRh7joCSP2Q==", - "dev": true, - "requires": { - "@jest/environment": "^26.6.2", - "@jest/fake-timers": "^26.6.2", - "@jest/types": "^26.6.2", - "@types/node": "*", - "jest-mock": "^26.6.2", - "jest-util": "^26.6.2", - "jsdom": "^16.4.0" - } - }, - "jest-environment-node": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-environment-node/-/jest-environment-node-26.6.2.tgz", - "integrity": "sha512-zhtMio3Exty18dy8ee8eJ9kjnRyZC1N4C1Nt/VShN1apyXc8rWGtJ9lI7vqiWcyyXS4BVSEn9lxAM2D+07/Tag==", - "dev": true, - "requires": { - "@jest/environment": "^26.6.2", - "@jest/fake-timers": "^26.6.2", - "@jest/types": "^26.6.2", - "@types/node": "*", - "jest-mock": "^26.6.2", - "jest-util": "^26.6.2" - } - }, - "jest-get-type": { - "version": "26.3.0", - "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-26.3.0.tgz", - "integrity": "sha512-TpfaviN1R2pQWkIihlfEanwOXK0zcxrKEE4MlU6Tn7keoXdN6/3gK/xl0yEh8DOunn5pOVGKf8hB4R9gVh04ig==", - "dev": true - }, - "jest-haste-map": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-26.6.2.tgz", - "integrity": "sha512-easWIJXIw71B2RdR8kgqpjQrbMRWQBgiBwXYEhtGUTaX+doCjBheluShdDMeR8IMfJiTqH4+zfhtg29apJf/8w==", - "dev": true, - "requires": { - "@jest/types": "^26.6.2", - "@types/graceful-fs": "^4.1.2", - "@types/node": "*", - "anymatch": "^3.0.3", - "fb-watchman": "^2.0.0", - "fsevents": "^2.1.2", - "graceful-fs": "^4.2.4", - "jest-regex-util": "^26.0.0", - "jest-serializer": "^26.6.2", - "jest-util": "^26.6.2", - "jest-worker": "^26.6.2", - "micromatch": "^4.0.2", - "sane": "^4.0.3", - "walker": "^1.0.7" - }, - "dependencies": { - "anymatch": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", - "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", - "dev": true, - "requires": { - "normalize-path": "^3.0.0", - "picomatch": "^2.0.4" - } - }, - "braces": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", - "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", - "dev": true, - "requires": { - "fill-range": "^7.0.1" - } - }, - "fill-range": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", - "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", - "dev": true, - "requires": { - "to-regex-range": "^5.0.1" - } - }, - "fsevents": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.2.1.tgz", - "integrity": "sha512-bTLYHSeC0UH/EFXS9KqWnXuOl/wHK5Z/d+ghd5AsFMYN7wIGkUCOJyzy88+wJKkZPGON8u4Z9f6U4FdgURE9qA==", - "dev": true, - "optional": true - }, - "is-number": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "dev": true - }, - "micromatch": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", - "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", - "dev": true, - "requires": { - "braces": "^3.0.1", - "picomatch": "^2.0.5" - } - }, - "to-regex-range": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", - "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "dev": true, - "requires": { - "is-number": "^7.0.0" - } - } - } - }, - "jest-jasmine2": { - "version": "26.6.3", - "resolved": "https://registry.npmjs.org/jest-jasmine2/-/jest-jasmine2-26.6.3.tgz", - "integrity": "sha512-kPKUrQtc8aYwBV7CqBg5pu+tmYXlvFlSFYn18ev4gPFtrRzB15N2gW/Roew3187q2w2eHuu0MU9TJz6w0/nPEg==", - "dev": true, - "requires": { - "@babel/traverse": "^7.1.0", - "@jest/environment": "^26.6.2", - "@jest/source-map": "^26.6.2", - "@jest/test-result": "^26.6.2", - "@jest/types": "^26.6.2", - "@types/node": "*", - "chalk": "^4.0.0", - "co": "^4.6.0", - "expect": "^26.6.2", - "is-generator-fn": "^2.0.0", - "jest-each": "^26.6.2", - "jest-matcher-utils": "^26.6.2", - "jest-message-util": "^26.6.2", - "jest-runtime": "^26.6.3", - "jest-snapshot": "^26.6.2", - "jest-util": "^26.6.2", - "pretty-format": "^26.6.2", - "throat": "^5.0.0" - } - }, - "jest-leak-detector": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-leak-detector/-/jest-leak-detector-26.6.2.tgz", - "integrity": "sha512-i4xlXpsVSMeKvg2cEKdfhh0H39qlJlP5Ex1yQxwF9ubahboQYMgTtz5oML35AVA3B4Eu+YsmwaiKVev9KCvLxg==", - "dev": true, - "requires": { - "jest-get-type": "^26.3.0", - "pretty-format": "^26.6.2" - } - }, - "jest-matcher-utils": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-26.6.2.tgz", - "integrity": "sha512-llnc8vQgYcNqDrqRDXWwMr9i7rS5XFiCwvh6DTP7Jqa2mqpcCBBlpCbn+trkG0KNhPu/h8rzyBkriOtBstvWhw==", - "dev": true, - "requires": { - "chalk": "^4.0.0", - "jest-diff": "^26.6.2", - "jest-get-type": "^26.3.0", - "pretty-format": "^26.6.2" - } - }, - "jest-message-util": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-26.6.2.tgz", - "integrity": "sha512-rGiLePzQ3AzwUshu2+Rn+UMFk0pHN58sOG+IaJbk5Jxuqo3NYO1U2/MIR4S1sKgsoYSXSzdtSa0TgrmtUwEbmA==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.0.0", - "@jest/types": "^26.6.2", - "@types/stack-utils": "^2.0.0", - "chalk": "^4.0.0", - "graceful-fs": "^4.2.4", - "micromatch": "^4.0.2", - "pretty-format": "^26.6.2", - "slash": "^3.0.0", - "stack-utils": "^2.0.2" - }, - "dependencies": { - "braces": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", - "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", - "dev": true, - "requires": { - "fill-range": "^7.0.1" - } - }, - "fill-range": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", - "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", - "dev": true, - "requires": { - "to-regex-range": "^5.0.1" - } - }, - "is-number": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "dev": true - }, - "micromatch": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", - "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", - "dev": true, - "requires": { - "braces": "^3.0.1", - "picomatch": "^2.0.5" - } - }, - "slash": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", - "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", - "dev": true - }, - "to-regex-range": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", - "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "dev": true, - "requires": { - "is-number": "^7.0.0" - } - } - } - }, - "jest-mock": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-mock/-/jest-mock-26.6.2.tgz", - "integrity": "sha512-YyFjePHHp1LzpzYcmgqkJ0nm0gg/lJx2aZFzFy1S6eUqNjXsOqTK10zNRff2dNfssgokjkG65OlWNcIlgd3zew==", - "dev": true, - "requires": { - "@jest/types": "^26.6.2", - "@types/node": "*" - } - }, - "jest-pnp-resolver": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/jest-pnp-resolver/-/jest-pnp-resolver-1.2.2.tgz", - "integrity": "sha512-olV41bKSMm8BdnuMsewT4jqlZ8+3TCARAXjZGT9jcoSnrfUnRCqnMoF9XEeoWjbzObpqF9dRhHQj0Xb9QdF6/w==", - "dev": true - }, - "jest-regex-util": { - "version": "26.0.0", - "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-26.0.0.tgz", - "integrity": "sha512-Gv3ZIs/nA48/Zvjrl34bf+oD76JHiGDUxNOVgUjh3j890sblXryjY4rss71fPtD/njchl6PSE2hIhvyWa1eT0A==", - "dev": true - }, - "jest-resolve": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-resolve/-/jest-resolve-26.6.2.tgz", - "integrity": "sha512-sOxsZOq25mT1wRsfHcbtkInS+Ek7Q8jCHUB0ZUTP0tc/c41QHriU/NunqMfCUWsL4H3MHpvQD4QR9kSYhS7UvQ==", - "dev": true, - "requires": { - "@jest/types": "^26.6.2", - "chalk": "^4.0.0", - "graceful-fs": "^4.2.4", - "jest-pnp-resolver": "^1.2.2", - "jest-util": "^26.6.2", - "read-pkg-up": "^7.0.1", - "resolve": "^1.18.1", - "slash": "^3.0.0" - }, - "dependencies": { - "slash": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", - "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", - "dev": true - } - } - }, - "jest-resolve-dependencies": { - "version": "26.6.3", - "resolved": "https://registry.npmjs.org/jest-resolve-dependencies/-/jest-resolve-dependencies-26.6.3.tgz", - "integrity": "sha512-pVwUjJkxbhe4RY8QEWzN3vns2kqyuldKpxlxJlzEYfKSvY6/bMvxoFrYYzUO1Gx28yKWN37qyV7rIoIp2h8fTg==", - "dev": true, - "requires": { - "@jest/types": "^26.6.2", - "jest-regex-util": "^26.0.0", - "jest-snapshot": "^26.6.2" - } - }, - "jest-runner": { - "version": "26.6.3", - "resolved": "https://registry.npmjs.org/jest-runner/-/jest-runner-26.6.3.tgz", - "integrity": "sha512-atgKpRHnaA2OvByG/HpGA4g6CSPS/1LK0jK3gATJAoptC1ojltpmVlYC3TYgdmGp+GLuhzpH30Gvs36szSL2JQ==", - "dev": true, - "requires": { - "@jest/console": "^26.6.2", - "@jest/environment": "^26.6.2", - "@jest/test-result": "^26.6.2", - "@jest/types": "^26.6.2", - "@types/node": "*", - "chalk": "^4.0.0", - "emittery": "^0.7.1", - "exit": "^0.1.2", - "graceful-fs": "^4.2.4", - "jest-config": "^26.6.3", - "jest-docblock": "^26.0.0", - "jest-haste-map": "^26.6.2", - "jest-leak-detector": "^26.6.2", - "jest-message-util": "^26.6.2", - "jest-resolve": "^26.6.2", - "jest-runtime": "^26.6.3", - "jest-util": "^26.6.2", - "jest-worker": "^26.6.2", - "source-map-support": "^0.5.6", - "throat": "^5.0.0" - } - }, - "jest-runtime": { - "version": "26.6.3", - "resolved": "https://registry.npmjs.org/jest-runtime/-/jest-runtime-26.6.3.tgz", - "integrity": "sha512-lrzyR3N8sacTAMeonbqpnSka1dHNux2uk0qqDXVkMv2c/A3wYnvQ4EXuI013Y6+gSKSCxdaczvf4HF0mVXHRdw==", - "dev": true, - "requires": { - "@jest/console": "^26.6.2", - "@jest/environment": "^26.6.2", - "@jest/fake-timers": "^26.6.2", - "@jest/globals": "^26.6.2", - "@jest/source-map": "^26.6.2", - "@jest/test-result": "^26.6.2", - "@jest/transform": "^26.6.2", - "@jest/types": "^26.6.2", - "@types/yargs": "^15.0.0", - "chalk": "^4.0.0", - "cjs-module-lexer": "^0.6.0", - "collect-v8-coverage": "^1.0.0", - "exit": "^0.1.2", - "glob": "^7.1.3", - "graceful-fs": "^4.2.4", - "jest-config": "^26.6.3", - "jest-haste-map": "^26.6.2", - "jest-message-util": "^26.6.2", - "jest-mock": "^26.6.2", - "jest-regex-util": "^26.0.0", - "jest-resolve": "^26.6.2", - "jest-snapshot": "^26.6.2", - "jest-util": "^26.6.2", - "jest-validate": "^26.6.2", - "slash": "^3.0.0", - "strip-bom": "^4.0.0", - "yargs": "^15.4.1" - }, - "dependencies": { - "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", - "dev": true - }, - "is-fullwidth-code-point": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", - "dev": true - }, - "slash": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", - "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", - "dev": true - }, - "string-width": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", - "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", - "dev": true, - "requires": { - "emoji-regex": "^8.0.0", - "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.0" - } - }, - "strip-ansi": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", - "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", - "dev": true, - "requires": { - "ansi-regex": "^5.0.0" - } - }, - "yargs": { - "version": "15.4.1", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.4.1.tgz", - "integrity": "sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A==", - "dev": true, - "requires": { - "cliui": "^6.0.0", - "decamelize": "^1.2.0", - "find-up": "^4.1.0", - "get-caller-file": "^2.0.1", - "require-directory": "^2.1.1", - "require-main-filename": "^2.0.0", - "set-blocking": "^2.0.0", - "string-width": "^4.2.0", - "which-module": "^2.0.0", - "y18n": "^4.0.0", - "yargs-parser": "^18.1.2" - } - } - } - }, - "jest-serializer": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-serializer/-/jest-serializer-26.6.2.tgz", - "integrity": "sha512-S5wqyz0DXnNJPd/xfIzZ5Xnp1HrJWBczg8mMfMpN78OJ5eDxXyf+Ygld9wX1DnUWbIbhM1YDY95NjR4CBXkb2g==", - "dev": true, - "requires": { - "@types/node": "*", - "graceful-fs": "^4.2.4" - } - }, - "jest-snapshot": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-snapshot/-/jest-snapshot-26.6.2.tgz", - "integrity": "sha512-OLhxz05EzUtsAmOMzuupt1lHYXCNib0ECyuZ/PZOx9TrZcC8vL0x+DUG3TL+GLX3yHG45e6YGjIm0XwDc3q3og==", - "dev": true, - "requires": { - "@babel/types": "^7.0.0", - "@jest/types": "^26.6.2", - "@types/babel__traverse": "^7.0.4", - "@types/prettier": "^2.0.0", - "chalk": "^4.0.0", - "expect": "^26.6.2", - "graceful-fs": "^4.2.4", - "jest-diff": "^26.6.2", - "jest-get-type": "^26.3.0", - "jest-haste-map": "^26.6.2", - "jest-matcher-utils": "^26.6.2", - "jest-message-util": "^26.6.2", - "jest-resolve": "^26.6.2", - "natural-compare": "^1.4.0", - "pretty-format": "^26.6.2", - "semver": "^7.3.2" - }, - "dependencies": { - "semver": { - "version": "7.3.2", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", - "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", - "dev": true - } - } - }, - "jest-util": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-26.6.2.tgz", - "integrity": "sha512-MDW0fKfsn0OI7MS7Euz6h8HNDXVQ0gaM9uW6RjfDmd1DAFcaxX9OqIakHIqhbnmF08Cf2DLDG+ulq8YQQ0Lp0Q==", - "dev": true, - "requires": { - "@jest/types": "^26.6.2", - "@types/node": "*", - "chalk": "^4.0.0", - "graceful-fs": "^4.2.4", - "is-ci": "^2.0.0", - "micromatch": "^4.0.2" - }, - "dependencies": { - "braces": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", - "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", - "dev": true, - "requires": { - "fill-range": "^7.0.1" - } - }, - "fill-range": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", - "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", - "dev": true, - "requires": { - "to-regex-range": "^5.0.1" - } - }, - "is-number": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "dev": true - }, - "micromatch": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", - "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", - "dev": true, - "requires": { - "braces": "^3.0.1", - "picomatch": "^2.0.5" - } - }, - "to-regex-range": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", - "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "dev": true, - "requires": { - "is-number": "^7.0.0" - } - } - } - }, - "jest-validate": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-validate/-/jest-validate-26.6.2.tgz", - "integrity": "sha512-NEYZ9Aeyj0i5rQqbq+tpIOom0YS1u2MVu6+euBsvpgIme+FOfRmoC4R5p0JiAUpaFvFy24xgrpMknarR/93XjQ==", - "dev": true, - "requires": { - "@jest/types": "^26.6.2", - "camelcase": "^6.0.0", - "chalk": "^4.0.0", - "jest-get-type": "^26.3.0", - "leven": "^3.1.0", - "pretty-format": "^26.6.2" - }, - "dependencies": { - "camelcase": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.2.0.tgz", - "integrity": "sha512-c7wVvbw3f37nuobQNtgsgG9POC9qMbNuMQmTCqZv23b6MIz0fcYpBiOlv9gEN/hdLdnZTDQhg6e9Dq5M1vKvfg==", - "dev": true - } - } - }, - "jest-watcher": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-watcher/-/jest-watcher-26.6.2.tgz", - "integrity": "sha512-WKJob0P/Em2csiVthsI68p6aGKTIcsfjH9Gsx1f0A3Italz43e3ho0geSAVsmj09RWOELP1AZ/DXyJgOgDKxXQ==", - "dev": true, - "requires": { - "@jest/test-result": "^26.6.2", - "@jest/types": "^26.6.2", - "@types/node": "*", - "ansi-escapes": "^4.2.1", - "chalk": "^4.0.0", - "jest-util": "^26.6.2", - "string-length": "^4.0.1" - } - }, - "jest-worker": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-26.6.2.tgz", - "integrity": "sha512-KWYVV1c4i+jbMpaBC+U++4Va0cp8OisU185o73T1vo99hqi7w8tSJfUXYswwqqrjzwxa6KpRK54WhPvwf5w6PQ==", - "dev": true, - "requires": { - "@types/node": "*", - "merge-stream": "^2.0.0", - "supports-color": "^7.0.0" - } - }, - "js-tokens": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", - "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", - "dev": true - }, - "js-yaml": { - "version": "3.14.0", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.0.tgz", - "integrity": "sha512-/4IbIeHcD9VMHFqDR/gQ7EdZdLimOvW2DdcxFjdyyZ9NsbS+ccrXqVWDtab/lRl5AlUqmpBx8EhPaWR+OtY17A==", - "requires": { - "argparse": "^1.0.7", - "esprima": "^4.0.0" - } - }, - "jsbn": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", - "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=", - "dev": true - }, - "jsdom": { - "version": "16.4.0", - "resolved": "https://registry.npmjs.org/jsdom/-/jsdom-16.4.0.tgz", - "integrity": "sha512-lYMm3wYdgPhrl7pDcRmvzPhhrGVBeVhPIqeHjzeiHN3DFmD1RBpbExbi8vU7BJdH8VAZYovR8DMt0PNNDM7k8w==", - "dev": true, - "requires": { - "abab": "^2.0.3", - "acorn": "^7.1.1", - "acorn-globals": "^6.0.0", - "cssom": "^0.4.4", - "cssstyle": "^2.2.0", - "data-urls": "^2.0.0", - "decimal.js": "^10.2.0", - "domexception": "^2.0.1", - "escodegen": "^1.14.1", - "html-encoding-sniffer": "^2.0.1", - "is-potential-custom-element-name": "^1.0.0", - "nwsapi": "^2.2.0", - "parse5": "5.1.1", - "request": "^2.88.2", - "request-promise-native": "^1.0.8", - "saxes": "^5.0.0", - "symbol-tree": "^3.2.4", - "tough-cookie": "^3.0.1", - "w3c-hr-time": "^1.0.2", - "w3c-xmlserializer": "^2.0.0", - "webidl-conversions": "^6.1.0", - "whatwg-encoding": "^1.0.5", - "whatwg-mimetype": "^2.3.0", - "whatwg-url": "^8.0.0", - "ws": "^7.2.3", - "xml-name-validator": "^3.0.0" - } - }, - "jsesc": { - "version": "2.5.2", - "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", - "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", - "dev": true - }, - "json-buffer": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz", - "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=", - "dev": true - }, - "json-parse-even-better-errors": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", - "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", - "dev": true - }, - "json-schema": { - "version": "0.2.3", - "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.3.tgz", - "integrity": "sha1-tIDIkuWaLwWVTOcnvT8qTogvnhM=", - "dev": true - }, - "json-schema-traverse": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", - "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", - "dev": true - }, - "json-stable-stringify-without-jsonify": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", - "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", - "dev": true - }, - "json-stringify-safe": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", - "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=", - "dev": true - }, - "json5": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/json5/-/json5-2.1.3.tgz", - "integrity": "sha512-KXPvOm8K9IJKFM0bmdn8QXh7udDh1g/giieX0NLCaMnb4hEiVFqnop2ImTXCc5e0/oHz3LTqmHGtExn5hfMkOA==", - "dev": true, - "requires": { - "minimist": "^1.2.5" - } - }, - "jsonwebtoken": { - "version": "8.5.1", - "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz", - "integrity": "sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==", - "requires": { - "jws": "^3.2.2", - "lodash.includes": "^4.3.0", - "lodash.isboolean": "^3.0.3", - "lodash.isinteger": "^4.0.4", - "lodash.isnumber": "^3.0.3", - "lodash.isplainobject": "^4.0.6", - "lodash.isstring": "^4.0.1", - "lodash.once": "^4.0.0", - "ms": "^2.1.1", - "semver": "^5.6.0" - } - }, - "jsprim": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", - "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=", - "dev": true, - "requires": { - "assert-plus": "1.0.0", - "extsprintf": "1.3.0", - "json-schema": "0.2.3", - "verror": "1.10.0" - } - }, - "jwa": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz", - "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==", - "requires": { - "buffer-equal-constant-time": "1.0.1", - "ecdsa-sig-formatter": "1.0.11", - "safe-buffer": "^5.0.1" - } - }, - "jws": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", - "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", - "requires": { - "jwa": "^1.4.1", - "safe-buffer": "^5.0.1" - } - }, - "keyv": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.1.0.tgz", - "integrity": "sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA==", - "dev": true, - "requires": { - "json-buffer": "3.0.0" - } - }, - "kind-of": { - "version": "6.0.3", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", - "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", - "dev": true - }, - "kleur": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", - "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==", - "dev": true - }, - "latest-version": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/latest-version/-/latest-version-5.1.0.tgz", - "integrity": "sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA==", - "dev": true, - "requires": { - "package-json": "^6.3.0" - } - }, - "leven": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/leven/-/leven-3.1.0.tgz", - "integrity": "sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A==", - "dev": true - }, - "levn": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", - "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=", - "dev": true, - "requires": { - "prelude-ls": "~1.1.2", - "type-check": "~0.3.2" - } - }, - "lines-and-columns": { - "version": "1.1.6", - "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.1.6.tgz", - "integrity": "sha1-HADHQ7QzzQpOgHWPe2SldEDZ/wA=", - "dev": true - }, - "load-json-file": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-2.0.0.tgz", - "integrity": "sha1-eUfkIUmvgNaWy/eXvKq8/h/inKg=", - "dev": true, - "requires": { - "graceful-fs": "^4.1.2", - "parse-json": "^2.2.0", - "pify": "^2.0.0", - "strip-bom": "^3.0.0" - }, - "dependencies": { - "parse-json": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz", - "integrity": "sha1-9ID0BDTvgHQfhGkJn43qGPVaTck=", - "dev": true, - "requires": { - "error-ex": "^1.2.0" - } - }, - "pify": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", - "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", - "dev": true - }, - "strip-bom": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", - "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", - "dev": true - } - } - }, - "locate-path": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", - "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", - "requires": { - "p-locate": "^4.1.0" - } - }, - "lodash": { - "version": "4.17.20", - "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.20.tgz", - "integrity": "sha512-PlhdFcillOINfeV7Ni6oF1TAEayyZBoZ8bcshTHqOYJYlrqzRK5hagpagky5o4HfCzzd1TRkXPMFq6cKk9rGmA==", - "dev": true - }, - "lodash.includes": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", - "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8=" - }, - "lodash.isboolean": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", - "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY=" - }, - "lodash.isinteger": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", - "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M=" - }, - "lodash.isnumber": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", - "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w=" - }, - "lodash.isplainobject": { - "version": "4.0.6", - "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", - "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs=" - }, - "lodash.isstring": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", - "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE=" - }, - "lodash.memoize": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-4.1.2.tgz", - "integrity": "sha1-vMbEmkKihA7Zl/Mj6tpezRguC/4=", - "dev": true - }, - "lodash.once": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", - "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w=" - }, - "lodash.sortby": { - "version": "4.7.0", - "resolved": "https://registry.npmjs.org/lodash.sortby/-/lodash.sortby-4.7.0.tgz", - "integrity": "sha1-7dFMgk4sycHgsKG0K7UhBRakJDg=", - "dev": true - }, - "long": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/long/-/long-4.0.0.tgz", - "integrity": "sha512-XsP+KhQif4bjX1kbuSiySJFNAehNxgLb6hPRGJ9QsUr8ajHkuXGdrHmFUTUUXhDwVX2R5bY4JNZEwbUiMhV+MA==" - }, - "lowercase-keys": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", - "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", - "dev": true - }, - "lru-cache": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", - "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", - "requires": { - "yallist": "^4.0.0" - }, - "dependencies": { - "yallist": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", - "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==" - } - } - }, - "make-dir": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-2.1.0.tgz", - "integrity": "sha512-LS9X+dc8KLxXCb8dni79fLIIUA5VyZoyjSMCwTluaXA0o27cCK0bhXkpgw+sTXVpPy/lSO57ilRixqk0vDmtRA==", - "dev": true, - "requires": { - "pify": "^4.0.1", - "semver": "^5.6.0" - } - }, - "make-error": { - "version": "1.3.6", - "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", - "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", - "dev": true - }, - "makeerror": { - "version": "1.0.11", - "resolved": "https://registry.npmjs.org/makeerror/-/makeerror-1.0.11.tgz", - "integrity": "sha1-4BpckQnyr3lmDk6LlYd5AYT1qWw=", - "dev": true, - "requires": { - "tmpl": "1.0.x" - } - }, - "map-cache": { - "version": "0.2.2", - "resolved": "https://registry.npmjs.org/map-cache/-/map-cache-0.2.2.tgz", - "integrity": "sha1-wyq9C9ZSXZsFFkW7TyasXcmKDb8=", - "dev": true - }, - "map-visit": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/map-visit/-/map-visit-1.0.0.tgz", - "integrity": "sha1-7Nyo8TFE5mDxtb1B8S80edmN+48=", - "dev": true, - "requires": { - "object-visit": "^1.0.0" - } - }, - "media-typer": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", - "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" - }, - "merge-descriptors": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", - "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" - }, - "merge-stream": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", - "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", - "dev": true - }, - "merge2": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", - "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", - "dev": true - }, - "methods": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", - "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" - }, - "micromatch": { - "version": "3.1.10", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", - "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", - "dev": true, - "requires": { - "arr-diff": "^4.0.0", - "array-unique": "^0.3.2", - "braces": "^2.3.1", - "define-property": "^2.0.2", - "extend-shallow": "^3.0.2", - "extglob": "^2.0.4", - "fragment-cache": "^0.2.1", - "kind-of": "^6.0.2", - "nanomatch": "^1.2.9", - "object.pick": "^1.3.0", - "regex-not": "^1.0.0", - "snapdragon": "^0.8.1", - "to-regex": "^3.0.2" - } - }, - "mime": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", - "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" - }, - "mime-db": { - "version": "1.44.0", - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.44.0.tgz", - "integrity": "sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg==" - }, - "mime-types": { - "version": "2.1.27", - "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.27.tgz", - "integrity": "sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==", - "requires": { - "mime-db": "1.44.0" - } - }, - "mimic-fn": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", - "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", - "dev": true - }, - "mimic-response": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz", - "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==", - "dev": true - }, - "minimatch": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", - "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", - "requires": { - "brace-expansion": "^1.1.7" - } - }, - "minimist": { - "version": "1.2.5", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", - "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==" - }, - "minipass": { - "version": "2.9.0", - "resolved": "https://registry.npmjs.org/minipass/-/minipass-2.9.0.tgz", - "integrity": "sha512-wxfUjg9WebH+CUDX/CdbRlh5SmfZiy/hpkxaRI16Y9W56Pa75sWgd/rvFilSgrauD9NyFymP/+JFV3KwzIsJeg==", - "requires": { - "safe-buffer": "^5.1.2", - "yallist": "^3.0.0" - } - }, - "minizlib": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-1.3.3.tgz", - "integrity": "sha512-6ZYMOEnmVsdCeTJVE0W9ZD+pVnE8h9Hma/iOwwRDsdQoePpoX56/8B6z3P9VNwppJuBKNRuFDRNRqRWexT9G9Q==", - "requires": { - "minipass": "^2.9.0" - } - }, - "mixin-deep": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/mixin-deep/-/mixin-deep-1.3.2.tgz", - "integrity": "sha512-WRoDn//mXBiJ1H40rqa3vH0toePwSsGb45iInWlTySa+Uu4k3tYUSxa2v1KqAiLtvlrSzaExqS1gtk96A9zvEA==", - "dev": true, - "requires": { - "for-in": "^1.0.2", - "is-extendable": "^1.0.1" - }, - "dependencies": { - "is-extendable": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", - "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", - "dev": true, - "requires": { - "is-plain-object": "^2.0.4" - } - } - } - }, - "mkdirp": { - "version": "0.5.5", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", - "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", - "requires": { - "minimist": "^1.2.5" - } - }, - "ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" - }, - "mysql2": { - "version": "2.2.5", - "resolved": "https://registry.npmjs.org/mysql2/-/mysql2-2.2.5.tgz", - "integrity": "sha512-XRqPNxcZTpmFdXbJqb+/CtYVLCx14x1RTeNMD4954L331APu75IC74GDqnZMEt1kwaXy6TySo55rF2F3YJS78g==", - "requires": { - "denque": "^1.4.1", - "generate-function": "^2.3.1", - "iconv-lite": "^0.6.2", - "long": "^4.0.0", - "lru-cache": "^6.0.0", - "named-placeholders": "^1.1.2", - "seq-queue": "^0.0.5", - "sqlstring": "^2.3.2" - }, - "dependencies": { - "iconv-lite": { - "version": "0.6.2", - "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.2.tgz", - "integrity": "sha512-2y91h5OpQlolefMPmUlivelittSWy0rP+oYVpn6A7GwVHNE8AWzoYOBNmlwks3LobaJxgHCYZAnyNo2GgpNRNQ==", - "requires": { - "safer-buffer": ">= 2.1.2 < 3.0.0" - } - } - } - }, - "mz": { - "version": "2.7.0", - "resolved": "https://registry.npmjs.org/mz/-/mz-2.7.0.tgz", - "integrity": "sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==", - "requires": { - "any-promise": "^1.0.0", - "object-assign": "^4.0.1", - "thenify-all": "^1.0.0" - } - }, - "named-placeholders": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/named-placeholders/-/named-placeholders-1.1.2.tgz", - "integrity": "sha512-wiFWqxoLL3PGVReSZpjLVxyJ1bRqe+KKJVbr4hGs1KWfTZTQyezHFBbuKj9hsizHyGV2ne7EMjHdxEGAybD5SA==", - "requires": { - "lru-cache": "^4.1.3" - }, - "dependencies": { - "lru-cache": { - "version": "4.1.5", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-4.1.5.tgz", - "integrity": "sha512-sWZlbEP2OsHNkXrMl5GYk/jKk70MBng6UU4YI/qGDYbgf6YbP4EvmqISbXCoJiRKs+1bSpFHVgQxvJ17F2li5g==", - "requires": { - "pseudomap": "^1.0.2", - "yallist": "^2.1.2" - } - }, - "yallist": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-2.1.2.tgz", - "integrity": "sha1-HBH5IY8HYImkfdUS+TxmmaaoHVI=" - } - } - }, - "nan": { - "version": "2.14.2", - "resolved": "https://registry.npmjs.org/nan/-/nan-2.14.2.tgz", - "integrity": "sha512-M2ufzIiINKCuDfBSAUr1vWQ+vuVcA9kqx8JJUsbQi6yf1uGRyb7HfpdfUr5qLXf3B/t8dPvcjhKMmlfnP47EzQ==", - "dev": true, - "optional": true - }, - "nanomatch": { - "version": "1.2.13", - "resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.13.tgz", - "integrity": "sha512-fpoe2T0RbHwBTBUOftAfBPaDEi06ufaUai0mE6Yn1kacc3SnTErfb/h+X94VXzI64rKFHYImXSvdwGGCmwOqCA==", - "dev": true, - "requires": { - "arr-diff": "^4.0.0", - "array-unique": "^0.3.2", - "define-property": "^2.0.2", - "extend-shallow": "^3.0.2", - "fragment-cache": "^0.2.1", - "is-windows": "^1.0.2", - "kind-of": "^6.0.2", - "object.pick": "^1.3.0", - "regex-not": "^1.0.0", - "snapdragon": "^0.8.1", - "to-regex": "^3.0.1" - } - }, - "natural-compare": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", - "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", - "dev": true - }, - "needle": { - "version": "2.5.2", - "resolved": "https://registry.npmjs.org/needle/-/needle-2.5.2.tgz", - "integrity": "sha512-LbRIwS9BfkPvNwNHlsA41Q29kL2L/6VaOJ0qisM5lLWsTV3nP15abO5ITL6L81zqFhzjRKDAYjpcBcwM0AVvLQ==", - "requires": { - "debug": "^3.2.6", - "iconv-lite": "^0.4.4", - "sax": "^1.2.4" - } - }, - "negotiator": { - "version": "0.6.2", - "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", - "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" - }, - "nice-try": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", - "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", - "dev": true - }, - "node-addon-api": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-3.0.2.tgz", - "integrity": "sha512-+D4s2HCnxPd5PjjI0STKwncjXTUKKqm74MDMz9OPXavjsGmjkvwgLtA5yoxJUdmpj52+2u+RrXgPipahKczMKg==" - }, - "node-environment-flags": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/node-environment-flags/-/node-environment-flags-1.0.6.tgz", - "integrity": "sha512-5Evy2epuL+6TM0lCQGpFIj6KwiEsGh1SrHUhTbNX+sLbBtjidPZFAnVK9y5yU1+h//RitLbRHTIMyxQPtxMdHw==", - "dev": true, - "requires": { - "object.getownpropertydescriptors": "^2.0.3", - "semver": "^5.7.0" - } - }, - "node-gyp": { - "version": "3.8.0", - "resolved": "https://registry.npmjs.org/node-gyp/-/node-gyp-3.8.0.tgz", - "integrity": "sha512-3g8lYefrRRzvGeSowdJKAKyks8oUpLEd/DyPV4eMhVlhJ0aNaZqIrNUIPuEWWTAoPqyFkfGrM67MC69baqn6vA==", - "dev": true, - "optional": true, - "requires": { - "fstream": "^1.0.0", - "glob": "^7.0.3", - "graceful-fs": "^4.1.2", - "mkdirp": "^0.5.0", - "nopt": "2 || 3", - "npmlog": "0 || 1 || 2 || 3 || 4", - "osenv": "0", - "request": "^2.87.0", - "rimraf": "2", - "semver": "~5.3.0", - "tar": "^2.0.0", - "which": "1" - }, - "dependencies": { - "nopt": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/nopt/-/nopt-3.0.6.tgz", - "integrity": "sha1-xkZdvwirzU2zWTF/eaxopkayj/k=", - "dev": true, - "optional": true, - "requires": { - "abbrev": "1" - } - }, - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "optional": true, - "requires": { - "glob": "^7.1.3" - } - }, - "semver": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.3.0.tgz", - "integrity": "sha1-myzl094C0XxgEq0yaqa00M9U+U8=", - "dev": true, - "optional": true - }, - "tar": { - "version": "2.2.2", - "resolved": "https://registry.npmjs.org/tar/-/tar-2.2.2.tgz", - "integrity": "sha512-FCEhQ/4rE1zYv9rYXJw/msRqsnmlje5jHP6huWeBZ704jUTy02c5AZyWujpMR1ax6mVw9NyJMfuK2CMDWVIfgA==", - "dev": true, - "optional": true, - "requires": { - "block-stream": "*", - "fstream": "^1.0.12", - "inherits": "2" - } - } - } - }, - "node-int64": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz", - "integrity": "sha1-h6kGXNs1XTGC2PlM4RGIuCXGijs=", - "dev": true - }, - "node-mocks-http": { - "version": "1.9.0", - "resolved": "https://registry.npmjs.org/node-mocks-http/-/node-mocks-http-1.9.0.tgz", - "integrity": "sha512-ILf7Ws8xyX9Rl2fLZ7xhZBovrRwgaP84M13esndP6V17M/8j25TpwNzb7Im8U9XCo6fRhdwqiQajWXpsas/E6w==", - "requires": { - "accepts": "^1.3.7", - "depd": "^1.1.0", - "fresh": "^0.5.2", - "merge-descriptors": "^1.0.1", - "methods": "^1.1.2", - "mime": "^1.3.4", - "parseurl": "^1.3.3", - "range-parser": "^1.2.0", - "type-is": "^1.6.18" - } - }, - "node-modules-regexp": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/node-modules-regexp/-/node-modules-regexp-1.0.0.tgz", - "integrity": "sha1-jZ2+KJZKSsVxLpExZCEHxx6Q7EA=", - "dev": true - }, - "node-notifier": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/node-notifier/-/node-notifier-8.0.0.tgz", - "integrity": "sha512-46z7DUmcjoYdaWyXouuFNNfUo6eFa94t23c53c+lG/9Cvauk4a98rAUp9672X5dxGdQmLpPzTxzu8f/OeEPaFA==", - "dev": true, - "optional": true, - "requires": { - "growly": "^1.3.0", - "is-wsl": "^2.2.0", - "semver": "^7.3.2", - "shellwords": "^0.1.1", - "uuid": "^8.3.0", - "which": "^2.0.2" - }, - "dependencies": { - "semver": { - "version": "7.3.2", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", - "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", - "dev": true, - "optional": true - }, - "which": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", - "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", - "dev": true, - "optional": true, - "requires": { - "isexe": "^2.0.0" - } - } - } - }, - "node-pre-gyp": { - "version": "0.15.0", - "resolved": "https://registry.npmjs.org/node-pre-gyp/-/node-pre-gyp-0.15.0.tgz", - "integrity": "sha512-7QcZa8/fpaU/BKenjcaeFF9hLz2+7S9AqyXFhlH/rilsQ/hPZKK32RtR5EQHJElgu+q5RfbJ34KriI79UWaorA==", - "requires": { - "detect-libc": "^1.0.2", - "mkdirp": "^0.5.3", - "needle": "^2.5.0", - "nopt": "^4.0.1", - "npm-packlist": "^1.1.6", - "npmlog": "^4.0.2", - "rc": "^1.2.7", - "rimraf": "^2.6.1", - "semver": "^5.3.0", - "tar": "^4.4.2" - }, - "dependencies": { - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "requires": { - "glob": "^7.1.3" - } - } - } - }, - "node-releases": { - "version": "1.1.67", - "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-1.1.67.tgz", - "integrity": "sha512-V5QF9noGFl3EymEwUYzO+3NTDpGfQB4ve6Qfnzf3UNydMhjQRVPR1DZTuvWiLzaFJYw2fmDwAfnRNEVb64hSIg==", - "dev": true - }, - "nodemon": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/nodemon/-/nodemon-2.0.6.tgz", - "integrity": "sha512-4I3YDSKXg6ltYpcnZeHompqac4E6JeAMpGm8tJnB9Y3T0ehasLa4139dJOcCrB93HHrUMsCrKtoAlXTqT5n4AQ==", - "dev": true, - "requires": { - "chokidar": "^3.2.2", - "debug": "^3.2.6", - "ignore-by-default": "^1.0.1", - "minimatch": "^3.0.4", - "pstree.remy": "^1.1.7", - "semver": "^5.7.1", - "supports-color": "^5.5.0", - "touch": "^3.1.0", - "undefsafe": "^2.0.3", - "update-notifier": "^4.1.0" - }, - "dependencies": { - "has-flag": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", - "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", - "dev": true - }, - "supports-color": { - "version": "5.5.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", - "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", - "dev": true, - "requires": { - "has-flag": "^3.0.0" - } - } - } - }, - "nopt": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/nopt/-/nopt-4.0.3.tgz", - "integrity": "sha512-CvaGwVMztSMJLOeXPrez7fyfObdZqNUK1cPAEzLHrTybIua9pMdmmPR5YwtfNftIOMv3DPUhFaxsZMNTQO20Kg==", - "requires": { - "abbrev": "1", - "osenv": "^0.1.4" - } - }, - "normalize-package-data": { - "version": "2.5.0", - "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz", - "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", - "dev": true, - "requires": { - "hosted-git-info": "^2.1.4", - "resolve": "^1.10.0", - "semver": "2 || 3 || 4 || 5", - "validate-npm-package-license": "^3.0.1" - } - }, - "normalize-path": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", - "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", - "dev": true - }, - "normalize-url": { - "version": "4.5.0", - "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-4.5.0.tgz", - "integrity": "sha512-2s47yzUxdexf1OhyRi4Em83iQk0aPvwTddtFz4hnSSw9dCEsLEGf6SwIO8ss/19S9iBb5sJaOuTvTGDeZI00BQ==", - "dev": true - }, - "npm-bundled": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/npm-bundled/-/npm-bundled-1.1.1.tgz", - "integrity": "sha512-gqkfgGePhTpAEgUsGEgcq1rqPXA+tv/aVBlgEzfXwA1yiUJF7xtEt3CtVwOjNYQOVknDk0F20w58Fnm3EtG0fA==", - "requires": { - "npm-normalize-package-bin": "^1.0.1" - } - }, - "npm-normalize-package-bin": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/npm-normalize-package-bin/-/npm-normalize-package-bin-1.0.1.tgz", - "integrity": "sha512-EPfafl6JL5/rU+ot6P3gRSCpPDW5VmIzX959Ob1+ySFUuuYHWHekXpwdUZcKP5C+DS4GEtdJluwBjnsNDl+fSA==" - }, - "npm-packlist": { - "version": "1.4.8", - "resolved": "https://registry.npmjs.org/npm-packlist/-/npm-packlist-1.4.8.tgz", - "integrity": "sha512-5+AZgwru5IevF5ZdnFglB5wNlHG1AOOuw28WhUq8/8emhBmLv6jX5by4WJCh7lW0uSYZYS6DXqIsyZVIXRZU9A==", - "requires": { - "ignore-walk": "^3.0.1", - "npm-bundled": "^1.0.1", - "npm-normalize-package-bin": "^1.0.1" - } - }, - "npm-run-path": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-2.0.2.tgz", - "integrity": "sha1-NakjLfo11wZ7TLLd8jV7GHFTbF8=", - "dev": true, - "requires": { - "path-key": "^2.0.0" - } - }, - "npmlog": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/npmlog/-/npmlog-4.1.2.tgz", - "integrity": "sha512-2uUqazuKlTaSI/dC8AzicUck7+IrEaOnN/e0jd3Xtt1KcGpwx30v50mL7oPyr/h9bL3E4aZccVwpwP+5W9Vjkg==", - "requires": { - "are-we-there-yet": "~1.1.2", - "console-control-strings": "~1.1.0", - "gauge": "~2.7.3", - "set-blocking": "~2.0.0" - } - }, - "number-is-nan": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", - "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=" - }, - "nwsapi": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/nwsapi/-/nwsapi-2.2.0.tgz", - "integrity": "sha512-h2AatdwYH+JHiZpv7pt/gSX1XoRGb7L/qSIeuqA6GwYoF9w1vP1cw42TO0aI2pNyshRK5893hNSl+1//vHK7hQ==", - "dev": true - }, - "oauth-sign": { - "version": "0.9.0", - "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz", - "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==", - "dev": true - }, - "object-assign": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", - "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=" - }, - "object-copy": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/object-copy/-/object-copy-0.1.0.tgz", - "integrity": "sha1-fn2Fi3gb18mRpBupde04EnVOmYw=", - "dev": true, - "requires": { - "copy-descriptor": "^0.1.0", - "define-property": "^0.2.5", - "kind-of": "^3.0.3" - }, - "dependencies": { - "define-property": { - "version": "0.2.5", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, - "requires": { - "is-descriptor": "^0.1.0" - } - }, - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } - } - }, - "object-inspect": { - "version": "1.8.0", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.8.0.tgz", - "integrity": "sha512-jLdtEOB112fORuypAyl/50VRVIBIdVQOSUUGQHzJ4xBSbit81zRarz7GThkEFZy1RceYrWYcPcBFPQwHyAc1gA==", - "dev": true - }, - "object-keys": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", - "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", - "dev": true - }, - "object-visit": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/object-visit/-/object-visit-1.0.1.tgz", - "integrity": "sha1-95xEk68MU3e1n+OdOV5BBC3QRbs=", - "dev": true, - "requires": { - "isobject": "^3.0.0" - } - }, - "object.assign": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.2.tgz", - "integrity": "sha512-ixT2L5THXsApyiUPYKmW+2EHpXXe5Ii3M+f4e+aJFAHao5amFRW6J0OO6c/LU8Be47utCx2GL89hxGB6XSmKuQ==", - "dev": true, - "requires": { - "call-bind": "^1.0.0", - "define-properties": "^1.1.3", - "has-symbols": "^1.0.1", - "object-keys": "^1.1.1" - } - }, - "object.entries": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/object.entries/-/object.entries-1.1.2.tgz", - "integrity": "sha512-BQdB9qKmb/HyNdMNWVr7O3+z5MUIx3aiegEIJqjMBbBf0YT9RRxTJSim4mzFqtyr7PDAHigq0N9dO0m0tRakQA==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.5", - "has": "^1.0.3" - } - }, - "object.getownpropertydescriptors": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.1.0.tgz", - "integrity": "sha512-Z53Oah9A3TdLoblT7VKJaTDdXdT+lQO+cNpKVnya5JDe9uLvzu1YyY1yFDFrcxrlRgWrEFH0jJtD/IbuwjcEVg==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.0-next.1" - } - }, - "object.pick": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/object.pick/-/object.pick-1.3.0.tgz", - "integrity": "sha1-h6EKxMFpS9Lhy/U1kaZhQftd10c=", - "dev": true, - "requires": { - "isobject": "^3.0.1" - } - }, - "object.values": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.1.tgz", - "integrity": "sha512-WTa54g2K8iu0kmS/us18jEmdv1a4Wi//BZ/DTVYEcH0XhLM5NYdpDHja3gt57VrZLcNAO2WGA+KpWsDBaHt6eA==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.0-next.1", - "function-bind": "^1.1.1", - "has": "^1.0.3" - } - }, - "on-finished": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", - "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", - "requires": { - "ee-first": "1.1.1" - } - }, - "once": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", - "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", - "requires": { - "wrappy": "1" - } - }, - "onetime": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", - "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", - "dev": true, - "requires": { - "mimic-fn": "^2.1.0" - } - }, - "optionator": { - "version": "0.8.3", - "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz", - "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==", - "dev": true, - "requires": { - "deep-is": "~0.1.3", - "fast-levenshtein": "~2.0.6", - "levn": "~0.3.0", - "prelude-ls": "~1.1.2", - "type-check": "~0.3.2", - "word-wrap": "~1.2.3" - } - }, - "os-homedir": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/os-homedir/-/os-homedir-1.0.2.tgz", - "integrity": "sha1-/7xJiDNuDoM94MFox+8VISGqf7M=" - }, - "os-tmpdir": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", - "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=" - }, - "osenv": { - "version": "0.1.5", - "resolved": "https://registry.npmjs.org/osenv/-/osenv-0.1.5.tgz", - "integrity": "sha512-0CWcCECdMVc2Rw3U5w9ZjqX6ga6ubk1xDVKxtBQPK7wis/0F2r9T6k4ydGYhecl7YUBxBVxhL5oisPsNxAPe2g==", - "requires": { - "os-homedir": "^1.0.0", - "os-tmpdir": "^1.0.0" - } - }, - "p-cancelable": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-1.1.0.tgz", - "integrity": "sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw==", - "dev": true - }, - "p-each-series": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/p-each-series/-/p-each-series-2.2.0.tgz", - "integrity": "sha512-ycIL2+1V32th+8scbpTvyHNaHe02z0sjgh91XXjAk+ZeXoPN4Z46DVUnzdso0aX4KckKw0FNNFHdjZ2UsZvxiA==", - "dev": true - }, - "p-finally": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", - "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=", - "dev": true - }, - "p-limit": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", - "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", - "requires": { - "p-try": "^2.0.0" - } - }, - "p-locate": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", - "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", - "requires": { - "p-limit": "^2.2.0" - } - }, - "p-try": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", - "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==" - }, - "package-json": { - "version": "6.5.0", - "resolved": "https://registry.npmjs.org/package-json/-/package-json-6.5.0.tgz", - "integrity": "sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ==", - "dev": true, - "requires": { - "got": "^9.6.0", - "registry-auth-token": "^4.0.0", - "registry-url": "^5.0.0", - "semver": "^6.2.0" - }, - "dependencies": { - "semver": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", - "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", - "dev": true - } - } - }, - "parent-module": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", - "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", - "dev": true, - "requires": { - "callsites": "^3.0.0" - } - }, - "parent-require": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/parent-require/-/parent-require-1.0.0.tgz", - "integrity": "sha1-dGoWdjgIOoYLDu9nMssn7UbDKXc=" - }, - "parse-json": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.1.0.tgz", - "integrity": "sha512-+mi/lmVVNKFNVyLXV31ERiy2CY5E1/F6QtJFEzoChPRwwngMNXRDQ9GJ5WdE2Z2P4AujsOi0/+2qHID68KwfIQ==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.0.0", - "error-ex": "^1.3.1", - "json-parse-even-better-errors": "^2.3.0", - "lines-and-columns": "^1.1.6" - } - }, - "parse-passwd": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/parse-passwd/-/parse-passwd-1.0.0.tgz", - "integrity": "sha1-bVuTSkVpk7I9N/QKOC1vFmao5cY=", - "dev": true - }, - "parse5": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/parse5/-/parse5-5.1.1.tgz", - "integrity": "sha512-ugq4DFI0Ptb+WWjAdOK16+u/nHfiIrcE+sh8kZMaM0WllQKLI9rOUq6c2b7cwPkXdzfQESqvoqK6ug7U/Yyzug==" - }, - "parse5-htmlparser2-tree-adapter": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-5.1.1.tgz", - "integrity": "sha512-CF+TKjXqoqyDwHqBhFQ+3l5t83xYi6fVT1tQNg+Ye0JRLnTxWvIroCjEp1A0k4lneHNBGnICUf0cfYVYGEazqw==", - "requires": { - "parse5": "^5.1.1" - } - }, - "parseurl": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", - "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" - }, - "pascalcase": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/pascalcase/-/pascalcase-0.1.1.tgz", - "integrity": "sha1-s2PlXoAGym/iF4TS2yK9FdeRfxQ=", - "dev": true - }, - "passport": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/passport/-/passport-0.4.1.tgz", - "integrity": "sha512-IxXgZZs8d7uFSt3eqNjM9NQ3g3uQCW5avD8mRNoXV99Yig50vjuaez6dQK2qC0kVWPRTujxY0dWgGfT09adjYg==", - "requires": { - "passport-strategy": "1.x.x", - "pause": "0.0.1" - } - }, - "passport-jwt": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/passport-jwt/-/passport-jwt-4.0.0.tgz", - "integrity": "sha512-BwC0n2GP/1hMVjR4QpnvqA61TxenUMlmfNjYNgK0ZAs0HK4SOQkHcSv4L328blNTLtHq7DbmvyNJiH+bn6C5Mg==", - "requires": { - "jsonwebtoken": "^8.2.0", - "passport-strategy": "^1.0.0" - } - }, - "passport-strategy": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/passport-strategy/-/passport-strategy-1.0.0.tgz", - "integrity": "sha1-tVOaqPwiWj0a0XlHbd8ja0QPUuQ=" - }, - "path-dirname": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/path-dirname/-/path-dirname-1.0.2.tgz", - "integrity": "sha1-zDPSTVJeCZpTiMAzbG4yuRYGCeA=", - "dev": true, - "optional": true - }, - "path-exists": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", - "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==" - }, - "path-is-absolute": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", - "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=" - }, - "path-key": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", - "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", - "dev": true - }, - "path-parse": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz", - "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==", - "dev": true - }, - "path-to-regexp": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", - "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" - }, - "path-type": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", - "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", - "dev": true - }, - "pause": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/pause/-/pause-0.0.1.tgz", - "integrity": "sha1-HUCLP9t2kjuVQ9lvtMnf1TXZy10=" - }, - "performance-now": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", - "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=", - "dev": true - }, - "picomatch": { - "version": "2.2.2", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz", - "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==", - "dev": true - }, - "pify": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", - "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", - "dev": true - }, - "pirates": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.1.tgz", - "integrity": "sha512-WuNqLTbMI3tmfef2TKxlQmAiLHKtFhlsCZnPIpuv2Ow0RDVO8lfy1Opf4NUzlMXLjPl+Men7AuVdX6TA+s+uGA==", - "dev": true, - "requires": { - "node-modules-regexp": "^1.0.0" - } - }, - "pkg-dir": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-3.0.0.tgz", - "integrity": "sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw==", - "dev": true, - "requires": { - "find-up": "^3.0.0" - }, - "dependencies": { - "find-up": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", - "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", - "dev": true, - "requires": { - "locate-path": "^3.0.0" - } - }, - "locate-path": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", - "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", - "dev": true, - "requires": { - "p-locate": "^3.0.0", - "path-exists": "^3.0.0" - } - }, - "p-locate": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", - "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", - "dev": true, - "requires": { - "p-limit": "^2.0.0" - } - }, - "path-exists": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", - "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", - "dev": true - } - } - }, - "posix-character-classes": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/posix-character-classes/-/posix-character-classes-0.1.1.tgz", - "integrity": "sha1-AerA/jta9xoqbAL+q7jB/vfgDqs=", - "dev": true - }, - "prelude-ls": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", - "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=", - "dev": true - }, - "prepend-http": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz", - "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=", - "dev": true - }, - "prettier": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.1.2.tgz", - "integrity": "sha512-16c7K+x4qVlJg9rEbXl7HEGmQyZlG4R9AgP+oHKRMsMsuk8s+ATStlf1NpDqyBI1HpVyfjLOeMhH2LvuNvV5Vg==", - "dev": true - }, - "prettier-linter-helpers": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/prettier-linter-helpers/-/prettier-linter-helpers-1.0.0.tgz", - "integrity": "sha512-GbK2cP9nraSSUF9N2XwUwqfzlAFlMNYYl+ShE/V+H8a9uNl/oUqB1w2EL54Jh0OlyRSd8RfWYJ3coVS4TROP2w==", - "dev": true, - "requires": { - "fast-diff": "^1.1.2" - } - }, - "pretty-format": { - "version": "26.6.2", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-26.6.2.tgz", - "integrity": "sha512-7AeGuCYNGmycyQbCqd/3PWH4eOoX/OiCa0uphp57NVTeAGdJGaAliecxwBDHYQCIvrW7aDBZCYeNTP/WX69mkg==", - "dev": true, - "requires": { - "@jest/types": "^26.6.2", - "ansi-regex": "^5.0.0", - "ansi-styles": "^4.0.0", - "react-is": "^17.0.1" - }, - "dependencies": { - "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", - "dev": true - } - } - }, - "process-nextick-args": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", - "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" - }, - "progress": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz", - "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==", - "dev": true - }, - "prompts": { - "version": "2.4.0", - "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.0.tgz", - "integrity": "sha512-awZAKrk3vN6CroQukBL+R9051a4R3zCZBlJm/HBfrSZ8iTpYix3VX1vU4mveiLpiwmOJT4wokTF9m6HUk4KqWQ==", - "dev": true, - "requires": { - "kleur": "^3.0.3", - "sisteransi": "^1.0.5" - } - }, - "proxy-addr": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz", - "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==", - "requires": { - "forwarded": "~0.1.2", - "ipaddr.js": "1.9.1" - } - }, - "pseudomap": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/pseudomap/-/pseudomap-1.0.2.tgz", - "integrity": "sha1-8FKijacOYYkX7wqKw0wa5aaChrM=" - }, - "psl": { - "version": "1.8.0", - "resolved": "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz", - "integrity": "sha512-RIdOzyoavK+hA18OGGWDqUTsCLhtA7IcZ/6NCs4fFJaHBDab+pDDmDIByWFRQJq2Cd7r1OoQxBGKOaztq+hjIQ==", - "dev": true - }, - "pstree.remy": { - "version": "1.1.8", - "resolved": "https://registry.npmjs.org/pstree.remy/-/pstree.remy-1.1.8.tgz", - "integrity": "sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w==", - "dev": true - }, - "pump": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", - "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", - "dev": true, - "requires": { - "end-of-stream": "^1.1.0", - "once": "^1.3.1" - } - }, - "punycode": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", - "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", - "dev": true - }, - "pupa": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/pupa/-/pupa-2.1.1.tgz", - "integrity": "sha512-l1jNAspIBSFqbT+y+5FosojNpVpF94nlI+wDUpqP9enwOTfHx9f0gh5nB96vl+6yTpsJsypeNrwfzPrKuHB41A==", - "dev": true, - "requires": { - "escape-goat": "^2.0.0" - } - }, - "qs": { - "version": "6.7.0", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", - "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==" - }, - "range-parser": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", - "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" - }, - "raw-body": { - "version": "2.4.0", - "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", - "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", - "requires": { - "bytes": "3.1.0", - "http-errors": "1.7.2", - "iconv-lite": "0.4.24", - "unpipe": "1.0.0" - } - }, - "rc": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", - "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", - "requires": { - "deep-extend": "^0.6.0", - "ini": "~1.3.0", - "minimist": "^1.2.0", - "strip-json-comments": "~2.0.1" - } - }, - "react-is": { - "version": "17.0.1", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.1.tgz", - "integrity": "sha512-NAnt2iGDXohE5LI7uBnLnqvLQMtzhkiAOLXTmv+qnF9Ky7xAPcX8Up/xWIhxvLVGJvuLiNc4xQLtuqDRzb4fSA==", - "dev": true - }, - "read-pkg": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-5.2.0.tgz", - "integrity": "sha512-Ug69mNOpfvKDAc2Q8DRpMjjzdtrnv9HcSMX+4VsZxD1aZ6ZzrIE7rlzXBtWTyhULSMKg076AW6WR5iZpD0JiOg==", - "dev": true, - "requires": { - "@types/normalize-package-data": "^2.4.0", - "normalize-package-data": "^2.5.0", - "parse-json": "^5.0.0", - "type-fest": "^0.6.0" - }, - "dependencies": { - "type-fest": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.6.0.tgz", - "integrity": "sha512-q+MB8nYR1KDLrgr4G5yemftpMC7/QLqVndBmEEdqzmNj5dcFOO4Oo8qlwZE3ULT3+Zim1F8Kq4cBnikNhlCMlg==", - "dev": true - } - } - }, - "read-pkg-up": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-7.0.1.tgz", - "integrity": "sha512-zK0TB7Xd6JpCLmlLmufqykGE+/TlOePD6qKClNW7hHDKFh/J7/7gCWGR7joEQEW1bKq3a3yUZSObOoWLFQ4ohg==", - "dev": true, - "requires": { - "find-up": "^4.1.0", - "read-pkg": "^5.2.0", - "type-fest": "^0.8.1" - } - }, - "readable-stream": { - "version": "2.3.7", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", - "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", - "requires": { - "core-util-is": "~1.0.0", - "inherits": "~2.0.3", - "isarray": "~1.0.0", - "process-nextick-args": "~2.0.0", - "safe-buffer": "~5.1.1", - "string_decoder": "~1.1.1", - "util-deprecate": "~1.0.1" - } - }, - "readdirp": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-2.2.1.tgz", - "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", - "dev": true, - "optional": true, - "requires": { - "graceful-fs": "^4.1.11", - "micromatch": "^3.1.10", - "readable-stream": "^2.0.2" - } - }, - "reflect-metadata": { - "version": "0.1.13", - "resolved": "https://registry.npmjs.org/reflect-metadata/-/reflect-metadata-0.1.13.tgz", - "integrity": "sha512-Ts1Y/anZELhSsjMcU605fU9RE4Oi3p5ORujwbIKXfWa+0Zxs510Qrmrce5/Jowq3cHSZSJqBjypxmHarc+vEWg==" - }, - "regenerate": { - "version": "1.4.2", - "resolved": "https://registry.npmjs.org/regenerate/-/regenerate-1.4.2.tgz", - "integrity": "sha512-zrceR/XhGYU/d/opr2EKO7aRHUeiBI8qjtfHqADTwZd6Szfy16la6kqD0MIUs5z5hx6AaKa+PixpPrR289+I0A==", - "dev": true - }, - "regenerate-unicode-properties": { - "version": "8.2.0", - "resolved": "https://registry.npmjs.org/regenerate-unicode-properties/-/regenerate-unicode-properties-8.2.0.tgz", - "integrity": "sha512-F9DjY1vKLo/tPePDycuH3dn9H1OTPIkVD9Kz4LODu+F2C75mgjAJ7x/gwy6ZcSNRAAkhNlJSOHRe8k3p+K9WhA==", - "dev": true, - "requires": { - "regenerate": "^1.4.0" - } - }, - "regenerator-runtime": { - "version": "0.13.7", - "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.7.tgz", - "integrity": "sha512-a54FxoJDIr27pgf7IgeQGxmqUNYrcV338lf/6gH456HZ/PhX+5BcwHXG9ajESmwe6WRO0tAzRUrRmNONWgkrew==", - "dev": true - }, - "regenerator-transform": { - "version": "0.14.5", - "resolved": "https://registry.npmjs.org/regenerator-transform/-/regenerator-transform-0.14.5.tgz", - "integrity": "sha512-eOf6vka5IO151Jfsw2NO9WpGX58W6wWmefK3I1zEGr0lOD0u8rwPaNqQL1aRxUaxLeKO3ArNh3VYg1KbaD+FFw==", - "dev": true, - "requires": { - "@babel/runtime": "^7.8.4" - } - }, - "regex-not": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/regex-not/-/regex-not-1.0.2.tgz", - "integrity": "sha512-J6SDjUgDxQj5NusnOtdFxDwN/+HWykR8GELwctJ7mdqhcyy1xEc4SRFHUXvxTp661YaVKAjfRLZ9cCqS6tn32A==", - "dev": true, - "requires": { - "extend-shallow": "^3.0.2", - "safe-regex": "^1.1.0" - } - }, - "regexpp": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.1.0.tgz", - "integrity": "sha512-ZOIzd8yVsQQA7j8GCSlPGXwg5PfmA1mrq0JP4nGhh54LaKN3xdai/vHUDu74pKwV8OxseMS65u2NImosQcSD0Q==", - "dev": true - }, - "regexpu-core": { - "version": "4.7.1", - "resolved": "https://registry.npmjs.org/regexpu-core/-/regexpu-core-4.7.1.tgz", - "integrity": "sha512-ywH2VUraA44DZQuRKzARmw6S66mr48pQVva4LBeRhcOltJ6hExvWly5ZjFLYo67xbIxb6W1q4bAGtgfEl20zfQ==", - "dev": true, - "requires": { - "regenerate": "^1.4.0", - "regenerate-unicode-properties": "^8.2.0", - "regjsgen": "^0.5.1", - "regjsparser": "^0.6.4", - "unicode-match-property-ecmascript": "^1.0.4", - "unicode-match-property-value-ecmascript": "^1.2.0" - } - }, - "registry-auth-token": { - "version": "4.2.1", - "resolved": "https://registry.npmjs.org/registry-auth-token/-/registry-auth-token-4.2.1.tgz", - "integrity": "sha512-6gkSb4U6aWJB4SF2ZvLb76yCBjcvufXBqvvEx1HbmKPkutswjW1xNVRY0+daljIYRbogN7O0etYSlbiaEQyMyw==", - "dev": true, - "requires": { - "rc": "^1.2.8" - } - }, - "registry-url": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/registry-url/-/registry-url-5.1.0.tgz", - "integrity": "sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw==", - "dev": true, - "requires": { - "rc": "^1.2.8" - } - }, - "regjsgen": { - "version": "0.5.2", - "resolved": "https://registry.npmjs.org/regjsgen/-/regjsgen-0.5.2.tgz", - "integrity": "sha512-OFFT3MfrH90xIW8OOSyUrk6QHD5E9JOTeGodiJeBS3J6IwlgzJMNE/1bZklWz5oTg+9dCMyEetclvCVXOPoN3A==", - "dev": true - }, - "regjsparser": { - "version": "0.6.4", - "resolved": "https://registry.npmjs.org/regjsparser/-/regjsparser-0.6.4.tgz", - "integrity": "sha512-64O87/dPDgfk8/RQqC4gkZoGyyWFIEUTTh80CU6CWuK5vkCGyekIx+oKcEIYtP/RAxSQltCZHCNu/mdd7fqlJw==", - "dev": true, - "requires": { - "jsesc": "~0.5.0" - }, - "dependencies": { - "jsesc": { - "version": "0.5.0", - "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-0.5.0.tgz", - "integrity": "sha1-597mbjXW/Bb3EP6R1c9p9w8IkR0=", - "dev": true - } - } - }, - "remove-trailing-separator": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz", - "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=", - "dev": true - }, - "repeat-element": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/repeat-element/-/repeat-element-1.1.3.tgz", - "integrity": "sha512-ahGq0ZnV5m5XtZLMb+vP76kcAM5nkLqk0lpqAuojSKGgQtn4eRi4ZZGm2olo2zKFH+sMsWaqOCW1dqAnOru72g==", - "dev": true - }, - "repeat-string": { - "version": "1.6.1", - "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", - "integrity": "sha1-jcrkcOHIirwtYA//Sndihtp15jc=", - "dev": true - }, - "request": { - "version": "2.88.2", - "resolved": "https://registry.npmjs.org/request/-/request-2.88.2.tgz", - "integrity": "sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw==", - "dev": true, - "requires": { - "aws-sign2": "~0.7.0", - "aws4": "^1.8.0", - "caseless": "~0.12.0", - "combined-stream": "~1.0.6", - "extend": "~3.0.2", - "forever-agent": "~0.6.1", - "form-data": "~2.3.2", - "har-validator": "~5.1.3", - "http-signature": "~1.2.0", - "is-typedarray": "~1.0.0", - "isstream": "~0.1.2", - "json-stringify-safe": "~5.0.1", - "mime-types": "~2.1.19", - "oauth-sign": "~0.9.0", - "performance-now": "^2.1.0", - "qs": "~6.5.2", - "safe-buffer": "^5.1.2", - "tough-cookie": "~2.5.0", - "tunnel-agent": "^0.6.0", - "uuid": "^3.3.2" - }, - "dependencies": { - "qs": { - "version": "6.5.2", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz", - "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==", - "dev": true - }, - "tough-cookie": { - "version": "2.5.0", - "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.5.0.tgz", - "integrity": "sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g==", - "dev": true, - "requires": { - "psl": "^1.1.28", - "punycode": "^2.1.1" - } - }, - "uuid": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", - "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", - "dev": true - } - } - }, - "request-promise-core": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/request-promise-core/-/request-promise-core-1.1.4.tgz", - "integrity": "sha512-TTbAfBBRdWD7aNNOoVOBH4pN/KigV6LyapYNNlAPA8JwbovRti1E88m3sYAwsLi5ryhPKsE9APwnjFTgdUjTpw==", - "dev": true, - "requires": { - "lodash": "^4.17.19" - } - }, - "request-promise-native": { - "version": "1.0.9", - "resolved": "https://registry.npmjs.org/request-promise-native/-/request-promise-native-1.0.9.tgz", - "integrity": "sha512-wcW+sIUiWnKgNY0dqCpOZkUbF/I+YPi+f09JZIDa39Ec+q82CpSYniDp+ISgTTbKmnpJWASeJBPZmoxH84wt3g==", - "dev": true, - "requires": { - "request-promise-core": "1.1.4", - "stealthy-require": "^1.1.1", - "tough-cookie": "^2.3.3" - }, - "dependencies": { - "tough-cookie": { - "version": "2.5.0", - "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.5.0.tgz", - "integrity": "sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g==", - "dev": true, - "requires": { - "psl": "^1.1.28", - "punycode": "^2.1.1" - } - } - } - }, - "require-directory": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", - "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=" - }, - "require-main-filename": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", - "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==" - }, - "resolve": { - "version": "1.19.0", - "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.19.0.tgz", - "integrity": "sha512-rArEXAgsBG4UgRGcynxWIWKFvh/XZCcS8UJdHhwy91zwAvCZIbcs+vAbflgBnNjYMs/i/i+/Ux6IZhML1yPvxg==", - "dev": true, - "requires": { - "is-core-module": "^2.1.0", - "path-parse": "^1.0.6" - } - }, - "resolve-cwd": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz", - "integrity": "sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==", - "dev": true, - "requires": { - "resolve-from": "^5.0.0" - } - }, - "resolve-from": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", - "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", - "dev": true - }, - "resolve-url": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/resolve-url/-/resolve-url-0.2.1.tgz", - "integrity": "sha1-LGN/53yJOv0qZj/iGqkIAGjiBSo=", - "dev": true - }, - "responselike": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz", - "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=", - "dev": true, - "requires": { - "lowercase-keys": "^1.0.0" - } - }, - "ret": { - "version": "0.1.15", - "resolved": "https://registry.npmjs.org/ret/-/ret-0.1.15.tgz", - "integrity": "sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg==", - "dev": true - }, - "reusify": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", - "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", - "dev": true - }, - "rimraf": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", - "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", - "requires": { - "glob": "^7.1.3" - } - }, - "rsvp": { - "version": "4.8.5", - "resolved": "https://registry.npmjs.org/rsvp/-/rsvp-4.8.5.tgz", - "integrity": "sha512-nfMOlASu9OnRJo1mbEk2cz0D56a1MBNrJ7orjRZQG10XDyuvwksKbuXNp6qa+kbn839HwjwhBzhFmdsaEAfauA==", - "dev": true - }, - "run-parallel": { - "version": "1.1.10", - "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.1.10.tgz", - "integrity": "sha512-zb/1OuZ6flOlH6tQyMPUrE3x3Ulxjlo9WIVXR4yVYi4H9UXQaeIsPbLn2R3O3vQCnDKkAl2qHiuocKKX4Tz/Sw==", - "dev": true - }, - "safe-buffer": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", - "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" - }, - "safe-regex": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/safe-regex/-/safe-regex-1.1.0.tgz", - "integrity": "sha1-QKNmnzsHfR6UPURinhV91IAjvy4=", - "dev": true, - "requires": { - "ret": "~0.1.10" - } - }, - "safer-buffer": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", - "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" - }, - "sane": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/sane/-/sane-4.1.0.tgz", - "integrity": "sha512-hhbzAgTIX8O7SHfp2c8/kREfEn4qO/9q8C9beyY6+tvZ87EpoZ3i1RIEvp27YBswnNbY9mWd6paKVmKbAgLfZA==", - "dev": true, - "requires": { - "@cnakazawa/watch": "^1.0.3", - "anymatch": "^2.0.0", - "capture-exit": "^2.0.0", - "exec-sh": "^0.3.2", - "execa": "^1.0.0", - "fb-watchman": "^2.0.0", - "micromatch": "^3.1.4", - "minimist": "^1.1.1", - "walker": "~1.0.5" - } - }, - "sax": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz", - "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==" - }, - "saxes": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/saxes/-/saxes-5.0.1.tgz", - "integrity": "sha512-5LBh1Tls8c9xgGjw3QrMwETmTMVk0oFgvrFSvWx62llR2hcEInrKNZ2GZCCuuy2lvWrdl5jhbpeqc5hRYKFOcw==", - "dev": true, - "requires": { - "xmlchars": "^2.2.0" - } - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" - }, - "semver-diff": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/semver-diff/-/semver-diff-3.1.1.tgz", - "integrity": "sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg==", - "dev": true, - "requires": { - "semver": "^6.3.0" - }, - "dependencies": { - "semver": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", - "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", - "dev": true - } - } - }, - "send": { - "version": "0.17.1", - "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", - "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", - "requires": { - "debug": "2.6.9", - "depd": "~1.1.2", - "destroy": "~1.0.4", - "encodeurl": "~1.0.2", - "escape-html": "~1.0.3", - "etag": "~1.8.1", - "fresh": "0.5.2", - "http-errors": "~1.7.2", - "mime": "1.6.0", - "ms": "2.1.1", - "on-finished": "~2.3.0", - "range-parser": "~1.2.1", - "statuses": "~1.5.0" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "requires": { - "ms": "2.0.0" - }, - "dependencies": { - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" - } - } - }, - "ms": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", - "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" - } - } - }, - "seq-queue": { - "version": "0.0.5", - "resolved": "https://registry.npmjs.org/seq-queue/-/seq-queue-0.0.5.tgz", - "integrity": "sha1-1WgS4cAXpuTnw+Ojeh2m143TyT4=" - }, - "serve-static": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", - "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", - "requires": { - "encodeurl": "~1.0.2", - "escape-html": "~1.0.3", - "parseurl": "~1.3.3", - "send": "0.17.1" - } - }, - "set-blocking": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", - "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=" - }, - "set-value": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/set-value/-/set-value-2.0.1.tgz", - "integrity": "sha512-JxHc1weCN68wRY0fhCoXpyK55m/XPHafOmK4UWD7m2CI14GMcFypt4w/0+NV5f/ZMby2F6S2wwA7fgynh9gWSw==", - "dev": true, - "requires": { - "extend-shallow": "^2.0.1", - "is-extendable": "^0.1.1", - "is-plain-object": "^2.0.3", - "split-string": "^3.0.1" - }, - "dependencies": { - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - } - } - }, - "setprototypeof": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", - "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" - }, - "sha.js": { - "version": "2.4.11", - "resolved": "https://registry.npmjs.org/sha.js/-/sha.js-2.4.11.tgz", - "integrity": "sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==", - "requires": { - "inherits": "^2.0.1", - "safe-buffer": "^5.0.1" - } - }, - "shebang-command": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", - "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", - "dev": true, - "requires": { - "shebang-regex": "^1.0.0" - } - }, - "shebang-regex": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", - "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", - "dev": true - }, - "shellwords": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/shellwords/-/shellwords-0.1.1.tgz", - "integrity": "sha512-vFwSUfQvqybiICwZY5+DAWIPLKsWO31Q91JSKl3UYv+K5c2QRPzn0qzec6QPu1Qc9eHYItiP3NdJqNVqetYAww==", - "dev": true, - "optional": true - }, - "shimmer": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/shimmer/-/shimmer-1.2.1.tgz", - "integrity": "sha512-sQTKC1Re/rM6XyFM6fIAGHRPVGvyXfgzIDvzoq608vM+jeyVD0Tu1E6Np0Kc2zAIFWIj963V2800iF/9LPieQw==" - }, - "signal-exit": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", - "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==" - }, - "sisteransi": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", - "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==", - "dev": true - }, - "slash": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/slash/-/slash-2.0.0.tgz", - "integrity": "sha512-ZYKh3Wh2z1PpEXWr0MpSBZ0V6mZHAQfYevttO11c51CaWjGTaadiKZ+wVt1PbMlDV5qhMFslpZCemhwOK7C89A==", - "dev": true - }, - "slice-ansi": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-2.1.0.tgz", - "integrity": "sha512-Qu+VC3EwYLldKa1fCxuuvULvSJOKEgk9pi8dZeCVK7TqBfUNTH4sFkk4joj8afVSfAYgJoSOetjx9QWOJ5mYoQ==", - "dev": true, - "requires": { - "ansi-styles": "^3.2.0", - "astral-regex": "^1.0.0", - "is-fullwidth-code-point": "^2.0.0" - }, - "dependencies": { - "ansi-styles": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", - "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", - "dev": true, - "requires": { - "color-convert": "^1.9.0" - } - }, - "color-convert": { - "version": "1.9.3", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", - "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", - "dev": true, - "requires": { - "color-name": "1.1.3" - } - }, - "color-name": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", - "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", - "dev": true - }, - "is-fullwidth-code-point": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", - "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", - "dev": true - } - } - }, - "snapdragon": { - "version": "0.8.2", - "resolved": "https://registry.npmjs.org/snapdragon/-/snapdragon-0.8.2.tgz", - "integrity": "sha512-FtyOnWN/wCHTVXOMwvSv26d+ko5vWlIDD6zoUJ7LW8vh+ZBC8QdljveRP+crNrtBwioEUWy/4dMtbBjA4ioNlg==", - "dev": true, - "requires": { - "base": "^0.11.1", - "debug": "^2.2.0", - "define-property": "^0.2.5", - "extend-shallow": "^2.0.1", - "map-cache": "^0.2.2", - "source-map": "^0.5.6", - "source-map-resolve": "^0.5.0", - "use": "^3.1.0" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "dev": true, - "requires": { - "ms": "2.0.0" - } - }, - "define-property": { - "version": "0.2.5", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, - "requires": { - "is-descriptor": "^0.1.0" - } - }, - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - } - } - }, - "snapdragon-node": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/snapdragon-node/-/snapdragon-node-2.1.1.tgz", - "integrity": "sha512-O27l4xaMYt/RSQ5TR3vpWCAB5Kb/czIcqUFOM/C4fYcLnbZUc1PkjTAMjof2pBWaSTwOUd6qUHcFGVGj7aIwnw==", - "dev": true, - "requires": { - "define-property": "^1.0.0", - "isobject": "^3.0.0", - "snapdragon-util": "^3.0.1" - }, - "dependencies": { - "define-property": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", - "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", - "dev": true, - "requires": { - "is-descriptor": "^1.0.0" - } - }, - "is-accessor-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", - "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-data-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", - "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-descriptor": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", - "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, - "requires": { - "is-accessor-descriptor": "^1.0.0", - "is-data-descriptor": "^1.0.0", - "kind-of": "^6.0.2" - } - } - } - }, - "snapdragon-util": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/snapdragon-util/-/snapdragon-util-3.0.1.tgz", - "integrity": "sha512-mbKkMdQKsjX4BAL4bRYTj21edOf8cN7XHdYUJEe+Zn99hVEYcMvKPct1IqNe7+AZPirn8BCDOQBHQZknqmKlZQ==", - "dev": true, - "requires": { - "kind-of": "^3.2.0" - }, - "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } - } - }, - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true - }, - "source-map-resolve": { - "version": "0.5.3", - "resolved": "https://registry.npmjs.org/source-map-resolve/-/source-map-resolve-0.5.3.tgz", - "integrity": "sha512-Htz+RnsXWk5+P2slx5Jh3Q66vhQj1Cllm0zvnaY98+NFx+Dv2CF/f5O/t8x+KaNdrdIAsruNzoh/KpialbqAnw==", - "dev": true, - "requires": { - "atob": "^2.1.2", - "decode-uri-component": "^0.2.0", - "resolve-url": "^0.2.1", - "source-map-url": "^0.4.0", - "urix": "^0.1.0" - } - }, - "source-map-support": { - "version": "0.5.19", - "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.19.tgz", - "integrity": "sha512-Wonm7zOCIJzBGQdB+thsPar0kYuCIzYvxZwlBa87yi/Mdjv7Tip2cyVbLj5o0cFPN4EVkuTwb3GDDyUx2DGnGw==", - "dev": true, - "requires": { - "buffer-from": "^1.0.0", - "source-map": "^0.6.0" - }, - "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } - } - }, - "source-map-url": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/source-map-url/-/source-map-url-0.4.0.tgz", - "integrity": "sha1-PpNdfd1zYxuXZZlW1VEo6HtQhKM=", - "dev": true - }, - "spdx-correct": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.1.1.tgz", - "integrity": "sha512-cOYcUWwhCuHCXi49RhFRCyJEK3iPj1Ziz9DpViV3tbZOwXD49QzIN3MpOLJNxh2qwq2lJJZaKMVw9qNi4jTC0w==", - "dev": true, - "requires": { - "spdx-expression-parse": "^3.0.0", - "spdx-license-ids": "^3.0.0" - } - }, - "spdx-exceptions": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.3.0.tgz", - "integrity": "sha512-/tTrYOC7PPI1nUAgx34hUpqXuyJG+DTHJTnIULG4rDygi4xu/tfgmq1e1cIRwRzwZgo4NLySi+ricLkZkw4i5A==", - "dev": true - }, - "spdx-expression-parse": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz", - "integrity": "sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==", - "dev": true, - "requires": { - "spdx-exceptions": "^2.1.0", - "spdx-license-ids": "^3.0.0" - } - }, - "spdx-license-ids": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.6.tgz", - "integrity": "sha512-+orQK83kyMva3WyPf59k1+Y525csj5JejicWut55zeTWANuN17qSiSLUXWtzHeNWORSvT7GLDJ/E/XiIWoXBTw==", - "dev": true - }, - "split-string": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/split-string/-/split-string-3.1.0.tgz", - "integrity": "sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw==", - "dev": true, - "requires": { - "extend-shallow": "^3.0.0" - } - }, - "sprintf-js": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", - "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=" - }, - "sqlite3": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/sqlite3/-/sqlite3-5.0.0.tgz", - "integrity": "sha512-rjvqHFUaSGnzxDy2AHCwhHy6Zp6MNJzCPGYju4kD8yi6bze4d1/zMTg6C7JI49b7/EM7jKMTvyfN/4ylBKdwfw==", - "dev": true, - "requires": { - "node-addon-api": "2.0.0", - "node-gyp": "3.x", - "node-pre-gyp": "^0.11.0" - }, - "dependencies": { - "node-addon-api": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-2.0.0.tgz", - "integrity": "sha512-ASCL5U13as7HhOExbT6OlWJJUV/lLzL2voOSP1UVehpRD8FbSrSDjfScK/KwAvVTI5AS6r4VwbOMlIqtvRidnA==", - "dev": true - }, - "node-pre-gyp": { - "version": "0.11.0", - "resolved": "https://registry.npmjs.org/node-pre-gyp/-/node-pre-gyp-0.11.0.tgz", - "integrity": "sha512-TwWAOZb0j7e9eGaf9esRx3ZcLaE5tQ2lvYy1pb5IAaG1a2e2Kv5Lms1Y4hpj+ciXJRofIxxlt5haeQ/2ANeE0Q==", - "dev": true, - "requires": { - "detect-libc": "^1.0.2", - "mkdirp": "^0.5.1", - "needle": "^2.2.1", - "nopt": "^4.0.1", - "npm-packlist": "^1.1.6", - "npmlog": "^4.0.2", - "rc": "^1.2.7", - "rimraf": "^2.6.1", - "semver": "^5.3.0", - "tar": "^4" - } - }, - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - } - } - }, - "sqlstring": { - "version": "2.3.2", - "resolved": "https://registry.npmjs.org/sqlstring/-/sqlstring-2.3.2.tgz", - "integrity": "sha512-vF4ZbYdKS8OnoJAWBmMxCQDkiEBkGQYU7UZPtL8flbDRSNkhaXvRJ279ZtI6M+zDaQovVU4tuRgzK5fVhvFAhg==" - }, - "sshpk": { - "version": "1.16.1", - "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.16.1.tgz", - "integrity": "sha512-HXXqVUq7+pcKeLqqZj6mHFUMvXtOJt1uoUx09pFW6011inTMxqI8BA8PM95myrIyyKwdnzjdFjLiE6KBPVtJIg==", - "dev": true, - "requires": { - "asn1": "~0.2.3", - "assert-plus": "^1.0.0", - "bcrypt-pbkdf": "^1.0.0", - "dashdash": "^1.12.0", - "ecc-jsbn": "~0.1.1", - "getpass": "^0.1.1", - "jsbn": "~0.1.0", - "safer-buffer": "^2.0.2", - "tweetnacl": "~0.14.0" - } - }, - "stack-chain": { - "version": "1.3.7", - "resolved": "https://registry.npmjs.org/stack-chain/-/stack-chain-1.3.7.tgz", - "integrity": "sha1-0ZLJ/06moiyUxN1FkXHj8AzqEoU=" - }, - "stack-utils": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/stack-utils/-/stack-utils-2.0.3.tgz", - "integrity": "sha512-gL//fkxfWUsIlFL2Tl42Cl6+HFALEaB1FU76I/Fy+oZjRreP7OPMXFlGbxM7NQsI0ZpUfw76sHnv0WNYuTb7Iw==", - "dev": true, - "requires": { - "escape-string-regexp": "^2.0.0" - }, - "dependencies": { - "escape-string-regexp": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-2.0.0.tgz", - "integrity": "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==", - "dev": true - } - } - }, - "static-extend": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/static-extend/-/static-extend-0.1.2.tgz", - "integrity": "sha1-YICcOcv/VTNyJv1eC1IPNB8ftcY=", - "dev": true, - "requires": { - "define-property": "^0.2.5", - "object-copy": "^0.1.0" - }, - "dependencies": { - "define-property": { - "version": "0.2.5", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, - "requires": { - "is-descriptor": "^0.1.0" - } - } - } - }, - "statuses": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", - "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" - }, - "stealthy-require": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/stealthy-require/-/stealthy-require-1.1.1.tgz", - "integrity": "sha1-NbCYdbT/SfJqd35QmzCQoyJr8ks=", - "dev": true - }, - "string-length": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/string-length/-/string-length-4.0.1.tgz", - "integrity": "sha512-PKyXUd0LK0ePjSOnWn34V2uD6acUWev9uy0Ft05k0E8xRW+SKcA0F7eMr7h5xlzfn+4O3N+55rduYyet3Jk+jw==", - "dev": true, - "requires": { - "char-regex": "^1.0.2", - "strip-ansi": "^6.0.0" - }, - "dependencies": { - "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", - "dev": true - }, - "strip-ansi": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", - "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", - "dev": true, - "requires": { - "ansi-regex": "^5.0.0" - } - } - } - }, - "string-width": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", - "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", - "requires": { - "code-point-at": "^1.0.0", - "is-fullwidth-code-point": "^1.0.0", - "strip-ansi": "^3.0.0" - } - }, - "string.prototype.trimend": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.2.tgz", - "integrity": "sha512-8oAG/hi14Z4nOVP0z6mdiVZ/wqjDtWSLygMigTzAb+7aPEDTleeFf+WrF+alzecxIRkckkJVn+dTlwzJXORATw==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.18.0-next.1" - }, - "dependencies": { - "es-abstract": { - "version": "1.18.0-next.1", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.18.0-next.1.tgz", - "integrity": "sha512-I4UGspA0wpZXWENrdA0uHbnhte683t3qT/1VFH9aX2dA5PPSf6QW5HHXf5HImaqPmjXaVeVk4RGWnaylmV7uAA==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-negative-zero": "^2.0.0", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - } - } - }, - "string.prototype.trimstart": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.2.tgz", - "integrity": "sha512-7F6CdBTl5zyu30BJFdzSTlSlLPwODC23Od+iLoVH8X6+3fvDPPuBVVj9iaB1GOsSTSIgVfsfm27R2FGrAPznWg==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.18.0-next.1" - }, - "dependencies": { - "es-abstract": { - "version": "1.18.0-next.1", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.18.0-next.1.tgz", - "integrity": "sha512-I4UGspA0wpZXWENrdA0uHbnhte683t3qT/1VFH9aX2dA5PPSf6QW5HHXf5HImaqPmjXaVeVk4RGWnaylmV7uAA==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-negative-zero": "^2.0.0", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - } - } - }, - "string_decoder": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", - "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", - "requires": { - "safe-buffer": "~5.1.0" - } - }, - "strip-ansi": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", - "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", - "requires": { - "ansi-regex": "^2.0.0" - } - }, - "strip-bom": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-4.0.0.tgz", - "integrity": "sha512-3xurFv5tEgii33Zi8Jtp55wEIILR9eh34FAW00PZf+JnSsTmV/ioewSgQl97JHvgjoRGwPShsWm+IdrxB35d0w==", - "dev": true - }, - "strip-eof": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/strip-eof/-/strip-eof-1.0.0.tgz", - "integrity": "sha1-u0P/VZim6wXYm1n80SnJgzE2Br8=", - "dev": true - }, - "strip-final-newline": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-2.0.0.tgz", - "integrity": "sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==", - "dev": true - }, - "strip-json-comments": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", - "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=" - }, - "superagent": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/superagent/-/superagent-6.1.0.tgz", - "integrity": "sha512-OUDHEssirmplo3F+1HWKUrUjvnQuA+nZI6i/JJBdXb5eq9IyEQwPyPpqND+SSsxf6TygpBEkUjISVRN4/VOpeg==", - "dev": true, - "requires": { - "component-emitter": "^1.3.0", - "cookiejar": "^2.1.2", - "debug": "^4.1.1", - "fast-safe-stringify": "^2.0.7", - "form-data": "^3.0.0", - "formidable": "^1.2.2", - "methods": "^1.1.2", - "mime": "^2.4.6", - "qs": "^6.9.4", - "readable-stream": "^3.6.0", - "semver": "^7.3.2" - }, - "dependencies": { - "debug": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", - "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", - "dev": true, - "requires": { - "ms": "2.1.2" - } - }, - "form-data": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/form-data/-/form-data-3.0.0.tgz", - "integrity": "sha512-CKMFDglpbMi6PyN+brwB9Q/GOw0eAnsrEZDgcsH5Krhz5Od/haKHAX0NmQfha2zPPz0JpWzA7GJHGSnvCRLWsg==", - "dev": true, - "requires": { - "asynckit": "^0.4.0", - "combined-stream": "^1.0.8", - "mime-types": "^2.1.12" - } - }, - "mime": { - "version": "2.4.6", - "resolved": "https://registry.npmjs.org/mime/-/mime-2.4.6.tgz", - "integrity": "sha512-RZKhC3EmpBchfTGBVb8fb+RL2cWyw/32lshnsETttkBAyAUXSGHxbEJWWRXc751DrIxG1q04b8QwMbAwkRPpUA==", - "dev": true - }, - "qs": { - "version": "6.9.4", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.9.4.tgz", - "integrity": "sha512-A1kFqHekCTM7cz0udomYUoYNWjBebHm/5wzU/XqrBRBNWectVH0QIiN+NEcZ0Dte5hvzHwbr8+XQmguPhJ6WdQ==", - "dev": true - }, - "readable-stream": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", - "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", - "dev": true, - "requires": { - "inherits": "^2.0.3", - "string_decoder": "^1.1.1", - "util-deprecate": "^1.0.1" - } - }, - "semver": { - "version": "7.3.2", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", - "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", - "dev": true - } - } - }, - "supertest": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/supertest/-/supertest-6.0.1.tgz", - "integrity": "sha512-8yDNdm+bbAN/jeDdXsRipbq9qMpVF7wRsbwLgsANHqdjPsCoecmlTuqEcLQMGpmojFBhxayZ0ckXmLXYq7e+0g==", - "dev": true, - "requires": { - "methods": "1.1.2", - "superagent": "6.1.0" - } - }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", - "requires": { - "has-flag": "^4.0.0" - } - }, - "supports-hyperlinks": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.1.0.tgz", - "integrity": "sha512-zoE5/e+dnEijk6ASB6/qrK+oYdm2do1hjoLWrqUC/8WEIW1gbxFcKuBof7sW8ArN6e+AYvsE8HBGiVRWL/F5CA==", - "dev": true, - "requires": { - "has-flag": "^4.0.0", - "supports-color": "^7.0.0" - } - }, - "symbol-tree": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/symbol-tree/-/symbol-tree-3.2.4.tgz", - "integrity": "sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw==", - "dev": true - }, - "table": { - "version": "5.4.6", - "resolved": "https://registry.npmjs.org/table/-/table-5.4.6.tgz", - "integrity": "sha512-wmEc8m4fjnob4gt5riFRtTu/6+4rSe12TpAELNSqHMfF3IqnA+CH37USM6/YR3qRZv7e56kAEAtd6nKZaxe0Ug==", - "dev": true, - "requires": { - "ajv": "^6.10.2", - "lodash": "^4.17.14", - "slice-ansi": "^2.1.0", - "string-width": "^3.0.0" - }, - "dependencies": { - "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", - "dev": true - }, - "emoji-regex": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", - "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", - "dev": true - }, - "is-fullwidth-code-point": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", - "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", - "dev": true - }, - "string-width": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", - "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", - "dev": true, - "requires": { - "emoji-regex": "^7.0.1", - "is-fullwidth-code-point": "^2.0.0", - "strip-ansi": "^5.1.0" - } - }, - "strip-ansi": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", - "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", - "dev": true, - "requires": { - "ansi-regex": "^4.1.0" - } - } - } - }, - "tar": { - "version": "4.4.13", - "resolved": "https://registry.npmjs.org/tar/-/tar-4.4.13.tgz", - "integrity": "sha512-w2VwSrBoHa5BsSyH+KxEqeQBAllHhccyMFVHtGtdMpF4W7IRWfZjFiQceJPChOeTsSDVUpER2T8FA93pr0L+QA==", - "requires": { - "chownr": "^1.1.1", - "fs-minipass": "^1.2.5", - "minipass": "^2.8.6", - "minizlib": "^1.2.1", - "mkdirp": "^0.5.0", - "safe-buffer": "^5.1.2", - "yallist": "^3.0.3" - } - }, - "term-size": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/term-size/-/term-size-2.2.1.tgz", - "integrity": "sha512-wK0Ri4fOGjv/XPy8SBHZChl8CM7uMc5VML7SqiQ0zG7+J5Vr+RMQDoHa2CNT6KHUnTGIXH34UDMkPzAUyapBZg==", - "dev": true - }, - "terminal-link": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/terminal-link/-/terminal-link-2.1.1.tgz", - "integrity": "sha512-un0FmiRUQNr5PJqy9kP7c40F5BOfpGlYTrxonDChEZB7pzZxRNp/bt+ymiy9/npwXya9KH99nJ/GXFIiUkYGFQ==", - "dev": true, - "requires": { - "ansi-escapes": "^4.2.1", - "supports-hyperlinks": "^2.0.0" - } - }, - "test-exclude": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-6.0.0.tgz", - "integrity": "sha512-cAGWPIyOHU6zlmg88jwm7VRyXnMN7iV68OGAbYDk/Mh/xC/pzVPlQtY6ngoIH/5/tciuhGfvESU8GrHrcxD56w==", - "dev": true, - "requires": { - "@istanbuljs/schema": "^0.1.2", - "glob": "^7.1.4", - "minimatch": "^3.0.4" - } - }, - "text-table": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", - "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=", - "dev": true - }, - "thenify": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/thenify/-/thenify-3.3.1.tgz", - "integrity": "sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==", - "requires": { - "any-promise": "^1.0.0" - } - }, - "thenify-all": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/thenify-all/-/thenify-all-1.6.0.tgz", - "integrity": "sha1-GhkY1ALY/D+Y+/I02wvMjMEOlyY=", - "requires": { - "thenify": ">= 3.1.0 < 4" - } - }, - "throat": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/throat/-/throat-5.0.0.tgz", - "integrity": "sha512-fcwX4mndzpLQKBS1DVYhGAcYaYt7vsHNIvQV+WXMvnow5cgjPphq5CaayLaGsjRdSCKZFNGt7/GYAuXaNOiYCA==", - "dev": true - }, - "tmpl": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/tmpl/-/tmpl-1.0.4.tgz", - "integrity": "sha1-I2QN17QtAEM5ERQIIOXPRA5SHdE=", - "dev": true - }, - "to-fast-properties": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", - "integrity": "sha1-3F5pjL0HkmW8c+A3doGk5Og/YW4=", - "dev": true - }, - "to-object-path": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/to-object-path/-/to-object-path-0.3.0.tgz", - "integrity": "sha1-KXWIt7Dn4KwI4E5nL4XB9JmeF68=", - "dev": true, - "requires": { - "kind-of": "^3.0.2" - }, - "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } - } - }, - "to-readable-stream": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/to-readable-stream/-/to-readable-stream-1.0.0.tgz", - "integrity": "sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q==", - "dev": true - }, - "to-regex": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/to-regex/-/to-regex-3.0.2.tgz", - "integrity": "sha512-FWtleNAtZ/Ki2qtqej2CXTOayOH9bHDQF+Q48VpWyDXjbYxA4Yz8iDB31zXOBUlOHHKidDbqGVrTUvQMPmBGBw==", - "dev": true, - "requires": { - "define-property": "^2.0.2", - "extend-shallow": "^3.0.2", - "regex-not": "^1.0.2", - "safe-regex": "^1.1.0" - } - }, - "to-regex-range": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", - "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", - "dev": true, - "requires": { - "is-number": "^3.0.0", - "repeat-string": "^1.6.1" - } - }, - "toidentifier": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", - "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" - }, - "touch": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz", - "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==", - "dev": true, - "requires": { - "nopt": "~1.0.10" - }, - "dependencies": { - "nopt": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz", - "integrity": "sha1-bd0hvSoxQXuScn3Vhfim83YI6+4=", - "dev": true, - "requires": { - "abbrev": "1" - } - } - } - }, - "tough-cookie": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-3.0.1.tgz", - "integrity": "sha512-yQyJ0u4pZsv9D4clxO69OEjLWYw+jbgspjTue4lTQZLfV0c5l1VmK2y1JK8E9ahdpltPOaAThPcp5nKPUgSnsg==", - "dev": true, - "requires": { - "ip-regex": "^2.1.0", - "psl": "^1.1.28", - "punycode": "^2.1.1" - } - }, - "tr46": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/tr46/-/tr46-2.0.2.tgz", - "integrity": "sha512-3n1qG+/5kg+jrbTzwAykB5yRYtQCTqOGKq5U5PE3b0a1/mzo6snDhjGS0zJVJunO0NrT3Dg1MLy5TjWP/UJppg==", - "dev": true, - "requires": { - "punycode": "^2.1.1" - } - }, - "ts-jest": { - "version": "26.4.4", - "resolved": "https://registry.npmjs.org/ts-jest/-/ts-jest-26.4.4.tgz", - "integrity": "sha512-3lFWKbLxJm34QxyVNNCgXX1u4o/RV0myvA2y2Bxm46iGIjKlaY0own9gIckbjZJPn+WaJEnfPPJ20HHGpoq4yg==", - "dev": true, - "requires": { - "@types/jest": "26.x", - "bs-logger": "0.x", - "buffer-from": "1.x", - "fast-json-stable-stringify": "2.x", - "jest-util": "^26.1.0", - "json5": "2.x", - "lodash.memoize": "4.x", - "make-error": "1.x", - "mkdirp": "1.x", - "semver": "7.x", - "yargs-parser": "20.x" - }, - "dependencies": { - "mkdirp": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", - "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", - "dev": true - }, - "semver": { - "version": "7.3.2", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", - "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", - "dev": true - }, - "yargs-parser": { - "version": "20.2.4", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.4.tgz", - "integrity": "sha512-WOkpgNhPTlE73h4VFAFsOnomJVaovO8VqLDzy5saChRBFQFBoMYirowyW+Q9HB4HFF4Z7VZTiG3iSzJJA29yRA==", - "dev": true - } - } - }, - "tsconfig-paths": { - "version": "3.9.0", - "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.9.0.tgz", - "integrity": "sha512-dRcuzokWhajtZWkQsDVKbWyY+jgcLC5sqJhg2PSgf4ZkH2aHPvaOY8YWGhmjb68b5qqTfasSsDO9k7RUiEmZAw==", - "dev": true, - "requires": { - "@types/json5": "^0.0.29", - "json5": "^1.0.1", - "minimist": "^1.2.0", - "strip-bom": "^3.0.0" - }, - "dependencies": { - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", - "dev": true, - "requires": { - "minimist": "^1.2.0" - } - }, - "strip-bom": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", - "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", - "dev": true - } - } - }, - "tslib": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.0.3.tgz", - "integrity": "sha512-uZtkfKblCEQtZKBF6EBXVZeQNl82yqtDQdv+eck8u7tdPxjLu2/lp5/uPW+um2tpuxINHWy3GhiccY7QgEaVHQ==" - }, - "tsutils": { - "version": "3.17.1", - "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-3.17.1.tgz", - "integrity": "sha512-kzeQ5B8H3w60nFY2g8cJIuH7JDpsALXySGtwGJ0p2LSjLgay3NdIpqq5SoOBe46bKDW2iq25irHCr8wjomUS2g==", - "dev": true, - "requires": { - "tslib": "^1.8.1" - }, - "dependencies": { - "tslib": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", - "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", - "dev": true - } - } - }, - "tunnel-agent": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", - "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", - "dev": true, - "requires": { - "safe-buffer": "^5.0.1" - } - }, - "tweetnacl": { - "version": "0.14.5", - "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz", - "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=", - "dev": true - }, - "type-check": { - "version": "0.3.2", - "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", - "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=", - "dev": true, - "requires": { - "prelude-ls": "~1.1.2" - } - }, - "type-detect": { - "version": "4.0.8", - "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz", - "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==", - "dev": true - }, - "type-fest": { - "version": "0.8.1", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", - "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", - "dev": true - }, - "type-is": { - "version": "1.6.18", - "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", - "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", - "requires": { - "media-typer": "0.3.0", - "mime-types": "~2.1.24" - } - }, - "typedarray-to-buffer": { - "version": "3.1.5", - "resolved": "https://registry.npmjs.org/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz", - "integrity": "sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q==", - "dev": true, - "requires": { - "is-typedarray": "^1.0.0" - } - }, - "typeorm": { - "version": "0.2.29", - "resolved": "https://registry.npmjs.org/typeorm/-/typeorm-0.2.29.tgz", - "integrity": "sha512-ih1vrTe3gEAGKRcWlcsTRxTL7gNjacQE498wVGuJ3ZRujtMqPZlbAWuC7xDzWCRjQnkZYNwZQeG9UgKfxSHB5g==", - "requires": { - "@sqltools/formatter": "1.2.2", - "app-root-path": "^3.0.0", - "buffer": "^5.5.0", - "chalk": "^4.1.0", - "cli-highlight": "^2.1.4", - "debug": "^4.1.1", - "dotenv": "^8.2.0", - "glob": "^7.1.6", - "js-yaml": "^3.14.0", - "mkdirp": "^1.0.4", - "reflect-metadata": "^0.1.13", - "sha.js": "^2.4.11", - "tslib": "^1.13.0", - "xml2js": "^0.4.23", - "yargonaut": "^1.1.2", - "yargs": "^16.0.3" - }, - "dependencies": { - "debug": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", - "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", - "requires": { - "ms": "2.1.2" - } - }, - "mkdirp": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", - "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==" - }, - "tslib": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", - "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==" - } - } - }, - "typeorm-transactional-cls-hooked": { - "version": "0.1.12", - "resolved": "https://registry.npmjs.org/typeorm-transactional-cls-hooked/-/typeorm-transactional-cls-hooked-0.1.12.tgz", - "integrity": "sha512-kCKYVzmE+re8AbMCUR2hgXYFjhsTUo2L0r2ypo1mU8KY7WG+7S5o2LDhONX3OYJ12V81SOWHHmxGRwU82nYtWg==", - "requires": { - "@types/cls-hooked": "^4.2.1", - "cls-hooked": "^4.2.2" - } - }, - "typescript": { - "version": "4.0.5", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.0.5.tgz", - "integrity": "sha512-ywmr/VrTVCmNTJ6iV2LwIrfG1P+lv6luD8sUJs+2eI9NLGigaN+nUQc13iHqisq7bra9lnmUSYqbJvegraBOPQ==" - }, - "undefsafe": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.3.tgz", - "integrity": "sha512-nrXZwwXrD/T/JXeygJqdCO6NZZ1L66HrxM/Z7mIq2oPanoN0F1nLx3lwJMu6AwJY69hdixaFQOuoYsMjE5/C2A==", - "dev": true, - "requires": { - "debug": "^2.2.0" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "dev": true, - "requires": { - "ms": "2.0.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - } - } - }, - "unicode-canonical-property-names-ecmascript": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-1.0.4.tgz", - "integrity": "sha512-jDrNnXWHd4oHiTZnx/ZG7gtUTVp+gCcTTKr8L0HjlwphROEW3+Him+IpvC+xcJEFegapiMZyZe02CyuOnRmbnQ==", - "dev": true - }, - "unicode-match-property-ecmascript": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/unicode-match-property-ecmascript/-/unicode-match-property-ecmascript-1.0.4.tgz", - "integrity": "sha512-L4Qoh15vTfntsn4P1zqnHulG0LdXgjSO035fEpdtp6YxXhMT51Q6vgM5lYdG/5X3MjS+k/Y9Xw4SFCY9IkR0rg==", - "dev": true, - "requires": { - "unicode-canonical-property-names-ecmascript": "^1.0.4", - "unicode-property-aliases-ecmascript": "^1.0.4" - } - }, - "unicode-match-property-value-ecmascript": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/unicode-match-property-value-ecmascript/-/unicode-match-property-value-ecmascript-1.2.0.tgz", - "integrity": "sha512-wjuQHGQVofmSJv1uVISKLE5zO2rNGzM/KCYZch/QQvez7C1hUhBIuZ701fYXExuufJFMPhv2SyL8CyoIfMLbIQ==", - "dev": true - }, - "unicode-property-aliases-ecmascript": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/unicode-property-aliases-ecmascript/-/unicode-property-aliases-ecmascript-1.1.0.tgz", - "integrity": "sha512-PqSoPh/pWetQ2phoj5RLiaqIk4kCNwoV3CI+LfGmWLKI3rE3kl1h59XpX2BjgDrmbxD9ARtQobPGU1SguCYuQg==", - "dev": true - }, - "union-value": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/union-value/-/union-value-1.0.1.tgz", - "integrity": "sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg==", - "dev": true, - "requires": { - "arr-union": "^3.1.0", - "get-value": "^2.0.6", - "is-extendable": "^0.1.1", - "set-value": "^2.0.1" - } - }, - "unique-string": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/unique-string/-/unique-string-2.0.0.tgz", - "integrity": "sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg==", - "dev": true, - "requires": { - "crypto-random-string": "^2.0.0" - } - }, - "unpipe": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", - "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" - }, - "unset-value": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/unset-value/-/unset-value-1.0.0.tgz", - "integrity": "sha1-g3aHP30jNRef+x5vw6jtDfyKtVk=", - "dev": true, - "requires": { - "has-value": "^0.3.1", - "isobject": "^3.0.0" - }, - "dependencies": { - "has-value": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/has-value/-/has-value-0.3.1.tgz", - "integrity": "sha1-ex9YutpiyoJ+wKIHgCVlSEWZXh8=", - "dev": true, - "requires": { - "get-value": "^2.0.3", - "has-values": "^0.1.4", - "isobject": "^2.0.0" - }, - "dependencies": { - "isobject": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", - "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=", - "dev": true, - "requires": { - "isarray": "1.0.0" - } - } - } - }, - "has-values": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/has-values/-/has-values-0.1.4.tgz", - "integrity": "sha1-bWHeldkd/Km5oCCJrThL/49it3E=", - "dev": true - } - } - }, - "upath": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/upath/-/upath-1.2.0.tgz", - "integrity": "sha512-aZwGpamFO61g3OlfT7OQCHqhGnW43ieH9WZeP7QxN/G/jS4jfqUkZxoryvJgVPEcrl5NL/ggHsSmLMHuH64Lhg==", - "dev": true, - "optional": true - }, - "update-notifier": { - "version": "4.1.3", - "resolved": "https://registry.npmjs.org/update-notifier/-/update-notifier-4.1.3.tgz", - "integrity": "sha512-Yld6Z0RyCYGB6ckIjffGOSOmHXj1gMeE7aROz4MG+XMkmixBX4jUngrGXNYz7wPKBmtoD4MnBa2Anu7RSKht/A==", - "dev": true, - "requires": { - "boxen": "^4.2.0", - "chalk": "^3.0.0", - "configstore": "^5.0.1", - "has-yarn": "^2.1.0", - "import-lazy": "^2.1.0", - "is-ci": "^2.0.0", - "is-installed-globally": "^0.3.1", - "is-npm": "^4.0.0", - "is-yarn-global": "^0.3.0", - "latest-version": "^5.0.0", - "pupa": "^2.0.1", - "semver-diff": "^3.1.1", - "xdg-basedir": "^4.0.0" - }, - "dependencies": { - "chalk": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", - "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", - "dev": true, - "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" - } - } - } - }, - "uri-js": { - "version": "4.4.0", - "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.0.tgz", - "integrity": "sha512-B0yRTzYdUCCn9n+F4+Gh4yIDtMQcaJsmYBDsTSG8g/OejKBodLQ2IHfN3bM7jUsRXndopT7OIXWdYqc1fjmV6g==", - "dev": true, - "requires": { - "punycode": "^2.1.0" - } - }, - "urix": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/urix/-/urix-0.1.0.tgz", - "integrity": "sha1-2pN/emLiH+wf0Y1Js1wpNQZ6bHI=", - "dev": true - }, - "url-parse-lax": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz", - "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=", - "dev": true, - "requires": { - "prepend-http": "^2.0.0" - } - }, - "use": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/use/-/use-3.1.1.tgz", - "integrity": "sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ==", - "dev": true - }, - "util-deprecate": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", - "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=" - }, - "utils-merge": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", - "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" - }, - "uuid": { - "version": "8.3.1", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.1.tgz", - "integrity": "sha512-FOmRr+FmWEIG8uhZv6C2bTgEVXsHk08kE7mPlrBbEe+c3r9pjceVPgupIfNIhc4yx55H69OXANrUaSuu9eInKg==", - "dev": true, - "optional": true - }, - "v8-compile-cache": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.2.0.tgz", - "integrity": "sha512-gTpR5XQNKFwOd4clxfnhaqvfqMpqEwr4tOtCyz4MtYZX2JYhfr1JvBFKdS+7K/9rfpZR3VLX+YWBbKoxCgS43Q==", - "dev": true - }, - "v8-to-istanbul": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/v8-to-istanbul/-/v8-to-istanbul-7.0.0.tgz", - "integrity": "sha512-fLL2rFuQpMtm9r8hrAV2apXX/WqHJ6+IC4/eQVdMDGBUgH/YMV4Gv3duk3kjmyg6uiQWBAA9nJwue4iJUOkHeA==", - "dev": true, - "requires": { - "@types/istanbul-lib-coverage": "^2.0.1", - "convert-source-map": "^1.6.0", - "source-map": "^0.7.3" - }, - "dependencies": { - "source-map": { - "version": "0.7.3", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.3.tgz", - "integrity": "sha512-CkCj6giN3S+n9qrYiBTX5gystlENnRW5jZeNLHpe6aue+SrHcG5VYwujhW9s4dY31mEGsxBDrHR6oI69fTXsaQ==", - "dev": true - } - } - }, - "v8flags": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/v8flags/-/v8flags-3.2.0.tgz", - "integrity": "sha512-mH8etigqMfiGWdeXpaaqGfs6BndypxusHHcv2qSHyZkGEznCd/qAXCWWRzeowtL54147cktFOC4P5y+kl8d8Jg==", - "dev": true, - "requires": { - "homedir-polyfill": "^1.0.1" - } - }, - "validate-npm-package-license": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz", - "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==", - "dev": true, - "requires": { - "spdx-correct": "^3.0.0", - "spdx-expression-parse": "^3.0.0" - } - }, - "validator": { - "version": "13.0.0", - "resolved": "https://registry.npmjs.org/validator/-/validator-13.0.0.tgz", - "integrity": "sha512-anYx5fURbgF04lQV18nEQWZ/3wHGnxiKdG4aL8J+jEDsm98n/sU/bey+tYk6tnGJzm7ioh5FoqrAiQ6m03IgaA==" - }, - "vary": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", - "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" - }, - "verror": { - "version": "1.10.0", - "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", - "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", - "dev": true, - "requires": { - "assert-plus": "^1.0.0", - "core-util-is": "1.0.2", - "extsprintf": "^1.2.0" - } - }, - "w3c-hr-time": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/w3c-hr-time/-/w3c-hr-time-1.0.2.tgz", - "integrity": "sha512-z8P5DvDNjKDoFIHK7q8r8lackT6l+jo/Ye3HOle7l9nICP9lf1Ci25fy9vHd0JOWewkIFzXIEig3TdKT7JQ5fQ==", - "dev": true, - "requires": { - "browser-process-hrtime": "^1.0.0" - } - }, - "w3c-xmlserializer": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/w3c-xmlserializer/-/w3c-xmlserializer-2.0.0.tgz", - "integrity": "sha512-4tzD0mF8iSiMiNs30BiLO3EpfGLZUT2MSX/G+o7ZywDzliWQ3OPtTZ0PTC3B3ca1UAf4cJMHB+2Bf56EriJuRA==", - "dev": true, - "requires": { - "xml-name-validator": "^3.0.0" - } - }, - "walker": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/walker/-/walker-1.0.7.tgz", - "integrity": "sha1-L3+bj9ENZ3JisYqITijRlhjgKPs=", - "dev": true, - "requires": { - "makeerror": "1.0.x" - } - }, - "webidl-conversions": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-6.1.0.tgz", - "integrity": "sha512-qBIvFLGiBpLjfwmYAaHPXsn+ho5xZnGvyGvsarywGNc8VyQJUMHJ8OBKGGrPER0okBeMDaan4mNBlgBROxuI8w==", - "dev": true - }, - "whatwg-encoding": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/whatwg-encoding/-/whatwg-encoding-1.0.5.tgz", - "integrity": "sha512-b5lim54JOPN9HtzvK9HFXvBma/rnfFeqsic0hSpjtDbVxR3dJKLc+KB4V6GgiGOvl7CY/KNh8rxSo9DKQrnUEw==", - "dev": true, - "requires": { - "iconv-lite": "0.4.24" - } - }, - "whatwg-mimetype": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-2.3.0.tgz", - "integrity": "sha512-M4yMwr6mAnQz76TbJm914+gPpB/nCwvZbJU28cUD6dR004SAxDLOOSUaB1JDRqLtaOV/vi0IC5lEAGFgrjGv/g==", - "dev": true - }, - "whatwg-url": { - "version": "8.4.0", - "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-8.4.0.tgz", - "integrity": "sha512-vwTUFf6V4zhcPkWp/4CQPr1TW9Ml6SF4lVyaIMBdJw5i6qUUJ1QWM4Z6YYVkfka0OUIzVo/0aNtGVGk256IKWw==", - "dev": true, - "requires": { - "lodash.sortby": "^4.7.0", - "tr46": "^2.0.2", - "webidl-conversions": "^6.1.0" - } - }, - "which": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", - "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", - "dev": true, - "requires": { - "isexe": "^2.0.0" - } - }, - "which-module": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", - "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=" - }, - "wide-align": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz", - "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==", - "requires": { - "string-width": "^1.0.2 || 2" - } - }, - "widest-line": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz", - "integrity": "sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg==", - "dev": true, - "requires": { - "string-width": "^4.0.0" - }, - "dependencies": { - "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", - "dev": true - }, - "is-fullwidth-code-point": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", - "dev": true - }, - "string-width": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", - "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", - "dev": true, - "requires": { - "emoji-regex": "^8.0.0", - "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.0" - } - }, - "strip-ansi": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", - "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", - "dev": true, - "requires": { - "ansi-regex": "^5.0.0" - } - } - } - }, - "word-wrap": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz", - "integrity": "sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ==", - "dev": true - }, - "wrap-ansi": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", - "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", - "requires": { - "ansi-styles": "^4.0.0", - "string-width": "^4.1.0", - "strip-ansi": "^6.0.0" - }, - "dependencies": { - "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==" - }, - "is-fullwidth-code-point": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==" - }, - "string-width": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", - "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", - "requires": { - "emoji-regex": "^8.0.0", - "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.0" - } - }, - "strip-ansi": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", - "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", - "requires": { - "ansi-regex": "^5.0.0" - } - } - } - }, - "wrappy": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", - "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" - }, - "write": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/write/-/write-1.0.3.tgz", - "integrity": "sha512-/lg70HAjtkUgWPVZhZcm+T4hkL8Zbtp1nFNOn3lRrxnlv50SRBv7cR7RqR+GMsd3hUXy9hWBo4CHTbFTcOYwig==", - "dev": true, - "requires": { - "mkdirp": "^0.5.1" - } - }, - "write-file-atomic": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", - "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", - "dev": true, - "requires": { - "imurmurhash": "^0.1.4", - "is-typedarray": "^1.0.0", - "signal-exit": "^3.0.2", - "typedarray-to-buffer": "^3.1.5" - } - }, - "ws": { - "version": "7.4.0", - "resolved": "https://registry.npmjs.org/ws/-/ws-7.4.0.tgz", - "integrity": "sha512-kyFwXuV/5ymf+IXhS6f0+eAFvydbaBW3zjpT6hUdAh/hbVjTIB5EHBGi0bPoCLSK2wcuz3BrEkB9LrYv1Nm4NQ==", - "dev": true - }, - "xdg-basedir": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/xdg-basedir/-/xdg-basedir-4.0.0.tgz", - "integrity": "sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q==", - "dev": true - }, - "xml-name-validator": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-3.0.0.tgz", - "integrity": "sha512-A5CUptxDsvxKJEU3yO6DuWBSJz/qizqzJKOMIfUJHETbBw/sFaDxgd6fxm1ewUaM0jZ444Fc5vC5ROYurg/4Pw==", - "dev": true - }, - "xml2js": { - "version": "0.4.23", - "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.4.23.tgz", - "integrity": "sha512-ySPiMjM0+pLDftHgXY4By0uswI3SPKLDw/i3UXbnO8M/p28zqexCUoPmQFrYD+/1BzhGJSs2i1ERWKJAtiLrug==", - "requires": { - "sax": ">=0.6.0", - "xmlbuilder": "~11.0.0" - } - }, - "xmlbuilder": { - "version": "11.0.1", - "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-11.0.1.tgz", - "integrity": "sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==" - }, - "xmlchars": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/xmlchars/-/xmlchars-2.2.0.tgz", - "integrity": "sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==", - "dev": true - }, - "y18n": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.0.tgz", - "integrity": "sha512-r9S/ZyXu/Xu9q1tYlpsLIsa3EeLXXk0VwlxqTcFRfg9EhMW+17kbt9G0NrgCmhGb5vT2hyhJZLfDGx+7+5Uj/w==" - }, - "yallist": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", - "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==" - }, - "yargonaut": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/yargonaut/-/yargonaut-1.1.4.tgz", - "integrity": "sha512-rHgFmbgXAAzl+1nngqOcwEljqHGG9uUZoPjsdZEs1w5JW9RXYzrSvH/u70C1JE5qFi0qjsdhnUX/dJRpWqitSA==", - "requires": { - "chalk": "^1.1.1", - "figlet": "^1.1.1", - "parent-require": "^1.0.0" - }, - "dependencies": { - "ansi-styles": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", - "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=" - }, - "chalk": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", - "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", - "requires": { - "ansi-styles": "^2.2.1", - "escape-string-regexp": "^1.0.2", - "has-ansi": "^2.0.0", - "strip-ansi": "^3.0.0", - "supports-color": "^2.0.0" - } - }, - "supports-color": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", - "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=" - } - } - }, - "yargs": { - "version": "16.1.1", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.1.1.tgz", - "integrity": "sha512-hAD1RcFP/wfgfxgMVswPE+z3tlPFtxG8/yWUrG2i17sTWGCGqWnxKcLTF4cUKDUK8fzokwsmO9H0TDkRbMHy8w==", - "requires": { - "cliui": "^7.0.2", - "escalade": "^3.1.1", - "get-caller-file": "^2.0.5", - "require-directory": "^2.1.1", - "string-width": "^4.2.0", - "y18n": "^5.0.5", - "yargs-parser": "^20.2.2" - }, - "dependencies": { - "ansi-regex": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==" - }, - "cliui": { - "version": "7.0.4", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", - "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", - "requires": { - "string-width": "^4.2.0", - "strip-ansi": "^6.0.0", - "wrap-ansi": "^7.0.0" - } - }, - "is-fullwidth-code-point": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==" - }, - "string-width": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", - "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", - "requires": { - "emoji-regex": "^8.0.0", - "is-fullwidth-code-point": "^3.0.0", - "strip-ansi": "^6.0.0" - } - }, - "strip-ansi": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", - "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", - "requires": { - "ansi-regex": "^5.0.0" - } - }, - "wrap-ansi": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", - "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", - "requires": { - "ansi-styles": "^4.0.0", - "string-width": "^4.1.0", - "strip-ansi": "^6.0.0" - } - }, - "y18n": { - "version": "5.0.5", - "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.5.tgz", - "integrity": "sha512-hsRUr4FFrvhhRH12wOdfs38Gy7k2FFzB9qgN9v3aLykRq0dRcdcpz5C9FxdS2NuhOrI/628b/KSTJ3rwHysYSg==" - }, - "yargs-parser": { - "version": "20.2.4", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.4.tgz", - "integrity": "sha512-WOkpgNhPTlE73h4VFAFsOnomJVaovO8VqLDzy5saChRBFQFBoMYirowyW+Q9HB4HFF4Z7VZTiG3iSzJJA29yRA==" - } - } - }, - "yargs-parser": { - "version": "18.1.3", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-18.1.3.tgz", - "integrity": "sha512-o50j0JeToy/4K6OZcaQmW6lyXXKhq7csREXcDwk2omFPJEwUNOVtJKvmDr9EI1fAJZUyZcRF7kxGBWmRXudrCQ==", - "requires": { - "camelcase": "^5.0.0", - "decamelize": "^1.2.0" - } - } - } -} diff --git a/backend/package.json b/backend/package.json deleted file mode 100644 index df43117f..00000000 --- a/backend/package.json +++ /dev/null @@ -1,66 +0,0 @@ -{ - "name": "backend", - "version": "1.0.0", - "description": "", - "main": "index.js", - "scripts": { - "prebuild": "rimraf dist", - "build": "babel src --out-dir dist", - "start": "babel-node src/main", - "start:dev": "nodemon -e js --watch src --exec babel-node src/main", - "start:prod": "node dist/main", - "test": "jest" - }, - "author": "", - "license": "ISC", - "dependencies": { - "bcrypt": "^5.0.0", - "class-transformer": "^0.3.1", - "class-validator": "^0.12.2", - "cookie-parser": "^1.4.5", - "cors": "^2.8.5", - "dotenv": "^8.2.0", - "express": "^4.17.1", - "express-async-errors": "^3.1.1", - "jsonwebtoken": "^8.5.1", - "mysql2": "^2.2.5", - "node-mocks-http": "^1.9.0", - "passport": "^0.4.1", - "passport-jwt": "^4.0.0", - "reflect-metadata": "^0.1.13", - "rimraf": "^3.0.2", - "typeorm": "^0.2.29", - "typeorm-transactional-cls-hooked": "^0.1.12", - "typescript": "^4.0.5" - }, - "devDependencies": { - "@babel/cli": "^7.12.1", - "@babel/core": "^7.12.3", - "@babel/eslint-parser": "^7.12.1", - "@babel/node": "^7.12.6", - "@babel/plugin-proposal-class-properties": "^7.12.1", - "@babel/plugin-proposal-decorators": "^7.12.1", - "@babel/plugin-proposal-private-methods": "^7.12.1", - "@babel/plugin-transform-runtime": "^7.12.1", - "@babel/preset-env": "^7.12.1", - "@types/dotenv": "^8.2.0", - "@types/express": "^4.17.9", - "@types/jest": "^26.0.15", - "@types/jsonwebtoken": "^8.5.0", - "@types/mysql2": "github:types/mysql2", - "@typescript-eslint/eslint-plugin": "^4.8.1", - "@typescript-eslint/parser": "^4.8.1", - "babel-plugin-parameter-decorator": "^1.0.16", - "eslint": "^7.13.0", - "eslint-config-airbnb-base": "^14.2.1", - "eslint-config-prettier": "^6.15.0", - "eslint-plugin-import": "^2.22.1", - "eslint-plugin-prettier": "^3.1.4", - "jest": "^26.6.3", - "nodemon": "^2.0.6", - "prettier": "^2.1.2", - "sqlite3": "^5.0.0", - "supertest": "^6.0.1", - "ts-jest": "^26.4.4" - } -} diff --git a/backend/src/__test__/TransactionWrapper.js b/backend/src/__test__/TransactionWrapper.js deleted file mode 100644 index c9b8b9f9..00000000 --- a/backend/src/__test__/TransactionWrapper.js +++ /dev/null @@ -1,10 +0,0 @@ -import { Transactional } from 'typeorm-transactional-cls-hooked'; - -class TransactionWrapper { - @Transactional() - static async transaction(cb) { - await cb(); - } -} - -export { TransactionWrapper }; diff --git a/backend/src/__test__/router/api.test.js b/backend/src/__test__/router/api.test.js deleted file mode 100644 index 373744c1..00000000 --- a/backend/src/__test__/router/api.test.js +++ /dev/null @@ -1,58 +0,0 @@ -import { getEntityManagerOrTransactionManager } from 'typeorm-transactional-cls-hooked'; -import { agent } from 'supertest'; -import { TransactionWrapper } from '../TransactionWrapper'; -import { ApplicationFactory } from '../../application'; -import { UserService } from '../../service'; - -const mockUsers = [ - { - id: 1, - email: 'newtest1@test.com', - name: '테스터테스터1', - password: 'test123!@#', - profileImage: 'profile image' - }, - { - id: 2, - email: 'newtest2@test.com', - name: '테스터테스터2', - password: 'test123!@#', - profileImage: 'profile image' - } -]; - -describe('User Router Test', () => { - let app = null; - - beforeAll(async () => { - app = await ApplicationFactory.create(); - }); - - test('모든 유저 정보를 조회할 수 있다', async () => { - await TransactionWrapper.transaction(async () => { - const entityManager = getEntityManagerOrTransactionManager(); - await entityManager.query('SAVEPOINT STARTPOINT'); - - // given - const userService = UserService.getInstance(); - const promises = mockUsers.map((mockUser) => userService.signup(mockUser)); - await Promise.all(promises); - - // when - const response = await agent(app.httpServer).get('/api/users').send(); - - // then - expect(response.status).toEqual(200); - - const users = await userService.getUsers(); - mockUsers.forEach((mockUser, idx) => { - expect(users[idx].id).toBe(mockUser.id); - expect(users[idx].name).toBe(mockUser.name); - expect(users[idx].password).toBe(mockUser.password); - expect(users[idx].profileImage).toBe(mockUser.profileImage); - }); - - await entityManager.query('ROLLBACK TO STARTPOINT'); - }); - }); -}); diff --git a/backend/src/__test__/router/auth.test.js b/backend/src/__test__/router/auth.test.js deleted file mode 100644 index 826f4164..00000000 --- a/backend/src/__test__/router/auth.test.js +++ /dev/null @@ -1,76 +0,0 @@ -import { getEntityManagerOrTransactionManager } from 'typeorm-transactional-cls-hooked'; -import { agent } from 'supertest'; -import { TransactionWrapper } from '../TransactionWrapper'; -import { ApplicationFactory } from '../../application'; -import { UserService } from '../../service'; - -const mockUser = { - id: 1, - email: 'newtest1@test.com', - name: '테스터테스터1', - password: 'test123!@#', - profileImage: 'http://test-image.com' -}; - -const InvalidMockUser = { - id: 1, - email: 'abcd', - name: 1000, - password: 2134, - profileImage: 'abcd' -}; - -describe('Auth Router Test', () => { - let app = null; - - beforeAll(async () => { - app = await ApplicationFactory.create(); - }); - - test('유효한 회원가입 정보(이메일, 이름, 프로필사진, 패스워드)로 회원가입 할 수 있다', async () => { - await TransactionWrapper.transaction(async () => { - const entityManager = getEntityManagerOrTransactionManager(); - await entityManager.query('SAVEPOINT STARTPOINT'); - - // given | when - const response = await agent(app.httpServer).post('/auth/signup').send({ - email: mockUser.email, - name: mockUser.name, - password: mockUser.password, - profileImage: mockUser.profileImage - }); - - // then - expect(response.status).toEqual(200); - - const userService = UserService.getInstance(); - const users = await userService.getUsers(); - expect(users[0].id).toBe(mockUser.id); - expect(users[0].name).toBe(mockUser.name); - expect(users[0].password).toBe(mockUser.password); - expect(users[0].profileImage).toBe(mockUser.profileImage); - - await entityManager.query('ROLLBACK TO STARTPOINT'); - }); - }); - - test('유효하지 않은 회원가입 정보(이메일, 이름, 프로필사진, 패스워드)로 회원가입 할 수 없다', async () => { - await TransactionWrapper.transaction(async () => { - const entityManager = getEntityManagerOrTransactionManager(); - await entityManager.query('SAVEPOINT STARTPOINT'); - - // given | when - const response = await agent(app.httpServer).post('/auth/signup').send({ - email: InvalidMockUser.email, - name: InvalidMockUser.name, - password: InvalidMockUser.password, - profileImage: InvalidMockUser.profileImage - }); - - // then - expect(response.status).toEqual(400); - - await entityManager.query('ROLLBACK TO STARTPOINT'); - }); - }); -}); diff --git a/backend/src/__test__/service/user-service.test.js b/backend/src/__test__/service/user-service.test.js deleted file mode 100644 index a64be8f8..00000000 --- a/backend/src/__test__/service/user-service.test.js +++ /dev/null @@ -1,79 +0,0 @@ -import { getEntityManagerOrTransactionManager } from 'typeorm-transactional-cls-hooked'; -import { Application } from '../../application'; -import { UserService } from '../../service'; -import { TransactionWrapper } from '../TransactionWrapper'; -import { User } from '../../model/user'; - -const mockUsers = [ - { - email: 'test1@test.com', - name: '테스터1', - password: 'test123!@#', - profileImage: 'http://test-image.com' - }, - { - email: 'test2@test.com', - name: '테스터2', - password: 'test123!@#', - profileImage: 'http://test-image.com' - } -]; - -describe('UserService Test', () => { - const app = new Application(); - - beforeAll(async () => { - await app.initEnvironment(); - await app.initDatabase(); - }); - - test('모든 유저 정보를 조회할 수 있다', async () => { - const userService = UserService.getInstance(); - - await TransactionWrapper.transaction(async () => { - const entityManager = getEntityManagerOrTransactionManager(); - await entityManager.query('SAVEPOINT STARTPOINT'); - - // given - const promises = mockUsers.map((user) => entityManager.save(User, user)); - await Promise.all(promises); - - // when - const users = await userService.getUsers(); - - // then - mockUsers.forEach((mockUser, idx) => { - expect(users[idx].id).toBe(mockUser.id); - expect(users[idx].name).toBe(mockUser.name); - expect(users[idx].password).toBe(mockUser.password); - expect(users[idx].profileImage).toBe(mockUser.profileImage); - }); - - await entityManager.query('ROLLBACK TO STARTPOINT'); - }); - }); - - test('올바른 이메일, 이름, 프로필사진, 패스워드 정보로 회원가입을 할 수 있다', async () => { - const userService = UserService.getInstance(); - - await TransactionWrapper.transaction(async () => { - const entityManager = getEntityManagerOrTransactionManager(); - await entityManager.query('SAVEPOINT STARTPOINT'); - - // given | when - const promises = mockUsers.map((mockUser) => userService.signup(mockUser)); - await Promise.all(promises); - - // then - const users = await userService.getUsers(); - mockUsers.forEach((mockUser, idx) => { - expect(users[idx].id).toBe(mockUser.id); - expect(users[idx].name).toBe(mockUser.name); - expect(users[idx].password).toBe(mockUser.password); - expect(users[idx].profileImage).toBe(mockUser.profileImage); - }); - - await entityManager.query('ROLLBACK TO STARTPOINT'); - }); - }); -}); diff --git a/backend/src/application/application-factory.js b/backend/src/application/application-factory.js deleted file mode 100644 index 049d85c8..00000000 --- a/backend/src/application/application-factory.js +++ /dev/null @@ -1,11 +0,0 @@ -import Application from './application'; - -class ApplicationFactory { - static async create() { - const application = new Application(); - await application.initialize(); - return application; - } -} - -export default ApplicationFactory; diff --git a/backend/src/application/application.js b/backend/src/application/application.js deleted file mode 100644 index e77964b7..00000000 --- a/backend/src/application/application.js +++ /dev/null @@ -1,79 +0,0 @@ -import 'reflect-metadata'; -import 'express-async-errors'; -import dotenv from 'dotenv'; -import express from 'express'; -import path from 'path'; -import cors from 'cors'; -import cookieParser from 'cookie-parser'; -import { validateOrReject } from 'class-validator'; -import { createConnection } from 'typeorm'; -import { initializeTransactionalContext, patchTypeORMRepositoryWithBaseRepository } from 'typeorm-transactional-cls-hooked'; -import { errorHandler } from '../common/middleware'; -import { router } from '../router'; -import { EnvType } from '../common/env/env-type'; -import { DatabaseEnv } from '../common/env/database-env'; -import { ConnectionOptionGenerator } from '../common/config/database/connection-option-generator'; - -class Application { - constructor() { - this.httpServer = express(); - this.databaseEnv = null; - this.connectionOptionGenerator = null; - } - - listen(port) { - return new Promise((resolve) => { - this.httpServer.listen(port, () => { - resolve(); - }); - }); - } - - async initialize() { - try { - await this.initEnvironment(); - this.registerMiddleware(); - await this.initDatabase(); - } catch (error) { - console.error(error); - process.exit(); - } - } - - async initEnvironment() { - dotenv.config(); - if (!EnvType.contains(process.env.NODE_ENV)) { - throw new Error('잘못된 NODE_ENV 입니다. {production, development, local, test} 중 하나를 선택하십시오.'); - } - dotenv.config({ - path: path.join(`${process.cwd()}/.env.${process.env.NODE_ENV}`) - }); - - this.databaseEnv = new DatabaseEnv(); - await validateOrReject(this.databaseEnv); - this.connectionOptionGenerator = new ConnectionOptionGenerator(this.databaseEnv); - } - - async initDatabase() { - initializeTransactionalContext(); - patchTypeORMRepositoryWithBaseRepository(); - await createConnection(this.connectionOptionGenerator.generateConnectionOption()); - } - - registerMiddleware() { - this.httpServer.use( - cors({ - origin: true, - credentials: true - }) - ); - this.httpServer.use(cookieParser()); - this.httpServer.use(express.json()); - this.httpServer.use(express.urlencoded({ extended: false })); - this.httpServer.use(router); - this.httpServer.use(express.static(`${process.cwd()}/public`)); - this.httpServer.use(errorHandler); - } -} - -export default Application; diff --git a/backend/src/application/index.js b/backend/src/application/index.js deleted file mode 100644 index a3b9168a..00000000 --- a/backend/src/application/index.js +++ /dev/null @@ -1,2 +0,0 @@ -export { default as ApplicationFactory } from './application-factory'; -export { default as Application } from './application'; diff --git a/backend/src/common/config/database/connection-option-generator.js b/backend/src/common/config/database/connection-option-generator.js deleted file mode 100644 index 53c8a8d5..00000000 --- a/backend/src/common/config/database/connection-option-generator.js +++ /dev/null @@ -1,38 +0,0 @@ -import path from 'path'; -import { DatabaseType } from './database-type'; - -class ConnectionOptionGenerator { - static SQLITE3_DATABASE = ':memory:'; - - constructor(databaseEnv) { - this.databaseEnv = databaseEnv; - } - - generateConnectionOption() { - const connectionOption = { - type: this.databaseEnv.getDatabaseType(), - entities: [path.resolve(`${__dirname}/../../../model/*.js`)], - logging: this.databaseEnv.getDatabaseLogging(), - dropSchema: this.databaseEnv.getDatabaseDropSchema(), - synchronize: this.databaseEnv.getDatabaseSynchronize(), - extra: {} - }; - - switch (this.databaseEnv.getDatabaseType()) { - case DatabaseType.MYSQL: - connectionOption.url = this.databaseEnv.getDatabaseUrl(); - connectionOption.connectTimeout = 3000; - connectionOption.acquireTimeout = 5000; - connectionOption.extra.connectionLimit = this.databaseEnv.getDatabaseConnectionLimit(); - break; - case DatabaseType.SQLITE3: - connectionOption.database = ConnectionOptionGenerator.SQLITE3_DATABASE; - break; - default: - } - - return connectionOption; - } -} - -export { ConnectionOptionGenerator }; diff --git a/backend/src/common/config/database/database-type.js b/backend/src/common/config/database/database-type.js deleted file mode 100644 index 52fcaf21..00000000 --- a/backend/src/common/config/database/database-type.js +++ /dev/null @@ -1,11 +0,0 @@ -const DatabaseType = { - MYSQL: 'mysql', - SQLITE3: 'sqlite', - valueOf: (typeString) => Object.keys(DatabaseType) - .filter((value) => DatabaseType[value] === typeString) - .reduce((value) => value, null), - values: () => Object.values(DatabaseType).filter((value) => typeof value === 'string'), - contains: (db) => DatabaseType.values().filter((value) => value === db).length !== 0 -}; - -export { DatabaseType }; diff --git a/backend/src/common/env/database-env.js b/backend/src/common/env/database-env.js deleted file mode 100644 index 184eaad6..00000000 --- a/backend/src/common/env/database-env.js +++ /dev/null @@ -1,72 +0,0 @@ -import { - IsBoolean, IsIn, IsNotEmpty, IsNumber, ValidateIf -} from 'class-validator'; -import { DatabaseType } from '../config/database/database-type'; - -class DatabaseEnv { - @IsNotEmpty() - @IsIn(DatabaseType.values()) - databaseType; - - @ValidateIf((o) => o.databaseType === DatabaseType.MYSQL) - @IsNotEmpty() - databaseUrl; - - @IsNotEmpty() - @IsBoolean() - databaseDropSchema; - - @IsNotEmpty() - @IsBoolean() - databaseSynchronize; - - @ValidateIf((o) => o.databaseType === DatabaseType.MYSQL) - @IsNotEmpty() - @IsNumber() - databaseConnectionLimit; - - databaseLogging; - - constructor() { - this.databaseType = process.env.DATABASE_TYPE; - this.databaseUrl = process.env.DATABASE_URL ?? null; - this.databaseConnectionLimit = process.env.DATABASE_CONNECTION_LIMIT === undefined ? 10 : parseInt(process.env.DATABASE_CONNECTION_LIMIT, 10); - this.databaseDropSchema = process.env.DATABASE_DROP_SCHEMA === undefined ? false : JSON.parse(process.env.DATABASE_DROP_SCHEMA.toLowerCase()); - this.databaseSynchronize = process.env.DATABASE_SYNCHRONIZE === undefined ? false : JSON.parse(process.env.DATABASE_SYNCHRONIZE.toLowerCase()); - this.databaseLogging = process.env.DATABASE_LOGGING === undefined ? ['error'] : this.splitDatabaseLogging(); - } - - splitDatabaseLogging() { - if (process.env.DATABASE_LOGGING === 'all') { - return 'all'; - } - - return process.env.DATABASE_LOGGING.split(','); - } - - getDatabaseType() { - return this.databaseType; - } - - getDatabaseUrl() { - return this.databaseUrl; - } - - getDatabaseDropSchema() { - return this.databaseDropSchema; - } - - getDatabaseSynchronize() { - return this.databaseSynchronize; - } - - getDatabaseConnectionLimit() { - return this.databaseConnectionLimit; - } - - getDatabaseLogging() { - return this.databaseLogging; - } -} - -export { DatabaseEnv }; diff --git a/backend/src/common/env/env-type.js b/backend/src/common/env/env-type.js deleted file mode 100644 index f8f24e0a..00000000 --- a/backend/src/common/env/env-type.js +++ /dev/null @@ -1,10 +0,0 @@ -const EnvType = { - PRODUCTION: "production", - DEVELOPMENT: "development", - LOCAL: "local", - TEST: "test", - values: () => Object.values(EnvType).filter((value) => typeof value === "string"), - contains: (env) => EnvType.values().filter((value) => value === env).length !== 0 -}; - -export { EnvType }; diff --git a/backend/src/common/error/business-error.js b/backend/src/common/error/business-error.js deleted file mode 100644 index 7fff23e2..00000000 --- a/backend/src/common/error/business-error.js +++ /dev/null @@ -1,8 +0,0 @@ -class BusinessError extends Error { - constructor(errorCode, message) { - super(message || errorCode.message); - this.errorCode = errorCode; - } -} - -export { BusinessError }; diff --git a/backend/src/common/error/entity-already-exist.js b/backend/src/common/error/entity-already-exist.js deleted file mode 100644 index 6a876c69..00000000 --- a/backend/src/common/error/entity-already-exist.js +++ /dev/null @@ -1,10 +0,0 @@ -import { BusinessError } from './business-error'; -import { ErrorCode } from './error-code'; - -class EntityAlreadyExist extends BusinessError { - constructor() { - super(ErrorCode.ENTITY_ALREADY_EXIST); - } -} - -export { EntityAlreadyExist }; diff --git a/backend/src/common/error/entity-not-found-error.js b/backend/src/common/error/entity-not-found-error.js deleted file mode 100644 index c48894a6..00000000 --- a/backend/src/common/error/entity-not-found-error.js +++ /dev/null @@ -1,10 +0,0 @@ -import { BusinessError } from './business-error'; -import { ErrorCode } from './error-code'; - -class EntityNotFoundError extends BusinessError { - constructor() { - super(ErrorCode.ENTITY_NOT_FOUND); - } -} - -export { EntityNotFoundError }; diff --git a/backend/src/common/error/error-code.js b/backend/src/common/error/error-code.js deleted file mode 100644 index 11b2b50f..00000000 --- a/backend/src/common/error/error-code.js +++ /dev/null @@ -1,10 +0,0 @@ -const ErrorCode = { - ENTITY_NOT_FOUND: { code: 1000, httpStatusCode: 404, message: 'Entity is not found' }, - ENTITY_ALREADY_EXIST: { code: 1001, httpStatusCode: 400, message: 'Entity already exists' }, - UNAUTHORIZED: { code: 1002, httpStatusCode: 401, message: 'Unauthorized' }, - FORBIDDEN: { code: 1003, httpStatusCode: 403, message: 'Forbidden' }, - BAD_REQUEST: { code: 1004, httpStatusCode: 400, message: 'Bad Request' }, - VALIDATION_ERROR: { code: 1005, httpStatusCode: 400, message: 'Validation error occurs' } -}; - -export { ErrorCode }; diff --git a/backend/src/common/middleware/error-handler.js b/backend/src/common/middleware/error-handler.js deleted file mode 100644 index 621dae28..00000000 --- a/backend/src/common/middleware/error-handler.js +++ /dev/null @@ -1,25 +0,0 @@ -import { ValidationError } from 'class-validator'; -import { BusinessError } from '../error/business-error'; -import { ErrorCode } from '../error/error-code'; - -const errorHandler = (err, req, res, next) => { - if (err instanceof BusinessError) { - res.status(err.errorCode.httpStatusCode).json({ - error: { - code: err.errorCode.code, - message: err.errorCode.message - } - }); - } else if (Array.isArray(err) && err[0] instanceof ValidationError) { - res.status(ErrorCode.VALIDATION_ERROR.httpStatusCode).json({ - error: { - code: ErrorCode.VALIDATION_ERROR.code, - message: ErrorCode.VALIDATION_ERROR.message - } - }); - } else { - throw err; - } -}; - -export default errorHandler; diff --git a/backend/src/common/middleware/index.js b/backend/src/common/middleware/index.js deleted file mode 100644 index d8875a1b..00000000 --- a/backend/src/common/middleware/index.js +++ /dev/null @@ -1,3 +0,0 @@ -export { default as transformer } from './transformer'; -export { default as validator } from './validator'; -export { default as errorHandler } from './error-handler'; diff --git a/backend/src/common/middleware/request-type.js b/backend/src/common/middleware/request-type.js deleted file mode 100644 index 07f6c402..00000000 --- a/backend/src/common/middleware/request-type.js +++ /dev/null @@ -1,7 +0,0 @@ -const RequestType = { - BODY: 'body', - PARAMS: 'params', - QUERY: 'query' -}; - -export { RequestType }; diff --git a/backend/src/common/middleware/transformer.js b/backend/src/common/middleware/transformer.js deleted file mode 100644 index 4b671252..00000000 --- a/backend/src/common/middleware/transformer.js +++ /dev/null @@ -1,10 +0,0 @@ -import { plainToClass } from 'class-transformer'; - -const transformer = (reqTypes, TArr) => (req, res, next) => { - reqTypes.forEach((type, index) => { - req[type] = plainToClass(TArr[index], req[type]); - }); - next(); -}; - -export default transformer; diff --git a/backend/src/common/middleware/validator.js b/backend/src/common/middleware/validator.js deleted file mode 100644 index 78b66b5a..00000000 --- a/backend/src/common/middleware/validator.js +++ /dev/null @@ -1,8 +0,0 @@ -import { validateOrReject } from 'class-validator'; - -const validator = (reqTypes) => async (req, res, next) => { - await Promise.all(reqTypes.map((type) => validateOrReject(req[type]))); - next(); -}; - -export default validator; diff --git a/backend/src/controller/index.js b/backend/src/controller/index.js deleted file mode 100644 index 25ec8c57..00000000 --- a/backend/src/controller/index.js +++ /dev/null @@ -1,3 +0,0 @@ -import * as userController from './user-controller'; - -export { userController }; diff --git a/backend/src/controller/user-controller.js b/backend/src/controller/user-controller.js deleted file mode 100644 index 59c9f60a..00000000 --- a/backend/src/controller/user-controller.js +++ /dev/null @@ -1,33 +0,0 @@ -import { UserService } from '../service'; - -const sendUsers = async (req, res, next) => { - try { - const userService = UserService.getInstance(); - const users = await userService.getUsers(); - - res.status(200).json(users); - } catch (error) { - next(error); - } -}; - -const signup = async (req, res, next) => { - try { - const { - email, name, password, profileImage - } = req.body; - const userService = UserService.getInstance(); - await userService.signup({ - email, - name, - password, - profileImage - }); - - res.status(200).end(); - } catch (error) { - next(error); - } -}; - -export { sendUsers, signup }; diff --git a/backend/src/dto/index.js b/backend/src/dto/index.js deleted file mode 100644 index 3d071ad6..00000000 --- a/backend/src/dto/index.js +++ /dev/null @@ -1 +0,0 @@ -export { default as SignupRequestBody } from './signup-request-body'; diff --git a/backend/src/dto/signup-request-body.js b/backend/src/dto/signup-request-body.js deleted file mode 100644 index 77c39f0c..00000000 --- a/backend/src/dto/signup-request-body.js +++ /dev/null @@ -1,20 +0,0 @@ -import { - IsString, IsEmail, Length, IsUrl -} from 'class-validator'; - -class SignupRequestBody { - @IsEmail() - email - - @IsString() - @Length(4, 20) - name; - - @IsString() - password; - - @IsUrl() - profileImage; -} - -export default SignupRequestBody; diff --git a/backend/src/main.js b/backend/src/main.js deleted file mode 100644 index 6567ca76..00000000 --- a/backend/src/main.js +++ /dev/null @@ -1,8 +0,0 @@ -import { ApplicationFactory } from './application'; - -const main = async () => { - const app = await ApplicationFactory.create(); - await app.listen(process.env.port || 5000); -}; - -main(); diff --git a/backend/src/model/user.js b/backend/src/model/user.js deleted file mode 100644 index 50103552..00000000 --- a/backend/src/model/user.js +++ /dev/null @@ -1,43 +0,0 @@ -import { - Column, CreateDateColumn, Entity, PrimaryGeneratedColumn, UpdateDateColumn, DeleteDateColumn -} from 'typeorm'; -import { - IsEmail, Length, IsString, IsUrl, IsOptional -} from 'class-validator'; - -@Entity({ name: 'user' }) -class User { - @PrimaryGeneratedColumn('increment', { type: 'int' }) - id; - - @Column({ name: 'email', type: 'varchar', unique: true }) - @IsEmail() - email; - - @Column({ - name: 'name', type: 'varchar', unique: true, charset: 'utf-8' - }) - @IsString() - @Length(4, 20) - name; - - @Column({ name: 'password', type: 'varchar', nullable: true }) - @IsOptional() - @IsString() - password; - - @Column({ name: 'profile_image', type: 'varchar' }) - @IsUrl() - profileImage; - - @CreateDateColumn({ name: 'created_at', type: 'datetime' }) - createdAt; - - @UpdateDateColumn({ name: 'updated_at', type: 'datetime' }) - updatedAt; - - @DeleteDateColumn({ name: 'deleted_at', type: 'datetime' }) - deletedAt; -} - -export { User }; diff --git a/backend/src/router/api.js b/backend/src/router/api.js deleted file mode 100644 index 417955ec..00000000 --- a/backend/src/router/api.js +++ /dev/null @@ -1,8 +0,0 @@ -import express from 'express'; -import { userController } from '../controller'; - -const apiRouter = express.Router(); - -apiRouter.get('/users', userController.sendUsers); - -export { apiRouter }; diff --git a/backend/src/router/auth.js b/backend/src/router/auth.js deleted file mode 100644 index c54a1c70..00000000 --- a/backend/src/router/auth.js +++ /dev/null @@ -1,11 +0,0 @@ -import express from 'express'; -import { userController } from '../controller'; -import { RequestType } from '../common/middleware/request-type'; -import { SignupRequestBody } from '../dto'; -import { transformer, validator } from '../common/middleware'; - -const authRouter = express.Router(); - -authRouter.post('/signup', transformer([RequestType.BODY], [SignupRequestBody]), validator([RequestType.BODY]), userController.signup); - -export { authRouter }; diff --git a/backend/src/router/index.js b/backend/src/router/index.js deleted file mode 100644 index 47af8347..00000000 --- a/backend/src/router/index.js +++ /dev/null @@ -1,12 +0,0 @@ -import express from 'express'; -import { apiRouter } from './api'; -import { oauthRouter } from './oauth'; -import { authRouter } from './auth'; - -const router = express.Router(); - -router.use('/api', apiRouter); -router.use('/oauth', oauthRouter); -router.use('/auth', authRouter); - -export { router }; diff --git a/backend/src/router/oauth.js b/backend/src/router/oauth.js deleted file mode 100644 index 19e5ded1..00000000 --- a/backend/src/router/oauth.js +++ /dev/null @@ -1,8 +0,0 @@ -import express from 'express'; - -const oauthRouter = express.Router(); - -oauthRouter.get('/github', (req, res, next) => { res.status(200).end(); }); -oauthRouter.get('/github/login', (req, res, next) => { res.status(200).end(); }); - -export { oauthRouter }; diff --git a/backend/src/service/index.js b/backend/src/service/index.js deleted file mode 100644 index d4bf57c4..00000000 --- a/backend/src/service/index.js +++ /dev/null @@ -1 +0,0 @@ -export { default as UserService } from './user-service'; diff --git a/backend/src/service/user-service.js b/backend/src/service/user-service.js deleted file mode 100644 index 802dd11b..00000000 --- a/backend/src/service/user-service.js +++ /dev/null @@ -1,46 +0,0 @@ -import { getRepository } from 'typeorm'; -import { Transactional } from 'typeorm-transactional-cls-hooked'; -import { User } from '../model/user'; -import { EntityAlreadyExist } from '../common/error/entity-already-exist'; -import { EntityNotFoundError } from '../common/error/entity-not-found-error'; - -class UserService { - constructor() { - this.userRepository = getRepository(User); - } - - static instance = null; - - static getInstance() { - if (UserService.instance === null) { - UserService.instance = new UserService(); - } - return UserService.instance; - } - - @Transactional() - async signup({ - email, name, password, profileImage - }) { - const promises = []; - promises.push(this.userRepository.findOne({ email })); - promises.push(this.userRepository.findOne({ name })); - - const [userByEmail, userByName] = await Promise.all(promises); - if (userByEmail || userByName) { throw new EntityAlreadyExist(); } - - const newUser = this.userRepository.create({ - email, name, password, profileImage - }); - await this.userRepository.save(newUser); - } - - async getUsers() { - const users = await this.userRepository.find(); - if (!users) { throw new EntityNotFoundError(); } - - return users; - } -} - -export default UserService; diff --git a/backend/tsconfig.json b/backend/tsconfig.json deleted file mode 100644 index eab68508..00000000 --- a/backend/tsconfig.json +++ /dev/null @@ -1,29 +0,0 @@ -{ - "compilerOptions": { - "experimentalDecorators": true, - "target": "es5", - "lib": [ - "dom", - "dom.iterable", - "esnext" - ], - "allowJs": true, - "skipLibCheck": true, - "esModuleInterop": true, - "allowSyntheticDefaultImports": true, - "strict": true, - "forceConsistentCasingInFileNames": true, - "noFallthroughCasesInSwitch": true, - "module": "esnext", - "moduleResolution": "node", - "resolveJsonModule": true, - "isolatedModules": true, - "noEmit": true, - "jsx": "react", - "noImplicitAny": false, - "strictPropertyInitialization": false - }, - "include": [ - "src" - ] -} diff --git a/frontend/env/.env.sample b/frontend/env/.env.sample deleted file mode 100644 index f588fb90..00000000 --- a/frontend/env/.env.sample +++ /dev/null @@ -1,2 +0,0 @@ -REACT_APP_SERVER_URL=http://localhost:2323 -REACT_APP_GET_USERS=/api/users diff --git a/frontend/package-lock.json b/frontend/package-lock.json index 216f3a4f..865ab318 100644 --- a/frontend/package-lock.json +++ b/frontend/package-lock.json @@ -4605,12 +4605,6 @@ "tslib": "^1.10.0" } }, - "dotenv": { - "version": "8.2.0", - "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-8.2.0.tgz", - "integrity": "sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw==", - "dev": true - }, "duplexify": { "version": "3.7.1", "resolved": "https://registry.npmjs.org/duplexify/-/duplexify-3.7.1.tgz", diff --git a/frontend/package.json b/frontend/package.json index bb4e8f16..0916ca2c 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -26,7 +26,6 @@ "clean-webpack-plugin": "^3.0.0", "core-js": "^3.6.5", "css-loader": "^4.3.0", - "dotenv": "^8.2.0", "eslint": "^7.13.0", "eslint-config-airbnb": "^18.2.1", "eslint-config-prettier": "^6.15.0", @@ -48,8 +47,8 @@ "webpack-merge": "^5.4.0" }, "scripts": { - "dev": "webpack-dev-server --env.ENVIRONMENT=development", - "build": "webpack --mode production --devtool hidden-source-map --env.ENVIRONMENT=production", + "dev": "webpack-dev-server", + "build": "webpack --mode production --devtool hidden-source-map", "test": "jest" }, "eslintConfig": { diff --git a/frontend/src/common/command/AddTrackCommand.ts b/frontend/src/common/command/AddTrackCommand.ts new file mode 100644 index 00000000..b99b39e0 --- /dev/null +++ b/frontend/src/common/command/AddTrackCommand.ts @@ -0,0 +1,46 @@ +import { Command } from '@command'; +import { Controller } from '@controllers'; +import { Track } from '@model'; +import { storeChannel } from '@store'; +import { StoreChannelType } from '@types'; + +class AddTrackCommand extends Command { + constructor(){ + super(); + } + + execute(){ + try{ + const newTrack = new Track({id: 0, trackSectionList: []}); + Controller.setTrack(newTrack); + + this.publishNewTrackList(); + }catch(e){ + console.log(e); + } + } + + undo(){ + try{ + Controller.popTrackWithIndex(); + this.publishNewTrackList(); + }catch(e){ + console.log(e); + } + } + + publishNewTrackList(): void { + const newTrackList = Controller.getTrackList(); + + storeChannel.publish(StoreChannelType.TRACK_CHANNEL, newTrackList); + storeChannel.publish(StoreChannelType.TRACK_LIST_CHANNEL, newTrackList); + newTrackList.forEach((track) => { + storeChannel.publish(StoreChannelType.TRACK_SECTION_LIST_CHANNEL, { + trackId: track.id, + trackSectionList: track.trackSectionList + }); + }); + } +} + +export default AddTrackCommand; diff --git a/frontend/src/common/command/Command.ts b/frontend/src/common/command/Command.ts new file mode 100644 index 00000000..6858cd90 --- /dev/null +++ b/frontend/src/common/command/Command.ts @@ -0,0 +1,6 @@ +class Command { + execute() { }; + undo() { }; +} + +export default Command; diff --git a/frontend/src/common/command/CommandManager.ts b/frontend/src/common/command/CommandManager.ts index 29a0282b..96c96ec5 100644 --- a/frontend/src/common/command/CommandManager.ts +++ b/frontend/src/common/command/CommandManager.ts @@ -1,45 +1,45 @@ -import ICommand from './ICommand' +import { Command } from '@command'; import { storeChannel } from '@store' import { StoreChannelType } from '@types' class CommandManager { - public undoList: ICommand[]; - public redoList: ICommand[]; + public undoList: Command[]; + public redoList: Command[]; constructor() { this.undoList = [] this.redoList = [] } - execute(command: ICommand) { + execute(command: Command) { command.execute(); this.undoList.push(command); this.redoList = []; - storeChannel.publish(StoreChannelType.EDIT_TOOLS_CHANNEL, ''); + storeChannel.publish(StoreChannelType.EDIT_TOOLS_CHANNEL, null); } undo() { if (this.undoList.length > 0) { - const command: ICommand | undefined = this.undoList.pop(); + const command: Command | undefined = this.undoList.pop(); if (command) { this.redoList.push(command); command.undo(); - storeChannel.publish(StoreChannelType.EDIT_TOOLS_CHANNEL, ''); + storeChannel.publish(StoreChannelType.EDIT_TOOLS_CHANNEL, null); } } } redo() { if (this.redoList.length > 0) { - const command: ICommand | undefined = this.redoList.pop(); + const command: Command | undefined = this.redoList.pop(); if (command) { this.undoList.push(command); command.execute(); - storeChannel.publish(StoreChannelType.EDIT_TOOLS_CHANNEL, ''); + storeChannel.publish(StoreChannelType.EDIT_TOOLS_CHANNEL, null); } } } } -export default new CommandManager(); \ No newline at end of file +export default new CommandManager(); diff --git a/frontend/src/common/command/DeleteCommand.ts b/frontend/src/common/command/DeleteCommand.ts index c52ebbfc..8e54c210 100644 --- a/frontend/src/common/command/DeleteCommand.ts +++ b/frontend/src/common/command/DeleteCommand.ts @@ -1,8 +1,9 @@ -import ICommand from './ICommand' +import { Command } from '@command'; import { Controller } from '@controllers' import { TrackSection } from '@model' +import { CopyUtil } from '@util' -export default class DeleteCommand extends ICommand { +class DeleteCommand extends Command { private deleteList: TrackSection[]; constructor() { @@ -17,17 +18,7 @@ export default class DeleteCommand extends ICommand { if (focusList.length === 0) return; this.deleteList = focusList.map(focus => { const trackSection = focus.trackSection; - return new TrackSection({ - id: trackSection.id, - sourceId: trackSection.sourceId, - trackId: trackSection.trackId, - channelStartTime: trackSection.channelStartTime, - channelEndTime: trackSection.channelEndTime, - parsedChannelStartTime: trackSection.parsedChannelStartTime, - parsedChannelEndTime: trackSection.parsedChannelEndTime, - trackStartTime: trackSection.trackStartTime, - audioStartTime: trackSection.audioStartTime - }) + return CopyUtil.copySection(trackSection); }); } @@ -59,3 +50,4 @@ export default class DeleteCommand extends ICommand { }; } +export default DeleteCommand; diff --git a/frontend/src/common/command/DeleteTrackCommand.ts b/frontend/src/common/command/DeleteTrackCommand.ts new file mode 100644 index 00000000..c47eeee2 --- /dev/null +++ b/frontend/src/common/command/DeleteTrackCommand.ts @@ -0,0 +1,60 @@ +import { Command } from '@command'; +import { Controller } from '@controllers'; +import { Track } from '@model'; +import { storeChannel } from '@store'; +import { StoreChannelType } from '@types'; + +class DeleteTrackCommand extends Command { + private trackId : number; + private removeIdx: number; + private removedTrack: Track | undefined; + + constructor(trackId: number){ + super(); + this.trackId = trackId; + this.removeIdx = 0; + this.removedTrack; + } + + execute(): void { + try{ + this.removeIdx = this.calculateRemoveIdx(); + this.removedTrack = Controller.removeTrackById(this.trackId); + + this.publishNewTrackList(); + }catch(e){ + console.log(e); + } + } + + calculateRemoveIdx(): number { + const trackList = Controller.getTrackList(); + return trackList.findIndex((track) => track.id === this.trackId); + } + + undo(): void { + try{ + if(!this.removedTrack) return; + Controller.insertTrack(this.removeIdx, this.removedTrack); + + this.publishNewTrackList(); + }catch(e){ + console.log(e); + } + } + + publishNewTrackList(): void { + const newTrackList = Controller.getTrackList(); + + storeChannel.publish(StoreChannelType.TRACK_CHANNEL, newTrackList); + storeChannel.publish(StoreChannelType.TRACK_LIST_CHANNEL, newTrackList); + newTrackList.forEach((track) => { + storeChannel.publish(StoreChannelType.TRACK_SECTION_LIST_CHANNEL, { + trackId: track.id, + trackSectionList: track.trackSectionList + }); + }); + } +} + +export default DeleteTrackCommand; diff --git a/frontend/src/common/command/ICommand.ts b/frontend/src/common/command/ICommand.ts deleted file mode 100644 index 2a157a29..00000000 --- a/frontend/src/common/command/ICommand.ts +++ /dev/null @@ -1,17 +0,0 @@ -export default class ICommand { - execute() { }; - undo() { }; -} - -/** - * 붙여넣기 <-> 삭제하기 - * 이동하기 <-> 이동하기 - * 자르기 <-> 합치기 - * 효과넣기 <-> 효과지우기 - */ - - - - - - diff --git a/frontend/src/common/command/MoveCommand.ts b/frontend/src/common/command/MoveCommand.ts new file mode 100644 index 00000000..acd2e842 --- /dev/null +++ b/frontend/src/common/command/MoveCommand.ts @@ -0,0 +1,72 @@ +import { Command } from '@command'; +import { Track, TrackSection } from '@model'; +import { StoreChannelType } from '@types'; +import { storeChannel } from '@store'; +import { CopyUtil, ValidUtil } from '@util'; +import { Controller } from '@controllers'; + +class MoveCommand extends Command { + private prevTrack: Track; + private currentTrack: Track; + private trackSection: TrackSection; + private movingCursorTime: number; + private prevCursorTime: number; + + constructor(prevTrack: Track, currentTrack: Track, trackSection: TrackSection, movingCursorTime: number, prevCursorTime: number) { + super(); + this.prevTrack = prevTrack; + this.currentTrack = currentTrack; + this.trackSection = trackSection; + this.movingCursorTime = movingCursorTime; + this.prevCursorTime = prevCursorTime; + } + + execute(): void { + const sectionIndex = this.prevTrack.trackSectionList.findIndex((section) => section.id === this.trackSection.id); + + let newTrackStartTime = this.movingCursorTime - (this.prevCursorTime - this.trackSection.trackStartTime); + + if (newTrackStartTime < 0) { + newTrackStartTime = 0; + } + + const newTrackEndTime = newTrackStartTime + this.trackSection.length; + + if (ValidUtil.checkEnterTrack(this.trackSection, this.currentTrack.trackSectionList, newTrackStartTime, newTrackEndTime)) return; + + const currentScrollTime = Controller.getCurrentScrollTime() || 0; + + const newTrackSection = CopyUtil.copySection(this.trackSection); + + + newTrackSection.trackId = this.currentTrack.id; + newTrackSection.trackStartTime = newTrackStartTime + currentScrollTime; + + if (newTrackSection.id !== 0) { + Controller.removeSection(this.prevTrack.id, sectionIndex); + } + + Controller.addTrackSection(this.currentTrack.id, newTrackSection); + } + + undo(): void { + Controller.setTrack(this.prevTrack); + Controller.setTrack(this.currentTrack); + + if (this.prevTrack.id !== this.currentTrack.id) { + storeChannel.publish(StoreChannelType.TRACK_SECTION_LIST_CHANNEL, { + trackId: this.prevTrack.id, + trackSectionList: this.prevTrack.trackSectionList + }); + storeChannel.publish(StoreChannelType.TRACK_CHANNEL, this.prevTrack.trackSectionList); + } + + storeChannel.publish(StoreChannelType.TRACK_SECTION_LIST_CHANNEL, { + trackId: this.currentTrack.id, + trackSectionList: this.currentTrack.trackSectionList + }); + storeChannel.publish(StoreChannelType.TRACK_CHANNEL, this.currentTrack.trackSectionList); + } +} + +export default MoveCommand; diff --git a/frontend/src/common/command/PasteCommand.ts b/frontend/src/common/command/PasteCommand.ts new file mode 100644 index 00000000..7ea624c6 --- /dev/null +++ b/frontend/src/common/command/PasteCommand.ts @@ -0,0 +1,52 @@ +import { Command } from '@command'; +import { Controller } from '@controllers' +import { StoreChannelType } from '@types' +import { storeChannel } from '@store' +import { TrackSection, Track } from '@model' +import { CopyUtil } from '@util' + +class PasteCommand extends Command { + private beforeTrack: Track; + private addTrackSection: TrackSection; + + constructor(track: Track, trackSection: TrackSection) { + super(); + this.beforeTrack = track; + this.addTrackSection = trackSection; + } + + execute() { + const newTrack: Track = CopyUtil.copyTrack(this.beforeTrack); + const newSection: TrackSection = CopyUtil.copySection(this.addTrackSection); + + const endTime: number = newSection.trackStartTime + newSection.length; + const firstDelayIndex: number = newTrack.trackSectionList.findIndex(section => section.trackStartTime >= newSection.trackStartTime && section.trackStartTime < endTime); + if (firstDelayIndex !== -1) { + const delayTime = endTime - newTrack.trackSectionList[firstDelayIndex].trackStartTime; + newTrack.trackSectionList = newTrack.trackSectionList.map((cur, idx) => { + if (idx >= firstDelayIndex) { + cur.trackStartTime += delayTime; + } + return cur; + }) + Controller.setTrack(newTrack); + Controller.addTrackSection(newTrack.id, newSection); + } else { + Controller.addTrackSection(this.beforeTrack.id, newSection); + } + }; + + undo() { + const newTrack = CopyUtil.copyTrack(this.beforeTrack); + Controller.setTrack(newTrack); + + storeChannel.publish(StoreChannelType.TRACK_SECTION_LIST_CHANNEL, { + trackId: newTrack.id, + trackSectionList: newTrack.trackSectionList + }); + + storeChannel.publish(StoreChannelType.TRACK_CHANNEL, newTrack.trackSectionList); + }; +} + +export default PasteCommand; diff --git a/frontend/src/common/command/SplitCommand.ts b/frontend/src/common/command/SplitCommand.ts new file mode 100644 index 00000000..1f129715 --- /dev/null +++ b/frontend/src/common/command/SplitCommand.ts @@ -0,0 +1,75 @@ +import { Command } from '@command'; +import { Controller } from '@controllers'; +import { StoreChannelType } from '@types'; +import { storeChannel } from '@store'; +import { Track, TrackSection } from '@model'; +import { CopyUtil, TimeUtil } from '@util'; + +class SplitCommand extends Command { + private beforeTrack: Track; + private cursorPosition: number; + private targetTrackSection: TrackSection; + + constructor(cursorPosition: number, currentTrack: Track, targetTrackSection: TrackSection) { + super(); + this.cursorPosition = cursorPosition; + this.beforeTrack = currentTrack; + this.targetTrackSection = targetTrackSection; + } + + execute() { + const trackAreaElement = document.querySelector('.audio-track-area'); + if(!trackAreaElement) return; + + const maxTrackPlayTime = Controller.getMaxTrackPlayTime(); + const trackAreaLeftX = trackAreaElement.getBoundingClientRect().left; + const trackAreaRightX = trackAreaElement.getBoundingClientRect().right; + const trackAreaWidth = trackAreaRightX - trackAreaLeftX; + + const timeOfCursorPosition = TimeUtil.calculateTimeOfCursorPosition(trackAreaLeftX, this.cursorPosition, trackAreaWidth, maxTrackPlayTime); + const [leftTrackSection, rightTrackSection] = this.splitTrackSection(timeOfCursorPosition); + this.updateSplitedSections(leftTrackSection, rightTrackSection); + } + + splitTrackSection(timeOfCursorPosition: number): TrackSection[] { + const channelSplitOffset = timeOfCursorPosition - this.targetTrackSection.trackStartTime; + const channelSplitTime = this.targetTrackSection.channelStartTime + channelSplitOffset; + + const leftTrackSection = CopyUtil.copySection(this.targetTrackSection); + leftTrackSection.id = 0; + leftTrackSection.channelEndTime = channelSplitTime; + leftTrackSection.length = leftTrackSection.channelEndTime - leftTrackSection.channelStartTime; + + const rightTrackSection = CopyUtil.copySection(this.targetTrackSection); + rightTrackSection.id = 0; + rightTrackSection.channelStartTime = channelSplitTime; + rightTrackSection.length = rightTrackSection.channelEndTime - rightTrackSection.channelStartTime; + rightTrackSection.trackStartTime += channelSplitOffset; + + return [leftTrackSection, rightTrackSection]; + } + + updateSplitedSections(leftTrackSection: TrackSection, rightTrackSection: TrackSection): void { + const targetTrackSectionIndex = this.beforeTrack.trackSectionList.findIndex(trackSection => trackSection.id === this.targetTrackSection.id); + if (targetTrackSectionIndex === -1) return; + + const trackId = this.beforeTrack.id; + Controller.removeSection(trackId, targetTrackSectionIndex); + Controller.addTrackSection(trackId, leftTrackSection); + Controller.addTrackSection(trackId, rightTrackSection); + } + + undo(): void { + const newTrack = CopyUtil.copyTrack(this.beforeTrack); + Controller.setTrack(newTrack); + + storeChannel.publish(StoreChannelType.TRACK_SECTION_LIST_CHANNEL, { + trackId: newTrack.id, + trackSectionList: newTrack.trackSectionList + }); + + storeChannel.publish(StoreChannelType.TRACK_CHANNEL, newTrack.trackSectionList); + } +} + +export default SplitCommand; diff --git a/frontend/src/common/command/index.js b/frontend/src/common/command/index.js new file mode 100644 index 00000000..4f939709 --- /dev/null +++ b/frontend/src/common/command/index.js @@ -0,0 +1,8 @@ +export { default as Command } from './Command'; +export { default as CommandManager } from './CommandManager' +export { default as DeleteCommand } from './DeleteCommand'; +export { default as PasteCommand } from './PasteCommand'; +export { default as SplitCommand } from './SplitCommand'; +export { default as MoveCommand } from './MoveCommand'; +export { default as AddTrackCommand} from './AddTrackCommand'; +export { default as DeleteTrackCommand} from './DeleteTrackCommand'; diff --git a/frontend/src/common/types/eventTypes.ts b/frontend/src/common/types/eventTypes.ts index 2ee34e0b..e493837e 100644 --- a/frontend/src/common/types/eventTypes.ts +++ b/frontend/src/common/types/eventTypes.ts @@ -22,9 +22,18 @@ enum EventKeyType { AUDIO_SKIP_NEXT = 'AUDIO_SKIP_NEXT', AUDIO_TRACK_CONTAINER_MULTIPLE = 'AUDIO_TRACK_CONTAINER_MULTIPLE', SOURCE_LIST_MULTIPLE = 'SOURCE_LIST_MULTIPLE', - AUDIO_TRACK_SECTION_CLICK = 'AUDIO_TRACK_SECTION_CLICK', + AUDIO_TRACK_SECTION_MULTIPLE = 'AUDIO_TRACK_SECTION_MULTIPLE', FOCUS_RESET_CLICK = 'FOCUS_RESET_CLICK', - EDIT_TOOLS_CLICK = 'EDIT_TOOLS_CLICK' + EDIT_TOOLS_CLICK = 'EDIT_TOOLS_CLICK', + PLAYBAR_EVENT_ZONE_DROP = 'PLAYBAR_EVENT_ZONE_DROP', + ZOOM_BAR_MOUSE_DOWN = 'ZOOM_BAR_MOUSE_DOWN', + AUDIO_TRACK_OPTION_CLICK = 'AUDIO_TRACK_OPTION_CLICK', + EFFECT_LIST_OPEN_MODAL_BTN_CLICK = 'EFFECT_LIST_OPEN_MODAL_BTN_CLICK', + AUDIO_TRACK_MULTIPLE = 'AUDIO_TRACK_MULTIPLE', + AUDIO_TRACK_AREA_MULTIPLE = 'AUDIO_TRACK_AREA_MULTIPLE', + TRACK_ADD_MENU_CLICK = 'TRACK_ADD_MENU_CLICK', + TRACK_LIST_SCROLL = 'TRACK_LIST_SCROLL', + AUDIO_TRACK_AFTERIMAGE_DROP = 'AUDIO_TRACK_AFTERIMAGE_DROP' } enum EventType { @@ -39,10 +48,29 @@ enum EventType { drop = 'drop', change = 'change', input = 'input', - mousemove = 'mousemove' + mousemove = 'mousemove', + mousedown = 'mousedown', + mouseout = 'mouseout', + scroll = 'scroll' } -const eventTypes = ['click', 'dblclick', 'keyup', 'dragstart', 'dragover', 'dragenter', 'dragleave', 'drop', 'change', 'input', 'mousemove', 'dragend']; +const eventTypes = [ + 'click', + 'dblclick', + 'keyup', + 'dragstart', + 'dragover', + 'dragenter', + 'dragleave', + 'drop', + 'change', + 'input', + 'mousemove', + 'dragend', + 'mousedown', + 'mouseout', + 'scroll' +]; interface EventTargetDataType { listener: EventListener; diff --git a/frontend/src/common/types/index.js b/frontend/src/common/types/index.js index 0c12598e..0b5d800b 100644 --- a/frontend/src/common/types/index.js +++ b/frontend/src/common/types/index.js @@ -5,4 +5,6 @@ export { ButtonType } from './buttonType'; export { IconType } from './iconType'; export { CompressorOption } from './CompressorOption'; export { EffectStateType, EffectType, EffectTitleType, EffectContentType, EffectButtonContentType } from './effectTypes' -export { KeyBoard } from './keyType' \ No newline at end of file +export { KeyBoard } from './keyType' +export { TrackOptionType } from './trackOptionTypes'; +export { SectionDataType } from './trackSectionTypes'; diff --git a/frontend/src/common/types/keyType.ts b/frontend/src/common/types/keyType.ts index 33f728e0..6d07e33d 100644 --- a/frontend/src/common/types/keyType.ts +++ b/frontend/src/common/types/keyType.ts @@ -8,7 +8,9 @@ enum KeyBoard { CTRL = 17, LEFT = 37, RIGHT = 39, - SPACE = 32 + SPACE = 32, + LEFT_BRACKET = 219, + RIGHT_BRACKET = 221, } export { KeyBoard } \ No newline at end of file diff --git a/frontend/src/common/types/storeTypes.ts b/frontend/src/common/types/storeTypes.ts index 6fbc1815..13f0da66 100644 --- a/frontend/src/common/types/storeTypes.ts +++ b/frontend/src/common/types/storeTypes.ts @@ -1,4 +1,4 @@ -import { Source, Track, AudioSourceInfoInTrack, TrackSection } from '@model'; +import { Source, Track, AudioSourceInfoInTrack, TrackSection, SectionDragStartData, SelectTrackData } from '@model'; import { ModalStateType } from '@types'; enum CursorType { @@ -8,7 +8,7 @@ enum CursorType { interface FocusInfo { trackSection: TrackSection; - element: HTMLElement; + element: HTMLCanvasElement; } enum StoreChannelType { @@ -22,12 +22,20 @@ enum StoreChannelType { EDIT_TOOLS_CHANNEL = 'EDIT_TOOLS_CHANNEL', MARKER_TIME_CHANNEL = 'MARKER_TIME_CHANNEL', IS_PAUSE_CHANNEL = 'IS_PAUSE_CHANNEL', - PLAY_TIME_CHANNEL = 'PLAY_TIME_CHANNEL' + PLAY_TIME_CHANNEL = 'PLAY_TIME_CHANNEL', + MAX_TRACK_WIDTH_CHANNEL = 'MAX_TRACK_WIDTH_CHANNEL', + CURSOR_MODE_CHANNEL = 'CURSOR_MODE_CHANNEL', + EDIT_MENU_CHANNEL = 'EDIT_MENU_CHANNEL', + PLAY_OR_PAUSE_CHANNEL = 'PLAY_OR_PAUSE_CHANNEL', + MAX_TRACK_PLAY_TIME_CHANNEL = 'MAX_TRACK_PLAY_TIME_CHANNEL', + CURRENT_SCROLL_AMOUNT_CHANNEL = 'CURRENT_SCROLL_AMOUNT_CHANNEL', + TRACK_LIST_CHANNEL = 'TRACK_LIST_CHANNEL', + SELECT_AUDIO_TRACK = 'SELECT_AUDIO_TRACK', } interface StoreStateType { - cursorTime: string; - playTime: string; + cursorStringTime: string; + playStringTime: string; sourceList: Source[]; modalState: ModalStateType; isTrackDraggable: Boolean; @@ -40,9 +48,15 @@ interface StoreStateType { trackIndex: number; sectionIndex: number; clipBoard: TrackSection | null; - markerTime: number; - totalCursorTime: number; + markerNumberTime: number; + cursorNumberTime: number; isPause: boolean; + isRepeat: boolean; + maxTrackWidth: number; + maxTrackPlayTime: number; + currentScrollAmount: number; + sectionDragStartData: SectionDragStartData | null; + selectTrackData: SelectTrackData; } interface StoreObserverData { @@ -50,4 +64,8 @@ interface StoreObserverData { bindObj: Object; } -export { StoreStateType, StoreChannelType, StoreObserverData, CursorType, FocusInfo }; +interface EffectList { + name: string; +} + +export { StoreStateType, StoreChannelType, StoreObserverData, CursorType, FocusInfo, EffectList }; diff --git a/frontend/src/common/types/trackOptionTypes.ts b/frontend/src/common/types/trackOptionTypes.ts new file mode 100644 index 00000000..f5341d29 --- /dev/null +++ b/frontend/src/common/types/trackOptionTypes.ts @@ -0,0 +1,9 @@ +enum TrackOptionType { + solo = 'solo', + mute = 'mute', + delete = 'delete', +} + +export { + TrackOptionType +} diff --git a/frontend/src/common/types/trackSectionTypes.ts b/frontend/src/common/types/trackSectionTypes.ts new file mode 100644 index 00000000..535c7afc --- /dev/null +++ b/frontend/src/common/types/trackSectionTypes.ts @@ -0,0 +1,8 @@ +interface SectionDataType { + sectionChannelData: number[]; + duration: number; +} + +export { + SectionDataType +} diff --git a/frontend/src/common/util/AudioUtil.ts b/frontend/src/common/util/AudioUtil.ts index 34cb8f99..3885e0ca 100644 --- a/frontend/src/common/util/AudioUtil.ts +++ b/frontend/src/common/util/AudioUtil.ts @@ -1,4 +1,4 @@ -const decodeArrayBufferToAudio = async(arrayBuffer) => { +const decodeArrayBufferToAudio = async (arrayBuffer) => { const audioContext = new AudioContext(); const audioBuffer = await audioContext.decodeAudioData(arrayBuffer); return audioBuffer; @@ -6,7 +6,7 @@ const decodeArrayBufferToAudio = async(arrayBuffer) => { const mergeChannels = async (audioBuffer) => { const numberOfChannels = 1; - const offlineCtx = new OfflineAudioContext(numberOfChannels,audioBuffer.length,audioBuffer.sampleRate); + const offlineCtx = new OfflineAudioContext(numberOfChannels, audioBuffer.length, audioBuffer.sampleRate); const source = offlineCtx.createBufferSource(); const merger = offlineCtx.createChannelMerger(2); @@ -19,36 +19,48 @@ const mergeChannels = async (audioBuffer) => { return renderBuffer; } -const parsePeaks = (audioBuffer: AudioBuffer, parsedBufferSize: number): Promise => new Promise((resolve, reject)=> { - const {length } = audioBuffer; - const sampleSize = length / parsedBufferSize; - const sampleStep = Math.floor(sampleSize / 10) || 1; - - const channelData :Float32Array = audioBuffer.getChannelData(0); - let parsedData: number[] = []; - - Array(parsedBufferSize).fill(0).forEach((v, newPeakIndex) => { - const start = Math.floor(newPeakIndex * sampleSize); - const end = Math.floor(start + sampleSize); - let min = channelData[0]; - let max = channelData[0]; - - for (let sampleIndex = start; sampleIndex < end; sampleIndex += sampleStep) { - const v = channelData[sampleIndex]; - - if (v > max) max = v; - else if (v < min) min = v; - } - - parsedData[2 * newPeakIndex] = max; - parsedData[2 * newPeakIndex + 1] = min; - }); - - resolve(parsedData); +const parsePeaks = (audioBuffer: AudioBuffer, parsedBufferSize: number): Promise => new Promise((resolve, reject) => { + const { length } = audioBuffer; + const sampleSize = length / parsedBufferSize; + const sampleStep = Math.floor(sampleSize / 10) || 1; + + const channelData: Float32Array = audioBuffer.getChannelData(0); + let parsedData: number[] = []; + + Array(parsedBufferSize).fill(0).forEach((v, newPeakIndex) => { + const start = Math.floor(newPeakIndex * sampleSize); + const end = Math.floor(start + sampleSize); + let min = channelData[0]; + let max = channelData[0]; + + for (let sampleIndex = start; sampleIndex < end; sampleIndex += sampleStep) { + const v = channelData[sampleIndex]; + + if (v > max) max = v; + else if (v < min) min = v; + } + + parsedData[2 * newPeakIndex] = max; + parsedData[2 * newPeakIndex + 1] = min; }); + resolve(parsedData); +}); + +const getDecibel = (array: Float32Array) => { + var sumOfSquares = 0; + var average = 0; + for (var i = 0; i < array.length; i++) { + sumOfSquares += array[i] ** 2; + } + average = 10 * Math.log10(sumOfSquares / array.length); + return average; +} + + export { decodeArrayBufferToAudio, mergeChannels, - parsePeaks + parsePeaks, + getDecibel } diff --git a/frontend/src/common/util/CopyUtil.ts b/frontend/src/common/util/CopyUtil.ts index 960f066f..b8e5091e 100644 --- a/frontend/src/common/util/CopyUtil.ts +++ b/frontend/src/common/util/CopyUtil.ts @@ -2,17 +2,25 @@ import { Source, Track, TrackSection } from '@model'; const copySection = (trackSection: TrackSection): TrackSection => { const newTrackSection = new TrackSection({ + id: trackSection.id, sourceId: trackSection.sourceId, trackId: trackSection.trackId, channelStartTime: trackSection.channelStartTime, channelEndTime: trackSection.channelEndTime, - parsedChannelStartTime: trackSection.parsedChannelStartTime, - parsedChannelEndTime: trackSection.parsedChannelEndTime, - trackStartTime: 0, - audioStartTime: 0 + trackStartTime: trackSection.trackStartTime + // effectList: trackSection.effectList, // Effect 기능 구현시 추가 }); return newTrackSection; -} +}; -export { copySection } \ No newline at end of file +const copyTrack = (track: Track): Track => { + const newTrack = new Track({ + id: track.id, + trackSectionList: track.trackSectionList.map((section) => copySection(section)) + }); + + return newTrack; +}; + +export { copySection, copyTrack }; diff --git a/frontend/src/common/util/DragUtil.ts b/frontend/src/common/util/DragUtil.ts new file mode 100644 index 00000000..f1c3303b --- /dev/null +++ b/frontend/src/common/util/DragUtil.ts @@ -0,0 +1,70 @@ +import { Controller, CommandController } from '@controllers'; +import { WidthUtil, TimeUtil, ValidUtil } from '@util'; + +const showAfterimage = (afterimage: HTMLElement, trackId: number, trackContainerWidth: number, currentCursorPosition: number): void => { + const dragData = Controller.getSectionDragStartData(); + + if (!dragData) return; + + const maxTrackPlayTime = Controller.getMaxTrackPlayTime(); + const secondPerPixel = WidthUtil.getPixelPerSecond(trackContainerWidth, maxTrackPlayTime); + let { offsetLeft, prevCursorTime } = dragData; + + + if (dragData.trackSection.id === 0) { + prevCursorTime = currentCursorPosition / secondPerPixel; + dragData.prevCursorTime = prevCursorTime; + dragData.trackSection.trackStartTime = prevCursorTime - (dragData.trackSection.length / 2); + } + + + if (!offsetLeft || !prevCursorTime) return; + const track = Controller.getTrack(trackId); + const movingCursorTime = TimeUtil.calculateTimeOfCursorPosition(offsetLeft, currentCursorPosition, trackContainerWidth, maxTrackPlayTime); + + + if (!dragData.trackSection || !track) return; + + let newTrackStartTime = movingCursorTime - (prevCursorTime - dragData.trackSection.trackStartTime); + + if (newTrackStartTime < 0) { + newTrackStartTime = 0; + } + const newTrackEndTime = newTrackStartTime + dragData.trackSection.length; + + if (ValidUtil.checkEnterTrack(dragData.trackSection, track.trackSectionList, newTrackStartTime, newTrackEndTime)) { + afterimage.style.display = 'none'; + afterimage.style.left = `0px`; + afterimage.style.width = `0px`; + return; + } else { + afterimage.style.display = 'block'; + } + + afterimage.style.left = `${newTrackStartTime * secondPerPixel}px`; + afterimage.style.width = `${dragData.trackSection.length * secondPerPixel}px`; +} + +const dropTrackSection = (trackId: number, currentCursorPosition: number, trackContainerWidth: number) => { + const dragData = Controller.getSectionDragStartData(); + + if (!dragData) return; + const { prevCursorTime, offsetLeft } = dragData; + const maxTrackPlayTime = Controller.getMaxTrackPlayTime(); + + if (dragData.trackSection.id === 0 && prevCursorTime) { + dragData.trackSection.trackStartTime = prevCursorTime - (dragData.trackSection.length / 2); + dragData.trackSection.trackId = trackId; + + } + + if (!offsetLeft || !prevCursorTime) return; + + const movingCursorTime = TimeUtil.calculateTimeOfCursorPosition(offsetLeft, currentCursorPosition, trackContainerWidth, maxTrackPlayTime); + + Controller.resetFocus(); + CommandController.executeMoveCommand(dragData.trackSection.trackId, trackId, dragData.trackSection, movingCursorTime, prevCursorTime); +} + + +export { showAfterimage, dropTrackSection }; \ No newline at end of file diff --git a/frontend/src/common/util/MarkerEventUtil.ts b/frontend/src/common/util/MarkerEventUtil.ts index 4ca0b3a0..62efb631 100644 --- a/frontend/src/common/util/MarkerEventUtil.ts +++ b/frontend/src/common/util/MarkerEventUtil.ts @@ -1,24 +1,34 @@ -import { getCursorPosition } from './PlayBarUtil'; +import { getDifferenceWidth } from './WidthUtil'; +import { splitTime, calculateTimeOfCursorPosition } from './TimeUtil'; import { Controller } from '@controllers'; -const mousemoveMarkerListener: Function = (element: HTMLElement, defaultStartX: number, mainWidth: number) => (e: Event): void => { +const mousemoveMarkerListener: Function = (element: HTMLElement, elementLeftX: number, elementWidth: number, currentScrollAmount: number, trackPlayTime: number) => (e: MouseEvent): void => { if (!element) return; - const cursorPosition = e.pageX; - const [minute, second, milsecond, location, totalCursorTime] = getCursorPosition(defaultStartX, cursorPosition, mainWidth); + const cursorPosition = e.pageX; + const scrolledCursorPosition = cursorPosition + currentScrollAmount + const timeOfCursorPosition = calculateTimeOfCursorPosition(elementLeftX, scrolledCursorPosition, elementWidth, trackPlayTime); + const [minute, second, milsecond] = splitTime(timeOfCursorPosition); + const offesetOfCursorPosition = getDifferenceWidth(elementLeftX, cursorPosition); if (minute < 0 && second < 0) return; - Controller.changeCurrentPosition(location); - Controller.changeCursorTime(minute.toString(), second.toString(), milsecond.toString()); - Controller.changeTotalCursorTime(totalCursorTime); + Controller.changeCurrentPosition(offesetOfCursorPosition); + Controller.changeCursorStringTime(minute, second, milsecond); + Controller.changeCursorNumberTime(timeOfCursorPosition); }; const clickMarkerListener = (element: HTMLElement) => (e: Event): void => { if (!element) return; - const [currentPosition, totalCursorTime] = Controller.getCurrentPosition(); + const [currentPosition, cursorNumberTime] = Controller.getCurrentPosition(); + + Controller.changeMarkerPlayStringTime(cursorNumberTime); + Controller.changeMarkerNumberTime(cursorNumberTime); + + if (!Controller.getIsPauseState()) { + + Controller.audioCursorPlay(); + } - Controller.cursorChangeMarkerTime(totalCursorTime); - Controller.resetPlayTime(totalCursorTime); if (currentPosition < 0) return; element.style.left = `${currentPosition}px`; }; diff --git a/frontend/src/common/util/PlayBarUtil.ts b/frontend/src/common/util/PlayBarUtil.ts deleted file mode 100644 index 6ca297bb..00000000 --- a/frontend/src/common/util/PlayBarUtil.ts +++ /dev/null @@ -1,59 +0,0 @@ -const SECTION_TIME = 15; -const DEFAULT_TIME = 300; - -const setTime = (time: number): number[] => { - let newMinute = Math.floor(time / 60); - let newSecond = time % 60; - let newMilsecond = Number((newSecond % 1).toFixed(3).split('.')[1]); - - return [newMinute, Math.floor(newSecond), newMilsecond]; -}; - -const getOnePixel = (mainWidth: number): number[] => { - //1초에 몇 픽셀인지 - const oneSecond: number = parseFloat((Number(mainWidth) / Number(DEFAULT_TIME)).toFixed(2)); - - //1px에 몇초인지 - const onePixel: number = parseFloat((1 / oneSecond).toFixed(2)); - - return [onePixel, oneSecond]; -}; - -const getSomePixel = (time: number): number => { - const playBarElement: HTMLElement | null = document.querySelector('audi-playbar'); - const mainWidth = playBarElement?.getBoundingClientRect().right - playBarElement?.getBoundingClientRect().left; - - const second = time / 1000; - //1초에 몇 픽셀인지 - const oneSecond: number = parseFloat((Number(mainWidth) / Number(DEFAULT_TIME)).toFixed(2)); - - const somePixel: number = oneSecond * second; - - return somePixel; -}; - -const getCursorPosition = (defaultStartX: number, currentX: number, mainWidth: number): number[] => { - const [onePixel] = getOnePixel(mainWidth); - const differenceWidth = currentX - defaultStartX; - const cursorTime = onePixel * differenceWidth; - - const [newMinute, newSecond, newMilsecond] = setTime(cursorTime); - - return [newMinute, newSecond, newMilsecond, differenceWidth, cursorTime]; -}; - -const getStringTime = (time: number): string[] => { - let gap = Math.round(time / SECTION_TIME); - const timeArray: string[] = []; - - for (let second = 0; second <= time; second += gap) { - const [newMinute, newSecond] = setTime(second); - - const stringTime = `${newMinute.toString().padStart(2, '0')}:${newSecond.toString().padStart(2, '0')}`; - - timeArray.push(stringTime); - } - return timeArray; -}; - -export { getStringTime, getCursorPosition, setTime, getSomePixel }; diff --git a/frontend/src/common/util/SectionEffectListUtil.ts b/frontend/src/common/util/SectionEffectListUtil.ts new file mode 100644 index 00000000..1b5d0f24 --- /dev/null +++ b/frontend/src/common/util/SectionEffectListUtil.ts @@ -0,0 +1,19 @@ +const hideEffectList = (): void => { + const effectListElement = document.querySelector('audi-section-effect-list'); + const sourceListElement = document.querySelector('audi-source-list'); + if (!effectListElement || !sourceListElement) return; + + effectListElement.hide(); + sourceListElement.show(); +}; + +const showEffectList = (): void => { + const effectListElement = document.querySelector('audi-section-effect-list'); + const sourceListElement = document.querySelector('audi-source-list'); + if (!effectListElement || !sourceListElement) return; + + effectListElement.show(); + sourceListElement.hide(); +}; + +export { hideEffectList, showEffectList }; diff --git a/frontend/src/common/util/TimeUtil.ts b/frontend/src/common/util/TimeUtil.ts new file mode 100644 index 00000000..f77d4f79 --- /dev/null +++ b/frontend/src/common/util/TimeUtil.ts @@ -0,0 +1,54 @@ +import { getSecondPerPixel, getDifferenceWidth } from './WidthUtil'; + +const SECTION_TIME = 15; + +const splitTime = (time: number): number[] => { + let newMinute = Math.floor(time / 60); + let newSecond = time % 60; + let newMilsecond = Number((newSecond % 1).toFixed(3).split('.')[1]); + + return [newMinute, Math.floor(newSecond), newMilsecond]; +}; + +const getPlayBarTimes = (time: number): string[] => { + const offsetOfPlayBarTimes = getOffsetOfPlayBarTimes(time); + const timeArray: string[] = []; + + for (let second = 0; second <= time; second += offsetOfPlayBarTimes) { + const [newMinute, newSecond] = splitTime(second); + + const stringTime = `${newMinute.toString().padStart(2, '0')}:${newSecond.toString().padStart(2, '0')}`; + + timeArray.push(stringTime); + } + return timeArray; +}; + +const getOffsetOfPlayBarTimes = (time: number): number => { + const offsetOfPlayBarTimes = Math.round(time / SECTION_TIME); + return offsetOfPlayBarTimes; +} + +const getNumberTime = (startX: number, currentX: number, mainWidth: number, time:number): number => { + const secondPerPixel = getSecondPerPixel(mainWidth, time); + const differenceWidth = getDifferenceWidth(startX, currentX); + const cursorNumberTime = secondPerPixel * differenceWidth; + + return cursorNumberTime; +}; + +const getStringTime = (minute: number, second: number, milsecond: number): string => { + const stringTime: string = `${minute.toString().padStart(2, '0')}:${second.toString().padStart(2, '0')}:${milsecond.toString().padStart(3, '0')}`; + + return stringTime; +}; + +const calculateTimeOfCursorPosition = (startX: number, currentX: number, mainWidth: number, time: number): number => { + const secondPerPixel = getSecondPerPixel(mainWidth, time); + const differenceWidth = getDifferenceWidth(startX, currentX); + const cursorTime = secondPerPixel * differenceWidth; + + return cursorTime; +}; + +export { getOffsetOfPlayBarTimes, splitTime, getPlayBarTimes, getNumberTime, getStringTime, calculateTimeOfCursorPosition }; diff --git a/frontend/src/common/util/ValidUtil.ts b/frontend/src/common/util/ValidUtil.ts new file mode 100644 index 00000000..33ff6821 --- /dev/null +++ b/frontend/src/common/util/ValidUtil.ts @@ -0,0 +1,10 @@ +import { TrackSection } from '@model' +export const checkEnterTrack = (trackSection: TrackSection, trackSectionList: TrackSection[], newTrackStartTime: number, newTrackEndTime: number): boolean => { + const nearLeftSection = trackSectionList.some( + (section) => section.id !== trackSection.id && section.trackStartTime <= newTrackStartTime && section.trackStartTime + section.length >= newTrackStartTime + ); + const nearRightSection = trackSectionList.some( + (section) => section.id !== trackSection.id && newTrackStartTime <= section.trackStartTime && section.trackStartTime <= newTrackEndTime); + + return nearLeftSection || nearRightSection; +} \ No newline at end of file diff --git a/frontend/src/common/util/WidthUtil.ts b/frontend/src/common/util/WidthUtil.ts new file mode 100644 index 00000000..a8c69094 --- /dev/null +++ b/frontend/src/common/util/WidthUtil.ts @@ -0,0 +1,29 @@ +const getPixelPerSecond = (elementWidth: number, trackPlayTime: number): number => { + const pixelPerSecond: number = parseFloat((Number(elementWidth) / Number(trackPlayTime)).toFixed(2)); + return pixelPerSecond; +}; + +const getSecondPerPixel = (elementWidth: number, trackPlayTime: number): number => { + const pixelPerSecond = getPixelPerSecond(elementWidth, trackPlayTime); + const secondPerPixel: number = parseFloat((1 / pixelPerSecond).toFixed(2)); + return secondPerPixel; +}; + +const getPerPixel = (time: number, trackPlayTime: number): number => { + const playBarElement: HTMLElement | null = document.querySelector('audi-playbar'); + if (!playBarElement) return 0; + + const playBarWidth = playBarElement.getBoundingClientRect().right - playBarElement.getBoundingClientRect().left; + const second = time / 1000; + const pixelPerSecond: number = getPixelPerSecond(playBarWidth, trackPlayTime); + const somePixel: number = pixelPerSecond * second; + + return somePixel; +}; + +const getDifferenceWidth = (startX: number, currentX: number): number => { + const differenceWidth = Math.abs(currentX - startX); + return differenceWidth; +}; + +export { getPixelPerSecond, getPerPixel, getSecondPerPixel, getDifferenceWidth }; diff --git a/frontend/src/common/util/compressor.ts b/frontend/src/common/util/compressor.ts index b6338101..45e53a8b 100644 --- a/frontend/src/common/util/compressor.ts +++ b/frontend/src/common/util/compressor.ts @@ -1,52 +1,40 @@ +import { Controller } from '@controllers'; import { CompressorOption } from '@types'; import * as lamejs from 'lamejs'; -export const saveFile = async (arrayBuffer: ArrayBuffer, options: CompressorOption) => { - const audioCtx: AudioContext = new AudioContext(); - const buffer = await audioCtx.decodeAudioData(arrayBuffer); - const name = `${options.fileName}.${options.extention}`; +const sampleRate = 48000; +const numberOfChannels = 2; + +export const saveFile = async (options: CompressorOption) => { + const arrayBufferList: ArrayBuffer[] = []; + const trackList = Controller.getTrackList(); - const wavBuffer = await makeWaveBlob(buffer); + for (let i = 0; i < trackList.length; i++) { + const trackArrayBuffer = await getTrackArrayBuffer(trackList[i].id); + + if (trackArrayBuffer) { + arrayBufferList.push(trackArrayBuffer); + } + } + + const mergedBuffer = await mergeTrackArrayBuffer(arrayBufferList); + const leftChannel = mergedBuffer.getChannelData(0); + const righttChannel = mergedBuffer.getChannelData(1); + const length = leftChannel.length; + const wavBuffer = ChannelDataToWave([leftChannel, righttChannel], length); + const name = `${options.fileName}.${options.extention}`; if (options.extention === 'wav') { const wavFile = new Blob([wavBuffer], { type: `audio/wav` }); makeDownload(wavFile, name); } else if (options.extention === 'mp3' && options.quality) { - const mp3Blob = await makeMP3(wavBuffer, options.quality); + const mp3Blob = makeMP3(wavBuffer, options.quality); makeDownload(mp3Blob, name); } } -const makeWaveBlob = async (buffer: AudioBuffer) => { - const offlineAudioCtx: OfflineAudioContext = new OfflineAudioContext({ - numberOfChannels: 2, - length: (44100) * buffer.duration, - sampleRate: 44100, - }); - const soundSource = offlineAudioCtx.createBufferSource(); - soundSource.buffer = buffer; - - const compressor = offlineAudioCtx.createDynamicsCompressor(); - compressor.threshold.setValueAtTime(-20, offlineAudioCtx.currentTime); - compressor.knee.setValueAtTime(30, offlineAudioCtx.currentTime); - compressor.ratio.setValueAtTime(5, offlineAudioCtx.currentTime); - compressor.attack.setValueAtTime(.05, offlineAudioCtx.currentTime); - compressor.release.setValueAtTime(.25, offlineAudioCtx.currentTime); - - const gainNode = offlineAudioCtx.createGain(); - gainNode.gain.setValueAtTime(1, offlineAudioCtx.currentTime); - soundSource.connect(compressor); - compressor.connect(gainNode); - gainNode.connect(offlineAudioCtx.destination); - soundSource.start(0); - soundSource.loop = false; - - const renderedBuffer = await offlineAudioCtx.startRendering(); - return bufferToWave(renderedBuffer, offlineAudioCtx.length); -} - -const bufferToWave = (abuffer: AudioBuffer, len: number) => { - const numOfChan: number = abuffer.numberOfChannels; +const ChannelDataToWave = (channelDatas: Float32Array[], len: number) => { + const numOfChan: number = numberOfChannels; const length: number = len * numOfChan * 2 + 44; const buffer: ArrayBuffer = new ArrayBuffer(length); const view: DataView = new DataView(buffer); // buffer를 다룰 때 사용 @@ -76,16 +64,17 @@ const bufferToWave = (abuffer: AudioBuffer, len: number) => { setUint32(16); // length = 16 setUint16(1); // PCM (uncompressed) setUint16(numOfChan); - setUint32(abuffer.sampleRate); - setUint32(abuffer.sampleRate * 2 * numOfChan); // avg. bytes/sec + setUint32(sampleRate); + setUint32(sampleRate * 2 * numOfChan); // avg. bytes/sec setUint16(numOfChan * 2); // block-align setUint16(16); // 16-bit (hardcoded in this demo) setUint32(0x61746164); // "data" - chunk setUint32(length - pos - 4); // chunk length - for (let i = 0; i < abuffer.numberOfChannels; i++) - channels.push(abuffer.getChannelData(i)); + for (let i = 0; i < channelDatas.length; i++) { + channels.push(channelDatas[i]); + } while (pos < length) { for (let i = 0; i < numOfChan; i++) { @@ -100,10 +89,78 @@ const bufferToWave = (abuffer: AudioBuffer, len: number) => { return buffer; } -const makeMP3 = async (wavBuffer: ArrayBuffer, quality: number) => { +const getTrackArrayBuffer = async (trackId: number) => { + const track = Controller.getTrack(trackId); + if (!track || track.trackSectionList.length === 0) return null; + + const lastSection = track.trackSectionList[track.trackSectionList.length - 1]; + + const bufferLength = (lastSection.trackStartTime + lastSection.length) * sampleRate; + + const leftChannel = new Float32Array(bufferLength); + const rightChannel = new Float32Array(bufferLength); + let offset = 0; + + for (let i = 0; i < track.trackSectionList.length; i++) { + const section = track.trackSectionList[i]; + const audioSource = Controller.getSourceBySourceId(section.sourceId); + + if (!audioSource) return; + + const duration = section.trackStartTime + section.length; + const offlineCtx = new OfflineAudioContext(numberOfChannels, duration * sampleRate, sampleRate); + const source = offlineCtx.createBufferSource(); + + source.buffer = audioSource.buffer; + source.connect(offlineCtx.destination); + source.start(section.trackStartTime, section.channelStartTime, duration); + + const renderBuffer = await offlineCtx.startRendering(); + const left = renderBuffer.getChannelData(0); + const right = renderBuffer.getChannelData(1); + + while (offset < left.length) { + leftChannel[offset] = left[offset]; + rightChannel[offset] = right[offset]; + offset++; + } + } + const len = leftChannel.length; + const wave = ChannelDataToWave([leftChannel, rightChannel], len); + + return wave; +} + +const mergeTrackArrayBuffer = async (arrayBufferList: ArrayBuffer[]) => { + const audioBuffers: AudioBuffer[] = []; + let maxLength = 0; + for (let i = 0; i < arrayBufferList.length; i++) { + const audioCtx: AudioContext = new AudioContext(); + const audioBuffer = await audioCtx.decodeAudioData(arrayBufferList[i]); + + audioBuffers.push(audioBuffer); + maxLength = Math.max(maxLength, audioBuffer.length); + } + const numberOfChannels = 2; + const offlineCtx = new OfflineAudioContext(numberOfChannels, maxLength, sampleRate); + const merger = offlineCtx.createChannelMerger(numberOfChannels); + + audioBuffers.forEach(buffer => { + const source = offlineCtx.createBufferSource(); + source.buffer = buffer; + source.connect(merger); + source.connect(offlineCtx.destination); + source.start(); + }) + const renderBuffer = await offlineCtx.startRendering(); + + return renderBuffer; +} + +const makeMP3 = (wavBuffer: ArrayBuffer, quality: number) => { const mp3Data: Int8Array[] = []; - const mp3encoder = new lamejs.Mp3Encoder(2, 44100, quality); + const mp3encoder = new lamejs.Mp3Encoder(numberOfChannels, sampleRate, quality); const wavHdr = lamejs.WavHeader.readHeader(new DataView(wavBuffer)); const wavSamples = new Int16Array(wavBuffer, wavHdr.dataOffset, wavHdr.dataLen / 2); @@ -117,7 +174,7 @@ const makeMP3 = async (wavBuffer: ArrayBuffer, quality: number) => { } const mp3buf = mp3encoder?.encodeBuffer(leftData, rightData); if (mp3buf.length > 0) { - mp3Data.push(mp3buf);//new Int8Array(mp3buf)); + mp3Data.push(mp3buf); } const d = mp3encoder?.flush(); diff --git a/frontend/src/common/util/index.js b/frontend/src/common/util/index.js index 42cabe60..0b70ad1f 100644 --- a/frontend/src/common/util/index.js +++ b/frontend/src/common/util/index.js @@ -1,7 +1,11 @@ export * as EventUtil from './EventUtil'; export * as FileUtil from './FileUtil'; export * as AudioUtil from './AudioUtil'; -export * as PlayBarUtil from './PlayBarUtil'; +export * as WidthUtil from './WidthUtil'; export * as MarkerEventUtil from './MarkerEventUtil'; export * as CopyUtil from './CopyUtil'; +export * as SectionEffectListUtil from './SectionEffectListUtil'; +export * as TimeUtil from './TimeUtil'; export { saveFile } from './compressor'; +export * as ValidUtil from './ValidUtil' +export * as DragUtil from './DragUtil'; diff --git a/frontend/src/components/App/App.ts b/frontend/src/components/App/App.ts index b4a9ff8d..84fdbe60 100644 --- a/frontend/src/components/App/App.ts +++ b/frontend/src/components/App/App.ts @@ -1,6 +1,6 @@ import './App.scss'; import { EventDataType, eventTypes, EventTargetDataType, KeyBoard, CursorType } from '@types'; -import { Controller } from "@controllers"; +import { Controller, CommandController } from "@controllers"; (() => { const App = class extends HTMLElement { @@ -38,7 +38,6 @@ import { Controller } from "@controllers"; } KeyDownListener(e): void { - const { } = Controller const isCtrl = Controller.getCtrlIsPressed(); if (e.which === KeyBoard.CTRL) { @@ -52,35 +51,45 @@ import { Controller } from "@controllers"; Controller.setCursorMode(CursorType.SELECT_MODE); } else if (e.which === KeyBoard.DELETE && !isCtrl) { - Controller.deleteCommand(); + CommandController.executeDeleteCommand(); } else if (e.which === KeyBoard.LEFT && !isCtrl) { - // console.log('왼쪽'); + e.preventDefault() + Controller.audioFastRewind(); } else if (e.which === KeyBoard.RIGHT && !isCtrl) { - // console.log('오른쪽'); + e.preventDefault() + Controller.audioFastForward(); + } + else if (e.which === KeyBoard.LEFT_BRACKET && !isCtrl) { + Controller.audioSkipPrev(); + } + else if (e.which === KeyBoard.RIGHT_BRACKET && !isCtrl) { + Controller.audioSkipNext(); } else if (e.which === KeyBoard.SPACE && !isCtrl) { - // console.log('스페이스바'); + e.preventDefault() + Controller.audioPlayOrPause(); } else if (e.which === KeyBoard.C && isCtrl) { Controller.setClipBoard(); } else if (e.which === KeyBoard.X && isCtrl) { - // console.log('잘라내기'); + CommandController.executeCutCommand(); } else if (e.which === KeyBoard.V && isCtrl) { - // console.log('붙여넣기'); + CommandController.executePasteCommand(); } else if (e.which === KeyBoard.Z && isCtrl) { - Controller.undoCommand(); + CommandController.executeUndoCommand(); } else if (e.which === KeyBoard.Y && isCtrl) { - Controller.redoCommand(); + CommandController.executeRedoCommand(); } } ctrlKeyUpListener(e): void { + e.preventDefault() if (e.which === KeyBoard.CTRL) { Controller.setCtrlIsPressed(false); } @@ -88,10 +97,11 @@ import { Controller } from "@controllers"; eventListenerForRegistrant(e): void { const { target } = e; - if (!target || !this.isEventTarget(target) || !this.eventListenerCollectors) return; + if (!target || !this.isEventTarget(target) && !this.isDelegationEvent(target)) return; const eventType = e.type; - const eventKey = target.getAttribute('event-key'); + const eventKey = this.parseEventKey(target); + if (!eventKey) return; this.excuteEventListenerForTarget(eventType, eventKey, e); } @@ -101,6 +111,22 @@ import { Controller } from "@controllers"; return eventKey ? true : false; } + isDelegationEvent(eventTarget: HTMLElement): Boolean { + const eventDelegation = eventTarget.getAttribute('event-delegation'); + return eventDelegation === '' ? true : false; + } + + parseEventKey(eventTarget: HTMLElement) { + const eventDelegation = eventTarget.getAttribute('event-delegation'); + + if (eventDelegation === '') { + const newEventTarget = eventTarget.closest('.delegation'); + return newEventTarget?.getAttribute('event-key'); + } else { + return eventTarget.getAttribute('event-key'); + } + } + excuteEventListenerForTarget(eventType: string, eventKey: string, e: Event): void { if (!this.eventListenerCollectors) return; diff --git a/frontend/src/components/AudioMeter/AudioMeter.scss b/frontend/src/components/AudioMeter/AudioMeter.scss new file mode 100644 index 00000000..7e2c4e63 --- /dev/null +++ b/frontend/src/components/AudioMeter/AudioMeter.scss @@ -0,0 +1,46 @@ +@use '@style/variables.scss'as variables; + +#audio-meter-container { + display: flex; + flex-direction: column; + background-color: black; +} + +#audio-meter-bar { + display: flex; + flex-direction: column; + height: 12px; + border: 1px solid gray; + border-radius: 3px; + margin: 10px 0; + background-color: variables.$theme-color; +} + +#audio-meter-fill { + margin: auto 0; + height: 100%; + border-radius: 3px; + width: 0px; + box-shadow: 0 0 3px rgba(0,0,0,.25), + inset 1px 1px 1px rgba(255,255,255,.75), + inset -1px -1px 1px rgba(0,0,0,.4); + bottom: 0; + color: rgba(0,0,0,.75); +} + +.decibel-markers{ + display: flex; + margin-top: auto; + white-space: nowrap; + font-size: 12px; + width: calc(100% - 10px); + + & .decibel-level { + width: calc(100% / 19); + display: flex; + flex-direction: column; + justify-content: flex-end; + align-items: center; + color: rgb(160,160,160); + } +} diff --git a/frontend/src/components/AudioMeter/AudioMeter.ts b/frontend/src/components/AudioMeter/AudioMeter.ts new file mode 100644 index 00000000..3813161d --- /dev/null +++ b/frontend/src/components/AudioMeter/AudioMeter.ts @@ -0,0 +1,47 @@ +import './AudioMeter.scss'; + +(() => { + const AudioMeter = class extends HTMLElement { + + constructor() { + super(); + } + + connectedCallback(): void { + this.render(); + } + + render(): void { + this.innerHTML = ` +
+
+
+
+
+ ${this.getDecibelMarkers()} +
+
+ `; + } + + getDecibelMarkers(): string { + let mindb = -68; + let acc = `
+ -INF +
`; + + while (mindb <= 0) { + acc += `
+ ${mindb} +
` + mindb += 4; + } + return acc; + } + }; + customElements.define('audi-audio-meter', AudioMeter); +})(); + + + +export { }; diff --git a/frontend/src/components/AudioTrack/AudioTrack.scss b/frontend/src/components/AudioTrack/AudioTrack.scss index 6a195f97..2250da45 100644 --- a/frontend/src/components/AudioTrack/AudioTrack.scss +++ b/frontend/src/components/AudioTrack/AudioTrack.scss @@ -1,14 +1,14 @@ +@import '@style/variables'; + audi-audio-track:not(:defined) { display: hidden; } .audio-track-container{ - display: flex; - flex-direction: column; position: relative; width: 100%; min-height: 150px; - padding: 10px; + padding: 10px 0px; background-color: #000000; box-sizing: border-box; @@ -17,26 +17,49 @@ audi-audio-track:not(:defined) { flex-wrap: nowrap; padding: 10px 0px; position: relative; - width: 100%; - min-height: 150px; + min-height: 175px; border-top: 1px solid #ffffff; border-bottom: 1px solid #ffffff; box-sizing: border-box; + overflow: visible; + + .cut-line { + position: absolute; + width: 1px; + height: 100%; + + top:0; + left: 0px; + border-left: 1px dashed white; + z-index: 999; + + pointer-events: none; + } + + .audio-track-afterimage { + display:none; + height: calc(100% - 20px); + border-radius: 5px; + top:10px; + left: 0px; + border: 1px dashed gray; + background-color: rgba(40,40,40, 0.5); + position: absolute; + pointer-events: none; + } - .audio-track-dropzone{ + .track-select-line { position: absolute; - top: 0; - left: 0; - width: 100%; + display:none; + width: 1px; height: 100%; - background-color: rgba(60,60,60, 0.4); - border: 1px dotted #ffffff; - z-index: 100; - - &.focus{ - background-color: rgba(43, 130, 211, 0.5); - box-shadow: rgb(43, 130, 211) 0 0 15px 5px; - } + + top:0; + left: 0px; + border-left: 1px dashed white; + z-index: 999; + + pointer-events: none; } } @@ -51,9 +74,20 @@ audi-audio-track:not(:defined) { transform: translate3d(-50%, -50%, 0); text-align:center; z-index: 50; - + pointer-events: none; span{ color:rgb(158, 158, 158) } } } + +@media screen and (max-width: 1680px) { + .audio-track-container{ + min-height: 100px; + padding: 5px 0; + + .audio-track-area{ + min-height: 125px; + } + } +} diff --git a/frontend/src/components/AudioTrack/AudioTrack.ts b/frontend/src/components/AudioTrack/AudioTrack.ts index 72f54822..1c3b7ef3 100644 --- a/frontend/src/components/AudioTrack/AudioTrack.ts +++ b/frontend/src/components/AudioTrack/AudioTrack.ts @@ -1,23 +1,35 @@ import { Controller } from '@controllers'; -import { EventKeyType, EventType, StoreChannelType } from "@types"; -import { EventUtil } from "@util"; -import { storeChannel } from "@store"; -import { TrackSection } from "@model"; -import "./AudioTrack.scss"; +import { EventKeyType, EventType, StoreChannelType } from '@types'; +import { EventUtil, WidthUtil, DragUtil } from '@util'; +import { storeChannel } from '@store'; +import { TrackSection, SectionDragStartData, SelectTrackData } from '@model'; +import './AudioTrack.scss'; (() => { const AudioTrack = class extends HTMLElement { private trackId: number; - private trackMessage: HTMLDivElement | null; - private trackDropzoneElement: HTMLDivElement | null; + private trackMessageElement: HTMLDivElement | null; private trackSectionList: TrackSection[]; + private trackAreaElement: HTMLDivElement | null; + private trackScrollAreaElement: HTMLDivElement | null; + private afterimageElement: HTMLDivElement | null; + private trackWidth: number; + private maxTrackWidth: number; + private maxTrackPlayTime: number; + private sectionDragData: SectionDragStartData | null; constructor() { super(); this.trackId = 0; - this.trackMessage = null; - this.trackDropzoneElement = null; + this.trackMessageElement = null; this.trackSectionList = []; + this.trackAreaElement = null; + this.trackScrollAreaElement = null; + this.afterimageElement = null; + this.trackWidth = 0; + this.maxTrackWidth = 0; + this.maxTrackPlayTime = 0; + this.sectionDragData = null; } static get observedAttributes(): string[] { @@ -40,6 +52,7 @@ import "./AudioTrack.scss"; this.render(); this.initElement(); this.initEvent(); + this.initPosition(); this.subscribe(); } catch (e) { console.log(e); @@ -48,123 +61,206 @@ import "./AudioTrack.scss"; render(): void { this.innerHTML = ` -
-
+
+
${this.getTrackSectionList()}
Drag & Drop
-
-
+
+
+
+
`; } - + getTrackSectionList(): string { - return this.trackSectionList.reduce((acc, trackSection, idx) => - acc += `` - , ""); + return this.trackSectionList.reduce( + (acc, trackSection) => (acc += ``), + '' + ); } initElement(): void { - this.trackMessage = this.querySelector('.audio-track-message'); - this.trackDropzoneElement = this.querySelector('.audio-track-dropzone'); + this.maxTrackWidth = Controller.getMaxTrackWidth(); + this.maxTrackPlayTime = Controller.getMaxTrackPlayTime(); + this.trackMessageElement = this.querySelector('.audio-track-message'); + this.trackAreaElement = this.querySelector('.audio-track-area'); + this.trackScrollAreaElement = document.querySelector('.audi-main-audio-track-scroll-area'); + this.afterimageElement = this.querySelector(`#afterimage-${this.trackId}`); + this.trackWidth = this.calculateTrackWidth(); + } + + calculateTrackWidth(): number { + let trackWidth = 0; + if (this.trackAreaElement) { + trackWidth = this.trackAreaElement.getBoundingClientRect().right - this.trackAreaElement.getBoundingClientRect().left; + } + return trackWidth; } initEvent(): void { EventUtil.registerEventToRoot({ - eventTypes: [EventType.dragover, EventType.drop, EventType.dragenter, EventType.dragleave], - eventKey: EventKeyType.AUDIO_TRACK_DRAGOVER_DROP + this.trackId, - listeners: [this.trackDragoverListener, this.trackDropListener, this.trackDragenterListener, this.trackDragleaveListener], + eventTypes: [EventType.click, EventType.dragover, EventType.dragleave, EventType.drop, EventType.dragenter], + eventKey: EventKeyType.AUDIO_TRACK_AREA_MULTIPLE + this.trackId, + listeners: [this.trackClickListener, this.dragoverAudioTrackListener, this.dragleaveAudioTrackListener, this.dropAudioTrackListener, this.dragenterAudioTrackListener], bindObj: this }); EventUtil.registerEventToRoot({ eventTypes: [EventType.click], - eventKey: EventKeyType.FOCUS_RESET_CLICK, + eventKey: EventKeyType.FOCUS_RESET_CLICK + this.trackId, listeners: [this.focusResetListener], bindObj: this }); } - - focusResetListener(e): void { - const ctrlIsPressed = Controller.getCtrlIsPressed(); - if (!ctrlIsPressed) { - Controller.resetFocus(); - } + + initPosition(): void { + let prevEndOffset = 0; + const trackSectionElements: NodeListOf = this.querySelectorAll('audi-track-section'); + + if (!this.trackAreaElement || trackSectionElements.length === 0) return; + + this.trackWidth = this.calculateTrackWidth(); + this.maxTrackPlayTime = Controller.getMaxTrackPlayTime(); + const secondPerPixel = WidthUtil.getPixelPerSecond(this.trackWidth, this.maxTrackPlayTime); + + this.trackSectionList.forEach((section, idx) => { + const marginValue = (section.trackStartTime - prevEndOffset) * secondPerPixel; + if (!trackSectionElements[idx]) return; + trackSectionElements[idx].style.marginLeft = `${marginValue}px`; + prevEndOffset = section.trackStartTime + section.length; + }) } - trackDragoverListener(e): void { + dragenterAudioTrackListener(e): void { e.preventDefault(); - e.dataTransfer.dropEffect = "link"; + e.stopPropagation(); + if (!this.afterimageElement) return; + this.sectionDragData = Controller.getSectionDragStartData(); + this.afterimageElement.style.display = 'block'; } - trackDropListener(e): void { + dragleaveAudioTrackListener(e): void { e.preventDefault(); e.stopPropagation(); - const sourceId = e.dataTransfer.getData("text/plain"); - const source = Controller.getSourceBySourceId(Number(sourceId)); - if(!source) return; + if (!this.afterimageElement) return; - const { duration } = source; + this.afterimageElement.style.display = 'none'; + this.afterimageElement.style.left = `0px`; + this.afterimageElement.style.width = `0px`; + } + + dragoverAudioTrackListener(e): void { + e.preventDefault(); + if (!this.afterimageElement) return; + + const scrollAmount = Controller.getCurrentScrollAmount(); + const currentCursorPosition = e.pageX + scrollAmount; - const trackSection = new TrackSection({ - sourceId : source.id, - trackId: this.trackId, - channelStartTime : 0, - channelEndTime : duration, - parsedChannelStartTime : 0, - parsedChannelEndTime: duration, - trackStartTime : 0, - audioStartTime : 0 - }); - - Controller.addTrackSection(this.trackId, trackSection); - this.hideMessage(); + DragUtil.showAfterimage(this.afterimageElement, this.trackId, this.trackWidth, currentCursorPosition); } - - trackDragenterListener(e): void { - e.preventDefault() - this.trackDropzoneElement?.classList.add('focus'); + + dropAudioTrackListener(e): void { + e.preventDefault(); + e.stopPropagation(); + + if (!this.afterimageElement) return; + this.afterimageElement.style.display = 'none'; + + const scrollAmount = Controller.getCurrentScrollAmount(); + const currentCursorPosition = e.pageX + scrollAmount; + const currentTrackId: number = Number(e.target.dataset.trackId); + + DragUtil.dropTrackSection(currentTrackId, currentCursorPosition, this.trackWidth); } - trackDragleaveListener(e): void { - e.preventDefault() - this.trackDropzoneElement?.classList.remove('focus'); + trackClickListener(e): void { + const ctrlIsPressed = Controller.getCtrlIsPressed(); + if (!ctrlIsPressed) { + Controller.resetFocus(); + + if (!this.trackAreaElement) return; + const cursorPosition = e.pageX; + const trackAreaElementLeftX = this.trackAreaElement.getBoundingClientRect().left; + this.maxTrackPlayTime = Controller.getMaxTrackPlayTime(); + const secondPerPixel = WidthUtil.getPixelPerSecond(this.trackWidth, this.maxTrackPlayTime); + + const cursorOffset = cursorPosition - trackAreaElementLeftX; + const selectedTime = cursorOffset / secondPerPixel; + const selectLine = document.getElementById(`track-select-line-${this.trackId}`); + if (!selectLine) return; + selectLine.style.display = 'block'; + selectLine.style.left = `${cursorOffset}px`; + + Controller.changeSelectTrackData(this.trackId, selectedTime); + } } - hideMessage(): void { - this.trackMessage?.classList.add('hide'); + focusResetListener(e): void { + const ctrlIsPressed = Controller.getCtrlIsPressed(); + if (!ctrlIsPressed) { + Controller.resetFocus(); + Controller.changeSelectTrackData(0, 0); + } } subscribe(): void { - storeChannel.subscribe(StoreChannelType.TRACK_DRAG_STATE_CHANNEL, this.trackDragStateObserverCallback, this); storeChannel.subscribe(StoreChannelType.TRACK_SECTION_LIST_CHANNEL, this.trackSectionListObserverCallback, this); + storeChannel.subscribe(StoreChannelType.MAX_TRACK_WIDTH_CHANNEL, this.maxTrackWidthObserverCallback, this); + storeChannel.subscribe(StoreChannelType.MAX_TRACK_PLAY_TIME_CHANNEL, this.maxTrackPlayTimeObserverCallback, this); + storeChannel.subscribe(StoreChannelType.SELECT_AUDIO_TRACK, this.selectTrackDataObserverCallback, this); } - trackDragStateObserverCallback(isTrackDraggable): void { - if(isTrackDraggable){ - this.activeTrackDropzone(); - return; - } - this.inactiveTrackDropzone(); + trackSectionListObserverCallback({ trackId, trackSectionList }): void { + if (trackId !== this.trackId || !this.trackScrollAreaElement) return; + + this.trackSectionList = trackSectionList; + this.render(); + this.initElement(); + this.messageDisplayHandler(); + + const scrollWidth = this.trackScrollAreaElement.scrollWidth; + Controller.changeMaxTrackWidth(scrollWidth); + Controller.changeMaxTrackPlayTime(trackSectionList); + + if (this.maxTrackWidth === scrollWidth) + this.resizeTrackArea(scrollWidth); + this.initPosition(); } - activeTrackDropzone(): void { - this.trackDropzoneElement?.classList.remove('hide'); + messageDisplayHandler(): void { + if (!this.trackMessageElement) return; + + if (this.trackSectionList.length > 0) this.trackMessageElement.classList.add('hide'); + else this.trackMessageElement.classList.remove('hide'); } - inactiveTrackDropzone(): void { - this.trackDropzoneElement?.classList.add('hide'); + maxTrackWidthObserverCallback(maxTrackWidth: number): void { + this.maxTrackWidth = maxTrackWidth; + this.resizeTrackArea(maxTrackWidth); } - trackSectionListObserverCallback({trackId, trackSectionList}): void { - if(trackId !== this.trackId) return; + selectTrackDataObserverCallback(selectTrackData: SelectTrackData): void { + if (selectTrackData.trackId !== this.trackId) { + const selectLine = document.getElementById(`track-select-line-${this.trackId}`); + if (!selectLine) return; + selectLine.style.display = 'none'; + } + } - this.trackSectionList = trackSectionList; - this.render(); - this.initElement(); + resizeTrackArea(maxTrackWidth: number) { + if (!this.trackAreaElement || !this.trackScrollAreaElement) return; + + const scrollAreaWidth = this.trackScrollAreaElement.getBoundingClientRect().right - this.trackScrollAreaElement.getBoundingClientRect().left; + const ratio = maxTrackWidth / scrollAreaWidth; + + this.trackAreaElement.style.width = `${100 * ratio}%`; + } + + maxTrackPlayTimeObserverCallback(maxTrackPlayTime: number): void { + this.maxTrackPlayTime = maxTrackPlayTime; } }; customElements.define('audi-audio-track', AudioTrack); })(); - diff --git a/frontend/src/components/AudioTrack/AudioTrackMenu/AudioTrackMenu.scss b/frontend/src/components/AudioTrack/AudioTrackMenu/AudioTrackMenu.scss new file mode 100644 index 00000000..ddcf3efd --- /dev/null +++ b/frontend/src/components/AudioTrack/AudioTrackMenu/AudioTrackMenu.scss @@ -0,0 +1,31 @@ +audi-track-menu{ + width: 100%; + + .audio-track-menu-container{ + display: flex; + flex-direction: column; + justify-content: center; + align-items: center; + width: 100%; + padding: 10px 0 5px 0; + border: 1px solid #ffffff; + border-radius: 3px; + + .track-menu-area{ + display: flex; + justify-content: center; + } + + .zoom-menu-area{ + display: flex; + justify-content: center; + } + + audi-icon-button{ + margin-right: 5px; + &:last-of-type{ + margin-right: 0; + } + } + } +} diff --git a/frontend/src/components/AudioTrack/AudioTrackMenu/AudioTrackMenu.ts b/frontend/src/components/AudioTrack/AudioTrackMenu/AudioTrackMenu.ts new file mode 100644 index 00000000..8a856dec --- /dev/null +++ b/frontend/src/components/AudioTrack/AudioTrackMenu/AudioTrackMenu.ts @@ -0,0 +1,74 @@ +import { IconType, EventType, EventKeyType } from '@types'; +import { EventUtil } from '@util'; +import { CommandController } from '@controllers'; +import './AudioTrackMenu.scss'; + +(() => { + const AudioTrackMenu = class extends HTMLElement { + constructor() { + super(); + } + + static ICON_SIZE: number = 25; + + connectedCallback(): void { + try { + this.render(); + this.initEvent(); + } catch (e) { + console.log(e); + } + } + + render(): void { + const { headset, colorLens, zoomIn, zoomOut } = IconType; + const { ICON_SIZE } = AudioTrackMenu; + + this.innerHTML = ` +
+
+ + + + +
+
+ + + + +
+
+ `; + } + + initEvent(): void { + EventUtil.registerEventToRoot({ + eventTypes: [EventType.click], + eventKey: EventKeyType.TRACK_ADD_MENU_CLICK, + listeners: [this.trackAddMenuClickListener], + bindObj: this + }); + } + + trackAddMenuClickListener(e): void { + try { + CommandController.executeAddTrackCommand(); + } catch (e) { + console.log(e); + } + } + }; + + customElements.define('audi-track-menu', AudioTrackMenu); +})(); diff --git a/frontend/src/components/AudioTrack/AudioTrackOption/AudioTrackOption.scss b/frontend/src/components/AudioTrack/AudioTrackOption/AudioTrackOption.scss new file mode 100644 index 00000000..2363ce89 --- /dev/null +++ b/frontend/src/components/AudioTrack/AudioTrackOption/AudioTrackOption.scss @@ -0,0 +1,50 @@ +@use '@style/variables.scss'as variables; + +.audio-track-option-container{ + display: flex; + flex-direction: column; + justify-content: center; + align-items: center; + height: 175px; + padding: 10px; + margin: 20px 0; + border: 1px solid variables.$white-color; + border-radius: 3px; + color: variables.$white-color; + box-sizing: border-box; + cursor: pointer; + + &:first-of-type{ + margin-top: 10px; + } + + li{ + margin-bottom: 10px; + + &:hover{ + color: variables.$green-color; + } + + &.clicked{ + color: variables.$green-color; + } + } +} + +@media screen and (max-width: 1680px) { + .audio-track-option-container{ + height: 125px; + margin: 10px 0; + padding: 5px; + + li{ + span{ + font-size: 12px; + } + } + + &:first-of-type{ + margin-top: 5px; + } + } +} diff --git a/frontend/src/components/AudioTrack/AudioTrackOption/AudioTrackOption.ts b/frontend/src/components/AudioTrack/AudioTrackOption/AudioTrackOption.ts new file mode 100644 index 00000000..fb4257de --- /dev/null +++ b/frontend/src/components/AudioTrack/AudioTrackOption/AudioTrackOption.ts @@ -0,0 +1,117 @@ +import { EventKeyType, EventType, TrackOptionType } from '@types'; +import { EventUtil } from '@util'; +import { Controller, CommandController } from '@controllers'; +import './AudioTrackOption.scss'; + +(() => { + const AudioTrackOption = class extends HTMLElement { + private trackOptionId: number; + private trackId: number; + private isSolo: boolean; + private isMute: boolean; + private isDelete: boolean; + + constructor() { + super(); + this.trackOptionId = 0; + this.trackId = 0; + this.isSolo = false; + this.isMute = false; + this.isDelete = false; + } + + static get observedAttributes(): string[] { + return ['data-id', 'data-track-id']; + } + + attributeChangedCallback(attrName: string, oldVal: string, newVal: string): void { + if (oldVal !== newVal) { + switch (attrName) { + case 'data-id': + this.trackOptionId = Number(newVal); + break; + case 'data-track-id': + this.trackId = Number(newVal); + break; + } + this[attrName] = newVal; + } + } + + connectedCallback(): void { + try { + this.render(); + this.initEvent(); + } catch (e) { + console.log(e); + } + } + + render(): void { + this.innerHTML = ` +
    +
  • Solo
  • +
  • Mute
  • +
  • Delete
  • +
+ `; + } + + initEvent(): void{ + EventUtil.registerEventToRoot({ + eventTypes: [EventType.click], + eventKey: EventKeyType.AUDIO_TRACK_OPTION_CLICK + this.trackOptionId, + listeners: [this.trackOptionClickListener], + bindObj: this + }); + } + + trackOptionClickListener(e): void{ + const {type } = e.target.dataset; + + switch(type){ + case TrackOptionType.solo: + this.soloClickListener(); + break; + case TrackOptionType.mute: + this.muteClickListener(); + break; + case TrackOptionType.delete: + this.deleteClickListener(); + break; + } + } + + soloClickListener(): void { + if(!this.isSolo) { + this.isSolo = true; + Controller.setSolo(this.trackId); + } + else { + this.isSolo = false; + Controller.unsetSolo(this.trackId); + } + + this.render(); + } + + muteClickListener(): void { + if(!this.isMute) { + this.isMute = true; + Controller.setMute(this.trackId); + } + else { + this.isMute = false; + Controller.unsetMute(this.trackId); + } + + this.render(); + } + + deleteClickListener(): void { + CommandController.executeDeleteTrackCommand(this.trackId); + } + }; + + customElements.define('audi-track-option', AudioTrackOption); +})(); diff --git a/frontend/src/components/AudioTrack/AudioTrackSection/AudioTrackSection.scss b/frontend/src/components/AudioTrack/AudioTrackSection/AudioTrackSection.scss index 5b9fce35..42deeeff 100644 --- a/frontend/src/components/AudioTrack/AudioTrackSection/AudioTrackSection.scss +++ b/frontend/src/components/AudioTrack/AudioTrackSection/AudioTrackSection.scss @@ -1,5 +1,8 @@ +@import '@style/variables'; + audi-track-section{ width: 100%; + flex-shrink: 0; } .audio-track-section{ @@ -8,20 +11,30 @@ audi-track-section{ margin: 0; padding: 0; box-sizing: border-box; - cursor: pointer; - + + border: 1px solid rgba(3,199,90, 0.5); + border-radius: 5px; + &:hover { - border: 1px solid aqua; + border: 1px solid $green-color; border-radius: 5px; } + + cursor: pointer; } .focused-section { - border: 4px solid aqua; + border: 3px solid $green-color; border-radius: 5px; &:hover { - border: 4px solid aqua; + border: 4px solid $green-color; border-radius: 5px; } } + +@media screen and (max-width: 1680px) { + .audio-track-section{ + height: 100px; + } +} diff --git a/frontend/src/components/AudioTrack/AudioTrackSection/AudioTrackSection.ts b/frontend/src/components/AudioTrack/AudioTrackSection/AudioTrackSection.ts index fb4025e4..0fb819b1 100644 --- a/frontend/src/components/AudioTrack/AudioTrackSection/AudioTrackSection.ts +++ b/frontend/src/components/AudioTrack/AudioTrackSection/AudioTrackSection.ts @@ -1,27 +1,46 @@ -import { Controller } from '@controllers'; -import { EventKeyType, EventType } from '@types'; -import { EventUtil } from '@util'; - +import { Controller, CommandController } from '@controllers'; +import { CursorType, EventKeyType, EventType, StoreChannelType, SectionDataType } from '@types'; +import { EventUtil, TimeUtil, ValidUtil, WidthUtil, DragUtil } from '@util'; +import { storeChannel } from '@store'; import './AudioTrackSection.scss'; -interface SectionData { - sectionChannelData: number[]; - duration: number; -} - (() => { const AudioTrackSection = class extends HTMLElement { private trackId: number; private sectionId: number; - private sectionData: SectionData | undefined; + private sectionData: SectionDataType | undefined; + private cursorMode: CursorType | undefined; + private canvasWidth: number; + private canvasHeight: number; + private maxTrackPlayTime: number; + private maxTrackWidth: number; + private currentScrollAmount: number; private trackCanvasElement: HTMLCanvasElement | undefined | null; + private trackContainerElement: HTMLElement | null; + private trackAreaElement: HTMLElement | null; + private trackScrollAreaElement: HTMLElement | null; + private cutLineElement: HTMLElement | undefined | null; + private trackContainerWidth: number; + private trackAfterimageElement: HTMLElement | null; constructor() { super(); this.trackId = 0; this.sectionId = 0; this.sectionData; + this.cursorMode; + this.canvasWidth = 0; + this.canvasHeight = 0; + this.maxTrackWidth = 0; + this.maxTrackPlayTime = 0; + this.currentScrollAmount = 0; this.trackCanvasElement; + this.trackContainerElement = null; + this.trackAreaElement = null; + this.trackScrollAreaElement = null; + this.cutLineElement; + this.trackContainerWidth = 0; + this.trackAfterimageElement = null; } static get observedAttributes(): string[] { @@ -45,9 +64,11 @@ interface SectionData { connectedCallback(): void { try { this.render(); - this.init(); - this.draw(); + this.initProperty(); + this.drawTrackSection(); this.initEvent(); + this.initState(); + this.subscribe(); } catch (e) { console.log(e); } @@ -55,41 +76,79 @@ interface SectionData { render(): void { this.innerHTML = ` - + `; } - init(): void { + initProperty(): void { + this.cursorMode = Controller.getCursorMode(); + this.sectionData = Controller.getSectionData(this.trackId, this.sectionId); + this.maxTrackWidth = Controller.getMaxTrackWidth(); + this.maxTrackPlayTime = Controller.getMaxTrackPlayTime(); + this.currentScrollAmount = Controller.getCurrentScrollAmount(); this.trackCanvasElement = this.querySelector('.audio-track-section'); - this.sectionData = Controller.getSectionChannelData(this.trackId, this.sectionId); + this.cutLineElement = document.getElementById(`section-cut-line-${this.trackId}`); + this.trackContainerElement = document.querySelector('.audi-main-audio-track-container'); + this.trackAreaElement = document.querySelector('.audio-track-area'); + this.trackScrollAreaElement = document.querySelector('.audi-main-audio-track-scroll-area'); + this.trackAfterimageElement = document.querySelector(`#afterimage-${this.trackId}`); + + if (this.trackContainerElement) { + this.trackContainerWidth = this.trackContainerElement.getBoundingClientRect().right - this.trackContainerElement.getBoundingClientRect().left; + } } - draw(): void { - if (!this.sectionData || !this.trackCanvasElement) return; + drawTrackSection(): void { + if (!this.sectionData || !this.trackCanvasElement || !this.trackContainerElement) return; const { sectionChannelData, duration } = this.sectionData; + this.calculateCanvasSize(duration); + this.resizeCanvas(); + this.drawCanvas(sectionChannelData); + this.resizeTrackArea(); + } + + calculateCanvasSize(duration: number): void { + if (!this.trackContainerElement || !this.trackAreaElement) return; + + const trackWidth = this.trackAreaElement.getBoundingClientRect().right - this.trackAreaElement.getBoundingClientRect().left; + const trackHeight = this.trackAreaElement.clientHeight; - const trackWidth = this.trackCanvasElement.clientWidth; - const canvasWidth = trackWidth / (300 / duration); - this.trackCanvasElement.width = canvasWidth; - this.trackCanvasElement.style.width = `${canvasWidth}px`; + if (this.maxTrackWidth > trackWidth) + this.canvasWidth = this.maxTrackWidth / (this.maxTrackPlayTime / duration); + else + this.canvasWidth = trackWidth / (this.maxTrackPlayTime / duration); + + this.canvasHeight = trackHeight;; + } + + resizeCanvas(): void { + if (!this.trackCanvasElement) return; + + this.style.width = `${this.canvasWidth}px`; + this.trackCanvasElement.width = this.canvasWidth; + this.trackCanvasElement.style.width = `${this.canvasWidth}px`; + this.trackCanvasElement.height = this.canvasHeight; + } + + drawCanvas(sectionChannelData: number[]): void { + if (!this.trackCanvasElement) return; - const canvasHeight = this.trackCanvasElement.clientHeight; const canvasCtx = this.trackCanvasElement.getContext('2d'); if (!canvasCtx) return; const numOfPeaks = sectionChannelData.length; - const middleHeight = canvasHeight / 2; + const middleHeight = this.canvasHeight / 2; const defaultLineWidth = 1; canvasCtx.strokeStyle = '#2196f3'; - canvasCtx.lineWidth = defaultLineWidth / (numOfPeaks / 2 / canvasWidth); + canvasCtx.lineWidth = defaultLineWidth / (numOfPeaks / 2 / this.canvasWidth); canvasCtx.beginPath(); let offsetX = 0; let offsetY; for (let i = 0; i < numOfPeaks; i++) { - offsetY = middleHeight + Math.floor((sectionChannelData[i] * canvasHeight) / 2); + offsetY = middleHeight + Math.floor((sectionChannelData[i] * this.canvasHeight) / 2); if (i % 2 == 0) canvasCtx.moveTo(offsetX, offsetY); else { canvasCtx.lineTo(offsetX, offsetY); @@ -99,17 +158,140 @@ interface SectionData { canvasCtx.stroke(); } + resizeTrackArea() { + if (!this.trackScrollAreaElement || !this.trackAreaElement) return; + + const scrollAreaWidth = this.trackScrollAreaElement.getBoundingClientRect().right - this.trackScrollAreaElement.getBoundingClientRect().left; + const ratio = this.maxTrackWidth / scrollAreaWidth; + + this.trackAreaElement.style.width = `${100 * ratio}%`; + } + initEvent(): void { EventUtil.registerEventToRoot({ - eventTypes: [EventType.click], - eventKey: EventKeyType.AUDIO_TRACK_SECTION_CLICK + this.sectionId, - listeners: [this.clickListener], + eventTypes: [EventType.click, EventType.mousemove, EventType.mouseout, EventType.dragstart, EventType.dragover, EventType.drop], + eventKey: EventKeyType.AUDIO_TRACK_SECTION_MULTIPLE + this.sectionId, + listeners: [ + this.trackSectionClickListener, + this.trackSectionMouseMoveListener, + this.trackSectionMouseoutListener, + this.trackSectiondragStartListener, + this.trackSectiondragoverListener, + this.trackSectiondropListener + ], bindObj: this }); + window.addEventListener('resize', this.windowResizeListener.bind(this)); + } + + trackSectiondragoverListener(e): void { + e.preventDefault(); + if (!this.trackAfterimageElement) return; + + const currentCursorPosition = e.pageX + this.currentScrollAmount; + + DragUtil.showAfterimage(this.trackAfterimageElement, this.trackId, this.trackContainerWidth, currentCursorPosition); + } + + trackSectiondropListener(e): void { + e.preventDefault(); + e.stopPropagation(); + + const currentCursorPosition = e.pageX + this.currentScrollAmount; + + DragUtil.dropTrackSection(this.trackId, currentCursorPosition, this.trackContainerWidth); + } + + trackSectiondragStartListener(e): void { + if (!this.trackContainerElement) return; + const offsetLeft = this.trackContainerElement.getBoundingClientRect().left; + + this.maxTrackPlayTime = Controller.getMaxTrackPlayTime(); + const prevCursorPosition = e.pageX; + const prevCursorTime = TimeUtil.calculateTimeOfCursorPosition(offsetLeft, prevCursorPosition, this.trackContainerWidth, this.maxTrackPlayTime); + const trackSection = Controller.getTrackSection(this.trackId, this.sectionId); + + if (!trackSection) return; + const currentScrollTime = Controller.getCurrentScrollTime(); + const sectionDragStartData = { trackSection, prevCursorTime: prevCursorTime + currentScrollTime, offsetLeft }; + + Controller.changeSectionDragStartData(sectionDragStartData); + e.dataTransfer.effectAllowed = 'move'; + + const dragImage = document.createElement('div'); + dragImage.style.visibility = 'hidden'; + e.dataTransfer.setDragImage(dragImage, 0, 0); + } + + trackSectionClickListener(e): void { + switch (this.cursorMode) { + case CursorType.SELECT_MODE: + this.selectModeClickHandler(e); + break; + case CursorType.CUT_MODE: + this.cutModeClickHandler(e); + break; + } + } + + selectModeClickHandler(e): void { + const trackSectionElement = e.target; + Controller.toggleFocus(this.trackId, this.sectionId, trackSectionElement); + } + + cutModeClickHandler(e): void { + const cursorPosition = e.pageX; + CommandController.executeSplitCommand(cursorPosition, this.trackId, this.sectionId); + } + + trackSectionMouseMoveListener(e): void { + if (!this.trackContainerElement || this.cursorMode !== CursorType.CUT_MODE) return; + const cursorPosition = e.pageX; + const trackContainerLeftX = this.trackContainerElement.getBoundingClientRect().left; + const cursorOffset = cursorPosition - trackContainerLeftX; + + this.showCutLine(cursorOffset + this.currentScrollAmount); + } + + trackSectionMouseoutListener(e): void { + this.hideCutLine(); + } + + hideCutLine(): void { + if (!this.cutLineElement) return; + this.cutLineElement?.classList.add('hide'); + } + + showCutLine(location: number): void { + if (!this.cutLineElement) return; + this.cutLineElement?.classList.remove('hide'); + this.cutLineElement.style.left = `${location}px`; + } + + windowResizeListener(e) { + this.drawTrackSection(); } - clickListener(e): void { - Controller.toggleFocus(this.trackId, this.sectionId, e.target); + subscribe(): void { + storeChannel.subscribe(StoreChannelType.CURSOR_MODE_CHANNEL, this.cursorModeObserverCallback, this); + storeChannel.subscribe(StoreChannelType.CURRENT_SCROLL_AMOUNT_CHANNEL, this.currentScrollAmountObserverCallback, this); + } + + cursorModeObserverCallback(newCursorMode) { + this.cursorMode = newCursorMode; + } + + currentScrollAmountObserverCallback(newCurrentScrollAmount: number): void { + this.currentScrollAmount = newCurrentScrollAmount; + } + + initState(): void { + const focusList = Controller.getFocusList(); + const focusInfo = focusList.find(focus => focus.trackSection.id === this.sectionId); + if (!focusInfo || !this.trackCanvasElement) return; + + focusInfo.element = this.trackCanvasElement; + focusInfo.element.classList.add('focused-section'); } }; diff --git a/frontend/src/components/AudioTrack/index.js b/frontend/src/components/AudioTrack/index.js index e70fd072..b6f40c01 100644 --- a/frontend/src/components/AudioTrack/index.js +++ b/frontend/src/components/AudioTrack/index.js @@ -1,2 +1,4 @@ import "./AudioTrackSection/AudioTrackSection"; +import "./AudioTrackOption/AudioTrackOption"; +import "./AudioTrackMenu/AudioTrackMenu"; import "./AudioTrack"; diff --git a/frontend/src/components/EditTools/EditTools.scss b/frontend/src/components/EditTools/EditTools.scss index de7bdffa..82b52d82 100644 --- a/frontend/src/components/EditTools/EditTools.scss +++ b/frontend/src/components/EditTools/EditTools.scss @@ -11,6 +11,4 @@ .edit-tools > audi-icon-button { margin: auto 5px; - - cursor: pointer; } diff --git a/frontend/src/components/EditTools/EditTools.ts b/frontend/src/components/EditTools/EditTools.ts index 2e7d72ad..1c257f21 100644 --- a/frontend/src/components/EditTools/EditTools.ts +++ b/frontend/src/components/EditTools/EditTools.ts @@ -1,7 +1,7 @@ -import CommandManager from '@command/CommandManager'; +import { CommandManager } from '@command'; import { CursorType, StoreChannelType, EventKeyType, EventType, IconType } from "@types"; import { storeChannel } from '@store'; -import { Controller } from '@controllers' +import { Controller, CommandController } from '@controllers' import { EventUtil } from '@util'; import './EditTools.scss' @@ -32,7 +32,7 @@ import './EditTools.scss' connectedCallback() { this.render(); - this.init() + this.initElement() this.initState(); this.initEvent(); this.subscribe(); @@ -41,12 +41,12 @@ import './EditTools.scss' render() { this.innerHTML = `
- ${this.iconlist.reduce((acc, icon) => acc + ``, '')} + ${this.iconlist.reduce((acc, icon) => acc + ``, '')}
`; } - init() { + initElement() { this.cursorElement = this.querySelector('#cursor'); this.bladeElement = this.querySelector('#blade'); this.copyElement = this.querySelector('#copy'); @@ -58,10 +58,50 @@ import './EditTools.scss' } initState() { - this.cursorState(); - this.focusState(); - this.clipBoardState(); - this.commandState(); + this.initCursorState(); + this.initFocusState(); + this.initClipBoardState(); + this.initCommandState(); + } + + initCursorState() { + const cursorMode = Controller.getCursorMode(); + if (cursorMode === CursorType.SELECT_MODE) { + this.cursorElement?.classList.add('selected'); + } else { + this.bladeElement?.classList.add('selected'); + } + } + + initFocusState() { + const focusList = Controller.getFocusList(); + if (focusList.length === 0) { + this.copyElement?.classList.add('disabled'); + this.cutElement?.classList.add('disabled'); + this.deleteElement?.classList.add('disabled'); + } else if (focusList.length > 1) { + this.copyElement?.classList.add('disabled'); + this.cutElement?.classList.add('disabled'); + } + } + + initClipBoardState() { + const clipBoard = Controller.getClipBoard(); + const focusList = Controller.getFocusList(); + + if (focusList.length > 1 || !clipBoard) { + this.pasteElement?.classList.add('disabled'); + } + } + + initCommandState() { + const { undoList, redoList } = CommandManager; + if (undoList.length === 0) { + this.undoElement?.classList.add('disabled'); + } + if (redoList.length === 0) { + this.redoElement?.classList.add('disabled'); + } } initEvent() { @@ -85,11 +125,11 @@ import './EditTools.scss' listeners: [this.undoListener], bindObj: this }); - + EventUtil.registerEventToRoot({ eventTypes: [EventType.click], eventKey: EventKeyType.EDIT_TOOLS_CLICK + IconType.blade, - listeners: [this.cutCursorListener], + listeners: [this.cuttingCursorListener], bindObj: this }); @@ -99,60 +139,34 @@ import './EditTools.scss' listeners: [this.redoListener], bindObj: this }); - + EventUtil.registerEventToRoot({ eventTypes: [EventType.click], eventKey: EventKeyType.EDIT_TOOLS_CLICK + IconType.copy, listeners: [this.copyListener], bindObj: this }); - } - - cursorState() { - const cursorMode = Controller.getCursorMode(); - if (cursorMode === CursorType.SELECT_MODE) { - this.cursorElement?.classList.add('selected'); - } else { - this.bladeElement?.classList.add('selected'); - } - } - focusState() { - const focusList = Controller.getFocusList(); - if (focusList.length === 0) { - this.copyElement?.classList.add('disabled'); - this.cutElement?.classList.add('disabled'); - this.deleteElement?.classList.add('disabled'); - } else if (focusList.length > 1) { - this.copyElement?.classList.add('disabled'); - this.cutElement?.classList.add('disabled'); - } - } - - clipBoardState() { - const clipBoard = Controller.getClipBoard(); - const focusList = Controller.getFocusList(); - - if (focusList.length > 1 || !clipBoard) { - this.pasteElement?.classList.add('disabled'); - } - } + EventUtil.registerEventToRoot({ + eventTypes: [EventType.click], + eventKey: EventKeyType.EDIT_TOOLS_CLICK + IconType.cut, + listeners: [this.cutListener], + bindObj: this + }); - commandState() { - const { undoList, redoList } = CommandManager; - if (undoList.length === 0) { - this.undoElement?.classList.add('disabled'); - } - if (redoList.length === 0) { - this.redoElement?.classList.add('disabled'); - } + EventUtil.registerEventToRoot({ + eventTypes: [EventType.click], + eventKey: EventKeyType.EDIT_TOOLS_CLICK + IconType.paste, + listeners: [this.pasteListener], + bindObj: this + }); } selectCursorListener(e) { Controller.setCursorMode(CursorType.SELECT_MODE); } - cutCursorListener(e) { + cuttingCursorListener(e) { Controller.setCursorMode(CursorType.CUT_MODE); } @@ -160,30 +174,38 @@ import './EditTools.scss' Controller.setClipBoard(); } - subscribe(): void { - storeChannel.subscribe(StoreChannelType.EDIT_TOOLS_CHANNEL, this.updateEditTools, this); + cutListener(e) { + CommandController.executeCutCommand(); } - updateEditTools(): void { - this.render(); - this.init(); - this.initState(); - this.initEvent(); + pasteListener(e) { + CommandController.executePasteCommand(); } deleteListener(): void { - Controller.deleteCommand(); + CommandController.executeDeleteCommand(); } undoListener(): void { - Controller.undoCommand(); + CommandController.executeUndoCommand(); } redoListener(): void { - Controller.redoCommand(); + CommandController.executeRedoCommand(); + } + + subscribe(): void { + storeChannel.subscribe(StoreChannelType.EDIT_TOOLS_CHANNEL, this.updateEditTools, this); + storeChannel.subscribe(StoreChannelType.CURSOR_MODE_CHANNEL, this.updateEditTools, this); + } + + updateEditTools(): void { + this.render(); + this.initElement(); + this.initState(); } }; - + customElements.define('audi-edit-tools', EditTools); })() diff --git a/frontend/src/components/EditorMenu/EditorMenu.scss b/frontend/src/components/EditorMenu/EditorMenu.scss index 1c7979df..91a26879 100644 --- a/frontend/src/components/EditorMenu/EditorMenu.scss +++ b/frontend/src/components/EditorMenu/EditorMenu.scss @@ -11,12 +11,14 @@ border: 1px solid white; border-radius: 6px; + + &:last-of-type{ + margin-right: 20px; + } } .icon-wrap > audi-icon-button { margin: auto 5px; - - cursor: pointer; } @@ -30,4 +32,14 @@ } } +@media screen and (max-width: 1680px) { + #editor-menu { + audi-icon-button{ + svg{ + width: 25px !important; + } + } + } +} + diff --git a/frontend/src/components/EditorMenu/EditorMenu.ts b/frontend/src/components/EditorMenu/EditorMenu.ts index 36226c07..7170ec87 100644 --- a/frontend/src/components/EditorMenu/EditorMenu.ts +++ b/frontend/src/components/EditorMenu/EditorMenu.ts @@ -1,25 +1,29 @@ import './EditorMenu.scss'; -import { Controller } from '@controllers'; -import { ModalType, EventType, EventKeyType } from '@types'; +import { ModalType, EventType, EventKeyType, StoreChannelType } from '@types'; +import { storeChannel } from '@store'; +import { Controller } from '@controllers' import { EventUtil } from '@util'; - + (() => { const EditorMenu = class extends HTMLElement { + constructor() { super(); } connectedCallback(): void { this.render(); + this.init(); this.initEvent(); + this.subscribe(); } render(): void { this.innerHTML = `
- - + +
@@ -34,6 +38,15 @@ import { EventUtil } from '@util'; `; } + init(): void { + const saveButton = this.querySelector('#save'); + if (!saveButton) return; + + const trackList = Controller.getTrackList(); + const possibleSave = trackList.find(track => track.trackSectionList.length > 0); + if (!possibleSave) saveButton.classList.add('disabled') + } + initEvent(): void { EventUtil.registerEventToRoot({ eventTypes: [EventType.click], @@ -50,16 +63,26 @@ import { EventUtil } from '@util'; }); } - openUploadModalBtnClickListener(): void{ + openUploadModalBtnClickListener(): void { Controller.changeModalState(ModalType.upload, false); } - openDownloadModalBtnClickListener(): void{ + openDownloadModalBtnClickListener(): void { Controller.changeModalState(ModalType.download, false); } + + subscribe(): void { + storeChannel.subscribe(StoreChannelType.EDIT_MENU_CHANNEL, this.updateEditorMenu, this); + } + + updateEditorMenu(): void { + this.render(); + this.init(); + this.initEvent(); + } }; customElements.define('audi-editor-menu', EditorMenu); })(); -export {}; +export { }; diff --git a/frontend/src/components/IconButton/IconButton.scss b/frontend/src/components/IconButton/IconButton.scss index b56bbcce..0bed858e 100644 --- a/frontend/src/components/IconButton/IconButton.scss +++ b/frontend/src/components/IconButton/IconButton.scss @@ -1,5 +1,9 @@ @use '@style/variables.scss' as variables; +audi-icon-button { + cursor: pointer; +} + .icon:hover { path { fill: variables.$green-color; @@ -18,4 +22,31 @@ path { fill: variables.$green-color } +} + +.icon-buttion-wrap { + position: relative; + &:hover { + .icon-info{ + transition-property: visibility; + transition-delay: 0.3s; + visibility: visible; + } + } +} +.icon-info { + visibility: hidden; + position:absolute; + left: -20%; + width: fit-content; + padding: 3px 5px; + z-index: 1000; + + color:white; + border: 0.5px solid gray; + border-radius: 3px; + background-color:rgb(80,80,80); + + white-space:nowrap; + font-size: 15px; } \ No newline at end of file diff --git a/frontend/src/components/IconButton/IconButton.ts b/frontend/src/components/IconButton/IconButton.ts index a0fa1f2d..62b86826 100644 --- a/frontend/src/components/IconButton/IconButton.ts +++ b/frontend/src/components/IconButton/IconButton.ts @@ -1,5 +1,5 @@ import { IconType } from '@types'; -import icons from './icons'; +import { icons, iconInfo } from './icons'; import './IconButton.scss'; (() => { @@ -7,7 +7,6 @@ import './IconButton.scss'; private icontype: IconType | null; private color: string; private size: string; - private eventKey: string; private isDoneInit: Boolean; constructor() { @@ -15,7 +14,6 @@ import './IconButton.scss'; this.icontype = null; this.color = 'white'; this.size = '20px'; - this.eventKey = ''; this.isDoneInit = false; } @@ -24,9 +22,9 @@ import './IconButton.scss'; } attributeChangedCallback(attrName: string, oldVal: string, newVal: string): void { - if(oldVal === newVal) return; + if (oldVal === newVal) return; - switch(attrName){ + switch (attrName) { case 'icontype': this.icontype = IconType[newVal]; break; @@ -36,12 +34,9 @@ import './IconButton.scss'; case 'size': this.size = newVal; break; - case 'data-event-key': - this.eventKey = newVal; - break; } this[attrName] = newVal; - if(this.isDoneInit) + if (this.isDoneInit) this.render(); } @@ -52,7 +47,7 @@ import './IconButton.scss'; render(): void { this.innerHTML = ` -
+
${this.getIcon()} +
${this.icontype ? iconInfo[this.icontype] : ''}
`; } - getIcon(){ - if(!this.color || !this.icontype) return; + getIcon() { + if (!this.color || !this.icontype) return; - switch(this.icontype){ + switch (this.icontype) { case IconType.record_on: - return ``; + return ``; case IconType.zoomIn: - return ` + return ` `; case IconType.blade: return ` @@ -83,7 +79,7 @@ import './IconButton.scss'; `; default: - return ``; + return ``; } } }; diff --git a/frontend/src/components/IconButton/icons.ts b/frontend/src/components/IconButton/icons.ts index a4e64487..29fb44ec 100644 --- a/frontend/src/components/IconButton/icons.ts +++ b/frontend/src/components/IconButton/icons.ts @@ -8,11 +8,11 @@ const icons = { fastRewind: "M11 18V6l-8.5 6 8.5 6zm.5-6l8.5 6V6l-8.5 6z", repeat: "M7 7h10v3l4-4-4-4v3H5v6h2V7zm10 10H7v-3l-4 4 4 4v-3h12v-6h-2v4z", record: "M12 14c1.66 0 2.99-1.34 2.99-3L15 5c0-1.66-1.34-3-3-3S9 3.34 9 5v6c0 1.66 1.34 3 3 3zm5.3-3c0 3-2.54 5.1-5.3 5.1S6.7 14 6.7 11H5c0 3.41 2.72 6.23 6 6.72V21h2v-3.28c3.28-.48 6-3.3 6-6.72h-1.7z", - user: "M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 3c1.66 0 3 1.34 3 3s-1.34 3-3 3-3-1.34-3-3 1.34-3 3-3zm0 14.2c-2.5 0-4.71-1.28-6-3.22.03-1.99 4-3.08 6-3.08 1.99 0 5.97 1.09 6 3.08-1.29 1.94-3.5 3.22-6 3.22z", - menu: "M3 18h18v-2H3v2zm0-5h18v-2H3v2zm0-7v2h18V6H3z", - listAdd: "M14 10H2v2h12v-2zm0-4H2v2h12V6zm4 8v-4h-2v4h-4v2h4v4h2v-4h4v-2h-4zM2 16h8v-2H2v2z", - zoomOut: "M15.5 14h-.79l-.28-.27C15.41 12.59 16 11.11 16 9.5 16 5.91 13.09 3 9.5 3S3 5.91 3 9.5 5.91 16 9.5 16c1.61 0 3.09-.59 4.23-1.57l.27.28v.79l5 4.99L20.49 19l-4.99-5zm-6 0C7.01 14 5 11.99 5 9.5S7.01 5 9.5 5 14 7.01 14 9.5 11.99 14 9.5 14zM7 9h5v1H7z", - headset: "M12 1c-4.97 0-9 4.03-9 9v7c0 1.66 1.34 3 3 3h3v-8H5v-2c0-3.87 3.13-7 7-7s7 3.13 7 7v2h-4v8h3c1.66 0 3-1.34 3-3v-7c0-4.97-4.03-9-9-9z", + user: "M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm0 3c1.66 0 3 1.34 3 3s-1.34 3-3 3-3-1.34-3-3 1.34-3 3-3zm0 14.2c-2.5 0-4.71-1.28-6-3.22.03-1.99 4-3.08 6-3.08 1.99 0 5.97 1.09 6 3.08-1.29 1.94-3.5 3.22-6 3.22z", + menu: "M3 18h18v-2H3v2zm0-5h18v-2H3v2zm0-7v2h18V6H3z", + listAdd: "M14 10H2v2h12v-2zm0-4H2v2h12V6zm4 8v-4h-2v4h-4v2h4v4h2v-4h4v-2h-4zM2 16h8v-2H2v2z", + zoomOut: "M15.5 14h-.79l-.28-.27C15.41 12.59 16 11.11 16 9.5 16 5.91 13.09 3 9.5 3S3 5.91 3 9.5 5.91 16 9.5 16c1.61 0 3.09-.59 4.23-1.57l.27.28v.79l5 4.99L20.49 19l-4.99-5zm-6 0C7.01 14 5 11.99 5 9.5S7.01 5 9.5 5 14 7.01 14 9.5 11.99 14 9.5 14zM7 9h5v1H7z", + headset: "M12 1c-4.97 0-9 4.03-9 9v7c0 1.66 1.34 3 3 3h3v-8H5v-2c0-3.87 3.13-7 7-7s7 3.13 7 7v2h-4v8h3c1.66 0 3-1.34 3-3v-7c0-4.97-4.03-9-9-9z", colorLens: "M12 3c-4.97 0-9 4.03-9 9s4.03 9 9 9c.83 0 1.5-.67 1.5-1.5 0-.39-.15-.74-.39-1.01-.23-.26-.38-.61-.38-.99 0-.83.67-1.5 1.5-1.5H16c2.76 0 5-2.24 5-5 0-4.42-4.03-8-9-8zm-5.5 9c-.83 0-1.5-.67-1.5-1.5S5.67 9 6.5 9 8 9.67 8 10.5 7.33 12 6.5 12zm3-4C8.67 8 8 7.33 8 6.5S8.67 5 9.5 5s1.5.67 1.5 1.5S10.33 8 9.5 8zm5 0c-.83 0-1.5-.67-1.5-1.5S13.67 5 14.5 5s1.5.67 1.5 1.5S15.33 8 14.5 8zm3 4c-.83 0-1.5-.67-1.5-1.5S16.67 9 17.5 9s1.5.67 1.5 1.5-.67 1.5-1.5 1.5z", save: "M17 3H5c-1.11 0-2 .9-2 2v14c0 1.1.89 2 2 2h14c1.1 0 2-.9 2-2V7l-4-4zm-5 16c-1.66 0-3-1.34-3-3s1.34-3 3-3 3 1.34 3 3-1.34 3-3 3zm3-10H5V5h10v4z", upload: "M20 6h-8l-2-2H4c-1.11 0-1.99.89-1.99 2L2 18c0 1.11.89 2 2 2h16c1.11 0 2-.89 2-2V8c0-1.11-.89-2-2-2zm-1 8h-3v3h-2v-3h-3v-2h3V9h2v3h3v2z", @@ -26,4 +26,36 @@ const icons = { link: "M3.9 12c0-1.71 1.39-3.1 3.1-3.1h4V7H7c-2.76 0-5 2.24-5 5s2.24 5 5 5h4v-1.9H7c-1.71 0-3.1-1.39-3.1-3.1zM8 13h8v-2H8v2zm9-6h-4v1.9h4c1.71 0 3.1 1.39 3.1 3.1s-1.39 3.1-3.1 3.1h-4V17h4c2.76 0 5-2.24 5-5s-2.24-5-5-5z", } -export default icons; +const iconInfo = { + play: '재생(Space)', + pause: '일시정지(Space)', + stop: '중지', + skipNext: '종료 지점( ] )', + skipPrev: '시작 지점( [ )', + fastForward: '건너뛰기(→)', + fastRewind: '건너뛰기(←)', + repeat: '반복 재생', + record: '녹음하기', + record_on: '녹음 중', + user: '회원메뉴', + menu: '회원메뉴', + listAdd: '이펙트 추가', + zoomIn: '확대하기', + zoomOut: '축소하기', + headset: '트랙 추가', + colorLens: '트랙 색상 변경', + save: '저장하기', + upload: '불러오기', + cursor: '선택 도구(V)', + copy: '복사하기(Ctrl + C)', + paste: '붙여넣기(Ctrl + V)', + cut: '잘라내기(Ctrl + X)', + delete: '삭제하기(Delete)', + redo: 'Redo(Ctrl + Y)', + undo: 'Undo(Ctrl + Z)', + blade: '자르기 도구(C)', + link: '이어붙이기', + none: '' +} + +export { icons, iconInfo }; diff --git a/frontend/src/components/Logo/Logo.scss b/frontend/src/components/Logo/Logo.scss new file mode 100644 index 00000000..427dc326 --- /dev/null +++ b/frontend/src/components/Logo/Logo.scss @@ -0,0 +1,8 @@ +@media screen and (max-width: 1680px) { + #logo{ + svg{ + width: 140pt; + } + } +} + diff --git a/frontend/src/components/Logo/Logo.ts b/frontend/src/components/Logo/Logo.ts index c0c42dce..fb65609a 100644 --- a/frontend/src/components/Logo/Logo.ts +++ b/frontend/src/components/Logo/Logo.ts @@ -1,3 +1,5 @@ +import './Logo.scss'; + (() => { const Logo = class extends HTMLElement { public color: string; diff --git a/frontend/src/components/Main/Main.scss b/frontend/src/components/Main/Main.scss index 63b162f0..08e4ddeb 100644 --- a/frontend/src/components/Main/Main.scss +++ b/frontend/src/components/Main/Main.scss @@ -6,40 +6,20 @@ .audi-main-content { display: flex; - justify-content: space-between; width: 100%; height: 100%; padding: 30px; border: 1px solid #ffffff; border-radius: 5px; background-color: #000000; - - aside { - width: 20%; - } - - .audi-main-audio-track-container { - position: relative; - width: 80%; - height: 100%; - background-color: rgba(0, 0, 0, 0); - - .audi-main-audio-track-container-event-zone { - position: absolute; - width: 100%; - height: 95%; - - top: 2rem; - left: 0; - - z-index: 90; - } - } } + .cursor-change { - cursor: url("https://user-images.githubusercontent.com/7006837/100991542-73da8780-3596-11eb-84fc-de0de7494be8.png") 6 12, auto !important; + cursor: url("https://user-images.githubusercontent.com/7006837/100991542-73da8780-3596-11eb-84fc-de0de7494be8.png") 7 12, auto !important; & * { - cursor: url("https://user-images.githubusercontent.com/7006837/100991542-73da8780-3596-11eb-84fc-de0de7494be8.png") 6 12, auto !important; - } + cursor: url("https://user-images.githubusercontent.com/7006837/100991542-73da8780-3596-11eb-84fc-de0de7494be8.png") 7 12, auto !important; } + } } + + diff --git a/frontend/src/components/Main/Main.ts b/frontend/src/components/Main/Main.ts index 35efe7cc..cb0e87ea 100644 --- a/frontend/src/components/Main/Main.ts +++ b/frontend/src/components/Main/Main.ts @@ -1,99 +1,29 @@ -import { Track } from '@model'; -import { Controller } from '@controllers'; -import { EventUtil, MarkerEventUtil } from '@util'; -import { EventType, EventKeyType } from '@types'; import './Main.scss'; (() => { const Main = class extends HTMLElement { - private trackList: Track[]; - private mainAudioTrackContainerEventZone: HTMLElement | null; - private defaultStartX: number; - private markerElement: HTMLElement | null; - private mainWidth: number; - constructor() { super(); - this.trackList = Controller.getTrackList(); - this.mainAudioTrackContainerEventZone = null; - this.defaultStartX = 0; - this.markerElement = null; - this.mainWidth = 0; } connectedCallback(): void { - this.init(); - } - - init(): void { this.render(); - this.initElement(); - this.initEvent(); } render(): void { this.innerHTML = `
- -
- - - ${this.getTrackList()} -
-
+ + +
`; } - - initElement(): void { - const playbarElement = document.querySelector('.playbar'); - this.mainWidth = playbarElement?.getBoundingClientRect().right - playbarElement?.getBoundingClientRect().left; - this.mainAudioTrackContainerEventZone = document.querySelector('.audi-main-audio-track-container-event-zone'); - this.defaultStartX = this.mainAudioTrackContainerEventZone?.getBoundingClientRect().left; - this.markerElement = document.querySelector('.marker'); - } - - initEvent(): void { - EventUtil.registerEventToRoot({ - eventTypes: [EventType.click], - eventKey: EventKeyType.FOCUS_RESET_CLICK, - listeners: [this.focusResetListener], - bindObj: this - }); - - if (!this.markerElement) return; - - EventUtil.registerEventToRoot({ - eventTypes: [EventType.mousemove, EventType.click], - eventKey: EventKeyType.AUDIO_TRACK_CONTAINER_MULTIPLE, - listeners: [ - MarkerEventUtil.mousemoveMarkerListener(this.mainAudioTrackContainerEventZone, this.defaultStartX, this.mainWidth), - MarkerEventUtil.clickMarkerListener(this.markerElement) - ], - bindObj: this.mainAudioTrackContainerEventZone - }); - } - - focusResetListener(e): void { - const ctrlIsPressed = Controller.getCtrlIsPressed(); - if (!ctrlIsPressed) { - Controller.resetFocus(); - } - } - - getTrackList(): string { - return this.trackList.reduce((acc, cur, idx) => - acc += ``, ""); - } }; customElements.define('audi-main', Main); })(); -export {}; +export { }; diff --git a/frontend/src/components/Main/MainAsideContent/MainAsideContent.scss b/frontend/src/components/Main/MainAsideContent/MainAsideContent.scss new file mode 100644 index 00000000..6fa4be84 --- /dev/null +++ b/frontend/src/components/Main/MainAsideContent/MainAsideContent.scss @@ -0,0 +1,9 @@ +audi-main-aside-content{ + width: 15%; + height: 100%; + + aside { + width: 100%; + height: 100%; + } +} diff --git a/frontend/src/components/Main/MainAsideContent/MainAsideContent.ts b/frontend/src/components/Main/MainAsideContent/MainAsideContent.ts new file mode 100644 index 00000000..84a8f8dd --- /dev/null +++ b/frontend/src/components/Main/MainAsideContent/MainAsideContent.ts @@ -0,0 +1,25 @@ +import './MainAsideContent.scss'; + +(() => { + const MainAsideContent = class extends HTMLElement { + constructor() { + super(); + } + + connectedCallback(): void { + this.render(); + } + + render(): void { + this.innerHTML = ` + + `; + } + }; + + customElements.define('audi-main-aside-content', MainAsideContent); +})(); + +export { }; diff --git a/frontend/src/components/Main/MainMiddleContent/MainMiddleContent.scss b/frontend/src/components/Main/MainMiddleContent/MainMiddleContent.scss new file mode 100644 index 00000000..63a8b63e --- /dev/null +++ b/frontend/src/components/Main/MainMiddleContent/MainMiddleContent.scss @@ -0,0 +1,18 @@ +audi-main-middle-content{ + width: 8%; + height: 100%; + + .audi-main-menu-container{ + display: flex; + flex-direction: column; + justify-content: space-between; + align-items: center; + width: 100%; + height: 100%; + padding: 0 12%; + + audi-track-menu{ + height: 10%; + } + } +} diff --git a/frontend/src/components/Main/MainMiddleContent/MainMiddleContent.ts b/frontend/src/components/Main/MainMiddleContent/MainMiddleContent.ts new file mode 100644 index 00000000..15562963 --- /dev/null +++ b/frontend/src/components/Main/MainMiddleContent/MainMiddleContent.ts @@ -0,0 +1,27 @@ +import './MainMiddleContent.scss'; + +(() => { + const MainMiddleContent = class extends HTMLElement { + + constructor() { + super(); + } + + connectedCallback(): void { + this.render(); + } + + render(): void { + this.innerHTML = ` +
+ + +
+ `; + } + }; + + customElements.define('audi-main-middle-content', MainMiddleContent); +})(); + +export { }; diff --git a/frontend/src/components/Main/MainMiddleContent/MainTrackOptionListArea/MainTrackOptionListArea.scss b/frontend/src/components/Main/MainMiddleContent/MainTrackOptionListArea/MainTrackOptionListArea.scss new file mode 100644 index 00000000..4949ae33 --- /dev/null +++ b/frontend/src/components/Main/MainMiddleContent/MainTrackOptionListArea/MainTrackOptionListArea.scss @@ -0,0 +1,18 @@ +audi-main-track-option-list-area{ + width: 100%; + height: 90%; + + .audi-main-track-option-area{ + width: 100%; + height: 90%; + margin-top: 1.8rem; + + overflow-y: scroll; + -ms-overflow-style: none; + scrollbar-width: none; + + &::-webkit-scrollbar { + display: none; + } + } +} diff --git a/frontend/src/components/Main/MainMiddleContent/MainTrackOptionListArea/MainTrackOptionListArea.ts b/frontend/src/components/Main/MainMiddleContent/MainTrackOptionListArea/MainTrackOptionListArea.ts new file mode 100644 index 00000000..125ef2e3 --- /dev/null +++ b/frontend/src/components/Main/MainMiddleContent/MainTrackOptionListArea/MainTrackOptionListArea.ts @@ -0,0 +1,51 @@ +import { Track } from '@model'; +import { Controller } from '@controllers'; +import { StoreChannelType } from '@types'; +import { storeChannel } from '@store'; +import './MainTrackOptionListArea.scss'; + +(() => { + const MainTrackOptionListArea = class extends HTMLElement { + private trackList: Track[]; + + constructor() { + super(); + this.trackList = Controller.getTrackList(); + } + + connectedCallback(): void { + this.render(); + this.subscribe(); + } + + render(): void { + this.innerHTML = ` +
+ ${this.getTrackOptions()} +
+ `; + } + + getTrackOptions(): string { + return this.trackList.reduce((acc, cur, idx) => + acc += ``, ""); + } + + subscribe(): void { + storeChannel.subscribe(StoreChannelType.TRACK_LIST_CHANNEL, this.trackListObserverCallback, this); + } + + trackListObserverCallback(newTrackList: Track[]): void { + try{ + this.trackList = newTrackList; + this.render(); + }catch(e){ + console.log(e); + } + } + }; + + customElements.define('audi-main-track-option-list-area', MainTrackOptionListArea); +})(); + +export { }; diff --git a/frontend/src/components/Main/MainRightContent/MainRightContent.scss b/frontend/src/components/Main/MainRightContent/MainRightContent.scss new file mode 100644 index 00000000..bd385225 --- /dev/null +++ b/frontend/src/components/Main/MainRightContent/MainRightContent.scss @@ -0,0 +1,44 @@ +audi-main-right-content{ + width: 77%; + + .audi-main-audio-track-container { + display: flex; + flex-direction: column; + justify-content: space-between; + position: relative; + width: 100%; + height: 100%; + background-color: rgba(0, 0, 0, 0); + overflow: hidden; + + .audi-main-right-top{ + height: 90%; + + .audi-main-audio-track-scroll-area{ + height: 100%; + overflow-x: scroll; + -ms-overflow-style: none; + scrollbar-width: none; + + &::-webkit-scrollbar { + display: none; + } + } + + .audi-main-audio-track-container-event-zone { + position: absolute; + width: 100%; + height: 95%; + + top: 2rem; + left: 0; + + z-index: 90; + } + } + + .audi-main-right-bottom{ + height: 10%; + } + } +} diff --git a/frontend/src/components/Main/MainRightContent/MainRightContent.ts b/frontend/src/components/Main/MainRightContent/MainRightContent.ts new file mode 100644 index 00000000..c9e4443a --- /dev/null +++ b/frontend/src/components/Main/MainRightContent/MainRightContent.ts @@ -0,0 +1,116 @@ +import './MainRightContent.scss'; +import { Track } from '@model'; +import { Controller } from '@controllers'; +import { EventUtil, MarkerEventUtil } from '@util'; +import { EventType, EventKeyType, StoreChannelType } from '@types'; +import { storeChannel } from "@store"; + +(() => { + const MainRightContent = class extends HTMLElement { + private trackList: Track[]; + private mainAudioTrackContainerEventZone: HTMLElement | null; + private defaultStartX: number; + private markerElement: HTMLElement | null; + private mainWidth: number; + private maxTrackPlayTime: number; + + constructor() { + super(); + this.trackList = Controller.getTrackList(); + this.defaultStartX = 0; + this.markerElement = null; + this.mainWidth = 0; + this.maxTrackPlayTime = Controller.getMaxTrackPlayTime(); + this.mainAudioTrackContainerEventZone = null; + } + + connectedCallback(): void { + try{ + this.init(); + }catch(e){ + console.log(e); + } + } + + init(): void { + this.render(); + this.initElement(); + this.initEvent(); + } + + render(): void { + this.innerHTML = ` +
+
+
+ + + +
+
+
+
+ + +
+
+ `; + } + + initElement(): void { + const playbarElement = document.querySelector('.playbar'); + this.mainAudioTrackContainerEventZone = document.querySelector('.audi-main-audio-track-container-event-zone'); + + if(!playbarElement || !this.mainAudioTrackContainerEventZone) return; + + this.mainWidth = playbarElement?.getBoundingClientRect().right - playbarElement?.getBoundingClientRect().left; + this.defaultStartX = this.mainAudioTrackContainerEventZone?.getBoundingClientRect().left; + this.markerElement = document.querySelector('.marker'); + } + + initEvent(): void { + EventUtil.registerEventToRoot({ + eventTypes: [EventType.click], + eventKey: EventKeyType.FOCUS_RESET_CLICK, + listeners: [this.focusResetListener], + bindObj: this + }); + + if (!this.markerElement || !this.mainAudioTrackContainerEventZone) return; + + EventUtil.registerEventToRoot({ + eventTypes: [EventType.mousemove, EventType.click], + eventKey: EventKeyType.AUDIO_TRACK_CONTAINER_MULTIPLE, + listeners: [ + MarkerEventUtil.mousemoveMarkerListener(this.mainAudioTrackContainerEventZone, this.defaultStartX, this.mainWidth, this.maxTrackPlayTime), + MarkerEventUtil.clickMarkerListener(this.markerElement) + ], + bindObj: this.mainAudioTrackContainerEventZone + }); + } + + focusResetListener(e): void { + const ctrlIsPressed = Controller.getCtrlIsPressed(); + if (!ctrlIsPressed) { + Controller.resetFocus(); + } + } + + subscribe(): void { + storeChannel.subscribe(StoreChannelType.MAX_TRACK_PLAY_TIME_CHANNEL, this.maxTrackPlayTimeObserverCallback, this); + } + + maxTrackPlayTimeObserverCallback(maxTrackPlayTime: number): void { + try{ + this.maxTrackPlayTime = maxTrackPlayTime; + this.initEvent(); + }catch(e){ + console.log(e); + } + } + }; + + customElements.define('audi-main-right-content', MainRightContent); +})(); + +export { }; diff --git a/frontend/src/components/Main/MainRightContent/MainTrackListArea/MainTrackListArea.scss b/frontend/src/components/Main/MainRightContent/MainTrackListArea/MainTrackListArea.scss new file mode 100644 index 00000000..2736e8fe --- /dev/null +++ b/frontend/src/components/Main/MainRightContent/MainTrackListArea/MainTrackListArea.scss @@ -0,0 +1,15 @@ +.audi-main-track-list-scroll-area{ + width: 100%; + height: 90%; + overflow-x: visible; + overflow-y: scroll; + -ms-overflow-style: none; + scrollbar-width: none; + + &::-webkit-scrollbar { + display: none; + } +} + + + diff --git a/frontend/src/components/Main/MainRightContent/MainTrackListArea/MainTrackListArea.ts b/frontend/src/components/Main/MainRightContent/MainTrackListArea/MainTrackListArea.ts new file mode 100644 index 00000000..805b5b62 --- /dev/null +++ b/frontend/src/components/Main/MainRightContent/MainTrackListArea/MainTrackListArea.ts @@ -0,0 +1,53 @@ +import './MainTrackListArea.scss'; +import { Track } from '@model'; +import { Controller } from '@controllers'; +import { StoreChannelType } from '@types'; +import { storeChannel } from '@store'; + +(() => { + const MainTrackListArea = class extends HTMLElement { + private trackList: Track[]; + + constructor() { + super(); + this.trackList = Controller.getTrackList(); + } + + connectedCallback(): void { + try{ + this.render(); + this.subscribe(); + }catch(e){ + console.log(e); + } + } + + render(): void { + this.innerHTML = ` + ${this.getTrackList()} + `; + } + + getTrackList(): string { + return this.trackList.reduce((acc, cur, idx) => + acc += ``, ""); + } + + subscribe(): void { + storeChannel.subscribe(StoreChannelType.TRACK_LIST_CHANNEL, this.trackListObserverCallback, this); + } + + trackListObserverCallback(newTrackList: Track[]): void { + try{ + this.trackList = newTrackList; + this.render(); + }catch(e){ + console.log(e); + } + } + }; + + customElements.define('audi-main-track-list-area', MainTrackListArea); +})(); + +export { }; diff --git a/frontend/src/components/Main/index.js b/frontend/src/components/Main/index.js new file mode 100644 index 00000000..1bed9c5b --- /dev/null +++ b/frontend/src/components/Main/index.js @@ -0,0 +1,6 @@ +import './MainRightContent/MainTrackListArea/MainTrackListArea'; +import './MainRightContent/MainRightContent'; +import './MainMiddleContent/MainTrackOptionListArea/MainTrackOptionListArea'; +import './MainMiddleContent/MainMiddleContent'; +import './MainAsideContent/MainAsideContent'; +import './Main'; diff --git a/frontend/src/components/Marker/Marker.ts b/frontend/src/components/Marker/Marker.ts index 3b82795e..e1b5f4e2 100644 --- a/frontend/src/components/Marker/Marker.ts +++ b/frontend/src/components/Marker/Marker.ts @@ -27,8 +27,20 @@ import { StoreChannelType } from '@types'; const markerElement: HTMLElement | null = document.querySelector('.marker'); if (!markerElement) return; + + if (!newCurrentPosition) { + markerElement.style.left = `${newCurrentPosition}px`; + return; + } + + if(newCurrentPosition[1] === 1) { + markerElement.style.left = `${newCurrentPosition[0]}px`; + return; + } + const prevCurrentPosition = Number(markerElement?.style.left.split('px')[0]); - const currentPosition = prevCurrentPosition + newCurrentPosition; + let currentPosition = prevCurrentPosition + newCurrentPosition; + if (currentPosition < 0) currentPosition = 0; markerElement.style.left = `${currentPosition}px`; } }; diff --git a/frontend/src/components/Modal/ModalContents/SourceDownload/SourceDownload.ts b/frontend/src/components/Modal/ModalContents/SourceDownload/SourceDownload.ts index c087a91e..5501419e 100644 --- a/frontend/src/components/Modal/ModalContents/SourceDownload/SourceDownload.ts +++ b/frontend/src/components/Modal/ModalContents/SourceDownload/SourceDownload.ts @@ -51,7 +51,7 @@ import './SourceDownload.scss';
@@ -129,11 +129,7 @@ import './SourceDownload.scss'; this.changeBtnValue('압축 중'); this.inactiveSaveButton(this.saveButton); - const URL = 'https://s3-us-west-2.amazonaws.com/s.cdpn.io/123941/Yodel_Sound_Effect.mp3'; - const response = await window.fetch(URL); - const arrayBuffer = await response.arrayBuffer(); - - await saveFile(arrayBuffer, compressorObject); + await saveFile(compressorObject); this.changeBtnValue('저장하기'); this.activeSaveButton(this.saveButton); @@ -152,6 +148,11 @@ import './SourceDownload.scss'; fileNameChangeListener(e): void { if (!this.saveButton) return; this.saveButtonActivationHandler(); + + if (!this.downloadLink || !this.downloadLink.getAttribute('download') || !this.formElement) return; + + const fileName = `${this.formElement.fileName.value}.${this.formElement.extention.value}`; + this.downloadLink.setAttribute('download', fileName); }; extentionChangeListener(e): void { diff --git a/frontend/src/components/PlayBar/PlayBar.scss b/frontend/src/components/PlayBar/PlayBar.scss index 13f8fb97..1d642302 100644 --- a/frontend/src/components/PlayBar/PlayBar.scss +++ b/frontend/src/components/PlayBar/PlayBar.scss @@ -11,22 +11,36 @@ color: variables.$white-color; font-size: 0.7rem; + + .playbar-event-zone { + width: 100%; + height: 100%; + position: absolute; + top: 0; + left: 0; + + z-index: 3; + } } -.playbar-time { +.playbar-time-container { display: flex; flex-direction: column; justify-content: flex-end; align-items: center; + position: absolute; - width: 10%; height: 0.5rem; pointer-events: none; + + .playbar-time{ + position: absolute; + top: -230%; + } } .playbar-time-none { - width: 5.5%; height: 0.5rem; pointer-events: none; diff --git a/frontend/src/components/PlayBar/PlayBar.ts b/frontend/src/components/PlayBar/PlayBar.ts index 4d4777bc..1650ff24 100644 --- a/frontend/src/components/PlayBar/PlayBar.ts +++ b/frontend/src/components/PlayBar/PlayBar.ts @@ -1,121 +1,293 @@ -import './PlayBar.scss'; -import { PlayBarUtil, MarkerEventUtil } from '@util'; -import { EventUtil } from '@util'; -import { EventType, EventKeyType } from '@types'; +import { TimeUtil, MarkerEventUtil, EventUtil } from '@util'; +import { EventType, EventKeyType, StoreChannelType } from '@types'; import { Controller } from '@controllers'; - -const INIT_DURATION = 300; +import { storeChannel } from "@store"; +import './PlayBar.scss'; (() => { const PlayBar = class extends HTMLElement { - private defaultStartX: number; - private playBarElement: HTMLElement | null; + private playBarLeftX: number; + private playBarWidth: number; + private maxTrackWidth: number; + private maxTrackPlayTime: number; + private playBarTimeDatas: string[]; + private totalPlayTime: string; + private markerID: string; + private currentScrollAmount: number; + private playBarContainerElement: HTMLElement | null; private playbarMarkerElementLeft: HTMLElement | null; private playbarMarkerElementRight: HTMLElement | null; - private mainWidth: number; + private playbarMarkerBlurZoneElementLeft: HTMLElement | null; + private playbarMarkerBlurZoneElementRight: HTMLElement | null; + private playbarTimeElements: NodeListOf | null; private markerElement: HTMLElement | null; - private playtime: string[]; - private totalPlayTime: string; + private trackScrollAreaElement: HTMLDivElement | null; constructor() { super(); - this.defaultStartX = 0; - this.mainWidth = 0; - this.playBarElement = null; + this.playBarLeftX = 0; + this.playBarWidth = 0; + this.maxTrackWidth = 0; + this.maxTrackPlayTime = Controller.getMaxTrackPlayTime(); + this.totalPlayTime = ''; + this.playBarTimeDatas = []; + this.markerID = ''; + this.currentScrollAmount = 0; + this.playBarContainerElement = null; this.playbarMarkerElementLeft = null; this.playbarMarkerElementRight = null; + this.playbarMarkerBlurZoneElementLeft = null; + this.playbarMarkerBlurZoneElementRight = null; + this.playbarTimeElements = null; this.markerElement = null; - this.playtime = []; - this.totalPlayTime = ''; + this.trackScrollAreaElement = null; + + this.setPlayBarTimeInfo(); + } + + setPlayBarTimeInfo(): void { + this.totalPlayTime = this.parseTotalPlayTime(this.maxTrackPlayTime); + this.playBarTimeDatas = TimeUtil.getPlayBarTimes(this.maxTrackPlayTime); + } + + parseTotalPlayTime(maxTrackPlayTime: number): string { + const [minute, second] = TimeUtil.splitTime(maxTrackPlayTime); + return `${minute.toString().padStart(2, '0')}:${second.toString().padStart(2, '0')}`; } connectedCallback(): void { - this.totalPlayTime = this.getTotalTime(); - this.playtime = PlayBarUtil.getStringTime(INIT_DURATION); - this.init(); + try{ + this.init(); + }catch(e){ + console.log(e); + } } init(): void { this.render(); - this.initElement(); + this.initProperty(); + this.spreadPlayTimes(); + this.initPlayBarMarkerLocation(); this.initEvent(); + this.subscribe(); } render(): void { this.innerHTML = ` -
- - - ${this.setStringTime()} +
+
+ + +
+ + +
+ ${this.getPlayBarTimes()} +
`; } + getPlayBarTimes(): string { + return this.playBarTimeDatas.reduce((acc, time) => { + if (time === '00:00' || time === this.totalPlayTime) { + return (acc += `
`); + } + + return (acc += ` +
+
${time}
+
|
+
+ `); + }, ''); + } + + initProperty(): void { + this.playBarContainerElement = this.querySelector('.playbar'); + this.playbarMarkerElementLeft = document.getElementById('playbar-marker-left'); + this.playbarMarkerElementRight = document.getElementById('playbar-marker-right'); + this.playbarMarkerBlurZoneElementLeft = document.getElementById('playbar-marker-blur-zone-left'); + this.playbarMarkerBlurZoneElementRight = document.getElementById('playbar-marker-blur-zone-right'); + this.playbarTimeElements = this.querySelectorAll('.playbar-time-container'); + this.markerElement = document.querySelector('.marker'); + this.trackScrollAreaElement = document.querySelector('.audi-main-audio-track-scroll-area'); + + if(this.playBarContainerElement){ + this.playBarLeftX = this.playBarContainerElement.getBoundingClientRect().left; + this.playBarWidth = this.playBarContainerElement.getBoundingClientRect().right - this.playBarLeftX; + } + } + + spreadPlayTimes(){ + if(!this.playbarTimeElements || !this.playBarTimeDatas) return; + + const pixelPerSecond = this.playBarWidth / this.maxTrackPlayTime; + const offsetOfPlayBarTimes = TimeUtil.getOffsetOfPlayBarTimes(this.maxTrackPlayTime); + + this.playbarTimeElements.forEach((playbarTimeElement, idx) => { + const playBarTimeLeftX = (pixelPerSecond * offsetOfPlayBarTimes) * (idx + 1); + const ratio = playBarTimeLeftX / this.playBarWidth; + + playbarTimeElement.style.left = `${100 * ratio}%`; + }); + } + + initPlayBarMarkerLocation(): void { + if ( + this.playbarMarkerElementLeft && + this.playbarMarkerElementRight && + this.playbarMarkerBlurZoneElementLeft && + this.playbarMarkerBlurZoneElementRight + ) { + this.playbarMarkerElementLeft.style.left = '99.3%'; + this.playbarMarkerElementRight.style.left = '0px'; + this.playbarMarkerBlurZoneElementLeft.style.left = '100%'; + this.playbarMarkerBlurZoneElementRight.style.left = '0px'; + } + } + initEvent(): void { + if(!this.markerElement) return; + EventUtil.registerEventToRoot({ - eventTypes: [EventType.mousemove, EventType.dblclick, EventType.click], + eventTypes: [EventType.mousemove, EventType.dblclick, EventType.click, EventType.dragover, EventType.drop], eventKey: EventKeyType.PLAYBAR_MULTIPLE, listeners: [ - MarkerEventUtil.mousemoveMarkerListener(this, this.defaultStartX, this.mainWidth), + MarkerEventUtil.mousemoveMarkerListener(this, this.playBarLeftX, this.playBarWidth, this.currentScrollAmount, this.maxTrackPlayTime), this.dblclickPlayBarListener, - MarkerEventUtil.clickMarkerListener(this.markerElement) + MarkerEventUtil.clickMarkerListener(this.markerElement), + this.dragoverPlayBarListener, + this.dropPlayBarListener ], bindObj: this }); + + this.addEventListener('dragstart', this.dragStartPlayBarMarkerListener.bind(this)); + window.addEventListener('resize', this.windowResizeListener.bind(this)); + } + + dragStartPlayBarMarkerListener(e): void { + this.markerID = e.target.getAttribute('id').split('-')[2]; + const dragImage = document.createElement('div'); + dragImage.style.visibility = 'hidden'; + e.dataTransfer.setDragImage(dragImage, 0, 0); } - initElement(): void { - this.playBarElement = document.querySelector('audi-playbar'); - this.playbarMarkerElementLeft = document.getElementById('playbar-marker-left'); - this.playbarMarkerElementRight = document.getElementById('playbar-marker-right'); - this.markerElement = document.querySelector('.marker'); - this.defaultStartX = this.playBarElement?.getBoundingClientRect().left; - this.mainWidth = this.playBarElement?.getBoundingClientRect().right - this.defaultStartX; + dragoverPlayBarListener(e): void { + e.preventDefault(); + const movingWidth: number = e.pageX - this.playBarLeftX; + + if (this.markerID === 'left' && this.playbarMarkerElementLeft) { + this.playbarMarkerElementLeft.style.left = `${movingWidth}px`; + + if (this.playbarMarkerBlurZoneElementLeft) { + const blurZone = this.playBarWidth - movingWidth; + this.playbarMarkerBlurZoneElementLeft.style.width = `${blurZone}px`; + this.playbarMarkerBlurZoneElementLeft.style.left = `${movingWidth}px`; + } + return; + } + + if (this.playbarMarkerElementRight) { + this.playbarMarkerElementRight.style.left = `${movingWidth}px`; + + if (this.playbarMarkerBlurZoneElementRight) { + this.playbarMarkerBlurZoneElementRight.style.width = `${movingWidth}px`; + } + } + } + + dropPlayBarListener(e): void { + e.preventDefault(); } dblclickPlayBarListener(): void { - if (!this.playbarMarkerElementLeft || !this.playbarMarkerElementRight) return; + if ( + !this.playbarMarkerElementLeft || + !this.playbarMarkerElementRight || + !this.playbarMarkerBlurZoneElementLeft || + !this.playbarMarkerBlurZoneElementRight + ) + return; const [currentPosition] = Controller.getCurrentPosition(); if (this.compareCloseMarker(currentPosition)) { + const blurZone = this.playBarWidth - currentPosition; this.playbarMarkerElementLeft.style.left = `${currentPosition}px`; + this.playbarMarkerBlurZoneElementLeft.style.width = `${blurZone}px`; + this.playbarMarkerBlurZoneElementLeft.style.left = `${currentPosition}px`; return; } this.playbarMarkerElementRight.style.left = `${currentPosition}px`; - } - - setStringTime(): string { - return this.playtime.reduce((acc, time) => { - if (time === '00:00' || time === this.totalPlayTime) { - return (acc += `
`); - } - - return (acc += ` -
-
${time}
-
|
-
- `); - }, ''); - } - - getTotalTime(): string { - const [minute, second] = PlayBarUtil.setTime(INIT_DURATION); - return `${minute.toString().padStart(2, '0')}:${second.toString().padStart(2, '0')}`; + this.playbarMarkerBlurZoneElementRight.style.width = `${currentPosition}px`; } compareCloseMarker(currentPosition: number): boolean { - const markerLeft = Number(this.playbarMarkerElementLeft?.style.left.split('px').join('')); + const markerLeft = Number(this.playbarMarkerElementLeft?.style.left.split(/px|%/).join('')); const markerRight = Number(this.playbarMarkerElementRight?.style.left.split('px').join('')); const offsetLeft = Math.abs(currentPosition - markerLeft); const offsetRight = Math.abs(currentPosition - markerRight); if (offsetLeft <= offsetRight) return true; - return false; } + + windowResizeListener(e){ + const zoombarControllerElement = document.querySelector('.audi-zoombar-cotroller'); + if(!this.playBarContainerElement || !this.trackScrollAreaElement || !zoombarControllerElement) return; + + this.playBarLeftX = this.playBarContainerElement.getBoundingClientRect().left; + this.playBarWidth = this.playBarContainerElement.getBoundingClientRect().right - this.playBarLeftX; + + const initScrollAmount = 0; + + this.trackScrollAreaElement.scrollLeft = initScrollAmount; + zoombarControllerElement.style.left = `${initScrollAmount}`; + this.currentScrollAmount = initScrollAmount; + this.initEvent(); + } + + subscribe(): void { + storeChannel.subscribe(StoreChannelType.MAX_TRACK_WIDTH_CHANNEL, this.maxTrackWidthObserverCallback, this); + storeChannel.subscribe(StoreChannelType.MAX_TRACK_PLAY_TIME_CHANNEL, this.maxTrackPlayTimeObserverCallback, this); + storeChannel.subscribe(StoreChannelType.CURRENT_SCROLL_AMOUNT_CHANNEL, this.currentScrollAmountObserverCallback, this); + } + + maxTrackWidthObserverCallback(maxTrackWidth: number): void { + this.maxTrackWidth = maxTrackWidth; + this.resizePlayBarContainer(); + this.spreadPlayTimes(); + } + + maxTrackPlayTimeObserverCallback(maxTrackPlayTime: number): void { + this.maxTrackPlayTime = maxTrackPlayTime; + this.setPlayBarTimeInfo(); + this.render(); + this.initProperty(); + this.initPlayBarMarkerLocation(); + this.resizePlayBarContainer(); + this.spreadPlayTimes(); + this.initEvent(); + } + + resizePlayBarContainer(){ + if(!this.playBarContainerElement || !this.trackScrollAreaElement) return; + + const scrollAreaWidth = this.trackScrollAreaElement.getBoundingClientRect().right - this.trackScrollAreaElement.getBoundingClientRect().left; + const ratio = this.maxTrackWidth / scrollAreaWidth; + + this.playBarContainerElement.style.width = `${100 * ratio}%`; + this.playBarLeftX = this.playBarContainerElement.getBoundingClientRect().left; + this.playBarWidth = this.playBarContainerElement.getBoundingClientRect().right - this.playBarLeftX; + } + + currentScrollAmountObserverCallback(newCurrentScrollAmount: number): void { + this.currentScrollAmount = newCurrentScrollAmount; + this.initEvent(); + } }; customElements.define('audi-playbar', PlayBar); diff --git a/frontend/src/components/PlayBar/PlayBarMarker/PlayBarMarker.scss b/frontend/src/components/PlayBar/PlayBarMarker/PlayBarMarker.scss index c1f33b8f..df4814cc 100644 --- a/frontend/src/components/PlayBar/PlayBarMarker/PlayBarMarker.scss +++ b/frontend/src/components/PlayBar/PlayBarMarker/PlayBarMarker.scss @@ -9,4 +9,8 @@ top: -0.2rem; left: 0px; + + cursor: pointer; + + z-index: 4; } diff --git a/frontend/src/components/PlayBar/PlayBarMarkerBlurZone/PlayBarMarkerBlurZone.scss b/frontend/src/components/PlayBar/PlayBarMarkerBlurZone/PlayBarMarkerBlurZone.scss new file mode 100644 index 00000000..e80af318 --- /dev/null +++ b/frontend/src/components/PlayBar/PlayBarMarkerBlurZone/PlayBarMarkerBlurZone.scss @@ -0,0 +1,12 @@ +@use '@style/variables' as variables; + +.playbar-marker-blur-zone { + width: 0; + height: 100%; + + position: absolute; + + top: 0; + left: 0; + background-color: rgba($color: variables.$green-color, $alpha: 0.4); +} diff --git a/frontend/src/components/PlayBar/PlayBarMarkerBlurZone/PlayBarMarkerBlurZone.ts b/frontend/src/components/PlayBar/PlayBarMarkerBlurZone/PlayBarMarkerBlurZone.ts new file mode 100644 index 00000000..64548b9f --- /dev/null +++ b/frontend/src/components/PlayBar/PlayBarMarkerBlurZone/PlayBarMarkerBlurZone.ts @@ -0,0 +1,43 @@ +import './PlayBarMarkerBlurZone.scss'; + +(() => { + const PlayBarMarkerBlurZone = class extends HTMLElement { + private type: string; + + constructor() { + super(); + this.type = ''; + } + + static get observedAttributes(): string[] { + return ['type']; + } + + connectedCallback(): void { + this.render(); + } + + attributeChangedCallback(attrName: string, oldVal: string, newVal: string): void { + if (!newVal) return; + + if (oldVal !== newVal) { + switch (attrName) { + case 'type': + this.type = newVal; + break; + } + this[attrName] = newVal; + } + } + + render(): void { + this.innerHTML = ` +
+ `; + } + }; + + customElements.define('audi-playbar-marker-blur-zone', PlayBarMarkerBlurZone); +})(); + +export {}; diff --git a/frontend/src/components/PlaybackTools/PlaybackToolClass.ts b/frontend/src/components/PlaybackTools/PlaybackToolClass.ts new file mode 100644 index 00000000..bd73d125 --- /dev/null +++ b/frontend/src/components/PlaybackTools/PlaybackToolClass.ts @@ -0,0 +1,555 @@ +import { WidthUtil, AudioUtil } from '@util'; +import { StoreChannelType } from '@types'; +import { Source, Track, TrackSection, AudioSourceInfoInTrack } from '@model'; +import { storeChannel } from '@store'; +import { Controller } from '@controllers'; + +const TIMER_TIME = 34; +const QUANTUM = 3; +const INF = 987654321; + +class PlaybackToolClass { + private audioContext: AudioContext; + private trackList: Track[]; + private sourceList: Source[]; + private sourceInfo: AudioSourceInfoInTrack[]; + private mutedTrackList: Number[]; + private soloTrackList: Number[]; + private analyser: AnalyserNode|null; + private maxPlayTime: number; + private loopStartTime: number; + private loopEndTime: number; + + constructor() { + this.audioContext = new AudioContext(); + this.trackList = []; + this.sourceList = []; + this.sourceInfo = []; + this.mutedTrackList = []; + this.soloTrackList = []; + this.analyser = null; + this.maxPlayTime = INF; + + this.loopStartTime = 0; + this.loopEndTime = 0; + this.subscribe(); + } + + subscribe(): void { + storeChannel.subscribe(StoreChannelType.TRACK_CHANNEL, this.trackListObserver, this); + storeChannel.subscribe(StoreChannelType.SOURCE_LIST_CHANNEL, this.sourceListObserver, this); + } + + trackListObserver(trackList): void { + this.trackList = trackList; + } + + sourceListObserver(sourceList): void { + this.sourceList = sourceList; + } + + setMute(trackId: number): void { + this.mutedTrackList.push(trackId); + + const isPause = Controller.getIsPauseState(); + if (!isPause) { + this.play(); + } + } + + unsetMute(trackId: number): void { + const idx = this.mutedTrackList.indexOf(trackId); + if (idx > -1) this.mutedTrackList.splice(idx, 1) + + const isPause = Controller.getIsPauseState(); + if (!isPause) { + this.play(); + } + } + + setSolo(trackId: number): void { + this.soloTrackList.push(trackId); + + const isPause = Controller.getIsPauseState(); + if (!isPause) { + this.play(); + } + } + + unsetSolo(trackId: number): void { + const idx = this.soloTrackList.indexOf(trackId); + if (idx > -1) this.soloTrackList.splice(idx, 1) + + const isPause = Controller.getIsPauseState(); + if (!isPause) { + this.play(); + } + } + + audioCursorPlay(): void { + this.play(); + } + + audioPlayOrPause(): number { + if (this.trackList.length == 0) return 0; + const isPause = Controller.getIsPauseState(); + + if (isPause) { + Controller.changeIsPauseState(false); + this.play(); + this.playTimer() + + return 1; + } + else { + Controller.changeIsPauseState(true); + this.pause(); + + return 2; + } + } + + audioStop(): void { + if (this.trackList.length == 0) return; + + Controller.changeIsPauseState(true); + this.stop(false); + } + + audioRepeat(): void { + if (this.trackList.length == 0) return; + + const isRepeat = Controller.getIsRepeatState(); + + if (isRepeat === false) { + Controller.changeIsRepeatState(true); + } + else { + Controller.changeIsRepeatState(false); + } + } + + audioFastRewind(): void { + if (this.trackList.length == 0) return; + + this.fastRewind(); + } + + audioFastForward(): void { + if (this.trackList.length == 0) return; + + this.fastForward(); + } + + audioSkipPrev(): void { + if (this.trackList.length == 0) return; + + const isPause = Controller.getIsPauseState(); + this.stop(!isPause); + } + + audioSkipNext(): void { + if (this.trackList.length == 0) return; + + this.skipNext(); + } + + stopAudioSources(): void { + this.sourceInfo.forEach((source) => { + try { + source.bufferSourceNode.stop(); + source.bufferSourceNode.buffer = null; + } catch (e) { } + }); + + this.audioContext.close(); + this.audioContext = new AudioContext(); + this.audioContext.suspend(); + } + + updateSourceInfo(sourceId: number, trackId: number, sectionId: number): void { + const bufferSourceNode = this.audioContext.createBufferSource(); + bufferSourceNode.buffer = this.sourceList[sourceId].buffer; + + //TODO: effect가 적용된 노드에 대해 effect를 적용하기 + //trackId와 sectionId는 effectList에 내용이 있을 때 사용된다. + //connect가 다르게 된다. + bufferSourceNode.connect(this.audioContext.destination); + + this.sourceInfo.push({ trackId: trackId, sectionId: sectionId, bufferSourceNode: bufferSourceNode }); + } + + setMaxPlayTime(): void { + let tempMaxPlayTime = 0; + this.trackList.forEach((track: Track) => { + track.trackSectionList.forEach((trackSection: TrackSection) => { + let time = trackSection.trackStartTime + trackSection.length; + if(time > tempMaxPlayTime){ + tempMaxPlayTime = time; + } + }); + }); + + this.maxPlayTime = tempMaxPlayTime; + } + + createAndConnectAnalyser(): void { + this.analyser = this.audioContext.createAnalyser(); + this.analyser.smoothingTimeConstant = 0.3; + this.analyser.fftSize = 1024; + this.sourceInfo.forEach((source) => { + try { + source.bufferSourceNode.connect(this.analyser); + } catch (e) { } + }); + this.analyser.connect(this.audioContext.destination); + } + + checkMarkerIsAtMaxPlayTime(): boolean { + if(Controller.getMarkerTime() >= this.maxPlayTime){ + return true; + } + return false; + } + + checkMarkerIsAtRepeatEndTime(): boolean { + //재생바 부분 반영되면 값 가져와서 사용 + const loopEndTime = this.maxPlayTime; + if(Controller.getMarkerTime() >= loopEndTime){ + return true; + } + return false; + } + + stopAtMaxPlayTime() { + Controller.changeIsPauseState(false); + Controller.audioPlayOrPause(); + } + + playTimer(): void { + const volumeBar = document.getElementById("audio-meter-fill"); + + let playTimer = setInterval(() => { + if(Controller.getIsRepeatState()) { + if(this.checkMarkerIsAtRepeatEndTime()){ + this.repeat(); + } + } + else { + if(this.checkMarkerIsAtMaxPlayTime()) { + this.stopAtMaxPlayTime(); + clearInterval(playTimer); + } + } + + if (Controller.getIsPauseState()) { + if (volumeBar) { + setTimeout(() => { + volumeBar.style.width = '0'; + }, TIMER_TIME) + } + clearInterval(playTimer); + } + const maxTrackPlayTime = Controller.getMaxTrackPlayTime(); + const widthPixel = WidthUtil.getPerPixel(TIMER_TIME, maxTrackPlayTime); + + const array = new Float32Array(1024); + this.analyser.getFloatTimeDomainData(array); + if (!volumeBar) return; + const colors = ['rgb(153, 194, 198)', 'rgb(110,204,136)', 'rgb(214,171,34)', 'rgb(209,81,16)'] + let decibel = AudioUtil.getDecibel(array); + if (decibel < -72) { + decibel = -72 + } + const scaledDecibel = (decibel / 72) * 100; + const percentage = 100 + scaledDecibel; + volumeBar.style.width = `${percentage}%`; + + if (percentage > 98) { + volumeBar.style.background = `linear-gradient(to right, ${colors[0]}, ${colors[1]} 70%, ${colors[2]} 95%, ${colors[3]} 99%)`; + } else if (percentage > 80) { + volumeBar.style.background = `linear-gradient(to right, ${colors[0]}, ${colors[1]} 80%,${colors[2]})`; + } else { + volumeBar.style.background = `linear-gradient(to right, ${colors[0]}, ${colors[1]} 90%)`; + } + + Controller.setMarkerWidth(widthPixel); + Controller.changePlayStringTime(TIMER_TIME); + Controller.pauseChangeMarkerNumberTime(TIMER_TIME / 1000); + }, TIMER_TIME); + } + + play(): void { + const markerTime = Controller.getMarkerTime(); + + this.stopAudioSources(); + this.sourceInfo = []; + + this.setMaxPlayTime(); + this.trackList.forEach((track: Track) => { + if (track.trackSectionList.length !== 0) { + if (this.soloTrackList.length !== 0) { + const soloTrackIdx = this.soloTrackList.indexOf(track.id); + if(soloTrackIdx === -1) return; + } + + const mutedTrackIdx = this.mutedTrackList.indexOf(track.id); + if (mutedTrackIdx > -1) return; + + track.trackSectionList.forEach((trackSection: TrackSection) => { + this.updateSourceInfo(trackSection.sourceId, trackSection.trackId, trackSection.id); + const sourceIdx = this.sourceInfo.length - 1; + + let waitTime: number = 0; + let audioStartTime: number = 0; + let playDuration: number = 0; + let diff: number = 0; + + if (markerTime <= trackSection.trackStartTime) { + waitTime = this.audioContext.currentTime + trackSection.trackStartTime - markerTime; + audioStartTime = trackSection.channelStartTime; + playDuration = trackSection.length; + + this.sourceInfo[sourceIdx].bufferSourceNode.start(waitTime, audioStartTime, playDuration); + } else if (trackSection.trackStartTime + trackSection.length < markerTime) { + //재생되지 않는 부분 + } else { + waitTime = 0; + diff = markerTime - trackSection.trackStartTime; + audioStartTime = trackSection.channelStartTime + diff; + playDuration = trackSection.length - diff; + + this.sourceInfo[sourceIdx].bufferSourceNode.start(waitTime, audioStartTime, playDuration); + } + }); + } + }); + this.audioContext.resume(); + this.createAndConnectAnalyser(); + } + + pause(): void { + this.audioContext.suspend(); + } + + stop(restart: boolean): void { + try { + this.audioContext.close(); + this.audioContext = new AudioContext(); + this.audioContext.suspend(); + + setTimeout(() => { + Controller.changeMarkerPlayStringTime(0); + Controller.changeMarkerNumberTime(0); + Controller.setMarkerWidth(0); + + if (restart) { + this.play(); + } + }, TIMER_TIME + 1); + + } + catch (e) { + console.log(e); + } + } + + repeat(): void { + const maxTrackPlayTime = Controller.getMaxTrackPlayTime(); + const widthPixel = WidthUtil.getPixelPerSecond(this.calculateTrackWidth(), maxTrackPlayTime); + + Controller.setMarkerWidth([widthPixel * this.loopStartTime, 1]); + Controller.changeMarkerPlayStringTime(this.loopStartTime); + Controller.changeMarkerNumberTime(this.loopStartTime); + + this.play(); + } + + fastRewind(): void { + let markerTime = Controller.getMarkerTime(); + + this.stopAudioSources(); + this.sourceInfo = []; + + this.createAndConnectAnalyser(); + + const maxTrackPlayTime = Controller.getMaxTrackPlayTime(); + const widthPixel = WidthUtil.getPerPixel(QUANTUM * 1000, maxTrackPlayTime); + Controller.setMarkerWidth(-widthPixel); + Controller.pauseChangeMarkerNumberTime(-QUANTUM); + Controller.changePlayStringTime(-QUANTUM * 1000); + + const isPause = Controller.getIsPauseState(); + + if (isPause) { + return; + } + + this.setMaxPlayTime(); + this.trackList.forEach((track: Track) => { + if (track.trackSectionList.length !== 0) { + + if (this.soloTrackList.length !== 0) { + const soloTrackIdx = this.soloTrackList.indexOf(track.id); + if(soloTrackIdx === -1) return; + } + + const idx = this.mutedTrackList.indexOf(track.id); + if (idx > -1) { + return; + } + + track.trackSectionList.forEach((trackSection: TrackSection) => { + this.updateSourceInfo(trackSection.sourceId, trackSection.trackId, trackSection.id); + + let waitTime: number = 0; + let audioStartTime: number = 0; + let playDuration: number = 0; + let diff: number = 0; + const sourceIdx = this.sourceInfo.length - 1; + + if (markerTime - QUANTUM <= trackSection.trackStartTime) { + if (markerTime - QUANTUM < 0) waitTime = trackSection.trackStartTime; + else waitTime = this.audioContext.currentTime + trackSection.trackStartTime - markerTime + QUANTUM; + audioStartTime = trackSection.channelStartTime; + playDuration = trackSection.length; + + this.sourceInfo[sourceIdx].bufferSourceNode.start(waitTime, audioStartTime, playDuration); + } else if (trackSection.trackStartTime <= markerTime - QUANTUM && markerTime <= trackSection.trackStartTime + trackSection.length) { + diff = markerTime - trackSection.trackStartTime - QUANTUM; + waitTime = 0; + audioStartTime = trackSection.channelStartTime + diff; + playDuration = trackSection.length - diff; + + this.sourceInfo[sourceIdx].bufferSourceNode.start(waitTime, audioStartTime, playDuration); + } else if ( + markerTime - QUANTUM <= trackSection.trackStartTime + trackSection.length && + trackSection.trackStartTime + trackSection.length <= markerTime + ) { + diff = markerTime - QUANTUM - trackSection.trackStartTime; + waitTime = 0; + audioStartTime = trackSection.channelStartTime + diff; + playDuration = trackSection.length - diff; + + this.sourceInfo[sourceIdx].bufferSourceNode.start(waitTime, audioStartTime, playDuration); + } else { + //재생 구간을 벗어난 경우 + } + }); + } + }); + this.audioContext.resume(); + this.createAndConnectAnalyser(); + } + + fastForward(): void { + let markerTime = Controller.getMarkerTime(); + + this.stopAudioSources(); + this.sourceInfo = []; + + this.createAndConnectAnalyser(); + + const maxTrackPlayTime = Controller.getMaxTrackPlayTime(); + const widthPixel = WidthUtil.getPerPixel(QUANTUM * 1000, maxTrackPlayTime); + Controller.setMarkerWidth(widthPixel); + Controller.pauseChangeMarkerNumberTime(QUANTUM); + Controller.changePlayStringTime(QUANTUM * 1000); + + const isPause = Controller.getIsPauseState(); + if (isPause) { + return; + } + + this.setMaxPlayTime(); + this.trackList.forEach((track: Track) => { + if (track.trackSectionList.length !== 0) { + + if (this.soloTrackList.length !== 0) { + const soloTrackIdx = this.soloTrackList.indexOf(track.id); + if(soloTrackIdx === -1) return; + } + + const idx = this.mutedTrackList.indexOf(track.id); + if (idx > -1) { + return; + } + + track.trackSectionList.forEach((trackSection: TrackSection) => { + this.updateSourceInfo(trackSection.sourceId, trackSection.trackId, trackSection.id); + + let waitTime: number = 0; + let audioStartTime: number = 0; + let playDuration: number = 0; + let diff: number = 0; + const sourceIdx = this.sourceInfo.length - 1; + + if (markerTime + QUANTUM <= trackSection.trackStartTime) { + waitTime = this.audioContext.currentTime + trackSection.trackStartTime - (markerTime + QUANTUM); + audioStartTime = trackSection.channelStartTime; + playDuration = trackSection.length; + + this.sourceInfo[sourceIdx].bufferSourceNode.start(waitTime, audioStartTime, playDuration); + } else if (trackSection.trackStartTime <= markerTime && markerTime + QUANTUM <= trackSection.trackStartTime + trackSection.length) { + diff = markerTime - trackSection.trackStartTime + QUANTUM; + waitTime = 0; + audioStartTime = trackSection.channelStartTime + diff; + playDuration = trackSection.length - diff; + + this.sourceInfo[sourceIdx].bufferSourceNode.start(waitTime, audioStartTime, playDuration); + } else if (markerTime <= trackSection.trackStartTime && trackSection.trackStartTime <= markerTime + QUANTUM) { + diff = markerTime + QUANTUM - trackSection.trackStartTime; + waitTime = 0; + audioStartTime = trackSection.channelStartTime + diff; + playDuration = trackSection.length - diff; + + this.sourceInfo[sourceIdx].bufferSourceNode.start(waitTime, audioStartTime, playDuration); + } else { + //재생 구간을 벗어난 경우 + } + }); + } + }); + this.audioContext.resume(); + this.createAndConnectAnalyser(); + } + + skipNext(): void { + try { + this.audioContext.close(); + this.audioContext = new AudioContext(); + this.audioContext.suspend(); + + //loop여부 관계없이 맨 마지막에 멈추게 하는 게 구현은 편할 것. + Controller.changeIsPauseState(true); + this.setMaxPlayTime(); + + setTimeout(() => { + const maxTrackPlayTime = Controller.getMaxTrackPlayTime(); + this.setMaxPlayTime(); + + const widthPixel = WidthUtil.getPixelPerSecond(this.calculateTrackWidth(), maxTrackPlayTime); + Controller.setMarkerWidth([widthPixel * this.maxPlayTime, 1]); + Controller.changeMarkerPlayStringTime(this.maxPlayTime); + Controller.changeMarkerNumberTime(this.maxPlayTime); + }, TIMER_TIME + 1); + } + catch (e) { + console.log(e); + } + } + + //AudioTrack에 있는 함수. + calculateTrackWidth(): number{ + let trackWidth = 0; + const trackAreaElement = document.querySelector('.audio-track-area'); + if(trackAreaElement){ + trackWidth = trackAreaElement.getBoundingClientRect().right - trackAreaElement.getBoundingClientRect().left; + } + return trackWidth; + } +} + +const playbackTool = new PlaybackToolClass(); + +export default playbackTool; \ No newline at end of file diff --git a/frontend/src/components/PlaybackTools/PlaybackTools.scss b/frontend/src/components/PlaybackTools/PlaybackTools.scss index 3b589a1e..5e2d46fe 100644 --- a/frontend/src/components/PlaybackTools/PlaybackTools.scss +++ b/frontend/src/components/PlaybackTools/PlaybackTools.scss @@ -1,16 +1,22 @@ +@use '@style/variables.scss'as variables; + .playback-tools { - display:flex; + display: flex; justify-content: space-around; height: 50px; margin: auto 10px; padding: 0 5px; - + border: 1px solid white; border-radius: 6px; } .playback-tools > audi-icon-button { margin: auto 5px; +} - cursor: pointer; +.clicked { + path { + fill: variables.$green-color; + } } diff --git a/frontend/src/components/PlaybackTools/PlaybackTools.ts b/frontend/src/components/PlaybackTools/PlaybackTools.ts index 803ffb94..23a464f7 100644 --- a/frontend/src/components/PlaybackTools/PlaybackTools.ts +++ b/frontend/src/components/PlaybackTools/PlaybackTools.ts @@ -1,27 +1,14 @@ import './PlaybackTools.scss'; -import { EventUtil, PlayBarUtil } from '@util'; +import { EventUtil } from '@util'; import { EventType, EventKeyType, StoreChannelType } from '@types'; -import { Source, Track, TrackSection, AudioSourceInfoInTrack } from '@model'; import { storeChannel } from '@store'; import { Controller } from '@controllers'; -const TIMER_TIME = 34; - (() => { const PlaybackTools = class extends HTMLElement { public iconlist: string[]; public eventKeyList: string[]; - - //TODO: 다른 ts로 옮기기 - private audioContext: AudioContext; - private isPause: boolean; - private trackList: Track[]; - private sourceList: Source[]; - private sourceInfo: AudioSourceInfoInTrack[]; - public markerTime: number; - - private passedTime: number; - + constructor() { super(); this.iconlist = ['play', 'stop', 'repeat', 'fastRewind', 'fastForward', 'skipPrev', 'skipNext']; @@ -34,15 +21,6 @@ const TIMER_TIME = 34; EventKeyType.AUDIO_SKIP_PREV, EventKeyType.AUDIO_SKIP_NEXT ]; - - //옮기기 - this.audioContext = new AudioContext(); - this.isPause = Controller.getIsPauseState(); - this.trackList = []; - this.sourceList = []; - this.sourceInfo = []; - this.markerTime = 0; - this.passedTime = 0; } connectedCallback() { @@ -54,12 +32,9 @@ const TIMER_TIME = 34; render() { this.innerHTML = `
- ${this.iconlist.reduce( - (acc, icon, idx) => - acc + - ``, - '' - )} + ${this.iconlist.reduce((acc, icon, idx) =>{ + const checkRepeat = Controller.getIsRepeatState() && (icon==='repeat'); + return acc + ``},'')}
`; } @@ -116,147 +91,56 @@ const TIMER_TIME = 34; } audioPlayOrPauseListener() { - if (this.trackList.length == 0) return; - - if (this.isPause) { - this.isPause = false; - Controller.changeIsPauseState(false); - - this.iconlist[0] = 'pause'; - - const markerTime = Controller.getMarkerTime(); - this.play(markerTime); - } else { - this.isPause = true; - Controller.changeIsPauseState(true); - - this.iconlist[0] = 'play'; - - this.pause(this.markerTime); - } - this.render(); + Controller.audioPlayOrPause(); } audioStopListener() { - if (this.trackList.length == 0) return; + Controller.audioStop(); + this.iconlist[0] = 'play'; + + this.render(); } audioRepeatListener() { - if (this.trackList.length == 0) return; + Controller.audioRepeat(); + + this.render() } audioFastRewindListener() { - if (this.trackList.length == 0) return; + Controller.audioFastRewind(); } audioFastForwardListener() { - if (this.trackList.length == 0) return; + Controller.audioFastForward(); } audioSkipPrevListener() { - if (this.trackList.length == 0) return; + Controller.audioSkipPrev(); } audioSkipNextListener() { - if (this.trackList.length == 0) return; + Controller.audioSkipNext(); } subscribe(): void { - storeChannel.subscribe(StoreChannelType.TRACK_CHANNEL, this.trackListObserver, this); - storeChannel.subscribe(StoreChannelType.SOURCE_LIST_CHANNEL, this.sourceListObserver, this); - } - - trackListObserver(trackList): void { - this.trackList = trackList; - } - - sourceListObserver(sourceList): void { - this.sourceList = sourceList; + storeChannel.subscribe(StoreChannelType.PLAY_OR_PAUSE_CHANNEL, this.changePlayOrPauseIcon, this); } - stopAudioSources() { - this.sourceInfo.forEach((source) => { - try { - source.bufferSourceNode.stop(); - source.bufferSourceNode.buffer = null; - } catch (e) {} - }); - - this.audioContext.close(); - this.audioContext = new AudioContext(); - this.audioContext.suspend(); - } - - updateSourceInfo(sourceId: number, trackId: number, sectionId: number) { - const bufferSourceNode = this.audioContext.createBufferSource(); - - bufferSourceNode.buffer = this.sourceList[sourceId].buffer; - - //TODO: effect가 적용된 노드에 대해 effect를 적용하기 - //trackId와 sectionId는 effectList에 내용이 있을 때 사용된다. - //connect가 다르게 된다. - bufferSourceNode.connect(this.audioContext.destination); - - this.sourceInfo.push({ trackId: trackId, sectionId: sectionId, bufferSourceNode: bufferSourceNode }); - } - - play(markerTime: number): void { - this.stopAudioSources(); - this.sourceInfo = []; - - this.trackList.forEach((track: Track) => { - if (track.trackSectionList.length != 0) { - track.trackSectionList.forEach((trackSection: TrackSection, idx: number) => { - this.updateSourceInfo(trackSection.sourceId, trackSection.trackId, trackSection.id); - - //when:얼마나 있다가 시작할 것인지, offset:음원을 몇 초에서 시작할 것인지, duration:몇 초 도안 재생할 것인지 - let when: number = 0; - let offset: number = 0; - let duration: number = 0; - let diff: number = 0; - - if (markerTime <= trackSection.trackStartTime) { - when = this.audioContext.currentTime + trackSection.trackStartTime - markerTime; - offset = trackSection.audioStartTime; - duration = trackSection.length; - - const sourceIdx = this.sourceInfo.length - 1; - this.sourceInfo[sourceIdx].bufferSourceNode.start(when, offset); - } else if (trackSection.trackStartTime + trackSection.length < markerTime) { - //재생되지 않는 부분 - } else { - when = 0; - diff = markerTime - trackSection.trackStartTime; - offset = trackSection.audioStartTime + diff; - duration = trackSection.length - diff; - - const sourceIdx = this.sourceInfo.length - 1; - this.sourceInfo[sourceIdx].bufferSourceNode.start(when, offset, duration); - } - }); - } - }); - this.audioContext.resume(); - this.passedTime = this.audioContext.currentTime; - - let playTimer = setInterval(() => { - if (Controller.getIsPauseState()) { - clearInterval(playTimer); - } - const widthPixel = PlayBarUtil.getSomePixel(TIMER_TIME); - Controller.setMarkerWidth(widthPixel); - Controller.changePlayTime(TIMER_TIME); - }, TIMER_TIME); - } - - pause(markerTime: number) { - this.markerTime = markerTime; - this.audioContext.suspend(); - - const timeDiff = this.audioContext.currentTime - this.passedTime; - Controller.pauseChangeMarkerTime(timeDiff); + changePlayOrPauseIcon(iconType: number) { + if(iconType === 0) { + return; + } + else if(iconType === 1) { + this.iconlist[0] = 'pause'; + } + else if(iconType === 2) { + this.iconlist[0] = 'play'; + } + this.render(); } }; customElements.define('audi-playback-tools', PlaybackTools); })(); -export {}; + +export { }; diff --git a/frontend/src/components/SectionEffectList/SectionEffectList.scss b/frontend/src/components/SectionEffectList/SectionEffectList.scss new file mode 100644 index 00000000..b10e8a98 --- /dev/null +++ b/frontend/src/components/SectionEffectList/SectionEffectList.scss @@ -0,0 +1,58 @@ +.effect-list-outer-wrap { + display: flex; + flex-direction: column; + align-items: center; + width: 100%; + height: 100%; + + color: #ffffff; + border: 1px solid white; + border-radius: 5px; + + .effect-list-title-wrap { + display: flex; + justify-content: center; + align-items: center; + + width: 95%; + height: 4rem; + + .effect-list-title { + width: 95%; + + display: flex; + justify-content: space-between; + align-items: center; + } + } + + .effect-list-wrap { + display: flex; + flex-direction: column; + align-items: center; + + width: 95%; + height: 90%; + } +} + +.effect-container { + display: flex; + justify-content: center; + align-items: center; + + width: 95%; + height: 2rem; + + background-color: black; + border: 1px solid white; + border-radius: 5px; + + .effect { + width: 90%; + + display: flex; + justify-content: space-between; + align-items: center; + } +} diff --git a/frontend/src/components/SectionEffectList/SectionEffectList.ts b/frontend/src/components/SectionEffectList/SectionEffectList.ts new file mode 100644 index 00000000..ec00b4ac --- /dev/null +++ b/frontend/src/components/SectionEffectList/SectionEffectList.ts @@ -0,0 +1,84 @@ +import { ModalType, EventType, EventKeyType, EffectList, IconType } from '@types'; +import { Controller } from '@controllers'; +import { EventUtil } from '@util'; +import './SectionEffectList.scss'; + +(() => { + const SectionEffectList = class extends HTMLElement { + private effectList: EffectList[]; + + constructor() { + super(); + this.effectList = []; + } + + connectedCallback(): void { + this.render(); + this.initEvent(); + } + + initEvent(): void { + EventUtil.registerEventToRoot({ + eventTypes: [EventType.click], + eventKey: EventKeyType.EFFECT_LIST_OPEN_MODAL_BTN_CLICK, + listeners: [this.openEffectListModalBtnClickListener], + bindObj: this + }); + } + render(): void { + this.innerHTML = ` +
+
+
+
Effect
+ +
+
+
    + ${this.getEffectList()} +
+
+ `; + } + + getEffectList(): string { + const focusList = Controller.getFocusList(); + + if (!focusList.length) return ''; + + return focusList[0].trackSection.effectList.reduce( + (acc, effect) => + (acc += ` +
  • +
    +
    ${effect.name}
    + +
    +
  • + `), + '' + ); + } + + openEffectListModalBtnClickListener(): void { + Controller.changeModalState(ModalType.effect, false); + } + + updateEffectList(newEffect: EffectList): void { + this.effectList = [...this.effectList, newEffect]; + this.render(); + } + + hide(): void { + this.classList.add('hide'); + } + + show(): void { + this.classList.remove('hide'); + } + }; + + customElements.define('audi-section-effect-list', SectionEffectList); +})(); + +export { }; diff --git a/frontend/src/components/Sidebar/Sidebar.scss b/frontend/src/components/Sidebar/Sidebar.scss index 71d3d77a..201ed39b 100644 --- a/frontend/src/components/Sidebar/Sidebar.scss +++ b/frontend/src/components/Sidebar/Sidebar.scss @@ -8,7 +8,7 @@ side-bar:not(:defined) { align-items:center; padding:1px; height:100%; - width:270px; + width:100%; border-radius: 5px; background-color: black; @@ -21,6 +21,7 @@ side-bar:not(:defined) { .sidebar-source-list { height:85%; } + .sidebar-time-info { height:15%; diff --git a/frontend/src/components/Sidebar/Sidebar.ts b/frontend/src/components/Sidebar/Sidebar.ts index 9c1320d4..dc1ebbfe 100644 --- a/frontend/src/components/Sidebar/Sidebar.ts +++ b/frontend/src/components/Sidebar/Sidebar.ts @@ -1,8 +1,7 @@ -import "./Sidebar.scss" +import './Sidebar.scss'; (() => { const Sidebar = class extends HTMLElement { - constructor() { super(); } @@ -17,6 +16,7 @@ import "./Sidebar.scss" `; } diff --git a/frontend/src/components/SourceList/SourceList.ts b/frontend/src/components/SourceList/SourceList.ts index a5e32eaf..b4133e3d 100644 --- a/frontend/src/components/SourceList/SourceList.ts +++ b/frontend/src/components/SourceList/SourceList.ts @@ -1,9 +1,9 @@ -import { EventKeyType, EventType, StoreChannelType } from "@types"; +import { EventKeyType, EventType, StoreChannelType } from '@types'; import { storeChannel } from '@store'; -import { Source } from '@model'; -import { EventUtil } from "@util"; -import { Controller } from "@controllers"; -import "./SourceList.scss"; +import { Source, SectionDragStartData } from '@model'; +import { EventUtil } from '@util'; +import { Controller } from '@controllers'; +import './SourceList.scss'; (() => { const SourceList = class extends HTMLElement { @@ -21,8 +21,10 @@ import "./SourceList.scss"; } getSources(): string { - return this.sourceList.reduce((acc, source) => - acc + `
  • + return this.sourceList.reduce( + (acc, source) => + acc + + `
  • ${source.fileName}
    • FileName: ${source.fileName}
    • @@ -31,12 +33,13 @@ import "./SourceList.scss";
    • Channel: ${source.numberOfChannels}
    • PlayTime: ${this.parsePlayTime(source.duration)}
    -
  • ` - , "") + `, + '' + ); } - - parseFileSize(fileSize: number): string{ - let parsedFileSize = fileSize / 1024 /1024 + + parseFileSize(fileSize: number): string { + let parsedFileSize = fileSize / 1024 / 1024; parsedFileSize = parsedFileSize * 100; parsedFileSize = Math.floor(parsedFileSize); parsedFileSize = parsedFileSize / 100; @@ -45,7 +48,7 @@ import "./SourceList.scss"; } parsePlayTime(playTime: number): string { - if(playTime < 60){ + if (playTime < 60) { const seconds = Math.round(playTime); return `${seconds}초`; } @@ -68,37 +71,54 @@ import "./SourceList.scss"; `; } - initEvent(): void{ + initEvent(): void { EventUtil.registerEventToRoot({ - eventTypes: [EventType.dragstart, EventType.dragend], + eventTypes: [EventType.dragstart], eventKey: EventKeyType.SOURCE_LIST_MULTIPLE, - listeners: [this.sourceListDragstartListener, this.sourceListDragendListener], + listeners: [this.sourceListDragstartListener], bindObj: this - }) + }); } - sourceListDragstartListener(e): void{ - e.dataTransfer.setData("text/plain", e.target.dataset.id); - e.dataTransfer.dropEffect = "link"; + sourceListDragstartListener(e): void { + const dragImage = document.createElement('div'); + dragImage.style.visibility = 'hidden'; + e.dataTransfer.setDragImage(dragImage, 0, 0); - Controller.changeTrackDragState(true); - } + const sourceId = Number(e.target.dataset.id); + + const trackSection = Controller.createTrackSectionFromSource(sourceId); + if (!trackSection) return; + const trackContainerElement = document.querySelector('.audio-track-container');; + + if (!trackContainerElement) return; + + const offsetLeft = trackContainerElement.getBoundingClientRect().left; + + const sectionDragStartData = new SectionDragStartData({ trackSection, offsetLeft }); + Controller.changeSectionDragStartData(sectionDragStartData); - sourceListDragendListener(e): void{ - Controller.changeTrackDragState(false); } - subscribe(): void{ - storeChannel.subscribe(StoreChannelType.SOURCE_LIST_CHANNEL,this.updateSourceList,this); + subscribe(): void { + storeChannel.subscribe(StoreChannelType.SOURCE_LIST_CHANNEL, this.updateSourceList, this); } - updateSourceList(sourceList: Source[]): void{ + updateSourceList(sourceList: Source[]): void { this.sourceList = sourceList; this.render(); } + + hide(): void { + this.classList.add('hide'); + } + + show(): void { + this.classList.remove('hide'); + } }; customElements.define('audi-source-list', SourceList); })(); -export {}; +export { }; diff --git a/frontend/src/components/TimeInfo/TimeInfo.scss b/frontend/src/components/TimeInfo/TimeInfo.scss index 5a5d3f47..b2f342a5 100644 --- a/frontend/src/components/TimeInfo/TimeInfo.scss +++ b/frontend/src/components/TimeInfo/TimeInfo.scss @@ -5,6 +5,7 @@ $green-color:rgb(22, 185, 22); flex-direction:$f-dir; align-items:center; justify-content: space-between; + flex-wrap: nowrap; } .time-info-outer-wrap { @@ -44,3 +45,38 @@ $green-color:rgb(22, 185, 22); color:white; } + +@media screen and (max-width: 1680px) { + .time-info-wrap { + .play-time { + font-size: 20px; + } + + .other-time { + flex-basis:50%; + .total-time, .cursor-time { + @include my-flex(); + margin:0 4px; + + font-size: 9px; + } + } + } +} + +@media screen and (max-width: 1360px) { + .time-info-wrap { + .play-time { + font-size: 15px; + } + + .other-time { + flex-basis:50%; + .total-time, .cursor-time { + @include my-flex(); + margin:0 4px; + font-size: 2px; + } + } + } +} diff --git a/frontend/src/components/ZoomBar/ZoomBar.scss b/frontend/src/components/ZoomBar/ZoomBar.scss new file mode 100644 index 00000000..bb660271 --- /dev/null +++ b/frontend/src/components/ZoomBar/ZoomBar.scss @@ -0,0 +1,21 @@ +.audi-zoombar-container{ + display: flex; + width: 100%; + padding: 2px 0; + border: 1px solid rgb(100, 100, 100); + border-radius: 2px; + box-sizing: border-box; + overflow: hidden; + + .audi-zoombar-cotroller { + position: relative; + top: 0; + left: 0; + width: 100%; + height: 10px; + background-color: #607D8B; + border-radius: 2px; + box-sizing: border-box; + cursor: grab; + } +} diff --git a/frontend/src/components/ZoomBar/ZoomBar.ts b/frontend/src/components/ZoomBar/ZoomBar.ts new file mode 100644 index 00000000..307a6861 --- /dev/null +++ b/frontend/src/components/ZoomBar/ZoomBar.ts @@ -0,0 +1,151 @@ +import { EventUtil } from '@util'; +import { EventKeyType, EventType, StoreChannelType } from '@types'; +import { storeChannel } from '@store'; +import { Controller } from '@controllers'; +import './ZoomBar.scss'; + +(() => { + const ZoomBar = class extends HTMLElement{ + private mouseDownX: number; + private lastLeft: number; + private scrollDistancePerPixel: number; + private zoombarContainerElement: HTMLDivElement | null; + private zoombarControllerElement: HTMLDivElement | null; + + constructor(){ + super(); + this.mouseDownX = 0; + this.lastLeft = 0; + this.scrollDistancePerPixel = 1; + this.zoombarContainerElement = null; + this.zoombarControllerElement = null; + } + + static DEFAULT_ZOOMBAR_CONTROLLER_SIZE_PERCENTAGE = 100; + static MIN_ZOOMBAR_CONTROLLER_SIZE_PERCENTAGE = 5; + static MAX_SCROLL_RATIO_TO_RESIZE = 1.0; + + connectedCallback(): void { + try { + this.render(); + this.initElement(); + this.initEvent(); + this.subscribe(); + } catch (e) { + console.log(e); + } + } + + render(): void { + this.innerHTML = ` +
    +
    +
    + `; + } + + initElement(): void { + this.zoombarContainerElement = this.querySelector('.audi-zoombar-container'); + this.zoombarControllerElement = this.querySelector('.audi-zoombar-cotroller'); + } + + initEvent(): void { + EventUtil.registerEventToRoot({ + eventTypes: [EventType.mousedown], + eventKey: EventKeyType.ZOOM_BAR_MOUSE_DOWN, + listeners: [this.zoomBarControllerMousedownListener], + bindObj: this + }); + } + + zoomBarControllerMousedownListener(e): void { + e.preventDefault(); + e.stopPropagation(); + + this.mouseDownX = e.clientX; + document.onmousemove = this.zoomBarControllerMousemoveListener.bind(this); + document.onmouseup = this.zoomBarControllerMouseupListener.bind(this); + } + + zoomBarControllerMousemoveListener(e): void { + e.preventDefault(); + e.stopPropagation(); + + if(!this.zoombarControllerElement || !this.zoombarContainerElement) return; + + const mouseMoveX = this.lastLeft + e.clientX - this.mouseDownX ; + const containerLeftX = this.zoombarContainerElement.getBoundingClientRect().left; + const containerRightX = this.zoombarContainerElement.getBoundingClientRect().right; + + if((containerLeftX + mouseMoveX) > containerLeftX && + (containerLeftX + this.zoombarControllerElement.offsetWidth + mouseMoveX) < containerRightX){ + this.zoombarControllerElement.style.left = mouseMoveX + "px"; + + const audioTrackScrollAreaElement = document.querySelector('.audi-main-audio-track-scroll-area'); + if(!audioTrackScrollAreaElement) return; + + const scrollAmount = this.scrollDistancePerPixel * mouseMoveX; + audioTrackScrollAreaElement.scrollLeft = scrollAmount; + + Controller.changeCurrentScrollAmount(scrollAmount); + } + } + + zoomBarControllerMouseupListener(e): void { + e.preventDefault(); + e.stopPropagation(); + + if(!this.zoombarControllerElement) return; + + this.lastLeft = parseInt(this.zoombarControllerElement.style.left.replace('px','')); + document.onmousemove = null; + document.onmouseup = null; + } + + subscribe(): void { + storeChannel.subscribe(StoreChannelType.MAX_TRACK_WIDTH_CHANNEL, this.maxTrackWidthObserverCallback, this); + } + + maxTrackWidthObserverCallback(maxTrackWidth: number): void { + this.adjustZoombarController(maxTrackWidth); + } + + adjustZoombarController(maxTrackWidth: number): void { + if(!this.zoombarContainerElement) return; + + const zoombarContainerWidth = this.zoombarContainerElement.clientWidth; + const scrollWidth = maxTrackWidth - zoombarContainerWidth; + + this.resizeZoomBarController(zoombarContainerWidth, scrollWidth); + this.calculateScrollDistancePerPixel(zoombarContainerWidth, scrollWidth); + } + + resizeZoomBarController(zoombarContainerWidth: number, scrollWidth: number): void { + if(!this.zoombarControllerElement) return; + + const { DEFAULT_ZOOMBAR_CONTROLLER_SIZE_PERCENTAGE, MIN_ZOOMBAR_CONTROLLER_SIZE_PERCENTAGE, MAX_SCROLL_RATIO_TO_RESIZE } = ZoomBar; + const scrollRatio = scrollWidth / zoombarContainerWidth; + + let zoombarControllerSize = DEFAULT_ZOOMBAR_CONTROLLER_SIZE_PERCENTAGE; + if(scrollRatio < MAX_SCROLL_RATIO_TO_RESIZE){ + zoombarControllerSize -= ( scrollRatio % 1 ) * 100; + if( zoombarControllerSize < MIN_ZOOMBAR_CONTROLLER_SIZE_PERCENTAGE ) + zoombarControllerSize = MIN_ZOOMBAR_CONTROLLER_SIZE_PERCENTAGE; + } + else{ + zoombarControllerSize = MIN_ZOOMBAR_CONTROLLER_SIZE_PERCENTAGE; + } + this.zoombarControllerElement.style.width = `${zoombarControllerSize}%`; + } + + calculateScrollDistancePerPixel(zoombarContainerWidth: number, scrollWidth: number): void { + if(!this.zoombarControllerElement) return; + + const zoombarControllerWidth = this.zoombarControllerElement.clientWidth; + const movementRange = zoombarContainerWidth - zoombarControllerWidth; + this.scrollDistancePerPixel = 1 / (scrollWidth / movementRange); + } + } + + customElements.define('audi-zoom-bar', ZoomBar); +})(); diff --git a/frontend/src/components/index.js b/frontend/src/components/index.js index c2fee05b..ac6b62e8 100644 --- a/frontend/src/components/index.js +++ b/frontend/src/components/index.js @@ -9,8 +9,9 @@ import './PlaybackTools/PlaybackTools'; import './Sidebar/Sidebar'; import './SourceList/SourceList'; import './TimeInfo/TimeInfo'; -import './Main/Main'; +import './Main'; import './Loading/Loading'; +import './PlayBar/PlayBarMarkerBlurZone/PlayBarMarkerBlurZone'; import './PlayBar/PlayBarMarker/PlayBarMarker'; import './PlayBar/PlayBar'; import './Marker/Marker'; @@ -18,3 +19,6 @@ import './Modal'; import './Pages'; import './Button'; import './EffectController'; +import './ZoomBar/ZoomBar'; +import './AudioMeter/AudioMeter'; +import './SectionEffectList/SectionEffectList'; diff --git a/frontend/src/components/sidebar/Sidebar.ts b/frontend/src/components/sidebar/Sidebar.ts deleted file mode 100644 index 9c1320d4..00000000 --- a/frontend/src/components/sidebar/Sidebar.ts +++ /dev/null @@ -1,27 +0,0 @@ -import "./Sidebar.scss" - -(() => { - const Sidebar = class extends HTMLElement { - - constructor() { - super(); - } - - connectedCallback() { - this.render(); - } - - render() { - this.innerHTML = ` - - - `; - } - }; - customElements.define('audi-side-bar', Sidebar); -})(); - -export {}; diff --git a/frontend/src/components/sidebar/sidebar.scss b/frontend/src/components/sidebar/sidebar.scss deleted file mode 100644 index 71d3d77a..00000000 --- a/frontend/src/components/sidebar/sidebar.scss +++ /dev/null @@ -1,28 +0,0 @@ -side-bar:not(:defined) { - display: hidden; -} - -#sidebar { - display:flex; - flex-direction:column; - align-items:center; - padding:1px; - height:100%; - width:270px; - - border-radius: 5px; - background-color: black; -} - -.sidebar-child { - width:100%; -} - -.sidebar-source-list { - height:85%; -} -.sidebar-time-info { - height:15%; - - margin-bottom: 10px; -} diff --git a/frontend/src/controllers/CommandController.ts b/frontend/src/controllers/CommandController.ts new file mode 100644 index 00000000..8b3a5197 --- /dev/null +++ b/frontend/src/controllers/CommandController.ts @@ -0,0 +1,126 @@ +import { store } from '@store'; +import { CopyUtil } from '@util'; +import { Controller } from "@controllers"; +import { TrackSection } from '@model'; +import { CommandManager, DeleteCommand, PasteCommand, SplitCommand, AddTrackCommand, DeleteTrackCommand, MoveCommand } from '@command'; + +const executeUndoCommand = () => { + const { isPause } = store.getState(); + if (CommandManager.undoList.length === 0 || !isPause) return; + CommandManager.undo(); +}; + +const executeRedoCommand = () => { + const { isPause } = store.getState(); + if (CommandManager.redoList.length === 0 || !isPause) return; + + CommandManager.redo(); +}; + +const executeDeleteCommand = () => { + const { focusList, isPause } = store.getState(); + if (focusList.length === 0 || !isPause) return; + + const command = new DeleteCommand(); + CommandManager.execute(command); +}; + +const executeCutCommand = () => { + const { isPause } = store.getState(); + if (!isPause || !Controller.setClipBoard()) return; + + const command = new DeleteCommand(); + CommandManager.execute(command); +}; + +const executePasteCommand = () => { + const { focusList, trackList, clipBoard, isPause, selectTrackData } = store.getState(); + + if (focusList.length > 1 || !isPause || !clipBoard) return; + if (focusList.length === 0 && selectTrackData.trackId === 0) return; + + const trackId = focusList.length === 1 ? focusList[0].trackSection.trackId : selectTrackData.trackId; + const track = trackList.find((track) => track.id === trackId); + + if (!track) return; + + const copyTrack = CopyUtil.copyTrack(track); + const copySection = CopyUtil.copySection(clipBoard); + + if (focusList.length === 0) { + copySection.trackStartTime = selectTrackData.selectedTime; + copySection.trackId = selectTrackData.trackId; + } else { + const focusSection = focusList[0].trackSection; + copySection.trackStartTime = focusSection.trackStartTime + focusSection.length; + copySection.trackId = focusSection.trackId; + } + + const command = new PasteCommand(copyTrack, copySection); + + CommandManager.execute(command); +}; + +const executeSplitCommand = (cursorPosition: number, trackId: number, sectionId: number): void => { + const { isPause } = store.getState(); + if (!isPause) return; + + const track = Controller.getTrack(trackId); + const trackSection = track?.trackSectionList.find((section) => section.id === sectionId); + if (!trackSection || !track) return; + + const splitCommand = new SplitCommand(cursorPosition, CopyUtil.copyTrack(track), CopyUtil.copySection(trackSection)); + CommandManager.execute(splitCommand); +}; + +const executeAddTrackCommand = (): void => { + const { isPause } = store.getState(); + if (!isPause) return; + + const addTrackCommand = new AddTrackCommand(); + CommandManager.execute(addTrackCommand); +}; + +const executeMoveCommand = (prevTrackId: number, currentTrackId: number, trackSection: TrackSection, movingCursorTime: number, prevCursorTime: number) => { + const { trackList, isPause } = store.getState(); + if (!isPause) return; + const prevTrack = trackList.find((track) => track.id === prevTrackId); + const currentTrack = trackList.find((track) => track.id === currentTrackId); + + if (!prevTrack || !currentTrack) return; + + const command = new MoveCommand(CopyUtil.copyTrack(prevTrack), CopyUtil.copyTrack(currentTrack), trackSection, movingCursorTime, prevCursorTime); + CommandManager.execute(command); +}; + +const executeDeleteTrackCommand = (trackId: number): void => { + const { isPause } = store.getState(); + if (!isPause) return; + + if(isMinLengthOfTrackList()) { + alert("트랙은 최소 3개 이상 존재해야합니다."); + return; + } + + const deleteTrackCommand = new DeleteTrackCommand(trackId); + CommandManager.execute(deleteTrackCommand); +}; + +const isMinLengthOfTrackList = (): Boolean => { + const { trackList } = store.getState(); + const minLength = 3; + + return trackList.length === minLength; +}; + +export default { + executeUndoCommand, + executeRedoCommand, + executeDeleteCommand, + executeCutCommand, + executePasteCommand, + executeSplitCommand, + executeAddTrackCommand, + executeMoveCommand, + executeDeleteTrackCommand +} diff --git a/frontend/src/controllers/controller.ts b/frontend/src/controllers/controller.ts index 52089ede..70623167 100644 --- a/frontend/src/controllers/controller.ts +++ b/frontend/src/controllers/controller.ts @@ -1,20 +1,34 @@ -import { Source, Track, TrackSection } from '@model'; -import { store } from "@store"; -import { ModalType, FocusInfo, CursorType } from "@types"; -import CommandManager from '@command/CommandManager'; -import DeleteCommand from '@command/DeleteCommand' -import { CopyUtil } from '@util' -import { PlayBarUtil } from '@util'; - -interface SectionData { - sectionChannelData: number[]; - duration: number; -} +import { Source, Track, TrackSection, SectionDragStartData } from '@model'; +import { store } from '@store'; +import { ModalType, FocusInfo, CursorType, SectionDataType } from '@types'; +import { CopyUtil, SectionEffectListUtil, TimeUtil, WidthUtil } from '@util'; +import playbackTool from '@components/PlaybackTools/PlaybackToolClass'; -const getSectionChannelData = (trackId: number, trackSectionId: number): SectionData | undefined => { - const { trackList, sourceList } = store.getState(); +const getTrackSection = (trackId: number, trackSectionId: number): TrackSection | undefined => { + const { trackList } = store.getState(); const track = trackList.find((track) => track.id === trackId); + if (!track) return; + + const { trackSectionList } = track; + const trackSection = trackSectionList.find((trackSection) => trackSection.id === trackSectionId); + if (!trackSection) return; + return trackSection; +}; +const getSource = (trackId: number, trackSectionId: number): Source | undefined => { + const { sourceList } = store.getState(); + const trackSection = getTrackSection(trackId, trackSectionId); + + if (!trackSection) return; + const source = sourceList.find((source) => source.id === trackSection.sourceId); + if (!source) return; + + return source; +}; + +const getSourceAndTrackSection = (trackId: number, trackSectionId: number): { source: Source; trackSection: TrackSection } | undefined => { + const { trackList, sourceList } = store.getState(); + const track = trackList.find((track) => track.id === trackId); if (!track) return; const { trackSectionList } = track; @@ -22,22 +36,47 @@ const getSectionChannelData = (trackId: number, trackSectionId: number): Section if (!trackSection) return; const source = sourceList.find((source) => source.id === trackSection.sourceId); - if (!source) return; - const { parsedChannelData, duration } = source; - const { parsedChannelStartTime, parsedChannelEndTime } = trackSection; + return { source, trackSection }; +}; + +const getSectionData = (trackId: number, trackSectionId: number): SectionDataType | undefined => { + const getSourceAndTrackSection = (trackId: number, trackSectionId: number): { source: Source; trackSection: TrackSection } | undefined => { + const { trackList, sourceList } = store.getState(); + const track = trackList.find((track) => track.id === trackId); + if (!track) return; + + const { trackSectionList } = track; + const trackSection = trackSectionList.find((trackSection) => trackSection.id === trackSectionId); + if (!trackSection) return; + + const source = sourceList.find((source) => source.id === trackSection.sourceId); + if (!source) return; - const numOfPeakPerSecond = parsedChannelData.length / duration; + return { source, trackSection }; + }; + + const parseSectionData = ({ source, trackSection }: { source: Source; trackSection: TrackSection }): SectionDataType | undefined => { + if (!source || !trackSection) return; + + const { parsedChannelData, duration } = source; + const { channelStartTime, channelEndTime } = trackSection; + + const numOfPeakPerSecond = parsedChannelData.length / duration; - const sectionChannelStartTime = numOfPeakPerSecond * parsedChannelStartTime; - const sectionChannelEndTime = numOfPeakPerSecond * parsedChannelEndTime; - const sectionChannelData = parsedChannelData.slice(sectionChannelStartTime, sectionChannelEndTime); + const sectionChannelStartTime = numOfPeakPerSecond * channelStartTime; + const sectionChannelEndTime = numOfPeakPerSecond * channelEndTime; + const sectionChannelData = parsedChannelData.slice(sectionChannelStartTime, sectionChannelEndTime); - return { - sectionChannelData: sectionChannelData, - duration: parsedChannelEndTime - parsedChannelStartTime + return { + sectionChannelData: sectionChannelData, + duration: channelEndTime - channelStartTime + }; }; + + const pipe = (f, g) => (x, y) => g(f(x, y)); + return pipe(getSourceAndTrackSection, parseSectionData)(trackId, trackSectionId); }; const getSourceBySourceId = (sourceId: number): Source | undefined => { @@ -47,6 +86,12 @@ const getSourceBySourceId = (sourceId: number): Source | undefined => { return source; }; +const getSourceList = (): Source[] => { + const { sourceList } = store.getState(); + + return sourceList; +}; + const addSource = (source: Source): void => { store.setSource(source); }; @@ -55,8 +100,9 @@ const changeModalState = (modalType: ModalType, isHidden: Boolean): void => { store.setModalState(modalType, isHidden); }; -const changeCursorTime = (minute: string, second: string, milsecond: string): void => { - store.setCursorTime(minute, second, milsecond); +const changeCursorStringTime = (minute: number, second: number, milsecond: number): void => { + const newCursorStringTime = TimeUtil.getStringTime(minute, second, milsecond); + store.setCursorStringTime(newCursorStringTime); }; const changeTrackDragState = (isTrackDraggable: Boolean): void => { @@ -68,10 +114,36 @@ const getTrackList = (): Track[] => { return trackList; }; -const addTrack = (track: Track): void => { +const getTrack = (trackId: number): Track | null => { + const { trackList } = store.getState(); + const track = trackList.find((track) => track.id === trackId); + + if (!track) return null; + + return track; +}; + +const setTrack = (track: Track): void => { store.setTrack(track); }; +const createTrackSectionFromSource = (sourceId: number): TrackSection | undefined => { + const { sourceList } = store.getState(); + const source = sourceList.find((source) => source.id === sourceId); + + if (!source) return; + const newTrackSection = new TrackSection({ + id: 0, + sourceId, + trackId: 0, + channelStartTime: 0, + channelEndTime: source.duration, + trackStartTime: 0 + }); + + return newTrackSection; +} + const addTrackSection = (trackId: number, trackSection: TrackSection): void => { store.setTrackSection(trackId, trackSection); }; @@ -81,9 +153,9 @@ const changeCurrentPosition = (currentPosition: number): void => { }; const getCurrentPosition = (): number[] => { - const { currentPosition, totalCursorTime } = store.getState(); + const { currentPosition, cursorNumberTime } = store.getState(); - return [currentPosition, totalCursorTime]; + return [currentPosition, cursorNumberTime]; }; const getCtrlIsPressed = (): boolean => { @@ -100,9 +172,8 @@ const getFocusList = () => { return focusList; }; -const toggleFocus = (trackId: number, sectionId: number, selectedElement: HTMLElement): void => { +const toggleFocus = (trackId: number, sectionId: number, selectedElement: HTMLCanvasElement): void => { const { trackList, focusList, ctrlIsPressed, cursorMode } = store.getState(); - if (cursorMode !== CursorType.SELECT_MODE) return; const track = trackList.find((track) => track.id === trackId); @@ -112,7 +183,6 @@ const toggleFocus = (trackId: number, sectionId: number, selectedElement: HTMLEl const existFocus = focusList.find((info) => info.trackSection.id === sectionId); - if (ctrlIsPressed) { if (existFocus) { removeFocus(sectionId, selectedElement); @@ -125,13 +195,16 @@ const toggleFocus = (trackId: number, sectionId: number, selectedElement: HTMLEl } }; -const addFocus = (trackSection: TrackSection, selectedElement: HTMLElement): void => { +const addFocus = (trackSection: TrackSection, selectedElement: HTMLCanvasElement): void => { selectedElement.classList.add('focused-section'); + store.resetSelectTrackData(); + const newFocusInfo: FocusInfo = { trackSection: trackSection, element: selectedElement }; store.addFocus(newFocusInfo); + SectionEffectListUtil.showEffectList(); }; const removeFocus = (sectionId: number, selectedElement: HTMLElement): void => { @@ -139,12 +212,14 @@ const removeFocus = (sectionId: number, selectedElement: HTMLElement): void => { const index = focusList.findIndex((focus) => focus.trackSection.id === sectionId); selectedElement.classList.remove('focused-section'); store.removeFocus(index); + SectionEffectListUtil.hideEffectList(); }; const resetFocus = (): void => { const { focusList } = store.getState(); focusList.forEach((focus) => focus.element.classList.remove('focused-section')); store.resetFocus(); + SectionEffectListUtil.hideEffectList(); }; const getCursorMode = (): CursorType => { @@ -152,18 +227,19 @@ const getCursorMode = (): CursorType => { return cursorMode; }; -const setCursorMode = (newType: CursorType) => { - const trackContainer = document.querySelector('.audi-main-audio-track-container'); +const setCursorMode = (newCursorType: CursorType) => { + const trackContainerElement = document.querySelector('.audi-main-audio-track-container'); - if (!trackContainer) return; + if (!trackContainerElement) return; - if (newType === CursorType.SELECT_MODE) { - trackContainer.classList.remove('cursor-change'); - } else if (newType === CursorType.CUT_MODE) { + if (newCursorType === CursorType.SELECT_MODE) { + trackContainerElement.classList.remove('cursor-change'); + } else if (newCursorType === CursorType.CUT_MODE) { resetFocus(); - trackContainer.classList.add('cursor-change'); + store.resetSelectTrackData(); + trackContainerElement.classList.add('cursor-change'); } - store.setCursorMode(newType); + store.setCursorMode(newCursorType); }; const getClipBoard = (): TrackSection | null => { @@ -171,24 +247,27 @@ const getClipBoard = (): TrackSection | null => { return clipBoard; }; -const pauseChangeMarkerTime = (playingTime: number): void => { - const { markerTime } = store.getState(); - const newMarkerTime = markerTime + playingTime; +const pauseChangeMarkerNumberTime = (playingTime: number): void => { + const { markerNumberTime } = store.getState(); + let newMarkerNumberTime = markerNumberTime + playingTime; + if (newMarkerNumberTime < 0) { + newMarkerNumberTime = 0; + } - store.setMarkerTime(newMarkerTime); + store.setMarkerNumberTime(newMarkerNumberTime); }; -const cursorChangeMarkerTime = (newMarkerTime): void => { - store.setMarkerTime(newMarkerTime); -}; +const changeMarkerNumberTime = (markerTime: number) => { + store.setMarkerNumberTime(markerTime); +} const getMarkerTime = (): number => { - const { markerTime } = store.getState(); - return markerTime; + const { markerNumberTime } = store.getState(); + return markerNumberTime; }; -const changeTotalCursorTime = (totalCursorTime: number): void => { - store.setTotalCursorTime(totalCursorTime); +const changeCursorNumberTime = (cursorNumberTime: number): void => { + store.setCursorNumberTime(cursorNumberTime); }; const changeIsPauseState = (isPauseState: boolean): void => { @@ -200,82 +279,247 @@ const getIsPauseState = (): boolean => { return isPause; }; -const setMarkerWidth = (markerWidth: number): void => { +const setMarkerWidth = (markerWidth: number|number[]): void => { store.setMarkerWidth(markerWidth); }; -const changePlayTime = (passedTime: number): void => { - const { playTime } = store.getState(); +const changePlayStringTime = (passedTime: number): void => { + const { playStringTime } = store.getState(); - const [minute, second, milsecond] = playTime.split(':'); + const [minute, second, milsecond] = playStringTime.split(':'); let newMinute = Number(minute); let newSecond = Number(second); let newMilsecond = Number(milsecond) + Math.floor(passedTime); - if (newMilsecond >= 1000) { - newMilsecond -= 1000; - newSecond += 1; - } + if (newMilsecond > 0) { + if (newMilsecond >= 1000) { + newMilsecond -= 1000; + newSecond += 1; + } - if (newSecond >= 60) { - newSecond -= 60; - newMinute += 1; + if (newSecond >= 60) { + newSecond -= 60; + newMinute += 1; + } + } else { + let totalMilsecond = newMinute * 1000 * 60 + newSecond * 1000 + newMilsecond; + if (totalMilsecond < 0) { + newMinute = 0; + newSecond = 0; + newMilsecond = 0; + } else { + newMinute = Math.floor(totalMilsecond / (1000 * 60)); + totalMilsecond -= newMinute * 1000 * 60; + newSecond = Math.floor(totalMilsecond / 1000); + totalMilsecond -= newSecond * 1000; + newMilsecond = totalMilsecond; + } } - const newPlayTime = `${newMinute.toString().padStart(2, '0')}:${newSecond.toString().padStart(2, '0')}:${newMilsecond.toString().padStart(3, '0')}`; + const newPlayStringTime = TimeUtil.getStringTime(newMinute, newSecond, newMilsecond); - store.setPlayTime(newPlayTime); + store.setPlayStringTime(newPlayStringTime); }; -const resetPlayTime = (cursorTime: number): void => { - const [minute, second, milsecond] = PlayBarUtil.setTime(cursorTime); +const changeMarkerPlayStringTime = (cursorNumberTime: number): void => { + const [minute, second, milsecond] = TimeUtil.splitTime(cursorNumberTime); - const newPlayTime = `${minute.toString().padStart(2, '0')}:${second.toString().padStart(2, '0')}:${milsecond.toString().padStart(3, '0')}`; + const newPlayStringTime = TimeUtil.getStringTime(minute, second, milsecond); - store.setPlayTime(newPlayTime); + store.setPlayStringTime(newPlayStringTime); }; const removeSection = (trackId: number, sectionIndex: number) => { store.removeSection(trackId, sectionIndex); }; -const deleteCommand = () => { +const setClipBoard = (): boolean => { const { focusList } = store.getState(); - if (focusList.length === 0) return; - const command = new DeleteCommand(); - CommandManager.execute(command); + + if (focusList.length !== 1) return false; + + const newSection: TrackSection = CopyUtil.copySection(focusList[0].trackSection); + newSection.id = 0; + store.setClipBoard(newSection); + + return true; }; -const undoCommand = () => { - if (CommandManager.undoList.length === 0) return; - CommandManager.undo(); +const changeMaxTrackWidth = (maxTrackWidth: number): void => { + store.setMaxTrackWidth(maxTrackWidth); }; -const redoCommand = () => { - if (CommandManager.redoList.length === 0) return; - CommandManager.redo(); +const getMaxTrackWidth = () => { + const { maxTrackWidth } = store.getState(); + return maxTrackWidth; }; - -const setClipBoard = () => { - const { focusList } = store.getState(); - if (focusList.length !== 1) return; +const changeMaxTrackPlayTime = (trackSectionList: TrackSection[]): void => { + const trackPlaytime = trackSectionList.reduce((acc, trackSection) => acc += trackSection.length, 0); + store.setMaxTrackPlayTime(trackPlaytime); +}; - const newSection: TrackSection = CopyUtil.copySection(focusList[0].trackSection); +const getMaxTrackPlayTime = () => { + const { maxTrackPlayTime } = store.getState(); + return maxTrackPlayTime; +}; - store.setClipBoard(newSection); +const changeCurrentScrollAmount = (newCurrentScrollAmount: number): void => { + store.setCurrentScrollAmount(newCurrentScrollAmount); +}; + +const getCurrentScrollAmount = (): number => { + const { currentScrollAmount } = store.getState(); + + return currentScrollAmount; +} + +const getCurrentScrollTime = (): number => { + const { currentScrollAmount, maxTrackWidth, maxTrackPlayTime } = store.getState(); + + const secondPerPixel = WidthUtil.getSecondPerPixel(maxTrackWidth, maxTrackPlayTime); + + return secondPerPixel * currentScrollAmount; +} + +const audioCursorPlay = () => { + playbackTool.audioCursorPlay(); +}; + +const audioPlayOrPause = (): void => { + const audioPlayType = playbackTool.audioPlayOrPause(); + store.changePlayOrPauseIcon(audioPlayType); +}; + +const audioStop = (): void => { + playbackTool.audioStop(); +}; + +const audioRepeat = (): void => { + playbackTool.audioRepeat(); +}; + +const changeIsRepeatState = (isRepeatState: boolean): void => { + store.setIsRepeatState(isRepeatState); +}; + +const getIsRepeatState = (): boolean => { + const { isRepeat } = store.getState(); + return isRepeat; +}; + +const audioFastRewind = () => { + playbackTool.audioFastRewind(); +}; + +const audioFastForward = () => { + playbackTool.audioFastForward(); +}; + +const audioSkipPrev = () => { + playbackTool.audioSkipPrev(); +}; + +const audioSkipNext = () => { + playbackTool.audioSkipNext(); +}; + +const setMute = (trackId: number) => { + playbackTool.setMute(trackId); +}; + +const unsetMute = (trackId: number) => { + playbackTool.unsetMute(trackId); +}; + +const setSolo = (trackId: number) => { + playbackTool.setSolo(trackId); +}; + +const unsetSolo = (trackId: number) => { + playbackTool.unsetSolo(trackId); +}; + +const changeSectionDragStartData = (sectionDragStartData: SectionDragStartData): void => { + store.setSectionDragStartData(sectionDragStartData); +}; + +const getSectionDragStartData = (): SectionDragStartData | null => { + const { sectionDragStartData } = store.getState(); + + return sectionDragStartData; +}; + +const changeSelectTrackData = (trackId: number, selectedTime: number): void => { + const track = getTrack(trackId); + const trackSection = track?.trackSectionList.find(section => section.trackStartTime <= selectedTime && selectedTime <= section.trackStartTime + section.length) + if (trackSection) { + store.setSelectTrackData(0, 0); + return; + } + store.setSelectTrackData(trackId, selectedTime); +}; + +const getSelectTrackData = () => { + const { selectTrackData } = store.getState(); + return selectTrackData; +}; + +const pushTrackWidthIndex = (newTrack: Track): void => { + const { trackList, trackIndex } = store.getState(); + const newTrackList = trackList.concat(newTrack); + + store.setTrackList(newTrackList); + store.setTrackIndex(trackIndex + 1); +}; + +const popTrackWithIndex = (): Track | undefined => { + const { trackList, trackIndex } = store.getState(); + const removedTrack = trackList.pop(); + + store.setTrackList(trackList); + store.setTrackIndex(trackIndex - 1); + return removedTrack; +}; + +const removeTrackById = (trackId: number): Track | undefined => { + const { trackList } = store.getState(); + const trackToRemove = trackList.find((track) => track.id === trackId); + + if(!trackToRemove) return; + const newTrackList = trackList.filter((track) => track.id !== trackId); + + store.setTrackList(newTrackList); + return trackToRemove; +}; + +const insertTrack = (insertIdx: number, trackToInsert: Track): void => { + const { trackList } = store.getState(); + + const newTrackList = Array(trackList.length + 1).fill(0).map(( _, idx) => { + if(idx < insertIdx) return trackList[idx]; + if(idx > insertIdx) return trackList[idx - 1]; + return trackToInsert; + }); + + store.setTrackList(newTrackList); }; export default { - getSourceBySourceId, - getSectionChannelData, + getTrackSection, + getSource, + getSourceAndTrackSection, + createTrackSectionFromSource, + getSectionData, + getSourceList, addSource, changeModalState, changeTrackDragState, getTrackList, - addTrack, + getTrack, + setTrack, addTrackSection, - changeCursorTime, + changeCursorStringTime, changeCurrentPosition, getCurrentPosition, getCtrlIsPressed, @@ -289,17 +533,44 @@ export default { setCursorMode, getClipBoard, setClipBoard, - pauseChangeMarkerTime, + pauseChangeMarkerNumberTime, getMarkerTime, - changeTotalCursorTime, - cursorChangeMarkerTime, + changeCursorNumberTime, setMarkerWidth, getIsPauseState, changeIsPauseState, - changePlayTime, - resetPlayTime, + changePlayStringTime, + changeMarkerPlayStringTime, removeSection, - deleteCommand, - undoCommand, - redoCommand + getMaxTrackPlayTime, + getSourceBySourceId, + changeMarkerNumberTime, + audioPlayOrPause, + audioStop, + audioFastRewind, + audioFastForward, + audioSkipPrev, + audioSkipNext, + audioCursorPlay, + audioRepeat, + changeIsRepeatState, + getIsRepeatState, + setMute, + unsetMute, + setSolo, + unsetSolo, + changeMaxTrackWidth, + changeMaxTrackPlayTime, + changeCurrentScrollAmount, + getMaxTrackWidth, + getCurrentScrollTime, + getCurrentScrollAmount, + changeSectionDragStartData, + getSectionDragStartData, + changeSelectTrackData, + getSelectTrackData, + popTrackWithIndex, + pushTrackWidthIndex, + removeTrackById, + insertTrack }; diff --git a/frontend/src/controllers/index.js b/frontend/src/controllers/index.js index 8cbbeb9e..5b5e9eae 100644 --- a/frontend/src/controllers/index.js +++ b/frontend/src/controllers/index.js @@ -1 +1,2 @@ export {default as Controller} from "./controller"; +export {default as CommandController} from './CommandController'; diff --git a/frontend/src/model/SectionDragStartData.ts b/frontend/src/model/SectionDragStartData.ts new file mode 100644 index 00000000..575d8d1d --- /dev/null +++ b/frontend/src/model/SectionDragStartData.ts @@ -0,0 +1,19 @@ +import { TrackSection } from "@model" + +class SectionDragStartData { + public trackSection: TrackSection; + public prevCursorTime: number | null; + public offsetLeft: number; + + constructor({ + trackSection, + prevCursorTime = null, + offsetLeft, + }) { + this.trackSection = trackSection;; + this.prevCursorTime = prevCursorTime; + this.offsetLeft = offsetLeft; + } +} + +export default SectionDragStartData; diff --git a/frontend/src/model/SelectTrackData.ts b/frontend/src/model/SelectTrackData.ts new file mode 100644 index 00000000..2eb38935 --- /dev/null +++ b/frontend/src/model/SelectTrackData.ts @@ -0,0 +1,14 @@ +class SelectTrackData { + public trackId: number; + public selectedTime: number; + + constructor({ + trackId, + selectedTime, + }) { + this.trackId = trackId; + this.selectedTime = selectedTime; + } +} + +export default SelectTrackData; diff --git a/frontend/src/model/TrackSection.ts b/frontend/src/model/TrackSection.ts index bfeca2bf..b8f5b112 100644 --- a/frontend/src/model/TrackSection.ts +++ b/frontend/src/model/TrackSection.ts @@ -1,38 +1,32 @@ -class TrackSection{ +import { EffectList } from '@types'; + +class TrackSection { public id: number; public trackId: number; public sourceId: number; - public channelStartTime : number; - public channelEndTime : number; - public parsedChannelStartTime : number; - public parsedChannelEndTime : number; - public trackStartTime : number; + public channelStartTime: number; + public channelEndTime: number; + public trackStartTime: number; public length: number; public effectList: object[]; - public audioStartTime: number; constructor({ - id=0, + id, trackId, - sourceId, - channelStartTime, - channelEndTime, - parsedChannelStartTime, - parsedChannelEndTime, + sourceId, + channelStartTime, + channelEndTime, trackStartTime, - audioStartTime - }){ + effectList = [] + }) { this.id = id; this.trackId = trackId; this.sourceId = sourceId; this.channelStartTime = channelStartTime; this.channelEndTime = channelEndTime; - this.parsedChannelStartTime = parsedChannelStartTime; - this.parsedChannelEndTime = parsedChannelEndTime; - this.trackStartTime = trackStartTime; + this.trackStartTime = trackStartTime; this.length = this.channelEndTime - this.channelStartTime; - this.effectList = []; - this.audioStartTime = 0; + this.effectList = effectList; } } diff --git a/frontend/src/model/index.js b/frontend/src/model/index.js index 450f94e3..7ec56073 100644 --- a/frontend/src/model/index.js +++ b/frontend/src/model/index.js @@ -1,4 +1,6 @@ -export {default as Source} from './Source'; -export {default as Track} from './Track'; -export {default as TrackSection} from './TrackSection' -export {default as AudioSourceInfoInTrack} from './AudioSourceInfoInTrack'; \ No newline at end of file +export { default as Source } from './Source'; +export { default as Track } from './Track'; +export { default as TrackSection } from './TrackSection' +export { default as AudioSourceInfoInTrack } from './AudioSourceInfoInTrack'; +export { default as SectionDragStartData } from './SectionDragStartData'; +export { default as SelectTrackData } from './SelectTrackData'; diff --git a/frontend/src/store/store.ts b/frontend/src/store/store.ts index dccf1372..f204ccc1 100644 --- a/frontend/src/store/store.ts +++ b/frontend/src/store/store.ts @@ -1,5 +1,5 @@ import { StoreStateType, CursorType } from '@types'; -import { Track, Source, TrackSection } from '@model'; +import { Track, Source, TrackSection, SectionDragStartData, SelectTrackData } from '@model'; import { StoreChannelType, ModalType, ModalStateType, FocusInfo } from '@types'; import { storeChannel } from '@store'; @@ -8,8 +8,8 @@ const store = new (class Store { constructor() { this.state = { - cursorTime: '00:00:000', - playTime: '00:00:000', + cursorStringTime: '00:00:000', + playStringTime: '00:00:000', sourceList: [], modalState: { modalType: ModalType.upload, @@ -20,14 +20,20 @@ const store = new (class Store { focusList: [], ctrlIsPressed: false, cursorMode: CursorType.SELECT_MODE, - trackIndex: 3, - sectionIndex: 0, + trackIndex: 4, + sectionIndex: 1, clipBoard: null, audioSourceInfoInTrackList: [], currentPosition: 0, - markerTime: 0, - totalCursorTime: 0, - isPause: true + markerNumberTime: 0, + cursorNumberTime: 0, + isPause: true, + isRepeat: false, + maxTrackWidth: 0, + maxTrackPlayTime: 300, + currentScrollAmount: 0, + sectionDragStartData: null, + selectTrackData: new SelectTrackData({ trackId: 0, selectedTime: 0 }), }; } @@ -35,7 +41,7 @@ const store = new (class Store { return Array(numOfTracks) .fill(0) .reduce((acc, cur, idx) => { - const track = new Track({ id: idx, trackSectionList: [] }); + const track = new Track({ id: idx + 1, trackSectionList: [] }); return acc.concat(track); }, []); } @@ -44,6 +50,19 @@ const store = new (class Store { return this.state; } + resetSelectTrackData(): void { + const { selectTrackData } = this.state; + selectTrackData.trackId = 0; + selectTrackData.selectedTime = 0; + + storeChannel.publish(StoreChannelType.SELECT_AUDIO_TRACK, selectTrackData); + } + + setSelectTrackData(trackId: number, selectedTime: number): void { + this.state = { ...this.state, selectTrackData: { trackId, selectedTime } }; + storeChannel.publish(StoreChannelType.SELECT_AUDIO_TRACK, this.state.selectTrackData); + } + setSource(source: Source): void { const { sourceList } = this.state; source.id = sourceList.length; @@ -64,15 +83,13 @@ const store = new (class Store { storeChannel.publish(StoreChannelType.MODAL_STATE_CHANNEL, newModalState); } - setCursorTime(newMinute: string, newSecond: string, newMilsecond): void { - const { cursorTime } = this.state; - const [minute, second, milsecond] = cursorTime.split(':'); + setCursorStringTime(newCursorStringTime: string): void { + const { cursorStringTime } = this.state; - if (minute === newMinute && second === newSecond && milsecond === newMilsecond) return; + if (cursorStringTime === newCursorStringTime) return; - const newCursorTime: string = `${newMinute.padStart(2, '0')}:${newSecond.padStart(2, '0')}:${newMilsecond.padStart(3, '0')}`; - this.state = { ...this.state, cursorTime: newCursorTime }; - storeChannel.publish(StoreChannelType.CURSOR_TIME_CHANNEL, newCursorTime); + this.state = { ...this.state, cursorStringTime: newCursorStringTime }; + storeChannel.publish(StoreChannelType.CURSOR_TIME_CHANNEL, newCursorStringTime); } setTrackDragState(newIsTrackDraggable: Boolean): void { @@ -86,10 +103,15 @@ const store = new (class Store { setTrack(newTrack: Track): void { const { trackList } = this.state; - newTrack.id = trackList.length; - const newAudioTrackList = trackList.concat(newTrack); + const track = trackList.find((track) => track.id === newTrack.id); + if (track) { + track.trackSectionList = [...newTrack.trackSectionList]; + } else { + newTrack.id = this.state.trackIndex++; + const newAudioTrackList = trackList.concat(newTrack); - this.state = { ...this.state, trackList: newAudioTrackList }; + this.state = { ...this.state, trackList: newAudioTrackList }; + } } setTrackSection(trackId: number, newTrackSection: TrackSection): void { @@ -98,7 +120,9 @@ const store = new (class Store { if (!track) return; const { trackSectionList } = track; - newTrackSection.id = this.state.sectionIndex++; + if (newTrackSection.id === 0) { + newTrackSection.id = this.state.sectionIndex++; + } const newTrackSectionList = trackSectionList.concat(newTrackSection).sort((a, b) => a.trackStartTime - b.trackStartTime); @@ -114,8 +138,8 @@ const store = new (class Store { trackId: trackId, trackSectionList: newTrackSectionList }); - storeChannel.publish(StoreChannelType.TRACK_CHANNEL, newTrackList); + storeChannel.publish(StoreChannelType.EDIT_MENU_CHANNEL, null); } setCurrentPosition(newCurrentPosition: number): void { @@ -135,7 +159,7 @@ const store = new (class Store { const newfocusList = focusList.concat(newFocusInfo); this.state = { ...this.state, focusList: newfocusList }; - storeChannel.publish(StoreChannelType.EDIT_TOOLS_CHANNEL, ''); + storeChannel.publish(StoreChannelType.EDIT_TOOLS_CHANNEL, null); } removeFocus(removeIndex: number) { @@ -143,48 +167,50 @@ const store = new (class Store { const newfocusList = [...focusList]; newfocusList.splice(removeIndex, 1); this.state = { ...this.state, focusList: newfocusList }; - storeChannel.publish(StoreChannelType.EDIT_TOOLS_CHANNEL, ''); + storeChannel.publish(StoreChannelType.EDIT_TOOLS_CHANNEL, null); } resetFocus(): void { this.state = { ...this.state, focusList: [] }; - storeChannel.publish(StoreChannelType.EDIT_TOOLS_CHANNEL, ''); + storeChannel.publish(StoreChannelType.EDIT_TOOLS_CHANNEL, null); } setClipBoard(newSection: TrackSection): void { this.state.clipBoard = newSection; - storeChannel.publish(StoreChannelType.EDIT_TOOLS_CHANNEL, ''); + storeChannel.publish(StoreChannelType.EDIT_TOOLS_CHANNEL, null); } - setMarkerTime(newMarkerTime: number): void { - const { markerTime } = this.state; + setMarkerNumberTime(newMarkerNumberTime: number): void { + const { markerNumberTime } = this.state; - if (markerTime === newMarkerTime) return; + if (markerNumberTime === newMarkerNumberTime) { + return; + }; - this.state = { ...this.state, markerTime: newMarkerTime }; + this.state = { ...this.state, markerNumberTime: newMarkerNumberTime }; } - setTotalCursorTime(newTotalCursorTime: number): void { - this.state = { ...this.state, totalCursorTime: newTotalCursorTime }; + setCursorNumberTime(newCursorNumberTime: number): void { + this.state = { ...this.state, cursorNumberTime: newCursorNumberTime }; } setIsPauseState(isPauseState: boolean): void { this.state = { ...this.state, isPause: isPauseState }; } - setMarkerWidth(newMarkerWidth: number): void { + setMarkerWidth(newMarkerWidth: number|number[]): void { storeChannel.publish(StoreChannelType.CURRENT_POSITION_CHANNEL, newMarkerWidth); } - setPlayTime(newPlayTime): void { - this.state = { ...this.state, playTime: newPlayTime }; - storeChannel.publish(StoreChannelType.PLAY_TIME_CHANNEL, newPlayTime); + setPlayStringTime(newPlayStringTime): void { + this.state = { ...this.state, playStringTime: newPlayStringTime }; + storeChannel.publish(StoreChannelType.PLAY_TIME_CHANNEL, newPlayStringTime); } removeSection(trackId: number, sectionIndex: number): void { const { trackList } = this.state; - const track = trackList.find(track => track.id === trackId) + const track = trackList.find((track) => track.id === trackId); if (!track) return; const { trackSectionList } = track; @@ -194,23 +220,72 @@ const store = new (class Store { newTrackSectionList.splice(sectionIndex, 1); const newTrack = new Track({ ...track, trackSectionList: newTrackSectionList }); - const newTrackList: Array = trackList.reduce((acc, track) => - (track.id === trackId) ? acc.concat(newTrack) : acc.concat(track), - []); + const newTrackList: Array = trackList.reduce( + (acc, track) => (track.id === trackId ? acc.concat(newTrack) : acc.concat(track)), + [] + ); this.state = { ...this.state, trackList: newTrackList }; storeChannel.publish(StoreChannelType.TRACK_SECTION_LIST_CHANNEL, { - trackId: trackId, - trackSectionList: newTrackSectionList + trackId: trackId, + trackSectionList: newTrackSectionList }); storeChannel.publish(StoreChannelType.TRACK_CHANNEL, newTrackList); + storeChannel.publish(StoreChannelType.EDIT_MENU_CHANNEL, null); + } + + setCursorMode(newCursorMode: CursorType): void { + const { cursorMode } = this.state; + if (cursorMode === newCursorMode) return; + + this.state = { ...this.state, cursorMode: newCursorMode }; + storeChannel.publish(StoreChannelType.CURSOR_MODE_CHANNEL, newCursorMode); + } + + setMaxTrackWidth(newMaxTrackWidth: number): void { + const { maxTrackWidth } = this.state; + if (maxTrackWidth >= newMaxTrackWidth) return; + + this.state = { ...this.state, maxTrackWidth: newMaxTrackWidth }; + storeChannel.publish(StoreChannelType.MAX_TRACK_WIDTH_CHANNEL, newMaxTrackWidth); + } + + changePlayOrPauseIcon(iconType: number) { + storeChannel.publish(StoreChannelType.PLAY_OR_PAUSE_CHANNEL, iconType); + } + + setIsRepeatState(isRepeat: boolean) { + this.state = { ...this.state, isRepeat: isRepeat }; + } + + setMaxTrackPlayTime(newMaxTrackPlayTime: number): void { + const { maxTrackPlayTime } = this.state; + if (maxTrackPlayTime >= newMaxTrackPlayTime) return; + + this.state = { ...this.state, maxTrackPlayTime: newMaxTrackPlayTime }; + storeChannel.publish(StoreChannelType.MAX_TRACK_PLAY_TIME_CHANNEL, newMaxTrackPlayTime); + } + + setCurrentScrollAmount(newCurrentScrollAmount: number): void { + const { currentScrollAmount } = this.state; + if (currentScrollAmount === newCurrentScrollAmount) return; + + this.state = { ...this.state, currentScrollAmount: newCurrentScrollAmount }; + storeChannel.publish(StoreChannelType.CURRENT_SCROLL_AMOUNT_CHANNEL, newCurrentScrollAmount); + } + + setSectionDragStartData(newDragStartData: SectionDragStartData): void { + this.state = { ...this.state, sectionDragStartData: newDragStartData }; + } + + setTrackList(newTrackList: Track[]): void { + this.state = {...this.state, trackList: newTrackList}; } - setCursorMode(newType: CursorType): void { - this.state.cursorMode = newType; - storeChannel.publish(StoreChannelType.EDIT_TOOLS_CHANNEL, ''); + setTrackIndex(newTrackIndex: number): void { + this.state = {...this.state, trackIndex: newTrackIndex}; } })(); diff --git a/frontend/src/store/storeChannel.ts b/frontend/src/store/storeChannel.ts index 3ac33ba4..766151af 100644 --- a/frontend/src/store/storeChannel.ts +++ b/frontend/src/store/storeChannel.ts @@ -9,7 +9,7 @@ class StoreChannel{ this.observers = new Map(); } - publish(channel: StoreChannelType, data: any): void { + publish(channel: StoreChannelType, data: any): void { this.channels.set(channel, data); this.notify(channel); } @@ -17,8 +17,6 @@ class StoreChannel{ subscribe(channel: StoreChannelType, callback: Function, bindObj: Object): void { let observerDatas: StoreObserverData[] | undefined = this.observers.get(channel); - // if(!observerDatas) observerDatas = [{callback, bindObj}]; - if(!observerDatas) { observerDatas = [{callback, bindObj}] this.observers.set(channel, observerDatas); diff --git a/frontend/webpack.config.common.js b/frontend/webpack.config.common.js index 0cf77589..94864776 100644 --- a/frontend/webpack.config.common.js +++ b/frontend/webpack.config.common.js @@ -1,28 +1,9 @@ const path = require('path'); -const fs = require('fs'); -const dotenv = require('dotenv'); -const webpack = require('webpack'); const HtmlWebpackPlugin = require('html-webpack-plugin'); const { CleanWebpackPlugin } = require('clean-webpack-plugin'); const MiniCssExtractPlugin = require('mini-css-extract-plugin'); -const parseEnvKeys = (env) => { - const currentPath = path.join(__dirname); - const basePath = `${currentPath}/env/.env`; - const envPath = `${basePath}.${env.ENVIRONMENT}`; - const finalPath = fs.existsSync(envPath) ? envPath : basePath; - const fileEnv = dotenv.config({ - path: finalPath - }).parsed; - - return Object.keys(fileEnv).reduce((keys, key) => { - keys[`process.env.${key}`] = JSON.stringify(fileEnv[key]); - return keys; - }, {}); -}; - module.exports = (env) => { - const envKeys = parseEnvKeys(env); return { resolve: { @@ -73,13 +54,12 @@ module.exports = (env) => { hash: true }), new CleanWebpackPlugin(), - new webpack.DefinePlugin(envKeys), new MiniCssExtractPlugin({ filename: '[name].css' }) ], output: { - path: path.join(__dirname, '../backend/public'), + path: path.join(__dirname, './dist'), publicPath: '/', filename: '[name].js' }