mirror of
https://github.com/OutCast3k/coinbin.git
synced 2025-01-22 09:23:38 +01:00
2231 lines
79 KiB
JavaScript
2231 lines
79 KiB
JavaScript
$(document).ready(function() {
|
|
|
|
/* open wallet code */
|
|
|
|
var explorer_tx = "https://coinb.in/tx/"
|
|
var explorer_addr = "https://coinb.in/addr/"
|
|
var explorer_block = "https://coinb.in/block/"
|
|
|
|
var wallet_timer = false;
|
|
|
|
$("#openBtn").click(function(){
|
|
var email = $("#openEmail").val().toLowerCase();
|
|
if(email.match(/[\s\w\d]+@[\s\w\d]+/g)){
|
|
if($("#openPass").val().length>=10){
|
|
if($("#openPass").val()==$("#openPassConfirm").val()){
|
|
var email = $("#openEmail").val().toLowerCase();
|
|
var pass = $("#openPass").val();
|
|
var s = email;
|
|
s += '|'+pass+'|';
|
|
s += s.length+'|!@'+((pass.length*7)+email.length)*7;
|
|
var regchars = (pass.match(/[a-z]+/g)) ? pass.match(/[a-z]+/g).length : 1;
|
|
var regupchars = (pass.match(/[A-Z]+/g)) ? pass.match(/[A-Z]+/g).length : 1;
|
|
var regnums = (pass.match(/[0-9]+/g)) ? pass.match(/[0-9]+/g).length : 1;
|
|
s += ((regnums+regchars)+regupchars)*pass.length+'3571';
|
|
s += (s+''+s);
|
|
|
|
for(i=0;i<=50;i++){
|
|
s = Crypto.SHA256(s);
|
|
}
|
|
|
|
coinjs.compressed = true;
|
|
var keys = coinjs.newKeys(s);
|
|
var address = keys.address;
|
|
var wif = keys.wif;
|
|
var pubkey = keys.pubkey;
|
|
var privkeyaes = CryptoJS.AES.encrypt(keys.wif, pass);
|
|
|
|
$("#walletKeys .walletSegWitRS").addClass("hidden");
|
|
if($("#walletSegwit").is(":checked")){
|
|
if($("#walletSegwitBech32").is(":checked")){
|
|
var sw = coinjs.bech32Address(pubkey);
|
|
address = sw.address;
|
|
} else {
|
|
|
|
var sw = coinjs.segwitAddress(pubkey);
|
|
address = sw.address;
|
|
}
|
|
|
|
$("#walletKeys .walletSegWitRS").removeClass("hidden");
|
|
$("#walletKeys .walletSegWitRS input:text").val(sw.redeemscript);
|
|
}
|
|
|
|
$("#walletAddress").html(address);
|
|
$("#walletHistory").attr('href',explorer_addr+address);
|
|
|
|
$("#walletQrCode").html("");
|
|
var qrcode = new QRCode("walletQrCode");
|
|
qrcode.makeCode("bitcoin:"+address);
|
|
|
|
$("#walletKeys .privkey").val(wif);
|
|
$("#walletKeys .pubkey").val(pubkey);
|
|
$("#walletKeys .privkeyaes").val(privkeyaes);
|
|
|
|
$("#openLogin").hide();
|
|
$("#openWallet").removeClass("hidden").show();
|
|
|
|
walletBalance();
|
|
} else {
|
|
$("#openLoginStatus").html("Your passwords do not match!").removeClass("hidden").fadeOut().fadeIn();
|
|
}
|
|
} else {
|
|
$("#openLoginStatus").html("Your password must be at least 10 chars long").removeClass("hidden").fadeOut().fadeIn();
|
|
}
|
|
} else {
|
|
$("#openLoginStatus").html("Your email address doesn't appear to be valid").removeClass("hidden").fadeOut().fadeIn();
|
|
}
|
|
|
|
$("#openLoginStatus").prepend('<span class="glyphicon glyphicon-exclamation-sign"></span> ');
|
|
});
|
|
|
|
$("#walletLogout").click(function(){
|
|
$("#openEmail").val("");
|
|
$("#openPass").val("");
|
|
$("#openPassConfirm").val("");
|
|
|
|
$("#openLogin").show();
|
|
$("#openWallet").addClass("hidden").show();
|
|
|
|
$("#walletAddress").html("");
|
|
$("#walletHistory").attr('href',explorer_addr);
|
|
|
|
$("#walletQrCode").html("");
|
|
var qrcode = new QRCode("walletQrCode");
|
|
qrcode.makeCode("bitcoin:");
|
|
|
|
$("#walletKeys .privkey").val("");
|
|
$("#walletKeys .pubkey").val("");
|
|
|
|
$("#openLoginStatus").html("").hide();
|
|
});
|
|
|
|
$("#walletSegwit").click(function(){
|
|
if($(this).is(":checked")){
|
|
$(".walletSegwitType").attr('disabled',false);
|
|
} else {
|
|
$(".walletSegwitType").attr('disabled',true);
|
|
}
|
|
});
|
|
|
|
$("#walletToSegWit").click(function(){
|
|
$("#walletToBtn").html('SegWit <span class="caret"></span>');
|
|
$("#walletSegwit")[0].checked = true;
|
|
$("#walletSegwitp2sh")[0].checked = true;
|
|
$("#openBtn").click();
|
|
});
|
|
|
|
$("#walletToSegWitBech32").click(function(){
|
|
$("#walletToBtn").html('Bech32 <span class="caret"></span>');
|
|
$("#walletSegwit")[0].checked = true;
|
|
$("#walletSegwitBech32")[0].checked = true;
|
|
$("#openBtn").click();
|
|
});
|
|
|
|
$("#walletToLegacy").click(function(){
|
|
$("#walletToBtn").html('Legacy <span class="caret"></span>');
|
|
$("#walletSegwit")[0].checked = false;
|
|
$("#openBtn").click();
|
|
});
|
|
|
|
$("#walletShowKeys").click(function(){
|
|
$(".walletOptions").removeClass("hidden").addClass("hidden");
|
|
$("#walletKeys").removeClass("hidden");
|
|
});
|
|
|
|
$("#walletShowBuy").click(function(){
|
|
$(".walletOptions").removeClass("hidden").addClass("hidden");
|
|
$("#walletBuy").removeClass("hidden");
|
|
});
|
|
|
|
$("#walletBalance, #walletAddress, #walletQrCode").click(function(){
|
|
walletBalance();
|
|
});
|
|
|
|
$("#walletConfirmSend").click(function(){
|
|
var thisbtn = $(this);
|
|
var tx = coinjs.transaction();
|
|
var txfee = $("#txFee");
|
|
var devaddr = coinjs.developer;
|
|
var devamount = $("#developerDonation");
|
|
|
|
if((devamount.val()*1)>0){
|
|
tx.addoutput(devaddr, devamount.val()*1);
|
|
}
|
|
|
|
var total = (devamount.val()*1) + (txfee.val()*1);
|
|
|
|
$.each($("#walletSpendTo .output"), function(i,o){
|
|
var addr = $('.addressTo',o);
|
|
var amount = $('.amount',o);
|
|
if(amount.val()*1>0){
|
|
total += amount.val()*1;
|
|
tx.addoutput(addr.val(), amount.val()*1);
|
|
}
|
|
});
|
|
|
|
thisbtn.attr('disabled',true);
|
|
|
|
var script = false;
|
|
if($("#walletSegwit").is(":checked")){
|
|
if($("#walletSegwitBech32").is(":checked")){
|
|
var sw = coinjs.bech32Address($("#walletKeys .pubkey").val());
|
|
} else {
|
|
var sw = coinjs.segwitAddress($("#walletKeys .pubkey").val());
|
|
}
|
|
script = sw.redeemscript;
|
|
}
|
|
|
|
var sequence = 0xffffffff-1;
|
|
if($("#walletRBF").is(":checked")){
|
|
sequence = 0xffffffff-2;
|
|
}
|
|
|
|
tx.addUnspent($("#walletAddress").html(), function(data){
|
|
|
|
var dvalue = (data.value/100000000).toFixed(8) * 1;
|
|
total = (total*1).toFixed(8) * 1;
|
|
|
|
if(dvalue>=total){
|
|
var change = dvalue-total;
|
|
if((change*1)>0){
|
|
tx.addoutput($("#walletAddress").html(), change);
|
|
}
|
|
|
|
// clone the transaction with out using coinjs.clone() function as it gives us trouble
|
|
var tx2 = coinjs.transaction();
|
|
var txunspent = tx2.deserialize(tx.serialize());
|
|
|
|
// then sign
|
|
var signed = txunspent.sign($("#walletKeys .privkey").val());
|
|
|
|
// and finally broadcast!
|
|
|
|
tx2.broadcast(function(data){
|
|
if($(data).find("result").text()=="1"){
|
|
$("#walletSendConfirmStatus").removeClass('hidden').addClass('alert-success').html('txid: <a href="https://coinb.in/tx/'+$(data).find("txid").text()+'" target="_blank">'+$(data).find("txid").text()+'</a>');
|
|
} else {
|
|
$("#walletSendConfirmStatus").removeClass('hidden').addClass('alert-danger').html(unescape($(data).find("response").text()).replace(/\+/g,' '));
|
|
$("#walletSendFailTransaction").removeClass('hidden');
|
|
$("#walletSendFailTransaction textarea").val(signed);
|
|
thisbtn.attr('disabled',false);
|
|
}
|
|
|
|
// update wallet balance
|
|
walletBalance();
|
|
|
|
}, signed);
|
|
} else {
|
|
$("#walletSendConfirmStatus").removeClass("hidden").addClass('alert-danger').html("You have a confirmed balance of "+dvalue+" BTC unable to send "+total+" BTC").fadeOut().fadeIn();
|
|
thisbtn.attr('disabled',false);
|
|
}
|
|
|
|
$("#walletLoader").addClass("hidden");
|
|
|
|
}, script, script, sequence);
|
|
});
|
|
|
|
$("#walletSendBtn").click(function(){
|
|
|
|
$("#walletSendFailTransaction").addClass('hidden');
|
|
$("#walletSendStatus").addClass("hidden").html("");
|
|
|
|
var thisbtn = $(this);
|
|
var txfee = $("#txFee");
|
|
var devamount = $("#developerDonation");
|
|
|
|
if((!isNaN(devamount.val())) && devamount.val()>=0){
|
|
$(devamount).parent().removeClass('has-error');
|
|
} else {
|
|
$(devamount).parent().addClass('has-error')
|
|
}
|
|
|
|
if((!isNaN(txfee.val())) && txfee.val()>=0){
|
|
$(txfee).parent().removeClass('has-error');
|
|
} else {
|
|
$(txfee).parent().addClass('has-error');
|
|
}
|
|
|
|
var total = (devamount.val()*1) + (txfee.val()*1);
|
|
|
|
$.each($("#walletSpendTo .output"), function(i,o){
|
|
var amount = $('.amount',o);
|
|
var address = $('.addressTo',o);
|
|
|
|
total += amount.val()*1;
|
|
|
|
if((!isNaN($(amount).val())) && $(amount).val()>0){
|
|
$(amount).parent().removeClass('has-error');
|
|
} else {
|
|
$(amount).parent().addClass('has-error');
|
|
}
|
|
|
|
if(coinjs.addressDecode($(address).val())){
|
|
$(address).parent().removeClass('has-error');
|
|
} else {
|
|
$(address).parent().addClass('has-error');
|
|
}
|
|
});
|
|
|
|
total = total.toFixed(8);
|
|
|
|
if($("#walletSpend .has-error").length==0){
|
|
var balance = ($("#walletBalance").html()).replace(/[^0-9\.]+/g,'')*1;
|
|
if(total<=balance){
|
|
$("#walletSendConfirmStatus").addClass("hidden").removeClass('alert-success').removeClass('alert-danger').html("");
|
|
$("#spendAmount").html(total);
|
|
$("#modalWalletConfirm").modal("show");
|
|
$("#walletConfirmSend").attr('disabled',false);
|
|
} else {
|
|
$("#walletSendStatus").removeClass("hidden").html("You are trying to spend "+total+' but have a balance of '+balance);
|
|
}
|
|
} else {
|
|
$("#walletSpend .has-error").fadeOut().fadeIn();
|
|
$("#walletSendStatus").removeClass("hidden").html('<span class="glyphicon glyphicon-exclamation-sign"></span> One or more input has an error');
|
|
}
|
|
});
|
|
|
|
$("#walletShowSpend").click(function(){
|
|
$(".walletOptions").removeClass("hidden").addClass("hidden");
|
|
$("#walletSpend").removeClass("hidden");
|
|
});
|
|
|
|
$("#walletSpendTo .addressAdd").click(function(){
|
|
var clone = '<div class="form-horizontal output">'+$(this).parent().html()+'</div>';
|
|
$("#walletSpendTo").append(clone);
|
|
$("#walletSpendTo .glyphicon-plus:last").removeClass('glyphicon-plus').addClass('glyphicon-minus');
|
|
$("#walletSpendTo .glyphicon-minus:last").parent().removeClass('addressAdd').addClass('addressRemove');
|
|
$("#walletSpendTo .addressRemove").unbind("");
|
|
$("#walletSpendTo .addressRemove").click(function(){
|
|
$(this).parent().fadeOut().remove();
|
|
});
|
|
});
|
|
|
|
function walletBalance(){
|
|
if($("#walletLoader").hasClass("hidden")){
|
|
var tx = coinjs.transaction();
|
|
$("#walletLoader").removeClass("hidden");
|
|
coinjs.addressBalance($("#walletAddress").html(),function(data){
|
|
if($(data).find("result").text()==1){
|
|
var v = $(data).find("balance").text()/100000000;
|
|
$("#walletBalance").html(v+" BTC").attr('rel',v).fadeOut().fadeIn();
|
|
} else {
|
|
$("#walletBalance").html("0.00 BTC").attr('rel',v).fadeOut().fadeIn();
|
|
}
|
|
|
|
$("#walletLoader").addClass("hidden");
|
|
});
|
|
}
|
|
}
|
|
|
|
/* new -> address code */
|
|
|
|
$("#newKeysBtn").click(function(){
|
|
coinjs.compressed = false;
|
|
if($("#newCompressed").is(":checked")){
|
|
coinjs.compressed = true;
|
|
}
|
|
var s = ($("#newBrainwallet").is(":checked")) ? $("#brainwallet").val() : null;
|
|
var coin = coinjs.newKeys(s);
|
|
$("#newBitcoinAddress").val(coin.address);
|
|
$("#newPubKey").val(coin.pubkey);
|
|
$("#newPrivKey").val(coin.wif);
|
|
|
|
/* encrypted key code */
|
|
if((!$("#encryptKey").is(":checked")) || $("#aes256pass").val()==$("#aes256pass_confirm").val()){
|
|
$("#aes256passStatus").addClass("hidden");
|
|
if($("#encryptKey").is(":checked")){
|
|
$("#aes256wifkey").removeClass("hidden");
|
|
}
|
|
} else {
|
|
$("#aes256passStatus").removeClass("hidden");
|
|
}
|
|
$("#newPrivKeyEnc").val(CryptoJS.AES.encrypt(coin.wif, $("#aes256pass").val())+'');
|
|
});
|
|
|
|
$("#newPaperwalletBtn").click(function(){
|
|
if($("#newBitcoinAddress").val()==""){
|
|
$("#newKeysBtn").click();
|
|
}
|
|
|
|
var paperwallet = window.open();
|
|
paperwallet.document.write('<h2>BTC PaperWallet</h2><hr><div style="margin-top: 5px; margin-bottom: 5px"><div><h3 style="margin-top: 0">Address (Share)</h3></div><div style="text-align: center;"><div id="qraddress"></div><p>'+$("#newBitcoinAddress").val()+'</p></div></div><hr><div style="margin-top: 5px; margin-bottom: 5px"><div><h3 style="margin-top: 0">Public Key</h3></div><div style="text-align: center;"><div id="qrpubkey"></div><p>'+$("#newPubKey").val()+'</p></div></div><hr><div style="margin-top: 5px; margin-bottom: 5px"><div><h3 style="margin-top: 0">Private Key (KEEP SECRET!)</h3></div><div style="text-align: center;"><div id="qrprivkey"></div><p>'+$("#newPrivKey").val()+'</p></div></div>');
|
|
paperwallet.document.close();
|
|
paperwallet.focus();
|
|
new QRCode(paperwallet.document.getElementById("qraddress"), {text: $("#newBitcoinAddress").val(), width: 125, height: 125});
|
|
new QRCode(paperwallet.document.getElementById("qrpubkey"), {text: $("#newPubKey").val(), width: 125, height: 125});
|
|
new QRCode(paperwallet.document.getElementById("qrprivkey"), {text: $("#newPrivKey").val(), width: 125, height: 125});
|
|
paperwallet.print();
|
|
paperwallet.close();
|
|
});
|
|
|
|
$("#newBrainwallet").click(function(){
|
|
if($(this).is(":checked")){
|
|
$("#brainwallet").removeClass("hidden");
|
|
} else {
|
|
$("#brainwallet").addClass("hidden");
|
|
}
|
|
});
|
|
|
|
$("#newSegWitBrainwallet").click(function(){
|
|
if($(this).is(":checked")){
|
|
$("#brainwalletSegWit").removeClass("hidden");
|
|
} else {
|
|
$("#brainwalletSegWit").addClass("hidden");
|
|
}
|
|
});
|
|
|
|
$("#encryptKey").click(function(){
|
|
if($(this).is(":checked")){
|
|
$("#aes256passform").removeClass("hidden");
|
|
} else {
|
|
$("#aes256wifkey, #aes256passform, #aes256passStatus").addClass("hidden");
|
|
}
|
|
});
|
|
|
|
/* new -> segwit code */
|
|
$("#newSegWitKeysBtn").click(function(){
|
|
var compressed = coinjs.compressed;
|
|
coinjs.compressed = true;
|
|
|
|
var s = ($("#newSegWitBrainwallet").is(":checked")) ? $("#brainwalletSegWit").val() : null;
|
|
var coin = coinjs.newKeys(s);
|
|
|
|
if($("#newSegWitBech32addr").is(":checked")){
|
|
var sw = coinjs.bech32Address(coin.pubkey);
|
|
} else {
|
|
var sw = coinjs.segwitAddress(coin.pubkey);
|
|
}
|
|
|
|
$("#newSegWitAddress").val(sw.address);
|
|
$("#newSegWitRedeemScript").val(sw.redeemscript);
|
|
$("#newSegWitPubKey").val(coin.pubkey);
|
|
$("#newSegWitPrivKey").val(coin.wif);
|
|
coinjs.compressed = compressed;
|
|
});
|
|
|
|
$("#newSegwitPaperwalletBtn").click(function(){
|
|
if($("#newSegWitAddress").val()==""){
|
|
$("#newSegWitKeysBtn").click();
|
|
}
|
|
|
|
var paperwallet = window.open();
|
|
paperwallet.document.write('<h2>BTC SegWit PaperWallet</h2><hr><div style="margin-top: 5px; margin-bottom: 5px"><div><h3 style="margin-top: 0">Address (Share)</h3></div><div style="text-align: center;"><div id="qraddress"></div><p>'+$("#newSegWitAddress").val()+'</p></div></div><hr><div style="margin-top: 5px; margin-bottom: 5px"><div><h3 style="margin-top: 0">Public Key</h3></div><div style="text-align: center;"><div id="qrpubkey"></div><p>'+$("#newSegWitPubKey").val()+'</p></div></div><hr><div style="margin-top: 5px; margin-bottom: 5px"><div><h3 style="margin-top: 0">Redeem Script</h3></div><div style="text-align: center;"><div id="qrredeem"></div><p>'+$("#newSegWitRedeemScript").val()+'</p></div></div><hr><div style="margin-top: 5px; margin-bottom: 5px"><div><h3 style="margin-top: 0">Private Key (KEEP SECRET!)</h3></div><div style="text-align: center;"><div id="qrprivkey"></div><p>'+$("#newSegWitPrivKey").val()+'</p></div></div>');
|
|
paperwallet.document.close();
|
|
paperwallet.focus();
|
|
new QRCode(paperwallet.document.getElementById("qraddress"), {text: $("#newSegWitAddress").val(), width: 110, height: 110});
|
|
new QRCode(paperwallet.document.getElementById("qrpubkey"), {text: $("#newSegWitPubKey").val(), width: 110, height: 110});
|
|
new QRCode(paperwallet.document.getElementById("qrredeem"), {text: $("#newSegWitRedeemScript").val(), width: 110, height: 110});
|
|
new QRCode(paperwallet.document.getElementById("qrprivkey"), {text: $("#newSegWitPrivKey").val(), width: 110, height: 110});
|
|
paperwallet.print();
|
|
paperwallet.close();
|
|
});
|
|
|
|
/* new -> multisig code */
|
|
|
|
$("#newMultiSigAddress").click(function(){
|
|
|
|
$("#multiSigData").removeClass('show').addClass('hidden').fadeOut();
|
|
$("#multisigPubKeys .pubkey").parent().removeClass('has-error');
|
|
$("#releaseCoins").parent().removeClass('has-error');
|
|
$("#multiSigErrorMsg").hide();
|
|
|
|
if((isNaN($("#releaseCoins option:selected").html())) || ((!isNaN($("#releaseCoins option:selected").html())) && ($("#releaseCoins option:selected").html()>$("#multisigPubKeys .pubkey").length || $("#releaseCoins option:selected").html()*1<=0 || $("#releaseCoins option:selected").html()*1>8))){
|
|
$("#releaseCoins").parent().addClass('has-error');
|
|
$("#multiSigErrorMsg").html('<span class="glyphicon glyphicon-exclamation-sign"></span> Minimum signatures required is greater than the amount of public keys provided').fadeIn();
|
|
return false;
|
|
}
|
|
|
|
var keys = [];
|
|
$.each($("#multisigPubKeys .pubkey"), function(i,o){
|
|
if(coinjs.pubkeydecompress($(o).val())){
|
|
keys.push($(o).val());
|
|
$(o).parent().removeClass('has-error');
|
|
} else {
|
|
$(o).parent().addClass('has-error');
|
|
}
|
|
});
|
|
|
|
if(($("#multisigPubKeys .pubkey").parent().hasClass('has-error')==false) && $("#releaseCoins").parent().hasClass('has-error')==false){
|
|
var sigsNeeded = $("#releaseCoins option:selected").html();
|
|
var multisig = coinjs.pubkeys2MultisigAddress(keys, sigsNeeded);
|
|
if(multisig.size <= 520){
|
|
$("#multiSigData .address").val(multisig['address']);
|
|
$("#multiSigData .script").val(multisig['redeemScript']);
|
|
$("#multiSigData .scriptUrl").val(document.location.origin+''+document.location.pathname+'?verify='+multisig['redeemScript']+'#verify');
|
|
$("#multiSigData").removeClass('hidden').addClass('show').fadeIn();
|
|
$("#releaseCoins").removeClass('has-error');
|
|
} else {
|
|
$("#multiSigErrorMsg").html('<span class="glyphicon glyphicon-exclamation-sign"></span> Your generated redeemscript is too large (>520 bytes) it can not be used safely').fadeIn();
|
|
}
|
|
} else {
|
|
$("#multiSigErrorMsg").html('<span class="glyphicon glyphicon-exclamation-sign"></span> One or more public key is invalid!').fadeIn();
|
|
}
|
|
});
|
|
|
|
$("#multisigPubKeys .pubkeyAdd").click(function(){
|
|
if($("#multisigPubKeys .pubkeyRemove").length<14){
|
|
var clone = '<div class="form-horizontal">'+$(this).parent().html()+'</div>';
|
|
$("#multisigPubKeys").append(clone);
|
|
$("#multisigPubKeys .glyphicon-plus:last").removeClass('glyphicon-plus').addClass('glyphicon-minus');
|
|
$("#multisigPubKeys .glyphicon-minus:last").parent().removeClass('pubkeyAdd').addClass('pubkeyRemove');
|
|
$("#multisigPubKeys .pubkeyRemove").unbind("");
|
|
$("#multisigPubKeys .pubkeyRemove").click(function(){
|
|
$(this).parent().fadeOut().remove();
|
|
});
|
|
}
|
|
});
|
|
|
|
$("#mediatorList").change(function(){
|
|
var data = ($(this).val()).split(";");
|
|
$("#mediatorPubkey").val(data[0]);
|
|
$("#mediatorEmail").val(data[1]);
|
|
$("#mediatorFee").val(data[2]);
|
|
}).change();
|
|
|
|
$("#mediatorAddKey").click(function(){
|
|
var count = 0;
|
|
var len = $(".pubkeyRemove").length;
|
|
if(len<14){
|
|
$.each($("#multisigPubKeys .pubkey"),function(i,o){
|
|
if($(o).val()==''){
|
|
$(o).val($("#mediatorPubkey").val()).fadeOut().fadeIn();
|
|
$("#mediatorClose").click();
|
|
return false;
|
|
} else if(count==len){
|
|
$("#multisigPubKeys .pubkeyAdd").click();
|
|
$("#mediatorAddKey").click();
|
|
return false;
|
|
}
|
|
count++;
|
|
});
|
|
|
|
$("#mediatorClose").click();
|
|
}
|
|
});
|
|
|
|
/* new -> time locked code */
|
|
|
|
$('#timeLockedDateTimePicker').datetimepicker({
|
|
format: "MM/DD/YYYY HH:mm",
|
|
});
|
|
|
|
$('#timeLockedRbTypeBox input').change(function(){
|
|
if ($('#timeLockedRbTypeDate').is(':checked')){
|
|
$('#timeLockedDateTimePicker').show();
|
|
$('#timeLockedBlockHeight').hide();
|
|
} else {
|
|
$('#timeLockedDateTimePicker').hide();
|
|
$('#timeLockedBlockHeight').removeClass('hidden').show();
|
|
}
|
|
});
|
|
|
|
$("#newTimeLockedAddress").click(function(){
|
|
|
|
$("#timeLockedData").removeClass('show').addClass('hidden').fadeOut();
|
|
$("#timeLockedPubKey").parent().removeClass('has-error');
|
|
$("#timeLockedDateTimePicker").parent().removeClass('has-error');
|
|
$("#timeLockedErrorMsg").hide();
|
|
|
|
if(!coinjs.pubkeydecompress($("#timeLockedPubKey").val())) {
|
|
$('#timeLockedPubKey').parent().addClass('has-error');
|
|
}
|
|
|
|
var nLockTime = -1;
|
|
|
|
if ($('#timeLockedRbTypeDate').is(':checked')){
|
|
// by date
|
|
var date = $('#timeLockedDateTimePicker').data("DateTimePicker").date();
|
|
if(!date || !date.isValid()) {
|
|
$('#timeLockedDateTimePicker').parent().addClass('has-error');
|
|
}
|
|
nLockTime = date.unix()
|
|
if (nLockTime < 500000000) {
|
|
$('#timeLockedDateTimePicker').parent().addClass('has-error');
|
|
}
|
|
} else {
|
|
nLockTime = parseInt($('#timeLockedBlockHeightVal').val(), 10);
|
|
if (nLockTime >= 500000000) {
|
|
$('#timeLockedDateTimePicker').parent().addClass('has-error');
|
|
}
|
|
}
|
|
|
|
if(($("#timeLockedPubKey").parent().hasClass('has-error')==false) && $("#timeLockedDateTimePicker").parent().hasClass('has-error')==false){
|
|
try {
|
|
var hodl = coinjs.simpleHodlAddress($("#timeLockedPubKey").val(), nLockTime);
|
|
$("#timeLockedData .address").val(hodl['address']);
|
|
$("#timeLockedData .script").val(hodl['redeemScript']);
|
|
$("#timeLockedData .scriptUrl").val(document.location.origin+''+document.location.pathname+'?verify='+hodl['redeemScript']+'#verify');
|
|
$("#timeLockedData").removeClass('hidden').addClass('show').fadeIn();
|
|
} catch(e) {
|
|
$("#timeLockedErrorMsg").html('<span class="glyphicon glyphicon-exclamation-sign"></span> ' + e).fadeIn();
|
|
}
|
|
} else {
|
|
$("#timeLockedErrorMsg").html('<span class="glyphicon glyphicon-exclamation-sign"></span> Public key and/or date is invalid!').fadeIn();
|
|
}
|
|
});
|
|
|
|
/* new -> Hd address code */
|
|
|
|
$(".deriveHDbtn").click(function(){
|
|
$("#verifyScript").val($("input[type='text']",$(this).parent().parent()).val());
|
|
window.location = "#verify";
|
|
$("#verifyBtn").click();
|
|
});
|
|
|
|
$("#newHDKeysBtn").click(function(){
|
|
coinjs.compressed = true;
|
|
var s = ($("#newHDBrainwallet").is(":checked")) ? $("#HDBrainwallet").val() : null;
|
|
var hd = coinjs.hd();
|
|
var pair = hd.master(s);
|
|
$("#newHDxpub").val(pair.pubkey);
|
|
$("#newHDxprv").val(pair.privkey);
|
|
|
|
});
|
|
|
|
$("#newHDBrainwallet").click(function(){
|
|
if($(this).is(":checked")){
|
|
$("#HDBrainwallet").removeClass("hidden");
|
|
} else {
|
|
$("#HDBrainwallet").addClass("hidden");
|
|
}
|
|
});
|
|
|
|
/* new -> transaction code */
|
|
|
|
$("#recipients .addressAddTo").click(function(){
|
|
if($("#recipients .addressRemoveTo").length<19){
|
|
var clone = '<div class="row recipient"><br>'+$(this).parent().parent().html()+'</div>';
|
|
$("#recipients").append(clone);
|
|
$("#recipients .glyphicon-plus:last").removeClass('glyphicon-plus').addClass('glyphicon-minus');
|
|
$("#recipients .glyphicon-minus:last").parent().removeClass('addressAdd').addClass('addressRemoveTo');
|
|
$("#recipients .addressRemoveTo").unbind("");
|
|
$("#recipients .addressRemoveTo").click(function(){
|
|
$(this).parent().parent().fadeOut().remove();
|
|
validateOutputAmount();
|
|
});
|
|
validateOutputAmount();
|
|
}
|
|
});
|
|
|
|
$("#inputs .txidAdd").click(function(){
|
|
var clone = '<div class="row inputs"><br>'+$(this).parent().parent().html()+'</div>';
|
|
$("#inputs").append(clone);
|
|
$("#inputs .txidClear:last").remove();
|
|
$("#inputs .glyphicon-plus:last").removeClass('glyphicon-plus').addClass('glyphicon-minus');
|
|
$("#inputs .glyphicon-minus:last").parent().removeClass('txidAdd').addClass('txidRemove');
|
|
$("#inputs .txidRemove").unbind("");
|
|
$("#inputs .txidRemove").click(function(){
|
|
$(this).parent().parent().fadeOut().remove();
|
|
totalInputAmount();
|
|
});
|
|
$("#inputs .row:last input").attr('disabled',false);
|
|
|
|
$("#inputs .txIdAmount").unbind("").change(function(){
|
|
totalInputAmount();
|
|
}).keyup(function(){
|
|
totalInputAmount();
|
|
});
|
|
|
|
});
|
|
|
|
$("#transactionBtn").click(function(){
|
|
var tx = coinjs.transaction();
|
|
var estimatedTxSize = 10; // <4:version><1:txInCount><1:txOutCount><4:nLockTime>
|
|
|
|
$("#transactionCreate, #transactionCreateStatus").addClass("hidden");
|
|
|
|
if(($("#nLockTime").val()).match(/^[0-9]+$/g)){
|
|
tx.lock_time = $("#nLockTime").val()*1;
|
|
}
|
|
|
|
$("#inputs .row").removeClass('has-error');
|
|
|
|
$('#putTabs a[href="#txinputs"], #putTabs a[href="#txoutputs"]').attr('style','');
|
|
|
|
$.each($("#inputs .row"), function(i,o){
|
|
if(!($(".txId",o).val()).match(/^[a-f0-9]+$/i)){
|
|
$(o).addClass("has-error");
|
|
} else if((!($(".txIdScript",o).val()).match(/^[a-f0-9]+$/i)) && $(".txIdScript",o).val()!=""){
|
|
$(o).addClass("has-error");
|
|
} else if (!($(".txIdN",o).val()).match(/^[0-9]+$/i)){
|
|
$(o).addClass("has-error");
|
|
}
|
|
|
|
if(!$(o).hasClass("has-error")){
|
|
var seq = 0xffffffff-1;
|
|
if($("#txRBF").is(":checked")){
|
|
seq = 0xffffffff-2;
|
|
}
|
|
|
|
var currentScript = $(".txIdScript",o).val();
|
|
if (currentScript.match(/^76a914[0-9a-f]{40}88ac$/)) {
|
|
estimatedTxSize += 147
|
|
} else if (currentScript.match(/^5[1-9a-f](?:210[23][0-9a-f]{64}){1,15}5[1-9a-f]ae$/)) {
|
|
// <74:persig <1:push><72:sig><1:sighash> ><34:perpubkey <1:push><33:pubkey> > <32:prevhash><4:index><4:nSequence><1:m><1:n><1:OP>
|
|
var scriptSigSize = (parseInt(currentScript.slice(1,2),16) * 74) + (parseInt(currentScript.slice(-3,-2),16) * 34) + 43
|
|
// varint 2 bytes if scriptSig is > 252
|
|
estimatedTxSize += scriptSigSize + (scriptSigSize > 252 ? 2 : 1)
|
|
} else {
|
|
// underestimating won't hurt. Just showing a warning window anyways.
|
|
estimatedTxSize += 147
|
|
}
|
|
|
|
tx.addinput($(".txId",o).val(), $(".txIdN",o).val(), $(".txIdScript",o).val(), seq);
|
|
} else {
|
|
$('#putTabs a[href="#txinputs"]').attr('style','color:#a94442;');
|
|
}
|
|
});
|
|
|
|
$("#recipients .row").removeClass('has-error');
|
|
|
|
$.each($("#recipients .row"), function(i,o){
|
|
var a = ($(".address",o).val());
|
|
var ad = coinjs.addressDecode(a);
|
|
if(((a!="") && (ad.version == coinjs.pub || ad.version == coinjs.multisig || ad.type=="bech32")) && $(".amount",o).val()!=""){ // address
|
|
// P2SH output is 32, P2PKH is 34
|
|
estimatedTxSize += (ad.version == coinjs.pub ? 34 : 32)
|
|
tx.addoutput(a, $(".amount",o).val());
|
|
} else if (((a!="") && ad.version === 42) && $(".amount",o).val()!=""){ // stealth address
|
|
// 1 P2PKH and 1 OP_RETURN with 36 bytes, OP byte, and 8 byte value
|
|
estimatedTxSize += 78
|
|
tx.addstealth(ad, $(".amount",o).val());
|
|
} else if (((($("#opReturn").is(":checked")) && a.match(/^[a-f0-9]+$/ig)) && a.length<160) && (a.length%2)==0) { // data
|
|
estimatedTxSize += (a.length / 2) + 1 + 8
|
|
tx.adddata(a);
|
|
} else { // neither address nor data
|
|
$(o).addClass('has-error');
|
|
$('#putTabs a[href="#txoutputs"]').attr('style','color:#a94442;');
|
|
}
|
|
});
|
|
|
|
|
|
if(!$("#recipients .row, #inputs .row").hasClass('has-error')){
|
|
|
|
$("#transactionCreate textarea").val(tx.serialize());
|
|
$("#transactionCreate .txSize").html(tx.size());
|
|
|
|
if($("#feesestnewtx").attr('est')=='y'){
|
|
$("#fees .txhex").val($("#transactionCreate textarea").val());
|
|
$("#feesAnalyseBtn").click();
|
|
$("#fees .txhex").val("");
|
|
window.location = "#fees";
|
|
} else {
|
|
|
|
$("#transactionCreate").removeClass("hidden");
|
|
|
|
// Check fee against hard 0.01 as well as fluid 200 satoshis per byte calculation.
|
|
if($("#transactionFee").val()>=0.01 || $("#transactionFee").val()>= estimatedTxSize * 200 * 1e-8){
|
|
$("#modalWarningFeeAmount").html($("#transactionFee").val());
|
|
$("#modalWarningFee").modal("show");
|
|
}
|
|
}
|
|
$("#feesestnewtx").attr('est','');
|
|
} else {
|
|
$("#transactionCreateStatus").removeClass("hidden").html("One or more input or output is invalid").fadeOut().fadeIn();
|
|
}
|
|
});
|
|
|
|
$("#feesestnewtx").click(function(){
|
|
$(this).attr('est','y');
|
|
$("#transactionBtn").click();
|
|
});
|
|
|
|
$("#feesestwallet").click(function(){
|
|
$(this).attr('est','y');
|
|
var outputs = $("#walletSpendTo .output").length;
|
|
|
|
$("#fees .inputno, #fees .outputno, #fees .bytes").html(0);
|
|
$("#fees .slider").val(0);
|
|
|
|
var tx = coinjs.transaction();
|
|
tx.listUnspent($("#walletAddress").html(), function(data){
|
|
var inputs = $(data).find("unspent").children().length;
|
|
if($("#walletSegwit").is(":checked")){
|
|
$("#fees .txi_segwit").val(inputs);
|
|
$("#fees .txi_segwit").trigger('input');
|
|
} else {
|
|
$("#fees .txi_regular").val(inputs);
|
|
$("#fees .txi_regular").trigger('input');
|
|
}
|
|
|
|
$.each($("#walletSpendTo .output"), function(i,o){
|
|
var addr = $('.addressTo',o);
|
|
var ad = coinjs.addressDecode(addr.val());
|
|
if (ad.version == coinjs.pub){ // p2pkh
|
|
$("#fees .txo_p2pkh").val(($("#fees .txo_p2pkh").val()*1)+1);
|
|
$("#fees .txo_p2pkh").trigger('input');
|
|
} else { // p2psh
|
|
$("#fees .txo_p2sh").val(($("#fees .txo_p2sh").val()*1)+1);
|
|
$("#fees .txo_p2sh").trigger('input');
|
|
}
|
|
});
|
|
|
|
if(($("#developerDonation").val()*1)>0){
|
|
var addr = coinjs.developer;
|
|
var ad = coinjs.addressDecode(addr);
|
|
if (ad.version == coinjs.pub){ // p2pkh
|
|
$("#fees .txo_p2pkh").val(($("#fees .txo_p2pkh").val()*1)+1);
|
|
$("#fees .txo_p2pkh").trigger('input');
|
|
} else { // p2psh
|
|
$("#fees .txo_p2sh").val(($("#fees .txo_p2sh").val()*1)+1);
|
|
$("#fees .txo_p2sh").trigger('input');
|
|
}
|
|
}
|
|
|
|
});
|
|
|
|
//feeStats();
|
|
window.location = "#fees";
|
|
});
|
|
|
|
$(".txidClear").click(function(){
|
|
$("#inputs .row:first input").attr('disabled',false);
|
|
$("#inputs .row:first input").val("");
|
|
totalInputAmount();
|
|
});
|
|
|
|
$("#inputs .txIdAmount").unbind("").change(function(){
|
|
totalInputAmount();
|
|
}).keyup(function(){
|
|
totalInputAmount();
|
|
});
|
|
|
|
$("#donateTxBtn").click(function(){
|
|
|
|
var exists = false;
|
|
|
|
$.each($("#recipients .address"), function(i,o){
|
|
if($(o).val() == coinjs.developer){
|
|
exists = true;
|
|
$(o).fadeOut().fadeIn();
|
|
return true;
|
|
}
|
|
});
|
|
|
|
if(!exists){
|
|
if($("#recipients .recipient:last .address:last").val() != ""){
|
|
$("#recipients .addressAddTo:first").click();
|
|
};
|
|
|
|
$("#recipients .recipient:last .address:last").val(coinjs.developer).fadeOut().fadeIn();
|
|
|
|
return true;
|
|
}
|
|
});
|
|
|
|
/* code for the qr code scanner */
|
|
|
|
$(".qrcodeScanner").click(function(){
|
|
if ((typeof MediaStreamTrack === 'function') && typeof MediaStreamTrack.getSources === 'function'){
|
|
MediaStreamTrack.getSources(function(sourceInfos){
|
|
var f = 0;
|
|
$("select#videoSource").html("");
|
|
for (var i = 0; i !== sourceInfos.length; ++i) {
|
|
var sourceInfo = sourceInfos[i];
|
|
var option = document.createElement('option');
|
|
option.value = sourceInfo.id;
|
|
if (sourceInfo.kind === 'video') {
|
|
option.text = sourceInfo.label || 'camera ' + ($("select#videoSource options").length + 1);
|
|
$(option).appendTo("select#videoSource");
|
|
}
|
|
}
|
|
});
|
|
|
|
$("#videoSource").unbind("change").change(function(){
|
|
scannerStart()
|
|
});
|
|
|
|
} else {
|
|
$("#videoSource").addClass("hidden");
|
|
}
|
|
scannerStart();
|
|
$("#qrcode-scanner-callback-to").html($(this).attr('forward-result'));
|
|
});
|
|
|
|
function scannerStart(){
|
|
navigator.getUserMedia = navigator.getUserMedia || navigator.webkitGetUserMedia || navigator.mozGetUserMedia || false;
|
|
if(navigator.getUserMedia){
|
|
if (!!window.stream) {
|
|
$("video").attr('src',null);
|
|
}
|
|
|
|
var videoSource = $("select#videoSource").val();
|
|
var constraints = {
|
|
video: {
|
|
optional: [{sourceId: videoSource}]
|
|
}
|
|
};
|
|
|
|
navigator.getUserMedia(constraints, function(stream){
|
|
window.stream = stream; // make stream available to console
|
|
var videoElement = document.querySelector('video');
|
|
try {
|
|
videoElement.srcObject = stream;
|
|
} catch {
|
|
videoElement.src = window.URL.createObjectURL(stream);
|
|
}
|
|
videoElement.play();
|
|
}, function(error){ });
|
|
|
|
|
|
QCodeDecoder().decodeFromVideo(document.getElementById('videoReader'), function(er,data){
|
|
if(!er){
|
|
var match = data.match(/^bitcoin\:([1|3|bc1][a-z0-9]{25,50})/i);
|
|
var result = match ? match[1] : data;
|
|
$(""+$("#qrcode-scanner-callback-to").html()).val(result);
|
|
$("#qrScanClose").click();
|
|
}
|
|
});
|
|
|
|
} else {
|
|
$("#videoReaderError").removeClass("hidden");
|
|
$("#videoReader, #videoSource").addClass("hidden");
|
|
}
|
|
}
|
|
|
|
/* redeem from button code */
|
|
|
|
$("#redeemFromBtn").click(function(){
|
|
var redeem = redeemingFrom($("#redeemFrom").val());
|
|
|
|
$("#redeemFromStatus, #redeemFromAddress").addClass('hidden');
|
|
|
|
if(redeem.from=='multisigAddress'){
|
|
$("#redeemFromStatus").removeClass('hidden').html('<span class="glyphicon glyphicon-exclamation-sign"></span> You should use the redeem script, not its address!');
|
|
return false;
|
|
}
|
|
|
|
if(redeem.from=='other'){
|
|
$("#redeemFromStatus").removeClass('hidden').html('<span class="glyphicon glyphicon-exclamation-sign"></span> The address or redeem script you have entered is invalid');
|
|
return false;
|
|
}
|
|
|
|
if($("#clearInputsOnLoad").is(":checked")){
|
|
$("#inputs .txidRemove, #inputs .txidClear").click();
|
|
}
|
|
|
|
$("#redeemFromBtn").html("Please wait, loading...").attr('disabled',true);
|
|
|
|
var host = $(this).attr('rel');
|
|
|
|
// api: blockcypher blockchair chain.so
|
|
// network name "btc" "bitcoin" "BTC"
|
|
// network name "ltc" "litecoin" "LTC"
|
|
// network name "doge" "dogecoin" "DOGE"
|
|
|
|
if(host=='chain.so_bitcoinmainnet'){
|
|
listUnspentChainso(redeem, "BTC");
|
|
} else if(host=='chain.so_litecoin'){
|
|
listUnspentChainso(redeem, "LTC");
|
|
} else if(host=='chain.so_dogecoin'){
|
|
listUnspentChainso(redeem, "DOGE");
|
|
|
|
} else if(host=='blockcypher_bitcoinmainnet'){
|
|
listUnspentBlockcypher(redeem, "btc");
|
|
} else if(host=='blockcypher_litecoin'){
|
|
listUnspentBlockcypher(redeem, "ltc");
|
|
} else if(host=='blockcypher_dogecoin'){
|
|
listUnspentBlockcypher(redeem, "doge");
|
|
|
|
} else if(host=='blockchair_bitcoinmainnet'){
|
|
listUnspentBlockchair(redeem, "bitcoin");
|
|
} else if(host=='blockchair_litecoin'){
|
|
listUnspentBlockchair(redeem, "litecoin");
|
|
} else if(host=='blockchair_dogecoin'){
|
|
listUnspentBlockchair(redeem, "dogecoin");
|
|
} else {
|
|
listUnspentDefault(redeem);
|
|
}
|
|
|
|
if($("#redeemFromStatus").hasClass("hidden")) {
|
|
// An ethical dilemma: Should we automatically set nLockTime?
|
|
if(redeem.from == 'redeemScript' && redeem.type == "hodl__") {
|
|
$("#nLockTime").val(redeem.decodescript.checklocktimeverify);
|
|
} else {
|
|
$("#nLockTime").val(0);
|
|
}
|
|
}
|
|
});
|
|
|
|
/* function to determine what we are redeeming from */
|
|
function redeemingFrom(string){
|
|
var r = {};
|
|
var decode = coinjs.addressDecode(string);
|
|
if(decode.version == coinjs.pub){ // regular address
|
|
r.addr = string;
|
|
r.from = 'address';
|
|
r.redeemscript = false;
|
|
} else if (decode.version == coinjs.priv){ // wif key
|
|
var a = coinjs.wif2address(string);
|
|
r.addr = a['address'];
|
|
r.from = 'wif';
|
|
r.redeemscript = false;
|
|
} else if (decode.version == coinjs.multisig){ // mulisig address
|
|
r.addr = '';
|
|
r.from = 'multisigAddress';
|
|
r.redeemscript = false;
|
|
} else if(decode.type == 'bech32'){
|
|
r.addr = string;
|
|
r.from = 'bech32';
|
|
r.decodedRs = decode.redeemscript;
|
|
r.redeemscript = true;
|
|
} else {
|
|
var script = coinjs.script();
|
|
var decodeRs = script.decodeRedeemScript(string);
|
|
if(decodeRs){ // redeem script
|
|
r.addr = decodeRs['address'];
|
|
r.from = 'redeemScript';
|
|
r.decodedRs = decodeRs.redeemscript;
|
|
r.type = decodeRs['type'];
|
|
r.redeemscript = true;
|
|
r.decodescript = decodeRs;
|
|
} else { // something else
|
|
if(string.match(/^[a-f0-9]{64}$/i)){
|
|
r.addr = string;
|
|
r.from = 'txid';
|
|
r.redeemscript = false;
|
|
} else {
|
|
r.addr = '';
|
|
r.from = 'other';
|
|
r.redeemscript = false;
|
|
}
|
|
}
|
|
}
|
|
return r;
|
|
}
|
|
|
|
/* mediator payment code for when you used a public key */
|
|
function mediatorPayment(redeem){
|
|
|
|
if(redeem.from=="redeemScript"){
|
|
|
|
$('#recipients .row[rel="'+redeem.addr+'"]').parent().remove();
|
|
|
|
$.each(redeem.decodedRs.pubkeys, function(i, o){
|
|
$.each($("#mediatorList option"), function(mi, mo){
|
|
|
|
var ms = ($(mo).val()).split(";");
|
|
|
|
var pubkey = ms[0]; // mediators pubkey
|
|
var fee = ms[2]*1; // fee in a percentage
|
|
var payto = coinjs.pubkey2address(pubkey); // pay to mediators address
|
|
|
|
if(o==pubkey){ // matched a mediators pubkey?
|
|
|
|
var clone = '<span><div class="row recipients mediator mediator_'+pubkey+'" rel="'+redeem.addr+'">'+$("#recipients .addressAddTo").parent().parent().html()+'</div><br></span>';
|
|
$("#recipients").prepend(clone);
|
|
|
|
$("#recipients .mediator_"+pubkey+" .glyphicon-plus:first").removeClass('glyphicon-plus');
|
|
$("#recipients .mediator_"+pubkey+" .address:first").val(payto).attr('disabled', true).attr('readonly',true).attr('title','Medation fee for '+$(mo).html());
|
|
|
|
var amount = ((fee*$("#totalInput").html())/100).toFixed(8);
|
|
$("#recipients .mediator_"+pubkey+" .amount:first").attr('disabled',(((amount*1)==0)?false:true)).val(amount).attr('title','Medation fee for '+$(mo).html());
|
|
}
|
|
});
|
|
});
|
|
|
|
validateOutputAmount();
|
|
}
|
|
}
|
|
|
|
/* global function to add outputs to page */
|
|
function addOutput(tx, n, script, amount) {
|
|
if(tx){
|
|
if($("#inputs .txId:last").val()!=""){
|
|
$("#inputs .txidAdd").click();
|
|
}
|
|
|
|
$("#inputs .row:last input").attr('disabled',true);
|
|
|
|
var txid = ((tx).match(/.{1,2}/g).reverse()).join("")+'';
|
|
|
|
$("#inputs .txId:last").val(txid);
|
|
$("#inputs .txIdN:last").val(n);
|
|
$("#inputs .txIdAmount:last").val(amount);
|
|
|
|
if(((script.match(/^00/) && script.length==44)) || (script.length==40 && script.match(/^[a-f0-9]+$/gi))){
|
|
s = coinjs.script();
|
|
s.writeBytes(Crypto.util.hexToBytes(script));
|
|
s.writeOp(0);
|
|
s.writeBytes(coinjs.numToBytes((amount*100000000).toFixed(0), 8));
|
|
script = Crypto.util.bytesToHex(s.buffer);
|
|
}
|
|
|
|
$("#inputs .txIdScript:last").val(script);
|
|
}
|
|
}
|
|
|
|
/* global function to add inputs to page */
|
|
function addInput(address, amount) {
|
|
if($("#recipients .recipient:last .address:last").val() != ""){
|
|
$("#recipients .addressAddTo:first").click();
|
|
};
|
|
|
|
$("#recipients .address:last").val(address);
|
|
$("#recipients .amount:last").val(amount);
|
|
}
|
|
|
|
|
|
/* default function to retreive unspent outputs*/
|
|
function listUnspentDefault(redeem){
|
|
|
|
var tx = coinjs.transaction();
|
|
|
|
// unspent from transaction; double spend and RBF.
|
|
|
|
if(redeem.from == 'txid'){
|
|
tx.getTransaction(redeem.addr, function(data){
|
|
|
|
$("#redeemFromAddress").removeClass('hidden').html('<span class="glyphicon glyphicon-info-sign"></span> Attempted to rebuild transaction id <a href="'+explorer_tx+redeem.addr+'" target="_blank">'+redeem.addr+'</a>');
|
|
|
|
$.each($(data).find("inputs").children(), function(i,o){
|
|
var tx = $(o).find("txid").text();
|
|
var n = $(o).find("output_no").text();
|
|
var amount = (($(o).find("value").text()*1)).toFixed(8);
|
|
|
|
var scr = $(o).find("script").text();
|
|
|
|
addOutput(tx, n, scr, amount);
|
|
|
|
});
|
|
|
|
$("#recipients .addressRemoveTo").click();
|
|
$("#recipients .address").val("");
|
|
$("#recipients .amount").val("");
|
|
|
|
$.each($(data).find("outputs").children(), function(i,o){
|
|
addInput($(o).find("address").text(), $(o).find("value").text());
|
|
});
|
|
|
|
$("#redeemFromBtn").html("Load").attr('disabled',false);
|
|
totalInputAmount();
|
|
validateOutputAmount();
|
|
|
|
});
|
|
|
|
return;
|
|
}
|
|
|
|
// unspent from address
|
|
|
|
tx.listUnspent(redeem.addr, function(data){
|
|
if(redeem.addr) {
|
|
$("#redeemFromAddress").removeClass('hidden').html('<span class="glyphicon glyphicon-info-sign"></span> Retrieved unspent inputs from address <a href="'+explorer_addr+redeem.addr+'" target="_blank">'+redeem.addr+'</a>');
|
|
|
|
$.each($(data).find("unspent").children(), function(i,o){
|
|
var tx = $(o).find("tx_hash").text();
|
|
var n = $(o).find("tx_output_n").text();
|
|
var script = (redeem.redeemscript==true) ? redeem.decodedRs : $(o).find("script").text();
|
|
var amount = (($(o).find("value").text()*1)/100000000).toFixed(8);
|
|
|
|
addOutput(tx, n, script, amount);
|
|
});
|
|
}
|
|
|
|
$("#redeemFromBtn").html("Load").attr('disabled',false);
|
|
totalInputAmount();
|
|
|
|
mediatorPayment(redeem);
|
|
});
|
|
}
|
|
|
|
|
|
/* retrieve unspent data from blockcypher */
|
|
function listUnspentBlockcypher(redeem,network){
|
|
$.ajax ({
|
|
type: "GET",
|
|
url: "https://api.blockcypher.com/v1/"+network+"/main/addrs/"+redeem.addr+"?includeScript=true&unspentOnly=true",
|
|
dataType: "json",
|
|
error: function(data) {
|
|
$("#redeemFromStatus").removeClass('hidden').html('<span class="glyphicon glyphicon-exclamation-sign"></span> Unexpected error, unable to retrieve unspent outputs!');
|
|
},
|
|
success: function(data) {
|
|
if (data.address) { // address field will always be present, txrefs is only present if there are UTXOs
|
|
$("#redeemFromAddress").removeClass('hidden').html('<span class="glyphicon glyphicon-info-sign"></span> Retrieved unspent inputs from address <a href="'+explorer_addr+redeem.addr+'" target="_blank">'+redeem.addr+'</a>');
|
|
for(var i in data.txrefs){
|
|
var o = data.txrefs[i];
|
|
var tx = ((""+o.tx_hash).match(/.{1,2}/g).reverse()).join("")+'';
|
|
if(tx.match(/^[a-f0-9]+$/)){
|
|
var n = o.tx_output_n;
|
|
var script = (redeem.redeemscript==true) ? redeem.decodedRs : o.script;
|
|
var amount = ((o.value.toString()*1)/100000000).toFixed(8);
|
|
addOutput(tx, n, script, amount);
|
|
}
|
|
}
|
|
} else {
|
|
$("#redeemFromStatus").removeClass('hidden').html('<span class="glyphicon glyphicon-exclamation-sign"></span> Unexpected error, unable to retrieve unspent outputs.');
|
|
}
|
|
},
|
|
complete: function(data, status) {
|
|
$("#redeemFromBtn").html("Load").attr('disabled',false);
|
|
totalInputAmount();
|
|
}
|
|
});
|
|
}
|
|
|
|
/* retrieve unspent data from blockchair */
|
|
function listUnspentBlockchair(redeem,network){
|
|
$.ajax ({
|
|
type: "GET",
|
|
url: "https://api.blockchair.com/"+network+"/dashboards/address/"+redeem.addr,
|
|
dataType: "json",
|
|
error: function(data) {
|
|
$("#redeemFromStatus").removeClass('hidden').html('<span class="glyphicon glyphicon-exclamation-sign"></span> Unexpected error, unable to retrieve unspent outputs!');
|
|
},
|
|
success: function(data) {
|
|
if((data.context && data.data) && data.context.code =='200'){
|
|
$("#redeemFromAddress").removeClass('hidden').html('<span class="glyphicon glyphicon-info-sign"></span> Retrieved unspent inputs from address <a href="'+explorer_addr+redeem.addr+'" target="_blank">'+redeem.addr+'</a>');
|
|
var all_info = data.data[redeem.addr];
|
|
for(var i in all_info.utxo){
|
|
var o = all_info.utxo[i];
|
|
var tx = ((""+o.transaction_hash).match(/.{1,2}/g).reverse()).join("")+'';
|
|
if(tx.match(/^[a-f0-9]+$/)){
|
|
var n = o.index;
|
|
var script = (redeem.redeemscript==true) ? redeem.decodedRs : all_info.address.script_hex;
|
|
var amount = ((o.value.toString()*1)/100000000).toFixed(8);
|
|
addOutput(tx, n, script, amount);
|
|
}
|
|
}
|
|
} else {
|
|
$("#redeemFromStatus").removeClass('hidden').html('<span class="glyphicon glyphicon-exclamation-sign"></span> Unexpected error, unable to retrieve unspent outputs.');
|
|
}
|
|
},
|
|
complete: function(data, status) {
|
|
$("#redeemFromBtn").html("Load").attr('disabled',false);
|
|
totalInputAmount();
|
|
}
|
|
});
|
|
}
|
|
|
|
|
|
/* retrieve unspent data from chainso */
|
|
function listUnspentChainso(redeem, network){
|
|
$.ajax ({
|
|
type: "GET",
|
|
url: "https://chain.so/api/v2/get_tx_unspent/"+network+"/"+redeem.addr,
|
|
dataType: "json",
|
|
error: function(data) {
|
|
$("#redeemFromStatus").removeClass('hidden').html('<span class="glyphicon glyphicon-exclamation-sign"></span> Unexpected error, unable to retrieve unspent outputs!');
|
|
},
|
|
success: function(data) {
|
|
if((data.status && data.data) && data.status=='success'){
|
|
$("#redeemFromAddress").removeClass('hidden').html('<span class="glyphicon glyphicon-info-sign"></span> Retrieved unspent inputs from address <a href="'+explorer_addr+redeem.addr+'" target="_blank">'+redeem.addr+'</a>');
|
|
for(var i in data.data.txs){
|
|
var o = data.data.txs[i];
|
|
var tx = ((""+o.txid).match(/.{1,2}/g).reverse()).join("")+'';
|
|
if(tx.match(/^[a-f0-9]+$/)){
|
|
var n = o.output_no;
|
|
var script = (redeem.redeemscript==true) ? redeem.decodedRs : o.script_hex;
|
|
var amount = o.value;
|
|
addOutput(tx, n, script, amount);
|
|
}
|
|
}
|
|
} else {
|
|
$("#redeemFromStatus").removeClass('hidden').html('<span class="glyphicon glyphicon-exclamation-sign"></span> Unexpected error, unable to retrieve unspent outputs.');
|
|
}
|
|
},
|
|
complete: function(data, status) {
|
|
$("#redeemFromBtn").html("Load").attr('disabled',false);
|
|
totalInputAmount();
|
|
}
|
|
});
|
|
}
|
|
|
|
|
|
/* math to calculate the inputs and outputs */
|
|
|
|
function totalInputAmount(){
|
|
$("#totalInput").html('0.00');
|
|
$.each($("#inputs .txIdAmount"), function(i,o){
|
|
if(isNaN($(o).val())){
|
|
$(o).parent().addClass('has-error');
|
|
} else {
|
|
$(o).parent().removeClass('has-error');
|
|
var f = 0;
|
|
if(!isNaN($(o).val())){
|
|
f += $(o).val()*1;
|
|
}
|
|
$("#totalInput").html((($("#totalInput").html()*1) + (f*1)).toFixed(8));
|
|
}
|
|
});
|
|
totalFee();
|
|
}
|
|
|
|
function validateOutputAmount(){
|
|
$("#recipients .amount").unbind('');
|
|
$("#recipients .amount").keyup(function(){
|
|
if(isNaN($(this).val())){
|
|
$(this).parent().addClass('has-error');
|
|
} else {
|
|
$(this).parent().removeClass('has-error');
|
|
var f = 0;
|
|
$.each($("#recipients .amount"),function(i,o){
|
|
if(!isNaN($(o).val())){
|
|
f += $(o).val()*1;
|
|
}
|
|
});
|
|
$("#totalOutput").html((f).toFixed(8));
|
|
}
|
|
totalFee();
|
|
}).keyup();
|
|
}
|
|
|
|
function totalFee(){
|
|
var fee = (($("#totalInput").html()*1) - ($("#totalOutput").html()*1)).toFixed(8);
|
|
$("#transactionFee").val((fee>0)?fee:'0.00');
|
|
}
|
|
|
|
$(".optionsCollapse").click(function(){
|
|
if($(".optionsAdvanced",$(this).parent()).hasClass('hidden')){
|
|
$(".glyphcollapse",$(this).parent()).removeClass('glyphicon-collapse-down').addClass('glyphicon-collapse-up');
|
|
$(".optionsAdvanced",$(this).parent()).removeClass("hidden");
|
|
} else {
|
|
$(".glyphcollapse",$(this).parent()).removeClass('glyphicon-collapse-up').addClass('glyphicon-collapse-down');
|
|
$(".optionsAdvanced",$(this).parent()).addClass("hidden");
|
|
}
|
|
});
|
|
|
|
/* broadcast a transaction */
|
|
|
|
$("#rawSubmitBtn").click(function(){
|
|
rawSubmitDefault(this);
|
|
});
|
|
|
|
// broadcast transaction via coinbin (default)
|
|
function rawSubmitDefault(btn){
|
|
var thisbtn = btn;
|
|
$(thisbtn).val('Please wait, loading...').attr('disabled',true);
|
|
$.ajax ({
|
|
type: "POST",
|
|
url: coinjs.host+'?uid='+coinjs.uid+'&key='+coinjs.key+'&setmodule=bitcoin&request=sendrawtransaction',
|
|
data: {'rawtx':$("#rawTransaction").val()},
|
|
dataType: "xml",
|
|
error: function(data) {
|
|
$("#rawTransactionStatus").addClass('alert-danger').removeClass('alert-success').removeClass("hidden").html(" There was an error submitting your request, please try again").prepend('<span class="glyphicon glyphicon-exclamation-sign"></span>');
|
|
},
|
|
success: function(data) {
|
|
$("#rawTransactionStatus").html(unescape($(data).find("response").text()).replace(/\+/g,' ')).removeClass('hidden');
|
|
if($(data).find("result").text()==1){
|
|
$("#rawTransactionStatus").addClass('alert-success').removeClass('alert-danger').removeClass("hidden").html(' TXID: ' + $(data).find("txid").text() + '<br> <a href="https://coinb.in/tx/' + $(data).find("txid").text() + '" target="_blank">View on Blockchain</a>');
|
|
} else {
|
|
$("#rawTransactionStatus").addClass('alert-danger').removeClass('alert-success').prepend('<span class="glyphicon glyphicon-exclamation-sign"></span> ');
|
|
}
|
|
},
|
|
complete: function(data, status) {
|
|
$("#rawTransactionStatus").fadeOut().fadeIn();
|
|
$(thisbtn).val('Submit').attr('disabled',false);
|
|
}
|
|
});
|
|
}
|
|
|
|
// broadcast transaction via chain.so (mainnet)
|
|
function rawSubmitChainso(thisbtn, network){
|
|
$(thisbtn).val('Please wait, loading...').attr('disabled',true);
|
|
$.ajax ({
|
|
type: "POST",
|
|
url: "https://chain.so/api/v2/send_tx/"+network+"/",
|
|
data: {"tx_hex":$("#rawTransaction").val()},
|
|
dataType: "json",
|
|
error: function(data) {
|
|
var obj = $.parseJSON(data.responseText);
|
|
var r = ' ';
|
|
r += (obj.data.tx_hex) ? obj.data.tx_hex : '';
|
|
r = (r!='') ? r : ' Failed to broadcast'; // build response
|
|
$("#rawTransactionStatus").addClass('alert-danger').removeClass('alert-success').removeClass("hidden").html(r).prepend('<span class="glyphicon glyphicon-exclamation-sign"></span>');
|
|
},
|
|
success: function(data) {
|
|
if(data.status && data.data.txid){
|
|
$("#rawTransactionStatus").addClass('alert-success').removeClass('alert-danger').removeClass("hidden").html(' TXID: ' + data.data.txid + '<br> <a href="https://chain.so/tx/'+network+'/' + data.data.txid + '" target="_blank">View on Blockchain Explorer</a>');
|
|
} else {
|
|
$("#rawTransactionStatus").addClass('alert-danger').removeClass('alert-success').removeClass("hidden").html(' Unexpected error, please try again').prepend('<span class="glyphicon glyphicon-exclamation-sign"></span>');
|
|
}
|
|
},
|
|
complete: function(data, status) {
|
|
$("#rawTransactionStatus").fadeOut().fadeIn();
|
|
$(thisbtn).val('Submit').attr('disabled',false);
|
|
}
|
|
});
|
|
}
|
|
|
|
// broadcast transaction via blockcypher.com (mainnet)
|
|
function rawSubmitblockcypher(thisbtn, network){
|
|
$(thisbtn).val('Please wait, loading...').attr('disabled',true);
|
|
$.ajax ({
|
|
type: "POST",
|
|
url: "https://api.blockcypher.com/v1/"+network+"/main/txs/push",
|
|
data: JSON.stringify({"tx":$("#rawTransaction").val()}),
|
|
error: function(data) {
|
|
var r = 'Failed to broadcast: error code=' + data.status.toString() + ' ' + data.statusText;
|
|
$("#rawTransactionStatus").addClass('alert-danger').removeClass('alert-success').removeClass("hidden").html(r).prepend('<span class="glyphicon glyphicon-exclamation-sign"></span>');
|
|
},
|
|
success: function(data) {
|
|
if((data.tx) && data.tx.hash){
|
|
$("#rawTransactionStatus").addClass('alert-success').removeClass('alert-danger').removeClass("hidden")
|
|
.html(' TXID: ' + data.tx.hash + '<br> <a href="https://live.blockcypher.com/'+network+'/tx/' + data.tx.hash + '" target="_blank">View on Blockchain Explorer</a>');
|
|
} else {
|
|
$("#rawTransactionStatus").addClass('alert-danger').removeClass('alert-success').removeClass("hidden").html(' Unexpected error, please try again').prepend('<span class="glyphicon glyphicon-exclamation-sign"></span>');
|
|
}
|
|
},
|
|
complete: function(data, status) {
|
|
$("#rawTransactionStatus").fadeOut().fadeIn();
|
|
$(thisbtn).val('Submit').attr('disabled',false);
|
|
}
|
|
});
|
|
}
|
|
|
|
// broadcast transaction via blockchair
|
|
function rawSubmitblockchair(thisbtn, network){
|
|
$(thisbtn).val('Please wait, loading...').attr('disabled',true);
|
|
$.ajax ({
|
|
type: "POST",
|
|
url: "https://api.blockchair.com/"+network+"/push/transaction",
|
|
data: {"data":$("#rawTransaction").val()},
|
|
dataType: "json",
|
|
error: function(data) {
|
|
var r = 'Failed to broadcast: error code=' + data.status.toString() + ' ' + data.statusText;
|
|
$("#rawTransactionStatus").addClass('alert-danger').removeClass('alert-success').removeClass("hidden").html(r).prepend('<span class="glyphicon glyphicon-exclamation-sign"></span>');
|
|
// console.error(JSON.stringify(data, null, 4));
|
|
},
|
|
success: function(data) {
|
|
// console.info(JSON.stringify(data, null, 4));
|
|
if((data.context && data.data) && data.context.code=='200'){
|
|
$("#rawTransactionStatus").addClass('alert-success').removeClass('alert-danger').removeClass("hidden")
|
|
.html(' TXID: ' + data.data.transaction_hash + '<br> <a href="https://blockchair.com/'+network+'/transaction/' + data.data.transaction_hash + '" target="_blank">View on Blockchain Explorer</a>');
|
|
} else {
|
|
$("#rawTransactionStatus").addClass('alert-danger').removeClass('alert-success').removeClass("hidden").html(' Unexpected error, please try again').prepend('<span class="glyphicon glyphicon-exclamation-sign"></span>');
|
|
}
|
|
},
|
|
complete: function(data, status) {
|
|
$("#rawTransactionStatus").fadeOut().fadeIn();
|
|
$(thisbtn).val('Submit').attr('disabled',false);
|
|
}
|
|
});
|
|
}
|
|
|
|
|
|
|
|
|
|
/* verify script code */
|
|
|
|
$("#verifyBtn").click(function(){
|
|
$(".verifyData").addClass("hidden");
|
|
$("#verifyStatus").hide();
|
|
if(!decodeRedeemScript()){
|
|
if(!decodeTransactionScript()){
|
|
if(!decodePrivKey()){
|
|
if(!decodePubKey()){
|
|
if(!decodeHDaddress()){
|
|
$("#verifyStatus").removeClass('hidden').fadeOut().fadeIn();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
});
|
|
|
|
function decodeRedeemScript(){
|
|
var script = coinjs.script();
|
|
var decode = script.decodeRedeemScript($("#verifyScript").val());
|
|
if(decode){
|
|
$("#verifyRsDataMultisig").addClass('hidden');
|
|
$("#verifyRsDataHodl").addClass('hidden');
|
|
$("#verifyRsDataSegWit").addClass('hidden');
|
|
$("#verifyRsData").addClass("hidden");
|
|
|
|
|
|
if(decode.type == "multisig__") {
|
|
$("#verifyRsDataMultisig .multisigAddress").val(decode['address']);
|
|
$("#verifyRsDataMultisig .signaturesRequired").html(decode['signaturesRequired']);
|
|
$("#verifyRsDataMultisig table tbody").html("");
|
|
for(var i=0;i<decode.pubkeys.length;i++){
|
|
var pubkey = decode.pubkeys[i];
|
|
var address = coinjs.pubkey2address(pubkey);
|
|
$('<tr><td width="30%"><input type="text" class="form-control" value="'+address+'" readonly></td><td><input type="text" class="form-control" value="'+pubkey+'" readonly></td></tr>').appendTo("#verifyRsDataMultisig table tbody");
|
|
}
|
|
$("#verifyRsData").removeClass("hidden");
|
|
$("#verifyRsDataMultisig").removeClass('hidden');
|
|
$(".verifyLink").attr('href','?verify='+$("#verifyScript").val());
|
|
return true;
|
|
} else if(decode.type == "segwit__"){
|
|
$("#verifyRsData").removeClass("hidden");
|
|
$("#verifyRsDataSegWit .segWitAddress").val(decode['address']);
|
|
$("#verifyRsDataSegWit").removeClass('hidden');
|
|
return true;
|
|
} else if(decode.type == "hodl__") {
|
|
var d = $("#verifyRsDataHodl .date").data("DateTimePicker");
|
|
$("#verifyRsDataHodl .address").val(decode['address']);
|
|
$("#verifyRsDataHodl .pubkey").val(coinjs.pubkey2address(decode['pubkey']));
|
|
$("#verifyRsDataHodl .date").val(decode['checklocktimeverify'] >= 500000000? moment.unix(decode['checklocktimeverify']).format("MM/DD/YYYY HH:mm") : decode['checklocktimeverify']);
|
|
$("#verifyRsData").removeClass("hidden");
|
|
$("#verifyRsDataHodl").removeClass('hidden');
|
|
$(".verifyLink").attr('href','?verify='+$("#verifyScript").val());
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
function decodeTransactionScript(){
|
|
var tx = coinjs.transaction();
|
|
try {
|
|
var decode = tx.deserialize($("#verifyScript").val());
|
|
$("#verifyTransactionData .transactionVersion").html(decode['version']);
|
|
$("#verifyTransactionData .transactionSize").html(decode.size()+' <i>bytes</i>');
|
|
$("#verifyTransactionData .transactionLockTime").html(decode['lock_time']);
|
|
$("#verifyTransactionData .transactionRBF").hide();
|
|
$("#verifyTransactionData .transactionSegWit").hide();
|
|
if (decode.witness.length>=1) {
|
|
$("#verifyTransactionData .transactionSegWit").show();
|
|
}
|
|
$("#verifyTransactionData").removeClass("hidden");
|
|
$("#verifyTransactionData tbody").html("");
|
|
|
|
var h = '';
|
|
$.each(decode.ins, function(i,o){
|
|
var s = decode.extractScriptKey(i);
|
|
h += '<tr>';
|
|
h += '<td><input class="form-control" type="text" value="'+o.outpoint.hash+'" readonly></td>';
|
|
h += '<td class="col-xs-1">'+o.outpoint.index+'</td>';
|
|
h += '<td class="col-xs-2"><input class="form-control" type="text" value="'+Crypto.util.bytesToHex(o.script.buffer)+'" readonly></td>';
|
|
h += '<td class="col-xs-1"> <span class="glyphicon glyphicon-'+((s.signed=='true' || (decode.witness[i] && decode.witness[i].length==2))?'ok':'remove')+'-circle"></span>';
|
|
if(s['type']=='multisig' && s['signatures']>=1){
|
|
h += ' '+s['signatures'];
|
|
}
|
|
h += '</td>';
|
|
h += '<td class="col-xs-1">';
|
|
if(s['type']=='multisig'){
|
|
var script = coinjs.script();
|
|
var rs = script.decodeRedeemScript(s.script);
|
|
h += rs['signaturesRequired']+' of '+rs['pubkeys'].length;
|
|
} else {
|
|
h += '<span class="glyphicon glyphicon-remove-circle"></span>';
|
|
}
|
|
h += '</td>';
|
|
h += '</tr>';
|
|
|
|
//debug
|
|
if(parseInt(o.sequence)<(0xFFFFFFFF-1)){
|
|
$("#verifyTransactionData .transactionRBF").show();
|
|
}
|
|
});
|
|
|
|
$(h).appendTo("#verifyTransactionData .ins tbody");
|
|
|
|
h = '';
|
|
$.each(decode.outs, function(i,o){
|
|
|
|
if(o.script.chunks.length==2 && o.script.chunks[0]==106){ // OP_RETURN
|
|
|
|
var data = Crypto.util.bytesToHex(o.script.chunks[1]);
|
|
var dataascii = hex2ascii(data);
|
|
|
|
if(dataascii.match(/^[\s\d\w]+$/ig)){
|
|
data = dataascii;
|
|
}
|
|
|
|
h += '<tr>';
|
|
h += '<td><input type="text" class="form-control" value="(OP_RETURN) '+data+'" readonly></td>';
|
|
h += '<td class="col-xs-1">'+(o.value/100000000).toFixed(8)+'</td>';
|
|
h += '<td class="col-xs-2"><input class="form-control" type="text" value="'+Crypto.util.bytesToHex(o.script.buffer)+'" readonly></td>';
|
|
h += '</tr>';
|
|
} else {
|
|
|
|
var addr = '';
|
|
if(o.script.chunks.length==5){
|
|
addr = coinjs.scripthash2address(Crypto.util.bytesToHex(o.script.chunks[2]));
|
|
} else if((o.script.chunks.length==2) && o.script.chunks[0]==0){
|
|
addr = coinjs.bech32_encode(coinjs.bech32.hrp, [coinjs.bech32.version].concat(coinjs.bech32_convert(o.script.chunks[1], 8, 5, true)));
|
|
} else {
|
|
var pub = coinjs.pub;
|
|
coinjs.pub = coinjs.multisig;
|
|
addr = coinjs.scripthash2address(Crypto.util.bytesToHex(o.script.chunks[1]));
|
|
coinjs.pub = pub;
|
|
}
|
|
|
|
h += '<tr>';
|
|
h += '<td><input class="form-control" type="text" value="'+addr+'" readonly></td>';
|
|
h += '<td class="col-xs-1">'+(o.value/100000000).toFixed(8)+'</td>';
|
|
h += '<td class="col-xs-2"><input class="form-control" type="text" value="'+Crypto.util.bytesToHex(o.script.buffer)+'" readonly></td>';
|
|
h += '</tr>';
|
|
}
|
|
});
|
|
$(h).appendTo("#verifyTransactionData .outs tbody");
|
|
|
|
$(".verifyLink").attr('href','?verify='+$("#verifyScript").val());
|
|
return true;
|
|
} catch(e) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
function hex2ascii(hex) {
|
|
var str = '';
|
|
for (var i = 0; i < hex.length; i += 2)
|
|
str += String.fromCharCode(parseInt(hex.substr(i, 2), 16));
|
|
return str;
|
|
}
|
|
|
|
function decodePrivKey(){
|
|
var wif = $("#verifyScript").val();
|
|
if(wif.length==51 || wif.length==52){
|
|
try {
|
|
var w2address = coinjs.wif2address(wif);
|
|
var w2pubkey = coinjs.wif2pubkey(wif);
|
|
var w2privkey = coinjs.wif2privkey(wif);
|
|
|
|
$("#verifyPrivKey .address").val(w2address['address']);
|
|
$("#verifyPrivKey .pubkey").val(w2pubkey['pubkey']);
|
|
$("#verifyPrivKey .privkey").val(w2privkey['privkey']);
|
|
$("#verifyPrivKey .iscompressed").html(w2address['compressed']?'true':'false');
|
|
|
|
$("#verifyPrivKey").removeClass("hidden");
|
|
return true;
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
function decodePubKey(){
|
|
var pubkey = $("#verifyScript").val();
|
|
if(pubkey.length==66 || pubkey.length==130){
|
|
try {
|
|
$("#verifyPubKey .verifyDataSw").addClass('hidden');
|
|
$("#verifyPubKey .address").val(coinjs.pubkey2address(pubkey));
|
|
if(pubkey.length == 66){
|
|
var sw = coinjs.segwitAddress(pubkey);
|
|
$("#verifyPubKey .addressSegWit").val(sw.address);
|
|
$("#verifyPubKey .addressSegWitRedeemScript").val(sw.redeemscript);
|
|
|
|
var b32 = coinjs.bech32Address(pubkey);
|
|
$("#verifyPubKey .addressBech32").val(b32.address);
|
|
$("#verifyPubKey .addressBech32RedeemScript").val(b32.redeemscript);
|
|
|
|
$("#verifyPubKey .verifyDataSw").removeClass('hidden');
|
|
}
|
|
$("#verifyPubKey").removeClass("hidden");
|
|
$(".verifyLink").attr('href','?verify='+$("#verifyScript").val());
|
|
return true;
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
function decodeHDaddress(){
|
|
coinjs.compressed = true;
|
|
var s = $("#verifyScript").val();
|
|
try {
|
|
var hex = Crypto.util.bytesToHex((coinjs.base58decode(s)).slice(0,4));
|
|
var hex_cmp_prv = Crypto.util.bytesToHex((coinjs.numToBytes(coinjs.hdkey.prv,4)).reverse());
|
|
var hex_cmp_pub = Crypto.util.bytesToHex((coinjs.numToBytes(coinjs.hdkey.pub,4)).reverse());
|
|
if(hex == hex_cmp_prv || hex == hex_cmp_pub){
|
|
var hd = coinjs.hd(s);
|
|
$("#verifyHDaddress .hdKey").html(s);
|
|
$("#verifyHDaddress .chain_code").val(Crypto.util.bytesToHex(hd.chain_code));
|
|
$("#verifyHDaddress .depth").val(hd.depth);
|
|
$("#verifyHDaddress .version").val('0x'+(hd.version).toString(16));
|
|
$("#verifyHDaddress .child_index").val(hd.child_index);
|
|
$("#verifyHDaddress .hdwifkey").val((hd.keys.wif)?hd.keys.wif:'');
|
|
$("#verifyHDaddress .key_type").html((((hd.depth==0 && hd.child_index==0)?'Master':'Derived')+' '+hd.type).toLowerCase());
|
|
$("#verifyHDaddress .parent_fingerprint").val(Crypto.util.bytesToHex(hd.parent_fingerprint));
|
|
$("#verifyHDaddress .derived_data table tbody").html("");
|
|
deriveHDaddress();
|
|
$(".verifyLink").attr('href','?verify='+$("#verifyScript").val());
|
|
$("#verifyHDaddress").removeClass("hidden");
|
|
return true;
|
|
}
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
function deriveHDaddress() {
|
|
var hd = coinjs.hd($("#verifyHDaddress .hdKey").html());
|
|
var index_start = $("#verifyHDaddress .derivation_index_start").val()*1;
|
|
var index_end = $("#verifyHDaddress .derivation_index_end").val()*1;
|
|
var html = '';
|
|
$("#verifyHDaddress .derived_data table tbody").html("");
|
|
for(var i=index_start;i<=index_end;i++){
|
|
if($("#hdpathtype option:selected").val()=='simple'){
|
|
var derived = hd.derive(i);
|
|
} else {
|
|
var derived = hd.derive_path(($("#hdpath input").val().replace(/\/+$/, ""))+'/'+i);
|
|
}
|
|
html += '<tr>';
|
|
html += '<td>'+i+'</td>';
|
|
html += '<td><input type="text" class="form-control" value="'+derived.keys.address+'" readonly></td>';
|
|
html += '<td><input type="text" class="form-control" value="'+((derived.keys.wif)?derived.keys.wif:'')+'" readonly></td>';
|
|
html += '<td><input type="text" class="form-control" value="'+derived.keys_extended.pubkey+'" readonly></td>';
|
|
html += '<td><input type="text" class="form-control" value="'+((derived.keys_extended.privkey)?derived.keys_extended.privkey:'')+'" readonly></td>';
|
|
html += '</tr>';
|
|
}
|
|
$(html).appendTo("#verifyHDaddress .derived_data table tbody");
|
|
}
|
|
|
|
|
|
$("#hdpathtype").change(function(){
|
|
if($(this).val()=='simple'){
|
|
$("#hdpath").removeClass().addClass("hidden");
|
|
} else {
|
|
$("#hdpath").removeClass();
|
|
}
|
|
});
|
|
|
|
|
|
/* sign code */
|
|
|
|
$("#signBtn").click(function(){
|
|
var wifkey = $("#signPrivateKey");
|
|
var script = $("#signTransaction");
|
|
|
|
if(coinjs.addressDecode(wifkey.val())){
|
|
$(wifkey).parent().removeClass('has-error');
|
|
} else {
|
|
$(wifkey).parent().addClass('has-error');
|
|
}
|
|
|
|
if((script.val()).match(/^[a-f0-9]+$/ig)){
|
|
$(script).parent().removeClass('has-error');
|
|
} else {
|
|
$(script).parent().addClass('has-error');
|
|
}
|
|
|
|
if($("#sign .has-error").length==0){
|
|
$("#signedDataError").addClass('hidden');
|
|
try {
|
|
var tx = coinjs.transaction();
|
|
var t = tx.deserialize(script.val());
|
|
|
|
var signed = t.sign(wifkey.val(), $("#sighashType option:selected").val());
|
|
$("#signedData textarea").val(signed);
|
|
$("#signedData .txSize").html(t.size());
|
|
$("#signedData").removeClass('hidden').fadeIn();
|
|
} catch(e) {
|
|
// console.log(e);
|
|
}
|
|
} else {
|
|
$("#signedDataError").removeClass('hidden');
|
|
$("#signedData").addClass('hidden');
|
|
}
|
|
});
|
|
|
|
$("#sighashType").change(function(){
|
|
$("#sighashTypeInfo").html($("option:selected",this).attr('rel')).fadeOut().fadeIn();
|
|
});
|
|
|
|
$("#signAdvancedCollapse").click(function(){
|
|
if($("#signAdvanced").hasClass('hidden')){
|
|
$("span",this).removeClass('glyphicon-collapse-down').addClass('glyphicon-collapse-up');
|
|
$("#signAdvanced").removeClass("hidden");
|
|
} else {
|
|
$("span",this).removeClass('glyphicon-collapse-up').addClass('glyphicon-collapse-down');
|
|
$("#signAdvanced").addClass("hidden");
|
|
}
|
|
});
|
|
|
|
/* page load code */
|
|
|
|
function _get(value) {
|
|
var dataArray = (document.location.search).match(/(([a-z0-9\_\[\]]+\=[a-z0-9\_\.\%\@]+))/gi);
|
|
var r = [];
|
|
if(dataArray) {
|
|
for(var x in dataArray) {
|
|
if((dataArray[x]) && typeof(dataArray[x])=='string') {
|
|
if((dataArray[x].split('=')[0].toLowerCase()).replace(/\[\]$/ig,'') == value.toLowerCase()) {
|
|
r.push(unescape(dataArray[x].split('=')[1]));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return r;
|
|
}
|
|
|
|
var _getBroadcast = _get("broadcast");
|
|
if(_getBroadcast[0]){
|
|
$("#rawTransaction").val(_getBroadcast[0]);
|
|
$("#rawSubmitBtn").click();
|
|
window.location.hash = "#broadcast";
|
|
}
|
|
|
|
var _getVerify = _get("verify");
|
|
if(_getVerify[0]){
|
|
$("#verifyScript").val(_getVerify[0]);
|
|
$("#verifyBtn").click();
|
|
window.location.hash = "#verify";
|
|
}
|
|
|
|
$('a[data-toggle="tab"]').on('shown.bs.tab', function (e) {
|
|
if(e.target.hash == "#fees"){
|
|
feeStats();
|
|
}
|
|
})
|
|
|
|
$(".qrcodeBtn").click(function(){
|
|
$("#qrcode").html("");
|
|
var thisbtn = $(this).parent().parent();
|
|
var qrstr = false;
|
|
var ta = $("textarea",thisbtn);
|
|
|
|
if(ta.length>0){
|
|
var w = (screen.availWidth > screen.availHeight ? screen.availWidth : screen.availHeight)/3;
|
|
var qrcode = new QRCode("qrcode", {width:w, height:w});
|
|
qrstr = $(ta).val();
|
|
if(qrstr.length > 1024){
|
|
$("#qrcode").html("<p>Sorry the data is too long for the QR generator.</p>");
|
|
}
|
|
} else {
|
|
var qrcode = new QRCode("qrcode");
|
|
qrstr = "bitcoin:"+$('.address',thisbtn).val();
|
|
}
|
|
|
|
if(qrstr){
|
|
qrcode.makeCode(qrstr);
|
|
}
|
|
});
|
|
|
|
$('input[title!=""], abbr[title!=""]').tooltip({'placement':'bottom'});
|
|
|
|
if (location.hash !== ''){
|
|
$('a[href="' + location.hash + '"]').tab('show');
|
|
}
|
|
|
|
$(".showKey").click(function(){
|
|
$("input[type='password']",$(this).parent().parent()).attr('type','text');
|
|
});
|
|
|
|
$("#homeBtn").click(function(e){
|
|
e.preventDefault();
|
|
history.pushState(null, null, '#home');
|
|
$("#header .active, #content .tab-content").removeClass("active");
|
|
$("#home").addClass("active");
|
|
});
|
|
|
|
$('a[data-toggle="tab"]').on('click', function(e) {
|
|
e.preventDefault();
|
|
if(e.target && $(e.target).attr('href')) {
|
|
history.pushState(null, null, '#'+$(e.target).attr('href').substr(1));
|
|
}
|
|
});
|
|
|
|
window.addEventListener("popstate", function(e) {
|
|
var activeTab = $('[href=' + location.hash + ']');
|
|
if (activeTab.length) {
|
|
activeTab.tab('show');
|
|
} else {
|
|
$('.nav-tabs a:first').tab('show');
|
|
}
|
|
});
|
|
|
|
for(i=1;i<3;i++){
|
|
$(".pubkeyAdd").click();
|
|
}
|
|
|
|
validateOutputAmount();
|
|
|
|
/* settings page code */
|
|
|
|
$("#coinjs_pub").val('0x'+(coinjs.pub).toString(16));
|
|
$("#coinjs_priv").val('0x'+(coinjs.priv).toString(16));
|
|
$("#coinjs_multisig").val('0x'+(coinjs.multisig).toString(16));
|
|
|
|
$("#coinjs_hdpub").val('0x'+(coinjs.hdkey.pub).toString(16));
|
|
$("#coinjs_hdprv").val('0x'+(coinjs.hdkey.prv).toString(16));
|
|
|
|
$("#settingsBtn").click(function(){
|
|
|
|
// log out of openwallet
|
|
$("#walletLogout").click();
|
|
|
|
$("#statusSettings").removeClass("alert-success").removeClass("alert-danger").addClass("hidden").html("");
|
|
$("#settings .has-error").removeClass("has-error");
|
|
|
|
$.each($(".coinjssetting"),function(i, o){
|
|
if(!$(o).val().match(/^0x[0-9a-f]+$/)){
|
|
$(o).parent().addClass("has-error");
|
|
}
|
|
});
|
|
|
|
if($("#settings .has-error").length==0){
|
|
|
|
coinjs.pub = $("#coinjs_pub").val()*1;
|
|
coinjs.priv = $("#coinjs_priv").val()*1;
|
|
coinjs.multisig = $("#coinjs_multisig").val()*1;
|
|
|
|
coinjs.hdkey.pub = $("#coinjs_hdpub").val()*1;
|
|
coinjs.hdkey.prv = $("#coinjs_hdprv").val()*1;
|
|
|
|
configureBroadcast();
|
|
configureGetUnspentTx();
|
|
|
|
if (coinjs.pub == 0x30){ // LTC
|
|
explorer_addr = "https://chain.so/address/LTC/";
|
|
coinjs.bech32.hrp = "ltc";
|
|
}
|
|
else if (coinjs.pub == 0x1e){ // DOGE
|
|
explorer_addr = "https://chain.so/address/DOGE/";
|
|
}
|
|
|
|
$("#statusSettings").addClass("alert-success").removeClass("hidden").html("<span class=\"glyphicon glyphicon-ok\"></span> Settings updates successfully").fadeOut().fadeIn();
|
|
} else {
|
|
$("#statusSettings").addClass("alert-danger").removeClass("hidden").html("There is an error with one or more of your settings");
|
|
}
|
|
});
|
|
|
|
$("#coinjs_coin").change(function(){
|
|
|
|
var o = ($("option:selected",this).attr("rel")).split(";");
|
|
|
|
// deal with broadcasting settings
|
|
if(o[5]=="false"){
|
|
$("#coinjs_broadcast, #rawTransaction, #rawSubmitBtn, #openBtn").attr('disabled',true);
|
|
$("#coinjs_broadcast").val("coinb.in");
|
|
} else {
|
|
$("#coinjs_broadcast").val(o[5]);
|
|
$("#coinjs_broadcast, #rawTransaction, #rawSubmitBtn, #openBtn").attr('disabled',false);
|
|
}
|
|
|
|
// deal with unspent output settings
|
|
if(o[6]=="false"){
|
|
$("#coinjs_utxo, #redeemFrom, #redeemFromBtn, #openBtn, .qrcodeScanner").attr('disabled',true);
|
|
$("#coinjs_utxo").val("coinb.in");
|
|
} else {
|
|
$("#coinjs_utxo").val(o[6]);
|
|
$("#coinjs_utxo, #redeemFrom, #redeemFromBtn, #openBtn, .qrcodeScanner").attr('disabled',false);
|
|
}
|
|
|
|
// deal with the reset
|
|
$("#coinjs_pub").val(o[0]);
|
|
$("#coinjs_priv").val(o[1]);
|
|
$("#coinjs_multisig").val(o[2]);
|
|
$("#coinjs_hdpub").val(o[3]);
|
|
$("#coinjs_hdprv").val(o[4]);
|
|
|
|
// hide/show custom screen
|
|
if($("option:selected",this).val()=="custom"){
|
|
$("#settingsCustom").removeClass("hidden");
|
|
} else {
|
|
$("#settingsCustom").addClass("hidden");
|
|
}
|
|
});
|
|
|
|
function configureBroadcast(){
|
|
var host = $("#coinjs_broadcast option:selected").val();
|
|
|
|
// api: blockcypher blockchair chain.so
|
|
// network name "btc" "bitcoin" "BTC"
|
|
// network name "ltc" "litecoin" "LTC"
|
|
// network name "doge" "dogecoin" "DOGE"
|
|
|
|
$("#rawSubmitBtn").unbind("");
|
|
if(host=="chain.so_bitcoinmainnet"){
|
|
$("#rawSubmitBtn").click(function(){
|
|
rawSubmitChainso(this, "BTC");
|
|
});
|
|
} else if(host=="chain.so_litecoin"){
|
|
$("#rawSubmitBtn").click(function(){
|
|
rawSubmitChainso(this, "LTC");
|
|
});
|
|
} else if(host=="chain.so_dogecoin"){
|
|
$("#rawSubmitBtn").click(function(){
|
|
rawSubmitChainso(this, "DOGE");
|
|
});
|
|
} else if(host=="blockcypher_bitcoinmainnet"){
|
|
$("#rawSubmitBtn").click(function(){
|
|
rawSubmitblockcypher(this, "btc");
|
|
});
|
|
} else if(host=="blockcypher_litecoin"){
|
|
$("#rawSubmitBtn").click(function(){
|
|
rawSubmitblockcypher(this, "ltc");
|
|
});
|
|
} else if(host=="blockcypher_dogecoin"){
|
|
$("#rawSubmitBtn").click(function(){
|
|
rawSubmitblockcypher(this, "doge");
|
|
});
|
|
} else if(host=="blockchair_bitcoinmainnet"){
|
|
$("#rawSubmitBtn").click(function(){
|
|
rawSubmitblockchair(this, "bitcoin");
|
|
});
|
|
} else if(host=="blockchair_litecoin"){
|
|
$("#rawSubmitBtn").click(function(){
|
|
rawSubmitblockchair(this, "litecoin");
|
|
});
|
|
} else if(host=="blockchair_dogecoin"){
|
|
$("#rawSubmitBtn").click(function(){
|
|
rawSubmitblockchair(this, "dogecoin");
|
|
});
|
|
} else {
|
|
$("#rawSubmitBtn").click(function(){
|
|
rawSubmitDefault(this); // revert to default
|
|
});
|
|
}
|
|
}
|
|
|
|
function configureGetUnspentTx(){
|
|
$("#redeemFromBtn").attr('rel',$("#coinjs_utxo option:selected").val());
|
|
}
|
|
|
|
|
|
/* fees page code */
|
|
|
|
$("#fees .slider").on('input', function(){
|
|
$('.'+$(this).attr('rel')+' .inputno, .'+$(this).attr('rel')+' .outputno',$("#fees")).html($(this).val());
|
|
$('.'+$(this).attr('rel')+' .estimate',$("#fees")).removeClass('hidden');
|
|
});
|
|
|
|
$("#fees .txo_p2pkh").on('input', function(){
|
|
var outputno = $('.'+$(this).attr('rel')+' .outputno',$("#fees .txoutputs")).html();
|
|
$('.'+$(this).attr('rel')+' .bytes',$("#fees .txoutputs")).html((outputno*$("#est_txo_p2pkh").val())+(outputno*9));
|
|
mathFees();
|
|
});
|
|
|
|
$("#fees .txo_p2sh").on('input', function(){
|
|
var outputno = $('.'+$(this).attr('rel')+' .outputno',$("#fees .txoutputs")).html();
|
|
$('.'+$(this).attr('rel')+' .bytes',$("#fees .txoutputs")).html((outputno*$("#est_txo_p2sh").val())+(outputno*9));
|
|
mathFees();
|
|
});
|
|
|
|
$("#fees .txi_regular").on('input', function(){
|
|
var inputno = $('.'+$(this).attr('rel')+' .inputno',$("#fees .txinputs")).html();
|
|
$('.'+$(this).attr('rel')+' .bytes',$("#fees .txinputs")).html((inputno*$("#est_txi_regular").val())+(inputno*41));
|
|
mathFees();
|
|
});
|
|
|
|
$("#fees .txi_segwit").on('input', function(){
|
|
var inputno = $('.'+$(this).attr('rel')+' .inputno',$("#fees .txinputs")).html();
|
|
var bytes = 0;
|
|
if(inputno >= 1){
|
|
bytes = 2;
|
|
bytes += (inputno*32);
|
|
bytes += (inputno*$("#est_txi_segwit").val());
|
|
bytes += (inputno*(41))
|
|
}
|
|
|
|
bytes = bytes.toFixed(0);
|
|
$('.'+$(this).attr('rel')+' .bytes',$("#fees .txinputs")).html(bytes);
|
|
mathFees();
|
|
});
|
|
|
|
$("#fees .txi_multisig").on('input', function(){
|
|
var inputno = $('.'+$(this).attr('rel')+' .inputno',$("#fees .txinputs")).html();
|
|
$('.'+$(this).attr('rel')+' .bytes',$("#fees .txinputs")).html((inputno*$("#est_txi_multisig").val())+(inputno*41));
|
|
mathFees();
|
|
});
|
|
|
|
$("#fees .txi_hodl").on('input', function(){
|
|
var inputno = $('.'+$(this).attr('rel')+' .inputno',$("#fees .txinputs")).html();
|
|
$('.'+$(this).attr('rel')+' .bytes',$("#fees .txinputs")).html((inputno*$("#est_txi_hodl").val())+(inputno*41));
|
|
mathFees();
|
|
});
|
|
|
|
$("#fees .txi_unknown").on('input', function(){
|
|
var inputno = $('.'+$(this).attr('rel')+' .inputno',$("#fees .txinputs")).html();
|
|
$('.'+$(this).attr('rel')+' .bytes',$("#fees .txinputs")).html((inputno*$("#est_txi_unknown").val())+(inputno*41));
|
|
mathFees();
|
|
});
|
|
|
|
$("#fees .sliderbtn.down").click(function(){
|
|
var val = $(".slider",$(this).parent().parent()).val()*1;
|
|
if(val>($(".slider",$(this).parent().parent()).attr('min')*1)){
|
|
$(".slider",$(this).parent().parent()).val(val-1);
|
|
$(".slider",$(this).parent().parent()).trigger('input');
|
|
}
|
|
});
|
|
|
|
$("#fees .sliderbtn.up").click(function(){
|
|
var val = $(".slider",$(this).parent().parent()).val()*1;
|
|
if(val<($(".slider",$(this).parent().parent()).attr('max')*1)){
|
|
$(".slider",$(this).parent().parent()).val(val+1);
|
|
$(".slider",$(this).parent().parent()).trigger('input');
|
|
}
|
|
});
|
|
|
|
$("#advancedFeesCollapse").click(function(){
|
|
if($("#advancedFees").hasClass('hidden')){
|
|
$("span",this).removeClass('glyphicon-collapse-down').addClass('glyphicon-collapse-up');
|
|
$("#advancedFees").removeClass("hidden");
|
|
} else {
|
|
$("span",this).removeClass('glyphicon-collapse-up').addClass('glyphicon-collapse-down');
|
|
$("#advancedFees").addClass("hidden");
|
|
}
|
|
});
|
|
|
|
$("#feesAnalyseBtn").click(function(){
|
|
if(!$("#fees .txhex").val().match(/^[a-f0-9]+$/ig)){
|
|
alert('You must provide a hex encoded transaction');
|
|
return;
|
|
}
|
|
|
|
var tx = coinjs.transaction();
|
|
var deserialized = tx.deserialize($("#fees .txhex").val());
|
|
|
|
$("#fees .txoutputs .outputno, #fees .txinputs .inputno").html("0");
|
|
$("#fees .txoutputs .bytes, #fees .txinputs .bytes").html("0");
|
|
$("#fees .slider").val(0);
|
|
|
|
for(var i = 0; i < deserialized.ins.length; i++){
|
|
var script = deserialized.extractScriptKey(i);
|
|
var size = 41;
|
|
if(script.type == 'segwit'){
|
|
if(deserialized.witness[i]){
|
|
size += deserialized.ins[i].script.buffer.length / 2;
|
|
for(w in deserialized.witness[i]){
|
|
size += (deserialized.witness[i][w].length / 2) /4;
|
|
}
|
|
} else {
|
|
size += $("#est_txi_segwit").val()*1;
|
|
}
|
|
$("#fees .segwit .inputno").html(($("#fees .segwit .inputno").html()*1)+1);
|
|
$("#fees .txi_segwit").val(($("#fees .txi_segwit").val()*1)+1);
|
|
$("#fees .segwit .bytes").html(($("#fees .segwit .bytes").html()*1)+size);
|
|
|
|
} else if(script.type == 'multisig'){
|
|
var s = coinjs.script();
|
|
var rs = s.decodeRedeemScript(script.script);
|
|
size += 4 + ((script.script.length / 2) + (73 * rs.signaturesRequired));
|
|
$("#fees .multisig .inputno").html(($("#fees .multisig .inputno").html()*1)+1);
|
|
$("#fees .txi_multisig").val(($("#fees .txi_multisig").val()*1)+1);
|
|
$("#fees .multisig .bytes").html(($("#fees .multisig .bytes").html()*1)+size);
|
|
|
|
} else if(script.type == 'hodl'){
|
|
size += 78;
|
|
$("#fees .hodl .inputno").html(($("#fees .hodl .inputno").html()*1)+1);
|
|
$("#fees .txi_hodl").val(($("#fees .txi_hodl").val()*1)+1);
|
|
$("#fees .hodl .bytes").html(($("#fees .hodl .bytes").html()*1)+size);
|
|
|
|
} else if(script.type == 'empty' || script.type == 'scriptpubkey'){
|
|
if(script.signatures == 1){
|
|
size += script.script.length / 2;
|
|
} else {
|
|
size += $("#est_txi_regular").val()*1;
|
|
}
|
|
|
|
$("#fees .regular .inputno").html(($("#fees .regular .inputno").html()*1)+1);
|
|
$("#fees .txi_regular").val(($("#fees .txi_regular").val()*1)+1);
|
|
$("#fees .regular .bytes").html(($("#fees .regular .bytes").html()*1)+size);
|
|
|
|
} else if(script.type == 'unknown'){
|
|
size += script.script.length / 2;
|
|
$("#fees .unknown .inputno").html(($("#fees .unknown .inputno").html()*1)+1);
|
|
$("#fees .txi_unknown").val(($("#fees .txi_unknown").val()*1)+1);
|
|
$("#fees .unknown .bytes").html(($("#fees .unknown .bytes").html()*1)+size);
|
|
}
|
|
}
|
|
|
|
for(var i = 0; i < deserialized.outs.length; i++){
|
|
if(deserialized.outs[i].script.buffer[0]==118){
|
|
$("#fees .txoutputs .p2pkh .outputno").html(($("#fees .txoutputs .p2pkh .outputno").html()*1)+1);
|
|
$("#fees .txoutputs .p2pkh .bytes").html(($("#fees .txoutputs .p2pkh .bytes").html()*1)+34);
|
|
$("#fees .txo_p2pkh").val(($("#fees .txo_p2pkh").val()*1)+1);
|
|
} else if (deserialized.outs[i].script.buffer[0]==169){
|
|
$("#fees .txoutputs .p2sh .outputno").html(($("#fees .txoutputs .p2sh .outputno").html()*1)+1);
|
|
$("#fees .txoutputs .p2sh .bytes").html(($("#fees .txoutputs .p2sh .bytes").html()*1)+32);
|
|
$("#fees .txo_p2sh").val(($("#fees .txo_p2sh").val()*1)+1);
|
|
}
|
|
}
|
|
|
|
feeStats();
|
|
});
|
|
|
|
$("#feeStatsReload").click(function(){
|
|
feeStats();
|
|
});
|
|
|
|
function mathFees(){
|
|
|
|
var inputsTotal = 0;
|
|
var inputsBytes = 0;
|
|
$.each($(".inputno"), function(i,o){
|
|
inputsTotal += ($(o).html()*1);
|
|
inputsBytes += ($(".bytes",$(o).parent()).html()*1);
|
|
});
|
|
|
|
$("#fees .txinputs .txsize").html(inputsBytes.toFixed(0));
|
|
$("#fees .txinputs .txtotal").html(inputsTotal.toFixed(0));
|
|
|
|
var outputsTotal = 0;
|
|
var outputsBytes = 0;
|
|
$.each($(".outputno"), function(i,o){
|
|
outputsTotal += ($(o).html()*1);
|
|
outputsBytes += ($(".bytes",$(o).parent()).html()*1);
|
|
});
|
|
|
|
$("#fees .txoutputs .txsize").html(outputsBytes.toFixed(0));
|
|
$("#fees .txoutputs .txtotal").html(outputsTotal.toFixed(0));
|
|
|
|
var totalBytes = 10 + outputsBytes + inputsBytes;
|
|
if((!isNaN($("#fees .feeSatByte:first").html())) && totalBytes > 10){
|
|
var recommendedFee = ((totalBytes * $(".feeSatByte").html())/100000000).toFixed(8);
|
|
$(".recommendedFee").html(recommendedFee);
|
|
$(".feeTxSize").html(totalBytes);
|
|
} else {
|
|
$(".recommendedFee").html((0).toFixed(8));
|
|
$(".feeTxSize").html(0);
|
|
}
|
|
};
|
|
|
|
function feeStats(){
|
|
$("#feeStatsReload").attr('disabled',true);
|
|
$.ajax ({
|
|
type: "GET",
|
|
url: "https://coinb.in/api/?uid=1&key=12345678901234567890123456789012&setmodule=fees&request=stats",
|
|
dataType: "xml",
|
|
error: function(data) {
|
|
},
|
|
success: function(data) {
|
|
$("#fees .recommended .blockHeight").html('<a href="https://coinb.in/height/'+$(data).find("height").text()+'" target="_blank">'+$(data).find("height").text()+'</a>');
|
|
$("#fees .recommended .blockHash").html($(data).find("block").text());
|
|
$("#fees .recommended .blockTime").html($(data).find("timestamp").text());
|
|
$("#fees .recommended .blockDateTime").html(unescape($(data).find("datetime").text()).replace(/\+/g,' '));
|
|
$("#fees .recommended .txId").html('<a href="https://coinb.in/tx/'+$(data).find("txid").text()+'" target="_blank">'+$(data).find("txid").text()+'</a>');
|
|
$("#fees .recommended .txSize").html($(data).find("txsize").text());
|
|
$("#fees .recommended .txFee").html($(data).find("txfee").text());
|
|
$("#fees .feeSatByte").html($(data).find("satbyte").text());
|
|
|
|
mathFees();
|
|
},
|
|
complete: function(data, status){
|
|
$("#feeStatsReload").attr('disabled', false);
|
|
}
|
|
});
|
|
}
|
|
|
|
/* capture mouse movement to add entropy */
|
|
var IE = document.all?true:false // Boolean, is browser IE?
|
|
if (!IE) document.captureEvents(Event.MOUSEMOVE)
|
|
document.onmousemove = getMouseXY;
|
|
function getMouseXY(e) {
|
|
var tempX = 0;
|
|
var tempY = 0;
|
|
if (IE) { // If browser is IE
|
|
tempX = event.clientX + document.body.scrollLeft;
|
|
tempY = event.clientY + document.body.scrollTop;
|
|
} else {
|
|
tempX = e.pageX;
|
|
tempY = e.pageY;
|
|
};
|
|
|
|
if (tempX < 0){tempX = 0};
|
|
if (tempY < 0){tempY = 0};
|
|
var xEnt = Crypto.util.bytesToHex([tempX]).slice(-2);
|
|
var yEnt = Crypto.util.bytesToHex([tempY]).slice(-2);
|
|
var addEnt = xEnt.concat(yEnt);
|
|
|
|
if ($("#entropybucket").html().indexOf(xEnt) == -1 && $("#entropybucket").html().indexOf(yEnt) == -1) {
|
|
$("#entropybucket").html(addEnt + $("#entropybucket").html());
|
|
};
|
|
|
|
if ($("#entropybucket").html().length > 128) {
|
|
$("#entropybucket").html($("#entropybucket").html().slice(0, 128))
|
|
};
|
|
|
|
return true;
|
|
};
|
|
|
|
});
|