Author: wunwisa.c

  • Fixed Table Header Scrolling

    ในปัจจุบันในการพัฒนา website  เราจะเห็นว่า Table เกือบจะเป็นส่วนหนึ่งในการพัฒนาของแต่ละหน้า  ซึ่งส่วนใหญ่จะเป็นการแสดงผลของการค้นหาข้อมูลต่างๆของเมนูหรือหัวข้อในแต่ล่ะเรื่อง บ้างก็อาจจะมีข้อมูลน้อย บ้างก็อาจจะมีข้อมูลเยอะ ทำให้การแสดงผลมีความยาวล้นหน้าเพจ ถึงแม้ว่าปัจจุบันจะมีการนำ paging มาใช้แต่ในบางกรณีอาจจะมีความจำเป็นที่ต้องการตรวจสอบข้อมูลในการค้นหาทั้งหมดในหน้าเดียวกัน  แต่ปัญหาที่ตามมาในการแสดงผลทั้งหมดในหน้าเดียวกันก็คือเมื่อเรา scroll ลงมาข้างล่างสุด Header ของตัวตารางจะไม่ Fixed ไว้ และเลื่อนหายไปข้างบน ทำให้ยากลำบากในการตรวจสอบข้อมูล เนื่องจากผู้ใช้งานอาจจะลืมว่าคอลัมภ์ที่กำลังตรวจสอบอยู่นั้นคือคอลัมภ์อะไร จึงต้อง scroll ขึ้นไปดู Header อีกครั้ง ทำให้มีความล่าช้าในการตรวจสอบข้อมูล วันนี้จึงนำวิธีการ Fixed Header ของตาราง โดยใช้ css มาเสนอเพื่อเป็นแนวทางในการพัฒนากับผู้อ่าน ดังนี้ค่ะ

     

    ตัวอย่างที่1 (CSS ในหน้า Html เดียวกัน))

    tbody {

    display:block;

    height:400px;

    overflow:auto;

    }

    thead, tbody tr {

    display:table;

    width:100%;

    table-layout:fixed;

    }

    thead {

    width: calc( 100% – 1em )

    }

    ตัวอย่างที่1(การเรียกใช้ CSS ในหน้า Html เดียวกัน)

    <table id=”tableSearch” class=”table table-responsive table-hover” style=”border-collapse:collapse;border:1px solid #C0C0C0;table-layout:auto;width:1500px;”>

    <thead>

    <tr class=”filters” style=”background-color:#C0C0C0″>

    <th align=”left”>Order</th>

    <th align=”left”>Name</th>

    </tr></thead>

    <tbody><tr>

    <td colspan=”14″ style=”text-align:center;color:#808080 “>Order</td>

    <td colspan=”14″ style=”text-align:center;color:#808080 “>Name</td>

    </tr></tbody>

    </table>

    ตัวอย่างที่2 (CSS)

    html, body{

    margin:0;

    padding:0;

    height:100%;

    }

    section {

    position: relative;

    border: 1px solid #000;

    padding-top: 37px;

    background: #500;

    }

    section.positioned {

    position: absolute;

    top:100px;

    left:100px;

    width:800px;

    box-shadow: 0 0 15px #333;

    }

    .container {

    overflow-y: auto;

    height: 200px;

    }

    table {

    border-spacing: 0;

    width:100%;

    }

    td + td {

    border-left:1px solid #eee;

    }

    td, th {

    border-bottom:1px solid #eee;

    background: #ddd;

    color: #000;

    padding: 10px 25px;

    }

    th {

    height: 0;

    line-height: 0;

    padding-top: 0;

    padding-bottom: 0;

    color: transparent;

    border: none;

    white-space: nowrap;

    }

    th div{

    position: absolute;

    background: transparent;

    color: #fff;

    padding: 9px 25px;

    top: 0;

    margin-left: -25px;

    line-height: normal;

    border-left: 1px solid #800;

    }

    th:first-child div{

    border: none;

    }

    ตัวอย่างที่2(การเรียกใช้ CSS)

    <section class=””>

    <div class=”container”>

    <table>

    <thead>

    <tr class=”header”>

    <th>

    Table attribute name

    <div>Table attribute name</div>

    </th>

    <th>

    Value

    <div>Value</div>

    </th>

    <th>

    Description

    <div>Description</div>

    </th>

    </tr>

    </thead>

    <tbody>

    <tr>

    <td>align</td>

    <td>left, center, right</td>

    <td>Not supported in HTML5. Deprecated in HTML 4.01. Specifies the alignment of a table according to surrounding text</td>

    </tr>

    <tr>

    <td>bgcolor</td>

    <td>rgb(x,x,x), #xxxxxx, colorname</td>

    <td>Not supported in HTML5. Deprecated in HTML 4.01. Specifies the background color for a table</td>

    </tr>

    <tr>

    <td>border</td>

    <td>1,””</td>

    <td>Specifies whether the table cells should have borders or not</td>

    </tr>

    <tr>

    <td>cellpadding</td>

    <td>pixels</td>

    <td>Not supported in HTML5. Specifies the space between the cell wall and the cell content</td>

    </tr>

    <tr>

    <td>cellspacing</td>

    <td>pixels</td>

    <td>Not supported in HTML5. Specifies the space between cells</td>

    </tr>

    <tr>

    <td>frame</td>

    <td>void, above, below, hsides, lhs, rhs, vsides, box, border</td>

    <td>Not supported in HTML5. Specifies which parts of the outside borders that should be visible</td>

    </tr>

    <tr>

    <td>rules</td>

    <td>none, groups, rows, cols, all</td>

    <td>Not supported in HTML5. Specifies which parts of the inside borders that should be visible</td>

    </tr>

    <tr>

    <td>summary</td>

    <td>text</td>

    <td>Not supported in HTML5. Specifies a summary of the content of a table</td>

    </tr>

    <tr>

    <td>width</td>

    <td>pixels, %</td>

    <td>Not supported in HTML5. Specifies the width of a table</td>

    </tr>

    </tbody>

    </table>

    </div>

    </section>

    ตัวอย่างที่3(การเรียกใช้ CSS)

    @header_background_color: #333;

    @header_text_color: #FDFDFD;

    @alternate_row_background_color: #DDD;

    @table_width: 750px;

    @table_body_height: 300px;

    @column_one_width: 200px;

    @column_two_width: 200px;

    @column_three_width: 350px;

    .fixed_headers {

    width: @table_width;

    table-layout: fixed;

    border-collapse: collapse;

    th { text-decoration: underline; }

    th, td {

    padding: 5px;

    text-align: left;

    }

    td:nth-child(1), th:nth-child(1) { min-width: @column_one_width; }

    td:nth-child(2), th:nth-child(2) { min-width: @column_two_width; }

    td:nth-child(3), th:nth-child(3) { width: @column_three_width; }

    thead {

    background-color: @header_background_color;

    color: @header_text_color;

    tr {

    display: block;

    position: relative;

    }

    }

    tbody {

    display: block;

    overflow: auto;

    width: 100%;

    height: @table_body_height;

    tr:nth-child(even) {

    background-color: @alternate_row_background_color;

    }}}

    .old_ie_wrapper {

    height: @table_body_height;

    width: @table_width;

    overflow-x: hidden;

    overflow-y: auto;

    tbody { height: auto; }

    }

    ตัวอย่างที่3(การเรียกใช้ CSS)

    <table class=”fixed_headers”>

    <thead>

    <tr>

    <th>Name</th>

    <th>Color</th>

    <th>Description</th>

    </tr>

    </thead>

    <tbody>

    <tr>

    <td>Apple</td>

    <td>Red</td>

    <td>These are red.</td>

    </tr>

    <tr>

    <td>Pear</td>

    <td>Green</td>

    <td>These are green.</td>

    </tr>

    <tr>

    <td>Grape</td>

    <td>Purple / Green</td>

    <td>These are purple and green.</td>

    </tr>

    <tr>

    <td>Orange</td>

    <td>Orange</td>

    <td>These are orange.</td>

    </tr>

    <tr>

    <td>Banana</td>

    <td>Yellow</td>

    <td>These are yellow.</td>

    </tr>

    <tr>

    <td>Kiwi</td>

    <td>Green</td>

    <td>These are green.</td>

    </tr>

    <tr>

    <td>Plum</td>

    <td>Purple</td>

    <td>These are Purple</td>

    </tr>

    <tr>

    <td>Watermelon</td>

    <td>Red</td>

    <td>These are red.</td>

    </tr>

    <tr>

    <td>Tomato</td>

    <td>Red</td>

    <td>These are red.</td>

    </tr>

    <tr>

    <td>Cherry</td>

    <td>Red</td>

    <td>These are red.</td>

    </tr>

    <tr>

    <td>Cantelope</td>

    <td>Orange</td>

    <td>These are orange inside.</td>

    </tr>

    <tr>

    <td>Honeydew</td>

    <td>Green</td>

    <td>These are green inside.</td>

    </tr>

    <tr>

    <td>Papaya</td>

    <td>Green</td>

    <td>These are green.</td>

    </tr>

    <tr>

    <td>Raspberry</td>

    <td>Red</td>

    <td>These are red.</td>

    </tr>

    <tr>

    <td>Blueberry</td>

    <td>Blue</td>

    <td>These are blue.</td>

    </tr>

    <tr>

    <td>Mango</td>

    <td>Orange</td>

    <td>These are orange.</td>

    </tr>

    <tr>

    <td>Passion Fruit</td>

    <td>Green</td>

    <td>These are green.</td>

    </tr>

    </tbody>

    </table>

    <!–[if lte IE 9]>

    </div>

    <!–<![endif]–>

    จากข้อความข้างต้นผู้เขียนได้นำตัวอย่างการ Fixed Header ของ Table โดยใช้ CSS มาให้เลือกใช้ 3 แบบ หวังว่าคงเป็นแนวทางสำหรับการนำไปใช้สำหรับการพัฒนาที่ต้องการแสดงข้อมูลจำนวนมากในหน้าเดียวกัน และต้องการ Freeze Header ของตารางเพื่อตรวจสอบความถูกต้องของข้อมูลนะคะ

    แหล่งอ้างอิง
    https://codepen.io/tjvantoll/pen/JEKIu

  • Pass special characters in JavaScript function

    ในการพัฒนาโปรแกรมไม่ว่าจะพัฒนาด้วยภาษาใดก็ตาม ทุกภาษาจะมี special characters ซึ่ง special characters บางตัวอาจจะถูกระบุให้เป็นตัวแปร หรือ ใช้อักขระพิเศษเพื่อระบุอยู่ลำดับสุดท้ายของข้อความของบรรทัด เป็นต้น   ซึ่งตรงส่วนนี้จะทำให้ประสบปัญหาในการนำ special characters ไปใช้งาน นั่นคือจะทำให้เกิด Error ทำให้โปรแกรมไม่สามารถทำงานต่อได้ ซึ่งJavaScript ก็เป็นอีกหนึ่งภาษาที่ประสบปัญหานี้ เช่น การส่ง parameter ผ่าน function ของ JavaScript โดยค่าของ parameter ที่ส่งไปเป็น special characters ทำให้เกิด Error และโปรแกรมไม่สามารถทำงานต่อได้

     

    วันนี้จะขอแนะนำเทคนิคในการแก้ปัญหาดังกล่าว นั่นก็คือ การ Encoding และ Decoding ในที่นี้ผู้เขียนได้พัฒนาในรูปแบบ MVC โดยใช้ JavaScript ในการเรียก Controller จึงใช้วิธีการ Encoding กับ JavaScript และ ทางฝั่ง Controller จะใช้ C# จึงใช้ C# ในการ Decoding ค่ะ
    1.วิธีการ Encoding in JavaScript ในการจะส่ง parameter ผ่านทาง function ของ JavaScript โดยข้อความนั้นมี special characters จำเป็นที่จะต้องมีการ Encoding ในที่นี้เราจะใช้ฟังก์ชัน Escape() ของ JavaScript
    ตัวอย่าง Encoding in JavaScript

    <script>   
    
      function SaveData() {
    
            var txtDocTitle = $('#txtDocTitle').val();
    
    
            $('#pleaseWaitDialog').modal();
            $.ajax({
                type: 'POST',
                contentType: 'application/json; charset=utf-8',
                url: "@Url.Action("SaveDataToTable", "I_EXPENSE")",
                data: "{'budgetYear':'" + valueYear +
                    "' ,'docTitle':'" + escape(txtDocTitle) +
                    "'}",
                success: function (data) {
    
                },
                error: function (data) {
                    alert(data);
                }
            });
        }
          </script>

    จากตัวอย่างกรณีป้อนข้อมูลที่มี special characters เช่น “I’m *&^$ special characters” เมื่อใช้ function escape(“I’m *&^$ special characters”) จะได้ผลลัพธ์ดังนี้

     ผลลัพธ์จากการเรียกใช้ escape()

    I%u2019m%20*%26%5E%24%20special%20characters
    

    หลังจากที่มีการเรียกใช้ escape() จะทำให้สามารถส่ง parameter ที่มี special characters ไปได้โดยไม่พบ Error โปรแกรมสามารถทำงานได้ปกติ เป็นอันผ่านไปหนึ่งด่านในส่วนของการเรียกใช้งาน function ของ JavaScript แต่ทว่าการที่เราจะนำข้อความที่ผ่านการ Encoding มาใช้งานไม่ได้ เนื่องจากจะไม่ถูกต้องตามความต้องการของผู้ใช้งาน จากตัวอย่างข้างต้น ผู้ใช้งานป้อนข้อมูล “I’m *&^$ special characters” สิ่งที่โปรแกรมต้องทำการบันทึกก็ต้องเป็น “I’m *&^$ special characters” ไม่ใช่ “I%u2019m%20*%26%5E%24%20special%20characters” ดังนั้นจึงจำเป็นจะต้องมีการ Decoding เพื่อให้ได้ข้อมูลที่ถูกต้อง
    2.Decoding in C# เมื่อมีการ Encoding เราจำเป็นที่จะต้อง Decoding เพื่อให้ได้ข้อมูลที่ถูกต้องตามที่ผู้ใช้งานได้ป้อนข้อมูล
    ตัวอย่าง Decoding in C#

          var docTitle = HttpUtility.UrlDecode(docTitle, System.Text.Encoding.Default);
           //ตรงส่วนนี้ตัวแปร docTitle จะมีค่าเท่ากับ “I’m *&^$ special characters”  ค่ะ
          

    เมื่อ Decoding แล้ว ก็จะได้ข้อความที่ถูกต้องตามที่ผู้ใช้งานได้ป้อนข้อมูลและสามารถนำไปใช้งานได้อย่างถูกต้องแล้วนะคะ

     

    สำหรับบทความนี้ก็ขอจบเพียงเท่านี้และหวังว่าผู้พัฒนาคงหมดกังวลในส่วนของ Special character ที่ส่งค่า parameter ผ่านทาง JavaScript ได้อย่างไม่มี error มากวนใจผู้พัฒนากันอีกแล้วนะคะ ^_^

     

     

    แหล่งอ้างอิง :

    https://developer.mozilla.org/th/docs/Web/JavaScript/Reference/Global_Objects/escape

    http://stackoverflow.com/questions/3778165/unescape-javascripts-escape-using-c-sharp

  • การใช้ jQuery ร่วมกับ library อื่นๆ ด้วยโหมด no-conflict

    ในบางครั้งท่านผู้อ่านอาจจะมีปัญหา ในการนำ jQuery ไปใช้กับ ร่วมสคริปต์อื่นๆ หรือแม้กระทั่งพวก css framework ที่บางตัวจะมีการฝัง script ไว้ข้างในด้วย  ซึ่งอาจทำให้ script ของท่านทำงานผิดเพี้ยนไปจากที่ควรเป็น  วันนี้จะขอแนะนำเทคนิคในการแก้ปัญหาดังกล่าว นั่นก็คือ การปรับ jQuery ไปใช้งานในโหมด no-conflict นั่นเองค่ะ

     

    สิ่งที่พึงระวังสำหรับมือใหม่ก็คือ ตัว jquery เองนั้นจะใช้ตัวแปร $  อ้างถึงฟังก์ชั่น jQuery  ดังนั้น ถ้ามีสคริปต์ตัวอื่นที่ใช้ตัวแปร $ นี้เหมือนกันล่ะก็  เกิดปัญหาขึ้นแน่นอนค่ะ

    วิธีแก้ปัญหา

     วิธีที่ 1 คือให้สร้างตัวแปรเพิ่มขึ้นมาอีกตัวเพื่อใช้แทนเครื่องหมาย $

    ตัวอย่างพร้อมคำอธิบาย

    <script src="other_library.js"></script>
    
    <script src="jquery.js"></script>
    
    <script>
    
    //ในที่นี้เราประกาศตัวแปร $j แทนตัว $
    
    var $j = jQuery.noConflict();
    
    $j(document).ready(function(){
    
    $j("div").hide();
    
    });
    
    
    // ถึงตรงนี้ตัวแปร $ จะหมายถึงตัวแปรจากสคริปต์ other_library.js
    
    // mainDiv ดังนั้นคำสั่งข้างล่างนี้ $('main') จึงหมายถึงการเรียกฟังก์ชั่น
    
    // จาก other_library ไม่ใช่การเรียกฟังก์ชั่นของ jquery
    
    window.onload = function(){
    
    var mainDiv = $('main');
    
    }
    
    </script>
    
    

    วิธีที่ 2 ในกรณีที่ไม่ต้องการประกาศตัวแปรเพิ่ม ในที่นี้เราสามารถใช้งานตัวแปร $ ได้แต่เป็นการเรียกใช้แบบ locally scope(เรียกใช้เฉพาะในบล็อคของ jQuery เอง)

    ตัวอย่างพร้อมคำอธิบาย

    <script src="other_library.js"></script>
    
    <script src="jquery.js"></script>
    
    <script>
    
    jQuery.noConflict();
    
    jQuery(document).ready(function($){
    
    // การใช้ตัวแปร $ ในนี้จะหมายถึง jQuery (locally scope)
    
    $("div").hide();
    
    });
    
    // ส่วนการใช้ตัวแปร $ นี้จะหมายถึงการเรียกใช้แบบ global scope
    
    // ซึ่งในที่นี้หมายถึงการเรียกใช้ $ ของ other_library นั่นเอง
    
    window.onload = function(){
    
    var mainDiv = $('main');
    
    }
    
    </script>
    
    

    การเรียกใช้งานแบบนี้จะทำให้เราไม่สับสนกรณีที่เขียนโค๊ดยาวๆ หรือใช้ไลบราลีหลายตัว

    สองวิธีข้างต้น จะสังเกตว่าเรา include สคริปต์ jQuery หลังสคริปต์ตัวอื่น กรณีที่เรามีการ include สคริปต์ jQuery ก่อนสคริปต์ตัวอื่น จะทำอย่างไร?

    ในกรณีที่เรา include jQuery มาก่อนและหากสคริปต์ที่ตามหลังใช้ตัวแปร $ จะถือว่าตัวแปร $ นั้นเป็นของสคริปต์ที่ประกาศที่หลัง(override) ซึ่งถ้าเราต้องการใช้งาน jQuery จะทำได้โดยการเรียกใช้ฟังก์ชั่น  jQuery โดยตรง  ดังตัวอย่างข้างล่างนี้

    ตัวอย่างพร้อมคำอธิบาย

    <script src="jquery.js"></script>
    
    <script src="other_library.js"></script>
    
    <script>
    
    //เรียกใช้ jQuery ฟังก์ชั่นแบบเต็ม
    
    jQuery(document).ready(function(){
    
    jQuery("div").hide();
    
    });
    
     
    
    // เรียกใช้ตัวแปร $ ที่ระบุไว้ใน other_library.js
    
    window.onload = function() {
    
    var mainDiv = $('main');
    
    };
    
     
    
    </script>
    
    

    สำหรับบทความนี้ก็ขอจบเพียงเท่านี้ หวังว่าต่อไปนี้ท่านผู้อ่านที่ใช้งาน jQuery ร่วมกับสคริปต์อื่นๆ คงจะไม่มีปัญหาสคริปต์ทำงานผิดเพี้ยนหรือสคริปต์ทำงานไม่ครบอีกแล้วนะคะ

    แหล่งอ้างอิงhttp://learn.jquery.com/using-jquery-core/avoid-conflicts-other-libraries/