├── .gitignore ├── .rubocop.yml ├── Gemfile ├── MIT-LICENSE ├── README.md ├── Rakefile ├── app ├── assets │ ├── config │ │ └── chat_manifest.js │ ├── images │ │ └── chat │ │ │ └── .keep │ ├── javascripts │ │ ├── chat.js │ │ └── chat │ │ │ ├── cable.js │ │ │ ├── channels │ │ │ ├── message.coffee │ │ │ ├── notification.coffee │ │ │ └── status.coffee │ │ │ ├── emojionearea.min.js │ │ │ └── messages.coffee │ └── stylesheets │ │ ├── chat.css │ │ └── chat │ │ ├── avatar_grid.sass │ │ ├── chat.sass │ │ ├── checkbox.sass │ │ ├── emojionearea.min.css │ │ ├── header.sass │ │ ├── launch.sass │ │ ├── list.sass │ │ ├── message_form.sass │ │ ├── new.sass │ │ └── transcript.sass ├── channels │ └── chat │ │ ├── messages_channel.rb │ │ ├── notification_channel.rb │ │ └── status_channel.rb ├── controllers │ └── chat │ │ ├── application_controller.rb │ │ ├── conversations_controller.rb │ │ └── messages_controller.rb ├── helpers │ └── chat │ │ └── application_helper.rb ├── jobs │ └── chat │ │ ├── application_job.rb │ │ ├── message_relay_job.rb │ │ ├── notification_relay_job.rb │ │ └── status_relay_job.rb ├── mailers │ └── chat │ │ └── application_mailer.rb ├── models │ └── chat │ │ ├── application_record.rb │ │ ├── conversation.rb │ │ ├── dot_command.rb │ │ ├── dot_command │ │ ├── gif.rb │ │ ├── shrug.rb │ │ └── validator.rb │ │ ├── message.rb │ │ └── session.rb └── views │ ├── chat │ ├── _chat.html.haml │ ├── conversations │ │ ├── _conversation.html.haml │ │ ├── _index.html.haml │ │ ├── _new.html.haml │ │ ├── _show.html.haml │ │ ├── create.js.erb │ │ └── show.js.erb │ └── messages │ │ └── _message.haml │ └── layouts │ └── chat │ └── application.html.erb ├── bin ├── rails └── test ├── chat.gemspec ├── config └── routes.rb ├── lib ├── chat.rb ├── chat │ ├── engine.rb │ ├── user.rb │ └── version.rb ├── generators │ └── chat │ │ └── install │ │ ├── install_generator.rb │ │ └── templates │ │ ├── add_chat_to_users.rb │ │ ├── chat.rb │ │ └── create_chat.rb └── tasks │ └── chat_tasks.rake └── test ├── chat_test.rb ├── dummy ├── Rakefile ├── app │ ├── assets │ │ ├── config │ │ │ └── manifest.js │ │ ├── images │ │ │ └── .keep │ │ ├── javascripts │ │ │ ├── application.js │ │ │ ├── cable.js │ │ │ ├── channels │ │ │ │ └── .keep │ │ │ ├── material-components-web.min.js │ │ │ └── material_design_components.js │ │ └── stylesheets │ │ │ ├── application.css │ │ │ ├── home.sass │ │ │ ├── material-components-web.min.css │ │ │ └── sessions.sass │ ├── channels │ │ └── application_cable │ │ │ ├── channel.rb │ │ │ └── connection.rb │ ├── controllers │ │ ├── application_controller.rb │ │ ├── concerns │ │ │ └── .keep │ │ ├── home_controller.rb │ │ └── users_controller.rb │ ├── helpers │ │ ├── application_helper.rb │ │ └── snackbar_helper.rb │ ├── jobs │ │ └── application_job.rb │ ├── mailers │ │ └── application_mailer.rb │ ├── models │ │ ├── application_record.rb │ │ ├── concerns │ │ │ └── .keep │ │ └── user.rb │ └── views │ │ ├── clearance │ │ └── users │ │ │ └── _form.html.haml │ │ ├── home │ │ └── index.html.haml │ │ ├── layouts │ │ ├── application.html.haml │ │ ├── mailer.html.erb │ │ └── mailer.text.erb │ │ ├── sessions │ │ └── new.html.haml │ │ └── users │ │ ├── edit.html.haml │ │ └── new.html.haml ├── bin │ ├── bundle │ ├── rails │ ├── rake │ ├── setup │ └── update ├── config.ru ├── config │ ├── application.rb │ ├── boot.rb │ ├── cable.yml │ ├── database.yml │ ├── environment.rb │ ├── environments │ │ ├── development.rb │ │ ├── production.rb │ │ └── test.rb │ ├── initializers │ │ ├── application_controller_renderer.rb │ │ ├── assets.rb │ │ ├── backtrace_silencers.rb │ │ ├── chat.rb │ │ ├── clearance.rb │ │ ├── cookies_serializer.rb │ │ ├── filter_parameter_logging.rb │ │ ├── inflections.rb │ │ ├── mime_types.rb │ │ ├── new_framework_defaults.rb │ │ ├── session_store.rb │ │ └── wrap_parameters.rb │ ├── locales │ │ └── en.yml │ ├── puma.rb │ ├── routes.rb │ ├── secrets.yml │ ├── sidekiq.yml │ └── spring.rb ├── db │ ├── migrate │ │ ├── 20160911032725_create_users.rb │ │ ├── 20160919005427_add_chat_to_users.rb │ │ └── 20160919005428_create_chat.rb │ ├── schema.rb │ └── seeds.rb ├── lib │ └── assets │ │ └── .keep └── public │ ├── 404.html │ ├── 422.html │ ├── 500.html │ ├── apple-touch-icon-precomposed.png │ ├── apple-touch-icon.png │ └── favicon.ico ├── integration └── navigation_test.rb ├── lib └── generators │ └── chat │ └── install_generator_test.rb └── test_helper.rb /.gitignore: -------------------------------------------------------------------------------- 1 | /.bundle/ 2 | /.yardoc 3 | /Gemfile.lock 4 | /_yardoc/ 5 | /coverage/ 6 | /doc/ 7 | /pkg/ 8 | /spec/reports/ 9 | /tmp/ 10 | /test/dummy/tmp/ 11 | /test/dummy/log/ 12 | /test/dummy/db/*.sqlite3 13 | *.gem 14 | /test/dummy/public/system 15 | -------------------------------------------------------------------------------- /.rubocop.yml: -------------------------------------------------------------------------------- 1 | --- 2 | AllCops: 3 | Exclude: 4 | - vendor/**/* 5 | - Gemfile.lock 6 | - db/schema.rb 7 | - test/dummy/db/schema.rb 8 | TargetRubyVersion: 2.4 9 | Lint/AmbiguousOperator: 10 | Description: Checks for ambiguous operators in the first argument of a method invocation 11 | without parentheses. 12 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#parens-as-args 13 | Enabled: true 14 | Lint/AmbiguousBlockAssociation: 15 | Enabled: false 16 | Lint/AmbiguousRegexpLiteral: 17 | Description: Checks for ambiguous regexp literals in the first argument of a method 18 | invocation without parenthesis. 19 | Enabled: true 20 | Lint/AssignmentInCondition: 21 | Description: Don't use assignment in conditions. 22 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#safe-assignment-in-condition 23 | Enabled: true 24 | Lint/BlockAlignment: 25 | Description: Align block ends correctly. 26 | Enabled: true 27 | Lint/CircularArgumentReference: 28 | Description: Don't refer to the keyword argument in the default value. 29 | Enabled: true 30 | Lint/ConditionPosition: 31 | Description: Checks for condition placed in a confusing position relative to the 32 | keyword. 33 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#same-line-condition 34 | Enabled: true 35 | Lint/Debugger: 36 | Description: Check for debugger calls. 37 | Enabled: true 38 | Lint/DefEndAlignment: 39 | Description: Align ends corresponding to defs correctly. 40 | Enabled: true 41 | Lint/DeprecatedClassMethods: 42 | Description: Check for deprecated class method calls. 43 | Enabled: true 44 | Lint/DuplicateMethods: 45 | Description: Check for duplicate methods calls. 46 | Enabled: true 47 | Lint/EachWithObjectArgument: 48 | Description: Check for immutable argument given to each_with_object. 49 | Enabled: true 50 | Lint/ElseLayout: 51 | Description: Check for odd code arrangement in an else block. 52 | Enabled: true 53 | Lint/EmptyEnsure: 54 | Description: Checks for empty ensure block. 55 | Enabled: true 56 | Lint/EmptyInterpolation: 57 | Description: Checks for empty string interpolation. 58 | Enabled: true 59 | Lint/EndAlignment: 60 | Description: Align ends correctly. 61 | Enabled: true 62 | Lint/EndInMethod: 63 | Description: END blocks should not be placed inside method definitions. 64 | Enabled: true 65 | Lint/EnsureReturn: 66 | Description: Do not use return in an ensure block. 67 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-return-ensure 68 | Enabled: true 69 | Lint/FormatParameterMismatch: 70 | Description: The number of parameters to format/sprint must match the fields. 71 | Enabled: true 72 | Lint/HandleExceptions: 73 | Description: Don't suppress exception. 74 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#dont-hide-exceptions 75 | Enabled: true 76 | Lint/InvalidCharacterLiteral: 77 | Description: Checks for invalid character literals with a non-escaped whitespace 78 | character. 79 | Enabled: true 80 | Lint/LiteralInCondition: 81 | Description: Checks of literals used in conditions. 82 | Enabled: true 83 | Lint/LiteralInInterpolation: 84 | Description: Checks for literals used in interpolation. 85 | Enabled: true 86 | Lint/Loop: 87 | Description: Use Kernel#loop with break rather than begin/end/until or begin/end/while 88 | for post-loop tests. 89 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#loop-with-break 90 | Enabled: true 91 | Lint/NestedMethodDefinition: 92 | Description: Do not use nested method definitions. 93 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-nested-methods 94 | Enabled: true 95 | Lint/NonLocalExitFromIterator: 96 | Description: Do not use return in iterator to cause non-local exit. 97 | Enabled: true 98 | Lint/ParenthesesAsGroupedExpression: 99 | Description: Checks for method calls with a space before the opening parenthesis. 100 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#parens-no-spaces 101 | Enabled: true 102 | Lint/RequireParentheses: 103 | Description: Use parentheses in the method call to avoid confusion about precedence. 104 | Enabled: true 105 | Lint/RescueException: 106 | Description: Avoid rescuing the Exception class. 107 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-blind-rescues 108 | Enabled: true 109 | Lint/ShadowingOuterLocalVariable: 110 | Description: Do not use the same name as outer local variable for block arguments 111 | or block local variables. 112 | Enabled: true 113 | Lint/StringConversionInInterpolation: 114 | Description: Checks for Object#to_s usage in string interpolation. 115 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-to-s 116 | Enabled: true 117 | Lint/UnderscorePrefixedVariableName: 118 | Description: Do not use prefix `_` for a variable that is used. 119 | Enabled: true 120 | Lint/UnneededDisable: 121 | Description: 'Checks for rubocop:disable comments that can be removed. Note: this 122 | cop is not disabled when disabling all cops. It must be explicitly disabled.' 123 | Enabled: true 124 | Lint/UnusedBlockArgument: 125 | Description: Checks for unused block arguments. 126 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#underscore-unused-vars 127 | Enabled: true 128 | Lint/UnusedMethodArgument: 129 | Description: Checks for unused method arguments. 130 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#underscore-unused-vars 131 | Enabled: true 132 | Lint/UnreachableCode: 133 | Description: Unreachable code. 134 | Enabled: true 135 | Lint/UselessAccessModifier: 136 | Description: Checks for useless access modifiers. 137 | Enabled: true 138 | Lint/UselessAssignment: 139 | Description: Checks for useless assignment to a local variable. 140 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#underscore-unused-vars 141 | Enabled: true 142 | Lint/UselessComparison: 143 | Description: Checks for comparison of something with itself. 144 | Enabled: true 145 | Lint/UselessElseWithoutRescue: 146 | Description: Checks for useless `else` in `begin..end` without `rescue`. 147 | Enabled: true 148 | Lint/UselessSetterCall: 149 | Description: Checks for useless setter call to a local variable. 150 | Enabled: true 151 | Lint/Void: 152 | Description: Possible use of operator/literal/variable in void context. 153 | Enabled: true 154 | Metrics/AbcSize: 155 | Description: A calculated magnitude based on number of assignments, branches, and 156 | conditions. 157 | Reference: http://c2.com/cgi/wiki?AbcMetric 158 | Enabled: true 159 | Max: 15 160 | Metrics/BlockNesting: 161 | Description: Avoid excessive block nesting 162 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#three-is-the-number-thou-shalt-count 163 | Enabled: true 164 | Max: 4 165 | Metrics/ClassLength: 166 | Description: Avoid classes longer than 100 lines of code. 167 | Enabled: true 168 | Max: 100 169 | Metrics/CyclomaticComplexity: 170 | Description: A complexity metric that is strongly correlated to the number of test 171 | cases needed to validate a method. 172 | Enabled: true 173 | Metrics/LineLength: 174 | Description: Limit lines to 80 characters. 175 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#80-character-limits 176 | Enabled: true 177 | Metrics/MethodLength: 178 | Description: Avoid methods longer than 10 lines of code. 179 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#short-methods 180 | Enabled: true 181 | Max: 10 182 | Metrics/ModuleLength: 183 | Description: Avoid modules longer than 100 lines of code. 184 | Enabled: true 185 | Max: 100 186 | Metrics/ParameterLists: 187 | Description: Avoid parameter lists longer than three or four parameters. 188 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#too-many-params 189 | Enabled: false 190 | Metrics/PerceivedComplexity: 191 | Description: A complexity metric geared towards measuring complexity for a human 192 | reader. 193 | Enabled: true 194 | Metrics/BlockLength: 195 | Exclude: 196 | - Rakefile 197 | - "**/*.rake" 198 | - spec/**/*.rb 199 | Performance/Count: 200 | Description: Use `count` instead of `select...size`, `reject...size`, `select...count`, 201 | `reject...count`, `select...length`, and `reject...length`. 202 | Enabled: true 203 | Performance/Detect: 204 | Description: Use `detect` instead of `select.first`, `find_all.first`, `select.last`, 205 | and `find_all.last`. 206 | Reference: https://github.com/JuanitoFatas/fast-ruby#enumerabledetect-vs-enumerableselectfirst-code 207 | Enabled: true 208 | Performance/FlatMap: 209 | Description: Use `Enumerable#flat_map` instead of `Enumerable#map...Array#flatten(1)` 210 | or `Enumberable#collect..Array#flatten(1)` 211 | Reference: https://github.com/JuanitoFatas/fast-ruby#enumerablemaparrayflatten-vs-enumerableflat_map-code 212 | Enabled: true 213 | EnabledForFlattenWithoutParams: false 214 | Performance/ReverseEach: 215 | Description: Use `reverse_each` instead of `reverse.each`. 216 | Reference: https://github.com/JuanitoFatas/fast-ruby#enumerablereverseeach-vs-enumerablereverse_each-code 217 | Enabled: true 218 | Performance/Sample: 219 | Description: Use `sample` instead of `shuffle.first`, `shuffle.last`, and `shuffle[Fixnum]`. 220 | Reference: https://github.com/JuanitoFatas/fast-ruby#arrayshufflefirst-vs-arraysample-code 221 | Enabled: true 222 | Performance/Size: 223 | Description: Use `size` instead of `count` for counting the number of elements in 224 | `Array` and `Hash`. 225 | Reference: https://github.com/JuanitoFatas/fast-ruby#arraycount-vs-arraysize-code 226 | Enabled: true 227 | Performance/StringReplacement: 228 | Description: Use `tr` instead of `gsub` when you are replacing the same number of 229 | characters. Use `delete` instead of `gsub` when you are deleting characters. 230 | Reference: https://github.com/JuanitoFatas/fast-ruby#stringgsub-vs-stringtr-code 231 | Enabled: true 232 | Rails/ActionFilter: 233 | Description: Enforces consistent use of action filter methods. 234 | Enabled: false 235 | Rails/Date: 236 | Description: Checks the correct usage of date aware methods, such as Date.today, 237 | Date.current etc. 238 | Enabled: true 239 | Rails/Delegate: 240 | Description: Prefer delegate method for delegations. 241 | Enabled: true 242 | Rails/FindBy: 243 | Description: Prefer find_by over where.first. 244 | Enabled: true 245 | Rails/FindEach: 246 | Description: Prefer all.find_each over all.find. 247 | Enabled: true 248 | Rails/HasAndBelongsToMany: 249 | Description: Prefer has_many :through to has_and_belongs_to_many. 250 | Enabled: true 251 | Rails/Output: 252 | Description: Checks for calls to puts, print, etc. 253 | Enabled: true 254 | Rails/ReadWriteAttribute: 255 | Description: Checks for read_attribute(:attr) and write_attribute(:attr, val). 256 | Enabled: false 257 | Rails/ScopeArgs: 258 | Description: Checks the arguments of ActiveRecord scopes. 259 | Enabled: true 260 | Rails/TimeZone: 261 | Description: Checks the correct usage of time zone aware methods. 262 | StyleGuide: https://github.com/bbatsov/rails-style-guide#time 263 | Reference: http://danilenko.org/2012/7/6/rails_timezones 264 | Enabled: true 265 | Rails/Validation: 266 | Description: Use validates :attribute, hash of validations. 267 | Enabled: true 268 | Layout/AccessModifierIndentation: 269 | Description: Check indentation of private/protected visibility modifiers. 270 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#indent-public-private-protected 271 | Enabled: false 272 | Style/AccessorMethodName: 273 | Description: Check the naming of accessor methods for get_/set_. 274 | Enabled: false 275 | Style/Alias: 276 | Description: Use alias_method instead of alias. 277 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#alias-method 278 | Enabled: false 279 | Layout/AlignArray: 280 | Description: Align the elements of an array literal if they span more than one line. 281 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#align-multiline-arrays 282 | Enabled: true 283 | Layout/AlignHash: 284 | Description: Align the elements of a hash literal if they span more than one line. 285 | Enabled: true 286 | Layout/AlignParameters: 287 | Description: Align the parameters of a method call if they span more than one line. 288 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-double-indent 289 | Enabled: true 290 | Style/AndOr: 291 | Description: Use &&/|| instead of and/or. 292 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-and-or-or 293 | Enabled: true 294 | Style/ArrayJoin: 295 | Description: Use Array#join instead of Array#*. 296 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#array-join 297 | Enabled: false 298 | Style/AsciiComments: 299 | Description: Use only ascii symbols in comments. 300 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#english-comments 301 | Enabled: false 302 | Style/AsciiIdentifiers: 303 | Description: Use only ascii symbols in identifiers. 304 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#english-identifiers 305 | Enabled: false 306 | Style/Attr: 307 | Description: Checks for uses of Module#attr. 308 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#attr 309 | Enabled: true 310 | Style/BeginBlock: 311 | Description: Avoid the use of BEGIN blocks. 312 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-BEGIN-blocks 313 | Enabled: true 314 | Style/BarePercentLiterals: 315 | Description: Checks if usage of %() or %Q() matches configuration. 316 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#percent-q-shorthand 317 | Enabled: true 318 | Style/BlockComments: 319 | Description: Do not use block comments. 320 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-block-comments 321 | Enabled: false 322 | Layout/BlockEndNewline: 323 | Description: Put end statement of multiline block on its own line. 324 | Enabled: true 325 | Style/BlockDelimiters: 326 | Description: Avoid using {...} for multi-line blocks (multiline chaining is always 327 | ugly). Prefer {...} over do...end for single-line blocks. 328 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#single-line-blocks 329 | Enabled: true 330 | Style/BracesAroundHashParameters: 331 | Description: Enforce braces style around hash parameters. 332 | Enabled: false 333 | Style/CaseEquality: 334 | Description: Avoid explicit use of the case equality operator(===). 335 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-case-equality 336 | Enabled: false 337 | Layout/CaseIndentation: 338 | Description: Indentation of when in a case/when/[else/]end. 339 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#indent-when-to-case 340 | Enabled: true 341 | Style/CharacterLiteral: 342 | Description: Checks for uses of character literals. 343 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-character-literals 344 | Enabled: false 345 | Style/ClassAndModuleCamelCase: 346 | Description: Use CamelCase for classes and modules. 347 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#camelcase-classes 348 | Enabled: true 349 | Style/ClassAndModuleChildren: 350 | Description: Checks style of children classes and modules. 351 | Enabled: false 352 | Style/ClassCheck: 353 | Description: Enforces consistent use of `Object#is_a?` or `Object#kind_of?`. 354 | Enabled: true 355 | Style/ClassMethods: 356 | Description: Use self when defining module/class methods. 357 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#def-self-class-methods 358 | Enabled: false 359 | Style/ClassVars: 360 | Description: Avoid the use of class variables. 361 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-class-vars 362 | Enabled: false 363 | Layout/ClosingParenthesisIndentation: 364 | Description: Checks the indentation of hanging closing parentheses. 365 | Enabled: false 366 | Style/ColonMethodCall: 367 | Description: 'Do not use :: for method call.' 368 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#double-colons 369 | Enabled: true 370 | Style/CommandLiteral: 371 | Description: Use `` or %x around command literals. 372 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#percent-x 373 | Enabled: false 374 | Style/CommentAnnotation: 375 | Description: Checks formatting of annotation comments. 376 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#annotate-keywords 377 | Enabled: false 378 | Layout/CommentIndentation: 379 | Description: Indentation of comments. 380 | Enabled: false 381 | Style/ConstantName: 382 | Description: Constants should use SCREAMING_SNAKE_CASE. 383 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#screaming-snake-case 384 | Enabled: true 385 | Style/DefWithParentheses: 386 | Description: Use def with parentheses when there are arguments. 387 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#method-parens 388 | Enabled: true 389 | Style/Documentation: 390 | Description: Document classes and non-namespace modules. 391 | Enabled: false 392 | Layout/DotPosition: 393 | Description: Checks the position of the dot in multi-line method calls. 394 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#consistent-multi-line-chains 395 | Enabled: true 396 | Style/DoubleNegation: 397 | Description: Checks for uses of double negation (!!). 398 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-bang-bang 399 | Enabled: true 400 | Style/EachWithObject: 401 | Description: Prefer `each_with_object` over `inject` or `reduce`. 402 | Enabled: false 403 | Layout/ElseAlignment: 404 | Description: Align elses and elsifs correctly. 405 | Enabled: true 406 | Style/EmptyElse: 407 | Description: Avoid empty else-clauses. 408 | Enabled: true 409 | Layout/EmptyLineBetweenDefs: 410 | Description: Use empty lines between defs. 411 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#empty-lines-between-methods 412 | Enabled: true 413 | Layout/EmptyLines: 414 | Description: Don't use several empty lines in a row. 415 | Enabled: true 416 | Layout/EmptyLinesAroundAccessModifier: 417 | Description: Keep blank lines around access modifiers. 418 | Enabled: false 419 | Layout/EmptyLinesAroundBlockBody: 420 | Description: Keeps track of empty lines around block bodies. 421 | Enabled: false 422 | Layout/EmptyLinesAroundClassBody: 423 | Description: Keeps track of empty lines around class bodies. 424 | Enabled: false 425 | Layout/EmptyLinesAroundModuleBody: 426 | Description: Keeps track of empty lines around module bodies. 427 | Enabled: false 428 | Layout/EmptyLinesAroundMethodBody: 429 | Description: Keeps track of empty lines around method bodies. 430 | Enabled: false 431 | Style/EmptyLiteral: 432 | Description: Prefer literals to Array.new/Hash.new/String.new. 433 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#literal-array-hash 434 | Enabled: false 435 | Style/EndBlock: 436 | Description: Avoid the use of END blocks. 437 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-END-blocks 438 | Enabled: false 439 | Layout/EndOfLine: 440 | Description: Use Unix-style line endings. 441 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#crlf 442 | Enabled: false 443 | Style/EvenOdd: 444 | Description: Favor the use of Fixnum#even? && Fixnum#odd? 445 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#predicate-methods 446 | Enabled: true 447 | Layout/ExtraSpacing: 448 | Description: Do not use unnecessary spacing. 449 | Enabled: true 450 | Style/FileName: 451 | Description: Use snake_case for source file names. 452 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#snake-case-files 453 | Enabled: true 454 | Layout/InitialIndentation: 455 | Description: Checks the indentation of the first non-blank non-comment line in a 456 | file. 457 | Enabled: false 458 | Layout/FirstParameterIndentation: 459 | Description: Checks the indentation of the first parameter in a method call. 460 | Enabled: false 461 | Style/FlipFlop: 462 | Description: Checks for flip flops 463 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-flip-flops 464 | Enabled: true 465 | Style/For: 466 | Description: Checks use of for or each in multiline loops. 467 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-for-loops 468 | Enabled: true 469 | Style/FormatString: 470 | Description: Enforce the use of Kernel#sprintf, Kernel#format or String#%. 471 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#sprintf 472 | Enabled: false 473 | Style/GlobalVars: 474 | Description: Do not introduce global variables. 475 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#instance-vars 476 | Reference: http://www.zenspider.com/Languages/Ruby/QuickRef.html 477 | Enabled: true 478 | Style/GuardClause: 479 | Description: Check for conditionals that can be replaced with guard clauses 480 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-nested-conditionals 481 | Enabled: true 482 | Style/HashSyntax: 483 | Description: 'Prefer Ruby 1.9 hash syntax { a: 1, b: 2 } over 1.8 syntax { :a => 484 | 1, :b => 2 }.' 485 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#hash-literals 486 | Enabled: true 487 | Style/IfUnlessModifier: 488 | Description: Favor modifier if/unless usage when you have a single-line body. 489 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#if-as-a-modifier 490 | Enabled: true 491 | Style/IfWithSemicolon: 492 | Description: Do not use if x; .... Use the ternary operator instead. 493 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-semicolon-ifs 494 | Enabled: true 495 | Layout/IndentationConsistency: 496 | Description: Keep indentation straight. 497 | Enabled: false 498 | Layout/IndentationWidth: 499 | Description: Use 2 spaces for indentation. 500 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#spaces-indentation 501 | Enabled: true 502 | Layout/IndentArray: 503 | Description: Checks the indentation of the first element in an array literal. 504 | Enabled: false 505 | Layout/IndentHash: 506 | Description: Checks the indentation of the first key in a hash literal. 507 | Enabled: false 508 | Style/InfiniteLoop: 509 | Description: Use Kernel#loop for infinite loops. 510 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#infinite-loop 511 | Enabled: true 512 | Style/Lambda: 513 | Description: Use the new lambda literal syntax for single-line blocks. 514 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#lambda-multi-line 515 | Enabled: true 516 | Style/LambdaCall: 517 | Description: Use lambda.call(...) instead of lambda.(...). 518 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#proc-call 519 | Enabled: false 520 | Layout/LeadingCommentSpace: 521 | Description: Comments should start with a space. 522 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#hash-space 523 | Enabled: true 524 | Style/LineEndConcatenation: 525 | Description: Use \ instead of + or << to concatenate two string literals at line 526 | end. 527 | Enabled: true 528 | Style/MethodCallWithoutArgsParentheses: 529 | Description: Do not use parentheses for method calls with no arguments. 530 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-args-no-parens 531 | Enabled: true 532 | Style/MethodDefParentheses: 533 | Description: Checks if the method definitions have or don't have parentheses. 534 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#method-parens 535 | Enabled: true 536 | Style/MethodName: 537 | Description: Use the configured style when naming methods. 538 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#snake-case-symbols-methods-vars 539 | Enabled: false 540 | Style/ModuleFunction: 541 | Description: Checks for usage of `extend self` in modules. 542 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#module-function 543 | Enabled: true 544 | Style/MultilineBlockChain: 545 | Description: Avoid multi-line chains of blocks. 546 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#single-line-blocks 547 | Enabled: false 548 | Layout/MultilineBlockLayout: 549 | Description: Ensures newlines after multiline block do statements. 550 | Enabled: true 551 | Style/MultilineIfThen: 552 | Description: Do not use then for multi-line if/unless. 553 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-then 554 | Enabled: true 555 | Layout/MultilineOperationIndentation: 556 | Description: Checks indentation of binary operations that span more than one line. 557 | Enabled: false 558 | Style/MultilineTernaryOperator: 559 | Description: 'Avoid multi-line ?: (the ternary operator); use if/unless instead.' 560 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-multiline-ternary 561 | Enabled: true 562 | Style/NegatedIf: 563 | Description: Favor unless over if for negative conditions (or control flow or). 564 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#unless-for-negatives 565 | Enabled: true 566 | Style/NegatedWhile: 567 | Description: Favor until over while for negative conditions. 568 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#until-for-negatives 569 | Enabled: true 570 | Style/NestedTernaryOperator: 571 | Description: Use one expression per branch in a ternary operator. 572 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-nested-ternary 573 | Enabled: true 574 | Style/Next: 575 | Description: Use `next` to skip iteration instead of a condition at the end. 576 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-nested-conditionals 577 | Enabled: true 578 | Style/NilComparison: 579 | Description: Prefer x.nil? to x == nil. 580 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#predicate-methods 581 | Enabled: true 582 | Style/NonNilCheck: 583 | Description: Checks for redundant nil checks. 584 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-non-nil-checks 585 | Enabled: true 586 | Style/Not: 587 | Description: Use ! instead of not. 588 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#bang-not-not 589 | Enabled: true 590 | Style/NumericLiterals: 591 | Description: Add underscores to large numeric literals to improve their readability. 592 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#underscores-in-numerics 593 | Enabled: true 594 | Style/OneLineConditional: 595 | Description: Favor the ternary operator(?:) over if/then/else/end constructs. 596 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#ternary-operator 597 | Enabled: false 598 | Style/OpMethod: 599 | Description: When defining binary operators, name the argument other. 600 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#other-arg 601 | Enabled: false 602 | Style/OptionalArguments: 603 | Description: Checks for optional arguments that do not appear at the end of the 604 | argument list 605 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#optional-arguments 606 | Enabled: false 607 | Style/ParallelAssignment: 608 | Description: Check for simple usages of parallel assignment. It will only warn when 609 | the number of variables matches on both sides of the assignment. This also provides 610 | performance benefits 611 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#parallel-assignment 612 | Enabled: false 613 | Style/ParenthesesAroundCondition: 614 | Description: Don't use parentheses around the condition of an if/unless/while. 615 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-parens-if 616 | Enabled: true 617 | Style/PercentLiteralDelimiters: 618 | Description: Use `%`-literal delimiters consistently 619 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#percent-literal-braces 620 | Enabled: true 621 | PreferredDelimiters: 622 | default: "()" 623 | "%i": "()" 624 | "%I": "()" 625 | "%r": "{}" 626 | "%w": "()" 627 | "%W": "()" 628 | Style/PercentQLiterals: 629 | Description: Checks if uses of %Q/%q match the configured preference. 630 | Enabled: true 631 | Style/PerlBackrefs: 632 | Description: Avoid Perl-style regex back references. 633 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-perl-regexp-last-matchers 634 | Enabled: false 635 | Style/PredicateName: 636 | Description: Check the names of predicate methods. 637 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#bool-methods-qmark 638 | Enabled: true 639 | Style/Proc: 640 | Description: Use proc instead of Proc.new. 641 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#proc 642 | Enabled: true 643 | Style/RaiseArgs: 644 | Description: Checks the arguments passed to raise/fail. 645 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#exception-class-messages 646 | Enabled: false 647 | Style/RedundantBegin: 648 | Description: Don't use begin blocks when they are not needed. 649 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#begin-implicit 650 | Enabled: true 651 | Style/RedundantException: 652 | Description: Checks for an obsolete RuntimeException argument in raise/fail. 653 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-explicit-runtimeerror 654 | Enabled: true 655 | Style/RedundantReturn: 656 | Description: Don't use return where it's not required. 657 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-explicit-return 658 | Enabled: true 659 | Style/RedundantSelf: 660 | Description: Don't use self where it's not needed. 661 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-self-unless-required 662 | Enabled: true 663 | Style/RegexpLiteral: 664 | Description: Use / or %r around regular expressions. 665 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#percent-r 666 | Enabled: false 667 | Layout/RescueEnsureAlignment: 668 | Description: Align rescues and ensures correctly. 669 | Enabled: false 670 | Style/RescueModifier: 671 | Description: Avoid using rescue in its modifier form. 672 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-rescue-modifiers 673 | Enabled: false 674 | Style/SelfAssignment: 675 | Description: Checks for places where self-assignment shorthand should have been 676 | used. 677 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#self-assignment 678 | Enabled: true 679 | Style/Semicolon: 680 | Description: Don't use semicolons to terminate expressions. 681 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-semicolon 682 | Enabled: true 683 | Style/SignalException: 684 | Description: Checks for proper usage of fail and raise. 685 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#fail-method 686 | Enabled: true 687 | Style/SingleLineBlockParams: 688 | Description: Enforces the names of some block params. 689 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#reduce-blocks 690 | Enabled: true 691 | Style/SingleLineMethods: 692 | Description: Avoid single-line methods. 693 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-single-line-methods 694 | Enabled: false 695 | Layout/SpaceBeforeFirstArg: 696 | Description: Checks that exactly one space is used between a method name and the 697 | first argument for method calls without parentheses. 698 | Enabled: true 699 | Layout/SpaceAfterColon: 700 | Description: Use spaces after colons. 701 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#spaces-operators 702 | Enabled: true 703 | Layout/SpaceAfterComma: 704 | Description: Use spaces after commas. 705 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#spaces-operators 706 | Enabled: true 707 | Layout/SpaceAroundKeyword: 708 | Description: Use spaces around keywords. 709 | Enabled: true 710 | Layout/SpaceAfterMethodName: 711 | Description: Do not put a space between a method name and the opening parenthesis 712 | in a method definition. 713 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#parens-no-spaces 714 | Enabled: true 715 | Layout/SpaceAfterNot: 716 | Description: Tracks redundant space after the ! operator. 717 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-space-bang 718 | Enabled: false 719 | Layout/SpaceAfterSemicolon: 720 | Description: Use spaces after semicolons. 721 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#spaces-operators 722 | Enabled: true 723 | Layout/SpaceBeforeBlockBraces: 724 | Description: Checks that the left block brace has or doesn't have space before it. 725 | Enabled: false 726 | Layout/SpaceBeforeComma: 727 | Description: No spaces before commas. 728 | Enabled: false 729 | Layout/SpaceBeforeComment: 730 | Description: Checks for missing space between code and a comment on the same line. 731 | Enabled: false 732 | Layout/SpaceBeforeSemicolon: 733 | Description: No spaces before semicolons. 734 | Enabled: false 735 | Layout/SpaceInsideBlockBraces: 736 | Description: Checks that block braces have or don't have surrounding space. For 737 | blocks taking parameters, checks that the left brace has or doesn't have trailing 738 | space. 739 | Enabled: false 740 | Layout/SpaceAroundBlockParameters: 741 | Description: Checks the spacing inside and after block parameters pipes. 742 | Enabled: true 743 | Layout/SpaceAroundEqualsInParameterDefault: 744 | Description: Checks that the equals signs in parameter default assignments have 745 | or don't have surrounding space depending on configuration. 746 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#spaces-around-equals 747 | Enabled: true 748 | Layout/SpaceAroundOperators: 749 | Description: Use a single space around operators. 750 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#spaces-operators 751 | Enabled: true 752 | Layout/SpaceInsideBrackets: 753 | Description: No spaces after [ or before ]. 754 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-spaces-braces 755 | Enabled: false 756 | Layout/SpaceInsideHashLiteralBraces: 757 | Description: Use spaces inside hash literal braces - or don't. 758 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#spaces-operators 759 | Enabled: true 760 | Layout/SpaceInsideParens: 761 | Description: No spaces after ( or before ). 762 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-spaces-braces 763 | Enabled: true 764 | Layout/SpaceInsideRangeLiteral: 765 | Description: No spaces inside range literals. 766 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-space-inside-range-literals 767 | Enabled: true 768 | Layout/SpaceInsideStringInterpolation: 769 | Description: Checks for padding/surrounding spaces inside string interpolation. 770 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#string-interpolation 771 | Enabled: false 772 | Style/SpecialGlobalVars: 773 | Description: Avoid Perl-style global variables. 774 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-cryptic-perlisms 775 | Enabled: false 776 | Style/StringLiterals: 777 | Description: Checks if uses of quotes match the configured preference. 778 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#consistent-string-literals 779 | EnforcedStyle: double_quotes 780 | Enabled: true 781 | Style/StringLiteralsInInterpolation: 782 | Description: Checks if uses of quotes inside expressions in interpolated strings 783 | match the configured preference. 784 | Enabled: true 785 | Style/StructInheritance: 786 | Description: Checks for inheritance from Struct.new. 787 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-extend-struct-new 788 | Enabled: false 789 | Style/SymbolLiteral: 790 | Description: Use plain symbols instead of string symbols when possible. 791 | Enabled: false 792 | Style/SymbolProc: 793 | Description: Use symbols as procs instead of blocks when possible. 794 | Enabled: false 795 | Layout/Tab: 796 | Description: No hard tabs. 797 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#spaces-indentation 798 | Enabled: false 799 | Layout/TrailingBlankLines: 800 | Description: Checks trailing blank lines and final newline. 801 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#newline-eof 802 | Enabled: false 803 | Style/TrailingCommaInArguments: 804 | Description: Checks for trailing comma in parameter lists. 805 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-trailing-params-comma 806 | Enabled: false 807 | Style/TrailingCommaInLiteral: 808 | Description: Checks for trailing comma in literals. 809 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-trailing-array-commas 810 | Enabled: false 811 | Layout/TrailingWhitespace: 812 | Description: Avoid trailing whitespace. 813 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-trailing-whitespace 814 | Enabled: false 815 | Style/TrivialAccessors: 816 | Description: Prefer attr_* methods to trivial readers/writers. 817 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#attr_family 818 | Enabled: false 819 | Style/UnlessElse: 820 | Description: Do not use unless with else. Rewrite these with the positive case first. 821 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-else-with-unless 822 | Enabled: true 823 | Style/UnneededCapitalW: 824 | Description: Checks for %W when interpolation is not needed. 825 | Enabled: false 826 | Style/UnneededPercentQ: 827 | Description: Checks for %q/%Q when single quotes or double quotes would do. 828 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#percent-q 829 | Enabled: false 830 | Style/TrailingUnderscoreVariable: 831 | Description: Checks for the usage of unneeded trailing underscores at the end of 832 | parallel variable assignment. 833 | Enabled: false 834 | Style/VariableInterpolation: 835 | Description: Don't interpolate global, instance and class variables directly in 836 | strings. 837 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#curlies-interpolate 838 | Enabled: true 839 | Style/VariableName: 840 | Description: Use the configured style when naming variables. 841 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#snake-case-symbols-methods-vars 842 | Enabled: true 843 | Style/WhenThen: 844 | Description: Use when x then ... for one-line cases. 845 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#one-line-cases 846 | Enabled: true 847 | Style/WhileUntilDo: 848 | Description: Checks for redundant do after while or until. 849 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#no-multiline-while-do 850 | Enabled: true 851 | Style/WhileUntilModifier: 852 | Description: Favor modifier while/until usage when you have a single-line body. 853 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#while-as-a-modifier 854 | Enabled: true 855 | Style/MutableConstant: 856 | Enabled: true 857 | Style/FrozenStringLiteralComment: 858 | Enabled: true 859 | Style/WordArray: 860 | Description: Use %w or %W for arrays of words. 861 | StyleGuide: https://github.com/bbatsov/ruby-style-guide#percent-w 862 | Enabled: true 863 | Style/EmptyMethod: 864 | Enabled: false 865 | Security/Eval: 866 | Description: The use of eval represents a serious security risk. 867 | Enabled: true 868 | -------------------------------------------------------------------------------- /Gemfile: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | source "https://rubygems.org" 4 | 5 | # Declare your gem's dependencies in chat.gemspec. 6 | # Bundler will treat runtime dependencies like base dependencies, and 7 | # development dependencies will be added by default to the :development group. 8 | gemspec 9 | 10 | # Declare any dependencies that are still in development here instead of in 11 | # your gemspec. These might include edge Rails or gems from your path or 12 | # Git. Remember to move these dependencies to your gemspec before releasing 13 | # your gem to rubygems.org. 14 | 15 | # To use a debugger 16 | # gem 'byebug', group: [:development, :test] 17 | -------------------------------------------------------------------------------- /MIT-LICENSE: -------------------------------------------------------------------------------- 1 | Copyright 2016 npezza93 2 | 3 | Permission is hereby granted, free of charge, to any person obtaining 4 | a copy of this software and associated documentation files (the 5 | "Software"), to deal in the Software without restriction, including 6 | without limitation the rights to use, copy, modify, merge, publish, 7 | distribute, sublicense, and/or sell copies of the Software, and to 8 | permit persons to whom the Software is furnished to do so, subject to 9 | the following conditions: 10 | 11 | The above copyright notice and this permission notice shall be 12 | included in all copies or substantial portions of the Software. 13 | 14 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 15 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 16 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 17 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE 18 | LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION 19 | OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION 20 | WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 21 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Chat 2 | Simple Rails chat plugin that allows users to talk to one or more people at a 3 | time that leverages ActionCable. 4 | 5 | A demo of functionality can be found: [https://chatgem.herokuapp.com/](https://chatgem.herokuapp.com/) 6 | 7 | ## Getting Started 8 | 9 | #### Prerequisites 10 | 1. This gem relies on ActionCable therefore Rails 5.0.0 is a minimum. 11 | 1. For help setting up ActionCable when coming from an older version of Rails, view 12 | [this](https://github.com/npezza93/chat/wiki/Setting-up-ActionCable). 13 | 1. An authentication system must be setup before installing `chat`, like 14 | Devise, Clearance, or your own custom solution. The generator will only run 15 | unless a users table exists. 16 | 1. Chat assumes that a `current_user` view helper 17 | method exists. 18 | 1. Your connection should be identified by `current_user`. You can learn 19 | more about how to set this up by reading 20 | [here](http://edgeguides.rubyonrails.org/action_cable_overview.html#server-side-components-connections). 21 | 22 | #### Installation 23 | Add Chat to your application's Gemfile: 24 | 25 | ```ruby 26 | gem 'chat' 27 | ``` 28 | 29 | And then execute: 30 | ```bash 31 | ❯ bundle 32 | ``` 33 | 34 | Run the installer to setup migrations and helpers and then migrate: 35 | ```bash 36 | ❯ rails generate chat:install 37 | ❯ rails db:migrate 38 | ``` 39 | 40 | Require chat in your app/assets/javascripts/application.js file: 41 | 42 | ```js 43 | //= require chat 44 | ``` 45 | 46 | Require chat in your app/assets/stylesheets/application.css file 47 | 48 | ```css 49 | *= require chat 50 | ``` 51 | 52 | Finally, add the following line to your application.html.erb file or any view file 53 | you'd like Chat available from: 54 | ```erb 55 | <%= render_chat %> 56 | ``` 57 | 58 | ### Dot Commands 59 | 60 | Dot commands behave differently than regular messages and begin with a dot(.). 61 | The following commands are available: 62 | - .shrug followed by an optional message outputs a srugging emoticon 63 | - .gif following by a label fetches a random gif from giphy with the given label 64 | 65 | ## Contributing 66 | Bug reports and pull requests are welcome on GitHub at https://github.com/npezza93/chat. 67 | 68 | ## License 69 | The gem is available as open source under the terms of the 70 | [MIT License](http://opensource.org/licenses/MIT). 71 | -------------------------------------------------------------------------------- /Rakefile: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | begin 4 | require "bundler/setup" 5 | rescue LoadError 6 | puts "You must `gem install bundler` and `bundle install` to run rake tasks" 7 | end 8 | 9 | require "rdoc/task" 10 | 11 | RDoc::Task.new(:rdoc) do |rdoc| 12 | rdoc.rdoc_dir = "rdoc" 13 | rdoc.title = "Chat" 14 | rdoc.options << "--line-numbers" 15 | rdoc.rdoc_files.include("README.md") 16 | rdoc.rdoc_files.include("lib/**/*.rb") 17 | end 18 | 19 | require "bundler/gem_tasks" 20 | 21 | require "rake/testtask" 22 | 23 | Rake::TestTask.new(:test) do |t| 24 | t.libs << "lib" 25 | t.libs << "test" 26 | t.pattern = "test/**/*_test.rb" 27 | t.verbose = false 28 | end 29 | 30 | task default: :test 31 | -------------------------------------------------------------------------------- /app/assets/config/chat_manifest.js: -------------------------------------------------------------------------------- 1 | //= link_directory ../javascripts/chat .js 2 | //= link_directory ../stylesheets/chat .css 3 | -------------------------------------------------------------------------------- /app/assets/images/chat/.keep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/npezza93/chat/c8462876bd068532f6c4ebcf1d32570ea60f1dec/app/assets/images/chat/.keep -------------------------------------------------------------------------------- /app/assets/javascripts/chat.js: -------------------------------------------------------------------------------- 1 | // This is a manifest file that'll be compiled into application.js, which will include all the files 2 | // listed below. 3 | // 4 | // Any JavaScript/Coffee file within this directory, lib/assets/javascripts, vendor/assets/javascripts, 5 | // or any plugin's vendor/assets/javascripts directory can be referenced here using a relative path. 6 | // 7 | // It's not advisable to add code directly here, but if you do, it'll appear at the bottom of the 8 | // compiled file. JavaScript code in this file should be added after the last require_* statement. 9 | // 10 | // Read Sprockets README (https://github.com/rails/sprockets#sprockets-directives) for details 11 | // about supported directives. 12 | // 13 | //= require jquery 14 | //= require jquery_ujs 15 | //= require jquery-fileupload/basic 16 | //= require_tree './chat' 17 | -------------------------------------------------------------------------------- /app/assets/javascripts/chat/cable.js: -------------------------------------------------------------------------------- 1 | // Action Cable provides the framework to deal with WebSockets in Rails. 2 | // You can generate new channels where WebSocket features live using the rails generate channel command. 3 | // 4 | //= require action_cable 5 | //= require_self 6 | //= require_tree ./channels 7 | 8 | (function() { 9 | this.App || (this.App = {}); 10 | 11 | App.cable = ActionCable.createConsumer(); 12 | 13 | }).call(this); 14 | -------------------------------------------------------------------------------- /app/assets/javascripts/chat/channels/message.coffee: -------------------------------------------------------------------------------- 1 | App.chat_message = 2 | App.cable.subscriptions.create 'Chat::MessagesChannel', 3 | collection: -> $("[data-channel='messages']") 4 | 5 | connected: -> 6 | @followCurrentChat() 7 | 8 | received: (data) -> 9 | @collection().append @generateMessage(data) 10 | height = $('.current_chat').prop('scrollHeight') 11 | $('.current_chat').animate { scrollTop: height }, 'slow' 12 | 13 | followCurrentChat: -> 14 | if chatId = @current_chat() 15 | @perform 'follow', chat_id: chatId 16 | else 17 | @perform 'unsubscribed' 18 | 19 | current_user: -> 20 | if @collection().length > 0 21 | @collection().data().currentUser 22 | 23 | current_chat: -> 24 | if @collection().length > 0 25 | @collection().data().currentChat 26 | 27 | div: (classes = "")-> 28 | element = document.createElement('div') 29 | element.classList += classes if classes.length > 0 30 | element 31 | 32 | render_container: (user) -> 33 | @div("message-container #{if user == @current_user() then 'right' else ''}") 34 | 35 | render_avatar: (data) -> 36 | last_message_classes = @collection().children().last()[0] 37 | last_message_classes = 38 | if last_message_classes == undefined 39 | [] 40 | else 41 | last_message_classes.classList 42 | 43 | if data.user == @current_user() && 44 | $.inArray("right", last_message_classes) >= 0 45 | @div("transcript_placeholder_avatar").outerHTML 46 | else 47 | data.avatar 48 | 49 | render_content: (data) -> 50 | el = @div("message #{if data.image then 'image' else ''}") 51 | if data.image 52 | el.innerHTML = data.message 53 | else 54 | content = @div("content") 55 | content.innerHTML = data.message 56 | el.innerHTML = content.outerHTML 57 | 58 | el 59 | 60 | generateMessage: (data) -> 61 | container = @render_container(data.user) 62 | container.innerHTML = 63 | @render_avatar(data) + @render_content(data).outerHTML 64 | 65 | container 66 | -------------------------------------------------------------------------------- /app/assets/javascripts/chat/channels/notification.coffee: -------------------------------------------------------------------------------- 1 | App.chat_notification = 2 | App.cable.subscriptions.create "Chat::NotificationChannel", 3 | connected: -> 4 | @perform 'follow', user_id: @current_user() 5 | 6 | disconnected: -> 7 | 8 | received: (data) -> 9 | if @current_chat() != data.chat_id || !@viewing_transcript() 10 | @add_notification(data.chat_id) 11 | $("#chat__show-#{data.chat_id}").addClass("chat__has_new_message") 12 | 13 | current_user: -> 14 | if $(".chat__launch").length > 0 15 | $(".chat__launch").data().currentUser 16 | 17 | current_chat: -> 18 | App.chat_message.current_chat() 19 | 20 | viewing_transcript: -> 21 | $(".current_chat").css("display") == "flex" 22 | 23 | add_notification: (id) -> 24 | classes = "#chat .chat__settings, .chat__launch" 25 | $(classes).addClass('chat__badge') 26 | $(classes).attr('data-badge', @new_notification_count(id)) 27 | 28 | new_notification_count: (id) -> 29 | @notification_count() + @notification_increment(id) 30 | 31 | notification_count: -> 32 | parseInt($("#chat .chat__settings").data('badge') || 0) 33 | 34 | notification_increment: (id) -> 35 | if $("#chat__show-#{id}").hasClass('chat__has_new_message') 36 | 0 37 | else 38 | 1 39 | 40 | decrement: -> 41 | classes = "#chat .chat__settings, .chat__launch" 42 | if @notification_count() == 1 43 | $(classes).removeClass('chat__badge') 44 | $(classes).attr('data-badge', @notification_count() - 1) 45 | -------------------------------------------------------------------------------- /app/assets/javascripts/chat/channels/status.coffee: -------------------------------------------------------------------------------- 1 | App.chat_status = 2 | App.cable.subscriptions.create "Chat::StatusChannel", 3 | connected: -> 4 | @perform "online", user_id: App.chat_notification.current_user() 5 | 6 | disconnected: -> 7 | @perform "unsubscribed", user_id: App.chat_notification.current_user() 8 | 9 | received: (data) -> 10 | label = $("label[for='conversation_user_ids_#{data.user_id}'] .chat__status") 11 | label.removeClass("offline") 12 | label.removeClass("online") 13 | label.addClass(data.status) 14 | -------------------------------------------------------------------------------- /app/assets/javascripts/chat/emojionearea.min.js: -------------------------------------------------------------------------------- 1 | !function(a,b,c){"use strict";function i(a){g?a():h.push(a)}function o(a,b,d){var f=!0,g=1;if(b){b=b.toLowerCase();do{var h=1==g?"@"+b:b;e[a.id][h]&&e[a.id][h].length&&c.each(e[a.id][h],function(b,c){return f=c.apply(a,d||[])!==!1})}while(f&&g--)}return f}function p(a,b,d,e){e=e||function(a,b){return c(b.currentTarget)},c.each(d,function(g,h){g=c.isArray(d)?h:g,(f[a.id][h]||(f[a.id][h]=[])).push([b,g,e])})}function q(a,b,c){var e,d=g.imageType;return e="svg"==d?g.imagePathSVG:g.imagePathPNG,a.replace("{name}",c||"").replace("{img}",e+(m<2?b.toUpperCase():b)+"."+d).replace("{uni}",b).replace("{alt}",g.convert(b))}function r(a,b,c){return a.replace(/:?\+?[\w_\-]+:?/g,function(a){a=":"+a.replace(/:$/,"").replace(/^:/,"")+":";var d=g.emojioneList[a];return d?(m>3&&(d=d.unicode),q(b,d[d.length-1],a)):c?"":a})}function s(c){var d,e;if(b.getSelection){if(d=b.getSelection(),d.getRangeAt&&d.rangeCount){e=d.getRangeAt(0),e.deleteContents();var f=a.createElement("div");f.innerHTML=c;for(var h,i,g=a.createDocumentFragment();h=f.firstChild;)i=g.appendChild(h);e.insertNode(g),i&&(e=e.cloneRange(),e.setStartAfter(i),e.collapse(!0),d.removeAllRanges(),d.addRange(e))}}else a.selection&&"Control"!=a.selection.type&&a.selection.createRange().pasteHTML(c)}function u(a){return"object"==typeof a}function v(a){var b=t();if(a&&a.filters){var d=b.filters;c.each(a.filters,function(a,b){return!u(b)||c.isEmptyObject(b)?void delete d[a]:void c.each(b,function(b,c){d[a][b]=c})}),a.filters=d}return c.extend({},b,a)}function z(a,b){return a.replace(y,function(a){var c=g[0===m?"jsecapeMap":"jsEscapeMap"];return"undefined"!=typeof a&&a in c?q(b,c[a]):a})}function A(a,b){return a=a.replace(/&/g,"&").replace(//g,">").replace(/"/g,""").replace(/'/g,"'").replace(/`/g,"`").replace(/(?:\r\n|\r|\n)/g,"\n").replace(/(\n+)/g,"
$1
").replace(/\n/g,"
").replace(/<\/div>/g,""),b.shortnames&&(a=g.shortnameToUnicode(a)),z(a,b.emojiTemplate).replace(/\t/g,"    ").replace(/ /g,"  ")}function B(a,b){switch(a=a.replace(/]*alt="([^"]+)"[^>]*>/gi,"$1").replace(/\n|\r/g,"").replace(/]*>/gi,"\n").replace(/(?:<(?:div|p|ol|ul|li|pre|code|object)[^>]*>)+/gi,"
").replace(/(?:<\/(?:div|p|ol|ul|li|pre|code|object)>)+/gi,"
").replace(/\n
<\/div>/gi,"\n").replace(/
<\/div>\n/gi,"\n").replace(/(?:
)+<\/div>/gi,"\n").replace(/([^\n])<\/div>
/gi,"$1\n").replace(/(?:<\/div>)+/gi,"
").replace(/([^\n])<\/div>([^\n])/gi,"$1\n$2").replace(/<\/div>/gi,"").replace(/([^\n])
/gi,"$1\n").replace(/\n
/gi,"\n").replace(/
\n/gi,"\n\n").replace(/<(?:[^>]+)?>/g,"").replace(new RegExp(n,"g"),"").replace(/ /g," ").replace(/</g,"<").replace(/>/g,">").replace(/"/g,'"').replace(/'/g,"'").replace(/`/g,"`").replace(/&/g,"&"),b.saveEmojisAs){case"image":a=z(a,b.emojiTemplate);break;case"shortname":a=g.toShort(a)}return a}function C(){var a=this,b=a.editor[0].offsetWidth-a.editor[0].clientWidth,c=parseInt(a.button.css("marginRight"));c!==b&&(a.button.css({marginRight:b}),a.floatingPicker&&a.picker.css({right:parseInt(a.picker.css("right"))-c+b}))}function D(){var a=this;if(!a.sprite&&a.lasyEmoji[0]){var b=a.picker.offset().top,d=b+a.picker.height()+20;a.lasyEmoji.each(function(){var a=c(this),e=a.offset().top;e>b&&e",u(a)?a:{class:E(a,!0)});return c.each(k.call(arguments).slice(1),function(a,d){c.isFunction(d)&&(d=d.call(b)),d&&c(d).appendTo(b)}),b}function G(){return localStorage.getItem("recent_emojis")||""}function H(a){var b=G();if(!a.recent||a.recent!==b){if(b.length){var e,f,d=a.scrollArea.is(".skinnable");d||(e=a.scrollArea.scrollTop(),f=a.recentCategory.is(":visible")?a.recentCategory.height():0);var g=r(b,a.emojiBtnTemplate,!0).split("|").join("");if(a.recentCategory.children(".emojibtn").remove(),c(g).insertAfter(a.recentCategory.children("h1")),a.recentCategory.children(".emojibtn").on("click",function(){a.trigger("emojibtn.click",c(this))}),a.recentFilter.show(),!d){a.recentCategory.show();var h=a.recentCategory.height();f!==h&&a.scrollArea.scrollTop(e+h-f)}}else a.recentFilter.hasClass("active")&&a.recentFilter.removeClass("active").next().addClass("active"),a.recentCategory.hide(),a.recentFilter.hide();a.recent=b}}function I(a,b){var c=G(),d=c.split("|"),e=d.indexOf(b);e!==-1&&d.splice(e,1),d.unshift(b),d.length>9&&d.pop(),localStorage.setItem("recent_emojis",d.join("|")),H(a)}function J(){var a="test";try{return localStorage.setItem(a,a),localStorage.removeItem(a),!0}catch(a){return!1}}function K(d,e,f){f=v(f),d.sprite=f.sprite&&m<3,d.inline=null===f.inline?e.is("INPUT"):f.inline,d.shortnames=f.shortnames,d.saveEmojisAs=f.saveEmojisAs,d.standalone=f.standalone,d.emojiTemplate='{alt}':'emoji" src="{img}"/>'),d.emojiTemplateAlt=d.sprite?'':'',d.emojiBtnTemplate=''+d.emojiTemplateAlt+"",d.recentEmojis=f.recentEmojis&&J();var h=f.pickerPosition;d.floatingPicker="top"===h||"bottom"===h;var k,q,t,y,z,K,L,M,N,O,i=e.is("TEXTAREA")||e.is("INPUT")?"val":"text",P=F({class:l+(d.standalone?" "+l+"-standalone ":" ")+(e.attr("class")||""),role:"application"},k=d.editor=F("editor").attr({contenteditable:!d.standalone,placeholder:f.placeholder||e.data("placeholder")||e.attr("placeholder")||"",tabindex:0}),N=d.bar=F("bar"),q=d.button=F("button",F("button-open"),F("button-close")).attr("title",f.buttonTitle),t=d.picker=F("picker",F("wrapper",z=F("filters"),O=F("scroll-area",L=F("emojis-list"),y=F("tones",function(){if(f.tones){this.addClass(E("tones-"+f.tonesStyle,!0));for(var a=0;a<=5;a++)this.append(c("",{class:"btn-tone btn-tone-"+a+(a?"":" active"),"data-skin":a,role:"button"}))}})))).addClass(E("picker-position-"+f.pickerPosition,!0)).addClass(E("filters-position-"+f.filtersPosition,!0)).addClass("hidden"));if(k.data(e.data()),c.each(f.attributes,function(a,b){k.attr(a,b)}),c.each(f.filters,function(a,b){var e=0;if("recent"!==a||d.recentEmojis){if("tones"!==a)c("",{class:E("filter",!0)+" "+E("filter-"+a,!0),"data-filter":a,title:b.title}).wrapInner(r(b.icon,d.emojiTemplateAlt)).appendTo(z);else{if(!f.tones)return;e=5}do{var g=F("category").attr({name:a,"data-tone":e}).appendTo(L),h=b.emoji.replace(/[\s,;]+/g,"|");e>0&&(g.hide(),h=h.split("|").join("_tone"+e+"|")+"_tone"+e),"recent"===a&&(h=G()),h=r(h,d.sprite?'':'',!0).split("|").join(""),g.html(h),c("

").text(b.title).prependTo(g)}while(--e>0)}}),f.filters=null,d.sprite||(d.lasyEmoji=L.find(".lazy-emoji")),K=z.find(E("filter")),K.eq(0).addClass("active"),M=L.find(E("category")),d.recentFilter=K.filter('[data-filter="recent"]'),d.recentCategory=M.filter("[name=recent]"),d.scrollArea=O,f.container?c(f.container).wrapInner(P):P.insertAfter(e),f.hideSource&&e.hide(),d.setText(e[i]()),e[i](d.getText()),C.apply(d),d.standalone&&!d.getText().length){var Q=c(e).data("emoji-placeholder")||f.emojiPlaceholder;d.setText(Q),k.addClass("has-placeholder")}p(d,L.find(".emojibtn"),{click:"emojibtn.click"}),p(d,b,{resize:"!resize"}),p(d,y.children(),{click:"tone.click"}),p(d,[t,q],{mousedown:"!mousedown"},k),p(d,q,{click:"button.click"}),p(d,k,{paste:"!paste"},k),p(d,k,["focus","blur"],function(){return!d.stayFocused&&k}),p(d,t,{mousedown:"picker.mousedown",mouseup:"picker.mouseup",click:"picker.click",keyup:"picker.keyup",keydown:"picker.keydown",keypress:"picker.keypress"}),p(d,k,["mousedown","mouseup","click","keyup","keydown","keypress"]),p(d,t.find(".emojionearea-filter"),{click:"filter.click"});var R=!1;if(O.on("scroll",function(){if(!R&&(D.call(d),O.is(":not(.skinnable)"))){var a=M.eq(0),b=O.offset().top;M.each(function(d,e){return!(c(e).offset().top-b>=10)&&void(a=c(e))});var e=K.filter('[data-filter="'+a.attr("name")+'"]');e[0]&&!e.is(".active")&&(K.removeClass("active"),e.addClass("active"))}}),d.on("@filter.click",function(a){var b=a.is(".active");if(O.is(".skinnable")){if(b)return;y.children().eq(0).click()}R=!0,b||(K.filter(".active").removeClass("active"),a.addClass("active"));var c=M.filter('[name="'+a.data("filter")+'"]').offset().top,e=O.scrollTop(),f=O.offset().top;O.stop().animate({scrollTop:c+e-f-2},200,"swing",function(){D.call(d),R=!1})}).on("@picker.show",function(){d.recentEmojis&&H(d),D.call(this)}).on("@tone.click",function(a){y.children().removeClass("active");var b=a.addClass("active").data("skin");b?(O.addClass("skinnable"),M.hide().filter("[data-tone="+b+"]").show(),K.eq(0).is('.active[data-filter="recent"]')&&K.eq(0).removeClass("active").next().addClass("active")):(O.removeClass("skinnable"),M.hide().filter("[data-tone=0]").show(),K.eq(0).click()),D.call(d)}).on("@button.click",function(a){a.is(".active")?d.hidePicker():d.showPicker()}).on("@!paste",function(a,e){var f=function(b){var e="caret-"+(new Date).getTime(),f=A(b,d);s(f),s(''),a.scrollTop(i);var g=c("#"+e),h=g.offset().top-a.offset().top,j=a.height();(i+h>=j||i>h)&&a.scrollTop(i+h-2*j/3),g.remove(),d.stayFocused=!1,C.apply(d),o(d,"paste",[a,b,f])};if(e.originalEvent.clipboardData){var g=e.originalEvent.clipboardData.getData("text/plain");return f(g),e.preventDefault?e.preventDefault():e.stop(),e.returnValue=!1,e.stopPropagation(),!1}d.stayFocused=!0,s(""+n+"");var h=w(a[0]),i=a.scrollTop(),j=c("
",{contenteditable:!0}).css({position:"fixed",left:"-999px",width:"1px",height:"1px",top:"20px",overflow:"hidden"}).appendTo(c("BODY")).focus();b.setTimeout(function(){a.focus(),x(a[0],h);var b=B(j.html().replace(/\r\n|\n|\r/g,"
"),d);j.remove(),f(b)},200)}).on("@emojibtn.click",function(a){k.removeClass("has-placeholder"),P.is(".focused")||k.focus(),d.standalone?(k.html(r(a.data("name"),d.emojiTemplate)),d.trigger("blur")):(w(k[0]),s(r(a.data("name"),d.emojiTemplate))),d.recentEmojis&&I(d,a.data("name"))}).on("@!resize @keyup @emojibtn.click",C).on("@!mousedown",function(a,b){return P.is(".focused")||a.focus(),b.preventDefault(),!1}).on("@change",function(){var a=d.editor.html().replace(/<\/?(?:div|span|p)[^>]*>/gi,"");a.length&&!/^]*>$/i.test(a)||d.editor.html(d.content=""),e[i](d.getText())}).on("@focus",function(){P.addClass("focused")}).on("@blur",function(){P.removeClass("focused"),f.hidePickerOnBlur&&d.hidePicker();var a=d.editor.html();d.content!==a?(d.content=a,o(d,"change",[d.editor]),e.blur().trigger("change")):e.blur()}),f.shortcuts&&d.on("@keydown",function(a,b){b.ctrlKey||(9==b.which?(b.preventDefault(),q.click()):27==b.which&&(b.preventDefault(),q.is(".active")&&d.hidePicker()))}),u(f.events)&&!c.isEmptyObject(f.events)&&c.each(f.events,function(a,b){d.on(a.replace(/_/g,"."),b)}),f.autocomplete){var S=function(){var a={maxCount:f.textcomplete.maxCount,placement:f.textcomplete.placement};f.shortcuts&&(a.onKeydown=function(a,b){if(!a.ctrlKey&&13==a.which)return b.KEY_ENTER});var b=c.map(g.emojioneList,function(a,b){return f.autocompleteTones?b:/_tone[12345]/.test(b)?null:b});b.sort(),k.textcomplete([{id:l,match:/\B(:[\-+\w]*)$/,search:function(a,d){d(c.map(b,function(b){return 0===b.indexOf(a)?b:null}))},template:function(a){return r(a,d.emojiTemplate)+" "+a.replace(/:/g,"")},replace:function(a){return r(a,d.emojiTemplate)},cache:!0,index:1}],a),f.textcomplete.placement&&"static"==k.data("textComplete").option.appendTo.css("position")&&k.data("textComplete").option.appendTo.css("position","relative")};c.fn.textcomplete?S():c.getScript("https://cdn.rawgit.com/yuku-t/jquery-textcomplete/v1.3.4/dist/jquery.textcomplete.js",S)}d.inline&&(P.addClass(E("inline",!0)),d.on("@keydown",function(a,b){13==b.which&&b.preventDefault()})),/firefox/i.test(navigator.userAgent)&&a.execCommand("enableObjectResizing",!1,!1)}function N(d){function e(a){var b=a.cacheBustParam;return u(a.jsEscapeMap)?"?v=1.2.4"===b?"2.0.0":"?v=2.0.1"===b?"2.1.0":"?v=2.1.1"===b?"2.1.1":"?v=2.1.2"===b?"2.1.2":"?v=2.1.3"===b?"2.1.3":"2.1.4":"1.5.2"}function f(a){switch(a){case"1.5.2":return 0;case"2.0.0":return 1;case"2.1.0":case"2.1.1":return 2;case"2.1.2":return 3;case"2.1.3":case"2.1.4":default:return 4}}d=v(d),M.isLoading||(!g||f(e(g))<2?(M.isLoading=!0,c.getScript(M.defaultBase+L+"/lib/js/emojione.min.js",function(){if(g=b.emojione,L=e(g),m=f(L),M.base=M.defaultBase+L+"/assets",d.sprite){var i=M.base+"/sprites/emojione.sprites.css";a.createStyleSheet?a.createStyleSheet(i):c("",{rel:"stylesheet",href:i}).appendTo("head")}for(;h.length;)h.shift().call();M.isLoading=!1})):(L=e(g),m=f(L),M.base=M.defaultBase+L+"/assets")),i(function(){d.useInternalCDN&&(g.imagePathPNG=M.base+"/png/",g.imagePathSVG=M.base+"/svg/",g.imagePathSVGSprites=M.base+"/sprites/emojione.sprites.svg",g.imageType=d.imageType),y=new RegExp("]*>.*?|]*>.*?|<(?:object|embed|svg|img|div|span|p|a)[^>]*>|("+g.unicodeRegexp+")","gi")})}function P(a,b){b=b.replace(/^@/,"");var d=a.id;f[d][b]&&(c.each(f[d][b],function(d,e){c.each(c.isArray(e[0])?e[0]:[e[0]],function(d,f){c(f).on(e[1],function(){var d=k.call(arguments),f=c.isFunction(e[2])?e[2].apply(a,[b].concat(d)):e[2];f&&o(a,b,[f].concat(d))})})}),f[d][b]=null)}var w,x,d=0,e={},f={},g=b.emojione,h=[],j="",k=[].slice,l="emojionearea",m=0,n="​",t=function(){return c.fn.emojioneArea&&c.fn.emojioneArea.defaults?c.fn.emojioneArea.defaults:{attributes:{dir:"ltr",spellcheck:!1,autocomplete:"off",autocorrect:"off",autocapitalize:"off"},placeholder:null,emojiPlaceholder:":smiley:",container:null,hideSource:!0,shortnames:!0,sprite:!0,pickerPosition:"top",filtersPosition:"top",hidePickerOnBlur:!0,buttonTitle:"Use the TAB key to insert emoji faster",tones:!0,tonesStyle:"bullet",inline:null,saveEmojisAs:"unicode",shortcuts:!0,autocomplete:!0,autocompleteTones:!1,standalone:!1,useInternalCDN:!0,imageType:"png",recentEmojis:!0,textcomplete:{maxCount:15,placement:null},filters:{tones:{title:"Diversity",emoji:"santa runner surfer swimmer lifter ear nose point_up_2 point_down point_left point_right punch wave ok_hand thumbsup thumbsdown clap open_hands boy girl man woman cop bride_with_veil person_with_blond_hair man_with_gua_pi_mao man_with_turban older_man grandma baby construction_worker princess angel information_desk_person guardsman dancer nail_care massage haircut muscle spy hand_splayed middle_finger vulcan no_good ok_woman bow raising_hand raised_hands person_frowning person_with_pouting_face pray rowboat bicyclist mountain_bicyclist walking bath metal point_up basketball_player fist raised_hand v writing_hand"},recent:{icon:"clock3",title:"Recent",emoji:""},smileys_people:{icon:"yum",title:"Smileys & People",emoji:"grinning grimacing grin joy smiley smile sweat_smile laughing innocent wink blush slight_smile upside_down relaxed yum relieved heart_eyes kissing_heart kissing kissing_smiling_eyes kissing_closed_eyes stuck_out_tongue_winking_eye stuck_out_tongue_closed_eyes stuck_out_tongue money_mouth nerd sunglasses hugging smirk no_mouth neutral_face expressionless unamused rolling_eyes thinking flushed disappointed worried angry rage pensive confused slight_frown frowning2 persevere confounded tired_face weary triumph open_mouth scream fearful cold_sweat hushed frowning anguished cry disappointed_relieved sleepy sweat sob dizzy_face astonished zipper_mouth mask thermometer_face head_bandage sleeping zzz poop smiling_imp imp japanese_ogre japanese_goblin skull ghost alien robot smiley_cat smile_cat joy_cat heart_eyes_cat smirk_cat kissing_cat scream_cat crying_cat_face pouting_cat raised_hands clap wave thumbsup thumbsdown punch fist v ok_hand raised_hand open_hands muscle pray point_up point_up_2 point_down point_left point_right middle_finger hand_splayed metal vulcan writing_hand nail_care lips tongue ear nose eye eyes bust_in_silhouette busts_in_silhouette speaking_head baby boy girl man woman person_with_blond_hair older_man older_woman man_with_gua_pi_mao man_with_turban cop construction_worker guardsman spy santa angel princess bride_with_veil walking runner dancer dancers couple two_men_holding_hands two_women_holding_hands bow information_desk_person no_good ok_woman raising_hand person_with_pouting_face person_frowning haircut massage couple_with_heart couple_ww couple_mm couplekiss kiss_ww kiss_mm family family_mwg family_mwgb family_mwbb family_mwgg family_wwb family_wwg family_wwgb family_wwbb family_wwgg family_mmb family_mmg family_mmgb family_mmbb family_mmgg womans_clothes shirt jeans necktie dress bikini kimono lipstick kiss footprints high_heel sandal boot mans_shoe athletic_shoe womans_hat tophat helmet_with_cross mortar_board crown school_satchel pouch purse handbag briefcase eyeglasses dark_sunglasses ring closed_umbrella"},animals_nature:{icon:"hamster",title:"Animals & Nature",emoji:"dog cat mouse hamster rabbit bear panda_face koala tiger lion_face cow pig pig_nose frog octopus monkey_face see_no_evil hear_no_evil speak_no_evil monkey chicken penguin bird baby_chick hatching_chick hatched_chick wolf boar horse unicorn bee bug snail beetle ant spider scorpion crab snake turtle tropical_fish fish blowfish dolphin whale whale2 crocodile leopard tiger2 water_buffalo ox cow2 dromedary_camel camel elephant goat ram sheep racehorse pig2 rat mouse2 rooster turkey dove dog2 poodle cat2 rabbit2 chipmunk feet dragon dragon_face cactus christmas_tree evergreen_tree deciduous_tree palm_tree seedling herb shamrock four_leaf_clover bamboo tanabata_tree leaves fallen_leaf maple_leaf ear_of_rice hibiscus sunflower rose tulip blossom cherry_blossom bouquet mushroom chestnut jack_o_lantern shell spider_web earth_americas earth_africa earth_asia full_moon waning_gibbous_moon last_quarter_moon waning_crescent_moon new_moon waxing_crescent_moon first_quarter_moon waxing_gibbous_moon new_moon_with_face full_moon_with_face first_quarter_moon_with_face last_quarter_moon_with_face sun_with_face crescent_moon star star2 dizzy sparkles comet sunny white_sun_small_cloud partly_sunny white_sun_cloud white_sun_rain_cloud cloud cloud_rain thunder_cloud_rain cloud_lightning zap fire boom snowflake cloud_snow snowman2 snowman wind_blowing_face dash cloud_tornado fog umbrella2 umbrella droplet sweat_drops ocean"},food_drink:{icon:"pizza",title:"Food & Drink",emoji:"green_apple apple pear tangerine lemon banana watermelon grapes strawberry melon cherries peach pineapple tomato eggplant hot_pepper corn sweet_potato honey_pot bread cheese poultry_leg meat_on_bone fried_shrimp egg hamburger fries hotdog pizza spaghetti taco burrito ramen stew fish_cake sushi bento curry rice_ball rice rice_cracker oden dango shaved_ice ice_cream icecream cake birthday custard candy lollipop chocolate_bar popcorn doughnut cookie beer beers wine_glass cocktail tropical_drink champagne sake tea coffee baby_bottle fork_and_knife fork_knife_plate"},activity:{icon:"basketball",title:"Activity",emoji:"soccer basketball football baseball tennis volleyball rugby_football 8ball golf golfer ping_pong badminton hockey field_hockey cricket ski skier snowboarder ice_skate bow_and_arrow fishing_pole_and_fish rowboat swimmer surfer bath basketball_player lifter bicyclist mountain_bicyclist horse_racing levitate trophy running_shirt_with_sash medal military_medal reminder_ribbon rosette ticket tickets performing_arts art circus_tent microphone headphones musical_score musical_keyboard saxophone trumpet guitar violin clapper video_game space_invader dart game_die slot_machine bowling"},travel_places:{icon:"rocket",title:"Travel & Places",emoji:"red_car taxi blue_car bus trolleybus race_car police_car ambulance fire_engine minibus truck articulated_lorry tractor motorcycle bike rotating_light oncoming_police_car oncoming_bus oncoming_automobile oncoming_taxi aerial_tramway mountain_cableway suspension_railway railway_car train monorail bullettrain_side bullettrain_front light_rail mountain_railway steam_locomotive train2 metro tram station helicopter airplane_small airplane airplane_departure airplane_arriving sailboat motorboat speedboat ferry cruise_ship rocket satellite_orbital seat anchor construction fuelpump busstop vertical_traffic_light traffic_light checkered_flag ship ferris_wheel roller_coaster carousel_horse construction_site foggy tokyo_tower factory fountain rice_scene mountain mountain_snow mount_fuji volcano japan camping tent park motorway railway_track sunrise sunrise_over_mountains desert beach island city_sunset city_dusk cityscape night_with_stars bridge_at_night milky_way stars sparkler fireworks rainbow homes european_castle japanese_castle stadium statue_of_liberty house house_with_garden house_abandoned office department_store post_office european_post_office hospital bank hotel convenience_store school love_hotel wedding classical_building church mosque synagogue kaaba shinto_shrine"},objects:{icon:"bulb",title:"Objects",emoji:"watch iphone calling computer keyboard desktop printer mouse_three_button trackball joystick compression minidisc floppy_disk cd dvd vhs camera camera_with_flash video_camera movie_camera projector film_frames telephone_receiver telephone pager fax tv radio microphone2 level_slider control_knobs stopwatch timer alarm_clock clock hourglass_flowing_sand hourglass satellite battery electric_plug bulb flashlight candle wastebasket oil money_with_wings dollar yen euro pound moneybag credit_card gem scales wrench hammer hammer_pick tools pick nut_and_bolt gear chains gun bomb knife dagger crossed_swords shield smoking skull_crossbones coffin urn amphora crystal_ball prayer_beads barber alembic telescope microscope hole pill syringe thermometer label bookmark toilet shower bathtub key key2 couch sleeping_accommodation bed door bellhop frame_photo map beach_umbrella moyai shopping_bags balloon flags ribbon gift confetti_ball tada dolls wind_chime crossed_flags izakaya_lantern envelope envelope_with_arrow incoming_envelope e-mail love_letter postbox mailbox_closed mailbox mailbox_with_mail mailbox_with_no_mail package postal_horn inbox_tray outbox_tray scroll page_with_curl bookmark_tabs bar_chart chart_with_upwards_trend chart_with_downwards_trend page_facing_up date calendar calendar_spiral card_index card_box ballot_box file_cabinet clipboard notepad_spiral file_folder open_file_folder dividers newspaper2 newspaper notebook closed_book green_book blue_book orange_book notebook_with_decorative_cover ledger books book link paperclip paperclips scissors triangular_ruler straight_ruler pushpin round_pushpin triangular_flag_on_post flag_white flag_black closed_lock_with_key lock unlock lock_with_ink_pen pen_ballpoint pen_fountain black_nib pencil pencil2 crayon paintbrush mag mag_right"},symbols:{icon:"heartpulse",title:"Symbols",emoji:"heart yellow_heart green_heart blue_heart purple_heart broken_heart heart_exclamation two_hearts revolving_hearts heartbeat heartpulse sparkling_heart cupid gift_heart heart_decoration peace cross star_and_crescent om_symbol wheel_of_dharma star_of_david six_pointed_star menorah yin_yang orthodox_cross place_of_worship ophiuchus aries taurus gemini cancer leo virgo libra scorpius sagittarius capricorn aquarius pisces id atom u7a7a u5272 radioactive biohazard mobile_phone_off vibration_mode u6709 u7121 u7533 u55b6 u6708 eight_pointed_black_star vs accept white_flower ideograph_advantage secret congratulations u5408 u6e80 u7981 a b ab cl o2 sos no_entry name_badge no_entry_sign x o anger hotsprings no_pedestrians do_not_litter no_bicycles non-potable_water underage no_mobile_phones exclamation grey_exclamation question grey_question bangbang interrobang 100 low_brightness high_brightness trident fleur-de-lis part_alternation_mark warning children_crossing beginner recycle u6307 chart sparkle eight_spoked_asterisk negative_squared_cross_mark white_check_mark diamond_shape_with_a_dot_inside cyclone loop globe_with_meridians m atm sa passport_control customs baggage_claim left_luggage wheelchair no_smoking wc parking potable_water mens womens baby_symbol restroom put_litter_in_its_place cinema signal_strength koko ng ok up cool new free zero one two three four five six seven eight nine ten 1234 arrow_forward pause_button play_pause stop_button record_button track_next track_previous fast_forward rewind twisted_rightwards_arrows repeat repeat_one arrow_backward arrow_up_small arrow_down_small arrow_double_up arrow_double_down arrow_right arrow_left arrow_up arrow_down arrow_upper_right arrow_lower_right arrow_lower_left arrow_upper_left arrow_up_down left_right_arrow arrows_counterclockwise arrow_right_hook leftwards_arrow_with_hook arrow_heading_up arrow_heading_down hash asterisk information_source abc abcd capital_abcd symbols musical_note notes wavy_dash curly_loop heavy_check_mark arrows_clockwise heavy_plus_sign heavy_minus_sign heavy_division_sign heavy_multiplication_x heavy_dollar_sign currency_exchange copyright registered tm end back on top soon ballot_box_with_check radio_button white_circle black_circle red_circle large_blue_circle small_orange_diamond small_blue_diamond large_orange_diamond large_blue_diamond small_red_triangle black_small_square white_small_square black_large_square white_large_square small_red_triangle_down black_medium_square white_medium_square black_medium_small_square white_medium_small_square black_square_button white_square_button speaker sound loud_sound mute mega loudspeaker bell no_bell black_joker mahjong spades clubs hearts diamonds flower_playing_cards thought_balloon anger_right speech_balloon clock1 clock2 clock3 clock4 clock5 clock6 clock7 clock8 clock9 clock10 clock11 clock12 clock130 clock230 clock330 clock430 clock530 clock630 clock730 clock830 clock930 clock1030 clock1130 clock1230 eye_in_speech_bubble"},flags:{icon:"flag_gb",title:"Flags",emoji:"ac af al dz ad ao ai ag ar am aw au at az bs bh bd bb by be bz bj bm bt bo ba bw br bn bg bf bi cv kh cm ca ky cf td flag_cl cn co km cg flag_cd cr hr cu cy cz dk dj dm do ec eg sv gq er ee et fk fo fj fi fr pf ga gm ge de gh gi gr gl gd gu gt gn gw gy ht hn hk hu is in flag_id ir iq ie il it ci jm jp je jo kz ke ki xk kw kg la lv lb ls lr ly li lt lu mo mk mg mw my mv ml mt mh mr mu mx fm md mc mn me ms ma mz mm na nr np nl nc nz ni ne flag_ng nu kp no om pk pw ps pa pg py pe ph pl pt pr qa ro ru rw sh kn lc vc ws sm st flag_sa sn rs sc sl sg sk si sb so za kr es lk sd sr sz se ch sy tw tj tz th tl tg to tt tn tr flag_tm flag_tm ug ua ae gb us vi uy uz vu va ve vn wf eh ye zm zw re ax ta io bq cx cc gg im yt nf pn bl pm gs tk bv hm sj um ic ea cp dg as aq vg ck cw eu gf tf gp mq mp sx ss tc "}}}};b.getSelection&&a.createRange?(w=function(a){var c=b.getSelection&&b.getSelection();if(c&&c.rangeCount>0){var d=c.getRangeAt(0),e=d.cloneRange();return e.selectNodeContents(a),e.setEnd(d.startContainer,d.startOffset),e.toString().length}},x=function(c,d){var e=0,f=a.createRange();f.setStart(c,0),f.collapse(!0);for(var h,g=[c],i=!1,j=!1;!j&&(h=g.pop());)if(3==h.nodeType){var k=e+h.length;!i&&d>=e&&d<=k&&(f.setStart(h,d-e),f.setEnd(h,d-e),j=!0),e=k}else for(var l=h.childNodes.length;l--;)g.push(h.childNodes[l]);d=b.getSelection(),d.removeAllRanges(),d.addRange(f)}):a.selection&&a.body.createTextRange&&(w=function(b){var c=a.selection.createRange(),d=a.body.createTextRange();d.moveToElementText(b),d.setEndPoint("EndToStart",c);var e=d.text.length;return e+c.text.length},x=function(b,c){var d=a.body.createTextRange();d.moveToElementText(b),d.collapse(!0),d.moveEnd("character",c),d.moveStart("character",c),d.select()});var y,L=b.emojioneVersion||"2.1.4",M={defaultBase:"https://cdnjs.cloudflare.com/ajax/libs/emojione/",base:null,isLoading:!1},O=function(a,b){var c=this;N(b),e[c.id=++d]={},f[c.id]={},i(function(){K(c,a,b)})};O.prototype.on=function(a,b){if(a&&c.isFunction(b)){var d=this;c.each(a.toLowerCase().split(" "),function(a,c){P(d,c),(e[d.id][c]||(e[d.id][c]=[])).push(b)})}return this},O.prototype.off=function(a,b){if(a){var d=this.id;c.each(a.toLowerCase().replace(/_/g,".").split(" "),function(a,f){e[d][f]&&!/^@/.test(f)&&(b?c.each(e[d][f],function(a,c){c===b&&(e[d][f]=e[d][f].splice(a,1))}):e[d][f]=[])})}return this},O.prototype.trigger=function(){var a=k.call(arguments),b=[this].concat(a.slice(0,1));return b.push(a.slice(1)),o.apply(this,b)},O.prototype.setFocus=function(){var a=this;return i(function(){a.editor.focus()}),a},O.prototype.setText=function(a){var b=this;return i(function(){b.editor.html(A(a,b)),b.content=b.editor.html(),o(b,"change",[b.editor]),C.apply(b)}),b},O.prototype.getText=function(){return B(this.editor.html(),this)},O.prototype.showPicker=function(){var a=this;return a._sh_timer&&b.clearTimeout(a._sh_timer),a.picker.removeClass("hidden"),a._sh_timer=b.setTimeout(function(){a.button.addClass("active")},50),o(a,"picker.show",[a.picker]),a},O.prototype.hidePicker=function(){var a=this;return a._sh_timer&&b.clearTimeout(a._sh_timer),a.button.removeClass("active"),a._sh_timer=b.setTimeout(function(){a.picker.addClass("hidden")},500),o(a,"picker.hide",[a.picker]),a},c.fn.emojioneArea=function(a){return this.each(function(){return this.emojioneArea?this.emojioneArea:(c.data(this,"emojioneArea",this.emojioneArea=new O(c(this),a)),this.emojioneArea)})},c.fn.emojioneArea.defaults=t()}(document,window,jQuery); 2 | -------------------------------------------------------------------------------- /app/assets/javascripts/chat/messages.coffee: -------------------------------------------------------------------------------- 1 | event = 2 | if typeof(Turbolinks) == "undefined" 3 | 'DOMContentLoaded' 4 | else 5 | 'turbolinks:load' 6 | 7 | document.addEventListener event, -> 8 | $('.chat__launch').click -> 9 | $('#chat').addClass('active') 10 | $('.chat__launch').addClass('chat__disappear') 11 | return 12 | 13 | $('.chat__close').click -> 14 | $('#chat').removeClass('active') 15 | $('.chat__launch').removeClass('chat__disappear') 16 | return 17 | 18 | $('.chat__settings').click -> 19 | $('.current_chat').removeClass("slide_left") 20 | $('.chat__list').removeClass("slide_left") 21 | $('.chat__list').removeClass("slide_right") 22 | $('.chat__start-new').removeClass("chat__disappear") 23 | $('.new_conversation').removeClass("slide_right") 24 | return 25 | 26 | $(document).on 'click', '.chat__start-new', (e) -> 27 | e.preventDefault() 28 | $('.chat__start-new').addClass("chat__disappear") 29 | $('.chat__list').addClass("slide_right") 30 | $('.new_conversation').addClass("slide_right") 31 | return 32 | 33 | $(document).on 'click', '.chat__insert-image', -> 34 | $(".new_message input[type='file']").click() 35 | return 36 | 37 | $(document).on 'keyup', '.new_message .emojionearea-editor', (e) -> 38 | if e.keyCode == 13 && !e.shiftKey 39 | str = $('.new_message .emojionearea-editor').html() 40 | $('.new_message textarea').val(str) 41 | $('.new_message').submit() 42 | $('#message_text').prop('value', '') 43 | el = $('.new_message textarea').emojioneArea() 44 | el[0].emojioneArea.setText('') 45 | return 46 | -------------------------------------------------------------------------------- /app/assets/stylesheets/chat.css: -------------------------------------------------------------------------------- 1 | /* 2 | * This is a manifest file that'll be compiled into application.css, which will include all the files 3 | * listed below. 4 | * 5 | * Any CSS and SCSS file within this directory, lib/assets/stylesheets, vendor/assets/stylesheets, 6 | * or any plugin's vendor/assets/stylesheets directory can be referenced here using a relative path. 7 | * 8 | * You're free to add application-wide styles to this file and they'll appear at the bottom of the 9 | * compiled file so the styles you add here take precedence over styles defined in any other CSS/SCSS 10 | * files in this directory. Styles in this file should be added after the last require_* statement. 11 | * It is generally better to create a new file per style scope. 12 | * 13 | *= require material_icons 14 | *= require_tree ./chat 15 | */ 16 | -------------------------------------------------------------------------------- /app/assets/stylesheets/chat/avatar_grid.sass: -------------------------------------------------------------------------------- 1 | #chat .chat__list .avatar_container 2 | width: 50px 3 | height: 50px 4 | position: relative 5 | margin: 10px 0 6 | color: #212121 7 | 8 | &.count_1 9 | img 10 | width: 100% 11 | height: 100% 12 | border-radius: 50% 13 | object-fit: cover 14 | 15 | &.count_2, &.count_default 16 | img 17 | width: 50% 18 | height: 50% 19 | border-radius: 50% 20 | object-fit: cover 21 | img:nth-of-type(2) 22 | position: absolute 23 | bottom: 0 24 | right: 0 25 | img:nth-of-type(1) 26 | position: absolute 27 | left: 0 28 | top: 0 29 | div 30 | width: 50% 31 | height: 50% 32 | border-radius: 50% 33 | background-color: #fff 34 | position: absolute 35 | bottom: 0 36 | right: 0 37 | span 38 | position: absolute 39 | left: 33% 40 | top: 6% 41 | -------------------------------------------------------------------------------- /app/assets/stylesheets/chat/chat.sass: -------------------------------------------------------------------------------- 1 | @import "https://fonts.googleapis.com/css?family=Lato:300,400,700,900" 2 | 3 | #chat 4 | font-family: "Lato" !important 5 | letter-spacing: 1.5px !important 6 | width: 375px 7 | height: 100% 8 | position: fixed 9 | right: 0 10 | top: 0 11 | opacity: 0 12 | background: transparent 13 | visibility: hidden 14 | transform: scale(0) 15 | transform-origin: bottom right 16 | transition: transform .3s ease-in-out, opacity .3s ease-in-out 17 | overflow: hidden 18 | @media (max-width: 450px) 19 | width: 100% 20 | &.active 21 | visibility: visible 22 | opacity: 1 23 | transform: scale(1) 24 | .chat__flex 25 | flex: 1 26 | #chat__background 27 | height: 100% 28 | width: 100% 29 | position: absolute 30 | background-color: rgba(40,44,52, 0.75) 31 | -------------------------------------------------------------------------------- /app/assets/stylesheets/chat/checkbox.sass: -------------------------------------------------------------------------------- 1 | .new_conversation 2 | [type="checkbox"]:not(:checked), [type="checkbox"]:checked 3 | position: absolute 4 | left: -9999px 5 | opacity: 0 6 | 7 | [type="checkbox"] + label 8 | position: relative 9 | padding-left: 35px 10 | cursor: pointer 11 | display: inline-block 12 | height: 25px 13 | line-height: 25px 14 | font-size: 1rem 15 | -webkit-user-select: none 16 | -moz-user-select: none 17 | -khtml-user-select: none 18 | -ms-user-select: none 19 | 20 | [type="checkbox"] + label:before, [type="checkbox"]:not(.filled-in) + label:after 21 | content: '' 22 | position: absolute 23 | top: -2px 24 | left: 0 25 | width: 21px 26 | height: 21px 27 | z-index: 0 28 | border: 2px solid #fff 29 | border-radius: 1px 30 | margin-top: 2px 31 | transition: .2s 32 | 33 | [type="checkbox"]:not(.filled-in) + label:after 34 | border: 0 35 | -webkit-transform: scale(0) 36 | transform: scale(0) 37 | 38 | [type="checkbox"]:not(:checked):disabled + label:before 39 | border: none 40 | background-color: rgba(0, 0, 0, 0.26) 41 | 42 | [type="checkbox"]:checked + label:before 43 | top: -8px 44 | left: -5px 45 | width: 14px 46 | height: 26px 47 | border-top: 2px solid transparent 48 | border-left: 2px solid transparent 49 | border-right: 2px solid #A3EEA0 50 | border-bottom: 2px solid #A3EEA0 51 | -webkit-transform: rotate(40deg) 52 | transform: rotate(40deg) 53 | -webkit-backface-visibility: hidden 54 | backface-visibility: hidden 55 | -webkit-transform-origin: 100% 100% 56 | transform-origin: 100% 100% 57 | 58 | [type="checkbox"]:checked:disabled + label:before 59 | border-right: 2px solid rgba(0, 0, 0, 0.26) 60 | border-bottom: 2px solid rgba(0, 0, 0, 0.26) 61 | 62 | [type="checkbox"]:indeterminate + label:before 63 | top: -11px 64 | left: -12px 65 | width: 10px 66 | height: 22px 67 | border-top: none 68 | border-left: none 69 | border-right: 2px solid #e50000 70 | border-bottom: none 71 | -webkit-transform: rotate(90deg) 72 | transform: rotate(90deg) 73 | -webkit-backface-visibility: hidden 74 | backface-visibility: hidden 75 | -webkit-transform-origin: 100% 100% 76 | transform-origin: 100% 100% 77 | 78 | [type="checkbox"]:indeterminate:disabled + label:before 79 | border-right: 2px solid rgba(0, 0, 0, 0.26) 80 | background-color: transparent 81 | -------------------------------------------------------------------------------- /app/assets/stylesheets/chat/emojionearea.min.css: -------------------------------------------------------------------------------- 1 | .dropdown-menu.textcomplete-dropdown[data-strategy=emojionearea]{position:absolute;z-index:1000;min-width:160px;padding:5px 0;margin:2px 0 0;font-size:14px;text-align:left;list-style:none;background-color:#fff;-webkit-background-clip:padding-box;background-clip:padding-box;border:1px solid #ccc;border:1px solid rgba(0,0,0,.15);-moz-border-radius:4px;-webkit-border-radius:4px;border-radius:4px;-moz-box-shadow:0 6px 12px rgba(0,0,0,.175);-webkit-box-shadow:0 6px 12px rgba(0,0,0,.175);box-shadow:0 6px 12px rgba(0,0,0,.175)}.dropdown-menu.textcomplete-dropdown[data-strategy=emojionearea] li.textcomplete-item{font-size:14px;padding:1px 3px;border:0}.dropdown-menu.textcomplete-dropdown[data-strategy=emojionearea] li.textcomplete-item a{text-decoration:none;display:block;height:100%;line-height:1.8em;padding:0 1.54em 0 .615em;color:#4f4f4f}.dropdown-menu.textcomplete-dropdown[data-strategy=emojionearea] li.textcomplete-item.active,.dropdown-menu.textcomplete-dropdown[data-strategy=emojionearea] li.textcomplete-item:hover{background-color:#e4e4e4}.dropdown-menu.textcomplete-dropdown[data-strategy=emojionearea] li.textcomplete-item.active a,.dropdown-menu.textcomplete-dropdown[data-strategy=emojionearea] li.textcomplete-item:hover a{color:#333}.dropdown-menu.textcomplete-dropdown[data-strategy=emojionearea] li.textcomplete-item .emojioneemoji{font-size:inherit;height:2ex;width:2.1ex;min-height:20px;min-width:20px;display:inline-block;margin:0 5px .2ex 0;line-height:normal;vertical-align:middle;max-width:100%;top:0}.emojionearea,.emojionearea *{-moz-box-sizing:border-box;-webkit-box-sizing:border-box;box-sizing:border-box}.emojionearea,.emojionearea.form-control{display:block;position:relative!important;width:100%;height:auto;padding:0;font-size:14px}.emojionearea .emojionearea-editor{display:block;height:auto;max-height:80px;overflow:auto;padding:6px 57px 6px 0;line-height:1.42857143;font-size:17px;color:#212121;background-color:transparent;border:0;cursor:text;margin-right:1px;border-bottom:1px solid rgba(0,0,0,.26);word-wrap:break-word}.emojionearea .emojionearea-editor:empty:before{content:attr(placeholder);display:block;color:#bbb}.emojionearea .emojionearea-editor:focus{border:0;outline:0;-moz-box-shadow:none;-webkit-box-shadow:none;box-shadow:none}.emojionearea .emojionearea-editor .emojioneemoji,.emojionearea .emojionearea-editor [class*=emojione-]{font-size:inherit;height:2ex;width:2.1ex;min-height:20px;min-width:20px;display:inline-block;margin:-.2ex .15em .2ex;line-height:normal;vertical-align:middle;max-width:100%;top:0}.emojionearea.emojionearea-inline{height:34px}.emojionearea.emojionearea-inline>.emojionearea-editor{height:32px;min-height:20px;overflow:hidden;white-space:nowrap;position:absolute;top:0;left:12px;right:24px;padding:6px 0}.emojionearea.emojionearea-inline>.emojionearea-button{top:4px}.emojionearea .emojionearea-button{z-index:5;position:absolute;right:3px;top:3px;width:24px;height:24px;cursor:pointer;-moz-transition:opacity .3s ease-in-out;-o-transition:opacity .3s ease-in-out;-webkit-transition:opacity .3s ease-in-out;transition:opacity .3s ease-in-out}.emojionearea .emojionearea-button:hover{opacity:1}.emojionearea .emojionearea-button>div{display:block;width:24px;height:24px;position:absolute;-moz-transition:all .3s ease-in-out;-o-transition:all .3s ease-in-out;-webkit-transition:all .3s ease-in-out;transition:all .3s ease-in-out}.emojionearea .emojionearea-button>div.emojionearea-button-open{background-position:0 -24px;filter:alpha(enabled=false);opacity:1}.emojionearea .emojionearea-button>div.emojionearea-button-close{background-position:0 0;}.emojionearea .emojionearea-button.active>div.emojionearea-button-open{filter:alpha(Opacity=0);opacity:0}.emojionearea .emojionearea-button.active>div.emojionearea-button-close{filter:alpha(enabled=false);}.emojionearea .emojionearea-picker{background:#fff;position:absolute;-moz-box-shadow:0 1px 5px rgba(0,0,0,.32);-webkit-box-shadow:0 1px 5px rgba(0,0,0,.32);box-shadow:0 1px 5px rgba(0,0,0,.32);-moz-border-radius:5px;-webkit-border-radius:5px;border-radius:5px;height:236px;width:316px;top:-15px;right:-15px;z-index:90;-moz-transition:all .25s ease-in-out;-o-transition:all .25s ease-in-out;-webkit-transition:all .25s ease-in-out;transition:all .25s ease-in-out;filter:alpha(Opacity=0);opacity:0;-moz-user-select:-moz-none;-ms-user-select:none;-webkit-user-select:none;user-select:none}.emojionearea .emojionearea-picker.hidden{display:none}.emojionearea .emojionearea-picker .emojionearea-wrapper{position:relative;height:236px;width:316px}.emojionearea .emojionearea-picker .emojionearea-wrapper:after{content:"";display:block;position:absolute;background-repeat:no-repeat;z-index:91}.emojionearea .emojionearea-picker .emojionearea-filters{width:100%;position:absolute;background:#F5F7F9;padding:0 0 0 7px;height:40px;z-index:95}.emojionearea .emojionearea-picker .emojionearea-filters .emojionearea-filter{display:block;float:left;height:40px;width:32px;padding:7px 1px 0;cursor:pointer;-webkit-filter:grayscale(1);filter:grayscale(1)}.emojionearea .emojionearea-picker .emojionearea-filters .emojionearea-filter.active{background:#fff}.emojionearea .emojionearea-picker .emojionearea-filters .emojionearea-filter.active,.emojionearea .emojionearea-picker .emojionearea-filters .emojionearea-filter:hover{-webkit-filter:grayscale(0);filter:grayscale(0)}.emojionearea .emojionearea-picker .emojionearea-filters .emojionearea-filter>i{width:24px;height:24px;top:0}.emojionearea .emojionearea-picker .emojionearea-filters .emojionearea-filter>img{width:24px;height:24px;margin:0 3px}.emojionearea .emojionearea-picker .emojionearea-scroll-area{height:196px;overflow:auto;overflow-x:hidden;width:100%;position:absolute;padding:0 0 5px}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-emojis-list{z-index:1}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones{position:absolute;top:6px;right:10px;height:22px;z-index:2}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones>.btn-tone{display:inline-block;padding:0;border:0;vertical-align:middle;outline:0;background:0 0;cursor:pointer;position:relative}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones>.btn-tone.btn-tone-0,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones>.btn-tone.btn-tone-0:after{background-color:#ffcf3e}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones>.btn-tone.btn-tone-1,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones>.btn-tone.btn-tone-1:after{background-color:#fae3c5}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones>.btn-tone.btn-tone-2,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones>.btn-tone.btn-tone-2:after{background-color:#e2cfa5}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones>.btn-tone.btn-tone-3,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones>.btn-tone.btn-tone-3:after{background-color:#daa478}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones>.btn-tone.btn-tone-4,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones>.btn-tone.btn-tone-4:after{background-color:#a78058}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones>.btn-tone.btn-tone-5,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones>.btn-tone.btn-tone-5:after{background-color:#5e4d43}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-bullet>.btn-tone,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-square>.btn-tone{width:20px;height:20px;margin:0;background-color:transparent}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-bullet>.btn-tone:after,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-square>.btn-tone:after{content:"";position:absolute;display:block;top:4px;left:4px;width:12px;height:12px}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-bullet>.btn-tone.active:after,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-square>.btn-tone.active:after{top:0;left:0;width:20px;height:20px}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-checkbox>.btn-tone,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-radio>.btn-tone{width:16px;height:16px;margin:0 2px}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-checkbox>.btn-tone.active:after,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-radio>.btn-tone.active:after{content:"";position:absolute;display:block;background-color:transparent;border:2px solid #fff;width:8px;height:8px;top:2px;left:2px}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-category:after,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-category:before,.emojionearea .emojionearea-picker .emojionearea-scroll-area h1:after,.emojionearea .emojionearea-picker .emojionearea-scroll-area h1:before{content:" ";clear:both;display:block}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-bullet>.btn-tone,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-bullet>.btn-tone:after,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-radio>.btn-tone,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-radio>.btn-tone:after{-moz-border-radius:100%;-webkit-border-radius:100%;border-radius:100%}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-checkbox>.btn-tone,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-checkbox>.btn-tone:after,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-square>.btn-tone,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-tones.emojionearea-tones-square>.btn-tone:after{-moz-border-radius:1px;-webkit-border-radius:1px;border-radius:1px}.emojionearea .emojionearea-picker .emojionearea-scroll-area h1{display:block;font-family:Arial,'Helvetica Neue',Helvetica,sans-serif;font-size:13px;font-weight:400;color:#b2b2b2;background:#fff;line-height:20px;margin:0;padding:7px 0 5px 6px}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojionearea-category{padding:0 0 0 7px}.emojionearea .emojionearea-picker .emojionearea-scroll-area [class*=emojione-]{-moz-box-sizing:content-box;-webkit-box-sizing:content-box;box-sizing:content-box;margin:0;width:24px;height:24px;top:0}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojibtn{-moz-box-sizing:content-box;-webkit-box-sizing:content-box;box-sizing:content-box;width:24px;height:24px;float:left;display:block;margin:1px;padding:3px}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojibtn:hover{-moz-border-radius:4px;-webkit-border-radius:4px;border-radius:4px;background-color:#e4e4e4;cursor:pointer}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojibtn i,.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojibtn img{float:left;display:block;width:24px;height:24px}.emojionearea .emojionearea-picker .emojionearea-scroll-area .emojibtn img.lazy-emoji{filter:alpha(Opacity=0);opacity:0}.emojionearea .emojionearea-picker.emojionearea-filters-position-top .emojionearea-filters{top:0;-moz-border-radius-topleft:5px;-webkit-border-top-left-radius:5px;border-top-left-radius:5px;-moz-border-radius-topright:5px;-webkit-border-top-right-radius:5px;border-top-right-radius:5px}.emojionearea .emojionearea-picker.emojionearea-filters-position-top .emojionearea-scroll-area{bottom:0}.emojionearea .emojionearea-picker.emojionearea-filters-position-bottom .emojionearea-filters{bottom:0;-moz-border-radius-bottomleft:5px;-webkit-border-bottom-left-radius:5px;border-bottom-left-radius:5px;-moz-border-radius-bottomright:5px;-webkit-border-bottom-right-radius:5px;border-bottom-right-radius:5px}.emojionearea .emojionearea-picker.emojionearea-filters-position-bottom .emojionearea-scroll-area{top:0}.emojionearea .emojionearea-picker.emojionearea-picker-position-top{margin-top:-246px;right:-14px}.emojionearea .emojionearea-picker.emojionearea-picker-position-top .emojionearea-wrapper:after{width:19px;height:10px;background-position:-2px -49px;bottom:-10px;right:20px}.emojionearea .emojionearea-picker.emojionearea-picker-position-top.emojionearea-filters-position-bottom .emojionearea-wrapper:after{background-position:-2px -80px}.emojionearea .emojionearea-picker.emojionearea-picker-position-left,.emojionearea .emojionearea-picker.emojionearea-picker-position-right{margin-right:-326px;top:-8px}.emojionearea .emojionearea-picker.emojionearea-picker-position-left .emojionearea-wrapper:after,.emojionearea .emojionearea-picker.emojionearea-picker-position-right .emojionearea-wrapper:after{width:10px;height:19px;background-position:0 -60px;top:13px;left:-10px}.emojionearea .emojionearea-picker.emojionearea-picker-position-left.emojionearea-filters-position-bottom .emojionearea-wrapper:after,.emojionearea .emojionearea-picker.emojionearea-picker-position-right.emojionearea-filters-position-bottom .emojionearea-wrapper:after{background-position:right -60px}.emojionearea .emojionearea-picker.emojionearea-picker-position-bottom{margin-top:10px;right:-14px;top:47px}.emojionearea .emojionearea-picker.emojionearea-picker-position-bottom .emojionearea-wrapper:after{width:19px;height:10px;background-position:-2px -100px;top:-10px;right:20px}.emojionearea .emojionearea-picker.emojionearea-picker-position-bottom.emojionearea-filters-position-bottom .emojionearea-wrapper:after{background-position:-2px -90px}.emojionearea .emojionearea-button.active+.emojionearea-picker{filter:alpha(enabled=false);opacity:1}.emojionearea .emojionearea-button.active+.emojionearea-picker-position-top{margin-top:-229px}.emojionearea .emojionearea-button.active+.emojionearea-picker-position-left,.emojionearea .emojionearea-button.active+.emojionearea-picker-position-right{margin-right:-309px}.emojionearea .emojionearea-button.active+.emojionearea-picker-position-bottom{margin-top:-7px}.emojionearea.emojionearea-standalone{display:inline-block;width:auto;box-shadow:none}.emojionearea.emojionearea-standalone .emojionearea-editor{min-height:33px;position:relative;padding:6px 42px 6px 6px}.emojionearea.emojionearea-standalone .emojionearea-editor::before{content:"";position:absolute;top:4px;left:50%;bottom:4px;border-left:1px solid #e6e6e6}.emojionearea.emojionearea-standalone .emojionearea-editor.has-placeholder .emojioneemoji{opacity:.4}.emojionearea.emojionearea-standalone .emojionearea-button{top:0;right:0;bottom:0;left:0;width:auto;height:auto}.emojionearea.emojionearea-standalone .emojionearea-button>div{right:6px;top:5px}.emojionearea.emojionearea-standalone .emojionearea-picker.emojionearea-picker-position-bottom .emojionearea-wrapper:after,.emojionearea.emojionearea-standalone .emojionearea-picker.emojionearea-picker-position-top .emojionearea-wrapper:after{right:23px}.emojionearea.emojionearea-standalone .emojionearea-picker.emojionearea-picker-position-left .emojionearea-wrapper:after,.emojionearea.emojionearea-standalone .emojionearea-picker.emojionearea-picker-position-right .emojionearea-wrapper:after{top:15px}.emojionearea .emojionearea-button>div,.emojionearea .emojionearea-picker .emojionearea-wrapper:after{background-image:url()!important;background-size:contain;}.emojionearea .emojionearea-button > div:hover, .emojionearea .emojionearea-button:hover, .emojionearea .emojionearea-button.active > div {background-image:url()!important;background-size:contain;}.emojionearea.emojionearea-standalone .emojionearea-editor.has-placeholder{background-repeat:no-repeat;background-position:20px 4px;background-image:url()!important} 2 | -------------------------------------------------------------------------------- /app/assets/stylesheets/chat/header.sass: -------------------------------------------------------------------------------- 1 | @import "https://fonts.googleapis.com/css?family=Lato:300,400,700,900" 2 | 3 | #chat .chat__header 4 | font-family: 'Lato' !important 5 | letter-spacing: 1.5px !important 6 | height: 50px 7 | display: flex 8 | flex-direction: row 9 | align-items: center 10 | position: relative 11 | z-index: 11 12 | .chat__settings 13 | width: 20px 14 | height: 20px 15 | margin: 0 25px 16 | cursor: pointer 17 | div 18 | height: 2px 19 | background: #fff 20 | border-radius: 2px 21 | div:nth-of-type(1) 22 | width: 94% 23 | margin-bottom: 4px 24 | margin-top: 5px 25 | div:nth-of-type(2) 26 | width: 57% 27 | 28 | h6 29 | font-family: 'Lato' !important 30 | letter-spacing: 1.5px !important 31 | margin: 0 32 | font-size: 19px 33 | color: #fff 34 | font-weight: 400 35 | i 36 | margin: 0 25px 37 | color: #fff 38 | transition: all .3s ease 39 | cursor: pointer 40 | &:hover 41 | color: #eee 42 | 43 | #chat .chat__badge 44 | position: relative 45 | #chat .chat__badge[data-badge]:after, .chat__launch.chat__badge[data-badge]:after 46 | content: attr(data-badge) 47 | display: flex 48 | flex-direction: row 49 | flex-wrap: wrap 50 | justify-content: center 51 | align-content: center 52 | align-items: center 53 | position: absolute 54 | top: -7px 55 | right: -10px 56 | font-size: 12px 57 | width: 22px 58 | height: 22px 59 | border-radius: 50% 60 | background: #ff4081 61 | color: #fff 62 | font-family: 'Lato' !important 63 | .chat__launch.chat__badge[data-badge]:after 64 | position: fixed 65 | bottom: 51px 66 | right: 14px 67 | top: initial 68 | -------------------------------------------------------------------------------- /app/assets/stylesheets/chat/launch.sass: -------------------------------------------------------------------------------- 1 | .chat__launch 2 | position: fixed 3 | right: 15px 4 | bottom: 15px 5 | border-radius: 50% 6 | height: 56px 7 | margin: auto 8 | min-width: 56px 9 | width: 56px 10 | padding: 0 11 | overflow: hidden 12 | box-shadow: 0 1px 1.5px 0 rgba(0, 0, 0, 0.12), 0 1px 1px 0 rgba(0, 0, 0, 0.24) 13 | line-height: normal 14 | border: none 15 | will-change: box-shadow 16 | transition: box-shadow 0.2s cubic-bezier(0.4, 0, 1, 1), background-color 0.2s cubic-bezier(0.4, 0, 0.2, 1), color 0.2s cubic-bezier(0.4, 0, 0.2, 1) 17 | outline: none 18 | cursor: pointer 19 | text-decoration: none 20 | transition: 0.5s 21 | transform: scale(1) 22 | background: #fff 23 | i 24 | font-size: 30px 25 | &:active 26 | box-shadow: 0 4px 5px 0 rgba(0, 0, 0, 0.14), 0 1px 10px 0 rgba(0, 0, 0, 0.12), 0 2px 4px -1px rgba(0, 0, 0, 0.2) 27 | 28 | 29 | .chat__start-new 30 | position: fixed 31 | right: 10px 32 | bottom: 15px 33 | border-radius: 50% 34 | height: 50px 35 | margin: auto 36 | min-width: 50px 37 | width: 50px 38 | padding: 0 39 | overflow: hidden 40 | box-shadow: 0 1px 1.5px 0 rgba(0, 0, 0, 0.12), 0 1px 2px 0 rgba(0, 0, 0, 0.24) 41 | line-height: normal 42 | border: none 43 | will-change: box-shadow 44 | transition: box-shadow 0.2s cubic-bezier(0.4, 0, 1, 1), background-color 0.2s cubic-bezier(0.4, 0, 0.2, 1), color 0.2s cubic-bezier(0.4, 0, 0.2, 1) 45 | outline: none 46 | cursor: pointer 47 | text-decoration: none 48 | transition: 0.5s 49 | transform: scale(1) 50 | z-index: 15 51 | background: #fff 52 | i 53 | position: absolute 54 | left: 12px 55 | top: 12px 56 | font-size: 25px 57 | color: #212121 58 | &:active 59 | box-shadow: 0 4px 5px 0 rgba(0, 0, 0, 0.14), 0 1px 10px 0 rgba(0, 0, 0, 0.12), 0 2px 4px -1px rgba(0, 0, 0, 0.2) 60 | 61 | .chat__disappear 62 | transform: scale(0) 63 | -------------------------------------------------------------------------------- /app/assets/stylesheets/chat/list.sass: -------------------------------------------------------------------------------- 1 | @keyframes pulse 2 | 50% 3 | background-color: #C8E6C9 4 | #chat .slide_left 5 | transform: translate3d(-100%,0,0) 6 | #chat .slide_right 7 | transform: translate3d(100%,0,0) 8 | #chat .chat__list 9 | position: absolute 10 | width: 100% 11 | height: calc(100% - 50px) 12 | left: 0px 13 | opacity: 1 14 | transition: 0.5s 15 | overflow-y: scroll 16 | z-index: 13 17 | &::-webkit-scrollbar 18 | visibility: hidden 19 | a:not(.chat__start-new) 20 | padding: 10px 25px 21 | text-align: center 22 | transition: all 0.3s ease 23 | display: flex 24 | .chat__has_new_message 25 | animation: pulse 1s ease-out infinite 26 | .chat__glance 27 | display: flex 28 | flex-direction: column 29 | flex: 1 30 | .chat__glance-name-date 31 | display: flex 32 | flex-direction: row 33 | flex: 1 34 | margin-left: 15px 35 | margin-top: 15px 36 | color: #fff 37 | font-weight: 600 38 | .flex 39 | flex: 1 40 | .name 41 | font-size: 15px 42 | .date 43 | font-size: 11px 44 | padding-top: 3px 45 | letter-spacing: 0.5px 46 | .chat__glance-last-message 47 | margin-left: 15px 48 | text-align: left 49 | margin-top: 7px 50 | font-size: 13px 51 | color: #fff 52 | word-break: break-all 53 | font-weight: 300 54 | img 55 | font-size: inherit 56 | height: 2ex 57 | width: 2.1ex 58 | min-height: 20px 59 | min-width: 20px 60 | display: inline-block 61 | margin: -.2ex .15em .2ex 62 | line-height: normal 63 | vertical-align: middle 64 | max-width: 100% 65 | top: 0 66 | #chat .chat__none-available 67 | height: 100% 68 | display: flex 69 | flex-direction: column 70 | align-items: center 71 | justify-content: center 72 | color: #bdbdbd 73 | h5 74 | font-weight: 400 75 | font-size: 15px 76 | margin: 5px 0px 5px 0px 77 | i 78 | font-size: 60px 79 | -------------------------------------------------------------------------------- /app/assets/stylesheets/chat/message_form.sass: -------------------------------------------------------------------------------- 1 | @import "https://fonts.googleapis.com/css?family=Lato:300,400,700,900" 2 | 3 | #chat .new_message 4 | margin: 0px 20px 25px 20px 5 | flex-direction: column 6 | display: flex 7 | position: relative 8 | .chat__insert-image 9 | position: absolute 10 | right: 28px 11 | top: 4px 12 | color: #bdbdbd 13 | transition: all 0.3s ease 14 | z-index: 10 15 | cursor: pointer 16 | font-size: 22px 17 | &:hover 18 | color: #fff 19 | input[type='text'], textarea 20 | font-size: 17px 21 | font-family: "Lato" !important 22 | letter-spacing: 1.5px !important 23 | background: transparent 24 | border: none 25 | font-weight: 300 26 | border-bottom: 1px solid rgba(0,0,0,.26) 27 | height: 28px 28 | outline: none 29 | resize: none 30 | .bar, .emojionearea-bar 31 | position: relative 32 | &:before, &:after 33 | content: '' 34 | height: 2px 35 | width: 0 36 | bottom: 0px 37 | position: absolute 38 | background: #fff 39 | transition: 0.2s ease all 40 | -moz-transition: 0.2s ease all 41 | -webkit-transition: 0.2s ease all 42 | &:before 43 | left: 50% 44 | &:after 45 | right: 50% 46 | input[type='text']:focus ~ .bar:before, input[type='text']:focus ~ .bar:after, 47 | textarea:focus ~ .bar:before, textarea:focus ~ .bar:after, .emojionearea-editor:focus ~ .emojionearea-bar:before, .emojionearea-editor:focus ~ .emojionearea-bar:after 48 | width: 50% 49 | input[type='file'] 50 | display: none !important 51 | .emojionearea .emojionearea-editor 52 | border-bottom: 1px solid #bdbdbd !important 53 | color: #fff !important 54 | -------------------------------------------------------------------------------- /app/assets/stylesheets/chat/new.sass: -------------------------------------------------------------------------------- 1 | @import "https://fonts.googleapis.com/css?family=Lato:300,400,700,900" 2 | 3 | #chat .new_conversation 4 | height: calc(100% - 50px) 5 | overflow-y: scroll 6 | position: absolute 7 | right: 100% 8 | transition: 0.5s 9 | width: 100% 10 | z-index: 13 11 | display: flex 12 | flex-direction: column 13 | &::-webkit-scrollbar 14 | visibility: hidden 15 | label 16 | position: relative 17 | margin: 1rem 18 | cursor: pointer 19 | font-family: "Lato" !important 20 | letter-spacing: 1.5px !important 21 | padding-left: 40px 22 | color: #fff 23 | width: calc(100% - 40px - 1rem) 24 | .chat__status 25 | &:after 26 | content: "\2022" 27 | &.offline 28 | &:after 29 | color: hsl(355, 65%, 65%) 30 | &.online 31 | &:after 32 | color: #AED581 33 | 34 | .chat__submit-container 35 | margin: 1rem 36 | padding: 10px 0 37 | width: calc(100% - 1rem) 38 | display: flex 39 | flex-direction: column 40 | justify-content: center 41 | align-items: center 42 | 43 | input[type='submit'] 44 | border: none 45 | font-family: "Lato" !important 46 | letter-spacing: 1.5px !important 47 | font-size: 17px 48 | background: transparent 49 | padding: 0 50 | cursor: pointer 51 | transition: all 0.3s ease 52 | z-index: 10 53 | color: #fff 54 | margin: 0px 55 | 56 | #chat .chat__errors 57 | text-align: center 58 | color: #F44336 59 | -------------------------------------------------------------------------------- /app/assets/stylesheets/chat/transcript.sass: -------------------------------------------------------------------------------- 1 | #chat .current_chat 2 | display: flex 3 | flex-direction: column 4 | overflow-y: scroll 5 | position: absolute 6 | width: 100% 7 | height: calc(100% - 50px) 8 | left: 100% 9 | opacity: 1 10 | transition: 0.5s 11 | z-index: 13 12 | &::-webkit-scrollbar 13 | visibility: hidden 14 | img.chat__user-avatar 15 | width: 50px 16 | height: 50px 17 | object-fit: cover 18 | border-radius: 50% 19 | .transcript_placeholder_avatar 20 | width: 50px 21 | height: 50px 22 | i.chat__user-avatar 23 | font-size: 50px 24 | .chat__transcript 25 | margin-left: 25px 26 | margin-right: 25px 27 | .message-container 28 | display: flex 29 | flex-direction: row 30 | &.right 31 | text-align: right 32 | flex-direction: row-reverse 33 | .message 34 | flex-direction: row-reverse 35 | .content 36 | border-top-right-radius: 0 !important 37 | border-top-left-radius: 10px !important 38 | &.image img 39 | border-top-right-radius: 0 !important 40 | border-top-left-radius: 10px !important 41 | .message 42 | min-height: 50px 43 | display: flex 44 | margin: 0 10px 5px 10px 45 | flex: 1 46 | flex-direction: row 47 | max-width: 100% 48 | &.image 49 | display: block 50 | img 51 | max-width: 100% 52 | border-radius: 10px 53 | border-top-left-radius: 0 54 | .content 55 | background: #fff 56 | padding: 10px 57 | border-radius: 10px 58 | color: #212121 59 | border-top-left-radius: 0 60 | word-break: break-all 61 | img 62 | font-size: inherit 63 | height: 2ex 64 | width: 2.1ex 65 | min-height: 20px 66 | min-width: 20px 67 | display: inline-block 68 | margin: -.2ex .15em .2ex 69 | line-height: normal 70 | vertical-align: middle 71 | max-width: 100% 72 | top: 0 73 | -------------------------------------------------------------------------------- /app/channels/chat/messages_channel.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class Chat::MessagesChannel < ApplicationCable::Channel 4 | def follow(data) 5 | stop_all_streams 6 | stream_from "chats::#{data['chat_id']}::messages" 7 | end 8 | 9 | def unsubscribed 10 | stop_all_streams 11 | end 12 | end 13 | -------------------------------------------------------------------------------- /app/channels/chat/notification_channel.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class Chat::NotificationChannel < ApplicationCable::Channel 4 | def follow(data) 5 | stop_all_streams 6 | stream_from "users::#{data['user_id']}::chats" 7 | end 8 | 9 | def unsubscribed 10 | stop_all_streams 11 | end 12 | end 13 | -------------------------------------------------------------------------------- /app/channels/chat/status_channel.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class Chat::StatusChannel < ApplicationCable::Channel 4 | def online 5 | stop_all_streams 6 | stream_from "chat::status" 7 | current_user.online 8 | end 9 | 10 | def unsubscribed 11 | current_user.offline 12 | stop_all_streams 13 | end 14 | end 15 | -------------------------------------------------------------------------------- /app/controllers/chat/application_controller.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Chat 4 | class ApplicationController < ::ApplicationController 5 | protect_from_forgery with: :exception 6 | 7 | before_action Chat.logged_in_check 8 | end 9 | end 10 | -------------------------------------------------------------------------------- /app/controllers/chat/conversations_controller.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Chat 4 | class ConversationsController < ApplicationController 5 | before_action :set_conversation, only: :show 6 | 7 | def show 8 | end 9 | 10 | def create 11 | @conversation = Chat::Conversation.create(conversation_params) 12 | 13 | if @conversation.persisted? || 14 | @conversation.errors.messages[:sessions].present? 15 | render template: "chat/conversations/create" 16 | else 17 | @conversation = find_existing_conversation 18 | render template: "chat/conversations/show" 19 | end 20 | end 21 | 22 | private 23 | 24 | def set_conversation 25 | @conversation = Chat::Conversation.includes( 26 | :users, messages: :user 27 | ).find(params[:id]) 28 | end 29 | 30 | def conversation_params 31 | chat_params = params.require(:conversation).permit(user_ids: []) 32 | if chat_params[:user_ids].reject!(&:blank?).present? 33 | chat_params[:user_ids] << current_user.id 34 | chat_params[:user_ids].map!(&:to_i) 35 | end 36 | 37 | chat_params 38 | end 39 | 40 | def find_existing_conversation 41 | Chat::Conversation.includes( 42 | :users, messages: :user 43 | ).find( 44 | Chat::Conversation.joins(:users).having( 45 | "COUNT(DISTINCT users.id) = ?", conversation_params[:user_ids].count 46 | ).group(:id).find_by( 47 | users: { id: conversation_params[:user_ids] } 48 | ).id 49 | ) 50 | end 51 | end 52 | end 53 | -------------------------------------------------------------------------------- /app/controllers/chat/messages_controller.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Chat 4 | class MessagesController < ApplicationController 5 | before_action :set_conversation 6 | before_action :set_session 7 | 8 | def create 9 | @conversation.messages.create(message_params) 10 | 11 | head :ok 12 | end 13 | 14 | def destroy 15 | @conversation.destroy 16 | end 17 | 18 | private 19 | 20 | def set_conversation 21 | @conversation = Chat::Conversation.find(params[:conversation_id]) 22 | end 23 | 24 | def set_session 25 | @session = @conversation.sessions.find_by(user: current_user) 26 | end 27 | 28 | def message_params 29 | params.require(:message).permit(:text, :image).merge( 30 | user_id: current_user.id, session_id: @session.id 31 | ) 32 | end 33 | end 34 | end 35 | -------------------------------------------------------------------------------- /app/helpers/chat/application_helper.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Chat 4 | module ApplicationHelper 5 | def render_chat 6 | return unless send Chat.signed_in 7 | 8 | render "chat/chat" 9 | end 10 | 11 | def launch_chat_fab 12 | attrs = { 13 | class: "chat__launch", data: { "current-user" => current_user.id } 14 | } 15 | button_tag(attrs) do 16 | material_icon.forum.css_class("md-dark").to_s.html_safe 17 | end 18 | end 19 | 20 | def chatable_users 21 | @chatable_users ||= ::User.where.not(id: current_user.id) 22 | end 23 | 24 | def chatable_user_check_boxes(f) 25 | f.collection_check_boxes(:user_ids, chatable_users, :id, :name) do |b| 26 | b.check_box + b.label(class: b.object.chat_status) do 27 | content_tag( 28 | :div, b.text, 29 | class: "chat__status #{b.object.chat_status}" 30 | ) 31 | end 32 | end 33 | end 34 | 35 | def chat_avatar(user) 36 | if user.chat_avatar? 37 | image_tag(user.chat_avatar, class: "chat__user-avatar") 38 | else 39 | MaterialIcon.new.css_class("chat__user-avatar").person.to_s 40 | end 41 | end 42 | 43 | def chat_list 44 | @chat_list ||= current_user.conversations.includes(:users).order( 45 | "chat_conversations.created_at desc" 46 | ) 47 | end 48 | 49 | def chat_avatars(conversation) 50 | (conversation.users - [current_user]).first(2).map do |u| 51 | chat_avatar(u) 52 | end 53 | end 54 | 55 | def chat_avatar_count(conversation) 56 | if (count = conversation_user_count(conversation)) <= 2 57 | "count_#{count}" 58 | else 59 | "count_default" 60 | end 61 | end 62 | 63 | def conversation_user_count(conversation) 64 | conversation.users.to_a.size - 1 65 | end 66 | 67 | def message_classes(message) 68 | css_class = message.user == current_user ? "right" : "left" 69 | css_class += message.image? ? " image" : "" 70 | css_class 71 | end 72 | end 73 | end 74 | -------------------------------------------------------------------------------- /app/jobs/chat/application_job.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Chat 4 | class ApplicationJob < ::ApplicationJob 5 | queue_as :default 6 | 7 | private 8 | 9 | def broadcast(channel, payload) 10 | ActionCable.server.broadcast(channel, payload) 11 | end 12 | end 13 | end 14 | -------------------------------------------------------------------------------- /app/jobs/chat/message_relay_job.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Chat 4 | class MessageRelayJob < ApplicationJob 5 | def perform(message_id) 6 | message = Chat::Message.find(message_id) 7 | broadcast( 8 | "chats::#{message.conversation_id}::messages", render_message(message) 9 | ) 10 | end 11 | 12 | def render_message(message) 13 | { 14 | user: message.user_id, avatar: renderer.chat_avatar(message.user).to_s 15 | }.merge(content(message)) 16 | end 17 | 18 | def content(message) 19 | if message.image? 20 | { message: renderer.image_tag(message.image.url), image: true } 21 | else 22 | { message: message.text, image: false } 23 | end 24 | end 25 | 26 | private 27 | 28 | def renderer 29 | ApplicationController.helpers 30 | end 31 | end 32 | end 33 | -------------------------------------------------------------------------------- /app/jobs/chat/notification_relay_job.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Chat 4 | class NotificationRelayJob < ApplicationJob 5 | def perform(message) 6 | user_ids(message).each do |user_id| 7 | broadcast("users::#{user_id}::chats", chat_id: message.conversation_id) 8 | end 9 | end 10 | 11 | private 12 | 13 | def user_ids(message) 14 | message.conversation.users.where.not(id: message.user_id).ids 15 | end 16 | end 17 | end 18 | -------------------------------------------------------------------------------- /app/jobs/chat/status_relay_job.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Chat 4 | class StatusRelayJob < ApplicationJob 5 | def perform(user) 6 | broadcast("chat::status", user_id: user.id, status: user.chat_status) 7 | end 8 | end 9 | end 10 | -------------------------------------------------------------------------------- /app/mailers/chat/application_mailer.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Chat 4 | class ApplicationMailer < ActionMailer::Base 5 | default from: "from@example.com" 6 | layout "mailer" 7 | end 8 | end 9 | -------------------------------------------------------------------------------- /app/models/chat/application_record.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class Chat::ApplicationRecord < ActiveRecord::Base 4 | self.abstract_class = true 5 | end 6 | -------------------------------------------------------------------------------- /app/models/chat/conversation.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class Chat::Conversation < ApplicationRecord 4 | has_many :messages 5 | has_many :sessions, dependent: :destroy 6 | has_many :users, through: :sessions, class_name: "::User" 7 | 8 | has_one :last_message, -> { order(created_at: :desc) }, 9 | class_name: "Chat::Message" 10 | 11 | validates :sessions, presence: { 12 | message: "At least one user must be selected" 13 | } 14 | validate do 15 | next unless Chat::Conversation.joins(:users).where.not(id: id).where( 16 | users: { id: user_ids } 17 | ).group(:id).having("COUNT(DISTINCT users.id) = ?", user_ids.count).exists? 18 | 19 | errors.add(:conversation, "already taking place") 20 | end 21 | end 22 | -------------------------------------------------------------------------------- /app/models/chat/dot_command.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Chat 4 | class DotCommand 5 | attr_reader :text, :message 6 | 7 | def initialize(message, text) 8 | @message = message 9 | @text = text 10 | end 11 | 12 | def command 13 | @command ||= text[/^\.[a-zA-z]*/] 14 | end 15 | 16 | def perform 17 | return false unless valid? 18 | 19 | ("Chat::DotCommand::" + command[1..-1].classify).constantize.new( 20 | message, text[/[^\.[a-zA-z]*].*/]&.squish 21 | ).perform 22 | end 23 | 24 | def valid? 25 | Chat::DotCommand::Validator.new(command).perform 26 | end 27 | end 28 | end 29 | -------------------------------------------------------------------------------- /app/models/chat/dot_command/gif.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Chat 4 | class DotCommand 5 | class Gif 6 | attr_reader :message, :text 7 | 8 | def initialize(message, text) 9 | @message = message 10 | @text = text 11 | end 12 | 13 | def perform 14 | message.image = open(::Giphy.random(text).image_url) 15 | message.text = nil 16 | end 17 | end 18 | end 19 | end 20 | -------------------------------------------------------------------------------- /app/models/chat/dot_command/shrug.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Chat 4 | class DotCommand 5 | class Shrug 6 | attr_reader :message, :text 7 | 8 | def initialize(message, text) 9 | @message = message 10 | @text = text 11 | end 12 | 13 | def perform 14 | message.text = 15 | if text.blank? 16 | shruggie 17 | else 18 | "#{text} #{shruggie}" 19 | end 20 | end 21 | 22 | private 23 | 24 | def shruggie 25 | '¯\_(ツ)_/¯' 26 | end 27 | end 28 | end 29 | end 30 | -------------------------------------------------------------------------------- /app/models/chat/dot_command/validator.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Chat 4 | class DotCommand 5 | class Validator 6 | attr_reader :command 7 | 8 | def initialize(command) 9 | @command = command 10 | end 11 | 12 | def perform 13 | (commands + custom_commands).include? command 14 | end 15 | 16 | def commands 17 | %w(.gif .shrug) 18 | end 19 | 20 | def custom_commands 21 | [] 22 | end 23 | end 24 | end 25 | end 26 | -------------------------------------------------------------------------------- /app/models/chat/message.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class Chat::Message < ApplicationRecord 4 | has_attached_file :image 5 | validates_attachment :image, content_type: { 6 | content_type: ["image/jpg", "image/jpeg", "image/png", "image/gif"] 7 | } 8 | 9 | belongs_to :user, class_name: "::User" 10 | belongs_to :conversation 11 | belongs_to :session 12 | 13 | delegate :name, to: :user 14 | before_save :remove_extra_new_line 15 | before_save :execute_dot_command 16 | 17 | validates :text, presence: true, unless: :image? 18 | 19 | after_create_commit do 20 | Chat::MessageRelayJob.send(Chat.perform_method.to_sym, id) 21 | Chat::NotificationRelayJob.send(Chat.perform_method.to_sym, self) 22 | end 23 | 24 | private 25 | 26 | def execute_dot_command 27 | return unless dot_command.valid? 28 | 29 | dot_command.perform 30 | end 31 | 32 | def dot_command 33 | @dot_command ||= Chat::DotCommand.new(self, text) 34 | end 35 | 36 | def remove_extra_new_line 37 | self.text = text[0..-16] 38 | end 39 | end 40 | -------------------------------------------------------------------------------- /app/models/chat/session.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class Chat::Session < ApplicationRecord 4 | belongs_to :conversation 5 | belongs_to :user, class_name: "::User" 6 | 7 | has_many :messages, dependent: :destroy 8 | 9 | validates :user_id, uniqueness: { scope: :conversation_id } 10 | end 11 | -------------------------------------------------------------------------------- /app/views/chat/_chat.html.haml: -------------------------------------------------------------------------------- 1 | = launch_chat_fab 2 | 3 | #chat 4 | #chat__background 5 | .chat__header 6 | .chat__settings 7 | %div 8 | %div 9 | .chat__flex 10 | %h6 Chat 11 | .chat__flex 12 | = material_icon.close.css_class("chat__close") 13 | = render "chat/conversations/new" 14 | .chat__list 15 | = render "chat/conversations/index" 16 | = link_to nil, class: "chat__start-new" do 17 | = material_icon.add 18 | .current_chat 19 | -------------------------------------------------------------------------------- /app/views/chat/conversations/_conversation.html.haml: -------------------------------------------------------------------------------- 1 | = link_to [chat, conversation], remote: true, id: "chat__show-#{conversation.id}" do 2 | .avatar_container{ class: chat_avatar_count(conversation) } 3 | - if chat_avatar_count(conversation) == "count_default" 4 | = chat_avatars(conversation).sample.html_safe 5 | .chat_count 6 | %span 7 | = conversation_user_count(conversation) 8 | - else 9 | = chat_avatars(conversation).join("").html_safe 10 | .chat__glance 11 | .chat__glance-name-date 12 | .name 13 | - if conversation.users.to_a.size > 2 14 | Group Chat 15 | - else 16 | = conversation.users.where.not(id: current_user.id).first.first_name 17 | .flex 18 | .date 19 | - if conversation.last_message.present? 20 | = time_ago_in_words conversation.last_message&.created_at 21 | ago 22 | .chat__glance-last-message 23 | = raw conversation.last_message&.text 24 | -------------------------------------------------------------------------------- /app/views/chat/conversations/_index.html.haml: -------------------------------------------------------------------------------- 1 | - if chat_list.present? 2 | = render chat_list 3 | - else 4 | .chat__none-available 5 | = material_icon.chat_bubble_outline 6 | %h5 No Chats 7 | -------------------------------------------------------------------------------- /app/views/chat/conversations/_new.html.haml: -------------------------------------------------------------------------------- 1 | = form_for [chat, Chat::Conversation.new], remote: true do |f| 2 | - if chatable_users.present? 3 | = chatable_user_check_boxes(f) 4 | .chat__submit-container= f.submit "Start Chatting" 5 | - else 6 | .chat__none-available 7 | = material_icon.people 8 | %h5 No Chatable Users 9 | -------------------------------------------------------------------------------- /app/views/chat/conversations/_show.html.haml: -------------------------------------------------------------------------------- 1 | .chat__transcript{ data: { channel: 'messages', current: { chat: @conversation.id, user: current_user.id }}} 2 | - @conversation.messages.each_with_index do |msg, i| 3 | = render "chat/messages/message", message: msg, previous: (i > 0 ? @conversation.messages[i-1] : nil) 4 | .chat__flex 5 | %div 6 | = form_for [chat, @conversation, Chat::Message.new], remote: true do |f| 7 | = material_icon.insert_photo.css_class("chat__insert-image") 8 | = f.text_area :text, placeholder: 'Send message...', autocomplete: 'off' 9 | %span.bar 10 | = f.file_field :image 11 | -------------------------------------------------------------------------------- /app/views/chat/conversations/create.js.erb: -------------------------------------------------------------------------------- 1 | <% if @conversation.errors.messages[:sessions].blank? %> 2 | $('#chat .chat__list').html("<%= j render 'chat/conversations/index' %>"); 3 | $('.new_conversation').replaceWith("<%= j render 'chat/conversations/new' %>"); 4 | $('.current_chat').html("<%= j render 'chat/conversations/show' %>"); 5 | $('.new_message textarea').emojioneArea(); 6 | App.chat_message.followCurrentChat(); 7 | $("#new_message").fileupload(); 8 | if($("#chat__show-<%= @conversation.id %>").hasClass('chat__has_new_message')) { 9 | App.chat_notification.decrement(); 10 | $("#chat__show-<%= @conversation.id %>").removeClass('chat__has_new_message') 11 | } 12 | $('.current_chat').toggleClass("slide_left"); 13 | $('.chat__list').addClass("slide_left"); 14 | $('.chat__start-new').addClass("chat__disappear"); 15 | $('.current_chat').scrollTop($('.current_chat').prop('scrollHeight')); 16 | <% else %> 17 | error_msg = 18 | `
19 | <%= j @conversation.errors.values.flatten.join(', ') %> 20 |
`; 21 | if($('.chat__errors').length == 0) { 22 | $('.new_conversation').scrollTop(0); 23 | $('.new_conversation').prepend(error_msg); 24 | } 25 | <% end %> 26 | -------------------------------------------------------------------------------- /app/views/chat/conversations/show.js.erb: -------------------------------------------------------------------------------- 1 | $('.new_conversation').replaceWith("<%= j render 'chat/conversations/new' %>"); 2 | $('.current_chat').html("<%= j render 'chat/conversations/show' %>"); 3 | $('.new_message textarea').emojioneArea(); 4 | App.chat_message.followCurrentChat(); 5 | $("#new_message").fileupload(); 6 | if($("#chat__show-<%= @conversation.id %>").hasClass('chat__has_new_message')) { 7 | App.chat_notification.decrement(); 8 | $("#chat__show-<%= @conversation.id %>").removeClass('chat__has_new_message') 9 | } 10 | $('.current_chat').toggleClass("slide_left"); 11 | $('.chat__list').removeClass("slide_right"); 12 | $('.chat__list').addClass("slide_left"); 13 | $('.chat__start-new').addClass("chat__disappear"); 14 | $('.current_chat').scrollTop($('.current_chat').prop('scrollHeight')); 15 | $(".new_conversation").removeClass("slide_right"); 16 | -------------------------------------------------------------------------------- /app/views/chat/messages/_message.haml: -------------------------------------------------------------------------------- 1 | .message-container{ class: message.user == current_user ? "right" : "left" } 2 | - if message.user == previous&.user 3 | .transcript_placeholder_avatar 4 | - else 5 | = chat_avatar(message.user) 6 | .message{ class: ("image" if message.image?).to_s } 7 | - if message.image? 8 | = image_tag message.image.url 9 | - else 10 | .content= raw(message.text) 11 | -------------------------------------------------------------------------------- /app/views/layouts/chat/application.html.erb: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | Chat 5 | <%= stylesheet_link_tag "chat/application", media: "all" %> 6 | <%= javascript_include_tag "chat/application" %> 7 | <%= csrf_meta_tags %> 8 | 9 | 10 | 11 | <%= yield %> 12 | 13 | 14 | 15 | -------------------------------------------------------------------------------- /bin/rails: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | # frozen_string_literal: true 3 | 4 | # This command will automatically be run when you run "rails" with Rails gems 5 | # installed from the root of your application. 6 | 7 | ENGINE_ROOT = File.expand_path("../..", __FILE__) 8 | ENGINE_PATH = File.expand_path("../../lib/chat/engine", __FILE__) 9 | 10 | # Set up gems listed in the Gemfile. 11 | ENV["BUNDLE_GEMFILE"] ||= File.expand_path("../../Gemfile", __FILE__) 12 | require "bundler/setup" if File.exist?(ENV["BUNDLE_GEMFILE"]) 13 | 14 | require "rails/all" 15 | require "rails/engine/commands" 16 | -------------------------------------------------------------------------------- /bin/test: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | # frozen_string_literal: true 3 | 4 | $: << File.expand_path(File.expand_path("../../test", __FILE__)) 5 | 6 | require "bundler/setup" 7 | require "rails/test_unit/minitest_plugin" 8 | 9 | Rails::TestUnitReporter.executable = "bin/test" 10 | 11 | exit Minitest.run(ARGV) 12 | -------------------------------------------------------------------------------- /chat.gemspec: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | $LOAD_PATH.push File.expand_path("../lib", __FILE__) 4 | 5 | require "chat/version" 6 | 7 | Gem::Specification.new do |s| 8 | s.add_dependency "material_icons" 9 | s.add_dependency "paperclip" 10 | s.add_dependency "jquery-fileupload-rails" 11 | s.add_dependency "rails", "~> 5.1" 12 | s.add_dependency "coffee-rails" 13 | s.add_dependency "jquery-rails" 14 | s.add_dependency "giphy" 15 | s.add_dependency "sidekiq" 16 | 17 | s.name = "chat" 18 | s.version = Chat::VERSION 19 | s.authors = ["npezza93"] 20 | s.email = ["npezza93@gmail.com"] 21 | s.summary = "A simple rails chat gem that leverages ActionCable" 22 | s.homepage = "https://github.com/npezza93/chat" 23 | s.license = "MIT" 24 | 25 | s.files = 26 | Dir["{app,config,db,lib}/**/*", "MIT-LICENSE", "Rakefile", "README.md"] 27 | 28 | s.add_development_dependency "haml" 29 | s.add_development_dependency "pry-rails" 30 | s.add_development_dependency "clearance" 31 | s.add_development_dependency "faker" 32 | s.add_development_dependency "puma" 33 | s.add_development_dependency "sqlite3" 34 | s.add_development_dependency "rubocop" 35 | s.add_development_dependency "turbolinks" 36 | end 37 | -------------------------------------------------------------------------------- /config/routes.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | Chat::Engine.routes.draw do 4 | resources :conversations, only: %i(show create) do 5 | resources :messages, only: %i(create destroy) 6 | end 7 | end 8 | -------------------------------------------------------------------------------- /lib/chat.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require "chat/engine" 4 | require "chat/version" 5 | require "chat/user" 6 | 7 | require "paperclip" 8 | require "material_icons" 9 | require "coffee-rails" 10 | require "jquery-rails" 11 | require "jquery-fileupload-rails" 12 | require "giphy" 13 | require "sidekiq" 14 | 15 | module Chat 16 | # The helper method used to distinguish if a user is logged in 17 | mattr_accessor :signed_in 18 | 19 | # The before_action method to validate users are logged in 20 | mattr_accessor :logged_in_check 21 | 22 | # The field that contains the users avatar 23 | mattr_accessor :user_avatar 24 | 25 | # Can be perform_now or perform_later. Called on jobs 26 | mattr_accessor :perform_method 27 | 28 | def self.table_name_prefix 29 | "chat_" 30 | end 31 | 32 | def self.setup 33 | yield self 34 | end 35 | end 36 | -------------------------------------------------------------------------------- /lib/chat/engine.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Chat 4 | class Engine < ::Rails::Engine 5 | isolate_namespace Chat 6 | end 7 | end 8 | -------------------------------------------------------------------------------- /lib/chat/user.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require "active_support/concern" 4 | 5 | module Chat 6 | module User 7 | extend ActiveSupport::Concern 8 | 9 | included do 10 | has_many :sessions, dependent: :destroy, class_name: "::Chat::Session" 11 | has_many :conversations, through: :sessions, 12 | class_name: "::Chat::Conversation" 13 | 14 | validates :chat_status, inclusion: { in: %w(online offline) } 15 | 16 | after_commit :broadcast_status, 17 | if: proc { |u| u.previous_changes.key?(:chat_status) } 18 | end 19 | 20 | def name 21 | "#{first_name} #{last_name}" 22 | end 23 | 24 | def online 25 | update(chat_status: "online") 26 | end 27 | 28 | def offline 29 | update(chat_status: "offline") 30 | end 31 | 32 | def chat_avatar 33 | send(Chat.user_avatar) 34 | end 35 | 36 | def chat_avatar? 37 | send("#{Chat.user_avatar}?") 38 | end 39 | 40 | private 41 | 42 | def broadcast_status 43 | Chat::StatusRelayJob.send(Chat.perform_method, self) 44 | end 45 | end 46 | end 47 | -------------------------------------------------------------------------------- /lib/chat/version.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module Chat 4 | VERSION = "1.0.0" 5 | end 6 | -------------------------------------------------------------------------------- /lib/generators/chat/install/install_generator.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require "rails/generators/base" 4 | require "rails/generators/active_record" 5 | require "rails/generators/migration" 6 | 7 | module Chat 8 | module Generators 9 | class InstallGenerator < Rails::Generators::Base 10 | include Rails::Generators::Migration 11 | source_root File.expand_path("../templates", __FILE__) 12 | 13 | desc "Adds view helpers, user methods, and migrations for chat" 14 | 15 | def check_user_table 16 | raise UsersTableNotImplemented unless users_table_exists? 17 | end 18 | 19 | def create_chat_initializer 20 | copy_file "chat.rb", "config/initializers/chat.rb" 21 | end 22 | 23 | def add_helper 24 | file = "app/controllers/application_controller.rb" 25 | after = "class ApplicationController < ActionController::Base\n" 26 | inject_into_file file, after: after do 27 | " helper Chat::Engine.helpers\n" 28 | end 29 | end 30 | 31 | def inject_chat_into_user_model 32 | file = "app/models/user.rb" 33 | after = "class User < ApplicationRecord\n" 34 | inject_into_file file, after: after do 35 | " include Chat::User\n" 36 | end 37 | end 38 | 39 | def add_chat_status_migration 40 | migration_template "add_chat_to_users.rb", 41 | "db/migrate/add_chat_to_users.rb" 42 | migration_template "create_chat.rb", "db/migrate/create_chat.rb" 43 | end 44 | 45 | def add_chat_route 46 | route "mount Chat::Engine => \"/chat\", as: \"chat\"" 47 | end 48 | 49 | class UsersTableNotImplemented < StandardError 50 | def initialize(msg = "User model not yet implemented") 51 | super 52 | end 53 | end 54 | 55 | def self.next_migration_number(dirname) 56 | ActiveRecord::Generators::Base.next_migration_number(dirname) 57 | end 58 | 59 | private 60 | 61 | def users_table_exists? 62 | if ActiveRecord::Base.connection.respond_to?(:data_source_exists?) 63 | ActiveRecord::Base.connection.data_source_exists?(:users) 64 | else 65 | ActiveRecord::Base.connection.table_exists?(:users) 66 | end 67 | end 68 | end 69 | end 70 | end 71 | -------------------------------------------------------------------------------- /lib/generators/chat/install/templates/add_chat_to_users.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class AddChatToUsers < ActiveRecord::Migration[5.1] 4 | def self.up 5 | add_column :users, :chat_status, :string, default: "offline" 6 | end 7 | 8 | def self.down 9 | remove_column :users, :chat_status, :string, default: "offline" 10 | end 11 | end 12 | -------------------------------------------------------------------------------- /lib/generators/chat/install/templates/chat.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | Chat.setup do |config| 4 | # Clearance => :signed_in? 5 | # Devise => :user_signed_in? 6 | config.signed_in = :signed_in? 7 | 8 | # Clearance => :require_login 9 | # Devise => :authenticate_user! 10 | config.logged_in_check = :require_login 11 | 12 | config.user_avatar = :image 13 | 14 | config.perform_method = :perform_now 15 | end 16 | -------------------------------------------------------------------------------- /lib/generators/chat/install/templates/create_chat.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class CreateChat < ActiveRecord::Migration[5.1] 4 | def self.up 5 | create_table :chat_conversations do |t| 6 | t.timestamps 7 | end 8 | 9 | create_sessions 10 | create_messages 11 | end 12 | 13 | def self.down 14 | drop_table :chat_conversations 15 | drop_table :chat_sessions 16 | drop_table :chat_messages 17 | end 18 | 19 | private 20 | 21 | def create_sessions 22 | create_table :chat_sessions do |t| 23 | t.belongs_to :conversation 24 | t.belongs_to :user 25 | 26 | t.timestamps 27 | end 28 | end 29 | 30 | def create_messages 31 | create_table :chat_messages do |t| 32 | t.belongs_to :user 33 | t.text :text 34 | t.belongs_to :conversation 35 | t.belongs_to :session 36 | t.attachment :image 37 | 38 | t.timestamps 39 | end 40 | end 41 | end 42 | -------------------------------------------------------------------------------- /lib/tasks/chat_tasks.rake: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | # desc "Explaining what the task does" 3 | # task :chat do 4 | # # Task goes here 5 | # end 6 | -------------------------------------------------------------------------------- /test/chat_test.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require "test_helper" 4 | 5 | class Chat::Test < ActiveSupport::TestCase 6 | test "truth" do 7 | assert_kind_of Chat, Module 8 | end 9 | end 10 | -------------------------------------------------------------------------------- /test/dummy/Rakefile: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | # Add your own tasks in files placed in lib/tasks ending in .rake, 4 | # for example lib/tasks/capistrano.rake, and they will automatically be 5 | # available to Rake. 6 | 7 | require_relative "config/application" 8 | 9 | Rails.application.load_tasks 10 | -------------------------------------------------------------------------------- /test/dummy/app/assets/config/manifest.js: -------------------------------------------------------------------------------- 1 | 2 | //= link_tree ../images 3 | //= link_directory ../javascripts .js 4 | //= link_directory ../stylesheets .css 5 | -------------------------------------------------------------------------------- /test/dummy/app/assets/images/.keep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/npezza93/chat/c8462876bd068532f6c4ebcf1d32570ea60f1dec/test/dummy/app/assets/images/.keep -------------------------------------------------------------------------------- /test/dummy/app/assets/javascripts/application.js: -------------------------------------------------------------------------------- 1 | // This is a manifest file that'll be compiled into application.js, which will include all the files 2 | // listed below. 3 | // 4 | // Any JavaScript/Coffee file within this directory, lib/assets/javascripts, vendor/assets/javascripts, 5 | // or any plugin's vendor/assets/javascripts directory can be referenced here using a relative path. 6 | // 7 | // It's not advisable to add code directly here, but if you do, it'll appear at the bottom of the 8 | // compiled file. JavaScript code in this file should be added after the last require_* statement. 9 | // 10 | // Read Sprockets README (https://github.com/rails/sprockets#sprockets-directives) for details 11 | // about supported directives. 12 | // 13 | //= require jquery 14 | //= require jquery_ujs 15 | //= require turbolinks 16 | //= require chat 17 | //= require_tree . 18 | -------------------------------------------------------------------------------- /test/dummy/app/assets/javascripts/cable.js: -------------------------------------------------------------------------------- 1 | // Action Cable provides the framework to deal with WebSockets in Rails. 2 | // You can generate new channels where WebSocket features live using the rails generate channel command. 3 | // 4 | //= require action_cable 5 | //= require_self 6 | //= require_tree ./channels 7 | 8 | (function() { 9 | this.App || (this.App = {}); 10 | 11 | App.cable = ActionCable.createConsumer(); 12 | 13 | }).call(this); 14 | -------------------------------------------------------------------------------- /test/dummy/app/assets/javascripts/channels/.keep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/npezza93/chat/c8462876bd068532f6c4ebcf1d32570ea60f1dec/test/dummy/app/assets/javascripts/channels/.keep -------------------------------------------------------------------------------- /test/dummy/app/assets/javascripts/material_design_components.js: -------------------------------------------------------------------------------- 1 | /*global mdc $:true*/ 2 | 3 | $(document).on("turbolinks:load", function() { 4 | initialize_mdc(); 5 | }); 6 | 7 | function initialize_mdc() { 8 | mdc.autoInit(); 9 | 10 | $.each($(".mdc-textfield"), function(index, element) { 11 | mdc.textfield.MDCTextfield.attachTo(element); 12 | }); 13 | $.each($(".mdc-fab, .mdc-button, .mdc-ripple-surface, .mdc-list-item.with-ripple"), function(index, element) { 14 | mdc.ripple.MDCRipple.attachTo(element); 15 | }); 16 | $.each($(".mdc-select"), function(index, element) { 17 | mdc.select.MDCSelect.attachTo(element).listen('MDCSelect:change', function(e) { 18 | $(this).find("input[type='hidden']").val(e.detail.value); 19 | }); 20 | }); 21 | $.each($(".mdc-simple-menu"), function(index, element) { 22 | var menu = mdc.menu.MDCSimpleMenu.attachTo(element); 23 | $(element).parent().find(".menu-toggle").click(function() { 24 | menu.open = !menu.open; 25 | }); 26 | }); 27 | 28 | if ($(".mdc-snackbar").length > 0) { 29 | var snackbar = mdc.snackbar.MDCSnackbar.attachTo($(".mdc-snackbar")[0]); 30 | snackbar.show({ message: $(".mdc-snackbar").find(".mdc-snackbar__text").text() }); 31 | } 32 | } 33 | -------------------------------------------------------------------------------- /test/dummy/app/assets/stylesheets/application.css: -------------------------------------------------------------------------------- 1 | /* 2 | * This is a manifest file that'll be compiled into application.css, which will include all the files 3 | * listed below. 4 | * 5 | * Any CSS and SCSS file within this directory, lib/assets/stylesheets, vendor/assets/stylesheets, 6 | * or any plugin's vendor/assets/stylesheets directory can be referenced here using a relative path. 7 | * 8 | * You're free to add application-wide styles to this file and they'll appear at the bottom of the 9 | * compiled file so the styles you add here take precedence over styles defined in any other CSS/SCSS 10 | * files in this directory. Styles in this file should be added after the last require_* statement. 11 | * It is generally better to create a new file per style scope. 12 | * 13 | *= require_tree . 14 | *= require_self 15 | *= require chat 16 | */ 17 | -------------------------------------------------------------------------------- /test/dummy/app/assets/stylesheets/home.sass: -------------------------------------------------------------------------------- 1 | @import "https://fonts.googleapis.com/css?family=Roboto:300,400,500" 2 | 3 | body, html 4 | height: 100% 5 | width: 100% 6 | margin: 0 7 | background-image: linear-gradient( 135deg, #5EFCE8 0%, #736EFE 100%) 8 | a 9 | color: var(--mdc-theme-text-primary-on-primary,#fff) 10 | text-decoration: none 11 | 12 | #homeContainer 13 | width: 100% 14 | height: 100% 15 | display: flex 16 | flex-direction: column 17 | align-items: center 18 | justify-content: center 19 | a 20 | font-weight: 300 21 | margin-top: 24px 22 | -------------------------------------------------------------------------------- /test/dummy/app/assets/stylesheets/sessions.sass: -------------------------------------------------------------------------------- 1 | @import "https://fonts.googleapis.com/css?family=Lato:300,400,700,900" 2 | 3 | \:root 4 | --mdc-theme-primary: #FAFAFA 5 | 6 | h1 7 | font-weight: 400 8 | .link-back 9 | position: absolute 10 | top: 0 11 | left: 0 12 | margin: 10px 13 | color: var(--mdc-theme-text-primary-on-primary,#fff) 14 | .sign_up_links 15 | margin-top: 30px 16 | #signed_in_user 17 | color: var(--mdc-theme-text-primary-on-primary,#fff) 18 | padding: 10px 19 | display: flex 20 | flex-direction: row 21 | align-items: flex-end 22 | position: absolute 23 | top: 0 24 | right: 0 25 | span 26 | margin: 0 10px 27 | .container 28 | display: flex 29 | flex-direction: column 30 | align-items: center 31 | justify-content: center 32 | height: 100% 33 | 34 | a 35 | margin-top: 32px 36 | 37 | form 38 | display: flex 39 | flex-direction: column 40 | 41 | button 42 | display: flex 43 | flex-direction: row 44 | justify-content: center 45 | label 46 | color: var(--mdc-theme-text-hint-on-dark, rgba(0, 0, 0, 0.5)) 47 | input[type="file"] 48 | margin: 10px 0 49 | color: var(--mdc-theme-text-primary-on-primary,#fff) 50 | -------------------------------------------------------------------------------- /test/dummy/app/channels/application_cable/channel.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module ApplicationCable 4 | class Channel < ActionCable::Channel::Base 5 | end 6 | end 7 | -------------------------------------------------------------------------------- /test/dummy/app/channels/application_cable/connection.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module ApplicationCable 4 | class Connection < ActionCable::Connection::Base 5 | identified_by :current_user 6 | 7 | def connect 8 | self.current_user = find_verified_user 9 | end 10 | 11 | protected 12 | 13 | def find_verified_user 14 | if (current_user = User.find_by(remember_token: token)) 15 | current_user 16 | else 17 | reject_unauthorized_connection 18 | end 19 | end 20 | 21 | def token 22 | request.cookie_jar[:remember_token] 23 | end 24 | end 25 | end 26 | -------------------------------------------------------------------------------- /test/dummy/app/controllers/application_controller.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class ApplicationController < ActionController::Base 4 | helper Chat::Engine.helpers 5 | include Clearance::Controller 6 | protect_from_forgery with: :exception 7 | end 8 | -------------------------------------------------------------------------------- /test/dummy/app/controllers/concerns/.keep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/npezza93/chat/c8462876bd068532f6c4ebcf1d32570ea60f1dec/test/dummy/app/controllers/concerns/.keep -------------------------------------------------------------------------------- /test/dummy/app/controllers/home_controller.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class HomeController < ApplicationController 4 | def index 5 | end 6 | end 7 | -------------------------------------------------------------------------------- /test/dummy/app/controllers/users_controller.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class UsersController < ApplicationController 4 | before_action :require_login 5 | before_action :set_user, only: %i(edit update) 6 | 7 | def edit 8 | end 9 | 10 | def update 11 | if @user.update(user_params) 12 | redirect_to root_path, notice: "User was successfully updated." 13 | else 14 | render :edit 15 | end 16 | end 17 | 18 | private 19 | 20 | def set_user 21 | @user = current_user 22 | end 23 | 24 | def user_params 25 | params.require(:user).permit(:first, :last, :image, :email, :password) 26 | end 27 | end 28 | -------------------------------------------------------------------------------- /test/dummy/app/helpers/application_helper.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module ApplicationHelper 4 | end 5 | -------------------------------------------------------------------------------- /test/dummy/app/helpers/snackbar_helper.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module SnackbarHelper 4 | def snackbar_notice(message, opts = nil) 5 | return if message.blank? 6 | 7 | content_tag( 8 | :div, snackbar_text(message), 9 | class: "mdc-snackbar #{opts}", 10 | aria: { live: :assertive, atomic: "", hidden: "" } 11 | ) 12 | end 13 | 14 | def snackbar_text(message) 15 | content_tag(:div, message, class: "mdc-snackbar__text") + snackbar_button 16 | end 17 | 18 | def snackbar_button 19 | content_tag(:div, class: "mdc-snackbar__action-wrapper") do 20 | content_tag( 21 | :button, nil, 22 | class: "mdc-button mdc-snackbar__action-button", type: :button 23 | ) 24 | end.html_safe 25 | end 26 | end 27 | -------------------------------------------------------------------------------- /test/dummy/app/jobs/application_job.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class ApplicationJob < ActiveJob::Base 4 | end 5 | -------------------------------------------------------------------------------- /test/dummy/app/mailers/application_mailer.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class ApplicationMailer < ActionMailer::Base 4 | default from: "from@example.com" 5 | layout "mailer" 6 | end 7 | -------------------------------------------------------------------------------- /test/dummy/app/models/application_record.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class ApplicationRecord < ActiveRecord::Base 4 | self.abstract_class = true 5 | end 6 | -------------------------------------------------------------------------------- /test/dummy/app/models/concerns/.keep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/npezza93/chat/c8462876bd068532f6c4ebcf1d32570ea60f1dec/test/dummy/app/models/concerns/.keep -------------------------------------------------------------------------------- /test/dummy/app/models/user.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class User < ApplicationRecord 4 | include Chat::User 5 | include Clearance::User 6 | 7 | has_attached_file :image 8 | validates_attachment :image, content_type: { 9 | content_type: ["image/jpg", "image/jpeg", "image/png", "image/gif"] 10 | } 11 | end 12 | -------------------------------------------------------------------------------- /test/dummy/app/views/clearance/users/_form.html.haml: -------------------------------------------------------------------------------- 1 | .mdc-textfield 2 | = f.email_field :email, autofocus: true, 3 | class: "mdc-textfield__input mdc-theme--text-primary-on-primary", 4 | required: true 5 | = f.label :email, "Email", class: "mdc-textfield__label" 6 | 7 | .mdc-textfield 8 | = f.password_field :password, 9 | class: "mdc-textfield__input mdc-theme--text-primary-on-primary", 10 | required: true 11 | = f.label :password, "Password", class: "mdc-textfield__label" 12 | -------------------------------------------------------------------------------- /test/dummy/app/views/home/index.html.haml: -------------------------------------------------------------------------------- 1 | #homeContainer.mdc-theme--text-primary-on-primary 2 | %h1 Chat 3 | - unless signed_in? 4 | = link_to "Sign In", sign_in_path, class: "mdc-button mdc-button--primary" 5 | 6 | 7 | = render_chat 8 | -------------------------------------------------------------------------------- /test/dummy/app/views/layouts/application.html.haml: -------------------------------------------------------------------------------- 1 | !!! 2 | %html 3 | %head 4 | %title Dummy 5 | = csrf_meta_tags 6 | = stylesheet_link_tag 'application', 7 | media: 'all', 'data-turbolinks-track': 'reload' 8 | = javascript_include_tag 'application', 9 | 'data-turbolinks-track': 'reload' 10 | = action_cable_meta_tag 11 | 12 | %meta{ content: "width=device-width, initial-scale=1.0", name: "viewport"}/ 13 | %body.mdc-typography 14 | = snackbar_notice(notice || alert) 15 | 16 | - if signed_in? 17 | #signed_in_user 18 | = link_to 'Settings', edit_users_path 19 | %span= "|" 20 | = link_to 'Sign out', sign_out_path, method: :delete 21 | 22 | = yield 23 | -------------------------------------------------------------------------------- /test/dummy/app/views/layouts/mailer.html.erb: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 8 | 9 | 10 | 11 | <%= yield %> 12 | 13 | 14 | -------------------------------------------------------------------------------- /test/dummy/app/views/layouts/mailer.text.erb: -------------------------------------------------------------------------------- 1 | <%= yield %> 2 | -------------------------------------------------------------------------------- /test/dummy/app/views/sessions/new.html.haml: -------------------------------------------------------------------------------- 1 | = link_to root_path, class: "link-back" do 2 | = material_icon.arrow_back 3 | 4 | .container 5 | %h1.mdc-theme--text-primary-on-primary= t('.title') 6 | = form_for :session, url: session_path do |f| 7 | .mdc-textfield 8 | = f.email_field :email, autofocus: true, 9 | class: "mdc-textfield__input mdc-theme--text-primary-on-primary", 10 | required: true 11 | = f.label :email, "Email", class: "mdc-textfield__label" 12 | 13 | .mdc-textfield 14 | = f.password_field :password, 15 | class: "mdc-textfield__input mdc-theme--text-primary-on-primary", 16 | required: true 17 | = f.label :password, "Password", class: "mdc-textfield__label" 18 | 19 | %button.mdc-button.mdc-button--primary{ type: :submit } Sign In 20 | 21 | - if Clearance.configuration.allow_sign_up? 22 | = link_to t('.sign_up'), sign_up_path, 23 | class: "mdc-button mdc-button--primary" 24 | -------------------------------------------------------------------------------- /test/dummy/app/views/users/edit.html.haml: -------------------------------------------------------------------------------- 1 | = link_to root_path, class: "link-back" do 2 | = material_icon.arrow_back 3 | 4 | .container 5 | %h1.mdc-theme--text-primary-on-primary Settings 6 | = form_for @user, url: users_path, method: :put do |f| 7 | = render 'clearance/users/form', f: f 8 | 9 | .mdc-textfield 10 | = f.text_field :first_name, 11 | class: "mdc-textfield__input mdc-theme--text-primary-on-primary", 12 | required: true 13 | = f.label :first_name, "First Name", class: "mdc-textfield__label" 14 | 15 | .mdc-textfield 16 | = f.text_field :last_name, 17 | class: "mdc-textfield__input mdc-theme--text-primary-on-primary", 18 | required: true 19 | = f.label :last_name, "Last Name", class: "mdc-textfield__label" 20 | 21 | = f.file_field :image, placeholder: 'Image' 22 | 23 | %button.mdc-button.mdc-button--primary{ type: :submit } Save 24 | -------------------------------------------------------------------------------- /test/dummy/app/views/users/new.html.haml: -------------------------------------------------------------------------------- 1 | = link_to sign_in_path, class: "link-back" do 2 | = material_icon.arrow_back 3 | 4 | .container 5 | %h1.mdc-theme--text-primary-on-primary= t('.title') 6 | = form_for @user do |f| 7 | = render 'clearance/users/form', f: f 8 | 9 | %button.mdc-button.mdc-button--primary{ type: :submit } Sign Up 10 | -------------------------------------------------------------------------------- /test/dummy/bin/bundle: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | # frozen_string_literal: true 3 | 4 | ENV["BUNDLE_GEMFILE"] ||= File.expand_path("../../Gemfile", __FILE__) 5 | load Gem.bin_path("bundler", "bundle") 6 | -------------------------------------------------------------------------------- /test/dummy/bin/rails: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | # frozen_string_literal: true 3 | 4 | APP_PATH = File.expand_path("../config/application", __dir__) 5 | require_relative "../config/boot" 6 | require "rails/commands" 7 | -------------------------------------------------------------------------------- /test/dummy/bin/rake: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | # frozen_string_literal: true 3 | 4 | require_relative "../config/boot" 5 | require "rake" 6 | Rake.application.run 7 | -------------------------------------------------------------------------------- /test/dummy/bin/setup: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | # frozen_string_literal: true 3 | 4 | require "pathname" 5 | require "fileutils" 6 | include FileUtils 7 | 8 | # path to your application root. 9 | APP_ROOT = Pathname.new File.expand_path("../../", __FILE__) 10 | 11 | def system!(*args) 12 | system(*args) || abort("\n== Command #{args} failed ==") 13 | end 14 | 15 | chdir APP_ROOT do 16 | # This script is a starting point to setup your application. 17 | # Add necessary setup steps to this file. 18 | 19 | puts "== Installing dependencies ==" 20 | system! "gem install bundler --conservative" 21 | system("bundle check") || system!("bundle install") 22 | 23 | # puts "\n== Copying sample files ==" 24 | # unless File.exist?('config/database.yml') 25 | # cp 'config/database.yml.sample', 'config/database.yml' 26 | # end 27 | 28 | puts "\n== Preparing database ==" 29 | system! "bin/rails db:setup" 30 | 31 | puts "\n== Removing old logs and tempfiles ==" 32 | system! "bin/rails log:clear tmp:clear" 33 | 34 | puts "\n== Restarting application server ==" 35 | system! "bin/rails restart" 36 | end 37 | -------------------------------------------------------------------------------- /test/dummy/bin/update: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | # frozen_string_literal: true 3 | 4 | require "pathname" 5 | require "fileutils" 6 | include FileUtils 7 | 8 | # path to your application root. 9 | APP_ROOT = Pathname.new File.expand_path("../../", __FILE__) 10 | 11 | def system!(*args) 12 | system(*args) || abort("\n== Command #{args} failed ==") 13 | end 14 | 15 | chdir APP_ROOT do 16 | # This script is a way to update your development environment automatically. 17 | # Add necessary update steps to this file. 18 | 19 | puts "== Installing dependencies ==" 20 | system! "gem install bundler --conservative" 21 | system("bundle check") || system!("bundle install") 22 | 23 | puts "\n== Updating database ==" 24 | system! "bin/rails db:migrate" 25 | 26 | puts "\n== Removing old logs and tempfiles ==" 27 | system! "bin/rails log:clear tmp:clear" 28 | 29 | puts "\n== Restarting application server ==" 30 | system! "bin/rails restart" 31 | end 32 | -------------------------------------------------------------------------------- /test/dummy/config.ru: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | # This file is used by Rack-based servers to start the application. 4 | 5 | require_relative "config/environment" 6 | 7 | run Rails.application 8 | -------------------------------------------------------------------------------- /test/dummy/config/application.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require_relative "boot" 4 | 5 | require "rails/all" 6 | 7 | Bundler.require(*Rails.groups) 8 | require "chat" 9 | 10 | module Dummy 11 | class Application < Rails::Application 12 | # Settings in config/environments/* take precedence over those specified 13 | # here. 14 | # Application configuration should go into files in config/initializers 15 | # -- all .rb files in that directory are automatically loaded. 16 | config.active_job.queue_adapter = :sidekiq 17 | end 18 | end 19 | -------------------------------------------------------------------------------- /test/dummy/config/boot.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | # Set up gems listed in the Gemfile. 4 | ENV["BUNDLE_GEMFILE"] ||= File.expand_path("../../../Gemfile", __dir__) 5 | 6 | require "bundler/setup" if File.exist?(ENV["BUNDLE_GEMFILE"]) 7 | $LOAD_PATH.unshift File.expand_path("../../../lib", __dir__) 8 | -------------------------------------------------------------------------------- /test/dummy/config/cable.yml: -------------------------------------------------------------------------------- 1 | default: &default 2 | adapter: redis 3 | url: redis://localhost:6379/1 4 | 5 | development: 6 | <<: *default 7 | 8 | test: 9 | <<: *default 10 | 11 | production: 12 | <<: *default 13 | -------------------------------------------------------------------------------- /test/dummy/config/database.yml: -------------------------------------------------------------------------------- 1 | # SQLite version 3.x 2 | # gem install sqlite3 3 | # 4 | # Ensure the SQLite 3 gem is defined in your Gemfile 5 | # gem 'sqlite3' 6 | # 7 | default: &default 8 | adapter: sqlite3 9 | pool: 5 10 | timeout: 5000 11 | 12 | development: 13 | <<: *default 14 | database: db/development.sqlite3 15 | 16 | # Warning: The database defined as "test" will be erased and 17 | # re-generated from your development database when you run "rake". 18 | # Do not set this db to the same as development or production. 19 | test: 20 | <<: *default 21 | database: db/test.sqlite3 22 | 23 | production: 24 | <<: *default 25 | database: db/production.sqlite3 26 | -------------------------------------------------------------------------------- /test/dummy/config/environment.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | # Load the Rails application. 4 | require_relative "application" 5 | 6 | # Initialize the Rails application. 7 | Rails.application.initialize! 8 | -------------------------------------------------------------------------------- /test/dummy/config/environments/development.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | Rails.application.configure do 4 | # Settings specified here will take precedence over those in 5 | # config/application.rb. 6 | 7 | # In the development environment your application's code is reloaded on 8 | # every request. This slows down response time but is perfect for development 9 | # since you don't have to restart the web server when you make code changes. 10 | config.cache_classes = false 11 | 12 | # Do not eager load code on boot. 13 | config.eager_load = false 14 | 15 | # Show full error reports. 16 | config.consider_all_requests_local = true 17 | 18 | # Enable/disable caching. By default caching is disabled. 19 | if Rails.root.join("tmp/caching-dev.txt").exist? 20 | config.action_controller.perform_caching = true 21 | 22 | config.cache_store = :memory_store 23 | config.public_file_server.headers = { 24 | "Cache-Control" => "public, max-age=172800" 25 | } 26 | else 27 | config.action_controller.perform_caching = false 28 | 29 | config.cache_store = :null_store 30 | end 31 | 32 | # Don't care if the mailer can't send. 33 | config.action_mailer.raise_delivery_errors = false 34 | 35 | config.action_mailer.perform_caching = false 36 | 37 | # Print deprecation notices to the Rails logger. 38 | config.active_support.deprecation = :log 39 | 40 | # Raise an error on page load if there are pending migrations. 41 | config.active_record.migration_error = :page_load 42 | 43 | # Debug mode disables concatenation and preprocessing of assets. 44 | # This option may cause significant delays in view rendering with a large 45 | # number of complex assets. 46 | config.assets.debug = true 47 | 48 | # Suppress logger output for asset requests. 49 | config.assets.quiet = true 50 | 51 | # Raises error for missing translations 52 | # config.action_view.raise_on_missing_translations = true 53 | 54 | # Use an evented file watcher to asynchronously detect changes in source code, 55 | # routes, locales, etc. This feature depends on the listen gem. 56 | # config.file_watcher = ActiveSupport::EventedFileUpdateChecker 57 | end 58 | -------------------------------------------------------------------------------- /test/dummy/config/environments/production.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | Rails.application.configure do 4 | # Settings specified here will take precedence over those in 5 | # config/application.rb. 6 | 7 | # Code is not reloaded between requests. 8 | config.cache_classes = true 9 | 10 | # Eager load code on boot. This eager loads most of Rails and 11 | # your application in memory, allowing both threaded web servers 12 | # and those relying on copy on write to perform better. 13 | # Rake tasks automatically ignore this option for performance. 14 | config.eager_load = true 15 | 16 | # Full error reports are disabled and caching is turned on. 17 | config.consider_all_requests_local = false 18 | config.action_controller.perform_caching = true 19 | 20 | # Disable serving static files from the `/public` folder by default since 21 | # Apache or NGINX already handles this. 22 | config.public_file_server.enabled = ENV["RAILS_SERVE_STATIC_FILES"].present? 23 | 24 | # Compress JavaScripts and CSS. 25 | config.assets.js_compressor = :uglifier 26 | # config.assets.css_compressor = :sass 27 | 28 | # Do not fallback to assets pipeline if a precompiled asset is missed. 29 | config.assets.compile = false 30 | 31 | # `config.assets.precompile` and `config.assets.version` have moved to 32 | # config/initializers/assets.rb 33 | 34 | # Enable serving of images, stylesheets, and JavaScripts from an asset server. 35 | # config.action_controller.asset_host = 'http://assets.example.com' 36 | 37 | # Specifies the header that your server uses for sending files. 38 | # config.action_dispatch.x_sendfile_header = 'X-Sendfile' # for Apache 39 | # config.action_dispatch.x_sendfile_header = 'X-Accel-Redirect' # for NGINX 40 | 41 | # Mount Action Cable outside main process or domain 42 | # config.action_cable.mount_path = nil 43 | # config.action_cable.url = 'wss://example.com/cable' 44 | # config.action_cable.allowed_request_origins = [ 'http://example.com', 45 | # /http:\/\/example.*/ ] 46 | 47 | # Force all access to the app over SSL, use Strict-Transport-Security, 48 | # and use secure cookies. 49 | # config.force_ssl = true 50 | 51 | # Use the lowest log level to ensure availability of diagnostic information 52 | # when problems arise. 53 | config.log_level = :debug 54 | 55 | # Prepend all log lines with the following tags. 56 | config.log_tags = [ :request_id ] 57 | 58 | # Use a different cache store in production. 59 | # config.cache_store = :mem_cache_store 60 | 61 | # Use a real queuing backend for Active Job 62 | # (and separate queues per environment) 63 | # config.active_job.queue_adapter = :resque 64 | # config.active_job.queue_name_prefix = "dummy_#{Rails.env}" 65 | config.action_mailer.perform_caching = false 66 | 67 | # Ignore bad email addresses and do not raise email delivery errors. 68 | # Set this to true and configure the email server for immediate delivery to 69 | # raise delivery errors. 70 | # config.action_mailer.raise_delivery_errors = false 71 | 72 | # Enable locale fallbacks for I18n (makes lookups for any locale fall back to 73 | # the I18n.default_locale when a translation cannot be found). 74 | config.i18n.fallbacks = true 75 | 76 | # Send deprecation notices to registered listeners. 77 | config.active_support.deprecation = :notify 78 | 79 | # Use default logging formatter so that PID and timestamp are not suppressed. 80 | config.log_formatter = ::Logger::Formatter.new 81 | 82 | # Use a different logger for distributed setups. 83 | # require 'syslog/logger' 84 | # config.logger = 85 | # ActiveSupport::TaggedLogging.new(Syslog::Logger.new 'app-name') 86 | 87 | if ENV["RAILS_LOG_TO_STDOUT"].present? 88 | logger = ActiveSupport::Logger.new(STDOUT) 89 | logger.formatter = config.log_formatter 90 | config.logger = ActiveSupport::TaggedLogging.new(logger) 91 | end 92 | 93 | # Do not dump schema after migrations. 94 | config.active_record.dump_schema_after_migration = false 95 | end 96 | -------------------------------------------------------------------------------- /test/dummy/config/environments/test.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | Rails.application.configure do 4 | # Settings specified here will take precedence over those in 5 | # config/application.rb. 6 | 7 | # The test environment is used exclusively to run your application's 8 | # test suite. You never need to work with it otherwise. Remember that 9 | # your test database is "scratch space" for the test suite and is wiped 10 | # and recreated between test runs. Don't rely on the data there! 11 | config.cache_classes = true 12 | 13 | # Do not eager load code on boot. This avoids loading your whole application 14 | # just for the purpose of running a single test. If you are using a tool that 15 | # preloads Rails for running tests, you may have to set it to true. 16 | config.eager_load = false 17 | 18 | # Configure public file server for tests with Cache-Control for performance. 19 | config.public_file_server.enabled = true 20 | config.public_file_server.headers = { 21 | "Cache-Control" => "public, max-age=3600" 22 | } 23 | 24 | # Show full error reports and disable caching. 25 | config.consider_all_requests_local = true 26 | config.action_controller.perform_caching = false 27 | 28 | # Raise exceptions instead of rendering exception templates. 29 | config.action_dispatch.show_exceptions = false 30 | 31 | # Disable request forgery protection in test environment. 32 | config.action_controller.allow_forgery_protection = false 33 | config.action_mailer.perform_caching = false 34 | 35 | # Tell Action Mailer not to deliver emails to the real world. 36 | # The :test delivery method accumulates sent emails in the 37 | # ActionMailer::Base.deliveries array. 38 | config.action_mailer.delivery_method = :test 39 | 40 | # Print deprecation notices to the stderr. 41 | config.active_support.deprecation = :stderr 42 | 43 | # Raises error for missing translations 44 | # config.action_view.raise_on_missing_translations = true 45 | end 46 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/application_controller_renderer.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | # Be sure to restart your server when you modify this file. 3 | 4 | # ApplicationController.renderer.defaults.merge!( 5 | # http_host: 'example.org', 6 | # https: false 7 | # ) 8 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/assets.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | # Be sure to restart your server when you modify this file. 4 | 5 | # Version of your assets, change this if you want to expire all your assets. 6 | Rails.application.config.assets.version = "1.0" 7 | 8 | # Add additional assets to the asset load path 9 | # Rails.application.config.assets.paths << Emoji.images_path 10 | 11 | # Precompile additional assets. 12 | # application.js, application.css, and all non-JS/CSS in app/assets folder are 13 | # already added. 14 | # Rails.application.config.assets.precompile += %w( search.js ) 15 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/backtrace_silencers.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | # Be sure to restart your server when you modify this file. 3 | 4 | # You can add backtrace silencers for libraries that you're using but don't wish 5 | # to see in your backtraces. 6 | # Rails.backtrace_cleaner.add_silencer { |line| line =~ /my_noisy_library/ } 7 | 8 | # You can also remove all the silencers if you're trying to debug a problem that 9 | # might stem from framework code. 10 | # Rails.backtrace_cleaner.remove_silencers! 11 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/chat.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | Chat.setup do |config| 4 | # Clearance => :signed_in? 5 | # Devise => :user_signed_in? 6 | config.signed_in = :signed_in? 7 | 8 | # Clearance => :require_login 9 | # Devise => :authenticate_user! 10 | config.logged_in_check = :require_login 11 | 12 | config.user_avatar = :image 13 | 14 | config.perform_method = :perform_now 15 | end 16 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/clearance.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | Clearance.configure do |config| 4 | config.mailer_sender = "reply@example.com" 5 | end 6 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/cookies_serializer.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | # Be sure to restart your server when you modify this file. 4 | 5 | # Specify a serializer for the signed and encrypted cookie jars. 6 | # Valid options are :json, :marshal, and :hybrid. 7 | Rails.application.config.action_dispatch.cookies_serializer = :json 8 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/filter_parameter_logging.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | # Be sure to restart your server when you modify this file. 4 | 5 | # Configure sensitive parameters which will be filtered from the log file. 6 | Rails.application.config.filter_parameters += [:password] 7 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/inflections.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | # Be sure to restart your server when you modify this file. 3 | 4 | # Add new inflection rules using the following format. Inflections 5 | # are locale specific, and you may define rules for as many different 6 | # locales as you wish. All of these examples are active by default: 7 | # ActiveSupport::Inflector.inflections(:en) do |inflect| 8 | # inflect.plural /^(ox)$/i, '\1en' 9 | # inflect.singular /^(ox)en/i, '\1' 10 | # inflect.irregular 'person', 'people' 11 | # inflect.uncountable %w( fish sheep ) 12 | # end 13 | 14 | # These inflection rules are supported but not enabled by default: 15 | # ActiveSupport::Inflector.inflections(:en) do |inflect| 16 | # inflect.acronym 'RESTful' 17 | # end 18 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/mime_types.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | # Be sure to restart your server when you modify this file. 3 | 4 | # Add new mime types for use in respond_to blocks: 5 | # Mime::Type.register "text/richtext", :rtf 6 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/new_framework_defaults.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | # Be sure to restart your server when you modify this file. 4 | # 5 | # This file contains migration options to ease your Rails 5.0 upgrade. 6 | # 7 | # Read the Rails 5.0 release notes for more info on each option. 8 | 9 | # Enable per-form CSRF tokens. Previous versions had false. 10 | Rails.application.config.action_controller.per_form_csrf_tokens = true 11 | 12 | # Enable origin-checking CSRF mitigation. Previous versions had false. 13 | Rails.application.config.action_controller.forgery_protection_origin_check = 14 | true 15 | 16 | # Make Ruby 2.4 preserve the timezone of the receiver when calling `to_time`. 17 | # Previous versions had false. 18 | ActiveSupport.to_time_preserves_timezone = true 19 | 20 | # Require `belongs_to` associations by default. Previous versions had false. 21 | Rails.application.config.active_record.belongs_to_required_by_default = true 22 | 23 | # Do not halt callback chains when a callback returns false. Previous versions 24 | # had true. 25 | ActiveSupport.halt_callback_chains_on_return_false = false 26 | 27 | # Configure SSL options to enable HSTS with subdomains. Previous versions had 28 | # false. 29 | Rails.application.config.ssl_options = { hsts: { subdomains: true } } 30 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/session_store.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | # Be sure to restart your server when you modify this file. 4 | 5 | Rails.application.config.session_store :cookie_store, key: "_dummy_session" 6 | -------------------------------------------------------------------------------- /test/dummy/config/initializers/wrap_parameters.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | # Be sure to restart your server when you modify this file. 4 | 5 | # This file contains settings for ActionController::ParamsWrapper which 6 | # is enabled by default. 7 | 8 | # Enable parameter wrapping for JSON. You can disable this by setting :format 9 | # to an empty array. 10 | ActiveSupport.on_load(:action_controller) do 11 | wrap_parameters format: [:json] 12 | end 13 | 14 | # To enable root element in JSON for ActiveRecord objects. 15 | # ActiveSupport.on_load(:active_record) do 16 | # self.include_root_in_json = true 17 | # end 18 | -------------------------------------------------------------------------------- /test/dummy/config/locales/en.yml: -------------------------------------------------------------------------------- 1 | # Files in the config/locales directory are used for internationalization 2 | # and are automatically loaded by Rails. If you want to use locales other 3 | # than English, add the necessary files in this directory. 4 | # 5 | # To use the locales, use `I18n.t`: 6 | # 7 | # I18n.t 'hello' 8 | # 9 | # In views, this is aliased to just `t`: 10 | # 11 | # <%= t('hello') %> 12 | # 13 | # To use a different locale, set it with `I18n.locale`: 14 | # 15 | # I18n.locale = :es 16 | # 17 | # This would use the information in config/locales/es.yml. 18 | # 19 | # To learn more, please read the Rails Internationalization guide 20 | # available at http://guides.rubyonrails.org/i18n.html. 21 | 22 | en: 23 | hello: "Hello world" 24 | -------------------------------------------------------------------------------- /test/dummy/config/puma.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | # Puma can serve each request in a thread from an internal thread pool. 4 | # The `threads` method setting takes two numbers a minimum and maximum. 5 | # Any libraries that use thread pools should be configured to match 6 | # the maximum value specified for Puma. Default is set to 5 threads for minimum 7 | # and maximum, this matches the default thread size of Active Record. 8 | # 9 | threads_count = ENV.fetch("RAILS_MAX_THREADS") { 5 }.to_i 10 | threads threads_count, threads_count 11 | 12 | # Specifies the `port` that Puma will listen on to receive requests, default is 13 | # 3000. 14 | port ENV.fetch("PORT") { 3000 } 15 | 16 | # Specifies the `environment` that Puma will run in. 17 | # 18 | environment ENV.fetch("RAILS_ENV") { "development" } 19 | 20 | # Specifies the number of `workers` to boot in clustered mode. 21 | # Workers are forked webserver processes. If using threads and workers together 22 | # the concurrency of the application would be max `threads` * `workers`. 23 | # Workers do not work on JRuby or Windows (both of which do not support 24 | # processes). 25 | # 26 | # workers ENV.fetch("WEB_CONCURRENCY") { 2 } 27 | 28 | # Use the `preload_app!` method when specifying a `workers` number. 29 | # This directive tells Puma to first boot the application and load code 30 | # before forking the application. This takes advantage of Copy On Write 31 | # process behavior so workers use less memory. If you use this option 32 | # you need to make sure to reconnect any threads in the `on_worker_boot` 33 | # block. 34 | # 35 | # preload_app! 36 | 37 | # The code in the `on_worker_boot` will be called if you are using 38 | # clustered mode by specifying a number of `workers`. After each worker 39 | # process is booted this block will be run, if you are using `preload_app!` 40 | # option you will want to use this block to reconnect to any threads 41 | # or connections that may have been created at application boot, Ruby 42 | # cannot share connections between processes. 43 | # 44 | # on_worker_boot do 45 | # ActiveRecord::Base.establish_connection if defined?(ActiveRecord) 46 | # end 47 | 48 | # Allow puma to be restarted by `rails restart` command. 49 | plugin :tmp_restart 50 | -------------------------------------------------------------------------------- /test/dummy/config/routes.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | Rails.application.routes.draw do 4 | mount Chat::Engine => "/chat", as: "chat" 5 | root to: "home#index", id: "home" 6 | 7 | resource :users, only: %i(edit update) 8 | 9 | mount ActionCable.server => "/cable" 10 | end 11 | -------------------------------------------------------------------------------- /test/dummy/config/secrets.yml: -------------------------------------------------------------------------------- 1 | # Be sure to restart your server when you modify this file. 2 | 3 | # Your secret key is used for verifying the integrity of signed cookies. 4 | # If you change this key, all old signed cookies will become invalid! 5 | 6 | # Make sure the secret is at least 30 characters and all random, 7 | # no regular words or you'll be exposed to dictionary attacks. 8 | # You can use `rails secret` to generate a secure secret key. 9 | 10 | # Make sure the secrets in this file are kept private 11 | # if you're sharing your code publicly. 12 | 13 | development: 14 | secret_key_base: a295d0eb38e63f477b38a7d5abac7dbf4f72c046d7994d9bb7f7eec6a877c3a90d0073033ea78f8d1277df07462a843f056e4cea112c7ef4368fdd5d7bf13b41 15 | 16 | test: 17 | secret_key_base: 4a7c0cf250513a95464864600a5cc56fce804622997f4fd5767ed77dce9cef2ace66c809941f87ab70470bb9df0c619ff1a86a550536c7fa74509fb3bcdfc621 18 | 19 | # Do not keep production secrets in the repository, 20 | # instead read values from the environment. 21 | production: 22 | secret_key_base: <%= ENV["SECRET_KEY_BASE"] %> 23 | -------------------------------------------------------------------------------- /test/dummy/config/sidekiq.yml: -------------------------------------------------------------------------------- 1 | --- 2 | :concurrency: 25 3 | development: 4 | :concurrency: 2 5 | staging: 6 | :concurrency: 10 7 | :queues: 8 | - default 9 | -------------------------------------------------------------------------------- /test/dummy/config/spring.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | %w( 4 | .ruby-version 5 | .rbenv-vars 6 | tmp/restart.txt 7 | tmp/caching-dev.txt 8 | ).each { |path| Spring.watch(path) } 9 | -------------------------------------------------------------------------------- /test/dummy/db/migrate/20160911032725_create_users.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class CreateUsers < ActiveRecord::Migration[5.1] 4 | def change 5 | create_table :users do |t| 6 | t.timestamps null: false 7 | t.string :email, null: false 8 | t.string :first_name 9 | t.string :last_name 10 | t.attachment :image 11 | add_tokens_password(t) 12 | end 13 | 14 | add_index :users, :email 15 | add_index :users, :remember_token 16 | end 17 | 18 | private 19 | 20 | def add_tokens_password(table) 21 | table.string :encrypted_password, limit: 128, null: false 22 | table.string :confirmation_token, limit: 128 23 | table.string :remember_token, limit: 128, null: false 24 | end 25 | end 26 | -------------------------------------------------------------------------------- /test/dummy/db/migrate/20160919005427_add_chat_to_users.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class AddChatToUsers < ActiveRecord::Migration[5.0] 4 | def self.up 5 | add_column :users, :chat_status, :string, default: "offline" 6 | end 7 | 8 | def self.down 9 | remove_column :users, :chat_status, :string, default: "offline" 10 | end 11 | end 12 | -------------------------------------------------------------------------------- /test/dummy/db/migrate/20160919005428_create_chat.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | class CreateChat < ActiveRecord::Migration[5.0] 4 | def self.up 5 | create_table :chat_conversations do |t| 6 | t.timestamps 7 | end 8 | 9 | create_sessions 10 | create_messages 11 | end 12 | 13 | def self.down 14 | drop_table :chat_conversations 15 | drop_table :chat_sessions 16 | drop_table :chat_messages 17 | end 18 | 19 | private 20 | 21 | def create_sessions 22 | create_table :chat_sessions do |t| 23 | t.belongs_to :conversation 24 | t.belongs_to :user 25 | 26 | t.timestamps 27 | end 28 | end 29 | 30 | def create_messages 31 | create_table :chat_messages do |t| 32 | t.belongs_to :user 33 | t.text :text 34 | t.belongs_to :conversation 35 | t.belongs_to :session 36 | t.attachment :image 37 | 38 | t.timestamps 39 | end 40 | end 41 | end 42 | -------------------------------------------------------------------------------- /test/dummy/db/schema.rb: -------------------------------------------------------------------------------- 1 | # This file is auto-generated from the current state of the database. Instead 2 | # of editing this file, please use the migrations feature of Active Record to 3 | # incrementally modify your database, and then regenerate this schema definition. 4 | # 5 | # Note that this schema.rb definition is the authoritative source for your 6 | # database schema. If you need to create the application database on another 7 | # system, you should be using db:schema:load, not running all the migrations 8 | # from scratch. The latter is a flawed and unsustainable approach (the more migrations 9 | # you'll amass, the slower it'll run and the greater likelihood for issues). 10 | # 11 | # It's strongly recommended that you check this file into your version control system. 12 | 13 | ActiveRecord::Schema.define(version: 20160919005428) do 14 | 15 | create_table "chat_conversations", force: :cascade do |t| 16 | t.datetime "created_at", null: false 17 | t.datetime "updated_at", null: false 18 | end 19 | 20 | create_table "chat_messages", force: :cascade do |t| 21 | t.integer "user_id" 22 | t.text "text" 23 | t.integer "conversation_id" 24 | t.integer "session_id" 25 | t.string "image_file_name" 26 | t.string "image_content_type" 27 | t.integer "image_file_size" 28 | t.datetime "image_updated_at" 29 | t.datetime "created_at", null: false 30 | t.datetime "updated_at", null: false 31 | t.index ["conversation_id"], name: "index_chat_messages_on_conversation_id" 32 | t.index ["session_id"], name: "index_chat_messages_on_session_id" 33 | t.index ["user_id"], name: "index_chat_messages_on_user_id" 34 | end 35 | 36 | create_table "chat_sessions", force: :cascade do |t| 37 | t.integer "conversation_id" 38 | t.integer "user_id" 39 | t.datetime "created_at", null: false 40 | t.datetime "updated_at", null: false 41 | t.index ["conversation_id"], name: "index_chat_sessions_on_conversation_id" 42 | t.index ["user_id"], name: "index_chat_sessions_on_user_id" 43 | end 44 | 45 | create_table "users", force: :cascade do |t| 46 | t.datetime "created_at", null: false 47 | t.datetime "updated_at", null: false 48 | t.string "email", null: false 49 | t.string "first_name" 50 | t.string "last_name" 51 | t.string "image_file_name" 52 | t.string "image_content_type" 53 | t.integer "image_file_size" 54 | t.datetime "image_updated_at" 55 | t.string "encrypted_password", limit: 128, null: false 56 | t.string "confirmation_token", limit: 128 57 | t.string "remember_token", limit: 128, null: false 58 | t.string "chat_status", default: "offline" 59 | t.index ["email"], name: "index_users_on_email" 60 | t.index ["remember_token"], name: "index_users_on_remember_token" 61 | end 62 | 63 | end 64 | -------------------------------------------------------------------------------- /test/dummy/db/seeds.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | admin = User.create( 4 | first_name: "Nick", 5 | last_name: "Pezza", 6 | password: "password", 7 | email: "admin@admin.com", 8 | image: open(Faker::Avatar.image) 9 | ) 10 | 11 | def create_user 12 | name = Faker::StarWars.character 13 | 14 | User.create( 15 | first_name: name.split(" ")[0], 16 | last_name: name.split(" ")[1], 17 | password: "password", 18 | email: "#{name.parameterize}@chat.com", 19 | image: open(Faker::Avatar.image) 20 | ).id 21 | end 22 | 23 | 5.times do |i| 24 | conversation = Chat::Conversation.create(user_ids: [create_user, admin.id]) 25 | 26 | i.times do 27 | conversation.reload.sessions.create(user_id: create_user) 28 | end 29 | end 30 | -------------------------------------------------------------------------------- /test/dummy/lib/assets/.keep: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/npezza93/chat/c8462876bd068532f6c4ebcf1d32570ea60f1dec/test/dummy/lib/assets/.keep -------------------------------------------------------------------------------- /test/dummy/public/404.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | The page you were looking for doesn't exist (404) 5 | 6 | 55 | 56 | 57 | 58 | 59 |
60 |
61 |

The page you were looking for doesn't exist.

62 |

You may have mistyped the address or the page may have moved.

63 |
64 |

If you are the application owner check the logs for more information.

65 |
66 | 67 | 68 | -------------------------------------------------------------------------------- /test/dummy/public/422.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | The change you wanted was rejected (422) 5 | 6 | 55 | 56 | 57 | 58 | 59 |
60 |
61 |

The change you wanted was rejected.

62 |

Maybe you tried to change something you didn't have access to.

63 |
64 |

If you are the application owner check the logs for more information.

65 |
66 | 67 | 68 | -------------------------------------------------------------------------------- /test/dummy/public/500.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | We're sorry, but something went wrong (500) 5 | 6 | 55 | 56 | 57 | 58 | 59 |
60 |
61 |

We're sorry, but something went wrong.

62 |
63 |

If you are the application owner check the logs for more information.

64 |
65 | 66 | 67 | -------------------------------------------------------------------------------- /test/dummy/public/apple-touch-icon-precomposed.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/npezza93/chat/c8462876bd068532f6c4ebcf1d32570ea60f1dec/test/dummy/public/apple-touch-icon-precomposed.png -------------------------------------------------------------------------------- /test/dummy/public/apple-touch-icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/npezza93/chat/c8462876bd068532f6c4ebcf1d32570ea60f1dec/test/dummy/public/apple-touch-icon.png -------------------------------------------------------------------------------- /test/dummy/public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/npezza93/chat/c8462876bd068532f6c4ebcf1d32570ea60f1dec/test/dummy/public/favicon.ico -------------------------------------------------------------------------------- /test/integration/navigation_test.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require "test_helper" 4 | 5 | class NavigationTest < ActionDispatch::IntegrationTest 6 | # test "the truth" do 7 | # assert true 8 | # end 9 | end 10 | -------------------------------------------------------------------------------- /test/lib/generators/chat/install_generator_test.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require "test_helper" 4 | require "generators/install/install_generator" 5 | 6 | module Chat 7 | class InstallGeneratorTest < Rails::Generators::TestCase 8 | tests InstallGenerator 9 | destination Rails.root.join("tmp/generators") 10 | setup :prepare_destination 11 | 12 | # test "generator runs without errors" do 13 | # assert_nothing_raised do 14 | # run_generator ["arguments"] 15 | # end 16 | # end 17 | end 18 | end 19 | -------------------------------------------------------------------------------- /test/test_helper.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | # Configure Rails Environment 4 | ENV["RAILS_ENV"] = "test" 5 | 6 | require File.expand_path("../../test/dummy/config/environment.rb", __FILE__) 7 | ActiveRecord::Migrator.migrations_paths = 8 | [File.expand_path("../../test/dummy/db/migrate", __FILE__)] 9 | ActiveRecord::Migrator.migrations_paths << 10 | File.expand_path("../../db/migrate", __FILE__) 11 | require "rails/test_help" 12 | 13 | # Filter out Minitest backtrace while allowing backtrace from other libraries 14 | # to be shown. 15 | Minitest.backtrace_filter = Minitest::BacktraceFilter.new 16 | 17 | # Load fixtures from the engine 18 | if ActiveSupport::TestCase.respond_to?(:fixture_path=) 19 | ActiveSupport::TestCase.fixture_path = 20 | File.expand_path("../fixtures", __FILE__) 21 | ActionDispatch::IntegrationTest.fixture_path = 22 | ActiveSupport::TestCase.fixture_path 23 | ActiveSupport::TestCase.file_fixture_path = 24 | ActiveSupport::TestCase.fixture_path + "/files" 25 | ActiveSupport::TestCase.fixtures :all 26 | end 27 | --------------------------------------------------------------------------------