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

편집 요약 없음
편집 요약 없음
1번째 줄: 1번째 줄:
/* ▣ Dictionary gadget ▣
/* ▣ Dictionary gadget ▣
   ─ 검색창+버튼 / Enter
   ─ 검색창 + 버튼/Enter
   ─ 전체·부분·완전 일치 모드
   ─ 완전 일치 → 카드형 결과
  ─ 미일치 → “단어가 없습니다” 한 줄
-------------------------------------------------- */
-------------------------------------------------- */
mw.loader.using( [
mw.loader.using(
    'mediawiki.util',
  ['oojs-ui-core', 'oojs-ui.styles.icons-interactions'],
    'oojs-ui-core',
  function () {
    'oojs-ui.styles.icons-interactions'
], function () {


     /* 사전 JSON 읽어 두기 (key → 정의, 소문자 기준) */
     /* Ⅰ. 사전 JSON을 메모리에 로드 (소문자 ) */
     var dictHolder  = document.getElementById( 'dictionary-json' );
     var dataNode  = document.getElementById('dictionary-json');
     var dictData    = dictHolder ? JSON.parse( dictHolder.textContent ) : {};
     var dict      = dataNode ? JSON.parse(dataNode.textContent) : {};
     var dictLower   = {};
     var dictLower = {};
     Object.keys( dictData ).forEach( function (k) {
     Object.keys(dict).forEach(function (k) {
        dictLower[ k.toLowerCase() ] = dictData[ k ];
      dictLower[k.toLowerCase()] = dict[k];
     } );
     });


     /* ② 페이지 내용이 로드·갱신될 때마다 실행 */
     /* Ⅱ. 한 번만 CSS 삽입 */
     mw.hook( 'wikipage.content' ).add( function ( $content ) {
     if (!document.getElementById('dict-card-style')) {
      mw.util.addCSS(`
        .dict-result-card {
          padding: 12px 16px; border: 1px solid #ccc; border-radius: 8px;
          background: var(--background-color, #f9f9f9); margin: 8px 0;
        }
        .dict-result-card .term {
          font-weight: 600; font-size: 1.1em; margin-right: 0.4em;
        }
        .dict-no-match {
          padding: 8px; color: #d33;
        }`).id = 'dict-card-style';
    }


        var $table = $content.find( '.mw-dictionary' ).first();
    /* Ⅲ. 페이지(또는 Ajax 미리보기)가 로드될 때마다 */
        if ( !$table.length ) return;
    mw.hook('wikipage.content').add(function ($content) {
        if ( $table.prev( '.dict-search-wrapper' ).length ) return;  // 중복 생성 방지


        /* ── 검색창 + 버튼 ── */
      var $table = $content.find('.mw-dictionary').first();
        var input  = new OO.ui.TextInputWidget( {
      if (!$table.length) return;
            placeholder: '단어·뜻 입력…',
      if ($table.prev('.dict-search-wrapper').length) return;   // 이미 처리된 페이지
            icons: [ 'search' ]
        } );
        var button = new OO.ui.ButtonWidget( {
            label: '검색',
            icon:  'search',
            flags: [ 'progressive' ]
        } );
        var field  = new OO.ui.ActionFieldLayout( input, button, { align: 'top' } )
                      .$element.addClass( 'dict-search-wrapper' )
                      .css( 'margin-bottom', '8px' );


         $table.before( field );
      /* 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.addClass('dict-search-wrapper')
                  .css('margin-bottom', '10px');
      $table.before(field);


        /* ── “완전 일치 전용” 표시 행(처음에는 숨김) ── */
      /* 2) 결과 2종 (카드 / 없음) ─ 처음에는 숨김 */
        var $exactRow = $( '<tr class="dict-exact" style="display:none"><td colspan="2"></td></tr>' );
      var $cardRow = $('<tr class="dict-row-card" style="display:none"><td colspan="2"></td></tr>');
        $table.find( 'tr' ).first().after( $exactRow );  // 헤더 바로 아래에 삽입
      var $noneRow = $('<tr class="dict-row-none" style="display:none"><td colspan="2" class="dict-no-match">해당 단어가 없습니다.</td></tr>');
        var $exactCell = $exactRow.children( 'td' );
      var $tbody  = $table;                // Lua 모듈이 <tbody> 없이 직접 <tr> 나열했음
      $tbody.find('tr').first().after($cardRow, $noneRow);  // 헤더 바로 삽입
      var $cardCell = $cardRow.children('td');


        /* ── 필터 함수 ── */
      /* 3) 검색 로직 */
        function applyFilter() {
      function apply() {
            var q = input.getValue().trim();
        var key  = input.getValue().trim();
            var qLow = q.toLowerCase();
        var lower = key.toLowerCase();


            /* 1) 빈 검색어 → 모든 행 표시, exact 숨김 */
        /* 입력 없으면 표 리셋 */
            if ( !q ) {
        if (!key) {
                $exactRow.hide();
          $cardRow.hide(); $noneRow.hide();
                $table.find( 'tr' ).show();
          $table.find('tr').not('.dict-row-card, .dict-row-none').show();
                return;
          return;
            }
        }


            /* 2) 완전 일치 → 전부 숨기고 한 줄로 치환 */
        /* 완전 일치 */
            if ( dictLower.hasOwnProperty( qLow ) ) {
        if (dictLower.hasOwnProperty(lower)) {
                $table.find( 'tr' ).not( '.dict-exact' ).not( ':has(th)' ).hide();
          var defi = mw.html.escape(dictLower[lower]);
                $exactCell.html( '<b>' + mw.html.escape( q ) + '</b> : ' +
          $cardCell.html(
                                mw.html.escape( dictLower[ qLow ] ) );
            '<div class="dict-result-card">' +
                $exactRow.show();
              '<span class="term">' + mw.html.escape(key) + '</span>' +
                return;
              '<span class="def">'  + defi + '</span>' +
            }
            '</div>'
          );
          $cardRow.show();  $noneRow.hide();
          $table.find('tr').not('.dict-row-card, .dict-row-none').not(':has(th)').hide();
          return;
        }


            /* 3) 부분 일치 → 기존 방식 */
        /* 미일치 */
            $exactRow.hide();
        $cardRow.hide();
            $table.find( 'tr' ).each( function () {
        $noneRow.show();
                var $row = $( this );
        $table.find('tr').not('.dict-row-card, .dict-row-none').not(':has(th)').hide();
                if ( $row.is( '.dict-exact' ) || $row.find( 'th' ).length ) return;
      }
 
                var term = $row.children( 'td' ).eq( 0 ).text().toLowerCase();
                var defi = $row.children( 'td' ).eq( 1 ).text().toLowerCase();
                $row.toggle( term.includes( qLow ) || defi.includes( qLow ) );
            } );
        }


        /* ── 이벤트 연결 ── */
      /* 4) 이벤트 연결 */
        button.on( 'click', applyFilter );   // 버튼
      button.on('click', apply);
        input.on( 'enter', applyFilter );   // Enter 키
      input.on('enter', apply);
     } );
     });
} );
  }
);

2025년 6월 29일 (일) 03:13 판

/* ▣ Dictionary gadget ▣
   ─ 검색창 + 버튼/Enter
   ─ 완전 일치 → 카드형 결과
   ─ 미일치 → “단어가 없습니다” 한 줄
-------------------------------------------------- */
mw.loader.using(
  ['oojs-ui-core', 'oojs-ui.styles.icons-interactions'],
  function () {

    /* Ⅰ. 사전 JSON을 메모리에 로드 (소문자 키) */
    var dataNode   = document.getElementById('dictionary-json');
    var dict       = dataNode ? JSON.parse(dataNode.textContent) : {};
    var dictLower  = {};
    Object.keys(dict).forEach(function (k) {
      dictLower[k.toLowerCase()] = dict[k];
    });

    /* Ⅱ. 한 번만 CSS 삽입 */
    if (!document.getElementById('dict-card-style')) {
      mw.util.addCSS(`
        .dict-result-card {
          padding: 12px 16px; border: 1px solid #ccc; border-radius: 8px;
          background: var(--background-color, #f9f9f9); margin: 8px 0;
        }
        .dict-result-card .term {
          font-weight: 600; font-size: 1.1em; margin-right: 0.4em;
        }
        .dict-no-match {
          padding: 8px; color: #d33;
        }`).id = 'dict-card-style';
    }

    /* Ⅲ. 페이지(또는 Ajax 미리보기)가 로드될 때마다 */
    mw.hook('wikipage.content').add(function ($content) {

      var $table = $content.find('.mw-dictionary').first();
      if (!$table.length) return;
      if ($table.prev('.dict-search-wrapper').length) return;   // 이미 처리된 페이지

      /* 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.addClass('dict-search-wrapper')
                   .css('margin-bottom', '10px');
      $table.before(field);

      /* 2) 결과 행 2종 (카드 / 없음) ─ 처음에는 숨김 */
      var $cardRow = $('<tr class="dict-row-card" style="display:none"><td colspan="2"></td></tr>');
      var $noneRow = $('<tr class="dict-row-none" style="display:none"><td colspan="2" class="dict-no-match">해당 단어가 없습니다.</td></tr>');
      var $tbody   = $table;                 // Lua 모듈이 <tbody> 없이 직접 <tr> 나열했음
      $tbody.find('tr').first().after($cardRow, $noneRow);  // 헤더 바로 뒤 삽입
      var $cardCell = $cardRow.children('td');

      /* 3) 검색 로직 */
      function apply() {
        var key   = input.getValue().trim();
        var lower = key.toLowerCase();

        /* 입력 없으면 표 리셋 */
        if (!key) {
          $cardRow.hide(); $noneRow.hide();
          $table.find('tr').not('.dict-row-card, .dict-row-none').show();
          return;
        }

        /* 완전 일치 */
        if (dictLower.hasOwnProperty(lower)) {
          var defi = mw.html.escape(dictLower[lower]);
          $cardCell.html(
            '<div class="dict-result-card">' +
              '<span class="term">' + mw.html.escape(key) + '</span>' +
              '<span class="def">'  + defi + '</span>' +
            '</div>'
          );
          $cardRow.show();   $noneRow.hide();
          $table.find('tr').not('.dict-row-card, .dict-row-none').not(':has(th)').hide();
          return;
        }

        /* 미일치 */
        $cardRow.hide();
        $noneRow.show();
        $table.find('tr').not('.dict-row-card, .dict-row-none').not(':has(th)').hide();
      }

      /* 4) 이벤트 연결 */
      button.on('click', apply);
      input.on('enter', apply);
    });
  }
);