├── .gitignore
├── README.md
├── __init__.py
├── data
└── pull_data.sh
├── data_utils.py
├── env_setup.sh
├── execute.py
├── neuralconvo.ini
├── seq2seq.ini
├── seq2seq_model.py
├── seq2seq_serve.ini
├── ui
├── .gitignore
├── README.md
├── __init.py__
├── app.py
├── requirements.txt
├── setup.sh
├── static
│ ├── css
│ │ ├── normalize.css
│ │ └── style.css
│ ├── js
│ │ ├── index.js
│ │ ├── jquery-latest.js
│ │ ├── jquery.mCustomScrollbar.concat.min.js
│ │ ├── jquery.mCustomScrollbar.min.css
│ │ └── jquery.min.js
│ ├── res
│ │ ├── botim.png
│ │ ├── hd1.jpg
│ │ └── hd2.jpg
│ └── scss
│ │ └── style.scss
└── templates
│ └── index.html
└── working_dir
├── vocab20000.dec
└── vocab20000.enc
/.gitignore:
--------------------------------------------------------------------------------
1 | data/
2 | *.pyc
3 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # easy\_seq2seq
2 |
3 | 
4 |
5 | * Go to [*practical_seq2seq*](https://github.com/suriyadeepan/practical_seq2seq)
6 |
7 | An implementation of Seq2Seq that actually works. I want to make it easy for people to train their own seq2seq model with any corpus. I am also adding the parameters of my trained model for people to just use it without training. If you have a model that works share your model params here, as external link or do a pull request. I have used Cornell Movie Dialog Corpus to train my model. A link to preprocessed data and scripts for preprocessing can be found in this repo.
8 |
9 | *Have Fun!*
10 |
11 | ## Update 1.1.2017
12 |
13 |
14 | I have created another repository - [practical_seq2seq](https://github.com/suriyadeepan/practical_seq2seq) to experiment with the seq2seq model. The new model trained on Twitter chat log and Cornell Movie Dialog corpus performs well. I wrote an article - [Practical seq2seq](http://suriyadeepan.github.io/2016-12-31-practical-seq2seq/), explaining the code.
15 |
16 | Happy New Year, **2017**
17 |
18 |
19 | ## Setup
20 |
21 |
22 | * Create temporary working directory prior to training
23 |
24 | ```bash
25 | mkdir working_dir
26 | ```
27 |
28 | * Download test/train data from Cornell Movie Dialog Corpus
29 |
30 | ```bash
31 | cd data/
32 | bash pull_data.sh
33 | ```
34 |
35 | ## Training
36 |
37 | ```bash
38 | # edit seq2seq.ini file to set
39 | # mode = train
40 | python execute.py
41 | # or use custom ini file
42 | # python execute.py my_custom_conf.ini
43 | ```
44 |
45 | ## Testing
46 |
47 | ```bash
48 | # edit seq2seq.ini file to set
49 | # mode = test
50 | python execute.py
51 | ```
52 |
53 | ## Serve
54 |
55 | ```bash
56 | # configuration : seq2seq_serve.ini
57 | python ui/app.py
58 | # wait until this message shows up
59 | # "Running on http://127.0.0.1:5000/ (Press CTRL+C to quit)"
60 | # open up the address in browser, chat with the bot
61 | ```
62 |
--------------------------------------------------------------------------------
/__init__.py:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/suriyadeepan/easy_seq2seq/c05b0d02f15e9bd1947ff79ddcd0b761a214e2a9/__init__.py
--------------------------------------------------------------------------------
/data/pull_data.sh:
--------------------------------------------------------------------------------
1 | wget -c 'https://www.dropbox.com/s/ncfa5t950gvtaeb/test.enc?dl=0' -O test.enc
2 | wget -c 'https://www.dropbox.com/s/48ro4759jaikque/test.dec?dl=0' -O test.dec
3 | wget -c 'https://www.dropbox.com/s/gu54ngk3xpwite4/train.enc?dl=0' -O train.enc
4 | wget -c 'https://www.dropbox.com/s/g3z2msjziqocndl/train.dec?dl=0' -O train.dec
5 |
--------------------------------------------------------------------------------
/data_utils.py:
--------------------------------------------------------------------------------
1 | # Copyright 2015 The TensorFlow Authors. All Rights Reserved.
2 | #
3 | # Licensed under the Apache License, Version 2.0 (the "License");
4 | # you may not use this file except in compliance with the License.
5 | # You may obtain a copy of the License at
6 | #
7 | # http://www.apache.org/licenses/LICENSE-2.0
8 | #
9 | # Unless required by applicable law or agreed to in writing, software
10 | # distributed under the License is distributed on an "AS IS" BASIS,
11 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 | # See the License for the specific language governing permissions and
13 | # limitations under the License.
14 | # ==============================================================================
15 |
16 | """Utilities for downloading data from WMT, tokenizing, vocabularies."""
17 | from __future__ import absolute_import
18 | from __future__ import division
19 | from __future__ import print_function
20 |
21 | import os
22 | import re
23 |
24 | from six.moves import urllib
25 |
26 | from tensorflow.python.platform import gfile
27 |
28 | # Special vocabulary symbols - we always put them at the start.
29 | _PAD = b"_PAD"
30 | _GO = b"_GO"
31 | _EOS = b"_EOS"
32 | _UNK = b"_UNK"
33 | _START_VOCAB = [_PAD, _GO, _EOS, _UNK]
34 |
35 | PAD_ID = 0
36 | GO_ID = 1
37 | EOS_ID = 2
38 | UNK_ID = 3
39 |
40 | # Regular expressions used to tokenize.
41 | _WORD_SPLIT = re.compile(b"([.,!?\"':;)(])")
42 | _DIGIT_RE = re.compile(br"\d")
43 |
44 |
45 |
46 | def basic_tokenizer(sentence):
47 | """Very basic tokenizer: split the sentence into a list of tokens."""
48 | words = []
49 | for space_separated_fragment in sentence.strip().split():
50 | if isinstance(space_separated_fragment, str):
51 | word = str.encode(space_separated_fragment)
52 | else:
53 | word = space_separated_fragment
54 | words.extend(re.split(_WORD_SPLIT, word))
55 | return [w for w in words if w]
56 |
57 |
58 | def create_vocabulary(vocabulary_path, data_path, max_vocabulary_size,
59 | tokenizer=None, normalize_digits=True):
60 |
61 | if not gfile.Exists(vocabulary_path):
62 | print("Creating vocabulary %s from %s" % (vocabulary_path, data_path))
63 | vocab = {}
64 | with gfile.GFile(data_path, mode="rb") as f:
65 | counter = 0
66 | for line in f:
67 | counter += 1
68 | if counter % 100000 == 0:
69 | print(" processing line %d" % counter)
70 | tokens = tokenizer(line) if tokenizer else basic_tokenizer(line)
71 | for w in tokens:
72 | word = re.sub(_DIGIT_RE, b"0", w) if normalize_digits else w
73 | if word in vocab:
74 | vocab[word] += 1
75 | else:
76 | vocab[word] = 1
77 | vocab_list = _START_VOCAB + sorted(vocab, key=vocab.get, reverse=True)
78 | print('>> Full Vocabulary Size :',len(vocab_list))
79 | if len(vocab_list) > max_vocabulary_size:
80 | vocab_list = vocab_list[:max_vocabulary_size]
81 | with gfile.GFile(vocabulary_path, mode="wb") as vocab_file:
82 | for w in vocab_list:
83 | vocab_file.write(w + b"\n")
84 |
85 |
86 | def initialize_vocabulary(vocabulary_path):
87 |
88 | if gfile.Exists(vocabulary_path):
89 | rev_vocab = []
90 | with gfile.GFile(vocabulary_path, mode="rb") as f:
91 | rev_vocab.extend(f.readlines())
92 | rev_vocab = [line.strip() for line in rev_vocab]
93 | vocab = dict([(x, y) for (y, x) in enumerate(rev_vocab)])
94 | return vocab, rev_vocab
95 | else:
96 | raise ValueError("Vocabulary file %s not found.", vocabulary_path)
97 |
98 |
99 | def sentence_to_token_ids(sentence, vocabulary, tokenizer=None, normalize_digits=True):
100 |
101 | if tokenizer:
102 | words = tokenizer(sentence)
103 | else:
104 | words = basic_tokenizer(sentence)
105 | if not normalize_digits:
106 | return [vocabulary.get(w, UNK_ID) for w in words]
107 | # Normalize digits by 0 before looking words up in the vocabulary.
108 | return [vocabulary.get(re.sub(_DIGIT_RE, b"0", w), UNK_ID) for w in words]
109 |
110 |
111 | def data_to_token_ids(data_path, target_path, vocabulary_path,
112 | tokenizer=None, normalize_digits=True):
113 |
114 | if not gfile.Exists(target_path):
115 | print("Tokenizing data in %s" % data_path)
116 | vocab, _ = initialize_vocabulary(vocabulary_path)
117 | with gfile.GFile(data_path, mode="rb") as data_file:
118 | with gfile.GFile(target_path, mode="w") as tokens_file:
119 | counter = 0
120 | for line in data_file:
121 | counter += 1
122 | if counter % 100000 == 0:
123 | print(" tokenizing line %d" % counter)
124 | token_ids = sentence_to_token_ids(line, vocab, tokenizer,
125 | normalize_digits)
126 | tokens_file.write(" ".join([str(tok) for tok in token_ids]) + "\n")
127 |
128 |
129 |
130 | def prepare_custom_data(working_directory, train_enc, train_dec, test_enc, test_dec, enc_vocabulary_size, dec_vocabulary_size, tokenizer=None):
131 |
132 | # Create vocabularies of the appropriate sizes.
133 | enc_vocab_path = os.path.join(working_directory, "vocab%d.enc" % enc_vocabulary_size)
134 | dec_vocab_path = os.path.join(working_directory, "vocab%d.dec" % dec_vocabulary_size)
135 | create_vocabulary(enc_vocab_path, train_enc, enc_vocabulary_size, tokenizer)
136 | create_vocabulary(dec_vocab_path, train_dec, dec_vocabulary_size, tokenizer)
137 |
138 | # Create token ids for the training data.
139 | enc_train_ids_path = train_enc + (".ids%d" % enc_vocabulary_size)
140 | dec_train_ids_path = train_dec + (".ids%d" % dec_vocabulary_size)
141 | data_to_token_ids(train_enc, enc_train_ids_path, enc_vocab_path, tokenizer)
142 | data_to_token_ids(train_dec, dec_train_ids_path, dec_vocab_path, tokenizer)
143 |
144 | # Create token ids for the development data.
145 | enc_dev_ids_path = test_enc + (".ids%d" % enc_vocabulary_size)
146 | dec_dev_ids_path = test_dec + (".ids%d" % dec_vocabulary_size)
147 | data_to_token_ids(test_enc, enc_dev_ids_path, enc_vocab_path, tokenizer)
148 | data_to_token_ids(test_dec, dec_dev_ids_path, dec_vocab_path, tokenizer)
149 |
150 | return (enc_train_ids_path, dec_train_ids_path, enc_dev_ids_path, dec_dev_ids_path, enc_vocab_path, dec_vocab_path)
151 |
--------------------------------------------------------------------------------
/env_setup.sh:
--------------------------------------------------------------------------------
1 | echo 'export PYTHONPATH=$PYTHONPATH:$PWD' >> ~/.bashrc
2 |
--------------------------------------------------------------------------------
/execute.py:
--------------------------------------------------------------------------------
1 | # Copyright 2015 The TensorFlow Authors. All Rights Reserved.
2 | #
3 | # Licensed under the Apache License, Version 2.0 (the "License");
4 | # you may not use this file except in compliance with the License.
5 | # You may obtain a copy of the License at
6 | #
7 | # http://www.apache.org/licenses/LICENSE-2.0
8 | #
9 | # Unless required by applicable law or agreed to in writing, software
10 | # distributed under the License is distributed on an "AS IS" BASIS,
11 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 | # See the License for the specific language governing permissions and
13 | # limitations under the License.
14 | # ==============================================================================
15 |
16 | from __future__ import absolute_import
17 | from __future__ import division
18 | from __future__ import print_function
19 |
20 | import math
21 | import os
22 | import random
23 | import sys
24 | import time
25 |
26 | import numpy as np
27 | from six.moves import xrange # pylint: disable=redefined-builtin
28 | import tensorflow as tf
29 |
30 | import data_utils
31 | import seq2seq_model
32 |
33 | try:
34 | from ConfigParser import SafeConfigParser
35 | except:
36 | from configparser import SafeConfigParser # In Python 3, ConfigParser has been renamed to configparser for PEP 8 compliance.
37 |
38 | gConfig = {}
39 |
40 | def get_config(config_file='seq2seq.ini'):
41 | parser = SafeConfigParser()
42 | parser.read(config_file)
43 | # get the ints, floats and strings
44 | _conf_ints = [ (key, int(value)) for key,value in parser.items('ints') ]
45 | _conf_floats = [ (key, float(value)) for key,value in parser.items('floats') ]
46 | _conf_strings = [ (key, str(value)) for key,value in parser.items('strings') ]
47 | return dict(_conf_ints + _conf_floats + _conf_strings)
48 |
49 | # We use a number of buckets and pad to the closest one for efficiency.
50 | # See seq2seq_model.Seq2SeqModel for details of how they work.
51 | _buckets = [(5, 10), (10, 15), (20, 25), (40, 50)]
52 |
53 |
54 | def read_data(source_path, target_path, max_size=None):
55 | """Read data from source and target files and put into buckets.
56 |
57 | Args:
58 | source_path: path to the files with token-ids for the source language.
59 | target_path: path to the file with token-ids for the target language;
60 | it must be aligned with the source file: n-th line contains the desired
61 | output for n-th line from the source_path.
62 | max_size: maximum number of lines to read, all other will be ignored;
63 | if 0 or None, data files will be read completely (no limit).
64 |
65 | Returns:
66 | data_set: a list of length len(_buckets); data_set[n] contains a list of
67 | (source, target) pairs read from the provided data files that fit
68 | into the n-th bucket, i.e., such that len(source) < _buckets[n][0] and
69 | len(target) < _buckets[n][1]; source and target are lists of token-ids.
70 | """
71 | data_set = [[] for _ in _buckets]
72 | with tf.gfile.GFile(source_path, mode="r") as source_file:
73 | with tf.gfile.GFile(target_path, mode="r") as target_file:
74 | source, target = source_file.readline(), target_file.readline()
75 | counter = 0
76 | while source and target and (not max_size or counter < max_size):
77 | counter += 1
78 | if counter % 100000 == 0:
79 | print(" reading data line %d" % counter)
80 | sys.stdout.flush()
81 | source_ids = [int(x) for x in source.split()]
82 | target_ids = [int(x) for x in target.split()]
83 | target_ids.append(data_utils.EOS_ID)
84 | for bucket_id, (source_size, target_size) in enumerate(_buckets):
85 | if len(source_ids) < source_size and len(target_ids) < target_size:
86 | data_set[bucket_id].append([source_ids, target_ids])
87 | break
88 | source, target = source_file.readline(), target_file.readline()
89 | return data_set
90 |
91 |
92 | def create_model(session, forward_only):
93 |
94 | """Create model and initialize or load parameters"""
95 | model = seq2seq_model.Seq2SeqModel( gConfig['enc_vocab_size'], gConfig['dec_vocab_size'], _buckets, gConfig['layer_size'], gConfig['num_layers'], gConfig['max_gradient_norm'], gConfig['batch_size'], gConfig['learning_rate'], gConfig['learning_rate_decay_factor'], forward_only=forward_only)
96 |
97 | if 'pretrained_model' in gConfig:
98 | model.saver.restore(session,gConfig['pretrained_model'])
99 | return model
100 |
101 | ckpt = tf.train.get_checkpoint_state(gConfig['working_directory'])
102 | if ckpt and ckpt.model_checkpoint_path:
103 | print("Reading model parameters from %s" % ckpt.model_checkpoint_path)
104 | model.saver.restore(session, ckpt.model_checkpoint_path)
105 | else:
106 | print("Created model with fresh parameters.")
107 | session.run(tf.global_variables_initializer())
108 | return model
109 |
110 |
111 | def train():
112 | # prepare dataset
113 | print("Preparing data in %s" % gConfig['working_directory'])
114 | enc_train, dec_train, enc_dev, dec_dev, _, _ = data_utils.prepare_custom_data(gConfig['working_directory'],gConfig['train_enc'],gConfig['train_dec'],gConfig['test_enc'],gConfig['test_dec'],gConfig['enc_vocab_size'],gConfig['dec_vocab_size'])
115 |
116 | # setup config to use BFC allocator
117 | config = tf.ConfigProto()
118 | config.gpu_options.allocator_type = 'BFC'
119 |
120 | with tf.Session(config=config) as sess:
121 | # Create model.
122 | print("Creating %d layers of %d units." % (gConfig['num_layers'], gConfig['layer_size']))
123 | model = create_model(sess, False)
124 |
125 | # Read data into buckets and compute their sizes.
126 | print ("Reading development and training data (limit: %d)."
127 | % gConfig['max_train_data_size'])
128 | dev_set = read_data(enc_dev, dec_dev)
129 | train_set = read_data(enc_train, dec_train, gConfig['max_train_data_size'])
130 | train_bucket_sizes = [len(train_set[b]) for b in xrange(len(_buckets))]
131 | train_total_size = float(sum(train_bucket_sizes))
132 |
133 | # A bucket scale is a list of increasing numbers from 0 to 1 that we'll use
134 | # to select a bucket. Length of [scale[i], scale[i+1]] is proportional to
135 | # the size if i-th training bucket, as used later.
136 | train_buckets_scale = [sum(train_bucket_sizes[:i + 1]) / train_total_size
137 | for i in xrange(len(train_bucket_sizes))]
138 |
139 | # This is the training loop.
140 | step_time, loss = 0.0, 0.0
141 | current_step = 0
142 | previous_losses = []
143 | while True:
144 | # Choose a bucket according to data distribution. We pick a random number
145 | # in [0, 1] and use the corresponding interval in train_buckets_scale.
146 | random_number_01 = np.random.random_sample()
147 | bucket_id = min([i for i in xrange(len(train_buckets_scale))
148 | if train_buckets_scale[i] > random_number_01])
149 |
150 | # Get a batch and make a step.
151 | start_time = time.time()
152 | encoder_inputs, decoder_inputs, target_weights = model.get_batch(
153 | train_set, bucket_id)
154 | _, step_loss, _ = model.step(sess, encoder_inputs, decoder_inputs,
155 | target_weights, bucket_id, False)
156 | step_time += (time.time() - start_time) / gConfig['steps_per_checkpoint']
157 | loss += step_loss / gConfig['steps_per_checkpoint']
158 | current_step += 1
159 |
160 | # Once in a while, we save checkpoint, print statistics, and run evals.
161 | if current_step % gConfig['steps_per_checkpoint'] == 0:
162 | # Print statistics for the previous epoch.
163 | perplexity = math.exp(loss) if loss < 300 else float('inf')
164 | print ("global step %d learning rate %.4f step-time %.2f perplexity "
165 | "%.2f" % (model.global_step.eval(), model.learning_rate.eval(),
166 | step_time, perplexity))
167 | # Decrease learning rate if no improvement was seen over last 3 times.
168 | if len(previous_losses) > 2 and loss > max(previous_losses[-3:]):
169 | sess.run(model.learning_rate_decay_op)
170 | previous_losses.append(loss)
171 | # Save checkpoint and zero timer and loss.
172 | checkpoint_path = os.path.join(gConfig['working_directory'], "seq2seq.ckpt")
173 | model.saver.save(sess, checkpoint_path, global_step=model.global_step)
174 | step_time, loss = 0.0, 0.0
175 | # Run evals on development set and print their perplexity.
176 | for bucket_id in xrange(len(_buckets)):
177 | if len(dev_set[bucket_id]) == 0:
178 | print(" eval: empty bucket %d" % (bucket_id))
179 | continue
180 | encoder_inputs, decoder_inputs, target_weights = model.get_batch(
181 | dev_set, bucket_id)
182 | _, eval_loss, _ = model.step(sess, encoder_inputs, decoder_inputs,
183 | target_weights, bucket_id, True)
184 | eval_ppx = math.exp(eval_loss) if eval_loss < 300 else float('inf')
185 | print(" eval: bucket %d perplexity %.2f" % (bucket_id, eval_ppx))
186 | sys.stdout.flush()
187 |
188 |
189 | def decode():
190 | with tf.Session() as sess:
191 | # Create model and load parameters.
192 | model = create_model(sess, True)
193 | model.batch_size = 1 # We decode one sentence at a time.
194 |
195 | # Load vocabularies.
196 | enc_vocab_path = os.path.join(gConfig['working_directory'],"vocab%d.enc" % gConfig['enc_vocab_size'])
197 | dec_vocab_path = os.path.join(gConfig['working_directory'],"vocab%d.dec" % gConfig['dec_vocab_size'])
198 |
199 | enc_vocab, _ = data_utils.initialize_vocabulary(enc_vocab_path)
200 | _, rev_dec_vocab = data_utils.initialize_vocabulary(dec_vocab_path)
201 |
202 | # Decode from standard input.
203 | sys.stdout.write("> ")
204 | sys.stdout.flush()
205 | sentence = sys.stdin.readline()
206 | while sentence:
207 | # Get token-ids for the input sentence.
208 | token_ids = data_utils.sentence_to_token_ids(tf.compat.as_bytes(sentence), enc_vocab)
209 | # Which bucket does it belong to?
210 | bucket_id = min([b for b in xrange(len(_buckets))
211 | if _buckets[b][0] > len(token_ids)])
212 | # Get a 1-element batch to feed the sentence to the model.
213 | encoder_inputs, decoder_inputs, target_weights = model.get_batch(
214 | {bucket_id: [(token_ids, [])]}, bucket_id)
215 | # Get output logits for the sentence.
216 | _, _, output_logits = model.step(sess, encoder_inputs, decoder_inputs,
217 | target_weights, bucket_id, True)
218 | # This is a greedy decoder - outputs are just argmaxes of output_logits.
219 | outputs = [int(np.argmax(logit, axis=1)) for logit in output_logits]
220 | # If there is an EOS symbol in outputs, cut them at that point.
221 | if data_utils.EOS_ID in outputs:
222 | outputs = outputs[:outputs.index(data_utils.EOS_ID)]
223 | # Print out French sentence corresponding to outputs.
224 | print(" ".join([tf.compat.as_str(rev_dec_vocab[output]) for output in outputs]))
225 | print("> ", end="")
226 | sys.stdout.flush()
227 | sentence = sys.stdin.readline()
228 |
229 |
230 | def self_test():
231 | """Test the translation model."""
232 | with tf.Session() as sess:
233 | print("Self-test for neural translation model.")
234 | # Create model with vocabularies of 10, 2 small buckets, 2 layers of 32.
235 | model = seq2seq_model.Seq2SeqModel(10, 10, [(3, 3), (6, 6)], 32, 2,
236 | 5.0, 32, 0.3, 0.99, num_samples=8)
237 | sess.run(tf.initialize_all_variables())
238 |
239 | # Fake data set for both the (3, 3) and (6, 6) bucket.
240 | data_set = ([([1, 1], [2, 2]), ([3, 3], [4]), ([5], [6])],
241 | [([1, 1, 1, 1, 1], [2, 2, 2, 2, 2]), ([3, 3, 3], [5, 6])])
242 | for _ in xrange(5): # Train the fake model for 5 steps.
243 | bucket_id = random.choice([0, 1])
244 | encoder_inputs, decoder_inputs, target_weights = model.get_batch(
245 | data_set, bucket_id)
246 | model.step(sess, encoder_inputs, decoder_inputs, target_weights,
247 | bucket_id, False)
248 |
249 |
250 | def init_session(sess, conf='seq2seq.ini'):
251 | global gConfig
252 | gConfig = get_config(conf)
253 |
254 | # Create model and load parameters.
255 | model = create_model(sess, True)
256 | model.batch_size = 1 # We decode one sentence at a time.
257 |
258 | # Load vocabularies.
259 | enc_vocab_path = os.path.join(gConfig['working_directory'],"vocab%d.enc" % gConfig['enc_vocab_size'])
260 | dec_vocab_path = os.path.join(gConfig['working_directory'],"vocab%d.dec" % gConfig['dec_vocab_size'])
261 |
262 | enc_vocab, _ = data_utils.initialize_vocabulary(enc_vocab_path)
263 | _, rev_dec_vocab = data_utils.initialize_vocabulary(dec_vocab_path)
264 |
265 | return sess, model, enc_vocab, rev_dec_vocab
266 |
267 | def decode_line(sess, model, enc_vocab, rev_dec_vocab, sentence):
268 | # Get token-ids for the input sentence.
269 | token_ids = data_utils.sentence_to_token_ids(tf.compat.as_bytes(sentence), enc_vocab)
270 |
271 | # Which bucket does it belong to?
272 | bucket_id = min([b for b in xrange(len(_buckets)) if _buckets[b][0] > len(token_ids)])
273 |
274 | # Get a 1-element batch to feed the sentence to the model.
275 | encoder_inputs, decoder_inputs, target_weights = model.get_batch({bucket_id: [(token_ids, [])]}, bucket_id)
276 |
277 | # Get output logits for the sentence.
278 | _, _, output_logits = model.step(sess, encoder_inputs, decoder_inputs, target_weights, bucket_id, True)
279 |
280 | # This is a greedy decoder - outputs are just argmaxes of output_logits.
281 | outputs = [int(np.argmax(logit, axis=1)) for logit in output_logits]
282 |
283 | # If there is an EOS symbol in outputs, cut them at that point.
284 | if data_utils.EOS_ID in outputs:
285 | outputs = outputs[:outputs.index(data_utils.EOS_ID)]
286 |
287 | return " ".join([tf.compat.as_str(rev_dec_vocab[output]) for output in outputs])
288 |
289 | if __name__ == '__main__':
290 | if len(sys.argv) - 1:
291 | gConfig = get_config(sys.argv[1])
292 | else:
293 | # get configuration from seq2seq.ini
294 | gConfig = get_config()
295 |
296 | print('\n>> Mode : %s\n' %(gConfig['mode']))
297 |
298 | if gConfig['mode'] == 'train':
299 | # start training
300 | train()
301 | elif gConfig['mode'] == 'test':
302 | # interactive decode
303 | decode()
304 | else:
305 | # wrong way to execute "serve"
306 | # Use : >> python ui/app.py
307 | # uses seq2seq_serve.ini as conf file
308 | print('Serve Usage : >> python ui/app.py')
309 | print('# uses seq2seq_serve.ini as conf file')
310 |
--------------------------------------------------------------------------------
/neuralconvo.ini:
--------------------------------------------------------------------------------
1 | [strings]
2 | # Mode : train, test, serve
3 | mode = train
4 | train_enc = data/train.enc
5 | train_dec = data/train.dec
6 | test_enc = data/test.enc
7 | test_dec = data/test.enc
8 | # folder where checkpoints, vocabulary, temporary data will be stored
9 | working_directory = working_dir/
10 | [ints]
11 | # vocabulary size
12 | # 20,000 is a reasonable size
13 | enc_vocab_size = 45000
14 | dec_vocab_size = 45000
15 | # number of LSTM layers : 1/2/3
16 | num_layers = 2
17 | # typical options : 128, 256, 512, 1024
18 | layer_size = 900
19 | # dataset size limit; typically none : no limit
20 | max_train_data_size = 50000
21 | batch_size = 10
22 | # steps per checkpoint
23 | # Note : At a checkpoint, models parameters are saved, model is evaluated
24 | # and results are printed
25 | steps_per_checkpoint = 200
26 | [floats]
27 | learning_rate = 0.001
28 | learning_rate_decay_factor = 0.9
29 | max_gradient_norm = 5.0
30 | ##############################################################################
31 | # Note : Edit the bucket sizes at line47 of execute.py (_buckets)
32 | #
33 | # Learn more about the configurations from this link
34 | # https://www.tensorflow.org/versions/r0.9/tutorials/seq2seq/index.html
35 | ##############################################################################
36 |
--------------------------------------------------------------------------------
/seq2seq.ini:
--------------------------------------------------------------------------------
1 | [strings]
2 | # Mode : train, test, serve
3 | mode = train
4 | train_enc = data/train.enc
5 | train_dec = data/train.dec
6 | test_enc = data/test.enc
7 | test_dec = data/test.dec
8 | # folder where checkpoints, vocabulary, temporary data will be stored
9 | working_directory = working_dir/
10 | [ints]
11 | # vocabulary size
12 | # 20,000 is a reasonable size
13 | enc_vocab_size = 20000
14 | dec_vocab_size = 20000
15 | # number of LSTM layers : 1/2/3
16 | num_layers = 3
17 | # typical options : 128, 256, 512, 1024
18 | layer_size = 256
19 | # dataset size limit; typically none : no limit
20 | max_train_data_size = 0
21 | batch_size = 64
22 | # steps per checkpoint
23 | # Note : At a checkpoint, models parameters are saved, model is evaluated
24 | # and results are printed
25 | steps_per_checkpoint = 300
26 | [floats]
27 | learning_rate = 0.5
28 | learning_rate_decay_factor = 0.99
29 | max_gradient_norm = 5.0
30 | ##############################################################################
31 | # Note : Edit the bucket sizes at line47 of execute.py (_buckets)
32 | #
33 | # Learn more about the configurations from this link
34 | # https://www.tensorflow.org/versions/r0.9/tutorials/seq2seq/index.html
35 | ##############################################################################
36 |
--------------------------------------------------------------------------------
/seq2seq_model.py:
--------------------------------------------------------------------------------
1 | # Copyright 2015 The TensorFlow Authors. All Rights Reserved.
2 | #
3 | # Licensed under the Apache License, Version 2.0 (the "License");
4 | # you may not use this file except in compliance with the License.
5 | # You may obtain a copy of the License at
6 | #
7 | # http://www.apache.org/licenses/LICENSE-2.0
8 | #
9 | # Unless required by applicable law or agreed to in writing, software
10 | # distributed under the License is distributed on an "AS IS" BASIS,
11 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 | # See the License for the specific language governing permissions and
13 | # limitations under the License.
14 | # ==============================================================================
15 |
16 | """Sequence-to-sequence model with an attention mechanism."""
17 |
18 | from __future__ import absolute_import
19 | from __future__ import division
20 | from __future__ import print_function
21 |
22 | import random
23 |
24 | import numpy as np
25 | from six.moves import xrange # pylint: disable=redefined-builtin
26 | import tensorflow as tf
27 |
28 | from tensorflow.models.rnn.translate import data_utils
29 |
30 |
31 | class Seq2SeqModel(object):
32 | """Sequence-to-sequence model with attention and for multiple buckets.
33 |
34 | This class implements a multi-layer recurrent neural network as encoder,
35 | and an attention-based decoder. This is the same as the model described in
36 | this paper: http://arxiv.org/abs/1412.7449 - please look there for details,
37 | or into the seq2seq library for complete model implementation.
38 | This class also allows to use GRU cells in addition to LSTM cells, and
39 | sampled softmax to handle large output vocabulary size. A single-layer
40 | version of this model, but with bi-directional encoder, was presented in
41 | http://arxiv.org/abs/1409.0473
42 | and sampled softmax is described in Section 3 of the following paper.
43 | http://arxiv.org/abs/1412.2007
44 | """
45 |
46 | def __init__(self, source_vocab_size, target_vocab_size, buckets, size,
47 | num_layers, max_gradient_norm, batch_size, learning_rate,
48 | learning_rate_decay_factor, use_lstm=False,
49 | num_samples=512, forward_only=False):
50 | """Create the model.
51 |
52 | Args:
53 | source_vocab_size: size of the source vocabulary.
54 | target_vocab_size: size of the target vocabulary.
55 | buckets: a list of pairs (I, O), where I specifies maximum input length
56 | that will be processed in that bucket, and O specifies maximum output
57 | length. Training instances that have inputs longer than I or outputs
58 | longer than O will be pushed to the next bucket and padded accordingly.
59 | We assume that the list is sorted, e.g., [(2, 4), (8, 16)].
60 | size: number of units in each layer of the model.
61 | num_layers: number of layers in the model.
62 | max_gradient_norm: gradients will be clipped to maximally this norm.
63 | batch_size: the size of the batches used during training;
64 | the model construction is independent of batch_size, so it can be
65 | changed after initialization if this is convenient, e.g., for decoding.
66 | learning_rate: learning rate to start with.
67 | learning_rate_decay_factor: decay learning rate by this much when needed.
68 | use_lstm: if true, we use LSTM cells instead of GRU cells.
69 | num_samples: number of samples for sampled softmax.
70 | forward_only: if set, we do not construct the backward pass in the model.
71 | """
72 | self.source_vocab_size = source_vocab_size
73 | self.target_vocab_size = target_vocab_size
74 | self.buckets = buckets
75 | self.batch_size = batch_size
76 | self.learning_rate = tf.Variable(float(learning_rate), trainable=False)
77 | self.learning_rate_decay_op = self.learning_rate.assign(
78 | self.learning_rate * learning_rate_decay_factor)
79 | self.global_step = tf.Variable(0, trainable=False)
80 |
81 | # If we use sampled softmax, we need an output projection.
82 | output_projection = None
83 | softmax_loss_function = None
84 | # Sampled softmax only makes sense if we sample less than vocabulary size.
85 | if num_samples > 0 and num_samples < self.target_vocab_size:
86 | w = tf.get_variable("proj_w", [size, self.target_vocab_size])
87 | w_t = tf.transpose(w)
88 | b = tf.get_variable("proj_b", [self.target_vocab_size])
89 | output_projection = (w, b)
90 |
91 | def sampled_loss(inputs, labels):
92 | labels = tf.reshape(labels, [-1, 1])
93 | return tf.nn.sampled_softmax_loss(w_t, b, inputs, labels, num_samples,
94 | self.target_vocab_size)
95 | softmax_loss_function = sampled_loss
96 |
97 | # Create the internal multi-layer cell for our RNN.
98 | single_cell = tf.nn.rnn_cell.GRUCell(size)
99 | if use_lstm:
100 | single_cell = tf.nn.rnn_cell.BasicLSTMCell(size)
101 | cell = single_cell
102 | cell = tf.nn.rnn_cell.DropoutWrapper(cell, output_keep_prob=0.5)
103 | if num_layers > 1:
104 | cell = tf.nn.rnn_cell.MultiRNNCell([single_cell] * num_layers)
105 |
106 | # The seq2seq function: we use embedding for the input and attention.
107 | def seq2seq_f(encoder_inputs, decoder_inputs, do_decode):
108 | return tf.nn.seq2seq.embedding_attention_seq2seq(
109 | encoder_inputs, decoder_inputs, cell,
110 | num_encoder_symbols=source_vocab_size,
111 | num_decoder_symbols=target_vocab_size,
112 | embedding_size=size,
113 | output_projection=output_projection,
114 | feed_previous=do_decode)
115 |
116 | # Feeds for inputs.
117 | self.encoder_inputs = []
118 | self.decoder_inputs = []
119 | self.target_weights = []
120 | for i in xrange(buckets[-1][0]): # Last bucket is the biggest one.
121 | self.encoder_inputs.append(tf.placeholder(tf.int32, shape=[None],
122 | name="encoder{0}".format(i)))
123 | for i in xrange(buckets[-1][1] + 1):
124 | self.decoder_inputs.append(tf.placeholder(tf.int32, shape=[None],
125 | name="decoder{0}".format(i)))
126 | self.target_weights.append(tf.placeholder(tf.float32, shape=[None],
127 | name="weight{0}".format(i)))
128 |
129 | # Our targets are decoder inputs shifted by one.
130 | targets = [self.decoder_inputs[i + 1]
131 | for i in xrange(len(self.decoder_inputs) - 1)]
132 |
133 | # Training outputs and losses.
134 | if forward_only:
135 | self.outputs, self.losses = tf.nn.seq2seq.model_with_buckets(
136 | self.encoder_inputs, self.decoder_inputs, targets,
137 | self.target_weights, buckets, lambda x, y: seq2seq_f(x, y, True),
138 | softmax_loss_function=softmax_loss_function)
139 | # If we use output projection, we need to project outputs for decoding.
140 | if output_projection is not None:
141 | for b in xrange(len(buckets)):
142 | self.outputs[b] = [
143 | tf.matmul(output, output_projection[0]) + output_projection[1]
144 | for output in self.outputs[b]
145 | ]
146 | else:
147 | self.outputs, self.losses = tf.nn.seq2seq.model_with_buckets(
148 | self.encoder_inputs, self.decoder_inputs, targets,
149 | self.target_weights, buckets,
150 | lambda x, y: seq2seq_f(x, y, False),
151 | softmax_loss_function=softmax_loss_function)
152 |
153 | # Gradients and SGD update operation for training the model.
154 | params = tf.trainable_variables()
155 | if not forward_only:
156 | self.gradient_norms = []
157 | self.updates = []
158 | opt = tf.train.AdamOptimizer()
159 | for b in xrange(len(buckets)):
160 | gradients = tf.gradients(self.losses[b], params)
161 | clipped_gradients, norm = tf.clip_by_global_norm(gradients,
162 | max_gradient_norm)
163 | self.gradient_norms.append(norm)
164 | self.updates.append(opt.apply_gradients(
165 | zip(clipped_gradients, params), global_step=self.global_step))
166 |
167 | self.saver = tf.train.Saver(tf.global_variables())
168 |
169 | def step(self, session, encoder_inputs, decoder_inputs, target_weights,
170 | bucket_id, forward_only):
171 | """Run a step of the model feeding the given inputs.
172 |
173 | Args:
174 | session: tensorflow session to use.
175 | encoder_inputs: list of numpy int vectors to feed as encoder inputs.
176 | decoder_inputs: list of numpy int vectors to feed as decoder inputs.
177 | target_weights: list of numpy float vectors to feed as target weights.
178 | bucket_id: which bucket of the model to use.
179 | forward_only: whether to do the backward step or only forward.
180 |
181 | Returns:
182 | A triple consisting of gradient norm (or None if we did not do backward),
183 | average perplexity, and the outputs.
184 |
185 | Raises:
186 | ValueError: if length of encoder_inputs, decoder_inputs, or
187 | target_weights disagrees with bucket size for the specified bucket_id.
188 | """
189 | # Check if the sizes match.
190 | encoder_size, decoder_size = self.buckets[bucket_id]
191 | if len(encoder_inputs) != encoder_size:
192 | raise ValueError("Encoder length must be equal to the one in bucket,"
193 | " %d != %d." % (len(encoder_inputs), encoder_size))
194 | if len(decoder_inputs) != decoder_size:
195 | raise ValueError("Decoder length must be equal to the one in bucket,"
196 | " %d != %d." % (len(decoder_inputs), decoder_size))
197 | if len(target_weights) != decoder_size:
198 | raise ValueError("Weights length must be equal to the one in bucket,"
199 | " %d != %d." % (len(target_weights), decoder_size))
200 |
201 | # Input feed: encoder inputs, decoder inputs, target_weights, as provided.
202 | input_feed = {}
203 | for l in xrange(encoder_size):
204 | input_feed[self.encoder_inputs[l].name] = encoder_inputs[l]
205 | for l in xrange(decoder_size):
206 | input_feed[self.decoder_inputs[l].name] = decoder_inputs[l]
207 | input_feed[self.target_weights[l].name] = target_weights[l]
208 |
209 | # Since our targets are decoder inputs shifted by one, we need one more.
210 | last_target = self.decoder_inputs[decoder_size].name
211 | input_feed[last_target] = np.zeros([self.batch_size], dtype=np.int32)
212 |
213 | # Output feed: depends on whether we do a backward step or not.
214 | if not forward_only:
215 | output_feed = [self.updates[bucket_id], # Update Op that does SGD.
216 | self.gradient_norms[bucket_id], # Gradient norm.
217 | self.losses[bucket_id]] # Loss for this batch.
218 | else:
219 | output_feed = [self.losses[bucket_id]] # Loss for this batch.
220 | for l in xrange(decoder_size): # Output logits.
221 | output_feed.append(self.outputs[bucket_id][l])
222 |
223 | outputs = session.run(output_feed, input_feed)
224 | if not forward_only:
225 | return outputs[1], outputs[2], None # Gradient norm, loss, no outputs.
226 | else:
227 | return None, outputs[0], outputs[1:] # No gradient norm, loss, outputs.
228 |
229 | def get_batch(self, data, bucket_id):
230 | """Get a random batch of data from the specified bucket, prepare for step.
231 |
232 | To feed data in step(..) it must be a list of batch-major vectors, while
233 | data here contains single length-major cases. So the main logic of this
234 | function is to re-index data cases to be in the proper format for feeding.
235 |
236 | Args:
237 | data: a tuple of size len(self.buckets) in which each element contains
238 | lists of pairs of input and output data that we use to create a batch.
239 | bucket_id: integer, which bucket to get the batch for.
240 |
241 | Returns:
242 | The triple (encoder_inputs, decoder_inputs, target_weights) for
243 | the constructed batch that has the proper format to call step(...) later.
244 | """
245 | encoder_size, decoder_size = self.buckets[bucket_id]
246 | encoder_inputs, decoder_inputs = [], []
247 |
248 | # Get a random batch of encoder and decoder inputs from data,
249 | # pad them if needed, reverse encoder inputs and add GO to decoder.
250 | for _ in xrange(self.batch_size):
251 | encoder_input, decoder_input = random.choice(data[bucket_id])
252 |
253 | # Encoder inputs are padded and then reversed.
254 | encoder_pad = [data_utils.PAD_ID] * (encoder_size - len(encoder_input))
255 | encoder_inputs.append(list(reversed(encoder_input + encoder_pad)))
256 |
257 | # Decoder inputs get an extra "GO" symbol, and are padded then.
258 | decoder_pad_size = decoder_size - len(decoder_input) - 1
259 | decoder_inputs.append([data_utils.GO_ID] + decoder_input +
260 | [data_utils.PAD_ID] * decoder_pad_size)
261 |
262 | # Now we create batch-major vectors from the data selected above.
263 | batch_encoder_inputs, batch_decoder_inputs, batch_weights = [], [], []
264 |
265 | # Batch encoder inputs are just re-indexed encoder_inputs.
266 | for length_idx in xrange(encoder_size):
267 | batch_encoder_inputs.append(
268 | np.array([encoder_inputs[batch_idx][length_idx]
269 | for batch_idx in xrange(self.batch_size)], dtype=np.int32))
270 |
271 | # Batch decoder inputs are re-indexed decoder_inputs, we create weights.
272 | for length_idx in xrange(decoder_size):
273 | batch_decoder_inputs.append(
274 | np.array([decoder_inputs[batch_idx][length_idx]
275 | for batch_idx in xrange(self.batch_size)], dtype=np.int32))
276 |
277 | # Create target_weights to be 0 for targets that are padding.
278 | batch_weight = np.ones(self.batch_size, dtype=np.float32)
279 | for batch_idx in xrange(self.batch_size):
280 | # We set weight to 0 if the corresponding target is a PAD symbol.
281 | # The corresponding target is decoder_input shifted by 1 forward.
282 | if length_idx < decoder_size - 1:
283 | target = decoder_inputs[batch_idx][length_idx + 1]
284 | if length_idx == decoder_size - 1 or target == data_utils.PAD_ID:
285 | batch_weight[batch_idx] = 0.0
286 | batch_weights.append(batch_weight)
287 | return batch_encoder_inputs, batch_decoder_inputs, batch_weights
288 |
--------------------------------------------------------------------------------
/seq2seq_serve.ini:
--------------------------------------------------------------------------------
1 | [strings]
2 | # Mode : train, test, serve
3 | mode = serve
4 | train_enc = data/train.enc
5 | train_dec = data/train.dec
6 | test_enc = data/test.enc
7 | test_dec = data/test.enc
8 | # folder where checkpoints, vocabulary, temporary data will be stored
9 | working_directory = working_dir/
10 | [ints]
11 | # vocabulary size
12 | # 20,000 is a reasonable size
13 | enc_vocab_size = 20000
14 | dec_vocab_size = 20000
15 | # number of LSTM layers : 1/2/3
16 | num_layers = 3
17 | # typical options : 128, 256, 512, 1024
18 | layer_size = 256
19 | # dataset size limit; typically none : no limit
20 | max_train_data_size = 0
21 | batch_size = 64
22 | # steps per checkpoint
23 | # Note : At a checkpoint, models parameters are saved, model is evaluated
24 | # and results are printed
25 | steps_per_checkpoint = 300
26 | [floats]
27 | learning_rate = 0.5
28 | learning_rate_decay_factor = 0.99
29 | max_gradient_norm = 5.0
30 | ##############################################################################
31 | # Note : Edit the bucket sizes at line47 of execute.py (_buckets)
32 | #
33 | # Learn more about the configurations from this link
34 | # https://www.tensorflow.org/versions/r0.9/tutorials/seq2seq/index.html
35 | ##############################################################################
36 |
--------------------------------------------------------------------------------
/ui/.gitignore:
--------------------------------------------------------------------------------
1 | .DS_Store
2 | *.pyc
3 | *.pyo
4 | env
5 | env*
6 | dist
7 | build
8 | *.egg
9 | *.egg-info
10 | _mailinglist
11 | .tox
12 | .cache/
13 | .idea/
14 |
--------------------------------------------------------------------------------
/ui/README.md:
--------------------------------------------------------------------------------
1 | # Flask Chatbox
2 |
3 | A simple flask application that provides an UI for chat and handles the model internally.
4 |
--------------------------------------------------------------------------------
/ui/__init.py__:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/suriyadeepan/easy_seq2seq/c05b0d02f15e9bd1947ff79ddcd0b761a214e2a9/ui/__init.py__
--------------------------------------------------------------------------------
/ui/app.py:
--------------------------------------------------------------------------------
1 | from flask import Flask, render_template, request
2 | from flask import jsonify
3 |
4 | app = Flask(__name__,static_url_path="/static")
5 |
6 | #############
7 | # Routing
8 | #
9 | @app.route('/message', methods=['POST'])
10 | def reply():
11 | return jsonify( { 'text': execute.decode_line(sess, model, enc_vocab, rev_dec_vocab, request.form['msg'] ) } )
12 |
13 | @app.route("/")
14 | def index():
15 | return render_template("index.html")
16 | #############
17 |
18 | '''
19 | Init seq2seq model
20 |
21 | 1. Call main from execute.py
22 | 2. Create decode_line function that takes message as input
23 | '''
24 | #_________________________________________________________________
25 | import sys
26 | import os.path
27 |
28 | sys.path.append(os.path.abspath(os.path.join(os.path.dirname(__file__), os.path.pardir)))
29 | import tensorflow as tf
30 | import execute
31 |
32 | sess = tf.Session()
33 | sess, model, enc_vocab, rev_dec_vocab = execute.init_session(sess, conf='seq2seq_serve.ini')
34 | #_________________________________________________________________
35 |
36 | # start app
37 | if (__name__ == "__main__"):
38 | app.run(port = 5000)
39 |
--------------------------------------------------------------------------------
/ui/requirements.txt:
--------------------------------------------------------------------------------
1 | click==6.6
2 | Flask==0.11.1
3 | itsdangerous==0.24
4 | Jinja2==2.8
5 | MarkupSafe==0.23
6 | Werkzeug==0.11.10
7 |
--------------------------------------------------------------------------------
/ui/setup.sh:
--------------------------------------------------------------------------------
1 | sudo -H pip install --upgrade pip
2 | sudo -H pip install -r requirements.txt
3 |
--------------------------------------------------------------------------------
/ui/static/css/normalize.css:
--------------------------------------------------------------------------------
1 | /*! normalize.css v4.0.0 | MIT License | github.com/necolas/normalize.css */html{font-family:sans-serif;-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%}body{margin:0}article,aside,details,figcaption,figure,footer,header,main,menu,nav,section,summary{display:block}audio,canvas,progress,video{display:inline-block}audio:not([controls]){display:none;height:0}progress{vertical-align:baseline}template,[hidden]{display:none}a{background-color:transparent}a:active,a:hover{outline-width:0}abbr[title]{border-bottom:none;text-decoration:underline;text-decoration:underline dotted}b,strong{font-weight:inherit}b,strong{font-weight:bolder}dfn{font-style:italic}h1{font-size:2em;margin:0.67em 0}mark{background-color:#ff0;color:#000}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-0.25em}sup{top:-0.5em}img{border-style:none}svg:not(:root){overflow:hidden}code,kbd,pre,samp{font-family:monospace, monospace;font-size:1em}figure{margin:1em 40px}hr{-webkit-box-sizing:content-box;-moz-box-sizing:content-box;box-sizing:content-box;height:0;overflow:visible}button,input,select,textarea{font:inherit;margin:0}optgroup{font-weight:bold}button,input,select{overflow:visible}button,select{text-transform:none}button,[type="button"],[type="reset"],[type="submit"]{cursor:pointer}[disabled]{cursor:default}button,html [type="button"],[type="reset"],[type="submit"]{-webkit-appearance:button}button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0}button:-moz-focusring,input:-moz-focusring{outline:1px dotted ButtonText}fieldset{border:1px solid #c0c0c0;margin:0 2px;padding:0.35em 0.625em 0.75em}legend{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;color:inherit;display:table;max-width:100%;padding:0;white-space:normal}textarea{overflow:auto}[type="checkbox"],[type="radio"]{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;padding:0}[type="number"]::-webkit-inner-spin-button,[type="number"]::-webkit-outer-spin-button{height:auto}[type="search"]{-webkit-appearance:textfield}[type="search"]::-webkit-search-cancel-button,[type="search"]::-webkit-search-decoration{-webkit-appearance:none}
2 |
--------------------------------------------------------------------------------
/ui/static/css/style.css:
--------------------------------------------------------------------------------
1 | /*--------------------
2 | Mixins
3 | --------------------*/
4 | /*--------------------
5 | Body
6 | --------------------*/
7 | *,
8 | *::before,
9 | *::after {
10 | box-sizing: border-box;
11 | }
12 |
13 | html,
14 | body {
15 | height: 100%;
16 | }
17 |
18 | body {
19 | background: -webkit-linear-gradient(315deg, #044f48, #2a7561);
20 | background: linear-gradient(135deg, #044f48, #2a7561);
21 | background-size: cover;
22 | font-family: 'Open Sans', sans-serif;
23 | font-size: 12px;
24 | line-height: 1.3;
25 | overflow: hidden;
26 | }
27 |
28 | .bg {
29 | width: 100%;
30 | height: 100%;
31 | top: 0;
32 | left: 0;
33 | z-index: 1;
34 | background: url("/static/res/hd2.jpg") no-repeat 0 0;
35 | -webkit-filter: blur(80px);
36 | filter: blur(80px);
37 | -webkit-transform: scale(1.2);
38 | transform: scale(1.2);
39 | }
40 |
41 | /*--------------------
42 | Chat
43 | --------------------*/
44 | .chat {
45 | position: absolute;
46 | top: 50%;
47 | left: 50%;
48 | -webkit-transform: translate(-50%, -50%);
49 | transform: translate(-50%, -50%);
50 | width: 300px;
51 | height: 80vh;
52 | max-height: 500px;
53 | z-index: 2;
54 | overflow: hidden;
55 | box-shadow: 0 5px 30px rgba(0, 0, 0, 0.2);
56 | background: rgba(0, 0, 0, 0.5);
57 | border-radius: 20px;
58 | display: -webkit-box;
59 | display: -webkit-flex;
60 | display: -ms-flexbox;
61 | display: flex;
62 | -webkit-box-pack: justify;
63 | -webkit-justify-content: space-between;
64 | -ms-flex-pack: justify;
65 | justify-content: space-between;
66 | -webkit-box-orient: vertical;
67 | -webkit-box-direction: normal;
68 | -webkit-flex-direction: column;
69 | -ms-flex-direction: column;
70 | flex-direction: column;
71 | }
72 |
73 | /*--------------------
74 | Chat Title
75 | --------------------*/
76 | .chat-title {
77 | -webkit-box-flex: 0;
78 | -webkit-flex: 0 1 45px;
79 | -ms-flex: 0 1 45px;
80 | flex: 0 1 45px;
81 | position: relative;
82 | z-index: 2;
83 | background: rgba(0, 0, 0, 0.2);
84 | color: #fff;
85 | text-transform: uppercase;
86 | text-align: left;
87 | padding: 10px 10px 10px 50px;
88 | }
89 | .chat-title h1, .chat-title h2 {
90 | font-weight: normal;
91 | font-size: 10px;
92 | margin: 0;
93 | padding: 0;
94 | }
95 | .chat-title h2 {
96 | color: rgba(255, 255, 255, 0.5);
97 | font-size: 8px;
98 | letter-spacing: 1px;
99 | }
100 | .chat-title .avatar {
101 | position: absolute;
102 | z-index: 1;
103 | top: 8px;
104 | left: 9px;
105 | border-radius: 30px;
106 | width: 30px;
107 | height: 30px;
108 | overflow: hidden;
109 | margin: 0;
110 | padding: 0;
111 | border: 2px solid rgba(255, 255, 255, 0.24);
112 | }
113 | .chat-title .avatar img {
114 | width: 100%;
115 | height: auto;
116 | }
117 |
118 | /*--------------------
119 | Messages
120 | --------------------*/
121 | .messages {
122 | -webkit-box-flex: 1;
123 | -webkit-flex: 1 1 auto;
124 | -ms-flex: 1 1 auto;
125 | flex: 1 1 auto;
126 | color: rgba(255, 255, 255, 0.5);
127 | overflow: hidden;
128 | position: relative;
129 | width: 100%;
130 | }
131 | .messages .messages-content {
132 | position: absolute;
133 | top: 0;
134 | left: 0;
135 | height: 101%;
136 | width: 100%;
137 | }
138 | .messages .message {
139 | clear: both;
140 | float: left;
141 | padding: 6px 10px 7px;
142 | border-radius: 10px 10px 10px 0;
143 | background: rgba(0, 0, 0, 0.3);
144 | margin: 8px 0;
145 | font-size: 11px;
146 | line-height: 1.4;
147 | margin-left: 35px;
148 | position: relative;
149 | text-shadow: 0 1px 1px rgba(0, 0, 0, 0.2);
150 | }
151 | .messages .message .timestamp {
152 | position: absolute;
153 | bottom: -15px;
154 | font-size: 9px;
155 | color: rgba(255, 255, 255, 0.3);
156 | }
157 | .messages .message::before {
158 | content: '';
159 | position: absolute;
160 | bottom: -6px;
161 | border-top: 6px solid rgba(0, 0, 0, 0.3);
162 | left: 0;
163 | border-right: 7px solid transparent;
164 | }
165 | .messages .message .avatar {
166 | position: absolute;
167 | z-index: 1;
168 | bottom: -15px;
169 | left: -35px;
170 | border-radius: 30px;
171 | width: 30px;
172 | height: 30px;
173 | overflow: hidden;
174 | margin: 0;
175 | padding: 0;
176 | border: 2px solid rgba(255, 255, 255, 0.24);
177 | }
178 | .messages .message .avatar img {
179 | width: 100%;
180 | height: auto;
181 | }
182 | .messages .message.message-personal {
183 | float: right;
184 | color: #fff;
185 | text-align: right;
186 | background: -webkit-linear-gradient(330deg, #248A52, #257287);
187 | background: linear-gradient(120deg, #248A52, #257287);
188 | border-radius: 10px 10px 0 10px;
189 | }
190 | .messages .message.message-personal::before {
191 | left: auto;
192 | right: 0;
193 | border-right: none;
194 | border-left: 5px solid transparent;
195 | border-top: 4px solid #257287;
196 | bottom: -4px;
197 | }
198 | .messages .message:last-child {
199 | margin-bottom: 30px;
200 | }
201 | .messages .message.new {
202 | -webkit-transform: scale(0);
203 | transform: scale(0);
204 | -webkit-transform-origin: 0 0;
205 | transform-origin: 0 0;
206 | -webkit-animation: bounce 500ms linear both;
207 | animation: bounce 500ms linear both;
208 | }
209 | .messages .message.loading::before {
210 | position: absolute;
211 | top: 50%;
212 | left: 50%;
213 | -webkit-transform: translate(-50%, -50%);
214 | transform: translate(-50%, -50%);
215 | content: '';
216 | display: block;
217 | width: 3px;
218 | height: 3px;
219 | border-radius: 50%;
220 | background: rgba(255, 255, 255, 0.5);
221 | z-index: 2;
222 | margin-top: 4px;
223 | -webkit-animation: ball 0.45s cubic-bezier(0, 0, 0.15, 1) alternate infinite;
224 | animation: ball 0.45s cubic-bezier(0, 0, 0.15, 1) alternate infinite;
225 | border: none;
226 | -webkit-animation-delay: .15s;
227 | animation-delay: .15s;
228 | }
229 | .messages .message.loading span {
230 | display: block;
231 | font-size: 0;
232 | width: 20px;
233 | height: 10px;
234 | position: relative;
235 | }
236 | .messages .message.loading span::before {
237 | position: absolute;
238 | top: 50%;
239 | left: 50%;
240 | -webkit-transform: translate(-50%, -50%);
241 | transform: translate(-50%, -50%);
242 | content: '';
243 | display: block;
244 | width: 3px;
245 | height: 3px;
246 | border-radius: 50%;
247 | background: rgba(255, 255, 255, 0.5);
248 | z-index: 2;
249 | margin-top: 4px;
250 | -webkit-animation: ball 0.45s cubic-bezier(0, 0, 0.15, 1) alternate infinite;
251 | animation: ball 0.45s cubic-bezier(0, 0, 0.15, 1) alternate infinite;
252 | margin-left: -7px;
253 | }
254 | .messages .message.loading span::after {
255 | position: absolute;
256 | top: 50%;
257 | left: 50%;
258 | -webkit-transform: translate(-50%, -50%);
259 | transform: translate(-50%, -50%);
260 | content: '';
261 | display: block;
262 | width: 3px;
263 | height: 3px;
264 | border-radius: 50%;
265 | background: rgba(255, 255, 255, 0.5);
266 | z-index: 2;
267 | margin-top: 4px;
268 | -webkit-animation: ball 0.45s cubic-bezier(0, 0, 0.15, 1) alternate infinite;
269 | animation: ball 0.45s cubic-bezier(0, 0, 0.15, 1) alternate infinite;
270 | margin-left: 7px;
271 | -webkit-animation-delay: .3s;
272 | animation-delay: .3s;
273 | }
274 |
275 | /*--------------------
276 | Message Box
277 | --------------------*/
278 | .message-box {
279 | -webkit-box-flex: 0;
280 | -webkit-flex: 0 1 40px;
281 | -ms-flex: 0 1 40px;
282 | flex: 0 1 40px;
283 | width: 100%;
284 | background: rgba(0, 0, 0, 0.3);
285 | padding: 10px;
286 | position: relative;
287 | }
288 | .message-box .message-input {
289 | background: none;
290 | border: none;
291 | outline: none !important;
292 | resize: none;
293 | color: rgba(255, 255, 255, 0.7);
294 | font-size: 11px;
295 | height: 17px;
296 | margin: 0;
297 | padding-right: 20px;
298 | width: 265px;
299 | }
300 | .message-box textarea:focus:-webkit-placeholder {
301 | color: transparent;
302 | }
303 | .message-box .message-submit {
304 | position: absolute;
305 | z-index: 1;
306 | top: 9px;
307 | right: 10px;
308 | color: #fff;
309 | border: none;
310 | background: #248A52;
311 | font-size: 10px;
312 | text-transform: uppercase;
313 | line-height: 1;
314 | padding: 6px 10px;
315 | border-radius: 10px;
316 | outline: none !important;
317 | -webkit-transition: background .2s ease;
318 | transition: background .2s ease;
319 | }
320 | .message-box .message-submit:hover {
321 | background: #1D7745;
322 | }
323 |
324 | /*--------------------
325 | Custom Srollbar
326 | --------------------*/
327 | .mCSB_scrollTools {
328 | margin: 1px -3px 1px 0;
329 | opacity: 0;
330 | }
331 |
332 | .mCSB_inside > .mCSB_container {
333 | margin-right: 0px;
334 | padding: 0 10px;
335 | }
336 |
337 | .mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar {
338 | background-color: rgba(0, 0, 0, 0.5) !important;
339 | }
340 |
341 | /*--------------------
342 | Bounce
343 | --------------------*/
344 | @-webkit-keyframes bounce {
345 | 0% {
346 | -webkit-transform: matrix3d(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
347 | transform: matrix3d(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
348 | }
349 | 4.7% {
350 | -webkit-transform: matrix3d(0.45, 0, 0, 0, 0, 0.45, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
351 | transform: matrix3d(0.45, 0, 0, 0, 0, 0.45, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
352 | }
353 | 9.41% {
354 | -webkit-transform: matrix3d(0.883, 0, 0, 0, 0, 0.883, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
355 | transform: matrix3d(0.883, 0, 0, 0, 0, 0.883, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
356 | }
357 | 14.11% {
358 | -webkit-transform: matrix3d(1.141, 0, 0, 0, 0, 1.141, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
359 | transform: matrix3d(1.141, 0, 0, 0, 0, 1.141, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
360 | }
361 | 18.72% {
362 | -webkit-transform: matrix3d(1.212, 0, 0, 0, 0, 1.212, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
363 | transform: matrix3d(1.212, 0, 0, 0, 0, 1.212, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
364 | }
365 | 24.32% {
366 | -webkit-transform: matrix3d(1.151, 0, 0, 0, 0, 1.151, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
367 | transform: matrix3d(1.151, 0, 0, 0, 0, 1.151, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
368 | }
369 | 29.93% {
370 | -webkit-transform: matrix3d(1.048, 0, 0, 0, 0, 1.048, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
371 | transform: matrix3d(1.048, 0, 0, 0, 0, 1.048, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
372 | }
373 | 35.54% {
374 | -webkit-transform: matrix3d(0.979, 0, 0, 0, 0, 0.979, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
375 | transform: matrix3d(0.979, 0, 0, 0, 0, 0.979, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
376 | }
377 | 41.04% {
378 | -webkit-transform: matrix3d(0.961, 0, 0, 0, 0, 0.961, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
379 | transform: matrix3d(0.961, 0, 0, 0, 0, 0.961, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
380 | }
381 | 52.15% {
382 | -webkit-transform: matrix3d(0.991, 0, 0, 0, 0, 0.991, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
383 | transform: matrix3d(0.991, 0, 0, 0, 0, 0.991, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
384 | }
385 | 63.26% {
386 | -webkit-transform: matrix3d(1.007, 0, 0, 0, 0, 1.007, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
387 | transform: matrix3d(1.007, 0, 0, 0, 0, 1.007, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
388 | }
389 | 85.49% {
390 | -webkit-transform: matrix3d(0.999, 0, 0, 0, 0, 0.999, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
391 | transform: matrix3d(0.999, 0, 0, 0, 0, 0.999, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
392 | }
393 | 100% {
394 | -webkit-transform: matrix3d(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
395 | transform: matrix3d(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
396 | }
397 | }
398 | @keyframes bounce {
399 | 0% {
400 | -webkit-transform: matrix3d(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
401 | transform: matrix3d(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
402 | }
403 | 4.7% {
404 | -webkit-transform: matrix3d(0.45, 0, 0, 0, 0, 0.45, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
405 | transform: matrix3d(0.45, 0, 0, 0, 0, 0.45, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
406 | }
407 | 9.41% {
408 | -webkit-transform: matrix3d(0.883, 0, 0, 0, 0, 0.883, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
409 | transform: matrix3d(0.883, 0, 0, 0, 0, 0.883, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
410 | }
411 | 14.11% {
412 | -webkit-transform: matrix3d(1.141, 0, 0, 0, 0, 1.141, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
413 | transform: matrix3d(1.141, 0, 0, 0, 0, 1.141, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
414 | }
415 | 18.72% {
416 | -webkit-transform: matrix3d(1.212, 0, 0, 0, 0, 1.212, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
417 | transform: matrix3d(1.212, 0, 0, 0, 0, 1.212, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
418 | }
419 | 24.32% {
420 | -webkit-transform: matrix3d(1.151, 0, 0, 0, 0, 1.151, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
421 | transform: matrix3d(1.151, 0, 0, 0, 0, 1.151, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
422 | }
423 | 29.93% {
424 | -webkit-transform: matrix3d(1.048, 0, 0, 0, 0, 1.048, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
425 | transform: matrix3d(1.048, 0, 0, 0, 0, 1.048, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
426 | }
427 | 35.54% {
428 | -webkit-transform: matrix3d(0.979, 0, 0, 0, 0, 0.979, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
429 | transform: matrix3d(0.979, 0, 0, 0, 0, 0.979, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
430 | }
431 | 41.04% {
432 | -webkit-transform: matrix3d(0.961, 0, 0, 0, 0, 0.961, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
433 | transform: matrix3d(0.961, 0, 0, 0, 0, 0.961, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
434 | }
435 | 52.15% {
436 | -webkit-transform: matrix3d(0.991, 0, 0, 0, 0, 0.991, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
437 | transform: matrix3d(0.991, 0, 0, 0, 0, 0.991, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
438 | }
439 | 63.26% {
440 | -webkit-transform: matrix3d(1.007, 0, 0, 0, 0, 1.007, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
441 | transform: matrix3d(1.007, 0, 0, 0, 0, 1.007, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
442 | }
443 | 85.49% {
444 | -webkit-transform: matrix3d(0.999, 0, 0, 0, 0, 0.999, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
445 | transform: matrix3d(0.999, 0, 0, 0, 0, 0.999, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
446 | }
447 | 100% {
448 | -webkit-transform: matrix3d(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
449 | transform: matrix3d(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1);
450 | }
451 | }
452 | @-webkit-keyframes ball {
453 | from {
454 | -webkit-transform: translateY(0) scaleY(0.8);
455 | transform: translateY(0) scaleY(0.8);
456 | }
457 | to {
458 | -webkit-transform: translateY(-10px);
459 | transform: translateY(-10px);
460 | }
461 | }
462 | @keyframes ball {
463 | from {
464 | -webkit-transform: translateY(0) scaleY(0.8);
465 | transform: translateY(0) scaleY(0.8);
466 | }
467 | to {
468 | -webkit-transform: translateY(-10px);
469 | transform: translateY(-10px);
470 | }
471 | }
472 |
--------------------------------------------------------------------------------
/ui/static/js/index.js:
--------------------------------------------------------------------------------
1 | var $messages = $('.messages-content'),
2 | d, h, m,
3 | i = 0;
4 |
5 | $(window).load(function() {
6 | $messages.mCustomScrollbar();
7 | setTimeout(function() {
8 | //fakeMessage();
9 | }, 100);
10 | });
11 |
12 | function updateScrollbar() {
13 | $messages.mCustomScrollbar("update").mCustomScrollbar('scrollTo', 'bottom', {
14 | scrollInertia: 10,
15 | timeout: 0
16 | });
17 | }
18 |
19 | function setDate(){
20 | d = new Date()
21 | if (m != d.getMinutes()) {
22 | m = d.getMinutes();
23 | $('
' + d.getHours() + ':' + m + '
').appendTo($('.message:last'));
24 | }
25 | }
26 |
27 | function insertMessage() {
28 | msg = $('.message-input').val();
29 | if ($.trim(msg) == '') {
30 | return false;
31 | }
32 | $('' + msg + '
').appendTo($('.mCSB_container')).addClass('new');
33 | setDate();
34 | $('.message-input').val(null);
35 | updateScrollbar();
36 | interact(msg);
37 | setTimeout(function() {
38 | //fakeMessage();
39 | }, 1000 + (Math.random() * 20) * 100);
40 | }
41 |
42 | $('.message-submit').click(function() {
43 | insertMessage();
44 | });
45 |
46 | $(window).on('keydown', function(e) {
47 | if (e.which == 13) {
48 | insertMessage();
49 | return false;
50 | }
51 | })
52 |
53 |
54 | function interact(message){
55 | // loading message
56 | $('').appendTo($('.mCSB_container'));
57 | // make a POST request [ajax call]
58 | $.post('/message', {
59 | msg: message,
60 | }).done(function(reply) {
61 | // Message Received
62 | // remove loading meassage
63 | $('.message.loading').remove();
64 | // Add message to chatbox
65 | $('
' + reply['text'] + '
').appendTo($('.mCSB_container')).addClass('new');
66 | setDate();
67 | updateScrollbar();
68 |
69 | }).fail(function() {
70 | alert('error calling function');
71 | });
72 | }
73 |
--------------------------------------------------------------------------------
/ui/static/js/jquery.mCustomScrollbar.concat.min.js:
--------------------------------------------------------------------------------
1 | /* == jquery mousewheel plugin == Version: 3.1.13, License: MIT License (MIT) */
2 | !function(a){"function"==typeof define&&define.amd?define(["jquery"],a):"object"==typeof exports?module.exports=a:a(jQuery)}(function(a){function b(b){var g=b||window.event,h=i.call(arguments,1),j=0,l=0,m=0,n=0,o=0,p=0;if(b=a.event.fix(g),b.type="mousewheel","detail"in g&&(m=-1*g.detail),"wheelDelta"in g&&(m=g.wheelDelta),"wheelDeltaY"in g&&(m=g.wheelDeltaY),"wheelDeltaX"in g&&(l=-1*g.wheelDeltaX),"axis"in g&&g.axis===g.HORIZONTAL_AXIS&&(l=-1*m,m=0),j=0===m?l:m,"deltaY"in g&&(m=-1*g.deltaY,j=m),"deltaX"in g&&(l=g.deltaX,0===m&&(j=-1*l)),0!==m||0!==l){if(1===g.deltaMode){var q=a.data(this,"mousewheel-line-height");j*=q,m*=q,l*=q}else if(2===g.deltaMode){var r=a.data(this,"mousewheel-page-height");j*=r,m*=r,l*=r}if(n=Math.max(Math.abs(m),Math.abs(l)),(!f||f>n)&&(f=n,d(g,n)&&(f/=40)),d(g,n)&&(j/=40,l/=40,m/=40),j=Math[j>=1?"floor":"ceil"](j/f),l=Math[l>=1?"floor":"ceil"](l/f),m=Math[m>=1?"floor":"ceil"](m/f),k.settings.normalizeOffset&&this.getBoundingClientRect){var s=this.getBoundingClientRect();o=b.clientX-s.left,p=b.clientY-s.top}return b.deltaX=l,b.deltaY=m,b.deltaFactor=f,b.offsetX=o,b.offsetY=p,b.deltaMode=0,h.unshift(b,j,l,m),e&&clearTimeout(e),e=setTimeout(c,200),(a.event.dispatch||a.event.handle).apply(this,h)}}function c(){f=null}function d(a,b){return k.settings.adjustOldDeltas&&"mousewheel"===a.type&&b%120===0}var e,f,g=["wheel","mousewheel","DOMMouseScroll","MozMousePixelScroll"],h="onwheel"in document||document.documentMode>=9?["wheel"]:["mousewheel","DomMouseScroll","MozMousePixelScroll"],i=Array.prototype.slice;if(a.event.fixHooks)for(var j=g.length;j;)a.event.fixHooks[g[--j]]=a.event.mouseHooks;var k=a.event.special.mousewheel={version:"3.1.12",setup:function(){if(this.addEventListener)for(var c=h.length;c;)this.addEventListener(h[--c],b,!1);else this.onmousewheel=b;a.data(this,"mousewheel-line-height",k.getLineHeight(this)),a.data(this,"mousewheel-page-height",k.getPageHeight(this))},teardown:function(){if(this.removeEventListener)for(var c=h.length;c;)this.removeEventListener(h[--c],b,!1);else this.onmousewheel=null;a.removeData(this,"mousewheel-line-height"),a.removeData(this,"mousewheel-page-height")},getLineHeight:function(b){var c=a(b),d=c["offsetParent"in a.fn?"offsetParent":"parent"]();return d.length||(d=a("body")),parseInt(d.css("fontSize"),10)||parseInt(c.css("fontSize"),10)||16},getPageHeight:function(b){return a(b).height()},settings:{adjustOldDeltas:!0,normalizeOffset:!0}};a.fn.extend({mousewheel:function(a){return a?this.bind("mousewheel",a):this.trigger("mousewheel")},unmousewheel:function(a){return this.unbind("mousewheel",a)}})});!function(a){"function"==typeof define&&define.amd?define(["jquery"],a):"object"==typeof exports?module.exports=a:a(jQuery)}(function(a){function b(b){var g=b||window.event,h=i.call(arguments,1),j=0,l=0,m=0,n=0,o=0,p=0;if(b=a.event.fix(g),b.type="mousewheel","detail"in g&&(m=-1*g.detail),"wheelDelta"in g&&(m=g.wheelDelta),"wheelDeltaY"in g&&(m=g.wheelDeltaY),"wheelDeltaX"in g&&(l=-1*g.wheelDeltaX),"axis"in g&&g.axis===g.HORIZONTAL_AXIS&&(l=-1*m,m=0),j=0===m?l:m,"deltaY"in g&&(m=-1*g.deltaY,j=m),"deltaX"in g&&(l=g.deltaX,0===m&&(j=-1*l)),0!==m||0!==l){if(1===g.deltaMode){var q=a.data(this,"mousewheel-line-height");j*=q,m*=q,l*=q}else if(2===g.deltaMode){var r=a.data(this,"mousewheel-page-height");j*=r,m*=r,l*=r}if(n=Math.max(Math.abs(m),Math.abs(l)),(!f||f>n)&&(f=n,d(g,n)&&(f/=40)),d(g,n)&&(j/=40,l/=40,m/=40),j=Math[j>=1?"floor":"ceil"](j/f),l=Math[l>=1?"floor":"ceil"](l/f),m=Math[m>=1?"floor":"ceil"](m/f),k.settings.normalizeOffset&&this.getBoundingClientRect){var s=this.getBoundingClientRect();o=b.clientX-s.left,p=b.clientY-s.top}return b.deltaX=l,b.deltaY=m,b.deltaFactor=f,b.offsetX=o,b.offsetY=p,b.deltaMode=0,h.unshift(b,j,l,m),e&&clearTimeout(e),e=setTimeout(c,200),(a.event.dispatch||a.event.handle).apply(this,h)}}function c(){f=null}function d(a,b){return k.settings.adjustOldDeltas&&"mousewheel"===a.type&&b%120===0}var e,f,g=["wheel","mousewheel","DOMMouseScroll","MozMousePixelScroll"],h="onwheel"in document||document.documentMode>=9?["wheel"]:["mousewheel","DomMouseScroll","MozMousePixelScroll"],i=Array.prototype.slice;if(a.event.fixHooks)for(var j=g.length;j;)a.event.fixHooks[g[--j]]=a.event.mouseHooks;var k=a.event.special.mousewheel={version:"3.1.12",setup:function(){if(this.addEventListener)for(var c=h.length;c;)this.addEventListener(h[--c],b,!1);else this.onmousewheel=b;a.data(this,"mousewheel-line-height",k.getLineHeight(this)),a.data(this,"mousewheel-page-height",k.getPageHeight(this))},teardown:function(){if(this.removeEventListener)for(var c=h.length;c;)this.removeEventListener(h[--c],b,!1);else this.onmousewheel=null;a.removeData(this,"mousewheel-line-height"),a.removeData(this,"mousewheel-page-height")},getLineHeight:function(b){var c=a(b),d=c["offsetParent"in a.fn?"offsetParent":"parent"]();return d.length||(d=a("body")),parseInt(d.css("fontSize"),10)||parseInt(c.css("fontSize"),10)||16},getPageHeight:function(b){return a(b).height()},settings:{adjustOldDeltas:!0,normalizeOffset:!0}};a.fn.extend({mousewheel:function(a){return a?this.bind("mousewheel",a):this.trigger("mousewheel")},unmousewheel:function(a){return this.unbind("mousewheel",a)}})});
3 | /* == malihu jquery custom scrollbar plugin == Version: 3.1.3, License: MIT License (MIT) */
4 | !function(e){"undefined"!=typeof module&&module.exports?module.exports=e:e(jQuery,window,document)}(function(e){!function(t){var o="function"==typeof define&&define.amd,a="undefined"!=typeof module&&module.exports,n="https:"==document.location.protocol?"https:":"http:",i="cdnjs.cloudflare.com/ajax/libs/jquery-mousewheel/3.1.13/jquery.mousewheel.min.js";o||(a?require("jquery-mousewheel")(e):e.event.special.mousewheel||e("head").append(decodeURI("%3Cscript src="+n+"//"+i+"%3E%3C/script%3E"))),t()}(function(){var t,o="mCustomScrollbar",a="mCS",n=".mCustomScrollbar",i={setTop:0,setLeft:0,axis:"y",scrollbarPosition:"inside",scrollInertia:950,autoDraggerLength:!0,alwaysShowScrollbar:0,snapOffset:0,mouseWheel:{enable:!0,scrollAmount:"auto",axis:"y",deltaFactor:"auto",disableOver:["select","option","keygen","datalist","textarea"]},scrollButtons:{scrollType:"stepless",scrollAmount:"auto"},keyboard:{enable:!0,scrollType:"stepless",scrollAmount:"auto"},contentTouchScroll:25,documentTouchScroll:!0,advanced:{autoScrollOnFocus:"input,textarea,select,button,datalist,keygen,a[tabindex],area,object,[contenteditable='true']",updateOnContentResize:!0,updateOnImageLoad:"auto",autoUpdateTimeout:60},theme:"light",callbacks:{onTotalScrollOffset:0,onTotalScrollBackOffset:0,alwaysTriggerOffsets:!0}},r=0,l={},s=window.attachEvent&&!window.addEventListener?1:0,c=!1,d=["mCSB_dragger_onDrag","mCSB_scrollTools_onDrag","mCS_img_loaded","mCS_disabled","mCS_destroyed","mCS_no_scrollbar","mCS-autoHide","mCS-dir-rtl","mCS_no_scrollbar_y","mCS_no_scrollbar_x","mCS_y_hidden","mCS_x_hidden","mCSB_draggerContainer","mCSB_buttonUp","mCSB_buttonDown","mCSB_buttonLeft","mCSB_buttonRight"],u={init:function(t){var t=e.extend(!0,{},i,t),o=f.call(this);if(t.live){var s=t.liveSelector||this.selector||n,c=e(s);if("off"===t.live)return void m(s);l[s]=setTimeout(function(){c.mCustomScrollbar(t),"once"===t.live&&c.length&&m(s)},500)}else m(s);return t.setWidth=t.set_width?t.set_width:t.setWidth,t.setHeight=t.set_height?t.set_height:t.setHeight,t.axis=t.horizontalScroll?"x":p(t.axis),t.scrollInertia=t.scrollInertia>0&&t.scrollInertia<17?17:t.scrollInertia,"object"!=typeof t.mouseWheel&&1==t.mouseWheel&&(t.mouseWheel={enable:!0,scrollAmount:"auto",axis:"y",preventDefault:!1,deltaFactor:"auto",normalizeDelta:!1,invert:!1}),t.mouseWheel.scrollAmount=t.mouseWheelPixels?t.mouseWheelPixels:t.mouseWheel.scrollAmount,t.mouseWheel.normalizeDelta=t.advanced.normalizeMouseWheelDelta?t.advanced.normalizeMouseWheelDelta:t.mouseWheel.normalizeDelta,t.scrollButtons.scrollType=g(t.scrollButtons.scrollType),h(t),e(o).each(function(){var o=e(this);if(!o.data(a)){o.data(a,{idx:++r,opt:t,scrollRatio:{y:null,x:null},overflowed:null,contentReset:{y:null,x:null},bindEvents:!1,tweenRunning:!1,sequential:{},langDir:o.css("direction"),cbOffsets:null,trigger:null,poll:{size:{o:0,n:0},img:{o:0,n:0},change:{o:0,n:0}}});var n=o.data(a),i=n.opt,l=o.data("mcs-axis"),s=o.data("mcs-scrollbar-position"),c=o.data("mcs-theme");l&&(i.axis=l),s&&(i.scrollbarPosition=s),c&&(i.theme=c,h(i)),v.call(this),n&&i.callbacks.onCreate&&"function"==typeof i.callbacks.onCreate&&i.callbacks.onCreate.call(this),e("#mCSB_"+n.idx+"_container img:not(."+d[2]+")").addClass(d[2]),u.update.call(null,o)}})},update:function(t,o){var n=t||f.call(this);return e(n).each(function(){var t=e(this);if(t.data(a)){var n=t.data(a),i=n.opt,r=e("#mCSB_"+n.idx+"_container"),l=e("#mCSB_"+n.idx),s=[e("#mCSB_"+n.idx+"_dragger_vertical"),e("#mCSB_"+n.idx+"_dragger_horizontal")];if(!r.length)return;n.tweenRunning&&N(t),o&&n&&i.callbacks.onBeforeUpdate&&"function"==typeof i.callbacks.onBeforeUpdate&&i.callbacks.onBeforeUpdate.call(this),t.hasClass(d[3])&&t.removeClass(d[3]),t.hasClass(d[4])&&t.removeClass(d[4]),l.css("max-height","none"),l.height()!==t.height()&&l.css("max-height",t.height()),_.call(this),"y"===i.axis||i.advanced.autoExpandHorizontalScroll||r.css("width",x(r)),n.overflowed=y.call(this),M.call(this),i.autoDraggerLength&&S.call(this),b.call(this),T.call(this);var c=[Math.abs(r[0].offsetTop),Math.abs(r[0].offsetLeft)];"x"!==i.axis&&(n.overflowed[0]?s[0].height()>s[0].parent().height()?B.call(this):(V(t,c[0].toString(),{dir:"y",dur:0,overwrite:"none"}),n.contentReset.y=null):(B.call(this),"y"===i.axis?k.call(this):"yx"===i.axis&&n.overflowed[1]&&V(t,c[1].toString(),{dir:"x",dur:0,overwrite:"none"}))),"y"!==i.axis&&(n.overflowed[1]?s[1].width()>s[1].parent().width()?B.call(this):(V(t,c[1].toString(),{dir:"x",dur:0,overwrite:"none"}),n.contentReset.x=null):(B.call(this),"x"===i.axis?k.call(this):"yx"===i.axis&&n.overflowed[0]&&V(t,c[0].toString(),{dir:"y",dur:0,overwrite:"none"}))),o&&n&&(2===o&&i.callbacks.onImageLoad&&"function"==typeof i.callbacks.onImageLoad?i.callbacks.onImageLoad.call(this):3===o&&i.callbacks.onSelectorChange&&"function"==typeof i.callbacks.onSelectorChange?i.callbacks.onSelectorChange.call(this):i.callbacks.onUpdate&&"function"==typeof i.callbacks.onUpdate&&i.callbacks.onUpdate.call(this)),X.call(this)}})},scrollTo:function(t,o){if("undefined"!=typeof t&&null!=t){var n=f.call(this);return e(n).each(function(){var n=e(this);if(n.data(a)){var i=n.data(a),r=i.opt,l={trigger:"external",scrollInertia:r.scrollInertia,scrollEasing:"mcsEaseInOut",moveDragger:!1,timeout:60,callbacks:!0,onStart:!0,onUpdate:!0,onComplete:!0},s=e.extend(!0,{},l,o),c=q.call(this,t),d=s.scrollInertia>0&&s.scrollInertia<17?17:s.scrollInertia;c[0]=Y.call(this,c[0],"y"),c[1]=Y.call(this,c[1],"x"),s.moveDragger&&(c[0]*=i.scrollRatio.y,c[1]*=i.scrollRatio.x),s.dur=oe()?0:d,setTimeout(function(){null!==c[0]&&"undefined"!=typeof c[0]&&"x"!==r.axis&&i.overflowed[0]&&(s.dir="y",s.overwrite="all",V(n,c[0].toString(),s)),null!==c[1]&&"undefined"!=typeof c[1]&&"y"!==r.axis&&i.overflowed[1]&&(s.dir="x",s.overwrite="none",V(n,c[1].toString(),s))},s.timeout)}})}},stop:function(){var t=f.call(this);return e(t).each(function(){var t=e(this);t.data(a)&&N(t)})},disable:function(t){var o=f.call(this);return e(o).each(function(){var o=e(this);if(o.data(a)){{o.data(a)}X.call(this,"remove"),k.call(this),t&&B.call(this),M.call(this,!0),o.addClass(d[3])}})},destroy:function(){var t=f.call(this);return e(t).each(function(){var n=e(this);if(n.data(a)){var i=n.data(a),r=i.opt,l=e("#mCSB_"+i.idx),s=e("#mCSB_"+i.idx+"_container"),c=e(".mCSB_"+i.idx+"_scrollbar");r.live&&m(r.liveSelector||e(t).selector),X.call(this,"remove"),k.call(this),B.call(this),n.removeData(a),K(this,"mcs"),c.remove(),s.find("img."+d[2]).removeClass(d[2]),l.replaceWith(s.contents()),n.removeClass(o+" _"+a+"_"+i.idx+" "+d[6]+" "+d[7]+" "+d[5]+" "+d[3]).addClass(d[4])}})}},f=function(){return"object"!=typeof e(this)||e(this).length<1?n:this},h=function(t){var o=["rounded","rounded-dark","rounded-dots","rounded-dots-dark"],a=["rounded-dots","rounded-dots-dark","3d","3d-dark","3d-thick","3d-thick-dark","inset","inset-dark","inset-2","inset-2-dark","inset-3","inset-3-dark"],n=["minimal","minimal-dark"],i=["minimal","minimal-dark"],r=["minimal","minimal-dark"];t.autoDraggerLength=e.inArray(t.theme,o)>-1?!1:t.autoDraggerLength,t.autoExpandScrollbar=e.inArray(t.theme,a)>-1?!1:t.autoExpandScrollbar,t.scrollButtons.enable=e.inArray(t.theme,n)>-1?!1:t.scrollButtons.enable,t.autoHideScrollbar=e.inArray(t.theme,i)>-1?!0:t.autoHideScrollbar,t.scrollbarPosition=e.inArray(t.theme,r)>-1?"outside":t.scrollbarPosition},m=function(e){l[e]&&(clearTimeout(l[e]),K(l,e))},p=function(e){return"yx"===e||"xy"===e||"auto"===e?"yx":"x"===e||"horizontal"===e?"x":"y"},g=function(e){return"stepped"===e||"pixels"===e||"step"===e||"click"===e?"stepped":"stepless"},v=function(){var t=e(this),n=t.data(a),i=n.opt,r=i.autoExpandScrollbar?" "+d[1]+"_expand":"",l=["",""],s="yx"===i.axis?"mCSB_vertical_horizontal":"x"===i.axis?"mCSB_horizontal":"mCSB_vertical",c="yx"===i.axis?l[0]+l[1]:"x"===i.axis?l[1]:l[0],u="yx"===i.axis?"":"",f=i.autoHideScrollbar?" "+d[6]:"",h="x"!==i.axis&&"rtl"===n.langDir?" "+d[7]:"";i.setWidth&&t.css("width",i.setWidth),i.setHeight&&t.css("height",i.setHeight),i.setLeft="y"!==i.axis&&"rtl"===n.langDir?"989999px":i.setLeft,t.addClass(o+" _"+a+"_"+n.idx+f+h).wrapInner("");var m=e("#mCSB_"+n.idx),p=e("#mCSB_"+n.idx+"_container");"y"===i.axis||i.advanced.autoExpandHorizontalScroll||p.css("width",x(p)),"outside"===i.scrollbarPosition?("static"===t.css("position")&&t.css("position","relative"),t.css("overflow","visible"),m.addClass("mCSB_outside").after(c)):(m.addClass("mCSB_inside").append(c),p.wrap(u)),w.call(this);var g=[e("#mCSB_"+n.idx+"_dragger_vertical"),e("#mCSB_"+n.idx+"_dragger_horizontal")];g[0].css("min-height",g[0].height()),g[1].css("min-width",g[1].width())},x=function(t){var o=[t[0].scrollWidth,Math.max.apply(Math,t.children().map(function(){return e(this).outerWidth(!0)}).get())],a=t.parent().width();return o[0]>a?o[0]:o[1]>a?o[1]:"100%"},_=function(){var t=e(this),o=t.data(a),n=o.opt,i=e("#mCSB_"+o.idx+"_container");if(n.advanced.autoExpandHorizontalScroll&&"y"!==n.axis){i.css({width:"auto","min-width":0,"overflow-x":"scroll"});var r=Math.ceil(i[0].scrollWidth);3===n.advanced.autoExpandHorizontalScroll||2!==n.advanced.autoExpandHorizontalScroll&&r>i.parent().width()?i.css({width:r,"min-width":"100%","overflow-x":"inherit"}):i.css({"overflow-x":"inherit",position:"absolute"}).wrap("").css({width:Math.ceil(i[0].getBoundingClientRect().right+.4)-Math.floor(i[0].getBoundingClientRect().left),"min-width":"100%",position:"relative"}).unwrap()}},w=function(){var t=e(this),o=t.data(a),n=o.opt,i=e(".mCSB_"+o.idx+"_scrollbar:first"),r=ee(n.scrollButtons.tabindex)?"tabindex='"+n.scrollButtons.tabindex+"'":"",l=["","","",""],s=["x"===n.axis?l[2]:l[0],"x"===n.axis?l[3]:l[1],l[2],l[3]];n.scrollButtons.enable&&i.prepend(s[0]).append(s[1]).next(".mCSB_scrollTools").prepend(s[2]).append(s[3])},S=function(){var t=e(this),o=t.data(a),n=e("#mCSB_"+o.idx),i=e("#mCSB_"+o.idx+"_container"),r=[e("#mCSB_"+o.idx+"_dragger_vertical"),e("#mCSB_"+o.idx+"_dragger_horizontal")],l=[n.height()/i.outerHeight(!1),n.width()/i.outerWidth(!1)],c=[parseInt(r[0].css("min-height")),Math.round(l[0]*r[0].parent().height()),parseInt(r[1].css("min-width")),Math.round(l[1]*r[1].parent().width())],d=s&&c[1]r&&(r=s),c>l&&(l=c),[r>n.height(),l>n.width()]},B=function(){var t=e(this),o=t.data(a),n=o.opt,i=e("#mCSB_"+o.idx),r=e("#mCSB_"+o.idx+"_container"),l=[e("#mCSB_"+o.idx+"_dragger_vertical"),e("#mCSB_"+o.idx+"_dragger_horizontal")];if(N(t),("x"!==n.axis&&!o.overflowed[0]||"y"===n.axis&&o.overflowed[0])&&(l[0].add(r).css("top",0),V(t,"_resetY")),"y"!==n.axis&&!o.overflowed[1]||"x"===n.axis&&o.overflowed[1]){var s=dx=0;"rtl"===o.langDir&&(s=i.width()-r.outerWidth(!1),dx=Math.abs(s/o.scrollRatio.x)),r.css("left",s),l[1].css("left",dx),V(t,"_resetX")}},T=function(){function t(){r=setTimeout(function(){e.event.special.mousewheel?(clearTimeout(r),R.call(o[0])):t()},100)}var o=e(this),n=o.data(a),i=n.opt;if(!n.bindEvents){if(I.call(this),i.contentTouchScroll&&D.call(this),E.call(this),i.mouseWheel.enable){var r;t()}L.call(this),P.call(this),i.advanced.autoScrollOnFocus&&z.call(this),i.scrollButtons.enable&&H.call(this),i.keyboard.enable&&U.call(this),n.bindEvents=!0}},k=function(){var t=e(this),o=t.data(a),n=o.opt,i=a+"_"+o.idx,r=".mCSB_"+o.idx+"_scrollbar",l=e("#mCSB_"+o.idx+",#mCSB_"+o.idx+"_container,#mCSB_"+o.idx+"_container_wrapper,"+r+" ."+d[12]+",#mCSB_"+o.idx+"_dragger_vertical,#mCSB_"+o.idx+"_dragger_horizontal,"+r+">a"),s=e("#mCSB_"+o.idx+"_container");n.advanced.releaseDraggableSelectors&&l.add(e(n.advanced.releaseDraggableSelectors)),n.advanced.extraDraggableSelectors&&l.add(e(n.advanced.extraDraggableSelectors)),o.bindEvents&&(e(document).add(e(!W()||top.document)).unbind("."+i),l.each(function(){e(this).unbind("."+i)}),clearTimeout(t[0]._focusTimeout),K(t[0],"_focusTimeout"),clearTimeout(o.sequential.step),K(o.sequential,"step"),clearTimeout(s[0].onCompleteTimeout),K(s[0],"onCompleteTimeout"),o.bindEvents=!1)},M=function(t){var o=e(this),n=o.data(a),i=n.opt,r=e("#mCSB_"+n.idx+"_container_wrapper"),l=r.length?r:e("#mCSB_"+n.idx+"_container"),s=[e("#mCSB_"+n.idx+"_scrollbar_vertical"),e("#mCSB_"+n.idx+"_scrollbar_horizontal")],c=[s[0].find(".mCSB_dragger"),s[1].find(".mCSB_dragger")];"x"!==i.axis&&(n.overflowed[0]&&!t?(s[0].add(c[0]).add(s[0].children("a")).css("display","block"),l.removeClass(d[8]+" "+d[10])):(i.alwaysShowScrollbar?(2!==i.alwaysShowScrollbar&&c[0].css("display","none"),l.removeClass(d[10])):(s[0].css("display","none"),l.addClass(d[10])),l.addClass(d[8]))),"y"!==i.axis&&(n.overflowed[1]&&!t?(s[1].add(c[1]).add(s[1].children("a")).css("display","block"),l.removeClass(d[9]+" "+d[11])):(i.alwaysShowScrollbar?(2!==i.alwaysShowScrollbar&&c[1].css("display","none"),l.removeClass(d[11])):(s[1].css("display","none"),l.addClass(d[11])),l.addClass(d[9]))),n.overflowed[0]||n.overflowed[1]?o.removeClass(d[5]):o.addClass(d[5])},O=function(t){var o=t.type,a=t.target.ownerDocument!==document?[e(frameElement).offset().top,e(frameElement).offset().left]:null,n=W()&&t.target.ownerDocument!==top.document?[e(t.view.frameElement).offset().top,e(t.view.frameElement).offset().left]:[0,0];switch(o){case"pointerdown":case"MSPointerDown":case"pointermove":case"MSPointerMove":case"pointerup":case"MSPointerUp":return a?[t.originalEvent.pageY-a[0]+n[0],t.originalEvent.pageX-a[1]+n[1],!1]:[t.originalEvent.pageY,t.originalEvent.pageX,!1];case"touchstart":case"touchmove":case"touchend":var i=t.originalEvent.touches[0]||t.originalEvent.changedTouches[0],r=t.originalEvent.touches.length||t.originalEvent.changedTouches.length;return t.target.ownerDocument!==document?[i.screenY,i.screenX,r>1]:[i.pageY,i.pageX,r>1];default:return a?[t.pageY-a[0]+n[0],t.pageX-a[1]+n[1],!1]:[t.pageY,t.pageX,!1]}},I=function(){function t(e){var t=m.find("iframe");if(t.length){var o=e?"auto":"none";t.css("pointer-events",o)}}function o(e,t,o,a){if(m[0].idleTimer=u.scrollInertia<233?250:0,n.attr("id")===h[1])var i="x",r=(n[0].offsetLeft-t+a)*d.scrollRatio.x;else var i="y",r=(n[0].offsetTop-e+o)*d.scrollRatio.y;V(l,r.toString(),{dir:i,drag:!0})}var n,i,r,l=e(this),d=l.data(a),u=d.opt,f=a+"_"+d.idx,h=["mCSB_"+d.idx+"_dragger_vertical","mCSB_"+d.idx+"_dragger_horizontal"],m=e("#mCSB_"+d.idx+"_container"),p=e("#"+h[0]+",#"+h[1]),g=u.advanced.releaseDraggableSelectors?p.add(e(u.advanced.releaseDraggableSelectors)):p,v=u.advanced.extraDraggableSelectors?e(!W()||top.document).add(e(u.advanced.extraDraggableSelectors)):e(!W()||top.document);p.bind("mousedown."+f+" touchstart."+f+" pointerdown."+f+" MSPointerDown."+f,function(o){if(o.stopImmediatePropagation(),o.preventDefault(),Z(o)){c=!0,s&&(document.onselectstart=function(){return!1}),t(!1),N(l),n=e(this);var a=n.offset(),d=O(o)[0]-a.top,f=O(o)[1]-a.left,h=n.height()+a.top,m=n.width()+a.left;h>d&&d>0&&m>f&&f>0&&(i=d,r=f),C(n,"active",u.autoExpandScrollbar)}}).bind("touchmove."+f,function(e){e.stopImmediatePropagation(),e.preventDefault();var t=n.offset(),a=O(e)[0]-t.top,l=O(e)[1]-t.left;o(i,r,a,l)}),e(document).add(v).bind("mousemove."+f+" pointermove."+f+" MSPointerMove."+f,function(e){if(n){var t=n.offset(),a=O(e)[0]-t.top,l=O(e)[1]-t.left;if(i===a&&r===l)return;o(i,r,a,l)}}).add(g).bind("mouseup."+f+" touchend."+f+" pointerup."+f+" MSPointerUp."+f,function(e){n&&(C(n,"active",u.autoExpandScrollbar),n=null),c=!1,s&&(document.onselectstart=null),t(!0)})},D=function(){function o(e){if(!$(e)||c||O(e)[2])return void(t=0);t=1,b=0,C=0,d=1,y.removeClass("mCS_touch_action");var o=I.offset();u=O(e)[0]-o.top,f=O(e)[1]-o.left,z=[O(e)[0],O(e)[1]]}function n(e){if($(e)&&!c&&!O(e)[2]&&(T.documentTouchScroll||e.preventDefault(),e.stopImmediatePropagation(),(!C||b)&&d)){g=G();var t=M.offset(),o=O(e)[0]-t.top,a=O(e)[1]-t.left,n="mcsLinearOut";if(E.push(o),R.push(a),z[2]=Math.abs(O(e)[0]-z[0]),z[3]=Math.abs(O(e)[1]-z[1]),B.overflowed[0])var i=D[0].parent().height()-D[0].height(),r=u-o>0&&o-u>-(i*B.scrollRatio.y)&&(2*z[3]0&&a-f>-(l*B.scrollRatio.x)&&(2*z[2]30)){_=1e3/(v-p);var n="mcsEaseOut",i=2.5>_,r=i?[E[E.length-2],R[R.length-2]]:[0,0];x=i?[o-r[0],a-r[1]]:[o-h,a-m];var u=[Math.abs(x[0]),Math.abs(x[1])];_=i?[Math.abs(x[0]/4),Math.abs(x[1]/4)]:[_,_];var f=[Math.abs(I[0].offsetTop)-x[0]*l(u[0]/_[0],_[0]),Math.abs(I[0].offsetLeft)-x[1]*l(u[1]/_[1],_[1])];w="yx"===T.axis?[f[0],f[1]]:"x"===T.axis?[null,f[1]]:[f[0],null],S=[4*u[0]+T.scrollInertia,4*u[1]+T.scrollInertia];var y=parseInt(T.contentTouchScroll)||0;w[0]=u[0]>y?w[0]:0,w[1]=u[1]>y?w[1]:0,B.overflowed[0]&&s(w[0],S[0],n,"y",L,!1),B.overflowed[1]&&s(w[1],S[1],n,"x",L,!1)}}}function l(e,t){var o=[1.5*t,2*t,t/1.5,t/2];return e>90?t>4?o[0]:o[3]:e>60?t>3?o[3]:o[2]:e>30?t>8?o[1]:t>6?o[0]:t>4?t:o[2]:t>8?t:o[3]}function s(e,t,o,a,n,i){e&&V(y,e.toString(),{dur:t,scrollEasing:o,dir:a,overwrite:n,drag:i})}var d,u,f,h,m,p,g,v,x,_,w,S,b,C,y=e(this),B=y.data(a),T=B.opt,k=a+"_"+B.idx,M=e("#mCSB_"+B.idx),I=e("#mCSB_"+B.idx+"_container"),D=[e("#mCSB_"+B.idx+"_dragger_vertical"),e("#mCSB_"+B.idx+"_dragger_horizontal")],E=[],R=[],A=0,L="yx"===T.axis?"none":"all",z=[],P=I.find("iframe"),H=["touchstart."+k+" pointerdown."+k+" MSPointerDown."+k,"touchmove."+k+" pointermove."+k+" MSPointerMove."+k,"touchend."+k+" pointerup."+k+" MSPointerUp."+k],U=void 0!==document.body.style.touchAction;I.bind(H[0],function(e){o(e)}).bind(H[1],function(e){n(e)}),M.bind(H[0],function(e){i(e)}).bind(H[2],function(e){r(e)}),P.length&&P.each(function(){e(this).load(function(){W(this)&&e(this.contentDocument||this.contentWindow.document).bind(H[0],function(e){o(e),i(e)}).bind(H[1],function(e){n(e)}).bind(H[2],function(e){r(e)})})})},E=function(){function o(){return window.getSelection?window.getSelection().toString():document.selection&&"Control"!=document.selection.type?document.selection.createRange().text:0}function n(e,t,o){d.type=o&&i?"stepped":"stepless",d.scrollAmount=10,F(r,e,t,"mcsLinearOut",o?60:null)}var i,r=e(this),l=r.data(a),s=l.opt,d=l.sequential,u=a+"_"+l.idx,f=e("#mCSB_"+l.idx+"_container"),h=f.parent();f.bind("mousedown."+u,function(e){t||i||(i=1,c=!0)}).add(document).bind("mousemove."+u,function(e){if(!t&&i&&o()){var a=f.offset(),r=O(e)[0]-a.top+f[0].offsetTop,c=O(e)[1]-a.left+f[0].offsetLeft;r>0&&r0&&cr?n("on",38):r>h.height()&&n("on",40)),"y"!==s.axis&&l.overflowed[1]&&(0>c?n("on",37):c>h.width()&&n("on",39)))}}).bind("mouseup."+u+" dragend."+u,function(e){t||(i&&(i=0,n("off",null)),c=!1)})},R=function(){function t(t,a){if(N(o),!A(o,t.target)){var r="auto"!==i.mouseWheel.deltaFactor?parseInt(i.mouseWheel.deltaFactor):s&&t.deltaFactor<100?100:t.deltaFactor||100,d=i.scrollInertia;if("x"===i.axis||"x"===i.mouseWheel.axis)var u="x",f=[Math.round(r*n.scrollRatio.x),parseInt(i.mouseWheel.scrollAmount)],h="auto"!==i.mouseWheel.scrollAmount?f[1]:f[0]>=l.width()?.9*l.width():f[0],m=Math.abs(e("#mCSB_"+n.idx+"_container")[0].offsetLeft),p=c[1][0].offsetLeft,g=c[1].parent().width()-c[1].width(),v=t.deltaX||t.deltaY||a;else var u="y",f=[Math.round(r*n.scrollRatio.y),parseInt(i.mouseWheel.scrollAmount)],h="auto"!==i.mouseWheel.scrollAmount?f[1]:f[0]>=l.height()?.9*l.height():f[0],m=Math.abs(e("#mCSB_"+n.idx+"_container")[0].offsetTop),p=c[0][0].offsetTop,g=c[0].parent().height()-c[0].height(),v=t.deltaY||a;"y"===u&&!n.overflowed[0]||"x"===u&&!n.overflowed[1]||((i.mouseWheel.invert||t.webkitDirectionInvertedFromDevice)&&(v=-v),i.mouseWheel.normalizeDelta&&(v=0>v?-1:1),(v>0&&0!==p||0>v&&p!==g||i.mouseWheel.preventDefault)&&(t.stopImmediatePropagation(),t.preventDefault()),t.deltaFactor<2&&!i.mouseWheel.normalizeDelta&&(h=t.deltaFactor,d=17),V(o,(m-v*h).toString(),{dir:u,dur:d}))}}if(e(this).data(a)){var o=e(this),n=o.data(a),i=n.opt,r=a+"_"+n.idx,l=e("#mCSB_"+n.idx),c=[e("#mCSB_"+n.idx+"_dragger_vertical"),e("#mCSB_"+n.idx+"_dragger_horizontal")],d=e("#mCSB_"+n.idx+"_container").find("iframe");d.length&&d.each(function(){e(this).load(function(){W(this)&&e(this.contentDocument||this.contentWindow.document).bind("mousewheel."+r,function(e,o){t(e,o)})})}),l.bind("mousewheel."+r,function(e,o){t(e,o)})}},W=function(e){var t=null;if(e){try{var o=e.contentDocument||e.contentWindow.document;t=o.body.innerHTML}catch(a){}return null!==t}try{var o=top.document;t=o.body.innerHTML}catch(a){}return null!==t},A=function(t,o){var n=o.nodeName.toLowerCase(),i=t.data(a).opt.mouseWheel.disableOver,r=["select","textarea"];return e.inArray(n,i)>-1&&!(e.inArray(n,r)>-1&&!e(o).is(":focus"))},L=function(){var t,o=e(this),n=o.data(a),i=a+"_"+n.idx,r=e("#mCSB_"+n.idx+"_container"),l=r.parent(),s=e(".mCSB_"+n.idx+"_scrollbar ."+d[12]);s.bind("mousedown."+i+" touchstart."+i+" pointerdown."+i+" MSPointerDown."+i,function(o){c=!0,e(o.target).hasClass("mCSB_dragger")||(t=1)}).bind("touchend."+i+" pointerup."+i+" MSPointerUp."+i,function(e){c=!1}).bind("click."+i,function(a){if(t&&(t=0,e(a.target).hasClass(d[12])||e(a.target).hasClass("mCSB_draggerRail"))){N(o);var i=e(this),s=i.find(".mCSB_dragger");if(i.parent(".mCSB_scrollTools_horizontal").length>0){if(!n.overflowed[1])return;var c="x",u=a.pageX>s.offset().left?-1:1,f=Math.abs(r[0].offsetLeft)-.9*u*l.width()}else{if(!n.overflowed[0])return;var c="y",u=a.pageY>s.offset().top?-1:1,f=Math.abs(r[0].offsetTop)-.9*u*l.height()}V(o,f.toString(),{dir:c,scrollEasing:"mcsEaseInOut"})}})},z=function(){var t=e(this),o=t.data(a),n=o.opt,i=a+"_"+o.idx,r=e("#mCSB_"+o.idx+"_container"),l=r.parent();r.bind("focusin."+i,function(o){var a=e(document.activeElement),i=r.find(".mCustomScrollBox").length,s=0;a.is(n.advanced.autoScrollOnFocus)&&(N(t),clearTimeout(t[0]._focusTimeout),t[0]._focusTimer=i?(s+17)*i:0,t[0]._focusTimeout=setTimeout(function(){var e=[te(a)[0],te(a)[1]],o=[r[0].offsetTop,r[0].offsetLeft],i=[o[0]+e[0]>=0&&o[0]+e[0]=0&&o[0]+e[1]a");s.bind("mousedown."+r+" touchstart."+r+" pointerdown."+r+" MSPointerDown."+r+" mouseup."+r+" touchend."+r+" pointerup."+r+" MSPointerUp."+r+" mouseout."+r+" pointerout."+r+" MSPointerOut."+r+" click."+r,function(a){function r(e,o){i.scrollAmount=n.scrollButtons.scrollAmount,F(t,e,o)}if(a.preventDefault(),Z(a)){var l=e(this).attr("class");switch(i.type=n.scrollButtons.scrollType,a.type){case"mousedown":case"touchstart":case"pointerdown":case"MSPointerDown":if("stepped"===i.type)return;c=!0,o.tweenRunning=!1,r("on",l);break;case"mouseup":case"touchend":case"pointerup":case"MSPointerUp":case"mouseout":case"pointerout":case"MSPointerOut":if("stepped"===i.type)return;c=!1,i.dir&&r("off",l);break;case"click":if("stepped"!==i.type||o.tweenRunning)return;r("on",l)}}})},U=function(){function t(t){function a(e,t){r.type=i.keyboard.scrollType,r.scrollAmount=i.keyboard.scrollAmount,"stepped"===r.type&&n.tweenRunning||F(o,e,t)}switch(t.type){case"blur":n.tweenRunning&&r.dir&&a("off",null);break;case"keydown":case"keyup":var l=t.keyCode?t.keyCode:t.which,s="on";if("x"!==i.axis&&(38===l||40===l)||"y"!==i.axis&&(37===l||39===l)){if((38===l||40===l)&&!n.overflowed[0]||(37===l||39===l)&&!n.overflowed[1])return;"keyup"===t.type&&(s="off"),e(document.activeElement).is(u)||(t.preventDefault(),t.stopImmediatePropagation(),a(s,l))}else if(33===l||34===l){if((n.overflowed[0]||n.overflowed[1])&&(t.preventDefault(),t.stopImmediatePropagation()),"keyup"===t.type){N(o);var f=34===l?-1:1;if("x"===i.axis||"yx"===i.axis&&n.overflowed[1]&&!n.overflowed[0])var h="x",m=Math.abs(c[0].offsetLeft)-.9*f*d.width();else var h="y",m=Math.abs(c[0].offsetTop)-.9*f*d.height();V(o,m.toString(),{dir:h,scrollEasing:"mcsEaseInOut"})}}else if((35===l||36===l)&&!e(document.activeElement).is(u)&&((n.overflowed[0]||n.overflowed[1])&&(t.preventDefault(),t.stopImmediatePropagation()),"keyup"===t.type)){if("x"===i.axis||"yx"===i.axis&&n.overflowed[1]&&!n.overflowed[0])var h="x",m=35===l?Math.abs(d.width()-c.outerWidth(!1)):0;else var h="y",m=35===l?Math.abs(d.height()-c.outerHeight(!1)):0;V(o,m.toString(),{dir:h,scrollEasing:"mcsEaseInOut"})}}}var o=e(this),n=o.data(a),i=n.opt,r=n.sequential,l=a+"_"+n.idx,s=e("#mCSB_"+n.idx),c=e("#mCSB_"+n.idx+"_container"),d=c.parent(),u="input,textarea,select,datalist,keygen,[contenteditable='true']",f=c.find("iframe"),h=["blur."+l+" keydown."+l+" keyup."+l];f.length&&f.each(function(){e(this).load(function(){W(this)&&e(this.contentDocument||this.contentWindow.document).bind(h[0],function(e){t(e)})})}),s.attr("tabindex","0").bind(h[0],function(e){t(e)})},F=function(t,o,n,i,r){function l(e){u.snapAmount&&(f.scrollAmount=u.snapAmount instanceof Array?"x"===f.dir[0]?u.snapAmount[1]:u.snapAmount[0]:u.snapAmount);var o="stepped"!==f.type,a=r?r:e?o?p/1.5:g:1e3/60,n=e?o?7.5:40:2.5,s=[Math.abs(h[0].offsetTop),Math.abs(h[0].offsetLeft)],d=[c.scrollRatio.y>10?10:c.scrollRatio.y,c.scrollRatio.x>10?10:c.scrollRatio.x],m="x"===f.dir[0]?s[1]+f.dir[1]*d[1]*n:s[0]+f.dir[1]*d[0]*n,v="x"===f.dir[0]?s[1]+f.dir[1]*parseInt(f.scrollAmount):s[0]+f.dir[1]*parseInt(f.scrollAmount),x="auto"!==f.scrollAmount?v:m,_=i?i:e?o?"mcsLinearOut":"mcsEaseInOut":"mcsLinear",w=e?!0:!1;return e&&17>a&&(x="x"===f.dir[0]?s[1]:s[0]),V(t,x.toString(),{dir:f.dir[0],scrollEasing:_,dur:a,onComplete:w}),e?void(f.dir=!1):(clearTimeout(f.step),void(f.step=setTimeout(function(){l()},a)))}function s(){clearTimeout(f.step),K(f,"step"),N(t)}var c=t.data(a),u=c.opt,f=c.sequential,h=e("#mCSB_"+c.idx+"_container"),m="stepped"===f.type?!0:!1,p=u.scrollInertia<26?26:u.scrollInertia,g=u.scrollInertia<1?17:u.scrollInertia;switch(o){case"on":if(f.dir=[n===d[16]||n===d[15]||39===n||37===n?"x":"y",n===d[13]||n===d[15]||38===n||37===n?-1:1],N(t),ee(n)&&"stepped"===f.type)return;l(m);break;case"off":s(),(m||c.tweenRunning&&f.dir)&&l(!0)}},q=function(t){var o=e(this).data(a).opt,n=[];return"function"==typeof t&&(t=t()),t instanceof Array?n=t.length>1?[t[0],t[1]]:"x"===o.axis?[null,t[0]]:[t[0],null]:(n[0]=t.y?t.y:t.x||"x"===o.axis?null:t,n[1]=t.x?t.x:t.y||"y"===o.axis?null:t),"function"==typeof n[0]&&(n[0]=n[0]()),"function"==typeof n[1]&&(n[1]=n[1]()),n},Y=function(t,o){if(null!=t&&"undefined"!=typeof t){var n=e(this),i=n.data(a),r=i.opt,l=e("#mCSB_"+i.idx+"_container"),s=l.parent(),c=typeof t;o||(o="x"===r.axis?"x":"y");var d="x"===o?l.outerWidth(!1):l.outerHeight(!1),f="x"===o?l[0].offsetLeft:l[0].offsetTop,h="x"===o?"left":"top";switch(c){case"function":return t();case"object":var m=t.jquery?t:e(t);if(!m.length)return;return"x"===o?te(m)[1]:te(m)[0];case"string":case"number":if(ee(t))return Math.abs(t);if(-1!==t.indexOf("%"))return Math.abs(d*parseInt(t)/100);if(-1!==t.indexOf("-="))return Math.abs(f-parseInt(t.split("-=")[1]));if(-1!==t.indexOf("+=")){var p=f+parseInt(t.split("+=")[1]);return p>=0?0:Math.abs(p)}if(-1!==t.indexOf("px")&&ee(t.split("px")[0]))return Math.abs(t.split("px")[0]);if("top"===t||"left"===t)return 0;if("bottom"===t)return Math.abs(s.height()-l.outerHeight(!1));if("right"===t)return Math.abs(s.width()-l.outerWidth(!1));if("first"===t||"last"===t){var m=l.find(":"+t);return"x"===o?te(m)[1]:te(m)[0]}return e(t).length?"x"===o?te(e(t))[1]:te(e(t))[0]:(l.css(h,t),void u.update.call(null,n[0]))}}},X=function(t){function o(){return clearTimeout(f[0].autoUpdate),0===l.parents("html").length?void(l=null):void(f[0].autoUpdate=setTimeout(function(){return c.advanced.updateOnSelectorChange&&(s.poll.change.n=i(),s.poll.change.n!==s.poll.change.o)?(s.poll.change.o=s.poll.change.n,void r(3)):c.advanced.updateOnContentResize&&(s.poll.size.n=l[0].scrollHeight+l[0].scrollWidth+f[0].offsetHeight+l[0].offsetHeight+l[0].offsetWidth,s.poll.size.n!==s.poll.size.o)?(s.poll.size.o=s.poll.size.n,void r(1)):!c.advanced.updateOnImageLoad||"auto"===c.advanced.updateOnImageLoad&&"y"===c.axis||(s.poll.img.n=f.find("img").length,s.poll.img.n===s.poll.img.o)?void((c.advanced.updateOnSelectorChange||c.advanced.updateOnContentResize||c.advanced.updateOnImageLoad)&&o()):(s.poll.img.o=s.poll.img.n,void f.find("img").each(function(){n(this)}))},c.advanced.autoUpdateTimeout))}function n(t){function o(e,t){return function(){return t.apply(e,arguments)}}function a(){this.onload=null,e(t).addClass(d[2]),r(2)}if(e(t).hasClass(d[2]))return void r();var n=new Image;n.onload=o(n,a),n.src=t.src}function i(){c.advanced.updateOnSelectorChange===!0&&(c.advanced.updateOnSelectorChange="*");var e=0,t=f.find(c.advanced.updateOnSelectorChange);
5 |
6 | return c.advanced.updateOnSelectorChange&&t.length>0&&t.each(function(){e+=this.offsetHeight+this.offsetWidth}),e}function r(e){clearTimeout(f[0].autoUpdate),u.update.call(null,l[0],e)}var l=e(this),s=l.data(a),c=s.opt,f=e("#mCSB_"+s.idx+"_container");return t?(clearTimeout(f[0].autoUpdate),void K(f[0],"autoUpdate")):void o()},j=function(e,t,o){return Math.round(e/t)*t-o},N=function(t){var o=t.data(a),n=e("#mCSB_"+o.idx+"_container,#mCSB_"+o.idx+"_container_wrapper,#mCSB_"+o.idx+"_dragger_vertical,#mCSB_"+o.idx+"_dragger_horizontal");n.each(function(){J.call(this)})},V=function(t,o,n){function i(e){return s&&c.callbacks[e]&&"function"==typeof c.callbacks[e]}function r(){return[c.callbacks.alwaysTriggerOffsets||w>=S[0]+y,c.callbacks.alwaysTriggerOffsets||-B>=w]}function l(){var e=[h[0].offsetTop,h[0].offsetLeft],o=[x[0].offsetTop,x[0].offsetLeft],a=[h.outerHeight(!1),h.outerWidth(!1)],i=[f.height(),f.width()];t[0].mcs={content:h,top:e[0],left:e[1],draggerTop:o[0],draggerLeft:o[1],topPct:Math.round(100*Math.abs(e[0])/(Math.abs(a[0])-i[0])),leftPct:Math.round(100*Math.abs(e[1])/(Math.abs(a[1])-i[1])),direction:n.dir}}var s=t.data(a),c=s.opt,d={trigger:"internal",dir:"y",scrollEasing:"mcsEaseOut",drag:!1,dur:c.scrollInertia,overwrite:"all",callbacks:!0,onStart:!0,onUpdate:!0,onComplete:!0},n=e.extend(d,n),u=[n.dur,n.drag?0:n.dur],f=e("#mCSB_"+s.idx),h=e("#mCSB_"+s.idx+"_container"),m=h.parent(),p=c.callbacks.onTotalScrollOffset?q.call(t,c.callbacks.onTotalScrollOffset):[0,0],g=c.callbacks.onTotalScrollBackOffset?q.call(t,c.callbacks.onTotalScrollBackOffset):[0,0];if(s.trigger=n.trigger,(0!==m.scrollTop()||0!==m.scrollLeft())&&(e(".mCSB_"+s.idx+"_scrollbar").css("visibility","visible"),m.scrollTop(0).scrollLeft(0)),"_resetY"!==o||s.contentReset.y||(i("onOverflowYNone")&&c.callbacks.onOverflowYNone.call(t[0]),s.contentReset.y=1),"_resetX"!==o||s.contentReset.x||(i("onOverflowXNone")&&c.callbacks.onOverflowXNone.call(t[0]),s.contentReset.x=1),"_resetY"!==o&&"_resetX"!==o){if(!s.contentReset.y&&t[0].mcs||!s.overflowed[0]||(i("onOverflowY")&&c.callbacks.onOverflowY.call(t[0]),s.contentReset.x=null),!s.contentReset.x&&t[0].mcs||!s.overflowed[1]||(i("onOverflowX")&&c.callbacks.onOverflowX.call(t[0]),s.contentReset.x=null),c.snapAmount){var v=c.snapAmount instanceof Array?"x"===n.dir?c.snapAmount[1]:c.snapAmount[0]:c.snapAmount;o=j(o,v,c.snapOffset)}switch(n.dir){case"x":var x=e("#mCSB_"+s.idx+"_dragger_horizontal"),_="left",w=h[0].offsetLeft,S=[f.width()-h.outerWidth(!1),x.parent().width()-x.width()],b=[o,0===o?0:o/s.scrollRatio.x],y=p[1],B=g[1],T=y>0?y/s.scrollRatio.x:0,k=B>0?B/s.scrollRatio.x:0;break;case"y":var x=e("#mCSB_"+s.idx+"_dragger_vertical"),_="top",w=h[0].offsetTop,S=[f.height()-h.outerHeight(!1),x.parent().height()-x.height()],b=[o,0===o?0:o/s.scrollRatio.y],y=p[0],B=g[0],T=y>0?y/s.scrollRatio.y:0,k=B>0?B/s.scrollRatio.y:0}b[1]<0||0===b[0]&&0===b[1]?b=[0,0]:b[1]>=S[1]?b=[S[0],S[1]]:b[0]=-b[0],t[0].mcs||(l(),i("onInit")&&c.callbacks.onInit.call(t[0])),clearTimeout(h[0].onCompleteTimeout),Q(x[0],_,Math.round(b[1]),u[1],n.scrollEasing),(s.tweenRunning||!(0===w&&b[0]>=0||w===S[0]&&b[0]<=S[0]))&&Q(h[0],_,Math.round(b[0]),u[0],n.scrollEasing,n.overwrite,{onStart:function(){n.callbacks&&n.onStart&&!s.tweenRunning&&(i("onScrollStart")&&(l(),c.callbacks.onScrollStart.call(t[0])),s.tweenRunning=!0,C(x),s.cbOffsets=r())},onUpdate:function(){n.callbacks&&n.onUpdate&&i("whileScrolling")&&(l(),c.callbacks.whileScrolling.call(t[0]))},onComplete:function(){if(n.callbacks&&n.onComplete){"yx"===c.axis&&clearTimeout(h[0].onCompleteTimeout);var e=h[0].idleTimer||0;h[0].onCompleteTimeout=setTimeout(function(){i("onScroll")&&(l(),c.callbacks.onScroll.call(t[0])),i("onTotalScroll")&&b[1]>=S[1]-T&&s.cbOffsets[0]&&(l(),c.callbacks.onTotalScroll.call(t[0])),i("onTotalScrollBack")&&b[1]<=k&&s.cbOffsets[1]&&(l(),c.callbacks.onTotalScrollBack.call(t[0])),s.tweenRunning=!1,h[0].idleTimer=0,C(x,"hide")},e)}}})}},Q=function(e,t,o,a,n,i,r){function l(){S.stop||(x||m.call(),x=G()-v,s(),x>=S.time&&(S.time=x>S.time?x+f-(x-S.time):x+f-1,S.time0?(S.currVal=u(S.time,_,b,a,n),w[t]=Math.round(S.currVal)+"px"):w[t]=o+"px",p.call()}function c(){f=1e3/60,S.time=x+f,h=window.requestAnimationFrame?window.requestAnimationFrame:function(e){return s(),setTimeout(e,.01)},S.id=h(l)}function d(){null!=S.id&&(window.requestAnimationFrame?window.cancelAnimationFrame(S.id):clearTimeout(S.id),S.id=null)}function u(e,t,o,a,n){switch(n){case"linear":case"mcsLinear":return o*e/a+t;case"mcsLinearOut":return e/=a,e--,o*Math.sqrt(1-e*e)+t;case"easeInOutSmooth":return e/=a/2,1>e?o/2*e*e+t:(e--,-o/2*(e*(e-2)-1)+t);case"easeInOutStrong":return e/=a/2,1>e?o/2*Math.pow(2,10*(e-1))+t:(e--,o/2*(-Math.pow(2,-10*e)+2)+t);case"easeInOut":case"mcsEaseInOut":return e/=a/2,1>e?o/2*e*e*e+t:(e-=2,o/2*(e*e*e+2)+t);case"easeOutSmooth":return e/=a,e--,-o*(e*e*e*e-1)+t;case"easeOutStrong":return o*(-Math.pow(2,-10*e/a)+1)+t;case"easeOut":case"mcsEaseOut":default:var i=(e/=a)*e,r=i*e;return t+o*(.499999999999997*r*i+-2.5*i*i+5.5*r+-6.5*i+4*e)}}e._mTween||(e._mTween={top:{},left:{}});var f,h,r=r||{},m=r.onStart||function(){},p=r.onUpdate||function(){},g=r.onComplete||function(){},v=G(),x=0,_=e.offsetTop,w=e.style,S=e._mTween[t];"left"===t&&(_=e.offsetLeft);var b=o-_;S.stop=0,"none"!==i&&d(),c()},G=function(){return window.performance&&window.performance.now?window.performance.now():window.performance&&window.performance.webkitNow?window.performance.webkitNow():Date.now?Date.now():(new Date).getTime()},J=function(){var e=this;e._mTween||(e._mTween={top:{},left:{}});for(var t=["top","left"],o=0;o=0&&a[0]+te(n)[0]=0&&a[1]+te(n)[1].mCSB_container{margin-right:30px}.mCSB_container.mCS_no_scrollbar_y.mCS_y_hidden{margin-right:0}.mCS-dir-rtl>.mCSB_inside>.mCSB_container{margin-right:0;margin-left:30px}.mCS-dir-rtl>.mCSB_inside>.mCSB_container.mCS_no_scrollbar_y.mCS_y_hidden{margin-left:0}.mCSB_scrollTools{position:absolute;width:16px;height:auto;left:auto;top:0;right:0;bottom:0;opacity:.75;filter:"alpha(opacity=75)";-ms-filter:"alpha(opacity=75)"}.mCSB_outside+.mCSB_scrollTools{right:-26px}.mCS-dir-rtl>.mCSB_inside>.mCSB_scrollTools,.mCS-dir-rtl>.mCSB_outside+.mCSB_scrollTools{right:auto;left:0}.mCS-dir-rtl>.mCSB_outside+.mCSB_scrollTools{left:-26px}.mCSB_scrollTools .mCSB_draggerContainer{position:absolute;top:0;left:0;bottom:0;right:0;height:auto}.mCSB_scrollTools a+.mCSB_draggerContainer{margin:20px 0}.mCSB_scrollTools .mCSB_draggerRail{width:2px;height:100%;margin:0 auto;-webkit-border-radius:16px;-moz-border-radius:16px;border-radius:16px}.mCSB_scrollTools .mCSB_dragger{cursor:pointer;width:100%;height:30px;z-index:1}.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{position:relative;width:4px;height:100%;margin:0 auto;-webkit-border-radius:16px;-moz-border-radius:16px;border-radius:16px;text-align:center}.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded .mCSB_dragger_bar,.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_dragger .mCSB_dragger_bar{width:12px}.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded+.mCSB_draggerRail,.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_draggerRail{width:8px}.mCSB_scrollTools .mCSB_buttonDown,.mCSB_scrollTools .mCSB_buttonUp{display:block;position:absolute;height:20px;width:100%;overflow:hidden;margin:0 auto;cursor:pointer}.mCSB_scrollTools .mCSB_buttonDown{bottom:0}.mCSB_horizontal.mCSB_inside>.mCSB_container{margin-right:0;margin-bottom:30px}.mCSB_horizontal.mCSB_outside>.mCSB_container{min-height:100%}.mCSB_horizontal>.mCSB_container.mCS_no_scrollbar_x.mCS_x_hidden{margin-bottom:0}.mCSB_scrollTools.mCSB_scrollTools_horizontal{width:auto;height:16px;top:auto;right:0;bottom:0;left:0}.mCustomScrollBox+.mCSB_scrollTools+.mCSB_scrollTools.mCSB_scrollTools_horizontal,.mCustomScrollBox+.mCSB_scrollTools.mCSB_scrollTools_horizontal{bottom:-26px}.mCSB_scrollTools.mCSB_scrollTools_horizontal a+.mCSB_draggerContainer{margin:0 20px}.mCSB_scrollTools.mCSB_scrollTools_horizontal .mCSB_draggerRail{width:100%;height:2px;margin:7px 0}.mCSB_scrollTools.mCSB_scrollTools_horizontal .mCSB_dragger{width:30px;height:100%;left:0}.mCSB_scrollTools.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar{width:100%;height:4px;margin:6px auto}.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded .mCSB_dragger_bar,.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_dragger .mCSB_dragger_bar{height:12px;margin:2px auto}.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded+.mCSB_draggerRail,.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_draggerRail{height:8px;margin:4px 0}.mCSB_scrollTools.mCSB_scrollTools_horizontal .mCSB_buttonLeft,.mCSB_scrollTools.mCSB_scrollTools_horizontal .mCSB_buttonRight{display:block;position:absolute;width:20px;height:100%;overflow:hidden;margin:0 auto;cursor:pointer}.mCSB_scrollTools.mCSB_scrollTools_horizontal .mCSB_buttonLeft{left:0}.mCSB_scrollTools.mCSB_scrollTools_horizontal .mCSB_buttonRight{right:0}.mCSB_container_wrapper{position:absolute;height:auto;width:auto;overflow:hidden;top:0;left:0;right:0;bottom:0;margin-right:30px;margin-bottom:30px}.mCSB_container_wrapper>.mCSB_container{padding-right:30px;padding-bottom:30px;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.mCSB_vertical_horizontal>.mCSB_scrollTools.mCSB_scrollTools_vertical{bottom:20px}.mCSB_vertical_horizontal>.mCSB_scrollTools.mCSB_scrollTools_horizontal{right:20px}.mCSB_container_wrapper.mCS_no_scrollbar_x.mCS_x_hidden+.mCSB_scrollTools.mCSB_scrollTools_vertical{bottom:0}.mCS-dir-rtl>.mCustomScrollBox.mCSB_vertical_horizontal.mCSB_inside>.mCSB_scrollTools.mCSB_scrollTools_horizontal,.mCSB_container_wrapper.mCS_no_scrollbar_y.mCS_y_hidden+.mCSB_scrollTools~.mCSB_scrollTools.mCSB_scrollTools_horizontal{right:0}.mCS-dir-rtl>.mCustomScrollBox.mCSB_vertical_horizontal.mCSB_inside>.mCSB_scrollTools.mCSB_scrollTools_horizontal{left:20px}.mCS-dir-rtl>.mCustomScrollBox.mCSB_vertical_horizontal.mCSB_inside>.mCSB_container_wrapper.mCS_no_scrollbar_y.mCS_y_hidden+.mCSB_scrollTools~.mCSB_scrollTools.mCSB_scrollTools_horizontal{left:0}.mCS-dir-rtl>.mCSB_inside>.mCSB_container_wrapper{margin-right:0;margin-left:30px}.mCSB_container_wrapper.mCS_no_scrollbar_y.mCS_y_hidden>.mCSB_container{padding-right:0}.mCSB_container_wrapper.mCS_no_scrollbar_x.mCS_x_hidden>.mCSB_container{padding-bottom:0}.mCustomScrollBox.mCSB_vertical_horizontal.mCSB_inside>.mCSB_container_wrapper.mCS_no_scrollbar_y.mCS_y_hidden{margin-right:0;margin-left:0}.mCustomScrollBox.mCSB_vertical_horizontal.mCSB_inside>.mCSB_container_wrapper.mCS_no_scrollbar_x.mCS_x_hidden{margin-bottom:0}.mCSB_scrollTools,.mCSB_scrollTools .mCSB_buttonDown,.mCSB_scrollTools .mCSB_buttonLeft,.mCSB_scrollTools .mCSB_buttonRight,.mCSB_scrollTools .mCSB_buttonUp,.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{-webkit-transition:opacity .2s ease-in-out,background-color .2s ease-in-out;-moz-transition:opacity .2s ease-in-out,background-color .2s ease-in-out;-o-transition:opacity .2s ease-in-out,background-color .2s ease-in-out;transition:opacity .2s ease-in-out,background-color .2s ease-in-out}.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_draggerRail,.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_dragger_bar,.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_draggerRail,.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_dragger_bar{-webkit-transition:width .2s ease-out .2s,height .2s ease-out .2s,margin-left .2s ease-out .2s,margin-right .2s ease-out .2s,margin-top .2s ease-out .2s,margin-bottom .2s ease-out .2s,opacity .2s ease-in-out,background-color .2s ease-in-out;-moz-transition:width .2s ease-out .2s,height .2s ease-out .2s,margin-left .2s ease-out .2s,margin-right .2s ease-out .2s,margin-top .2s ease-out .2s,margin-bottom .2s ease-out .2s,opacity .2s ease-in-out,background-color .2s ease-in-out;-o-transition:width .2s ease-out .2s,height .2s ease-out .2s,margin-left .2s ease-out .2s,margin-right .2s ease-out .2s,margin-top .2s ease-out .2s,margin-bottom .2s ease-out .2s,opacity .2s ease-in-out,background-color .2s ease-in-out;transition:width .2s ease-out .2s,height .2s ease-out .2s,margin-left .2s ease-out .2s,margin-right .2s ease-out .2s,margin-top .2s ease-out .2s,margin-bottom .2s ease-out .2s,opacity .2s ease-in-out,background-color .2s ease-in-out}.mCS-autoHide>.mCustomScrollBox>.mCSB_scrollTools,.mCS-autoHide>.mCustomScrollBox~.mCSB_scrollTools{opacity:0;filter:"alpha(opacity=0)";-ms-filter:"alpha(opacity=0)"}.mCS-autoHide:hover>.mCustomScrollBox>.mCSB_scrollTools,.mCS-autoHide:hover>.mCustomScrollBox~.mCSB_scrollTools,.mCustomScrollBox:hover>.mCSB_scrollTools,.mCustomScrollBox:hover~.mCSB_scrollTools,.mCustomScrollbar>.mCustomScrollBox>.mCSB_scrollTools.mCSB_scrollTools_onDrag,.mCustomScrollbar>.mCustomScrollBox~.mCSB_scrollTools.mCSB_scrollTools_onDrag{opacity:1;filter:"alpha(opacity=100)";-ms-filter:"alpha(opacity=100)"}.mCSB_scrollTools .mCSB_draggerRail{background-color:#000;background-color:rgba(0,0,0,.4);filter:"alpha(opacity=40)";-ms-filter:"alpha(opacity=40)"}.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{background-color:#fff;background-color:rgba(255,255,255,.75);filter:"alpha(opacity=75)";-ms-filter:"alpha(opacity=75)"}.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar{background-color:#fff;background-color:rgba(255,255,255,.85);filter:"alpha(opacity=85)";-ms-filter:"alpha(opacity=85)"}.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar{background-color:#fff;background-color:rgba(255,255,255,.9);filter:"alpha(opacity=90)";-ms-filter:"alpha(opacity=90)"}.mCSB_scrollTools .mCSB_buttonDown,.mCSB_scrollTools .mCSB_buttonLeft,.mCSB_scrollTools .mCSB_buttonRight,.mCSB_scrollTools .mCSB_buttonUp{background-image:url(mCSB_buttons.png);background-repeat:no-repeat;opacity:.4;filter:"alpha(opacity=40)";-ms-filter:"alpha(opacity=40)"}.mCSB_scrollTools .mCSB_buttonUp{background-position:0 0}.mCSB_scrollTools .mCSB_buttonDown{background-position:0 -20px}.mCSB_scrollTools .mCSB_buttonLeft{background-position:0 -40px}.mCSB_scrollTools .mCSB_buttonRight{background-position:0 -56px}.mCSB_scrollTools .mCSB_buttonDown:hover,.mCSB_scrollTools .mCSB_buttonLeft:hover,.mCSB_scrollTools .mCSB_buttonRight:hover,.mCSB_scrollTools .mCSB_buttonUp:hover{opacity:.75;filter:"alpha(opacity=75)";-ms-filter:"alpha(opacity=75)"}.mCSB_scrollTools .mCSB_buttonDown:active,.mCSB_scrollTools .mCSB_buttonLeft:active,.mCSB_scrollTools .mCSB_buttonRight:active,.mCSB_scrollTools .mCSB_buttonUp:active{opacity:.9;filter:"alpha(opacity=90)";-ms-filter:"alpha(opacity=90)"}.mCS-dark.mCSB_scrollTools .mCSB_draggerRail{background-color:#000;background-color:rgba(0,0,0,.15)}.mCS-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.75)}.mCS-dark.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar{background-color:rgba(0,0,0,.85)}.mCS-dark.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-dark.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar{background-color:rgba(0,0,0,.9)}.mCS-dark.mCSB_scrollTools .mCSB_buttonUp{background-position:-80px 0}.mCS-dark.mCSB_scrollTools .mCSB_buttonDown{background-position:-80px -20px}.mCS-dark.mCSB_scrollTools .mCSB_buttonLeft{background-position:-80px -40px}.mCS-dark.mCSB_scrollTools .mCSB_buttonRight{background-position:-80px -56px}.mCS-dark-2.mCSB_scrollTools .mCSB_draggerRail,.mCS-light-2.mCSB_scrollTools .mCSB_draggerRail{width:4px;background-color:#fff;background-color:rgba(255,255,255,.1);-webkit-border-radius:1px;-moz-border-radius:1px;border-radius:1px}.mCS-dark-2.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-light-2.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{width:4px;background-color:#fff;background-color:rgba(255,255,255,.75);-webkit-border-radius:1px;-moz-border-radius:1px;border-radius:1px}.mCS-dark-2.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-dark-2.mCSB_scrollTools_horizontal .mCSB_draggerRail,.mCS-light-2.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-light-2.mCSB_scrollTools_horizontal .mCSB_draggerRail{width:100%;height:4px;margin:6px auto}.mCS-light-2.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar{background-color:#fff;background-color:rgba(255,255,255,.85)}.mCS-light-2.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-light-2.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar{background-color:#fff;background-color:rgba(255,255,255,.9)}.mCS-light-2.mCSB_scrollTools .mCSB_buttonUp{background-position:-32px 0}.mCS-light-2.mCSB_scrollTools .mCSB_buttonDown{background-position:-32px -20px}.mCS-light-2.mCSB_scrollTools .mCSB_buttonLeft{background-position:-40px -40px}.mCS-light-2.mCSB_scrollTools .mCSB_buttonRight{background-position:-40px -56px}.mCS-dark-2.mCSB_scrollTools .mCSB_draggerRail{background-color:#000;background-color:rgba(0,0,0,.1);-webkit-border-radius:1px;-moz-border-radius:1px;border-radius:1px}.mCS-dark-2.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.75);-webkit-border-radius:1px;-moz-border-radius:1px;border-radius:1px}.mCS-dark-2.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.85)}.mCS-dark-2.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-dark-2.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.9)}.mCS-dark-2.mCSB_scrollTools .mCSB_buttonUp{background-position:-112px 0}.mCS-dark-2.mCSB_scrollTools .mCSB_buttonDown{background-position:-112px -20px}.mCS-dark-2.mCSB_scrollTools .mCSB_buttonLeft{background-position:-120px -40px}.mCS-dark-2.mCSB_scrollTools .mCSB_buttonRight{background-position:-120px -56px}.mCS-dark-thick.mCSB_scrollTools .mCSB_draggerRail,.mCS-light-thick.mCSB_scrollTools .mCSB_draggerRail{width:4px;background-color:#fff;background-color:rgba(255,255,255,.1);-webkit-border-radius:2px;-moz-border-radius:2px;border-radius:2px}.mCS-dark-thick.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-light-thick.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{width:6px;background-color:#fff;background-color:rgba(255,255,255,.75);-webkit-border-radius:2px;-moz-border-radius:2px;border-radius:2px}.mCS-dark-thick.mCSB_scrollTools_horizontal .mCSB_draggerRail,.mCS-light-thick.mCSB_scrollTools_horizontal .mCSB_draggerRail{width:100%;height:4px;margin:6px 0}.mCS-dark-thick.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-light-thick.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar{width:100%;height:6px;margin:5px auto}.mCS-light-thick.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar{background-color:#fff;background-color:rgba(255,255,255,.85)}.mCS-light-thick.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-light-thick.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar{background-color:#fff;background-color:rgba(255,255,255,.9)}.mCS-light-thick.mCSB_scrollTools .mCSB_buttonUp{background-position:-16px 0}.mCS-light-thick.mCSB_scrollTools .mCSB_buttonDown{background-position:-16px -20px}.mCS-light-thick.mCSB_scrollTools .mCSB_buttonLeft{background-position:-20px -40px}.mCS-light-thick.mCSB_scrollTools .mCSB_buttonRight{background-position:-20px -56px}.mCS-dark-thick.mCSB_scrollTools .mCSB_draggerRail{background-color:#000;background-color:rgba(0,0,0,.1);-webkit-border-radius:2px;-moz-border-radius:2px;border-radius:2px}.mCS-dark-thick.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.75);-webkit-border-radius:2px;-moz-border-radius:2px;border-radius:2px}.mCS-dark-thick.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.85)}.mCS-dark-thick.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-dark-thick.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.9)}.mCS-dark-thick.mCSB_scrollTools .mCSB_buttonUp{background-position:-96px 0}.mCS-dark-thick.mCSB_scrollTools .mCSB_buttonDown{background-position:-96px -20px}.mCS-dark-thick.mCSB_scrollTools .mCSB_buttonLeft{background-position:-100px -40px}.mCS-dark-thick.mCSB_scrollTools .mCSB_buttonRight{background-position:-100px -56px}.mCS-light-thin.mCSB_scrollTools .mCSB_draggerRail{background-color:#fff;background-color:rgba(255,255,255,.1)}.mCS-dark-thin.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-light-thin.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{width:2px}.mCS-dark-thin.mCSB_scrollTools_horizontal .mCSB_draggerRail,.mCS-light-thin.mCSB_scrollTools_horizontal .mCSB_draggerRail{width:100%}.mCS-dark-thin.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-light-thin.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar{width:100%;height:2px;margin:7px auto}.mCS-dark-thin.mCSB_scrollTools .mCSB_draggerRail{background-color:#000;background-color:rgba(0,0,0,.15)}.mCS-dark-thin.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.75)}.mCS-dark-thin.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.85)}.mCS-dark-thin.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-dark-thin.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.9)}.mCS-dark-thin.mCSB_scrollTools .mCSB_buttonUp{background-position:-80px 0}.mCS-dark-thin.mCSB_scrollTools .mCSB_buttonDown{background-position:-80px -20px}.mCS-dark-thin.mCSB_scrollTools .mCSB_buttonLeft{background-position:-80px -40px}.mCS-dark-thin.mCSB_scrollTools .mCSB_buttonRight{background-position:-80px -56px}.mCS-rounded.mCSB_scrollTools .mCSB_draggerRail{background-color:#fff;background-color:rgba(255,255,255,.15)}.mCS-rounded-dark.mCSB_scrollTools .mCSB_dragger,.mCS-rounded-dots-dark.mCSB_scrollTools .mCSB_dragger,.mCS-rounded-dots.mCSB_scrollTools .mCSB_dragger,.mCS-rounded.mCSB_scrollTools .mCSB_dragger{height:14px}.mCS-rounded-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-rounded-dots-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-rounded-dots.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-rounded.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{width:14px;margin:0 1px}.mCS-rounded-dark.mCSB_scrollTools_horizontal .mCSB_dragger,.mCS-rounded-dots-dark.mCSB_scrollTools_horizontal .mCSB_dragger,.mCS-rounded-dots.mCSB_scrollTools_horizontal .mCSB_dragger,.mCS-rounded.mCSB_scrollTools_horizontal .mCSB_dragger{width:14px}.mCS-rounded-dark.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-rounded-dots-dark.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-rounded-dots.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-rounded.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar{height:14px;margin:1px 0}.mCS-rounded-dark.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded .mCSB_dragger_bar,.mCS-rounded-dark.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_dragger .mCSB_dragger_bar,.mCS-rounded.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded .mCSB_dragger_bar,.mCS-rounded.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_dragger .mCSB_dragger_bar{width:16px;height:16px;margin:-1px 0}.mCS-rounded-dark.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded+.mCSB_draggerRail,.mCS-rounded-dark.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_draggerRail,.mCS-rounded.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded+.mCSB_draggerRail,.mCS-rounded.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_draggerRail{width:4px}.mCS-rounded-dark.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded .mCSB_dragger_bar,.mCS-rounded-dark.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_dragger .mCSB_dragger_bar,.mCS-rounded.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded .mCSB_dragger_bar,.mCS-rounded.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_dragger .mCSB_dragger_bar{height:16px;width:16px;margin:0 -1px}.mCS-rounded-dark.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded+.mCSB_draggerRail,.mCS-rounded-dark.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_draggerRail,.mCS-rounded.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded+.mCSB_draggerRail,.mCS-rounded.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_draggerRail{height:4px;margin:6px 0}.mCS-rounded.mCSB_scrollTools .mCSB_buttonUp{background-position:0 -72px}.mCS-rounded.mCSB_scrollTools .mCSB_buttonDown{background-position:0 -92px}.mCS-rounded.mCSB_scrollTools .mCSB_buttonLeft{background-position:0 -112px}.mCS-rounded.mCSB_scrollTools .mCSB_buttonRight{background-position:0 -128px}.mCS-rounded-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-rounded-dots-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.75)}.mCS-rounded-dark.mCSB_scrollTools .mCSB_draggerRail{background-color:#000;background-color:rgba(0,0,0,.15)}.mCS-rounded-dark.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar,.mCS-rounded-dots-dark.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.85)}.mCS-rounded-dark.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-rounded-dark.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar,.mCS-rounded-dots-dark.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-rounded-dots-dark.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.9)}.mCS-rounded-dark.mCSB_scrollTools .mCSB_buttonUp{background-position:-80px -72px}.mCS-rounded-dark.mCSB_scrollTools .mCSB_buttonDown{background-position:-80px -92px}.mCS-rounded-dark.mCSB_scrollTools .mCSB_buttonLeft{background-position:-80px -112px}.mCS-rounded-dark.mCSB_scrollTools .mCSB_buttonRight{background-position:-80px -128px}.mCS-rounded-dots-dark.mCSB_scrollTools_vertical .mCSB_draggerRail,.mCS-rounded-dots.mCSB_scrollTools_vertical .mCSB_draggerRail{width:4px}.mCS-rounded-dots-dark.mCSB_scrollTools .mCSB_draggerRail,.mCS-rounded-dots-dark.mCSB_scrollTools_horizontal .mCSB_draggerRail,.mCS-rounded-dots.mCSB_scrollTools .mCSB_draggerRail,.mCS-rounded-dots.mCSB_scrollTools_horizontal .mCSB_draggerRail{background-color:transparent;background-position:center}.mCS-rounded-dots-dark.mCSB_scrollTools .mCSB_draggerRail,.mCS-rounded-dots.mCSB_scrollTools .mCSB_draggerRail{background-image:url();background-repeat:repeat-y;opacity:.3;filter:"alpha(opacity=30)";-ms-filter:"alpha(opacity=30)"}.mCS-rounded-dots-dark.mCSB_scrollTools_horizontal .mCSB_draggerRail,.mCS-rounded-dots.mCSB_scrollTools_horizontal .mCSB_draggerRail{height:4px;margin:6px 0;background-repeat:repeat-x}.mCS-rounded-dots.mCSB_scrollTools .mCSB_buttonUp{background-position:-16px -72px}.mCS-rounded-dots.mCSB_scrollTools .mCSB_buttonDown{background-position:-16px -92px}.mCS-rounded-dots.mCSB_scrollTools .mCSB_buttonLeft{background-position:-20px -112px}.mCS-rounded-dots.mCSB_scrollTools .mCSB_buttonRight{background-position:-20px -128px}.mCS-rounded-dots-dark.mCSB_scrollTools .mCSB_draggerRail{background-image:url()}.mCS-rounded-dots-dark.mCSB_scrollTools .mCSB_buttonUp{background-position:-96px -72px}.mCS-rounded-dots-dark.mCSB_scrollTools .mCSB_buttonDown{background-position:-96px -92px}.mCS-rounded-dots-dark.mCSB_scrollTools .mCSB_buttonLeft{background-position:-100px -112px}.mCS-rounded-dots-dark.mCSB_scrollTools .mCSB_buttonRight{background-position:-100px -128px}.mCS-3d-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-3d-thick-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-3d-thick.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-3d.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{background-repeat:repeat-y;background-image:-moz-linear-gradient(left,rgba(255,255,255,.5) 0,rgba(255,255,255,0) 100%);background-image:-webkit-gradient(linear,left top,right top,color-stop(0,rgba(255,255,255,.5)),color-stop(100%,rgba(255,255,255,0)));background-image:-webkit-linear-gradient(left,rgba(255,255,255,.5) 0,rgba(255,255,255,0) 100%);background-image:-o-linear-gradient(left,rgba(255,255,255,.5) 0,rgba(255,255,255,0) 100%);background-image:-ms-linear-gradient(left,rgba(255,255,255,.5) 0,rgba(255,255,255,0) 100%);background-image:linear-gradient(to right,rgba(255,255,255,.5) 0,rgba(255,255,255,0) 100%)}.mCS-3d-dark.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-3d-thick-dark.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-3d-thick.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-3d.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar{background-repeat:repeat-x;background-image:-moz-linear-gradient(top,rgba(255,255,255,.5) 0,rgba(255,255,255,0) 100%);background-image:-webkit-gradient(linear,left top,left bottom,color-stop(0,rgba(255,255,255,.5)),color-stop(100%,rgba(255,255,255,0)));background-image:-webkit-linear-gradient(top,rgba(255,255,255,.5) 0,rgba(255,255,255,0) 100%);background-image:-o-linear-gradient(top,rgba(255,255,255,.5) 0,rgba(255,255,255,0) 100%);background-image:-ms-linear-gradient(top,rgba(255,255,255,.5) 0,rgba(255,255,255,0) 100%);background-image:linear-gradient(to bottom,rgba(255,255,255,.5) 0,rgba(255,255,255,0) 100%)}.mCS-3d-dark.mCSB_scrollTools_vertical .mCSB_dragger,.mCS-3d.mCSB_scrollTools_vertical .mCSB_dragger{height:70px}.mCS-3d-dark.mCSB_scrollTools_horizontal .mCSB_dragger,.mCS-3d.mCSB_scrollTools_horizontal .mCSB_dragger{width:70px}.mCS-3d-dark.mCSB_scrollTools,.mCS-3d.mCSB_scrollTools{opacity:1;filter:"alpha(opacity=30)";-ms-filter:"alpha(opacity=30)"}.mCS-3d-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-3d-dark.mCSB_scrollTools .mCSB_draggerRail,.mCS-3d.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-3d.mCSB_scrollTools .mCSB_draggerRail{-webkit-border-radius:16px;-moz-border-radius:16px;border-radius:16px}.mCS-3d-dark.mCSB_scrollTools .mCSB_draggerRail,.mCS-3d.mCSB_scrollTools .mCSB_draggerRail{width:8px;background-color:#000;background-color:rgba(0,0,0,.2);box-shadow:inset 1px 0 1px rgba(0,0,0,.5),inset -1px 0 1px rgba(255,255,255,.2)}.mCS-3d-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-3d-dark.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-3d-dark.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar,.mCS-3d-dark.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar,.mCS-3d.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-3d.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-3d.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar,.mCS-3d.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar{background-color:#555}.mCS-3d-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-3d.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{width:8px}.mCS-3d-dark.mCSB_scrollTools_horizontal .mCSB_draggerRail,.mCS-3d.mCSB_scrollTools_horizontal .mCSB_draggerRail{width:100%;height:8px;margin:4px 0;box-shadow:inset 0 1px 1px rgba(0,0,0,.5),inset 0 -1px 1px rgba(255,255,255,.2)}.mCS-3d-dark.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-3d.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar{width:100%;height:8px;margin:4px auto}.mCS-3d.mCSB_scrollTools .mCSB_buttonUp{background-position:-32px -72px}.mCS-3d.mCSB_scrollTools .mCSB_buttonDown{background-position:-32px -92px}.mCS-3d.mCSB_scrollTools .mCSB_buttonLeft{background-position:-40px -112px}.mCS-3d.mCSB_scrollTools .mCSB_buttonRight{background-position:-40px -128px}.mCS-3d-dark.mCSB_scrollTools .mCSB_draggerRail{background-color:#000;background-color:rgba(0,0,0,.1);box-shadow:inset 1px 0 1px rgba(0,0,0,.1)}.mCS-3d-dark.mCSB_scrollTools_horizontal .mCSB_draggerRail{box-shadow:inset 0 1px 1px rgba(0,0,0,.1)}.mCS-3d-dark.mCSB_scrollTools .mCSB_buttonUp{background-position:-112px -72px}.mCS-3d-dark.mCSB_scrollTools .mCSB_buttonDown{background-position:-112px -92px}.mCS-3d-dark.mCSB_scrollTools .mCSB_buttonLeft{background-position:-120px -112px}.mCS-3d-dark.mCSB_scrollTools .mCSB_buttonRight{background-position:-120px -128px}.mCS-3d-thick-dark.mCSB_scrollTools,.mCS-3d-thick.mCSB_scrollTools{opacity:1;filter:"alpha(opacity=30)";-ms-filter:"alpha(opacity=30)"}.mCS-3d-thick-dark.mCSB_scrollTools,.mCS-3d-thick-dark.mCSB_scrollTools .mCSB_draggerContainer,.mCS-3d-thick.mCSB_scrollTools,.mCS-3d-thick.mCSB_scrollTools .mCSB_draggerContainer{-webkit-border-radius:7px;-moz-border-radius:7px;border-radius:7px}.mCSB_inside+.mCS-3d-thick-dark.mCSB_scrollTools_vertical,.mCSB_inside+.mCS-3d-thick.mCSB_scrollTools_vertical{right:1px}.mCS-3d-thick-dark.mCSB_scrollTools_vertical,.mCS-3d-thick.mCSB_scrollTools_vertical{box-shadow:inset 1px 0 1px rgba(0,0,0,.1),inset 0 0 14px rgba(0,0,0,.5)}.mCS-3d-thick-dark.mCSB_scrollTools_horizontal,.mCS-3d-thick.mCSB_scrollTools_horizontal{bottom:1px;box-shadow:inset 0 1px 1px rgba(0,0,0,.1),inset 0 0 14px rgba(0,0,0,.5)}.mCS-3d-thick-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-3d-thick.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{-webkit-border-radius:5px;-moz-border-radius:5px;border-radius:5px;box-shadow:inset 1px 0 0 rgba(255,255,255,.4);width:12px;margin:2px;position:absolute;height:auto;top:0;bottom:0;left:0;right:0}.mCS-3d-thick-dark.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-3d-thick.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar{box-shadow:inset 0 1px 0 rgba(255,255,255,.4);height:12px;width:auto}.mCS-3d-thick.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-3d-thick.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-3d-thick.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar,.mCS-3d-thick.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar{background-color:#555}.mCS-3d-thick.mCSB_scrollTools .mCSB_draggerContainer{background-color:#000;background-color:rgba(0,0,0,.05);box-shadow:inset 1px 1px 16px rgba(0,0,0,.1)}.mCS-3d-thick.mCSB_scrollTools .mCSB_draggerRail{background-color:transparent}.mCS-3d-thick.mCSB_scrollTools .mCSB_buttonUp{background-position:-32px -72px}.mCS-3d-thick.mCSB_scrollTools .mCSB_buttonDown{background-position:-32px -92px}.mCS-3d-thick.mCSB_scrollTools .mCSB_buttonLeft{background-position:-40px -112px}.mCS-3d-thick.mCSB_scrollTools .mCSB_buttonRight{background-position:-40px -128px}.mCS-3d-thick-dark.mCSB_scrollTools{box-shadow:inset 0 0 14px rgba(0,0,0,.2)}.mCS-3d-thick-dark.mCSB_scrollTools_horizontal{box-shadow:inset 0 1px 1px rgba(0,0,0,.1),inset 0 0 14px rgba(0,0,0,.2)}.mCS-3d-thick-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{box-shadow:inset 1px 0 0 rgba(255,255,255,.4),inset -1px 0 0 rgba(0,0,0,.2)}.mCS-3d-thick-dark.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar{box-shadow:inset 0 1px 0 rgba(255,255,255,.4),inset 0 -1px 0 rgba(0,0,0,.2)}.mCS-3d-thick-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-3d-thick-dark.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-3d-thick-dark.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar,.mCS-3d-thick-dark.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar{background-color:#777}.mCS-3d-thick-dark.mCSB_scrollTools .mCSB_draggerContainer{background-color:#fff;background-color:rgba(0,0,0,.05);box-shadow:inset 1px 1px 16px rgba(0,0,0,.1)}.mCS-3d-thick-dark.mCSB_scrollTools .mCSB_draggerRail,.mCS-minimal-dark.mCSB_scrollTools .mCSB_draggerRail,.mCS-minimal.mCSB_scrollTools .mCSB_draggerRail{background-color:transparent}.mCS-3d-thick-dark.mCSB_scrollTools .mCSB_buttonUp{background-position:-112px -72px}.mCS-3d-thick-dark.mCSB_scrollTools .mCSB_buttonDown{background-position:-112px -92px}.mCS-3d-thick-dark.mCSB_scrollTools .mCSB_buttonLeft{background-position:-120px -112px}.mCS-3d-thick-dark.mCSB_scrollTools .mCSB_buttonRight{background-position:-120px -128px}.mCSB_outside+.mCS-minimal-dark.mCSB_scrollTools_vertical,.mCSB_outside+.mCS-minimal.mCSB_scrollTools_vertical{right:0;margin:12px 0}.mCustomScrollBox.mCS-minimal+.mCSB_scrollTools+.mCSB_scrollTools.mCSB_scrollTools_horizontal,.mCustomScrollBox.mCS-minimal+.mCSB_scrollTools.mCSB_scrollTools_horizontal,.mCustomScrollBox.mCS-minimal-dark+.mCSB_scrollTools+.mCSB_scrollTools.mCSB_scrollTools_horizontal,.mCustomScrollBox.mCS-minimal-dark+.mCSB_scrollTools.mCSB_scrollTools_horizontal{bottom:0;margin:0 12px}.mCS-dir-rtl>.mCSB_outside+.mCS-minimal-dark.mCSB_scrollTools_vertical,.mCS-dir-rtl>.mCSB_outside+.mCS-minimal.mCSB_scrollTools_vertical{left:0;right:auto}.mCS-minimal-dark.mCSB_scrollTools_vertical .mCSB_dragger,.mCS-minimal.mCSB_scrollTools_vertical .mCSB_dragger{height:50px}.mCS-minimal-dark.mCSB_scrollTools_horizontal .mCSB_dragger,.mCS-minimal.mCSB_scrollTools_horizontal .mCSB_dragger{width:50px}.mCS-minimal.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{background-color:#fff;background-color:rgba(255,255,255,.2);filter:"alpha(opacity=20)";-ms-filter:"alpha(opacity=20)"}.mCS-minimal.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-minimal.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar{background-color:#fff;background-color:rgba(255,255,255,.5);filter:"alpha(opacity=50)";-ms-filter:"alpha(opacity=50)"}.mCS-minimal-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.2);filter:"alpha(opacity=20)";-ms-filter:"alpha(opacity=20)"}.mCS-minimal-dark.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-minimal-dark.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.5);filter:"alpha(opacity=50)";-ms-filter:"alpha(opacity=50)"}.mCS-dark-3.mCSB_scrollTools .mCSB_draggerRail,.mCS-light-3.mCSB_scrollTools .mCSB_draggerRail{width:6px;background-color:#000;background-color:rgba(0,0,0,.2)}.mCS-dark-3.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-light-3.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{width:6px}.mCS-dark-3.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-dark-3.mCSB_scrollTools_horizontal .mCSB_draggerRail,.mCS-light-3.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-light-3.mCSB_scrollTools_horizontal .mCSB_draggerRail{width:100%;height:6px;margin:5px 0}.mCS-dark-3.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded+.mCSB_draggerRail,.mCS-dark-3.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_draggerRail,.mCS-light-3.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded+.mCSB_draggerRail,.mCS-light-3.mCSB_scrollTools_vertical.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_draggerRail{width:12px}.mCS-dark-3.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded+.mCSB_draggerRail,.mCS-dark-3.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_draggerRail,.mCS-light-3.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_dragger.mCSB_dragger_onDrag_expanded+.mCSB_draggerRail,.mCS-light-3.mCSB_scrollTools_horizontal.mCSB_scrollTools_onDrag_expand .mCSB_draggerContainer:hover .mCSB_draggerRail{height:12px;margin:2px 0}.mCS-light-3.mCSB_scrollTools .mCSB_buttonUp{background-position:-32px -72px}.mCS-light-3.mCSB_scrollTools .mCSB_buttonDown{background-position:-32px -92px}.mCS-light-3.mCSB_scrollTools .mCSB_buttonLeft{background-position:-40px -112px}.mCS-light-3.mCSB_scrollTools .mCSB_buttonRight{background-position:-40px -128px}.mCS-dark-3.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.75)}.mCS-dark-3.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.85)}.mCS-dark-3.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-dark-3.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.9)}.mCS-dark-3.mCSB_scrollTools .mCSB_draggerRail{background-color:#000;background-color:rgba(0,0,0,.1)}.mCS-dark-3.mCSB_scrollTools .mCSB_buttonUp{background-position:-112px -72px}.mCS-dark-3.mCSB_scrollTools .mCSB_buttonDown{background-position:-112px -92px}.mCS-dark-3.mCSB_scrollTools .mCSB_buttonLeft{background-position:-120px -112px}.mCS-dark-3.mCSB_scrollTools .mCSB_buttonRight{background-position:-120px -128px}.mCS-inset-2-dark.mCSB_scrollTools .mCSB_draggerRail,.mCS-inset-2.mCSB_scrollTools .mCSB_draggerRail,.mCS-inset-3-dark.mCSB_scrollTools .mCSB_draggerRail,.mCS-inset-3.mCSB_scrollTools .mCSB_draggerRail,.mCS-inset-dark.mCSB_scrollTools .mCSB_draggerRail,.mCS-inset.mCSB_scrollTools .mCSB_draggerRail{width:12px;background-color:#000;background-color:rgba(0,0,0,.2)}.mCS-inset-2-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-inset-2.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-inset-3-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-inset-3.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-inset-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-inset.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{width:6px;margin:3px 5px;position:absolute;height:auto;top:0;bottom:0;left:0;right:0}.mCS-inset-2-dark.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-inset-2.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-inset-3-dark.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-inset-3.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-inset-dark.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar,.mCS-inset.mCSB_scrollTools_horizontal .mCSB_dragger .mCSB_dragger_bar{height:6px;margin:5px 3px;position:absolute;width:auto;top:0;bottom:0;left:0;right:0}.mCS-inset-2-dark.mCSB_scrollTools_horizontal .mCSB_draggerRail,.mCS-inset-2.mCSB_scrollTools_horizontal .mCSB_draggerRail,.mCS-inset-3-dark.mCSB_scrollTools_horizontal .mCSB_draggerRail,.mCS-inset-3.mCSB_scrollTools_horizontal .mCSB_draggerRail,.mCS-inset-dark.mCSB_scrollTools_horizontal .mCSB_draggerRail,.mCS-inset.mCSB_scrollTools_horizontal .mCSB_draggerRail{width:100%;height:12px;margin:2px 0}.mCS-inset-2.mCSB_scrollTools .mCSB_buttonUp,.mCS-inset-3.mCSB_scrollTools .mCSB_buttonUp,.mCS-inset.mCSB_scrollTools .mCSB_buttonUp{background-position:-32px -72px}.mCS-inset-2.mCSB_scrollTools .mCSB_buttonDown,.mCS-inset-3.mCSB_scrollTools .mCSB_buttonDown,.mCS-inset.mCSB_scrollTools .mCSB_buttonDown{background-position:-32px -92px}.mCS-inset-2.mCSB_scrollTools .mCSB_buttonLeft,.mCS-inset-3.mCSB_scrollTools .mCSB_buttonLeft,.mCS-inset.mCSB_scrollTools .mCSB_buttonLeft{background-position:-40px -112px}.mCS-inset-2.mCSB_scrollTools .mCSB_buttonRight,.mCS-inset-3.mCSB_scrollTools .mCSB_buttonRight,.mCS-inset.mCSB_scrollTools .mCSB_buttonRight{background-position:-40px -128px}.mCS-inset-2-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-inset-3-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar,.mCS-inset-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.75)}.mCS-inset-2-dark.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar,.mCS-inset-3-dark.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar,.mCS-inset-dark.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.85)}.mCS-inset-2-dark.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-inset-2-dark.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar,.mCS-inset-3-dark.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-inset-3-dark.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar,.mCS-inset-dark.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-inset-dark.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.9)}.mCS-inset-2-dark.mCSB_scrollTools .mCSB_draggerRail,.mCS-inset-3-dark.mCSB_scrollTools .mCSB_draggerRail,.mCS-inset-dark.mCSB_scrollTools .mCSB_draggerRail{background-color:#000;background-color:rgba(0,0,0,.1)}.mCS-inset-2-dark.mCSB_scrollTools .mCSB_buttonUp,.mCS-inset-3-dark.mCSB_scrollTools .mCSB_buttonUp,.mCS-inset-dark.mCSB_scrollTools .mCSB_buttonUp{background-position:-112px -72px}.mCS-inset-2-dark.mCSB_scrollTools .mCSB_buttonDown,.mCS-inset-3-dark.mCSB_scrollTools .mCSB_buttonDown,.mCS-inset-dark.mCSB_scrollTools .mCSB_buttonDown{background-position:-112px -92px}.mCS-inset-2-dark.mCSB_scrollTools .mCSB_buttonLeft,.mCS-inset-3-dark.mCSB_scrollTools .mCSB_buttonLeft,.mCS-inset-dark.mCSB_scrollTools .mCSB_buttonLeft{background-position:-120px -112px}.mCS-inset-2-dark.mCSB_scrollTools .mCSB_buttonRight,.mCS-inset-3-dark.mCSB_scrollTools .mCSB_buttonRight,.mCS-inset-dark.mCSB_scrollTools .mCSB_buttonRight{background-position:-120px -128px}.mCS-inset-2-dark.mCSB_scrollTools .mCSB_draggerRail,.mCS-inset-2.mCSB_scrollTools .mCSB_draggerRail{background-color:transparent;border-width:1px;border-style:solid;border-color:#fff;border-color:rgba(255,255,255,.2);-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.mCS-inset-2-dark.mCSB_scrollTools .mCSB_draggerRail{border-color:#000;border-color:rgba(0,0,0,.2)}.mCS-inset-3.mCSB_scrollTools .mCSB_draggerRail{background-color:#fff;background-color:rgba(255,255,255,.6)}.mCS-inset-3-dark.mCSB_scrollTools .mCSB_draggerRail{background-color:#000;background-color:rgba(0,0,0,.6)}.mCS-inset-3.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.75)}.mCS-inset-3.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.85)}.mCS-inset-3.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-inset-3.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar{background-color:#000;background-color:rgba(0,0,0,.9)}.mCS-inset-3-dark.mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar{background-color:#fff;background-color:rgba(255,255,255,.75)}.mCS-inset-3-dark.mCSB_scrollTools .mCSB_dragger:hover .mCSB_dragger_bar{background-color:#fff;background-color:rgba(255,255,255,.85)}.mCS-inset-3-dark.mCSB_scrollTools .mCSB_dragger.mCSB_dragger_onDrag .mCSB_dragger_bar,.mCS-inset-3-dark.mCSB_scrollTools .mCSB_dragger:active .mCSB_dragger_bar{background-color:#fff;background-color:rgba(255,255,255,.9)}
--------------------------------------------------------------------------------
/ui/static/res/botim.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/suriyadeepan/easy_seq2seq/c05b0d02f15e9bd1947ff79ddcd0b761a214e2a9/ui/static/res/botim.png
--------------------------------------------------------------------------------
/ui/static/res/hd1.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/suriyadeepan/easy_seq2seq/c05b0d02f15e9bd1947ff79ddcd0b761a214e2a9/ui/static/res/hd1.jpg
--------------------------------------------------------------------------------
/ui/static/res/hd2.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/suriyadeepan/easy_seq2seq/c05b0d02f15e9bd1947ff79ddcd0b761a214e2a9/ui/static/res/hd2.jpg
--------------------------------------------------------------------------------
/ui/static/scss/style.scss:
--------------------------------------------------------------------------------
1 | /*--------------------
2 | Mixins
3 | --------------------*/
4 | @mixin center {
5 | position: absolute;
6 | top: 50%;
7 | left: 50%;
8 | transform: translate(-50%, -50%);
9 | }
10 |
11 | @mixin ball {
12 | @include center;
13 | content: '';
14 | display: block;
15 | width: 3px;
16 | height: 3px;
17 | border-radius: 50%;
18 | background: rgba(255, 255, 255, .5);
19 | z-index: 2;
20 | margin-top: 4px;
21 | animation: ball .45s cubic-bezier(0, 0, 0.15, 1) alternate infinite;
22 | }
23 |
24 |
25 | /*--------------------
26 | Body
27 | --------------------*/
28 | *,
29 | *::before,
30 | *::after {
31 | box-sizing: border-box;
32 | }
33 |
34 | html,
35 | body {
36 | height: 100%;
37 | }
38 |
39 | body {
40 | background: linear-gradient(135deg, #044f48, #2a7561);
41 | background-size: cover;
42 | font-family: 'Open Sans', sans-serif;
43 | font-size: 12px;
44 | line-height: 1.3;
45 | overflow: hidden;
46 | }
47 |
48 | .bg {
49 | width: 100%;
50 | height: 100%;
51 | top: 0;
52 | left: 0;
53 | z-index: 1;
54 | background: url('https://images.unsplash.com/photo-1451186859696-371d9477be93?crop=entropy&fit=crop&fm=jpg&h=975&ixjsv=2.1.0&ixlib=rb-0.3.5&q=80&w=1925') no-repeat 0 0;
55 | filter: blur(80px);
56 | transform: scale(1.2);
57 | }
58 |
59 |
60 | /*--------------------
61 | Chat
62 | --------------------*/
63 | .chat {
64 | @include center;
65 | width: 300px;
66 | height: 80vh;
67 | max-height: 500px;
68 | z-index: 2;
69 | overflow: hidden;
70 | box-shadow: 0 5px 30px rgba(0, 0, 0, .2);
71 | background: rgba(0, 0, 0, .5);
72 | border-radius: 20px;
73 | display: flex;
74 | justify-content: space-between;
75 | flex-direction: column;
76 | }
77 |
78 |
79 | /*--------------------
80 | Chat Title
81 | --------------------*/
82 | .chat-title {
83 | flex: 0 1 45px;
84 | position: relative;
85 | z-index: 2;
86 | background: rgba(0, 0, 0, 0.2);
87 | color: #fff;
88 | text-transform: uppercase;
89 | text-align: left;
90 | padding: 10px 10px 10px 50px;
91 |
92 | h1, h2 {
93 | font-weight: normal;
94 | font-size: 10px;
95 | margin: 0;
96 | padding: 0;
97 | }
98 |
99 | h2 {
100 | color: rgba(255, 255, 255, .5);
101 | font-size: 8px;
102 | letter-spacing: 1px;
103 | }
104 |
105 | .avatar {
106 | position: absolute;
107 | z-index: 1;
108 | top: 8px;
109 | left: 9px;
110 | border-radius: 30px;
111 | width: 30px;
112 | height: 30px;
113 | overflow: hidden;
114 | margin: 0;
115 | padding: 0;
116 | border: 2px solid rgba(255, 255, 255, 0.24);
117 |
118 | img {
119 | width: 100%;
120 | height: auto;
121 | }
122 | }
123 | }
124 |
125 |
126 | /*--------------------
127 | Messages
128 | --------------------*/
129 | .messages {
130 | flex: 1 1 auto;
131 | color: rgba(255, 255, 255, .5);
132 | overflow: hidden;
133 | position: relative;
134 | width: 100%;
135 |
136 | & .messages-content {
137 | position: absolute;
138 | top: 0;
139 | left: 0;
140 | height: 101%;
141 | width: 100%;
142 | }
143 |
144 |
145 | .message {
146 | clear: both;
147 | float: left;
148 | padding: 6px 10px 7px;
149 | border-radius: 10px 10px 10px 0;
150 | background: rgba(0, 0, 0, .3);
151 | margin: 8px 0;
152 | font-size: 11px;
153 | line-height: 1.4;
154 | margin-left: 35px;
155 | position: relative;
156 | text-shadow: 0 1px 1px rgba(0, 0, 0, .2);
157 |
158 | .timestamp {
159 | position: absolute;
160 | bottom: -15px;
161 | font-size: 9px;
162 | color: rgba(255, 255, 255, .3);
163 | }
164 |
165 | &::before {
166 | content: '';
167 | position: absolute;
168 | bottom: -6px;
169 | border-top: 6px solid rgba(0, 0, 0, .3);
170 | left: 0;
171 | border-right: 7px solid transparent;
172 | }
173 |
174 | .avatar {
175 | position: absolute;
176 | z-index: 1;
177 | bottom: -15px;
178 | left: -35px;
179 | border-radius: 30px;
180 | width: 30px;
181 | height: 30px;
182 | overflow: hidden;
183 | margin: 0;
184 | padding: 0;
185 | border: 2px solid rgba(255, 255, 255, 0.24);
186 |
187 | img {
188 | width: 100%;
189 | height: auto;
190 | }
191 | }
192 |
193 | &.message-personal {
194 | float: right;
195 | color: #fff;
196 | text-align: right;
197 | background: linear-gradient(120deg, #248A52, #257287);
198 | border-radius: 10px 10px 0 10px;
199 |
200 | &::before {
201 | left: auto;
202 | right: 0;
203 | border-right: none;
204 | border-left: 5px solid transparent;
205 | border-top: 4px solid #257287;
206 | bottom: -4px;
207 | }
208 | }
209 |
210 | &:last-child {
211 | margin-bottom: 30px;
212 | }
213 |
214 | &.new {
215 | transform: scale(0);
216 | transform-origin: 0 0;
217 | animation: bounce 500ms linear both;
218 | }
219 |
220 | &.loading {
221 |
222 | &::before {
223 | @include ball;
224 | border: none;
225 | animation-delay: .15s;
226 | }
227 |
228 | & span {
229 | display: block;
230 | font-size: 0;
231 | width: 20px;
232 | height: 10px;
233 | position: relative;
234 |
235 | &::before {
236 | @include ball;
237 | margin-left: -7px;
238 | }
239 |
240 | &::after {
241 | @include ball;
242 | margin-left: 7px;
243 | animation-delay: .3s;
244 | }
245 | }
246 | }
247 |
248 | }
249 | }
250 |
251 |
252 | /*--------------------
253 | Message Box
254 | --------------------*/
255 | .message-box {
256 | flex: 0 1 40px;
257 | width: 100%;
258 | background: rgba(0, 0, 0, 0.3);
259 | padding: 10px;
260 | position: relative;
261 |
262 | & .message-input {
263 | background: none;
264 | border: none;
265 | outline: none!important;
266 | resize: none;
267 | color: rgba(255, 255, 255, .7);
268 | font-size: 11px;
269 | height: 17px;
270 | margin: 0;
271 | padding-right: 20px;
272 | width: 265px;
273 | }
274 |
275 | textarea:focus:-webkit-placeholder{
276 | color: transparent;
277 | }
278 |
279 | & .message-submit {
280 | position: absolute;
281 | z-index: 1;
282 | top: 9px;
283 | right: 10px;
284 | color: #fff;
285 | border: none;
286 | background: #248A52;
287 | font-size: 10px;
288 | text-transform: uppercase;
289 | line-height: 1;
290 | padding: 6px 10px;
291 | border-radius: 10px;
292 | outline: none!important;
293 | transition: background .2s ease;
294 |
295 | &:hover {
296 | background: #1D7745;
297 | }
298 | }
299 | }
300 |
301 |
302 | /*--------------------
303 | Custom Srollbar
304 | --------------------*/
305 | .mCSB_scrollTools {
306 | margin: 1px -3px 1px 0;
307 | opacity: 0;
308 | }
309 |
310 | .mCSB_inside > .mCSB_container {
311 | margin-right: 0px;
312 | padding: 0 10px;
313 | }
314 |
315 | .mCSB_scrollTools .mCSB_dragger .mCSB_dragger_bar {
316 | background-color: rgba(0, 0, 0, 0.5)!important;
317 | }
318 |
319 |
320 | /*--------------------
321 | Bounce
322 | --------------------*/
323 | @keyframes bounce {
324 | 0% { transform: matrix3d(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1); }
325 | 4.7% { transform: matrix3d(0.45, 0, 0, 0, 0, 0.45, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1); }
326 | 9.41% { transform: matrix3d(0.883, 0, 0, 0, 0, 0.883, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1); }
327 | 14.11% { transform: matrix3d(1.141, 0, 0, 0, 0, 1.141, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1); }
328 | 18.72% { transform: matrix3d(1.212, 0, 0, 0, 0, 1.212, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1); }
329 | 24.32% { transform: matrix3d(1.151, 0, 0, 0, 0, 1.151, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1); }
330 | 29.93% { transform: matrix3d(1.048, 0, 0, 0, 0, 1.048, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1); }
331 | 35.54% { transform: matrix3d(0.979, 0, 0, 0, 0, 0.979, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1); }
332 | 41.04% { transform: matrix3d(0.961, 0, 0, 0, 0, 0.961, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1); }
333 | 52.15% { transform: matrix3d(0.991, 0, 0, 0, 0, 0.991, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1); }
334 | 63.26% { transform: matrix3d(1.007, 0, 0, 0, 0, 1.007, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1); }
335 | 85.49% { transform: matrix3d(0.999, 0, 0, 0, 0, 0.999, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1); }
336 | 100% { transform: matrix3d(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1); }
337 | }
338 |
339 |
340 | @keyframes ball {
341 | from {
342 | transform: translateY(0) scaleY(.8);
343 | }
344 | to {
345 | transform: translateY(-10px);
346 | }
347 | }
348 |
--------------------------------------------------------------------------------
/ui/templates/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | @Janus
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
JANUS
20 |
Ask Me Anything
21 |
22 | 
23 |
24 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
--------------------------------------------------------------------------------
/working_dir/vocab20000.dec:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/suriyadeepan/easy_seq2seq/c05b0d02f15e9bd1947ff79ddcd0b761a214e2a9/working_dir/vocab20000.dec
--------------------------------------------------------------------------------
/working_dir/vocab20000.enc:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/suriyadeepan/easy_seq2seq/c05b0d02f15e9bd1947ff79ddcd0b761a214e2a9/working_dir/vocab20000.enc
--------------------------------------------------------------------------------