oaf 动态创建table vo (转)

原文地址:如何动态创建table

需求:

因为系统中有几千个QA plan 但是不能手动创建几千个 质量收集页面
所有需要根据 不同的plan 动态创建对应的 质量收集页面。

但是创建tabel 都要绑定一个 具体的vo  而我需要一个动态的
如果用 select ... from dual 的话 字段的个数如何动态?

感谢答主sumury

方案0.1版本

//用以下的代码可以实现,但是有一点不足,就是当数据量比较多的时候,无法按照指定的行数进行分页显示。

//先在页面上创建一个advancedTable,名字叫“region5”,即可。

public void processRequest(OAPageContext pageContext, OAWebBean webBean)
  {
    super.processRequest(pageContext, webBean);

    final String CHILD_DATA_LIST = "childDataList";
    final String TEXT = "text";
    int COLUMN_COUNT = 0;

    // write your Personalization SQL in here.
    StringBuffer sbfSQL = new StringBuffer(200);
    // set title in the advancedTable
    sbfSQL.append("SELECT ‘名‘ user_name, ‘ID‘ user_id, ‘開始日‘  FROM dual  \r\n");
    sbfSQL.append("UNION ALL \r\n");
    sbfSQL.append("SELECT fu.user_name, to_char(fu.user_id), to_char(fu.start_date) FROM fnd_user fu");

    ResultSet rs = null;
    Statement s = null;
    Connection con = pageContext.getApplicationModule(webBean).getOADBTransaction().getJdbcConnection();

    try
    {
      s = con.createStatement(ResultSet.TYPE_SCROLL_SENSITIVE, ResultSet.CONCUR_UPDATABLE);
      rs = s.executeQuery(sbfSQL.toString());
      ResultSetMetaData rsmd = rs.getMetaData();
      COLUMN_COUNT = rsmd.getColumnCount();
      rs.last();
      int intRowCount = rs.getRow();
      rs.first();

      // find the advancedTable
      OAAdvancedTableBean tableBean = (OAAdvancedTableBean)webBean.findChildRecursive("region5");
      tableBean.setViewUsageName("");

      for (int i = 1; i <= COLUMN_COUNT; i++)
      {
        // create a column
        OAColumnBean cb = (OAColumnBean)createWebBean(pageContext, COLUMN_BEAN, null, null);
        tableBean.addIndexedChild(cb);

        // create MessageStyledText
        OAMessageStyledTextBean mst = (OAMessageStyledTextBean)createWebBean(pageContext, MESSAGE_STYLED_TEXT_BEAN, null, null);
        mst.setTextBinding(TEXT + i);
        cb.addIndexedChild(mst);

        // create title
        OASortableHeaderBean shb = (OASortableHeaderBean)createWebBean(pageContext, SORTABLE_HEADER_BEAN, null, null);
        shb.setPrompt(rs.getString(i));
        cb.setColumnHeader(shb);

        if (i != 1)
        {
          UINodeList colList = new DataObjectListNodeList(mst, new DataBoundValue(CHILD_DATA_LIST + i));
          cb.setIndexedNodeList(colList);
        }
      }

      // get row count;
      DictionaryData rowData[] = new DictionaryData[intRowCount - 1];
      int intRowLoop = 2;
      // loop row
      while (rs.next())
      {
        // setting 1 column cell value
        rowData[intRowLoop - 2] = new DictionaryData(TEXT + "1", rs.getString(1));

        // setting 2 to N column cell value
        DictionaryData otherColumn[] = new DictionaryData[COLUMN_COUNT];
        // loop column
        for (int j = 2; j <= COLUMN_COUNT; j++)
        {
          otherColumn[j - 2] = new DictionaryData(TEXT + j, rs.getString(j));
          rowData[intRowLoop - 2].put(CHILD_DATA_LIST + j , new ArrayDataSet(otherColumn));
        }
        intRowLoop++;
      }
      tableBean.setTableData(new ArrayDataSet(rowData));
    }
    catch (SQLException se)
    {
      se.printStackTrace();
    }
    finally
    {
      if (s != null)
      {
        try
        {
          s.close();
        }
        catch (SQLException se)
        {
          se.printStackTrace();
        }
      }

      if (rs != null)
      {
        try
        {
          rs.close();
        }
        catch (SQLException se)
        {
          se.printStackTrace();
        }
      }
    }
  }

方案0.2版本

/*今天空下来,又研究了一下这个问题,有种比之前的方法更好些的方法。贴出来大家看看。
1、创建一个DynVO,里面不需要有SQL,而且只需要生成xml,就可以了
2、把DynVO加到AM中去。
3、在页面上创建advancedTable,取名region6,并且绑定DynVO1
4、在CO中追加以下代码:*/
  public void processRequest(OAPageContext pageContext, OAWebBean webBean)
  {
    super.processRequest(pageContext, webBean);
    OAApplicationModule am = pageContext.getApplicationModule(webBean);
    String[][] strTitleAndAttributeName = (String[][])am.invokeMethod("initDynVO");
    OAAdvancedTableBean tableBean = (OAAdvancedTableBean)webBean.findChildRecursive("region6");
    for (int i = 0; i < strTitleAndAttributeName[0].length; i++)
    {
      // create a column
      OAColumnBean cb = (OAColumnBean)createWebBean(pageContext, COLUMN_BEAN, null, null);
      tableBean.addIndexedChild(cb);
      // create MessageStyledText
      OAMessageStyledTextBean mst = (OAMessageStyledTextBean)createWebBean(pageContext, MESSAGE_STYLED_TEXT_BEAN, null, null);
      mst.setViewAttributeName(strTitleAndAttributeName[1][i]);
      cb.addIndexedChild(mst);
      // create title
      OASortableHeaderBean shb = (OASortableHeaderBean)createWebBean(pageContext, SORTABLE_HEADER_BEAN, null, null);
      shb.setPrompt(strTitleAndAttributeName[0][i]);
      cb.setColumnHeader(shb);
    }
  }

//5、在AM中追加以下代码

  public String[][] initDynVO()
  {
    StringBuffer sbfSQLTitle = new StringBuffer(200);
    // set title in the advancedTable
    sbfSQLTitle.append("SELECT ‘名‘ user_name, ‘ID‘ user_id, ‘開始日‘ start_date FROM dual");
    StringBuffer sbfSQLValue = new StringBuffer(200);
    sbfSQLValue.append("SELECT fu.user_name, to_char(fu.user_id), to_char(fu.start_date) FROM fnd_user fu");

    OAViewObjectImpl vo = getDynVO1();
    vo.setQuery(sbfSQLTitle.toString());
    vo.executeQuery();
    // set attribute to updateable
    Row row = vo.first();
    AttributeDefImpl[] ad = (AttributeDefImpl[])vo.getViewDefinition().getAttributeDefs();
    int intAttributeCount = ad.length;
    String[] strTitle = new String[intAttributeCount];
    for (int i = 0; i < intAttributeCount; i++)
    {
      ad[i].setUpdateableFlag(AttributeDefImpl.UPDATEABLE);
      // get title from SQL result.
      strTitle[i] = (String)row.getAttribute(i);
    }
    vo.setQuery(sbfSQLValue.toString());
    vo.executeQuery();
    return new String[][]{strTitle, row.getAttributeNames()};
  }

/*完毕。

残留问题:
虽然在AM的代码中写了ad[i].setUpdateableFlag(AttributeDefImpl.UPDATEABLE);
发现,对于VO中的数据,我们是可以更新了。
但是确不能insert和remove数据。
目前还想不出怎么解决这个问题。*/

终极版1.0

/*通过一天的研究,解决了原来的代码中,无法对row进行insert和remove的处理。

原来的代码,对VO进行了2次查询,第一次检索出advancedTable的标题,第二次检索出实际数据。

但是比较好的做法是,使用UNION ALL,将两段查询合并,进行一次查询。

对于查询结果而言,第一行是标题,待我们提取出来以后,再删除它,剩下的就是要在advancedTable中显示的数据了。

然而,使用原来的代码,运行到row.remove()方法的时候,就是抛出错误消息Oracle.jbo.ReadOnlyViewObjectException: JBO-25016.

所以,我们需要跟踪row.remove()方法看看,到底哪里抛出了这个错误消息。

通过反编译 ViewRowImpl.class,发现*/
void doRemove(int i)
{
  ArrayList arraylist = null;
  try
  {
    useInner();
    if(mInner.mVO.isReadOnly())
      throw new ReadOnlyViewObjectException(mInner.mVO.getName());
......
}

//然后再查看ViewObjectImpl.class 文件
public boolean isReadOnly()
{
  return mViewDef.isReadOnly() && getDynamicAttributeCount() == 0;
}

/*到这里发现一点苗头了,我们可以通过修改mViewDef.isReadOnly()或getDynamicAttributeCount() == 0;中的任何一个值,来实现我们的目的。

通过进一步的调查发现很难修改mViewDef.isReadOnly()的值,所以,只能将重点放在修改getDynamicAttributeCount()上了。

通过读源代码发现,使用vo.addDynamicAttribute()方法,可以改变getDynamicAttributeCount()的返回值,于是就实验了一把。结果成功了。

以下是修改好的代码,可以对VO中的数据进行insert、remove、update操作了。*/

   public String[][] initDynVO()
  {
    StringBuffer sbfSQL = new StringBuffer(200);
    // set title in the advancedTable
    sbfSQL.append("SELECT ‘名‘ user_name, ‘ID‘ user_id, ‘開始日‘ start_date FROM dual ");
    sbfSQL.append("UNION ALL ");
    sbfSQL.append("SELECT fu.user_name, to_char(fu.user_id), to_char(fu.start_date) FROM fnd_user fu");

    OAViewObjectImpl vo = getDynVO1();
    vo.setQuery(sbfSQL.toString());
    vo.executeQuery();
    // set attribute to updateable
    Row row = vo.first();
    AttributeDefImpl[] ad = (AttributeDefImpl[])vo.getViewDefinition().getAttributeDefs();
    int intAttributeCount = ad.length;
    String[] strTitle = new String[intAttributeCount];
    String[] strAttributeNames = row.getAttributeNames();
    for (int i = 0; i < intAttributeCount; i++)
    {
      // for update the attribute
      ad.setUpdateableFlag(AttributeDefImpl.UPDATEABLE);

      // get title from SQL result.
      strTitle = (String)row.getAttribute(i);
    }
    // for insert and remove row.
    vo.addDynamicAttribute("DUMMY_ATTRIBUTE");
    row.remove();
    return new String[][]{strTitle, strAttributeNames};
  }

感谢这些热爱思考且愿意分享代码的人们。

时间: 2024-08-07 04:31:43

oaf 动态创建table vo (转)的相关文章

js基础例子动态创建table实例

<style> table{ width:500px; font-weight: bold; border: 1px solid #000; border-collapse:collapse; font-size:20px; text-align:center; margin: 0 auto; } td{ border: 1px solid #000000; } </style> <script> window.onload=function () { //创建一个数据

OAF 动态创建组件以及动态绑定属性

在开发中,我们遇到以下一个需求. 一个表格左侧有5列是固定存在的,右侧有N列是动态生成的,并且该N列中第一列可输入,第二列是不可编辑的,但是是数字,如果小于0,那么就要显示为红色,重点标识出来. 首先,我们假设你的VO中的SQL查询已经做好了,并且每次能动态确定到需要插入几列. //获得你要生成的列总数 int n=Integer.valueOf(String.valueOf(am.invokeMethod("getCloumnCount"))) ; OATableBean table

JS动态创建table

<!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8"> <title></title> <style> *{ margin: 0; padding: 0; } table{ width: 500px; border-color: chartreuse; border-collapse: collapse; } ta

使用createElement函数来动态创建table的问题

在我们使用ajax时,最重要的就是success函数中对于页面的动态生成,现在的情景就是使用ajax动态生成table或者其他列表格式来进行查询的输出.我去网上查过一些资料,关于动态生成table的例子,大多是通过拼写html文本来实现,这个缺点就是如果数据量很大,js的性能不高.我参考了一些资料后,决定使用动态生成html元素来进行显示.核心函数就是document.createElement(string tagName)来生成table,thead,tbody,tr,th,td等元素.代码

动态创建table

<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> Insert title here <div> <input id="add" type="button" value="新增" onclick="add()">

js制作动态创建table行与修改等编辑表单功能特效

<script type="text/javascript"> //Create TR var nName = new Array("Tony","Mika","Neo","Oi","Kim","Park","Mr Lee","Tasky","Saco","Novel"); v

《项目经验》--后台一般处理程序向前台JS文件传递JSON,JS解析JSON,将数据显示在界面--显示在DropDownList 或 显示在动态创建的table中

http://blog.csdn.net/mazhaojuan/article/details/8599167 先看一下我要实现的功能界面: 这篇文章主要介绍:后台一般处理程序把从数据库查找的数据,转换成JSON,然后传递到前台JS文件中,JS解析JSON数据,并将数据显示在界面,主要介绍两种显示方式,显示在DropDownList控件 or 显示在动态创建的Table表中.   本文主要介绍两个地方: 1.根据学年查询学期信息的实现--JS将解析的JSON数据绑定到DropDownList框

用javascript动态创建并提交表单form,表格table

<script> //helper function to create the formfunction getNewSubmitForm(){ var submitForm = document.createElement("FORM"); document.body.appendChild(submitForm); submitForm.method = "POST"; return submitForm;} //helper function t

mybatis3动态创建表,判断表是否存在,删除表

1.mybatis3动态创建表,判断表是否存在,删除表 mapper配置文件: <span style="font-size:18px;"><?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/