単体テストランナー「Karma」をインストール
テストフレームワーク「Mocha」をインストール
JS、CSSを1ファイルにまとめる webpack を利用
設定ファイル「karma.conf.js」が作成され、指定ブラウザにKarmaが起動
親のVue にコンポーネントを登録することで、子のイベントを管理できる
var コンポーネント変数 = Vue.extend({
template: 'HTML', //methods で登録したメソッドを実行可能
data: function () { return { データ名 : 値 } }, //コンポーネント内で利用できる変数を登録
methods: { コンポーネント内のメソッド } //template 内で利用可能
)}
親に登録しない場合は、Vue.component() でもよい
var コンポーネント変数 = Vue.component('counter-button', {
template: 'HTML',
data: function () { return { データ名 : 値 } },
methods: { コンポーネント内のメソッド }
})
new Vue({
el: "#id",
components : { 'HTMLタグ名' : コンポーネント変数 }, //タグ名はケバブケース(kebab-case)
data: { オブジェクト } //Element内のオブジェクト
methods: { メソッド } //Element内のメソッド
}
{{fruit.name}}
:
//v-on:increment カスタムイベント設定し、他のメソッドを実行
Total: {{total}}
var counterButton = Vue.extend({
template: '{{counter}} amounts ',
data: function () {
return {
counter: 0
}
},
methods: {
addToCart: function () {
this.counter += 1
this.$emit('increment') //add custom event
}
}
})
new Vue({
el: '#fruits-counter',
components: {
'counter-button': counterButton
},
data: {
total: 0,
fruits: [
{ name: 'Pear'},
{ name: 'Strawbery'}
]
},
methods: {
incrementCartStatus: function() {
this.total += 1
}
}
});
「jQuery LoadingOverlay」をネイティブで実装してみる。
Github: yoo16/html_samples
This is loading1 element areas.
This is loading1 element areas.
This is loading1 element areas.
This is loading1 element areas.
This is loading1 element areas.
This is loading1 element areas.
This is loading2 element areas.
This is loading2 element areas.
This is loading2 element areas.
This is loading2 element areas.
This is loading2 element areas.
This is loading2 element areas.
var LoadingOverlay = function() {
var _this = this
this.loading_overlay_name = 'loading_overlay';
this.show = function(selector, options) {
_this.hide(selector);
var style = "";
style+= "position: absolute;";
style+= "z-index: 1050;";
style+= "background-color: rgba(255, 255, 255, 0.2);";
style+= "background-position: center center;";
style+= "background-repeat: no-repeat;";
var top = "0;";
var left = "0;";
var width = "100%;";
var height = "100%;";
var background_size = 'contains';
var element = document.body;
if (selector) {
element = document.getElementById(selector);
var rect = element.getBoundingClientRect();
top = rect.top;
left = rect.left;
width = element.clientWidth;
height = element.clientHeight;
background_size = _this.backgroundSize(element);
let td = element.closest('td');
if (td) {
top = 0;
left = 0;
}
}
style+= "background-size: " + background_size + ";";
style+= "top: " + top + "px;";
style+= "left: " + left + "px;";
style+= "width: " + width + "px;";
style+= "height: " + height + "px;";
style+="background-image: url(" + loading_image + ");";
_this.loading_overlay_element = document.createElement('div');
_this.loading_overlay_element.classList.add(_this.loading_overlay_name);
_this.loading_overlay_element.style = style;
element.appendChild(_this.loading_overlay_element);
//timer
function hideHandler() {
clearInterval(_this.timer);
_this.timer = null;
_this.hide(selector);
}
if (options && options.timeout > 0) {
if (!_this.timer) _this.timer = setInterval(hideHandler, options.timeout);
}
}
this.hide = function(selector) {
var parent_element;
if (selector) {
parent_element = document.getElementById(selector);
} else {
parent_element = document.body;
}
[].forEach.call(parent_element.children, function(element) {
if (element.classList.contains(_this.loading_overlay_name)) {
parent_element.removeChild(element);
}
});
}
this.backgroundSize = function(element) {
background_size = "contains";
var img = new Image();
img.src = loading_image;
var width_rate = 0;
var height_rate = 0;
if (img.width > element.clientWidth) {
width_rate = element.clientWidth / img.width;
}
if (img.height > element.clientHeight) {
height_rate = element.clientHeight / img.height;
}
var rate = 0;
if (width_rate > height_rate) {
rate = width_rate;
} else {
rate = height_rate;
}
if (rate > 0) background_size = rate * 30 + "%;"
return background_size;
}
}
var loading_image = "";
var loading_overlay = new LoadingOverlay();
document.createElement('div') で LoadingOver Element を作成し、style を動的に設定後、ターゲットに追加(appendChild()) している。 (ターゲットがない場合は document.body)
styleはターゲットの位置やサイズを取得して absolute で位置合わせしている。 ローディング画像(loading_image)は、画像をBase64に変換しているが、URLでも構わない。 backgroundSize() ではローディング画像のサイズを調整している。 ローディング画像が親より小さければ「background_size: contains」、大きければ「%」で調整する。
指定したIDの子要素の中の、LoadingOver Element のクラスを検索して削除(removeChild()) これで要素毎のローディングにも対応できる。
クラス名や「rel」アトリビュートを指定して、新規ウィンドウで開く
rel New Window
class New Window
HTML属性にウィンドウ名「window_name」、オプション「window_option」を指定した
rel属性を指定した querySelectorAll() で NodeList を取得して click イベントを登録 (IEだと単にループで回せないので、一工夫必要)
function loadPopupRelEvent(rel_name) {
var popupEvent = function(event) {
var window_name = '_blank';
var window_option = null;
if (this.getAttribute('window_name')) window_name = this.getAttribute('window_name');
if (this.getAttribute('window_option')) window_option = this.getAttribute('window_option');
window.open(this.href, window_name, window_option).focus()
event.preventDefault();
event.stopPropagation();
}
var query_string = '[rel = "' + rel_name + '"]';
let elements = document.querySelectorAll(query_string);
if (!elements) return;
[].forEach.call(elements, function (element) {
element.addEventListener('click', popupEvent, false);
});
}
document.addEventListener('DOMContentLoaded', function () {
loadPopupRelEvent('popup', params);
});
getElementsByClassName() で HTMLCollection を取得して click イベントを登録 (IEだと単にループで回せないので、一工夫必要)
function loadPopupClassEvent(class_name) {
var popupEvent = function(event) {
var window_name = '_blank';
var window_option = null;
if (this.getAttribute('window_name')) window_name = this.getAttribute('window_name');
if (this.getAttribute('window_option')) window_option = this.getAttribute('window_option');
window.open(this.href, window_name, window_option).focus()
event.preventDefault();
event.stopPropagation();
}
let elements = document.getElementsByClassName(class_name);
if (!elements) return;
[].forEach.call(elements, function (element) {
element.addEventListener('click', popupEvent, false);
});
}
document.addEventListener('DOMContentLoaded', function () {
loadPopupClassEvent('popup_class', params);
});
要素にclickイベントを登録し、正規表現でURLパースする
テーブルソートは、jQuery(tablednd.js)で比較的簡単に実装できる。 tablednd はマウスイベントを駆使して実装している感じだが、ここではHTML5のDND(Drag & Dropイベント)を利用して実装してみる。
まずは仕様を確認 ・HTML Living Standard:Drag and drop ・ネイティブ HTML5 ドラッグ&ドロップ
tableの並び替え&アップデートスクリプトのライブラリ(PwJS)を作成してみた。
PwJSライブラリ *PwTableDND.js, PwNode.jsを利用 html_samples/sotrable_table.html
色々と制御する必要があるので、ここでは主要部分だけ記述 *並び替えの保持やAPI更新は記述なし
「dragstart」「dragover」「drop」などのDNDイベントを登録して、Element取得・操作を行う
//各イベントハンドラ
function handleDragStart(event) {
//ここにドラッグアイテムの取得処理
event.target.style.opacity = '0.4'; //透明度を有効
event.stopPropagation();
}
function handleDrag(event) {
}
function handleDragEnter(event) {
}
function handleDragOver(event) {
event.dataTransfer.dropEffect = 'move'; //仕様でサポートしている dropEffect
event.preventDefault(); //伝播を止める
return false;
}
function handleDragLeave(event) {
}
function handleDrop(event) {
//ここにターゲットアイテムの取得処理
event.preventDefault(); //伝播を止める
event.stopPropagation(); //リンクキャンセル
}
function handleDragEnd(event) {
event.target.style.opacity = '' //透明度を無効
}
//各イベントの登録
document.addEventListener('dragstart', handleDragStart, false);
document.addEventListener('drag', handleDrag, false);
document.addEventListener('dragenter', handleDragEnter, false)
document.addEventListener('dragover', handleDragOver, false);
document.addEventListener('dragleave', handleDragLeave, false);
document.addEventListener('drop', handleDrop, false);
document.addEventListener('dragend', handleDragEnd, false);
伝播しないように preventDefault(), ドラッグでリンク誤動作しないように stopPropagation() を実行しているが、実際に実行してみるとよくわかる。
並び順のデータID「row-id」属性を持った「tr」で構成 「tbody」内の「tr」を入れ替えるようなスクリプトを作成する
City
Tokyo
Osaka
Nagoya
Kobe
querySelectorAll() などでElement や HTMLCollection を取得して、各行の属性を動的に設定 HTML5では「draggable = true」でドラッグ可能になる
....
var _this = this;
this.table_id = 'sortable-table';
this.row_id_column = 'row-id';
this.body_selector = '';
this.tr_selector = '';
this.sortable_tr_selector = '';
....
this.enableDrag = function() {
_this.loadSelectorColumn();
if (!this.table_id) return;
var row_id;
[].forEach.call(this.getElements(), function(element, index) {
_this.before_rows.push(element);
row_node = PwNode.byElement(element);
if (row_id = row_node.attr(_this.row_id_column)) {
row_node.setAttr('id', pw_row_id_column + row_id);
row_node.setAttr('order', index + 1);
row_node.setAttr('draggable', true);
if (!pw_app.isIE()) {
row_node.setAttr('ondragstart', "event.dataTransfer.setData('text/plain', null)");
}
}
});
}
this.loadSelectorColumn = function() {
_this.body_selector = '#' + _this.table_id + ' tbody';
_this.tr_selector = '#' + _this.table_id + ' tr';
_this.sortable_tr_selector = _this.body_selector + ' tr';
}
this.getElements = function() {
let elements = document.querySelectorAll(_this.sortable_tr_selector);
return elements;
}
FirefoxではHTMLタグに「ondragstart="event.dataTransfer.setData('text/plain', null)"」を指定しておかないと、何故か動作しなかった。 *pw_app.isIE() はライブラリのIE判別関数
dropイベントで、Drag Element, Drop Elementの取得・入れ替えを行う
function handleDragStart(event) {
let tr = event.target.closest('tr');
if (tr) _this.drag_item = event.target.closest('tr');
event.target.style.opacity = '0.4';
event.stopPropagation();
}
function handleDrop(event) {
var tr = event.target.closest('tr');
if (tr && _this.drag_item != tr) {
_this.target_item = tr;
}
event.target.style.opacity = '1.0'
let row_id = _this.drag_item.getAttribute(_this.row_id_column);
if (row_id && _this.target_item && _this.drag_item != _this.target_item) {
var tbody = PwNode.byQuery(_this.body_selector).first();
let drag_order = _this.drag_item.getAttribute('order');
let target_order = _this.target_item.getAttribute('order');
if (drag_order > target_order) {
tbody.insertBefore(_this.drag_item, _this.target_item);
} else if (drag_order < target_order) {
tbody.insertBefore(_this.drag_item, _this.target_item.nextElementSibling);
}
//テーブル並び替え後の処理を記述
//並び順管理、tr.order を振り直す etc..
}
event.preventDefault();
event.stopPropagation();
}
event.target.closest() で「tr」を取得 「tbody」内で insertBefore() を利用し、並び順を考慮しつつ Element を書き換える 色々な方法があると思うが、上記は「tr」の order が並び替え後に順番を振り直す事を前提にしている
jQuery の $(document).on() をネイティブでコーディング document.addEventListner() で記述するが、ネイティブの場合はターゲット(Event.target)が何かによって条件分岐が必要
yoo16/html_samples/blob/master/document_on.html
//jQuery
$(document).on('click', '#id_name', function() {
//working
});
//native
document.addEventListener('click', function(event) {
if (event.target.id === 'id_name') {
//working
}
});
//jQuery
$(document).on('click', '.class_name', function() {
//working
});
//native
document.addEventListener('click', function(event) {
if(event.target.classList.contains('class_name'))
//working
}
});
querySelectorAll() で取得する方法もある。 (この方がjQueryに近いか?) ただ、addEventListener() を事前登録するため、後にHTMLを発生さした要素では動作しないのでイベント管理が必要
document.querySelectorAll('.class_name').forEach(function(element) {
element.addEventListener('click', function(event) {
//working
});
});
上記はIEの場合「NodeList」をループで回せない可能性があるので、
let elements = document.querySelectorAll('.class_name');
[].forEach.call(elements, function(element) {
element.addEventListener('click', function(event) {
//working
});
});
としても良い。
・git clone が重い
・秘密のファイルを Commit してしまった
など、過去にファイルを Commit & Pull してしまった場合、問題ファイルをなかったことにしたい場合があります。
履歴の削除は「git filter-branch」を利用してできますが、破壊的に履歴変更するため事前バックアップするなど作業は自己責任で行ってください。
git_find_big.sh で大きなオブジェクトを抽出
$ du -sh .git/objects 12M .git/objects
$ sh git_find_big.sh
「git_find_big.sh」は、gitコマンドを用いて「.git/objects/pack/pack」内の容量の多いオブジェクト順で抽出します。
抽出したファイルやディレクトリを削除 (refs/heads/master と refs/remotes/origin/master を書き換え)
※ディレクトリの場合は「git rm -r」
$ git filter-branch -f --index-filter 'git rm --cached --ignore-unmatch ファイルパス' --prune-empty -- --all
ガベージコレクションを利用して、不要なオブジェクトやログを削除します。
$ git gc --aggressive --prune=now
git オブジェクトのリパックします。
$ git repack -A -d
$ git push --force origin master
共同作業している場合は、clone してもらった方が安全かもしれません。
Gitignore にファイルを記述しても反映されない場合、キャッシュをクリアして commit & push してみる。
ファイル名に「.」を指定して、全ファイルをターゲットにする事もできるが、ちょっと怖いので1ファイルずつ実行した。
UFJ が API を虎視眈々と開発していた。
カブドットコムのAPIも使えたりと結構楽しそう、というか近々実用化されそうな予感がします。
カブドットコムのAPI(kabu.com API)は本番環境で利用できるとのことだが、
法人契約が必要なのと当然ながら審査が必要です。
2019/05/15現在、個人口座、法人口座、振込申請、総合振込申請、給与賞与振込申請、特別徴収地方税のAPIが用意されている。
当然ながらAPIを利用するには、Client Secret の発行が必要となる。
各サービス毎にリファレンスは用意されている。
cURL,Ruby,Python,PHP,Java,Node,Go,Swift のサンプルが記載されており、
結構本気度が伺えられる。 また、レスポンスは基本 json のようだ。
ただ本番利用の場合は、金融業界なので審査が厳しいと予想される。
使いやすいように、Github にクラスライブラリを作成してました。
https://github.com/yoo16/ufj_api (2019/05/15現在アカウント情報取得のみ)
また、検証用データが用意されていて、Account ID一覧は以下の通り
001001110001 002001110002 325001110003 135011110005 118011110008
329011110009 002021110013 002022220003 325022220004 909022220013
002022220014 002023330003 325023330004 480023330005 777023330008
118023330009 427383330012 909383330013 002383330014 002283330016
$ php get_account.php 002023330003
{"branchNo":"002","branchName":"丸の内","accountTypeCode":"02",
"accountTypeDetailCode":"00099",
"accountTypeName":"定期","accountNo":"3330003",
"accountName":null,"accountNameKana":null,
"balance":100000,"withdrawableAmount":null,
"currencyCode":"JPY"
,"foreignCurrencyBalance":null,
"yenEquivalent":null,"exchangeRate":null,
"totalMarketValue":null,
"totalUnrealizedProfitAndLoss":null,
"mutualFundAccountType":null,"fundBaseDate":null}
30日前より前のメールをラベル抽出して、ゴミ箱に移動するスクリプト GitHub: yoo16/gas_samples
function removeGmail() {
var before_days = 30;
var spreadsheet_id = 'スプレードシートID';
var sheet = SpreadsheetApp.openById(spreadsheet_id).getActiveSheet();
var values = sheet.getDataRange().getValues();
for (var i = 0; i < values.length; i++) {
removeMail(values[i], before_days);
}
}
function removeMail(label, before_days) {
var condition = 'older_than:' + before_days + 'd label:'+ label;
var threads = GmailApp.search(condition);
for (var i = 0; i < threads.length; i++) {
threads[i].moveToTrash();
}
//TODO loop for all threads by label
//var threads = GmailApp.search(condition, 0, 100);
//GmailApp.moveThreadsToTrash(threads);
}
GmailApp の機能はClass GmailAppを参照
Gmail検索ラベルは複数あるので、Googleスプレッドシートに記載して読み取る スプレッドシートの指定は、ファイル名検索でもできるが、あらかじめID取得しておいた方が楽
var sheet = SpreadsheetApp.openById(spreadsheet_id).getActiveSheet();
GmailApp.search() メソッドで Gmailを検索できるが、引数は Gmail のブラウザ操作で現れるものと同じと考えて良い。
var threads = GmailApp.search('older_than:' + before_days + 'd label:'+ label);
ただし件数制限がある模様(threads.length を調べると500件)
for (var i = 0; i < threads.length; i++) {
threads[i].moveToTrash();
}
1スレッドづつ削除するが、完全削除ではなくゴミ箱に移動する (完全削除メソッドは GAS では用意されていない)
上記だと forループで1スレッド毎にゴミ箱に移動するが、一括でゴミ箱移動する「GmailApp.moveThreadsToTrash()」がメソッドが存在する
var threads = GmailApp.search('older_than:' + before_days + 'd label:'+ label);
GmailApp.moveThreadsToTrash(threads);
ただし「この操作は最大 100 件のスレッドにのみ適用できます。」と表示され、100件までしか動作しないようなので、search() するときに件数を指定
var condition = 'older_than:' + before_days + 'd label:'+ label;
var threads = GmailApp.search(condition, 0, 100);
GmailApp.moveThreadsToTrash(threads);
これを、メールがなくなるまでループで回せば全削除はできそう
定期的にスクリプトを実行するには「編集 > 現在のプロジェクトのトリガー」から設定できる。
「トリガーを追加」でスケジュールを設定できる(詳細設定はわかりやすいので割愛)