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

편집 요약 없음
편집 요약 없음
1번째 줄: 1번째 줄:
/* ▣ Dictionary gadget ▣
/* ▣ Dictionary gadget ▣
   검색창 + 버튼 → 클릭(또는 Enter) 때만 필터 적용
   검색창+버튼 / Enter
  ─ 전체·부분·완전 일치 모드
-------------------------------------------------- */
-------------------------------------------------- */
mw.loader.using( [
mw.loader.using( [
'oojs-ui-core',
    'mediawiki.util',
'oojs-ui.styles.icons-interactions'
    'oojs-ui-core',
    'oojs-ui.styles.icons-interactions'
], function () {
], function () {


/* 페이지 내용이 로드·갱신될 때마다 실행 */
    /* ① 사전 JSON 읽어 두기 (key → 정의, 소문자 기준) */
mw.hook( 'wikipage.content' ).add( function ( $content ) {
    var dictHolder  = document.getElementById( 'dictionary-json' );
    var dictData    = dictHolder ? JSON.parse( dictHolder.textContent ) : {};
    var dictLower  = {};
    Object.keys( dictData ).forEach( function (k) {
        dictLower[ k.toLowerCase() ] = dictData[ k ];
    } );


var $table = $content.find( '.mw-dictionary' ).first();
    /* ② 페이지 내용이 로드·갱신될 때마다 실행 */
if ( !$table.length ) return;                              // 사전 표가 없으면 패스
    mw.hook( 'wikipage.content' ).add( function ( $content ) {
if ( $table.prev( '.dict-search-wrapper' ).length ) return; // 중복 생성 방지


/* ---------- 1) 검색창 & 버튼 만들기 ---------- */
        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' ],
indicator: null
} );


var button = new OO.ui.ButtonWidget( {
        /* ── 검색창 + 버튼 ── */
label: '검색',
        var input  = new OO.ui.TextInputWidget( {
icon: 'search',
            placeholder: '단어·뜻 입력…',
flags: [ 'progressive' ]
            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' );


/* OOUI ActionFieldLayout → 두 요소를 한 줄에 배치 */
        $table.before( field );
var field  = new OO.ui.ActionFieldLayout( input, button, {
align: 'top'
} ).$element
  .addClass( 'dict-search-wrapper' )
  .css( 'margin-bottom', '8px' );


$table.before( field );
        /* ── “완전 일치 전용” 표시 행(처음에는 숨김) ── */
        var $exactRow = $( '<tr class="dict-exact" style="display:none"><td colspan="2"></td></tr>' );
        $table.find( 'tr' ).first().after( $exactRow );  // 헤더 바로 아래에 삽입
        var $exactCell = $exactRow.children( 'td' );


/* ---------- 2) 필터 함수 ---------- */
        /* ── 필터 함수 ── */
function applyFilter() {
        function applyFilter() {
var q = input.getValue().trim().toLowerCase();
            var q = input.getValue().trim();
            var qLow = q.toLowerCase();


$table.find( 'tr' ).each( function () {
            /* 1) 빈 검색어 → 모든 행 표시, exact 숨김 */
var $row = $( this );
            if ( !q ) {
if ( $row.find( 'th' ).length ) return;           // 헤더 row
                $exactRow.hide();
                $table.find( 'tr' ).show();
                return;
            }


var term = $row.children( 'td' ).eq( 0 ).text().toLowerCase();
            /* 2) 완전 일치 → 전부 숨기고 한 줄로 치환 */
var defi = $row.children( 'td' ).eq( 1 ).text().toLowerCase();
            if ( dictLower.hasOwnProperty( qLow ) ) {
                $table.find( 'tr' ).not( '.dict-exact' ).not( ':has(th)' ).hide();
                $exactCell.html( '<b>' + mw.html.escape( q ) + '</b> : ' +
                                mw.html.escape( dictLower[ qLow ] ) );
                $exactRow.show();
                return;
            }


$row.toggle( !q || term.includes( q ) || defi.includes( q ) );
            /* 3) 부분 일치 → 기존 방식 */
} );
            $exactRow.hide();
}
            $table.find( 'tr' ).each( function () {
                var $row = $( this );
                if ( $row.is( '.dict-exact' ) || $row.find( 'th' ).length ) return;


/* ---------- 3) 이벤트 연결 ---------- */
                var term = $row.children( 'td' ).eq( 0 ).text().toLowerCase();
button.on( 'click', applyFilter );     // 버튼 클릭
                var defi = $row.children( 'td' ).eq( 1 ).text().toLowerCase();
input.on( 'enter', applyFilter );       // Enter 키
                $row.toggle( term.includes( qLow ) || defi.includes( qLow ) );
} );
            } );
        }
 
        /* ── 이벤트 연결 ── */
        button.on( 'click', applyFilter );   // 버튼
        input.on( 'enter', applyFilter );   // Enter 키
    } );
} );
} );

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

/* ▣ Dictionary gadget ▣
   ─ 검색창+버튼 / Enter
   ─ 전체·부분·완전 일치 모드
-------------------------------------------------- */
mw.loader.using( [
    'mediawiki.util',
    'oojs-ui-core',
    'oojs-ui.styles.icons-interactions'
], function () {

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

    /* ② 페이지 내용이 로드·갱신될 때마다 실행 */
    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;  // 중복 생성 방지

        /* ── 검색창 + 버튼 ── */
        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', '8px' );

        $table.before( field );

        /* ── “완전 일치 전용” 표시 행(처음에는 숨김) ── */
        var $exactRow = $( '<tr class="dict-exact" style="display:none"><td colspan="2"></td></tr>' );
        $table.find( 'tr' ).first().after( $exactRow );  // 헤더 바로 아래에 삽입
        var $exactCell = $exactRow.children( 'td' );

        /* ── 필터 함수 ── */
        function applyFilter() {
            var q = input.getValue().trim();
            var qLow = q.toLowerCase();

            /* 1) 빈 검색어 → 모든 행 표시, exact 숨김 */
            if ( !q ) {
                $exactRow.hide();
                $table.find( 'tr' ).show();
                return;
            }

            /* 2) 완전 일치 → 전부 숨기고 한 줄로 치환 */
            if ( dictLower.hasOwnProperty( qLow ) ) {
                $table.find( 'tr' ).not( '.dict-exact' ).not( ':has(th)' ).hide();
                $exactCell.html( '<b>' + mw.html.escape( q ) + '</b> : ' +
                                 mw.html.escape( dictLower[ qLow ] ) );
                $exactRow.show();
                return;
            }

            /* 3) 부분 일치 → 기존 방식 */
            $exactRow.hide();
            $table.find( 'tr' ).each( function () {
                var $row = $( this );
                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 ) );
            } );
        }

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