에루샤
erusya
Back-end Developer
Web Geek
Anime Otaku
에루샤 프로필 이미지
개발

코드 하이라이터(highlight.js) 적용 및 플로라 에디터 연동

174 views as of November 5, 2024.
개발 관련 글을 쓰다보면 코드 하이라이팅이 필요한 경우 많다.

그냥 하이라이팅을 무시하고 텍스트로 적을 수 있긴하지만 그러면 코드의 가독성이 나빠진다.

그래서 기존엔 컬러 스크립터 라는 온라인 에디터를 통해서 변환한다음에 HTML 아웃 풋을 에디터에 복사하는 방식으로 코드 하이라이팅을 진행했다.

Color Scripter
Color Scripter
Simple & Flexible Syntax HighLighter
https://colorscripter.com

본문 이미지컬러 스크립터 출력물은 얼추 이런느낌

근데 이런방식은 결국 웹사이트를 2개 틀어놓고 매번 복붙 과정을 해야하므로 이런 과정을 에디터 내부에서 한방에 실행할 수 있는 기능이 없을까? 해서 리서치를 좀 해보았다.


코드 하이라이터 highlight.js

물론 좋은 에디터는 이런 코드뷰를 자동으로 제공하지만 대부분의 에디터는 그부분을 사용자 커스텀으로 남겨두고있다.

위지윅 에디터와 코드 하이라이팅은 아예 다른기술이고, 좋은 하이라이터는 결국 기업단위다보니 쉽게 자체적 마이그레이션을 지원을 하지않는건가 싶다.

그와중에 고성능 코드 하아라이터를 찾아보니 하이라이트JS 라는 라이브러리를 찾게되었다.

highlight.js
highlight.js
The Internet's favorite JavaScript syntax highlighter supporting Node.js and the web.
https://highlightjs.org

 생각보다 많은 유저가 사용하고 있었고 대응하는 언어도 진짜 여러가지였다.

무엇보다 실제적인 표출이 단순 코드 테이블에서 뭔가 내 블로그와 어울리는 리턴을 주는거같아서 과감하게 이걸 적용해보기 마음을 먹었다.


highlight.js 데모

적용하는 과정은 생각보다 간단했다.

<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.4.0/styles/atom-one-dark.min.css">
<script src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.4.0/highlight.min.js"></script>Copy

CSS와 JS는 이 두줄만 입력하면 끝이다.
여기서 자기가 원하는 테마가 있다면 css 주소 마지막에 atom-one-dark 이부분을 highlight.js가 제공하는 테마명으로 바꿔주면 된다.

일단 내 사이트에 라이브러리를 한번 띄우고 테스트하고 싶어서 아래와 같이 스타일, 본문, 스크립트를 구현했다.

1. 스타일

<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.4.0/styles/atom-one-dark.min.css">
<style>
    body {
        font-family: Arial, sans-serif;
        padding: 20px;
    }
    textarea {
        width: 100%;
        height: 200px;
        font-family: monospace;
        background-color: #1e1e1e;
        color: #dcdcdc;
        border: 1px solid #333;
        padding: 10px;
        margin-bottom: 20px;
        resize: vertical;
    }
    .highlighted-code {
        background: #282c34; /* Atom One Dark 테마와 일치하는 배경색 */
        padding: 20px;
        margin-top: 20px;
        border-radius: 5px;
        overflow-x: auto;
    }
</style>Copy

2. 본문

<div class="row">
    <div class="col-6">
        <textarea id="codeInput" placeholder="Enter your code here..."></textarea>
    </div>
    <div class="col-6">
        <div class="highlighted-code">
            <button class="copy-button" onclick="copyToClipboard()">Copy</button>
            <pre><code id="highlightedCode" class="hljs"></code></pre>
        </div>
    </div>
</div>Copy


3. 스크립트

<script src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.4.0/highlight.min.js"></script>
<script>
    const codeInput = document.getElementById('codeInput');
    const highlightedCode = document.getElementById('highlightedCode');

    codeInput.addEventListener('input', () => {
        // 코드 입력값을 텍스트로 설정
        const code = codeInput.value;
        highlightedCode.textContent = code;

        // highlightAuto로 자동 언어 감지 및 하이라이팅 적용
        const result = hljs.highlightAuto(code);
        highlightedCode.innerHTML = result.value;
    });

    function copyToClipboard() {
        navigator.clipboard.writeText(codeInput.value)
            .then(() => alert("Copied to clipboard!"))
            .catch(err => alert("Failed to copy: ", err));
    }
</script>Copy

4. 결과

본문 이미지좌측은 코드 입력단, 우측은 하이라이터된 코드 출력

일단 이렇게 하이라이터를 적용할 수 있다는 걸 인지하고 다음으로 넘어가서 플로라 에디터와 연동작업을 진행했다.


플로라 에디터와 연동 작업

플로라 에디터 연동은 별도의 커스텀 버튼을 만들어서 에디터 툴바에 등록하고 클릭 시 코드 입력 모달을 띄우는 과정까지만 작동되도록 구현했다.

// 'highlightCode' 커맨드 정의
FroalaEditor.DefineIcon('highlightCode', { NAME: 'code', SVG_KEY: 'codeView' });
FroalaEditor.RegisterCommand('highlightCode', {
    title: 'Insert Highlighted Code',
    focus: true,
    undo: true,
    refreshAfterCallback: true,
    callback: function () {
        editorInstance = this; // 현재 에디터 인스턴스 저장
        this.selection.save(); // 커서 위치 저장
        $('#highlightModal').modal('show');

        // 모달이 표시된 후 #codeInput에 포커스 설정
        $('#highlightModal').on('shown.bs.modal', function () {
            $('#codeInput').focus();
        });
    }
});Copy

우선 커스텀 버튼 hilightCode 를 플로라 객체를 이용해 선언하고,

new FroalaEditor(id, {

    ...

    toolbarButtons: {
        'moreText': {
            'buttons': ['strikeThrough', 'fontSize', 'bold', 'fontFamily', 'textColor', 'underline', 'clearFormatting'],
            'buttonsVisible': 1
        },
        'moreParagraph': {
            'buttons': ['alignLeft', 'alignCenter', 'paragraphFormat', 'alignRight', 'alignJustify', 'formatOL', 'formatUL'],
            'buttonsVisible': 3
        },
        'moreRich': {
            'buttons': ['insertImage', 'highlightCode', 'insertLink', 'insertTable', 'insertCode', 'insertHR'],
            'buttonsVisible': 3
        },
        'moreMisc': {
            'buttons': ['html'],
            'buttonsVisible': 1,
            'align': 'right'
        }
    },

    ....Copy

이런식으로 툴바에 highlightCode 버튼을 등록한다.

본문 이미지

그러면 위의 이미지처럼 지정한 툴바에 코드 버튼이 표시된다.

이 코드버튼이 활성화 시킬 모달도 마찬가지로 HTML 문서에 정의해준다.

<div class="modal fade" id="highlightModal" tabindex="-1" aria-labelledby="highlightModalLabel" aria-hidden="true">
    <div class="modal-dialog modal-dialog-centered modal-lg">
        <div class="modal-content" style="background-color: transparent; box-shadow: none;">
            <textarea id="codeInput" class="form-control" placeholder="Enter your code here..."></textarea>
            <button type="button" class="btn" onclick="insertHighlightedCode()">Insert</button>
        </div>
    </div>
</div>Copy

그러면 저 툴바의 버튼을 클릭하면 아래와 같이 입력 모달이 뜨게된다.

본문 이미지


이제 마지막으로 모달에 정보를 입력하고 Insert 버튼을 클릭했을때 하이라이터를 통해 코드를 변환하고 에디터 커서위치에 내용을 넣어주는 마지막 자바스크립트 함수를 작성한다.

// 코드 하이라이팅 및 에디터에 삽입
function insertHighlightedCode() {
    const codeInput = document.getElementById('codeInput');
    if(codeInput.value === '') {
        toast('입력된 코드가 없습니다.', TOAST_WARNING);
        return false;
    }
    const highlightedCode = hljs.highlightAuto(codeInput.value).value;
    const codeHtml = `<pre><code class="hljs">${highlightedCode}</code><span class="hljs-ctc">Copy</span></pre><div><br></div>`;

    // Froala Editor에 HTML 삽입
    if (editorInstance) {
        editorInstance.selection.restore(); // 저장된 커서 위치로 복원
        editorInstance.html.insert(codeHtml); // 현재 커서 위치에 HTML 삽입
    }

    // codeInput 내용 초기화
    codeInput.value = '';

    // 모달 닫기
    $('#highlightModal').modal('hide');
}Copy


코드 하이라이터 스타일 변경 비교

본문 이미지이전 하이라이팅
본문 이미지개선 하이라이팅

딱 놓고보면 스타일 차이가 크게 난다.
카피 버튼도 추가해놔서 코드 퍼가기도 편하게 구성해보았다.

전반적으로 나름 맘에들어서 그동안 썻던 개발글의 코드뷰를 전부 교체하게되었다.
(거의 80개정도의 글을...)

#JavaScript #FroalaEditor
0 개의 댓글
×