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

편집 요약 없음
편집 요약 없음
1번째 줄: 1번째 줄:
/* ▣ Dictionary gadget
/* ▣ Dictionary “검색 전용” 가젯
   ─ 검색창 + 버튼/Enter
   - 페이지에 .dictionary-container 가 있을 때만 동작
   ─ 완전 일치 카드형 결과
   - 표 없이: 검색 → 결과 카드 or “없음” 메시지
  ─ 미일치 → “단어가 없습니다” 한 줄
-------------------------------------------------- */
-------------------------------------------------- */
mw.loader.using(
mw.loader.using(
8번째 줄: 7번째 줄:
   function () {
   function () {


     /* Ⅰ. 사전 JSON을 메모리에 로드 (소문자 키) */
     /* 사전 JSON 로드 */
     var dataNode  = document.getElementById('dictionary-json');
     var jsonNode = document.getElementById('dictionary-json');
     var dict      = dataNode ? JSON.parse(dataNode.textContent) : {};
    if (!jsonNode) return;                        // 사전 없는 페이지
     var dictLower  = {};
     var dictRaw  = JSON.parse(jsonNode.textContent);
     Object.keys(dict).forEach(function (k) {
     var dict    = {};                           // 소문자 키 → 정의
       dictLower[k.toLowerCase()] = dict[k];
     Object.keys(dictRaw).forEach(function (k) {
       dict[k.toLowerCase()] = dictRaw[k];
     });
     });


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


     /* Ⅲ. 페이지(또는 Ajax 미리보기)가 로드될 때마다 */
     /* 콘텐츠 훅: 컨테이너마다 UI 주입 */
     mw.hook('wikipage.content').add(function ($content) {
     mw.hook('wikipage.content').add(function ($content) {
      $content.find('.dictionary-container').each(function () {
        var $box = $(this);
        if ($box.children().length) return;      // 중복 방지


      var $table = $content.find('.mw-dictionary').first();
        /* 1) 입력창 + 버튼 */
      if (!$table.length) return;
        var input  = new OO.ui.TextInputWidget({
      if ($table.prev('.dict-search-wrapper').length) return;  // 이미 처리된 페이지
          placeholder: '단어 입력…', icons:['search']
 
        });
      /* 1) 검색창 + 버튼 */
        var button = new OO.ui.ButtonWidget({
      var input  = new OO.ui.TextInputWidget({
          label:'검색', icon:'search', flags:['progressive']
        placeholder: '단어 입력…',
        });
        icons: ['search']
        var field  = new OO.ui.ActionFieldLayout(input, button, {align:'top'})
      });
                    .$element.css('margin-bottom','10px');
      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종 (카드 / 없음) ─ 처음에는 숨김 */
        /* 2) 결과 영역 */
      var $cardRow = $('<tr class="dict-row-card" style="display:none"><td colspan="2"></td></tr>');
        var $result = $('<div class="dict-result"></div>');
      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) 검색 로직 */
         $box.append(field, $result);
      function apply() {
        var key  = input.getValue().trim();
         var lower = key.toLowerCase();


         /* 입력 없으면 표 리셋 */
         /* 3) 검색 실행 */
         if (!key) {
         function run() {
           $cardRow.hide(); $noneRow.hide();
           var q = input.getValue().trim();
           $table.find('tr').not('.dict-row-card, .dict-row-none').show();
           var key = q.toLowerCase();
          return;
        }


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


         /* 미일치 */
         button.on('click', run);
        $cardRow.hide();
        input.on('enter', run);
        $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);
     });
     });
   }
   }
);
);

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

/* ▣ Dictionary “검색 전용” 가젯 ▣
   - 페이지에 .dictionary-container 가 있을 때만 동작
   - 표 없이: 검색 → 결과 카드 or “없음” 메시지
-------------------------------------------------- */
mw.loader.using(
  ['oojs-ui-core', 'oojs-ui.styles.icons-interactions'],
  function () {

    /* 사전 JSON 로드 */
    var jsonNode = document.getElementById('dictionary-json');
    if (!jsonNode) return;                        // 사전 없는 페이지
    var dictRaw  = JSON.parse(jsonNode.textContent);
    var dict     = {};                            // 소문자 키 → 정의
    Object.keys(dictRaw).forEach(function (k) {
      dict[k.toLowerCase()] = dictRaw[k];
    });

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

    /* 콘텐츠 훅: 컨테이너마다 UI 주입 */
    mw.hook('wikipage.content').add(function ($content) {
      $content.find('.dictionary-container').each(function () {
        var $box = $(this);
        if ($box.children().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.css('margin-bottom','10px');

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

        $box.append(field, $result);

        /* 3) 검색 실행 */
        function run() {
          var q = input.getValue().trim();
          var key = q.toLowerCase();

          if (!q) {
            $result.empty();
            return;
          }
          if (dict.hasOwnProperty(key)) {
            $result.html(
              '<div class="dict-card">' +
                '<span class="term">' + mw.html.escape(q) + '</span>' +
                '<span class="def">'  + mw.html.escape(dict[key]) + '</span>' +
              '</div>'
            );
          } else {
            $result.html('<div class="dict-none">해당 단어가 없습니다.</div>');
          }
        }

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