import 'bootstrap.css';
import '@mescius/wijmo.styles/wijmo.css';
import './styles.css';
//
import * as wijmo from '@mescius/wijmo';
import * as grid from '@mescius/wijmo.grid';
import { countries, getData } from './data';
//
document.readyState === 'complete' ? init() : window.onload = init;
//
function init() {
// create a CollectionView
let view = new wijmo.CollectionView(getData(), {
sortDescriptions: ['country'],
getError,
});
//
// CollectionView validation function
function getError(item, propName) {
switch (propName) {
case 'country':
return countries.indexOf(item.country) < 0 ? '無効な国です。' : '';
case 'downloads':
case 'sales':
case 'expenses':
return item[propName] < 0 ? '負の値を入力できません。' : '';
case 'active':
return item.active && item.country.match(/^(アメリカ|イギリス)$/)
? 'アメリカとイギリスでは有効にできません。'
: '';
case null:
let errors = [];
for (let key in item) {
let err = getError(item, key);
if (err)
errors.push(err);
}
return errors.length > 1
? 'この項目では ' + errors.length + ' 個のエラーがあります。'
: (errors.length == 1 ? errors[0] : null);
}
return null;
}
//
// create grid for editing with validation
let theGrid = new grid.FlexGrid('#theGrid', {
autoGenerateColumns: false,
columns: [
{ binding: 'country', header: '国' },
{ binding: 'downloads', header: 'DL数', width: 80 },
{ binding: 'sales', header: '売上', width: 80 },
{ binding: 'expenses', header: '費用', width: 80 },
{ binding: 'active', header: '有効' },
],
itemsSource: view
});
//
// use getError to provide form validation
let theItem = {};
//
document.getElementById('theForm').addEventListener('input', e => {
let input = e.target;
let propName = input.id;
//
theItem[propName] = input.value;
input.setCustomValidity(view.getError(theItem, propName));
});
//
document.getElementById('theForm').addEventListener('submit', e => e.preventDefault());
}
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<title>MESCIUS Wijmo CollectionView Validation</title>
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<!-- SystemJS -->
<script src="https://cdnjs.cloudflare.com/ajax/libs/systemjs/0.21.5/system.src.js" integrity="sha512-skZbMyvYdNoZfLmiGn5ii6KmklM82rYX2uWctBhzaXPxJgiv4XBwJnFGr5k8s+6tE1pcR1nuTKghozJHyzMcoA==" crossorigin="anonymous"></script>
<script src="systemjs.config.js"></script>
<script>
System.import('./src/app');
</script>
</head>
<body>
<div class="container-fluid">
<h2>
FlexGridの検証
</h2>
<p>
検証に<b>CollectionView.getError</b>メソッドを使用すると、
<b>FlexGrid</b>はデータセルと行ヘッダーセルにエラーを表示します。
</p>
<p>
デフォルトでは、行ヘッダーセルにアイテム内のすべてのエラーが表示されます。
その動作をカスタマイズするには、 <b>getError</b>が特定のプロパティなしで呼び出されたときに
行ヘッダーに表示するエラーメッセージを返します(prop == null)。
</p>
<p>
無効な国または負の数を入力して、グリッドがエラーを識別し、無効なエントリがコミットされないようにする方法を確認してください。
</p>
<div id="theGrid">
</div>
<h2>
フォームの検証
</h2>
<p>
<b>CollectionView.getError</b>プロパティを使用してフォームを検証することもできます。HTML5検証APIの一部である<b>setCustomValidity</b>メソッドを使用して、関数を呼び出して適切な入力要素に結果を適用するだけです。
</p>
<form id="theForm" class="form-inline">
<div class="form-group">
<label for="country">国</label>
<input id="country" type="text" class="form-control" required value="アメリカ">
</div>
<div class="form-group">
<label for="downloads">DL数</label>
<input id="downloads" type="number" class="form-control" required value="123">
</div>
<div class="form-group">
<label for="sales">売上</label>
<input id="sales" type="number" class="form-control" required value="123">
</div>
<button type="submit" class="btn btn-primary">送信</button>
</form>
</div>
</body>
</html>
//
export const countries = ['アメリカ', 'ドイツ', 'イギリス', '日本', 'イタリア', 'ギリシャ', 'スペイン', 'ポルトガル', 'オーストラリア'];
//
export function getData() {
// create some random data
let data = [];
//
for (let i = 0; i < countries.length; i++) {
data.push({
country: countries[i],
downloads: Math.round(Math.random() * 20000),
sales: (countries[i] == 'ドイツ' ? -1 : 1) * Math.random() * 10000,
expenses: (countries[i] == 'アメリカ' ? -1 : 1) * Math.random() * 5000,
active: i % 4 == 0
});
}
//
return data;
}
.form-group {
margin-right: 8px;
}
.form-group > label {
margin-right: 3px;
}
.form-group input:invalid {
border-color: red;
box-shadow: inset 0 1px 1px rgba(0,0,0,.075), 0 0 8px rgba(255,0,0,.6);
}
(function (global) {
System.config({
transpiler: 'plugin-babel',
babelOptions: {
es2015: true
},
meta: {
'*.css': { loader: 'css' }
},
paths: {
// paths serve as alias
'npm:': 'node_modules/'
},
// map tells the System loader where to look for things
map: {
'jszip': 'npm:jszip/dist/jszip.js',
'@mescius/wijmo': 'npm:@mescius/wijmo/index.js',
'@mescius/wijmo.input': 'npm:@mescius/wijmo.input/index.js',
'@mescius/wijmo.styles': 'npm:@mescius/wijmo.styles',
'@mescius/wijmo.cultures': 'npm:@mescius/wijmo.cultures',
'@mescius/wijmo.chart': 'npm:@mescius/wijmo.chart/index.js',
'@mescius/wijmo.chart.analytics': 'npm:@mescius/wijmo.chart.analytics/index.js',
'@mescius/wijmo.chart.animation': 'npm:@mescius/wijmo.chart.animation/index.js',
'@mescius/wijmo.chart.annotation': 'npm:@mescius/wijmo.chart.annotation/index.js',
'@mescius/wijmo.chart.finance': 'npm:@mescius/wijmo.chart.finance/index.js',
'@mescius/wijmo.chart.finance.analytics': 'npm:@mescius/wijmo.chart.finance.analytics/index.js',
'@mescius/wijmo.chart.hierarchical': 'npm:@mescius/wijmo.chart.hierarchical/index.js',
'@mescius/wijmo.chart.interaction': 'npm:@mescius/wijmo.chart.interaction/index.js',
'@mescius/wijmo.chart.radar': 'npm:@mescius/wijmo.chart.radar/index.js',
'@mescius/wijmo.chart.render': 'npm:@mescius/wijmo.chart.render/index.js',
'@mescius/wijmo.chart.webgl': 'npm:@mescius/wijmo.chart.webgl/index.js',
'@mescius/wijmo.chart.map': 'npm:@mescius/wijmo.chart.map/index.js',
'@mescius/wijmo.gauge': 'npm:@mescius/wijmo.gauge/index.js',
'@mescius/wijmo.grid': 'npm:@mescius/wijmo.grid/index.js',
'@mescius/wijmo.grid.detail': 'npm:@mescius/wijmo.grid.detail/index.js',
'@mescius/wijmo.grid.filter': 'npm:@mescius/wijmo.grid.filter/index.js',
'@mescius/wijmo.grid.search': 'npm:@mescius/wijmo.grid.search/index.js',
'@mescius/wijmo.grid.grouppanel': 'npm:@mescius/wijmo.grid.grouppanel/index.js',
'@mescius/wijmo.grid.multirow': 'npm:@mescius/wijmo.grid.multirow/index.js',
'@mescius/wijmo.grid.transposed': 'npm:@mescius/wijmo.grid.transposed/index.js',
'@mescius/wijmo.grid.transposedmultirow': 'npm:@mescius/wijmo.grid.transposedmultirow/index.js',
'@mescius/wijmo.grid.pdf': 'npm:@mescius/wijmo.grid.pdf/index.js',
'@mescius/wijmo.grid.sheet': 'npm:@mescius/wijmo.grid.sheet/index.js',
'@mescius/wijmo.grid.xlsx': 'npm:@mescius/wijmo.grid.xlsx/index.js',
'@mescius/wijmo.grid.selector': 'npm:@mescius/wijmo.grid.selector/index.js',
'@mescius/wijmo.grid.cellmaker': 'npm:@mescius/wijmo.grid.cellmaker/index.js',
'@mescius/wijmo.nav': 'npm:@mescius/wijmo.nav/index.js',
'@mescius/wijmo.odata': 'npm:@mescius/wijmo.odata/index.js',
'@mescius/wijmo.olap': 'npm:@mescius/wijmo.olap/index.js',
'@mescius/wijmo.rest': 'npm:@mescius/wijmo.rest/index.js',
'@mescius/wijmo.pdf': 'npm:@mescius/wijmo.pdf/index.js',
'@mescius/wijmo.pdf.security': 'npm:@mescius/wijmo.pdf.security/index.js',
'@mescius/wijmo.viewer': 'npm:@mescius/wijmo.viewer/index.js',
'@mescius/wijmo.xlsx': 'npm:@mescius/wijmo.xlsx/index.js',
'@mescius/wijmo.undo': 'npm:@mescius/wijmo.undo/index.js',
'@mescius/wijmo.interop.grid': 'npm:@mescius/wijmo.interop.grid/index.js',
'@mescius/wijmo.touch': 'npm:@mescius/wijmo.touch/index.js',
'@mescius/wijmo.cloud': 'npm:@mescius/wijmo.cloud/index.js',
'@mescius/wijmo.barcode': 'npm:@mescius/wijmo.barcode/index.js',
'@mescius/wijmo.barcode.common': 'npm:@mescius/wijmo.barcode.common/index.js',
'@mescius/wijmo.barcode.composite': 'npm:@mescius/wijmo.barcode.composite/index.js',
'@mescius/wijmo.barcode.specialized': 'npm:@mescius/wijmo.barcode.specialized/index.js',
'jszip': 'npm:jszip/dist/jszip.js',
'bootstrap.css': 'npm:bootstrap/dist/css/bootstrap.min.css',
'css': 'npm:systemjs-plugin-css/css.js',
'plugin-babel': 'npm:systemjs-plugin-babel/plugin-babel.js',
'systemjs-babel-build':'npm:systemjs-plugin-babel/systemjs-babel-browser.js'
},
// packages tells the System loader how to load when no filename and/or no extension
packages: {
src: {
defaultExtension: 'js'
},
"node_modules": {
defaultExtension: 'js'
},
}
});
})(this);