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

편집 요약 없음
편집 요약 없음
1번째 줄: 1번째 줄:
/* ▣ Dictionary gadget ▣
/* ▣ Dictionary gadget ▣
   * 표(class="mw-dictionary")가 있는 페이지에만 작동
   검색창 + 버튼 → 클릭(또는 Enter) 때만 필터 적용
  * 검색창은 OOUI → 실패 시 HTML <input>로 폴백
-------------------------------------------------- */
-------------------------------------------------- */
mw.loader.using( [
mw.loader.using( [
    'mediawiki.util',
'oojs-ui-core',
    'oojs-ui-core',
'oojs-ui.styles.icons-interactions'
    'oojs-ui.styles.icons-interactions'
], function () {
], function () {


    /* 페이지 내용이 처음 로드되거나, Ajax로 다시 삽입될 때마다 실행 */
/* 페이지 내용이 로드·갱신될 때마다 실행 */
    mw.hook( 'wikipage.content' ).add( function ( $content ) {
mw.hook( 'wikipage.content' ).add( function ( $content ) {


        var $table = $content.find( '.mw-dictionary' ).first();
var $table = $content.find( '.mw-dictionary' ).first();
        if ( !$table.length ) return;                         // 표가 없다면 skip
if ( !$table.length ) return;                               // 사전 표가 없으면 패스
        if ( $table.prev( '.dict-search-wrapper' ).length ) return; // 중복 생성 방지
if ( $table.prev( '.dict-search-wrapper' ).length ) return; // 중복 생성 방지


        /* ---- 1) 검색창 만들기 ---- */
/* ---------- 1) 검색창 & 버튼 만들기 ---------- */
        var $wrapper = $( '<div>' ).addClass( 'dict-search-wrapper' )
var input  = new OO.ui.TextInputWidget( {
                                  .css( 'margin-bottom', '8px' );
placeholder: '단어·뜻 입력…',
icons: [ 'search' ],
indicator: null
} );


        var makeFilter = function ( getValue ) {
var button = new OO.ui.ButtonWidget( {
            return function () {
label: '검색',
                var q = getValue().trim().toLowerCase();
icon: 'search',
                $table.find( 'tr' ).each( function () {
flags: [ 'progressive' ]
                    var $row = $( this );
} );
                    if ( $row.find( 'th' ).length ) return;    // 헤더 row
                    var term = $row.children( 'td' ).eq(0).text().toLowerCase();
                    var defi = $row.children( 'td' ).eq(1).text().toLowerCase();
                    $row.toggle( !q || term.includes( q ) || defi.includes( q ) );
                } );
            };
        };


        /* 1-A) OOUI 위젯 시도 */
/* OOUI ActionFieldLayout → 두 요소를 한 줄에 배치 */
        try {
var field  = new OO.ui.ActionFieldLayout( input, button, {
            var widget = new OO.ui.TextInputWidget( {
align: 'top'
                placeholder: '단어·뜻 검색…',
} ).$element
                icons: [ 'search' ],
  .addClass( 'dict-search-wrapper' )
                indicator: null
  .css( 'margin-bottom', '8px' );
            } );
            widget.on( 'change', makeFilter( function () {
                return widget.getValue();
            } ) );
            $wrapper.append( widget.$element );
        } catch ( e ) {
            /* 1-B) 실패하면 순수 HTML 입력창 */
            var $input = $( '<input type="search" placeholder="검색…">' )
                          .css( { width: '100%', padding: '4px' } );
            $input.on( 'input', makeFilter( function () {
                return $input.val();
            } ) );
            $wrapper.append( $input );
        }


        $table.before( $wrapper );   // 표 앞에 검색창 끼우기
$table.before( field );
    } );
 
/* ---------- 2) 필터 함수 ---------- */
function applyFilter() {
var q = input.getValue().trim().toLowerCase();
 
$table.find( 'tr' ).each( function () {
var $row = $( this );
if ( $row.find( 'th' ).length ) return;            // 헤더 row
 
var term = $row.children( 'td' ).eq( 0 ).text().toLowerCase();
var defi = $row.children( 'td' ).eq( 1 ).text().toLowerCase();
 
$row.toggle( !q || term.includes( q ) || defi.includes( q ) );
} );
}
 
/* ---------- 3) 이벤트 연결 ---------- */
button.on( 'click', applyFilter );      // 버튼 클릭
input.on( 'enter', applyFilter );       // Enter 키
} );
} );
} );

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

/* ▣ Dictionary gadget ▣
   검색창 + 버튼 → 클릭(또는 Enter) 때만 필터 적용
-------------------------------------------------- */
mw.loader.using( [
	'oojs-ui-core',
	'oojs-ui.styles.icons-interactions'
], function () {

	/* 페이지 내용이 로드·갱신될 때마다 실행 */
	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' ],
			indicator: null
		} );

		var button = new OO.ui.ButtonWidget( {
			label: '검색',
			icon: 'search',
			flags: [ 'progressive' ]
		} );

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

		$table.before( field );

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

			$table.find( 'tr' ).each( function () {
				var $row = $( this );
				if ( $row.find( 'th' ).length ) return;            // 헤더 row

				var term = $row.children( 'td' ).eq( 0 ).text().toLowerCase();
				var defi = $row.children( 'td' ).eq( 1 ).text().toLowerCase();

				$row.toggle( !q || term.includes( q ) || defi.includes( q ) );
			} );
		}

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