原文地址:如何动态创建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};
}

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

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

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

    <style> table{ width:500px; font-weight: bold; border: 1px solid #000; border-collapse:collaps ...

  2. 初探原生js根据json数据动态创建table

    初探原生js根据json数据动态创建table 小生以实习生的职位进入了一家非纯软件的公司做asp.net开发,大半个月下来发现公司里居然没有前端工程师,这令我很诧异,跟着公司做项目,发现前端后台没有 ...

  3. JS动态创建Table,Tr,Td并赋值

    JS动态创建Table,Tr,Td并赋值. 成果库修改: 要求主题列表随成果类型改变而改变 网上查询资料后开工,在成果类型下拉框添加change()事件触发Dwr,查询主题集合——动态创建/编辑Tab ...

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

    在开发中,我们遇到以下一个需求. 一个表格左侧有5列是固定存在的,右侧有N列是动态生成的,并且该N列中第一列可输入,第二列是不可编辑的,但是是数字,如果小于0,那么就要显示为红色,重点标识出来. 首先 ...

  5. JS动态创建table

    <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8&quo ...

  6. 动态创建table表格页面出现undefined原因以及修改

    源代码: var html: if(lists) { html += '<a href="https://www.4001149114.com/NLJJ/member/sharecel ...

  7. DOM – 7.动态创建DOM + 8.innerText innerHTML value

    7.动态创建DOM 8.innerText  innerHTML  value 7+8 练习:案例:点击按钮动态增加网站列表,分两列,第一列为网站的名字,第二列为带网站超链接的网站名.增加三行常见网站 ...

  8. JS,Jquery,ExtJs不同脚本动态创建DOM对象

    好久不来写东西了,这段时间太慢了,闲了下来看了几篇文章,觉得很好,同时也许咱们大家都能遇到,所以就把它记录下来... 简单使用JavaScript.JQuery.ExtJs进行DOM对象创建的测试,主 ...

  9. JS、JQuery和ExtJs动态创建DOM对象

    做了个简单使用JavaScript.JQuery.ExtJs进行DOM对象创建的测试,主要是使用JavaScript.JQuery.ExtJs动态创建Table对象.动态Table数据填充.多选控制. ...

随机推荐

  1. Fine报表权限流程分析记录

    Fine报表权限流程分析记录 URL访问三种类型的报表:第一个:BI报表 例如: http://192.25.103.250:37799/WebReport/ReportServer?op=fr_bi ...

  2. 20145315《网络对抗》——注入shellcode以及 Return-to-libc攻击实验

    shellcode 准备一段Shellcode 我用的老师的shellcode:\x31\xc0\x50\x68\x2f\x2f\x73\x68\x68\x2f\x62\x69\x6e\x89\xe3 ...

  3. Educational Codeforces Round 21 Problem F (Codeforces 808F) - 最小割 - 二分答案

    Digital collectible card games have become very popular recently. So Vova decided to try one of thes ...

  4. Python3基础 super 子类调用父类的__init__

             Python : 3.7.0          OS : Ubuntu 18.04.1 LTS         IDE : PyCharm 2018.2.4       Conda ...

  5. eclipse 在线安装反编译插件

    打开eclipse help>install New Software…..>add Name:jd-eclipse_update_site Location:http://jd.beno ...

  6. [不屈的复习] - http://how2j.cn/

    http://how2j.cn/ 该教程网站分得比较规整!

  7. bootstrap datarangepicker如何使用

    本文为博主原创,未经允许不得转载: 下载资源文件: 地址:Github:https://github.com/dangrossman/bootstrap-daterangepicker/ 1.页面引用 ...

  8. OpFlex

    参考: OpFlex Main OpFlex: Building and Running OpFlex Building mkdir -p ~/work pushd work git clone ht ...

  9. http post发送请求

    一: 用java自带URL发送 public synchronized JSONObject getJSON(String url2, String param) { try { URL url = ...

  10. python pandas demo

    1. import pandas as pd web_stats = {,,,,,], ,,,,,], ,,,,,]} df = pd.DataFrame(web_stats) print(df.he ...