Eclipse Birt的列绑定错误
今天用eclipse birt做报表时出现了一个错误困扰半天啊...
Column Binding "XXX" is incorrect:the parent query column bindings which include aggregations cannot be used in column bindings of subquery.翻译一下大概是:"父查询绑定列包含了聚合函数不能被用于子查询绑定列",什么父查询子查询?晕了!!
goole一下吧,只找到一篇相关的文章看了一下大概是说"原因是表格上的字段绑定,必需设置bingding到table,而不能设置binding到data元素",跟我这个完全没有关系,不过受到启发我查看了一下我的报表结构,一个Gird中放了一个Chart,Grid上绑定了数据集并且还自定义了几个列(使用了聚合函数),Chart上没有绑定数据,但是它自动使用了Grid的绑定数据。
-------我明白了原来什么父啊子的指的是容器的父子关系啊,Chart放在了Grid中因此Grid是父,Chart是它的子,那就给Chart绑定一个数据集呗!
搞定,收工。
eclipse birt使用脚本实现换行
使用Eclipse Birt有一段时间了,对birt的功能还是比较满意的,基本能满足工作中的需要,当然也有一些不尽人意的地方,比如birt中较长数据的换行问题,笔者使用过程中发现birt自动换行的功能似乎只能换一次行,当换行后的下一行文本仍较长时就不换了,而且换行的临界选择似乎也有点问题,比如有中英文结合的情况时通常是在中英文交界处换行而不是根据长度换行,如:"中华人民共和国The people's republic of China"换行后变成
"中华人民共和国
The people's republic of China"这样出来的格式显得参差不齐非常不爽。幸好birt支持js脚本,可以通过脚本实现自己需要的功能,个人觉得这也是birt强大之处。因为没有用过其他报表工具所以无从比较,birt足够满足日常工作需要了。
下面就介绍脚本的写法,有两种方法:
一、在onRender事件中写脚本,选中要换行的列,切换到脚本页面,选择onRender事件,输入以下脚本
var name = this.getValue(); name = name.toString(); var interval = 35;//指定换行的条件,当字符超过这个数时换一行,可以根据列的宽度指定合适的长度 var n = name.length() / interval|0; var re = name.length() % interval; var rslt = ""; for(var i=0; i<n; i++){ var subs = name.substr(i*interval,interval); rslt += ( subs + "\n"); } if(re > 0){ rslt += name.substr(n*interval,re); } this.setDisplayValue(rslt);
二、使用text控件,text控件支持动态表达式,这就为我们的脚本有了用武之地
<VALUE-OF format="HTML">//使用html格式,就可以使用<BR>换行符 var name = row["M_NAME"]; var interval = 35;//指定换行的条件,当字符超过这个数时换一行 var n = name.length / interval | 0; var re = name.length % interval; var rslt = ""; for(var i=0; i<n; i++){ var subs = name.substr(i*interval,interval); rslt += ( subs + "<BR>"); } if(re > 0){ rslt += name.substr(n*interval,re); } this.value = rslt; </VALUE-OF>
总结:其实两种方法基本是一样的,除了一些个别细节的地方,如在text表达式中求字符串的长度用的是字符串变量的length属性,而在onRender事件中用的是变量的length()方法;在text表达式中给text赋值用的是this.value属性,而在onRender事件中用的是this.setDisplayValue()方法,还有很多其他地方的细小差别,大家可以自己比较。
还有一个问题是因为中文字符的宽度和西文字符以及数字的宽度都不同因此按字符个数来换行并不是十分准确的,期待朋友们提供更好的解决方法...
Eclipse Birt使用动态sql创建data set
Eclipse Birt可以使用sql query来创建数据集,通过在报表中设置参数(Parameter)来传递客户端的用户输入,同时可以设置data set参数并且与报表参数关联(这两个参数是不同的),这样可以将客户端的用户输入传递给data set的参数供sql query的where语句使用,这是通常的创建sql语句的做法,但是种方法只适合固定的sql语句,对于动态的sql就不适用了。
如有这样一条sql query
select user.first_name firstName,user.second_name secondName,user.sex sex,user.age age,user.birth_day birthDay from user_table user where user.second_name = ? and user.sex = ?
定义两个报表参数SecondName,Sex定义两个data set参数secondname,sex并与报表参数关联,用户查看报表时必须要输入姓和性别作为查询条件,如果不输则对应的参数值就是null,假如查姓李的人,性别不输,则sql语句就变成这样select ... from ... where user.second_name = '李' and user.sex = null 显然这样的sql是查不出结果的,哪有sex是null的人呢?对于这种情况应该是如果用户某一个参数不输则不将它做为查询条件,上面的sql应该变成select ... from ... where user.second_name = '李'就合理了,把所有姓李的人不分男女都查出来,这就需要使用动态sql语句了,方法有两种:
一、date set编辑器中有property binging的属性,可以在右边的query text框中设输入sql语句,或者用expression生成器辅助生成sql语句,上面的例子用种方法应该写成
var query = " select user.first_name firstName,user.second_name secondName,user.sex sex,user.age age,user.birth_day birthDay"; query += " from user_table user"; query += " where user.second_name = ?" if(param["Sex"].value != null){ query += " and user.sex = '" + param["Sex"].value + "'"; } this.text = query;
二、利用script,在data set的before open事件中
var query = " select user.first_name firstName,user.second_name secondName,user.sex sex,user.age age,user.birth_day birthDay"; query += " from user_table user"; query += " where user.second_name = ?" var sex = reportContext.getParameterValue("Sex"); if(sex != null){ query += " and user.sex = '" + sex + "'"; } this.queryText = query;
这两种方法的区别在于:第一种方法是在报表run的时候创建data set,因此在data set编辑器里是没有办法像通常那样预览的 而第二种方法是写在data set的beforeOpen中的,所以在打开data set之前会创建data set因此可以像通常那样预览数据集结果等等
第一种方法中用param["sex"].value获得Sex参数值,而第二种方法需要利用reportContext来获取Sex参数的值
另外需要注意的是在拼凑sql时要细心,例如如果参数是String类型时前后一定要加上一对''(具体的取决于所使用的数据库,笔者使用的是oracle)