Source: keypad.js

/**
 * @see https://github.com/v-garcia/boursorama-auto-login/blob/master/bourso_autologin.user.js
 */
const compareImages = require('resemblejs/compareImages')

const DIGIT_B64 = {
  0: '',
  1: '',
  2: '',
  3: '',
  4: '',
  5: '',
  6: '',
  7: '',
  8: '',
  9: ''
}

function getPassword($, password) {
  // Password input
  const $pwdInput = $('ul.password-input')

  // Get b64 image from a button HTMLElement, by extraction for style attribute
  const buttonToB64Pic = button => {
    const $button = $(button)

    // Get the style attribute
    const btnStyle = $button.attr('style')

    // Parse style attribute for b64 picture
    const [b64pic] = /data:image\/([a-zA-Z]*);base64,([^)]*)/.exec(btnStyle)

    return b64pic
  }

  const digitImgs = $pwdInput
    .find('[data-matrix-key]')
    .map((x, y) => buttonToB64Pic(y))
    .get()

  const imageCompareWrapperOptions = {
    ignore: 'antialiasing'
  }

  // Wrap resemble.js with Promise because this lib use old fashion callbacks
  const imageCompareWrapper = async (p1b64, p2b64) => {
    const r = await compareImages(p1b64, p2b64, imageCompareWrapperOptions)
    return { ...r, p1: p1b64, p2: p2b64 }
  }

  // Get a digit and return the HTMLElement associated by comparing resemblance with reference base 64 pictures
  const digitToButton = digit => {
    const referencePicture = DIGIT_B64[digit]
    const comparisons = digitImgs.map(elem =>
      imageCompareWrapper(referencePicture, elem)
    )

    return Promise.all(comparisons).then(values => {
      // Order images by resemblance
      values.sort((a, b) => a.misMatchPercentage - b.misMatchPercentage)

      // Pick the button whith the most resembling pic in the DOM
      const targetPic = $('[data-matrix-key]')
        .toArray()
        .find(el => buttonToB64Pic(el) === values[0].p2)

      return $(targetPic).data('matrix-key')
    })
  }

  // Get DOM buttons that match digit images (Array<Promise<HtmlElement>>)
  return [...password].map(digitToButton)
}

module.exports = {
  getPassword
}