js正则验证会计公式

因工作要求,需要验证会计使用的公式,写了一个js类,在此一做记录


/**************************************************
* 此类用来验证公式合法性<br>
* 公式输入说明:<br>
* 1、可以输入数值常量。50、20.3<br>
* 2、可以报表单元格列表中选取需要的单元格。R.001.A04<br>
* 3、可以使用+,-,*,/运算符和括号(,)。<br>
* <br>
* 正确公式示例:<br>
* //正确公式 (R.001.A04+R.001.A04+R.001.A03)/50-R.001.A06/8.05*166 <br>
* //错误公式 (R.0011.A04+R.001.A04+R.001.A03)/50-R.001.A06/8.05*166<br>
* <br>
* <br>
* 使用示例:<br>
* 创建公式<br>
* var tStr = "(R.001.A04+R.001.A04+R.001.A03)/50-R.001.A06/8.05*166";<br>
* 创建验证公式对象<br>
* var fromsV = new FromsUtil();<br>
* 调用验证方法<br>
* fromsV.validateFroms(tStr);<br>
* 取得验证结果<br>
* alert(fromsV.validateFlag);<br>
* <br>
*************************************************/
FromsUtil = function() {
//加、减、乘、除
this.patt = /[+\-\/\*]/;
//验证公式元素:R.001.A04--(R.001.A04--R.001.A04)--22--22.22
//var patt2 = /^\({0,}R\.\d{3}\.[A-Z]\d{2}\){0,}$|^\d{1,}$|^\d{1,}\.\d{1,}$/;
//常量前后也可有括号
this.patt2 = /^\({0,}R\.\d{3}\.[A-Z]\d{2}\){0,}$|^\({0,}\d{1,}\){0,}$|^\({0,}\d{1,}\.\d{1,}\){0,}$/;
//验证结果
this.validateFlag = true;
//括号个数
this.patt3 = /\(/g;
this.patt4 = /\)/g;
var leftBracket=0;
var rigthBracket=0;
var count = 0; //递归次数--来确定是否有计算符号
//验证公式的主要方法
this.validateFroms = function (str) {
//取得匹配的字符串
var tem1 = str.match(this.patt);
//alert(tem1);
//假如为空为最后一个元素
if(tem1 == null) {
//假如无计算符返回false
if(count==0) {
if(str.match(this.patt) == null) {
this.validateFlag = false;
return;
}
}
//验证最后一个元素
if(!(this.patt2.test(str))) {
this.validateFlag = this.patt2.test(str);
return;
}
//取得括号的个数
while (this.patt3.exec(str) != null) {
leftBracket++;
}
while (this.patt4.exec(str) != null) {
rigthBracket++;
}
//验证括号个烽
if(leftBracket!=rigthBracket) {
this.validateFlag=false;
}
return;
};
//取得位置
var index = str.indexOf(tem1);
//取得要验证的字符串
var str1=str.substring(0,index);
//alert(str1);
//alert(patt2.test(str1));
//验证结果
if(!(this.patt2.test(str1))) {
this.validateFlag = this.patt2.test(str1);
return;
}
//取得括号的个数
while (this.patt3.exec(str1) != null) {
leftBracket++;
}
while (this.patt4.exec(str1) != null) {
rigthBracket++;
}
count ++;
//使用递归
var str2=str.substring(index+1,str.length);
this.validateFroms(str2);
}
}
function processEntries() { const voucher = document.querySelector('input[type="text"]:nth-of-type(1)').value; const date = document.querySelector('input[type="text"]:nth-of-type(2)').value; document.querySelectorAll('.tab tr:not(:first-child)').forEach(row => { const subject = row.querySelector('td:nth-child(2) input').value; if (!subject || !TABLE_CONFIG[subject]) return; // 构造唯一标识(基于凭证字号、日期、摘要) const entryId = `${voucher}-${date}-${row.querySelector('td:first-child input').value}`; //const entryId = `${voucher}-${date}-${row.querySelector('td:first-child input').value}-${row.querySelector('td:nth-child(3) input').value}-${row.querySelector('td:nth-child(4) input').value}` // 查找是否已存在相同标识的行 const table = document.querySelector(TABLE_CONFIG[subject].table); const existingRow = findRowById(table, entryId); if (existingRow) { // 如果行已存在,更新当前行的字段内容 updateRowFields(existingRow, row); } else { // 如果行不存在,插入新行 const entry = { date: date, voucher: voucher, abstract: row.querySelector('td:first-child input').value, debit: parseFloat(row.querySelector('td:nth-child(3) input').value) || 0, credit: parseFloat(row.querySelector('td:nth-child(4) input').value) || 0 }; updateDetailTable(TABLE_CONFIG[subject].table, entry); } }); } // 根据唯一标识查找已存在的行 function findRowById(table, entryId) { const rows = Array.from(table.rows).slice(2, -2); // 跳过表头和合计行 return rows.find(row => { const rowId = `${row.cells[1].textContent}-${row.cells[0].textContent}-${row.cells[2].textContent}`; return rowId === entryId; }); } // 更新已存在行的字段内容 function updateRowFields(existingRow, sourceRow) { // 更新日期 existingRow.cells[0].textContent = sourceRow.querySelector('td:nth-child(1) input').value; // 更新凭证字号 existingRow.cells[1].textContent = sourceRow.querySelector('td:nth-child(2) input').value; // 更新摘要 existingRow.cells[2].textContent = sourceRow.querySelector('td:nth-child(3) input').value; // 更新借方金额 existingRow.cells[4].textContent = sourceRow.querySelector('td:nth-child(4) input').value; // 更新贷方金额 existingRow.cells[5].textContent = sourceRow.querySelector('td:nth-child(5) input').value; // 重新计算方向和余额 const debit = parseFloat(existingRow.cells[4].textContent.replace(/[^0-9.-]/g, '')) || 0; const credit = parseFloat(existingRow.cells[5].textContent.replace(/[^0-9.-]/g, '')) || 0; const balance = debit - credit; existingRow.cells[6].textContent = balance > 0 ? '借' : '贷'; // 方向 existingRow.cells[7].textContent = Math.abs(balance).toFixed(2); // 余额 // 更新合计数据 updateTotals(existingRow.closest('table')); }
03-13
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值