devise-otp-rails5 0.2.4

Sign up to get free protection for your applications and to get access to all the features.
Files changed (89) hide show
  1. checksums.yaml +7 -0
  2. data/.gitignore +42 -0
  3. data/.travis.yml +12 -0
  4. data/Gemfile +25 -0
  5. data/LICENSE.txt +22 -0
  6. data/README.md +140 -0
  7. data/Rakefile +42 -0
  8. data/app/assets/javascripts/devise-otp.js +1 -0
  9. data/app/assets/javascripts/qrcode.js +609 -0
  10. data/app/controllers/devise_otp/credentials_controller.rb +106 -0
  11. data/app/controllers/devise_otp/tokens_controller.rb +111 -0
  12. data/app/views/devise_otp/credentials/refresh.html.erb +20 -0
  13. data/app/views/devise_otp/credentials/show.html.erb +23 -0
  14. data/app/views/devise_otp/tokens/_token_secret.html.erb +19 -0
  15. data/app/views/devise_otp/tokens/_trusted_devices.html.erb +10 -0
  16. data/app/views/devise_otp/tokens/recovery.html.erb +21 -0
  17. data/app/views/devise_otp/tokens/recovery_codes.text.erb +3 -0
  18. data/app/views/devise_otp/tokens/show.html.erb +19 -0
  19. data/config/locales/en.yml +66 -0
  20. data/devise-otp.gemspec +25 -0
  21. data/lib/devise-otp.rb +83 -0
  22. data/lib/devise-otp/version.rb +5 -0
  23. data/lib/devise_otp_authenticatable/controllers/helpers.rb +168 -0
  24. data/lib/devise_otp_authenticatable/controllers/url_helpers.rb +33 -0
  25. data/lib/devise_otp_authenticatable/engine.rb +23 -0
  26. data/lib/devise_otp_authenticatable/hooks.rb +13 -0
  27. data/lib/devise_otp_authenticatable/hooks/sessions.rb +59 -0
  28. data/lib/devise_otp_authenticatable/mapping.rb +19 -0
  29. data/lib/devise_otp_authenticatable/models/otp_authenticatable.rb +137 -0
  30. data/lib/devise_otp_authenticatable/routes.rb +32 -0
  31. data/lib/generators/active_record/devise_otp_generator.rb +13 -0
  32. data/lib/generators/active_record/templates/migration.rb +27 -0
  33. data/lib/generators/devise_otp/devise_otp_generator.rb +17 -0
  34. data/lib/generators/devise_otp/install_generator.rb +53 -0
  35. data/lib/generators/devise_otp/views_generator.rb +19 -0
  36. data/test/dummy/README.rdoc +261 -0
  37. data/test/dummy/Rakefile +7 -0
  38. data/test/dummy/app/assets/javascripts/application.js +13 -0
  39. data/test/dummy/app/assets/stylesheets/application.css +13 -0
  40. data/test/dummy/app/controllers/application_controller.rb +4 -0
  41. data/test/dummy/app/controllers/posts_controller.rb +83 -0
  42. data/test/dummy/app/helpers/application_helper.rb +2 -0
  43. data/test/dummy/app/helpers/posts_helper.rb +2 -0
  44. data/test/dummy/app/mailers/.gitkeep +0 -0
  45. data/test/dummy/app/models/post.rb +2 -0
  46. data/test/dummy/app/models/user.rb +20 -0
  47. data/test/dummy/app/views/layouts/application.html.erb +14 -0
  48. data/test/dummy/app/views/posts/_form.html.erb +25 -0
  49. data/test/dummy/app/views/posts/edit.html.erb +6 -0
  50. data/test/dummy/app/views/posts/index.html.erb +25 -0
  51. data/test/dummy/app/views/posts/new.html.erb +5 -0
  52. data/test/dummy/app/views/posts/show.html.erb +15 -0
  53. data/test/dummy/config.ru +4 -0
  54. data/test/dummy/config/application.rb +67 -0
  55. data/test/dummy/config/boot.rb +10 -0
  56. data/test/dummy/config/database.yml +25 -0
  57. data/test/dummy/config/environment.rb +5 -0
  58. data/test/dummy/config/environments/development.rb +30 -0
  59. data/test/dummy/config/environments/production.rb +69 -0
  60. data/test/dummy/config/environments/test.rb +36 -0
  61. data/test/dummy/config/initializers/backtrace_silencers.rb +7 -0
  62. data/test/dummy/config/initializers/devise.rb +253 -0
  63. data/test/dummy/config/initializers/inflections.rb +15 -0
  64. data/test/dummy/config/initializers/mime_types.rb +5 -0
  65. data/test/dummy/config/initializers/secret_token.rb +8 -0
  66. data/test/dummy/config/initializers/session_store.rb +8 -0
  67. data/test/dummy/config/initializers/wrap_parameters.rb +14 -0
  68. data/test/dummy/config/locales/en.yml +5 -0
  69. data/test/dummy/config/routes.rb +6 -0
  70. data/test/dummy/db/migrate/20130125101430_create_users.rb +9 -0
  71. data/test/dummy/db/migrate/20130131092406_add_devise_to_users.rb +53 -0
  72. data/test/dummy/db/migrate/20130131142320_create_posts.rb +10 -0
  73. data/test/dummy/db/migrate/20130131160351_devise_otp_add_to_users.rb +28 -0
  74. data/test/dummy/lib/assets/.gitkeep +0 -0
  75. data/test/dummy/public/404.html +26 -0
  76. data/test/dummy/public/422.html +26 -0
  77. data/test/dummy/public/500.html +25 -0
  78. data/test/dummy/public/favicon.ico +0 -0
  79. data/test/dummy/script/rails +6 -0
  80. data/test/integration/persistence_test.rb +65 -0
  81. data/test/integration/refresh_test.rb +106 -0
  82. data/test/integration/sign_in_test.rb +87 -0
  83. data/test/integration/token_test.rb +34 -0
  84. data/test/integration_tests_helper.rb +66 -0
  85. data/test/model_tests_helper.rb +22 -0
  86. data/test/models/otp_authenticatable_test.rb +122 -0
  87. data/test/orm/active_record.rb +4 -0
  88. data/test/test_helper.rb +22 -0
  89. metadata +253 -0
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: b432f81632e695540637cfe195d2749f1f1bfc48
4
+ data.tar.gz: 2a8266cabe0c8e501a521b7b5e17ad03c1bab9b7
5
+ SHA512:
6
+ metadata.gz: 5d835e0a85746b177efa6998314518317551319e1953284b8ce453b1ea673be9c687dd3ea0b7de450a152739929c5a256a5b324bb25c68b90b1fa9ae2583c714
7
+ data.tar.gz: 25a0643d9f1c6d8d906aa5ae285bfd8e2aada608cf40065ddd7c64ad1cb4cfdf1e174de2dc665873a6a864a630df922a6a93f7ddaf7e5da87936d953d476ef0e
@@ -0,0 +1,42 @@
1
+ ## RubyMine
2
+ .idea
3
+
4
+ ## MAC OS
5
+ .DS_Store
6
+
7
+ ## TEXTMATE
8
+ *.tmproj
9
+ tmtags
10
+
11
+ ## EMACS
12
+ *~
13
+ \#*
14
+ .\#*
15
+
16
+ ## VIM
17
+ *.swp
18
+
19
+ *.gem
20
+ *.rbc
21
+ .bundle
22
+ .config
23
+ .yardoc
24
+
25
+ ## PROJECT::GENERAL
26
+ _yardoc
27
+ doc/
28
+ coverage
29
+ rdoc
30
+ pkg
31
+ spec/reports
32
+ lib/bundler/man
33
+
34
+ ## PROJECT::SPECIFIC
35
+ test/dummy/log/**
36
+ test/dummy/tmp/**
37
+ test/dummy/db/*.sqlite3
38
+
39
+ Gemfile.lock
40
+
41
+ # Generated test files
42
+ tmp/*
@@ -0,0 +1,12 @@
1
+ language: ruby
2
+ rvm:
3
+ - 1.9.3
4
+ - 2.0.0
5
+ - 2.1.2
6
+ - rbx-19mode
7
+ - rbx-20mode
8
+ script: rake test
9
+ env:
10
+ - DEVISE_ORM=active_record
11
+ matrix:
12
+ allow_failures:
data/Gemfile ADDED
@@ -0,0 +1,25 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in devise-otp.gemspec
4
+ gemspec
5
+
6
+ gem "rdoc"
7
+
8
+ group :test do
9
+ platforms :jruby do
10
+ gem 'activerecord-jdbcsqlite3-adapter'
11
+ end
12
+
13
+ platforms :ruby do
14
+ gem "sqlite3"
15
+ end
16
+
17
+ gem "rails", "~> 5.0.0"
18
+
19
+ gem "capybara"
20
+ gem 'shoulda'
21
+ gem 'selenium-webdriver'
22
+
23
+ gem 'minitest-reporters', '>= 0.5.0'
24
+
25
+ end
@@ -0,0 +1,22 @@
1
+ Copyright (c) 2013 Lele Forzani
2
+
3
+ MIT License
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining
6
+ a copy of this software and associated documentation files (the
7
+ "Software"), to deal in the Software without restriction, including
8
+ without limitation the rights to use, copy, modify, merge, publish,
9
+ distribute, sublicense, and/or sell copies of the Software, and to
10
+ permit persons to whom the Software is furnished to do so, subject to
11
+ the following conditions:
12
+
13
+ The above copyright notice and this permission notice shall be
14
+ included in all copies or substantial portions of the Software.
15
+
16
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
20
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
21
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
22
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
@@ -0,0 +1,140 @@
1
+ # Devise::Otp
2
+ [![Build Status](https://travis-ci.org/wmlele/devise-otp.png?branch=master)](https://travis-ci.org/wmlele/devise-otp)
3
+
4
+ Devise OTP implements two-factors authentication for Devise, using an rfc6238 compatible Time-Based One-Time Password Algorithm.
5
+ It uses the [rotp library](https://github.com/mdp/rotp) for generation and verification of codes.
6
+
7
+ **If you are upgrading from version 0.1.x, you will need to regenerate your views.**
8
+
9
+ It currently has the following features:
10
+
11
+ * Url based provisioning of token devices, compatible with **Google Authenticator**.
12
+ * Browsers can be set as 'trusted' for a limited time. During that time no OTP challenge is asked again when logging from that browser (but normal login will).
13
+ * Two factors authentication can be **optional** at user discretion, **recommended** (it nags the user on every sign-in) or **mandatory** (users must enroll OTP after signing-in next time, before they can navigate the site). The settings is global, or per-user. ( **incomplete**, see below)
14
+ * Optionally, users can obtain a list of HOTP recovery tokens to be used for emergency log-in in case the token device is lost or unavailable.
15
+
16
+ Compatible token devices are:
17
+
18
+ * [Google Authenticator](https://code.google.com/p/google-authenticator/)
19
+ * [FreeOTP](https://fedorahosted.org/freeotp/)
20
+
21
+ ## Quick overview of Two Factors Authentication, using OTPs.
22
+
23
+ * A shared secret is generated on the server, and stored both on the token device (ie: the phone) and the server itself.
24
+ * The secret is used to generate short numerical tokens that are either time or sequence based.
25
+ * Tokens can be generated on a phone without internet connectivity.
26
+ * The token provides an additional layer of security against password theft.
27
+ * OTP's should always be used as a second factor of authentication(if your phone is lost, you account is still secured with a password)
28
+ * Google Authenticator allows you to store multiple OTP secrets and provision those using a QR Code
29
+
30
+ Although there's an adjustable drift window, it is important that both the server and the token device (phone) have their clocks set (eg: using NTP).
31
+
32
+
33
+ ## Installation
34
+
35
+ Add this line to your application's Gemfile:
36
+
37
+ gem 'devise'
38
+ gem 'devise-otp'
39
+
40
+ And then execute:
41
+
42
+ $ bundle
43
+
44
+ Or install it yourself as:
45
+
46
+ $ gem install devise-otp
47
+
48
+
49
+ ### Devise Installation
50
+
51
+ To setup Devise, you need to do the following (but refer to https://github.com/plataformatec/devise for more information)
52
+
53
+ Install Devise:
54
+
55
+ rails g devise:install
56
+
57
+ Setup the User or Admin model
58
+
59
+ rails g devise MODEL
60
+
61
+ Configure your app for authorisation, edit your Controller and add this before_action:
62
+
63
+ before_action :authenticate_user!
64
+
65
+ Make sure your "root" route is configured in config/routes.rb
66
+
67
+ ### Automatic Installation
68
+
69
+ Run the following generator to add the necessary configuration options to Devise's config file:
70
+
71
+ rails g devise_otp:install
72
+
73
+ After you've created your Devise user models (which is usually done with a "rails g devise MODEL"), set up your Devise OTP additions:
74
+
75
+ rails g devise_otp MODEL
76
+
77
+ Don't forget to migrate:
78
+
79
+ rake db:migrate
80
+
81
+ Add the gem's javascript to you application.js
82
+
83
+ //= require devise-otp
84
+
85
+
86
+
87
+ ### Custom Views
88
+
89
+ If you want to customise your views (which you likely will want to), you can use the generator:
90
+
91
+ rails g devise_otp:views
92
+
93
+ ### I18n
94
+
95
+ The install generator also installs an english copy of a Devise OTP i18n file. This can be modified (or used to create other language versions) and is located at: _config/locales/devise.otp.en.yml_
96
+
97
+
98
+ ## Usage
99
+
100
+ With this extension enabled, the following is expected behaviour:
101
+
102
+ * Users may go to _/MODEL/otp/token_ and enable their OTP state, they might be asked to provide their password again (and OTP token, if it's enabled)
103
+ * Once enabled they're shown an alphanumeric code (for manual provisioning) and a QR code, for automatic provisioning of their authetication device (for instance, Google Authenticator)
104
+ * If config.otp_mandatory or model_instance.otp_mandatory, users will be required to enable, and provision, next time they successfully sign-in.
105
+
106
+
107
+ ### Configuration Options
108
+
109
+ The install generator adds some options to the end of your Devise config file (config/initializers/devise.rb)
110
+
111
+ * `config.otp_mandatory` - OTP is mandatory, users are going to be asked to enroll the next time they sign in, before they can successfully complete the session establishment.
112
+ * `config.otp_authentication_timeout` - how long the user has to authenticate with their token. (defaults to `3.minutes`)
113
+ * `config.otp_drift_window` - a window which provides allowance for drift between a user's token device clock (and therefore their OTP tokens) and the authentication server's clock. Expressed in minutes centered at the current time. (default: `3`)
114
+ * `config.otp_credentials_refresh` - Users that have logged in longer than this time ago, are going to be asked their password (and an OTP challenge, if enabled) before they can see or change their otp informations. (defaults to `15.minutes`)
115
+ * `config.otp_recovery_tokens` - Whether the users are given a list of one-time recovery tokens, for emergency access (default: `10`, set to `false` to disable)
116
+ * `config.otp_trust_persistence` - The user is allowed to set his browser as "trusted", no more OTP challenges will be asked for that browser, for a limited time. (default: `1.month`, set to false to disable setting the browser as trusted)
117
+ * `config.otp_issuer` - The name of the token issuer, to be added to the provisioning url. Display will vary based on token application. (defaults to the Rails application class)
118
+
119
+ ## Todo
120
+
121
+ * 2D barcodes for provisioning are currently produced with the google charts api. You can, of course, use your own source in the template, but I am looking for a solution with no external dependencies (feedback welcome).
122
+ * **recommended** mode (nag the user each time) is not fully implemented. Right now you can make 2FA mandatory, or leave it to the user.
123
+
124
+
125
+ ## Contributing
126
+
127
+ 1. Fork it
128
+ 2. Create your feature branch (`git checkout -b my-new-feature`)
129
+ 3. Commit your changes (`git commit -am 'Add some feature'`)
130
+ 4. Push to the branch (`git push origin my-new-feature`)
131
+ 5. Create new Pull Request
132
+
133
+ ## Thanks
134
+
135
+ I started this extension by forking [devise_google_authenticator](https://github.com/AsteriskLabs/devise_google_authenticator), and this project still contains some chunk of code from it, esp. in the tests and generators.
136
+ At some point, my design goals were significantly diverging, so I refactored most of its code. Still, I want to thank the original author for his relevant contribution.
137
+
138
+ ## License
139
+
140
+ MIT Licensed
@@ -0,0 +1,42 @@
1
+ #!/usr/bin/env rake
2
+ begin
3
+ require 'bundler/setup'
4
+ rescue LoadError
5
+ puts 'You must `gem install bundler` and `bundle install` to run rake tasks'
6
+ end
7
+ begin
8
+ require 'rdoc/task'
9
+ rescue LoadError
10
+ require 'rdoc/rdoc'
11
+ require 'rake/rdoctask'
12
+ RDoc::Task = Rake::RDocTask
13
+ end
14
+
15
+ RDoc::Task.new(:rdoc) do |rdoc|
16
+ rdoc.rdoc_dir = 'rdoc'
17
+ rdoc.title = 'Foobar'
18
+ rdoc.options << '--line-numbers'
19
+ rdoc.rdoc_files.include('README.rdoc')
20
+ rdoc.rdoc_files.include('lib/**/*.rb')
21
+ end
22
+
23
+
24
+ Bundler::GemHelper.install_tasks
25
+
26
+ require 'rake/testtask'
27
+ Rake::TestTask.new(:test) do |test|
28
+ test.libs << 'lib' << 'test'
29
+ test.pattern = 'test/**/*_test.rb'
30
+ test.verbose = true
31
+ end
32
+
33
+ desc 'Run Devise tests for all ORMs.'
34
+ task :tests do
35
+ Dir[File.join(File.dirname(__FILE__), 'test', 'orm', '*.rb')].each do |file|
36
+ orm = File.basename(file).split(".").first
37
+ system "rake test DEVISE_ORM=#{orm}"
38
+ end
39
+ end
40
+
41
+ desc 'Default: run tests for all ORMs.'
42
+ task :default => :tests
@@ -0,0 +1 @@
1
+ //= require_tree .
@@ -0,0 +1,609 @@
1
+ /**
2
+ * @fileoverview
3
+ * - Using the 'QRCode for Javascript library'
4
+ * - Fixed dataset of 'QRCode for Javascript library' for support full-spec.
5
+ * - this library has no dependencies.
6
+ *
7
+ * @author davidshimjs
8
+ * @see <a href="http://www.d-project.com/" target="_blank">http://www.d-project.com/</a>
9
+ * @see <a href="http://jeromeetienne.github.com/jquery-qrcode/" target="_blank">http://jeromeetienne.github.com/jquery-qrcode/</a>
10
+ */
11
+ var QRCode;
12
+
13
+ (function () {
14
+ //---------------------------------------------------------------------
15
+ // QRCode for JavaScript
16
+ //
17
+ // Copyright (c) 2009 Kazuhiko Arase
18
+ //
19
+ // URL: http://www.d-project.com/
20
+ //
21
+ // Licensed under the MIT license:
22
+ // http://www.opensource.org/licenses/mit-license.php
23
+ //
24
+ // The word "QR Code" is registered trademark of
25
+ // DENSO WAVE INCORPORATED
26
+ // http://www.denso-wave.com/qrcode/faqpatent-e.html
27
+ //
28
+ //---------------------------------------------------------------------
29
+ function QR8bitByte(data) {
30
+ this.mode = QRMode.MODE_8BIT_BYTE;
31
+ this.data = data;
32
+ this.parsedData = [];
33
+
34
+ // Added to support UTF-8 Characters
35
+ for (var i = 0, l = this.data.length; i < l; i++) {
36
+ var byteArray = [];
37
+ var code = this.data.charCodeAt(i);
38
+
39
+ if (code > 0x10000) {
40
+ byteArray[0] = 0xF0 | ((code & 0x1C0000) >>> 18);
41
+ byteArray[1] = 0x80 | ((code & 0x3F000) >>> 12);
42
+ byteArray[2] = 0x80 | ((code & 0xFC0) >>> 6);
43
+ byteArray[3] = 0x80 | (code & 0x3F);
44
+ } else if (code > 0x800) {
45
+ byteArray[0] = 0xE0 | ((code & 0xF000) >>> 12);
46
+ byteArray[1] = 0x80 | ((code & 0xFC0) >>> 6);
47
+ byteArray[2] = 0x80 | (code & 0x3F);
48
+ } else if (code > 0x80) {
49
+ byteArray[0] = 0xC0 | ((code & 0x7C0) >>> 6);
50
+ byteArray[1] = 0x80 | (code & 0x3F);
51
+ } else {
52
+ byteArray[0] = code;
53
+ }
54
+
55
+ this.parsedData.push(byteArray);
56
+ }
57
+
58
+ this.parsedData = Array.prototype.concat.apply([], this.parsedData);
59
+
60
+ if (this.parsedData.length != this.data.length) {
61
+ this.parsedData.unshift(191);
62
+ this.parsedData.unshift(187);
63
+ this.parsedData.unshift(239);
64
+ }
65
+ }
66
+
67
+ QR8bitByte.prototype = {
68
+ getLength: function (buffer) {
69
+ return this.parsedData.length;
70
+ },
71
+ write: function (buffer) {
72
+ for (var i = 0, l = this.parsedData.length; i < l; i++) {
73
+ buffer.put(this.parsedData[i], 8);
74
+ }
75
+ }
76
+ };
77
+
78
+ function QRCodeModel(typeNumber, errorCorrectLevel) {
79
+ this.typeNumber = typeNumber;
80
+ this.errorCorrectLevel = errorCorrectLevel;
81
+ this.modules = null;
82
+ this.moduleCount = 0;
83
+ this.dataCache = null;
84
+ this.dataList = [];
85
+ }
86
+
87
+ QRCodeModel.prototype={addData:function(data){var newData=new QR8bitByte(data);this.dataList.push(newData);this.dataCache=null;},isDark:function(row,col){if(row<0||this.moduleCount<=row||col<0||this.moduleCount<=col){throw new Error(row+","+col);}
88
+ return this.modules[row][col];},getModuleCount:function(){return this.moduleCount;},make:function(){this.makeImpl(false,this.getBestMaskPattern());},makeImpl:function(test,maskPattern){this.moduleCount=this.typeNumber*4+17;this.modules=new Array(this.moduleCount);for(var row=0;row<this.moduleCount;row++){this.modules[row]=new Array(this.moduleCount);for(var col=0;col<this.moduleCount;col++){this.modules[row][col]=null;}}
89
+ this.setupPositionProbePattern(0,0);this.setupPositionProbePattern(this.moduleCount-7,0);this.setupPositionProbePattern(0,this.moduleCount-7);this.setupPositionAdjustPattern();this.setupTimingPattern();this.setupTypeInfo(test,maskPattern);if(this.typeNumber>=7){this.setupTypeNumber(test);}
90
+ if(this.dataCache==null){this.dataCache=QRCodeModel.createData(this.typeNumber,this.errorCorrectLevel,this.dataList);}
91
+ this.mapData(this.dataCache,maskPattern);},setupPositionProbePattern:function(row,col){for(var r=-1;r<=7;r++){if(row+r<=-1||this.moduleCount<=row+r)continue;for(var c=-1;c<=7;c++){if(col+c<=-1||this.moduleCount<=col+c)continue;if((0<=r&&r<=6&&(c==0||c==6))||(0<=c&&c<=6&&(r==0||r==6))||(2<=r&&r<=4&&2<=c&&c<=4)){this.modules[row+r][col+c]=true;}else{this.modules[row+r][col+c]=false;}}}},getBestMaskPattern:function(){var minLostPoint=0;var pattern=0;for(var i=0;i<8;i++){this.makeImpl(true,i);var lostPoint=QRUtil.getLostPoint(this);if(i==0||minLostPoint>lostPoint){minLostPoint=lostPoint;pattern=i;}}
92
+ return pattern;},createMovieClip:function(target_mc,instance_name,depth){var qr_mc=target_mc.createEmptyMovieClip(instance_name,depth);var cs=1;this.make();for(var row=0;row<this.modules.length;row++){var y=row*cs;for(var col=0;col<this.modules[row].length;col++){var x=col*cs;var dark=this.modules[row][col];if(dark){qr_mc.beginFill(0,100);qr_mc.moveTo(x,y);qr_mc.lineTo(x+cs,y);qr_mc.lineTo(x+cs,y+cs);qr_mc.lineTo(x,y+cs);qr_mc.endFill();}}}
93
+ return qr_mc;},setupTimingPattern:function(){for(var r=8;r<this.moduleCount-8;r++){if(this.modules[r][6]!=null){continue;}
94
+ this.modules[r][6]=(r%2==0);}
95
+ for(var c=8;c<this.moduleCount-8;c++){if(this.modules[6][c]!=null){continue;}
96
+ this.modules[6][c]=(c%2==0);}},setupPositionAdjustPattern:function(){var pos=QRUtil.getPatternPosition(this.typeNumber);for(var i=0;i<pos.length;i++){for(var j=0;j<pos.length;j++){var row=pos[i];var col=pos[j];if(this.modules[row][col]!=null){continue;}
97
+ for(var r=-2;r<=2;r++){for(var c=-2;c<=2;c++){if(r==-2||r==2||c==-2||c==2||(r==0&&c==0)){this.modules[row+r][col+c]=true;}else{this.modules[row+r][col+c]=false;}}}}}},setupTypeNumber:function(test){var bits=QRUtil.getBCHTypeNumber(this.typeNumber);for(var i=0;i<18;i++){var mod=(!test&&((bits>>i)&1)==1);this.modules[Math.floor(i/3)][i%3+this.moduleCount-8-3]=mod;}
98
+ for(var i=0;i<18;i++){var mod=(!test&&((bits>>i)&1)==1);this.modules[i%3+this.moduleCount-8-3][Math.floor(i/3)]=mod;}},setupTypeInfo:function(test,maskPattern){var data=(this.errorCorrectLevel<<3)|maskPattern;var bits=QRUtil.getBCHTypeInfo(data);for(var i=0;i<15;i++){var mod=(!test&&((bits>>i)&1)==1);if(i<6){this.modules[i][8]=mod;}else if(i<8){this.modules[i+1][8]=mod;}else{this.modules[this.moduleCount-15+i][8]=mod;}}
99
+ for(var i=0;i<15;i++){var mod=(!test&&((bits>>i)&1)==1);if(i<8){this.modules[8][this.moduleCount-i-1]=mod;}else if(i<9){this.modules[8][15-i-1+1]=mod;}else{this.modules[8][15-i-1]=mod;}}
100
+ this.modules[this.moduleCount-8][8]=(!test);},mapData:function(data,maskPattern){var inc=-1;var row=this.moduleCount-1;var bitIndex=7;var byteIndex=0;for(var col=this.moduleCount-1;col>0;col-=2){if(col==6)col--;while(true){for(var c=0;c<2;c++){if(this.modules[row][col-c]==null){var dark=false;if(byteIndex<data.length){dark=(((data[byteIndex]>>>bitIndex)&1)==1);}
101
+ var mask=QRUtil.getMask(maskPattern,row,col-c);if(mask){dark=!dark;}
102
+ this.modules[row][col-c]=dark;bitIndex--;if(bitIndex==-1){byteIndex++;bitIndex=7;}}}
103
+ row+=inc;if(row<0||this.moduleCount<=row){row-=inc;inc=-inc;break;}}}}};QRCodeModel.PAD0=0xEC;QRCodeModel.PAD1=0x11;QRCodeModel.createData=function(typeNumber,errorCorrectLevel,dataList){var rsBlocks=QRRSBlock.getRSBlocks(typeNumber,errorCorrectLevel);var buffer=new QRBitBuffer();for(var i=0;i<dataList.length;i++){var data=dataList[i];buffer.put(data.mode,4);buffer.put(data.getLength(),QRUtil.getLengthInBits(data.mode,typeNumber));data.write(buffer);}
104
+ var totalDataCount=0;for(var i=0;i<rsBlocks.length;i++){totalDataCount+=rsBlocks[i].dataCount;}
105
+ if(buffer.getLengthInBits()>totalDataCount*8){throw new Error("code length overflow. ("
106
+ +buffer.getLengthInBits()
107
+ +">"
108
+ +totalDataCount*8
109
+ +")");}
110
+ if(buffer.getLengthInBits()+4<=totalDataCount*8){buffer.put(0,4);}
111
+ while(buffer.getLengthInBits()%8!=0){buffer.putBit(false);}
112
+ while(true){if(buffer.getLengthInBits()>=totalDataCount*8){break;}
113
+ buffer.put(QRCodeModel.PAD0,8);if(buffer.getLengthInBits()>=totalDataCount*8){break;}
114
+ buffer.put(QRCodeModel.PAD1,8);}
115
+ return QRCodeModel.createBytes(buffer,rsBlocks);};QRCodeModel.createBytes=function(buffer,rsBlocks){var offset=0;var maxDcCount=0;var maxEcCount=0;var dcdata=new Array(rsBlocks.length);var ecdata=new Array(rsBlocks.length);for(var r=0;r<rsBlocks.length;r++){var dcCount=rsBlocks[r].dataCount;var ecCount=rsBlocks[r].totalCount-dcCount;maxDcCount=Math.max(maxDcCount,dcCount);maxEcCount=Math.max(maxEcCount,ecCount);dcdata[r]=new Array(dcCount);for(var i=0;i<dcdata[r].length;i++){dcdata[r][i]=0xff&buffer.buffer[i+offset];}
116
+ offset+=dcCount;var rsPoly=QRUtil.getErrorCorrectPolynomial(ecCount);var rawPoly=new QRPolynomial(dcdata[r],rsPoly.getLength()-1);var modPoly=rawPoly.mod(rsPoly);ecdata[r]=new Array(rsPoly.getLength()-1);for(var i=0;i<ecdata[r].length;i++){var modIndex=i+modPoly.getLength()-ecdata[r].length;ecdata[r][i]=(modIndex>=0)?modPoly.get(modIndex):0;}}
117
+ var totalCodeCount=0;for(var i=0;i<rsBlocks.length;i++){totalCodeCount+=rsBlocks[i].totalCount;}
118
+ var data=new Array(totalCodeCount);var index=0;for(var i=0;i<maxDcCount;i++){for(var r=0;r<rsBlocks.length;r++){if(i<dcdata[r].length){data[index++]=dcdata[r][i];}}}
119
+ for(var i=0;i<maxEcCount;i++){for(var r=0;r<rsBlocks.length;r++){if(i<ecdata[r].length){data[index++]=ecdata[r][i];}}}
120
+ return data;};var QRMode={MODE_NUMBER:1<<0,MODE_ALPHA_NUM:1<<1,MODE_8BIT_BYTE:1<<2,MODE_KANJI:1<<3};var QRErrorCorrectLevel={L:1,M:0,Q:3,H:2};var QRMaskPattern={PATTERN000:0,PATTERN001:1,PATTERN010:2,PATTERN011:3,PATTERN100:4,PATTERN101:5,PATTERN110:6,PATTERN111:7};var QRUtil={PATTERN_POSITION_TABLE:[[],[6,18],[6,22],[6,26],[6,30],[6,34],[6,22,38],[6,24,42],[6,26,46],[6,28,50],[6,30,54],[6,32,58],[6,34,62],[6,26,46,66],[6,26,48,70],[6,26,50,74],[6,30,54,78],[6,30,56,82],[6,30,58,86],[6,34,62,90],[6,28,50,72,94],[6,26,50,74,98],[6,30,54,78,102],[6,28,54,80,106],[6,32,58,84,110],[6,30,58,86,114],[6,34,62,90,118],[6,26,50,74,98,122],[6,30,54,78,102,126],[6,26,52,78,104,130],[6,30,56,82,108,134],[6,34,60,86,112,138],[6,30,58,86,114,142],[6,34,62,90,118,146],[6,30,54,78,102,126,150],[6,24,50,76,102,128,154],[6,28,54,80,106,132,158],[6,32,58,84,110,136,162],[6,26,54,82,110,138,166],[6,30,58,86,114,142,170]],G15:(1<<10)|(1<<8)|(1<<5)|(1<<4)|(1<<2)|(1<<1)|(1<<0),G18:(1<<12)|(1<<11)|(1<<10)|(1<<9)|(1<<8)|(1<<5)|(1<<2)|(1<<0),G15_MASK:(1<<14)|(1<<12)|(1<<10)|(1<<4)|(1<<1),getBCHTypeInfo:function(data){var d=data<<10;while(QRUtil.getBCHDigit(d)-QRUtil.getBCHDigit(QRUtil.G15)>=0){d^=(QRUtil.G15<<(QRUtil.getBCHDigit(d)-QRUtil.getBCHDigit(QRUtil.G15)));}
121
+ return((data<<10)|d)^QRUtil.G15_MASK;},getBCHTypeNumber:function(data){var d=data<<12;while(QRUtil.getBCHDigit(d)-QRUtil.getBCHDigit(QRUtil.G18)>=0){d^=(QRUtil.G18<<(QRUtil.getBCHDigit(d)-QRUtil.getBCHDigit(QRUtil.G18)));}
122
+ return(data<<12)|d;},getBCHDigit:function(data){var digit=0;while(data!=0){digit++;data>>>=1;}
123
+ return digit;},getPatternPosition:function(typeNumber){return QRUtil.PATTERN_POSITION_TABLE[typeNumber-1];},getMask:function(maskPattern,i,j){switch(maskPattern){case QRMaskPattern.PATTERN000:return(i+j)%2==0;case QRMaskPattern.PATTERN001:return i%2==0;case QRMaskPattern.PATTERN010:return j%3==0;case QRMaskPattern.PATTERN011:return(i+j)%3==0;case QRMaskPattern.PATTERN100:return(Math.floor(i/2)+Math.floor(j/3))%2==0;case QRMaskPattern.PATTERN101:return(i*j)%2+(i*j)%3==0;case QRMaskPattern.PATTERN110:return((i*j)%2+(i*j)%3)%2==0;case QRMaskPattern.PATTERN111:return((i*j)%3+(i+j)%2)%2==0;default:throw new Error("bad maskPattern:"+maskPattern);}},getErrorCorrectPolynomial:function(errorCorrectLength){var a=new QRPolynomial([1],0);for(var i=0;i<errorCorrectLength;i++){a=a.multiply(new QRPolynomial([1,QRMath.gexp(i)],0));}
124
+ return a;},getLengthInBits:function(mode,type){if(1<=type&&type<10){switch(mode){case QRMode.MODE_NUMBER:return 10;case QRMode.MODE_ALPHA_NUM:return 9;case QRMode.MODE_8BIT_BYTE:return 8;case QRMode.MODE_KANJI:return 8;default:throw new Error("mode:"+mode);}}else if(type<27){switch(mode){case QRMode.MODE_NUMBER:return 12;case QRMode.MODE_ALPHA_NUM:return 11;case QRMode.MODE_8BIT_BYTE:return 16;case QRMode.MODE_KANJI:return 10;default:throw new Error("mode:"+mode);}}else if(type<41){switch(mode){case QRMode.MODE_NUMBER:return 14;case QRMode.MODE_ALPHA_NUM:return 13;case QRMode.MODE_8BIT_BYTE:return 16;case QRMode.MODE_KANJI:return 12;default:throw new Error("mode:"+mode);}}else{throw new Error("type:"+type);}},getLostPoint:function(qrCode){var moduleCount=qrCode.getModuleCount();var lostPoint=0;for(var row=0;row<moduleCount;row++){for(var col=0;col<moduleCount;col++){var sameCount=0;var dark=qrCode.isDark(row,col);for(var r=-1;r<=1;r++){if(row+r<0||moduleCount<=row+r){continue;}
125
+ for(var c=-1;c<=1;c++){if(col+c<0||moduleCount<=col+c){continue;}
126
+ if(r==0&&c==0){continue;}
127
+ if(dark==qrCode.isDark(row+r,col+c)){sameCount++;}}}
128
+ if(sameCount>5){lostPoint+=(3+sameCount-5);}}}
129
+ for(var row=0;row<moduleCount-1;row++){for(var col=0;col<moduleCount-1;col++){var count=0;if(qrCode.isDark(row,col))count++;if(qrCode.isDark(row+1,col))count++;if(qrCode.isDark(row,col+1))count++;if(qrCode.isDark(row+1,col+1))count++;if(count==0||count==4){lostPoint+=3;}}}
130
+ for(var row=0;row<moduleCount;row++){for(var col=0;col<moduleCount-6;col++){if(qrCode.isDark(row,col)&&!qrCode.isDark(row,col+1)&&qrCode.isDark(row,col+2)&&qrCode.isDark(row,col+3)&&qrCode.isDark(row,col+4)&&!qrCode.isDark(row,col+5)&&qrCode.isDark(row,col+6)){lostPoint+=40;}}}
131
+ for(var col=0;col<moduleCount;col++){for(var row=0;row<moduleCount-6;row++){if(qrCode.isDark(row,col)&&!qrCode.isDark(row+1,col)&&qrCode.isDark(row+2,col)&&qrCode.isDark(row+3,col)&&qrCode.isDark(row+4,col)&&!qrCode.isDark(row+5,col)&&qrCode.isDark(row+6,col)){lostPoint+=40;}}}
132
+ var darkCount=0;for(var col=0;col<moduleCount;col++){for(var row=0;row<moduleCount;row++){if(qrCode.isDark(row,col)){darkCount++;}}}
133
+ var ratio=Math.abs(100*darkCount/moduleCount/moduleCount-50)/5;lostPoint+=ratio*10;return lostPoint;}};var QRMath={glog:function(n){if(n<1){throw new Error("glog("+n+")");}
134
+ return QRMath.LOG_TABLE[n];},gexp:function(n){while(n<0){n+=255;}
135
+ while(n>=256){n-=255;}
136
+ return QRMath.EXP_TABLE[n];},EXP_TABLE:new Array(256),LOG_TABLE:new Array(256)};for(var i=0;i<8;i++){QRMath.EXP_TABLE[i]=1<<i;}
137
+ for(var i=8;i<256;i++){QRMath.EXP_TABLE[i]=QRMath.EXP_TABLE[i-4]^QRMath.EXP_TABLE[i-5]^QRMath.EXP_TABLE[i-6]^QRMath.EXP_TABLE[i-8];}
138
+ for(var i=0;i<255;i++){QRMath.LOG_TABLE[QRMath.EXP_TABLE[i]]=i;}
139
+ function QRPolynomial(num,shift){if(num.length==undefined){throw new Error(num.length+"/"+shift);}
140
+ var offset=0;while(offset<num.length&&num[offset]==0){offset++;}
141
+ this.num=new Array(num.length-offset+shift);for(var i=0;i<num.length-offset;i++){this.num[i]=num[i+offset];}}
142
+ QRPolynomial.prototype={get:function(index){return this.num[index];},getLength:function(){return this.num.length;},multiply:function(e){var num=new Array(this.getLength()+e.getLength()-1);for(var i=0;i<this.getLength();i++){for(var j=0;j<e.getLength();j++){num[i+j]^=QRMath.gexp(QRMath.glog(this.get(i))+QRMath.glog(e.get(j)));}}
143
+ return new QRPolynomial(num,0);},mod:function(e){if(this.getLength()-e.getLength()<0){return this;}
144
+ var ratio=QRMath.glog(this.get(0))-QRMath.glog(e.get(0));var num=new Array(this.getLength());for(var i=0;i<this.getLength();i++){num[i]=this.get(i);}
145
+ for(var i=0;i<e.getLength();i++){num[i]^=QRMath.gexp(QRMath.glog(e.get(i))+ratio);}
146
+ return new QRPolynomial(num,0).mod(e);}};function QRRSBlock(totalCount,dataCount){this.totalCount=totalCount;this.dataCount=dataCount;}
147
+ QRRSBlock.RS_BLOCK_TABLE=[[1,26,19],[1,26,16],[1,26,13],[1,26,9],[1,44,34],[1,44,28],[1,44,22],[1,44,16],[1,70,55],[1,70,44],[2,35,17],[2,35,13],[1,100,80],[2,50,32],[2,50,24],[4,25,9],[1,134,108],[2,67,43],[2,33,15,2,34,16],[2,33,11,2,34,12],[2,86,68],[4,43,27],[4,43,19],[4,43,15],[2,98,78],[4,49,31],[2,32,14,4,33,15],[4,39,13,1,40,14],[2,121,97],[2,60,38,2,61,39],[4,40,18,2,41,19],[4,40,14,2,41,15],[2,146,116],[3,58,36,2,59,37],[4,36,16,4,37,17],[4,36,12,4,37,13],[2,86,68,2,87,69],[4,69,43,1,70,44],[6,43,19,2,44,20],[6,43,15,2,44,16],[4,101,81],[1,80,50,4,81,51],[4,50,22,4,51,23],[3,36,12,8,37,13],[2,116,92,2,117,93],[6,58,36,2,59,37],[4,46,20,6,47,21],[7,42,14,4,43,15],[4,133,107],[8,59,37,1,60,38],[8,44,20,4,45,21],[12,33,11,4,34,12],[3,145,115,1,146,116],[4,64,40,5,65,41],[11,36,16,5,37,17],[11,36,12,5,37,13],[5,109,87,1,110,88],[5,65,41,5,66,42],[5,54,24,7,55,25],[11,36,12],[5,122,98,1,123,99],[7,73,45,3,74,46],[15,43,19,2,44,20],[3,45,15,13,46,16],[1,135,107,5,136,108],[10,74,46,1,75,47],[1,50,22,15,51,23],[2,42,14,17,43,15],[5,150,120,1,151,121],[9,69,43,4,70,44],[17,50,22,1,51,23],[2,42,14,19,43,15],[3,141,113,4,142,114],[3,70,44,11,71,45],[17,47,21,4,48,22],[9,39,13,16,40,14],[3,135,107,5,136,108],[3,67,41,13,68,42],[15,54,24,5,55,25],[15,43,15,10,44,16],[4,144,116,4,145,117],[17,68,42],[17,50,22,6,51,23],[19,46,16,6,47,17],[2,139,111,7,140,112],[17,74,46],[7,54,24,16,55,25],[34,37,13],[4,151,121,5,152,122],[4,75,47,14,76,48],[11,54,24,14,55,25],[16,45,15,14,46,16],[6,147,117,4,148,118],[6,73,45,14,74,46],[11,54,24,16,55,25],[30,46,16,2,47,17],[8,132,106,4,133,107],[8,75,47,13,76,48],[7,54,24,22,55,25],[22,45,15,13,46,16],[10,142,114,2,143,115],[19,74,46,4,75,47],[28,50,22,6,51,23],[33,46,16,4,47,17],[8,152,122,4,153,123],[22,73,45,3,74,46],[8,53,23,26,54,24],[12,45,15,28,46,16],[3,147,117,10,148,118],[3,73,45,23,74,46],[4,54,24,31,55,25],[11,45,15,31,46,16],[7,146,116,7,147,117],[21,73,45,7,74,46],[1,53,23,37,54,24],[19,45,15,26,46,16],[5,145,115,10,146,116],[19,75,47,10,76,48],[15,54,24,25,55,25],[23,45,15,25,46,16],[13,145,115,3,146,116],[2,74,46,29,75,47],[42,54,24,1,55,25],[23,45,15,28,46,16],[17,145,115],[10,74,46,23,75,47],[10,54,24,35,55,25],[19,45,15,35,46,16],[17,145,115,1,146,116],[14,74,46,21,75,47],[29,54,24,19,55,25],[11,45,15,46,46,16],[13,145,115,6,146,116],[14,74,46,23,75,47],[44,54,24,7,55,25],[59,46,16,1,47,17],[12,151,121,7,152,122],[12,75,47,26,76,48],[39,54,24,14,55,25],[22,45,15,41,46,16],[6,151,121,14,152,122],[6,75,47,34,76,48],[46,54,24,10,55,25],[2,45,15,64,46,16],[17,152,122,4,153,123],[29,74,46,14,75,47],[49,54,24,10,55,25],[24,45,15,46,46,16],[4,152,122,18,153,123],[13,74,46,32,75,47],[48,54,24,14,55,25],[42,45,15,32,46,16],[20,147,117,4,148,118],[40,75,47,7,76,48],[43,54,24,22,55,25],[10,45,15,67,46,16],[19,148,118,6,149,119],[18,75,47,31,76,48],[34,54,24,34,55,25],[20,45,15,61,46,16]];QRRSBlock.getRSBlocks=function(typeNumber,errorCorrectLevel){var rsBlock=QRRSBlock.getRsBlockTable(typeNumber,errorCorrectLevel);if(rsBlock==undefined){throw new Error("bad rs block @ typeNumber:"+typeNumber+"/errorCorrectLevel:"+errorCorrectLevel);}
148
+ var length=rsBlock.length/3;var list=[];for(var i=0;i<length;i++){var count=rsBlock[i*3+0];var totalCount=rsBlock[i*3+1];var dataCount=rsBlock[i*3+2];for(var j=0;j<count;j++){list.push(new QRRSBlock(totalCount,dataCount));}}
149
+ return list;};QRRSBlock.getRsBlockTable=function(typeNumber,errorCorrectLevel){switch(errorCorrectLevel){case QRErrorCorrectLevel.L:return QRRSBlock.RS_BLOCK_TABLE[(typeNumber-1)*4+0];case QRErrorCorrectLevel.M:return QRRSBlock.RS_BLOCK_TABLE[(typeNumber-1)*4+1];case QRErrorCorrectLevel.Q:return QRRSBlock.RS_BLOCK_TABLE[(typeNumber-1)*4+2];case QRErrorCorrectLevel.H:return QRRSBlock.RS_BLOCK_TABLE[(typeNumber-1)*4+3];default:return undefined;}};function QRBitBuffer(){this.buffer=[];this.length=0;}
150
+ QRBitBuffer.prototype={get:function(index){var bufIndex=Math.floor(index/8);return((this.buffer[bufIndex]>>>(7-index%8))&1)==1;},put:function(num,length){for(var i=0;i<length;i++){this.putBit(((num>>>(length-i-1))&1)==1);}},getLengthInBits:function(){return this.length;},putBit:function(bit){var bufIndex=Math.floor(this.length/8);if(this.buffer.length<=bufIndex){this.buffer.push(0);}
151
+ if(bit){this.buffer[bufIndex]|=(0x80>>>(this.length%8));}
152
+ this.length++;}};var QRCodeLimitLength=[[17,14,11,7],[32,26,20,14],[53,42,32,24],[78,62,46,34],[106,84,60,44],[134,106,74,58],[154,122,86,64],[192,152,108,84],[230,180,130,98],[271,213,151,119],[321,251,177,137],[367,287,203,155],[425,331,241,177],[458,362,258,194],[520,412,292,220],[586,450,322,250],[644,504,364,280],[718,560,394,310],[792,624,442,338],[858,666,482,382],[929,711,509,403],[1003,779,565,439],[1091,857,611,461],[1171,911,661,511],[1273,997,715,535],[1367,1059,751,593],[1465,1125,805,625],[1528,1190,868,658],[1628,1264,908,698],[1732,1370,982,742],[1840,1452,1030,790],[1952,1538,1112,842],[2068,1628,1168,898],[2188,1722,1228,958],[2303,1809,1283,983],[2431,1911,1351,1051],[2563,1989,1423,1093],[2699,2099,1499,1139],[2809,2213,1579,1219],[2953,2331,1663,1273]];
153
+
154
+ function _isSupportCanvas() {
155
+ return typeof CanvasRenderingContext2D != "undefined";
156
+ }
157
+
158
+ // android 2.x doesn't support Data-URI spec
159
+ function _getAndroid() {
160
+ var android = false;
161
+ var sAgent = navigator.userAgent;
162
+
163
+ if (/android/i.test(sAgent)) { // android
164
+ android = true;
165
+ aMat = sAgent.toString().match(/android ([0-9]\.[0-9])/i);
166
+
167
+ if (aMat && aMat[1]) {
168
+ android = parseFloat(aMat[1]);
169
+ }
170
+ }
171
+
172
+ return android;
173
+ }
174
+
175
+ var svgDrawer = (function() {
176
+
177
+ var Drawing = function (el, htOption) {
178
+ this._el = el;
179
+ this._htOption = htOption;
180
+ };
181
+
182
+ Drawing.prototype.draw = function (oQRCode) {
183
+ var _htOption = this._htOption;
184
+ var _el = this._el;
185
+ var nCount = oQRCode.getModuleCount();
186
+ var nWidth = Math.floor(_htOption.width / nCount);
187
+ var nHeight = Math.floor(_htOption.height / nCount);
188
+
189
+ this.clear();
190
+
191
+ function makeSVG(tag, attrs) {
192
+ var el = document.createElementNS('http://www.w3.org/2000/svg', tag);
193
+ for (var k in attrs)
194
+ if (attrs.hasOwnProperty(k)) el.setAttribute(k, attrs[k]);
195
+ return el;
196
+ }
197
+
198
+ var svg = makeSVG("svg" , {'viewBox': '0 0 ' + String(nCount) + " " + String(nCount), 'width': '100%', 'height': '100%', 'fill': _htOption.colorLight});
199
+ svg.setAttributeNS("http://www.w3.org/2000/xmlns/", "xmlns:xlink", "http://www.w3.org/1999/xlink");
200
+ _el.appendChild(svg);
201
+
202
+ svg.appendChild(makeSVG("rect", {"fill": _htOption.colorDark, "width": "1", "height": "1", "id": "template"}));
203
+
204
+ for (var row = 0; row < nCount; row++) {
205
+ for (var col = 0; col < nCount; col++) {
206
+ if (oQRCode.isDark(row, col)) {
207
+ var child = makeSVG("use", {"x": String(row), "y": String(col)});
208
+ child.setAttributeNS("http://www.w3.org/1999/xlink", "href", "#template")
209
+ svg.appendChild(child);
210
+ }
211
+ }
212
+ }
213
+ };
214
+ Drawing.prototype.clear = function () {
215
+ while (this._el.hasChildNodes())
216
+ this._el.removeChild(this._el.lastChild);
217
+ };
218
+ return Drawing;
219
+ })();
220
+
221
+ var useSVG = document.documentElement.tagName.toLowerCase() === "svg";
222
+
223
+ // Drawing in DOM by using Table tag
224
+ var Drawing = useSVG ? svgDrawer : !_isSupportCanvas() ? (function () {
225
+ var Drawing = function (el, htOption) {
226
+ this._el = el;
227
+ this._htOption = htOption;
228
+ };
229
+
230
+ /**
231
+ * Draw the QRCode
232
+ *
233
+ * @param {QRCode} oQRCode
234
+ */
235
+ Drawing.prototype.draw = function (oQRCode) {
236
+ var _htOption = this._htOption;
237
+ var _el = this._el;
238
+ var nCount = oQRCode.getModuleCount();
239
+ var nWidth = Math.floor(_htOption.width / nCount);
240
+ var nHeight = Math.floor(_htOption.height / nCount);
241
+ var aHTML = ['<table style="border:0;border-collapse:collapse;">'];
242
+
243
+ for (var row = 0; row < nCount; row++) {
244
+ aHTML.push('<tr>');
245
+
246
+ for (var col = 0; col < nCount; col++) {
247
+ aHTML.push('<td style="border:0;border-collapse:collapse;padding:0;margin:0;width:' + nWidth + 'px;height:' + nHeight + 'px;background-color:' + (oQRCode.isDark(row, col) ? _htOption.colorDark : _htOption.colorLight) + ';"></td>');
248
+ }
249
+
250
+ aHTML.push('</tr>');
251
+ }
252
+
253
+ aHTML.push('</table>');
254
+ _el.innerHTML = aHTML.join('');
255
+
256
+ // Fix the margin values as real size.
257
+ var elTable = _el.childNodes[0];
258
+ var nLeftMarginTable = (_htOption.width - elTable.offsetWidth) / 2;
259
+ var nTopMarginTable = (_htOption.height - elTable.offsetHeight) / 2;
260
+
261
+ if (nLeftMarginTable > 0 && nTopMarginTable > 0) {
262
+ elTable.style.margin = nTopMarginTable + "px " + nLeftMarginTable + "px";
263
+ }
264
+ };
265
+
266
+ /**
267
+ * Clear the QRCode
268
+ */
269
+ Drawing.prototype.clear = function () {
270
+ this._el.innerHTML = '';
271
+ };
272
+
273
+ return Drawing;
274
+ })() : (function () { // Drawing in Canvas
275
+ function _onMakeImage() {
276
+ this._elImage.src = this._elCanvas.toDataURL("image/png");
277
+ this._elImage.style.display = "block";
278
+ this._elCanvas.style.display = "none";
279
+ }
280
+
281
+ // Android 2.1 bug workaround
282
+ // http://code.google.com/p/android/issues/detail?id=5141
283
+ if (this._android && this._android <= 2.1) {
284
+ var factor = 1 / window.devicePixelRatio;
285
+ var drawImage = CanvasRenderingContext2D.prototype.drawImage;
286
+ CanvasRenderingContext2D.prototype.drawImage = function (image, sx, sy, sw, sh, dx, dy, dw, dh) {
287
+ if (("nodeName" in image) && /img/i.test(image.nodeName)) {
288
+ for (var i = arguments.length - 1; i >= 1; i--) {
289
+ arguments[i] = arguments[i] * factor;
290
+ }
291
+ } else if (typeof dw == "undefined") {
292
+ arguments[1] *= factor;
293
+ arguments[2] *= factor;
294
+ arguments[3] *= factor;
295
+ arguments[4] *= factor;
296
+ }
297
+
298
+ drawImage.apply(this, arguments);
299
+ };
300
+ }
301
+
302
+ /**
303
+ * Check whether the user's browser supports Data URI or not
304
+ *
305
+ * @private
306
+ * @param {Function} fSuccess Occurs if it supports Data URI
307
+ * @param {Function} fFail Occurs if it doesn't support Data URI
308
+ */
309
+ function _safeSetDataURI(fSuccess, fFail) {
310
+ var self = this;
311
+ self._fFail = fFail;
312
+ self._fSuccess = fSuccess;
313
+
314
+ // Check it just once
315
+ if (self._bSupportDataURI === null) {
316
+ var el = document.createElement("img");
317
+ var fOnError = function() {
318
+ self._bSupportDataURI = false;
319
+
320
+ if (self._fFail) {
321
+ _fFail.call(self);
322
+ }
323
+ };
324
+ var fOnSuccess = function() {
325
+ self._bSupportDataURI = true;
326
+
327
+ if (self._fSuccess) {
328
+ self._fSuccess.call(self);
329
+ }
330
+ };
331
+
332
+ el.onabort = fOnError;
333
+ el.onerror = fOnError;
334
+ el.onload = fOnSuccess;
335
+ el.src = "data:image/gif;base64,iVBORw0KGgoAAAANSUhEUgAAAAUAAAAFCAYAAACNbyblAAAAHElEQVQI12P4//8/w38GIAXDIBKE0DHxgljNBAAO9TXL0Y4OHwAAAABJRU5ErkJggg=="; // the Image contains 1px data.
336
+ return;
337
+ } else if (self._bSupportDataURI === true && self._fSuccess) {
338
+ self._fSuccess.call(self);
339
+ } else if (self._bSupportDataURI === false && self._fFail) {
340
+ self._fFail.call(self);
341
+ }
342
+ };
343
+
344
+ /**
345
+ * Drawing QRCode by using canvas
346
+ *
347
+ * @constructor
348
+ * @param {HTMLElement} el
349
+ * @param {Object} htOption QRCode Options
350
+ */
351
+ var Drawing = function (el, htOption) {
352
+ this._bIsPainted = false;
353
+ this._android = _getAndroid();
354
+
355
+ this._htOption = htOption;
356
+ this._elCanvas = document.createElement("canvas");
357
+ this._elCanvas.width = htOption.width;
358
+ this._elCanvas.height = htOption.height;
359
+ el.appendChild(this._elCanvas);
360
+ this._el = el;
361
+ this._oContext = this._elCanvas.getContext("2d");
362
+ this._bIsPainted = false;
363
+ this._elImage = document.createElement("img");
364
+ this._elImage.alt = "Scan me!";
365
+ this._elImage.style.display = "none";
366
+ this._el.appendChild(this._elImage);
367
+ this._bSupportDataURI = null;
368
+ };
369
+
370
+ /**
371
+ * Draw the QRCode
372
+ *
373
+ * @param {QRCode} oQRCode
374
+ */
375
+ Drawing.prototype.draw = function (oQRCode) {
376
+ var _elImage = this._elImage;
377
+ var _oContext = this._oContext;
378
+ var _htOption = this._htOption;
379
+
380
+ var nCount = oQRCode.getModuleCount();
381
+ var nWidth = _htOption.width / nCount;
382
+ var nHeight = _htOption.height / nCount;
383
+ var nRoundedWidth = Math.round(nWidth);
384
+ var nRoundedHeight = Math.round(nHeight);
385
+
386
+ _elImage.style.display = "none";
387
+ this.clear();
388
+
389
+ for (var row = 0; row < nCount; row++) {
390
+ for (var col = 0; col < nCount; col++) {
391
+ var bIsDark = oQRCode.isDark(row, col);
392
+ var nLeft = col * nWidth;
393
+ var nTop = row * nHeight;
394
+ _oContext.strokeStyle = bIsDark ? _htOption.colorDark : _htOption.colorLight;
395
+ _oContext.lineWidth = 1;
396
+ _oContext.fillStyle = bIsDark ? _htOption.colorDark : _htOption.colorLight;
397
+ _oContext.fillRect(nLeft, nTop, nWidth, nHeight);
398
+
399
+ // 안티 앨리어싱 방지 처리
400
+ _oContext.strokeRect(
401
+ Math.floor(nLeft) + 0.5,
402
+ Math.floor(nTop) + 0.5,
403
+ nRoundedWidth,
404
+ nRoundedHeight
405
+ );
406
+
407
+ _oContext.strokeRect(
408
+ Math.ceil(nLeft) - 0.5,
409
+ Math.ceil(nTop) - 0.5,
410
+ nRoundedWidth,
411
+ nRoundedHeight
412
+ );
413
+ }
414
+ }
415
+
416
+ this._bIsPainted = true;
417
+ };
418
+
419
+ /**
420
+ * Make the image from Canvas if the browser supports Data URI.
421
+ */
422
+ Drawing.prototype.makeImage = function () {
423
+ if (this._bIsPainted) {
424
+ _safeSetDataURI.call(this, _onMakeImage);
425
+ }
426
+ };
427
+
428
+ /**
429
+ * Return whether the QRCode is painted or not
430
+ *
431
+ * @return {Boolean}
432
+ */
433
+ Drawing.prototype.isPainted = function () {
434
+ return this._bIsPainted;
435
+ };
436
+
437
+ /**
438
+ * Clear the QRCode
439
+ */
440
+ Drawing.prototype.clear = function () {
441
+ this._oContext.clearRect(0, 0, this._elCanvas.width, this._elCanvas.height);
442
+ this._bIsPainted = false;
443
+ };
444
+
445
+ /**
446
+ * @private
447
+ * @param {Number} nNumber
448
+ */
449
+ Drawing.prototype.round = function (nNumber) {
450
+ if (!nNumber) {
451
+ return nNumber;
452
+ }
453
+
454
+ return Math.floor(nNumber * 1000) / 1000;
455
+ };
456
+
457
+ return Drawing;
458
+ })();
459
+
460
+ /**
461
+ * Get the type by string length
462
+ *
463
+ * @private
464
+ * @param {String} sText
465
+ * @param {Number} nCorrectLevel
466
+ * @return {Number} type
467
+ */
468
+ function _getTypeNumber(sText, nCorrectLevel) {
469
+ var nType = 1;
470
+ var length = _getUTF8Length(sText);
471
+
472
+ for (var i = 0, len = QRCodeLimitLength.length; i <= len; i++) {
473
+ var nLimit = 0;
474
+
475
+ switch (nCorrectLevel) {
476
+ case QRErrorCorrectLevel.L :
477
+ nLimit = QRCodeLimitLength[i][0];
478
+ break;
479
+ case QRErrorCorrectLevel.M :
480
+ nLimit = QRCodeLimitLength[i][1];
481
+ break;
482
+ case QRErrorCorrectLevel.Q :
483
+ nLimit = QRCodeLimitLength[i][2];
484
+ break;
485
+ case QRErrorCorrectLevel.H :
486
+ nLimit = QRCodeLimitLength[i][3];
487
+ break;
488
+ }
489
+
490
+ if (length <= nLimit) {
491
+ break;
492
+ } else {
493
+ nType++;
494
+ }
495
+ }
496
+
497
+ if (nType > QRCodeLimitLength.length) {
498
+ throw new Error("Too long data");
499
+ }
500
+
501
+ return nType;
502
+ }
503
+
504
+ function _getUTF8Length(sText) {
505
+ var replacedText = encodeURI(sText).toString().replace(/\%[0-9a-fA-F]{2}/g, 'a');
506
+ return replacedText.length + (replacedText.length != sText ? 3 : 0);
507
+ }
508
+
509
+ /**
510
+ * @class QRCode
511
+ * @constructor
512
+ * @example
513
+ * new QRCode(document.getElementById("test"), "http://jindo.dev.naver.com/collie");
514
+ *
515
+ * @example
516
+ * var oQRCode = new QRCode("test", {
517
+ * text : "http://naver.com",
518
+ * width : 128,
519
+ * height : 128
520
+ * });
521
+ *
522
+ * oQRCode.clear(); // Clear the QRCode.
523
+ * oQRCode.makeCode("http://map.naver.com"); // Re-create the QRCode.
524
+ *
525
+ * @param {HTMLElement|String} el target element or 'id' attribute of element.
526
+ * @param {Object|String} vOption
527
+ * @param {String} vOption.text QRCode link data
528
+ * @param {Number} [vOption.width=256]
529
+ * @param {Number} [vOption.height=256]
530
+ * @param {String} [vOption.colorDark="#000000"]
531
+ * @param {String} [vOption.colorLight="#ffffff"]
532
+ * @param {QRCode.CorrectLevel} [vOption.correctLevel=QRCode.CorrectLevel.H] [L|M|Q|H]
533
+ */
534
+ QRCode = function (el, vOption) {
535
+ this._htOption = {
536
+ width : 256,
537
+ height : 256,
538
+ typeNumber : 4,
539
+ colorDark : "#000000",
540
+ colorLight : "#ffffff",
541
+ correctLevel : QRErrorCorrectLevel.H
542
+ };
543
+
544
+ if (typeof vOption === 'string') {
545
+ vOption = {
546
+ text : vOption
547
+ };
548
+ }
549
+
550
+ // Overwrites options
551
+ if (vOption) {
552
+ for (var i in vOption) {
553
+ this._htOption[i] = vOption[i];
554
+ }
555
+ }
556
+
557
+ if (typeof el == "string") {
558
+ el = document.getElementById(el);
559
+ }
560
+
561
+ this._android = _getAndroid();
562
+ this._el = el;
563
+ this._oQRCode = null;
564
+ this._oDrawing = new Drawing(this._el, this._htOption);
565
+
566
+ if (this._htOption.text) {
567
+ this.makeCode(this._htOption.text);
568
+ }
569
+ };
570
+
571
+ /**
572
+ * Make the QRCode
573
+ *
574
+ * @param {String} sText link data
575
+ */
576
+ QRCode.prototype.makeCode = function (sText) {
577
+ this._oQRCode = new QRCodeModel(_getTypeNumber(sText, this._htOption.correctLevel), this._htOption.correctLevel);
578
+ this._oQRCode.addData(sText);
579
+ this._oQRCode.make();
580
+ this._el.title = sText;
581
+ this._oDrawing.draw(this._oQRCode);
582
+ this.makeImage();
583
+ };
584
+
585
+ /**
586
+ * Make the Image from Canvas element
587
+ * - It occurs automatically
588
+ * - Android below 3 doesn't support Data-URI spec.
589
+ *
590
+ * @private
591
+ */
592
+ QRCode.prototype.makeImage = function () {
593
+ if (typeof this._oDrawing.makeImage == "function" && (!this._android || this._android >= 3)) {
594
+ this._oDrawing.makeImage();
595
+ }
596
+ };
597
+
598
+ /**
599
+ * Clear the QRCode
600
+ */
601
+ QRCode.prototype.clear = function () {
602
+ this._oDrawing.clear();
603
+ };
604
+
605
+ /**
606
+ * @name QRCode.CorrectLevel
607
+ */
608
+ QRCode.CorrectLevel = QRErrorCorrectLevel;
609
+ })();