วันพุธที่ 3 พฤศจิกายน พ.ศ. 2553

สร้างเงื่อนไขในการแสดง field ใน form

ในการใช้  form ของ OFBiz หากมีเงื่อนไขในการแสดงผล field ต่างกัน ก็สามารถใช้ form เดียวกันได้ เพียงแต่ต้องกำหนดเงื่อนไขในการแสดงผลของ field นั้นให้ถูกต้อง
ในกรณีที่ต้องการแสดงผล field โดยใช้เงื่อนไขนี้ OFBiz จะเตรียมคำสั่ง use-when ไว้ให้ใช้ ดังนี้
<field name="USER_TITLE" use-when="roleTypeId==SUPPLIER||roleTypeId==CUSTOMER||roleTypeId==INTERNAL_ORGANIZATION" title="${uiLabelMap.CommonTitle}${roleTypeId}"><text size="10" maxlength="30"/></field>
<field name="USER_FIRST_NAME" use-when="roleTypeId!=SUPPLIER||roleTypeId!=CUSTOMER||roleTypeId!=INTERNAL_ORGANIZATION" title="${uiLabelMap.PartyFirstName}" tooltip="${uiLabelMap.CommonRequired}" widget-style="required"><text size="30" maxlength="60"/></field>
<field name="USER_MIDDLE_NAME" use-when="roleTypeId==SUPPLIER||roleTypeId==CUSTOMER||roleTypeId==INTERNAL_ORGANIZATION" title="${uiLabelMap.PartyMiddleInitial}"><text size="4" maxlength="4"/></field>

จากโค้ดสังเกตว่า field ชื่อ USER_TITLE นั้น จะแสดงก็ต่อเมื่อมี roleTypeId = SUPPLIER หรือ มี roleTypeId = CUSTOMER หรือ roleTypeId = INTERNAL_ORGANIZATION หากมี roleTypeId เป็นอื่นๆ ที่ไม่ใช่ SUPPLIER หรือ CUSTOMER หรือ INTERNAL_ORGANIZATION ก็จะไม่แสดง

การใช้คำสั่ง ignore-if-empty ใน drop down list

ในการคิวรี่ข้อมูลออกมาจาก entity บางครั้งมีข้อมูลที่เป็น empty ติดมาด้วย
หากไม่ต้องการนำมาแสดงใน drop down list ให้ใช้คำสั่ง ignore-if-empty ต่อท้ายการ select ข้อมูลออกมา


จะเห็นว่าข้อมูลในส่วนของ group name เป็นค่าว่าง อาจเนื่องมาจากการทดสอบการเพิ่มข้อมูลและการสร้าง services ต่างๆ ในขั้นตอนการพัฒนาที่ไม่ได้ถูกลบออกจากฐานข้อมูล หากต้องการซ่อนค่าดังกล่าว ให้ทำการเพิ่มโค้ดในส่วนของ entity-constrain ดังนี้

<field name="partyIdFrom" title="${uiLabelMap.PartyGroupDepartmentName}" tooltip="${uiLabelMap.CommonRequired}">
            <drop-down allow-empty="true" >
               <entity-options key-field-name="partyId" description="${groupName}" entity-name="PartyRoleDetailAndPartyDetail">
                <entity-constraint name="roleTypeId" operator="equals" value="DEPARTMENT"/>
                <entity-constraint name="groupName" operator="like" value="%" ignore-if-empty="true"/>
                    <entity-order-by field-name="groupName"/>
                </entity-options>
            </drop-down>
</field>

ในส่วนของการ entity-constrain มีการ filter เอาฟิล์ด groupname ที่มีค่าเป็น empty ออก โดยใช้ property ignore-if-empty
ผลลัพธ์ที่ได้จะเป็นดังนี้


ซึ่งจะไม่แสดงค่าว่างตามเงื่อนไขที่กำหนดไว้

วันจันทร์ที่ 1 พฤศจิกายน พ.ศ. 2553

การ convert ชนิดของข้อมูลใน simple method ของ OFBiz

ในบางกรณีที่ชนิดของข้อมูลที่รับมาจาก field ในฟอร์มเป็นคนละชนิดกับที่อยู่ใน entity เราสามารถใช้ simple method แปลงข้อมูลที่รับมาจาก field เพื่อเอาไปทำงานต่อไปได้ ดังนี้

<simple-method method-name="addProductionRunRoutingTask" short-description="Check parameters and add a production run task.">
        <call-map-processor in-map-name="parameters" out-map-name="context">
            <simple-map-processor name="prepareAddRoutingTask">
                <process field="productionRunId">
                    <copy/>
                </process>
                <process field="routingTaskId">
                    <copy/>
                    <not-empty>
                        <fail-property resource="ManufacturingUiLabels" property="ManufacturingRoutingTaskIdMissing"/>
                    </not-empty>
                </process>
                <process field="priority">
                    <copy/>
                    <not-empty>
                        <fail-property resource="ManufacturingUiLabels" property="ManufacturingProductionRunPriorityMissing"/>
                    </not-empty>
                    <convert type="Long">
                        <fail-property resource="ManufacturingUiLabels" property="ManufacturingRoutingSeqIdFormatNotCorrect"/>
                    </convert>
                </process>
                <process field="estimatedStartDate">
                    <copy/>
                    <convert type="Timestamp">
                        <fail-property resource="ManufacturingUiLabels" property="ManufacturingProductionRunStartDateNotCorrect"/>
                    </convert>
                </process>
                <process field="estimatedCompletionDate">
                    <copy/>
                    <convert type="Timestamp">
                        <fail-property resource="ManufacturingUiLabels" property="ManufacturingProductionRunCompletionDateNotCorrect"/>
                    </convert>
                </process>
                <process field="estimatedSetupMillis">
                    <copy/>
                    <convert type="BigDecimal">
                        <fail-property resource="ManufacturingUiLabels" property="ManufacturingProductionRunQuantityNotCorrect"/>
                    </convert>
                </process>
                <process field="estimatedMilliSeconds">
                    <copy/>
                    <convert type="BigDecimal">
                        <fail-property resource="ManufacturingUiLabels" property="ManufacturingProductionRunQuantityNotCorrect"/>
                    </convert>
                </process>
                <process field="workEffortName">
                    <copy/>
                </process>
                <process field="description">
                    <copy/>
                </process>
            </simple-map-processor>
        </call-map-processor>
        <check-errors/>
        <call-service service-name="addProductionRunRoutingTask" in-map-name="context">
        </call-service>
        
        <set field="workEffortAssocMap.workEffortIdFrom" from-field="parameters.workEffortIdFrom"/>
        <set field="workEffortAssocMap.workEffortIdTo" from-field="parameters.routingTaskId"/>
        <set field="workEffortAssocMap.workEffortAssocTypeId" value="WORK_EFF_BREAKDOWN"/>
        <set field="workEffortAssocMap.fromDate" from-field="parameters.estimatedStartDate"/>
        <call-service service-name="createWorkEffortAssoc" in-map-name="workEffortAssocMap"/>
        
        <property-to-field resource="DefaultMessages" property="service.default.message" field="successMessage"/>
    </simple-method>

จะเห็นว่าในส่วนของการรับค่าจากฟอร์ม จะมีการแปลงข้อมูลที่รับจาก field ดังนี้

<process field="priority">
                    <copy/>
                    <not-empty>
                        <fail-property resource="ManufacturingUiLabels" property="ManufacturingProductionRunPriorityMissing"/>
                    </not-empty>
                    <convert type="Long">
                        <fail-property resource="ManufacturingUiLabels" property="ManufacturingRoutingSeqIdFormatNotCorrect"/>
                    </convert>
</process> 

ในส่วนของ convert-type จะเป็นการแปลงข้อมูลจาก field ชื่อ priority ที่ได้รับมาจากฟอร์ม ให้กลายเป็นชนิด Long เพื่อนำไปใช้ในเซอร์วิส addProductionRunRoutingTask ต่อไป

วันพุธที่ 20 ตุลาคม พ.ศ. 2553

การเขียน beanshell เพื่อคำนวนค่าใน row-actions

ใน OFBiz เราสามารถคำนวนค่าใดๆ ในแต่ละแถวของการแสดงผลของ form ได้ โดยใช้คำสั่ง beanshell script ใน action หรือ row-actions ของ form ดังนี้

<row-actions>
            <set field="estimatedTotalMilliSeconds" value="${bsh:estimatedMilliSeconds * quantity}" type="BigDecimal"/>
</row-actions>

จากโค้ด ได้ทำการคำนวน โดยการนำค่าของ estimatedMilliSeconds มาทำการคูณกับ quantity แล้วเก็บไว้ในตัวแปร estimatedTotalMilliSeconds
โดยหากใช้คำสั่ง <row-actions> นั่นจะหมายถึง มีการคำนวนค่านี้ในทุกแถวที่มีการ process ค่านี้ออกมา

วันจันทร์ที่ 18 ตุลาคม พ.ศ. 2553

การเรียงลำดับฟิล์ดในฟอร์มของ OFBiz

ใน form ของ OFBiz แล้ว บางครั้งการวางตำแหน่งของโค้ดตามลำดับก่อนหลัง ไม่ใช่ลำดับของการแสดงผลก่อนหลังของฟอร์มแต่อย่างใด หากต้องการเรียงลำดับฟิล์ดให้ได้ตามต้องการแล้ว ควรเพิ่มโค้ดของ <sort-order></sort-order> ไว้ก่อนจะจบฟอร์มด้วย ดังนี้

<form default-map-name="partyGroup" default-table-style="basic-table" focus-field-name="groupName" header-row-style="header-row" name="EditDepartment" target="updatePartyGroup" type="single">
    <field name="groupName" title="${uiLabelMap.DepartmentName}" tooltip="${uiLabelMap.CommonRequired}" widget-style="required"></field>
    <field name="partyId" title="${uiLabelMap.Department}" tooltip="${uiLabelMap.CommonRequired}" widget-style="required"><text></text></field>
    <field name="description" title="${uiLabelMap.CommonDescription}"><text></text>
    <field name="annualRevenue" title="${uiLabelMap.iMAS_AnnualRevenue}"><text></text>
    <field name="numEmployees" title="${uiLabelMap.iMAS_NumberOfEmployees}"><text></text>
<sort-order>
        <sort-field name="groupName"/>
        <sort-field name="description"/>
        <sort-field name="numEmployees"/>
        <sort-field name="annualRevenue"/>
        <sort-field name="partyId"/>
</sort-order>
</form>

การเขียนในรูปแบบนี้ ถึงแม้ว่าจะเรียงลำดับฟิล์ดในฟอร์มดังนี้
1. groupName
2. partyId
3. description
4. annualRevenue
5. numEmployees

แต่เวลาแสดงผลในฟอร์ม จะแสดงผลตามลำดับที่บังคับไว้ใน sort-order ดังนี้
1. groupName
2. description
3. numEmployees
4. annualRevenue
5. partyId

วันพุธที่ 25 สิงหาคม พ.ศ. 2553

การเพิ่มข้อความ successMessage ให้แสดงข้อความแจ้งว่าทำงานสำเร็จแล้ว

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


ทำได้โดยการเพิ่มโค้ดดังนี้ ใน service ที่มีการทำงาน

<property-to-field resource="DefaultMessages" property="service.default.message" field="successMessage"/>

เช่นต้องการสร้างข้อความแสดงผลของการ update Project Phase จึงเพิ่มโค้ดในไฟล์ ProjectsServices.xml เป็นต้น

วันพฤหัสบดีที่ 19 สิงหาคม พ.ศ. 2553

การสร้าง List โดยใช้ Dojo

ในการสร้างฟอร์มเพื่อช่วยคัดกรองรายงานในโมดูล report ของ imas จะมีขั้นตอนดังบทความ การเพิ่ม filter ในการแสดงรายงานโดยใช้ dojo และ การทำ lookup ให้กับการเลือกดูฟอร์ม โดยใช้ dojo ดังนั้น ในบทความนี้จะแสดงเฉพาะในส่วนของการสร้าง List โดยใช้ Dojo มีขั้นตอนดังนี้

1. ใน form ทำการสร้าง drop down list ดังนี้

<select id="ProductTypeInventory" name="ProductTypeInventory" dojoType="dijit.form.ComboBox" autocomplete="false" >
                        <option value="">${uiLabelMap.CommonSelect}</option>
                            <#list productTypeList as productTypeListMap>
                                <option value="${productTypeListMap.productTypeId}">${productTypeListMap.description}</option>
                             </#list>
</select>

2. สร้าง screen ดังนี้

<screen name="ProductSummaryDialog">
        <section>
            <actions>
                <property-map resource="iMAS-ReportUiLabels" map-name="uiLabelMap" global="true"/>
                <property-map resource="iMAS-AccountingUiLabels" map-name="uiLabelMap" global="true"/>
                <property-map resource="CommonUiLabels" map-name="uiLabelMap" global="true"/>
                <property-map resource="iMAS-InventoryUiLabels" map-name="uiLabelMap" global="true"/>
                <set field="headerItem" value="ProductSummaryDialog"/>
                <entity-condition list="productTypeList" entity-name="ProductType">
                </entity-condition>
            </actions>
            <widgets>
                    <platform-specific>
                        <html>
                            <html-template location="component://imas-report/webapp/imas-report/report/inventory/page/ProductSummaryDialog.ftl"/>
                        </html>
                    </platform-specific>
            </widgets>
        </section>
    </screen>

อธิบายโค้ด
กำหนดว่าจะดึงข้อมูลจาก entity ไหน โดยใช้คำสั่ง
<entity-condition list="productTypeList" entity-name="ProductType"></entity-condition>
จากตัวอย่าง จะเป็นการเลือกข้อมูลจาก entity ชื่อ ProductType

จะได้ฟอร์มที่สามารถเลือกข้อมูลจาก drop down list ของ ProductType ได้ดังนี้


การทำ lookup ให้กับการเลือกดูฟอร์ม โดยใช้ dojo

จากบทความ การเพิ่ม filter ในการแสดงรายงานโดยใช้ dojo ก่อนหน้านี้ ได้พูดถึงการอำนวยความสะดวกให้แก่ผู้ใช้งานโปรแกรม โดยการใช้ calendar ของ dojo ในการเลือกวันที่ ส่วนในบทความนี้จะพูดถึงการใช้ dojo ดึงเอา Lookup ของ OFBiz มาแสดงเป็น popup แบบ ajax ที่มีลักษณะดังรูป


มีขั้นตอนดังนี้

1. เพิ่มฟิลด์ที่ลงในไฟล์ ARAccruedInvoiceDialog.ftl ดังนี้

<input name="partyIdFrom" size="25" id="findCommEvents_partyIdFrom" type="text">
            <a href="javascript:call_fieldlookup2(document.ARCardForm.partyIdFrom,'LookupPartyName');">
                <img src="/images/fieldlookup.gif" alt="Lookup" border="0" height="14" width="15">
            </a>

อธิบายโค้ด
สร้าง input ชนิด textbox เพิ่มเข้าไปในฟอร์ม และสร้าง link โดยเรียกใช้ function ของ java script ชื่อ call_fieldlookup2 ซึ่งต้องส่ง parameter ดังนี้

call_fieldlookup2(target, viewName)

2. ทำการ include controller ของ module ที่มี lookup เข้าไปใน controller ของ component ที่ทำงานอยู่ เช่น จะเรียกใช้ lookup ที่มีการเรียก party id ใน component report ต้องทำการ include controller ของ party มา ดังนี้

<include location="component://imas-partymgr/webapp/partymgr/WEB-INF/controller.xml"/>

ผลที่ได้ จะสามารถใช้งาน lookup ได้

วันพุธที่ 18 สิงหาคม พ.ศ. 2553

การเพิ่ม filter ในการแสดงรายงานโดยใช้ dojo

โดยปกติการเรียกดู report ของ OFBiz จะมีข้อมูลที่จะต้องแสดงในปริมาณมาก จึงควรมีส่วนที่ให้ผู้ใช้งานสามารถสามารถเลือกดูได้ว่า จะดู report ของส่วนไหน และช่วงเวลาไหน

เพื่อให้ง่ายต่อการใช้งานของผู้ใช้ Imas จึงเลือกที่จะนำเอา Dojo มาสร้าง User Interface ซึ่งจะได้ form ในรูปแบบนี้


จาก form นี้ ผู้ใช้งานสามารถเลือกดู report ได้ว่า ต้องการช่วงไหนบ้าง

การนำเอา dojo มาช่วยใน user interface ในส่วนของ form มีขั้นตอนดังนี้

1. ค้นหารายงานที่ต้องการสร้าง dojo ui เช่น ในส่วนของ Payable - Recievable (เจ้าหนี้-ลูกหนี้) ต้องการสร้าง form แบบ calendar เพื่อให้ผู้ใช้สามารถเลือกช่วงเวลาที่ต้องการแสดงรายงานได้ โดย list ของรายงานเจ้าหนี้-ลูกหนี้ แสดงตาม url นี้
https://localhost:8443/imas-report/control/fourth

2. สร้าง form ของ calendar โดยการสร้างไฟล์ชื่อ ARAccruedInvoiceDialog.ftl ลงในโฟลเดอร์ ofbiz.9.04/hot-deploy/imas-report/webapp/imas-report/report/receivable/page ดังนี้

<form id="ARAccruedInvoiceForm" name="ARAccruedInvoiceForm" dojoType="dijit.form.Form" target="_blank" action="/imas-report/control/ARAccruedInvoice?externalLoginKey=${externalLoginKey}" method="POST">
    <table cellspacing="0">
      <tr>
        <td>${uiLabelMap.CommonFromDate}:</td>
        <td>
            <input id="ARAccruedInvoicefromDate" type="text" name="fromDate" dojoType="dijit.form.DateTextBox" constraints={datePattern:'dd-MM-yyyy'}
            onChange="dijit.byId('ARAccruedInvoicetoDate').constraints.min = arguments[0];"
            />
        </td>
        <td>${uiLabelMap.CommonThruDate}</td>
        <td>
            <input id="ARAccruedInvoicetoDate" type="text" name="toDate" dojoType="dijit.form.DateTextBox" constraints={datePattern:'dd-MM-yyyy'}
            onChange="dijit.byId('ARAccruedInvoicefromDate').constraints.max = arguments[0];"
            />
        </td>
      </tr>
      <tr>
      <td colspan="3">&nbsp;</td>
      </tr>
      <tr>
        <td>&nbsp;</td>
        <td colspan="3">
            <div align="center">
                <button id="ARAccruedInvoicesubmit" name="ARAccruedInvoicesubmitButton" dojoType="dijit.form.Button" type="submit" iconClass="dijitEditorIcon dijitEditorIconSave" onClick="dijit.byId('ARAccruedInvoiceDialog').hide();">${uiLabelMap.CommonSubmit}</button>
                <button id="ARAccruedInvoicecancel" name="cancelButton" dojoType="dijit.form.Button" type="button" iconClass="dijitEditorIcon dijitEditorIconCancel" onClick="javascript:dijit.byId('ARAccruedInvoiceDialog').hide();">${uiLabelMap.CommonCancelled}</button>
            </div>
        </td>
     </tr>
    </table>
</form>

อธิบายโค้ด
ไฟล์ ARAccruedInvoiceDialog.ftl นี้จะสร้างฟอร์มที่มี id = ARAccruedInvoiceForm และ name = ARAccruedInvoiceForm และมี action ไปที่ uri-map ชื่อ ARAccruedInvoice

ทำการสร้าง text field ที่มี id = ARAccruedInvoicefromDate โดยมี dojoType="dijit.form.DateTextBox" เพราะต้องการแสดงผลเป็น calendar ให้ผู้ใช้เลือกวันที่ได้ โดยกำหนด datePattern เป็น dd-MM-yyyy และส่งค่าที่เลือกนี้ไปกับ form

ทำการสร้าง text field ที่มี id = ARAccruedInvoicetoDate โดยมี dojoType="dijit.form.DateTextBox" เพราะต้องการแสดงผลเป็น calendar ให้ผู้ใช้เลือกวันที่ได้ โดยกำหนด datePattern เป็น dd-MM-yyyy และส่งค่าที่เลือกนี้ไปกับ form เช่นกัน

สร้างปุ่มที่มี id="ARAccruedInvoicesubmit" และ name="ARAccruedInvoicesubmitButton" เมื่อมีการกดปุ่มนี้ จะทำการซ่อน form ที่มี id = ARAccruedInvoiceDialog ในไฟล์ iMAS-ReportScreens.xml

สร้างปุ่มที่มี id="ARAccruedInvoicecancel" และ name="cancelButton" เมื่อมีการกดปุ่มนี้ จะทำการซ่อน form ที่มี id = ARAccruedInvoiceDialog ในไฟล์ iMAS-ReportScreens.xml

3. ที่ controller สร้าง request-map และ view-map ดังนี้
<request-map uri="ARAccruedInvoiceDialog"><security https="true" auth="true"/><response name="success" type="view" value="ARAccruedInvoiceDialog"/></request-map>
<view-map name="ARAccruedInvoiceDialog" type="screen" page="component://imas-report/widget/iMAS-ReportScreens.xml#ARAccruedInvoiceDialog"/>

อธิบายโค้ด
 สร้าง request-map ชื่อ ARAccruedInvoiceDialog ที่มีการเรียกใช้ view-map ชื่อ ARAccruedInvoiceDialog ที่มีการเรียกใช้ screen ชื่อ ARAccruedInvoiceDialog ในไฟล์ imas-report/widget/iMAS-ReportScreens.xml มาแสดงผล

4. ที่ imas-report/widget/iMAS-ReportScreens.xml สร้าง screen ชื่อ ARAccruedInvoiceDialog ดังนี้
<screen name="ARAccruedInvoiceDialog">
        <section>
            <actions>
                <property-map resource="iMAS-ReportUiLabels" map-name="uiLabelMap" global="true"/>
                <property-map resource="iMAS-PurchaseUiLabels" map-name="uiLabelMap" global="true"/>
                <property-map resource="CommonUiLabels" map-name="uiLabelMap" global="true"/>
                <set field="headerItem" value="ARAccruedInvoiceDialog"/>
            </actions>
            <widgets>
                    <platform-specific>
                        <html>
                            <html-template location="component://imas-report/webapp/imas-report/report/receivable/page/ARAccruedInvoiceDialog.ftl"/>
                        </html>
                    </platform-specific>
            </widgets>
        </section>
    </screen>
</screens>

อธิบายโค้ด
สร้าง screen ชื่อ ARAccruedInvoiceDialog ที่ทำการ include ไฟล์ ftl ชื่อ ARAccruedInvoiceDialog.ftl มาแสดงผล

5. แก้ไข link ที่ fourth.ftl เพื่อให้เรียกใช้ screen ชื่อ ARAccruedInvoiceForm ได้
<div class="browsecategorytext"><a href="#" id="ARAccruedInvoiceButton" onclick="javascript:dijit.byId('ARAccruedInvoiceDialog').show();">${uiLabelMap.ARAccruedInvoice} FF</button></div>
<div dojoType="dijit.Dialog" id="ARAccruedInvoiceDialog"  title="${uiLabelMap.ARAccruedInvoice}" href="/imas-report/control/ARAccruedInvoiceDialog?formTarget=ARAccruedInvoice&amp;dialogName=ARAccruedInvoiceDialog" ></div>

อธิบายโค้ด
เมื่อมีการกดลิ้ง จะมีการเรียก ARAccruedInvoiceDialog มาแสดงผล

ในบทความต่อไปจะพูดถึงเรื่องการทำ lookup ให้กับการเลือกดูฟอร์ม โดยใช้ dojo

วันพุธที่ 28 กรกฎาคม พ.ศ. 2553

การใส่ paging ให้ list form

ในไฟล์ที่มีการแสดงรายการ จำเป็นต้องใส่ paging ให้กับ list form นั้น เนื่องจากหากมีข้อมูลในปริมาณมาก จะทำให้การโหลดหน้านั้นมาแสดงผลช้า สามารถใส่ paging ให้ list form ได้ดังนี้

1. ใน form ที่มีการ list รายการทั้งหมดมาแสดง มีโค้ดดังนี้

<form name="listLookupWorkEffort" list-name="listIt" target="" title="" type="list" paginate-target="LookupWorkEffort" odd-row-style="alternate-row" default-table-style="basic-table hover-bar">

อธิบายโค้ด
ใส่ paginate-target="LookupWorkEffort" เพื่อแสดงถึงการแบ่ง list จาก form LookupWorkEffort ออกมาเป็นส่วนๆ

2. ใน screen ที่มีการเรียก list form นี้มาแสดงผล หากต้องการกำหนดจำนวนรายการที่แสดงในแต่ละหน้า ให้เขียนโค้ดเพิ่มในส่วนของ &lt;actions&gt;&lt;/actions&gt; ดังนี้

<set field="viewSize" from-field="parameters.VIEW_SIZE" type="Integer" default-value="20"/>

อธิบายโค้ด
default-value คือ จำนวนข้อมูลที่ต้องการให้แสดงในแต่ละหน้า

การส่ง parameter จาก field เป็น hidden field ไปให้ form และ screen

กรณีศึกษา
ในการค้นหาโครงการของโมดูล project management จะมี lookup เพื่อให้ผู้ใช้สามารถเลือก project ที่ต้องการค้นหาได้ แต่เมื่อกด search แล้ว การค้นหาเจอทั้งข้อมูลที่มี workEffortTypeId เป็น PROJECT, TASK, EVENT, PHASE และอื่นๆ ด้วย ทำให้มีข้อมูลมาแสดงเยอะมาก ดังนั้นจึงต้องการกรองเฉพาะข้อมูลที่มี workEffortTypeId เป็น PROJECT เท่านั้น สามารถทำได้ดังนี้

1. ที่ screen ในส่วนของ actions ให้เพิ่มโค้ด

<set field="workEffortTypeId" from-field="parameters.workEffortTypeId" default-value="PROJECT" />

อธิบายโค้ด
ให้ field ชื่อ workEffortTypeId ส่งค่า workEffortTypeId ที่ได้มาจาก parameter โดยที่ตัวแปร workEffortTypeId นี้ ถูกกำหนดค่าไว้ให้มีค่าเท่ากับ PROJECT (default-value="PROJECT")

2. ที่ form ให้เพิ่ม value และส่งค่าไปพร้อมกับ field ชื่อ workEffortTypeId ดังนี้

<field name="workEffortTypeId"><hidden value="${parameters.workEffortTypeId}"/></field>

ใน lookup ก็จะสามารถกรองเฉพาะข้อมูลที่มี workEffortTypeId ที่มีค่าเป็น PROJECT มาแสดงได้

3. ที่ lookup ในหน้า  ProjectForms.xml ให้เพิ่มโค้ด
<lookup target-form-name="LookupWorkEffort?workEffortTypeId=PROJECT">

ใส่ paremeter workEffortTypeId=PROJECT เป็น query string ไปในโค้ด เพื่อให้ส่งค่าเข้าไปค้นหาเฉพาะ workEffortTypeId=PROJECT

วันอังคารที่ 27 กรกฎาคม พ.ศ. 2553

การสร้าง screenlet เพื่อสร้างข้อความระหว่างการสร้างข้อมูลใหม่และการแก้ไขข้อมูลในฟอร์ม

เมื่อมีการเรียกใช้ form เดียวกันจาก 2 actions คือ มีการสร้างข้อมูลใหม่ผ่าน form และแก้ไขข้อมูลเดิมผ่าน form หากไม่มีการแก้ไข ในส่วนของ labels จะไปดึง uiLabelMap เดียวกันมาแสดงผล ซึ่งก่อให้เกิดความสับสนระหว่างการสร้างและการแก้ไขข้อมูล สามารถแก้ปัญหาได้ดังนี้

1. เพิ่มโค้ดลงใน screen ดังนี้

<container style="screenlet-title-bar">
  <container style="h3">
   <section>
    <condition>
     <if-empty field="projectId"/>
    </condition>
    <widgets>
     <label style="h3" text="${uiLabelMap.PageTitleEditProject}"/>
    </widgets>
    <fail-widgets>
     <label style="h3" text="${uiLabelMap.iMAS_ProjectMgrEditProject}"/>
    </fail-widgets>
    </section>
  </container>
 </container>

อธิบายโค้ด
ในส่วน section ของ titlebar ให้กำหนดเงื่อนไขว่า เมื่อไม่มีค่า projectId ส่งมา ให้เรียกใช้ widget ที่มี label ที่เรียกใช้ uiLabelMap ชื่อ PageTitleEditProject
แต่เมื่อมีค่า projectId ส่งมาด้วย ให้เรียกใช้ widget ที่มี label ที่เรียกใช้ uiLabelMap ชื่อ iMAS_ProjectMgrEditProject

2. ในส่วน screen ที่มีการเรียกใช้ form ให้เพิ่มโค้ดดังนี้

<container style="screenlet-body">                       
   <section>
    <widgets>
     <platform-specific>
      <html><html-template location="component://imas-project/webapp/imas-project/page/NewCustomerButton.ftl"/></html>
                                        </platform-specific>
     <include-form name="EditProject" location="component://imas-project/widget/forms/ProjectForms.xml"/>
    </widgets>
    <fail-widgets>
     <include-form name="EditProject" location="component://imas-project/widget/forms/ProjectForms.xml"/>
    </fail-widgets>
   </section>
 </container>

อธิบายโค้ด
จากข้อ 1 ในส่วนของ condition ถ้ามีการส่งค่า projectId จะเรียกใช้ widget ที่เรียกไฟล์ NewCustomerButton.ftl มาแสดงผลพร้อมกับ form ชื่อ EditProject
แต่ในทางกลับกัน ถ้าไม่มีการส่งค่า projectId ก็จะเรียก form ชื่อ EditProject มาแสดงผลเพียงอย่างเดียว

การสร้างไฟล์ diff เพื่อทำ patch

บางครั้งในการทำงานในโปรเจ็คท์ต้องมีการแก้ไขไฟล์หลักของ OFBiz โดยเฉพาะไฟล์ในโฟลเดอร์หลัก คือ applications, framework และ specialpurpose ซึ่งเมื่อแก้ไขแล้วหากทำการ commit ขึ้นไปที่ svn หลักของ OFBiz จะเกิดความสับสนกับ framework หลัก ดังนั้นจึงต้องทำไฟล์ .diff เพื่อให้เฉพาะโปรเจ็คท์ที่เราทำเท่านั้นได้ทราบถึงความเปลี่ยนแปลงของไฟล์ สามารถทำไฟล์ .diff ได้ดังนี้

1. ใช้คำสั่งดังนี้ใน terminal

svn diff applications >>/home/toonztudio/Documents/imas_28072010_toon.diff

อธิบายคำสั่ง
ใช้คำสั่ง svn diff เพื่อความแตกต่างของไฟล์ในโฟลเดอร์ applications แล้วสร้างไว้เป็นไฟล์ชื่อ imas_28072010.diff แล้วเก็บไว้ที่ /home/toonztudio/Documents

2. ทำการ copy ไฟล์ชื่อ imas_28072010.diff จาก /home/toonztudio/Documents มาเก็บไว้ใน โฟลเดอร์ patches ของโปรเจ็คท์ ในกรณีนี้คือ /home/toonztudio/Desktop/projects/imaserp/trunk/patches

3. ใช้คำสั่ง svn add ไฟล์ .diff

4. ใช้คำสั่ง svn commit ไฟล์ .diff เข้าสู่ repositories ต่อไป

วันพฤหัสบดีที่ 22 กรกฎาคม พ.ศ. 2553

การแยก column ในหัวตารางของ form

โดยปกติแล้วการแสดงข้อความในตารางควรจะมีหัวตาราง เพื่อแสดงข้อมูลว่าเป็นข้อมูลจากตารางไหน ซึ่งบางครั้งใน OFBiz จะแสดงหัวตารางในลักษณะติดกัน ทำให้ไม่สามารถทราบได้ว่า เป็นข้อมูลจากตารางไหน ดังรูป


ต้องทำการใส่แท็กเพื่อให้แสดงผลเป็นตาราง ดังนี้

separate-columns="true"

จะได้ตารางที่แสดงผลหัวตารางสมบูรณ์

วันจันทร์ที่ 12 กรกฎาคม พ.ศ. 2553

การเซ็ต trackpoint ของ thinkpad ใน Ubuntu

โดยปกติแล้วใน windows จะมี software ชื่อ UltraNav สำหรับควบคุมการทำงานของ trackpoint ทำให้ผู้ใช้ thinkpad บน windows สามารถควบคุม trackpoint ได้ แต่ใน ubuntu ต้องทำการแก้ไขไฟล์บางไฟล์เล็กน้อย เพื่อให้สามารถใช้ความสามารถของ trackpoint อย่างเต็มที่ได้ มีขั้นตอนดังนี้

1. สร้างไฟล์ /usr/lib/X11/xorg.conf.d/20-thinkpad.conf โดยใช้คำสั่ง
sudo gedit /usr/lib/X11/xorg.conf.d/20-thinkpad.conf

2. เพิ่มข้อความดังนี้ลงไปในไฟล์
Section "InputClass"
    Identifier "Trackpoint Wheel Emulation"
    MatchProduct "TrackPoint"
    MatchDevicePath "/dev/input/event*"
    Driver "evdev"
    Option "EmulateWheel" "true"
    Option "EmulateWheelButton" "2"
    Option "Emulate3Buttons" "false"
    Option "XAxisMapping" "6 7"
    Option "YAxisMapping" "4 5"
EndSection

3. save ไฟล์ และทำการ restart โดยใช้คำสั่ง
sudo /etc/init.d/gdm restart

4. จะสามารถใช้ trackpoint ร่วมกับปุ่มกลางของ touchpad ได้

reference:
http://psung.blogspot.com
http://thaipats.buddythai.com

วันพุธที่ 7 กรกฎาคม พ.ศ. 2553

การเปลี่ยนค่า screen resolution ของ ubuntu

ในการ install ubuntu บางครั้ง resolution ใน monitor ได้ค่าไม่ตรงกับค่าที่เคยเซ็ตใน windows ทำให้จอไม่ชัดหรือเบลอ สามารถแก้ไขได้ตามวิธีการดังนี้

1. เปิด terminal พิมพ์คำสั่ง

xrandr

2. จะได้ข้อความดังนี้
Screen 0: minimum 320 x 200, current 2464 x 900, maximum 8192 x 8192
VGA1 connected 1440x900+1024+0 (normal left inverted right x axis y axis) 0mm x 0mm
   1360x768       59.8  *
   1024x768       60.0  
   800x600        60.3     56.2  
   848x480        60.0  
   640x480        59.9     59.9  
LVDS1 connected 1024x768+0+132 (normal left inverted right x axis y axis) 304mm x 228mm
   1024x768       60.0*+   85.0     75.0     70.1     60.0*    50.0  
   832x624        74.6  
   800x600        85.1     72.2     75.0     60.3     56.2  
   640x480        85.0     72.8     75.0     60.0     59.9  
   720x400        85.0  
   640x400        85.1  
   640x350        85.1  
TV1 disconnected (normal left inverted right x axis y axis)

* หมายเหตุ VGA1 คือ Acer monitor 17" ที่ต่อเพิ่ม ส่วน LVDS1 คือ monitor ของ Thinkpad R61

3. สิ่งที่ต้องการคือ ต้องการเพิ่มความละเอียด 1440x900 ในมอนิเตอร์ VGA1 ให้ทำการเช็ค modeline โดยใช้คำสั่ง

cvt 1440 900

*หมายเหตุ 1440 900 คือความละเอียด 1440x900 พิกเซล

4. จะแสดงผลดังนี้

# 1440x900 59.89 Hz (CVT 1.30MA) hsync: 55.93 kHz; pclk: 106.50 MHz
Modeline "1440x900_60.00"  106.50  1440 1528 1672 1904  900 903 909 934 -hsync +vsync

โดยเราจะใช้โค้ดที่ต่อจาก modeline คือ
"1440x900_60.00"  106.50  1440 1528 1672 1904  900 903 909 934 -hsync +vsync
ในการเซ็ตค่าที่ไฟล์ /etc/gdm/Init/Default

5. เปิดไฟล์ /etc/gdm/Init/Default โดยใช้คำสั่ง

sudo gedit /etc/gdm/Init/Default

6. ค้นหาบรรทัดดังนี้
PATH=/usr/bin:$PATH
OLD_IFS=$IFS

7. เพิ่มโค้ดต่อท้ายดังนี้
xrandr --newmode "1440x900_60.00"  106.50  1440 1528 1672 1904  900 903 909 934 -hsync +vsync
xrandr --addmode VGA1 1440x900_60.00
xrandr --output VGA1 --mode 1440x900_60.00
โดยค่าที่ใส่ในบรรทัด newmode คือค่าที่ได้จากการเช็ค modeline

8. restart ubuntu และเช็คค่าที่ System > Preference > Monitor จะพบ resolution ใหม่ที่เพิ่มเข้าไปแล้ว

Reference: http://www.ubuntugeek.com/how-change-display-resolution-settings-using-xrandr.html

การแก้ปัญหาเมื่อ Eclipse หา JVM ไม่เจอ

เมื่อทำการถอน jdk ตัวเดิมออกแล้วติดตั้งใหม่ Eclipse มักจะหา path ของ JAVA_HOME ไม่เจอ และจะแจ้งเตือนว่า

A Java Runtime Environment (JRE) or Java Development Kit (JDK)
must be available in order to run Eclipse. No Java virtual machine
was found after searching the following locations:
/home/toonztudio/Downloads/eclipse_reporting/jre/bin/java
java in your current PATH

หากยังไม่ได้เซ็ท JAVA_HOME ให้ทำการเซ็ตตามนี้
การ เซ็ต JAVA_HOME และ CLASSPATH บน Ubuntu

หากทำการเซ็ตแล้ว ให้เปิด Eclipse จะเจอ Error ด้านบน ให้แก้ไขไฟล์ eclipse.ini
โดยเพิ่มบรรทัด

-vm
/home/toonztudio/jdk1.6.0_20/bin

โดย /home/toonztudio/jdk1.6.0_20/bin คือพาธที่ทำการ install jdk ไว้

เปิด eclipse อีกครั้ง จะไม่เจอข้อความแสดง error

reference: http://wiki.eclipse.org/Eclipse.ini#Linux_Example

วันจันทร์ที่ 5 กรกฎาคม พ.ศ. 2553

การเซ็ต JAVA_HOME และ CLASSPATH บน Ubuntu

1. เปิด terminal พิมพ์ดังนี้
sudo gedit /etc/bash.bashrc

2. ในไฟล์ bash.bashrc เพิ่มคำสั่งดังนี้
export JAVA_HOME="/usr/lib/jvm/java-6-sun"
export JRE_HOME="/usr/lib/jvm/java-6-sun/jre"
export CLASSPATH="/usr/lib/jvm/java-6-sun/lib/tools.jar"
export PATH="/usr/lib/jvm/java-6-sun/bin:$PATH"

สามารถเช็คการตั้งค่าได้จาก terminal
พิมพ์ echo $JAVA_HOME เพื่อเช็คพาธ JAVA_HOME
พิมพ์ echo $JRE_HOME เพื่อเช็คพาธ JRE_HOME
พิมพ์ echo $CLASSPATH เพื่อเช็คพาธ CLASSPATH
พิมพ์ echo $PATH เพื่อเช็ค PATH

การเพิ่ม auto complete ให้กับ Eclipse

ในกรณีที่ Eclipse ไม่รู้จักคำสั่งใน namespace จะไม่สามารถสร้าง auto complete ได้ สามารถทำการเพิ่ม namespace ได้ ดังนี้

Windows -> Preference -> XML -> XML Catalog

ทำการเพิ่ม namespace เช่น http://ofbiz.apache.org/dtds/widget-screen.xsd
แล้วกด OK

วันอาทิตย์ที่ 4 กรกฎาคม พ.ศ. 2553

การเพิ่ม confirm dialog box ใน form ของ OFBiz 9.04

ใน OFBiz 10.04 การทำ Confirm Dialog Box สามารถใช้ confirmation-message ใน button ได้โดยตรง ดังนี้

<field name="searchButton" title="Search" widget-style="smallSubmit"><submit button-type="button" confirmation-message="Test"/></field>

แต่ใน OFBiz 9.04 ไม่สามารถใช้ confirmation-message ได้ จึงได้ทำการทดลองหาวิธีต่างๆ มาใช้ ดังนี้

ทดลองสร้าง javascript แล้วนำไปใช้กับ onClick ของปุ่มของฟอร์มใน OFBiz ดังนี้
<input type="submit" name="Submit" value="Submit" onClick="fncSubmit()" />

พบว่าไม่สามารถทำได้ เนื่องจาก OFBiz จะทำการ generate แท็ก onSubmit ให้โดยอัติโนมัติดังนี้

onSubmit="javascript:submitFormDisableSubmits(this)"

เมื่อเขียน javascript ที่ปุ่ม submit ของ form ทำให้เกิดการ submit ไปยัง target ของ form ถึงแม้ว่าจะเป็นการกดปุ่ม cancel

ได้รับคำแนะนำจากพี่แซนด์ให้ทำการสร้างปุ่มของฟอร์มในไฟล์ ftl แล้ว include เข้ามาใช้แทนปุ่มของ form โดยตรง จึงได้ทำการสร้างไฟล์ดังนี้

1. สร้างไฟล์ submitButton.ftl โดยมีรายละเอียดดังนี้ (ใช้ dojo มาช่วยในการเขียน javascript ศึกษาเพิ่มเติมได้จาก http://www.dojotoolkit.org/)

<button dojoType="dijit.form.Button" id="btn">Submit
    <script type="dojo/method" event="onClick">showDlg1("FindProject")</script>
</button>

2. ทำการ comment ปุ่มของ form เดิม เพื่อไม่ให้ปุ่มเดิมสามารถทำงานได้
3. เขียนฟังก์ชั่น javascript ในไฟล์ชื่อ testDialog.js ดังนี้

dojo.addOnLoad(function(){
 dojo.parser.parse();
});

function removeThis(){
 this.destroyRecursive()
}

function showDlg1(formIdName){
 var d = new dijit.Dialog({id:'dlg',title:'Dojo Dialogs',content: "<h2>Do you need to do this thing ?</h2>" +
        "<button dojoType='dijit.form.Button' type='submit' id='buttonalertyes' iconClass='mailIconOk' onClick='document." + formIdName + ".submit()'>Yes</button><button dojoType='dijit.form.Button' type='submit' id='buttonalertno' iconClass='mailIconCancel'>No</button>",onExecute:removeThis,onCancel:removeThis});
 d.show()
}

4. ที่ form ทำการ include ไฟล์ submitButton.ftl เข้าไปดังนี้
<platform-specific>
    <html><html-template location="component://imas-project/webapp/imas-project/project/submitButton.ftl"/></html>
</platform-specific>

5. ทำการเพิ่มไฟล์ testDialog.js ใน framework/common/widget/CommonScreens.xml เพื่อให้ทุกเพจสามารถเรียกใช้ไฟล์ javscript นี้ได้ ดังนี้
<set field="layoutSettings.javaScripts[+0]" value="/dojotoolkit/imas/testDialog.js" global="true"/>

6.เมื่อทดลองกดปุ่ม จะแสดง dialog box ดังนี้

วันพุธที่ 9 มิถุนายน พ.ศ. 2553

การ สร้าง service เพื่อ edit ข้อมูลในฐานข้อมูลโดยใช้ simple method

ในบทความนี้จะแสดงการสร้าง service ที่ใช้แก้ไขข้อมูลใน entity โดยใช้ simple method ในไฟล์ *Services.xml มีวิธีทำดังนี้

1. สร้างฟอร์มชื่อ EditBlog ที่ *Forms.xml
<form name="EditBlog" type="single" target="editBlog" title="" 
            default-map-name="resultMap" default-entity-name="NewBlog">
        <auto-fields-entity entity-name="NewBlog" default-field-type="edit"/>
        <field name="statusId">
            <drop-down allow-empty="false">
                <entity-options description="${description}" entity-name="StatusItem" key-field-name="statusId">
                <entity-constraint name="statusTypeId" value="CONTENT_STATUS"/>
                </entity-options>
            </drop-down>
        </field>
        <field name="editButton" title="Save" ><submit button-type="button"/></field>
</form>

2. include form ชื่อ EditBlog เข้าไปในไฟล์ *Screens.xml
<screen name="EditBlog">
        <section>
            <actions>
                <set field="newBlogId" from-field="parameters.newBlogId" global="true"/>
                <entity-one value-field="resultMap" entity-name="NewBlog" auto-field-map="true"></entity-one>
            </actions>
            <widgets>
                <decorator-screen name="CommonDecorator" location="component://myblog/widget/CommonScreens.xml">
                    <decorator-section name="body">
                        <container id="content">
                             <container style="post">
                                   <container style="entry">
                                        <include-form location="component://myblog/widget/MyBlogForms.xml" name="EditBlog"/>
                                   </container>
                             </container>
                        </container>
                    </decorator-section>
                </decorator-screen>
            </widgets>
        </section>
    </screen>

3. กลับไปดูที่ไฟล์ *Forms.xml ฟอร์มชื่อ EditBlog มี target="editBlog"
<form name="EditBlog" type="single" target="editBlog" title="" 
            default-map-name="resultMap" default-entity-name="NewBlog">
        <auto-fields-entity entity-name="NewBlog" default-field-type="edit"/>
        <field name="statusId">
            <drop-down allow-empty="false">
                <entity-options description="${description}" entity-name="StatusItem" key-field-name="statusId">
                <entity-constraint name="statusTypeId" value="CONTENT_STATUS"/>
                </entity-options>
            </drop-down>
        </field>
        <field name="editButton" title="Save" ><submit button-type="button"/></field>
</form

4. ใน controller.xml ทำการสร้าง request-map และสั่งให้มีการการ invoke simple method ชื่อ editBlog ที่ services.xml
<request-map uri="editBlog">
        <security https="true" auth="false"/>
        <event type="service" invoke="editBlog"/>
        <response name="success" type="request-redirect-noparam" value="home"></response>
</request-map>

5. ใน services.xml มีการ invoke editBlog ในไฟล์ *Services.xml
<service name="editBlog" engine="simple" auth="false"
            location="component://myblog/script/com/org/toon/myblog/MyBlogServices.xml" invoke="editBlog">
            <attribute name="newBlogId" mode="INOUT" type="String" optional="false"></attribute>
            <attribute name="newBlogTitle" mode="IN" type="String" optional="false"></attribute>
            <attribute name="newBlogBody" mode="IN" type="String" optional="false"></attribute>
            <attribute name="statusId" mode="IN" type="String" optional="true"></attribute>
</service>

6. สร้าง service ชื่อ editBlog ในไฟล์ MyBlogServices.xml
<simple-method method-name="editBlog" short-description="edit a Blog">
        <entity-one value-field="newData" entity-name="NewBlog" auto-field-map="true"></entity-one>
        <set-nonpk-fields value-field="newData" map="parameters"/>
        <store-value value-field="newData"/>
</simple-metho

7. จบส่วนในการสร้างและเรียกใช้ Service ต่อไปจะทำการ controller ให้เรียก view ที่มีหน้า form ของการ edit new blog มาแสดงผล

8.  เพิ่ม controller ชื่อ EditBlog ใน controller.xml ให้เรียก view ชื่อ CreateBlog มาแสดงผล เมื่อมีการเรียกเข้ามาที่ controller นี้
<request-map uri="EditBlog"><security https="true" auth="false"/>
        <response name="success" type="view" value="EditBlog"/>
    </request-map>

9. สร้าง view ชื่อ EditBlog เรียกฟอร์มการสร้างมาแสดงผล
<view-map name="EditBlog" type="screen" page="component://myblog/widget/MyBlogScreens.xml#EditBlog"/>

10. ทำการ ./startofbiz.sh ใหม่อีกครั้ง แล้วทดลองเแก้ไขข้อมูลลงใน entity

วันอังคารที่ 8 มิถุนายน พ.ศ. 2553

การสร้าง service เพื่อ insert ข้อมูลลงในฐานข้อมูลโดยใช้ simple method

ในการสร้าง service ใน OFBiz นั้น สามารถใช้ engine ในการสร้าง method ได้ 3 วิธี ดังนี้
1. engine="java" ทำได้โดยการเขียนโปรแกรมภาษา java เก็บไว้ในโฟลเดอร์ src แล้วเรียกใช้งานผ่าน package
2. engine="simple" เขียนเป็น simple method เก็บไว้ในไฟล์ *Services.xml
3. engine เป็น webservice

ในบทความนี้จะแสดงการสร้าง service โดยใช้ simple method ในไฟล์ *Services.xml มีวิธีทำดังนี้

1. สร้าง form ชื่อ CreateBlog ที่ *Forms.xml

<form name="CreateBlog" type="single" target="createBlog" default-entity-name="NewBlog">
        <auto-fields-entity entity-name="NewBlog" default-field-type="edit"/>
        <field name="saveButton" title="Save" ><submit button-type="button"/></field>
</form>

2. include form ชื่อ CreateBlog เข้าไปในไฟล์ *Screens.xml
<screen name="CreateBlog">
        <section>
            <widgets>
                <decorator-screen name="CommonDecorator" location="component://myblog/widget/CommonScreens.xml">
                    <decorator-section name="body">
                        <container id="content">
                            <container id="content-bgtop">
                                <container id="content-bgbtm">
                                    <container style="post">
                                        <include-form location="component://myblog/widget/MyBlogForms.xml" name="CreateBlog"/>
                                    </container>
                                </container>
                            </container>
                        </container>
                    </decorator-section>
                </decorator-screen>
            </widgets>
        </section>
    </screen>

3. กลับไปดูที่ไฟล์ *Forms.xml ฟอร์มชื่อ CreateBlog มี target="createBlog"
<form name="CreateBlog" type="single" target="createBlog" default-entity-name="NewBlog">
        <auto-fields-entity entity-name="NewBlog" default-field-type="edit"/>
        <field name="saveButton" title="Save" ><submit button-type="button"/></field>
</form>

4. ใน controller.xml ทำการสร้าง request-map และสั่งให้มีการการ invoke simple method ชื่อ createBlog ที่ services.xml

<request-map uri="createBlog">
        <event type="service" invoke="createBlog"/>
        <response name="success" type="request-redirect-noparam" value="blog"></response>
</request-map>

5. ใน services.xml มีการ invoke createBlog ในไฟล์ *Services.xml
<service name="createBlog" engine="simple" auth="false"
            location="component://myblog/script/com/org/toon/myblog/MyBlogServices.xml" invoke="createBlog">
            <attribute name="newBlogId" mode="INOUT" type="String" optional="false"></attribute>
            <attribute name="newBlogTitle" mode="IN" type="String" optional="false"></attribute>
            <attribute name="newBlogBody" mode="IN" type="String" optional="false"></attribute>
            <attribute name="statusId" mode="IN" type="String" optional="true"></attribute>
</service>

6. สร้าง service ชื่อ createBlog ในไฟล์ MyBlogServices.xml
<simple-method method-name="createBlog" short-description="create a Blog">
        <make-value entity-name="NewBlog" value-field="newEntity"/>
        <set-pk-fields map="parameters" value-field="newEntity"/>
        <set-nonpk-fields map="parameters" value-field="newEntity"/>
        
        <create-value value-field="newEntity"/>

        <field-to-result field="parameters.newBlogId" result-name="newBlogId"/>
</simple-method>


7.จบส่วนในการสร้างและเรียกใช้ Service ต่อไปจะทำการ controller ให้เรียก view ที่มีหน้า form ของการ create new blog มาแสดงผล

8. เพิ่ม controller ชื่อ CreateBlog ใน controller.xml ให้เรียก view ชื่อ CreateBlog มาแสดงผล เมื่อมีการเรียกเข้ามาที่ controller นี้

<request-map uri="CreateBlog">
        <security https="true" auth="false"/>
        <response name="success" type="view" value="CreateBlog"></response>
</request-map>

9. สร้าง view ชื่อ CreateBlog เรียกฟอร์มการสร้างมาแสดงผล
<view-map name="CreateBlog" type="screen" page="component://myblog/widget/MyBlogScreens.xml#CreateBlog"/>

10. ทำการ ./startofbiz.sh ใหม่อีกครั้ง แล้วทดลองเพิ่มข้อมูลลงใน entity

ในบทความต่อไปจะแสดงการแก้ไขข้อมูลในฐานข้อมูลผ่านการสร้าง service โดยใช้ simple method แบบเดียวกันนี้

การใช้ form widget ใน OFBiz - การสร้าง list form widget

จากบทความที่แล้ว (การใช้ form widget ใน OFBiz - การสร้าง find form widget) ได้ทำการสร้างฟอร์มการค้นหา ขั้นตอนต่อไปจะทำการสร้างฟอร์มที่แสดลผลการค้นหา หรือ list form ดังนี้

1. define form ของการแสดงผลของ list form ในไฟล์ *Forms.xml

<form name="ListExamples" type="list" target="example" default-entity-name="Example" list-name="listIt">
        <actions>
            <service service-name="performFind" result-map="result" result-map-list="listIt">
                <field-map field-name="inputFields" from-field="parameters"/>
                <field-map field-name="entityName" value="Example"/>
            </service>
        </actions>
        <auto-fields-entity entity-name="Example" default-field-type="display"/>
    </form>

อธิบายโค้ด
ในส่วนของ actions จะมีการเรียกใช้ service ในไฟล์ services.xml ดังนี้
service-name="performFind" result-map="result" result-map-list="listIt" มีการเรียกใช้ service ชื่อ performFind ซึ่งเป็น service ของ OFBiz ที่สร้างขึ้นมาเพื่อให้เรียกใช้งาน find form ได้

2. ที่ screen ที่ต้องการให้ข้อมูลผลลัพธ์ของการค้นหาแสดงผล ให้เพิ่มโค้ดในส่วนของ actions และ include form ListExamples เข้าไป ดังนี้

<actions>
                <set field="exampleId" global="true" />
                <entity-one value-field="resultMap" entity-name="Example"></entity-one>
                <get-related relation-name="ExampleItem" list="resultList" value-field="resultMap"/>
            </actions>

<include-form location="component://culinary/widget/CulinaryForms.xml" name="ListExamples"/>

โค้ดในส่วนของ screen name="findExample" เมื่อสมบูรณ์แล้ว จะได้ดังนี้

<screen name="findExample">
        <section>
            <actions>
                <set field="exampleId" global="true" />
                <entity-one value-field="resultMap" entity-name="Example"></entity-one>
                <get-related relation-name="ExampleItem" list="resultList" value-field="resultMap"/>
            </actions>
            <widgets>
                <decorator-screen name="CommonDecorator" location="component://culinary/widget/CommonScreens.xml">
                    <decorator-section name="body">
                        <container id="content">
                             <container style="post">
                                   <container style="entry">
                                        <include-form location="component://culinary/widget/CulinaryForms.xml" name="FindExample"/>
                                        <include-form location="component://culinary/widget/CulinaryForms.xml" name="ListExamples"/>
                                   </container>
                             </container>
                        </container>
                    </decorator-section>
                </decorator-screen>
            </widgets>
        </section>
    </screen>

อธิบายโค้ด
ส่วนของ actions จะทำการประมวลผลก่อน ถึงจะส่งไปให้ในส่วนของ widgets แสดงผลต่อไป
field="exampleId" global="true" ทำการเซ็ตค่าที่ได้จากตัวแปร exampleId ให้เป็น global

entity-one value-field="resultMap" entity-name="Example" ทำการส่งค่าที่ได้จากการค้นหาใน entity ชื่อ Example มาเก็บไว้ในตัวแปรชนิด map ชื่อ resultMap

get-related relation-name="ExampleItem" list="resultList" value-field="resultMap" ทำการลิสต์ item ที่มีความสัมพันธ์กับ entity ชื่อ Example นี้มาแสดงผลทีละ item

entity ชื่อ Example นี้ จะมี entity ชื่อ ExampleItem ที่มีความสัมพันธ์กันแบบ one to many โค้ดในส่วนของ get-related จะช่วยทำการลิสต์ item ใน ExampleItem มาแสดงผล

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

ส่วนหัวของรายงานเปรียบได้กับ entity ชื่อ Example ที่มี entity ชื่อ ExampleItem มามีความสัมพันธ์ด้วย เพื่อเรียกใช้ส่วนหัวของรายงาน นั่นหมายความว่า หัวรายงาน 1 หัวรายงาน สามารถแสดง item ใน entity ชื่อ ExampleItem ได้หลายไอเท็มนั่นเอง ซึ่งคำสั่งนี้จะทำการลิสต์ไอเท็มที่ต้องการแสดงภายใต้หัวรายงานออกมาทั้งหมด

เมื่อจบในส่วนของ actions แล้ว จึง include เพื่อเรียกใช้งาน form ชื่อ ListExamples มาแสดงผลต่อจากส่วน FindExample

3. ทำการทดลองค้นหาผ่าน url http://localhost:8080/culinary/control/example จะพบว่า เมื่อกดปุ่ม Search แล้ว ผลลัพธ์ในการค้นหาจะแสดงต่อจาก find form ทางด้านล่าง

การใช้ form widget ใน OFBiz - การสร้าง find form widget

ใน OFBiz สามารถนำ form แสดงผลใน screen ได้ โดย form ที่นำมาแสดง สามารถแบ่งประเภทได้ดังนี้
1. find form เป็นฟอร์มที่ใช้ค้นหา item หรือ content ต่างๆ ใน module
โดย form ชนิดนี้จะมี type เป็น single


2. list form เป็นฟอร์มที่ใช้แสดงรายการ โดยจะลิสต์แสดงรายการออกมาเป็น table
form ชนิดนี้จะมี type เป็น list


3. display and edit เป็นฟอร์มที่ใช้แสดงรายการแก้ไข โดยจะสามารถแก้ไขข้อความใน field ที่อนุญาตให้แก้ไขได้ หากต้องการให้ field ไหนแก้ไขได้ สามารถ เซ็ต type เป็น input หรือ textbox หรือแล้วแต่ชนิดของ field ที่ต้องการแก้ไข แต่หากไม่ต้องการให้สามารถแก้ไขได้ ให้เซ็ต type เป็น display จะเป็นการสั่งให้แสดงผลเท่านั้น แต่ไม่สามารถแก้ไขได้



ขั้นตอนการสร้าง find form widget สามารถทำได้ดังนี้

1. ทำการ define form เข้าไปในไฟล์ *Form.xml ดังนี้
หมายเหตุ *Form.xml คือชื่อของ Component ต่อด้วย Form.xml จะอยู่ในโฟลเดอร์ widget

<form default-entity-name="Example" name="FindExample" target="example" type="single">
<auto-fields-entity default-field-type="find" entity-name="Example">
        <field name="noConditionFind"><hidden value="Y"></hidden>
        <field name="searchButton" title="Search"><submit button-type="button"></submit>
    </field></field></auto-fields-entity></form>

อธิบายโค้ด
form name="FindExample" ตั้งชื่อฟอร์มว่า  FindExample
type="single" target="example" ให้เป็นชนิด single และ route ไปที่ example controller
default-entity-name="Example" ชื่อ entity ที่เรียกใช้ข้อมูลคือ Example

auto-fields-entity entity-name="Example" ทำการเรียกข้อมูลใน entity ชื่อ Example มาแสดงโดยอัติโนมัติทุก field โดยที่ไม่ต้องระบุชื่อฟิลด์
default-field-type="find" ระบุชนิดของ field ใน form นี้ให้เป็น find

field name="searchButton" title="Search"
submit button-type="button"เพิ่มปุ่ม Search ให้ฟอร์มนี้

2. ทำการเรียก form ที่ define ไว้ใน *Form.xml มาแสดงผลใน *Screen.xml
<screen name="findExample">
        <section>
            <widgets>
                <decorator-screen location="component://culinary/widget/CommonScreens.xml" name="CommonDecorator">
                    <decorator-section name="body">
                        <container id="content">
                             <container>
                                   <container>
                                        <include-form location="component://culinary/widget/CulinaryForms.xml" name="FindExample">
                                   </include-form>
                             </container>
                        </container>
                    </container>
                </decorator-section>
            </decorator-screen>
        </widgets>
    </section></screen>
อธิบายโค้ด
include-form location="component://culinary/widget/CulinaryForms.xml" name="FindExample" ทำการ include form ชื่อ FindExample ไปแสดงผลที่ screen widget ที่ชื่อ findExample

3. ทำการกำหนด request-map ให้กับไฟล์ controller.xml

<request-map uri="example">
        <security auth="true" https="false">
        <response name="success" type="view" value="example"></response>
    </security>
</request-map>

4.สร้าง view ให้กับ example controller

<view-map name="example" page="component://culinary/widget/CulinaryScreens.xml#findExample" type="screen"></view-map>

5. ทดลองเรียกใช้ url http://localhost:8080/culinary/control/example จะปรากฏ find form ดังนี้

ซึ่งฟอร์มนี้จะแสดงเฉพาะฟอร์มการค้นหา ไม่สามารถแสดงผลการค้นหาได้ เนื่องจากยังไม่กำหนด actions ให้กับ screen
ในบทความต่อไปจะแสดงผลลัพธ์จากการค้นหาที่ต่อเนื่องจากการสร้างฟอร์มในบทความนี้

การดึงข้อมูลจาก Entity มาแสดงผลในหน้าเพจ

OFBiz สามารถเรียกข้อมูลจาก Entity มาแสดงผลในหน้าเวบเพจได้ 2 วิธี คือ
1. ดึงจาก Screen หรือ Widget
2. ดึงจาก Script

ในบทความจะแสดงการดึงค่าจาก Widget ดังนี้
1. เพิ่มแท็ก <action> ลงภายใต้แท็ก <screen><section> ในไฟล์ CulinaryScreens.xml ในโฟลเดอร์ widget
หมายเหตุ: ชื่อไฟล์ CulinaryScreens.xml ตาม component ชื่อ culinary หากทำการสร้าง component ชื่ออื่น ให้ทำการแก้ไขที่ไฟล์ ชื่อคอมโพเน้นท์Screens.xml
ดังนี้

<actions>
                <set field="exampleId" value="10000" global="true" />
                <entity-one value-field="resultMap" entity-name="Example"></entity-one>
                <get-related relation-name="ExampleItem" list="resultList" value-field="resultMap"/>
</actions>

อธิบายโค้ด

<set field="exampleId" global="true" value="10000">
</set>
ทำการเลือก record มาแสดงผล โดยทำการเลือก record ที่มี exampleId = 10000 และทำการเซ็ตให้มีค่าเป็น Global


<entity-one value-field="resultMap" entity-name="Example"></entity-one>
เก็บผลลัพธ์ (record) ที่ได้จากบรรทัดแรกมาเก็บในตัวแปร resultMap

<get-related relation-name="ExampleItem" list="resultList" value-field="resultMap"/>
ดึง item ที่มีความสัมพันธ์กับ exampleId = 10000 มาเก็บไว้ใน list ชื่อ resultList

2. ที่หน้า view ของ home เพิ่มโค้ดดังนี้

<#list resultList as resultItem>
    ${resultItem.description}<br />
</#list>

อธิบายโค้ด
ใช้แท็ก FreeMarker มาช่วยในการแสดงผล โดยทำการ list ค่าของ ExampleItem ทั้งหมดที่อยู่ใน resultList มาแสดงผลทีละไอเทม

3. ทดลองเปิดหน้าเว็บไซต์ดู

วันจันทร์ที่ 7 มิถุนายน พ.ศ. 2553

การสร้างและการเรียกใช้งาน Screen Decorator ใน OFBiz

ในการสร้างหน้าเพจ มักจะมีส่วนของการแสดงผลที่ซ้ำๆ กัน เช่น ส่วนของ menu, header, footer เป็นต้น ซึ่งใน OFBiz จะมีการสร้าง Screen Decorator ไว้สำหรับเป็น template และหากมีการเปลี่ยนแปลงเฉพาะส่วนย่อยๆ จะเปลี่ยนเฉพาะ Screen บางส่วนเท่านั้น ซึ่งในการสร้าง Screen Decorator สามารถสร้างได้ดังนี้

1. ในเทมเพลนมีส่วนที่ซ้ำกันทุกหน้าคือ header และ footer ให้ทำการแยกโค้ดในส่วนของ header และ footer เก็บไว้ในเพจ .ftl เช่น header.ftl และ footer.ftl
2. โค้ดในส่วนที่เหลือจาก header.ftl และ footer.ftl ให้เซฟเก็บไว้เป็นไฟล์ index.ftl
3. เปิดไฟล์ CommonScreens.xml ซึ่งจะอยู่ในโฟลเดอร์ hot-deploy/culinary/widget/CommonScreens.xml
4. ทำการสร้าง Screen สำหรับที่จะใช้เป็น template ภายใต้แท็ก <screens></screens> ดังโค้ดต่อไปนี้

<screen name="CommonDecorator">
        <section>
            <widgets>
                <!-- header section -->
                <platform-specific>
                    <html>
                        <html-template location="component://culinary/webapp/culinary/includes/header.ftl"/>
                    </html>
                </platform-specific>
                <decorator-section-include name="body"/>
                <!-- right -->
                <platform-specific>
                    <html>
                        <html-template location="component://culinary/webapp/culinary/includes/right.ftl"/>
                    </html>
                </platform-specific>
                <!-- footer section -->
                <platform-specific>
                    <html>
                        <html-template location="component://culinary/webapp/culinary/includes/footer.ftl"/>
                    </html>
                </platform-specific>
            </widgets>
        </section>
    </screen>

อธิบายโค้ด
 <screen name="CommonDecorator"> ตั้งชื่อ screen นี้ว่า CommonDecorator

<platform-specific>
      <html>
              <html-template location="component://culinary/webapp/culinary/includes/header.ftl"/>
      </html>
</platform-specific>
ดึงไฟล์ header.ftl มาเป็นส่วนหนึ่งของเทมเพลต


<decorator-section-include name="body"/>
เตรียมพื้นที่ไว้รอแสดงส่วนเนื้อหาจากหน้าอื่น

<platform-specific>
    <html>
        <html-template location="component://culinary/webapp/culinary/includes/footer.ftl"/>
    </html>
</platform-specific>
ดึงไฟล์ header.ftl มาเป็นส่วนหนึ่งของเทมเพลต

เมื่อเสร็จขั้นตอนนี้ จะได้ เทมเพลตที่ประกอบไปด้วยส่วน header และ footer ที่ไม่มีการเปลี่ยนแปลงของ screen และมีส่วน body ที่ define ไว้รอสำหรับการนำเอาเนื้อหาที่มีการเปลี่ยนแปลงในแต่ละหน้ามาแสดงผล

5. ในโฟลเดอร์ widget จะมีไฟล์ xml ที่ใช้จัดการกับ screen อีก 3 ไฟล์ ซึ่งจะมี 1 ไฟล์ที่ชื่อเดียวกับชื่อ component เช่น ถ้า component ชื่อ culinary ไฟล์นี้จะชื่อ CulinaryScreens.xml ทำการเปิดไฟล์นี้และเพิ่มโค้ดดังนี้

<screen name="home">
        <section>
            <widgets>
                <decorator-screen name="CommonDecorator" location="component://culinary/widget/CommonScreens.xml">
                    <decorator-section name="body">
                        <platform-specific>
                            <html>
                                <html-template location="component://culinary/webapp/culinary/home.ftl"/>
                            </html>
                        </platform-specific>
                    </decorator-section>
                </decorator-screen>
            </widgets>
        </section>
    </screen>

อธิบายโค้ด
<decorator-screen location="component://culinary/widget/CommonScreens.xml" name="CommonDecorator">
ตั้งชื่อ screen ว่า CommonDecorator และ url ของ screen นี้ถูกกำหนดลงใน location

<decorator-section name="body">
    <platform-specific>
        <html>
            <html-template location="component://culinary/webapp/culinary/home.ftl"/>
        </html>
    </platform-specific>
</decorator-section>
อธิบายโค้ด
ในส่วน decorator-section ชื่อ body ให้เรียกไฟล์ home.ftl มาแสดงผล

6. ทำการกำหนด controller และ view ในไฟล์ webapp/culinary/WEB-INF/controller.xml ดังนี้

<request-map uri="home">
<security https="false" auth="true"/>
<response name="success" type="view" value="home"/></request-map>
<view-map name="home" type="screen" page="component://culinary/widget/CulinaryScreens.xml#home"/>

7. ทดสอบเรียกหน้า home ขึ้นมาแสดงผล

การติดตั้ง OFBiz โดยใช้ PostgreSQL เป็นฐานข้อมูล

PostgreSQL คือ ระบบจัดการฐานข้อมูลเชิงสัมพันธ์ (Object-Relational DataBase Management System หรือ ORDBMS) มีความสามารถในการทำงานร่วมกับ OFBiz ได้อย่างดี ทั้งนี้ยังเป็นซอฟต์แวร์ฟรีอีกด้วย จึงสามารถนำมาช่วยลดค่าใช้จ่ายในองค์กรได้

จากบทความเรื่องการติดตั้ง OFBiz ได้ใช้ฐานข้อมูล Derby ซึ่งเป็นฐานข้อมูลขนาดเล็กที่มาพร้อม OFBiz มีความสามารถในระดับหนึ่ง แต่ไม่เทียบเท่า PostgreSQL และไม่สามารถรองรับการใช้งานที่มีการเรียกใช้ฐานข้อมูลจำนวนมากได้ จึงได้นำเอา PostgreSQL มาช่วยให้การใช้งาน OFBiz มีประสิทธิภาพมากขึ้น

ขั้นตอนการติดตั้ง PostgreSQL สำหรับ OFBiz ทำได้ดังนี้

1. ติดตั้ง PostgreSQL จาก Synaptic Package Manager
2. ติดตั้ง pgadmin จาก Synaptic Package Manager
3. ทำการตั้งค่า password ให้กับ account ของ PostgreSQL โดยเปิด terminal และเข้าไปที่โฟลเดอร์ /etc/postgresql/8.4/main
4. สั่งแก้ไขไฟล์ชื่อ pg_hba.conf โดยใช้คำสั่ง
    sudo gedit pg_hba.conf
5. เปลี่ยนค่าจาก ident เป็น trust ทั้งหมด และ เปลี่ยนค่าจาก md5 เป็น trust ทั้งหมด
6. สั่งแก้ไขไฟล์ชื่อ postgresql.conf โดยใช้คำสั่ง
sudo gedit postgresql.conf 
7. แก้ไขบรรทัด #listen_addresses = 'localhost' โดยการเอา # ออกและเปลี่ยนจาก localhost เป็น * จะได้โค้ดดังนี้
listen_addresses = '*'   
8. สั่ง start serveice ของ postgresql โดยใช้คำสั่ง
cd /etc/init.d
9. สั่ง restart service โดยใช้คำสั่ง
sudo service postgresql-8.4 restart

เมื่อตั้งค่าเสร็จแล้ว ลงมือทำการสร้าง connection ดังนี้
1. เปิด pgadmin
2. กดปุ่ม add a connection to a server
3. ตั้งค่าดังนี้
    Name = localhost
    Host = localhost
แล้วกด ok
4. ทำการสร้าง login role ให้แก่ database โดยการคลิ้กขวาที่ Login Roles แล้วเลือก New Login Role
5. กรอกรายละเอียด Role name, Password และ Password(again) แล้วกด OK
6. ในการสร้างฐานข้อมูลสำหรับ OFBiz ต้องสร้างทั้งหมด 3 database โดยกำหนดดังนี้
    Database1
        name = ofbiz
        login role = ofbiz
    Database2
        name = ofbizolap
        login role = ofbiz
    Database3
        name = ofbiztenance
        login role = ofbiz
7. ทำการดาวน์โหลด jdbc driver สำหรับ postgresql โดยสามารถดาวน์โหลดได้ที่ http://jdbc.postgresql.org/download.html
เลือกดาวน์โหลดไฟล์ JDBC4 Postgresql Driver, Version 8.4-701 สำหรับ 1.6 JVM ขึ้นไป
8. ทำการ copy ไฟล์ที่ดาวน์โหลดไปไว้ในโฟลเดอร์ framework/entity/lib/jdbc ของโปรเจ็ค
9. ทำการแก้ไขไฟล์ framework/entity/config/entityengine.xml โดยการเพิ่ม datasource ดังนี้
<datasource name="localpostnew"
            helper-class="org.ofbiz.entity.datasource.GenericHelperDAO"
            schema-name="public"
            field-type-name="postnew"
            check-on-start="true"
            add-missing-on-start="true"
            use-fk-initially-deferred="false"
            alias-view-columns="false"
            join-style="ansi"
            result-fetch-size="50"
            use-binary-type-for-blob="true">
        <read-data reader-name="seed"/>
        <read-data reader-name="seed-initial"/>
        <read-data reader-name="demo"/>
        <read-data reader-name="ext"/>
        <inline-jdbc
                jdbc-driver="org.postgresql.Driver"
                jdbc-uri="jdbc:postgresql://127.0.0.1/org"
                jdbc-username="org"
                jdbc-password="orgpass"
                isolation-level="ReadCommitted"
                pool-minsize="2"
                pool-maxsize="250"
                time-between-eviction-runs-millis="600000"/>
    </datasource>
    
    <datasource name="localpostnewolap"
            helper-class="org.ofbiz.entity.datasource.GenericHelperDAO"
            schema-name="public"
            field-type-name="postnew"
            check-on-start="true"
            add-missing-on-start="true"
            use-fk-initially-deferred="false"
            alias-view-columns="false"
            join-style="ansi"
            result-fetch-size="50"
            use-binary-type-for-blob="true">
        <read-data reader-name="seed"/>
        <read-data reader-name="seed-initial"/>
        <read-data reader-name="demo"/>
        <read-data reader-name="ext"/>
        <inline-jdbc
                jdbc-driver="org.postgresql.Driver"
                jdbc-uri="jdbc:postgresql://127.0.0.1/orgolap"
                jdbc-username="org"
                jdbc-password="orgpass"
                isolation-level="ReadCommitted"
                pool-minsize="2"
                pool-maxsize="250"
                time-between-eviction-runs-millis="600000"/>
    </datasource>
    
    <datasource name="localpostnewtenant"
            helper-class="org.ofbiz.entity.datasource.GenericHelperDAO"
            schema-name="public"
            field-type-name="postnew"
            check-on-start="true"
            add-missing-on-start="true"
            use-fk-initially-deferred="false"
            alias-view-columns="false"
            join-style="ansi"
            result-fetch-size="50"
            use-binary-type-for-blob="true">
        <read-data reader-name="seed"/>
        <read-data reader-name="seed-initial"/>
        <read-data reader-name="demo"/>
        <read-data reader-name="ext"/>
        <inline-jdbc
                jdbc-driver="org.postgresql.Driver"
                jdbc-uri="jdbc:postgresql://127.0.0.1/orgtenant"
                jdbc-username="org"
                jdbc-password="orgpass"
                isolation-level="ReadCommitted"
                pool-minsize="2"
                pool-maxsize="250"
                time-between-eviction-runs-millis="600000"/>
    </datasource>

และแก้ไข delegetor ของ delegator name="default" และ delegator name="default-no-eca" ดังนี้

<delegator name="default" entity-model-reader="main" entity-group-reader="main" entity-eca-reader="main" distributed-cache-clear-enabled="false">
        <group-map group-name="org.ofbiz" datasource-name="localpostnew"/>
        <group-map group-name="org.ofbiz.olap" datasource-name="localpostnewolap"/>
        <group-map group-name="org.ofbiz.tenant" datasource-name="localpostnewtenant"/>
    </delegator>
    <delegator name="default-no-eca" entity-model-reader="main" entity-group-reader="main" entity-eca-reader="main" entity-eca-enabled="false" distributed-cache-clear-enabled="false">
        <group-map group-name="org.ofbiz" datasource-name="localpostnew"/>
        <group-map group-name="org.ofbiz.olap" datasource-name="localpostnewolap"/>
        <group-map group-name="org.ofbiz.tenant" datasource-name="localpostnewtenant"/>
    </delegator> 

10. พิมพ์คำสั่งใน terminal ดังนี้
./ant clean-all
11. พิมพ์คำสั่งใน terminal ดังนี้
./ant run-install
12. สั่ง start ofbiz ดังนี้
./startofbiz.sh

การใช้ Subversion ในการควบคุม Source Code

ในการสร้างโปรเจ็ค ควรจะทำการควบคุม source code เพื่อป้องกันความผิดพลาดจากการลบไฟล์โดยไม่ได้ตั้งใจ และเพื่อสามารถย้อนกลับไปใช้ซอร์สโค้ดเดิมได้ หากมีข้อผิดพลาดเกิดขึ้นกับ souce code เวอร์ชั่นไหม่

ในบทความจะแสดงวิธีการใช้งาน Subversion (svn) ควบคุม source code บนพื้นที่ฟรีเช่น google code สามารถทำได้ดังนี้

1. เปิดเว็บไซต์กูเกิ้ลโค้ด โดยเข้าไปที่ http://code.google.com
2. เลือก Project Hosting
3. เลือก Project Hosting on Google Code
4. ทำการ sign in เข้าใช้งาน โดยใช้ account ของ google หรือ gmail
5. ทำการสร้างโปรเจ็คโดยการเลือก Create a new project
6. ในหน้า Create Project ให้กำหนดรายละเอียดของโปรเจ็ค โดยกำหนดเพิ่มเติมดังนี้
Version control system: เลือก subversion
Source code license: เลือก Apache License 2.0
Use a separate content license: เลือก Creative Common 3.0 BY
Project labels: ใส่ tag ที่เกี่ยวข้องกับโปรเจ็ค
7. กดปุ่ม Create Project
8. เมื่อสร้างโปรเจ็คแล้ว กดแท็บ Source จะได้คำสั่งในการ checkout code
9. ทำการก็อปปี้คำสั่งไปวางใน terminal และกด enter เพื่อทำการ check out code ลงมาที่เครื่อง
10. เมื่อ check out ลงมาแล้ว หากต้องการดูข้อมูลของโปรเจ็ค ให้พิมพ์ svn info
11. ทดลองทำการเพิ่มไฟล์ลงในโปรเจ็ค เสร็จแล้วพิมพ์ svn st เพื่อดูสถานะ ที่ terminal จะแสดงค่า
? test.txt
หมายความว่า ไฟล์ที่เครื่องคอมพิวเตอร์ไม่ตรงกับไฟล์บน google code
12. หากต้องการทำการนำไฟล์นี้ขึ้นไปบน google code ให้ใช้คำสั่ง
svn add ชื่อไฟล์หรือโฟลเดอร์ เช่น
svn add test.txt
เพื่อทำการเพิ่มไฟล์เข้าไปในระบบ
13. เมื่อทำการ add แล้ว status ของไฟล์จะเป็น A ซึ่งหมายความว่า add ไฟล์นี้เข้าสู่โปรเจ็คแล้ว แต่ยังไม่ได้อัพโหลดขึ้นบน google code
หากต้องการนำโค้ดขึ้นไปบนโฮสต์ให้ใช้คำสั่ง
svn commit -m"ข้อความที่ต้องการอธิบาย" เช่น
svn commit -m"my first commit"
เพื่อทำการ commit file ขึ้นไปบนโฮสต์ แล้วกด enter
10. เมื่อมีการแสดงข้อความร้องขอให้ใส่พาสเวิร์ดบน terminal ให้กลับไปที่หน้า source ของ google code แล้วกดที่ข้อความ
When prompted, enter your generated googlecode.com password กูเกิ้ลโค้ดจะทำการสร้างพาสเวิร์ดให้ และก๊อปปี้พาสเวิร์ดนั้นไปใส่ใน terminal
11. svn จะทำการ commit ไฟล์ขึ้นสู่ google code และจะแสดง Committed revision 2 หมายถึงได้เพิ่มโค้ดเรียบร้อยแล้ว

คำสั่งของ svn ที่ใช้บ่อย จะมีดังนี้
svn checkout หรือ svn co สั่งให้ check out source code จากโฮสต์ลงมาเก็บไว้ในเครื่อง
svn add เมื่อมีการสร้างไฟล์หรือไดเร็คทอรี่ คำสั่งนี้จะเป็นการบอกให้ svn ทราบว่ามีการเพิ่มไฟล์เข้าไปในโปรเจ็ค แต่จะยังไม่ทำการอัพโหลดขึ้นไปยังโฮสต์ จนกว่าจะสั่ง svn commit
svn delete คำสั่งนี้จะเป็นการบอกให้ svn ทราบว่ามีการเพิ่มไฟล์ลบไฟล์ออกจากโปรเจ็ค และจะทำการลบไฟล์บนโฮสต์ทันทีที่สั่ง svn commit
svn status เช็คสถานะของไฟล์และโฟลเดอร์ ว่าอยู่ในสถานะไหน สถานะต่างๆ เช่น
A(Add), M(Commit), G(Merge), U(Update)
svn update หรือ svn up จะทำการ sync ไฟล์และโฟลเดอร์ในเครื่องกับบนโฮสต์ให้ตรงกัน
svn commit เป็นการสั่งนำไฟล์ขึ้นไปเก็บไว้บนโฮสต์

โดยคำสั่งที่จำเป็นของ svn สามารถศึกษาเพิ่มเติมได้ ดังนี้
http://www.linuxfromscratch.org/blfs/edguide/chapter04.html
http://oslc.wordpress.com/2007/01/20/subversion-basic-work-cycle-command-line/

วันศุกร์ที่ 4 มิถุนายน พ.ศ. 2553

การนำเอา FreeMarker Template Engine มาใช้ใน OFBiz

FreeMarker เป็น Template Engine ที่ใช้แนวคิด MVC เพื่อแยกส่วนแสดงผลออกจากส่วน source ของ java สามารถศึกษารายละเอียดเพิ่มเติมได้ที่ http://freemarker.sourceforge.net/

ในบทความจะแสดงตัวอย่างในการนำเอา freemarker มาใช้งานใน OFBiz project

โดยปกติแล้วหน้าเพจมักจะมีส่วนที่ซ้ำซ้อนกันที่ต้องแสดงทุกหน้า อย่างเช่น เมนู, วันที่ ฯลฯ เราสามารถ assigned ค่าให้กับค่าตัวแปรที่มีการแสดงผลซ้ำๆ กัน แล้วเก็บไว้ที่ไฟล์อื่น เพื่อความง่ายในการแก้ไขหากต้องมีการเปลี่ยนค่า ไม่ต้องเปลี่ยนทุกหน้า แต่เปลี่ยนที่ไฟล์ที่ assigned ค่าไว้แค่ไฟล์เดียว

หรืออีกประโยชน์หนึ่ง เพื่อให้หน้าเพจมีการเขียนโค้ดที่สั้นลง เนื่องจากโค้ดที่ซับซ้อนถูกกำหนดไว้ที่ไฟล์อื่น สามารถลดความยากในการอ่านโค้ดในเพจนั้นได้

ทดลองทำตามขั้นตอนดังนี้
1. สร้างไฟล์ .ftl ใหม่ใน webapp ของโปรเจ็ค ชื่อ configtemplate.ftl
2. เพิ่มโค้ดใน configtemplate.ftl ดังนี้









อธิบายโค้ด

<#macro category> สร้าง macro โดยกำหนดชื่อ macro ว่า category

3. ในหน้า testpage.ftl ทำการเรียก macro ชื่อ category ดังนี้
<#import "component://testtest/webapp/testtest/configtemplate.ftl" as myconf>
<@myconf.category />

อธิบายโค้ด
<#import "component://testtest/webapp/testtest/configtemplate.ftl" as myconf> ทำการ import ไฟล์ที่กำหนด macro ไว้ แล้วตั้ง alias ว่า myconf
<@myconf.category /> เรียกใช้ macro ชื่อ category ผ่าน alias myconf

4. ทดสอบเรียกหน้าเพจมาแสดงผล


ทดลองทำการ assigned string ลงในไฟล์ ดังนี้
1. ในหน้า configtemplate.ftl เพิ่มโค้ดดังนี้
<#assign addtext="test add string in template">
2. ในหน้า testpage.ftl ทำการเรียกใช้ตัวแปรที่ assigned ดังนี้
${myconf.addtext}
3. ทดสอบเรียกหน้าเพจมาแสดงผล


FreeMarker ยังสามารถสร้างและแสดง template ได้อีกหลายวิธี ให้ทดลองอ่าน Online Documentation ได้ที่เว็บไซต์ http://freemarker.sourceforge.net/docs/index.html

การเพิ่ม page ใน OFBiz

webapp ของ OFBiz อาศัยหลักการ MVC หรือ Model-View-Controller ในการแสดงผล โดยแยกออปเจคที่เก็บข้อมูล (model) ออปเจคที่แสดงข้อมูล (view) และออปเจคที่ติดต่อกับผู้ใช้ (controller) ออกจากกันอย่างชัดเจน โดย webapp ใน OFBiz จะมีไฟล์ชื่อ controller.xml ในโฟลเดอร์ WEB-INF เป็นตัวกำหนดว่าจะเรียก view ไหนขึ้นมาแสดงผล เมื่อถูก request ผ่านทาง url

ทดลองสร้าง views ขึ้นมาใน webapp ของ component โดยสร้างไฟล์ชนิด ftl ดังนี้
หมายเหตุ ftl เป็นไฟล์ชนิด free marker template engine

1. เลือก File > New > Others > Web > Html page
2. ตั้งชื่อไฟล์ เช่น testpage จะได้ไฟล์ testpage.html
3. ทำการ rename นามสกุลไฟล์เป็น .ftl จะได้ไฟล์ testpage.ftl
4. ในไฟล์ controller.xml ทำการสร้าง request-map ดังนี้

<request-map uri="testpage">
    <security auth="true" https="false">
        <response name="success" type="view" value="testpage"></response>
    </security>
</request-map>

อธิบายโค้ด

<request-map uri="testpage"></request-map>

ทำการเซ็ตค่าว่า ถ้าหาก uri ชื่อ testpage ถูก request ผ่าน webapp มา จะให้ทำอะไรบ้าง



security https="false" กำหนดให้ ไม่สามารถเรียกผ่าน https protocol ได้
auth="true" กำหนดให้ต้องมีการ login ก่อน ถึงจะสามารถเรียกใช้ page นี้ได้


response name="success" เมื่อมีการ success ในการเรียกใช้เพจนี้
type="view" ชนิดของ object ที่จะเรียกให้แสดงผลเมื่อถูก request url นี้เป็นชนิด view
value="testpage" ชื่อของ view ที่จะถูกเรียกมาแสดงผล

5. เมื่อทำการเพิ่ม view map ดังนี้



อธิบายโค้ด
view-map name ="testpage" ชื่อของ view นี้
type="ftl" เป็นชนิด ftl
page="component://testtest/webapp/testtest/testpage.ftl" uri ของ view นี้

6. ทดลองเรียกใช้ page ที่สร้างขึ้น โดยเรียกผ่าน url

https://localhost:8443/testtest/control/testpage

เอกสารอ้างอิง
http://www.narisa.com/forums/index.php?showtopic=1036
http://freemarker.sourceforge.net/

วันพฤหัสบดีที่ 3 มิถุนายน พ.ศ. 2553

การสร้าง Component ใน OFBiz

ในการพัฒนา OFBiz จะเรียกการสร้าง application หรือ module ว่า component โดยแต่ละ component จะเป็นอิสระต่อกัน และแต่ละ component จะมี data model, views, controller เป็นของ component เอง แต่สามารถทำงานร่วมกับ component อื่นได้

component หลักของ OFBiz จะอยู่ใน folder ชื่อ applications ส่วน component ที่ถูกสร้างขึ้นจะถูกสร้างไว้ในโฟลเดอร์ hot-deploy โดยขั้นตอนในการสร้าง component สำหรับ OFBiz มีดังนี้
1. เปิด Terminal แล้วเข้าไปยัง folder ที่มีการรัน OFBiz
2. ทำการเรียก ant ให้สร้าง component โดยใช้คำสั่งดังนี้
./ant create-component
3. ant จะให้ตั้งค่าต่างๆ ของ component รูปแบบดังนี้
- ชื่อ component รูปแบบดังนี้ Component name: (e.g. mycomponent)
- Component resource name รูปแบบดังนี้ Component resource name: (e.g. MyComponent)
- ชื่อ web application รูปแบบดังนี้ Webapp name: (e.g. mycomponent)
- Base permission รูปแบบดังนี้ Base permission: (e.g. MYCOMPONENT)
4. เมื่อกำหนดค่าทั้งหมดแล้ว ant จะให้ confirm สิ่งที่ตั้งค่าไป เมื่อถูกต้องทั้งหมดแล้วกด y เพื่อยืนยัน ant จะเริ่มทำการสร้าง component ให้
5. ทดสอบการสร้าง component โดยลองเรียกผ่านทาง web browser ดังนี้
https://localhost:8443/ชื่อคอมโพเน้นท์/control/main หรือ
http://localhost:8080/ชื่อคอมโพเน้นท์

เมื่อสร้าง component เสร็จแล้ว จะยังไม่สามารถล็อกอินเข้าใช้คอมโพเน้นท์ได้ ให้ทำการแอด data security ให้กับ component ก่อน มีขั้นตอนดังนี้
1. login เข้าไปใน webtools ของ OFBiz ที่ได้ติดตั้งไว้ ตาม url ดังนี้
https://localhost:8443/webtools/control/main หรือ
https://localhost:8080/webtools
2. ค่า default ของการ login เข้าใช้งาน OFBiz ดังนี้
username = admin
password = ofbiz
3. ในการ import data security เข้าไปใน webtools สามารถทำได้หลายวิธี ดังนี้
วิธีที่ 1 import security data xml file
3.1.1 เลือก XML Data Import ใน webtools
3.1.2 ใส่ค่า absolute path ของ security data file ลงในช่อง Absolute Filename or URL: เช่น
/home/toonztudio/Desktop/projects/ofbiz.10.04/hot-deploy/testtest/data/TestTestSecurityData.xml

หมายเหตุ: ไฟล์ security data จะอยู่ในโฟลเดอร์ hot-deploy/ชื่อคอมโพเน้นท์/data/ชื่อ Component resource name ต่อด้วย SecurityData.xml

3.1.3 กดปุ่ม Import file
3.1.4 หากขึ้นข้อความว่าไฟล์ถูกอิมพอร์ตเข้าไปแล้ว และไม่มี error แสดงว่าการนำเข้า security data สำเร็จ ดังนี้
Results:
Got 12 entities to write to the datasource.
วิธีที่ 2 import xml element
3.2.1 เลือก XML Data Import ใน webtools
3.2.2 เปิดไฟล์ SecurityData.xml แล้ว copy element ระหว่าง ลงในช่อง Complete XML document (root tag: entity-engine-xml):
3.2.3 กดปุ่ม import text
3.2.4 หากไม่มี error แสดงว่าสามารถนำเข้าได้ทั้งหมด
Results:
Got 12 entities to write to the datasource.

วิธีที่ 3 XML Data Import Dir
3.3.1 เลือก XML Data Import ใน webtools
3.3.2 ใส่ค่า absolute path ของ data directory ลงในช่อง Absolute directory path: เช่น
/home/toonztudio/Desktop/projects/ofbiz.10.04/hot-deploy/testtest/data

3.3.3 กดปุ่ม import file หากไม่มี error แจ้งว่า fail แสดงว่าการนำเข้าสำเร็จ

เมื่อทำการเพิ่ม data security ให้กับ component นี้แล้ว แอคเคาท์ admin จะสามารถ login เข้าใช้งาน component นี้ได้ ทดสอบได้โดยการ login เข้าไปใน url ดังนี้
https://localhost:8443/ชื่อคอมโพเน้นท์/control/main หรือ
http://localhost:8080/ชื่อคอมโพเน้นท์

เมื่อล็อกอินเข้ามาแล้วจะพบหน้า OFBiz ว่าง ซึ่งต้องทำการเซ็ต controller ของ webapp ใน component ของ OFBiz ให้เรียก views มาแสดงผลได้ ซึ่งจะแสดงในบทความต่อไป

วันพุธที่ 2 มิถุนายน พ.ศ. 2553

Linux, OFBiz and Eclipse Installation

ที่ Orange Gears พัฒนาและทำการปรับแต่ง OFBiz ให้ตรงกับความต้องการของลูกค้า โดยใช้สิ่งแวดล้อมในการพัฒนาคือ
1. Ubuntu Linux
2. Eclipse IDE
3. Subversion
4. Sub Java 1.5+ SDK
5. PostgreSQL
6. Apache OFBiz ERP

Linux Installation for OFBiz Development มีขั้นตอนดังนี้
1. ดาวน์โหลด Ubuntu Linux เวอร์ชั่น 10.04 จากเว็บไซต์ http://www.ubuntu.com/
2. install ubuntu 10.04 และอัพเดตไฟล์ที่จำเป็นของระบบ (Update Manager)
3. install JAVA SDK 1.5+
3.1 หากในระบบมีการลง Open Java ไว้อยู่แล้ว ให้ทำการเซ็ตให้ Sun Java กลายเป็น default jvm ของระบบ สามารถทำได้โดยใช้คำสั่งใน terminal ดังนี้

sudo update-alternatives --config java


Eclipse Installation
การติดตั้ง Eclipse สามารถทำได้ 2 วิธี ดังนี้
1. Download Eclipse IDE for Java and Report Developers (221 MB) เวอร์ชั่น Linux 32bit จากเว็บไซต์ eclipse.org แล้วทำการ extract เพื่อทำการเรียกใช้โปรแกรม
2. ติดตั้งจาก Synaptic Package Manager ของ Ubuntu โดยตรง

เมื่อทำการติดตั้ง Eclipse แล้ว ต้องทำการติดตั้ง Plug-ins เพิ่มเติม โดย Plug-ins ที่ต้องติดตั้งเพิ่ม มีดังนี้
1. Subclipse จากเว็บไซต์ http://subclipse.tigris.org/
2. Groovy จากเว็บไซต์ http://groovy.codehaus.org/
3. JBoss Tools เลือกเฉพาะ plug-ins ชื่อ FreeMarker จากเว็บไซต์ http://www.jboss.org/tools


เอกสารอ้างอิง
http://www.eclipse.org/downloads/
https://help.ubuntu.com/community/Java
http://subclipse.tigris.org/
http://groovy.codehaus.org/
http://www.jboss.org/tools

วันอังคารที่ 1 มิถุนายน พ.ศ. 2553

Apache OFBiz ERP/Framework


ERP
เป็นการวางแผนการจัดการทรัพยากรขององค์กร เพื่อให้สามารถจัดการทรัพยากรของระบบให้มีประสิทธิภาพและเกิดประโยชน์สูงสุดต่อองค์กร

โดยทั่วไปในองค์กรมักจะใช้ซอฟต์แวร์สำเร็จรูปในการจัดการข้อมูลในองค์กร ทำให้เกิดความไม่เป็นอันหนึ่งอันเดียวกัน และข้อมูลไม่เป็นรูปแบบเดียวกัน เมื่อมีความต้องการจะนำเอาข้อมูลมาใช้ ความหลากหลายของซอฟต์แวร์ทำให้เกิดความไม่สะดวกในการเรียกข้อมูล ก่อให้เกิดความล่าช้าในการแข่งขันในธุรกิจยุคปัจจุบัน

ดังนั้นจึงมีแนวคิดที่จะนำเอาการวางแผนการจัดการทรัพยากรขององค์กร หรือ ERP มารวมกับระบบสารสนเทศ เพื่อรวบรวมข้อมูลทั้งหมดในองค์กรให้เป็นระบบเดียวกัน ส่งผลให้ง่ายต่อการเรียกใช้ข้อมูล, การควบคุมข้อมูล และข้อมูลทั้งหมดสามารถเชื่อมโยงกันได้อย่างทันเหตุการณ์ (real time)

OFBiz
เป็น Open Source Software ที่ช่วยให้องค์กรหรือธุรกิจสามารถวางแผนการใช้งานทรัพยากรให้เกิดประโยชน์สูงสุด โดยสามารถเชื่อมต่อกับทุกระบบ หรือเชื่อมต่อระบบสำคัญๆ ในองค์กร ช่วยให้ผู้ที่มีความสามารถเข้าถึงข้อมูลใช้ประโยชน์จากข้อมูลที่เชื่อมโยงกันได้อย่างทันเหตุการณ์ (real time)

OFBiz มาจากคำว่า Open For Business อยู่ภายใต้การดูแลของ Apache Foundation ซึ่งมี Developer จากทั่วโลกมาช่วยกันพัฒนา แล้วแจกจ่ายให้ผู้สนใจนำไปใช้งานและพัฒนาต่อโดยไม่เสียค่าใช้จ่าย แต่อยู่ภายใต้เงื่อนไขของ Open Source Software

OFBiz ประกอบด้วยแอพพลิเคชั่นที่ช่วยในการทำงานหลายด้าน เช่น Customer Relationship Management(CRM), Human Resource Management(HRM), Supply Chain Management(SCM), Purchase Management, Inventory, Sales Management, IT Management, Production Management, Data Analysis/Report เป็นต้น

ปัจจุบันมี Open Source ERP จากหลายบริษัท แต่หลักๆ แล้ว จะแบ่งตาม Framework ที่ใช้เป็นพื้นฐานในการพัฒนาได้ดังนี้

1. Apache OFBiz - ERP ที่นำเอา OFBiz ไปเป็นพื้นฐานในการพัฒนา เช่น Opentabs ERP/CRM, Shared OFBiz, OrangeGears ERP/CRM, Atlassian JIRA, Neogia ERP เป็นต้น
2. Open ERP
3. Compiere - ERP ที่นำเอา OFBiz ไปเป็นพื้นฐานในการพัฒนา เช่น Adempier, Open Bravo, Seree ERP เป็นต้น

องค์ประกอบหลักของ OFBiz
1. Application Component
2. Application Development Framework
3. Application Data Model

โครงสร้างของ OFBiz Framework แบ่งเป็น 3 ส่วน คือ
1. Presentation
2. Business Logic
3. Data

บรรยายสำหรับนักศึกษาฝึกงาน Orange Gears
เมื่อวันที่ 1 มิ.ย. 2553 โดย คุณธนกฤติ วงศ์ยืด

เอกสารอ้างอิง
http://www.slideshare.net/orangegears/apache-ofbiz-erp