미디어위키:Gadget-dictionary.js: 두 판 사이의 차이

편집 요약 없음
편집 요약 없음
 
(같은 사용자의 중간 판 34개는 보이지 않습니다)
1번째 줄: 1번째 줄:
/* ▣ Dictionary gadget
/* ▣ Dictionary “검색 전용” 가젯
   * 표(class="mw-dictionary")있는 페이지에만 작동
   - 없이: 검색 → 결과 카드 or “없음” 메시지
  * 검색창은 OOUI → 실패 시 HTML <input>로 폴백
  - 단어()에 들어 있는 위키문법을 실제로 렌더링
-------------------------------------------------- */
-------------------------------------------------- */
mw.loader.using( [
mw.loader.using(
    'mediawiki.util',
  ['oojs-ui-core', 'oojs-ui.styles.icons-interactions', 'mediawiki.api'],
    'oojs-ui-core',
  function () {
    'oojs-ui.styles.icons-interactions'
], function () {


     /* 페이지 내용이 처음 로드되거나, Ajax로 다시 삽입될 때마다 실행 */
     /* ────────── Ⅰ. 사전 JSON 로드 ────────── */
     mw.hook( 'wikipage.content' ).add( function ( $content ) {
     var jsonNode = document.getElementById('dictionary-json');
    if (!jsonNode) return;                        // 사전 없는 페이지
    var dictRaw  = JSON.parse(jsonNode.textContent);
    var dict    = {};                            // 소문자+정규화 키 → {rawKey, def}


        var $table = $content.find( '.mw-dictionary' ).first();
    Object.keys(dictRaw).forEach(function (k) {
        if ( !$table.length ) return;                          // 표가 없다면 skip
      const normKey = k.normalize('NFC').toLowerCase();   // ★ 정규화 추가
        if ( $table.prev( '.dict-search-wrapper' ).length ) return; // 중복 생성 방지
      dict[normKey] = { raw: k, def: dictRaw[k] };
    });


        /* ---- 1) 검색창 만들기 ---- */
    /* ────────── Ⅱ. CSS 삽입(1회) ────────── */
         var $wrapper = $( '<div>' ).addClass( 'dict-search-wrapper' )
    if (!document.getElementById('dict-card-style')) {
                                  .css( 'margin-bottom', '8px' );
      mw.util.addCSS(`
.dictionary-container .oo-ui-inputWidget-input, .dictionary-container .oo-ui-buttonElement-button {
background: var(--bg) !important;
color: var(--text) !important;
border: 1px solid var(--border) !important;
height: 36px !important;
}
.oo-ui-inputWidget-input {
border-radius: 0.5rem 0 0 0.5rem !important;
padding-left: 11px !important;
}
.oo-ui-buttonElement-button {
border-radius: 0 0.5rem 0.5rem 0 !important;
padding-top: 6px;
padding-bottom: 6px;
}
         .dict-card    {padding:20px 20px 8px !important; border:1px solid light-dark(#ccc, #555);border-radius:0.8rem;margin: 1em 0 0.5em !important;
                      background:var(--altbg); padding-top: }
        .dict-card .term{font-weight:600;font-size:1.5em;margin-right:.4em; padding:8px; padding-bottom: 3px; margin-bottom: -15px; padding-top: 4px; }
.dict-card .def{padding-left:8px; padding-right:8px; margin-bottom: -3px;}
        .dict-none    {padding:8px;color:light-dark(#d33, HSL(0, 71%, 75%));}
      `).id = 'dict-card-style';
    }


        var makeFilter = function ( getValue ) {
    /* ────────── Ⅲ. 위키텍스트 → HTML 파서 헬퍼 ────────── */
            return function () {
    var api = new mw.Api();
                var q = getValue().trim().toLowerCase();
    function parseWikitext(wikitext) {
                $table.find( 'tr' ).each( function () {
      return api.get({
                    var $row = $( this );
        action:  'parse',
                    if ( $row.find( 'th' ).length ) return;   // 헤더 row
        format:  'json',
                    var term = $row.children( 'td' ).eq(0).text().toLowerCase();
        contentmodel: 'wikitext',
                    var defi = $row.children( 'td' ).eq(1).text().toLowerCase();
        prop:   'text',
                    $row.toggle( !q || term.includes( q ) || defi.includes( q ) );
        text:    wikitext,
                } );
        disablelimitreport: 1,
            };
        disableeditsection: 1,
        };
        pst: 0, wrapshtml: 1
      }).then(function (data) {
        return (data.parse && data.parse.text) ? data.parse.text['*'] : mw.html.escape(wikitext);
      });
    }


         /* 1-A) OOUI 위젯 시도 */
    /* ────────── Ⅳ. 페이지 로드 때 UI 주입 ────────── */
         try {
    mw.hook('wikipage.content').add(function ($content) {
            var widget = new OO.ui.TextInputWidget( {
      $content.find('.dictionary-container').each(function () {
                placeholder: '단어·뜻 검색…',
        var $box = $(this);
                icons: [ 'search' ],
        if ($box.children().length) return;      // 이미 UI가 있음
                indicator: null
 
            } );
         /* 1) 검색창 + 버튼 */
            widget.on( 'change', makeFilter( function () {
         var input  = new OO.ui.TextInputWidget({ placeholder: '단어 입력…', icons:['search'] });
                return widget.getValue();
        var button = new OO.ui.ButtonWidget({ label:'검색', icon:'search', flags:['progressive'] });
            } ) );
        var field  = new OO.ui.ActionFieldLayout(input, button, {align:'top'})
             $wrapper.append( widget.$element );
                    .$element.css('margin-bottom','10px');
        } catch ( e ) {
 
             /* 1-B) 실패하면 순수 HTML 입력창 */
        /* 2) 결과 영역 */
             var $input = $( '<input type="search" placeholder="검색…">' )
        var $result = $('<div class="dict-result"></div>');
                          .css( { width: '100%', padding: '4px' } );
        $box.append(field, $result);
            $input.on( 'input', makeFilter( function () {
 
                 return $input.val();
        /* 3) 검색 실행 – 단어·뜻 모두 위키텍스트 → HTML */
             } ) );
        function run() {
             $wrapper.append( $input );
          const qRaw  = input.getValue().trim();
          const q    = qRaw.normalize('NFC');        // ★ 정규화 추가
          const keyL  = q.toLowerCase();
 
          if (!qRaw) {                    // 검색어 비우면 결과 초기화
             $result.empty();
            return;
          }
 
          if (dict.hasOwnProperty(keyL)) {
             const entry = dict[keyL];
 
            Promise.all([
              parseWikitext(entry.raw),  // ① 단어
              parseWikitext(entry.def)   // ② 정의
             ]).then(function (parts) {
              const htmlKey = parts[0], htmlDef = parts[1];
 
              $result.html(
                '<div class="term">' + htmlKey + '</div>' +
                '<div class="def">' + htmlDef + '</div>'
              );
            }).catch(function () {
              $result.html(
                '<div class="term">' + mw.html.escape(entry.raw) + '</div>' +
                 '<div class="def">'  + mw.html.escape(entry.def) + '</div>'
              );
             });
 
          } else {
             $result.html('<div class="dict-none">해당 단어가 없습니다.</div>');
          }
         }
         }


         $table.before( $wrapper );   // 표 앞에 검색창 끼우기
         button.on('click', run);
     } );
        input.on('enter', run);
} );
      });
     });
  }
);

2026년 1월 25일 (일) 05:02 기준 최신판

/* ▣ Dictionary “검색 전용” 가젯 ▣
   - 표 없이: 검색 → 결과 카드 or “없음” 메시지
   - 단어(키)에 들어 있는 위키문법을 실제로 렌더링
-------------------------------------------------- */
mw.loader.using(
  ['oojs-ui-core', 'oojs-ui.styles.icons-interactions', 'mediawiki.api'],
  function () {

    /* ────────── Ⅰ. 사전 JSON 로드 ────────── */
    var jsonNode = document.getElementById('dictionary-json');
    if (!jsonNode) return;                        // 사전 없는 페이지
    var dictRaw  = JSON.parse(jsonNode.textContent);
    var dict     = {};                            // 소문자+정규화 키 → {rawKey, def}

    Object.keys(dictRaw).forEach(function (k) {
      const normKey = k.normalize('NFC').toLowerCase();   // ★ 정규화 추가
      dict[normKey] = { raw: k, def: dictRaw[k] };
    });

    /* ────────── Ⅱ. CSS 삽입(1회) ────────── */
    if (!document.getElementById('dict-card-style')) {
      mw.util.addCSS(`
.dictionary-container .oo-ui-inputWidget-input, .dictionary-container .oo-ui-buttonElement-button {
background: var(--bg) !important;
color: var(--text) !important;
border: 1px solid var(--border) !important;
height: 36px !important;
}
.oo-ui-inputWidget-input {
border-radius: 0.5rem 0 0 0.5rem !important;
padding-left: 11px !important;
}
.oo-ui-buttonElement-button {
border-radius: 0 0.5rem 0.5rem 0 !important;
padding-top: 6px;
padding-bottom: 6px;
}
        .dict-card    {padding:20px 20px 8px !important; border:1px solid light-dark(#ccc, #555);border-radius:0.8rem;margin: 1em 0 0.5em !important;
                       background:var(--altbg); padding-top: }
        .dict-card .term{font-weight:600;font-size:1.5em;margin-right:.4em; padding:8px; padding-bottom: 3px; margin-bottom: -15px; padding-top: 4px; }
.dict-card .def{padding-left:8px; padding-right:8px; margin-bottom: -3px;}
        .dict-none    {padding:8px;color:light-dark(#d33, HSL(0, 71%, 75%));}
      `).id = 'dict-card-style';
    }

    /* ────────── Ⅲ. 위키텍스트 → HTML 파서 헬퍼 ────────── */
    var api = new mw.Api();
    function parseWikitext(wikitext) {
      return api.get({
        action:  'parse',
        format:  'json',
        contentmodel: 'wikitext',
        prop:    'text',
        text:    wikitext,
        disablelimitreport: 1,
        disableeditsection: 1,
        pst: 0, wrapshtml: 1
      }).then(function (data) {
        return (data.parse && data.parse.text) ? data.parse.text['*'] : mw.html.escape(wikitext);
      });
    }

    /* ────────── Ⅳ. 페이지 로드 때 UI 주입 ────────── */
    mw.hook('wikipage.content').add(function ($content) {
      $content.find('.dictionary-container').each(function () {
        var $box = $(this);
        if ($box.children().length) return;       // 이미 UI가 있음

        /* 1) 검색창 + 버튼 */
        var input  = new OO.ui.TextInputWidget({ placeholder: '단어 입력…', icons:['search'] });
        var button = new OO.ui.ButtonWidget({ label:'검색', icon:'search', flags:['progressive'] });
        var field  = new OO.ui.ActionFieldLayout(input, button, {align:'top'})
                     .$element.css('margin-bottom','10px');

        /* 2) 결과 영역 */
        var $result = $('<div class="dict-result"></div>');
        $box.append(field, $result);

        /* 3) 검색 실행 – 단어·뜻 모두 위키텍스트 → HTML */
        function run() {
          const qRaw  = input.getValue().trim();
          const q     = qRaw.normalize('NFC');        // ★ 정규화 추가
          const keyL  = q.toLowerCase();

          if (!qRaw) {                    // 검색어 비우면 결과 초기화
            $result.empty();
            return;
          }

          if (dict.hasOwnProperty(keyL)) {
            const entry = dict[keyL];

            Promise.all([
              parseWikitext(entry.raw),   // ① 단어
              parseWikitext(entry.def)    // ② 정의
            ]).then(function (parts) {
              const htmlKey = parts[0], htmlDef = parts[1];

              $result.html(
                '<div class="term">' + htmlKey + '</div>' +
                '<div class="def">'  + htmlDef + '</div>'
              );
            }).catch(function () {
              $result.html(
                '<div class="term">' + mw.html.escape(entry.raw) + '</div>' +
                '<div class="def">'  + mw.html.escape(entry.def) + '</div>'
              );
            });

          } else {
            $result.html('<div class="dict-none">해당 단어가 없습니다.</div>');
          }
        }

        button.on('click', run);
        input.on('enter', run);
      });
    });
  }
);