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

편집 요약 없음
편집 요약 없음
 
(같은 사용자의 중간 판 54개는 보이지 않습니다)
1번째 줄: 1번째 줄:
/* ▣ Dictionary gadget ▣
  ─ 검색창 + 버튼/Enter
  ─ 완전 일치 → 카드형 결과
  ─ 미일치 → “단어가 없습니다” 한 줄
-------------------------------------------------- */
mw.loader.using(
mw.loader.using(
   ['oojs-ui-core', 'oojs-ui.styles.icons-interactions'],
   ['oojs-ui-core', 'oojs-ui.styles.icons-interactions', 'mediawiki.api'],
 
   function () {
   function () {
    var jsonNode = document.getElementById('dictionary-json');
    if (!jsonNode) return;
    var dictRaw  = JSON.parse(jsonNode.textContent);
    var dict    = {}; 


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


    /* Ⅱ. 한 번만 CSS 삽입 */
     if (!document.getElementById('dict-card-style')) {
     if (!document.getElementById('dict-card-style')) {
       mw.util.addCSS(`
       mw.util.addCSS(`
         .dict-result-card {
         .dictionary-container .oo-ui-inputWidget-input, .dictionary-container .oo-ui-buttonElement-button {
          padding: 12px 16px; border: 1px solid #ccc; border-radius: 8px;
            background: var(--bg) !important;
          background: var(--background-color, #f9f9f9); margin: 8px 0;
            color: var(--text) !important;
            border: 1px solid var(--border) !important;
            height: 36px !important;
            padding-top: 6.25px !important;
            padding-bottom: 6.25px !important;
            font-family: 'Source', 'Noto Sans KR', 'Noto Sans', 'Noto Sans CJK KR';
        }
        .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;
            width: 100px;
         }
         }
         .dict-result-card .term {
         .dict-card {
          font-weight: 600; font-size: 1.1em; margin-right: 0.4em;
            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:
            font-family: 'Source', 'Noto Sans KR', 'Noto Sans', 'Noto Sans CJK KR';  
         }
         }
         .dict-no-match {
         .dict-card .term p, .dict-card .def p,
          padding: 8px; color: #d33;
        .dict-card .term p *, .dict-card .def p * {
         }`).id = 'dict-card-style';
            margin-top: 0 !important;
            margin-bottom: 0 !important;
            font-family: 'Source', 'Noto Sans KR', 'Noto Sans', 'Noto Sans CJK KR';
        }
        .dict-card .term {
            font-weight: 600;
            font-size: 1.5em;
            margin-right: .4em;
            padding: 8px;
            padding-bottom: 8px;
            padding-top: 6px;
        }
        .dict-card .def {
            padding-left: 8px;
            padding-right: 8px;
            margin-bottom: 10px;
            }
        .dict-none {
            padding: 8px;
            color: light-dark(#d33, HSL(0, 71%, 75%));
         }
        .dict-result { margin-top: 5px; }
      `).id = 'dict-card-style';
    }
 
    var api = new mw.Api();
 
    function debounce(fn, delay) {
      let timer;
      return function () {
        clearTimeout(timer);
        timer = setTimeout(fn, delay);
      };
    }
 
    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);
      });
    }
 
    function matchLevel(entry, q) {
      if (entry.rawNorm === q) return 1;
      if (entry.rawNorm.startsWith(q)) return 2;
      if (entry.rawNorm.includes(q)) return 3;
      if (entry.defNorm.includes(q)) return 4;
      return 0;
    }
 
    async function renderResults($result, results) {
      if (!results.length) {
        $result.html('<div class="dict-none">해당 단어가 없습니다.</div>');
        return;
      }
 
      const chunks = await Promise.all(
        results.map(async r => {
          const termHtml = await parseWikitext(r.entry.raw);
          const defHtml  = await parseWikitext(r.entry.def);
 
          return `<div class="term">${termHtml}</div><div class="def">${defHtml}</div>`;
        })
      );
 
      $result.html(chunks.join(''));
     }
     }


    /* Ⅲ. 페이지(또는 Ajax 미리보기)가 로드될 때마다 */
     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 input  = new OO.ui.TextInputWidget({ placeholder: '검색어 입력…', icons: ['search'] });
        var button = new OO.ui.ButtonWidget({ label: '검색', flags: ['progressive'] });
        var field  = new OO.ui.ActionFieldLayout(input, button, {align:'top'})
                    .$element.css('margin-bottom','10px');
        var $result = $('<div class="dict-result"></div>');
        $box.append(field, $result);


      var $table = $content.find('.mw-dictionary').first();
        function run() {
      if (!$table.length) return;
          const qRaw = input.getValue();
      if ($table.prev('.dict-search-wrapper').length) return;   // 이미 처리된 페이지
          const q = qRaw.trim().normalize('NFC').toLowerCase();


      /* 1) 검색창 + 버튼 */
          if (!q) {
      var input  = new OO.ui.TextInputWidget({
            $result.empty();
        placeholder: '단어 입력…',
            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', '10px');
      $table.before(field);


      /* 2) 결과 행 2종 (카드 / 없음) ─ 처음에는 숨김 */
          const hits = [];
      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) 검색 로직 */
          Object.values(dict).forEach(entry => {
      function apply() {
            const level = matchLevel(entry, q);
        var key  = input.getValue().trim();
            if (level) {
        var lower = key.toLowerCase();
              hits.push({ entry, level });
            }
          });


        /* 입력 없으면 표 리셋 */
          hits.sort((a, b) => {
        if (!key) {
            if (a.level !== b.level) return a.level - b.level;
          $cardRow.hide(); $noneRow.hide();
            return a.entry.rawNorm.localeCompare(b.entry.rawNorm);
          $table.find('tr').not('.dict-row-card, .dict-row-none').show();
           });
           return;
        }


        /* 완전 일치 */
           renderResults($result, hits.slice(0, 11));
        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;
         }
         }


         /* 미일치 */
         button.on('click', run);
        $cardRow.hide();
         input.on('enter', run);
         $noneRow.show();
         const runDebounced = debounce(run, 200);
         $table.find('tr').not('.dict-row-card, .dict-row-none').not(':has(th)').hide();
        input.on('change', runDebounced);
      }
       });
 
      /* 4) 이벤트 연결 */
      button.on('click', apply);
       input.on('enter', apply);
     });
     });
   }
   }
);
);

2026년 3월 21일 (토) 16:41 기준 최신판

mw.loader.using(
  ['oojs-ui-core', 'oojs-ui.styles.icons-interactions', 'mediawiki.api'],

  function () {
    var jsonNode = document.getElementById('dictionary-json');
    if (!jsonNode) return; 
    var dictRaw  = JSON.parse(jsonNode.textContent);
    var dict     = {};  

    Object.keys(dictRaw).forEach(function (k) {
      const normKey = k.normalize('NFC').toLowerCase(); 
      dict[normKey] = {
        raw: k,
        def: dictRaw[k],
        rawNorm: k.normalize('NFC').toLowerCase(),
        defNorm: dictRaw[k].normalize('NFC').toLowerCase()
      };
    });

    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;
            padding-top: 6.25px !important;
            padding-bottom: 6.25px !important;
            font-family: 'Source', 'Noto Sans KR', 'Noto Sans', 'Noto Sans CJK KR'; 
        }
        .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;
            width: 100px;
        }
        .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: 
            font-family: 'Source', 'Noto Sans KR', 'Noto Sans', 'Noto Sans CJK KR'; 
        }
        .dict-card .term p, .dict-card .def p,
        .dict-card .term p *, .dict-card .def p * {
            margin-top: 0 !important;
            margin-bottom: 0 !important;
            font-family: 'Source', 'Noto Sans KR', 'Noto Sans', 'Noto Sans CJK KR'; 
        }
        .dict-card .term {
            font-weight: 600;
            font-size: 1.5em;
            margin-right: .4em;
            padding: 8px;
            padding-bottom: 8px;
            padding-top: 6px;
        }
        .dict-card .def {
            padding-left: 8px;
            padding-right: 8px;
            margin-bottom: 10px;
            }
        .dict-none {
            padding: 8px;
            color: light-dark(#d33, HSL(0, 71%, 75%));
        }
        .dict-result { margin-top: 5px; }
      `).id = 'dict-card-style';
    }

    var api = new mw.Api();

    function debounce(fn, delay) {
      let timer;
      return function () {
        clearTimeout(timer);
        timer = setTimeout(fn, delay);
      };
    }

    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);
      });
    }

    function matchLevel(entry, q) {
      if (entry.rawNorm === q) return 1;
      if (entry.rawNorm.startsWith(q)) return 2;
      if (entry.rawNorm.includes(q)) return 3;
      if (entry.defNorm.includes(q)) return 4;
      return 0;
    }

    async function renderResults($result, results) {
      if (!results.length) {
        $result.html('<div class="dict-none">해당 단어가 없습니다.</div>');
        return;
      }

      const chunks = await Promise.all(
        results.map(async r => {
          const termHtml = await parseWikitext(r.entry.raw);
          const defHtml  = await parseWikitext(r.entry.def);

          return `<div class="term">${termHtml}</div><div class="def">${defHtml}</div>`;
        })
      );

      $result.html(chunks.join(''));
    }

    mw.hook('wikipage.content').add(function ($content) {
      $content.find('.dictionary-container').each(function () {
        var $box = $(this);
        if ($box.children().length) return; 

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

        var $result = $('<div class="dict-result"></div>');
        $box.append(field, $result);

        function run() {
          const qRaw = input.getValue();
          const q = qRaw.trim().normalize('NFC').toLowerCase();

          if (!q) {
            $result.empty();
            return;
          }

          const hits = [];

          Object.values(dict).forEach(entry => {
            const level = matchLevel(entry, q);
            if (level) {
              hits.push({ entry, level });
            }
          });

          hits.sort((a, b) => {
            if (a.level !== b.level) return a.level - b.level;
            return a.entry.rawNorm.localeCompare(b.entry.rawNorm);
          });

          renderResults($result, hits.slice(0, 11));
        }

        button.on('click', run);
        input.on('enter', run);
        const runDebounced = debounce(run, 200);
        input.on('change', runDebounced);
      });
    });
  }
);