'css'에 해당되는 글 2건

  1. 2015.08.26 하이브리드 아키텍처
  2. 2010.02.09 Practice : jQuery DataTables

작심 3일로 끝나버릴까봐 더 늦기 전에 다음 진도를 나가봅니다.

구성도는 아래와 같습니다.

  • FrontEnd
    • 이전에 글에 썼던 것처럼 nodejs 기반으로 뷰에 대한 렌더링이 클라이언트에서 된다고 생각하시면 됩니다.
    • Ionic Framework는 AngularJS와 Sass기반으로 구현되어 있습니다. Ionic을 단순하게 말하면 UI Framework라고 생각하시면 됩니다. CSS기반으로 작성된 앱스럽게 보이는 컨트롤 및 자바스크립트를 제공해준다고 생각하시면 됩니다.
    • 모바일 웹에서는 모바일 디바이스 API(디바이스 정보, 카메라 기타 등등)에 접근하는 것이 어렵지만(불가능?) 하이드리드 앱에서는 이러한 부분을 Apache Cordova를 통해서 가능하게 해줍니다. Cordova 디바이스 API는 플러그인 형태로 제공되며 추가적인 플러그인(PushPlugin 등등)도 Github에 공개되어 있습니다.
    • ngCordova는 AngularJS 컨트롤러에서 Cordova 디바이스 API를 사용하기 쉽게 확장해둔 라이브러리라고 생각하시면 됩니다.
    • Ionic Material은 Google Material 디자인처럼 Ionic 컨트롤을 확장한 라이브러리라고 생각하시면 됩니다.
      참고 https://www.materialup.com/
    • FrontEnd는 atom으로 코딩을 하고 gulp로 빌드를 해서 개발을 진행하고 있습니다.
      참고 : http://programmingsummaries.tistory.com/356

  • BackEnd
    • 서버는 Web API로 json 데이터를 제공해줍니다. Nancy로 구현하였는데 Nginx에도 올릴 수도 있다고 해서 다른 프로젝트에서 적용을 해보았는데 특별한 문제는 없고 IoC부터 라이팅 규칙 작성도 쉽고 직렬화 등 라이브러리 교체와 같은 부분들이 쉽게 되어 있습니다.
    • 보통 .NET에서는 직렬화 라이브러리로 JSON.NET를 많이 사용하는데 ServiceStack.Text(4.0이상은 유료입니다;;)가 성능이 제일 괜찮다고 하여 사용해보았습니다. 이전에 말한 것처럼 Nancy에서는 라이브러리 교체가 쉽게 때문에 문제가 생기면 바로 교체를 하려고 했었는데 크게 문제는 없었습니다.(json 데이터 파일을 읽다가 공백과 관련된 버그는 있었습니다.) 아 또 한가지 CSV와 JSV 다양한 데이터 포맷을 제공합니다. 동일 주소를 호출을 하여 다양한 포맷으로 데이터를 받을 경우(Content Negotiation)에 활용하면 괜찮습니다.
    • PushSharp은 GCM, Apns 기타 등등과 관련된 푸쉬 서비스에 대한 프로바이더입니다. GCM에 쏘는건 실제로 Http Post방식으로 쏘면 되는거라 어렵지 않아 굳이 쓸 필요까지는 없었는데 저는 오픈 소스를 사랑하기 때문에 ^^; 소스도 볼겸 사용해보았습니다.
    • SignalR은 실시간(양방향) 통신이 필요할 경우에 사용하려고 고려만 해둔 상태입니다.
      참고 : 
      초록색으로 표시한 부분은 직접 구현한 부분들이 아닌 외부와의 연동을 의미합니다.

  • GCM(Apns)
    • 푸쉬 서비스를 사용하는 이유는 앱이 실행중이지 않더라도 알림을 받으려고 하는 것입니다. 디바이스에서 먼저 GCM Service에 등록을 해줘야 하며 등록하는 과정에서 Token를 받습니다. 이러한 Token은 사용자별로 푸쉬 프로바이더로 전송을 해줘서 저장을 해줘야 합니다. 푸쉬 프로바이더에서는 저장된 Token을 사용하여 메세지를 GCM Service로 보냅니다. GCM Service에서는 받은 메세지를 디바이스로 보냅니다.
    • PushSharp에서는 GCM, Apns에 대한 구현이 각각 되어 있었는데 보통 아이폰은 Apns를 안드로이드폰은 GCM를 사용합니다. GCM 3.0부터는 등록 프로세스를 간소화하여 아이폰, 안드로이드폰, 크롬까지 지원을 합니다.
    • GCM를 사용하더라도 Apns에 인증을 하는 절차를 거쳐야 합니다. 실제로 GCM Service에서 Apns에 인증만 받는 건지 아니면 중간 역할을 해주는 건지는 확실하지 않지만 등록 프로세스를 간소화를 통하여 GCM의 Topic과 Device Group를 사용하기 위함입니다. 관련 내용은 슬라이드를 보시면 아실 수 있을 겁니다~ ^^

아직까지도 보여줄 코딩한줄이 없네요 ㅎㅎ;; 블로그질을 오랫만에 하니 시간을 은근히 잡아먹네요 ㅜㅜ
다음에는 GCM 3.0의 등록 프로세스 간소화를 위한 iOS와 관련된 설정 및 코드로.........


Posted by resisa
,

이전 포스트에서 jQuery Table Plugin 중에 DataTables을 소개해드렸습니다. 이번 포스트에서는 DataTables를 사용하는 제가 알고 있는 가장 Best Practics한 사용법을 설명드리도록 하겠습니다.

먼저 필요한 기능들을 나열해보도록 하겠습니다.
 - 헤더를 클릭하여 열 정렬이 가능해야 한다.
 - Paging이 가능해야 한다. (1, 2, 3, 4 숫자로 표시되는 Paging)
 - 해당 Page의 보이는 Data만을 조회하여야 한다. (성능이 떨어지면 안된다.)
 - CSS를 통해서 사용자 정의 스타일이 가능해야 한다.

가장 중요한 기능은 Page를 나타내는 테이블을 구현하려고 하다 보니 성능적인 문제가 제일 문제가 되는데 이러한 것은 DataTables의 Server-Side를 호출해주는 방법으로 해결할 수 있습니다. DataTables의 Example을 살펴보면 Server-Side코드들이 php로 작성된 코드들이라 ASP.NET MVC에 맞는 코드를 Forum에서 찾아보았습니다. 여기서 한가지 알아야 할 사항이 있는데 Client에서 Sever쪽 액션 메소드(MVC)를 호출하고 액션 메소드의 결과를 Json타입으로 직렬화하여 리턴해줄 때의 Data형태를 DataTables이 알 수 있는 형태로 리턴을 해주어야 합니다. (물론 일반적인 List<T>타입을 Json타입으로 Serialize해준 후에 Client측에서 DataTables이 원하는 Data형태로 변경해줄 수도 있습니다.)

1. http://weblogs.asp.net/zowens/archive/2010/01/19/jquery-datatables-plugin-meets-c.aspx
2. http://datatables.net/forums/comments.php?DiscussionID=932&page=1#Item_0

첫 번째 방법으로 먼저 시도를 해보았는데 사용하는 방법이 쉽지만 'Caveat'부분을 보면 버그가 있다는 경고 메세지가 있고 실제로 시도해보았을 경우에 단순히 하나의 테이블에서 string의 타입만 있을 경우에는 문제가 없었지만 여러 가지 타입이나 테이블 조인의 경우에 문제가 있어 아쉽게 사용하지 못하게 되었습니다.

두 번째 방법을 살짝 Custom하게 고쳐서 사용해보도록 하겠습니다.

ORM은 Entity Framework를 사용하였으면 DB모델은 아래 그림과 같으며 Script파일을 첨부하였습니다.



Client-Side
var logTable;

 

$(function() {

 

    $("#logList tbody").click(function(event) {

        $(logTable.fnSettings().aoData).each(function() {

            $(this.nTr).removeClass('row_selected');

        });

        $(event.target.parentNode).addClass('row_selected');

    });

 

    logTable = $("#logList").dataTable({

        "bAutoWidth": false,

        "iDisplayLength": 10,

        "bLengthChange": false,

        "bInfo": false,

        "sPaginationType": "full_numbers",

        "bProcessing": true,

        "bServerSide": true,

        "aaSorting": [[0, "asc"]],

        "aoColumns": [

                        { "bVisible": true, "sClass": "Center" },

                        null,

                        null

                            ],

        "sAjaxSource": '<%= Url.Action("GetAllChildren", "Home") %>'

    });

});

 

<table id="logList" class="display">

    <thead>

        <tr>

            <th>P_ID</th>

            <th>P_Name</th>

            <th>C_COUNT</th>

        </tr>

    </thead>

    <tbody></tbody>

</table>

=> click()부분은 Table에서 해당 로우를 선택하였을 경우에 해당 로우만 다른 색깔로 표시하는 것이며 클릭하였을 경우에 액션을 취하고 싶을 경우에 사용하시면 됩니다. logTable이라는 변수를 선언해주는 이유는 DataTable의 sAjaxSource부분이 MVC의 액션 메소드를 호출해주는 부분인데 액션 메소드 호출시에 특정한 상황에 따라서 동적으로 변하는 매개변수를 넘길 경우가 있을 수 있습니다. 이럴경우에 logTable이라는 변수를 사용해서 sAjaxSource를 내용을 고치고 Table의 내용도 변경된 내용으로 Refresh하기 위함입니다. 기타 옵션들은 그 단어만 봐도 어떤 것인지 어느 정도는 알 수 있으며 자세한 사항은 DataTables 사이트를 참조하시기 바랍니다. DataTables를 다운 받아보시면 Demo_Table.css파일이 있는데 이 파일에서 스타일을 원하는 형태로 변경하셔서 사용하실 수 있습니다.

Server-Side Helper Class
public class GridParams

{

    public int iDisplayStart { get; set; }

    public int iDisplayLength { get; set; }

    public string sSearch { get; set; }

    public bool bEscapeRegex { get; set; }

    public int iColumns { get; set; }

    public int iSortingCols { get; set; }

    public int iSortCol_0 { get; set; }

    public string sSortDir_0 { get; set; }

    public int sEcho { get; set; }

    public bool bSortable_0 { get; set; }

    public bool bSearchable_0 { get; set; }

}

 

public class FormatedList

{

    public int sEcho { get; set; }

    public int iTotalRecords { get; set; }

    public int iTotalDisplayRecords { get; set; }

    public string[][] aaData { get; set; }

}

 

public static class QuerableExtensions

{

    public static IQueryable<T> OrderBy<T>(this IQueryable<T> source, string propertyName, bool asc)

    {

        var type = typeof(T);

        string methodName = asc ? "OrderBy" : "OrderByDescending";

        var property = type.GetProperty(propertyName);

        var parameter = Expression.Parameter(type, "p");

        var propertyAccess = Expression.MakeMemberAccess(parameter, property);

        var orderByExp = Expression.Lambda(propertyAccess, parameter);

        MethodCallExpression resultExp = Expression.Call(typeof(Queryable), methodName, new Type[] { type, property.PropertyType }, source.Expression, Expression.Quote(orderByExp));

        return source.Provider.CreateQuery<T>(resultExp);

    }

}

 

public static class GridsUtils

{

    public delegate string[] GetStringDelegate<T>(T obj);

 

    public static string[][] GetStrings<T>(this IQueryable<T> source, GetStringDelegate<T> del)

    {

        string[][] retArray = new string[source.Count()][];

        int i = 0;

        foreach (var s in source)

        {

            retArray.SetValue(del(s), i);

            i++;

        }

        return retArray;

    }

}

=> GridParams클래스는 열을 클릭하거나 페이지를 클릭하거나 조회 상자에서 조회하는 내용 등의 정보를 Server쪽에서 받기 쉽게 정의해놓은 클래스입니다. FormatList클래스는 위의에 말했듯이 DataTable이 인식할 수 있는 정보를 받기 위해 정의해놓은 클래스입니다. 참고적으로 iTotalDisplayRecord는 Page에 나타나는 숫자를 의미합니다. QuerableExtensions클래스는 정렬 시에 사용하기 위해서 확장해놓은 클래스인데 해당 클래스의 프로퍼티와 asc, desc인지 여부를 Expression으로 구현해놓았습니다. 그런데 헤더 정렬시에 꼭 프로퍼티 이름만으로 정렬을 하는 것이 아니라 자식 프로퍼티의 개수로도 소트할 수 있기 때문에 꼭 사용하지는 않습니다. GridsUtils클래스는 IQeuryable<T>형태로 반환된 결과물을 string[][]의 형태로 만들어주기 위한 Helper 클래스입니다.

Server-Side 액션 메소드
public ActionResult GetAllParent(GridParams gridParams)

{

    using (TestEntities Context = new TestEntities())

    {

        //data

        var query = Context.Parent.Include("Children").AsQueryable();

 

        int total = query.Count();

 

        //search whatever property you like

        if (!string.IsNullOrEmpty(gridParams.sSearch))

        {

            query = from x in query

                    where x.P_Name.Contains(gridParams.sSearch)

                    select x;

        }

 

        //sort

        if (gridParams.iSortingCols > 0)

        {

            var propertyName = "P_ID";

 

            switch (gridParams.iSortCol_0)

            {

                case 0: propertyName = "P_ID"; break;

                case 1: propertyName = "P_Name"; break;

            }

 

            query = query.OrderBy(propertyName, (gridParams.sSortDir_0 == "asc"));

        }

 

        //display

        int totaldisp = query.Count();

        if (gridParams.iDisplayLength > 0)

            query = query.Skip(gridParams.iDisplayStart).Take(gridParams.iDisplayLength);

 

        return Json(new FormatedList

        {

            sEcho = gridParams.sEcho,

            iTotalRecords = total,

            iTotalDisplayRecords = totaldisp,

            aaData = query.GetStrings(t => new string[]

            {

                t.P_ID.ToString(),

                t.P_Name,

                t.Children.Count.ToString()

            })

        });

    }

}

=> 여기서 제일 중요한 부분은 AsQueryable()로 IQueryable<T>로 리턴해주는 부분입니다. 필요한 기능목록에서 Sort와 Page시에 사용되는 매개변수(GridParams)와 함께 실질적인 쿼리만을 만들어주며 Json타입으로 Serialize해주는 부분에서 실질적인 쿼리가 DB에 질의 됩니다. 물론 그 전에 Count를 알기 위한 쿼리는 Count()를 호출해줄 때마다 질의됩니다.

그런데 아마 Page, Sort에 대해서 궁금증이 생기실 수 있습니다. 과연 얼마나 효율적으로 데이터를 가져올까요? 전체 데이터를 가져와서 asc, desc로 정렬한 이후에 가져오는 걸까요? 정답은 row_number()라는 SQL함수에 있습니다. (이전에 row_number() 대한 포스트가 있는데 참고하시기 바랍니다.)
SQL Server Profiler로 질의된 쿼리들을 살펴보도록 하겠습니다.

SELECT

[Project2].[P_ID] AS [P_ID],

[Project2].[P_Name] AS [P_Name],

[Project2].[C1] AS [C1],

[Project2].[C3] AS [C2],

[Project2].[C2] AS [C3],

[Project2].[C_ID] AS [C_ID],

[Project2].[C_Name] AS [C_Name],

[Project2].[P_ID1] AS [P_ID1]

FROM ( SELECT

       [Limit1].[P_ID] AS [P_ID],

       [Limit1].[P_Name] AS [P_Name],

       [Limit1].[C1] AS [C1],

       [Extent2].[C_ID] AS [C_ID],

       [Extent2].[C_Name] AS [C_Name],

       [Extent2].[P_ID] AS [P_ID1],

       CASE WHEN ([Extent2].[C_ID] IS NULL) THEN CAST(NULL AS int) ELSE 1 END AS [C2],

       CASE WHEN ([Extent2].[C_ID] IS NULL) THEN CAST(NULL AS int) ELSE 1 END AS [C3]

       FROM   (SELECT TOP (10) [Project1].[P_ID] AS [P_ID], [Project1].[P_Name] AS [P_Name], [Project1].[C1] AS [C1]

             FROM ( SELECT [Project1].[P_ID] AS [P_ID], [Project1].[P_Name] AS [P_Name], [Project1].[C1] AS [C1], row_number() OVER (ORDER BY [Project1].[P_ID] ASC) AS [row_number]

                    FROM ( SELECT

                           [Extent1].[P_ID] AS [P_ID],

                           [Extent1].[P_Name] AS [P_Name],

                           1 AS [C1]

                           FROM [dbo].[Parent] AS [Extent1]

                    )  AS [Project1]

             )  AS [Project1]

             WHERE [Project1].[row_number] > 0

             ORDER BY [Project1].[P_ID] ASC ) AS [Limit1]

       LEFT OUTER JOIN [dbo].[Children] AS [Extent2] ON [Limit1].[P_ID] = [Extent2].[P_ID]

)  AS [Project2]

ORDER BY [Project2].[P_ID] ASC, [Project2].[C3] ASC

=> 빨간색으로 표시된 부분을 보면 row_number()함수를 사용하였고 select절에는 TOP(10)과 where절에 row_number > 0이 눈에 들어옵니다. Page시에 사용하는 Skip()과 Take()가 쿼리로 변경되는 부분이 바로 이 부분입니다. Sort시에는 row_number()함수안에서 사용하는 Order By절과 제일 마지막에 보이는 Order By절에 의해서 Data가 정렬됩니다.
예를 들어 일반적인 게시판을 생각해보시면 보여지는 글의 건수를 10건이라고 하고 첫 번째 페이지를 보여준다면 쿼리는 TOP(10), row_number > 0이라는 조건절이 만들어져 질의됩니다. 만약에 세 번째 페이지를 보여준다면 TOP(10), row_number > 20이 조건절로 되는 것입니다. 마찬가지로 Order By는 해당 열을 처음 클릭하면 asc 두 번째로 클릭하면 desc으로 조건절이 만들어져 질의되는 것입니다. 실질적으로 row_number()가 어떻게 만들어져 있는지는 알 수 없지만 최소한 화면에 보이는 Data를 빠르게 가져온다는 사실만은 분명합니다. 또한 P_ID가 아닌 P_Name으로 정렬을 해보면 속도가 P_ID보다 느린 것을 알 수 있는데 이것은 row_number()가 인덱스와 상관이 있다는 것을 간접적으로 알 수 있습니다.
Posted by resisa
,