1459 lines
49 KiB
JavaScript
1459 lines
49 KiB
JavaScript
$(document).ready(function() {
|
|
|
|
/* open wallet code */
|
|
|
|
$("#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);
|
|
|
|
$("#walletAddress").html(keys.address);
|
|
$("#walletHistory").attr('href','https://btc.blockr.io/address/info/'+keys.address);
|
|
|
|
$("#walletQrCode").html("");
|
|
var qrcode = new QRCode("walletQrCode");
|
|
qrcode.makeCode("bitcoin:"+keys.address);
|
|
|
|
$("#walletKeys .privkey").val(keys.wif);
|
|
$("#walletKeys .pubkey").val(keys.pubkey);
|
|
$("#walletKeys .privkeyaes").val(CryptoJS.AES.encrypt(keys.wif, pass));
|
|
|
|
$("#openLogin").hide();
|
|
$("#openWallet").removeClass("hidden").show();
|
|
|
|
walletBalance();
|
|
checkBalanceLoop();
|
|
} 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','https://btc.blockr.io/address/info/');
|
|
|
|
$("#walletQrCode").html("");
|
|
var qrcode = new QRCode("walletQrCode");
|
|
qrcode.makeCode("bitcoin:");
|
|
|
|
$("#walletKeys .privkey").val("");
|
|
$("#walletKeys .pubkey").val("");
|
|
|
|
});
|
|
|
|
$("#walletShowKeys").click(function(){
|
|
$("#walletKeys").removeClass("hidden");
|
|
$("#walletSpend").removeClass("hidden").addClass("hidden");
|
|
});
|
|
|
|
$("#walletBalance").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);
|
|
total += amount.val()*1;
|
|
tx.addoutput(addr.val(), amount.val()*1);
|
|
});
|
|
|
|
thisbtn.attr('disabled',true);
|
|
|
|
tx.addUnspent($("#walletAddress").html(), function(data){
|
|
var dvalue = data.value/100000000
|
|
if(dvalue>=total){
|
|
var change = dvalue-total;
|
|
if(change>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: "+$(data).find("txid").text());
|
|
} else {
|
|
$("#walletSendConfirmStatus").removeClass('hidden').addClass('alert-danger').html(unescape($(data).find("response").text()).replace(/\+/g,' '));
|
|
thisbtn.attr('disabled',false);
|
|
}
|
|
|
|
// update wallet balance
|
|
walletBalance();
|
|
|
|
}, signed);
|
|
} else {
|
|
$("#walletSendConfirmStatus").removeClass("hidden").addClass('alert-danger').html("You have a confirmed balance of "+data.value+" BTC unable to send "+total+" BTC").fadeOut().fadeIn();
|
|
thisbtn.attr('disabled',false);
|
|
}
|
|
|
|
$("#walletLoader").addClass("hidden");
|
|
});
|
|
});
|
|
|
|
$("#walletSendBtn").click(function(){
|
|
|
|
$("#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(){
|
|
$("#walletSpend").removeClass("hidden");
|
|
$("#walletKeys").removeClass("hidden").addClass("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(){
|
|
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");
|
|
});
|
|
}
|
|
|
|
function checkBalanceLoop(){
|
|
setTimeout(function(){
|
|
walletBalance();
|
|
checkBalanceLoop();
|
|
},45000);
|
|
}
|
|
|
|
/* 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())+'');
|
|
|
|
});
|
|
|
|
$("#newBrainwallet").click(function(){
|
|
if($(this).is(":checked")){
|
|
$("#brainwallet").removeClass("hidden");
|
|
} else {
|
|
$("#brainwallet").addClass("hidden");
|
|
}
|
|
});
|
|
|
|
$("#encryptKey").click(function(){
|
|
if($(this).is(":checked")){
|
|
$("#aes256passform").removeClass("hidden");
|
|
} else {
|
|
$("#aes256wifkey, #aes256passform, #aes256passStatus").addClass("hidden");
|
|
}
|
|
});
|
|
|
|
/* 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);
|
|
$("#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> 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",
|
|
});
|
|
|
|
$("#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 date = $('#timeLockedDateTimePicker').data("DateTimePicker").date();
|
|
if(!date || !date.isValid()) {
|
|
$('#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(), date.unix());
|
|
$("#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();
|
|
|
|
$("#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")){
|
|
tx.addinput($(".txId",o).val(), $(".txIdN",o).val(), $(".txIdScript",o).val());
|
|
} 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)) && $(".amount",o).val()!=""){ // address
|
|
tx.addoutput(a, $(".amount",o).val());
|
|
} else if (((a!="") && ad.version === 42) && $(".amount",o).val()!=""){ // stealth address
|
|
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
|
|
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());
|
|
|
|
$("#transactionCreate").removeClass("hidden");
|
|
|
|
if($("#transactionFee").val()>=0.01){
|
|
$("#modalWarningFeeAmount").html($("#transactionFee").val());
|
|
$("#modalWarningFee").modal("show");
|
|
}
|
|
} else {
|
|
$("#transactionCreateStatus").removeClass("hidden").html("One or more input or output is invalid").fadeOut().fadeIn();
|
|
}
|
|
});
|
|
|
|
$(".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();
|
|
});
|
|
|
|
/* 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);
|
|
window.stream.stop();
|
|
}
|
|
|
|
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');
|
|
videoElement.src = window.URL.createObjectURL(stream);
|
|
videoElement.play();
|
|
}, function(error){ });
|
|
|
|
QCodeDecoder().decodeFromCamera(document.getElementById('videoReader'), function(er,data){
|
|
if(!er){
|
|
var match = data.match(/^bitcoin\:([13][a-z0-9]{26,33})/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 the multisig address!');
|
|
return false;
|
|
}
|
|
|
|
if(redeem.from=='other'){
|
|
$("#redeemFromStatus").removeClass('hidden').html('<span class="glyphicon glyphicon-exclamation-sign"></span> The address or multisig 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');
|
|
if(host=='blockr.io_bitcoinmainnet'){
|
|
listUnspentBlockrio_BitcoinMainnet(redeem);
|
|
} else if(host=='chain.so_litecoin'){
|
|
listUnspentChainso_Litecoin(redeem);
|
|
} else {
|
|
listUnspentDefault(redeem);
|
|
}
|
|
|
|
if($("#redeemFromStatus").hasClass("hidden")) {
|
|
// An ethical dilemma: Should we automatically set nLockTime?
|
|
if(redeem.from == 'redeemScript' && redeem.decodedRs.type == "hodl__") {
|
|
$("#nLockTime").val(redeem.decodedRs.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.isMultisig = false;
|
|
} else if (decode.version == coinjs.priv){ // wif key
|
|
var a = coinjs.wif2address(string);
|
|
r.addr = a['address'];
|
|
r.from = 'wif';
|
|
r.isMultisig = false;
|
|
} else if (decode.version == coinjs.multisig){ // mulisig address
|
|
r.addr = '';
|
|
r.from = 'multisigAddress';
|
|
r.isMultisig = false;
|
|
} else {
|
|
var script = coinjs.script();
|
|
var decodeRs = script.decodeRedeemScript(string);
|
|
if(decodeRs){ // redeem script
|
|
r.addr = decodeRs['address'];
|
|
r.from = 'redeemScript';
|
|
r.decodedRs = decodeRs;
|
|
r.isMultisig = true; // not quite, may be hodl
|
|
} else { // something else
|
|
r.addr = '';
|
|
r.from = 'other';
|
|
r.isMultisig = 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);
|
|
$("#inputs .txIdScript:last").val(script);
|
|
}
|
|
}
|
|
|
|
/* default function to retreive unspent outputs*/
|
|
function listUnspentDefault(redeem){
|
|
var tx = coinjs.transaction();
|
|
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="https://btc.blockr.io/address/info/'+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.isMultisig==true) ? $("#redeemFrom").val() : $(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 blockrio for mainnet */
|
|
function listUnspentBlockrio_BitcoinMainnet(redeem){
|
|
$.ajax ({
|
|
type: "POST",
|
|
url: "https://btc.blockr.io/api/v1/address/unspent/"+redeem.addr+"?unconfirmed=1",
|
|
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="https://btc.blockr.io/address/info/'+redeem.addr+'" target="_blank">'+redeem.addr+'</a>');
|
|
for(var i in data.data.unspent){
|
|
var o = data.data.unspent[i];
|
|
var tx = o.tx;
|
|
var n = o.n;
|
|
var script = (redeem.isMultisig==true) ? $("#redeemFrom").val() : o.script;
|
|
var amount = o.amount;
|
|
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 blockrio for litecoin */
|
|
function listUnspentChainso_Litecoin(redeem){
|
|
$.ajax ({
|
|
type: "GET",
|
|
url: "https://chain.so/api/v2/get_tx_unspent/ltc/"+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) {
|
|
console.log(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="https://btc.blockr.io/address/info/'+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("")+'';
|
|
var n = o.output_no;
|
|
var script = (redeem.isMultisig==true) ? $("#redeemFrom").val() : 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").hasClass('hidden')){
|
|
$("#glyphcollapse").removeClass('glyphicon-collapse-down').addClass('glyphicon-collapse-up');
|
|
$("#optionsAdvanced").removeClass("hidden");
|
|
} else {
|
|
$("#glyphcollapse").removeClass('glyphicon-collapse-up').addClass('glyphicon-collapse-down');
|
|
$("#optionsAdvanced").addClass("hidden");
|
|
}
|
|
});
|
|
|
|
/* broadcast a transaction */
|
|
|
|
$("#rawSubmitBtn").click(function(){
|
|
rawSubmitDefault(this);
|
|
});
|
|
|
|
// broadcast transaction vai 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');
|
|
$("#rawTransactionStatus").html('txid: '+$(data).find("txid").text());
|
|
} 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 blockr.io (mainnet)
|
|
function rawSubmitBlockrio_BitcoinMainnet(thisbtn){
|
|
$(thisbtn).val('Please wait, loading...').attr('disabled',true);
|
|
$.ajax ({
|
|
type: "POST",
|
|
url: "https://btc.blockr.io/api/v1/tx/push",
|
|
data: {"hex":$("#rawTransaction").val()},
|
|
dataType: "json",
|
|
error: function(data) {
|
|
var obj = $.parseJSON(data.responseText);
|
|
var r = ' ';
|
|
r += (obj.data) ? obj.data : '';
|
|
r += (obj.message) ? ' '+obj.message : '';
|
|
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) {
|
|
var obj = $.parseJSON(data.responseText);
|
|
if((obj.status && obj.data) && obj.status=='success'){
|
|
$("#rawTransactionStatus").addClass('alert-success').removeClass('alert-danger').removeClass("hidden").html(' Txid: '+obj.data);
|
|
} 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 blockr.io for litecoin
|
|
function rawSubmitBlockrio_litecoin(thisbtn){
|
|
$(thisbtn).val('Please wait, loading...').attr('disabled',true);
|
|
$.ajax ({
|
|
type: "POST",
|
|
url: "https://ltc.blockr.io/api/v1/tx/push",
|
|
data: {"hex":$("#rawTransaction").val()},
|
|
dataType: "json",
|
|
error: function(data) {
|
|
var obj = $.parseJSON(data.responseText);
|
|
var r = ' ';
|
|
r += (obj.data) ? obj.data : '';
|
|
r += (obj.message) ? ' '+obj.message : '';
|
|
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) {
|
|
var obj = $.parseJSON(data.responseText);
|
|
if((obj.status && obj.data) && obj.status=='success'){
|
|
$("#rawTransactionStatus").addClass('alert-success').removeClass('alert-danger').removeClass("hidden").html(' Txid: '+obj.data);
|
|
} 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');
|
|
|
|
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 == "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());
|
|
// console.log(decode);
|
|
$("#verifyTransactionData .transactionVersion").html(decode['version']);
|
|
$("#verifyTransactionData .transactionSize").html(decode.size()+' <i>bytes</i>');
|
|
$("#verifyTransactionData .transactionLockTime").html(decode['lock_time']);
|
|
$("#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')?'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>';
|
|
});
|
|
|
|
$(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 {
|
|
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 .address").val(coinjs.pubkey2address(pubkey));
|
|
$("#verifyPubKey").removeClass("hidden");
|
|
$(".verifyLink").attr('href','?verify='+$("#verifyScript").val());
|
|
return true;
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
function decodeHDaddress(){
|
|
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++){
|
|
var derived = hd.derive(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");
|
|
}
|
|
|
|
|
|
/* 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());
|
|
$("#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');
|
|
}
|
|
});
|
|
|
|
|
|
/* 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;
|
|
}
|
|
|
|
$("#newKeysBtn, #newHDKeysBtn").click();
|
|
|
|
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";
|
|
}
|
|
|
|
$(".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){
|
|
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();
|
|
|
|
$("#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();
|
|
$("#rawSubmitBtn").unbind("");
|
|
if(host=="blockr.io_litecoin"){
|
|
$("#rawSubmitBtn").click(function(){
|
|
rawSubmitBlockrio_litecoin(this)
|
|
});
|
|
} else if(host=="blockr.io_bitcoinmainnet"){
|
|
$("#rawSubmitBtn").click(function(){
|
|
rawSubmitBlockrio_BitcoinMainnet(this);
|
|
});
|
|
} else {
|
|
$("#rawSubmitBtn").click(function(){
|
|
rawSubmitDefault(this); // revert to default
|
|
});
|
|
}
|
|
}
|
|
|
|
function configureGetUnspentTx(){
|
|
$("#redeemFromBtn").attr('rel',$("#coinjs_utxo option:selected").val());
|
|
}
|
|
|
|
/* 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;
|
|
};
|
|
});
|