diff --git a/README.md b/README.md index 18606e495302ce918518f618d55d3a641d2128fb..fd86c9e7b8f90d44d1f487f5d154b41bd86ada58 100644 --- a/README.md +++ b/README.md @@ -29,7 +29,7 @@ duniter4j has four main components : sudo apt-get install openjdk-8-jre ``` - - Install [libsodium](https://download.libsodium.org/doc/index.html) (Linux only) + - Install [libsodium](https://download.libsodium.org/doc/index.html) v1.0.11 (Linux only) - Linux: See [installation](https://download.libsodium.org/doc/installation/index.html). After installation, make sure the file 'libsodium.so' exists on: /usr/local/lib or /opt/local/lib. If not, create a symbolic link. diff --git a/doc/fr/development_tutorial.md b/doc/fr/development_tutorial.md index 11f04346fa33deaac7d09e58383d007b3335005b..69fd5d6dc05ef8bfb5d3c9cf2ac0e8cd03343a88 100644 --- a/doc/fr/development_tutorial.md +++ b/doc/fr/development_tutorial.md @@ -16,7 +16,7 @@ Le projet Duniter4j est composé de plusieurs sous-modules : - `duniter4j-es-*`: Les plugins ElasticSearch, qui implémentent : * `duniter4j-es-core`: Indexation de BlockChain Duniter (ESA ou ES API); * `duniter4j-es-user`: Indexation de données utilisateurs (profils, des messages privées, paramètres chiffrés) (ESUA ou ES USER API); - * `duniter4j-es-gchange`: Indexation d'annonces, registre des profesionnels (GChange API ). Note : cette partie sera pas la suite sortie dans un autre projet. + * `duniter4j-es-subscription`: Indexation d'annonces, registre des profesionnels (GChange API ). Note : cette partie sera pas la suite sortie dans un autre projet. ## Niveau I : récupérer le code source @@ -394,10 +394,10 @@ Duniter4j permet aussi de stocker et d'indexer les données hors BlockChain, com - `/user/profile` : les profiles utilisateurs (nom complet, réseaux sociaux, avatar, etc.) - `/message/inbox` : les messages privées envoyés -- `/market/record` : les annonces de la [place de maché](http://cesium.duniter.fr/#/app/market/lg) Ğchange; - * `/market/record` : les commentaires sur les annonces -- `/registry/record` : les référencement de l'[annuaire pro](http://cesium.duniter.fr/#/app/registry/lg) Ğchange; - * `/market/comment` : les commentaires sur les référencements +- `/mail/record` : les annonces de la [place de maché](http://cesium.duniter.fr/#/app/market/lg) Ğchange; + * `/mail/record` : les commentaires sur les annonces +- `/mail/record` : les référencement de l'[annuaire pro](http://cesium.duniter.fr/#/app/registry/lg) Ğchange; + * `/mail/comment` : les commentaires sur les référencements > La document de l'API HTTP est disponible [ici](../API.md). diff --git a/duniter4j-client/src/main/resources/i18n/duniter4j-client_en_GB.properties b/duniter4j-client/src/main/resources/i18n/duniter4j-client_en_GB.properties index ff30b8ca5c5e5f35a83d8b020f2b09c4c1f0e94f..e161ce0fb1fdd770a4208f13de9e97195a58c051 100644 --- a/duniter4j-client/src/main/resources/i18n/duniter4j-client_en_GB.properties +++ b/duniter4j-client/src/main/resources/i18n/duniter4j-client_en_GB.properties @@ -10,10 +10,14 @@ duniter4j.client.network.noPeers=No peers found duniter4j.client.network.params.continue=Continue scanning? (Will refresh on new peer/block). duniter4j.client.network.params.output=Output CSV file duniter4j.client.network.ssl=SSL +duniter4j.client.params.authScrypt=Authenticate using salt (Scrypt) ? duniter4j.client.params.authScrypt.ask.passwd=Please enter your Scrypt password\: duniter4j.client.params.authScrypt.ask.salt=Please enter your Scrypt Salt (Secret identifier)\: duniter4j.client.params.authScrypt.ask.scryptParams=Please enter your Scrypt parameters (N,r,p)\: [%d,%d,%d] duniter4j.client.params.authScrypt.error.scryptParams=Invalid Scrypt parameters (expected 3 values)" +duniter4j.client.params.authScrypt.passwd=Password +duniter4j.client.params.authScrypt.salt=Secret identifier (Salt) +duniter4j.client.params.authScrypt.scryptParams=Scrypt parameters (N,r,p) duniter4j.client.params.config=Configuration file path duniter4j.client.params.debug=Show debug logs duniter4j.client.params.error.invalidOption=Invalid value of option [%s] @@ -34,7 +38,3 @@ duniter4j.client.transaction.loadingMemberPeers=Retrieving member's peers... duniter4j.client.transaction.params.amount.ask=Please enter the amount (integer value)\: duniter4j.client.transaction.params.output.ask=Please enter output (public key)\: duniter4j.client.transaction.sent=Transaction successfully sent. -duniter4j.client.params.authScrypt=Authenticate using salt (Scrypt) ? -duniter4j.client.params.authScrypt.salt=Secret identifier (Salt) -duniter4j.client.params.authScrypt.passwd=Password -duniter4j.client.params.authScrypt.scryptParams=Scrypt parameters (N,r,p) diff --git a/duniter4j-client/src/main/resources/i18n/duniter4j-client_fr_FR.properties b/duniter4j-client/src/main/resources/i18n/duniter4j-client_fr_FR.properties index 958f291e0c35c2b76a06785ed7c590cf805310a1..0e5a4cb730831dd2df1b544677d7b1a345dc45bb 100644 --- a/duniter4j-client/src/main/resources/i18n/duniter4j-client_fr_FR.properties +++ b/duniter4j-client/src/main/resources/i18n/duniter4j-client_fr_FR.properties @@ -10,10 +10,14 @@ duniter4j.client.network.noPeers=Aucun noeud trouvé duniter4j.client.network.params.continue=Continue scanning? (Will refresh on new peer/block). duniter4j.client.network.params.output=Output CSV file duniter4j.client.network.ssl=SSL +duniter4j.client.params.authScrypt=Authentification par salage Scrypt ? duniter4j.client.params.authScrypt.ask.passwd=Veuillez entrer votre mot de passe Scrypt (password) \: duniter4j.client.params.authScrypt.ask.salt=Veuillez entrer votre identifiant secret Scrypt (Salt)\: duniter4j.client.params.authScrypt.ask.scryptParams=Veuillez entrer les paramètres de Scrypt (N,r,p)\: [%d,%d,%d] duniter4j.client.params.authScrypt.error.scryptParams=Paramètre Scrypt non valide (3 valeurs attendues - format 'N,r,p')" +duniter4j.client.params.authScrypt.passwd=Mot de passe +duniter4j.client.params.authScrypt.salt=Identifiant secret (salt) +duniter4j.client.params.authScrypt.scryptParams=Paramètre de salage Scrypt (N,r,p) duniter4j.client.params.config=Fichier de configuration duniter4j.client.params.debug=Activer les logs de débuggage duniter4j.client.params.error.invalidOption=Valeur d'option [%s] invalid @@ -34,7 +38,3 @@ duniter4j.client.transaction.loadingMemberPeers=Récupération des noeuds membre duniter4j.client.transaction.params.amount.ask=Veuillez entrer le montant (valeur entière) \: duniter4j.client.transaction.params.output.ask=Veuillez entrer le destinataire (clef publique) \: duniter4j.client.transaction.sent=Transaction envoyé avec succès. -duniter4j.client.params.authScrypt=Authentification par salage Scrypt ? -duniter4j.client.params.authScrypt.salt=Identifiant secret (salt) -duniter4j.client.params.authScrypt.passwd=Mot de passe -duniter4j.client.params.authScrypt.scryptParams=Paramètre de salage Scrypt (N,r,p) diff --git a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/TxSource.java b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/TxSource.java index 4c0a05e072f76d26619c495338979818a868e256..36e4901a5c9d62cfa718405c7d18885c2e2b1ab5 100644 --- a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/TxSource.java +++ b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/TxSource.java @@ -89,7 +89,7 @@ public class TxSource { } /** - * Source sortType : <ul> + * Source type : <ul> * <li><code>D</code> : Universal Dividend</li> * <li><code>T</code> : Transaction</li> * </ul> diff --git a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/jackson/JacksonUtils.java b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/jackson/JacksonUtils.java index 11aed3fdc065c8878bd9023129578c1493236028..9f754aef2c877cdb3a0aa8c4dcc336051ee1b316 100644 --- a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/jackson/JacksonUtils.java +++ b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/bma/jackson/JacksonUtils.java @@ -22,7 +22,9 @@ package org.duniter.core.client.model.bma.jackson; * #L% */ +import com.fasterxml.jackson.core.util.MinimalPrettyPrinter; import com.fasterxml.jackson.databind.ObjectMapper; +import com.fasterxml.jackson.databind.SerializationConfig; import com.fasterxml.jackson.databind.module.SimpleModule; import org.duniter.core.client.model.bma.BlockchainBlock; import org.duniter.core.client.model.bma.NetworkPeering; @@ -36,7 +38,7 @@ import java.util.List; */ public abstract class JacksonUtils extends SimpleModule { - public static final String REGEX_ATTRIBUTE_REPLACE = "[,]?[\"\\s\\n\\r]*%s[\"]?[\\s\\n\\r]*:[\\s\\n\\r]*\"[^\"]+\""; + public static final String REGEX_ATTRIBUTE_REPLACE = "[,]?(?:\"%s\"|%s)[\\s\\n\\r]*:[\\s\\n\\r]*(?:\"[^\"]+\"|null)"; public static ObjectMapper newObjectMapper() { @@ -80,7 +82,8 @@ public abstract class JacksonUtils extends SimpleModule { } public static String removeAttribute(String jsonString, String attributeName) { - return jsonString.replaceAll(String.format(REGEX_ATTRIBUTE_REPLACE, attributeName), ""); + String regex = String.format(REGEX_ATTRIBUTE_REPLACE, attributeName, attributeName); + return jsonString.replaceAll(regex, ""); } } diff --git a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/elasticsearch/DeleteRecord.java b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/elasticsearch/DeleteRecord.java index d87291d7e20b990145f5dc11539bce7ecea63dbd..c09fe0d125212863155ea662b67b071fa39676f1 100644 --- a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/elasticsearch/DeleteRecord.java +++ b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/elasticsearch/DeleteRecord.java @@ -28,7 +28,7 @@ package org.duniter.core.client.model.elasticsearch; public class DeleteRecord extends Record { public static final String PROPERTY_INDEX="index"; - public static final String PROPERTY_TYPE="sortType"; + public static final String PROPERTY_TYPE="type"; public static final String PROPERTY_ID="id"; private String index; diff --git a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/elasticsearch/Record.java b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/elasticsearch/Record.java index 65ad743ccdbb7c891e72934d1d83bb6b3b8cb8cf..38a94d02bc2b01742a1c2e8f4e6bbcce5ec48d9e 100644 --- a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/elasticsearch/Record.java +++ b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/elasticsearch/Record.java @@ -22,16 +22,20 @@ package org.duniter.core.client.model.elasticsearch; * #L% */ +import com.fasterxml.jackson.annotation.JsonIgnore; +import org.duniter.core.client.model.local.LocalEntity; + /** * Created by blavenie on 01/03/16. */ -public class Record { +public class Record implements LocalEntity<String> { public static final String PROPERTY_ISSUER="issuer"; public static final String PROPERTY_HASH="hash"; public static final String PROPERTY_SIGNATURE="signature"; public static final String PROPERTY_TIME="time"; + private String id; private String issuer; private String hash; private String signature; @@ -41,12 +45,23 @@ public class Record { } public Record(Record another) { + this.id = another.getId(); this.issuer = another.getIssuer(); this.hash = another.getHash(); this.signature = another.getSignature(); this.time = another.getTime(); } + @JsonIgnore + public String getId() { + return id; + } + + @JsonIgnore + public void setId(String id) { + this.id = id; + } + public String getIssuer() { return issuer; } diff --git a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/local/Wallet.java b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/local/Wallet.java index 675f5d9bbec4cf3c8e71d78754adc2da87a3cd8e..c22dbf1251a4ff881ff72f0b188c73e94912ef59 100644 --- a/duniter4j-core-client/src/main/java/org/duniter/core/client/model/local/Wallet.java +++ b/duniter4j-core-client/src/main/java/org/duniter/core/client/model/local/Wallet.java @@ -151,7 +151,7 @@ public class Wallet extends KeyPair implements LocalEntity<Long>, Serializable { } public String toString() { - return name; + return name != null ? name : identity.getPubkey(); } public String getUid() { diff --git a/duniter4j-core-client/src/main/java/org/duniter/core/client/service/local/NetworkServiceImpl.java b/duniter4j-core-client/src/main/java/org/duniter/core/client/service/local/NetworkServiceImpl.java index 70d4dbefcc577caa34a6150d00de895c2a943274..52e2fb97be703e86df53d328cb65fb4d98d92ea2 100644 --- a/duniter4j-core-client/src/main/java/org/duniter/core/client/service/local/NetworkServiceImpl.java +++ b/duniter4j-core-client/src/main/java/org/duniter/core/client/service/local/NetworkServiceImpl.java @@ -174,6 +174,9 @@ public class NetworkServiceImpl extends BaseRemoteServiceImpl implements Network return peersFuture.join().stream() .map(peer -> { if (mainPeer.getUrl().equals(peer.getUrl())) { + mainPeer.setPubkey(peer.getPubkey()); + mainPeer.setHash(peer.getHash()); + mainPeer.setCurrency(peer.getCurrency()); return asyncRefreshPeer(mainPeer, memberUids, pool); } return asyncRefreshPeer(peer, memberUids, pool); diff --git a/duniter4j-core-shared/src/main/java/org/duniter/core/model/SmtpConfig.java b/duniter4j-core-shared/src/main/java/org/duniter/core/model/SmtpConfig.java new file mode 100644 index 0000000000000000000000000000000000000000..67ebaa21c4d02f143d3be87beb0124361596dbb9 --- /dev/null +++ b/duniter4j-core-shared/src/main/java/org/duniter/core/model/SmtpConfig.java @@ -0,0 +1,69 @@ +package org.duniter.core.model; + +public class SmtpConfig { + + + private String smtpHost; + private int smtpPort; + private String smtpUsername; + private String smtpPassword; + private String senderAddress; + private boolean useSsl; + private boolean startTLS; + + public String getSmtpHost() { + return smtpHost; + } + + public void setSmtpHost(String smtpHost) { + this.smtpHost = smtpHost; + } + + public int getSmtpPort() { + return smtpPort; + } + + public void setSmtpPort(int smtpPort) { + this.smtpPort = smtpPort; + } + + public String getSmtpUsername() { + return smtpUsername; + } + + public void setSmtpUsername(String smtpUsername) { + this.smtpUsername = smtpUsername; + } + + public String getSmtpPassword() { + return smtpPassword; + } + + public void setSmtpPassword(String smtpPassword) { + this.smtpPassword = smtpPassword; + } + + public String getSenderAddress() { + return senderAddress; + } + + public void setSenderAddress(String senderAddress) { + this.senderAddress = senderAddress; + } + + public boolean isUseSsl() { + return useSsl; + } + + public void setUseSsl(boolean useSsl) { + this.useSsl = useSsl; + } + + public boolean isStartTLS() { + return startTLS; + } + + public void setStartTLS(boolean startTLS) { + this.startTLS = startTLS; + } +} \ No newline at end of file diff --git a/duniter4j-core-shared/src/main/java/org/duniter/core/service/CryptoService.java b/duniter4j-core-shared/src/main/java/org/duniter/core/service/CryptoService.java index 4702a46c4d8268c45a240eaa6fa0fba108ae5838..7c3a6458fd523a21707a6a1a7a9863b82e3b70af 100644 --- a/duniter4j-core-shared/src/main/java/org/duniter/core/service/CryptoService.java +++ b/duniter4j-core-shared/src/main/java/org/duniter/core/service/CryptoService.java @@ -68,6 +68,16 @@ public interface CryptoService extends Bean { String sign(String message, String secretKey); + String box(String message, byte[] nonce, String senderSignSk, String receiverSignPk); + + String box(String message, byte[] nonce, byte[] senderSignSk, byte[] receiverSignPk); + + byte[] getBoxRandomNonce(); + + String openBox(String cypherText, String nonce, String senderSignPk, String receiverSignSk); + + String openBox(String cypherText, byte[] nonce, byte[] senderSignPk, byte[] receiverSignSk); + boolean verify(String message, String signature, String publicKey); /** diff --git a/duniter4j-core-shared/src/main/java/org/duniter/core/service/Ed25519CryptoServiceImpl.java b/duniter4j-core-shared/src/main/java/org/duniter/core/service/Ed25519CryptoServiceImpl.java index 6448f79aad940ff0c4906b5083fbad1fd66c3803..59bc0264af49496b0a7ed2487a73c64911a21205 100644 --- a/duniter4j-core-shared/src/main/java/org/duniter/core/service/Ed25519CryptoServiceImpl.java +++ b/duniter4j-core-shared/src/main/java/org/duniter/core/service/Ed25519CryptoServiceImpl.java @@ -32,8 +32,15 @@ import org.abstractj.kalium.NaCl; import org.abstractj.kalium.NaCl.Sodium; import org.abstractj.kalium.crypto.Util; +import java.nio.charset.Charset; import java.security.GeneralSecurityException; +import static org.abstractj.kalium.NaCl.Sodium.CRYPTO_BOX_CURVE25519XSALSA20POLY1305_BOXZEROBYTES; +import static org.abstractj.kalium.NaCl.Sodium.CRYPTO_BOX_CURVE25519XSALSA20POLY1305_NONCEBYTES; +import static org.abstractj.kalium.NaCl.Sodium.CRYPTO_BOX_CURVE25519XSALSA20POLY1305_ZEROBYTES; +import static org.abstractj.kalium.NaCl.sodium; +import static org.abstractj.kalium.crypto.Util.*; + /** * Crypto services (sign...) @@ -132,12 +139,71 @@ public class Ed25519CryptoServiceImpl implements CryptoService { @Override public String hash(String message) { - byte[] hash = new byte[Sodium.SHA256BYTES]; + byte[] hash = new byte[Sodium.CRYPTO_HASH_SHA256_BYTES]; byte[] messageBinary = CryptoUtils.decodeUTF8(message); naCl.crypto_hash_sha256(hash, messageBinary, messageBinary.length); return bytesToHex(hash).toUpperCase(); } + @Override + public byte[] getBoxRandomNonce() { + byte[] nonce = new byte[Sodium.CRYPTO_BOX_CURVE25519XSALSA20POLY1305_NONCEBYTES]; + naCl.randombytes(nonce, nonce.length); + + return nonce; + } + + @Override + public String box(String message, byte[] nonce, String senderSignSk, String receiverSignPk) { + byte[] senderSignSkBinary = CryptoUtils.decodeBase58(senderSignSk); + byte[] receiverSignPkBinary = CryptoUtils.decodeBase58(receiverSignPk); + return box(message, nonce, senderSignSkBinary, receiverSignPkBinary); + } + + @Override + public String box(String message, byte[] nonce, byte[] senderSignSk, byte[] receiverSignPk) { + checkLength(nonce, CRYPTO_BOX_CURVE25519XSALSA20POLY1305_NONCEBYTES); + + byte[] messageBinary = prependZeros(CRYPTO_BOX_CURVE25519XSALSA20POLY1305_ZEROBYTES, CryptoUtils.decodeBase64(message)); + + byte[] senderBoxSk = new byte[Sodium.CRYPTO_BOX_CURVE25519XSALSA20POLY1305_SECRETKEYBYTES]; + naCl.crypto_sign_ed25519_sk_to_curve25519(senderBoxSk, senderSignSk); + + byte[] receiverBoxPk = new byte[Sodium.CRYPTO_BOX_CURVE25519XSALSA20POLY1305_PUBLICKEYBYTES]; + naCl.crypto_sign_ed25519_pk_to_curve25519(receiverBoxPk, receiverSignPk); + + byte[] cypherTextBinary = new byte[messageBinary.length]; + isValid(sodium().crypto_box_curve25519xsalsa20poly1305(cypherTextBinary, messageBinary, + cypherTextBinary.length, nonce, senderBoxSk, receiverBoxPk), "Encryption failed"); + return CryptoUtils.encodeBase64(removeZeros(CRYPTO_BOX_CURVE25519XSALSA20POLY1305_BOXZEROBYTES, cypherTextBinary)); + } + + @Override + public String openBox(String cypherText, String nonce, String senderSignPk, String receiverSignSk) { + return openBox(cypherText, + CryptoUtils.decodeBase58(nonce), + CryptoUtils.decodeBase58(senderSignPk), + CryptoUtils.decodeBase58(receiverSignSk)); + } + + @Override + public String openBox(String cypherText, byte[] nonce, byte[] senderSignPk, byte[] receiverSignSk) { + checkLength(nonce, CRYPTO_BOX_CURVE25519XSALSA20POLY1305_NONCEBYTES); + byte[] cypherTextBinary = prependZeros(CRYPTO_BOX_CURVE25519XSALSA20POLY1305_BOXZEROBYTES, CryptoUtils.decodeBase64(cypherText)); + + byte[] receiverBoxSk = new byte[Sodium.CRYPTO_BOX_CURVE25519XSALSA20POLY1305_SECRETKEYBYTES]; + naCl.crypto_sign_ed25519_sk_to_curve25519(receiverBoxSk, receiverSignSk); + + byte[] senderBoxPk = new byte[Sodium.CRYPTO_BOX_CURVE25519XSALSA20POLY1305_PUBLICKEYBYTES]; + naCl.crypto_sign_ed25519_pk_to_curve25519(senderBoxPk, senderSignPk); + + byte[] messageBinary = new byte[cypherTextBinary.length]; + isValid(sodium().crypto_box_curve25519xsalsa20poly1305_open( + messageBinary, cypherTextBinary, cypherTextBinary.length, nonce, senderBoxPk, receiverBoxSk), + "Decryption failed. Ciphertext failed verification."); + return CryptoUtils.encodeUTF8(removeZeros(CRYPTO_BOX_CURVE25519XSALSA20POLY1305_ZEROBYTES, messageBinary)); + } + /* -- Internal methods -- */ protected byte[] sign(byte[] message, byte[] secretKey) { diff --git a/duniter4j-core-shared/src/main/java/org/duniter/core/service/MailService.java b/duniter4j-core-shared/src/main/java/org/duniter/core/service/MailService.java index 64975b25ce5240bdece733be974e4a995e0f3977..1a11960e61b1852241675983bdef0cf7cd88b555 100644 --- a/duniter4j-core-shared/src/main/java/org/duniter/core/service/MailService.java +++ b/duniter4j-core-shared/src/main/java/org/duniter/core/service/MailService.java @@ -24,6 +24,7 @@ package org.duniter.core.service; import org.duniter.core.beans.Bean; import org.duniter.core.exception.TechnicalException; +import org.duniter.core.model.SmtpConfig; import javax.mail.internet.ContentType; import javax.mail.internet.ParseException; @@ -33,22 +34,21 @@ import javax.mail.internet.ParseException; */ public interface MailService extends Bean { - void sendTextEmail(String smtpHost, - int smtpPort, - String smtpUsername, - String smtpPassword, - String issuer, - String recipients, - String subject, - String textContent); - - void sendEmail(String smtpHost, - int smtpPort, - String smtpUsername, - String smtpPassword, - String issuer, - String recipients, + void setSmtpConfig(SmtpConfig config); + + void sendTextEmail(String subject, + String textContent, + String... recipients); + + void sendHtmlEmail( String subject, - ContentType contentType, - String content); + String utf8HtmlContent, + String... recipients); + + void sendHtmlEmailWithText( + String subject, + String utf8textContent, + String utf8HtmlContent, + String... recipients); + } diff --git a/duniter4j-core-shared/src/main/java/org/duniter/core/service/MailServiceImpl.java b/duniter4j-core-shared/src/main/java/org/duniter/core/service/MailServiceImpl.java index ec7bb898b1ffd51c2e62049835e3fe7ad6257931..6a19564a43c44a5a158f7be7050741d891d3da22 100644 --- a/duniter4j-core-shared/src/main/java/org/duniter/core/service/MailServiceImpl.java +++ b/duniter4j-core-shared/src/main/java/org/duniter/core/service/MailServiceImpl.java @@ -22,19 +22,24 @@ package org.duniter.core.service; * #L% */ +import com.google.common.base.Joiner; import org.duniter.core.exception.TechnicalException; +import org.duniter.core.model.SmtpConfig; +import org.duniter.core.util.CollectionUtils; import org.duniter.core.util.StringUtils; +import javax.activation.CommandMap; +import javax.activation.MailcapCommandMap; import javax.mail.*; -import javax.mail.internet.ContentType; -import javax.mail.internet.InternetAddress; -import javax.mail.internet.MimeMessage; -import javax.mail.internet.ParseException; +import javax.mail.internet.*; import java.io.Closeable; +import java.util.Arrays; import java.util.Properties; +import java.util.stream.Collectors; public class MailServiceImpl implements MailService, Closeable { + private SmtpConfig smtpConfig; private static Session session; private static Transport transport; @@ -43,82 +48,85 @@ public class MailServiceImpl implements MailService, Closeable { } @Override - public void sendTextEmail(String smtpHost, - int smtpPort, - String smtpUsername, - String smtpPassword, - String issuer, - String recipients, - String subject, - String textContent) { + public void setSmtpConfig(SmtpConfig smtpConfig) { + this.smtpConfig = smtpConfig; + } + + @Override + public void sendTextEmail(String subject, + String textContent, + String... recipients) { try{ ContentType contentType = new ContentType("text/plain"); contentType.setParameter("charset", "UTF-8"); - - sendEmail(smtpHost, smtpPort, smtpUsername, smtpPassword, - issuer, - recipients, - subject, - contentType, - textContent); + sendEmail(subject, contentType.toString(), textContent, recipients); } catch(ParseException e) { // Should never occur throw new TechnicalException(e); } - } @Override - public void sendEmail(String smtpHost, - int smtpPort, - String smtpUsername, - String smtpPassword, - String issuer, - String recipients, - String subject, - ContentType contentType, - String content) { + public void sendHtmlEmail(String subject, + String htmlContent, + String... recipients) { + try{ + ContentType contentType = new ContentType("text/html"); + contentType.setParameter("charset", "UTF-8"); - // check arguments - if (StringUtils.isBlank(smtpHost) || smtpPort <= 0) { - throw new TechnicalException("Invalid arguments: 'smtpHost' could not be null or empty, and 'smtpPort' could not be <= 0"); - } - if (StringUtils.isBlank(issuer)) { - throw new TechnicalException("Invalid arguments: 'issuer' could not be null or empty"); + Multipart content = new MimeMultipart(); + MimeBodyPart mbp = new MimeBodyPart(); + mbp.setContent(htmlContent, contentType.toString()); + content.addBodyPart(mbp); + + sendEmail(subject, content.getContentType(), content, recipients); } - if (StringUtils.isBlank(recipients) || StringUtils.isBlank(subject) || StringUtils.isBlank(content) || contentType == null) { - throw new TechnicalException("Invalid arguments: 'recipients', 'subject', 'contentType' or 'content' could not be null or empty"); + catch(MessagingException e) { + // Should never occur + throw new TechnicalException(e); } - if (!isConnected()) { - connect(smtpHost, smtpPort, smtpUsername, smtpPassword, issuer); - } + } - // send email to recipients - try { - Message message = new MimeMessage(session); - message.setFrom(new InternetAddress(issuer)); - message.setRecipients(Message.RecipientType.TO, InternetAddress.parse(recipients)); - message.setSubject(subject); + @Override + public void sendHtmlEmailWithText(String subject, + String textContent, + String htmlContent, + String... recipients) { + try{ - message.setContent(content, contentType.toString()); - message.saveChanges(); - transport.sendMessage(message, message.getAllRecipients()); + Multipart content = new MimeMultipart("alternative"); - } catch (MessagingException e) { - throw new TechnicalException(String.format("Error while sending email to [%s] using smtp server [%s]", - recipients, - getSmtpServerAsString(smtpHost, smtpPort, smtpUsername) - ), e); - } catch (IllegalStateException e) { - throw new TechnicalException(String.format("Error while sending email to [%s] using smtp server [%s]", - recipients, - getSmtpServerAsString(smtpHost, smtpPort, smtpUsername) - ), e); + // Add text part + { + MimeBodyPart mbp = new MimeBodyPart(); + ContentType contentType = new ContentType("text/plain"); + contentType.setParameter("charset", "UTF-8"); + mbp.setContent(textContent, contentType.toString()); + content.addBodyPart(mbp); + } + + // Add html part + { + MimeBodyPart mbp = new MimeBodyPart(); + ContentType contentType = new ContentType("text/html"); + contentType.setParameter("charset", "UTF-8"); + mbp.setContent(htmlContent, contentType.toString()); + content.addBodyPart(mbp); + } + + sendEmail(subject, content.getContentType(), content, recipients); + } + catch(MessagingException e) { + // Should never occur + throw new TechnicalException(e); } + } + + public void close() { if (isConnected()) { try { @@ -134,6 +142,54 @@ public class MailServiceImpl implements MailService, Closeable { /* -- private methods -- */ + public void sendEmail(String subject, + String contentType, + Object content, + String... recipients) { + + if (CollectionUtils.isEmpty(recipients) || StringUtils.isBlank(subject) || content == null || contentType == null) { + throw new TechnicalException("Invalid arguments: 'recipients', 'subject', 'contentType' or 'content' could not be null or empty"); + } + + if (!isConnected()) { + connect(smtpConfig); + } + + // send email to recipients + try { + Message message = new MimeMessage(session); + message.setFrom(new InternetAddress(smtpConfig.getSenderAddress())); + + Address[] recipientsAddresses = Arrays.asList(recipients).stream().map(recipient -> { + try { + return new InternetAddress(recipient); + } + catch (AddressException e) { + throw new TechnicalException(String.format("Error while sending email. Bad recipient address [%s]", recipient), e); + } + }).collect(Collectors.toList()).toArray(new InternetAddress[recipients.length]); + + message.setRecipients(Message.RecipientType.TO, recipientsAddresses); + message.setSubject(subject); + + message.setContent(content, contentType); + message.setSentDate(new java.util.Date()); + message.saveChanges(); + transport.sendMessage(message, message.getAllRecipients()); + + } catch (MessagingException e) { + throw new TechnicalException(String.format("Error while sending email to [%s] using smtp server [%s]: %s", + Joiner.on(',').join(recipients), getSmtpServerAsString(), + e.getMessage() + ), e); + } + } + + private String getSmtpServerAsString() { + if (smtpConfig == null) return ""; + return getSmtpServerAsString(smtpConfig.getSmtpHost(), smtpConfig.getSmtpPort(), smtpConfig.getSmtpUsername()); + } + private String getSmtpServerAsString(String smtpHost, int smtpPort, String smtpUsername) { StringBuilder buffer = new StringBuilder(); if (StringUtils.isNotBlank(smtpUsername)) { @@ -149,19 +205,64 @@ public class MailServiceImpl implements MailService, Closeable { private void connect(String smtpHost, int smtpPort, String smtpUsername, String smtpPassword, - String issuer) { + String issuer, + boolean useSsl, + boolean starttls) { + // check arguments + if (StringUtils.isBlank(smtpHost) || smtpPort <= 0) { + throw new TechnicalException("Invalid arguments: 'smtpHost' could not be null or empty, and 'smtpPort' could not be <= 0"); + } + if (StringUtils.isBlank(issuer)) { + throw new TechnicalException("Invalid arguments: 'issuer' could not be null or empty"); + } + + this.smtpConfig = new SmtpConfig(); + smtpConfig.setSmtpHost(smtpHost); + smtpConfig.setSmtpPort(smtpPort); + smtpConfig.setSmtpUsername(smtpUsername); + smtpConfig.setSmtpPassword(smtpPassword); + smtpConfig.setSenderAddress(issuer); + smtpConfig.setUseSsl(useSsl); + smtpConfig.setStartTLS(starttls); + connect(this.smtpConfig); + } + + private void connect(SmtpConfig config) { + + // Workaround, to avoid error on content type + // http://stackoverflow.com/questions/21856211/javax-activation-unsupporteddatatypeexception-no-object-dch-for-mime-type-multi + Thread.currentThread().setContextClassLoader( getClass().getClassLoader() ); + + configureJavaMailMimeTypes(); + Properties props = new Properties(); - props.put("mail.smtp.host", smtpHost); - props.put("mail.smtp.port", smtpPort); - if (StringUtils.isNotBlank(issuer)) { - props.put("mail.from", issuer); + // check arguments + if (StringUtils.isBlank(config.getSmtpHost()) || config.getSmtpPort() <= 0) { + throw new TechnicalException("Invalid arguments: 'smtpHost' could not be null or empty, and 'smtpPort' could not be <= 0"); + } + if (StringUtils.isBlank(config.getSenderAddress())) { + throw new TechnicalException("Invalid arguments: 'senderAddress' could not be null or empty"); + } + + props.put("mail.smtp.host", config.getSmtpHost()); + props.put("mail.smtp.port", config.getSmtpPort()); + if (StringUtils.isNotBlank(config.getSenderAddress())) { + props.put("mail.from", config.getSenderAddress()); + } + if (config.isUseSsl()) { + props.put("mail.smtp.socketFactory.port", config.getSmtpPort()); + props.put("mail.smtp.socketFactory.class", "javax.net.ssl.SSLSocketFactory"); + props.put("mail.smtp.socketFactory.fallback", "false"); + } + if (config.isStartTLS()) { + props.put("mail.smtp.starttls.enable", "true"); } + boolean useAuth = false; // auto set authentification if smtp user name is provided - if (StringUtils.isNotBlank(smtpUsername)) { - props.put("mail.smtp.auth", "true"); - //props.put("mail.smtp.starttls.enable", "true"); + if (StringUtils.isNotBlank(config.getSmtpUsername())) { + props.put("mail.smtp.auth", true); useAuth = true; } @@ -170,7 +271,7 @@ public class MailServiceImpl implements MailService, Closeable { try { transport = session.getTransport("smtp"); if (useAuth) { - transport.connect(smtpUsername, smtpPassword); + transport.connect(config.getSmtpUsername(), config.getSmtpPassword()); } else { transport.connect(); } @@ -188,4 +289,18 @@ public class MailServiceImpl implements MailService, Closeable { return transport.isConnected(); } + + /** + * Workaround to define javax.mail MIME types to classes WITHOUT using classpath file. + * See http://stackoverflow.com/questions/21856211/javax-activation-unsupporteddatatypeexception-no-object-dch-for-mime-type-multi + */ + protected void configureJavaMailMimeTypes() { + + MailcapCommandMap mc = (MailcapCommandMap) CommandMap.getDefaultCommandMap(); + mc.addMailcap("text/html;; x-java-content-handler=com.sun.mail.handlers.text_html"); + mc.addMailcap("text/xml;; x-java-content-handler=com.sun.mail.handlers.text_xml"); + mc.addMailcap("text/plain;; x-java-content-handler=com.sun.mail.handlers.text_plain"); + mc.addMailcap("multipart/*;; x-java-content-handler=com.sun.mail.handlers.multipart_mixed"); + mc.addMailcap("message/rfc822;; x-java-content- handler=com.sun.mail.handlers.message_rfc822"); + } } \ No newline at end of file diff --git a/duniter4j-core-shared/src/main/java/org/duniter/core/util/crypto/CryptoUtils.java b/duniter4j-core-shared/src/main/java/org/duniter/core/util/crypto/CryptoUtils.java index 3451ff62fa3bd4e5511777fe11a6ad559cbd67ba..ae062df9a4a4c3913d3f3b7fa33cee0a03e97c1a 100644 --- a/duniter4j-core-shared/src/main/java/org/duniter/core/util/crypto/CryptoUtils.java +++ b/duniter4j-core-shared/src/main/java/org/duniter/core/util/crypto/CryptoUtils.java @@ -60,6 +60,10 @@ public class CryptoUtils extends Util { public static byte[] decodeUTF8(String string) { return string.getBytes(CHARSET_UTF8); } + + public static String encodeUTF8(byte[] bytes) { + return new String(bytes, CHARSET_UTF8); + } public static byte[] decodeAscii(String string) { return string.getBytes(CHARSET_ASCII); diff --git a/duniter4j-core-shared/src/main/java/org/duniter/core/util/protocols/classpath/Handler.java b/duniter4j-core-shared/src/main/java/org/duniter/core/util/protocols/classpath/Handler.java new file mode 100644 index 0000000000000000000000000000000000000000..a0c08c289cea6c93cf11eca91d3699cb1967bb12 --- /dev/null +++ b/duniter4j-core-shared/src/main/java/org/duniter/core/util/protocols/classpath/Handler.java @@ -0,0 +1,32 @@ +package org.duniter.core.util.protocols.classpath; + +import org.duniter.core.exception.TechnicalException; + +import java.io.FileNotFoundException; +import java.io.IOException; +import java.net.URL; +import java.net.URLConnection; +import java.net.URLStreamHandler; + +/** A {@link URLStreamHandler} that handles resources on the classpath. */ +public class Handler extends URLStreamHandler { + /** The classloader to find resources from. */ + private final ClassLoader classLoader; + + public Handler() { + this.classLoader = getClass().getClassLoader(); + } + + public Handler(ClassLoader classLoader) { + this.classLoader = classLoader; + } + + @Override + protected URLConnection openConnection(final URL u) throws IOException { + final URL resourceUrl = classLoader.getResource(u.getPath()); + if (resourceUrl == null) { + throw new FileNotFoundException("Unable to load classpath resources: " + u.getPath()); + } + return resourceUrl.openConnection(); + } +} \ No newline at end of file diff --git a/duniter4j-core-shared/src/main/java/org/duniter/core/util/url/URLs.java b/duniter4j-core-shared/src/main/java/org/duniter/core/util/url/URLs.java new file mode 100644 index 0000000000000000000000000000000000000000..1d4f99b7ae3e1c0e490244cb909ea54ea33ca844 --- /dev/null +++ b/duniter4j-core-shared/src/main/java/org/duniter/core/util/url/URLs.java @@ -0,0 +1,44 @@ +package org.duniter.core.util.url; + +import org.duniter.core.exception.TechnicalException; + +import java.net.MalformedURLException; +import java.net.URL; + +/** + * Created by blavenie on 08/04/17. + */ +public final class URLs { + + static { + init(); + } + + private void URLs() { + // helper class + } + + public static URL newClasspathURL(String classpathResource, ClassLoader cl) throws MalformedURLException { + final URL resourceUrl = cl.getResource(classpathResource); + return resourceUrl; + } + + protected static void init() { + // Extend default JRE protocols (add classpath://) + // System.setProperty("java.protocol.handler.pkgs", "org.duniter.core.util.protocols"); + + } + + public static URL getClasspathResourceURL(String aClasspathFile, ClassLoader cl) throws MalformedURLException { + final URL resourceUrl = cl.getResource(aClasspathFile); + if (resourceUrl == null) { + throw new TechnicalException("File not found : " + aClasspathFile); + } + return resourceUrl; + } + + public static URL getParentURL(URL resourceUrl) throws MalformedURLException { + String filePath = resourceUrl.getPath(); + return new URL(filePath.substring(0, filePath.lastIndexOf('/'))); + } +} diff --git a/duniter4j-core-shared/src/test/java/org/duniter/core/service/Ed25519CryptoServiceTest.java b/duniter4j-core-shared/src/test/java/org/duniter/core/service/Ed25519CryptoServiceTest.java index 9473e6fd87b7abccc831c98fc04d9d1beecc2f4f..1595285cfb25a270f5a67e0e74f580f87be74824 100644 --- a/duniter4j-core-shared/src/test/java/org/duniter/core/service/Ed25519CryptoServiceTest.java +++ b/duniter4j-core-shared/src/test/java/org/duniter/core/service/Ed25519CryptoServiceTest.java @@ -23,26 +23,39 @@ package org.duniter.core.service; */ +import com.google.common.primitives.UnsignedBytes; +import org.abstractj.kalium.NaCl; +import org.abstractj.kalium.crypto.Box; +import org.abstractj.kalium.crypto.Util; import org.duniter.core.test.TestFixtures; import org.duniter.core.util.crypto.Base58; +import org.duniter.core.util.crypto.CryptoUtils; import org.duniter.core.util.crypto.SecretBox; import org.junit.Assert; import org.junit.Before; import org.junit.Test; import java.io.UnsupportedEncodingException; +import java.nio.charset.Charset; + +import static org.abstractj.kalium.NaCl.Sodium.CRYPTO_BOX_CURVE25519XSALSA20POLY1305_ZEROBYTES; +import static org.abstractj.kalium.NaCl.Sodium.CRYPTO_SECRETBOX_XSALSA20POLY1305_NONCEBYTES; +import static org.abstractj.kalium.crypto.Util.checkLength; +import static org.abstractj.kalium.crypto.Util.isValid; public class Ed25519CryptoServiceTest { private String message; private byte[] messageAsBytes; private CryptoService service; + private TestFixtures fixtures; @Before public void setUp() throws UnsupportedEncodingException { message = "my message to encrypt !"; messageAsBytes = message.getBytes("UTF-8"); service = new Ed25519CryptoServiceImpl(); + fixtures = new TestFixtures(); } @Test @@ -75,12 +88,51 @@ public class Ed25519CryptoServiceTest { Assert.assertEquals("AC31F1E3EAEB7A535A3BF1182AA53100BB3B610C5D63643ACB145EB99764B0CA", hash); } + @Test + public void packThenOpenBox() throws Exception { + + // + String originalMessage = "test@test"; + String nonce = "AHHfny8igAJp1h7P5d8bEobKZfgoRcXs9"; + + // Sender + SecretBox receiver = createSecretBox(); + SecretBox sender = createSecretBox(); + + // Create box + String cypherText = service.box(originalMessage, + CryptoUtils.decodeBase58(nonce), + receiver.getSecretKey(), sender.getPublicKey()); + + // Open box + String decryptedText = service.openBox( + cypherText, + nonce, + sender.getPublicKey(), receiver.getSecretKey()); + + Assert.assertEquals(originalMessage, decryptedText); + + } + + @Test + public void openBox() throws Exception { + // Receiver & Receiver + SecretBox receiver = createSecretBox(); + SecretBox sender = createSecretBox(); + + // Open box + String decryptedText = service.openBox( + "RZwBeUpjGH2f4GUP0UW32WC82iiWfcrKbw==", + "AHHfny8igAJp1h7P5d8bEobKZfgoRcXs9", + sender.getPublicKey(), receiver.getSecretKey()); + + Assert.assertEquals("test@test", decryptedText); + } /* -- internal methods */ protected SecretBox createSecretBox() { - TestFixtures fixtures = new TestFixtures(); String salt = fixtures.getUserSalt(); String password = fixtures.getUserPassword(); SecretBox secretBox = new SecretBox(salt, password); diff --git a/duniter4j-es-assembly/pom.xml b/duniter4j-es-assembly/pom.xml index e89d4dd9270c82e4154463cd1f35bf96af369e4a..07c3f36da2d8733bd4227e6828c3c114fc9e0004 100644 --- a/duniter4j-es-assembly/pom.xml +++ b/duniter4j-es-assembly/pom.xml @@ -163,9 +163,9 @@ </configuration> </execution> - <!-- unpack ES gchange plugin --> + <!-- unpack ES subscription plugin --> <execution> - <id>unpack-es-gchange-plugin</id> + <id>unpack-es-subscription-plugin</id> <goals> <goal>unpack</goal> </goals> @@ -174,12 +174,12 @@ <artifactItems> <artifactItem> <groupId>org.duniter</groupId> - <artifactId>duniter4j-es-gchange</artifactId> + <artifactId>duniter4j-es-subscription</artifactId> <version>${project.version}</version> <type>zip</type> </artifactItem> </artifactItems> - <outputDirectory>${project.build.directory}/elasticsearch-${elasticsearch.version}/plugins/duniter4j-es-gchange</outputDirectory> + <outputDirectory>${project.build.directory}/elasticsearch-${elasticsearch.version}/plugins/duniter4j-es-subscription</outputDirectory> <silent>true</silent> <skip>${assembly.skip}</skip> </configuration> @@ -270,8 +270,8 @@ <fileset dir="../duniter4j-es-user/target" includes="duniter4j-*${project.version}.jar"> </fileset> </copy> - <copy todir="${run.es.home}/plugins/duniter4j-es-gchange" overwrite="true"> - <fileset dir="../duniter4j-es-gchange/target" includes="duniter4j-*${project.version}.jar"> + <copy todir="${run.es.home}/plugins/duniter4j-es-subscription" overwrite="true"> + <fileset dir="../duniter4j-es-subscription/target" includes="duniter4j-*${project.version}.jar"> </fileset> </copy> </then> diff --git a/duniter4j-es-assembly/src/main/assembly/config/elasticsearch.yml b/duniter4j-es-assembly/src/main/assembly/config/elasticsearch.yml index 31be3af2b8b1b7210eabbfeda9f008eee2a50d77..a7f0e815444f0415d87276ac32b35f6b74843dd7 100644 --- a/duniter4j-es-assembly/src/main/assembly/config/elasticsearch.yml +++ b/duniter4j-es-assembly/src/main/assembly/config/elasticsearch.yml @@ -176,3 +176,7 @@ duniter.mail.enable: false duniter.changes.listenSource: '*/block' duniter.ws.port: 9400 + +duniter.subscription.enable: true + +duniter.subscription.email.cesium.url: 'https://g1.duniter.fr' \ No newline at end of file diff --git a/duniter4j-es-assembly/src/test/es-home/config/elasticsearch.yml b/duniter4j-es-assembly/src/test/es-home/config/elasticsearch.yml index b11292099d8346807434f4f1d4a26ff04de2ebe1..67c8e332503447d2c82d6a154c6b36f1e8d3d9b8 100644 --- a/duniter4j-es-assembly/src/test/es-home/config/elasticsearch.yml +++ b/duniter4j-es-assembly/src/test/es-home/config/elasticsearch.yml @@ -162,14 +162,20 @@ duniter.data.sync.port: 443 # # SMTP server configuration (host and port) # -duniter.mail.enable: false -#duniter.mail.smtp.host: localhost -#duniter.mail.smtp.port: 25 +duniter.mail.enable: true +duniter.mail.smtp.host: smtp.gmail.com +duniter.mail.smtp.port: 465 +duniter.mail.smtp.ssl: true +duniter.mail.smtp.starttls: true +duniter.mail.smtp.username: benoit.lavenier@e-is.pro +duniter.mail.smtp.password: .2hainepourDieu + # # Mail 'from' address # #duniter.mail.from: no-reply@domain.com -#duniter.mail.from: root@EIS-DEV +duniter.mail.from: benoit.lavenier@e-is.pro + # # Mail: admin address # @@ -182,3 +188,7 @@ duniter.mail.enable: false duniter.changes.listenSource: '*/block' duniter.ws.port: 9400 + +# = 10s +duniter.subscription.enable: true +duniter.subscription.email.interval: 10000 \ No newline at end of file diff --git a/duniter4j-es-assembly/src/test/es-home/config/logging.yml b/duniter4j-es-assembly/src/test/es-home/config/logging.yml index c9b46939f307aa5433a2217e0e80075467c8e6b2..ad829d2029d741802ca251053c7068e5d7158a6a 100644 --- a/duniter4j-es-assembly/src/test/es-home/config/logging.yml +++ b/duniter4j-es-assembly/src/test/es-home/config/logging.yml @@ -20,14 +20,18 @@ logger: org.duniter: INFO org.duniter.core.beans: DEBUG + org.duniter.core.client.service: DEBUG org.duniter.elasticsearch: DEBUG org.duniter.elasticsearch.service: DEBUG - org.duniter.core.client.service: DEBUG + org.duniter.elasticsearch.user.service: DEBUG + org.duniter.elasticsearch.subscription.service: DEBUG + - duniter : DEBUG - duniter.security : ERROR - duniter.user.event : INFO + duniter: DEBUG + duniter.security: ERROR + duniter.user.event: INFO duniter.network.p2p: INFO + duniter.mail: DEBUG security: DEBUG diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/Plugin.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/Plugin.java index 4c8b851d4b7da36e83f02eb1e3c946c40694fa1a..693feb1be7183fdbc39332cae472d5301b4d7df7 100644 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/Plugin.java +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/Plugin.java @@ -46,6 +46,7 @@ public class Plugin extends org.elasticsearch.plugins.Plugin { @Inject public Plugin(Settings settings) { this.enable = settings.getAsBoolean("duniter.enabled", true); + } @Override diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/PluginInit.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/PluginInit.java index d91277ecc49c437e4978e99ee960ec19720659b8..6f285aed865ca1079dc87a097cb60880e4fed133 100644 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/PluginInit.java +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/PluginInit.java @@ -50,7 +50,7 @@ public class PluginInit extends AbstractLifecycleComponent<PluginInit> { private final static ESLogger logger = Loggers.getLogger("duniter.core"); @Inject - public PluginInit(Client client, Settings settings, PluginSettings pluginSettings, ThreadPool threadPool, final Injector injector) { + public PluginInit(Settings settings, PluginSettings pluginSettings, ThreadPool threadPool, final Injector injector) { super(settings); this.pluginSettings = pluginSettings; this.threadPool = threadPool; @@ -62,10 +62,9 @@ public class PluginInit extends AbstractLifecycleComponent<PluginInit> { threadPool.scheduleOnClusterHealthStatus(() -> { createIndices(); - // Waiting cluster back to GREEN or YELLOW state, before synchronize - threadPool.scheduleOnClusterHealthStatus(() -> { - synchronize(); - }, ClusterHealthStatus.YELLOW, ClusterHealthStatus.GREEN); + // Waiting cluster back to GREEN or YELLOW state, before doAfterStart + threadPool.scheduleOnClusterHealthStatus(this::doAfterStart, ClusterHealthStatus.YELLOW, ClusterHealthStatus.GREEN); + }, ClusterHealthStatus.YELLOW, ClusterHealthStatus.GREEN); } @@ -110,7 +109,9 @@ public class PluginInit extends AbstractLifecycleComponent<PluginInit> { } } - protected void synchronize() { + protected void doAfterStart() { + + // Synchronize blockchain if (pluginSettings.enableBlockchainSync()) { Peer peer = pluginSettings.checkAndGetPeer(); diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/PluginSettings.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/PluginSettings.java index e95270d390d0a7cb0ee960d11e5113b5efb94578..2a6140c64e21b9fe7ee03263b3b85e8ccc465429 100644 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/PluginSettings.java +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/PluginSettings.java @@ -324,6 +324,10 @@ public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> { } } + public Locale getI18nLocale() { + return clientConfig.getI18nLocale(); + } + /** * Override the version default option, from the MANIFEST implementation version (if any) * @param applicationConfig diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClient.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClient.java index a00760357cb0f36a2d023db2d616d0208a5dd20b..cdaea5b3b2d63b5ff9bc88cb66685a84a426197a 100644 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClient.java +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClient.java @@ -1,10 +1,12 @@ package org.duniter.elasticsearch.client; import org.duniter.core.beans.Bean; +import org.duniter.core.client.model.local.LocalEntity; import org.duniter.elasticsearch.dao.handler.StringReaderHandler; import org.elasticsearch.ElasticsearchException; import org.elasticsearch.action.bulk.BulkRequestBuilder; import org.elasticsearch.client.Client; +import org.elasticsearch.search.SearchHit; import java.io.File; import java.io.InputStream; @@ -56,6 +58,8 @@ public interface Duniter4jClient extends Bean, Client { */ <T extends Object> T getSourceById(String index, String type, String docId, Class<T> classOfT, String... fieldNames); + <C extends LocalEntity<String>> C readSourceOrNull(SearchHit searchHit, Class<? extends C> clazz); + void bulkFromClasspathFile(String classpathFile, String indexName, String indexType); void bulkFromClasspathFile(String classpathFile, String indexName, String indexType, StringReaderHandler handler); diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClientImpl.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClientImpl.java index 158b86b4b3cecbfedca3fcebf5da1ceaeb5210fb..54942a82c7dbd3151d25a6041f069cf10ee30f60 100644 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClientImpl.java +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/client/Duniter4jClientImpl.java @@ -28,6 +28,7 @@ import com.google.common.base.Joiner; import org.apache.commons.collections4.MapUtils; import org.duniter.core.client.model.bma.jackson.JacksonUtils; import org.duniter.core.client.model.elasticsearch.Record; +import org.duniter.core.client.model.local.LocalEntity; import org.duniter.core.exception.TechnicalException; import org.duniter.core.util.CollectionUtils; import org.duniter.core.util.ObjectUtils; @@ -377,6 +378,19 @@ public class Duniter4jClientImpl implements Duniter4jClient { } } + @Override + public <C extends LocalEntity<String>> C readSourceOrNull(SearchHit searchHit, Class<? extends C> clazz) { + try { + C value = objectMapper.readValue(searchHit.getSourceRef().streamInput(), clazz); + value.setId(searchHit.getId()); + return value; + } + catch(IOException e) { + logger.warn(String.format("Unable to deserialize source [%s/%s/%s] into [%s]: %s", searchHit.getIndex(), searchHit.getType(), searchHit.getId(), clazz.getName(), e.getMessage())); + return null; + } + } + @Override public void bulkFromClasspathFile(String classpathFile, String indexName, String indexType) { bulkFromClasspathFile(classpathFile, indexName, indexType, null); diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/IndexTypeDao.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/IndexTypeDao.java index 047280fbf0b7c93504df0eb68e8aea7b8403ef3a..dd4ff0614e2e3cfbcfc9aad64df080c594abd45a 100644 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/IndexTypeDao.java +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/dao/IndexTypeDao.java @@ -8,7 +8,7 @@ import java.util.Map; /** * Created by blavenie on 03/04/17. */ -public interface IndexTypeDao<T extends IndexTypeDao> { +public interface IndexTypeDao<T extends IndexTypeDao> extends IndexDao<T> { T createIndexIfNotExists(); diff --git a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/ServiceLocator.java b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/ServiceLocator.java index 7c773b054cdbe51833eed2556030a5bc96b072c3..cdc9b9c3ff25f96095ac87668c0b67b849cf2dbc 100644 --- a/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/ServiceLocator.java +++ b/duniter4j-es-core/src/main/java/org/duniter/elasticsearch/service/ServiceLocator.java @@ -30,8 +30,8 @@ import org.duniter.core.client.service.DataContext; import org.duniter.core.client.service.HttpService; import org.duniter.core.client.service.HttpServiceImpl; import org.duniter.core.client.service.bma.*; -import org.duniter.core.client.service.local.*; import org.duniter.core.client.service.local.CurrencyService; +import org.duniter.core.client.service.local.*; import org.duniter.core.exception.TechnicalException; import org.duniter.core.service.CryptoService; import org.duniter.core.service.Ed25519CryptoServiceImpl; @@ -39,10 +39,8 @@ import org.duniter.core.service.MailService; import org.duniter.core.service.MailServiceImpl; import org.duniter.elasticsearch.beans.ESBeanFactory; import org.duniter.elasticsearch.dao.BlockDao; -import org.duniter.elasticsearch.client.Duniter4jClient; import org.duniter.elasticsearch.dao.impl.BlockDaoImpl; import org.duniter.elasticsearch.dao.impl.CurrencyDaoImpl; -import org.duniter.elasticsearch.client.Duniter4jClientImpl; import org.duniter.elasticsearch.dao.impl.PeerDaoImpl; import org.elasticsearch.common.inject.Inject; import org.elasticsearch.common.inject.Injector; @@ -61,15 +59,12 @@ public class ServiceLocator private static ESBeanFactory beanFactory = null; @Inject - public ServiceLocator(Injector injector/*, PeerDao peerDao, CurrencyDao currencyDao*/) { + public ServiceLocator(Injector injector) { super(getOrCreateBeanFactory()); if (logger.isDebugEnabled()) { logger.debug("Starting Duniter4j ServiceLocator..."); } beanFactory.setInjector(injector); -/* - beanFactory.setBean(peerDao, PeerDao.class); - beanFactory.setBean(currencyDao, CurrencyDao.class);*/ org.duniter.core.client.service.ServiceLocator.setInstance(this); } @@ -122,10 +117,7 @@ public class ServiceLocator } public T get() { - logger.debug("Loading class [" + clazz.getName() + "]..."); - T result = getOrCreateBeanFactory().getBean(clazz); - logger.debug("...end of loading [" + clazz.getName() + "]"); - return result; + return getOrCreateBeanFactory().getBean(clazz); } } } diff --git a/duniter4j-es-core/src/main/resources/META-INF/services/org.duniter.core.beans.Bean b/duniter4j-es-core/src/main/resources/META-INF/services/org.duniter.core.beans.Bean index 1cf6cb3c777043f71eecb37be3c997efedf6cae7..c3c319867f1d4438cb5f6f538a1ff0a494b1232f 100644 --- a/duniter4j-es-core/src/main/resources/META-INF/services/org.duniter.core.beans.Bean +++ b/duniter4j-es-core/src/main/resources/META-INF/services/org.duniter.core.beans.Bean @@ -10,6 +10,4 @@ org.duniter.core.client.service.DataContext org.duniter.core.client.service.local.PeerServiceImpl org.duniter.core.client.service.local.CurrencyServiceImpl org.duniter.core.client.service.local.NetworkServiceImpl -org.duniter.core.client.dao.mem.MemoryCurrencyDaoImpl -org.duniter.core.client.dao.mem.MemoryPeerDaoImpl diff --git a/duniter4j-es-core/src/main/resources/META-INF/services/org.nuiton.config.ApplicationConfigProvider b/duniter4j-es-core/src/main/resources/META-INF/services/org.nuiton.config.ApplicationConfigProvider deleted file mode 100644 index f70e2417e5e77245a868ea71b4cf87e3650e853b..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/main/resources/META-INF/services/org.nuiton.config.ApplicationConfigProvider +++ /dev/null @@ -1 +0,0 @@ -org.duniter.elasticsearch.config.ConfigurationProvider diff --git a/duniter4j-es-core/src/test/es-home/config/elasticsearch.yml b/duniter4j-es-core/src/test/es-home/config/elasticsearch.yml index 828ddbd04543ef810f380ec8e0dbf229e6c2c48c..b9d916475573ca46815ff863b2268b8386cb1941 100644 --- a/duniter4j-es-core/src/test/es-home/config/elasticsearch.yml +++ b/duniter4j-es-core/src/test/es-home/config/elasticsearch.yml @@ -125,8 +125,8 @@ duniter.blockchain.sync.enable: false # # Duniter node to synchronize # -duniter.host: cgeek.fr -duniter.port: 9330 +duniter.host: gtest.duniter.org +duniter.port: 10900 # # ---------------------------------- Duniter4j security ------------------------- # @@ -174,5 +174,5 @@ duniter.mail.admin: blavenie@EIS-DEV # #duniter.mail.subject.prefix: [Duniter4j ES] -duniter.changes.listenSource: */block +duniter.changes.listenSource: '*/block' duniter.ws.port: 9400 diff --git a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/TestResource.java b/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/TestResource.java index 2b27a3e55828026c4e9b37c4a3dba40020e161c4..828607961ea176218af3fd86ca0ff8348574f2a1 100644 --- a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/TestResource.java +++ b/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/TestResource.java @@ -24,10 +24,13 @@ package org.duniter.elasticsearch; import com.google.common.collect.Lists; +import org.apache.commons.io.FileUtils; +import org.duniter.core.client.config.Configuration; import org.duniter.core.client.config.ConfigurationOption; import org.duniter.core.client.service.ServiceLocator; -import org.apache.commons.io.FileUtils; +import org.elasticsearch.bootstrap.Elasticsearch; import org.junit.runner.Description; +import org.nuiton.config.ApplicationConfig; import org.nuiton.i18n.I18n; import org.nuiton.i18n.init.DefaultI18nInitializer; import org.nuiton.i18n.init.UserI18nInitializer; @@ -64,14 +67,15 @@ public class TestResource extends org.duniter.core.test.TestResource { protected void before(Description description) throws Throwable { super.before(description); - // Initialize configuration - initConfiguration(getConfigFileName()); + // Prepare ES home + File esHomeDir = getResourceDirectory("es-home"); - // Init i18n - initI18n(); + System.setProperty("es.path.home", esHomeDir.getCanonicalPath()); - // Initialize service locator - ServiceLocator.instance().init(); + FileUtils.copyDirectory(new File("src/test/es-home"), esHomeDir); + FileUtils.copyDirectory(new File("target/classes"), new File(esHomeDir, "plugins/duniter4j-es-core")); + + Elasticsearch.main(new String[]{"start"}); } /** @@ -81,61 +85,7 @@ public class TestResource extends org.duniter.core.test.TestResource { * @return the prefix to use to retrieve configuration files */ protected String getConfigFilesPrefix() { - return "duniter4j-elasticsearch-test"; - } - - protected String getI18nBundleName() { - return "duniter4j-elasticsearch-i18n"; - } - - /* -- -- */ - - /** - * Convenience methods that could be override to initialize other configuration - * - * @param configFilename - * @param configArgs - */ - protected void initConfiguration(String configFilename) { - String[] configArgs = getConfigArgs(); - //PluginSettings config = new PluginSettings(configFilename, configArgs); - //PluginSettings.setInstance(config); - } - - protected void initI18n() throws IOException { - /*PluginSettings config ;//= PluginSettings.instance(); - - // --------------------------------------------------------------------// - // init i18n - // --------------------------------------------------------------------// - File i18nDirectory = new File(config.getDataDirectory(), "i18n"); - if (i18nDirectory.exists()) { - // clean i18n cache - FileUtils.cleanDirectory(i18nDirectory); - } - - FileUtils.forceMkdir(i18nDirectory); - - if (log.isDebugEnabled()) { - log.debug("I18N directory: " + i18nDirectory); - } - - Locale i18nLocale = config.getI18nLocale(); - - if (log.isInfoEnabled()) { - log.info(String.format("Starts i18n with locale [%s] at [%s]", - i18nLocale, i18nDirectory)); - } - I18n.init(new UserI18nInitializer( - i18nDirectory, new DefaultI18nInitializer(getI18nBundleName())), - i18nLocale);*/ - } - - protected String[] getConfigArgs() { - List<String> configArgs = Lists.newArrayList(); - configArgs.addAll(Lists.newArrayList( - "--option", ConfigurationOption.BASEDIR.getKey(), getResourceDirectory().getAbsolutePath())); - return configArgs.toArray(new String[configArgs.size()]); + return "duniter4j-es-core-test"; } } diff --git a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/BlockchainServiceTest.java b/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/BlockchainServiceTest.java index a305ac1b40ff0fca3ed27b843d7c737d1f117d50..bb17bab81c3cf5f1fcab3272eeb860c9ed723444 100644 --- a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/BlockchainServiceTest.java +++ b/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/BlockchainServiceTest.java @@ -23,18 +23,22 @@ package org.duniter.elasticsearch.service; */ +import com.fasterxml.jackson.databind.ObjectMapper; import org.duniter.core.client.config.Configuration; import org.duniter.core.client.model.bma.BlockchainBlock; +import org.duniter.core.client.model.bma.jackson.JacksonUtils; import org.duniter.core.client.model.local.Peer; import org.duniter.core.client.service.bma.BlockchainRemoteService; import org.duniter.elasticsearch.TestResource; -import org.junit.*; +import org.junit.Assert; +import org.junit.Before; +import org.junit.ClassRule; +import org.junit.Test; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import java.util.List; -@Ignore public class BlockchainServiceTest { private static final Logger log = LoggerFactory.getLogger(BlockchainServiceTest.class); @@ -43,115 +47,47 @@ public class BlockchainServiceTest { public static final TestResource resource = TestResource.create(); private BlockchainService service; - private BlockchainRemoteService blockchainRemoteService; + private BlockchainRemoteService remoteService; private Configuration config; private Peer peer; + private ObjectMapper objectMapper; @Before public void setUp() throws Exception { - //service = ServiceLocator.instance().getBlockIndexerService(); - //blockchainRemoteService = ServiceLocator.instance().getBlockchainRemoteService(); + service = ServiceLocator.instance().getBean(BlockchainService.class); + remoteService = ServiceLocator.instance().getBlockchainRemoteService(); config = Configuration.instance(); peer = createTestPeer(); + objectMapper = JacksonUtils.newObjectMapper(); - initLocalNode(); + // Init the currency + CurrencyService currencyService = ServiceLocator.instance().getBean(CurrencyService.class); + currencyService.createIndexIfNotExists() + .indexCurrencyFromPeer(peer); } @Test - public void createIndex() throws Exception { - String currencyName = resource.getFixtures().getCurrency(); - - // drop and recreate index - service.deleteIndex(currencyName); - - service.createIndex(currencyName); + public void indexLastBlocks() { + service.indexLastBlocks(peer); } - @Test - public void indexBlock() throws Exception { - // Read a block - BlockchainBlock currentBlock = blockchainRemoteService.getCurrentBlock(peer); - - // Create a new non-existing block - service.indexBlock(currentBlock, true); + public void indexBlock() { + BlockchainBlock current = remoteService.getCurrentBlock(peer); + service.indexCurrentBlock(current, true/*wait*/); - // Update a existing block - { - currentBlock.setMembersCount(1000000); + try { + String blockStr = objectMapper.writeValueAsString(current); - service.indexBlock(currentBlock, true); + service.indexBlockFromJson(peer, blockStr, true/*is rurrent*/, false/*detected fork*/, true/*wait*/); + } + catch(Exception e) { + Assert.fail(e.getMessage()); } - } - - @Test - public void indexCurrentBlock() throws Exception { - // Create a block with a fake hash - BlockchainBlock aBlock = blockchainRemoteService.getBlock(peer, 8450); - service.indexCurrentBlock(aBlock, true); - } - - @Test - // FIXME make this works - @Ignore - public void searchBlocks() throws Exception { - String currencyName = resource.getFixtures().getCurrency(); - - // Create a block with a fake hash - BlockchainBlock aBlock = blockchainRemoteService.getCurrentBlock(peer); - aBlock.setHash("myUnitTestHash"); - service.saveBlock(aBlock, true, true); - - Thread.sleep(5 * 1000); // wait 5s that ES process the block - - // match multi words - String queryText = aBlock.getHash(); - List<BlockchainBlock> blocks = service.findBlocksByHash(currencyName, queryText); - //assertResults(queryText, blocks); - - Thread.sleep(5 * 1000); // wait 5s that ES process the block - - BlockchainBlock loadBlock = service.getBlockById(currencyName, aBlock.getNumber()); - Assert.assertNotNull(loadBlock); - Assert.assertEquals(aBlock.getHash(), loadBlock.getHash()); - } - - @Test - public void getMaxBlockNumber() throws Exception { - String currencyName = resource.getFixtures().getCurrency(); - - // match multi words - Integer maxBlockNumber = service.getMaxBlockNumber(currencyName); - Assert.assertNotNull(maxBlockNumber); - } - - - @Test - @Ignore - public void allInOne() throws Exception { - - createIndex(); - indexBlock(); - searchBlocks(); } /* -- internal methods */ - protected void initLocalNode() throws Exception { - String currencyName = resource.getFixtures().getCurrency(); - - // Make sure the index exists - service.deleteIndex(currencyName); - service.createIndex(currencyName); - - // Get the first block from peer - BlockchainBlock firstBlock = blockchainRemoteService.getBlock(peer, 0); - - // Make sure the block has been indexed - service.indexBlock(firstBlock, true); - - } - protected void assertResults(String queryText, List<BlockchainBlock> result) { log.info(String.format("Results for a search on [%s]", queryText)); Assert.assertNotNull(result); diff --git a/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/CurrencyServiceTest.java b/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/CurrencyServiceTest.java new file mode 100644 index 0000000000000000000000000000000000000000..a7626d6e3d6c0008a0788161d3dc67827d628216 --- /dev/null +++ b/duniter4j-es-core/src/test/java/org/duniter/elasticsearch/service/CurrencyServiceTest.java @@ -0,0 +1,81 @@ +package org.duniter.elasticsearch.service; + +/* + * #%L + * UCoin Java Client :: Core API + * %% + * Copyright (C) 2014 - 2015 EIS + * %% + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as + * published by the Free Software Foundation, either version 3 of the + * License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public + * License along with this program. If not, see + * <http://www.gnu.org/licenses/gpl-3.0.html>. + * #L% + */ + + +import org.duniter.core.client.config.Configuration; +import org.duniter.core.client.model.bma.BlockchainBlock; +import org.duniter.core.client.model.local.Peer; +import org.duniter.core.client.service.bma.BlockchainRemoteService; +import org.duniter.elasticsearch.TestResource; +import org.junit.*; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.util.List; + +@Ignore +public class CurrencyServiceTest { + + private static final Logger log = LoggerFactory.getLogger(CurrencyServiceTest.class); + + @ClassRule + public static final TestResource resource = TestResource.create(); + + private BlockchainRemoteService blockchainRemoteService; + private CurrencyService service; + private Configuration config; + private Peer peer; + + @Before + public void setUp() throws Exception { + service = ServiceLocator.instance().getBean(CurrencyService.class); + blockchainRemoteService = ServiceLocator.instance().getBlockchainRemoteService(); + config = Configuration.instance(); + peer = createTestPeer(); + } + + @Test + public void createIndexIfNotExists() throws Exception { + + // drop and recreate index + service.deleteIndex().createIndexIfNotExists(); + } + + @Test + public void indexCurrencyFromPeer() throws Exception { + service.createIndexIfNotExists() + .indexCurrencyFromPeer(peer); + } + + /* -- internal methods */ + + protected Peer createTestPeer() { + Peer peer = new Peer( + Configuration.instance().getNodeHost(), + Configuration.instance().getNodePort()); + + return peer; + } + +} diff --git a/duniter4j-es-core/src/test/resources/META-INF/services/org.duniter.core.beans.Bean b/duniter4j-es-core/src/test/resources/META-INF/services/org.duniter.core.beans.Bean index 1d327a744e4eec6703b417254f5b19dfdbc09fa4..6613b03e0d6f746558b3ac98f065d36d74cf3411 100644 --- a/duniter4j-es-core/src/test/resources/META-INF/services/org.duniter.core.beans.Bean +++ b/duniter4j-es-core/src/test/resources/META-INF/services/org.duniter.core.beans.Bean @@ -8,7 +8,6 @@ org.duniter.core.client.service.HttpServiceImpl org.duniter.core.client.service.DataContext org.duniter.core.client.service.local.PeerServiceImpl org.duniter.core.client.service.local.CurrencyServiceImpl -org.duniter.core.client.dao.mem.MemoryCurrencyDaoImpl -org.duniter.core.client.dao.mem.MemoryPeerDaoImpl -org.duniter.elasticsearch.service.ElasticSearchService -org.duniter.elasticsearch.service.registry.CurrencyRegistryService \ No newline at end of file +org.duniter.elasticsearch.dao.impl.CurrencyDaoImpl +org.duniter.elasticsearch.dao.impl.PeerDaoImpl +org.duniter.elasticsearch.dao.impl.BlockDaoImpl diff --git a/duniter4j-es-core/src/test/resources/duniter4j-elasticsearch-localhost-node.properties b/duniter4j-es-core/src/test/resources/duniter4j-elasticsearch-localhost-node.properties deleted file mode 100644 index a7974bcf8a4a70cc95c3504514407d8612e1e258..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/resources/duniter4j-elasticsearch-localhost-node.properties +++ /dev/null @@ -1,12 +0,0 @@ -duniter4j.node.host=g1.duniter.fr -duniter4j.node.port=10901 - -duniter4j.elasticsearch.embedded.enable=false -duniter4j.elasticsearch.local=false -duniter4j.elasticsearch.http.enable=false -duniter4j.elasticsearch.cluster.name=duniter4j-elacticsearch-test - -#duniter4j.elasticsearch.cluster.name=duniter4j-elacticsearch - -duniter4j.elasticsearch.host=localhost -duniter4j.elasticsearch.port=9300 diff --git a/duniter4j-es-core/src/test/resources/duniter4j-elasticsearch-test.properties b/duniter4j-es-core/src/test/resources/duniter4j-elasticsearch-test.properties deleted file mode 100644 index 301e47352113728eb7614b69e672b6bcfe4c92b7..0000000000000000000000000000000000000000 --- a/duniter4j-es-core/src/test/resources/duniter4j-elasticsearch-test.properties +++ /dev/null @@ -1,16 +0,0 @@ -duniter4j.node.host=g1.duniter.org -duniter4j.node.port=10901 - -duniter4j.basedir=target/es-home - -#duniter4j.elasticsearch.data -#duniter4j.elasticsearch.embedded.enable=true -duniter4j.elasticsearch.local=false -duniter4j.elasticsearch.http.enable=true -duniter4j.elasticsearch.cluster.name=duniter4j-elasticsearch - -#duniter4j.elasticsearch.cluster.name=duniter4j-elacticsearch - -duniter4j.elasticsearch.embedded.enable=false -duniter4j.elasticsearch.host=192.168.0.5 -duniter4j.elasticsearch.port=9300 diff --git a/duniter4j-es-core/src/test/resources/duniter4j-es-core-test.properties b/duniter4j-es-core/src/test/resources/duniter4j-es-core-test.properties new file mode 100644 index 0000000000000000000000000000000000000000..608cea0ef49f2330cc7a26f58af9bacd76da0e1a --- /dev/null +++ b/duniter4j-es-core/src/test/resources/duniter4j-es-core-test.properties @@ -0,0 +1 @@ +#Empty test file (need for inherited TestResource). See files 'src/test/es-home/config' \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/AbstractCommentDaoImpl.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/AbstractCommentDaoImpl.java deleted file mode 100644 index 52944b007df0f9f5232db32243a85e36361e83ab..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/AbstractCommentDaoImpl.java +++ /dev/null @@ -1,155 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao; - -/* - * #%L - * UCoin Java Client :: Core API - * %% - * Copyright (C) 2014 - 2015 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - - -import com.fasterxml.jackson.core.JsonProcessingException; -import org.duniter.core.client.model.elasticsearch.RecordComment; -import org.duniter.core.exception.TechnicalException; -import org.duniter.elasticsearch.dao.AbstractIndexTypeDao; -import org.duniter.elasticsearch.gchange.PluginSettings; -import org.duniter.elasticsearch.gchange.dao.CommentDao; -import org.elasticsearch.action.search.SearchPhaseExecutionException; -import org.elasticsearch.action.search.SearchRequestBuilder; -import org.elasticsearch.action.search.SearchResponse; -import org.elasticsearch.action.search.SearchType; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; -import org.elasticsearch.index.query.QueryBuilders; -import org.elasticsearch.index.query.TermQueryBuilder; - -import java.io.IOException; - -/** - * Created by Benoit on 30/03/2015. - */ -public class AbstractCommentDaoImpl<T extends AbstractCommentDaoImpl> extends AbstractIndexTypeDao<T> implements CommentDao<T> { - - - protected PluginSettings pluginSettings; - - public AbstractCommentDaoImpl(String index, PluginSettings pluginSettings) { - super(index, CommentDao.TYPE); - this.pluginSettings = pluginSettings; - } - - @Override - protected void createIndex() throws JsonProcessingException { - throw new TechnicalException("not implemented"); - } - - public String create(final String json) { - return super.indexDocumentFromJson(json); - } - - public void update(final String id, final String json) { - super.updateDocumentFromJson(id, json); - } - - @Override - public long countReplies(String id) { - - // Prepare count request - SearchRequestBuilder searchRequest = client - .prepareSearch(getIndex()) - .setTypes(getType()) - .setFetchSource(false) - .setSearchType(SearchType.QUERY_AND_FETCH) - .setSize(0); - - // Query = filter on reference - TermQueryBuilder query = QueryBuilders.termQuery(RecordComment.PROPERTY_REPLY_TO_JSON, id); - searchRequest.setQuery(query); - - // Execute query - try { - SearchResponse response = searchRequest.execute().actionGet(); - return response.getHits().getTotalHits(); - } - catch(SearchPhaseExecutionException e) { - // Failed or no item on index - logger.error(String.format("Error while counting comment replies: %s", e.getMessage()), e); - } - return 1; - } - - - public XContentBuilder createTypeMapping() { - String stringAnalyzer = pluginSettings.getDefaultStringAnalyzer(); - - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject().startObject(getType()) - .startObject("properties") - - // issuer - .startObject("issuer") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // time - .startObject("time") - .field("type", "integer") - .endObject() - - // message - .startObject("message") - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - - // record - .startObject("record") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // reply to - .startObject("reply_to") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // aggregations - .startObject("aggregations") - .field("type", "nested") - .field("dynamic", "true") - .startObject("properties") - .startObject("reply_count") - .field("type", "integer") - .field("index", "not_analyzed") - .endObject() - .endObject() - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s]: %s", getType(), ioe.getMessage()), ioe); - } - } - -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/AbstractRecordDaoImpl.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/AbstractRecordDaoImpl.java deleted file mode 100644 index 648137245cd7df2464ea102ac1c8384179ba770e..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/AbstractRecordDaoImpl.java +++ /dev/null @@ -1,210 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao; - -/* - * #%L - * UCoin Java Client :: Core API - * %% - * Copyright (C) 2014 - 2015 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - - -import com.fasterxml.jackson.core.JsonProcessingException; -import org.duniter.core.client.model.elasticsearch.Record; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.ObjectUtils; -import org.duniter.elasticsearch.dao.AbstractIndexTypeDao; -import org.duniter.elasticsearch.gchange.PluginSettings; -import org.duniter.elasticsearch.gchange.dao.RecordDao; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; - -import java.io.IOException; - -/** - * Created by Benoit on 30/03/2015. - */ -public class AbstractRecordDaoImpl<T extends AbstractRecordDaoImpl> extends AbstractIndexTypeDao<T> implements RecordDao<T> { - - protected PluginSettings pluginSettings; - - public AbstractRecordDaoImpl(String index, PluginSettings pluginSettings) { - super(index, RecordDao.TYPE); - this.pluginSettings = pluginSettings; - } - - @Override - protected void createIndex() throws JsonProcessingException { - throw new TechnicalException("not implemented"); - } - - @Override - public void checkSameDocumentIssuer(String id, String expectedIssuer) { - String issuer = getMandatoryFieldsById(id, Record.PROPERTY_ISSUER).get(Record.PROPERTY_ISSUER).toString(); - if (!ObjectUtils.equals(expectedIssuer, issuer)) { - throw new TechnicalException("Not same issuer"); - } - } - - public XContentBuilder createTypeMapping() { - String stringAnalyzer = pluginSettings.getDefaultStringAnalyzer(); - - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject().startObject(getType()) - .startObject("properties") - - // title - .startObject("title") - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - - // description - .startObject("description") - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - - // creationTime - .startObject("creationTime") - .field("type", "integer") - .endObject() - - // time - .startObject("time") - .field("type", "integer") - .endObject() - - // issuer - .startObject("issuer") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // pubkey - .startObject("pubkey") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // address - .startObject("address") - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - - // city - .startObject("city") - .field("type", "string") - .endObject() - - // geoPoint - .startObject("geoPoint") - .field("type", "geo_point") - .endObject() - - // thumbnail - .startObject("thumbnail") - .field("type", "attachment") - .startObject("fields") // src - .startObject("content") // title - .field("index", "no") - .endObject() - .startObject("title") // title - .field("type", "string") - .field("store", "no") - .endObject() - .startObject("author") // title - .field("store", "no") - .endObject() - .startObject("content_type") // title - .field("store", "yes") - .endObject() - .endObject() - .endObject() - - // pictures - .startObject("pictures") - .field("type", "nested") - .field("dynamic", "false") - .startObject("properties") - .startObject("file") // file - .field("type", "attachment") - .startObject("fields") - .startObject("content") // content - .field("index", "no") - .endObject() - .startObject("title") // title - .field("type", "string") - .field("store", "yes") - .field("analyzer", stringAnalyzer) - .endObject() - .startObject("author") // author - .field("type", "string") - .field("store", "no") - .endObject() - .startObject("content_type") // content_type - .field("store", "yes") - .endObject() - .endObject() - .endObject() - .endObject() - .endObject() - - // picturesCount - .startObject("picturesCount") - .field("type", "integer") - .endObject() - - // category - .startObject("category") - .field("type", "nested") - .field("dynamic", "false") - .startObject("properties") - .startObject("id") // id - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .startObject("parent") // parent - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .startObject("name") // name - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - .endObject() - .endObject() - - // tags - .startObject("tags") - .field("type", "completion") - .field("search_analyzer", "simple") - .field("analyzer", "simple") - .field("preserve_separators", "false") - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", getIndex(), getType(), ioe.getMessage()), ioe); - } - } -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/CommentDao.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/CommentDao.java deleted file mode 100644 index 3cefe3997f7fdf4e74ed24cbba6a6a6d882c2d83..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/CommentDao.java +++ /dev/null @@ -1,41 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao; - -/* - * #%L - * UCoin Java Client :: Core API - * %% - * Copyright (C) 2014 - 2015 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - - -import org.duniter.elasticsearch.dao.IndexTypeDao; - -/** - * Created by Benoit on 30/03/2015. - */ -public interface CommentDao<T extends CommentDao> extends IndexTypeDao<T> { - - String TYPE = "comment"; - - String create(final String json); - - void update(final String id, final String json); - - long countReplies(String id); - -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/DaoModule.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/DaoModule.java deleted file mode 100644 index 65625ab03abd553e5dbf9c4a304dad3e2bdd9d6a..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/DaoModule.java +++ /dev/null @@ -1,52 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao; - -/* - * #%L - * duniter4j-elasticsearch-plugin - * %% - * Copyright (C) 2014 - 2016 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.core.beans.Bean; -import org.duniter.elasticsearch.client.Duniter4jClientImpl; -import org.duniter.elasticsearch.gchange.dao.market.*; -import org.duniter.elasticsearch.gchange.dao.registry.*; -import org.duniter.elasticsearch.service.ServiceLocator; -import org.elasticsearch.common.inject.AbstractModule; -import org.elasticsearch.common.inject.Module; - -public class DaoModule extends AbstractModule implements Module { - - @Override protected void configure() { - - bind(RegistryIndexDao.class).to(RegistryIndexDaoImpl.class).asEagerSingleton(); - bind(RegistryCommentDao.class).to(RegistryCommentDaoImpl.class).asEagerSingleton(); - bind(RegistryRecordDao.class).to(RegistryRecordDaoImpl.class).asEagerSingleton(); - - bind(MarketIndexDao.class).to(MarketIndexDaoImpl.class).asEagerSingleton(); - bind(MarketCommentDao.class).to(MarketCommentDaoImpl.class).asEagerSingleton(); - bind(MarketRecordDao.class).to(MarketRecordDaoImpl.class).asEagerSingleton(); - } - - /* protected methods */ - - protected <T extends Bean> void bindWithLocator(Class<T> clazz) { - bind(clazz).toProvider(new ServiceLocator.Provider<>(clazz)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/location/CitiesLocationDaoImpl.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/location/CitiesLocationDaoImpl.java deleted file mode 100644 index bcdf9807663fa165369a3834a3bf41bf7fc95cd9..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/location/CitiesLocationDaoImpl.java +++ /dev/null @@ -1,349 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao.location; - -/* - * #%L - * UCoin Java Client :: Core API - * %% - * Copyright (C) 2014 - 2015 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - - -import com.fasterxml.jackson.core.JsonProcessingException; -import org.duniter.core.exception.TechnicalException; -import org.duniter.elasticsearch.PluginSettings; -import org.duniter.elasticsearch.dao.AbstractDao; -import org.elasticsearch.action.admin.indices.create.CreateIndexRequestBuilder; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.logging.ESLogger; -import org.elasticsearch.common.logging.ESLoggerFactory; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; - -import java.io.IOException; - -/** - * Created by Benoit on 30/03/2015. - */ -public class CitiesLocationDaoImpl extends AbstractDao { - - private static final ESLogger log = ESLoggerFactory.getLogger(CitiesLocationDaoImpl.class.getName()); - - private static final String CITIES_BULK_FILENAME = "registry-cities-bulk-insert.json"; - - private static final String CITIES_SOURCE_CLASSPATH_FILE = "cities/countriesToCities.json"; - - private static final String CITIES_SOURCE_FILE2 = "/home/blavenie/git/ucoin-io/duniter4j/duniter4j-elasticsearch/src/main/misc/geoflar-communes-2015.geojson"; - - public static final String INDEX = "location"; - public static final String CITY_TYPE = "city"; - - public CitiesLocationDaoImpl() { - super("gchange.location.cities"); - } - - /** - * Delete blockchain index, and all data - * @throws JsonProcessingException - */ - public void deleteIndex() throws JsonProcessingException { - client.deleteIndexIfExists(INDEX); - } - - - public boolean existsIndex() { - return client.existsIndex(INDEX); - } - - /** - * Create index need for blockchain registry, if need - */ - public void createIndexIfNotExists() { - try { - if (!client.existsIndex(INDEX)) { - createIndex(); - } - } - catch(JsonProcessingException e) { - throw new TechnicalException(String.format("Error while creating index [%s]", INDEX)); - } - } - - /** - * Create index need for category registry - * @throws JsonProcessingException - */ - public void createIndex() throws JsonProcessingException { - log.info(String.format("Creating index [%s/%s]", INDEX, CITY_TYPE)); - - CreateIndexRequestBuilder createIndexRequestBuilder = client.admin().indices().prepareCreate(INDEX); - org.elasticsearch.common.settings.Settings indexSettings = org.elasticsearch.common.settings.Settings.settingsBuilder() - .put("number_of_shards", 1) - .put("number_of_replicas", 1) - //.put("analyzer", createDefaultAnalyzer()) - .build(); - createIndexRequestBuilder.setSettings(indexSettings); - createIndexRequestBuilder.addMapping(CITY_TYPE, createIndexMapping()); - createIndexRequestBuilder.execute().actionGet(); - } - - public void initCities() { - if (log.isDebugEnabled()) { - log.debug("Initializing allOfToList registry cities"); - } - - //File bulkFile = createCitiesBulkFile2(); - - // Insert cities - //bulkFromFile(bulkFile, INDEX, CITY_TYPE); - } - - - /* -- Internal methods -- */ - - - public XContentBuilder createIndexMapping() { - - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject().startObject(CITY_TYPE) - .startObject("properties") - - // city - .startObject("name") - .field("type", "string") - .endObject() - - // country - .startObject("country") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", INDEX, CITY_TYPE, ioe.getMessage()), ioe); - } - } - - /* - public File createCitiesBulkFile() { - - File result = new File(pluginSettings.getTempDirectory(), CITIES_BULK_FILENAME); - - InputStream ris = null; - BufferedReader bf = null; - FileWriter fw = null; - try { - if (result.exists()) { - FileUtils.forceDelete(result); - } - else if (!result.getParentFile().exists()) { - FileUtils.forceMkdir(result.getParentFile()); - } - - ris = getClass().getClassLoader().getResourceAsStream(CITIES_SOURCE_CLASSPATH_FILE); - if (ris == null) { - throw new TechnicalException(String.format("Could not retrieve file [%s] from test classpath. Make sure git submodules has been initialized before building.", CITIES_SOURCE_CLASSPATH_FILE)); - } - - boolean firstLine = true; - java.lang.reflect.Type typeOfHashMap = new TypeToken<Map<String, String[]>>() { }.getType(); - - Gson gson = GsonUtils.newBuilder().create(); - - StringBuilder builder = new StringBuilder(); - bf = new BufferedReader( - new InputStreamReader( - ris, "UTF-16LE"), 2048); - - fw = new FileWriter(result); - char[] buf = new char[2048]; - int len; - - while((len = bf.read(buf)) != -1) { - String bufStr = new String(buf, 0, len); - - if (firstLine) { - // Remove UTF-16 BOM char - int objectStartIndex = bufStr.indexOf('\uFEFF'); - if (objectStartIndex != -1) { - bufStr = bufStr.substring(objectStartIndex); - } - firstLine=false; - } - - int arrayEndIndex = bufStr.indexOf("],\""); - if (arrayEndIndex == -1) { - arrayEndIndex = bufStr.indexOf("]}"); - } - - if (arrayEndIndex == -1) { - builder.append(bufStr); - } - else { - builder.append(bufStr.substring(0, arrayEndIndex+1)); - builder.append("}"); - if (log.isTraceEnabled()) { - log.trace(builder.toString()); - } - Map<String, String[]> citiesByCountry = gson.fromJson(builder.toString(), typeOfHashMap); - - builder.setLength(0); - for (String country: citiesByCountry.keySet()) { - if (StringUtils.isNotBlank(country)) { - for (String city : citiesByCountry.get(country)) { - if (StringUtils.isNotBlank(city)) { - fw.write(String.format("{\"index\":{\"_id\" : \"%s-%s\"}}\n", country, city)); - fw.write(String.format("{\"country\":\"%s\", \"name\":\"%s\"}\n", country, city)); - } - } - } - } - - fw.flush(); - - // reset and prepare buffer for next country - builder.setLength(0); - builder.append("{"); - if (arrayEndIndex+2 < bufStr.length()) { - builder.append(bufStr.substring(arrayEndIndex+2)); - } - } - } - - fw.close(); - bf.close(); - - } catch(Exception e) { - throw new TechnicalException(String.format("Error while creating cities file [%s]", result.getName()), e); - } - finally { - IOUtils.closeQuietly(bf); - IOUtils.closeQuietly(ris); - IOUtils.closeQuietly(fw); - } - - return result; - } - - public File createCitiesBulkFile2() { - - File result = new File(pluginSettings.getTempDirectory(), CITIES_BULK_FILENAME); - File inputFile = new File(CITIES_SOURCE_FILE2); - - InputStream ris = null; - BufferedReader bf = null; - FileWriter fw = null; - try { - if (result.exists()) { - FileUtils.forceDelete(result); - } - else if (!result.getParentFile().exists()) { - FileUtils.forceMkdir(result.getParentFile()); - } - - ris = new BufferedInputStream(new FileInputStream(inputFile)); - if (ris == null) { - throw new TechnicalException(String.format("Could not retrieve file [%s] from test classpath. Make sure git submodules has been initialized before building.", CITIES_SOURCE_FILE2)); - } - - boolean firstLine = true; - java.lang.reflect.Type typeOfHashMap = new TypeToken<Map<String, String[]>>() { }.getType(); - - Gson gson = GsonUtils.newBuilder().create(); - - StringBuilder builder = new StringBuilder(); - bf = new BufferedReader( - new InputStreamReader( - ris, "UTF-16LE"), 2048); - - fw = new FileWriter(result); - char[] buf = new char[2048]; - int len; - - while((len = bf.read(buf)) != -1) { - String bufStr = new String(buf, 0, len); - - if (firstLine) { - // Remove UTF-16 BOM char - int objectStartIndex = bufStr.indexOf('\uFEFF'); - if (objectStartIndex != -1) { - bufStr = bufStr.substring(objectStartIndex); - } - firstLine=false; - } - - int arrayEndIndex = bufStr.indexOf("],\""); - if (arrayEndIndex == -1) { - arrayEndIndex = bufStr.indexOf("]}"); - } - - if (arrayEndIndex == -1) { - builder.append(bufStr); - } - else { - builder.append(bufStr.substring(0, arrayEndIndex+1)); - builder.append("}"); - if (log.isTraceEnabled()) { - log.trace(builder.toString()); - } - Map<String, String[]> citiesByCountry = gson.fromJson(builder.toString(), typeOfHashMap); - - builder.setLength(0); - for (String country: citiesByCountry.keySet()) { - if (StringUtils.isNotBlank(country)) { - for (String city : citiesByCountry.get(country)) { - if (StringUtils.isNotBlank(city)) { - fw.write(String.format("{\"index\":{\"_id\" : \"%s-%s\"}}\n", country, city)); - fw.write(String.format("{\"country\":\"%s\", \"name\":\"%s\"}\n", country, city)); - } - } - } - } - - fw.flush(); - - // reset and prepare buffer for next country - builder.setLength(0); - builder.append("{"); - if (arrayEndIndex+2 < bufStr.length()) { - builder.append(bufStr.substring(arrayEndIndex+2)); - } - } - } - - fw.close(); - bf.close(); - - } catch(Exception e) { - throw new TechnicalException(String.format("Error while creating cities file [%s]", result.getName()), e); - } - finally { - IOUtils.closeQuietly(bf); - IOUtils.closeQuietly(ris); - IOUtils.closeQuietly(fw); - } - - return result; - } - */ -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketCommentDao.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketCommentDao.java deleted file mode 100644 index a3f16efab314f737cbf942d0124c0a90faad29ef..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketCommentDao.java +++ /dev/null @@ -1,9 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao.market; - -import org.duniter.elasticsearch.gchange.dao.CommentDao; - -/** - * Created by blavenie on 03/04/17. - */ -public interface MarketCommentDao extends CommentDao { -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketCommentDaoImpl.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketCommentDaoImpl.java deleted file mode 100644 index 415f3ba193d83e012de37530d5ceb95f3d872038..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketCommentDaoImpl.java +++ /dev/null @@ -1,22 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao.market; - -import org.duniter.core.exception.TechnicalException; -import org.duniter.elasticsearch.gchange.PluginSettings; -import org.duniter.elasticsearch.gchange.dao.AbstractCommentDaoImpl; -import org.duniter.elasticsearch.gchange.dao.AbstractRecordDaoImpl; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; - -import java.io.IOException; - -/** - * Created by blavenie on 03/04/17. - */ -public class MarketCommentDaoImpl extends AbstractCommentDaoImpl implements MarketCommentDao { - - @Inject - public MarketCommentDaoImpl(PluginSettings pluginSettings) { - super(MarketIndexDao.INDEX, pluginSettings); - } -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketIndexDao.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketIndexDao.java deleted file mode 100644 index 2e3f36177728d5f8cee36ff2230b6ba875b35ce0..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketIndexDao.java +++ /dev/null @@ -1,11 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao.market; - -import org.duniter.elasticsearch.dao.IndexDao; - -/** - * Created by blavenie on 03/04/17. - */ -public interface MarketIndexDao extends IndexDao<MarketIndexDao> { - String INDEX = "market"; - String CATEGORY_TYPE = "category"; -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketIndexDaoImpl.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketIndexDaoImpl.java deleted file mode 100644 index 69548e97c3b231274df263fdc2d9df79cc7af787..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketIndexDaoImpl.java +++ /dev/null @@ -1,111 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao.market; - -import com.fasterxml.jackson.core.JsonProcessingException; -import org.duniter.core.exception.TechnicalException; -import org.duniter.elasticsearch.dao.AbstractIndexDao; -import org.duniter.elasticsearch.dao.AbstractIndexTypeDao; -import org.duniter.elasticsearch.dao.IndexTypeDao; -import org.duniter.elasticsearch.dao.handler.AddSequenceAttributeHandler; -import org.duniter.elasticsearch.gchange.PluginSettings; -import org.duniter.elasticsearch.gchange.dao.CommentDao; -import org.duniter.elasticsearch.gchange.dao.RecordDao; -import org.elasticsearch.action.admin.indices.create.CreateIndexRequestBuilder; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; - -import java.io.IOException; - -/** - * Created by blavenie on 03/04/17. - */ -public class MarketIndexDaoImpl extends AbstractIndexDao<MarketIndexDao> implements MarketIndexDao { - - private static final String CATEGORIES_BULK_CLASSPATH_FILE = "registry-categories-bulk-insert.json"; - - private PluginSettings pluginSettings; - private IndexTypeDao<?> categoryDao; - private RecordDao recordDao; - private CommentDao commentDao; - - @Inject - public MarketIndexDaoImpl(PluginSettings pluginSettings, MarketRecordDao recordDao, MarketCommentDao commentDao) { - super(MarketIndexDao.INDEX); - - this.pluginSettings = pluginSettings; - this.commentDao = commentDao; - this.recordDao = recordDao; - this.categoryDao = createCategoryDao(pluginSettings); - } - - - @Override - protected void createIndex() throws JsonProcessingException { - logger.info(String.format("Creating index [%s]", INDEX)); - - CreateIndexRequestBuilder createIndexRequestBuilder = client.admin().indices().prepareCreate(INDEX); - org.elasticsearch.common.settings.Settings indexSettings = org.elasticsearch.common.settings.Settings.settingsBuilder() - .put("number_of_shards", 3) - .put("number_of_replicas", 1) - //.put("analyzer", createDefaultAnalyzer()) - .build(); - createIndexRequestBuilder.setSettings(indexSettings); - createIndexRequestBuilder.addMapping(recordDao.getType(), recordDao.createTypeMapping()); - createIndexRequestBuilder.addMapping(commentDao.getType(), commentDao.createTypeMapping()); - createIndexRequestBuilder.addMapping(categoryDao.getType(), categoryDao.createTypeMapping()); - createIndexRequestBuilder.execute().actionGet(); - - // Fill categories - fillRecordCategories(); - } - - public void fillRecordCategories() { - if (logger.isDebugEnabled()) { - logger.debug(String.format("[%s/%s] Fill data", INDEX, MarketIndexDao.CATEGORY_TYPE)); - } - - // Insert categories - categoryDao.bulkFromClasspathFile(CATEGORIES_BULK_CLASSPATH_FILE, - // Add order attribute - new AddSequenceAttributeHandler("order", "\\{.*\"name\".*\\}", 1)); - } - - - protected IndexTypeDao<?> createCategoryDao(final PluginSettings settings) { - return new AbstractIndexTypeDao(INDEX, MarketIndexDao.CATEGORY_TYPE) { - @Override - protected void createIndex() throws JsonProcessingException { - throw new TechnicalException("not implemented"); - } - - @Override - public XContentBuilder createTypeMapping() { - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject() - .startObject(getType()) - .startObject("properties") - - // name - .startObject("name") - .field("type", "string") - .field("analyzer", settings.getDefaultStringAnalyzer()) - .endObject() - - // parent - .startObject("parent") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", getIndex(), getType(), ioe.getMessage()), ioe); - } - } - }; - } -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketRecordDao.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketRecordDao.java deleted file mode 100644 index 7b4238a9dc5dfecb935e17012f688a1bf601d271..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketRecordDao.java +++ /dev/null @@ -1,9 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao.market; - -import org.duniter.elasticsearch.gchange.dao.RecordDao; - -/** - * Created by blavenie on 03/04/17. - */ -public interface MarketRecordDao extends RecordDao { -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketRecordDaoImpl.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketRecordDaoImpl.java deleted file mode 100644 index f5f855d7326e75f9ee15a6b61710835ff6349cdc..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/market/MarketRecordDaoImpl.java +++ /dev/null @@ -1,182 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao.market; - -import org.duniter.core.exception.TechnicalException; -import org.duniter.elasticsearch.gchange.PluginSettings; -import org.duniter.elasticsearch.gchange.dao.AbstractRecordDaoImpl; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; - -import java.io.IOException; - -/** - * Created by blavenie on 03/04/17. - */ -public class MarketRecordDaoImpl extends AbstractRecordDaoImpl implements MarketRecordDao { - - @Inject - public MarketRecordDaoImpl(PluginSettings pluginSettings) { - super(MarketIndexDao.INDEX, pluginSettings); - } - - @Override - public XContentBuilder createTypeMapping() { - String stringAnalyzer = pluginSettings.getDefaultStringAnalyzer(); - - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject().startObject(getType()) - .startObject("properties") - - // title - .startObject("title") - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - - // description - .startObject("description") - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - - // creationTime - .startObject("creationTime") - .field("type", "integer") - .endObject() - - // time - .startObject("time") - .field("type", "integer") - .endObject() - - // price - .startObject("price") - .field("type", "double") - .endObject() - - // price Unit - .startObject("unit") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // currency - .startObject("currency") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // issuer - .startObject("issuer") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // type (offer, need, ...) - .startObject("type") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - // location - .startObject("location") - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - - // geoPoint - .startObject("geoPoint") - .field("type", "geo_point") - .endObject() - - // thumbnail - .startObject("thumbnail") - .field("type", "attachment") - .startObject("fields") // src - .startObject("content") // title - .field("index", "no") - .endObject() - .startObject("title") // title - .field("type", "string") - .field("store", "no") - .endObject() - .startObject("author") // title - .field("store", "no") - .endObject() - .startObject("content_type") // title - .field("store", "yes") - .endObject() - .endObject() - .endObject() - - // pictures - .startObject("pictures") - .field("type", "nested") - .field("dynamic", "false") - .startObject("properties") - .startObject("file") // file - .field("type", "attachment") - .startObject("fields") - .startObject("content") // content - .field("index", "no") - .endObject() - .startObject("title") // title - .field("type", "string") - .field("store", "yes") - .field("analyzer", stringAnalyzer) - .endObject() - .startObject("author") // author - .field("type", "string") - .field("store", "no") - .endObject() - .startObject("content_type") // content_type - .field("store", "yes") - .endObject() - .endObject() - .endObject() - .endObject() - .endObject() - - // picturesCount - .startObject("picturesCount") - .field("type", "integer") - .endObject() - - // category - .startObject("category") - .field("type", "nested") - .field("dynamic", "false") - .startObject("properties") - .startObject("id") // id - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .startObject("parent") // parent - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - .startObject("name") // name - .field("type", "string") - .field("analyzer", stringAnalyzer) - .endObject() - .endObject() - .endObject() - - // tags - .startObject("tags") - .field("type", "completion") - .field("search_analyzer", "simple") - .field("analyzer", "simple") - .field("preserve_separators", "false") - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", getIndex(), getType(), ioe.getMessage()), ioe); - } - } -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryCommentDao.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryCommentDao.java deleted file mode 100644 index e70be6d0e371069d344a3ffc4a773a394d584758..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryCommentDao.java +++ /dev/null @@ -1,9 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao.registry; - -import org.duniter.elasticsearch.gchange.dao.CommentDao; - -/** - * Created by blavenie on 03/04/17. - */ -public interface RegistryCommentDao extends CommentDao { -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryCommentDaoImpl.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryCommentDaoImpl.java deleted file mode 100644 index c9d4b28ad4f71078b88ebc76800971cdeee2671b..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryCommentDaoImpl.java +++ /dev/null @@ -1,17 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao.registry; - -import org.duniter.elasticsearch.gchange.PluginSettings; -import org.duniter.elasticsearch.gchange.dao.AbstractCommentDaoImpl; -import org.elasticsearch.common.inject.Inject; - -/** - * Created by blavenie on 03/04/17. - */ -public class RegistryCommentDaoImpl extends AbstractCommentDaoImpl implements RegistryCommentDao { - - - @Inject - public RegistryCommentDaoImpl(PluginSettings pluginSettings) { - super(RegistryIndexDao.INDEX, pluginSettings); - } -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryIndexDao.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryIndexDao.java deleted file mode 100644 index f803d1733524c1e3e163c3a5d6ba6847afacffe3..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryIndexDao.java +++ /dev/null @@ -1,11 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao.registry; - -import org.duniter.elasticsearch.dao.IndexDao; - -/** - * Created by blavenie on 03/04/17. - */ -public interface RegistryIndexDao extends IndexDao<RegistryIndexDao> { - String INDEX = "registry"; - String CATEGORY_TYPE = "category"; -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryIndexDaoImpl.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryIndexDaoImpl.java deleted file mode 100644 index c62ded0962cd57c6008e7c2ba9aa88a1a9a85841..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryIndexDaoImpl.java +++ /dev/null @@ -1,117 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao.registry; - -import com.fasterxml.jackson.core.JsonProcessingException; -import org.duniter.core.exception.TechnicalException; -import org.duniter.elasticsearch.dao.AbstractIndexDao; -import org.duniter.elasticsearch.dao.AbstractIndexTypeDao; -import org.duniter.elasticsearch.dao.IndexDao; -import org.duniter.elasticsearch.dao.IndexTypeDao; -import org.duniter.elasticsearch.dao.handler.AddSequenceAttributeHandler; -import org.duniter.elasticsearch.gchange.PluginSettings; -import org.duniter.elasticsearch.gchange.dao.AbstractCommentDaoImpl; -import org.duniter.elasticsearch.gchange.dao.AbstractRecordDaoImpl; -import org.duniter.elasticsearch.gchange.dao.CommentDao; -import org.duniter.elasticsearch.gchange.dao.RecordDao; -import org.duniter.elasticsearch.gchange.service.RegistryService; -import org.elasticsearch.action.admin.indices.create.CreateIndexRequestBuilder; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.inject.Injector; -import org.elasticsearch.common.xcontent.XContentBuilder; -import org.elasticsearch.common.xcontent.XContentFactory; - -import java.io.IOException; - -/** - * Created by blavenie on 03/04/17. - */ -public class RegistryIndexDaoImpl extends AbstractIndexDao<RegistryIndexDao> implements RegistryIndexDao { - - - private static final String CATEGORIES_BULK_CLASSPATH_FILE = "registry-categories-bulk-insert.json"; - - private PluginSettings pluginSettings; - private IndexTypeDao<?> categoryDao; - private RecordDao recordDao; - private CommentDao commentDao; - - @Inject - public RegistryIndexDaoImpl(PluginSettings pluginSettings, RegistryRecordDao recordDao, RegistryCommentDao commentDao) { - super(RegistryIndexDao.INDEX); - - this.pluginSettings = pluginSettings; - this.commentDao = commentDao; - this.recordDao = recordDao; - this.categoryDao = createCategoryDao(pluginSettings); - } - - - @Override - protected void createIndex() throws JsonProcessingException { - logger.info(String.format("Creating index [%s]", INDEX)); - - CreateIndexRequestBuilder createIndexRequestBuilder = client.admin().indices().prepareCreate(INDEX); - org.elasticsearch.common.settings.Settings indexSettings = org.elasticsearch.common.settings.Settings.settingsBuilder() - .put("number_of_shards", 3) - .put("number_of_replicas", 1) - //.put("analyzer", createDefaultAnalyzer()) - .build(); - createIndexRequestBuilder.setSettings(indexSettings); - createIndexRequestBuilder.addMapping(recordDao.getType(), recordDao.createTypeMapping()); - createIndexRequestBuilder.addMapping(commentDao.getType(), commentDao.createTypeMapping()); - createIndexRequestBuilder.addMapping(categoryDao.getType(), categoryDao.createTypeMapping()); - createIndexRequestBuilder.execute().actionGet(); - - // Fill categories - fillRecordCategories(); - } - - public void fillRecordCategories() { - if (logger.isDebugEnabled()) { - logger.debug(String.format("[%s/%s] Fill data", INDEX, RegistryIndexDao.CATEGORY_TYPE)); - } - - // Insert categories - categoryDao.bulkFromClasspathFile(CATEGORIES_BULK_CLASSPATH_FILE, - // Add order attribute - new AddSequenceAttributeHandler("order", "\\{.*\"name\".*\\}", 1)); - } - - - protected IndexTypeDao<?> createCategoryDao(final PluginSettings settings) { - return new AbstractIndexTypeDao(INDEX, RegistryIndexDao.CATEGORY_TYPE) { - @Override - protected void createIndex() throws JsonProcessingException { - throw new TechnicalException("not implemented"); - } - - @Override - public XContentBuilder createTypeMapping() { - try { - XContentBuilder mapping = XContentFactory.jsonBuilder().startObject() - .startObject(getType()) - .startObject("properties") - - // name - .startObject("name") - .field("type", "string") - .field("analyzer", settings.getDefaultStringAnalyzer()) - .endObject() - - // parent - .startObject("parent") - .field("type", "string") - .field("index", "not_analyzed") - .endObject() - - .endObject() - .endObject().endObject(); - - return mapping; - } - catch(IOException ioe) { - throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", getIndex(), getType(), ioe.getMessage()), ioe); - } - } - }; - } -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryRecordDao.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryRecordDao.java deleted file mode 100644 index 163ebea84380c11af3bb8f4a977c97b84cc68979..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryRecordDao.java +++ /dev/null @@ -1,9 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao.registry; - -import org.duniter.elasticsearch.gchange.dao.RecordDao; - -/** - * Created by blavenie on 03/04/17. - */ -public interface RegistryRecordDao extends RecordDao { -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryRecordDaoImpl.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryRecordDaoImpl.java deleted file mode 100644 index b9ae257b326024b1d88d4d197b5729326c060f70..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/registry/RegistryRecordDaoImpl.java +++ /dev/null @@ -1,16 +0,0 @@ -package org.duniter.elasticsearch.gchange.dao.registry; - -import org.duniter.elasticsearch.gchange.PluginSettings; -import org.duniter.elasticsearch.gchange.dao.AbstractRecordDaoImpl; -import org.elasticsearch.common.inject.Inject; - -/** - * Created by blavenie on 03/04/17. - */ -public class RegistryRecordDaoImpl extends AbstractRecordDaoImpl implements RegistryRecordDao { - - @Inject - public RegistryRecordDaoImpl(PluginSettings pluginSettings) { - super(RegistryIndexDao.INDEX, pluginSettings); - } -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/model/event/GchangeEventCodes.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/model/event/GchangeEventCodes.java deleted file mode 100644 index 3ab5a4010ad98ba4e837ab0bf5fdea2c12f212e9..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/model/event/GchangeEventCodes.java +++ /dev/null @@ -1,34 +0,0 @@ -package org.duniter.elasticsearch.gchange.model.event; - -/* - * #%L - * Duniter4j :: ElasticSearch GChange plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -/** - * Created by blavenie on 01/12/16. - */ -public enum GchangeEventCodes { - - NEW_COMMENT, - UPDATE_COMMENT, - NEW_REPLY_COMMENT, - UPDATE_REPLY_COMMENT, -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/model/market/MarketRecord.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/model/market/MarketRecord.java deleted file mode 100644 index 33ab56540dfbbe73a6eb7293277526cabcf813ca..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/model/market/MarketRecord.java +++ /dev/null @@ -1,96 +0,0 @@ -package org.duniter.elasticsearch.gchange.model.market; - -/* - * #%L - * Duniter4j :: ElasticSearch GChange plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.core.client.model.elasticsearch.Record; - -import java.util.HashMap; -import java.util.Map; - -/** - * Created by blavenie on 01/12/16. - */ -public class MarketRecord extends Record{ - - public static final String PROPERTY_TITLE="title"; - public static final String PROPERTY_DESCRIPTION="description"; - public static final String PROPERTY_PRICE="price"; - public static final String PROPERTY_UNIT="unit"; - public static final String PROPERTY_CURRENCY="currency"; - public static final String PROPERTY_THUMBNAIL="thumbnail"; - - private String title; - private String description; - private Map<String, String> thumbnail = new HashMap<>(); - private Double price; - private String unit; - private String currency; - - public String getTitle() { - return title; - } - - public void setTitle(String title) { - this.title = title; - } - - public String getDescription() { - return description; - } - - public void setDescription(String description) { - this.description = description; - } - - public Double getPrice() { - return price; - } - - public void setPrice(Double price) { - this.price = price; - } - - public String getUnit() { - return unit; - } - - public void setUnit(String unit) { - this.unit = unit; - } - - public String getCurrency() { - return currency; - } - - public void setCurrency(String currency) { - this.currency = currency; - } - - public Map<String, String> getThumbnail() { - return thumbnail; - } - - public void setThumbnail(Map<String, String> thumbnail) { - this.thumbnail = thumbnail; - } -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/model/registry/RegistryRecord.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/model/registry/RegistryRecord.java deleted file mode 100644 index e04d30131b5d068909bff27913cd46a790bb5443..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/model/registry/RegistryRecord.java +++ /dev/null @@ -1,67 +0,0 @@ -package org.duniter.elasticsearch.gchange.model.registry; - -/* - * #%L - * Duniter4j :: ElasticSearch GChange plugin - * %% - * Copyright (C) 2014 - 2017 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.core.client.model.elasticsearch.Record; - -import java.util.HashMap; -import java.util.Map; - -/** - * Created by blavenie on 01/12/16. - */ -public class RegistryRecord extends Record{ - - public static final String PROPERTY_TITLE="title"; - public static final String PROPERTY_DESCRIPTION="description"; - public static final String PROPERTY_THUMBNAIL="thumbnail"; - - private String title; - private String description; - private Map<String, String> thumbnail = new HashMap<>(); - - public String getTitle() { - return title; - } - - public void setTitle(String title) { - this.title = title; - } - - public String getDescription() { - return description; - } - - public void setDescription(String description) { - this.description = description; - } - - public Map<String, String> getThumbnail() { - return thumbnail; - } - - public void setThumbnail(Map<String, String> thumbnail) { - this.thumbnail = thumbnail; - } - -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/RestModule.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/RestModule.java deleted file mode 100644 index d7ee15f72cfa2fa87b0bb298a7d2e6980d3f5aa6..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/RestModule.java +++ /dev/null @@ -1,50 +0,0 @@ -package org.duniter.elasticsearch.gchange.rest; - -/* - * #%L - * duniter4j-elasticsearch-plugin - * %% - * Copyright (C) 2014 - 2016 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.elasticsearch.gchange.rest.market.*; -import org.duniter.elasticsearch.gchange.rest.registry.*; -import org.elasticsearch.common.inject.AbstractModule; -import org.elasticsearch.common.inject.Module; - -public class RestModule extends AbstractModule implements Module { - - @Override protected void configure() { - - // Market - bind(RestMarketRecordIndexAction.class).asEagerSingleton(); - bind(RestMarketRecordUpdateAction.class).asEagerSingleton(); - bind(RestMarketCommentIndexAction.class).asEagerSingleton(); - bind(RestMarketCommentUpdateAction.class).asEagerSingleton(); - bind(RestMarketCategoryAction.class).asEagerSingleton(); - bind(RestMarketImageAction.class).asEagerSingleton(); - - // Registry - bind(RestRegistryRecordIndexAction.class).asEagerSingleton(); - bind(RestRegistryRecordUpdateAction.class).asEagerSingleton(); - bind(RestRegistryCommentIndexAction.class).asEagerSingleton(); - bind(RestRegistryCommentUpdateAction.class).asEagerSingleton(); - bind(RestRegistryCategoryAction.class).asEagerSingleton(); - bind(RestRegistryImageAction.class).asEagerSingleton(); - } -} \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketCommentIndexAction.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketCommentIndexAction.java deleted file mode 100644 index e0a39533181d76af21abf5a5c1234d73323630b1..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketCommentIndexAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.gchange.rest.market; - -/* - * #%L - * duniter4j-elasticsearch-plugin - * %% - * Copyright (C) 2014 - 2016 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.elasticsearch.gchange.dao.market.MarketCommentDao; -import org.duniter.elasticsearch.gchange.dao.market.MarketIndexDao; -import org.duniter.elasticsearch.rest.AbstractRestPostIndexAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.gchange.service.MarketService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestMarketCommentIndexAction extends AbstractRestPostIndexAction { - - @Inject - public RestMarketCommentIndexAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - MarketService service) { - super(settings, controller, client, securityController, - MarketIndexDao.INDEX, MarketCommentDao.TYPE, - json -> service.indexCommentFromJson(json)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketCommentUpdateAction.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketCommentUpdateAction.java deleted file mode 100644 index 0052f6aafa158b32eacd1d607e9e0bb8c8bb68a3..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketCommentUpdateAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.gchange.rest.market; - -/* - * #%L - * duniter4j-elasticsearch-plugin - * %% - * Copyright (C) 2014 - 2016 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.elasticsearch.gchange.dao.market.MarketCommentDao; -import org.duniter.elasticsearch.gchange.dao.market.MarketIndexDao; -import org.duniter.elasticsearch.rest.AbstractRestPostUpdateAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.gchange.service.MarketService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestMarketCommentUpdateAction extends AbstractRestPostUpdateAction { - - @Inject - public RestMarketCommentUpdateAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - MarketService service) { - super(settings, controller, client, securityController, - MarketIndexDao.INDEX, MarketCommentDao.TYPE, - (id, json) -> service.updateCommentFromJson(id, json)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketImageAction.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketImageAction.java deleted file mode 100644 index dbc054d50def487ac70f8652d8ca7dd3d3f85163..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketImageAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.gchange.rest.market; - -/* - * #%L - * duniter4j-elasticsearch-plugin - * %% - * Copyright (C) 2014 - 2016 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.elasticsearch.gchange.dao.market.MarketIndexDao; -import org.duniter.elasticsearch.gchange.dao.market.MarketRecordDao; -import org.duniter.elasticsearch.gchange.model.market.MarketRecord; -import org.duniter.elasticsearch.gchange.model.registry.RegistryRecord; -import org.duniter.elasticsearch.gchange.service.MarketService; -import org.duniter.elasticsearch.gchange.service.RegistryService; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.rest.RestRequest; - -public class RestMarketImageAction { - - @Inject - public RestMarketImageAction(RestSecurityController securityController) { - - // Allow to get thumbnail - securityController.allowImageAttachment(MarketIndexDao.INDEX, MarketRecordDao.TYPE, MarketRecord.PROPERTY_THUMBNAIL); - - // TODO : allow to get pictures - } -} \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketRecordIndexAction.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketRecordIndexAction.java deleted file mode 100644 index 7af1deee1241626079c0dc5238852db60bd6cac1..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketRecordIndexAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.gchange.rest.market; - -/* - * #%L - * duniter4j-elasticsearch-plugin - * %% - * Copyright (C) 2014 - 2016 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.elasticsearch.gchange.dao.market.MarketIndexDao; -import org.duniter.elasticsearch.gchange.dao.market.MarketRecordDao; -import org.duniter.elasticsearch.rest.AbstractRestPostIndexAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.gchange.service.MarketService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestMarketRecordIndexAction extends AbstractRestPostIndexAction { - - @Inject - public RestMarketRecordIndexAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - MarketService service) { - super(settings, controller, client, securityController, - MarketIndexDao.INDEX, MarketRecordDao.TYPE, - json -> service.indexRecordFromJson(json)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketRecordUpdateAction.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketRecordUpdateAction.java deleted file mode 100644 index 7fd0a76e0907402d6a426f24a1962d31533734ff..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketRecordUpdateAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.gchange.rest.market; - -/* - * #%L - * duniter4j-elasticsearch-plugin - * %% - * Copyright (C) 2014 - 2016 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.elasticsearch.gchange.dao.market.MarketIndexDao; -import org.duniter.elasticsearch.gchange.dao.market.MarketRecordDao; -import org.duniter.elasticsearch.gchange.service.MarketService; -import org.duniter.elasticsearch.rest.AbstractRestPostUpdateAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestMarketRecordUpdateAction extends AbstractRestPostUpdateAction { - - @Inject - public RestMarketRecordUpdateAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - MarketService service) { - super(settings, controller, client, securityController, - MarketIndexDao.INDEX, MarketRecordDao.TYPE, - (id, json) -> service.updateRecordFromJson(id, json)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryCommentUpdateAction.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryCommentUpdateAction.java deleted file mode 100644 index e724f60bf5cef85dc8c6f4dc66c2c51040466f18..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryCommentUpdateAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.gchange.rest.registry; - -/* - * #%L - * duniter4j-elasticsearch-plugin - * %% - * Copyright (C) 2014 - 2016 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.elasticsearch.gchange.dao.registry.RegistryCommentDao; -import org.duniter.elasticsearch.gchange.dao.registry.RegistryIndexDao; -import org.duniter.elasticsearch.gchange.service.RegistryService; -import org.duniter.elasticsearch.rest.AbstractRestPostUpdateAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestRegistryCommentUpdateAction extends AbstractRestPostUpdateAction { - - @Inject - public RestRegistryCommentUpdateAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - RegistryService service) { - super(settings, controller, client, securityController, - RegistryIndexDao.INDEX, RegistryCommentDao.TYPE, - (id, json) -> service.updateCommentFromJson(id, json)); - } - -} \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryImageAction.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryImageAction.java deleted file mode 100644 index 197ba6066dd9004c64e43762387ae459a583325e..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryImageAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.gchange.rest.registry; - -/* - * #%L - * duniter4j-elasticsearch-plugin - * %% - * Copyright (C) 2014 - 2016 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.elasticsearch.gchange.dao.registry.RegistryCommentDao; -import org.duniter.elasticsearch.gchange.dao.registry.RegistryIndexDao; -import org.duniter.elasticsearch.gchange.dao.registry.RegistryRecordDao; -import org.duniter.elasticsearch.gchange.model.registry.RegistryRecord; -import org.duniter.elasticsearch.gchange.service.RegistryService; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.user.service.UserService; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.rest.RestRequest; - -public class RestRegistryImageAction { - - @Inject - public RestRegistryImageAction(RestSecurityController securityController) { - - // Allow to get thumbnail - securityController.allowImageAttachment(RegistryIndexDao.INDEX, RegistryRecordDao.TYPE, RegistryRecord.PROPERTY_THUMBNAIL); - - // TODO : allow to get pictures - } -} \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryRecordIndexAction.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryRecordIndexAction.java deleted file mode 100644 index 297562526cf4b88ee97016a223618902e545545e..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryRecordIndexAction.java +++ /dev/null @@ -1,45 +0,0 @@ -package org.duniter.elasticsearch.gchange.rest.registry; - -/* - * #%L - * duniter4j-elasticsearch-plugin - * %% - * Copyright (C) 2014 - 2016 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.elasticsearch.gchange.dao.registry.RegistryIndexDao; -import org.duniter.elasticsearch.gchange.dao.registry.RegistryRecordDao; -import org.duniter.elasticsearch.rest.AbstractRestPostIndexAction; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.gchange.service.RegistryService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.elasticsearch.rest.RestController; - -public class RestRegistryRecordIndexAction extends AbstractRestPostIndexAction { - - - @Inject - public RestRegistryRecordIndexAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - RegistryService service) { - super(settings, controller, client, securityController, - RegistryIndexDao.INDEX, RegistryRecordDao.TYPE, - json -> service.indexRecordFromJson(json)); - } -} \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/CommentUserEventService.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/CommentUserEventService.java deleted file mode 100644 index 76da0a6e0267fe649aab2a9eba646a5d7f22f8c0..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/CommentUserEventService.java +++ /dev/null @@ -1,273 +0,0 @@ -package org.duniter.elasticsearch.gchange.service; - -/* - * #%L - * UCoin Java Client :: Core API - * %% - * Copyright (C) 2014 - 2015 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - - -import com.fasterxml.jackson.core.JsonProcessingException; -import com.fasterxml.jackson.databind.DeserializationFeature; -import com.google.common.collect.ImmutableList; -import org.apache.commons.collections4.MapUtils; -import org.duniter.core.client.model.ModelUtils; -import org.duniter.core.client.model.elasticsearch.RecordComment; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.service.CryptoService; -import org.duniter.core.util.StringUtils; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.gchange.PluginSettings; -import org.duniter.elasticsearch.gchange.dao.RecordDao; -import org.duniter.elasticsearch.gchange.dao.market.MarketCommentDao; -import org.duniter.elasticsearch.gchange.dao.market.MarketIndexDao; -import org.duniter.elasticsearch.gchange.dao.registry.RegistryCommentDao; -import org.duniter.elasticsearch.gchange.dao.registry.RegistryIndexDao; -import org.duniter.elasticsearch.gchange.model.event.GchangeEventCodes; -import org.duniter.elasticsearch.gchange.model.market.MarketRecord; -import org.duniter.elasticsearch.service.changes.ChangeEvent; -import org.duniter.elasticsearch.service.changes.ChangeService; -import org.duniter.elasticsearch.service.changes.ChangeSource; -import org.duniter.elasticsearch.user.model.UserEvent; -import org.duniter.elasticsearch.user.service.UserEventService; -import org.duniter.elasticsearch.user.service.UserService; -import org.elasticsearch.common.inject.Inject; -import org.nuiton.i18n.I18n; - -import java.io.IOException; -import java.util.Collection; -import java.util.List; -import java.util.Map; - -/** - * Created by Benoit on 30/03/2015. - */ -public class CommentUserEventService extends AbstractService implements ChangeService.ChangeListener { - - static { - I18n.n("duniter.market.error.comment.recordNotFound"); - I18n.n("duniter.market.event.newComment"); - I18n.n("duniter.market.event.updateComment"); - I18n.n("duniter.market.event.newReplyComment"); - I18n.n("duniter.market.event.updateReplyComment"); - - I18n.n("duniter.registry.error.comment.recordNotFound"); - I18n.n("duniter.registry.event.newComment"); - I18n.n("duniter.registry.event.updateComment"); - I18n.n("duniter.registry.event.newReplyComment"); - I18n.n("duniter.registry.event.updateReplyComment"); - } - - private final UserService userService; - private final UserEventService userEventService; - private final List<ChangeSource> changeListenSources; - private final String recordType; - private final boolean trace; - - @Inject - public CommentUserEventService(Duniter4jClient client, - PluginSettings settings, - CryptoService cryptoService, - UserService userService, - UserEventService userEventService) { - super("duniter.user.event.comment", client, settings, cryptoService); - this.userService = userService; - this.userEventService = userEventService; - objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false); - this.changeListenSources = ImmutableList.of( - new ChangeSource(MarketIndexDao.INDEX, MarketCommentDao.TYPE), - new ChangeSource(RegistryIndexDao.INDEX, RegistryCommentDao.TYPE)); - ChangeService.registerListener(this); - - this.trace = logger.isTraceEnabled(); - - this.recordType = RecordDao.TYPE; - } - - @Override - public String getId() { - return "duniter.user.event.comment"; - } - - @Override - public void onChange(ChangeEvent change) { - - RecordComment comment; - - switch (change.getOperation()) { - case CREATE: - comment = readComment(change); - if (comment != null) { - processCreateComment(change.getIndex(), change.getType(), change.getId(), comment); - } - break; - case INDEX: - comment = readComment(change); - if (comment != null) { - processUpdateComment(change.getIndex(), change.getType(), change.getId(), comment); - } - break; - - // on DELETE : remove user event on block (using link - case DELETE: - processCommentDelete(change); - break; - } - - } - - @Override - public Collection<ChangeSource> getChangeSources() { - return changeListenSources; - } - - /* -- internal method -- */ - - /** - * Send notification from a new comment - * - * @param index - * @param type - * @param commentId - * @param comment - */ - private void processCreateComment(String index, String type, String commentId, RecordComment comment) { - - processUpdateOrCreateComment(index, type, commentId, comment, - GchangeEventCodes.NEW_COMMENT, String.format("duniter.%s.event.newComment", index.toLowerCase()), - GchangeEventCodes.NEW_REPLY_COMMENT, String.format("duniter.%s.event.newReplyComment", index.toLowerCase())); - } - - /** - * Same as processCreateComment(), but with other code and message. - * - * @param index - * @param type - * @param commentId - * @param comment - */ - private void processUpdateComment(String index, String type, String commentId, RecordComment comment) { - - processUpdateOrCreateComment(index, type, commentId, comment, - GchangeEventCodes.UPDATE_COMMENT, String.format("duniter.%s.event.updateComment", index.toLowerCase()), - GchangeEventCodes.UPDATE_REPLY_COMMENT, String.format("duniter.%s.event.updateReplyComment", index.toLowerCase())); - } - - - /** - * Same as processCreateComment(), but with other code and message. - * - * @param index - * @param type - * @param commentId - * @param comment - */ - private void processUpdateOrCreateComment(String index, String type, String commentId, RecordComment comment, - GchangeEventCodes eventCodeForRecordIssuer, String messageKeyForRecordIssuer, - GchangeEventCodes eventCodeForParentCommentIssuer, String messageKeyForParentCommentIssuer) { - // Get record issuer - String recordId = comment.getRecord(); - Map<String, Object> record = client.getFieldsById(index, this.recordType, recordId, - MarketRecord.PROPERTY_TITLE, MarketRecord.PROPERTY_ISSUER); - - // Record not found : nothing to emit - if (MapUtils.isEmpty(record)) { - logger.warn(I18n.t(String.format("duniter.%s.error.comment.recordNotFound", index.toLowerCase()), recordId)); - return; - } - - // Fetch record info - String recordIssuer = record.get(MarketRecord.PROPERTY_ISSUER).toString(); - String recordTitle = record.get(MarketRecord.PROPERTY_TITLE).toString(); - - // Get comment issuer title - String issuer = comment.getIssuer(); - String issuerTitle = userService.getProfileTitle(issuer); - - // Notify issuer of record (is not same as comment writer) - if (!issuer.equals(recordIssuer)) { - userEventService.notifyUser( - UserEvent.newBuilder(UserEvent.EventType.INFO, eventCodeForRecordIssuer.name()) - .setMessage( - messageKeyForRecordIssuer, - issuer, - issuerTitle != null ? issuerTitle : ModelUtils.minifyPubkey(issuer), - recordTitle - ) - .setRecipient(recordIssuer) - .setReference(index, recordType, recordId) - .setReferenceAnchor(commentId) - .setTime(comment.getTime()) - .build()); - } - - // Notify comment is a reply to another comment - if (StringUtils.isNotBlank(comment.getReplyTo())) { - - String parentCommentIssuer = client.getTypedFieldById(index, type, comment.getReplyTo(), RecordComment.PROPERTY_ISSUER); - - if (StringUtils.isNotBlank(parentCommentIssuer) && - !issuer.equals(parentCommentIssuer) && - !recordIssuer.equals(parentCommentIssuer)) { - - userEventService.notifyUser( - UserEvent.newBuilder(UserEvent.EventType.INFO, eventCodeForParentCommentIssuer.name()) - .setMessage( - messageKeyForParentCommentIssuer, - issuer, - issuerTitle != null ? issuerTitle : ModelUtils.minifyPubkey(issuer), - recordTitle - ) - .setRecipient(parentCommentIssuer) - .setReference(index, recordType, recordId) - .setReferenceAnchor(commentId) - /*.setTime(comment.getTime()) - DO NOT set time, has the comment time is NOT the update time*/ - .build()); - } - } - - } - - private void processCommentDelete(ChangeEvent change) { - if (change.getId() == null) return; - - // Delete events that reference this block - userEventService.deleteEventsByReference(new UserEvent.Reference(change.getIndex(), change.getType(), change.getId())); - } - - private RecordComment readComment(ChangeEvent change) { - try { - if (change.getSource() != null) { - return objectMapper.readValue(change.getSource().streamInput(), RecordComment.class); - } - return null; - } catch (JsonProcessingException e) { - if (trace) { - logger.warn(String.format("Bad format for comment [%s]: %s. Skip this comment", change.getId(), e.getMessage()), e); - } - else { - logger.warn(String.format("Bad format for comment [%s]: %s. Skip this comment", change.getId(), e.getMessage())); - } - return null; - } - catch (IOException e) { - throw new TechnicalException(String.format("Unable to parse received comment %s", change.getId()), e); - } - } -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/MarketService.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/MarketService.java deleted file mode 100644 index 78581d99962e758a6ee1953867c8e3f7a6945816..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/MarketService.java +++ /dev/null @@ -1,150 +0,0 @@ -package org.duniter.elasticsearch.gchange.service; - -/* - * #%L - * UCoin Java Client :: Core API - * %% - * Copyright (C) 2014 - 2015 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - - -import com.fasterxml.jackson.databind.JsonNode; -import org.duniter.core.client.model.elasticsearch.RecordComment; -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.exception.NotFoundException; -import org.duniter.elasticsearch.gchange.PluginSettings; -import org.duniter.elasticsearch.gchange.dao.market.MarketCommentDao; -import org.duniter.elasticsearch.gchange.dao.market.MarketIndexDao; -import org.duniter.elasticsearch.gchange.dao.market.MarketRecordDao; -import org.duniter.elasticsearch.user.service.HistoryService; -import org.elasticsearch.client.Client; -import org.elasticsearch.common.inject.Inject; - -/** - * Created by Benoit on 30/03/2015. - */ -public class MarketService extends AbstractService { - - private MarketIndexDao indexDao; - private MarketRecordDao recordDao; - private MarketCommentDao commentDao; - private HistoryService historyService; - - @Inject - public MarketService(Duniter4jClient client, PluginSettings settings, - CryptoService cryptoService, - HistoryService historyService, - MarketIndexDao indexDao, - MarketCommentDao commentDao, - MarketRecordDao recordDao - ) { - super("gchange.service.market", client, settings, cryptoService); - this.indexDao = indexDao; - this.commentDao = commentDao; - this.recordDao = recordDao; - - this.historyService = historyService; - } - - - /** - * Create index need for blockchain registry, if need - */ - public MarketService createIndexIfNotExists() { - indexDao.createIndexIfNotExists(); - return this; - } - - public MarketService deleteIndex() { - indexDao.deleteIndex(); - return this; - } - - - public String indexRecordFromJson(String json) { - JsonNode actualObj = readAndVerifyIssuerSignature(json); - String issuer = getIssuer(actualObj); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Indexing a %s from issuer [%s]", recordDao.getType(), issuer.substring(0, 8))); - } - - return recordDao.create(json); - } - - public void updateRecordFromJson(String id, String json) { - JsonNode actualObj = readAndVerifyIssuerSignature(json); - String issuer = getIssuer(actualObj); - - // Check same document issuer - recordDao.checkSameDocumentIssuer(id, issuer); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Updating %s [%s] from issuer [%s]", recordDao.getType(), id, issuer.substring(0, 8))); - } - - recordDao.update(id, json); - } - - public String indexCommentFromJson(String json) { - JsonNode commentObj = readAndVerifyIssuerSignature(json); - String issuer = getMandatoryField(commentObj, RecordComment.PROPERTY_ISSUER).asText(); - - // Check the record document exists - String recordId = getMandatoryField(commentObj, RecordComment.PROPERTY_RECORD).asText(); - checkRecordExistsOrDeleted(recordId); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Indexing a %s from issuer [%s]", commentDao.getType(), issuer.substring(0, 8))); - } - return commentDao.create(json); - } - - public void updateCommentFromJson(String id, String json) { - JsonNode commentObj = readAndVerifyIssuerSignature(json); - - // Check the record document exists - String recordId = getMandatoryField(commentObj, RecordComment.PROPERTY_RECORD).asText(); - checkRecordExistsOrDeleted(recordId); - - if (logger.isDebugEnabled()) { - String issuer = getMandatoryField(commentObj, RecordComment.PROPERTY_ISSUER).asText(); - logger.debug(String.format("[%s] Indexing a %s from issuer [%s] on [%s]", commentDao.getType(), commentDao.getType(), issuer.substring(0, 8))); - } - - commentDao.update(id, json); - } - - - /* -- Internal methods -- */ - - // Check the record document exists (or has been deleted) - private void checkRecordExistsOrDeleted(String id) { - boolean recordExists; - try { - recordExists = recordDao.isExists(id); - } catch (NotFoundException e) { - // Check if exists in delete history - recordExists = historyService.existsInDeleteHistory(recordDao.getIndex(), recordDao.getType(), id); - } - if (!recordExists) { - throw new NotFoundException(String.format("Comment refers a non-existent document [%s/%s/%s].", recordDao.getIndex(), recordDao.getType(), id)); - } - } -} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/RegistryService.java b/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/RegistryService.java deleted file mode 100644 index ca51cb87372a987f264b201909c9c76dfdb9dd93..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/RegistryService.java +++ /dev/null @@ -1,147 +0,0 @@ -package org.duniter.elasticsearch.gchange.service; - -/* - * #%L - * UCoin Java Client :: Core API - * %% - * Copyright (C) 2014 - 2015 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - - -import com.fasterxml.jackson.databind.JsonNode; -import org.duniter.core.client.model.elasticsearch.RecordComment; -import org.duniter.core.service.CryptoService; -import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.exception.NotFoundException; -import org.duniter.elasticsearch.gchange.PluginSettings; -import org.duniter.elasticsearch.gchange.dao.registry.RegistryCommentDao; -import org.duniter.elasticsearch.gchange.dao.registry.RegistryIndexDao; -import org.duniter.elasticsearch.gchange.dao.registry.RegistryRecordDao; -import org.duniter.elasticsearch.user.service.HistoryService; -import org.elasticsearch.common.inject.Inject; - -/** - * Created by Benoit on 30/03/2015. - */ -public class RegistryService extends AbstractService { - - private RegistryIndexDao indexDao; - private RegistryRecordDao recordDao; - private RegistryCommentDao commentDao; - private HistoryService historyService; - - @Inject - public RegistryService(Duniter4jClient client, - PluginSettings settings, - CryptoService cryptoService, - HistoryService historyService, - RegistryIndexDao registryIndexDao, - RegistryCommentDao commentDao, - RegistryRecordDao recordDao) { - super("gchange.service.registry", client, settings, cryptoService); - this.indexDao = registryIndexDao; - this.commentDao = commentDao; - this.recordDao = recordDao; - this.historyService = historyService; - } - - /** - * Create index need for blockchain registry, if need - */ - public RegistryService createIndexIfNotExists() { - indexDao.createIndexIfNotExists(); - return this; - } - - public RegistryService deleteIndex() { - indexDao.deleteIndex(); - return this; - } - - public String indexRecordFromJson(String json) { - JsonNode actualObj = readAndVerifyIssuerSignature(json); - String issuer = getIssuer(actualObj); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Indexing a %s from issuer [%s]", recordDao.getType(), issuer.substring(0, 8))); - } - - return recordDao.create(json); - } - - public void updateRecordFromJson(String id, String json) { - JsonNode actualObj = readAndVerifyIssuerSignature(json); - String issuer = getIssuer(actualObj); - - // Check same document issuer - recordDao.checkSameDocumentIssuer(id, issuer); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Updating %s [%s] from issuer [%s]", recordDao.getType(), id, issuer.substring(0, 8))); - } - - recordDao.update(id, json); - } - - public String indexCommentFromJson(String json) { - JsonNode commentObj = readAndVerifyIssuerSignature(json); - String issuer = getMandatoryField(commentObj, RecordComment.PROPERTY_ISSUER).asText(); - - // Check the record document exists - String recordId = getMandatoryField(commentObj, RecordComment.PROPERTY_RECORD).asText(); - checkRecordExistsOrDeleted(recordId); - - if (logger.isDebugEnabled()) { - logger.debug(String.format("Indexing a %s from issuer [%s]", commentDao.getType(), issuer.substring(0, 8))); - } - return commentDao.create(json); - } - - public void updateCommentFromJson(String id, String json) { - JsonNode commentObj = readAndVerifyIssuerSignature(json); - - // Check the record document exists - String recordId = getMandatoryField(commentObj, RecordComment.PROPERTY_RECORD).asText(); - checkRecordExistsOrDeleted(recordId); - - if (logger.isDebugEnabled()) { - String issuer = getMandatoryField(commentObj, RecordComment.PROPERTY_ISSUER).asText(); - logger.debug(String.format("[%s] Indexing a %s from issuer [%s] on [%s]", commentDao.getType(), commentDao.getType(), issuer.substring(0, 8))); - } - - commentDao.update(id, json); - } - - - /* -- Internal methods -- */ - - // Check the record document exists (or has been deleted) - private void checkRecordExistsOrDeleted(String id) { - boolean recordExists; - try { - recordExists = recordDao.isExists(id); - } catch (NotFoundException e) { - // Check if exists in delete history - recordExists = historyService.existsInDeleteHistory(recordDao.getIndex(), recordDao.getType(), id); - } - if (!recordExists) { - throw new NotFoundException(String.format("Comment refers a non-existent document [%s/%s/%s].", recordDao.getIndex(), recordDao.getType(), id)); - } - } - -} diff --git a/duniter4j-es-gchange/src/main/misc/cities-fr.geoJson.txt b/duniter4j-es-gchange/src/main/misc/cities-fr.geoJson.txt deleted file mode 100644 index bb4b1bd422f6de2c147bba0ab39ce74aadf01f6c..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/misc/cities-fr.geoJson.txt +++ /dev/null @@ -1,8 +0,0 @@ -Les données Francaise des communes provient de ce fichier : -http://public.opendatasoft.com/explore/dataset/geoflar-communes-2015/export/ - ->> Cliquer sur Export > GeoJSON - - -Ou directement via : - http://public.opendatasoft.com/explore/dataset/geoflar-communes-2015/download/?format=geojson&timezone=Europe/Berlin \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/misc/registry-categories-naf2008_liste_n5.ods b/duniter4j-es-gchange/src/main/misc/registry-categories-naf2008_liste_n5.ods deleted file mode 100644 index acc6ea091339d82dacf7533c99e3cf9876599984..0000000000000000000000000000000000000000 Binary files a/duniter4j-es-gchange/src/main/misc/registry-categories-naf2008_liste_n5.ods and /dev/null differ diff --git a/duniter4j-es-gchange/src/main/resources/i18n/duniter4j-es-gchange_en_GB.properties b/duniter4j-es-gchange/src/main/resources/i18n/duniter4j-es-gchange_en_GB.properties deleted file mode 100644 index dd29729a2e711db27e13cfc3f804fa441d74b608..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/resources/i18n/duniter4j-es-gchange_en_GB.properties +++ /dev/null @@ -1,12 +0,0 @@ -duniter.event.NODE_BMA_DOWN= -duniter.event.NODE_BMA_UP= -duniter.market.error.comment.recordNotFound=Ad [%s] referenced by comment not found -duniter.market.event.newComment= -duniter.market.event.newReplyComment= -duniter.market.event.updateComment= -duniter.market.event.updateReplyComment= -duniter.registry.error.comment.recordNotFound=Record [%s] referenced by comment not found -duniter.registry.event.newComment= -duniter.registry.event.newReplyComment= -duniter.registry.event.updateComment= -duniter.registry.event.updateReplyComment= diff --git a/duniter4j-es-gchange/src/main/resources/i18n/duniter4j-es-gchange_fr_FR.properties b/duniter4j-es-gchange/src/main/resources/i18n/duniter4j-es-gchange_fr_FR.properties deleted file mode 100644 index 40d7fb34f50c9be5947023487bda6434025fa243..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/resources/i18n/duniter4j-es-gchange_fr_FR.properties +++ /dev/null @@ -1,12 +0,0 @@ -duniter.event.NODE_BMA_DOWN= -duniter.event.NODE_BMA_UP= -duniter.market.error.comment.recordNotFound=L'annonce [%s] référencée par le commentaire n'existe pas. -duniter.market.event.newComment=%2$s a commenté votre annonce '%3$s' -duniter.market.event.newReplyComment=%2$s a répondu à votre commentaire sur l'annonce '%3$s' -duniter.market.event.updateComment=%2$s a mise à jour son commentaire sur votre annonce '%3$s' -duniter.market.event.updateReplyComment=%2$s a modifié sa réponse à votre commentaire, sur l'annonce '%3$s' -duniter.registry.error.comment.recordNotFound=Le référencement [%s] référencée par le commentaire n'existe pas. -duniter.registry.event.newComment=%2$s a commenté votre référencement '%3$s' -duniter.registry.event.newReplyComment=%2$s a répondu à votre commentaire sur le référencement '%3$s' -duniter.registry.event.updateComment=%2$s a modifié son commentaire sur votre référencement '%3$s' -duniter.registry.event.updateReplyComment=%2$s a modifié sa réponse à votre commentaire, sur le référencement '%3$s' diff --git a/duniter4j-es-gchange/src/main/resources/market-categories-bulk-insert.json b/duniter4j-es-gchange/src/main/resources/market-categories-bulk-insert.json deleted file mode 100644 index c9527a96a6f1a8d350067029670a10b4c1f2de74..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/resources/market-categories-bulk-insert.json +++ /dev/null @@ -1,151 +0,0 @@ -{ "index": { "_id": "cat71"}} -{ "name": "Emploi" , "parent": null} -{ "index": { "_id": "cat33"}} -{ "name": "Offres d'emploi", "parent": "cat71" } - -{ "index": { "_id": "cat1" }} -{ "name": "Véhicules" , "parent": null} -{ "index": { "_id": "cat2" }} -{ "name": "Voitures" , "parent": "cat2" } -{ "index": { "_id": "cat3" }} -{ "name": "Motos" , "parent": "cat3" } -{ "index": { "_id": "cat4" }} -{ "name": "Caravaning" , "parent": "cat3" } -{ "index": { "_id": "cat5" }} -{ "name": "Utilitaires" , "parent": "cat3" } -{ "index": { "_id": "cat6" }} -{ "name": "Equipement Auto" , "parent": "cat3" } -{ "index": { "_id": "cat44" }} -{ "name": "Equipement Moto" , "parent": "cat3" } -{ "index": { "_id": "cat50" }} -{ "name": "Equipement Caravaning" , "parent": "cat3" } -{ "index": { "_id": "cat7" }} -{ "name": "Nautisme" , "parent": "cat3" } -{ "index": { "_id": "cat51" }} -{ "name": "Equipement Nautisme" , "parent": "cat3" } - -{ "index": { "_id": "cat8" }} -{ "name": "Immobilier" , "parent": null} -{ "index": { "_id": "cat9" }} -{ "name": "Ventes immobilières" , "parent": "cat8" } -{ "index": { "_id": "cat10" }} -{ "name": "Locations" , "parent": "cat8" } -{ "index": { "_id": "cat11" }} -{ "name": "Colocations" , "parent": "cat8" } -{ "index": { "_id": "cat13" }} -{ "name": "Bureaux & Commerces" , "parent": "cat8" } - -{ "index": { "_id": "cat66" }} -{ "name": "Vacances" , "parent": null} -{ "index": { "_id": "cat12" }} -{ "name": "Locations & Gîtes" , "parent": "cat66" } -{ "index": { "_id": "cat67" }} -{ "name": "Chambres d'hôtes" , "parent": "cat66" } -{ "index": { "_id": "cat68" }} -{ "name": "Campings" , "parent": "cat66" } -{ "index": { "_id": "cat69" }} -{ "name": "Hôtels" , "parent": "cat66" } -{ "index": { "_id": "cat70" }} -{ "name": "Hébergements insolites" , "parent": "cat66" } - -{ "index": { "_id": "cat14" }} -{ "name": "Multimédia" , "parent": null} -{ "index": { "_id": "cat15" }} -{ "name": "Informatique" , "parent": "cat14" } -{ "index": { "_id": "cat43" }} -{ "name": "Consoles & Jeux vidéo" , "parent": "cat14" } -{ "index": { "_id": "cat16" }} -{ "name": "Image & Son" , "parent": "cat14" } -{ "index": { "_id": "cat17" }} -{ "name": "Téléphonie" , "parent": "cat14" } - -{ "index": { "_id": "cat18" }} -{ "name": "Maison" , "parent": null} -{ "index": { "_id": "cat19" }} -{ "name": "Ameublement" , "parent": "cat18" } -{ "index": { "_id": "cat20" }} -{ "name": "Electroménager" , "parent": "cat18" } -{ "index": { "_id": "cat45" }} -{ "name": "Arts de la table" , "parent": "cat18" } -{ "index": { "_id": "cat39" }} -{ "name": "Décoration" , "parent": "cat18" } -{ "index": { "_id": "cat46" }} -{ "name": "Linge de maison" , "parent": "cat18" } -{ "index": { "_id": "cat21" }} -{ "name": "Bricolage" , "parent": "cat18" } -{ "index": { "_id": "cat52" }} -{ "name": "Jardinage" , "parent": "cat18" } -{ "index": { "_id": "cat22" }} -{ "name": "Vêtements" , "parent": "cat18" } -{ "index": { "_id": "cat53" }} -{ "name": "Chaussures" , "parent": "cat18" } -{ "index": { "_id": "cat47" }} -{ "name": "Accessoires & Bagagerie" , "parent": "cat18" } -{ "index": { "_id": "cat42" }} -{ "name": "Montres & Bijoux" , "parent": "cat18" } -{ "index": { "_id": "cat23" }} -{ "name": "Equipement bébé" , "parent": "cat18" } -{ "index": { "_id": "cat54" }} -{ "name": "Vêtements bébé" , "parent": "cat18" } - -{ "index": { "_id": "cat24" }} -{ "name": "Loisirs" , "parent": null} -{ "index": { "_id": "cat25" }} -{ "name": "DVD / Films" , "parent": "cat24" } -{ "index": { "_id": "cat26" }} -{ "name": "CD / Musique" , "parent": "cat24" } -{ "index": { "_id": "cat27" }} -{ "name": "Livres" , "parent": "cat24" } -{ "index": { "_id": "cat28" }} -{ "name": "Animaux" , "parent": "cat24" } -{ "index": { "_id": "cat55" }} -{ "name": "Vélos" , "parent": "cat24" } -{ "index": { "_id": "cat29" }} -{ "name": "Sports & Hobbies" } -{ "index": { "_id": "cat30" }} -{ "name": "Instruments de musique" , "parent": "cat24" } -{ "index": { "_id": "cat40" }} -{ "name": "Collection" , "parent": "cat24" } -{ "index": { "_id": "cat41" }} -{ "name": "Jeux & Jouets" , "parent": "cat24" } -{ "index": { "_id": "cat48" }} -{ "name": "Vins & Gastronomie" , "parent": "cat24" } - -{ "index": { "_id": "cat56" }} -{ "name": "Matériel professionnel" , "parent": null} -{ "index": { "_id": "cat57" }} -{ "name": "Matériel Agricole" , "parent": "cat56" } -{ "index": { "_id": "cat58" }} -{ "name": "Transport - Manutention" , "parent": "cat56" } -{ "index": { "_id": "cat59" }} -{ "name": "BTP - Chantier Gros-oeuvre" , "parent": "cat56" } -{ "index": { "_id": "cat60" }} -{ "name": "Outillage - Matériaux 2nd-oeuvre" , "parent": "cat56" } -{ "index": { "_id": "cat32" }} -{ "name": "Équipements Industriels" , "parent": "cat56" } -{ "index": { "_id": "cat61" }} -{ "name": "Restauration - Hôtellerie" , "parent": "cat56" } -{ "index": { "_id": "cat62" }} -{ "name": "Fournitures de Bureau" , "parent": "cat56" } -{ "index": { "_id": "cat63" }} -{ "name": "Commerces & Marchés" , "parent": "cat56" } -{ "index": { "_id": "cat64" }} -{ "name": "Matériel Médical" , "parent": "cat56" } - -{ "index": { "_id": "cat31" }} -{ "name": "Services" , "parent": null} -{ "index": { "_id": "cat34" }} -{ "name": "Prestations de services" , "parent": "cat31" } -{ "index": { "_id": "cat35" }} -{ "name": "Billetterie" , "parent": "cat31" } -{ "index": { "_id": "cat49" }} -{ "name": "Evénements" , "parent": "cat31" } -{ "index": { "_id": "cat36" }} -{ "name": "Cours particuliers" , "parent": "cat31" } -{ "index": { "_id": "cat65" }} -{ "name": "Covoiturage" , "parent": "cat31" } -{ "index": { "_id": "cat37" }} - -{ "name": "Divers" , "parent": null} -{ "index": { "_id": "cat38" }} -{ "name": "Autres" , "parent": "cat37" } diff --git a/duniter4j-es-gchange/src/main/resources/registry-categories-bulk-insert.json b/duniter4j-es-gchange/src/main/resources/registry-categories-bulk-insert.json deleted file mode 100644 index 3ee70de3d3aa58dfd3ff01f1a578a2f917508d66..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/main/resources/registry-categories-bulk-insert.json +++ /dev/null @@ -1,1506 +0,0 @@ -{"index": {"_id": "A"}} -{"name": "Agriculture, sylviculture et pêche", "parent": null} -{"index": {"_id": "01.11Z"}} -{"name": "Culture de céréales (à l'exception du riz), de légumineuses et de graines oléagineuses", "parent": "A"} -{"index": {"_id": "01.12Z"}} -{"name": "Culture du riz", "parent": "A"} -{"index": {"_id": "01.13Z"}} -{"name": "Culture de légumes, de melons, de racines et de tubercules", "parent": "A"} -{"index": {"_id": "01.14Z"}} -{"name": "Culture de la canne à sucre", "parent": "A"} -{"index": {"_id": "01.15Z"}} -{"name": "Culture du tabac", "parent": "A"} -{"index": {"_id": "01.16Z"}} -{"name": "Culture de plantes à fibres", "parent": "A"} -{"index": {"_id": "01.19Z"}} -{"name": "Autres cultures non permanentes", "parent": "A"} -{"index": {"_id": "01.21Z"}} -{"name": "Culture de la vigne", "parent": "A"} -{"index": {"_id": "01.22Z"}} -{"name": "Culture de fruits tropicaux et subtropicaux", "parent": "A"} -{"index": {"_id": "01.23Z"}} -{"name": "Culture d'agrumes", "parent": "A"} -{"index": {"_id": "01.24Z"}} -{"name": "Culture de fruits à pépins et à noyau", "parent": "A"} -{"index": {"_id": "01.25Z"}} -{"name": "Culture d'autres fruits d'arbres ou d'arbustes et de fruits à coque", "parent": "A"} -{"index": {"_id": "01.26Z"}} -{"name": "Culture de fruits oléagineux", "parent": "A"} -{"index": {"_id": "01.27Z"}} -{"name": "Culture de plantes à boissons", "parent": "A"} -{"index": {"_id": "01.28Z"}} -{"name": "Culture de plantes à épices, aromatiques, médicinales et pharmaceutiques", "parent": "A"} -{"index": {"_id": "01.29Z"}} -{"name": "Autres cultures permanentes", "parent": "A"} -{"index": {"_id": "01.30Z"}} -{"name": "Reproduction de plantes", "parent": "A"} -{"index": {"_id": "01.41Z"}} -{"name": "Élevage de vaches laitières", "parent": "A"} -{"index": {"_id": "01.42Z"}} -{"name": "Élevage d'autres bovins et de buffles", "parent": "A"} -{"index": {"_id": "01.43Z"}} -{"name": "Élevage de chevaux et d'autres équidés", "parent": "A"} -{"index": {"_id": "01.44Z"}} -{"name": "Élevage de chameaux et d'autres camélidés", "parent": "A"} -{"index": {"_id": "01.45Z"}} -{"name": "Élevage d'ovins et de caprins", "parent": "A"} -{"index": {"_id": "01.46Z"}} -{"name": "Élevage de porcins", "parent": "A"} -{"index": {"_id": "01.47Z"}} -{"name": "Élevage de volailles", "parent": "A"} -{"index": {"_id": "01.49Z"}} -{"name": "Élevage d'autres animaux", "parent": "A"} -{"index": {"_id": "01.50Z"}} -{"name": "Culture et élevage associés", "parent": "A"} -{"index": {"_id": "01.61Z"}} -{"name": "Activités de soutien aux cultures", "parent": "A"} -{"index": {"_id": "01.62Z"}} -{"name": "Activités de soutien à la production animale", "parent": "A"} -{"index": {"_id": "01.63Z"}} -{"name": "Traitement primaire des récoltes", "parent": "A"} -{"index": {"_id": "01.64Z"}} -{"name": "Traitement des semences", "parent": "A"} -{"index": {"_id": "01.70Z"}} -{"name": "Chasse, piégeage et services annexes", "parent": "A"} -{"index": {"_id": "02.10Z"}} -{"name": "Sylviculture et autres activités forestières", "parent": "A"} -{"index": {"_id": "02.20Z"}} -{"name": "Exploitation forestière", "parent": "A"} -{"index": {"_id": "02.30Z"}} -{"name": "Récolte de produits forestiers non ligneux poussant à l'état sauvage", "parent": "A"} -{"index": {"_id": "02.40Z"}} -{"name": "Services de soutien à l'exploitation forestière", "parent": "A"} -{"index": {"_id": "03.11Z"}} -{"name": "Pêche en mer", "parent": "A"} -{"index": {"_id": "03.12Z"}} -{"name": "Pêche en eau douce", "parent": "A"} -{"index": {"_id": "03.21Z"}} -{"name": "Aquaculture en mer", "parent": "A"} -{"index": {"_id": "03.22Z"}} -{"name": "Aquaculture en eau douce", "parent": "A"} -{"index": {"_id": "B"}} -{"name": "Industries extractives", "parent": null} -{"index": {"_id": "05.10Z"}} -{"name": "Extraction de houille", "parent": "B"} -{"index": {"_id": "05.20Z"}} -{"name": "Extraction de lignite", "parent": "B"} -{"index": {"_id": "06.10Z"}} -{"name": "Extraction de pétrole brut", "parent": "B"} -{"index": {"_id": "06.20Z"}} -{"name": "Extraction de gaz naturel", "parent": "B"} -{"index": {"_id": "07.10Z"}} -{"name": "Extraction de minerais de fer", "parent": "B"} -{"index": {"_id": "07.21Z"}} -{"name": "Extraction de minerais d'uranium et de thorium", "parent": "B"} -{"index": {"_id": "07.29Z"}} -{"name": "Extraction d'autres minerais de métaux non ferreux", "parent": "B"} -{"index": {"_id": "08.11Z"}} -{"name": "Extraction de pierres ornementales et de construction, de calcaire industriel, de gypse, de craie et d'ardoise", "parent": "B"} -{"index": {"_id": "08.12Z"}} -{"name": "Exploitation de gravières et sablières, extraction d'argiles et de kaolin", "parent": "B"} -{"index": {"_id": "08.91Z"}} -{"name": "Extraction des minéraux chimiques et d'engrais minéraux", "parent": "B"} -{"index": {"_id": "08.92Z"}} -{"name": "Extraction de tourbe", "parent": "B"} -{"index": {"_id": "08.93Z"}} -{"name": "Production de sel", "parent": "B"} -{"index": {"_id": "08.99Z"}} -{"name": "Autres activités extractives n.c.a.", "parent": "B"} -{"index": {"_id": "09.10Z"}} -{"name": "Activités de soutien à l'extraction d'hydrocarbures", "parent": "B"} -{"index": {"_id": "09.90Z"}} -{"name": "Activités de soutien aux autres industries extractives", "parent": "B"} -{"index": {"_id": "C"}} -{"name": "Industrie manufacturière", "parent": null} -{"index": {"_id": "10.11Z"}} -{"name": "Transformation et conservation de la viande de boucherie", "parent": "C"} -{"index": {"_id": "10.12Z"}} -{"name": "Transformation et conservation de la viande de volaille", "parent": "C"} -{"index": {"_id": "10.13A"}} -{"name": "Préparation industrielle de produits à base de viande", "parent": "C"} -{"index": {"_id": "10.13B"}} -{"name": "Charcuterie", "parent": "C"} -{"index": {"_id": "10.20Z"}} -{"name": "Transformation et conservation de poisson, de crustacés et de mollusques", "parent": "C"} -{"index": {"_id": "10.31Z"}} -{"name": "Transformation et conservation de pommes de terre", "parent": "C"} -{"index": {"_id": "10.32Z"}} -{"name": "Préparation de jus de fruits et légumes", "parent": "C"} -{"index": {"_id": "10.39A"}} -{"name": "Autre transformation et conservation de légumes", "parent": "C"} -{"index": {"_id": "10.39B"}} -{"name": "Transformation et conservation de fruits", "parent": "C"} -{"index": {"_id": "10.41A"}} -{"name": "Fabrication d'huiles et graisses brutes", "parent": "C"} -{"index": {"_id": "10.41B"}} -{"name": "Fabrication d'huiles et graisses raffinées", "parent": "C"} -{"index": {"_id": "10.42Z"}} -{"name": "Fabrication de margarine et graisses comestibles similaires", "parent": "C"} -{"index": {"_id": "10.51A"}} -{"name": "Fabrication de lait liquide et de produits frais", "parent": "C"} -{"index": {"_id": "10.51B"}} -{"name": "Fabrication de beurre", "parent": "C"} -{"index": {"_id": "10.51C"}} -{"name": "Fabrication de fromage", "parent": "C"} -{"index": {"_id": "10.51D"}} -{"name": "Fabrication d'autres produits laitiers", "parent": "C"} -{"index": {"_id": "10.52Z"}} -{"name": "Fabrication de glaces et sorbets", "parent": "C"} -{"index": {"_id": "10.61A"}} -{"name": "Meunerie", "parent": "C"} -{"index": {"_id": "10.61B"}} -{"name": "Autres activités du travail des grains", "parent": "C"} -{"index": {"_id": "10.62Z"}} -{"name": "Fabrication de produits amylacés", "parent": "C"} -{"index": {"_id": "10.71A"}} -{"name": "Fabrication industrielle de pain et de pâtisserie fraîche", "parent": "C"} -{"index": {"_id": "10.71B"}} -{"name": "Cuisson de produits de boulangerie", "parent": "C"} -{"index": {"_id": "10.71C"}} -{"name": "Boulangerie et boulangerie-pâtisserie", "parent": "C"} -{"index": {"_id": "10.71D"}} -{"name": "Pâtisserie", "parent": "C"} -{"index": {"_id": "10.72Z"}} -{"name": "Fabrication de biscuits, biscottes et pâtisseries de conservation", "parent": "C"} -{"index": {"_id": "10.73Z"}} -{"name": "Fabrication de pâtes alimentaires", "parent": "C"} -{"index": {"_id": "10.81Z"}} -{"name": "Fabrication de sucre", "parent": "C"} -{"index": {"_id": "10.82Z"}} -{"name": "Fabrication de cacao, chocolat et de produits de confiserie", "parent": "C"} -{"index": {"_id": "10.83Z"}} -{"name": "Transformation du thé et du café", "parent": "C"} -{"index": {"_id": "10.84Z"}} -{"name": "Fabrication de condiments et assaisonnements", "parent": "C"} -{"index": {"_id": "10.85Z"}} -{"name": "Fabrication de plats préparés", "parent": "C"} -{"index": {"_id": "10.86Z"}} -{"name": "Fabrication d'aliments homogénéisés et diététiques", "parent": "C"} -{"index": {"_id": "10.89Z"}} -{"name": "Fabrication d'autres produits alimentaires n.c.a.", "parent": "C"} -{"index": {"_id": "10.91Z"}} -{"name": "Fabrication d'aliments pour animaux de ferme", "parent": "C"} -{"index": {"_id": "10.92Z"}} -{"name": "Fabrication d'aliments pour animaux de compagnie", "parent": "C"} -{"index": {"_id": "11.01Z"}} -{"name": "Production de boissons alcooliques distillées", "parent": "C"} -{"index": {"_id": "11.02A"}} -{"name": "Fabrication de vins effervescents", "parent": "C"} -{"index": {"_id": "11.02B"}} -{"name": "Vinification", "parent": "C"} -{"index": {"_id": "11.03Z"}} -{"name": "Fabrication de cidre et de vins de fruits", "parent": "C"} -{"index": {"_id": "11.04Z"}} -{"name": "Production d'autres boissons fermentées non distillées", "parent": "C"} -{"index": {"_id": "11.05Z"}} -{"name": "Fabrication de bière", "parent": "C"} -{"index": {"_id": "11.06Z"}} -{"name": "Fabrication de malt", "parent": "C"} -{"index": {"_id": "11.07A"}} -{"name": "Industrie des eaux de table", "parent": "C"} -{"index": {"_id": "11.07B"}} -{"name": "Production de boissons rafraîchissantes", "parent": "C"} -{"index": {"_id": "12.00Z"}} -{"name": "Fabrication de produits à base de tabac", "parent": "C"} -{"index": {"_id": "13.10Z"}} -{"name": "Préparation de fibres textiles et filature", "parent": "C"} -{"index": {"_id": "13.20Z"}} -{"name": "Tissage", "parent": "C"} -{"index": {"_id": "13.30Z"}} -{"name": "Ennoblissement textile", "parent": "C"} -{"index": {"_id": "13.91Z"}} -{"name": "Fabrication d'étoffes à mailles", "parent": "C"} -{"index": {"_id": "13.92Z"}} -{"name": "Fabrication d'articles textiles, sauf habillement", "parent": "C"} -{"index": {"_id": "13.93Z"}} -{"name": "Fabrication de tapis et moquettes", "parent": "C"} -{"index": {"_id": "13.94Z"}} -{"name": "Fabrication de ficelles, cordes et filets", "parent": "C"} -{"index": {"_id": "13.95Z"}} -{"name": "Fabrication de non-tissés, sauf habillement", "parent": "C"} -{"index": {"_id": "13.96Z"}} -{"name": "Fabrication d'autres textiles techniques et industriels", "parent": "C"} -{"index": {"_id": "13.99Z"}} -{"name": "Fabrication d'autres textiles n.c.a.", "parent": "C"} -{"index": {"_id": "14.11Z"}} -{"name": "Fabrication de vêtements en cuir", "parent": "C"} -{"index": {"_id": "14.12Z"}} -{"name": "Fabrication de vêtements de travail", "parent": "C"} -{"index": {"_id": "14.13Z"}} -{"name": "Fabrication de vêtements de dessus", "parent": "C"} -{"index": {"_id": "14.14Z"}} -{"name": "Fabrication de vêtements de dessous", "parent": "C"} -{"index": {"_id": "14.19Z"}} -{"name": "Fabrication d'autres vêtements et accessoires", "parent": "C"} -{"index": {"_id": "14.20Z"}} -{"name": "Fabrication d'articles en fourrure", "parent": "C"} -{"index": {"_id": "14.31Z"}} -{"name": "Fabrication d'articles chaussants à mailles", "parent": "C"} -{"index": {"_id": "14.39Z"}} -{"name": "Fabrication d'autres articles à mailles", "parent": "C"} -{"index": {"_id": "15.11Z"}} -{"name": "Apprêt et tannage des cuirs ; préparation et teinture des fourrures", "parent": "C"} -{"index": {"_id": "15.12Z"}} -{"name": "Fabrication d'articles de voyage, de maroquinerie et de sellerie", "parent": "C"} -{"index": {"_id": "15.20Z"}} -{"name": "Fabrication de chaussures", "parent": "C"} -{"index": {"_id": "16.10A"}} -{"name": "Sciage et rabotage du bois, hors imprégnation", "parent": "C"} -{"index": {"_id": "16.10B"}} -{"name": "Imprégnation du bois", "parent": "C"} -{"index": {"_id": "16.21Z"}} -{"name": "Fabrication de placage et de panneaux de bois", "parent": "C"} -{"index": {"_id": "16.22Z"}} -{"name": "Fabrication de parquets assemblés", "parent": "C"} -{"index": {"_id": "16.23Z"}} -{"name": "Fabrication de charpentes et d'autres menuiseries", "parent": "C"} -{"index": {"_id": "16.24Z"}} -{"name": "Fabrication d'emballages en bois", "parent": "C"} -{"index": {"_id": "16.29Z"}} -{"name": "Fabrication d'objets divers en bois ; fabrication d'objets en liège, vannerie et sparterie", "parent": "C"} -{"index": {"_id": "17.11Z"}} -{"name": "Fabrication de pâte à papier", "parent": "C"} -{"index": {"_id": "17.12Z"}} -{"name": "Fabrication de papier et de carton", "parent": "C"} -{"index": {"_id": "17.21A"}} -{"name": "Fabrication de carton ondulé", "parent": "C"} -{"index": {"_id": "17.21B"}} -{"name": "Fabrication de cartonnages", "parent": "C"} -{"index": {"_id": "17.21C"}} -{"name": "Fabrication d'emballages en papier", "parent": "C"} -{"index": {"_id": "17.22Z"}} -{"name": "Fabrication d'articles en papier à usage sanitaire ou domestique", "parent": "C"} -{"index": {"_id": "17.23Z"}} -{"name": "Fabrication d'articles de papeterie", "parent": "C"} -{"index": {"_id": "17.24Z"}} -{"name": "Fabrication de papiers peints", "parent": "C"} -{"index": {"_id": "17.29Z"}} -{"name": "Fabrication d'autres articles en papier ou en carton", "parent": "C"} -{"index": {"_id": "18.11Z"}} -{"name": "Imprimerie de journaux", "parent": "C"} -{"index": {"_id": "18.12Z"}} -{"name": "Autre imprimerie (labeur)", "parent": "C"} -{"index": {"_id": "18.13Z"}} -{"name": "Activités de pré-presse", "parent": "C"} -{"index": {"_id": "18.14Z"}} -{"name": "Reliure et activités connexes", "parent": "C"} -{"index": {"_id": "18.20Z"}} -{"name": "Reproduction d'enregistrements", "parent": "C"} -{"index": {"_id": "19.10Z"}} -{"name": "Cokéfaction", "parent": "C"} -{"index": {"_id": "19.20Z"}} -{"name": "Raffinage du pétrole", "parent": "C"} -{"index": {"_id": "20.11Z"}} -{"name": "Fabrication de gaz industriels", "parent": "C"} -{"index": {"_id": "20.12Z"}} -{"name": "Fabrication de colorants et de pigments", "parent": "C"} -{"index": {"_id": "20.13A"}} -{"name": "Enrichissement et retraitement de matières nucléaires", "parent": "C"} -{"index": {"_id": "20.13B"}} -{"name": "Fabrication d'autres produits chimiques inorganiques de base n.c.a.", "parent": "C"} -{"index": {"_id": "20.14Z"}} -{"name": "Fabrication d'autres produits chimiques organiques de base", "parent": "C"} -{"index": {"_id": "20.15Z"}} -{"name": "Fabrication de produits azotés et d'engrais", "parent": "C"} -{"index": {"_id": "20.16Z"}} -{"name": "Fabrication de matières plastiques de base", "parent": "C"} -{"index": {"_id": "20.17Z"}} -{"name": "Fabrication de caoutchouc synthétique", "parent": "C"} -{"index": {"_id": "20.20Z"}} -{"name": "Fabrication de pesticides et d'autres produits agrochimiques", "parent": "C"} -{"index": {"_id": "20.30Z"}} -{"name": "Fabrication de peintures, vernis, encres et mastics", "parent": "C"} -{"index": {"_id": "20.41Z"}} -{"name": "Fabrication de savons, détergents et produits d'entretien", "parent": "C"} -{"index": {"_id": "20.42Z"}} -{"name": "Fabrication de parfums et de produits pour la toilette", "parent": "C"} -{"index": {"_id": "20.51Z"}} -{"name": "Fabrication de produits explosifs", "parent": "C"} -{"index": {"_id": "20.52Z"}} -{"name": "Fabrication de colles", "parent": "C"} -{"index": {"_id": "20.53Z"}} -{"name": "Fabrication d'huiles essentielles", "parent": "C"} -{"index": {"_id": "20.59Z"}} -{"name": "Fabrication d'autres produits chimiques n.c.a.", "parent": "C"} -{"index": {"_id": "20.60Z"}} -{"name": "Fabrication de fibres artificielles ou synthétiques", "parent": "C"} -{"index": {"_id": "21.10Z"}} -{"name": "Fabrication de produits pharmaceutiques de base", "parent": "C"} -{"index": {"_id": "21.20Z"}} -{"name": "Fabrication de préparations pharmaceutiques", "parent": "C"} -{"index": {"_id": "22.11Z"}} -{"name": "Fabrication et rechapage de pneumatiques", "parent": "C"} -{"index": {"_id": "22.19Z"}} -{"name": "Fabrication d'autres articles en caoutchouc", "parent": "C"} -{"index": {"_id": "22.21Z"}} -{"name": "Fabrication de plaques, feuilles, tubes et profilés en matières plastiques", "parent": "C"} -{"index": {"_id": "22.22Z"}} -{"name": "Fabrication d'emballages en matières plastiques", "parent": "C"} -{"index": {"_id": "22.23Z"}} -{"name": "Fabrication d'éléments en matières plastiques pour la construction", "parent": "C"} -{"index": {"_id": "22.29A"}} -{"name": "Fabrication de pièces techniques à base de matières plastiques", "parent": "C"} -{"index": {"_id": "22.29B"}} -{"name": "Fabrication de produits de consommation courante en matières plastiques", "parent": "C"} -{"index": {"_id": "23.11Z"}} -{"name": "Fabrication de verre plat", "parent": "C"} -{"index": {"_id": "23.12Z"}} -{"name": "Façonnage et transformation du verre plat", "parent": "C"} -{"index": {"_id": "23.13Z"}} -{"name": "Fabrication de verre creux", "parent": "C"} -{"index": {"_id": "23.14Z"}} -{"name": "Fabrication de fibres de verre", "parent": "C"} -{"index": {"_id": "23.19Z"}} -{"name": "Fabrication et façonnage d'autres articles en verre, y compris verre technique", "parent": "C"} -{"index": {"_id": "23.20Z"}} -{"name": "Fabrication de produits réfractaires", "parent": "C"} -{"index": {"_id": "23.31Z"}} -{"name": "Fabrication de carreaux en céramique", "parent": "C"} -{"index": {"_id": "23.32Z"}} -{"name": "Fabrication de briques, tuiles et produits de construction, en terre cuite", "parent": "C"} -{"index": {"_id": "23.41Z"}} -{"name": "Fabrication d'articles céramiques à usage domestique ou ornemental", "parent": "C"} -{"index": {"_id": "23.42Z"}} -{"name": "Fabrication d'appareils sanitaires en céramique", "parent": "C"} -{"index": {"_id": "23.43Z"}} -{"name": "Fabrication d'isolateurs et pièces isolantes en céramique", "parent": "C"} -{"index": {"_id": "23.44Z"}} -{"name": "Fabrication d'autres produits céramiques à usage technique", "parent": "C"} -{"index": {"_id": "23.49Z"}} -{"name": "Fabrication d'autres produits céramiques", "parent": "C"} -{"index": {"_id": "23.51Z"}} -{"name": "Fabrication de ciment", "parent": "C"} -{"index": {"_id": "23.52Z"}} -{"name": "Fabrication de chaux et plâtre", "parent": "C"} -{"index": {"_id": "23.61Z"}} -{"name": "Fabrication d'éléments en béton pour la construction", "parent": "C"} -{"index": {"_id": "23.62Z"}} -{"name": "Fabrication d'éléments en plâtre pour la construction", "parent": "C"} -{"index": {"_id": "23.63Z"}} -{"name": "Fabrication de béton prêt à l'emploi", "parent": "C"} -{"index": {"_id": "23.64Z"}} -{"name": "Fabrication de mortiers et bétons secs", "parent": "C"} -{"index": {"_id": "23.65Z"}} -{"name": "Fabrication d'ouvrages en fibre-ciment", "parent": "C"} -{"index": {"_id": "23.69Z"}} -{"name": "Fabrication d'autres ouvrages en béton, en ciment ou en plâtre", "parent": "C"} -{"index": {"_id": "23.70Z"}} -{"name": "Taille, façonnage et finissage de pierres", "parent": "C"} -{"index": {"_id": "23.91Z"}} -{"name": "Fabrication de produits abrasifs", "parent": "C"} -{"index": {"_id": "23.99Z"}} -{"name": "Fabrication d'autres produits minéraux non métalliques n.c.a.", "parent": "C"} -{"index": {"_id": "24.10Z"}} -{"name": "Sidérurgie", "parent": "C"} -{"index": {"_id": "24.20Z"}} -{"name": "Fabrication de tubes, tuyaux, profilés creux et accessoires correspondants en acier", "parent": "C"} -{"index": {"_id": "24.31Z"}} -{"name": "Étirage à froid de barres", "parent": "C"} -{"index": {"_id": "24.32Z"}} -{"name": "Laminage à froid de feuillards", "parent": "C"} -{"index": {"_id": "24.33Z"}} -{"name": "Profilage à froid par formage ou pliage", "parent": "C"} -{"index": {"_id": "24.34Z"}} -{"name": "Tréfilage à froid", "parent": "C"} -{"index": {"_id": "24.41Z"}} -{"name": "Production de métaux précieux", "parent": "C"} -{"index": {"_id": "24.42Z"}} -{"name": "Métallurgie de l'aluminium", "parent": "C"} -{"index": {"_id": "24.43Z"}} -{"name": "Métallurgie du plomb, du zinc ou de l'étain", "parent": "C"} -{"index": {"_id": "24.44Z"}} -{"name": "Métallurgie du cuivre", "parent": "C"} -{"index": {"_id": "24.45Z"}} -{"name": "Métallurgie des autres métaux non ferreux", "parent": "C"} -{"index": {"_id": "24.46Z"}} -{"name": "Élaboration et transformation de matières nucléaires", "parent": "C"} -{"index": {"_id": "24.51Z"}} -{"name": "Fonderie de fonte", "parent": "C"} -{"index": {"_id": "24.52Z"}} -{"name": "Fonderie d'acier", "parent": "C"} -{"index": {"_id": "24.53Z"}} -{"name": "Fonderie de métaux légers", "parent": "C"} -{"index": {"_id": "24.54Z"}} -{"name": "Fonderie d'autres métaux non ferreux", "parent": "C"} -{"index": {"_id": "25.11Z"}} -{"name": "Fabrication de structures métalliques et de parties de structures", "parent": "C"} -{"index": {"_id": "25.12Z"}} -{"name": "Fabrication de portes et fenêtres en métal", "parent": "C"} -{"index": {"_id": "25.21Z"}} -{"name": "Fabrication de radiateurs et de chaudières pour le chauffage central", "parent": "C"} -{"index": {"_id": "25.29Z"}} -{"name": "Fabrication d'autres réservoirs, citernes et conteneurs métalliques", "parent": "C"} -{"index": {"_id": "25.30Z"}} -{"name": "Fabrication de générateurs de vapeur, à l'exception des chaudières pour le chauffage central", "parent": "C"} -{"index": {"_id": "25.40Z"}} -{"name": "Fabrication d'armes et de munitions", "parent": "C"} -{"index": {"_id": "25.50A"}} -{"name": "Forge, estampage, matriçage ; métallurgie des poudres", "parent": "C"} -{"index": {"_id": "25.50B"}} -{"name": "Découpage, emboutissage", "parent": "C"} -{"index": {"_id": "25.61Z"}} -{"name": "Traitement et revêtement des métaux", "parent": "C"} -{"index": {"_id": "25.62A"}} -{"name": "Décolletage", "parent": "C"} -{"index": {"_id": "25.62B"}} -{"name": "Mécanique industrielle", "parent": "C"} -{"index": {"_id": "25.71Z"}} -{"name": "Fabrication de coutellerie", "parent": "C"} -{"index": {"_id": "25.72Z"}} -{"name": "Fabrication de serrures et de ferrures", "parent": "C"} -{"index": {"_id": "25.73A"}} -{"name": "Fabrication de moules et modèles", "parent": "C"} -{"index": {"_id": "25.73B"}} -{"name": "Fabrication d'autres outillages", "parent": "C"} -{"index": {"_id": "25.91Z"}} -{"name": "Fabrication de fûts et emballages métalliques similaires", "parent": "C"} -{"index": {"_id": "25.92Z"}} -{"name": "Fabrication d'emballages métalliques légers", "parent": "C"} -{"index": {"_id": "25.93Z"}} -{"name": "Fabrication d'articles en fils métalliques, de chaînes et de ressorts", "parent": "C"} -{"index": {"_id": "25.94Z"}} -{"name": "Fabrication de vis et de boulons", "parent": "C"} -{"index": {"_id": "25.99A"}} -{"name": "Fabrication d'articles métalliques ménagers", "parent": "C"} -{"index": {"_id": "25.99B"}} -{"name": "Fabrication d'autres articles métalliques", "parent": "C"} -{"index": {"_id": "26.11Z"}} -{"name": "Fabrication de composants électroniques", "parent": "C"} -{"index": {"_id": "26.12Z"}} -{"name": "Fabrication de cartes électroniques assemblées", "parent": "C"} -{"index": {"_id": "26.20Z"}} -{"name": "Fabrication d'ordinateurs et d'équipements périphériques", "parent": "C"} -{"index": {"_id": "26.30Z"}} -{"name": "Fabrication d'équipements de communication", "parent": "C"} -{"index": {"_id": "26.40Z"}} -{"name": "Fabrication de produits électroniques grand public", "parent": "C"} -{"index": {"_id": "26.51A"}} -{"name": "Fabrication d'équipements d'aide à la navigation", "parent": "C"} -{"index": {"_id": "26.51B"}} -{"name": "Fabrication d'instrumentation scientifique et technique", "parent": "C"} -{"index": {"_id": "26.52Z"}} -{"name": "Horlogerie", "parent": "C"} -{"index": {"_id": "26.60Z"}} -{"name": "Fabrication d'équipements d'irradiation médicale, d'équipements électromédicaux et électrothérapeutiques", "parent": "C"} -{"index": {"_id": "26.70Z"}} -{"name": "Fabrication de matériels optique et photographique", "parent": "C"} -{"index": {"_id": "26.80Z"}} -{"name": "Fabrication de supports magnétiques et optiques", "parent": "C"} -{"index": {"_id": "27.11Z"}} -{"name": "Fabrication de moteurs, génératrices et transformateurs électriques", "parent": "C"} -{"index": {"_id": "27.12Z"}} -{"name": "Fabrication de matériel de distribution et de commande électrique", "parent": "C"} -{"index": {"_id": "27.20Z"}} -{"name": "Fabrication de piles et d'accumulateurs électriques", "parent": "C"} -{"index": {"_id": "27.31Z"}} -{"name": "Fabrication de câbles de fibres optiques", "parent": "C"} -{"index": {"_id": "27.32Z"}} -{"name": "Fabrication d'autres fils et câbles électroniques ou électriques", "parent": "C"} -{"index": {"_id": "27.33Z"}} -{"name": "Fabrication de matériel d'installation électrique", "parent": "C"} -{"index": {"_id": "27.40Z"}} -{"name": "Fabrication d'appareils d'éclairage électrique", "parent": "C"} -{"index": {"_id": "27.51Z"}} -{"name": "Fabrication d'appareils électroménagers", "parent": "C"} -{"index": {"_id": "27.52Z"}} -{"name": "Fabrication d'appareils ménagers non électriques", "parent": "C"} -{"index": {"_id": "27.90Z"}} -{"name": "Fabrication d'autres matériels électriques", "parent": "C"} -{"index": {"_id": "28.11Z"}} -{"name": "Fabrication de moteurs et turbines, à l'exception des moteurs d'avions et de véhicules", "parent": "C"} -{"index": {"_id": "28.12Z"}} -{"name": "Fabrication d'équipements hydrauliques et pneumatiques", "parent": "C"} -{"index": {"_id": "28.13Z"}} -{"name": "Fabrication d'autres pompes et compresseurs", "parent": "C"} -{"index": {"_id": "28.14Z"}} -{"name": "Fabrication d'autres articles de robinetterie", "parent": "C"} -{"index": {"_id": "28.15Z"}} -{"name": "Fabrication d'engrenages et d'organes mécaniques de transmission", "parent": "C"} -{"index": {"_id": "28.21Z"}} -{"name": "Fabrication de fours et brûleurs", "parent": "C"} -{"index": {"_id": "28.22Z"}} -{"name": "Fabrication de matériel de levage et de manutention", "parent": "C"} -{"index": {"_id": "28.23Z"}} -{"name": "Fabrication de machines et d'équipements de bureau (à l'exception des ordinateurs et équipements périphériques)", "parent": "C"} -{"index": {"_id": "28.24Z"}} -{"name": "Fabrication d'outillage portatif à moteur incorporé", "parent": "C"} -{"index": {"_id": "28.25Z"}} -{"name": "Fabrication d'équipements aérauliques et frigorifiques industriels", "parent": "C"} -{"index": {"_id": "28.29A"}} -{"name": "Fabrication d'équipements d'emballage, de conditionnement et de pesage", "parent": "C"} -{"index": {"_id": "28.29B"}} -{"name": "Fabrication d'autres machines d'usage général", "parent": "C"} -{"index": {"_id": "28.30Z"}} -{"name": "Fabrication de machines agricoles et forestières", "parent": "C"} -{"index": {"_id": "28.41Z"}} -{"name": "Fabrication de machines-outils pour le travail des métaux", "parent": "C"} -{"index": {"_id": "28.49Z"}} -{"name": "Fabrication d'autres machines-outils", "parent": "C"} -{"index": {"_id": "28.91Z"}} -{"name": "Fabrication de machines pour la métallurgie", "parent": "C"} -{"index": {"_id": "28.92Z"}} -{"name": "Fabrication de machines pour l'extraction ou la construction", "parent": "C"} -{"index": {"_id": "28.93Z"}} -{"name": "Fabrication de machines pour l'industrie agro-alimentaire", "parent": "C"} -{"index": {"_id": "28.94Z"}} -{"name": "Fabrication de machines pour les industries textiles", "parent": "C"} -{"index": {"_id": "28.95Z"}} -{"name": "Fabrication de machines pour les industries du papier et du carton", "parent": "C"} -{"index": {"_id": "28.96Z"}} -{"name": "Fabrication de machines pour le travail du caoutchouc ou des plastiques", "parent": "C"} -{"index": {"_id": "28.99A"}} -{"name": "Fabrication de machines d'imprimerie", "parent": "C"} -{"index": {"_id": "28.99B"}} -{"name": "Fabrication d'autres machines spécialisées", "parent": "C"} -{"index": {"_id": "29.10Z"}} -{"name": "Construction de véhicules automobiles", "parent": "C"} -{"index": {"_id": "29.20Z"}} -{"name": "Fabrication de carrosseries et remorques", "parent": "C"} -{"index": {"_id": "29.31Z"}} -{"name": "Fabrication d'équipements électriques et électroniques automobiles", "parent": "C"} -{"index": {"_id": "29.32Z"}} -{"name": "Fabrication d'autres équipements automobiles", "parent": "C"} -{"index": {"_id": "30.11Z"}} -{"name": "Construction de navires et de structures flottantes", "parent": "C"} -{"index": {"_id": "30.12Z"}} -{"name": "Construction de bateaux de plaisance", "parent": "C"} -{"index": {"_id": "30.20Z"}} -{"name": "Construction de locomotives et d'autre matériel ferroviaire roulant", "parent": "C"} -{"index": {"_id": "30.30Z"}} -{"name": "Construction aéronautique et spatiale", "parent": "C"} -{"index": {"_id": "30.40Z"}} -{"name": "Construction de véhicules militaires de combat", "parent": "C"} -{"index": {"_id": "30.91Z"}} -{"name": "Fabrication de motocycles", "parent": "C"} -{"index": {"_id": "30.92Z"}} -{"name": "Fabrication de bicyclettes et de véhicules pour invalides", "parent": "C"} -{"index": {"_id": "30.99Z"}} -{"name": "Fabrication d'autres équipements de transport n.c.a.", "parent": "C"} -{"index": {"_id": "31.01Z"}} -{"name": "Fabrication de meubles de bureau et de magasin", "parent": "C"} -{"index": {"_id": "31.02Z"}} -{"name": "Fabrication de meubles de cuisine", "parent": "C"} -{"index": {"_id": "31.03Z"}} -{"name": "Fabrication de matelas", "parent": "C"} -{"index": {"_id": "31.09A"}} -{"name": "Fabrication de sièges d'ameublement d'intérieur", "parent": "C"} -{"index": {"_id": "31.09B"}} -{"name": "Fabrication d'autres meubles et industries connexes de l'ameublement", "parent": "C"} -{"index": {"_id": "32.11Z"}} -{"name": "Frappe de monnaie", "parent": "C"} -{"index": {"_id": "32.12Z"}} -{"name": "Fabrication d'articles de joaillerie et bijouterie", "parent": "C"} -{"index": {"_id": "32.13Z"}} -{"name": "Fabrication d'articles de bijouterie fantaisie et articles similaires", "parent": "C"} -{"index": {"_id": "32.20Z"}} -{"name": "Fabrication d'instruments de musique", "parent": "C"} -{"index": {"_id": "32.30Z"}} -{"name": "Fabrication d'articles de sport", "parent": "C"} -{"index": {"_id": "32.40Z"}} -{"name": "Fabrication de jeux et jouets", "parent": "C"} -{"index": {"_id": "32.50A"}} -{"name": "Fabrication de matériel médico-chirurgical et dentaire", "parent": "C"} -{"index": {"_id": "32.50B"}} -{"name": "Fabrication de lunettes", "parent": "C"} -{"index": {"_id": "32.91Z"}} -{"name": "Fabrication d'articles de brosserie", "parent": "C"} -{"index": {"_id": "32.99Z"}} -{"name": "Autres activités manufacturières n.c.a.", "parent": "C"} -{"index": {"_id": "33.11Z"}} -{"name": "Réparation d'ouvrages en métaux", "parent": "C"} -{"index": {"_id": "33.12Z"}} -{"name": "Réparation de machines et équipements mécaniques", "parent": "C"} -{"index": {"_id": "33.13Z"}} -{"name": "Réparation de matériels électroniques et optiques", "parent": "C"} -{"index": {"_id": "33.14Z"}} -{"name": "Réparation d'équipements électriques", "parent": "C"} -{"index": {"_id": "33.15Z"}} -{"name": "Réparation et maintenance navale", "parent": "C"} -{"index": {"_id": "33.16Z"}} -{"name": "Réparation et maintenance d'aéronefs et d'engins spatiaux", "parent": "C"} -{"index": {"_id": "33.17Z"}} -{"name": "Réparation et maintenance d'autres équipements de transport", "parent": "C"} -{"index": {"_id": "33.19Z"}} -{"name": "Réparation d'autres équipements", "parent": "C"} -{"index": {"_id": "33.20A"}} -{"name": "Installation de structures métalliques, chaudronnées et de tuyauterie", "parent": "C"} -{"index": {"_id": "33.20B"}} -{"name": "Installation de machines et équipements mécaniques", "parent": "C"} -{"index": {"_id": "33.20C"}} -{"name": "Conception d'ensemble et assemblage sur site industriel d'équipements de contrôle des processus industriels", "parent": "C"} -{"index": {"_id": "33.20D"}} -{"name": "Installation d'équipements électriques, de matériels électroniques et optiques ou d'autres matériels", "parent": "C"} -{"index": {"_id": "D"}} -{"name": "Production et distribution d'électricité, de gaz, de vapeur et d'air conditionné", "parent": null} -{"index": {"_id": "35.11Z"}} -{"name": "Production d'électricité", "parent": "D"} -{"index": {"_id": "35.12Z"}} -{"name": "Transport d'électricité", "parent": "D"} -{"index": {"_id": "35.13Z"}} -{"name": "Distribution d'électricité", "parent": "D"} -{"index": {"_id": "35.14Z"}} -{"name": "Commerce d'électricité", "parent": "D"} -{"index": {"_id": "35.21Z"}} -{"name": "Production de combustibles gazeux", "parent": "D"} -{"index": {"_id": "35.22Z"}} -{"name": "Distribution de combustibles gazeux par conduites", "parent": "D"} -{"index": {"_id": "35.23Z"}} -{"name": "Commerce de combustibles gazeux par conduites", "parent": "D"} -{"index": {"_id": "35.30Z"}} -{"name": "Production et distribution de vapeur et d'air conditionné", "parent": "D"} -{"index": {"_id": "E"}} -{"name": "Production et distribution d'eau ; assainissement, gestion des déchets et dépollution", "parent": null} -{"index": {"_id": "36.00Z"}} -{"name": "Captage, traitement et distribution d'eau", "parent": "E"} -{"index": {"_id": "37.00Z"}} -{"name": "Collecte et traitement des eaux usées", "parent": "E"} -{"index": {"_id": "38.11Z"}} -{"name": "Collecte des déchets non dangereux", "parent": "E"} -{"index": {"_id": "38.12Z"}} -{"name": "Collecte des déchets dangereux", "parent": "E"} -{"index": {"_id": "38.21Z"}} -{"name": "Traitement et élimination des déchets non dangereux", "parent": "E"} -{"index": {"_id": "38.22Z"}} -{"name": "Traitement et élimination des déchets dangereux", "parent": "E"} -{"index": {"_id": "38.31Z"}} -{"name": "Démantèlement d'épaves", "parent": "E"} -{"index": {"_id": "38.32Z"}} -{"name": "Récupération de déchets triés", "parent": "E"} -{"index": {"_id": "39.00Z"}} -{"name": "Dépollution et autres services de gestion des déchets", "parent": "E"} -{"index": {"_id": "F"}} -{"name": "Construction", "parent": null} -{"index": {"_id": "41.10A"}} -{"name": "Promotion immobilière de logements", "parent": "F"} -{"index": {"_id": "41.10B"}} -{"name": "Promotion immobilière de bureaux", "parent": "F"} -{"index": {"_id": "41.10C"}} -{"name": "Promotion immobilière d'autres bâtiments", "parent": "F"} -{"index": {"_id": "41.10D"}} -{"name": "Supports juridiques de programmes", "parent": "F"} -{"index": {"_id": "41.20A"}} -{"name": "Construction de maisons individuelles", "parent": "F"} -{"index": {"_id": "41.20B"}} -{"name": "Construction d'autres bâtiments", "parent": "F"} -{"index": {"_id": "42.11Z"}} -{"name": "Construction de routes et autoroutes", "parent": "F"} -{"index": {"_id": "42.12Z"}} -{"name": "Construction de voies ferrées de surface et souterraines", "parent": "F"} -{"index": {"_id": "42.13A"}} -{"name": "Construction d'ouvrages d'art", "parent": "F"} -{"index": {"_id": "42.13B"}} -{"name": "Construction et entretien de tunnels", "parent": "F"} -{"index": {"_id": "42.21Z"}} -{"name": "Construction de réseaux pour fluides", "parent": "F"} -{"index": {"_id": "42.22Z"}} -{"name": "Construction de réseaux électriques et de télécommunications", "parent": "F"} -{"index": {"_id": "42.91Z"}} -{"name": "Construction d'ouvrages maritimes et fluviaux", "parent": "F"} -{"index": {"_id": "42.99Z"}} -{"name": "Construction d'autres ouvrages de génie civil n.c.a.", "parent": "F"} -{"index": {"_id": "43.11Z"}} -{"name": "Travaux de démolition", "parent": "F"} -{"index": {"_id": "43.12A"}} -{"name": "Travaux de terrassement courants et travaux préparatoires", "parent": "F"} -{"index": {"_id": "43.12B"}} -{"name": "Travaux de terrassement spécialisés ou de grande masse", "parent": "F"} -{"index": {"_id": "43.13Z"}} -{"name": "Forages et sondages", "parent": "F"} -{"index": {"_id": "43.21A"}} -{"name": "Travaux d'installation électrique dans tous locaux", "parent": "F"} -{"index": {"_id": "43.21B"}} -{"name": "Travaux d'installation électrique sur la voie publique", "parent": "F"} -{"index": {"_id": "43.22A"}} -{"name": "Travaux d'installation d'eau et de gaz en tous locaux", "parent": "F"} -{"index": {"_id": "43.22B"}} -{"name": "Travaux d'installation d'équipements thermiques et de climatisation", "parent": "F"} -{"index": {"_id": "43.29A"}} -{"name": "Travaux d'isolation", "parent": "F"} -{"index": {"_id": "43.29B"}} -{"name": "Autres travaux d'installation n.c.a.", "parent": "F"} -{"index": {"_id": "43.31Z"}} -{"name": "Travaux de plâtrerie", "parent": "F"} -{"index": {"_id": "43.32A"}} -{"name": "Travaux de menuiserie bois et PVC", "parent": "F"} -{"index": {"_id": "43.32B"}} -{"name": "Travaux de menuiserie métallique et serrurerie", "parent": "F"} -{"index": {"_id": "43.32C"}} -{"name": "Agencement de lieux de vente", "parent": "F"} -{"index": {"_id": "43.33Z"}} -{"name": "Travaux de revêtement des sols et des murs", "parent": "F"} -{"index": {"_id": "43.34Z"}} -{"name": "Travaux de peinture et vitrerie", "parent": "F"} -{"index": {"_id": "43.39Z"}} -{"name": "Autres travaux de finition", "parent": "F"} -{"index": {"_id": "43.91A"}} -{"name": "Travaux de charpente", "parent": "F"} -{"index": {"_id": "43.91B"}} -{"name": "Travaux de couverture par éléments", "parent": "F"} -{"index": {"_id": "43.99A"}} -{"name": "Travaux d'étanchéification", "parent": "F"} -{"index": {"_id": "43.99B"}} -{"name": "Travaux de montage de structures métalliques", "parent": "F"} -{"index": {"_id": "43.99C"}} -{"name": "Travaux de maçonnerie générale et gros œuvre de bâtiment", "parent": "F"} -{"index": {"_id": "43.99D"}} -{"name": "Autres travaux spécialisés de construction", "parent": "F"} -{"index": {"_id": "43.99E"}} -{"name": "Location avec opérateur de matériel de construction", "parent": "F"} -{"index": {"_id": "G"}} -{"name": "Commerce ; réparation d'automobiles et de motocycles", "parent": null} -{"index": {"_id": "45.11Z"}} -{"name": "Commerce de voitures et de véhicules automobiles légers", "parent": "G"} -{"index": {"_id": "45.19Z"}} -{"name": "Commerce d'autres véhicules automobiles", "parent": "G"} -{"index": {"_id": "45.20A"}} -{"name": "Entretien et réparation de véhicules automobiles légers", "parent": "G"} -{"index": {"_id": "45.20B"}} -{"name": "Entretien et réparation d'autres véhicules automobiles", "parent": "G"} -{"index": {"_id": "45.31Z"}} -{"name": "Commerce de gros d'équipements automobiles", "parent": "G"} -{"index": {"_id": "45.32Z"}} -{"name": "Commerce de détail d'équipements automobiles", "parent": "G"} -{"index": {"_id": "45.40Z"}} -{"name": "Commerce et réparation de motocycles", "parent": "G"} -{"index": {"_id": "46.11Z"}} -{"name": "Intermédiaires du commerce en matières premières agricoles, animaux vivants, matières premières textiles et produits semi-finis", "parent": "G"} -{"index": {"_id": "46.12A"}} -{"name": "Centrales d'achat de carburant", "parent": "G"} -{"index": {"_id": "46.12B"}} -{"name": "Autres intermédiaires du commerce en combustibles, métaux, minéraux et produits chimiques", "parent": "G"} -{"index": {"_id": "46.13Z"}} -{"name": "Intermédiaires du commerce en bois et matériaux de construction", "parent": "G"} -{"index": {"_id": "46.14Z"}} -{"name": "Intermédiaires du commerce en machines, équipements industriels, navires et avions", "parent": "G"} -{"index": {"_id": "46.15Z"}} -{"name": "Intermédiaires du commerce en meubles, articles de ménage et quincaillerie", "parent": "G"} -{"index": {"_id": "46.16Z"}} -{"name": "Intermédiaires du commerce en textiles, habillement, fourrures, chaussures et articles en cuir", "parent": "G"} -{"index": {"_id": "46.17A"}} -{"name": "Centrales d'achat alimentaires", "parent": "G"} -{"index": {"_id": "46.17B"}} -{"name": "Autres intermédiaires du commerce en denrées, boissons et tabac", "parent": "G"} -{"index": {"_id": "46.18Z"}} -{"name": "Intermédiaires spécialisés dans le commerce d'autres produits spécifiques", "parent": "G"} -{"index": {"_id": "46.19A"}} -{"name": "Centrales d'achat non alimentaires", "parent": "G"} -{"index": {"_id": "46.19B"}} -{"name": "Autres intermédiaires du commerce en produits divers", "parent": "G"} -{"index": {"_id": "46.21Z"}} -{"name": "Commerce de gros (commerce interentreprises) de céréales, de tabac non manufacturé, de semences et d'aliments pour le bétail", "parent": "G"} -{"index": {"_id": "46.22Z"}} -{"name": "Commerce de gros (commerce interentreprises) de fleurs et plantes", "parent": "G"} -{"index": {"_id": "46.23Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'animaux vivants", "parent": "G"} -{"index": {"_id": "46.24Z"}} -{"name": "Commerce de gros (commerce interentreprises) de cuirs et peaux", "parent": "G"} -{"index": {"_id": "46.31Z"}} -{"name": "Commerce de gros (commerce interentreprises) de fruits et légumes", "parent": "G"} -{"index": {"_id": "46.32A"}} -{"name": "Commerce de gros (commerce interentreprises) de viandes de boucherie", "parent": "G"} -{"index": {"_id": "46.32B"}} -{"name": "Commerce de gros (commerce interentreprises) de produits à base de viande", "parent": "G"} -{"index": {"_id": "46.32C"}} -{"name": "Commerce de gros (commerce interentreprises) de volailles et gibier", "parent": "G"} -{"index": {"_id": "46.33Z"}} -{"name": "Commerce de gros (commerce interentreprises) de produits laitiers, œufs, huiles et matières grasses comestibles", "parent": "G"} -{"index": {"_id": "46.34Z"}} -{"name": "Commerce de gros (commerce interentreprises) de boissons", "parent": "G"} -{"index": {"_id": "46.35Z"}} -{"name": "Commerce de gros (commerce interentreprises) de produits à base de tabac", "parent": "G"} -{"index": {"_id": "46.36Z"}} -{"name": "Commerce de gros (commerce interentreprises) de sucre, chocolat et confiserie", "parent": "G"} -{"index": {"_id": "46.37Z"}} -{"name": "Commerce de gros (commerce interentreprises) de café, thé, cacao et épices", "parent": "G"} -{"index": {"_id": "46.38A"}} -{"name": "Commerce de gros (commerce interentreprises) de poissons, crustacés et mollusques", "parent": "G"} -{"index": {"_id": "46.38B"}} -{"name": "Commerce de gros (commerce interentreprises) alimentaire spécialisé divers", "parent": "G"} -{"index": {"_id": "46.39A"}} -{"name": "Commerce de gros (commerce interentreprises) de produits surgelés", "parent": "G"} -{"index": {"_id": "46.39B"}} -{"name": "Commerce de gros (commerce interentreprises) alimentaire non spécialisé", "parent": "G"} -{"index": {"_id": "46.41Z"}} -{"name": "Commerce de gros (commerce interentreprises) de textiles", "parent": "G"} -{"index": {"_id": "46.42Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'habillement et de chaussures", "parent": "G"} -{"index": {"_id": "46.43Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'appareils électroménagers", "parent": "G"} -{"index": {"_id": "46.44Z"}} -{"name": "Commerce de gros (commerce interentreprises) de vaisselle, verrerie et produits d'entretien", "parent": "G"} -{"index": {"_id": "46.45Z"}} -{"name": "Commerce de gros (commerce interentreprises) de parfumerie et de produits de beauté", "parent": "G"} -{"index": {"_id": "46.46Z"}} -{"name": "Commerce de gros (commerce interentreprises) de produits pharmaceutiques", "parent": "G"} -{"index": {"_id": "46.47Z"}} -{"name": "Commerce de gros (commerce interentreprises) de meubles, de tapis et d'appareils d'éclairage", "parent": "G"} -{"index": {"_id": "46.48Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'articles d'horlogerie et de bijouterie", "parent": "G"} -{"index": {"_id": "46.49Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'autres biens domestiques", "parent": "G"} -{"index": {"_id": "46.51Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'ordinateurs, d'équipements informatiques périphériques et de logiciels", "parent": "G"} -{"index": {"_id": "46.52Z"}} -{"name": "Commerce de gros (commerce interentreprises) de composants et d'équipements électroniques et de télécommunication", "parent": "G"} -{"index": {"_id": "46.61Z"}} -{"name": "Commerce de gros (commerce interentreprises) de matériel agricole", "parent": "G"} -{"index": {"_id": "46.62Z"}} -{"name": "Commerce de gros (commerce interentreprises) de machines-outils", "parent": "G"} -{"index": {"_id": "46.63Z"}} -{"name": "Commerce de gros (commerce interentreprises) de machines pour l'extraction, la construction et le génie civil", "parent": "G"} -{"index": {"_id": "46.64Z"}} -{"name": "Commerce de gros (commerce interentreprises) de machines pour l'industrie textile et l'habillement", "parent": "G"} -{"index": {"_id": "46.65Z"}} -{"name": "Commerce de gros (commerce interentreprises) de mobilier de bureau", "parent": "G"} -{"index": {"_id": "46.66Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'autres machines et équipements de bureau", "parent": "G"} -{"index": {"_id": "46.69A"}} -{"name": "Commerce de gros (commerce interentreprises) de matériel électrique", "parent": "G"} -{"index": {"_id": "46.69B"}} -{"name": "Commerce de gros (commerce interentreprises) de fournitures et équipements industriels divers", "parent": "G"} -{"index": {"_id": "46.69C"}} -{"name": "Commerce de gros (commerce interentreprises) de fournitures et équipements divers pour le commerce et les services", "parent": "G"} -{"index": {"_id": "46.71Z"}} -{"name": "Commerce de gros (commerce interentreprises) de combustibles et de produits annexes", "parent": "G"} -{"index": {"_id": "46.72Z"}} -{"name": "Commerce de gros (commerce interentreprises) de minerais et métaux", "parent": "G"} -{"index": {"_id": "46.73A"}} -{"name": "Commerce de gros (commerce interentreprises) de bois et de matériaux de construction", "parent": "G"} -{"index": {"_id": "46.73B"}} -{"name": "Commerce de gros (commerce interentreprises) d'appareils sanitaires et de produits de décoration", "parent": "G"} -{"index": {"_id": "46.74A"}} -{"name": "Commerce de gros (commerce interentreprises) de quincaillerie", "parent": "G"} -{"index": {"_id": "46.74B"}} -{"name": "Commerce de gros (commerce interentreprises) de fournitures pour la plomberie et le chauffage", "parent": "G"} -{"index": {"_id": "46.75Z"}} -{"name": "Commerce de gros (commerce interentreprises) de produits chimiques", "parent": "G"} -{"index": {"_id": "46.76Z"}} -{"name": "Commerce de gros (commerce interentreprises) d'autres produits intermédiaires", "parent": "G"} -{"index": {"_id": "46.77Z"}} -{"name": "Commerce de gros (commerce interentreprises) de déchets et débris", "parent": "G"} -{"index": {"_id": "46.90Z"}} -{"name": "Commerce de gros (commerce interentreprises) non spécialisé", "parent": "G"} -{"index": {"_id": "47.11A"}} -{"name": "Commerce de détail de produits surgelés", "parent": "G"} -{"index": {"_id": "47.11B"}} -{"name": "Commerce d'alimentation générale", "parent": "G"} -{"index": {"_id": "47.11C"}} -{"name": "Supérettes", "parent": "G"} -{"index": {"_id": "47.11D"}} -{"name": "Supermarchés", "parent": "G"} -{"index": {"_id": "47.11E"}} -{"name": "Magasins multi-commerces", "parent": "G"} -{"index": {"_id": "47.11F"}} -{"name": "Hypermarchés", "parent": "G"} -{"index": {"_id": "47.19A"}} -{"name": "Grands magasins", "parent": "G"} -{"index": {"_id": "47.19B"}} -{"name": "Autres commerces de détail en magasin non spécialisé", "parent": "G"} -{"index": {"_id": "47.21Z"}} -{"name": "Commerce de détail de fruits et légumes en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.22Z"}} -{"name": "Commerce de détail de viandes et de produits à base de viande en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.23Z"}} -{"name": "Commerce de détail de poissons, crustacés et mollusques en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.24Z"}} -{"name": "Commerce de détail de pain, pâtisserie et confiserie en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.25Z"}} -{"name": "Commerce de détail de boissons en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.26Z"}} -{"name": "Commerce de détail de produits à base de tabac en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.29Z"}} -{"name": "Autres commerces de détail alimentaires en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.30Z"}} -{"name": "Commerce de détail de carburants en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.41Z"}} -{"name": "Commerce de détail d'ordinateurs, d'unités périphériques et de logiciels en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.42Z"}} -{"name": "Commerce de détail de matériels de télécommunication en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.43Z"}} -{"name": "Commerce de détail de matériels audio et vidéo en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.51Z"}} -{"name": "Commerce de détail de textiles en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.52A"}} -{"name": "Commerce de détail de quincaillerie, peintures et verres en petites surfaces (moins de 400 m²)", "parent": "G"} -{"index": {"_id": "47.52B"}} -{"name": "Commerce de détail de quincaillerie, peintures et verres en grandes surfaces (400 m² et plus)", "parent": "G"} -{"index": {"_id": "47.53Z"}} -{"name": "Commerce de détail de tapis, moquettes et revêtements de murs et de sols en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.54Z"}} -{"name": "Commerce de détail d'appareils électroménagers en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.59A"}} -{"name": "Commerce de détail de meubles", "parent": "G"} -{"index": {"_id": "47.59B"}} -{"name": "Commerce de détail d'autres équipements du foyer", "parent": "G"} -{"index": {"_id": "47.61Z"}} -{"name": "Commerce de détail de livres en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.62Z"}} -{"name": "Commerce de détail de journaux et papeterie en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.63Z"}} -{"name": "Commerce de détail d'enregistrements musicaux et vidéo en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.64Z"}} -{"name": "Commerce de détail d'articles de sport en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.65Z"}} -{"name": "Commerce de détail de jeux et jouets en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.71Z"}} -{"name": "Commerce de détail d'habillement en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.72A"}} -{"name": "Commerce de détail de la chaussure", "parent": "G"} -{"index": {"_id": "47.72B"}} -{"name": "Commerce de détail de maroquinerie et d'articles de voyage", "parent": "G"} -{"index": {"_id": "47.73Z"}} -{"name": "Commerce de détail de produits pharmaceutiques en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.74Z"}} -{"name": "Commerce de détail d'articles médicaux et orthopédiques en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.75Z"}} -{"name": "Commerce de détail de parfumerie et de produits de beauté en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.76Z"}} -{"name": "Commerce de détail de fleurs, plantes, graines, engrais, animaux de compagnie et aliments pour ces animaux en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.77Z"}} -{"name": "Commerce de détail d'articles d'horlogerie et de bijouterie en magasin spécialisé", "parent": "G"} -{"index": {"_id": "47.78A"}} -{"name": "Commerces de détail d'optique", "parent": "G"} -{"index": {"_id": "47.78B"}} -{"name": "Commerces de détail de charbons et combustibles", "parent": "G"} -{"index": {"_id": "47.78C"}} -{"name": "Autres commerces de détail spécialisés divers", "parent": "G"} -{"index": {"_id": "47.79Z"}} -{"name": "Commerce de détail de biens d'occasion en magasin", "parent": "G"} -{"index": {"_id": "47.81Z"}} -{"name": "Commerce de détail alimentaire sur éventaires et marchés", "parent": "G"} -{"index": {"_id": "47.82Z"}} -{"name": "Commerce de détail de textiles, d'habillement et de chaussures sur éventaires et marchés", "parent": "G"} -{"index": {"_id": "47.89Z"}} -{"name": "Autres commerces de détail sur éventaires et marchés", "parent": "G"} -{"index": {"_id": "47.91A"}} -{"name": "Vente à distance sur catalogue général", "parent": "G"} -{"index": {"_id": "47.91B"}} -{"name": "Vente à distance sur catalogue spécialisé", "parent": "G"} -{"index": {"_id": "47.99A"}} -{"name": "Vente à domicile", "parent": "G"} -{"index": {"_id": "47.99B"}} -{"name": "Vente par automates et autres commerces de détail hors magasin, éventaires ou marchés n.c.a.", "parent": "G"} -{"index": {"_id": "H"}} -{"name": "Transports et entreposage", "parent": null} -{"index": {"_id": "49.10Z"}} -{"name": "Transport ferroviaire interurbain de voyageurs", "parent": "H"} -{"index": {"_id": "49.20Z"}} -{"name": "Transports ferroviaires de fret", "parent": "H"} -{"index": {"_id": "49.31Z"}} -{"name": "Transports urbains et suburbains de voyageurs", "parent": "H"} -{"index": {"_id": "49.32Z"}} -{"name": "Transports de voyageurs par taxis", "parent": "H"} -{"index": {"_id": "49.39A"}} -{"name": "Transports routiers réguliers de voyageurs", "parent": "H"} -{"index": {"_id": "49.39B"}} -{"name": "Autres transports routiers de voyageurs", "parent": "H"} -{"index": {"_id": "49.39C"}} -{"name": "Téléphériques et remontées mécaniques", "parent": "H"} -{"index": {"_id": "49.41A"}} -{"name": "Transports routiers de fret interurbains", "parent": "H"} -{"index": {"_id": "49.41B"}} -{"name": "Transports routiers de fret de proximité", "parent": "H"} -{"index": {"_id": "49.41C"}} -{"name": "Location de camions avec chauffeur", "parent": "H"} -{"index": {"_id": "49.42Z"}} -{"name": "Services de déménagement", "parent": "H"} -{"index": {"_id": "49.50Z"}} -{"name": "Transports par conduites", "parent": "H"} -{"index": {"_id": "50.10Z"}} -{"name": "Transports maritimes et côtiers de passagers", "parent": "H"} -{"index": {"_id": "50.20Z"}} -{"name": "Transports maritimes et côtiers de fret", "parent": "H"} -{"index": {"_id": "50.30Z"}} -{"name": "Transports fluviaux de passagers", "parent": "H"} -{"index": {"_id": "50.40Z"}} -{"name": "Transports fluviaux de fret", "parent": "H"} -{"index": {"_id": "51.10Z"}} -{"name": "Transports aériens de passagers", "parent": "H"} -{"index": {"_id": "51.21Z"}} -{"name": "Transports aériens de fret", "parent": "H"} -{"index": {"_id": "51.22Z"}} -{"name": "Transports spatiaux", "parent": "H"} -{"index": {"_id": "52.10A"}} -{"name": "Entreposage et stockage frigorifique", "parent": "H"} -{"index": {"_id": "52.10B"}} -{"name": "Entreposage et stockage non frigorifique", "parent": "H"} -{"index": {"_id": "52.21Z"}} -{"name": "Services auxiliaires des transports terrestres", "parent": "H"} -{"index": {"_id": "52.22Z"}} -{"name": "Services auxiliaires des transports par eau", "parent": "H"} -{"index": {"_id": "52.23Z"}} -{"name": "Services auxiliaires des transports aériens", "parent": "H"} -{"index": {"_id": "52.24A"}} -{"name": "Manutention portuaire", "parent": "H"} -{"index": {"_id": "52.24B"}} -{"name": "Manutention non portuaire", "parent": "H"} -{"index": {"_id": "52.29A"}} -{"name": "Messagerie, fret express", "parent": "H"} -{"index": {"_id": "52.29B"}} -{"name": "Affrètement et organisation des transports", "parent": "H"} -{"index": {"_id": "53.10Z"}} -{"name": "Activités de poste dans le cadre d'une obligation de service universel", "parent": "H"} -{"index": {"_id": "53.20Z"}} -{"name": "Autres activités de poste et de courrier", "parent": "H"} -{"index": {"_id": "I"}} -{"name": "Hébergement et restauration", "parent": null} -{"index": {"_id": "55.10Z"}} -{"name": "Hôtels et hébergement similaire", "parent": "I"} -{"index": {"_id": "55.20Z"}} -{"name": "Hébergement touristique et autre hébergement de courte durée", "parent": "I"} -{"index": {"_id": "55.30Z"}} -{"name": "Terrains de camping et parcs pour caravanes ou véhicules de loisirs", "parent": "I"} -{"index": {"_id": "55.90Z"}} -{"name": "Autres hébergements", "parent": "I"} -{"index": {"_id": "56.10A"}} -{"name": "Restauration traditionnelle", "parent": "I"} -{"index": {"_id": "56.10B"}} -{"name": "Cafétérias et autres libres-services", "parent": "I"} -{"index": {"_id": "56.10C"}} -{"name": "Restauration de type rapide", "parent": "I"} -{"index": {"_id": "56.21Z"}} -{"name": "Services des traiteurs", "parent": "I"} -{"index": {"_id": "56.29A"}} -{"name": "Restauration collective sous contrat", "parent": "I"} -{"index": {"_id": "56.29B"}} -{"name": "Autres services de restauration n.c.a.", "parent": "I"} -{"index": {"_id": "56.30Z"}} -{"name": "Débits de boissons", "parent": "I"} -{"index": {"_id": "J"}} -{"name": "Information et communication", "parent": null} -{"index": {"_id": "58.11Z"}} -{"name": "Édition de livres", "parent": "J"} -{"index": {"_id": "58.12Z"}} -{"name": "Édition de répertoires et de fichiers d'adresses", "parent": "J"} -{"index": {"_id": "58.13Z"}} -{"name": "Édition de journaux", "parent": "J"} -{"index": {"_id": "58.14Z"}} -{"name": "Édition de revues et périodiques", "parent": "J"} -{"index": {"_id": "58.19Z"}} -{"name": "Autres activités d'édition", "parent": "J"} -{"index": {"_id": "58.21Z"}} -{"name": "Édition de jeux électroniques", "parent": "J"} -{"index": {"_id": "58.29A"}} -{"name": "Édition de logiciels système et de réseau", "parent": "J"} -{"index": {"_id": "58.29B"}} -{"name": "Édition de logiciels outils de développement et de langages", "parent": "J"} -{"index": {"_id": "58.29C"}} -{"name": "Édition de logiciels applicatifs", "parent": "J"} -{"index": {"_id": "59.11A"}} -{"name": "Production de films et de programmes pour la télévision", "parent": "J"} -{"index": {"_id": "59.11B"}} -{"name": "Production de films institutionnels et publicitaires", "parent": "J"} -{"index": {"_id": "59.11C"}} -{"name": "Production de films pour le cinéma", "parent": "J"} -{"index": {"_id": "59.12Z"}} -{"name": "Post-production de films cinématographiques, de vidéo et de programmes de télévision", "parent": "J"} -{"index": {"_id": "59.13A"}} -{"name": "Distribution de films cinématographiques", "parent": "J"} -{"index": {"_id": "59.13B"}} -{"name": "Édition et distribution vidéo", "parent": "J"} -{"index": {"_id": "59.14Z"}} -{"name": "Projection de films cinématographiques", "parent": "J"} -{"index": {"_id": "59.20Z"}} -{"name": "Enregistrement sonore et édition musicale", "parent": "J"} -{"index": {"_id": "60.10Z"}} -{"name": "Édition et diffusion de programmes radio", "parent": "J"} -{"index": {"_id": "60.20A"}} -{"name": "Édition de chaînes généralistes", "parent": "J"} -{"index": {"_id": "60.20B"}} -{"name": "Édition de chaînes thématiques", "parent": "J"} -{"index": {"_id": "61.10Z"}} -{"name": "Télécommunications filaires", "parent": "J"} -{"index": {"_id": "61.20Z"}} -{"name": "Télécommunications sans fil", "parent": "J"} -{"index": {"_id": "61.30Z"}} -{"name": "Télécommunications par satellite", "parent": "J"} -{"index": {"_id": "61.90Z"}} -{"name": "Autres activités de télécommunication", "parent": "J"} -{"index": {"_id": "62.01Z"}} -{"name": "Programmation informatique", "parent": "J"} -{"index": {"_id": "62.02A"}} -{"name": "Conseil en systèmes et logiciels informatiques", "parent": "J"} -{"index": {"_id": "62.02B"}} -{"name": "Tierce maintenance de systèmes et d'applications informatiques", "parent": "J"} -{"index": {"_id": "62.03Z"}} -{"name": "Gestion d'installations informatiques", "parent": "J"} -{"index": {"_id": "62.09Z"}} -{"name": "Autres activités informatiques", "parent": "J"} -{"index": {"_id": "63.11Z"}} -{"name": "Traitement de données, hébergement et activités connexes", "parent": "J"} -{"index": {"_id": "63.12Z"}} -{"name": "Portails Internet", "parent": "J"} -{"index": {"_id": "63.91Z"}} -{"name": "Activités des agences de presse", "parent": "J"} -{"index": {"_id": "63.99Z"}} -{"name": "Autres services d'information n.c.a.", "parent": "J"} -{"index": {"_id": "K"}} -{"name": "Activités financières et d'assurance", "parent": null} -{"index": {"_id": "64.11Z"}} -{"name": "Activités de banque centrale", "parent": "K"} -{"index": {"_id": "64.19Z"}} -{"name": "Autres intermédiations monétaires", "parent": "K"} -{"index": {"_id": "64.20Z"}} -{"name": "Activités des sociétés holding", "parent": "K"} -{"index": {"_id": "64.30Z"}} -{"name": "Fonds de placement et entités financières similaires", "parent": "K"} -{"index": {"_id": "64.91Z"}} -{"name": "Crédit-bail", "parent": "K"} -{"index": {"_id": "64.92Z"}} -{"name": "Autre distribution de crédit", "parent": "K"} -{"index": {"_id": "64.99Z"}} -{"name": "Autres activités des services financiers, hors assurance et caisses de retraite, n.c.a.", "parent": "K"} -{"index": {"_id": "65.11Z"}} -{"name": "Assurance vie", "parent": "K"} -{"index": {"_id": "65.12Z"}} -{"name": "Autres assurances", "parent": "K"} -{"index": {"_id": "65.20Z"}} -{"name": "Réassurance", "parent": "K"} -{"index": {"_id": "65.30Z"}} -{"name": "Caisses de retraite", "parent": "K"} -{"index": {"_id": "66.11Z"}} -{"name": "Administration de marchés financiers", "parent": "K"} -{"index": {"_id": "66.12Z"}} -{"name": "Courtage de valeurs mobilières et de marchandises", "parent": "K"} -{"index": {"_id": "66.19A"}} -{"name": "Supports juridiques de gestion de patrimoine mobilier", "parent": "K"} -{"index": {"_id": "66.19B"}} -{"name": "Autres activités auxiliaires de services financiers, hors assurance et caisses de retraite, n.c.a.", "parent": "K"} -{"index": {"_id": "66.21Z"}} -{"name": "Évaluation des risques et dommages", "parent": "K"} -{"index": {"_id": "66.22Z"}} -{"name": "Activités des agents et courtiers d'assurances", "parent": "K"} -{"index": {"_id": "66.29Z"}} -{"name": "Autres activités auxiliaires d'assurance et de caisses de retraite", "parent": "K"} -{"index": {"_id": "66.30Z"}} -{"name": "Gestion de fonds", "parent": "K"} -{"index": {"_id": "L"}} -{"name": "Activités immobilières", "parent": null} -{"index": {"_id": "68.10Z"}} -{"name": "Activités des marchands de biens immobiliers", "parent": "L"} -{"index": {"_id": "68.20A"}} -{"name": "Location de logements", "parent": "L"} -{"index": {"_id": "68.20B"}} -{"name": "Location de terrains et d'autres biens immobiliers", "parent": "L"} -{"index": {"_id": "68.31Z"}} -{"name": "Agences immobilières", "parent": "L"} -{"index": {"_id": "68.32A"}} -{"name": "Administration d'immeubles et autres biens immobiliers", "parent": "L"} -{"index": {"_id": "68.32B"}} -{"name": "Supports juridiques de gestion de patrimoine immobilier", "parent": "L"} -{"index": {"_id": "M"}} -{"name": "Activités spécialisées, scientifiques et techniques", "parent": null} -{"index": {"_id": "69.10Z"}} -{"name": "Activités juridiques", "parent": "M"} -{"index": {"_id": "69.20Z"}} -{"name": "Activités comptables", "parent": "M"} -{"index": {"_id": "70.10Z"}} -{"name": "Activités des sièges sociaux", "parent": "M"} -{"index": {"_id": "70.21Z"}} -{"name": "Conseil en relations publiques et communication", "parent": "M"} -{"index": {"_id": "70.22Z"}} -{"name": "Conseil pour les affaires et autres conseils de gestion", "parent": "M"} -{"index": {"_id": "71.11Z"}} -{"name": "Activités d'architecture", "parent": "M"} -{"index": {"_id": "71.12A"}} -{"name": "Activité des géomètres", "parent": "M"} -{"index": {"_id": "71.12B"}} -{"name": "Ingénierie, études techniques", "parent": "M"} -{"index": {"_id": "71.20A"}} -{"name": "Contrôle technique automobile", "parent": "M"} -{"index": {"_id": "71.20B"}} -{"name": "Analyses, essais et inspections techniques", "parent": "M"} -{"index": {"_id": "72.11Z"}} -{"name": "Recherche-développement en biotechnologie", "parent": "M"} -{"index": {"_id": "72.19Z"}} -{"name": "Recherche-développement en autres sciences physiques et naturelles", "parent": "M"} -{"index": {"_id": "72.20Z"}} -{"name": "Recherche-développement en sciences humaines et sociales", "parent": "M"} -{"index": {"_id": "73.11Z"}} -{"name": "Activités des agences de publicité", "parent": "M"} -{"index": {"_id": "73.12Z"}} -{"name": "Régie publicitaire de médias", "parent": "M"} -{"index": {"_id": "73.20Z"}} -{"name": "Études de marché et sondages", "parent": "M"} -{"index": {"_id": "74.10Z"}} -{"name": "Activités spécialisées de design", "parent": "M"} -{"index": {"_id": "74.20Z"}} -{"name": "Activités photographiques", "parent": "M"} -{"index": {"_id": "74.30Z"}} -{"name": "Traduction et interprétation", "parent": "M"} -{"index": {"_id": "74.90A"}} -{"name": "Activité des économistes de la construction", "parent": "M"} -{"index": {"_id": "74.90B"}} -{"name": "Activités spécialisées, scientifiques et techniques diverses", "parent": "M"} -{"index": {"_id": "N"}} -{"name": "Activités de services administratifs et de soutien", "parent": null} -{"index": {"_id": "75.00Z"}} -{"name": "Activités vétérinaires", "parent": "N"} -{"index": {"_id": "77.11A"}} -{"name": "Location de courte durée de voitures et de véhicules automobiles légers", "parent": "N"} -{"index": {"_id": "77.11B"}} -{"name": "Location de longue durée de voitures et de véhicules automobiles légers", "parent": "N"} -{"index": {"_id": "77.12Z"}} -{"name": "Location et location-bail de camions", "parent": "N"} -{"index": {"_id": "77.21Z"}} -{"name": "Location et location-bail d'articles de loisirs et de sport", "parent": "N"} -{"index": {"_id": "77.22Z"}} -{"name": "Location de vidéocassettes et disques vidéo", "parent": "N"} -{"index": {"_id": "77.29Z"}} -{"name": "Location et location-bail d'autres biens personnels et domestiques", "parent": "N"} -{"index": {"_id": "77.31Z"}} -{"name": "Location et location-bail de machines et équipements agricoles", "parent": "N"} -{"index": {"_id": "77.32Z"}} -{"name": "Location et location-bail de machines et équipements pour la construction", "parent": "N"} -{"index": {"_id": "77.33Z"}} -{"name": "Location et location-bail de machines de bureau et de matériel informatique", "parent": "N"} -{"index": {"_id": "77.34Z"}} -{"name": "Location et location-bail de matériels de transport par eau", "parent": "N"} -{"index": {"_id": "77.35Z"}} -{"name": "Location et location-bail de matériels de transport aérien", "parent": "N"} -{"index": {"_id": "77.39Z"}} -{"name": "Location et location-bail d'autres machines, équipements et biens matériels n.c.a.", "parent": "N"} -{"index": {"_id": "77.40Z"}} -{"name": "Location-bail de propriété intellectuelle et de produits similaires, à l'exception des œuvres soumises à copyright", "parent": "N"} -{"index": {"_id": "78.10Z"}} -{"name": "Activités des agences de placement de main-d'œuvre", "parent": "N"} -{"index": {"_id": "78.20Z"}} -{"name": "Activités des agences de travail temporaire", "parent": "N"} -{"index": {"_id": "78.30Z"}} -{"name": "Autre mise à disposition de ressources humaines", "parent": "N"} -{"index": {"_id": "79.11Z"}} -{"name": "Activités des agences de voyage", "parent": "N"} -{"index": {"_id": "79.12Z"}} -{"name": "Activités des voyagistes", "parent": "N"} -{"index": {"_id": "79.90Z"}} -{"name": "Autres services de réservation et activités connexes", "parent": "N"} -{"index": {"_id": "80.10Z"}} -{"name": "Activités de sécurité privée", "parent": "N"} -{"index": {"_id": "80.20Z"}} -{"name": "Activités liées aux systèmes de sécurité", "parent": "N"} -{"index": {"_id": "80.30Z"}} -{"name": "Activités d'enquête", "parent": "N"} -{"index": {"_id": "81.10Z"}} -{"name": "Activités combinées de soutien lié aux bâtiments", "parent": "N"} -{"index": {"_id": "81.21Z"}} -{"name": "Nettoyage courant des bâtiments", "parent": "N"} -{"index": {"_id": "81.22Z"}} -{"name": "Autres activités de nettoyage des bâtiments et nettoyage industriel", "parent": "N"} -{"index": {"_id": "81.29A"}} -{"name": "Désinfection, désinsectisation, dératisation", "parent": "N"} -{"index": {"_id": "81.29B"}} -{"name": "Autres activités de nettoyage n.c.a.", "parent": "N"} -{"index": {"_id": "81.30Z"}} -{"name": "Services d'aménagement paysager", "parent": "N"} -{"index": {"_id": "82.11Z"}} -{"name": "Services administratifs combinés de bureau", "parent": "N"} -{"index": {"_id": "82.19Z"}} -{"name": "Photocopie, préparation de documents et autres activités spécialisées de soutien de bureau", "parent": "N"} -{"index": {"_id": "82.20Z"}} -{"name": "Activités de centres d'appels", "parent": "N"} -{"index": {"_id": "82.30Z"}} -{"name": "Organisation de foires, salons professionnels et congrès", "parent": "N"} -{"index": {"_id": "82.91Z"}} -{"name": "Activités des agences de recouvrement de factures et des sociétés d'information financière sur la clientèle", "parent": "N"} -{"index": {"_id": "82.92Z"}} -{"name": "Activités de conditionnement", "parent": "N"} -{"index": {"_id": "82.99Z"}} -{"name": "Autres activités de soutien aux entreprises n.c.a.", "parent": "N"} -{"index": {"_id": "O"}} -{"name": "Administration publique", "parent": null} -{"index": {"_id": "84.11Z"}} -{"name": "Administration publique générale", "parent": "O"} -{"index": {"_id": "84.12Z"}} -{"name": "Administration publique (tutelle) de la santé, de la formation, de la culture et des services sociaux, autre que sécurité sociale", "parent": "O"} -{"index": {"_id": "84.13Z"}} -{"name": "Administration publique (tutelle) des activités économiques", "parent": "O"} -{"index": {"_id": "84.21Z"}} -{"name": "Affaires étrangères", "parent": "O"} -{"index": {"_id": "84.22Z"}} -{"name": "Défense", "parent": "O"} -{"index": {"_id": "84.23Z"}} -{"name": "Justice", "parent": "O"} -{"index": {"_id": "84.24Z"}} -{"name": "Activités d'ordre public et de sécurité", "parent": "O"} -{"index": {"_id": "84.25Z"}} -{"name": "Services du feu et de secours", "parent": "O"} -{"index": {"_id": "84.30A"}} -{"name": "Activités générales de sécurité sociale", "parent": "O"} -{"index": {"_id": "84.30B"}} -{"name": "Gestion des retraites complémentaires", "parent": "O"} -{"index": {"_id": "84.30C"}} -{"name": "Distribution sociale de revenus", "parent": "O"} -{"index": {"_id": "P"}} -{"name": "Enseignement", "parent": null} -{"index": {"_id": "85.10Z"}} -{"name": "Enseignement pré-primaire", "parent": "P"} -{"index": {"_id": "85.20Z"}} -{"name": "Enseignement primaire", "parent": "P"} -{"index": {"_id": "85.31Z"}} -{"name": "Enseignement secondaire général", "parent": "P"} -{"index": {"_id": "85.32Z"}} -{"name": "Enseignement secondaire technique ou professionnel", "parent": "P"} -{"index": {"_id": "85.41Z"}} -{"name": "Enseignement post-secondaire non supérieur", "parent": "P"} -{"index": {"_id": "85.42Z"}} -{"name": "Enseignement supérieur", "parent": "P"} -{"index": {"_id": "85.51Z"}} -{"name": "Enseignement de disciplines sportives et d'activités de loisirs", "parent": "P"} -{"index": {"_id": "85.52Z"}} -{"name": "Enseignement culturel", "parent": "P"} -{"index": {"_id": "85.53Z"}} -{"name": "Enseignement de la conduite", "parent": "P"} -{"index": {"_id": "85.59A"}} -{"name": "Formation continue d'adultes", "parent": "P"} -{"index": {"_id": "85.59B"}} -{"name": "Autres enseignements", "parent": "P"} -{"index": {"_id": "85.60Z"}} -{"name": "Activités de soutien à l'enseignement", "parent": "P"} -{"index": {"_id": "Q"}} -{"name": "Santé humaine et action sociale", "parent": null} -{"index": {"_id": "86.10Z"}} -{"name": "Activités hospitalières", "parent": "Q"} -{"index": {"_id": "86.21Z"}} -{"name": "Activité des médecins généralistes", "parent": "Q"} -{"index": {"_id": "86.22A"}} -{"name": "Activités de radiodiagnostic et de radiothérapie", "parent": "Q"} -{"index": {"_id": "86.22B"}} -{"name": "Activités chirurgicales", "parent": "Q"} -{"index": {"_id": "86.22C"}} -{"name": "Autres activités des médecins spécialistes", "parent": "Q"} -{"index": {"_id": "86.23Z"}} -{"name": "Pratique dentaire", "parent": "Q"} -{"index": {"_id": "86.90A"}} -{"name": "Ambulances", "parent": "Q"} -{"index": {"_id": "86.90B"}} -{"name": "Laboratoires d'analyses médicales", "parent": "Q"} -{"index": {"_id": "86.90C"}} -{"name": "Centres de collecte et banques d'organes", "parent": "Q"} -{"index": {"_id": "86.90D"}} -{"name": "Activités des infirmiers et des sages-femmes", "parent": "Q"} -{"index": {"_id": "86.90E"}} -{"name": "Activités des professionnels de la rééducation, de l'appareillage et des pédicures-podologues", "parent": "Q"} -{"index": {"_id": "86.90F"}} -{"name": "Activités de santé humaine non classées ailleurs", "parent": "Q"} -{"index": {"_id": "87.10A"}} -{"name": "Hébergement médicalisé pour personnes âgées", "parent": "Q"} -{"index": {"_id": "87.10B"}} -{"name": "Hébergement médicalisé pour enfants handicapés", "parent": "Q"} -{"index": {"_id": "87.10C"}} -{"name": "Hébergement médicalisé pour adultes handicapés et autre hébergement médicalisé", "parent": "Q"} -{"index": {"_id": "87.20A"}} -{"name": "Hébergement social pour handicapés mentaux et malades mentaux", "parent": "Q"} -{"index": {"_id": "87.20B"}} -{"name": "Hébergement social pour toxicomanes", "parent": "Q"} -{"index": {"_id": "87.30A"}} -{"name": "Hébergement social pour personnes âgées", "parent": "Q"} -{"index": {"_id": "87.30B"}} -{"name": "Hébergement social pour handicapés physiques", "parent": "Q"} -{"index": {"_id": "87.90A"}} -{"name": "Hébergement social pour enfants en difficultés", "parent": "Q"} -{"index": {"_id": "87.90B"}} -{"name": "Hébergement social pour adultes et familles en difficultés et autre hébergement social", "parent": "Q"} -{"index": {"_id": "88.10A"}} -{"name": "Aide à domicile", "parent": "Q"} -{"index": {"_id": "88.10B"}} -{"name": "Accueil ou accompagnement sans hébergement d'adultes handicapés ou de personnes âgées", "parent": "Q"} -{"index": {"_id": "88.10C"}} -{"name": "Aide par le travail", "parent": "Q"} -{"index": {"_id": "88.91A"}} -{"name": "Accueil de jeunes enfants", "parent": "Q"} -{"index": {"_id": "88.91B"}} -{"name": "Accueil ou accompagnement sans hébergement d'enfants handicapés", "parent": "Q"} -{"index": {"_id": "88.99A"}} -{"name": "Autre accueil ou accompagnement sans hébergement d'enfants et d'adolescents", "parent": "Q"} -{"index": {"_id": "88.99B"}} -{"name": "Action sociale sans hébergement n.c.a.", "parent": "Q"} -{"index": {"_id": "R"}} -{"name": "Arts, spectacles et activités récréatives", "parent": null} -{"index": {"_id": "90.01Z"}} -{"name": "Arts du spectacle vivant", "parent": "R"} -{"index": {"_id": "90.02Z"}} -{"name": "Activités de soutien au spectacle vivant", "parent": "R"} -{"index": {"_id": "90.03A"}} -{"name": "Création artistique relevant des arts plastiques", "parent": "R"} -{"index": {"_id": "90.03B"}} -{"name": "Autre création artistique", "parent": "R"} -{"index": {"_id": "90.04Z"}} -{"name": "Gestion de salles de spectacles", "parent": "R"} -{"index": {"_id": "91.01Z"}} -{"name": "Gestion des bibliothèques et des archives", "parent": "R"} -{"index": {"_id": "91.02Z"}} -{"name": "Gestion des musées", "parent": "R"} -{"index": {"_id": "91.03Z"}} -{"name": "Gestion des sites et monuments historiques et des attractions touristiques similaires", "parent": "R"} -{"index": {"_id": "91.04Z"}} -{"name": "Gestion des jardins botaniques et zoologiques et des réserves naturelles", "parent": "R"} -{"index": {"_id": "92.00Z"}} -{"name": "Organisation de jeux de hasard et d'argent", "parent": "R"} -{"index": {"_id": "93.11Z"}} -{"name": "Gestion d'installations sportives", "parent": "R"} -{"index": {"_id": "93.12Z"}} -{"name": "Activités de clubs de sports", "parent": "R"} -{"index": {"_id": "93.13Z"}} -{"name": "Activités des centres de culture physique", "parent": "R"} -{"index": {"_id": "93.19Z"}} -{"name": "Autres activités liées au sport", "parent": "R"} -{"index": {"_id": "93.21Z"}} -{"name": "Activités des parcs d'attractions et parcs à thèmes", "parent": "R"} -{"index": {"_id": "93.29Z"}} -{"name": "Autres activités récréatives et de loisirs", "parent": "R"} -{"index": {"_id": "S"}} -{"name": "Autres activités de services", "parent": null} -{"index": {"_id": "94.11Z"}} -{"name": "Activités des organisations patronales et consulaires", "parent": "S"} -{"index": {"_id": "94.12Z"}} -{"name": "Activités des organisations professionnelles", "parent": "S"} -{"index": {"_id": "94.20Z"}} -{"name": "Activités des syndicats de salariés", "parent": "S"} -{"index": {"_id": "94.91Z"}} -{"name": "Activités des organisations religieuses", "parent": "S"} -{"index": {"_id": "94.92Z"}} -{"name": "Activités des organisations politiques", "parent": "S"} -{"index": {"_id": "94.99Z"}} -{"name": "Autres organisations fonctionnant par adhésion volontaire", "parent": "S"} -{"index": {"_id": "95.11Z"}} -{"name": "Réparation d'ordinateurs et d'équipements périphériques", "parent": "S"} -{"index": {"_id": "95.12Z"}} -{"name": "Réparation d'équipements de communication", "parent": "S"} -{"index": {"_id": "95.21Z"}} -{"name": "Réparation de produits électroniques grand public", "parent": "S"} -{"index": {"_id": "95.22Z"}} -{"name": "Réparation d'appareils électroménagers et d'équipements pour la maison et le jardin", "parent": "S"} -{"index": {"_id": "95.23Z"}} -{"name": "Réparation de chaussures et d'articles en cuir", "parent": "S"} -{"index": {"_id": "95.24Z"}} -{"name": "Réparation de meubles et d'équipements du foyer", "parent": "S"} -{"index": {"_id": "95.25Z"}} -{"name": "Réparation d'articles d'horlogerie et de bijouterie", "parent": "S"} -{"index": {"_id": "95.29Z"}} -{"name": "Réparation d'autres biens personnels et domestiques", "parent": "S"} -{"index": {"_id": "96.01A"}} -{"name": "Blanchisserie-teinturerie de gros", "parent": "S"} -{"index": {"_id": "96.01B"}} -{"name": "Blanchisserie-teinturerie de détail", "parent": "S"} -{"index": {"_id": "96.02A"}} -{"name": "Coiffure", "parent": "S"} -{"index": {"_id": "96.02B"}} -{"name": "Soins de beauté", "parent": "S"} -{"index": {"_id": "96.03Z"}} -{"name": "Services funéraires", "parent": "S"} -{"index": {"_id": "96.04Z"}} -{"name": "Entretien corporel", "parent": "S"} -{"index": {"_id": "96.09Z"}} -{"name": "Autres services personnels n.c.a.", "parent": "S"} -{"index": {"_id": "T"}} -{"name": "Activités des ménages en tant qu'employeurs ; activités indifférenciées des ménages en tant que producteurs de biens et services pour usage propre", "parent": null} -{"index": {"_id": "97.00Z"}} -{"name": "Activités des ménages en tant qu'employeurs de personnel domestique", "parent": "T"} -{"index": {"_id": "98.10Z"}} -{"name": "Activités indifférenciées des ménages en tant que producteurs de biens pour usage propre", "parent": "T"} -{"index": {"_id": "98.20Z"}} -{"name": "Activités indifférenciées des ménages en tant que producteurs de services pour usage propre", "parent": "T"} -{"index": {"_id": "U"}} -{"name": "Activités extra-territoriales", "parent": null} -{"index": {"_id": "99.00Z"}} -{"name": "Activités des organisations et organismes extraterritoriaux", "parent": "U"} diff --git a/duniter4j-es-gchange/src/test/java/org/duniter/elasticsearch/TestResource.java b/duniter4j-es-gchange/src/test/java/org/duniter/elasticsearch/TestResource.java deleted file mode 100644 index 2b27a3e55828026c4e9b37c4a3dba40020e161c4..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/test/java/org/duniter/elasticsearch/TestResource.java +++ /dev/null @@ -1,141 +0,0 @@ -package org.duniter.elasticsearch; - -/* - * #%L - * UCoin Java Client :: Core API - * %% - * Copyright (C) 2014 - 2015 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - - -import com.google.common.collect.Lists; -import org.duniter.core.client.config.ConfigurationOption; -import org.duniter.core.client.service.ServiceLocator; -import org.apache.commons.io.FileUtils; -import org.junit.runner.Description; -import org.nuiton.i18n.I18n; -import org.nuiton.i18n.init.DefaultI18nInitializer; -import org.nuiton.i18n.init.UserI18nInitializer; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import java.io.File; -import java.io.IOException; -import java.util.List; -import java.util.Locale; - -public class TestResource extends org.duniter.core.test.TestResource { - - private static final Logger log = LoggerFactory.getLogger(TestResource.class); - - public static TestResource create() { - return new TestResource(null); - } - - public static TestResource create(String configName) { - return new TestResource(configName); - } - - private TestFixtures fixtures = new TestFixtures(); - - protected TestResource(String configName) { - super(configName); - } - - public TestFixtures getFixtures() { - return fixtures; - } - - protected void before(Description description) throws Throwable { - super.before(description); - - // Initialize configuration - initConfiguration(getConfigFileName()); - - // Init i18n - initI18n(); - - // Initialize service locator - ServiceLocator.instance().init(); - } - - /** - * Return configuration files prefix (i.e. 'allegro-test') - * Could be override by external project - * - * @return the prefix to use to retrieve configuration files - */ - protected String getConfigFilesPrefix() { - return "duniter4j-elasticsearch-test"; - } - - protected String getI18nBundleName() { - return "duniter4j-elasticsearch-i18n"; - } - - /* -- -- */ - - /** - * Convenience methods that could be override to initialize other configuration - * - * @param configFilename - * @param configArgs - */ - protected void initConfiguration(String configFilename) { - String[] configArgs = getConfigArgs(); - //PluginSettings config = new PluginSettings(configFilename, configArgs); - //PluginSettings.setInstance(config); - } - - protected void initI18n() throws IOException { - /*PluginSettings config ;//= PluginSettings.instance(); - - // --------------------------------------------------------------------// - // init i18n - // --------------------------------------------------------------------// - File i18nDirectory = new File(config.getDataDirectory(), "i18n"); - if (i18nDirectory.exists()) { - // clean i18n cache - FileUtils.cleanDirectory(i18nDirectory); - } - - FileUtils.forceMkdir(i18nDirectory); - - if (log.isDebugEnabled()) { - log.debug("I18N directory: " + i18nDirectory); - } - - Locale i18nLocale = config.getI18nLocale(); - - if (log.isInfoEnabled()) { - log.info(String.format("Starts i18n with locale [%s] at [%s]", - i18nLocale, i18nDirectory)); - } - I18n.init(new UserI18nInitializer( - i18nDirectory, new DefaultI18nInitializer(getI18nBundleName())), - i18nLocale);*/ - } - - protected String[] getConfigArgs() { - List<String> configArgs = Lists.newArrayList(); - configArgs.addAll(Lists.newArrayList( - "--option", ConfigurationOption.BASEDIR.getKey(), getResourceDirectory().getAbsolutePath())); - return configArgs.toArray(new String[configArgs.size()]); - } - -} diff --git a/duniter4j-es-gchange/src/test/java/org/duniter/elasticsearch/service/RegistryRecordIndexerServiceTest.java b/duniter4j-es-gchange/src/test/java/org/duniter/elasticsearch/service/RegistryRecordIndexerServiceTest.java deleted file mode 100644 index 4987281db4a472a7657256319818daccb07a7bd4..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/test/java/org/duniter/elasticsearch/service/RegistryRecordIndexerServiceTest.java +++ /dev/null @@ -1,57 +0,0 @@ -package org.duniter.elasticsearch.service; - -/* - * #%L - * UCoin Java Client :: ElasticSearch Indexer - * %% - * Copyright (C) 2014 - 2016 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ - -import org.duniter.elasticsearch.TestResource; -import org.duniter.elasticsearch.gchange.service.RegistryService; -import org.junit.Before; -import org.junit.ClassRule; -import org.junit.Ignore; -import org.junit.Test; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -/** - * Created by Benoit on 06/05/2015. - */ -@Ignore -public class RegistryRecordIndexerServiceTest { - private static final Logger log = LoggerFactory.getLogger(RegistryRecordIndexerServiceTest.class); - - @ClassRule - public static final TestResource resource = TestResource.create(); - - private RegistryService service; - - @Before - public void setUp() throws Exception { - // FIXME use google guice ? - //service = ServiceLocator.instance().getRegistryRecordIndexerService(); - } - - @Test - public void insertTestData() { - //service.insertRecordFromBulkFile(new File("src/test/resources/registry-test-records.json")); - } - -} diff --git a/duniter4j-es-gchange/src/test/resources/META-INF/services/org.duniter.core.beans.Bean b/duniter4j-es-gchange/src/test/resources/META-INF/services/org.duniter.core.beans.Bean deleted file mode 100644 index 1d327a744e4eec6703b417254f5b19dfdbc09fa4..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/test/resources/META-INF/services/org.duniter.core.beans.Bean +++ /dev/null @@ -1,14 +0,0 @@ -org.duniter.core.client.service.bma.BlockchainRemoteServiceImpl -org.duniter.core.client.service.bma.NetworkRemoteServiceImpl -org.duniter.core.client.service.bma.WotRemoteServiceImpl -org.duniter.core.client.service.bma.TransactionRemoteServiceImpl -org.duniter.core.service.Ed25519CryptoServiceImpl -org.duniter.core.service.MailServiceImpl -org.duniter.core.client.service.HttpServiceImpl -org.duniter.core.client.service.DataContext -org.duniter.core.client.service.local.PeerServiceImpl -org.duniter.core.client.service.local.CurrencyServiceImpl -org.duniter.core.client.dao.mem.MemoryCurrencyDaoImpl -org.duniter.core.client.dao.mem.MemoryPeerDaoImpl -org.duniter.elasticsearch.service.ElasticSearchService -org.duniter.elasticsearch.service.registry.CurrencyRegistryService \ No newline at end of file diff --git a/duniter4j-es-gchange/src/test/resources/duniter4j-elasticsearch-localhost-node.properties b/duniter4j-es-gchange/src/test/resources/duniter4j-elasticsearch-localhost-node.properties deleted file mode 100644 index 38d7a5d9655c9a5bb5babc7487d246139417a8ff..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/test/resources/duniter4j-elasticsearch-localhost-node.properties +++ /dev/null @@ -1,12 +0,0 @@ -duniter4j.node.host=metab.ucoin.fr -duniter4j.node.port=9201 - -duniter4j.elasticsearch.embedded.enable=false -duniter4j.elasticsearch.local=fals -duniter4j.elasticsearch.http.enable=false -duniter4j.elasticsearch.cluster.name=duniter4j-elacticsearch-test - -#duniter4j.elasticsearch.cluster.name=duniter4j-elacticsearch - -duniter4j.elasticsearch.host=localhost -duniter4j.elasticsearch.port=9300 diff --git a/duniter4j-es-gchange/src/test/resources/duniter4j-elasticsearch-test.properties b/duniter4j-es-gchange/src/test/resources/duniter4j-elasticsearch-test.properties deleted file mode 100644 index 27e326f1e7a5e4a4ee67c86df9ca03e8c5b6d2df..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/test/resources/duniter4j-elasticsearch-test.properties +++ /dev/null @@ -1,16 +0,0 @@ -duniter4j.node.host=metab.ucoin.fr -duniter4j.node.port=9201 - -duniter4j.basedir=target/es-home - -#duniter4j.elasticsearch.data -#duniter4j.elasticsearch.embedded.enable=true -duniter4j.elasticsearch.local=false -duniter4j.elasticsearch.http.enable=true -duniter4j.elasticsearch.cluster.name=duniter4j-elasticsearch - -#duniter4j.elasticsearch.cluster.name=duniter4j-elacticsearch - -duniter4j.elasticsearch.embedded.enable=false -duniter4j.elasticsearch.host=192.168.0.5 -duniter4j.elasticsearch.port=9300 diff --git a/duniter4j-es-gchange/src/test/resources/registry-test-records.json b/duniter4j-es-gchange/src/test/resources/registry-test-records.json deleted file mode 100644 index c9c11c01be29242dca860f9e0e692539c101e07c..0000000000000000000000000000000000000000 --- a/duniter4j-es-gchange/src/test/resources/registry-test-records.json +++ /dev/null @@ -1,23 +0,0 @@ -/* - * #%L - * Duniter4j :: ElasticSearch Plugin - * %% - * Copyright (C) 2014 - 2016 EIS - * %% - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as - * published by the Free Software Foundation, either version 3 of the - * License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public - * License along with this program. If not, see - * <http://www.gnu.org/licenses/gpl-3.0.html>. - * #L% - */ -{"index": {"_id": "AVOt3cmVo7-63byx1Jow"}} -{"title": "Benoit Lavenier (kimamila)", "description": "Pasionné de plein de trucs...", "category": "particulier", "location":"Martigné-sur-Mayenne", "pictures": [{"src": ""}], "issuer": "G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU", "hash": "7zPfFwyXGZKYMmLTzVynFGVEwLDfj48a5E3EA2RWtifq", "signature": "CP4p0+Nby/Fs7exp/mlLnOAu8iMJwyLRW6UHkZXZX8q8WbqxHYBRo2uzpcFAT0zEYSig7j3HqYdeoA3MpU1BCQ=="} diff --git a/duniter4j-es-gchange/pom.xml b/duniter4j-es-subscription/pom.xml similarity index 68% rename from duniter4j-es-gchange/pom.xml rename to duniter4j-es-subscription/pom.xml index ab8ce385a9635160987197cd812a1f98779430dc..219fb1e6992ce2b9d79f4f6d78385258f2775872 100644 --- a/duniter4j-es-gchange/pom.xml +++ b/duniter4j-es-subscription/pom.xml @@ -7,14 +7,14 @@ <version>0.9.2-SNAPSHOT</version> </parent> - <artifactId>duniter4j-es-gchange</artifactId> + <artifactId>duniter4j-es-subscription</artifactId> <packaging>jar</packaging> - <name>Duniter4j :: ElasticSearch GChange plugin</name> + <name>Duniter4j :: ElasticSearch Subscription plugin</name> <properties> <!-- i18n configuration --> - <i18n.bundleOutputName>duniter4j-es-gchange-i18n</i18n.bundleOutputName> + <i18n.bundleOutputName>duniter4j-es-subscription-i18n</i18n.bundleOutputName> <i18n.generateCsvFile>true</i18n.generateCsvFile> <i18n.bundleCsvFile> ${maven.gen.dir}/resources/META-INF/${i18n.bundleOutputName}.csv @@ -41,12 +41,65 @@ <scope>provided</scope> </dependency> + <dependency> + <groupId>org.antlr</groupId> + <artifactId>stringtemplate</artifactId> + </dependency> + <!-- Unit test --> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <scope>test</scope> </dependency> + + <!-- LOGGING DEPENDENCIES - SLF4J --> + <dependency> + <groupId>org.slf4j</groupId> + <artifactId>slf4j-api</artifactId> + <scope>test</scope> + </dependency> + <dependency> + <groupId>org.slf4j</groupId> + <artifactId>slf4j-log4j12</artifactId> + <scope>test</scope> + </dependency> + <dependency> + <groupId>log4j</groupId> + <artifactId>log4j</artifactId> + <optional>true</optional> + <scope>test</scope> + </dependency> + <dependency> + <groupId>log4j</groupId> + <artifactId>log4j</artifactId> + <optional>true</optional> + <scope>test</scope> + </dependency> + + <!-- JNA (need for OS shutdown hook) --> + <dependency> + <groupId>net.java.dev.jna</groupId> + <artifactId>jna</artifactId> + <scope>test</scope> + </dependency> + <dependency> + <groupId>net.java.dev.jna</groupId> + <artifactId>jna-platform</artifactId> + <scope>test</scope> + <exclusions> + <exclusion> + <groupId>net.java.dev.jna</groupId> + <artifactId>jna</artifactId> + </exclusion> + </exclusions> + </dependency> + <dependency> + <groupId>org.elasticsearch.plugin</groupId> + <artifactId>mapper-attachments</artifactId> + <version>${elasticsearch.version}</version> + <scope>test</scope> + </dependency> </dependencies> <build> diff --git a/duniter4j-es-gchange/src/license/THIRD-PARTY.properties b/duniter4j-es-subscription/src/license/THIRD-PARTY.properties similarity index 100% rename from duniter4j-es-gchange/src/license/THIRD-PARTY.properties rename to duniter4j-es-subscription/src/license/THIRD-PARTY.properties diff --git a/duniter4j-es-gchange/src/main/assembly/plugin.xml b/duniter4j-es-subscription/src/main/assembly/plugin.xml similarity index 100% rename from duniter4j-es-gchange/src/main/assembly/plugin.xml rename to duniter4j-es-subscription/src/main/assembly/plugin.xml diff --git a/duniter4j-es-gchange/src/main/filtered-resources/log4j.properties b/duniter4j-es-subscription/src/main/filtered-resources/log4j.properties similarity index 100% rename from duniter4j-es-gchange/src/main/filtered-resources/log4j.properties rename to duniter4j-es-subscription/src/main/filtered-resources/log4j.properties diff --git a/duniter4j-es-gchange/src/main/filtered-resources/plugin-descriptor.properties b/duniter4j-es-subscription/src/main/filtered-resources/plugin-descriptor.properties similarity index 73% rename from duniter4j-es-gchange/src/main/filtered-resources/plugin-descriptor.properties rename to duniter4j-es-subscription/src/main/filtered-resources/plugin-descriptor.properties index 9e05cd421741b01be916ac21a8b18930af1454af..78fb7eb1b62a95a65b96bca22b9aebe4e24a7d04 100644 --- a/duniter4j-es-gchange/src/main/filtered-resources/plugin-descriptor.properties +++ b/duniter4j-es-subscription/src/main/filtered-resources/plugin-descriptor.properties @@ -3,7 +3,7 @@ description=Plugin for Gchange API version=${project.version} site=false jvm=true -classname=org.duniter.elasticsearch.gchange.Plugin +classname=org.duniter.elasticsearch.subscription.Plugin java.version=1.7 elasticsearch.version=2.3.3 isolated=false diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/Plugin.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/Plugin.java similarity index 87% rename from duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/Plugin.java rename to duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/Plugin.java index f4db995937fa9f104742f7418679abc48b200ebe..48b5b967e00f1cb37de8c6ed31d715bf8ad454b7 100644 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/Plugin.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/Plugin.java @@ -1,4 +1,4 @@ -package org.duniter.elasticsearch.gchange; +package org.duniter.elasticsearch.subscription; /* * #%L @@ -23,9 +23,9 @@ package org.duniter.elasticsearch.gchange; */ import com.google.common.collect.Lists; -import org.duniter.elasticsearch.gchange.dao.DaoModule; -import org.duniter.elasticsearch.gchange.rest.RestModule; -import org.duniter.elasticsearch.gchange.service.ServiceModule; +import org.duniter.elasticsearch.subscription.dao.DaoModule; +import org.duniter.elasticsearch.subscription.rest.RestModule; +import org.duniter.elasticsearch.subscription.service.ServiceModule; import org.elasticsearch.common.component.LifecycleComponent; import org.elasticsearch.common.inject.Inject; import org.elasticsearch.common.inject.Module; @@ -42,12 +42,12 @@ public class Plugin extends org.elasticsearch.plugins.Plugin { private boolean enable; @Inject public Plugin(Settings settings) { - this.enable = settings.getAsBoolean("gchange.enabled", true); + this.enable = settings.getAsBoolean("subscription.enabled", true); } @Override public String name() { - return "gchange"; + return "subscription"; } @Override diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/PluginInit.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginInit.java similarity index 73% rename from duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/PluginInit.java rename to duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginInit.java index 5b781ea3cd5004a63ca7d3e14058911029364492..6dc6e5ade0a86c39a2f6fa67d52c08872547cd13 100644 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/PluginInit.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginInit.java @@ -1,4 +1,4 @@ -package org.duniter.elasticsearch.gchange; +package org.duniter.elasticsearch.subscription; /* * #%L @@ -22,11 +22,10 @@ package org.duniter.elasticsearch.gchange; * #L% */ -import org.duniter.elasticsearch.gchange.service.MarketService; -import org.duniter.elasticsearch.gchange.service.RegistryService; -import org.duniter.elasticsearch.gchange.service.SynchroService; +import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexDao; +import org.duniter.elasticsearch.subscription.service.SubscriptionService; +import org.duniter.elasticsearch.subscription.service.SynchroService; import org.duniter.elasticsearch.threadpool.ThreadPool; -import org.duniter.elasticsearch.user.PluginSettings; import org.elasticsearch.cluster.health.ClusterHealthStatus; import org.elasticsearch.common.component.AbstractLifecycleComponent; import org.elasticsearch.common.inject.Inject; @@ -43,7 +42,7 @@ public class PluginInit extends AbstractLifecycleComponent<PluginInit> { private final PluginSettings pluginSettings; private final ThreadPool threadPool; private final Injector injector; - private final static ESLogger logger = Loggers.getLogger("gchange"); + private final static ESLogger logger = Loggers.getLogger("duniter.subscription"); @Inject public PluginInit(Settings settings, PluginSettings pluginSettings, ThreadPool threadPool, final Injector injector) { @@ -59,7 +58,7 @@ public class PluginInit extends AbstractLifecycleComponent<PluginInit> { createIndices(); // Waiting cluster back to GREEN or YELLOW state, before synchronize - threadPool.scheduleOnClusterHealthStatus(() -> synchronize(), + threadPool.scheduleOnClusterHealthStatus(this::synchronize, ClusterHealthStatus.YELLOW, ClusterHealthStatus.GREEN); }, ClusterHealthStatus.YELLOW, ClusterHealthStatus.GREEN); } @@ -80,28 +79,24 @@ public class PluginInit extends AbstractLifecycleComponent<PluginInit> { if (reloadIndices) { if (logger.isInfoEnabled()) { - logger.info("Reloading all Gchange indices..."); + logger.info("Reloading all subscription indices..."); } - injector.getInstance(RegistryService.class) - .deleteIndex() - .createIndexIfNotExists(); - injector.getInstance(MarketService.class) + injector.getInstance(SubscriptionIndexDao.class) .deleteIndex() .createIndexIfNotExists(); if (logger.isInfoEnabled()) { - logger.info("Reloading all Gchange indices... [OK]"); + logger.info("Reloading all subscription indices... [OK]"); } } else { if (logger.isInfoEnabled()) { - logger.info("Checking Gchange indices..."); + logger.info("Checking subscription indices..."); } - injector.getInstance(RegistryService.class).createIndexIfNotExists(); - injector.getInstance(MarketService.class).createIndexIfNotExists(); + injector.getInstance(SubscriptionIndexDao.class).createIndexIfNotExists(); if (logger.isInfoEnabled()) { - logger.info("Checking Gchange indices... [OK]"); + logger.info("Checking subscription indices... [OK]"); } } } @@ -112,5 +107,11 @@ public class PluginInit extends AbstractLifecycleComponent<PluginInit> { // Synchronize injector.getInstance(SynchroService.class).synchronize(); } + + // Start subscription services + if (pluginSettings.enableSubscription()) { + + injector.getInstance(SubscriptionService.class).startScheduling(); + } } } diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/PluginSettings.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginSettings.java similarity index 75% rename from duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/PluginSettings.java rename to duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginSettings.java index 826e938d4bd47129ab98c31c7dc24e804c6bc25c..4151475acaa5e70884ffd946285f5fd4f2f75ee6 100644 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/PluginSettings.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/PluginSettings.java @@ -1,4 +1,4 @@ -package org.duniter.elasticsearch.gchange; +package org.duniter.elasticsearch.subscription; /* * #%L @@ -23,31 +23,10 @@ package org.duniter.elasticsearch.gchange; */ -import com.google.common.collect.ImmutableSet; -import org.apache.commons.io.FileUtils; -import org.duniter.core.client.config.Configuration; -import org.duniter.core.client.config.ConfigurationOption; -import org.duniter.core.client.config.ConfigurationProvider; -import org.duniter.core.client.model.local.Peer; -import org.duniter.core.exception.TechnicalException; -import org.duniter.core.util.StringUtils; +import org.duniter.core.util.crypto.KeyPair; import org.elasticsearch.common.component.*; import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.common.settings.Settings; -import org.nuiton.config.ApplicationConfig; -import org.nuiton.config.ApplicationConfigHelper; -import org.nuiton.config.ApplicationConfigProvider; -import org.nuiton.config.ArgumentsParserException; -import org.nuiton.i18n.I18n; -import org.nuiton.i18n.init.DefaultI18nInitializer; -import org.nuiton.i18n.init.UserI18nInitializer; - -import java.io.File; -import java.io.IOException; -import java.util.Locale; -import java.util.Set; - -import static org.nuiton.i18n.I18n.t; +import org.elasticsearch.common.inject.Singleton; /** * Access to configuration options @@ -58,6 +37,12 @@ public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> { private org.duniter.elasticsearch.user.PluginSettings delegate; + private static PluginSettings instance; + + public static final PluginSettings instance() { + return instance; + } + @Inject public PluginSettings(org.elasticsearch.common.settings.Settings settings, org.duniter.elasticsearch.user.PluginSettings delegate) { @@ -67,6 +52,7 @@ public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> { // Add i18n bundle name delegate.addI18nBundleName(getI18nBundleName()); + instance = this; } @Override @@ -88,6 +74,23 @@ public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> { return delegate; } + + public boolean enableSubscription() { + return settings.getAsBoolean("duniter.subscription.enable", Boolean.TRUE); + } + + public String getCesiumUrl() { + return this.settings.get("duniter.subscription.email.cesium.url", "https://g1.duniter.fr"); + } + + /** + * Time interval (millisecond) to send email ? (default: 3600000 = 1h) + * @return + */ + public long getExecuteEmailSubscriptionsInterval() { + return settings.getAsLong("duniter.subscription.email.interval", 36000000l) /*1hour*/; + } + /* -- delegate methods -- */ public boolean reloadIndices() { @@ -170,9 +173,21 @@ public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> { return delegate.getDefaultStringAnalyzer(); } + public KeyPair getNodeKeypair() { + return delegate.getNodeKeypair(); + } + + public boolean isRandomNodeKeypair() { + return delegate.isRandomNodeKeypair(); + } + + public String getNodePubkey() { + return delegate.getNodePubkey(); + } + /* -- protected methods -- */ protected String getI18nBundleName() { - return "duniter4j-es-gchange-i18n"; + return "duniter4j-es-subscription-i18n"; } } diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/AbstractSubscriptionIndexTypeDao.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/AbstractSubscriptionIndexTypeDao.java new file mode 100644 index 0000000000000000000000000000000000000000..693b04d9bc49f07749ca4574e76a642ebeb46a1c --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/AbstractSubscriptionIndexTypeDao.java @@ -0,0 +1,63 @@ +package org.duniter.elasticsearch.subscription.dao; + +/* + * #%L + * UCoin Java Client :: Core API + * %% + * Copyright (C) 2014 - 2015 EIS + * %% + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as + * published by the Free Software Foundation, either version 3 of the + * License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public + * License along with this program. If not, see + * <http://www.gnu.org/licenses/gpl-3.0.html>. + * #L% + */ + + +import com.fasterxml.jackson.core.JsonProcessingException; +import org.duniter.core.client.model.elasticsearch.Record; +import org.duniter.core.client.model.local.LocalEntity; +import org.duniter.core.exception.TechnicalException; +import org.duniter.core.util.ObjectUtils; +import org.duniter.elasticsearch.subscription.PluginSettings; +import org.elasticsearch.search.SearchHit; + +import java.io.IOException; +import java.util.function.Function; +import java.util.function.Predicate; + +/** + * Created by Benoit on 30/03/2015. + */ +public abstract class AbstractSubscriptionIndexTypeDao<T extends AbstractSubscriptionIndexTypeDao> extends org.duniter.elasticsearch.dao.AbstractIndexTypeDao<T> implements SubscriptionIndexTypeDao<T> { + + protected PluginSettings pluginSettings; + + public AbstractSubscriptionIndexTypeDao(String index, String type, PluginSettings pluginSettings) { + super(index, type); + this.pluginSettings = pluginSettings; + } + + @Override + protected void createIndex() throws JsonProcessingException { + throw new TechnicalException("not implemented"); + } + + @Override + public void checkSameDocumentIssuer(String id, String expectedIssuer) { + String issuer = getMandatoryFieldsById(id, Record.PROPERTY_ISSUER).get(Record.PROPERTY_ISSUER).toString(); + if (!ObjectUtils.equals(expectedIssuer, issuer)) { + throw new TechnicalException("Not same issuer"); + } + } + +} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryCategoryAction.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/DaoModule.java similarity index 53% rename from duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryCategoryAction.java rename to duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/DaoModule.java index ff58d64bcb6b65cd5f2d321b66410147daaeaea7..7851afbe668669e3e5a3196a266ca857f1f29b72 100644 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryCategoryAction.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/DaoModule.java @@ -1,4 +1,4 @@ -package org.duniter.elasticsearch.gchange.rest.registry; +package org.duniter.elasticsearch.subscription.dao; /* * #%L @@ -22,18 +22,20 @@ package org.duniter.elasticsearch.gchange.rest.registry; * #L% */ -import org.duniter.elasticsearch.gchange.dao.registry.RegistryIndexDao; -import org.duniter.elasticsearch.gchange.service.RegistryService; -import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.elasticsearch.common.inject.Inject; -import org.elasticsearch.rest.RestRequest; +import org.duniter.elasticsearch.subscription.dao.record.SubscriptionRecordDao; +import org.duniter.elasticsearch.subscription.dao.record.SubscriptionRecordDaoImpl; +import org.elasticsearch.common.inject.AbstractModule; +import org.elasticsearch.common.inject.Module; -public class RestRegistryCategoryAction { +public class DaoModule extends AbstractModule implements Module { - @Inject - public RestRegistryCategoryAction(RestSecurityController securityController) { - // Add security rule for category - securityController.allowIndexType(RestRequest.Method.GET, RegistryIndexDao.INDEX, RegistryIndexDao.CATEGORY_TYPE); + @Override protected void configure() { + + // Subscription index + bind(SubscriptionIndexDao.class).to(SubscriptionIndexDaoImpl.class).asEagerSingleton(); + + // Subscription types + bind(SubscriptionRecordDao.class).to(SubscriptionRecordDaoImpl.class).asEagerSingleton(); } } \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexDao.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexDao.java new file mode 100644 index 0000000000000000000000000000000000000000..f252212c742aee5b7ff7befb95650dbe6465efb1 --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexDao.java @@ -0,0 +1,14 @@ +package org.duniter.elasticsearch.subscription.dao; + +import org.duniter.elasticsearch.dao.IndexDao; +import org.duniter.elasticsearch.dao.IndexTypeDao; + +/** + * Created by blavenie on 03/04/17. + */ +public interface SubscriptionIndexDao extends IndexDao<SubscriptionIndexDao> { + String INDEX = "subscription"; + String CATEGORY_TYPE = "category"; + + SubscriptionIndexDao register(IndexTypeDao<?> indexTypeDao); +} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexDaoImpl.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexDaoImpl.java new file mode 100644 index 0000000000000000000000000000000000000000..a3494113948d677a3939cbe87dc0c90800a59ff3 --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexDaoImpl.java @@ -0,0 +1,100 @@ +package org.duniter.elasticsearch.subscription.dao; + +import com.fasterxml.jackson.core.JsonProcessingException; +import org.duniter.core.exception.TechnicalException; +import org.duniter.elasticsearch.dao.AbstractIndexDao; +import org.duniter.elasticsearch.dao.IndexTypeDao; +import org.duniter.elasticsearch.dao.handler.AddSequenceAttributeHandler; +import org.duniter.elasticsearch.subscription.PluginSettings; +import org.elasticsearch.action.admin.indices.create.CreateIndexRequestBuilder; +import org.elasticsearch.common.inject.Inject; +import org.elasticsearch.common.xcontent.XContentBuilder; +import org.elasticsearch.common.xcontent.XContentFactory; + +import java.io.IOException; +import java.util.ArrayList; +import java.util.List; + +/** + * Created by blavenie on 03/04/17. + */ +public class SubscriptionIndexDaoImpl extends AbstractIndexDao<SubscriptionIndexDao> implements SubscriptionIndexDao { + + + private static final String CATEGORIES_BULK_CLASSPATH_FILE = "subscription-categories-bulk-insert.json"; + + private PluginSettings pluginSettings; + private List<IndexTypeDao<?>> indexTypeDaos = new ArrayList<>(); + + @Inject + public SubscriptionIndexDaoImpl(PluginSettings pluginSettings) { + super(SubscriptionIndexDao.INDEX); + + this.pluginSettings = pluginSettings; + } + + public SubscriptionIndexDao register(IndexTypeDao<?> indexTypeDao) { + indexTypeDaos.add(indexTypeDao); + return this; + } + + @Override + protected void createIndex() throws JsonProcessingException { + logger.info(String.format("Creating index [%s]", INDEX)); + + CreateIndexRequestBuilder createIndexRequestBuilder = client.admin().indices().prepareCreate(INDEX); + org.elasticsearch.common.settings.Settings indexSettings = org.elasticsearch.common.settings.Settings.settingsBuilder() + .put("number_of_shards", 3) + .put("number_of_replicas", 1) + //.put("analyzer", createDefaultAnalyzer()) + .build(); + createIndexRequestBuilder.setSettings(indexSettings); + indexTypeDaos.forEach(indexTypeDao -> createIndexRequestBuilder.addMapping(indexTypeDao.getType(), indexTypeDao.createTypeMapping())); + createIndexRequestBuilder.addMapping(CATEGORY_TYPE, createCategoryTypeMapping()); + createIndexRequestBuilder.execute().actionGet(); + + // Fill categories + fillCategories(); + } + + + + protected XContentBuilder createCategoryTypeMapping() { + try { + XContentBuilder mapping = XContentFactory.jsonBuilder().startObject() + .startObject(CATEGORY_TYPE) + .startObject("properties") + + // name + .startObject("name") + .field("type", "string") + .field("analyzer", pluginSettings.getDefaultStringAnalyzer()) + .endObject() + + // parent + .startObject("parent") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + + .endObject() + .endObject().endObject(); + + return mapping; + } + catch(IOException ioe) { + throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", getIndex(), CATEGORY_TYPE, ioe.getMessage()), ioe); + } + } + + protected void fillCategories() { + if (logger.isDebugEnabled()) { + logger.debug(String.format("[%s/%s] Fill data", getIndex(), CATEGORY_TYPE)); + } + + // Insert categories + client.bulkFromClasspathFile(CATEGORIES_BULK_CLASSPATH_FILE, getIndex(), CATEGORY_TYPE, + // Add order attribute + new AddSequenceAttributeHandler("order", "\\{.*\"name\".*\\}", 1)); + } +} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/RecordDao.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexTypeDao.java similarity index 86% rename from duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/RecordDao.java rename to duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexTypeDao.java index 3313806a9bf69254e10819135870302603557624..1b3637b6fed4f8e8a7ec1fab7d9e81251cb89bcf 100644 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/dao/RecordDao.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/SubscriptionIndexTypeDao.java @@ -1,4 +1,4 @@ -package org.duniter.elasticsearch.gchange.dao; +package org.duniter.elasticsearch.subscription.dao; /* * #%L @@ -28,9 +28,7 @@ import org.duniter.elasticsearch.dao.IndexTypeDao; /** * Created by Benoit on 30/03/2015. */ -public interface RecordDao<T extends RecordDao> extends IndexTypeDao<T> { - - String TYPE = "record"; +public interface SubscriptionIndexTypeDao<T extends SubscriptionIndexTypeDao> extends IndexTypeDao<T> { String create(final String json); diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDao.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDao.java new file mode 100644 index 0000000000000000000000000000000000000000..b8b2703c25d5cb67aadd6f130ca38603ab1f99bf --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDao.java @@ -0,0 +1,16 @@ +package org.duniter.elasticsearch.subscription.dao.record; + +import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexTypeDao; +import org.duniter.elasticsearch.subscription.model.Subscription; + +import java.util.List; + +/** + * Created by blavenie on 03/04/17. + */ +public interface SubscriptionRecordDao<T extends SubscriptionIndexTypeDao> extends SubscriptionIndexTypeDao<T> { + + String TYPE = "record"; + + List<Subscription> getSubscriptions(int from, int size, String recipient, String... types); +} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDaoImpl.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDaoImpl.java new file mode 100644 index 0000000000000000000000000000000000000000..983ae4054bfe7c537a0be9fdca54e1de4ebd1014 --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/dao/record/SubscriptionRecordDaoImpl.java @@ -0,0 +1,141 @@ +package org.duniter.elasticsearch.subscription.dao.record; + +import org.duniter.core.exception.TechnicalException; +import org.duniter.core.util.CollectionUtils; +import org.duniter.elasticsearch.subscription.PluginSettings; +import org.duniter.elasticsearch.subscription.dao.AbstractSubscriptionIndexTypeDao; +import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexDao; +import org.duniter.elasticsearch.subscription.model.Subscription; +import org.duniter.elasticsearch.subscription.model.email.EmailSubscription; +import org.elasticsearch.action.search.SearchResponse; +import org.elasticsearch.action.search.SearchType; +import org.elasticsearch.common.inject.Inject; +import org.elasticsearch.common.xcontent.XContentBuilder; +import org.elasticsearch.common.xcontent.XContentFactory; +import org.elasticsearch.index.query.BoolQueryBuilder; +import org.elasticsearch.index.query.QueryBuilder; +import org.elasticsearch.index.query.QueryBuilders; +import org.elasticsearch.search.SearchHit; + +import java.io.IOException; +import java.util.Arrays; +import java.util.List; +import java.util.Objects; +import java.util.stream.Collectors; + +/** + * Created by blavenie on 03/04/17. + */ +public class SubscriptionRecordDaoImpl extends AbstractSubscriptionIndexTypeDao<SubscriptionRecordDaoImpl> implements SubscriptionRecordDao<SubscriptionRecordDaoImpl> { + + @Inject + public SubscriptionRecordDaoImpl(PluginSettings pluginSettings, SubscriptionIndexDao indexDao) { + super(SubscriptionIndexDao.INDEX, TYPE, pluginSettings); + + indexDao.register(this); + } + + @Override + public List<Subscription> getSubscriptions(int from, int size, String recipient, String... types) { + + BoolQueryBuilder query = QueryBuilders.boolQuery() + .must(QueryBuilders.termQuery(Subscription.PROPERTY_RECIPIENT, recipient)); + if (CollectionUtils.isNotEmpty(types)) { + query.must(QueryBuilders.termsQuery(Subscription.PROPERTY_TYPE, types)); + } + + SearchResponse response = client.prepareSearch(SubscriptionIndexDao.INDEX) + .setTypes(SubscriptionRecordDao.TYPE) + .setSearchType(SearchType.DFS_QUERY_THEN_FETCH) + .setQuery(query) + .setFetchSource(true) + .setFrom(from).setSize(size) + .get(); + + return Arrays.asList(response.getHits().getHits()).stream() + .map(this::toSubscription) + .filter(Objects::nonNull) + .collect(Collectors.toList()); + } + + @Override + public XContentBuilder createTypeMapping() { + try { + XContentBuilder mapping = XContentFactory.jsonBuilder().startObject() + .startObject(getType()) + .startObject("properties") + + // issuer + .startObject("issuer") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + + // recipient + .startObject("recipient") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + + // time + .startObject("time") + .field("type", "integer") + .endObject() + + // nonce + .startObject("nonce") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + + // issuer content + .startObject("issuer_content") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + + // receiver content + .startObject("receiver_content") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + + // hash + .startObject("hash") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + + // signature + .startObject("signature") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + + + .endObject() + .endObject().endObject(); + + return mapping; + } + catch(IOException ioe) { + throw new TechnicalException(String.format("Error while getting mapping for index [%s/%s]: %s", getIndex(), getType(), ioe.getMessage()), ioe); + } + } + + protected Subscription toSubscription(SearchHit searchHit) { + + Subscription record = null; + + if (SubscriptionRecordDao.TYPE.equals(searchHit.getType())) { + record = client.readSourceOrNull(searchHit, EmailSubscription.class); + } + + if (record != null) { + record.setId(searchHit.getId()); + } + + return record; + } + +} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/model/Protocol.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/Protocol.java similarity index 90% rename from duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/model/Protocol.java rename to duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/Protocol.java index 3d5539e649b4b87f468be8a1bcdb9276b2a8613f..ea9d484afe92a2e81ed9c2c32cde18a6730b9d90 100644 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/model/Protocol.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/Protocol.java @@ -1,4 +1,4 @@ -package org.duniter.elasticsearch.gchange.model; +package org.duniter.elasticsearch.subscription.model; /* * #%L @@ -29,5 +29,5 @@ public interface Protocol { String VERSION = "1"; - String GCHANGE_API = "GCHANGE_API"; + String EMAIL_API = "EMAIL_API"; } diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/Subscription.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/Subscription.java new file mode 100644 index 0000000000000000000000000000000000000000..2ffc884ed26b2d120c0a19d6b4facb093119c816 --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/Subscription.java @@ -0,0 +1,101 @@ +package org.duniter.elasticsearch.subscription.model; + +/* + * #%L + * Duniter4j :: ElasticSearch GChange plugin + * %% + * Copyright (C) 2014 - 2017 EIS + * %% + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as + * published by the Free Software Foundation, either version 3 of the + * License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public + * License along with this program. If not, see + * <http://www.gnu.org/licenses/gpl-3.0.html>. + * #L% + */ + +import com.fasterxml.jackson.annotation.JsonIgnore; +import org.duniter.core.client.model.elasticsearch.Record; + +/** + * Created by blavenie on 01/12/16. + */ +public class Subscription<T> extends Record{ + + public static final String PROPERTY_RECIPIENT = "recipient"; + + public static final String PROPERTY_NONCE = "nonce"; + + public static final String PROPERTY_RECIPIENT_CONTENT = "recipientContent"; + + public static final String PROPERTY_ISSUER_CONTENT = "issuerContent"; + + public static final String PROPERTY_CONTENT = "content"; + + public static final String PROPERTY_TYPE = "type"; + + private String recipient; + private String nonce; + private String recipientContent; + private String issuerContent; + private String type; + private T content; + + public String getRecipient() { + return recipient; + } + + public void setRecipient(String recipient) { + this.recipient = recipient; + } + + public String getNonce() { + return nonce; + } + + public void setNonce(String nonce) { + this.nonce = nonce; + } + + public String getRecipientContent() { + return recipientContent; + } + + public void setRecipientContent(String recipientContent) { + this.recipientContent = recipientContent; + } + + public String getIssuerContent() { + return issuerContent; + } + + public void setIssuerContent(String issuerContent) { + this.issuerContent = issuerContent; + } + + public String getType() { + return type; + } + + public void setType(String type) { + this.type = type; + } + + @JsonIgnore + public T getContent() { + return content; + } + + @JsonIgnore + public void setContent(T content) { + this.content = content; + } +} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/email/EmailSubscription.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/email/EmailSubscription.java new file mode 100644 index 0000000000000000000000000000000000000000..663297dfe9dd2d0c1f09ef433d07960690c3c3bd --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/model/email/EmailSubscription.java @@ -0,0 +1,83 @@ +package org.duniter.elasticsearch.subscription.model.email; + +/* + * #%L + * Duniter4j :: ElasticSearch GChange plugin + * %% + * Copyright (C) 2014 - 2017 EIS + * %% + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as + * published by the Free Software Foundation, either version 3 of the + * License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public + * License along with this program. If not, see + * <http://www.gnu.org/licenses/gpl-3.0.html>. + * #L% + */ + +import org.duniter.elasticsearch.subscription.model.Subscription; + +/** + * Created by blavenie on 01/12/16. + */ +public class EmailSubscription extends Subscription<EmailSubscription.Content> { + + public static final String TYPE = "email"; + + public static Content newContent() { + return new EmailSubscription.Content(); + } + + public static class Content { + + public static final String PROPERTY_EMAIL = "email"; + public static final String PROPERTY_LOCALE = "locale"; + public static final String PROPERTY_INCLUDES = "includes"; + public static final String PROPERTY_EXCLUDES = "excludes"; + + private String email; + private String[] includes; + private String[] excludes; + private String locale; + + public String getEmail() { + return email; + } + + public void setEmail(String email) { + this.email = email; + } + + public String[] getIncludes() { + return includes; + } + + public void setIncludes(String[] includes) { + this.includes = includes; + } + + public String[] getExcludes() { + return excludes; + } + + public void setExcludes(String[] excludes) { + this.excludes = excludes; + } + + public String getLocale() { + return locale; + } + + public void setLocale(String locale) { + this.locale = locale; + } + } + +} diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/RestModule.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/RestModule.java new file mode 100644 index 0000000000000000000000000000000000000000..4552038adc7271088fb2397b243f92a1d6d73e3d --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/RestModule.java @@ -0,0 +1,40 @@ +package org.duniter.elasticsearch.subscription.rest; + +/* + * #%L + * duniter4j-elasticsearch-plugin + * %% + * Copyright (C) 2014 - 2016 EIS + * %% + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as + * published by the Free Software Foundation, either version 3 of the + * License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public + * License along with this program. If not, see + * <http://www.gnu.org/licenses/gpl-3.0.html>. + * #L% + */ + +import org.duniter.elasticsearch.subscription.rest.record.RestSubscriptionRecordIndexAction; +import org.duniter.elasticsearch.subscription.rest.record.RestSubscriptionRecordUpdateAction; +import org.duniter.elasticsearch.subscription.rest.record.RestSubscriptionCategoryGetAction; +import org.elasticsearch.common.inject.AbstractModule; +import org.elasticsearch.common.inject.Module; + +public class RestModule extends AbstractModule implements Module { + + @Override protected void configure() { + + // Mail + bind(RestSubscriptionRecordIndexAction.class).asEagerSingleton(); + bind(RestSubscriptionRecordUpdateAction.class).asEagerSingleton(); + bind(RestSubscriptionCategoryGetAction.class).asEagerSingleton(); + } +} \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketCategoryAction.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionCategoryGetAction.java similarity index 75% rename from duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketCategoryAction.java rename to duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionCategoryGetAction.java index f82ce5a75403b02fcede06741eb484dec6a7e4f6..010c12cec297d6f93ddb1dc08ff88f58f075dfb0 100644 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/market/RestMarketCategoryAction.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionCategoryGetAction.java @@ -1,4 +1,4 @@ -package org.duniter.elasticsearch.gchange.rest.market; +package org.duniter.elasticsearch.subscription.rest.record; /* * #%L @@ -22,17 +22,17 @@ package org.duniter.elasticsearch.gchange.rest.market; * #L% */ -import org.duniter.elasticsearch.gchange.dao.market.MarketIndexDao; +import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexDao; import org.duniter.elasticsearch.rest.security.RestSecurityController; import org.elasticsearch.common.inject.Inject; import org.elasticsearch.rest.RestRequest; -public class RestMarketCategoryAction { +public class RestSubscriptionCategoryGetAction { @Inject - public RestMarketCategoryAction(RestSecurityController securityController) { + public RestSubscriptionCategoryGetAction(RestSecurityController securityController) { // Add security rule for category - securityController.allowIndexType(RestRequest.Method.GET, MarketIndexDao.INDEX, MarketIndexDao.CATEGORY_TYPE); + securityController.allowIndexType(RestRequest.Method.GET, SubscriptionIndexDao.INDEX, SubscriptionIndexDao.CATEGORY_TYPE); } } \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryCommentIndexAction.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionRecordIndexAction.java similarity index 62% rename from duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryCommentIndexAction.java rename to duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionRecordIndexAction.java index 7ee815c944e65835bd32e0758803ceb53e0220f8..8a4bb1a6037d68a6bd6809f9b2f7752a5aab593e 100644 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryCommentIndexAction.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionRecordIndexAction.java @@ -1,4 +1,4 @@ -package org.duniter.elasticsearch.gchange.rest.registry; +package org.duniter.elasticsearch.subscription.rest.record; /* * #%L @@ -22,24 +22,24 @@ package org.duniter.elasticsearch.gchange.rest.registry; * #L% */ -import org.duniter.elasticsearch.gchange.dao.registry.RegistryCommentDao; -import org.duniter.elasticsearch.gchange.dao.registry.RegistryIndexDao; +import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexDao; +import org.duniter.elasticsearch.subscription.dao.record.SubscriptionRecordDao; import org.duniter.elasticsearch.rest.AbstractRestPostIndexAction; import org.duniter.elasticsearch.rest.security.RestSecurityController; -import org.duniter.elasticsearch.gchange.service.RegistryService; +import org.duniter.elasticsearch.subscription.service.SubscriptionService; import org.elasticsearch.client.Client; import org.elasticsearch.common.inject.Inject; import org.elasticsearch.common.settings.Settings; import org.elasticsearch.rest.RestController; -public class RestRegistryCommentIndexAction extends AbstractRestPostIndexAction { +public class RestSubscriptionRecordIndexAction extends AbstractRestPostIndexAction { + @Inject - public RestRegistryCommentIndexAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - RegistryService service) { + public RestSubscriptionRecordIndexAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, + SubscriptionService service) { super(settings, controller, client, securityController, - RegistryIndexDao.INDEX, RegistryCommentDao.TYPE, - json -> service.indexCommentFromJson(json)); + SubscriptionIndexDao.INDEX, SubscriptionRecordDao.TYPE, + json -> service.create(json)); } - } \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryRecordUpdateAction.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionRecordUpdateAction.java similarity index 62% rename from duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryRecordUpdateAction.java rename to duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionRecordUpdateAction.java index 4131c882dac6b3142f192462c16f9611e72ef8e8..2a2659578808e5cababc1446d6ef75e9855b36d0 100644 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/rest/registry/RestRegistryRecordUpdateAction.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/rest/record/RestSubscriptionRecordUpdateAction.java @@ -1,4 +1,4 @@ -package org.duniter.elasticsearch.gchange.rest.registry; +package org.duniter.elasticsearch.subscription.rest.record; /* * #%L @@ -22,9 +22,9 @@ package org.duniter.elasticsearch.gchange.rest.registry; * #L% */ -import org.duniter.elasticsearch.gchange.dao.registry.RegistryIndexDao; -import org.duniter.elasticsearch.gchange.dao.registry.RegistryRecordDao; -import org.duniter.elasticsearch.gchange.service.RegistryService; +import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexDao; +import org.duniter.elasticsearch.subscription.dao.record.SubscriptionRecordDao; +import org.duniter.elasticsearch.subscription.service.SubscriptionService; import org.duniter.elasticsearch.rest.AbstractRestPostUpdateAction; import org.duniter.elasticsearch.rest.security.RestSecurityController; import org.elasticsearch.client.Client; @@ -32,14 +32,14 @@ import org.elasticsearch.common.inject.Inject; import org.elasticsearch.common.settings.Settings; import org.elasticsearch.rest.RestController; -public class RestRegistryRecordUpdateAction extends AbstractRestPostUpdateAction { +public class RestSubscriptionRecordUpdateAction extends AbstractRestPostUpdateAction { @Inject - public RestRegistryRecordUpdateAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, - RegistryService service) { + public RestSubscriptionRecordUpdateAction(Settings settings, RestController controller, Client client, RestSecurityController securityController, + SubscriptionService service) { super(settings, controller, client, securityController, - RegistryIndexDao.INDEX, RegistryRecordDao.TYPE, - (id, json) -> service.updateRecordFromJson(id, json)); + SubscriptionIndexDao.INDEX, SubscriptionRecordDao.TYPE, + (id, json) -> service.update(id, json)); } } \ No newline at end of file diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/AbstractService.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/AbstractService.java similarity index 88% rename from duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/AbstractService.java rename to duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/AbstractService.java index 9ac606915729fb4b9c6370e5daf2b12191dc232b..a900a2b283f8d414186542b8a876a1f2a4343baa 100644 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/AbstractService.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/AbstractService.java @@ -1,4 +1,4 @@ -package org.duniter.elasticsearch.gchange.service; +package org.duniter.elasticsearch.subscription.service; /* * #%L @@ -24,8 +24,7 @@ package org.duniter.elasticsearch.gchange.service; import org.duniter.core.service.CryptoService; import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.gchange.PluginSettings; -import org.elasticsearch.client.Client; +import org.duniter.elasticsearch.subscription.PluginSettings; /** * Created by blavenie on 10/01/17. @@ -43,7 +42,7 @@ public abstract class AbstractService extends org.duniter.elasticsearch.user.ser } public AbstractService(Duniter4jClient client, PluginSettings pluginSettings, CryptoService cryptoService) { - this("duniter.gchange", client, pluginSettings, cryptoService); + this("duniter.subscription", client, pluginSettings, cryptoService); } public AbstractService(String loggerName, Duniter4jClient client, PluginSettings pluginSettings, CryptoService cryptoService) { diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/ServiceModule.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/ServiceModule.java similarity index 82% rename from duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/ServiceModule.java rename to duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/ServiceModule.java index a3d5f3d32290d7d4d21cc65003dceb5772fb27c7..2bb9c7b904fd442e307682c72d1b5bb3544dfe1a 100644 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/ServiceModule.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/ServiceModule.java @@ -1,4 +1,4 @@ -package org.duniter.elasticsearch.gchange.service; +package org.duniter.elasticsearch.subscription.service; /* * #%L @@ -28,9 +28,10 @@ import org.elasticsearch.common.inject.Module; public class ServiceModule extends AbstractModule implements Module { @Override protected void configure() { - bind(RegistryService.class).asEagerSingleton(); - bind(CommentUserEventService.class).asEagerSingleton(); - bind(MarketService.class).asEagerSingleton(); + // Subscription services + bind(SubscriptionService.class).asEagerSingleton(); + + // Synchro service bind(SynchroService.class).asEagerSingleton(); } } \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/SubscriptionService.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/SubscriptionService.java new file mode 100644 index 0000000000000000000000000000000000000000..7571a7f6cbb2f3bde4d7df979e5678fc4f0019d4 --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/SubscriptionService.java @@ -0,0 +1,358 @@ +package org.duniter.elasticsearch.subscription.service; + +/* + * #%L + * UCoin Java Client :: Core API + * %% + * Copyright (C) 2014 - 2015 EIS + * %% + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as + * published by the Free Software Foundation, either version 3 of the + * License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public + * License along with this program. If not, see + * <http://www.gnu.org/licenses/gpl-3.0.html>. + * #L% + */ + + +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.databind.JsonNode; +import com.google.common.collect.ImmutableSet; +import org.duniter.core.client.model.ModelUtils; +import org.duniter.core.client.model.bma.jackson.JacksonUtils; +import org.duniter.core.client.model.elasticsearch.Record; +import org.duniter.core.exception.TechnicalException; +import org.duniter.core.service.CryptoService; +import org.duniter.core.util.CollectionUtils; +import org.duniter.core.util.Preconditions; +import org.duniter.core.util.StringUtils; +import org.duniter.core.util.crypto.CryptoUtils; +import org.duniter.elasticsearch.client.Duniter4jClient; +import org.duniter.elasticsearch.subscription.PluginSettings; +import org.duniter.elasticsearch.subscription.dao.record.SubscriptionRecordDao; +import org.duniter.elasticsearch.subscription.model.Subscription; +import org.duniter.elasticsearch.subscription.model.email.EmailSubscription; +import org.duniter.elasticsearch.subscription.util.stringtemplate.DateRenderer; +import org.duniter.elasticsearch.subscription.util.stringtemplate.I18nRenderer; +import org.duniter.elasticsearch.threadpool.ThreadPool; +import org.duniter.elasticsearch.user.model.UserEvent; +import org.duniter.elasticsearch.user.service.AdminService; +import org.duniter.elasticsearch.user.service.MailService; +import org.duniter.elasticsearch.user.service.UserEventService; +import org.duniter.elasticsearch.user.service.UserService; +import org.elasticsearch.common.inject.Inject; +import org.elasticsearch.common.unit.TimeValue; +import org.nuiton.i18n.I18n; +import org.stringtemplate.v4.*; + +import java.util.*; +import java.util.stream.Collectors; + +/** + * Created by Benoit on 30/03/2015. + */ +public class SubscriptionService extends AbstractService { + + private SubscriptionRecordDao subscriptionRecordDao; + private ThreadPool threadPool; + private MailService mailService; + private AdminService adminService; + private UserEventService userEventService; + private UserService userService; + private String emailSubjectPrefix; + + @Inject + public SubscriptionService(Duniter4jClient client, + PluginSettings settings, + CryptoService cryptoService, + SubscriptionRecordDao subscriptionRecordDao, + ThreadPool threadPool, + MailService mailService, + AdminService adminService, + UserService userService, + UserEventService userEventService) { + super("subscription.service", client, settings, cryptoService); + this.subscriptionRecordDao = subscriptionRecordDao; + this.threadPool = threadPool; + this.mailService = mailService; + this.adminService = adminService; + this.userService = userService; + this.userEventService = userEventService; + this.emailSubjectPrefix = pluginSettings.getMailSubjectPrefix().trim(); + if (StringUtils.isNotBlank(emailSubjectPrefix)) { + emailSubjectPrefix += " "; // add one trailing space + } + } + + public String create(String json) { + JsonNode actualObj = readAndVerifyIssuerSignature(json); + String issuer = getIssuer(actualObj); + + if (logger.isDebugEnabled()) { + logger.debug(String.format("Indexing a subscription from issuer [%s]", issuer.substring(0, 8))); + } + + return subscriptionRecordDao.create(json); + } + + public void update(String id, String json) { + JsonNode actualObj = readAndVerifyIssuerSignature(json); + String issuer = getIssuer(actualObj); + + // Check same document issuer + subscriptionRecordDao.checkSameDocumentIssuer(id, issuer); + + if (logger.isDebugEnabled()) { + logger.debug(String.format("Updating subscription [%s] from issuer [%s]", id, issuer.substring(0, 8))); + } + + subscriptionRecordDao.update(id, json); + } + + public SubscriptionService startScheduling() { + if (!pluginSettings.getMailEnable()) { + logger.warn(I18n.t("duniter4j.es.subscription.error.mailDisabling")); + return this; + } + + threadPool.scheduleWithFixedDelay( + this::executeEmailSubscriptions, + new TimeValue(pluginSettings.getExecuteEmailSubscriptionsInterval())); + + return this; + } + + public void executeEmailSubscriptions() { + + final String senderPubkey = pluginSettings.getNodePubkey(); + + int from = 0; + int size = 10; + + boolean hasMore = true; + while (hasMore) { + List<Subscription> subscriptions = subscriptionRecordDao.getSubscriptions(from, size, senderPubkey, EmailSubscription.TYPE); + + // Get profiles titles, for issuers and the sender + Set<String> issuers = subscriptions.stream() + .map(Subscription::getIssuer) + .distinct() + .collect(Collectors.toSet()); + final Map<String, String> profileTitles = userService.getProfileTitles( + ImmutableSet.<String>builder().addAll(issuers).add(senderPubkey).build()); + final String senderName = (profileTitles != null && profileTitles.containsKey(senderPubkey)) ? profileTitles.get(senderPubkey) : + ModelUtils.minifyPubkey(senderPubkey); + + subscriptions.stream() + .map(record -> decryptEmailSubscription((EmailSubscription)record)) + .filter(Objects::nonNull) + .map(record -> processEmailSubscription(record, senderPubkey, senderName, profileTitles)) + .filter(Objects::nonNull) + .forEach(this::saveSubscription); + + hasMore = CollectionUtils.size(subscriptions) >= size; + from += size; + } + } + + /* -- protected methods -- */ + + + protected EmailSubscription decryptEmailSubscription(EmailSubscription subscription) { + Preconditions.checkNotNull(subscription); + Preconditions.checkNotNull(subscription.getId()); + + if (StringUtils.isBlank(subscription.getRecipientContent()) || StringUtils.isBlank(subscription.getNonce()) || + StringUtils.isBlank(subscription.getIssuer())) { + logger.error(String.format("Invalid subscription [%s]. Missing field 'recipientContent', 'nonce' or 'issuer'.", subscription.getId())); + return null; + } + + String jsonContent; + try { + jsonContent = cryptoService.openBox(subscription.getRecipientContent(), + CryptoUtils.decodeBase58(subscription.getNonce()), + CryptoUtils.decodeBase58(subscription.getIssuer()), + pluginSettings.getNodeKeypair().getSecKey() + ); + } catch(Exception e) { + logger.error(String.format("Could not decrypt email subscription content for subscription [%s]", subscription.getId())); + return null; + } + + try { + EmailSubscription.Content content = objectMapper.readValue(jsonContent, EmailSubscription.Content.class); + subscription.setContent(content); + } catch(Exception e) { + logger.error(String.format("Could not parse email subscription content [%s]: %s", jsonContent, e.getMessage())); + return null; + } + + return subscription; + } + + protected EmailSubscription processEmailSubscription(final EmailSubscription subscription, + final String senderPubkey, + final String senderName, + final Map<String, String> profileTitles) { + Preconditions.checkNotNull(subscription); + + logger.info(String.format("Processing email subscription [%s]", subscription.getId())); + + Long lastTime = 0l; // TODO get it from subscription ? + + // Get last user events + String[] includes = subscription.getContent() == null ? null : subscription.getContent().getIncludes(); + String[] excludes = subscription.getContent() == null ? null : subscription.getContent().getExcludes(); + List<UserEvent> userEvents = userEventService.getUserEvents(subscription.getIssuer(), lastTime, includes, excludes); + + + STGroup templates = new STGroupDir("templates", '$', '$'); + templates.registerRenderer(Date.class, new DateRenderer()); + //templates.registerRenderer(String.class, new StringRenderer()); + //templates.registerRenderer(Number.class, new NumberRenderer()); + templates.registerRenderer(String.class, new I18nRenderer()); + String[] localParts = subscription.getContent() != null && subscription.getContent().getLocale() != null ? + subscription.getContent().getLocale().split("-") : new String[]{"en", "GB"}; + + Locale issuerLocale = localParts.length >= 2 ? new Locale(localParts[0].toLowerCase(), localParts[1].toUpperCase()) : new Locale(localParts[0].toLowerCase()); + + // Compute text + String text = fillTemplate( + templates.getInstanceOf("text_email"), + subscription, + senderPubkey, + senderName, + profileTitles, + userEvents, + pluginSettings.getCesiumUrl()) + .render(issuerLocale); + + // Compute HTML content + String html = fillTemplate( + templates.getInstanceOf("html_email_content"), + subscription, + senderPubkey, + senderName, + profileTitles, + userEvents, + pluginSettings.getCesiumUrl()) + .render(issuerLocale); + + mailService.sendHtmlEmailWithText( + emailSubjectPrefix + I18n.t("duniter4j.es.subscription.email.subject", userEvents.size()), + text, + "<body>" + html + "</body>", + subscription.getContent().getEmail()); + return subscription; + } + + + public static ST fillTemplate(ST template, + EmailSubscription subscription, + String senderPubkey, + String senderName, + Map<String, String> issuerProfilNames, + List<UserEvent> userEvents, + String cesiumSiteUrl) { + String issuerName = issuerProfilNames != null && issuerProfilNames.containsKey(subscription.getIssuer()) ? + issuerProfilNames.get(subscription.getIssuer()) : + ModelUtils.minifyPubkey(subscription.getIssuer()); + + + try { + // Compute body + template.add("url", cesiumSiteUrl); + template.add("issuer", issuerName); + template.add("senderPubkey", senderPubkey); + template.add("senderName", senderName); + userEvents.forEach(userEvent -> { + String description = userEvent.getParams() != null ? + I18n.t("duniter.user.event." + userEvent.getCode().toUpperCase(), userEvent.getParams()) : + I18n.t("duniter.user.event." + userEvent.getCode().toUpperCase()); + template.addAggr("events.{description, time}", new Object[]{ + description, + new Date(userEvent.getTime() * 1000) + }); + + }); + + return template; + + } + catch (Exception e) { + throw new TechnicalException(e); + } + } + + + public static String computeTextEmail(STGroup templates, + Locale issuerLocale, + EmailSubscription subscription, + String senderPubkey, + String senderName, + Map<String, String> issuerProfilNames, + List<UserEvent> userEvents, + String cesiumSiteUrl) { + String issuerName = issuerProfilNames != null && issuerProfilNames.containsKey(subscription.getIssuer()) ? + issuerProfilNames.get(subscription.getIssuer()) : + ModelUtils.minifyPubkey(subscription.getIssuer()); + + try { + // Compute text content + ST tpl = templates.getInstanceOf("text_email"); + tpl.add("url", cesiumSiteUrl); + tpl.add("issuer", issuerName); + tpl.add("url", cesiumSiteUrl); + tpl.add("senderPubkey", senderPubkey); + tpl.add("senderName", senderName); + userEvents.forEach(userEvent -> { + String description = userEvent.getParams() != null ? + I18n.t("duniter.user.event." + userEvent.getCode().toUpperCase(), userEvent.getParams()) : + I18n.t("duniter.user.event." + userEvent.getCode().toUpperCase()); + tpl.addAggr("events.{description, time}", new Object[]{ + description, + new Date(userEvent.getTime() * 1000) + }); + + }); + + return tpl.render(); + } + catch (Exception e) { + throw new TechnicalException(e); + } + } + + protected EmailSubscription saveSubscription(EmailSubscription subscription) { + Preconditions.checkNotNull(subscription); + + //mailService.sendEmail(); + return subscription; + } + + private String toJson(EmailSubscription subscription) { + return toJson(subscription, false); + } + + private String toJson(EmailSubscription subscription, boolean cleanHashAndSignature) { + try { + String json = objectMapper.writeValueAsString(subscription); + if (cleanHashAndSignature) { + json = JacksonUtils.removeAttribute(json, Record.PROPERTY_SIGNATURE); + json = JacksonUtils.removeAttribute(json, Record.PROPERTY_HASH); + } + return json; + } catch(JsonProcessingException e) { + throw new TechnicalException("Unable to serialize UserEvent object", e); + } + } +} diff --git a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/SynchroService.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/SynchroService.java similarity index 56% rename from duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/SynchroService.java rename to duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/SynchroService.java index 977f23303b90c82c3565935d680579c1392b13a8..94421128d7fda00c7af6cfb5314658be26c6d57f 100644 --- a/duniter4j-es-gchange/src/main/java/org/duniter/elasticsearch/gchange/service/SynchroService.java +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/service/SynchroService.java @@ -1,4 +1,4 @@ -package org.duniter.elasticsearch.gchange.service; +package org.duniter.elasticsearch.subscription.service; /* * #%L @@ -25,19 +25,14 @@ package org.duniter.elasticsearch.gchange.service; import org.duniter.core.client.model.local.Peer; import org.duniter.core.service.CryptoService; import org.duniter.elasticsearch.client.Duniter4jClient; -import org.duniter.elasticsearch.gchange.dao.market.MarketCommentDao; -import org.duniter.elasticsearch.gchange.dao.market.MarketIndexDao; -import org.duniter.elasticsearch.gchange.dao.market.MarketRecordDao; -import org.duniter.elasticsearch.gchange.dao.registry.RegistryCommentDao; -import org.duniter.elasticsearch.gchange.dao.registry.RegistryIndexDao; -import org.duniter.elasticsearch.gchange.dao.registry.RegistryRecordDao; -import org.duniter.elasticsearch.gchange.model.Protocol; import org.duniter.elasticsearch.model.SynchroResult; import org.duniter.elasticsearch.service.AbstractSynchroService; import org.duniter.elasticsearch.service.ServiceLocator; +import org.duniter.elasticsearch.subscription.dao.SubscriptionIndexDao; +import org.duniter.elasticsearch.subscription.dao.record.SubscriptionRecordDao; +import org.duniter.elasticsearch.subscription.model.Protocol; import org.duniter.elasticsearch.threadpool.ThreadPool; import org.duniter.elasticsearch.user.PluginSettings; -import org.elasticsearch.client.Client; import org.elasticsearch.common.inject.Inject; /** @@ -52,8 +47,8 @@ public class SynchroService extends AbstractSynchroService { } public void synchronize() { - logger.info("Synchronizing Gchange data..."); - Peer peer = getPeerFromAPI(Protocol.GCHANGE_API); + logger.info("Synchronizing subscription data..."); + Peer peer = getPeerFromAPI(Protocol.EMAIL_API); synchronize(peer); } @@ -63,25 +58,18 @@ public class SynchroService extends AbstractSynchroService { long sinceTime = 0; // TODO: get last sync time from somewhere ? (e.g. a specific index) - logger.info(String.format("[%s] Synchronizing gchange data since %s...", peer.toString(), sinceTime)); + logger.info(String.format("[%s] Synchronizing subscription data since %s...", peer.toString(), sinceTime)); SynchroResult result = new SynchroResult(); long time = System.currentTimeMillis(); - importMarketChanges(result, peer, sinceTime); - importRegistryChanges(result, peer, sinceTime); + importMailChanges(result, peer, sinceTime); long duration = System.currentTimeMillis() - time; - logger.info(String.format("[%s] Synchronizing gchange data since %s [OK] %s (in %s ms)", peer.toString(), sinceTime, result.toString(), duration)); + logger.info(String.format("[%s] Synchronizing subscription data since %s [OK] %s (in %s ms)", peer.toString(), sinceTime, result.toString(), duration)); } - protected void importMarketChanges(SynchroResult result, Peer peer, long sinceTime) { - importChanges(result, peer, MarketIndexDao.INDEX, MarketRecordDao.TYPE, sinceTime); - importChanges(result, peer, MarketIndexDao.INDEX, MarketCommentDao.TYPE, sinceTime); - } - - protected void importRegistryChanges(SynchroResult result, Peer peer, long sinceTime) { - importChanges(result, peer, RegistryIndexDao.INDEX, RegistryRecordDao.TYPE, sinceTime); - importChanges(result, peer, RegistryIndexDao.INDEX, RegistryCommentDao.TYPE, sinceTime); + protected void importMailChanges(SynchroResult result, Peer peer, long sinceTime) { + importChanges(result, peer, SubscriptionIndexDao.INDEX, SubscriptionRecordDao.TYPE, sinceTime); } } diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/DateRenderer.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/DateRenderer.java new file mode 100644 index 0000000000000000000000000000000000000000..eb84ec8d3e5a9c10b48a9683c89e8451eb0ed964 --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/DateRenderer.java @@ -0,0 +1,43 @@ +package org.duniter.elasticsearch.subscription.util.stringtemplate; + +import org.stringtemplate.v4.AttributeRenderer; + +import java.text.DateFormat; +import java.text.SimpleDateFormat; +import java.util.*; + +public class DateRenderer implements AttributeRenderer { + + public DateRenderer() { + } + + public String toString(Object o, String formatString, Locale locale) { + if(formatString == null) { + formatString = "short"; + } + + Date d; + if(o instanceof Calendar) { + d = ((Calendar)o).getTime(); + } else { + d = (Date)o; + } + + Integer styleI = (Integer)org.stringtemplate.v4.DateRenderer.formatToInt.get(formatString); + Object f; + if(styleI == null) { + f = new SimpleDateFormat(formatString, locale); + } else { + int style = styleI.intValue(); + if(formatString.startsWith("date:")) { + f = DateFormat.getDateInstance(style, locale); + } else if(formatString.startsWith("time:")) { + f = DateFormat.getTimeInstance(style, locale); + } else { + f = DateFormat.getDateTimeInstance(style, style, locale); + } + } + + return ((DateFormat)f).format(d); + } +} \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/I18nRenderer.java b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/I18nRenderer.java new file mode 100644 index 0000000000000000000000000000000000000000..fbcd6a9815c6b4f90681bbe9dfa1e3dfac6d77c1 --- /dev/null +++ b/duniter4j-es-subscription/src/main/java/org/duniter/elasticsearch/subscription/util/stringtemplate/I18nRenderer.java @@ -0,0 +1,24 @@ +package org.duniter.elasticsearch.subscription.util.stringtemplate; + +import org.duniter.core.util.CollectionUtils; +import org.duniter.core.util.StringUtils; +import org.nuiton.i18n.I18n; +import org.stringtemplate.v4.AttributeRenderer; + +import java.util.Locale; + +/** + * Created by blavenie on 10/04/17. + */ +public class I18nRenderer implements AttributeRenderer{ + + @Override + public String toString(Object key, String formatString, Locale locale) { + if (formatString == null || !formatString.startsWith("i18n")) return key.toString(); + String[] params = formatString.startsWith("i18n:") ? formatString.substring(5).split(",") : null; + if (CollectionUtils.isNotEmpty(params)) { + return I18n.l(locale, key.toString(), params); + } + return I18n.l(locale, key.toString()); + } +} diff --git a/duniter4j-es-gchange/src/main/misc/index.sh b/duniter4j-es-subscription/src/main/misc/index.sh similarity index 100% rename from duniter4j-es-gchange/src/main/misc/index.sh rename to duniter4j-es-subscription/src/main/misc/index.sh diff --git a/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_en_GB.properties b/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_en_GB.properties new file mode 100644 index 0000000000000000000000000000000000000000..d179ccf76f11ef48db2992de8b2786ca3605d8b1 --- /dev/null +++ b/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_en_GB.properties @@ -0,0 +1,7 @@ +duniter4j.es.subscription.email.footer.disableHelp=You can disable this email notification service in the page <a href\="%s">Online services</a> of Cesium+. +duniter4j.es.subscription.email.footer.sendBy=This email has sent you the Cesium+ node of <a href\="%s">%s</a>. +duniter4j.es.subscription.email.header=Hello <b>%s</b>\!<br/>You received %s new notifications\: +duniter4j.es.subscription.email.notificationsDivider=Notifications list\: +duniter4j.es.subscription.email.openCesium=Open Cesium+ +duniter4j.es.subscription.email.subject=You received %s new notifications +duniter4j.es.subscription.error.mailDisabling=Unable to process email subscriptions\: Email sending is disabled in the configuration diff --git a/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_fr_FR.properties b/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_fr_FR.properties new file mode 100644 index 0000000000000000000000000000000000000000..37f541ed87399b8b713af82885a4606d43de043c --- /dev/null +++ b/duniter4j-es-subscription/src/main/resources/i18n/duniter4j-es-subscription_fr_FR.properties @@ -0,0 +1,11 @@ +duniter4j.es.subscription.email.footer=Cet email vous a été envoyé par Cesium+.<br/><small>vous pouvez désactiver ce service de notification par email, dans la rubrique <a href\="%s">Services en ligne</a> de Cesium+. +duniter4j.es.subscription.email.footer.disableHelp=Vous pouvez désactiver ce service de notification par email, dans la rubrique <a href\="%s">Services en ligne</a> de Cesium+. +duniter4j.es.subscription.email.footer.disableHelp.text=Vous pouvez désactiver ce service de notification par email, dans la rubrique "Services en ligne" de Cesium+ (%s). +duniter4j.es.subscription.email.footer.sendBy=Cet email vous a été envoyé le noeud Cesium+ de <a href\="%1$s">%2$s</a>. +duniter4j.es.subscription.email.footer.sendBy.text=Cet email vous a été envoyé le noeud Cesium+ de %2$s (%1$s). +duniter4j.es.subscription.email.header=Bonjour <b>%s</b> \!<br/>Vous avez %s notifications non lues. +duniter4j.es.subscription.email.header.text=Bonjour %s \!\nVous avez %s notifications non lues. +duniter4j.es.subscription.email.notificationsDivider=Liste des notifications \: +duniter4j.es.subscription.email.openCesium=Ouvrir Cesium+ +duniter4j.es.subscription.email.subject=%s nouvelles notifications non lues +duniter4j.es.subscription.error.mailDisabling=Impossible de traiter les abonnements email\: la fonction d'envoi d'email est désactivée dans la configuration diff --git a/duniter4j-es-gchange/src/main/resources/plugin-security.policy b/duniter4j-es-subscription/src/main/resources/plugin-security.policy similarity index 100% rename from duniter4j-es-gchange/src/main/resources/plugin-security.policy rename to duniter4j-es-subscription/src/main/resources/plugin-security.policy diff --git a/duniter4j-es-subscription/src/main/resources/subscription-categories-bulk-insert.json b/duniter4j-es-subscription/src/main/resources/subscription-categories-bulk-insert.json new file mode 100644 index 0000000000000000000000000000000000000000..37e83c19b85c81b895b879e06500aad785ba21ce --- /dev/null +++ b/duniter4j-es-subscription/src/main/resources/subscription-categories-bulk-insert.json @@ -0,0 +1,9 @@ +{ "index": { "_id": "_notification"}} +{ "name": "Notifications" , "parent": null} +{ "index": { "_id": "email"}} +{ "name": "Email", "parent": "_notification"} + +{ "index": { "_id": "_service"}} +{ "name": "Services de paiement" , "parent": null} +{ "index": { "_id": "transfer"}} +{ "name": "Virement automatique", "parent": "_service"} \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/templates/css.st b/duniter4j-es-subscription/src/main/resources/templates/css.st new file mode 100644 index 0000000000000000000000000000000000000000..30a453a57d32d5e0454ee6fb2328e570ffb69a36 --- /dev/null +++ b/duniter4j-es-subscription/src/main/resources/templates/css.st @@ -0,0 +1,10738 @@ +css() ::= << + html, body, div, span, applet, object, iframe, + h1, h2, h3, h4, h5, h6, p, blockquote, pre, + a, abbr, acronym, address, big, cite, code, + del, dfn, em, img, ins, kbd, q, s, samp, + small, strike, strong, sub, sup, tt, var, + b, i, u, center, + dl, dt, dd, ol, ul, li, + fieldset, form, label, legend, + table, caption, tbody, tfoot, thead, tr, th, td, + article, aside, canvas, details, embed, fieldset, + figure, figcaption, footer, header, hgroup, + menu, nav, output, ruby, section, summary, + time, mark, audio, video { + margin: 0; + padding: 0; + border: 0; + vertical-align: baseline; + font: inherit; + font-size: 100%; } + + /** + * Hide the `template` element in IE, Safari, and Firefox < 22. + */ + [hidden], + template { + display: none; } + + script { + display: none !important; } + + /* ========================================================================== + Base + ========================================================================== */ + /** + * 1. Set default font family to sans-serif. + * 2. Prevent iOS text size adjust after orientation change, without disabling + * user zoom. + */ + html { + -webkit-user-select: none; + -moz-user-select: none; + -ms-user-select: none; + user-select: none; + font-family: sans-serif; + /* 1 */ + -webkit-text-size-adjust: 100%; + -ms-text-size-adjust: 100%; + /* 2 */ + -webkit-text-size-adjust: 100%; + /* 2 */ } + + /** + * Remove default margin. + */ + body { + margin: 0; + line-height: 1; } + + /** + * Remove default outlines. + */ + a, + button, + :focus, + a:focus, + button:focus, + a:active, + a:hover { + outline: 0; } + + /* * + * Remove tap highlight color + */ + a { + -webkit-user-drag: none; + -webkit-tap-highlight-color: transparent; + -webkit-tap-highlight-color: transparent; } + a[href]:hover { + cursor: pointer; } + + /* ========================================================================== + Typography + ========================================================================== */ + /** + * Address style set to `bolder` in Firefox 4+, Safari 5, and Chrome. + */ + b, + strong { + font-weight: bold; } + + /** + * Address styling not present in Safari 5 and Chrome. + */ + dfn { + font-style: italic; } + + /** + * Address differences between Firefox and other browsers. + */ + hr { + -moz-box-sizing: content-box; + box-sizing: content-box; + height: 0; } + + /** + * Correct font family set oddly in Safari 5 and Chrome. + */ + code, + kbd, + pre, + samp { + font-size: 1em; + font-family: monospace, serif; } + + /** + * Improve readability of pre-formatted text in all browsers. + */ + pre { + white-space: pre-wrap; } + + /** + * Set consistent quote types. + */ + q { + quotes: "\201C" "\201D" "\2018" "\2019"; } + + /** + * Address inconsistent and variable font size in all browsers. + */ + small { + font-size: 80%; } + + /** + * Prevent `sub` and `sup` affecting `line-height` in all browsers. + */ + sub, + sup { + position: relative; + vertical-align: baseline; + font-size: 75%; + line-height: 0; } + + sup { + top: -0.5em; } + + sub { + bottom: -0.25em; } + + /** + * Define consistent border, margin, and padding. + */ + fieldset { + margin: 0 2px; + padding: 0.35em 0.625em 0.75em; + border: 1px solid #c0c0c0; } + + /** + * 1. Correct `color` not being inherited in IE 8/9. + * 2. Remove padding so people aren't caught out if they zero out fieldsets. + */ + legend { + padding: 0; + /* 2 */ + border: 0; + /* 1 */ } + + /** + * 1. Correct font family not being inherited in all browsers. + * 2. Correct font size not being inherited in all browsers. + * 3. Address margins set differently in Firefox 4+, Safari 5, and Chrome. + * 4. Remove any default :focus styles + * 5. Make sure webkit font smoothing is being inherited + * 6. Remove default gradient in Android Firefox / FirefoxOS + */ + button, + input, + select, + textarea { + margin: 0; + /* 3 */ + font-size: 100%; + /* 2 */ + font-family: inherit; + /* 1 */ + outline-offset: 0; + /* 4 */ + outline-style: none; + /* 4 */ + outline-width: 0; + /* 4 */ + -webkit-font-smoothing: inherit; + /* 5 */ + background-image: none; + /* 6 */ } + + /** + * Address Firefox 4+ setting `line-height` on `input` using `importnt` in + * the UA stylesheet. + */ + button, + input { + line-height: normal; } + + /** + * Address inconsistent `text-transform` inheritance for `button` and `select`. + * All other form control elements do not inherit `text-transform` values. + * Correct `button` style inheritance in Chrome, Safari 5+, and IE 8+. + * Correct `select` style inheritance in Firefox 4+ and Opera. + */ + button, + select { + text-transform: none; } + + /** + * 1. Avoid the WebKit bug in Android 4.0.* where (2) destroys native `audio` + * and `video` controls. + * 2. Correct inability to style clickable `input` types in iOS. + * 3. Improve usability and consistency of cursor style between image-type + * `input` and others. + */ + button, + html input[type="button"], + input[type="reset"], + input[type="submit"] { + cursor: pointer; + /* 3 */ + -webkit-appearance: button; + /* 2 */ } + + /** + * Re-set default cursor for disabled elements. + */ + button[disabled], + html input[disabled] { + cursor: default; } + + /** + * Remove inner padding and border in Firefox 4+. + */ + button::-moz-focus-inner, + input::-moz-focus-inner { + padding: 0; + border: 0; } + + img { + -webkit-user-drag: none; } + + /** + * Scaffolding + * -------------------------------------------------- + */ + *, + *:before, + *:after { + -webkit-box-sizing: border-box; + -moz-box-sizing: border-box; + box-sizing: border-box; } + + html { + overflow: hidden; + -ms-touch-action: pan-y; + touch-action: pan-y; } + + body, + .ionic-body { + -webkit-touch-callout: none; + -webkit-font-smoothing: antialiased; + font-smoothing: antialiased; + -webkit-text-size-adjust: none; + -moz-text-size-adjust: none; + text-size-adjust: none; + -webkit-tap-highlight-color: transparent; + -webkit-tap-highlight-color: transparent; + -webkit-user-select: none; + -moz-user-select: none; + -ms-user-select: none; + user-select: none; + top: 0; + right: 0; + bottom: 0; + left: 0; + overflow: hidden; + margin: 0; + padding: 0; + color: #000; + word-wrap: break-word; + font-size: 14px; + font-family: -apple-system; + font-family: "-apple-system", "Helvetica Neue", "Roboto", "Segoe UI", sans-serif; + line-height: 20px; + text-rendering: optimizeLegibility; + -webkit-backface-visibility: hidden; + -webkit-user-drag: none; + -ms-content-zooming: none; } + + .content { + position: relative; } + + /* If you change these, change platform.scss as well */ + .has-header { + top: 44px; } + + .no-header { + top: 0; } + + .has-subheader { + top: 88px; } + + .has-tabs-top { + top: 93px; } + + .has-header.has-subheader.has-tabs-top { + top: 137px; } + + .has-footer { + bottom: 44px; } + + .has-subfooter { + bottom: 88px; } + + .has-tabs, + .bar-footer.has-tabs { + bottom: 49px; } + .has-tabs.pane, + .bar-footer.has-tabs.pane { + bottom: 49px; + height: auto; } + + .bar-subfooter.has-tabs { + bottom: 93px; } + + .has-footer.has-tabs { + bottom: 93px; } + + /** + * Typography + * -------------------------------------------------- + */ + p { + margin: 0 0 10px; } + + small { + font-size: 85%; } + + cite { + font-style: normal; } + + .text-left { + text-align: left; } + + .text-right { + text-align: right; } + + .text-center, .item.large-button-bar { + text-align: center; } + + h1, h2, h3, h4, h5, h6, + .h1, .h2, .h3, .h4, .h5, .h6 { + color: #000; + font-weight: 500; + font-family: "-apple-system", "Helvetica Neue", "Roboto", "Segoe UI", sans-serif; + line-height: 1.2; } + h1 small, h2 small, h3 small, h4 small, h5 small, h6 small, + .h1 small, .h2 small, .h3 small, .h4 small, .h5 small, .h6 small { + font-weight: normal; + line-height: 1; } + + h1, .h1, + h2, .h2, + h3, .h3 { + margin-top: 20px; + margin-bottom: 10px; } + h1:first-child, .h1:first-child, + h2:first-child, .h2:first-child, + h3:first-child, .h3:first-child { + margin-top: 0; } + h1 + h1, h1 + .h1, + h1 + h2, h1 + .h2, + h1 + h3, h1 + .h3, .h1 + h1, .h1 + .h1, + .h1 + h2, .h1 + .h2, + .h1 + h3, .h1 + .h3, + h2 + h1, + h2 + .h1, + h2 + h2, + h2 + .h2, + h2 + h3, + h2 + .h3, .h2 + h1, .h2 + .h1, + .h2 + h2, .h2 + .h2, + .h2 + h3, .h2 + .h3, + h3 + h1, + h3 + .h1, + h3 + h2, + h3 + .h2, + h3 + h3, + h3 + .h3, .h3 + h1, .h3 + .h1, + .h3 + h2, .h3 + .h2, + .h3 + h3, .h3 + .h3 { + margin-top: 10px; } + + h4, .h4, + h5, .h5, + h6, .h6 { + margin-top: 10px; + margin-bottom: 10px; } + + h1, .h1 { + font-size: 36px; } + + h2, .h2 { + font-size: 30px; } + + h3, .h3 { + font-size: 24px; } + + h4, .h4 { + font-size: 18px; } + + h5, .h5 { + font-size: 14px; } + + h6, .h6 { + font-size: 12px; } + + h1 small, .h1 small { + font-size: 24px; } + + h2 small, .h2 small { + font-size: 18px; } + + h3 small, .h3 small, + h4 small, .h4 small { + font-size: 14px; } + + dl { + margin-bottom: 20px; } + + dt, + dd { + line-height: 1.42857; } + + dt { + font-weight: bold; } + + blockquote { + margin: 0 0 20px; + padding: 10px 20px; + border-left: 5px solid gray; } + blockquote p { + font-weight: 300; + font-size: 17.5px; + line-height: 1.25; } + blockquote p:last-child { + margin-bottom: 0; } + blockquote small { + display: block; + line-height: 1.42857; } + blockquote small:before { + content: '\2014 \00A0'; } + + q:before, + q:after, + blockquote:before, + blockquote:after { + content: ""; } + + address { + display: block; + margin-bottom: 20px; + font-style: normal; + line-height: 1.42857; } + + a { + color: #387ef5; } + + a.subdued { + padding-right: 10px; + color: #888; + text-decoration: none; } + a.subdued:hover { + text-decoration: none; } + a.subdued:last-child { + padding-right: 0; } + + /** + * Bar (Headers and Footers) + * -------------------------------------------------- + */ + .bar { + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); + -webkit-user-select: none; + -moz-user-select: none; + -ms-user-select: none; + user-select: none; + position: absolute; + right: 0; + left: 0; + z-index: 9; + -webkit-box-sizing: border-box; + -moz-box-sizing: border-box; + box-sizing: border-box; + padding: 5px; + width: 100%; + height: 44px; + border-width: 0; + border-style: solid; + border-top: 1px solid transparent; + border-bottom: 1px solid #ddd; + background-color: white; + /* border-width: 1px will actually create 2 device pixels on retina */ + /* this nifty trick sets an actual 1px border on hi-res displays */ + background-size: 0; } + @media (min--moz-device-pixel-ratio: 1.5), (-webkit-min-device-pixel-ratio: 1.5), (min-device-pixel-ratio: 1.5), (min-resolution: 144dpi), (min-resolution: 1.5dppx) { + .bar { + border: none; + background-image: linear-gradient(0deg, #ddd, #ddd 50%, transparent 50%); + background-position: bottom; + background-size: 100% 1px; + background-repeat: no-repeat; } } + .bar.bar-clear { + border: none; + background: none; + color: #fff; } + .bar.bar-clear .button { + color: #fff; } + .bar.bar-clear .title { + color: #fff; } + .bar.item-input-inset .item-input-wrapper { + margin-top: -1px; } + .bar.item-input-inset .item-input-wrapper input { + padding-left: 8px; + width: 94%; + height: 28px; + background: transparent; } + .bar.bar-light { + border-color: #ddd; + background-color: white; + background-image: linear-gradient(0deg, #ddd, #ddd 50%, transparent 50%); + color: #444; } + .bar.bar-light .title { + color: #444; } + .bar.bar-light.bar-footer { + background-image: linear-gradient(180deg, #ddd, #ddd 50%, transparent 50%); } + .bar.bar-stable { + border-color: #b2b2b2; + background-color: #f8f8f8; + background-image: linear-gradient(0deg, #b2b2b2, #b2b2b2 50%, transparent 50%); + color: #444; } + .bar.bar-stable .title { + color: #444; } + .bar.bar-stable.bar-footer { + background-image: linear-gradient(180deg, #b2b2b2, #b2b2b2 50%, transparent 50%); } + .bar.bar-positive { + border-color: #0c60ee; + background-color: #387ef5; + background-image: linear-gradient(0deg, #0c60ee, #0c60ee 50%, transparent 50%); + color: #fff; } + .bar.bar-positive .title { + color: #fff; } + .bar.bar-positive.bar-footer { + background-image: linear-gradient(180deg, #0c60ee, #0c60ee 50%, transparent 50%); } + .bar.bar-calm { + border-color: #0a9dc7; + background-color: #11c1f3; + background-image: linear-gradient(0deg, #0a9dc7, #0a9dc7 50%, transparent 50%); + color: #fff; } + .bar.bar-calm .title { + color: #fff; } + .bar.bar-calm.bar-footer { + background-image: linear-gradient(180deg, #0a9dc7, #0a9dc7 50%, transparent 50%); } + .bar.bar-assertive { + border-color: #e42112; + background-color: #ef473a; + background-image: linear-gradient(0deg, #e42112, #e42112 50%, transparent 50%); + color: #fff; } + .bar.bar-assertive .title { + color: #fff; } + .bar.bar-assertive.bar-footer { + background-image: linear-gradient(180deg, #e42112, #e42112 50%, transparent 50%); } + .bar.bar-balanced { + border-color: #28a54c; + background-color: #33cd5f; + background-image: linear-gradient(0deg, #28a54c, #28a54c 50%, transparent 50%); + color: #fff; } + .bar.bar-balanced .title { + color: #fff; } + .bar.bar-balanced.bar-footer { + background-image: linear-gradient(180deg, #28a54c, #28a54c 50%, transparent 50%); } + .bar.bar-energized { + border-color: #e6b500; + background-color: #ffc900; + background-image: linear-gradient(0deg, #e6b500, #e6b500 50%, transparent 50%); + color: #fff; } + .bar.bar-energized .title { + color: #fff; } + .bar.bar-energized.bar-footer { + background-image: linear-gradient(180deg, #e6b500, #e6b500 50%, transparent 50%); } + .bar.bar-royal { + border-color: #6b46e5; + background-color: #886aea; + background-image: linear-gradient(0deg, #6b46e5, #6b46e5 50%, transparent 50%); + color: #fff; } + .bar.bar-royal .title { + color: #fff; } + .bar.bar-royal.bar-footer { + background-image: linear-gradient(180deg, #6b46e5, #6b46e5 50%, transparent 50%); } + .bar.bar-dark { + border-color: #111; + background-color: #444444; + background-image: linear-gradient(0deg, #111, #111 50%, transparent 50%); + color: #fff; } + .bar.bar-dark .title { + color: #fff; } + .bar.bar-dark.bar-footer { + background-image: linear-gradient(180deg, #111, #111 50%, transparent 50%); } + .bar .title { + display: block; + position: absolute; + top: 0; + right: 0; + left: 0; + z-index: 0; + overflow: hidden; + margin: 0 10px; + min-width: 30px; + height: 43px; + text-align: center; + text-overflow: ellipsis; + white-space: nowrap; + font-size: 17px; + font-weight: 500; + line-height: 44px; } + .bar .title.title-left { + text-align: left; } + .bar .title.title-right { + text-align: right; } + .bar .title a { + color: inherit; } + .bar .button, .bar button { + z-index: 1; + padding: 0 8px; + min-width: initial; + min-height: 31px; + font-weight: 400; + font-size: 13px; + line-height: 32px; } + .bar .button.button-icon:before, + .bar .button .icon:before, + .bar .button .icon-help:before, + .bar .button .icon-alert:before, + .bar .button #menu .footer .icon-help:before, #menu .footer + .bar .button .icon-help:before, .bar .button.icon:before, .bar .button.icon-help:before, .bar .button.icon-alert:before, .bar #menu .footer .button.icon-help:before, #menu .footer .bar .button.icon-help:before, .bar .button.icon-left:before, .bar .button.icon-right:before, .bar button.button-icon:before, + .bar button .icon:before, + .bar button .icon-help:before, + .bar button .icon-alert:before, + .bar button #menu .footer .icon-help:before, #menu .footer + .bar button .icon-help:before, .bar button.icon:before, .bar button.icon-help:before, .bar button.icon-alert:before, .bar #menu .footer button.icon-help:before, #menu .footer .bar button.icon-help:before, .bar button.icon-left:before, .bar button.icon-right:before { + padding-right: 2px; + padding-left: 2px; + font-size: 20px; + line-height: 32px; } + .bar .button.button-icon, .bar button.button-icon { + font-size: 17px; } + .bar .button.button-icon .icon:before, .bar .button.button-icon .icon-help:before, .bar .button.button-icon .icon-alert:before, .bar .button.button-icon #menu .footer .icon-help:before, #menu .footer .bar .button.button-icon .icon-help:before, .bar .button.button-icon:before, .bar .button.button-icon.icon-left:before, .bar .button.button-icon.icon-right:before, .bar button.button-icon .icon:before, .bar button.button-icon .icon-help:before, .bar button.button-icon .icon-alert:before, .bar button.button-icon #menu .footer .icon-help:before, #menu .footer .bar button.button-icon .icon-help:before, .bar button.button-icon:before, .bar button.button-icon.icon-left:before, .bar button.button-icon.icon-right:before { + vertical-align: top; + font-size: 32px; + line-height: 32px; } + .bar .button.button-clear, .bar .button.button-text, .bar button.button-clear, .bar button.button-text { + padding-right: 2px; + padding-left: 2px; + font-weight: 300; + font-size: 17px; } + .bar .button.button-clear .icon:before, .bar .button.button-text .icon:before, .bar .button.button-clear .icon-help:before, .bar .button.button-text .icon-help:before, .bar .button.button-clear .icon-alert:before, .bar .button.button-text .icon-alert:before, .bar .button.button-clear #menu .footer .icon-help:before, #menu .footer .bar .button.button-clear .icon-help:before, .bar .button.button-text #menu .footer .icon-help:before, #menu .footer .bar .button.button-text .icon-help:before, .bar .button.button-clear.icon:before, .bar .button.icon.button-text:before, .bar .button.button-text.icon-help:before, .bar .button.button-text.icon-alert:before, .bar #menu .footer .button.button-text.icon-help:before, #menu .footer .bar .button.button-text.icon-help:before, .bar .button.button-clear.icon-help:before, .bar .button.button-clear.icon-alert:before, .bar #menu .footer .button.button-clear.icon-help:before, #menu .footer .bar .button.button-clear.icon-help:before, .bar .button.button-clear.icon-left:before, .bar .button.icon-left.button-text:before, .bar .button.button-clear.icon-right:before, .bar .button.icon-right.button-text:before, .bar button.button-clear .icon:before, .bar button.button-text .icon:before, .bar button.button-clear .icon-help:before, .bar button.button-text .icon-help:before, .bar button.button-clear .icon-alert:before, .bar button.button-text .icon-alert:before, .bar button.button-clear #menu .footer .icon-help:before, #menu .footer .bar button.button-clear .icon-help:before, .bar button.button-text #menu .footer .icon-help:before, #menu .footer .bar button.button-text .icon-help:before, .bar button.button-clear.icon:before, .bar button.icon.button-text:before, .bar button.button-text.icon-help:before, .bar button.button-text.icon-alert:before, .bar #menu .footer button.button-text.icon-help:before, #menu .footer .bar button.button-text.icon-help:before, .bar button.button-clear.icon-help:before, .bar button.button-clear.icon-alert:before, .bar #menu .footer button.button-clear.icon-help:before, #menu .footer .bar button.button-clear.icon-help:before, .bar button.button-clear.icon-left:before, .bar button.icon-left.button-text:before, .bar button.button-clear.icon-right:before, .bar button.icon-right.button-text:before { + font-size: 32px; + line-height: 32px; } + .bar .button.back-button, .bar button.back-button { + display: block; + margin-right: 5px; + padding: 0; + white-space: nowrap; + font-weight: 400; } + .bar .button.back-button.active, .bar .button.back-button.activated, .bar button.back-button.active, .bar button.back-button.activated { + opacity: 0.2; } + .bar .button-bar > .button, + .bar .buttons > .button { + min-height: 31px; + line-height: 32px; } + .bar .button-bar + .button, + .bar .button + .button-bar { + margin-left: 5px; } + .bar .buttons, + .bar .buttons.primary-buttons, + .bar .buttons.secondary-buttons { + display: inherit; } + .bar .buttons span { + display: inline-block; } + .bar .buttons-left span { + margin-right: 5px; + display: inherit; } + .bar .buttons-right span { + margin-left: 5px; + display: inherit; } + .bar .title + .button:last-child, + .bar > .button + .button:last-child, + .bar > .button.pull-right, .popover-helptip + .bar > .button.icon.icon-right, .popover-helptip + .bar > .button.icon-right.icon-help, .popover-helptip + .bar > .button.icon-right.icon-alert, .popover-helptip #menu .footer + .bar > .button.icon-right.icon-help, #menu .footer .popover-helptip + .bar > .button.icon-right.icon-help, .popover-helptip + .bar > .button.icon.icon-center, .popover-helptip + .bar > .button.icon-center.icon-help, .popover-helptip + .bar > .button.icon-center.icon-alert, .popover-helptip #menu .footer + .bar > .button.icon-center.icon-help, #menu .footer .popover-helptip + .bar > .button.icon-center.icon-help, .popover-helptip + .bar > .button.icon.icon-bottom-right, .popover-helptip + .bar > .button.icon-bottom-right.icon-help, .popover-helptip + .bar > .button.icon-bottom-right.icon-alert, .popover-helptip #menu .footer + .bar > .button.icon-bottom-right.icon-help, #menu .footer .popover-helptip + .bar > .button.icon-bottom-right.icon-help, .popover-helptip + .bar > .button.icon.icon-bottom-center, .popover-helptip + .bar > .button.icon-bottom-center.icon-help, .popover-helptip + .bar > .button.icon-bottom-center.icon-alert, .popover-helptip #menu .footer + .bar > .button.icon-bottom-center.icon-help, #menu .footer .popover-helptip + .bar > .button.icon-bottom-center.icon-help, + .bar .buttons.pull-right, + .bar .popover-helptip .buttons.icon.icon-right, .popover-helptip + .bar .buttons.icon.icon-right, + .bar .popover-helptip .buttons.icon-right.icon-help, .popover-helptip + .bar .buttons.icon-right.icon-help, + .bar .popover-helptip .buttons.icon-right.icon-alert, .popover-helptip + .bar .buttons.icon-right.icon-alert, + .bar .popover-helptip #menu .footer .buttons.icon-right.icon-help, .popover-helptip #menu .footer + .bar .buttons.icon-right.icon-help, + .bar #menu .footer .popover-helptip .buttons.icon-right.icon-help, #menu .footer .popover-helptip + .bar .buttons.icon-right.icon-help, + .bar .popover-helptip .buttons.icon.icon-center, .popover-helptip + .bar .buttons.icon.icon-center, + .bar .popover-helptip .buttons.icon-center.icon-help, .popover-helptip + .bar .buttons.icon-center.icon-help, + .bar .popover-helptip .buttons.icon-center.icon-alert, .popover-helptip + .bar .buttons.icon-center.icon-alert, + .bar .popover-helptip #menu .footer .buttons.icon-center.icon-help, .popover-helptip #menu .footer + .bar .buttons.icon-center.icon-help, + .bar #menu .footer .popover-helptip .buttons.icon-center.icon-help, #menu .footer .popover-helptip + .bar .buttons.icon-center.icon-help, + .bar .popover-helptip .buttons.icon.icon-bottom-right, .popover-helptip + .bar .buttons.icon.icon-bottom-right, + .bar .popover-helptip .buttons.icon-bottom-right.icon-help, .popover-helptip + .bar .buttons.icon-bottom-right.icon-help, + .bar .popover-helptip .buttons.icon-bottom-right.icon-alert, .popover-helptip + .bar .buttons.icon-bottom-right.icon-alert, + .bar .popover-helptip #menu .footer .buttons.icon-bottom-right.icon-help, .popover-helptip #menu .footer + .bar .buttons.icon-bottom-right.icon-help, + .bar #menu .footer .popover-helptip .buttons.icon-bottom-right.icon-help, #menu .footer .popover-helptip + .bar .buttons.icon-bottom-right.icon-help, + .bar .popover-helptip .buttons.icon.icon-bottom-center, .popover-helptip + .bar .buttons.icon.icon-bottom-center, + .bar .popover-helptip .buttons.icon-bottom-center.icon-help, .popover-helptip + .bar .buttons.icon-bottom-center.icon-help, + .bar .popover-helptip .buttons.icon-bottom-center.icon-alert, .popover-helptip + .bar .buttons.icon-bottom-center.icon-alert, + .bar .popover-helptip #menu .footer .buttons.icon-bottom-center.icon-help, .popover-helptip #menu .footer + .bar .buttons.icon-bottom-center.icon-help, + .bar #menu .footer .popover-helptip .buttons.icon-bottom-center.icon-help, #menu .footer .popover-helptip + .bar .buttons.icon-bottom-center.icon-help, + .bar .title + .buttons { + position: absolute; + top: 5px; + right: 5px; + bottom: 5px; } + + .platform-android .nav-bar-has-subheader .bar { + background-image: none; } + + .platform-android .bar .back-button .icon:before, .platform-android .bar .back-button .icon-help:before, .platform-android .bar .back-button .icon-alert:before, .platform-android .bar .back-button #menu .footer .icon-help:before, #menu .footer .platform-android .bar .back-button .icon-help:before { + font-size: 24px; } + + .platform-android .bar .title { + font-size: 19px; + line-height: 44px; } + + .bar-light .button { + border-color: #ddd; + background-color: white; + color: #444; } + .bar-light .button:hover { + color: #444; + text-decoration: none; } + .bar-light .button.active, .bar-light .button.activated { + border-color: #ccc; + background-color: #fafafa; } + .bar-light .button.button-clear, .bar-light .button.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #444; + font-size: 17px; } + .bar-light .button.button-icon { + border-color: transparent; + background: none; } + + .bar-stable .button { + border-color: #b2b2b2; + background-color: #f8f8f8; + color: #444; } + .bar-stable .button:hover { + color: #444; + text-decoration: none; } + .bar-stable .button.active, .bar-stable .button.activated { + border-color: #a2a2a2; + background-color: #e5e5e5; } + .bar-stable .button.button-clear, .bar-stable .button.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #444; + font-size: 17px; } + .bar-stable .button.button-icon { + border-color: transparent; + background: none; } + + .bar-positive .button { + border-color: #0c60ee; + background-color: #387ef5; + color: #fff; } + .bar-positive .button:hover { + color: #fff; + text-decoration: none; } + .bar-positive .button.active, .bar-positive .button.activated { + border-color: #0c60ee; + background-color: #0c60ee; } + .bar-positive .button.button-clear, .bar-positive .button.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #fff; + font-size: 17px; } + .bar-positive .button.button-icon { + border-color: transparent; + background: none; } + + .bar-calm .button { + border-color: #0a9dc7; + background-color: #11c1f3; + color: #fff; } + .bar-calm .button:hover { + color: #fff; + text-decoration: none; } + .bar-calm .button.active, .bar-calm .button.activated { + border-color: #0a9dc7; + background-color: #0a9dc7; } + .bar-calm .button.button-clear, .bar-calm .button.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #fff; + font-size: 17px; } + .bar-calm .button.button-icon { + border-color: transparent; + background: none; } + + .bar-assertive .button { + border-color: #e42112; + background-color: #ef473a; + color: #fff; } + .bar-assertive .button:hover { + color: #fff; + text-decoration: none; } + .bar-assertive .button.active, .bar-assertive .button.activated { + border-color: #e42112; + background-color: #e42112; } + .bar-assertive .button.button-clear, .bar-assertive .button.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #fff; + font-size: 17px; } + .bar-assertive .button.button-icon { + border-color: transparent; + background: none; } + + .bar-balanced .button { + border-color: #28a54c; + background-color: #33cd5f; + color: #fff; } + .bar-balanced .button:hover { + color: #fff; + text-decoration: none; } + .bar-balanced .button.active, .bar-balanced .button.activated { + border-color: #28a54c; + background-color: #28a54c; } + .bar-balanced .button.button-clear, .bar-balanced .button.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #fff; + font-size: 17px; } + .bar-balanced .button.button-icon { + border-color: transparent; + background: none; } + + .bar-energized .button { + border-color: #e6b500; + background-color: #ffc900; + color: #fff; } + .bar-energized .button:hover { + color: #fff; + text-decoration: none; } + .bar-energized .button.active, .bar-energized .button.activated { + border-color: #e6b500; + background-color: #e6b500; } + .bar-energized .button.button-clear, .bar-energized .button.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #fff; + font-size: 17px; } + .bar-energized .button.button-icon { + border-color: transparent; + background: none; } + + .bar-royal .button { + border-color: #6b46e5; + background-color: #886aea; + color: #fff; } + .bar-royal .button:hover { + color: #fff; + text-decoration: none; } + .bar-royal .button.active, .bar-royal .button.activated { + border-color: #6b46e5; + background-color: #6b46e5; } + .bar-royal .button.button-clear, .bar-royal .button.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #fff; + font-size: 17px; } + .bar-royal .button.button-icon { + border-color: transparent; + background: none; } + + .bar-dark .button { + border-color: #111; + background-color: #444444; + color: #fff; } + .bar-dark .button:hover { + color: #fff; + text-decoration: none; } + .bar-dark .button.active, .bar-dark .button.activated { + border-color: #000; + background-color: #262626; } + .bar-dark .button.button-clear, .bar-dark .button.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #fff; + font-size: 17px; } + .bar-dark .button.button-icon { + border-color: transparent; + background: none; } + + .bar-header { + top: 0; + border-top-width: 0; + border-bottom-width: 1px; } + .bar-header.has-tabs-top { + border-bottom-width: 0px; + background-image: none; } + + .tabs-top .bar-header { + border-bottom-width: 0px; + background-image: none; } + + .bar-footer { + bottom: 0; + border-top-width: 1px; + border-bottom-width: 0; + background-position: top; + height: 44px; } + .bar-footer.item-input-inset { + position: absolute; } + .bar-footer .title { + height: 43px; + line-height: 44px; } + + .bar-tabs { + padding: 0; } + + .bar-subheader { + top: 44px; + height: 44px; } + .bar-subheader .title { + height: 43px; + line-height: 44px; } + + .bar-subfooter { + bottom: 44px; + height: 44px; } + .bar-subfooter .title { + height: 43px; + line-height: 44px; } + + .nav-bar-block { + position: absolute; + top: 0; + right: 0; + left: 0; + z-index: 9; } + + .bar .back-button.hide, + .bar .buttons .hide { + display: none; } + + .nav-bar-tabs-top .bar { + background-image: none; } + + /** + * Tabs + * -------------------------------------------------- + * A navigation bar with any number of tab items supported. + */ + .tabs { + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + -webkit-box-direction: normal; + -webkit-box-orient: horizontal; + -webkit-flex-direction: horizontal; + -moz-flex-direction: horizontal; + -ms-flex-direction: horizontal; + flex-direction: horizontal; + -webkit-box-pack: center; + -ms-flex-pack: center; + -webkit-justify-content: center; + -moz-justify-content: center; + justify-content: center; + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); + border-color: #b2b2b2; + background-color: #f8f8f8; + background-image: linear-gradient(0deg, #b2b2b2, #b2b2b2 50%, transparent 50%); + color: #444; + position: absolute; + bottom: 0; + z-index: 5; + width: 100%; + height: 49px; + border-style: solid; + border-top-width: 1px; + background-size: 0; + line-height: 49px; } + .tabs .tab-item .badge { + background-color: #444; + color: #f8f8f8; } + @media (min--moz-device-pixel-ratio: 1.5), (-webkit-min-device-pixel-ratio: 1.5), (min-device-pixel-ratio: 1.5), (min-resolution: 144dpi), (min-resolution: 1.5dppx) { + .tabs { + padding-top: 2px; + border-top: none !important; + border-bottom: none; + background-position: top; + background-size: 100% 1px; + background-repeat: no-repeat; } } + + /* Allow parent element of tabs to define color, or just the tab itself */ + .tabs-light > .tabs, + .tabs.tabs-light { + border-color: #ddd; + background-color: #fff; + background-image: linear-gradient(0deg, #ddd, #ddd 50%, transparent 50%); + color: #444; } + .tabs-light > .tabs .tab-item .badge, + .tabs.tabs-light .tab-item .badge { + background-color: #444; + color: #fff; } + + .tabs-stable > .tabs, + .tabs.tabs-stable { + border-color: #b2b2b2; + background-color: #f8f8f8; + background-image: linear-gradient(0deg, #b2b2b2, #b2b2b2 50%, transparent 50%); + color: #444; } + .tabs-stable > .tabs .tab-item .badge, + .tabs.tabs-stable .tab-item .badge { + background-color: #444; + color: #f8f8f8; } + + .tabs-positive > .tabs, + .tabs.tabs-positive { + border-color: #0c60ee; + background-color: #387ef5; + background-image: linear-gradient(0deg, #0c60ee, #0c60ee 50%, transparent 50%); + color: #fff; } + .tabs-positive > .tabs .tab-item .badge, + .tabs.tabs-positive .tab-item .badge { + background-color: #fff; + color: #387ef5; } + + .tabs-calm > .tabs, + .tabs.tabs-calm { + border-color: #0a9dc7; + background-color: #11c1f3; + background-image: linear-gradient(0deg, #0a9dc7, #0a9dc7 50%, transparent 50%); + color: #fff; } + .tabs-calm > .tabs .tab-item .badge, + .tabs.tabs-calm .tab-item .badge { + background-color: #fff; + color: #11c1f3; } + + .tabs-assertive > .tabs, + .tabs.tabs-assertive { + border-color: #e42112; + background-color: #ef473a; + background-image: linear-gradient(0deg, #e42112, #e42112 50%, transparent 50%); + color: #fff; } + .tabs-assertive > .tabs .tab-item .badge, + .tabs.tabs-assertive .tab-item .badge { + background-color: #fff; + color: #ef473a; } + + .tabs-balanced > .tabs, + .tabs.tabs-balanced { + border-color: #28a54c; + background-color: #33cd5f; + background-image: linear-gradient(0deg, #28a54c, #28a54c 50%, transparent 50%); + color: #fff; } + .tabs-balanced > .tabs .tab-item .badge, + .tabs.tabs-balanced .tab-item .badge { + background-color: #fff; + color: #33cd5f; } + + .tabs-energized > .tabs, + .tabs.tabs-energized { + border-color: #e6b500; + background-color: #ffc900; + background-image: linear-gradient(0deg, #e6b500, #e6b500 50%, transparent 50%); + color: #fff; } + .tabs-energized > .tabs .tab-item .badge, + .tabs.tabs-energized .tab-item .badge { + background-color: #fff; + color: #ffc900; } + + .tabs-royal > .tabs, + .tabs.tabs-royal { + border-color: #6b46e5; + background-color: #886aea; + background-image: linear-gradient(0deg, #6b46e5, #6b46e5 50%, transparent 50%); + color: #fff; } + .tabs-royal > .tabs .tab-item .badge, + .tabs.tabs-royal .tab-item .badge { + background-color: #fff; + color: #886aea; } + + .tabs-dark > .tabs, + .tabs.tabs-dark { + border-color: #111; + background-color: #444; + background-image: linear-gradient(0deg, #111, #111 50%, transparent 50%); + color: #fff; } + .tabs-dark > .tabs .tab-item .badge, + .tabs.tabs-dark .tab-item .badge { + background-color: #fff; + color: #444; } + + .tabs-striped .tabs { + background-color: white; + background-image: none; + border: none; + border-bottom: 1px solid #ddd; + padding-top: 2px; } + + .tabs-striped .tab-item.tab-item-active, .tabs-striped .tab-item.active, .tabs-striped .tab-item.activated { + margin-top: -2px; + border-style: solid; + border-width: 2px 0 0 0; + border-color: #444; } + .tabs-striped .tab-item.tab-item-active .badge, .tabs-striped .tab-item.active .badge, .tabs-striped .tab-item.activated .badge { + top: 2px; + opacity: 1; } + + .tabs-striped.tabs-light .tabs { + background-color: #fff; } + + .tabs-striped.tabs-light .tab-item { + color: rgba(68, 68, 68, 0.4); + opacity: 1; } + .tabs-striped.tabs-light .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-light .tab-item.tab-item-active, .tabs-striped.tabs-light .tab-item.active, .tabs-striped.tabs-light .tab-item.activated { + margin-top: -2px; + color: #444; + border-style: solid; + border-width: 2px 0 0 0; + border-color: #444; } + + .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { + top: 4%; } + + .tabs-striped.tabs-stable .tabs { + background-color: #f8f8f8; } + + .tabs-striped.tabs-stable .tab-item { + color: rgba(68, 68, 68, 0.4); + opacity: 1; } + .tabs-striped.tabs-stable .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-stable .tab-item.tab-item-active, .tabs-striped.tabs-stable .tab-item.active, .tabs-striped.tabs-stable .tab-item.activated { + margin-top: -2px; + color: #444; + border-style: solid; + border-width: 2px 0 0 0; + border-color: #444; } + + .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { + top: 4%; } + + .tabs-striped.tabs-positive .tabs { + background-color: #387ef5; } + + .tabs-striped.tabs-positive .tab-item { + color: rgba(255, 255, 255, 0.4); + opacity: 1; } + .tabs-striped.tabs-positive .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-positive .tab-item.tab-item-active, .tabs-striped.tabs-positive .tab-item.active, .tabs-striped.tabs-positive .tab-item.activated { + margin-top: -2px; + color: #fff; + border-style: solid; + border-width: 2px 0 0 0; + border-color: #fff; } + + .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { + top: 4%; } + + .tabs-striped.tabs-calm .tabs { + background-color: #11c1f3; } + + .tabs-striped.tabs-calm .tab-item { + color: rgba(255, 255, 255, 0.4); + opacity: 1; } + .tabs-striped.tabs-calm .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-calm .tab-item.tab-item-active, .tabs-striped.tabs-calm .tab-item.active, .tabs-striped.tabs-calm .tab-item.activated { + margin-top: -2px; + color: #fff; + border-style: solid; + border-width: 2px 0 0 0; + border-color: #fff; } + + .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { + top: 4%; } + + .tabs-striped.tabs-assertive .tabs { + background-color: #ef473a; } + + .tabs-striped.tabs-assertive .tab-item { + color: rgba(255, 255, 255, 0.4); + opacity: 1; } + .tabs-striped.tabs-assertive .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-assertive .tab-item.tab-item-active, .tabs-striped.tabs-assertive .tab-item.active, .tabs-striped.tabs-assertive .tab-item.activated { + margin-top: -2px; + color: #fff; + border-style: solid; + border-width: 2px 0 0 0; + border-color: #fff; } + + .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { + top: 4%; } + + .tabs-striped.tabs-balanced .tabs { + background-color: #33cd5f; } + + .tabs-striped.tabs-balanced .tab-item { + color: rgba(255, 255, 255, 0.4); + opacity: 1; } + .tabs-striped.tabs-balanced .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-balanced .tab-item.tab-item-active, .tabs-striped.tabs-balanced .tab-item.active, .tabs-striped.tabs-balanced .tab-item.activated { + margin-top: -2px; + color: #fff; + border-style: solid; + border-width: 2px 0 0 0; + border-color: #fff; } + + .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { + top: 4%; } + + .tabs-striped.tabs-energized .tabs { + background-color: #ffc900; } + + .tabs-striped.tabs-energized .tab-item { + color: rgba(255, 255, 255, 0.4); + opacity: 1; } + .tabs-striped.tabs-energized .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-energized .tab-item.tab-item-active, .tabs-striped.tabs-energized .tab-item.active, .tabs-striped.tabs-energized .tab-item.activated { + margin-top: -2px; + color: #fff; + border-style: solid; + border-width: 2px 0 0 0; + border-color: #fff; } + + .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { + top: 4%; } + + .tabs-striped.tabs-royal .tabs { + background-color: #886aea; } + + .tabs-striped.tabs-royal .tab-item { + color: rgba(255, 255, 255, 0.4); + opacity: 1; } + .tabs-striped.tabs-royal .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-royal .tab-item.tab-item-active, .tabs-striped.tabs-royal .tab-item.active, .tabs-striped.tabs-royal .tab-item.activated { + margin-top: -2px; + color: #fff; + border-style: solid; + border-width: 2px 0 0 0; + border-color: #fff; } + + .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { + top: 4%; } + + .tabs-striped.tabs-dark .tabs { + background-color: #444; } + + .tabs-striped.tabs-dark .tab-item { + color: rgba(255, 255, 255, 0.4); + opacity: 1; } + .tabs-striped.tabs-dark .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-dark .tab-item.tab-item-active, .tabs-striped.tabs-dark .tab-item.active, .tabs-striped.tabs-dark .tab-item.activated { + margin-top: -2px; + color: #fff; + border-style: solid; + border-width: 2px 0 0 0; + border-color: #fff; } + + .tabs-striped.tabs-top .tab-item.tab-item-active .badge, .tabs-striped.tabs-top .tab-item.active .badge, .tabs-striped.tabs-top .tab-item.activated .badge { + top: 4%; } + + .tabs-striped.tabs-background-light .tabs { + background-color: #fff; + background-image: none; } + + .tabs-striped.tabs-background-stable .tabs { + background-color: #f8f8f8; + background-image: none; } + + .tabs-striped.tabs-background-positive .tabs { + background-color: #387ef5; + background-image: none; } + + .tabs-striped.tabs-background-calm .tabs { + background-color: #11c1f3; + background-image: none; } + + .tabs-striped.tabs-background-assertive .tabs { + background-color: #ef473a; + background-image: none; } + + .tabs-striped.tabs-background-balanced .tabs { + background-color: #33cd5f; + background-image: none; } + + .tabs-striped.tabs-background-energized .tabs { + background-color: #ffc900; + background-image: none; } + + .tabs-striped.tabs-background-royal .tabs { + background-color: #886aea; + background-image: none; } + + .tabs-striped.tabs-background-dark .tabs { + background-color: #444; + background-image: none; } + + .tabs-striped.tabs-color-light .tab-item { + color: rgba(255, 255, 255, 0.4); + opacity: 1; } + .tabs-striped.tabs-color-light .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-color-light .tab-item.tab-item-active, .tabs-striped.tabs-color-light .tab-item.active, .tabs-striped.tabs-color-light .tab-item.activated { + margin-top: -2px; + color: #fff; + border: 0 solid #fff; + border-top-width: 2px; } + .tabs-striped.tabs-color-light .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-light .tab-item.active .badge, .tabs-striped.tabs-color-light .tab-item.activated .badge { + top: 2px; + opacity: 1; } + + .tabs-striped.tabs-color-stable .tab-item { + color: rgba(248, 248, 248, 0.4); + opacity: 1; } + .tabs-striped.tabs-color-stable .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-color-stable .tab-item.tab-item-active, .tabs-striped.tabs-color-stable .tab-item.active, .tabs-striped.tabs-color-stable .tab-item.activated { + margin-top: -2px; + color: #f8f8f8; + border: 0 solid #f8f8f8; + border-top-width: 2px; } + .tabs-striped.tabs-color-stable .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-stable .tab-item.active .badge, .tabs-striped.tabs-color-stable .tab-item.activated .badge { + top: 2px; + opacity: 1; } + + .tabs-striped.tabs-color-positive .tab-item { + color: rgba(56, 126, 245, 0.4); + opacity: 1; } + .tabs-striped.tabs-color-positive .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-color-positive .tab-item.tab-item-active, .tabs-striped.tabs-color-positive .tab-item.active, .tabs-striped.tabs-color-positive .tab-item.activated { + margin-top: -2px; + color: #387ef5; + border: 0 solid #387ef5; + border-top-width: 2px; } + .tabs-striped.tabs-color-positive .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-positive .tab-item.active .badge, .tabs-striped.tabs-color-positive .tab-item.activated .badge { + top: 2px; + opacity: 1; } + + .tabs-striped.tabs-color-calm .tab-item { + color: rgba(17, 193, 243, 0.4); + opacity: 1; } + .tabs-striped.tabs-color-calm .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-color-calm .tab-item.tab-item-active, .tabs-striped.tabs-color-calm .tab-item.active, .tabs-striped.tabs-color-calm .tab-item.activated { + margin-top: -2px; + color: #11c1f3; + border: 0 solid #11c1f3; + border-top-width: 2px; } + .tabs-striped.tabs-color-calm .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-calm .tab-item.active .badge, .tabs-striped.tabs-color-calm .tab-item.activated .badge { + top: 2px; + opacity: 1; } + + .tabs-striped.tabs-color-assertive .tab-item { + color: rgba(239, 71, 58, 0.4); + opacity: 1; } + .tabs-striped.tabs-color-assertive .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-color-assertive .tab-item.tab-item-active, .tabs-striped.tabs-color-assertive .tab-item.active, .tabs-striped.tabs-color-assertive .tab-item.activated { + margin-top: -2px; + color: #ef473a; + border: 0 solid #ef473a; + border-top-width: 2px; } + .tabs-striped.tabs-color-assertive .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-assertive .tab-item.active .badge, .tabs-striped.tabs-color-assertive .tab-item.activated .badge { + top: 2px; + opacity: 1; } + + .tabs-striped.tabs-color-balanced .tab-item { + color: rgba(51, 205, 95, 0.4); + opacity: 1; } + .tabs-striped.tabs-color-balanced .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-color-balanced .tab-item.tab-item-active, .tabs-striped.tabs-color-balanced .tab-item.active, .tabs-striped.tabs-color-balanced .tab-item.activated { + margin-top: -2px; + color: #33cd5f; + border: 0 solid #33cd5f; + border-top-width: 2px; } + .tabs-striped.tabs-color-balanced .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-balanced .tab-item.active .badge, .tabs-striped.tabs-color-balanced .tab-item.activated .badge { + top: 2px; + opacity: 1; } + + .tabs-striped.tabs-color-energized .tab-item { + color: rgba(255, 201, 0, 0.4); + opacity: 1; } + .tabs-striped.tabs-color-energized .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-color-energized .tab-item.tab-item-active, .tabs-striped.tabs-color-energized .tab-item.active, .tabs-striped.tabs-color-energized .tab-item.activated { + margin-top: -2px; + color: #ffc900; + border: 0 solid #ffc900; + border-top-width: 2px; } + .tabs-striped.tabs-color-energized .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-energized .tab-item.active .badge, .tabs-striped.tabs-color-energized .tab-item.activated .badge { + top: 2px; + opacity: 1; } + + .tabs-striped.tabs-color-royal .tab-item { + color: rgba(136, 106, 234, 0.4); + opacity: 1; } + .tabs-striped.tabs-color-royal .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-color-royal .tab-item.tab-item-active, .tabs-striped.tabs-color-royal .tab-item.active, .tabs-striped.tabs-color-royal .tab-item.activated { + margin-top: -2px; + color: #886aea; + border: 0 solid #886aea; + border-top-width: 2px; } + .tabs-striped.tabs-color-royal .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-royal .tab-item.active .badge, .tabs-striped.tabs-color-royal .tab-item.activated .badge { + top: 2px; + opacity: 1; } + + .tabs-striped.tabs-color-dark .tab-item { + color: rgba(68, 68, 68, 0.4); + opacity: 1; } + .tabs-striped.tabs-color-dark .tab-item .badge { + opacity: 0.4; } + .tabs-striped.tabs-color-dark .tab-item.tab-item-active, .tabs-striped.tabs-color-dark .tab-item.active, .tabs-striped.tabs-color-dark .tab-item.activated { + margin-top: -2px; + color: #444; + border: 0 solid #444; + border-top-width: 2px; } + .tabs-striped.tabs-color-dark .tab-item.tab-item-active .badge, .tabs-striped.tabs-color-dark .tab-item.active .badge, .tabs-striped.tabs-color-dark .tab-item.activated .badge { + top: 2px; + opacity: 1; } + + .tabs-background-light .tabs, + .tabs-background-light > .tabs { + background-color: #fff; + background-image: linear-gradient(0deg, #ddd, #ddd 50%, transparent 50%); + border-color: #ddd; } + + .tabs-background-stable .tabs, + .tabs-background-stable > .tabs { + background-color: #f8f8f8; + background-image: linear-gradient(0deg, #b2b2b2, #b2b2b2 50%, transparent 50%); + border-color: #b2b2b2; } + + .tabs-background-positive .tabs, + .tabs-background-positive > .tabs { + background-color: #387ef5; + background-image: linear-gradient(0deg, #0c60ee, #0c60ee 50%, transparent 50%); + border-color: #0c60ee; } + + .tabs-background-calm .tabs, + .tabs-background-calm > .tabs { + background-color: #11c1f3; + background-image: linear-gradient(0deg, #0a9dc7, #0a9dc7 50%, transparent 50%); + border-color: #0a9dc7; } + + .tabs-background-assertive .tabs, + .tabs-background-assertive > .tabs { + background-color: #ef473a; + background-image: linear-gradient(0deg, #e42112, #e42112 50%, transparent 50%); + border-color: #e42112; } + + .tabs-background-balanced .tabs, + .tabs-background-balanced > .tabs { + background-color: #33cd5f; + background-image: linear-gradient(0deg, #28a54c, #28a54c 50%, transparent 50%); + border-color: #28a54c; } + + .tabs-background-energized .tabs, + .tabs-background-energized > .tabs { + background-color: #ffc900; + background-image: linear-gradient(0deg, #e6b500, #e6b500 50%, transparent 50%); + border-color: #e6b500; } + + .tabs-background-royal .tabs, + .tabs-background-royal > .tabs { + background-color: #886aea; + background-image: linear-gradient(0deg, #6b46e5, #6b46e5 50%, transparent 50%); + border-color: #6b46e5; } + + .tabs-background-dark .tabs, + .tabs-background-dark > .tabs { + background-color: #444; + background-image: linear-gradient(0deg, #111, #111 50%, transparent 50%); + border-color: #111; } + + .tabs-color-light .tab-item { + color: rgba(255, 255, 255, 0.4); + opacity: 1; } + .tabs-color-light .tab-item .badge { + opacity: 0.4; } + .tabs-color-light .tab-item.tab-item-active, .tabs-color-light .tab-item.active, .tabs-color-light .tab-item.activated { + color: #fff; + border: 0 solid #fff; } + .tabs-color-light .tab-item.tab-item-active .badge, .tabs-color-light .tab-item.active .badge, .tabs-color-light .tab-item.activated .badge { + opacity: 1; } + + .tabs-color-stable .tab-item { + color: rgba(248, 248, 248, 0.4); + opacity: 1; } + .tabs-color-stable .tab-item .badge { + opacity: 0.4; } + .tabs-color-stable .tab-item.tab-item-active, .tabs-color-stable .tab-item.active, .tabs-color-stable .tab-item.activated { + color: #f8f8f8; + border: 0 solid #f8f8f8; } + .tabs-color-stable .tab-item.tab-item-active .badge, .tabs-color-stable .tab-item.active .badge, .tabs-color-stable .tab-item.activated .badge { + opacity: 1; } + + .tabs-color-positive .tab-item { + color: rgba(56, 126, 245, 0.4); + opacity: 1; } + .tabs-color-positive .tab-item .badge { + opacity: 0.4; } + .tabs-color-positive .tab-item.tab-item-active, .tabs-color-positive .tab-item.active, .tabs-color-positive .tab-item.activated { + color: #387ef5; + border: 0 solid #387ef5; } + .tabs-color-positive .tab-item.tab-item-active .badge, .tabs-color-positive .tab-item.active .badge, .tabs-color-positive .tab-item.activated .badge { + opacity: 1; } + + .tabs-color-calm .tab-item { + color: rgba(17, 193, 243, 0.4); + opacity: 1; } + .tabs-color-calm .tab-item .badge { + opacity: 0.4; } + .tabs-color-calm .tab-item.tab-item-active, .tabs-color-calm .tab-item.active, .tabs-color-calm .tab-item.activated { + color: #11c1f3; + border: 0 solid #11c1f3; } + .tabs-color-calm .tab-item.tab-item-active .badge, .tabs-color-calm .tab-item.active .badge, .tabs-color-calm .tab-item.activated .badge { + opacity: 1; } + + .tabs-color-assertive .tab-item { + color: rgba(239, 71, 58, 0.4); + opacity: 1; } + .tabs-color-assertive .tab-item .badge { + opacity: 0.4; } + .tabs-color-assertive .tab-item.tab-item-active, .tabs-color-assertive .tab-item.active, .tabs-color-assertive .tab-item.activated { + color: #ef473a; + border: 0 solid #ef473a; } + .tabs-color-assertive .tab-item.tab-item-active .badge, .tabs-color-assertive .tab-item.active .badge, .tabs-color-assertive .tab-item.activated .badge { + opacity: 1; } + + .tabs-color-balanced .tab-item { + color: rgba(51, 205, 95, 0.4); + opacity: 1; } + .tabs-color-balanced .tab-item .badge { + opacity: 0.4; } + .tabs-color-balanced .tab-item.tab-item-active, .tabs-color-balanced .tab-item.active, .tabs-color-balanced .tab-item.activated { + color: #33cd5f; + border: 0 solid #33cd5f; } + .tabs-color-balanced .tab-item.tab-item-active .badge, .tabs-color-balanced .tab-item.active .badge, .tabs-color-balanced .tab-item.activated .badge { + opacity: 1; } + + .tabs-color-energized .tab-item { + color: rgba(255, 201, 0, 0.4); + opacity: 1; } + .tabs-color-energized .tab-item .badge { + opacity: 0.4; } + .tabs-color-energized .tab-item.tab-item-active, .tabs-color-energized .tab-item.active, .tabs-color-energized .tab-item.activated { + color: #ffc900; + border: 0 solid #ffc900; } + .tabs-color-energized .tab-item.tab-item-active .badge, .tabs-color-energized .tab-item.active .badge, .tabs-color-energized .tab-item.activated .badge { + opacity: 1; } + + .tabs-color-royal .tab-item { + color: rgba(136, 106, 234, 0.4); + opacity: 1; } + .tabs-color-royal .tab-item .badge { + opacity: 0.4; } + .tabs-color-royal .tab-item.tab-item-active, .tabs-color-royal .tab-item.active, .tabs-color-royal .tab-item.activated { + color: #886aea; + border: 0 solid #886aea; } + .tabs-color-royal .tab-item.tab-item-active .badge, .tabs-color-royal .tab-item.active .badge, .tabs-color-royal .tab-item.activated .badge { + opacity: 1; } + + .tabs-color-dark .tab-item { + color: rgba(68, 68, 68, 0.4); + opacity: 1; } + .tabs-color-dark .tab-item .badge { + opacity: 0.4; } + .tabs-color-dark .tab-item.tab-item-active, .tabs-color-dark .tab-item.active, .tabs-color-dark .tab-item.activated { + color: #444; + border: 0 solid #444; } + .tabs-color-dark .tab-item.tab-item-active .badge, .tabs-color-dark .tab-item.active .badge, .tabs-color-dark .tab-item.activated .badge { + opacity: 1; } + + ion-tabs.tabs-color-active-light .tab-item { + color: #444; } + ion-tabs.tabs-color-active-light .tab-item.tab-item-active, ion-tabs.tabs-color-active-light .tab-item.active, ion-tabs.tabs-color-active-light .tab-item.activated { + color: #fff; } + + ion-tabs.tabs-striped.tabs-color-active-light .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-light .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-light .tab-item.activated { + border-color: #fff; + color: #fff; } + + ion-tabs.tabs-color-active-stable .tab-item { + color: #444; } + ion-tabs.tabs-color-active-stable .tab-item.tab-item-active, ion-tabs.tabs-color-active-stable .tab-item.active, ion-tabs.tabs-color-active-stable .tab-item.activated { + color: #f8f8f8; } + + ion-tabs.tabs-striped.tabs-color-active-stable .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-stable .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-stable .tab-item.activated { + border-color: #f8f8f8; + color: #f8f8f8; } + + ion-tabs.tabs-color-active-positive .tab-item { + color: #444; } + ion-tabs.tabs-color-active-positive .tab-item.tab-item-active, ion-tabs.tabs-color-active-positive .tab-item.active, ion-tabs.tabs-color-active-positive .tab-item.activated { + color: #387ef5; } + + ion-tabs.tabs-striped.tabs-color-active-positive .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-positive .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-positive .tab-item.activated { + border-color: #387ef5; + color: #387ef5; } + + ion-tabs.tabs-color-active-calm .tab-item { + color: #444; } + ion-tabs.tabs-color-active-calm .tab-item.tab-item-active, ion-tabs.tabs-color-active-calm .tab-item.active, ion-tabs.tabs-color-active-calm .tab-item.activated { + color: #11c1f3; } + + ion-tabs.tabs-striped.tabs-color-active-calm .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-calm .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-calm .tab-item.activated { + border-color: #11c1f3; + color: #11c1f3; } + + ion-tabs.tabs-color-active-assertive .tab-item { + color: #444; } + ion-tabs.tabs-color-active-assertive .tab-item.tab-item-active, ion-tabs.tabs-color-active-assertive .tab-item.active, ion-tabs.tabs-color-active-assertive .tab-item.activated { + color: #ef473a; } + + ion-tabs.tabs-striped.tabs-color-active-assertive .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-assertive .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-assertive .tab-item.activated { + border-color: #ef473a; + color: #ef473a; } + + ion-tabs.tabs-color-active-balanced .tab-item { + color: #444; } + ion-tabs.tabs-color-active-balanced .tab-item.tab-item-active, ion-tabs.tabs-color-active-balanced .tab-item.active, ion-tabs.tabs-color-active-balanced .tab-item.activated { + color: #33cd5f; } + + ion-tabs.tabs-striped.tabs-color-active-balanced .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-balanced .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-balanced .tab-item.activated { + border-color: #33cd5f; + color: #33cd5f; } + + ion-tabs.tabs-color-active-energized .tab-item { + color: #444; } + ion-tabs.tabs-color-active-energized .tab-item.tab-item-active, ion-tabs.tabs-color-active-energized .tab-item.active, ion-tabs.tabs-color-active-energized .tab-item.activated { + color: #ffc900; } + + ion-tabs.tabs-striped.tabs-color-active-energized .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-energized .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-energized .tab-item.activated { + border-color: #ffc900; + color: #ffc900; } + + ion-tabs.tabs-color-active-royal .tab-item { + color: #444; } + ion-tabs.tabs-color-active-royal .tab-item.tab-item-active, ion-tabs.tabs-color-active-royal .tab-item.active, ion-tabs.tabs-color-active-royal .tab-item.activated { + color: #886aea; } + + ion-tabs.tabs-striped.tabs-color-active-royal .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-royal .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-royal .tab-item.activated { + border-color: #886aea; + color: #886aea; } + + ion-tabs.tabs-color-active-dark .tab-item { + color: #fff; } + ion-tabs.tabs-color-active-dark .tab-item.tab-item-active, ion-tabs.tabs-color-active-dark .tab-item.active, ion-tabs.tabs-color-active-dark .tab-item.activated { + color: #444; } + + ion-tabs.tabs-striped.tabs-color-active-dark .tab-item.tab-item-active, ion-tabs.tabs-striped.tabs-color-active-dark .tab-item.active, ion-tabs.tabs-striped.tabs-color-active-dark .tab-item.activated { + border-color: #444; + color: #444; } + + .tabs-top.tabs-striped { + padding-bottom: 0; } + .tabs-top.tabs-striped .tab-item { + background: transparent; + -webkit-transition: color .1s ease; + -moz-transition: color .1s ease; + -ms-transition: color .1s ease; + -o-transition: color .1s ease; + transition: color .1s ease; } + .tabs-top.tabs-striped .tab-item.tab-item-active, .tabs-top.tabs-striped .tab-item.active, .tabs-top.tabs-striped .tab-item.activated { + margin-top: 1px; + border-width: 0px 0px 2px 0px !important; + border-style: solid; } + .tabs-top.tabs-striped .tab-item.tab-item-active > .badge, .tabs-top.tabs-striped .tab-item.tab-item-active > i, .tabs-top.tabs-striped .tab-item.active > .badge, .tabs-top.tabs-striped .tab-item.active > i, .tabs-top.tabs-striped .tab-item.activated > .badge, .tabs-top.tabs-striped .tab-item.activated > i { + margin-top: -1px; } + .tabs-top.tabs-striped .tab-item .badge { + -webkit-transition: color .2s ease; + -moz-transition: color .2s ease; + -ms-transition: color .2s ease; + -o-transition: color .2s ease; + transition: color .2s ease; } + .tabs-top.tabs-striped:not(.tabs-icon-left):not(.tabs-icon-top) .tab-item.tab-item-active .tab-title, .tabs-top.tabs-striped:not(.tabs-icon-left):not(.tabs-icon-top) .tab-item.tab-item-active i, .tabs-top.tabs-striped:not(.tabs-icon-left):not(.tabs-icon-top) .tab-item.active .tab-title, .tabs-top.tabs-striped:not(.tabs-icon-left):not(.tabs-icon-top) .tab-item.active i, .tabs-top.tabs-striped:not(.tabs-icon-left):not(.tabs-icon-top) .tab-item.activated .tab-title, .tabs-top.tabs-striped:not(.tabs-icon-left):not(.tabs-icon-top) .tab-item.activated i { + display: block; + margin-top: -1px; } + .tabs-top.tabs-striped.tabs-icon-left .tab-item { + margin-top: 1px; } + .tabs-top.tabs-striped.tabs-icon-left .tab-item.tab-item-active .tab-title, .tabs-top.tabs-striped.tabs-icon-left .tab-item.tab-item-active i, .tabs-top.tabs-striped.tabs-icon-left .tab-item.active .tab-title, .tabs-top.tabs-striped.tabs-icon-left .tab-item.active i, .tabs-top.tabs-striped.tabs-icon-left .tab-item.activated .tab-title, .tabs-top.tabs-striped.tabs-icon-left .tab-item.activated i { + margin-top: -0.1em; } + + /* Allow parent element to have tabs-top */ + /* If you change this, change platform.scss as well */ + .tabs-top > .tabs, + .tabs.tabs-top { + top: 44px; + padding-top: 0; + background-position: bottom; + border-top-width: 0; + border-bottom-width: 1px; } + .tabs-top > .tabs .tab-item.tab-item-active .badge, .tabs-top > .tabs .tab-item.active .badge, .tabs-top > .tabs .tab-item.activated .badge, + .tabs.tabs-top .tab-item.tab-item-active .badge, + .tabs.tabs-top .tab-item.active .badge, + .tabs.tabs-top .tab-item.activated .badge { + top: 4%; } + + .tabs-top ~ .bar-header { + border-bottom-width: 0; } + + .tab-item { + -webkit-box-flex: 1; + -webkit-flex: 1; + -moz-box-flex: 1; + -moz-flex: 1; + -ms-flex: 1; + flex: 1; + display: block; + overflow: hidden; + max-width: 150px; + height: 100%; + color: inherit; + text-align: center; + text-decoration: none; + text-overflow: ellipsis; + white-space: nowrap; + font-weight: 400; + font-size: 14px; + font-family: "-apple-system", "Helvetica Neue", "Roboto", "Segoe UI", sans-serif; + opacity: 0.7; } + .tab-item:hover { + cursor: pointer; } + .tab-item.tab-hidden { + display: none; } + + .tabs-item-hide > .tabs, + .tabs.tabs-item-hide { + display: none; } + + .tabs-icon-top > .tabs .tab-item, + .tabs-icon-top.tabs .tab-item, + .tabs-icon-bottom > .tabs .tab-item, + .tabs-icon-bottom.tabs .tab-item { + font-size: 10px; + line-height: 14px; } + + .tab-item .icon, .tab-item .icon-help, .tab-item .icon-alert, .tab-item #menu .footer .icon-help, #menu .footer .tab-item .icon-help { + display: block; + margin: 0 auto; + height: 32px; + font-size: 32px; } + + .tabs-icon-left.tabs .tab-item, + .tabs-icon-left > .tabs .tab-item, + .tabs-icon-right.tabs .tab-item, + .tabs-icon-right > .tabs .tab-item { + font-size: 10px; } + .tabs-icon-left.tabs .tab-item .icon, .tabs-icon-left.tabs .tab-item .icon-help, .tabs-icon-left.tabs .tab-item .icon-alert, .tabs-icon-left.tabs .tab-item #menu .footer .icon-help, #menu .footer .tabs-icon-left.tabs .tab-item .icon-help, .tabs-icon-left.tabs .tab-item .tab-title, + .tabs-icon-left > .tabs .tab-item .icon, + .tabs-icon-left > .tabs .tab-item .icon-help, + .tabs-icon-left > .tabs .tab-item .icon-alert, + .tabs-icon-left > .tabs .tab-item #menu .footer .icon-help, #menu .footer + .tabs-icon-left > .tabs .tab-item .icon-help, + .tabs-icon-left > .tabs .tab-item .tab-title, + .tabs-icon-right.tabs .tab-item .icon, + .tabs-icon-right.tabs .tab-item .icon-help, + .tabs-icon-right.tabs .tab-item .icon-alert, + .tabs-icon-right.tabs .tab-item #menu .footer .icon-help, #menu .footer + .tabs-icon-right.tabs .tab-item .icon-help, + .tabs-icon-right.tabs .tab-item .tab-title, + .tabs-icon-right > .tabs .tab-item .icon, + .tabs-icon-right > .tabs .tab-item .icon-help, + .tabs-icon-right > .tabs .tab-item .icon-alert, + .tabs-icon-right > .tabs .tab-item #menu .footer .icon-help, #menu .footer + .tabs-icon-right > .tabs .tab-item .icon-help, + .tabs-icon-right > .tabs .tab-item .tab-title { + display: inline-block; + vertical-align: top; + margin-top: -.1em; } + .tabs-icon-left.tabs .tab-item .icon:before, .tabs-icon-left.tabs .tab-item .icon-help:before, .tabs-icon-left.tabs .tab-item .icon-alert:before, .tabs-icon-left.tabs .tab-item #menu .footer .icon-help:before, #menu .footer .tabs-icon-left.tabs .tab-item .icon-help:before, .tabs-icon-left.tabs .tab-item .tab-title:before, + .tabs-icon-left > .tabs .tab-item .icon:before, + .tabs-icon-left > .tabs .tab-item .icon-help:before, + .tabs-icon-left > .tabs .tab-item .icon-alert:before, + .tabs-icon-left > .tabs .tab-item #menu .footer .icon-help:before, #menu .footer + .tabs-icon-left > .tabs .tab-item .icon-help:before, + .tabs-icon-left > .tabs .tab-item .tab-title:before, + .tabs-icon-right.tabs .tab-item .icon:before, + .tabs-icon-right.tabs .tab-item .icon-help:before, + .tabs-icon-right.tabs .tab-item .icon-alert:before, + .tabs-icon-right.tabs .tab-item #menu .footer .icon-help:before, #menu .footer + .tabs-icon-right.tabs .tab-item .icon-help:before, + .tabs-icon-right.tabs .tab-item .tab-title:before, + .tabs-icon-right > .tabs .tab-item .icon:before, + .tabs-icon-right > .tabs .tab-item .icon-help:before, + .tabs-icon-right > .tabs .tab-item .icon-alert:before, + .tabs-icon-right > .tabs .tab-item #menu .footer .icon-help:before, #menu .footer + .tabs-icon-right > .tabs .tab-item .icon-help:before, + .tabs-icon-right > .tabs .tab-item .tab-title:before { + font-size: 24px; + line-height: 49px; } + + .tabs-icon-left > .tabs .tab-item .icon, .tabs-icon-left > .tabs .tab-item .icon-help, .tabs-icon-left > .tabs .tab-item .icon-alert, .tabs-icon-left > .tabs .tab-item #menu .footer .icon-help, #menu .footer .tabs-icon-left > .tabs .tab-item .icon-help, + .tabs-icon-left.tabs .tab-item .icon, + .tabs-icon-left.tabs .tab-item .icon-help, + .tabs-icon-left.tabs .tab-item .icon-alert, + .tabs-icon-left.tabs .tab-item #menu .footer .icon-help, #menu .footer + .tabs-icon-left.tabs .tab-item .icon-help { + padding-right: 3px; } + + .tabs-icon-right > .tabs .tab-item .icon, .tabs-icon-right > .tabs .tab-item .icon-help, .tabs-icon-right > .tabs .tab-item .icon-alert, .tabs-icon-right > .tabs .tab-item #menu .footer .icon-help, #menu .footer .tabs-icon-right > .tabs .tab-item .icon-help, + .tabs-icon-right.tabs .tab-item .icon, + .tabs-icon-right.tabs .tab-item .icon-help, + .tabs-icon-right.tabs .tab-item .icon-alert, + .tabs-icon-right.tabs .tab-item #menu .footer .icon-help, #menu .footer + .tabs-icon-right.tabs .tab-item .icon-help { + padding-left: 3px; } + + .tabs-icon-only > .tabs .icon, .tabs-icon-only > .tabs .icon-help, .tabs-icon-only > .tabs .icon-alert, .tabs-icon-only > .tabs #menu .footer .icon-help, #menu .footer .tabs-icon-only > .tabs .icon-help, + .tabs-icon-only.tabs .icon, + .tabs-icon-only.tabs .icon-help, + .tabs-icon-only.tabs .icon-alert, + .tabs-icon-only.tabs #menu .footer .icon-help, #menu .footer + .tabs-icon-only.tabs .icon-help { + line-height: inherit; } + + .tab-item.has-badge { + position: relative; } + + .tab-item .badge { + position: absolute; + top: 4%; + right: 33%; + right: calc(50% - 26px); + padding: 1px 6px; + height: auto; + font-size: 12px; + line-height: 16px; } + + /* Navigational tab */ + /* Active state for tab */ + .tab-item.tab-item-active, + .tab-item.active, + .tab-item.activated { + opacity: 1; } + .tab-item.tab-item-active.tab-item-light, + .tab-item.active.tab-item-light, + .tab-item.activated.tab-item-light { + color: #fff; } + .tab-item.tab-item-active.tab-item-stable, + .tab-item.active.tab-item-stable, + .tab-item.activated.tab-item-stable { + color: #f8f8f8; } + .tab-item.tab-item-active.tab-item-positive, + .tab-item.active.tab-item-positive, + .tab-item.activated.tab-item-positive { + color: #387ef5; } + .tab-item.tab-item-active.tab-item-calm, + .tab-item.active.tab-item-calm, + .tab-item.activated.tab-item-calm { + color: #11c1f3; } + .tab-item.tab-item-active.tab-item-assertive, + .tab-item.active.tab-item-assertive, + .tab-item.activated.tab-item-assertive { + color: #ef473a; } + .tab-item.tab-item-active.tab-item-balanced, + .tab-item.active.tab-item-balanced, + .tab-item.activated.tab-item-balanced { + color: #33cd5f; } + .tab-item.tab-item-active.tab-item-energized, + .tab-item.active.tab-item-energized, + .tab-item.activated.tab-item-energized { + color: #ffc900; } + .tab-item.tab-item-active.tab-item-royal, + .tab-item.active.tab-item-royal, + .tab-item.activated.tab-item-royal { + color: #886aea; } + .tab-item.tab-item-active.tab-item-dark, + .tab-item.active.tab-item-dark, + .tab-item.activated.tab-item-dark { + color: #444; } + + .item.tabs { + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + padding: 0; } + .item.tabs .icon:before, .item.tabs .icon-help:before, .item.tabs .icon-alert:before, .item.tabs #menu .footer .icon-help:before, #menu .footer .item.tabs .icon-help:before { + position: relative; } + + .tab-item.disabled, + .tab-item[disabled] { + opacity: .4; + cursor: default; + pointer-events: none; } + + .nav-bar-tabs-top.hide ~ .view-container .tabs-top .tabs { + top: 0; } + + .pane[hide-nav-bar="true"] .has-tabs-top { + top: 49px; } + + /** + * Menus + * -------------------------------------------------- + * Side panel structure + */ + .menu { + position: absolute; + top: 0; + bottom: 0; + z-index: 0; + overflow: hidden; + min-height: 100%; + max-height: 100%; + width: 275px; + background-color: #fff; } + .menu .scroll-content { + z-index: 10; } + .menu .bar-header { + z-index: 11; } + + .menu-content { + -webkit-transform: none; + transform: none; + box-shadow: -1px 0px 2px rgba(0, 0, 0, 0.2), 1px 0px 2px rgba(0, 0, 0, 0.2); } + + .menu-open .menu-content .pane, + .menu-open .menu-content .scroll-content { + pointer-events: none; } + + .menu-open .menu-content .scroll-content .scroll { + pointer-events: none; } + + .menu-open .menu-content .scroll-content:not(.overflow-scroll) { + overflow: hidden; } + + .grade-b .menu-content, + .grade-c .menu-content { + -webkit-box-sizing: content-box; + -moz-box-sizing: content-box; + box-sizing: content-box; + right: -1px; + left: -1px; + border-right: 1px solid #ccc; + border-left: 1px solid #ccc; + box-shadow: none; } + + .menu-left { + left: 0; } + + .menu-right { + right: 0; } + + .aside-open.aside-resizing .menu-right { + display: none; } + + .menu-animated { + -webkit-transition: -webkit-transform 200ms ease; + transition: transform 200ms ease; } + + /** + * Modals + * -------------------------------------------------- + * Modals are independent windows that slide in from off-screen. + */ + .modal-backdrop, + .modal-backdrop-bg { + position: fixed; + top: 0; + left: 0; + z-index: 10; + width: 100%; + height: 100%; } + + .modal-backdrop-bg { + pointer-events: none; } + + .modal { + display: block; + position: absolute; + top: 0; + z-index: 10; + overflow: hidden; + min-height: 100%; + width: 100%; + background-color: #fff; } + + @media (min-width: 680px) { + .modal { + top: 20%; + right: 20%; + bottom: 20%; + left: 20%; + min-height: 240px; + width: 60%; } + .modal.ng-leave-active { + bottom: 0; } + .platform-ios.platform-cordova .modal-wrapper .modal .bar-header:not(.bar-subheader) { + height: 44px; } + .platform-ios.platform-cordova .modal-wrapper .modal .bar-header:not(.bar-subheader) > * { + margin-top: 0; } + .platform-ios.platform-cordova .modal-wrapper .modal .tabs-top > .tabs, + .platform-ios.platform-cordova .modal-wrapper .modal .tabs.tabs-top { + top: 44px; } + .platform-ios.platform-cordova .modal-wrapper .modal .has-header, + .platform-ios.platform-cordova .modal-wrapper .modal .bar-subheader { + top: 44px; } + .platform-ios.platform-cordova .modal-wrapper .modal .has-subheader { + top: 88px; } + .platform-ios.platform-cordova .modal-wrapper .modal .has-header.has-tabs-top { + top: 93px; } + .platform-ios.platform-cordova .modal-wrapper .modal .has-header.has-subheader.has-tabs-top { + top: 137px; } + .modal-backdrop-bg { + -webkit-transition: opacity 300ms ease-in-out; + transition: opacity 300ms ease-in-out; + background-color: #000; + opacity: 0; } + .active .modal-backdrop-bg { + opacity: 0.5; } } + + .modal-open { + pointer-events: none; } + .modal-open .modal, + .modal-open .modal-backdrop { + pointer-events: auto; } + .modal-open.loading-active .modal, + .modal-open.loading-active .modal-backdrop { + pointer-events: none; } + + /** + * Popovers + * -------------------------------------------------- + * Popovers are independent views which float over content + */ + .popover-backdrop { + position: fixed; + top: 0; + left: 0; + z-index: 10; + width: 100%; + height: 100%; + background-color: transparent; } + .popover-backdrop.active { + background-color: rgba(0, 0, 0, 0.1); } + + .popover { + position: absolute; + top: 25%; + left: 50%; + z-index: 10; + display: block; + margin-top: 12px; + margin-left: -110px; + height: 280px; + width: 220px; + background-color: #fff; + box-shadow: 0 1px 3px rgba(0, 0, 0, 0.4); + opacity: 0; } + .popover .item:first-child { + border-top: 0; } + .popover .item:last-child { + border-bottom: 0; } + .popover.popover-bottom { + margin-top: -12px; } + + .popover, + .popover .bar-header { + border-radius: 2px; } + + .popover .scroll-content { + z-index: 1; + margin: 2px 0; } + + .popover .bar-header { + border-bottom-right-radius: 0; + border-bottom-left-radius: 0; } + + .popover .has-header { + border-top-right-radius: 0; + border-top-left-radius: 0; } + + .popover-arrow { + display: none; } + + .platform-ios .popover { + box-shadow: 0 0 40px rgba(0, 0, 0, 0.08); + border-radius: 10px; } + + .platform-ios .popover .bar-header { + -webkit-border-top-right-radius: 10px; + border-top-right-radius: 10px; + -webkit-border-top-left-radius: 10px; + border-top-left-radius: 10px; } + + .platform-ios .popover .scroll-content { + margin: 8px 0; + border-radius: 10px; } + + .platform-ios .popover .scroll-content.has-header { + margin-top: 0; } + + .platform-ios .popover-arrow { + position: absolute; + display: block; + top: -17px; + width: 30px; + height: 19px; + overflow: hidden; } + .platform-ios .popover-arrow:after { + position: absolute; + top: 12px; + left: 5px; + width: 20px; + height: 20px; + background-color: #fff; + border-radius: 3px; + content: ''; + -webkit-transform: rotate(-45deg); + transform: rotate(-45deg); } + + .platform-ios .popover-bottom .popover-arrow { + top: auto; + bottom: -10px; } + .platform-ios .popover-bottom .popover-arrow:after { + top: -6px; } + + .platform-android .popover { + margin-top: -32px; + background-color: #fafafa; + box-shadow: 0 2px 6px rgba(0, 0, 0, 0.35); } + .platform-android .popover .item { + border-color: #fafafa; + background-color: #fafafa; + color: #4d4d4d; } + .platform-android .popover.popover-bottom { + margin-top: 32px; } + + .platform-android .popover-backdrop, + .platform-android .popover-backdrop.active { + background-color: transparent; } + + .popover-open { + pointer-events: none; } + .popover-open .popover, + .popover-open .popover-backdrop { + pointer-events: auto; } + .popover-open.loading-active .popover, + .popover-open.loading-active .popover-backdrop { + pointer-events: none; } + + @media (min-width: 680px) { + .popover { + width: 360px; + margin-left: -180px; } } + + /** + * Popups + * -------------------------------------------------- + */ + .popup-container { + position: absolute; + top: 0; + left: 0; + bottom: 0; + right: 0; + background: transparent; + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + -webkit-box-pack: center; + -ms-flex-pack: center; + -webkit-justify-content: center; + -moz-justify-content: center; + justify-content: center; + -webkit-box-align: center; + -ms-flex-align: center; + -webkit-align-items: center; + -moz-align-items: center; + align-items: center; + z-index: 12; + visibility: hidden; } + .popup-container.popup-showing { + visibility: visible; } + .popup-container.popup-hidden .popup { + -webkit-animation-name: scaleOut; + animation-name: scaleOut; + -webkit-animation-duration: 0.1s; + animation-duration: 0.1s; + -webkit-animation-timing-function: ease-in-out; + animation-timing-function: ease-in-out; + -webkit-animation-fill-mode: both; + animation-fill-mode: both; } + .popup-container.active .popup { + -webkit-animation-name: superScaleIn; + animation-name: superScaleIn; + -webkit-animation-duration: 0.2s; + animation-duration: 0.2s; + -webkit-animation-timing-function: ease-in-out; + animation-timing-function: ease-in-out; + -webkit-animation-fill-mode: both; + animation-fill-mode: both; } + .popup-container .popup { + width: 250px; + max-width: 100%; + max-height: 90%; + border-radius: 0px; + background-color: rgba(255, 255, 255, 0.9); + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + -webkit-box-direction: normal; + -webkit-box-orient: vertical; + -webkit-flex-direction: column; + -moz-flex-direction: column; + -ms-flex-direction: column; + flex-direction: column; } + .popup-container input, + .popup-container textarea { + width: 100%; } + + .popup-head { + padding: 15px 10px; + border-bottom: 1px solid #eee; + text-align: center; } + + .popup-title { + margin: 0; + padding: 0; + font-size: 15px; } + + .popup-sub-title { + margin: 5px 0 0 0; + padding: 0; + font-weight: normal; + font-size: 11px; } + + .popup-body { + padding: 10px; + overflow: auto; } + + .popup-buttons { + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + -webkit-box-direction: normal; + -webkit-box-orient: horizontal; + -webkit-flex-direction: row; + -moz-flex-direction: row; + -ms-flex-direction: row; + flex-direction: row; + padding: 10px; + min-height: 65px; } + .popup-buttons .button { + -webkit-box-flex: 1; + -webkit-flex: 1; + -moz-box-flex: 1; + -moz-flex: 1; + -ms-flex: 1; + flex: 1; + display: block; + min-height: 45px; + border-radius: 2px; + line-height: 20px; + margin-right: 5px; } + .popup-buttons .button:last-child { + margin-right: 0px; } + + .popup-open { + pointer-events: none; } + .popup-open.modal-open .modal { + pointer-events: none; } + .popup-open .popup-backdrop, .popup-open .popup { + pointer-events: auto; } + + /** + * Loading + * -------------------------------------------------- + */ + .loading-container { + position: absolute; + left: 0; + top: 0; + right: 0; + bottom: 0; + z-index: 13; + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + -webkit-box-pack: center; + -ms-flex-pack: center; + -webkit-justify-content: center; + -moz-justify-content: center; + justify-content: center; + -webkit-box-align: center; + -ms-flex-align: center; + -webkit-align-items: center; + -moz-align-items: center; + align-items: center; + -webkit-transition: 0.2s opacity linear; + transition: 0.2s opacity linear; + visibility: hidden; + opacity: 0; } + .loading-container:not(.visible) .icon, .loading-container:not(.visible) .icon-help, .loading-container:not(.visible) .icon-alert, .loading-container:not(.visible) #menu .footer .icon-help, #menu .footer .loading-container:not(.visible) .icon-help, + .loading-container:not(.visible) .spinner { + display: none; } + .loading-container.visible { + visibility: visible; } + .loading-container.active { + opacity: 1; } + .loading-container .loading { + padding: 20px; + border-radius: 5px; + background-color: rgba(0, 0, 0, 0.7); + color: #fff; + text-align: center; + text-overflow: ellipsis; + font-size: 15px; } + .loading-container .loading h1, .loading-container .loading h2, .loading-container .loading h3, .loading-container .loading h4, .loading-container .loading h5, .loading-container .loading h6 { + color: #fff; } + + /** + * Items + * -------------------------------------------------- + */ + .item { + border-color: #ddd; + background-color: #fff; + color: #444; + position: relative; + z-index: 2; + display: block; + margin: -1px; + padding: 16px; + border-width: 1px; + border-style: solid; + font-size: 16px; } + .item h2 { + margin: 0 0 2px 0; + font-size: 16px; + font-weight: normal; } + .item h3 { + margin: 0 0 4px 0; + font-size: 14px; } + .item h4 { + margin: 0 0 4px 0; + font-size: 12px; } + .item h5, .item h6 { + margin: 0 0 3px 0; + font-size: 10px; } + .item p { + color: #666; + font-size: 14px; + margin-bottom: 2px; } + .item h1:last-child, + .item h2:last-child, + .item h3:last-child, + .item h4:last-child, + .item h5:last-child, + .item h6:last-child, + .item p:last-child { + margin-bottom: 0; } + .item .badge { + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + position: absolute; + top: 16px; + right: 32px; } + .item.item-button-right .badge { + right: 67px; } + .item.item-divider .badge { + top: 8px; } + .item .badge + .badge { + margin-right: 5px; } + .item.item-light { + border-color: #ddd; + background-color: #fff; + color: #444; } + .item.item-stable { + border-color: #b2b2b2; + background-color: #f8f8f8; + color: #444; } + .item.item-positive { + border-color: #0c60ee; + background-color: #387ef5; + color: #fff; } + .item.item-calm { + border-color: #0a9dc7; + background-color: #11c1f3; + color: #fff; } + .item.item-assertive { + border-color: #e42112; + background-color: #ef473a; + color: #fff; } + .item.item-balanced { + border-color: #28a54c; + background-color: #33cd5f; + color: #fff; } + .item.item-energized { + border-color: #e6b500; + background-color: #ffc900; + color: #fff; } + .item.item-royal { + border-color: #6b46e5; + background-color: #886aea; + color: #fff; } + .item.item-dark { + border-color: #111; + background-color: #444; + color: #fff; } + .item[ng-click]:hover { + cursor: pointer; } + + .list-borderless .item, + .item-borderless { + border-width: 0; } + + .item.active, + .item.activated, + .item-complex.active .item-content, + .item-complex.activated .item-content, + .item .item-content.active, + .item .item-content.activated { + border-color: #ccc; + background-color: #D9D9D9; } + .item.active.item-complex > .item-content, + .item.activated.item-complex > .item-content, + .item-complex.active .item-content.item-complex > .item-content, + .item-complex.activated .item-content.item-complex > .item-content, + .item .item-content.active.item-complex > .item-content, + .item .item-content.activated.item-complex > .item-content { + border-color: #ccc; + background-color: #D9D9D9; } + .item.active.item-light, + .item.activated.item-light, + .item-complex.active .item-content.item-light, + .item-complex.activated .item-content.item-light, + .item .item-content.active.item-light, + .item .item-content.activated.item-light { + border-color: #ccc; + background-color: #fafafa; } + .item.active.item-light.item-complex > .item-content, + .item.activated.item-light.item-complex > .item-content, + .item-complex.active .item-content.item-light.item-complex > .item-content, + .item-complex.activated .item-content.item-light.item-complex > .item-content, + .item .item-content.active.item-light.item-complex > .item-content, + .item .item-content.activated.item-light.item-complex > .item-content { + border-color: #ccc; + background-color: #fafafa; } + .item.active.item-stable, + .item.activated.item-stable, + .item-complex.active .item-content.item-stable, + .item-complex.activated .item-content.item-stable, + .item .item-content.active.item-stable, + .item .item-content.activated.item-stable { + border-color: #a2a2a2; + background-color: #e5e5e5; } + .item.active.item-stable.item-complex > .item-content, + .item.activated.item-stable.item-complex > .item-content, + .item-complex.active .item-content.item-stable.item-complex > .item-content, + .item-complex.activated .item-content.item-stable.item-complex > .item-content, + .item .item-content.active.item-stable.item-complex > .item-content, + .item .item-content.activated.item-stable.item-complex > .item-content { + border-color: #a2a2a2; + background-color: #e5e5e5; } + .item.active.item-positive, + .item.activated.item-positive, + .item-complex.active .item-content.item-positive, + .item-complex.activated .item-content.item-positive, + .item .item-content.active.item-positive, + .item .item-content.activated.item-positive { + border-color: #0c60ee; + background-color: #0c60ee; } + .item.active.item-positive.item-complex > .item-content, + .item.activated.item-positive.item-complex > .item-content, + .item-complex.active .item-content.item-positive.item-complex > .item-content, + .item-complex.activated .item-content.item-positive.item-complex > .item-content, + .item .item-content.active.item-positive.item-complex > .item-content, + .item .item-content.activated.item-positive.item-complex > .item-content { + border-color: #0c60ee; + background-color: #0c60ee; } + .item.active.item-calm, + .item.activated.item-calm, + .item-complex.active .item-content.item-calm, + .item-complex.activated .item-content.item-calm, + .item .item-content.active.item-calm, + .item .item-content.activated.item-calm { + border-color: #0a9dc7; + background-color: #0a9dc7; } + .item.active.item-calm.item-complex > .item-content, + .item.activated.item-calm.item-complex > .item-content, + .item-complex.active .item-content.item-calm.item-complex > .item-content, + .item-complex.activated .item-content.item-calm.item-complex > .item-content, + .item .item-content.active.item-calm.item-complex > .item-content, + .item .item-content.activated.item-calm.item-complex > .item-content { + border-color: #0a9dc7; + background-color: #0a9dc7; } + .item.active.item-assertive, + .item.activated.item-assertive, + .item-complex.active .item-content.item-assertive, + .item-complex.activated .item-content.item-assertive, + .item .item-content.active.item-assertive, + .item .item-content.activated.item-assertive { + border-color: #e42112; + background-color: #e42112; } + .item.active.item-assertive.item-complex > .item-content, + .item.activated.item-assertive.item-complex > .item-content, + .item-complex.active .item-content.item-assertive.item-complex > .item-content, + .item-complex.activated .item-content.item-assertive.item-complex > .item-content, + .item .item-content.active.item-assertive.item-complex > .item-content, + .item .item-content.activated.item-assertive.item-complex > .item-content { + border-color: #e42112; + background-color: #e42112; } + .item.active.item-balanced, + .item.activated.item-balanced, + .item-complex.active .item-content.item-balanced, + .item-complex.activated .item-content.item-balanced, + .item .item-content.active.item-balanced, + .item .item-content.activated.item-balanced { + border-color: #28a54c; + background-color: #28a54c; } + .item.active.item-balanced.item-complex > .item-content, + .item.activated.item-balanced.item-complex > .item-content, + .item-complex.active .item-content.item-balanced.item-complex > .item-content, + .item-complex.activated .item-content.item-balanced.item-complex > .item-content, + .item .item-content.active.item-balanced.item-complex > .item-content, + .item .item-content.activated.item-balanced.item-complex > .item-content { + border-color: #28a54c; + background-color: #28a54c; } + .item.active.item-energized, + .item.activated.item-energized, + .item-complex.active .item-content.item-energized, + .item-complex.activated .item-content.item-energized, + .item .item-content.active.item-energized, + .item .item-content.activated.item-energized { + border-color: #e6b500; + background-color: #e6b500; } + .item.active.item-energized.item-complex > .item-content, + .item.activated.item-energized.item-complex > .item-content, + .item-complex.active .item-content.item-energized.item-complex > .item-content, + .item-complex.activated .item-content.item-energized.item-complex > .item-content, + .item .item-content.active.item-energized.item-complex > .item-content, + .item .item-content.activated.item-energized.item-complex > .item-content { + border-color: #e6b500; + background-color: #e6b500; } + .item.active.item-royal, + .item.activated.item-royal, + .item-complex.active .item-content.item-royal, + .item-complex.activated .item-content.item-royal, + .item .item-content.active.item-royal, + .item .item-content.activated.item-royal { + border-color: #6b46e5; + background-color: #6b46e5; } + .item.active.item-royal.item-complex > .item-content, + .item.activated.item-royal.item-complex > .item-content, + .item-complex.active .item-content.item-royal.item-complex > .item-content, + .item-complex.activated .item-content.item-royal.item-complex > .item-content, + .item .item-content.active.item-royal.item-complex > .item-content, + .item .item-content.activated.item-royal.item-complex > .item-content { + border-color: #6b46e5; + background-color: #6b46e5; } + .item.active.item-dark, + .item.activated.item-dark, + .item-complex.active .item-content.item-dark, + .item-complex.activated .item-content.item-dark, + .item .item-content.active.item-dark, + .item .item-content.activated.item-dark { + border-color: #000; + background-color: #262626; } + .item.active.item-dark.item-complex > .item-content, + .item.activated.item-dark.item-complex > .item-content, + .item-complex.active .item-content.item-dark.item-complex > .item-content, + .item-complex.activated .item-content.item-dark.item-complex > .item-content, + .item .item-content.active.item-dark.item-complex > .item-content, + .item .item-content.activated.item-dark.item-complex > .item-content { + border-color: #000; + background-color: #262626; } + + .item, + .item h1, + .item h2, + .item h3, + .item h4, + .item h5, + .item h6, + .item p, + .item-content, + .item-content h1, + .item-content h2, + .item-content h3, + .item-content h4, + .item-content h5, + .item-content h6, + .item-content p { + overflow: hidden; + text-overflow: ellipsis; + white-space: nowrap; } + + a.item { + color: inherit; + text-decoration: none; } + a.item:hover, a.item:focus { + text-decoration: none; } + + /** + * Complex Items + * -------------------------------------------------- + * Adding .item-complex allows the .item to be slidable and + * have options underneath the button, but also requires an + * additional .item-content element inside .item. + * Basically .item-complex removes any default settings which + * .item added, so that .item-content looks them as just .item. + */ + .item-complex, + a.item.item-complex, + button.item.item-complex { + padding: 0; } + + .item-complex .item-content, + .item-radio .item-content { + position: relative; + z-index: 2; + padding: 16px 49px 16px 16px; + border: none; + background-color: #fff; } + + a.item-content { + display: block; + color: inherit; + text-decoration: none; } + + .item-text-wrap .item, + .item-text-wrap .item-content, + .item-text-wrap, + .item-text-wrap h1, + .item-text-wrap h2, + .item-text-wrap h3, + .item-text-wrap h4, + .item-text-wrap h5, + .item-text-wrap h6, + .item-text-wrap p, + .item-complex.item-text-wrap .item-content, + .item-body h1, + .item-body h2, + .item-body h3, + .item-body h4, + .item-body h5, + .item-body h6, + .item-body p { + overflow: visible; + white-space: normal; } + + .item-complex.item-text-wrap, + .item-complex.item-text-wrap h1, + .item-complex.item-text-wrap h2, + .item-complex.item-text-wrap h3, + .item-complex.item-text-wrap h4, + .item-complex.item-text-wrap h5, + .item-complex.item-text-wrap h6, + .item-complex.item-text-wrap p { + overflow: visible; + white-space: normal; } + + .item-complex.item-light > .item-content { + border-color: #ddd; + background-color: #fff; + color: #444; } + .item-complex.item-light > .item-content.active, .item-complex.item-light > .item-content:active { + border-color: #ccc; + background-color: #fafafa; } + .item-complex.item-light > .item-content.active.item-complex > .item-content, .item-complex.item-light > .item-content:active.item-complex > .item-content { + border-color: #ccc; + background-color: #fafafa; } + + .item-complex.item-stable > .item-content { + border-color: #b2b2b2; + background-color: #f8f8f8; + color: #444; } + .item-complex.item-stable > .item-content.active, .item-complex.item-stable > .item-content:active { + border-color: #a2a2a2; + background-color: #e5e5e5; } + .item-complex.item-stable > .item-content.active.item-complex > .item-content, .item-complex.item-stable > .item-content:active.item-complex > .item-content { + border-color: #a2a2a2; + background-color: #e5e5e5; } + + .item-complex.item-positive > .item-content { + border-color: #0c60ee; + background-color: #387ef5; + color: #fff; } + .item-complex.item-positive > .item-content.active, .item-complex.item-positive > .item-content:active { + border-color: #0c60ee; + background-color: #0c60ee; } + .item-complex.item-positive > .item-content.active.item-complex > .item-content, .item-complex.item-positive > .item-content:active.item-complex > .item-content { + border-color: #0c60ee; + background-color: #0c60ee; } + + .item-complex.item-calm > .item-content { + border-color: #0a9dc7; + background-color: #11c1f3; + color: #fff; } + .item-complex.item-calm > .item-content.active, .item-complex.item-calm > .item-content:active { + border-color: #0a9dc7; + background-color: #0a9dc7; } + .item-complex.item-calm > .item-content.active.item-complex > .item-content, .item-complex.item-calm > .item-content:active.item-complex > .item-content { + border-color: #0a9dc7; + background-color: #0a9dc7; } + + .item-complex.item-assertive > .item-content { + border-color: #e42112; + background-color: #ef473a; + color: #fff; } + .item-complex.item-assertive > .item-content.active, .item-complex.item-assertive > .item-content:active { + border-color: #e42112; + background-color: #e42112; } + .item-complex.item-assertive > .item-content.active.item-complex > .item-content, .item-complex.item-assertive > .item-content:active.item-complex > .item-content { + border-color: #e42112; + background-color: #e42112; } + + .item-complex.item-balanced > .item-content { + border-color: #28a54c; + background-color: #33cd5f; + color: #fff; } + .item-complex.item-balanced > .item-content.active, .item-complex.item-balanced > .item-content:active { + border-color: #28a54c; + background-color: #28a54c; } + .item-complex.item-balanced > .item-content.active.item-complex > .item-content, .item-complex.item-balanced > .item-content:active.item-complex > .item-content { + border-color: #28a54c; + background-color: #28a54c; } + + .item-complex.item-energized > .item-content { + border-color: #e6b500; + background-color: #ffc900; + color: #fff; } + .item-complex.item-energized > .item-content.active, .item-complex.item-energized > .item-content:active { + border-color: #e6b500; + background-color: #e6b500; } + .item-complex.item-energized > .item-content.active.item-complex > .item-content, .item-complex.item-energized > .item-content:active.item-complex > .item-content { + border-color: #e6b500; + background-color: #e6b500; } + + .item-complex.item-royal > .item-content { + border-color: #6b46e5; + background-color: #886aea; + color: #fff; } + .item-complex.item-royal > .item-content.active, .item-complex.item-royal > .item-content:active { + border-color: #6b46e5; + background-color: #6b46e5; } + .item-complex.item-royal > .item-content.active.item-complex > .item-content, .item-complex.item-royal > .item-content:active.item-complex > .item-content { + border-color: #6b46e5; + background-color: #6b46e5; } + + .item-complex.item-dark > .item-content { + border-color: #111; + background-color: #444; + color: #fff; } + .item-complex.item-dark > .item-content.active, .item-complex.item-dark > .item-content:active { + border-color: #000; + background-color: #262626; } + .item-complex.item-dark > .item-content.active.item-complex > .item-content, .item-complex.item-dark > .item-content:active.item-complex > .item-content { + border-color: #000; + background-color: #262626; } + + /** + * Item Icons + * -------------------------------------------------- + */ + .item-icon-left .icon, .item-icon-left .icon-help, .item-icon-left .icon-alert, .item-icon-left #menu .footer .icon-help, #menu .footer .item-icon-left .icon-help, + .item-icon-right .icon, + .item-icon-right .icon-help, + .item-icon-right .icon-alert, + .item-icon-right #menu .footer .icon-help, #menu .footer + .item-icon-right .icon-help { + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + -webkit-box-align: center; + -ms-flex-align: center; + -webkit-align-items: center; + -moz-align-items: center; + align-items: center; + position: absolute; + top: 0; + height: 100%; + font-size: 32px; } + .item-icon-left .icon:before, .item-icon-left .icon-help:before, .item-icon-left .icon-alert:before, .item-icon-left #menu .footer .icon-help:before, #menu .footer .item-icon-left .icon-help:before, + .item-icon-right .icon:before, + .item-icon-right .icon-help:before, + .item-icon-right .icon-alert:before, + .item-icon-right #menu .footer .icon-help:before, #menu .footer + .item-icon-right .icon-help:before { + display: block; + width: 32px; + text-align: center; } + + .item .fill-icon { + min-width: 30px; + min-height: 30px; + font-size: 28px; } + + .item-icon-left { + padding-left: 54px; } + .item-icon-left .icon, .item-icon-left .icon-help, .item-icon-left .icon-alert, .item-icon-left #menu .footer .icon-help, #menu .footer .item-icon-left .icon-help { + left: 11px; } + + .item-complex.item-icon-left { + padding-left: 0; } + .item-complex.item-icon-left .item-content { + padding-left: 54px; } + + .item-icon-right { + padding-right: 54px; } + .item-icon-right .icon, .item-icon-right .icon-help, .item-icon-right .icon-alert, .item-icon-right #menu .footer .icon-help, #menu .footer .item-icon-right .icon-help { + right: 11px; } + + .item-complex.item-icon-right { + padding-right: 0; } + .item-complex.item-icon-right .item-content { + padding-right: 54px; } + + .item-icon-left.item-icon-right .icon:first-child, .item-icon-left.item-icon-right .icon-help:first-child, .item-icon-left.item-icon-right .icon-alert:first-child, .item-icon-left.item-icon-right #menu .footer .icon-help:first-child, #menu .footer .item-icon-left.item-icon-right .icon-help:first-child { + right: auto; } + + .item-icon-left.item-icon-right .icon:last-child, .item-icon-left.item-icon-right .icon-help:last-child, .item-icon-left.item-icon-right .icon-alert:last-child, .item-icon-left.item-icon-right #menu .footer .icon-help:last-child, #menu .footer .item-icon-left.item-icon-right .icon-help:last-child, + .item-icon-left .item-delete .icon, + .item-icon-left .item-delete .icon-help, + .item-icon-left .item-delete .icon-alert, + .item-icon-left .item-delete #menu .footer .icon-help, #menu .footer + .item-icon-left .item-delete .icon-help { + left: auto; } + + .item-icon-left .icon-accessory, + .item-icon-right .icon-accessory { + color: #ccc; + font-size: 16px; } + + .item-icon-left .icon-accessory { + left: 3px; } + + .item-icon-right .icon-accessory { + right: 3px; } + + /** + * Item Button + * -------------------------------------------------- + * An item button is a child button inside an .item (not the entire .item) + */ + .item-button-left { + padding-left: 72px; } + + .item-button-left > .button, + .item-button-left .item-content > .button { + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + -webkit-box-align: center; + -ms-flex-align: center; + -webkit-align-items: center; + -moz-align-items: center; + align-items: center; + position: absolute; + top: 8px; + left: 11px; + min-width: 34px; + min-height: 34px; + font-size: 18px; + line-height: 32px; } + .item-button-left > .button .icon:before, .item-button-left > .button .icon-help:before, .item-button-left > .button .icon-alert:before, .item-button-left > .button #menu .footer .icon-help:before, #menu .footer .item-button-left > .button .icon-help:before, + .item-button-left .item-content > .button .icon:before, + .item-button-left .item-content > .button .icon-help:before, + .item-button-left .item-content > .button .icon-alert:before, + .item-button-left .item-content > .button #menu .footer .icon-help:before, #menu .footer + .item-button-left .item-content > .button .icon-help:before { + position: relative; + left: auto; + width: auto; + line-height: 31px; } + .item-button-left > .button > .button, + .item-button-left .item-content > .button > .button { + margin: 0px 2px; + min-height: 34px; + font-size: 18px; + line-height: 32px; } + + .item-button-right, + a.item.item-button-right, + button.item.item-button-right { + padding-right: 80px; } + + .item-button-right > .button, + .item-button-right .item-content > .button, + .item-button-right > .buttons, + .item-button-right .item-content > .buttons { + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + -webkit-box-align: center; + -ms-flex-align: center; + -webkit-align-items: center; + -moz-align-items: center; + align-items: center; + position: absolute; + top: 8px; + right: 16px; + min-width: 34px; + min-height: 34px; + font-size: 18px; + line-height: 32px; } + .item-button-right > .button .icon:before, .item-button-right > .button .icon-help:before, .item-button-right > .button .icon-alert:before, .item-button-right > .button #menu .footer .icon-help:before, #menu .footer .item-button-right > .button .icon-help:before, + .item-button-right .item-content > .button .icon:before, + .item-button-right .item-content > .button .icon-help:before, + .item-button-right .item-content > .button .icon-alert:before, + .item-button-right .item-content > .button #menu .footer .icon-help:before, #menu .footer + .item-button-right .item-content > .button .icon-help:before, + .item-button-right > .buttons .icon:before, + .item-button-right > .buttons .icon-help:before, + .item-button-right > .buttons .icon-alert:before, + .item-button-right > .buttons #menu .footer .icon-help:before, #menu .footer + .item-button-right > .buttons .icon-help:before, + .item-button-right .item-content > .buttons .icon:before, + .item-button-right .item-content > .buttons .icon-help:before, + .item-button-right .item-content > .buttons .icon-alert:before, + .item-button-right .item-content > .buttons #menu .footer .icon-help:before, #menu .footer + .item-button-right .item-content > .buttons .icon-help:before { + position: relative; + left: auto; + width: auto; + line-height: 31px; } + .item-button-right > .button > .button, + .item-button-right .item-content > .button > .button, + .item-button-right > .buttons > .button, + .item-button-right .item-content > .buttons > .button { + margin: 0px 2px; + min-width: 34px; + min-height: 34px; + font-size: 18px; + line-height: 32px; } + + .item-button-left.item-button-right .button:first-child { + right: auto; } + + .item-button-left.item-button-right .button:last-child { + left: auto; } + + .item-avatar, + .item-avatar .item-content, + .item-avatar-left, + .item-avatar-left .item-content { + padding-left: 72px; + min-height: 72px; } + .item-avatar > img:first-child, + .item-avatar .item-image, + .item-avatar .item-content > img:first-child, + .item-avatar .item-content .item-image, + .item-avatar-left > img:first-child, + .item-avatar-left .item-image, + .item-avatar-left .item-content > img:first-child, + .item-avatar-left .item-content .item-image { + position: absolute; + top: 16px; + left: 16px; + max-width: 40px; + max-height: 40px; + width: 100%; + height: 100%; + border-radius: 50%; } + + .item-avatar-right, + .item-avatar-right .item-content { + padding-right: 72px; + min-height: 72px; } + .item-avatar-right > img:first-child, + .item-avatar-right .item-image, + .item-avatar-right .item-content > img:first-child, + .item-avatar-right .item-content .item-image { + position: absolute; + top: 16px; + right: 16px; + max-width: 40px; + max-height: 40px; + width: 100%; + height: 100%; + border-radius: 50%; } + + .item-thumbnail-left, + .item-thumbnail-left .item-content { + padding-top: 8px; + padding-left: 106px; + min-height: 100px; } + .item-thumbnail-left > img:first-child, + .item-thumbnail-left .item-image, + .item-thumbnail-left .item-content > img:first-child, + .item-thumbnail-left .item-content .item-image { + position: absolute; + top: 10px; + left: 10px; + max-width: 80px; + max-height: 80px; + width: 100%; + height: 100%; } + + .item-avatar.item-complex, + .item-avatar-left.item-complex, + .item-thumbnail-left.item-complex { + padding-top: 0; + padding-left: 0; } + + .item-thumbnail-right, + .item-thumbnail-right .item-content { + padding-top: 8px; + padding-right: 106px; + min-height: 100px; } + .item-thumbnail-right > img:first-child, + .item-thumbnail-right .item-image, + .item-thumbnail-right .item-content > img:first-child, + .item-thumbnail-right .item-content .item-image { + position: absolute; + top: 10px; + right: 10px; + max-width: 80px; + max-height: 80px; + width: 100%; + height: 100%; } + + .item-avatar-right.item-complex, + .item-thumbnail-right.item-complex { + padding-top: 0; + padding-right: 0; } + + .item-image { + padding: 0; + text-align: center; } + .item-image img:first-child, .item-image .list-img { + width: 100%; + vertical-align: middle; } + + .item-body { + overflow: auto; + padding: 16px; + text-overflow: inherit; + white-space: normal; } + .item-body h1, .item-body h2, .item-body h3, .item-body h4, .item-body h5, .item-body h6, .item-body p { + margin-top: 16px; + margin-bottom: 16px; } + + .item-divider { + padding-top: 8px; + padding-bottom: 8px; + min-height: 30px; + background-color: #f5f5f5; + color: #222; + font-weight: 500; } + + .platform-ios .item-divider-platform, + .item-divider-ios { + padding-top: 26px; + text-transform: uppercase; + font-weight: 300; + font-size: 13px; + background-color: #efeff4; + color: #555; } + + .platform-android .item-divider-platform, + .item-divider-android { + font-weight: 300; + font-size: 13px; } + + .item-note { + float: right; + color: #aaa; + font-size: 14px; } + + .item-left-editable .item-content, + .item-right-editable .item-content { + -webkit-transition-duration: 250ms; + transition-duration: 250ms; + -webkit-transition-timing-function: ease-in-out; + transition-timing-function: ease-in-out; + -webkit-transition-property: -webkit-transform; + -moz-transition-property: -moz-transform; + transition-property: transform; } + + .list-left-editing .item-left-editable .item-content, + .item-left-editing.item-left-editable .item-content { + -webkit-transform: translate3d(50px, 0, 0); + transform: translate3d(50px, 0, 0); } + + .item-remove-animate.ng-leave { + -webkit-transition-duration: 300ms; + transition-duration: 300ms; } + + .item-remove-animate.ng-leave .item-content, .item-remove-animate.ng-leave:last-of-type { + -webkit-transition-duration: 300ms; + transition-duration: 300ms; + -webkit-transition-timing-function: ease-in; + transition-timing-function: ease-in; + -webkit-transition-property: all; + transition-property: all; } + + .item-remove-animate.ng-leave.ng-leave-active .item-content { + opacity: 0; + -webkit-transform: translate3d(-100%, 0, 0) !important; + transform: translate3d(-100%, 0, 0) !important; } + + .item-remove-animate.ng-leave.ng-leave-active:last-of-type { + opacity: 0; } + + .item-remove-animate.ng-leave.ng-leave-active ~ ion-item:not(.ng-leave) { + -webkit-transform: translate3d(0, -webkit-calc(-100% + 1px), 0); + transform: translate3d(0, calc(-100% + 1px), 0); + -webkit-transition-duration: 300ms; + transition-duration: 300ms; + -webkit-transition-timing-function: cubic-bezier(0.25, 0.81, 0.24, 1); + transition-timing-function: cubic-bezier(0.25, 0.81, 0.24, 1); + -webkit-transition-property: all; + transition-property: all; } + + .item-left-edit { + -webkit-transition: all ease-in-out 125ms; + transition: all ease-in-out 125ms; + position: absolute; + top: 0; + left: 0; + z-index: 0; + width: 50px; + height: 100%; + line-height: 100%; + display: none; + opacity: 0; + -webkit-transform: translate3d(-21px, 0, 0); + transform: translate3d(-21px, 0, 0); } + .item-left-edit .button { + height: 100%; } + .item-left-edit .button.icon, .item-left-edit .button.icon-help, .item-left-edit .button.icon-alert, .item-left-edit #menu .footer .button.icon-help, #menu .footer .item-left-edit .button.icon-help { + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + -webkit-box-align: center; + -ms-flex-align: center; + -webkit-align-items: center; + -moz-align-items: center; + align-items: center; + position: absolute; + top: 0; + height: 100%; } + .item-left-edit.visible { + display: block; } + .item-left-edit.visible.active { + opacity: 1; + -webkit-transform: translate3d(8px, 0, 0); + transform: translate3d(8px, 0, 0); } + + .list-left-editing .item-left-edit { + -webkit-transition-delay: 125ms; + transition-delay: 125ms; } + + .item-delete .button.icon, .item-delete .button.icon-help, .item-delete .button.icon-alert, .item-delete #menu .footer .button.icon-help, #menu .footer .item-delete .button.icon-help { + color: #ef473a; + font-size: 24px; } + .item-delete .button.icon:hover, .item-delete .button.icon-help:hover, .item-delete .button.icon-alert:hover, .item-delete #menu .footer .button.icon-help:hover, #menu .footer .item-delete .button.icon-help:hover { + opacity: .7; } + + .item-right-edit { + -webkit-transition: all ease-in-out 250ms; + transition: all ease-in-out 250ms; + position: absolute; + top: 0; + right: 0; + z-index: 3; + width: 75px; + height: 100%; + background: inherit; + padding-left: 20px; + display: block; + opacity: 0; + -webkit-transform: translate3d(75px, 0, 0); + transform: translate3d(75px, 0, 0); } + .item-right-edit .button { + min-width: 50px; + height: 100%; } + .item-right-edit .button.icon, .item-right-edit .button.icon-help, .item-right-edit .button.icon-alert, .item-right-edit #menu .footer .button.icon-help, #menu .footer .item-right-edit .button.icon-help { + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + -webkit-box-align: center; + -ms-flex-align: center; + -webkit-align-items: center; + -moz-align-items: center; + align-items: center; + position: absolute; + top: 0; + height: 100%; + font-size: 32px; } + .item-right-edit.visible { + display: block; } + .item-right-edit.visible.active { + opacity: 1; + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); } + + .item-reorder .button.icon, .item-reorder .button.icon-help, .item-reorder .button.icon-alert, .item-reorder #menu .footer .button.icon-help, #menu .footer .item-reorder .button.icon-help { + color: #444; + font-size: 32px; } + + .item-reordering { + position: absolute; + left: 0; + top: 0; + z-index: 9; + width: 100%; + box-shadow: 0px 0px 10px 0px #aaa; } + .item-reordering .item-reorder { + z-index: 9; } + + .item-placeholder { + opacity: 0.7; } + + /** + * The hidden right-side buttons that can be exposed under a list item + * with dragging. + */ + .item-options { + position: absolute; + top: 0; + right: 0; + z-index: 1; + height: 100%; } + .item-options .button { + height: 100%; + border: none; + border-radius: 0; + display: -webkit-inline-box; + display: -webkit-inline-flex; + display: -moz-inline-flex; + display: -ms-inline-flexbox; + display: inline-flex; + -webkit-box-align: center; + -ms-flex-align: center; + -webkit-align-items: center; + -moz-align-items: center; + align-items: center; } + .item-options .button:before { + margin: 0 auto; } + + /** + * Lists + * -------------------------------------------------- + */ + .list { + position: relative; + padding-top: 1px; + padding-bottom: 1px; + padding-left: 0; + margin-bottom: 20px; } + + .list:last-child { + margin-bottom: 0px; } + .list:last-child.card { + margin-bottom: 40px; } + + /** + * List Header + * -------------------------------------------------- + */ + .list-header { + margin-top: 20px; + padding: 5px 15px; + background-color: transparent; + color: #222; + font-weight: bold; } + + .card.list .list-item { + padding-right: 1px; + padding-left: 1px; } + + /** + * Cards and Inset Lists + * -------------------------------------------------- + * A card and list-inset are close to the same thing, except a card as a box shadow. + */ + .card, + .list-inset { + overflow: hidden; + margin: 20px 10px; + border-radius: 2px; + background-color: #fff; } + + .card { + padding-top: 1px; + padding-bottom: 1px; + box-shadow: 0 1px 3px rgba(0, 0, 0, 0.3); } + .card .item { + border-left: 0; + border-right: 0; } + .card .item:first-child { + border-top: 0; } + .card .item:last-child { + border-bottom: 0; } + + .padding .card, .item.large-button-bar .card, .padding .list-inset, .item.large-button-bar .list-inset { + margin-left: 0; + margin-right: 0; } + + .card .item:first-child, + .list-inset .item:first-child, + .padding > .list .item:first-child, .item.large-button-bar > .list .item:first-child { + border-top-left-radius: 2px; + border-top-right-radius: 2px; } + .card .item:first-child .item-content, + .list-inset .item:first-child .item-content, + .padding > .list .item:first-child .item-content, .item.large-button-bar > .list .item:first-child .item-content { + border-top-left-radius: 2px; + border-top-right-radius: 2px; } + + .card .item:last-child, + .list-inset .item:last-child, + .padding > .list .item:last-child, .item.large-button-bar > .list .item:last-child { + border-bottom-right-radius: 2px; + border-bottom-left-radius: 2px; } + .card .item:last-child .item-content, + .list-inset .item:last-child .item-content, + .padding > .list .item:last-child .item-content, .item.large-button-bar > .list .item:last-child .item-content { + border-bottom-right-radius: 2px; + border-bottom-left-radius: 2px; } + + .card .item:last-child, + .list-inset .item:last-child { + margin-bottom: -1px; } + + .card .item, + .list-inset .item, + .padding > .list .item, .item.large-button-bar > .list .item, + .padding-horizontal > .list .item { + margin-right: 0; + margin-left: 0; } + .card .item.item-input input, + .list-inset .item.item-input input, + .padding > .list .item.item-input input, .item.large-button-bar > .list .item.item-input input, + .padding-horizontal > .list .item.item-input input { + padding-right: 44px; } + + .padding-left > .list .item { + margin-left: 0; } + + .padding-right > .list .item, .popover-share .bar-footer .button-close > .list .item { + margin-right: 0; } + + /** + * Badges + * -------------------------------------------------- + */ + .badge { + background-color: transparent; + color: #AAAAAA; + z-index: 1; + display: inline-block; + padding: 3px 8px; + min-width: 10px; + border-radius: 10px; + vertical-align: baseline; + text-align: center; + white-space: nowrap; + font-weight: bold; + font-size: 14px; + line-height: 16px; } + .badge:empty { + display: none; } + + .tabs .tab-item .badge.badge-light, + .badge.badge-light { + background-color: #fff; + color: #444; } + + .tabs .tab-item .badge.badge-stable, + .badge.badge-stable { + background-color: #f8f8f8; + color: #444; } + + .tabs .tab-item .badge.badge-positive, + .badge.badge-positive { + background-color: #387ef5; + color: #fff; } + + .tabs .tab-item .badge.badge-calm, + .badge.badge-calm { + background-color: #11c1f3; + color: #fff; } + + .tabs .tab-item .badge.badge-assertive, .tabs .tab-item .badge.badge-editable:hover, + .badge.badge-assertive, + .badge.badge-editable:hover { + background-color: #ef473a; + color: #fff; } + + .tabs .tab-item .badge.badge-balanced, + .badge.badge-balanced { + background-color: #33cd5f; + color: #fff; } + + .tabs .tab-item .badge.badge-energized, + .badge.badge-energized { + background-color: #ffc900; + color: #fff; } + + .tabs .tab-item .badge.badge-royal, + .badge.badge-royal { + background-color: #886aea; + color: #fff; } + + .tabs .tab-item .badge.badge-dark, + .badge.badge-dark { + background-color: #444; + color: #fff; } + + .button .badge { + position: relative; + top: -1px; } + + /** + * Slide Box + * -------------------------------------------------- + */ + .slider { + position: relative; + visibility: hidden; + overflow: hidden; } + + .slider-slides { + position: relative; + height: 100%; } + + .slider-slide { + position: relative; + display: block; + float: left; + width: 100%; + height: 100%; + vertical-align: top; } + + .slider-slide-image > img { + width: 100%; } + + .slider-pager { + position: absolute; + bottom: 20px; + z-index: 1; + width: 100%; + height: 15px; + text-align: center; } + .slider-pager .slider-pager-page { + display: inline-block; + margin: 0px 3px; + width: 15px; + color: #000; + text-decoration: none; + opacity: 0.3; } + .slider-pager .slider-pager-page.active { + -webkit-transition: opacity 0.4s ease-in; + transition: opacity 0.4s ease-in; + opacity: 1; } + + .slider-slide.ng-enter, .slider-slide.ng-leave, .slider-slide.ng-animate, + .slider-pager-page.ng-enter, + .slider-pager-page.ng-leave, + .slider-pager-page.ng-animate { + -webkit-transition: none !important; + transition: none !important; } + + .slider-slide.ng-animate, + .slider-pager-page.ng-animate { + -webkit-animation: none 0s; + animation: none 0s; } + + /** + * Swiper 3.2.7 + * Most modern mobile touch slider and framework with hardware accelerated transitions + * + * http://www.idangero.us/swiper/ + * + * Copyright 2015, Vladimir Kharlampidi + * The iDangero.us + * http://www.idangero.us/ + * + * Licensed under MIT + * + * Released on: December 7, 2015 + */ + .swiper-container { + margin: 0 auto; + position: relative; + overflow: hidden; + /* Fix of Webkit flickering */ + z-index: 1; } + + .swiper-container-no-flexbox .swiper-slide { + float: left; } + + .swiper-container-vertical > .swiper-wrapper { + -webkit-box-orient: vertical; + -moz-box-orient: vertical; + -ms-flex-direction: column; + -webkit-flex-direction: column; + flex-direction: column; } + + .swiper-wrapper { + position: relative; + width: 100%; + height: 100%; + z-index: 1; + display: -webkit-box; + display: -moz-box; + display: -ms-flexbox; + display: -webkit-flex; + display: flex; + -webkit-transition-property: -webkit-transform; + -moz-transition-property: -moz-transform; + -o-transition-property: -o-transform; + -ms-transition-property: -ms-transform; + transition-property: transform; + -webkit-box-sizing: content-box; + -moz-box-sizing: content-box; + box-sizing: content-box; } + + .swiper-container-android .swiper-slide, + .swiper-wrapper { + -webkit-transform: translate3d(0px, 0, 0); + -moz-transform: translate3d(0px, 0, 0); + -o-transform: translate(0px, 0px); + -ms-transform: translate3d(0px, 0, 0); + transform: translate3d(0px, 0, 0); } + + .swiper-container-multirow > .swiper-wrapper { + -webkit-box-lines: multiple; + -moz-box-lines: multiple; + -ms-flex-wrap: wrap; + -webkit-flex-wrap: wrap; + flex-wrap: wrap; } + + .swiper-container-free-mode > .swiper-wrapper { + -webkit-transition-timing-function: ease-out; + -moz-transition-timing-function: ease-out; + -ms-transition-timing-function: ease-out; + -o-transition-timing-function: ease-out; + transition-timing-function: ease-out; + margin: 0 auto; } + + .swiper-slide { + display: block; + -webkit-flex-shrink: 0; + -ms-flex: 0 0 auto; + flex-shrink: 0; + width: 100%; + height: 100%; + position: relative; } + + /* Auto Height */ + .swiper-container-autoheight, + .swiper-container-autoheight .swiper-slide { + height: auto; } + + .swiper-container-autoheight .swiper-wrapper { + -webkit-box-align: start; + -ms-flex-align: start; + -webkit-align-items: flex-start; + align-items: flex-start; + -webkit-transition-property: -webkit-transform, height; + -moz-transition-property: -moz-transform; + -o-transition-property: -o-transform; + -ms-transition-property: -ms-transform; + transition-property: transform, height; } + + /* a11y */ + .swiper-container .swiper-notification { + position: absolute; + left: 0; + top: 0; + pointer-events: none; + opacity: 0; + z-index: -1000; } + + /* IE10 Windows Phone 8 Fixes */ + .swiper-wp8-horizontal { + -ms-touch-action: pan-y; + touch-action: pan-y; } + + .swiper-wp8-vertical { + -ms-touch-action: pan-x; + touch-action: pan-x; } + + /* Arrows */ + .swiper-button-prev, + .swiper-button-next { + position: absolute; + top: 50%; + width: 27px; + height: 44px; + margin-top: -22px; + z-index: 10; + cursor: pointer; + -moz-background-size: 27px 44px; + -webkit-background-size: 27px 44px; + background-size: 27px 44px; + background-position: center; + background-repeat: no-repeat; } + + .swiper-button-prev.swiper-button-disabled, + .swiper-button-next.swiper-button-disabled { + opacity: 0.35; + cursor: auto; + pointer-events: none; } + + .swiper-button-prev, + .swiper-container-rtl .swiper-button-next { + background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20viewBox%3D'0%200%2027%2044'%3E%3Cpath%20d%3D'M0%2C22L22%2C0l2.1%2C2.1L4.2%2C22l19.9%2C19.9L22%2C44L0%2C22L0%2C22L0%2C22z'%20fill%3D'%23007aff'%2F%3E%3C%2Fsvg%3E"); + left: 10px; + right: auto; } + + .swiper-button-prev.swiper-button-black, + .swiper-container-rtl .swiper-button-next.swiper-button-black { + background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20viewBox%3D'0%200%2027%2044'%3E%3Cpath%20d%3D'M0%2C22L22%2C0l2.1%2C2.1L4.2%2C22l19.9%2C19.9L22%2C44L0%2C22L0%2C22L0%2C22z'%20fill%3D'%23000000'%2F%3E%3C%2Fsvg%3E"); } + + .swiper-button-prev.swiper-button-white, + .swiper-container-rtl .swiper-button-next.swiper-button-white { + background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20viewBox%3D'0%200%2027%2044'%3E%3Cpath%20d%3D'M0%2C22L22%2C0l2.1%2C2.1L4.2%2C22l19.9%2C19.9L22%2C44L0%2C22L0%2C22L0%2C22z'%20fill%3D'%23ffffff'%2F%3E%3C%2Fsvg%3E"); } + + .swiper-button-next, + .swiper-container-rtl .swiper-button-prev { + background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20viewBox%3D'0%200%2027%2044'%3E%3Cpath%20d%3D'M27%2C22L27%2C22L5%2C44l-2.1-2.1L22.8%2C22L2.9%2C2.1L5%2C0L27%2C22L27%2C22z'%20fill%3D'%23007aff'%2F%3E%3C%2Fsvg%3E"); + right: 10px; + left: auto; } + + .swiper-button-next.swiper-button-black, + .swiper-container-rtl .swiper-button-prev.swiper-button-black { + background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20viewBox%3D'0%200%2027%2044'%3E%3Cpath%20d%3D'M27%2C22L27%2C22L5%2C44l-2.1-2.1L22.8%2C22L2.9%2C2.1L5%2C0L27%2C22L27%2C22z'%20fill%3D'%23000000'%2F%3E%3C%2Fsvg%3E"); } + + .swiper-button-next.swiper-button-white, + .swiper-container-rtl .swiper-button-prev.swiper-button-white { + background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20viewBox%3D'0%200%2027%2044'%3E%3Cpath%20d%3D'M27%2C22L27%2C22L5%2C44l-2.1-2.1L22.8%2C22L2.9%2C2.1L5%2C0L27%2C22L27%2C22z'%20fill%3D'%23ffffff'%2F%3E%3C%2Fsvg%3E"); } + + /* Pagination Styles */ + .swiper-pagination { + position: absolute; + text-align: center; + -webkit-transition: 300ms; + -moz-transition: 300ms; + -o-transition: 300ms; + transition: 300ms; + -webkit-transform: translate3d(0, 0, 0); + -ms-transform: translate3d(0, 0, 0); + -o-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); + z-index: 10; } + + .swiper-pagination.swiper-pagination-hidden { + opacity: 0; } + + .swiper-pagination-bullet { + width: 8px; + height: 8px; + display: inline-block; + border-radius: 100%; + background: #000; + opacity: 0.2; } + + button.swiper-pagination-bullet { + border: none; + margin: 0; + padding: 0; + box-shadow: none; + -moz-appearance: none; + -ms-appearance: none; + -webkit-appearance: none; + appearance: none; } + + .swiper-pagination-clickable .swiper-pagination-bullet { + cursor: pointer; } + + .swiper-pagination-white .swiper-pagination-bullet { + background: #fff; } + + .swiper-pagination-bullet-active { + opacity: 1; } + + .swiper-pagination-white .swiper-pagination-bullet-active { + background: #fff; } + + .swiper-pagination-black .swiper-pagination-bullet-active { + background: #000; } + + .swiper-container-vertical > .swiper-pagination { + right: 10px; + top: 50%; + -webkit-transform: translate3d(0px, -50%, 0); + -moz-transform: translate3d(0px, -50%, 0); + -o-transform: translate(0px, -50%); + -ms-transform: translate3d(0px, -50%, 0); + transform: translate3d(0px, -50%, 0); } + + .swiper-container-vertical > .swiper-pagination .swiper-pagination-bullet { + margin: 5px 0; + display: block; } + + .swiper-container-horizontal > .swiper-pagination { + bottom: 10px; + left: 0; + width: 100%; } + + .swiper-container-horizontal > .swiper-pagination .swiper-pagination-bullet { + margin: 0 5px; } + + /* 3D Container */ + .swiper-container-3d { + -webkit-perspective: 1200px; + -moz-perspective: 1200px; + -o-perspective: 1200px; + perspective: 1200px; } + + .swiper-container-3d .swiper-wrapper, + .swiper-container-3d .swiper-slide, + .swiper-container-3d .swiper-slide-shadow-left, + .swiper-container-3d .swiper-slide-shadow-right, + .swiper-container-3d .swiper-slide-shadow-top, + .swiper-container-3d .swiper-slide-shadow-bottom, + .swiper-container-3d .swiper-cube-shadow { + -webkit-transform-style: preserve-3d; + -moz-transform-style: preserve-3d; + -ms-transform-style: preserve-3d; + transform-style: preserve-3d; } + + .swiper-container-3d .swiper-slide-shadow-left, + .swiper-container-3d .swiper-slide-shadow-right, + .swiper-container-3d .swiper-slide-shadow-top, + .swiper-container-3d .swiper-slide-shadow-bottom { + position: absolute; + left: 0; + top: 0; + width: 100%; + height: 100%; + pointer-events: none; + z-index: 10; } + + .swiper-container-3d .swiper-slide-shadow-left { + background-image: -webkit-gradient(linear, left top, right top, from(rgba(0, 0, 0, 0.5)), to(transparent)); + /* Safari 4+, Chrome */ + background-image: -webkit-linear-gradient(right, rgba(0, 0, 0, 0.5), transparent); + /* Chrome 10+, Safari 5.1+, iOS 5+ */ + background-image: -moz-linear-gradient(right, rgba(0, 0, 0, 0.5), transparent); + /* Firefox 3.6-15 */ + background-image: -o-linear-gradient(right, rgba(0, 0, 0, 0.5), transparent); + /* Opera 11.10-12.00 */ + background-image: linear-gradient(to left, rgba(0, 0, 0, 0.5), transparent); + /* Firefox 16+, IE10, Opera 12.50+ */ } + + .swiper-container-3d .swiper-slide-shadow-right { + background-image: -webkit-gradient(linear, right top, left top, from(rgba(0, 0, 0, 0.5)), to(transparent)); + /* Safari 4+, Chrome */ + background-image: -webkit-linear-gradient(left, rgba(0, 0, 0, 0.5), transparent); + /* Chrome 10+, Safari 5.1+, iOS 5+ */ + background-image: -moz-linear-gradient(left, rgba(0, 0, 0, 0.5), transparent); + /* Firefox 3.6-15 */ + background-image: -o-linear-gradient(left, rgba(0, 0, 0, 0.5), transparent); + /* Opera 11.10-12.00 */ + background-image: linear-gradient(to right, rgba(0, 0, 0, 0.5), transparent); + /* Firefox 16+, IE10, Opera 12.50+ */ } + + .swiper-container-3d .swiper-slide-shadow-top { + background-image: -webkit-gradient(linear, left top, left bottom, from(rgba(0, 0, 0, 0.5)), to(transparent)); + /* Safari 4+, Chrome */ + background-image: -webkit-linear-gradient(bottom, rgba(0, 0, 0, 0.5), transparent); + /* Chrome 10+, Safari 5.1+, iOS 5+ */ + background-image: -moz-linear-gradient(bottom, rgba(0, 0, 0, 0.5), transparent); + /* Firefox 3.6-15 */ + background-image: -o-linear-gradient(bottom, rgba(0, 0, 0, 0.5), transparent); + /* Opera 11.10-12.00 */ + background-image: linear-gradient(to top, rgba(0, 0, 0, 0.5), transparent); + /* Firefox 16+, IE10, Opera 12.50+ */ } + + .swiper-container-3d .swiper-slide-shadow-bottom { + background-image: -webkit-gradient(linear, left bottom, left top, from(rgba(0, 0, 0, 0.5)), to(transparent)); + /* Safari 4+, Chrome */ + background-image: -webkit-linear-gradient(top, rgba(0, 0, 0, 0.5), transparent); + /* Chrome 10+, Safari 5.1+, iOS 5+ */ + background-image: -moz-linear-gradient(top, rgba(0, 0, 0, 0.5), transparent); + /* Firefox 3.6-15 */ + background-image: -o-linear-gradient(top, rgba(0, 0, 0, 0.5), transparent); + /* Opera 11.10-12.00 */ + background-image: linear-gradient(to bottom, rgba(0, 0, 0, 0.5), transparent); + /* Firefox 16+, IE10, Opera 12.50+ */ } + + /* Coverflow */ + .swiper-container-coverflow .swiper-wrapper { + /* Windows 8 IE 10 fix */ + -ms-perspective: 1200px; } + + /* Fade */ + .swiper-container-fade.swiper-container-free-mode .swiper-slide { + -webkit-transition-timing-function: ease-out; + -moz-transition-timing-function: ease-out; + -ms-transition-timing-function: ease-out; + -o-transition-timing-function: ease-out; + transition-timing-function: ease-out; } + + .swiper-container-fade .swiper-slide { + pointer-events: none; } + + .swiper-container-fade .swiper-slide .swiper-slide { + pointer-events: none; } + + .swiper-container-fade .swiper-slide-active, + .swiper-container-fade .swiper-slide-active .swiper-slide-active { + pointer-events: auto; } + + /* Cube */ + .swiper-container-cube { + overflow: visible; } + + .swiper-container-cube .swiper-slide { + pointer-events: none; + visibility: hidden; + -webkit-transform-origin: 0 0; + -moz-transform-origin: 0 0; + -ms-transform-origin: 0 0; + transform-origin: 0 0; + -webkit-backface-visibility: hidden; + -moz-backface-visibility: hidden; + -ms-backface-visibility: hidden; + backface-visibility: hidden; + width: 100%; + height: 100%; + z-index: 1; } + + .swiper-container-cube.swiper-container-rtl .swiper-slide { + -webkit-transform-origin: 100% 0; + -moz-transform-origin: 100% 0; + -ms-transform-origin: 100% 0; + transform-origin: 100% 0; } + + .swiper-container-cube .swiper-slide-active, + .swiper-container-cube .swiper-slide-next, + .swiper-container-cube .swiper-slide-prev, + .swiper-container-cube .swiper-slide-next + .swiper-slide { + pointer-events: auto; + visibility: visible; } + + .swiper-container-cube .swiper-slide-shadow-top, + .swiper-container-cube .swiper-slide-shadow-bottom, + .swiper-container-cube .swiper-slide-shadow-left, + .swiper-container-cube .swiper-slide-shadow-right { + z-index: 0; + -webkit-backface-visibility: hidden; + -moz-backface-visibility: hidden; + -ms-backface-visibility: hidden; + backface-visibility: hidden; } + + .swiper-container-cube .swiper-cube-shadow { + position: absolute; + left: 0; + bottom: 0px; + width: 100%; + height: 100%; + background: #000; + opacity: 0.6; + -webkit-filter: blur(50px); + filter: blur(50px); + z-index: 0; } + + /* Scrollbar */ + .swiper-scrollbar { + border-radius: 10px; + position: relative; + -ms-touch-action: none; + background: rgba(0, 0, 0, 0.1); } + + .swiper-container-horizontal > .swiper-scrollbar { + position: absolute; + left: 1%; + bottom: 3px; + z-index: 50; + height: 5px; + width: 98%; } + + .swiper-container-vertical > .swiper-scrollbar { + position: absolute; + right: 3px; + top: 1%; + z-index: 50; + width: 5px; + height: 98%; } + + .swiper-scrollbar-drag { + height: 100%; + width: 100%; + position: relative; + background: rgba(0, 0, 0, 0.5); + border-radius: 10px; + left: 0; + top: 0; } + + .swiper-scrollbar-cursor-drag { + cursor: move; } + + /* Preloader */ + .swiper-lazy-preloader { + width: 42px; + height: 42px; + position: absolute; + left: 50%; + top: 50%; + margin-left: -21px; + margin-top: -21px; + z-index: 10; + -webkit-transform-origin: 50%; + -moz-transform-origin: 50%; + transform-origin: 50%; + -webkit-animation: swiper-preloader-spin 1s steps(12, end) infinite; + -moz-animation: swiper-preloader-spin 1s steps(12, end) infinite; + animation: swiper-preloader-spin 1s steps(12, end) infinite; } + + .swiper-lazy-preloader:after { + display: block; + content: ""; + width: 100%; + height: 100%; + background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20viewBox%3D'0%200%20120%20120'%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20xmlns%3Axlink%3D'http%3A%2F%2Fwww.w3.org%2F1999%2Fxlink'%3E%3Cdefs%3E%3Cline%20id%3D'l'%20x1%3D'60'%20x2%3D'60'%20y1%3D'7'%20y2%3D'27'%20stroke%3D'%236c6c6c'%20stroke-width%3D'11'%20stroke-linecap%3D'round'%2F%3E%3C%2Fdefs%3E%3Cg%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(30%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(60%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(90%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(120%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(150%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.37'%20transform%3D'rotate(180%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.46'%20transform%3D'rotate(210%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.56'%20transform%3D'rotate(240%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.66'%20transform%3D'rotate(270%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.75'%20transform%3D'rotate(300%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.85'%20transform%3D'rotate(330%2060%2C60)'%2F%3E%3C%2Fg%3E%3C%2Fsvg%3E"); + background-position: 50%; + -webkit-background-size: 100%; + background-size: 100%; + background-repeat: no-repeat; } + + .swiper-lazy-preloader-white:after { + background-image: url("data:image/svg+xml;charset=utf-8,%3Csvg%20viewBox%3D'0%200%20120%20120'%20xmlns%3D'http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg'%20xmlns%3Axlink%3D'http%3A%2F%2Fwww.w3.org%2F1999%2Fxlink'%3E%3Cdefs%3E%3Cline%20id%3D'l'%20x1%3D'60'%20x2%3D'60'%20y1%3D'7'%20y2%3D'27'%20stroke%3D'%23fff'%20stroke-width%3D'11'%20stroke-linecap%3D'round'%2F%3E%3C%2Fdefs%3E%3Cg%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(30%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(60%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(90%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(120%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.27'%20transform%3D'rotate(150%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.37'%20transform%3D'rotate(180%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.46'%20transform%3D'rotate(210%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.56'%20transform%3D'rotate(240%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.66'%20transform%3D'rotate(270%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.75'%20transform%3D'rotate(300%2060%2C60)'%2F%3E%3Cuse%20xlink%3Ahref%3D'%23l'%20opacity%3D'.85'%20transform%3D'rotate(330%2060%2C60)'%2F%3E%3C%2Fg%3E%3C%2Fsvg%3E"); } + + @-webkit-keyframes swiper-preloader-spin { + 100% { + -webkit-transform: rotate(360deg); } } + + @keyframes swiper-preloader-spin { + 100% { + transform: rotate(360deg); } } + + ion-slides { + width: 100%; + height: 100%; + display: block; } + + .slide-zoom { + display: block; + width: 100%; + text-align: center; } + + .swiper-container { + width: 100%; + height: 100%; + padding: 0; + overflow: hidden; } + + .swiper-wrapper { + position: absolute; + left: 0; + top: 0; + width: 100%; + height: 100%; + padding: 0; } + + .swiper-slide { + width: 100%; + height: 100%; + box-sizing: border-box; + /* Center slide text vertically */ } + .swiper-slide img { + width: auto; + height: auto; + max-width: 100%; + max-height: 100%; } + + .scroll-refresher { + position: absolute; + top: -60px; + right: 0; + left: 0; + overflow: hidden; + margin: auto; + height: 60px; } + .scroll-refresher .ionic-refresher-content { + position: absolute; + bottom: 15px; + left: 0; + width: 100%; + color: #666666; + text-align: center; + font-size: 30px; } + .scroll-refresher .ionic-refresher-content .text-refreshing, + .scroll-refresher .ionic-refresher-content .text-pulling { + font-size: 16px; + line-height: 16px; } + .scroll-refresher .ionic-refresher-content.ionic-refresher-with-text { + bottom: 10px; } + .scroll-refresher .icon-refreshing, + .scroll-refresher .icon-pulling { + width: 100%; + -webkit-backface-visibility: hidden; + backface-visibility: hidden; + -webkit-transform-style: preserve-3d; + transform-style: preserve-3d; } + .scroll-refresher .icon-pulling { + -webkit-animation-name: refresh-spin-back; + animation-name: refresh-spin-back; + -webkit-animation-duration: 200ms; + animation-duration: 200ms; + -webkit-animation-timing-function: linear; + animation-timing-function: linear; + -webkit-animation-fill-mode: none; + animation-fill-mode: none; + -webkit-transform: translate3d(0, 0, 0) rotate(0deg); + transform: translate3d(0, 0, 0) rotate(0deg); } + .scroll-refresher .icon-refreshing, + .scroll-refresher .text-refreshing { + display: none; } + .scroll-refresher .icon-refreshing { + -webkit-animation-duration: 1.5s; + animation-duration: 1.5s; } + .scroll-refresher.active .icon-pulling:not(.pulling-rotation-disabled) { + -webkit-animation-name: refresh-spin; + animation-name: refresh-spin; + -webkit-transform: translate3d(0, 0, 0) rotate(-180deg); + transform: translate3d(0, 0, 0) rotate(-180deg); } + .scroll-refresher.active.refreshing { + -webkit-transition: -webkit-transform 0.2s; + transition: -webkit-transform 0.2s; + -webkit-transition: transform 0.2s; + transition: transform 0.2s; + -webkit-transform: scale(1, 1); + transform: scale(1, 1); } + .scroll-refresher.active.refreshing .icon-pulling, + .scroll-refresher.active.refreshing .text-pulling { + display: none; } + .scroll-refresher.active.refreshing .icon-refreshing, + .scroll-refresher.active.refreshing .text-refreshing { + display: block; } + .scroll-refresher.active.refreshing.refreshing-tail { + -webkit-transform: scale(0, 0); + transform: scale(0, 0); } + + .overflow-scroll > .scroll { + -webkit-overflow-scrolling: touch; + width: 100%; } + .overflow-scroll > .scroll.overscroll { + position: fixed; + right: 0; + left: 0; } + + .overflow-scroll.padding > .scroll.overscroll, .overflow-scroll.item.large-button-bar > .scroll.overscroll { + padding: 10px; } + + @-webkit-keyframes refresh-spin { + 0% { + -webkit-transform: translate3d(0, 0, 0) rotate(0); } + 100% { + -webkit-transform: translate3d(0, 0, 0) rotate(180deg); } } + + @keyframes refresh-spin { + 0% { + transform: translate3d(0, 0, 0) rotate(0); } + 100% { + transform: translate3d(0, 0, 0) rotate(180deg); } } + + @-webkit-keyframes refresh-spin-back { + 0% { + -webkit-transform: translate3d(0, 0, 0) rotate(180deg); } + 100% { + -webkit-transform: translate3d(0, 0, 0) rotate(0); } } + + @keyframes refresh-spin-back { + 0% { + transform: translate3d(0, 0, 0) rotate(180deg); } + 100% { + transform: translate3d(0, 0, 0) rotate(0); } } + + /** + * Spinners + * -------------------------------------------------- + */ + .spinner { + stroke: #444; + fill: #444; } + .spinner svg { + width: 28px; + height: 28px; } + .spinner.spinner-light { + stroke: #fff; + fill: #fff; } + .spinner.spinner-stable { + stroke: #f8f8f8; + fill: #f8f8f8; } + .spinner.spinner-positive { + stroke: #387ef5; + fill: #387ef5; } + .spinner.spinner-calm { + stroke: #11c1f3; + fill: #11c1f3; } + .spinner.spinner-balanced { + stroke: #33cd5f; + fill: #33cd5f; } + .spinner.spinner-assertive { + stroke: #ef473a; + fill: #ef473a; } + .spinner.spinner-energized { + stroke: #ffc900; + fill: #ffc900; } + .spinner.spinner-royal { + stroke: #886aea; + fill: #886aea; } + .spinner.spinner-dark { + stroke: #444; + fill: #444; } + + .spinner-android { + stroke: #4b8bf4; } + + .spinner-ios, + .spinner-ios-small { + stroke: #69717d; } + + .spinner-spiral .stop1 { + stop-color: #fff; + stop-opacity: 0; } + + .spinner-spiral.spinner-light .stop1 { + stop-color: #444; } + + .spinner-spiral.spinner-light .stop2 { + stop-color: #fff; } + + .spinner-spiral.spinner-stable .stop2 { + stop-color: #f8f8f8; } + + .spinner-spiral.spinner-positive .stop2 { + stop-color: #387ef5; } + + .spinner-spiral.spinner-calm .stop2 { + stop-color: #11c1f3; } + + .spinner-spiral.spinner-balanced .stop2 { + stop-color: #33cd5f; } + + .spinner-spiral.spinner-assertive .stop2 { + stop-color: #ef473a; } + + .spinner-spiral.spinner-energized .stop2 { + stop-color: #ffc900; } + + .spinner-spiral.spinner-royal .stop2 { + stop-color: #886aea; } + + .spinner-spiral.spinner-dark .stop2 { + stop-color: #444; } + + /** + * Forms + * -------------------------------------------------- + */ + form { + margin: 0 0 1.42857; } + + legend { + display: block; + margin-bottom: 1.42857; + padding: 0; + width: 100%; + border: 1px solid #ddd; + color: #444; + font-size: 21px; + line-height: 2.85714; } + legend small { + color: #f8f8f8; + font-size: 1.07143; } + + label, + input, + button, + select, + textarea { + font-weight: normal; + font-size: 14px; + line-height: 1.42857; } + + input, + button, + select, + textarea { + font-family: "-apple-system", "Helvetica Neue", "Roboto", "Segoe UI", sans-serif; } + + .item-input { + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + -webkit-box-align: center; + -ms-flex-align: center; + -webkit-align-items: center; + -moz-align-items: center; + align-items: center; + position: relative; + overflow: hidden; + padding: 6px 0 5px 16px; } + .item-input input { + -webkit-border-radius: 0; + border-radius: 0; + -webkit-box-flex: 1; + -webkit-flex: 1 220px; + -moz-box-flex: 1; + -moz-flex: 1 220px; + -ms-flex: 1 220px; + flex: 1 220px; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; + margin: 0; + padding-right: 24px; + background-color: transparent; } + .item-input .button .icon, .item-input .button .icon-help, .item-input .button .icon-alert, .item-input .button #menu .footer .icon-help, #menu .footer .item-input .button .icon-help { + -webkit-box-flex: 0; + -webkit-flex: 0 0 24px; + -moz-box-flex: 0; + -moz-flex: 0 0 24px; + -ms-flex: 0 0 24px; + flex: 0 0 24px; + position: static; + display: inline-block; + height: auto; + text-align: center; + font-size: 16px; } + .item-input .button-bar { + -webkit-border-radius: 0; + border-radius: 0; + -webkit-box-flex: 1; + -webkit-flex: 1 0 220px; + -moz-box-flex: 1; + -moz-flex: 1 0 220px; + -ms-flex: 1 0 220px; + flex: 1 0 220px; + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; } + .item-input .icon, .item-input .icon-help, .item-input .icon-alert, .item-input #menu .footer .icon-help, #menu .footer .item-input .icon-help { + min-width: 14px; } + + .platform-windowsphone .item-input input { + flex-shrink: 1; } + + .item-input-inset { + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + -webkit-box-align: center; + -ms-flex-align: center; + -webkit-align-items: center; + -moz-align-items: center; + align-items: center; + position: relative; + overflow: hidden; + padding: 10.66667px; } + + .item-input-wrapper { + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + -webkit-box-flex: 1; + -webkit-flex: 1 0; + -moz-box-flex: 1; + -moz-flex: 1 0; + -ms-flex: 1 0; + flex: 1 0; + -webkit-box-align: center; + -ms-flex-align: center; + -webkit-align-items: center; + -moz-align-items: center; + align-items: center; + -webkit-border-radius: 4px; + border-radius: 4px; + padding-right: 8px; + padding-left: 8px; + background: #eee; } + + .item-input-inset .item-input-wrapper input { + padding-left: 4px; + height: 29px; + background: transparent; + line-height: 18px; } + + .item-input-wrapper ~ .button { + margin-left: 10.66667px; } + + .input-label { + display: table; + padding: 7px 10px 7px 0px; + max-width: 200px; + width: 35%; + color: #444; + font-size: 16px; } + + .placeholder-icon { + color: #aaa; } + .placeholder-icon:first-child { + padding-right: 6px; } + .placeholder-icon:last-child { + padding-left: 6px; } + + .item-stacked-label { + display: block; + background-color: transparent; + box-shadow: none; } + .item-stacked-label .input-label, .item-stacked-label .icon, .item-stacked-label .icon-help, .item-stacked-label .icon-alert, .item-stacked-label #menu .footer .icon-help, #menu .footer .item-stacked-label .icon-help { + display: inline-block; + padding: 4px 0 0 0px; + vertical-align: middle; } + + .item-stacked-label input, + .item-stacked-label textarea { + -webkit-border-radius: 2px; + border-radius: 2px; + padding: 4px 8px 3px 0; + border: none; + background-color: #fff; } + + .item-stacked-label input { + overflow: hidden; + height: 46px; } + + .item-select.item-stacked-label select { + position: relative; + padding: 0px; + max-width: 90%; + direction: ltr; + white-space: pre-wrap; + margin: -3px; } + + .item-floating-label { + display: block; + background-color: transparent; + box-shadow: none; } + .item-floating-label .input-label { + position: relative; + padding: 5px 0 0 0; + opacity: 0; + top: 10px; + -webkit-transition: opacity 0.15s ease-in, top 0.2s linear; + transition: opacity 0.15s ease-in, top 0.2s linear; } + .item-floating-label .input-label.has-input { + opacity: 1; + top: 0; + -webkit-transition: opacity 0.15s ease-in, top 0.2s linear; + transition: opacity 0.15s ease-in, top 0.2s linear; } + + textarea, + input[type="text"], + input[type="password"], + input[type="datetime"], + input[type="datetime-local"], + input[type="date"], + input[type="month"], + input[type="time"], + input[type="week"], + input[type="number"], + input[type="email"], + input[type="url"], + input[type="search"], + input[type="tel"], + input[type="color"] { + display: block; + padding-top: 2px; + padding-left: 0; + height: 34px; + color: #111; + vertical-align: middle; + font-size: 14px; + line-height: 16px; } + + .platform-ios input[type="datetime-local"], + .platform-ios input[type="date"], + .platform-ios input[type="month"], + .platform-ios input[type="time"], + .platform-ios input[type="week"], + .platform-android input[type="datetime-local"], + .platform-android input[type="date"], + .platform-android input[type="month"], + .platform-android input[type="time"], + .platform-android input[type="week"] { + padding-top: 8px; } + + .item-input input, + .item-input textarea { + width: 100%; } + + textarea { + padding-left: 0; } + textarea::-moz-placeholder { + color: #aaaaaa; } + textarea:-ms-input-placeholder { + color: #aaaaaa; } + textarea::-webkit-input-placeholder { + color: #aaaaaa; + text-indent: -3px; } + + textarea { + height: auto; } + + textarea, + input[type="text"], + input[type="password"], + input[type="datetime"], + input[type="datetime-local"], + input[type="date"], + input[type="month"], + input[type="time"], + input[type="week"], + input[type="number"], + input[type="email"], + input[type="url"], + input[type="search"], + input[type="tel"], + input[type="color"] { + border: 0; } + + input[type="radio"], + input[type="checkbox"] { + margin: 0; + line-height: normal; } + + .item-input input[type="file"], + .item-input input[type="image"], + .item-input input[type="submit"], + .item-input input[type="reset"], + .item-input input[type="button"], + .item-input input[type="radio"], + .item-input input[type="checkbox"] { + width: auto; } + + input[type="file"] { + line-height: 34px; } + + .previous-input-focus, + .cloned-text-input + input, + .cloned-text-input + textarea { + position: absolute !important; + left: -9999px; + width: 200px; } + + input::-moz-placeholder, + textarea::-moz-placeholder { + color: #aaaaaa; } + + input:-ms-input-placeholder, + textarea:-ms-input-placeholder { + color: #aaaaaa; } + + input::-webkit-input-placeholder, + textarea::-webkit-input-placeholder { + color: #aaaaaa; + text-indent: 0; } + + input[disabled], + select[disabled], + textarea[disabled], + input[readonly]:not(.cloned-text-input), + textarea[readonly]:not(.cloned-text-input), + select[readonly] { + background-color: #f8f8f8; + cursor: not-allowed; } + + input[type="radio"][disabled], + input[type="checkbox"][disabled], + input[type="radio"][readonly], + input[type="checkbox"][readonly] { + background-color: transparent; } + + /** + * Checkbox + * -------------------------------------------------- + */ + .checkbox { + position: relative; + display: inline-block; + padding: 7px 7px; + cursor: pointer; } + .checkbox input:before, + .checkbox .checkbox-icon:before { + border-color: #ddd; } + .checkbox input:checked:before, + .checkbox input:checked + .checkbox-icon:before { + background: #387ef5; + border-color: #387ef5; } + + .checkbox-light input:before, + .checkbox-light .checkbox-icon:before { + border-color: #ddd; } + + .checkbox-light input:checked:before, + .checkbox-light input:checked + .checkbox-icon:before { + background: #ddd; + border-color: #ddd; } + + .checkbox-stable input:before, + .checkbox-stable .checkbox-icon:before { + border-color: #b2b2b2; } + + .checkbox-stable input:checked:before, + .checkbox-stable input:checked + .checkbox-icon:before { + background: #b2b2b2; + border-color: #b2b2b2; } + + .checkbox-positive input:before, + .checkbox-positive .checkbox-icon:before { + border-color: #387ef5; } + + .checkbox-positive input:checked:before, + .checkbox-positive input:checked + .checkbox-icon:before { + background: #387ef5; + border-color: #387ef5; } + + .checkbox-calm input:before, + .checkbox-calm .checkbox-icon:before { + border-color: #11c1f3; } + + .checkbox-calm input:checked:before, + .checkbox-calm input:checked + .checkbox-icon:before { + background: #11c1f3; + border-color: #11c1f3; } + + .checkbox-assertive input:before, + .checkbox-assertive .checkbox-icon:before { + border-color: #ef473a; } + + .checkbox-assertive input:checked:before, + .checkbox-assertive input:checked + .checkbox-icon:before { + background: #ef473a; + border-color: #ef473a; } + + .checkbox-balanced input:before, + .checkbox-balanced .checkbox-icon:before { + border-color: #33cd5f; } + + .checkbox-balanced input:checked:before, + .checkbox-balanced input:checked + .checkbox-icon:before { + background: #33cd5f; + border-color: #33cd5f; } + + .checkbox-energized input:before, + .checkbox-energized .checkbox-icon:before { + border-color: #ffc900; } + + .checkbox-energized input:checked:before, + .checkbox-energized input:checked + .checkbox-icon:before { + background: #ffc900; + border-color: #ffc900; } + + .checkbox-royal input:before, + .checkbox-royal .checkbox-icon:before { + border-color: #886aea; } + + .checkbox-royal input:checked:before, + .checkbox-royal input:checked + .checkbox-icon:before { + background: #886aea; + border-color: #886aea; } + + .checkbox-dark input:before, + .checkbox-dark .checkbox-icon:before { + border-color: #444; } + + .checkbox-dark input:checked:before, + .checkbox-dark input:checked + .checkbox-icon:before { + background: #444; + border-color: #444; } + + .checkbox input:disabled:before, + .checkbox input:disabled + .checkbox-icon:before { + border-color: #ddd; } + + .checkbox input:disabled:checked:before, + .checkbox input:disabled:checked + .checkbox-icon:before { + background: #ddd; } + + .checkbox.checkbox-input-hidden input { + display: none !important; } + + .checkbox input, + .checkbox-icon { + position: relative; + width: 28px; + height: 28px; + display: block; + border: 0; + background: transparent; + cursor: pointer; + -webkit-appearance: none; } + .checkbox input:before, + .checkbox-icon:before { + display: table; + width: 100%; + height: 100%; + border-width: 1px; + border-style: solid; + border-radius: 28px; + background: #fff; + content: ' '; + -webkit-transition: background-color 20ms ease-in-out; + transition: background-color 20ms ease-in-out; } + + .checkbox input:checked:before, + input:checked + .checkbox-icon:before { + border-width: 2px; } + + .checkbox input:after, + .checkbox-icon:after { + -webkit-transition: opacity 0.05s ease-in-out; + transition: opacity 0.05s ease-in-out; + -webkit-transform: rotate(-45deg); + transform: rotate(-45deg); + position: absolute; + top: 33%; + left: 25%; + display: table; + width: 14px; + height: 6px; + border: 1px solid #fff; + border-top: 0; + border-right: 0; + content: ' '; + opacity: 0; } + + .platform-android .checkbox-platform input:before, + .platform-android .checkbox-platform .checkbox-icon:before, + .checkbox-square input:before, + .checkbox-square .checkbox-icon:before { + border-radius: 2px; + width: 72%; + height: 72%; + margin-top: 14%; + margin-left: 14%; + border-width: 2px; } + + .platform-android .checkbox-platform input:after, + .platform-android .checkbox-platform .checkbox-icon:after, + .checkbox-square input:after, + .checkbox-square .checkbox-icon:after { + border-width: 2px; + top: 19%; + left: 25%; + width: 13px; + height: 7px; } + + .platform-android .item-checkbox-right .checkbox-square .checkbox-icon::after { + top: 31%; } + + .grade-c .checkbox input:after, + .grade-c .checkbox-icon:after { + -webkit-transform: rotate(0); + transform: rotate(0); + top: 3px; + left: 4px; + border: none; + color: #fff; + content: '\2713'; + font-weight: bold; + font-size: 20px; } + + .checkbox input:checked:after, + input:checked + .checkbox-icon:after { + opacity: 1; } + + .item-checkbox { + padding-left: 60px; } + .item-checkbox.active { + box-shadow: none; } + + .item-checkbox .checkbox { + position: absolute; + top: 50%; + right: 8px; + left: 8px; + z-index: 3; + margin-top: -21px; } + + .item-checkbox.item-checkbox-right { + padding-right: 60px; + padding-left: 16px; } + + .item-checkbox-right .checkbox input, + .item-checkbox-right .checkbox-icon { + float: right; } + + /** + * Toggle + * -------------------------------------------------- + */ + .item-toggle { + pointer-events: none; } + + .toggle { + position: relative; + display: inline-block; + pointer-events: auto; + margin: -5px; + padding: 5px; } + .toggle input:checked + .track { + border-color: #4cd964; + background-color: #4cd964; } + .toggle.dragging .handle { + background-color: #f2f2f2 !important; } + + .toggle.toggle-light input:checked + .track { + border-color: #ddd; + background-color: #ddd; } + + .toggle.toggle-stable input:checked + .track { + border-color: #b2b2b2; + background-color: #b2b2b2; } + + .toggle.toggle-positive input:checked + .track { + border-color: #387ef5; + background-color: #387ef5; } + + .toggle.toggle-calm input:checked + .track { + border-color: #11c1f3; + background-color: #11c1f3; } + + .toggle.toggle-assertive input:checked + .track { + border-color: #ef473a; + background-color: #ef473a; } + + .toggle.toggle-balanced input:checked + .track { + border-color: #33cd5f; + background-color: #33cd5f; } + + .toggle.toggle-energized input:checked + .track { + border-color: #ffc900; + background-color: #ffc900; } + + .toggle.toggle-royal input:checked + .track { + border-color: #886aea; + background-color: #886aea; } + + .toggle.toggle-dark input:checked + .track { + border-color: #444; + background-color: #444; } + + .toggle input { + display: none; } + + /* the track appearance when the toggle is "off" */ + .toggle .track { + -webkit-transition-timing-function: ease-in-out; + transition-timing-function: ease-in-out; + -webkit-transition-duration: 0.3s; + transition-duration: 0.3s; + -webkit-transition-property: background-color, border; + transition-property: background-color, border; + display: inline-block; + box-sizing: border-box; + width: 51px; + height: 31px; + border: solid 2px #e6e6e6; + border-radius: 20px; + background-color: #fff; + content: ' '; + cursor: pointer; + pointer-events: none; } + + /* Fix to avoid background color bleeding */ + /* (occurred on (at least) Android 4.2, Asus MeMO Pad HD7 ME173X) */ + .platform-android4_2 .toggle .track { + -webkit-background-clip: padding-box; } + + /* the handle (circle) thats inside the toggle's track area */ + /* also the handle's appearance when it is "off" */ + .toggle .handle { + -webkit-transition: 0.3s cubic-bezier(0, 1.1, 1, 1.1); + transition: 0.3s cubic-bezier(0, 1.1, 1, 1.1); + -webkit-transition-property: background-color, transform; + transition-property: background-color, transform; + position: absolute; + display: block; + width: 27px; + height: 27px; + border-radius: 27px; + background-color: #fff; + top: 7px; + left: 7px; + box-shadow: 0 2px 7px rgba(0, 0, 0, 0.35), 0 1px 1px rgba(0, 0, 0, 0.15); } + .toggle .handle:before { + position: absolute; + top: -4px; + left: -21.5px; + padding: 18.5px 34px; + content: " "; } + + .toggle input:checked + .track .handle { + -webkit-transform: translate3d(20px, 0, 0); + transform: translate3d(20px, 0, 0); + background-color: #fff; } + + .item-toggle.active { + box-shadow: none; } + + .item-toggle, + .item-toggle.item-complex .item-content { + padding-right: 99px; } + + .item-toggle.item-complex { + padding-right: 0; } + + .item-toggle .toggle { + position: absolute; + top: 10px; + right: 16px; + z-index: 3; } + + .toggle input:disabled + .track { + opacity: .6; } + + .toggle-small .track { + border: 0; + width: 34px; + height: 15px; + background: #9e9e9e; } + + .toggle-small input:checked + .track { + background: rgba(0, 150, 137, 0.5); } + + .toggle-small .handle { + top: 2px; + left: 4px; + width: 21px; + height: 21px; + box-shadow: 0 2px 5px rgba(0, 0, 0, 0.25); } + + .toggle-small input:checked + .track .handle { + -webkit-transform: translate3d(16px, 0, 0); + transform: translate3d(16px, 0, 0); + background: #009689; } + + .toggle-small.item-toggle .toggle { + top: 19px; } + + .toggle-small .toggle-light input:checked + .track { + background-color: rgba(221, 221, 221, 0.5); } + + .toggle-small .toggle-light input:checked + .track .handle { + background-color: #ddd; } + + .toggle-small .toggle-stable input:checked + .track { + background-color: rgba(178, 178, 178, 0.5); } + + .toggle-small .toggle-stable input:checked + .track .handle { + background-color: #b2b2b2; } + + .toggle-small .toggle-positive input:checked + .track { + background-color: rgba(56, 126, 245, 0.5); } + + .toggle-small .toggle-positive input:checked + .track .handle { + background-color: #387ef5; } + + .toggle-small .toggle-calm input:checked + .track { + background-color: rgba(17, 193, 243, 0.5); } + + .toggle-small .toggle-calm input:checked + .track .handle { + background-color: #11c1f3; } + + .toggle-small .toggle-assertive input:checked + .track { + background-color: rgba(239, 71, 58, 0.5); } + + .toggle-small .toggle-assertive input:checked + .track .handle { + background-color: #ef473a; } + + .toggle-small .toggle-balanced input:checked + .track { + background-color: rgba(51, 205, 95, 0.5); } + + .toggle-small .toggle-balanced input:checked + .track .handle { + background-color: #33cd5f; } + + .toggle-small .toggle-energized input:checked + .track { + background-color: rgba(255, 201, 0, 0.5); } + + .toggle-small .toggle-energized input:checked + .track .handle { + background-color: #ffc900; } + + .toggle-small .toggle-royal input:checked + .track { + background-color: rgba(136, 106, 234, 0.5); } + + .toggle-small .toggle-royal input:checked + .track .handle { + background-color: #886aea; } + + .toggle-small .toggle-dark input:checked + .track { + background-color: rgba(68, 68, 68, 0.5); } + + .toggle-small .toggle-dark input:checked + .track .handle { + background-color: #444; } + + /** + * Radio Button Inputs + * -------------------------------------------------- + */ + .item-radio { + padding: 0; } + .item-radio:hover { + cursor: pointer; } + + .item-radio .item-content { + /* give some room to the right for the checkmark icon */ + padding-right: 64px; } + + .item-radio .radio-icon { + /* checkmark icon will be hidden by default */ + position: absolute; + top: 0; + right: 0; + z-index: 3; + visibility: hidden; + padding: 14px; + height: 100%; + font-size: 24px; } + + .item-radio input { + /* hide any radio button inputs elements (the ugly circles) */ + position: absolute; + left: -9999px; } + .item-radio input:checked + .radio-content .item-content { + /* style the item content when its checked */ + background: #f7f7f7; } + .item-radio input:checked + .radio-content .radio-icon { + /* show the checkmark icon when its checked */ + visibility: visible; } + + /** + * Buttons + * -------------------------------------------------- + */ + .button { + border-color: transparent; + background-color: #f8f8f8; + color: #444; + position: relative; + display: inline-block; + margin: 0; + padding: 0 12px; + min-width: 52px; + min-height: 47px; + border-width: 1px; + border-style: solid; + border-radius: 4px; + vertical-align: top; + text-align: center; + text-overflow: ellipsis; + font-size: 16px; + line-height: 42px; + cursor: pointer; } + .button:hover { + color: #444; + text-decoration: none; } + .button.active, .button.activated { + border-color: #a2a2a2; + background-color: #e5e5e5; } + .button:after { + position: absolute; + top: -6px; + right: -6px; + bottom: -6px; + left: -6px; + content: ' '; } + .button .icon, .button .icon-help, .button .icon-alert, .button #menu .footer .icon-help, #menu .footer .button .icon-help { + vertical-align: top; + pointer-events: none; } + .button .icon:before, .button .icon-help:before, .button .icon-alert:before, .button #menu .footer .icon-help:before, #menu .footer .button .icon-help:before, .button.icon:before, .button.icon-help:before, .button.icon-alert:before, #menu .footer .button.icon-help:before, .button.icon-left:before, .button.icon-right:before { + display: inline-block; + padding: 0 0 1px 0; + vertical-align: inherit; + font-size: 24px; + line-height: 41px; + pointer-events: none; } + .button.icon-left:before { + float: left; + padding-right: .2em; + padding-left: 0; } + .button.icon-right:before { + float: right; + padding-right: 0; + padding-left: .2em; } + .button.button-block, .button.button-full { + margin-top: 10px; + margin-bottom: 10px; } + .button.button-light { + border-color: transparent; + background-color: #fff; + color: #444; } + .button.button-light:hover { + color: #444; + text-decoration: none; } + .button.button-light.active, .button.button-light.activated { + border-color: #a2a2a2; + background-color: #fafafa; } + .button.button-light.button-clear, .button.button-light.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #ddd; } + .button.button-light.button-icon { + border-color: transparent; + background: none; } + .button.button-light.button-outline { + border-color: #ddd; + background: transparent; + color: #ddd; } + .button.button-light.button-outline.active, .button.button-light.button-outline.activated { + background-color: #ddd; + box-shadow: none; + color: #fff; } + .button.button-stable { + border-color: transparent; + background-color: #f8f8f8; + color: #444; } + .button.button-stable:hover { + color: #444; + text-decoration: none; } + .button.button-stable.active, .button.button-stable.activated { + border-color: #a2a2a2; + background-color: #e5e5e5; } + .button.button-stable.button-clear, .button.button-stable.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #b2b2b2; } + .button.button-stable.button-icon { + border-color: transparent; + background: none; } + .button.button-stable.button-outline { + border-color: #b2b2b2; + background: transparent; + color: #b2b2b2; } + .button.button-stable.button-outline.active, .button.button-stable.button-outline.activated { + background-color: #b2b2b2; + box-shadow: none; + color: #fff; } + .button.button-positive, .button.button-text { + border-color: transparent; + background-color: #387ef5; + color: #fff; } + .button.button-positive:hover, .button.button-text:hover { + color: #fff; + text-decoration: none; } + .button.button-positive.active, .button.active.button-text, .button.button-positive.activated, .button.activated.button-text { + border-color: #a2a2a2; + background-color: #0c60ee; } + .button.button-positive.button-clear, .button.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #387ef5; } + .button.button-positive.button-icon, .button.button-icon.button-text { + border-color: transparent; + background: none; } + .button.button-positive.button-outline, .button.button-outline.button-text { + border-color: #387ef5; + background: transparent; + color: #387ef5; } + .button.button-positive.button-outline.active, .button.button-outline.active.button-text, .button.button-positive.button-outline.activated, .button.button-outline.activated.button-text { + background-color: #387ef5; + box-shadow: none; + color: #fff; } + .button.button-calm { + border-color: transparent; + background-color: #11c1f3; + color: #fff; } + .button.button-calm:hover { + color: #fff; + text-decoration: none; } + .button.button-calm.active, .button.button-calm.activated { + border-color: #a2a2a2; + background-color: #0a9dc7; } + .button.button-calm.button-clear, .button.button-calm.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #11c1f3; } + .button.button-calm.button-icon { + border-color: transparent; + background: none; } + .button.button-calm.button-outline { + border-color: #11c1f3; + background: transparent; + color: #11c1f3; } + .button.button-calm.button-outline.active, .button.button-calm.button-outline.activated { + background-color: #11c1f3; + box-shadow: none; + color: #fff; } + .button.button-assertive { + border-color: transparent; + background-color: #ef473a; + color: #fff; } + .button.button-assertive:hover { + color: #fff; + text-decoration: none; } + .button.button-assertive.active, .button.button-assertive.activated { + border-color: #a2a2a2; + background-color: #e42112; } + .button.button-assertive.button-clear, .button.button-assertive.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #ef473a; } + .button.button-assertive.button-icon { + border-color: transparent; + background: none; } + .button.button-assertive.button-outline { + border-color: #ef473a; + background: transparent; + color: #ef473a; } + .button.button-assertive.button-outline.active, .button.button-assertive.button-outline.activated { + background-color: #ef473a; + box-shadow: none; + color: #fff; } + .button.button-balanced { + border-color: transparent; + background-color: #33cd5f; + color: #fff; } + .button.button-balanced:hover { + color: #fff; + text-decoration: none; } + .button.button-balanced.active, .button.button-balanced.activated { + border-color: #a2a2a2; + background-color: #28a54c; } + .button.button-balanced.button-clear, .button.button-balanced.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #33cd5f; } + .button.button-balanced.button-icon { + border-color: transparent; + background: none; } + .button.button-balanced.button-outline { + border-color: #33cd5f; + background: transparent; + color: #33cd5f; } + .button.button-balanced.button-outline.active, .button.button-balanced.button-outline.activated { + background-color: #33cd5f; + box-shadow: none; + color: #fff; } + .button.button-energized { + border-color: transparent; + background-color: #ffc900; + color: #fff; } + .button.button-energized:hover { + color: #fff; + text-decoration: none; } + .button.button-energized.active, .button.button-energized.activated { + border-color: #a2a2a2; + background-color: #e6b500; } + .button.button-energized.button-clear, .button.button-energized.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #ffc900; } + .button.button-energized.button-icon { + border-color: transparent; + background: none; } + .button.button-energized.button-outline { + border-color: #ffc900; + background: transparent; + color: #ffc900; } + .button.button-energized.button-outline.active, .button.button-energized.button-outline.activated { + background-color: #ffc900; + box-shadow: none; + color: #fff; } + .button.button-royal { + border-color: transparent; + background-color: #886aea; + color: #fff; } + .button.button-royal:hover { + color: #fff; + text-decoration: none; } + .button.button-royal.active, .button.button-royal.activated { + border-color: #a2a2a2; + background-color: #6b46e5; } + .button.button-royal.button-clear, .button.button-royal.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #886aea; } + .button.button-royal.button-icon { + border-color: transparent; + background: none; } + .button.button-royal.button-outline { + border-color: #886aea; + background: transparent; + color: #886aea; } + .button.button-royal.button-outline.active, .button.button-royal.button-outline.activated { + background-color: #886aea; + box-shadow: none; + color: #fff; } + .button.button-dark { + border-color: transparent; + background-color: #444; + color: #fff; } + .button.button-dark:hover { + color: #fff; + text-decoration: none; } + .button.button-dark.active, .button.button-dark.activated { + border-color: #a2a2a2; + background-color: #262626; } + .button.button-dark.button-clear, .button.button-dark.button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: #444; } + .button.button-dark.button-icon { + border-color: transparent; + background: none; } + .button.button-dark.button-outline { + border-color: #444; + background: transparent; + color: #444; } + .button.button-dark.button-outline.active, .button.button-dark.button-outline.activated { + background-color: #444; + box-shadow: none; + color: #fff; } + + .button-small, .button-text.button-small { + padding: 2px 4px 1px; + min-width: 28px; + min-height: 30px; + font-size: 12px; + line-height: 26px; } + .button-small .icon:before, .button-small .icon-help:before, .button-small .icon-alert:before, .button-small #menu .footer .icon-help:before, #menu .footer .button-small .icon-help:before, .button-small.icon:before, .button-small.icon-help:before, .button-small.icon-alert:before, #menu .footer .button-small.icon-help:before, .button-small.icon-left:before, .button-small.icon-right:before { + font-size: 16px; + line-height: 19px; + margin-top: 3px; } + + .button-large { + padding: 0 16px; + min-width: 68px; + min-height: 59px; + font-size: 20px; + line-height: 53px; } + .button-large .icon:before, .button-large .icon-help:before, .button-large .icon-alert:before, .button-large #menu .footer .icon-help:before, #menu .footer .button-large .icon-help:before, .button-large.icon:before, .button-large.icon-help:before, .button-large.icon-alert:before, #menu .footer .button-large.icon-help:before, .button-large.icon-left:before, .button-large.icon-right:before { + padding-bottom: 2px; + font-size: 32px; + line-height: 51px; } + + .button-icon { + -webkit-transition: opacity 0.1s; + transition: opacity 0.1s; + padding: 0 6px; + min-width: initial; + border-color: transparent; + background: none; } + .button-icon.button.active, .button-icon.button.activated { + border-color: transparent; + background: none; + box-shadow: none; + opacity: 0.3; } + .button-icon .icon:before, .button-icon .icon-help:before, .button-icon .icon-alert:before, .button-icon #menu .footer .icon-help:before, #menu .footer .button-icon .icon-help:before, .button-icon.icon:before, .button-icon.icon-help:before, .button-icon.icon-alert:before, #menu .footer .button-icon.icon-help:before { + font-size: 32px; } + + .button-clear, .button-text { + -webkit-transition: opacity 0.1s; + transition: opacity 0.1s; + padding: 0 6px; + max-height: 42px; + border-color: transparent; + background: none; + box-shadow: none; } + .button-clear.button-clear, .button-text { + border-color: transparent; + background: none; + box-shadow: none; + color: transparent; } + .button-clear.button-icon, .button-icon.button-text { + border-color: transparent; + background: none; } + .button-clear.active, .active.button-text, .button-clear.activated, .activated.button-text { + opacity: 0.3; } + + .button-outline { + -webkit-transition: opacity 0.1s; + transition: opacity 0.1s; + background: none; + box-shadow: none; } + .button-outline.button-outline { + border-color: transparent; + background: transparent; + color: transparent; } + .button-outline.button-outline.active, .button-outline.button-outline.activated { + background-color: transparent; + box-shadow: none; + color: #fff; } + + .padding > .button.button-block:first-child, .item.large-button-bar > .button.button-block:first-child { + margin-top: 0; } + + .button-block { + display: block; + clear: both; } + .button-block:after { + clear: both; } + + .button-full, + .button-full > .button { + display: block; + margin-right: 0; + margin-left: 0; + border-right-width: 0; + border-left-width: 0; + border-radius: 0; } + + button.button-block, + button.button-full, + .button-full > button.button, + input.button.button-block { + width: 100%; } + + a.button { + text-decoration: none; } + a.button .icon:before, a.button .icon-help:before, a.button .icon-alert:before, a.button #menu .footer .icon-help:before, #menu .footer a.button .icon-help:before, a.button.icon:before, a.button.icon-help:before, a.button.icon-alert:before, #menu .footer a.button.icon-help:before, a.button.icon-left:before, a.button.icon-right:before { + margin-top: 2px; } + + .button.disabled, + .button[disabled] { + opacity: .4; + cursor: default !important; + pointer-events: none; } + + /** + * Button Bar + * -------------------------------------------------- + */ + .button-bar { + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + -webkit-box-flex: 1; + -webkit-flex: 1; + -moz-box-flex: 1; + -moz-flex: 1; + -ms-flex: 1; + flex: 1; + width: 100%; } + .button-bar.button-bar-inline { + display: block; + width: auto; + *zoom: 1; } + .button-bar.button-bar-inline:before, .button-bar.button-bar-inline:after { + display: table; + content: ""; + line-height: 0; } + .button-bar.button-bar-inline:after { + clear: both; } + .button-bar.button-bar-inline > .button { + width: auto; + display: inline-block; + float: left; } + .button-bar.bar-light > .button { + border-color: #ddd; } + .button-bar.bar-stable > .button { + border-color: #b2b2b2; } + .button-bar.bar-positive > .button { + border-color: #0c60ee; } + .button-bar.bar-calm > .button { + border-color: #0a9dc7; } + .button-bar.bar-assertive > .button { + border-color: #e42112; } + .button-bar.bar-balanced > .button { + border-color: #28a54c; } + .button-bar.bar-energized > .button { + border-color: #e6b500; } + .button-bar.bar-royal > .button { + border-color: #6b46e5; } + .button-bar.bar-dark > .button { + border-color: #111; } + + .button-bar > .button { + -webkit-box-flex: 1; + -webkit-flex: 1; + -moz-box-flex: 1; + -moz-flex: 1; + -ms-flex: 1; + flex: 1; + display: block; + overflow: hidden; + padding: 0 16px; + width: 0; + border-width: 1px 0px 1px 1px; + border-radius: 0; + text-align: center; + text-overflow: ellipsis; + white-space: nowrap; } + .button-bar > .button:before, + .button-bar > .button .icon:before, + .button-bar > .button .icon-help:before, + .button-bar > .button .icon-alert:before, + .button-bar > .button #menu .footer .icon-help:before, #menu .footer + .button-bar > .button .icon-help:before { + line-height: 44px; } + .button-bar > .button:first-child { + border-radius: 4px 0px 0px 4px; } + .button-bar > .button:last-child { + border-right-width: 1px; + border-radius: 0px 4px 4px 0px; } + .button-bar > .button:only-child { + border-radius: 4px; } + + .button-bar > .button-small:before, + .button-bar > .button-small .icon:before, + .button-bar > .button-small .icon-help:before, + .button-bar > .button-small .icon-alert:before, + .button-bar > .button-small #menu .footer .icon-help:before, #menu .footer + .button-bar > .button-small .icon-help:before { + line-height: 28px; } + + /** + * Grid + * -------------------------------------------------- + * Using flexbox for the grid, inspired by Philip Walton: + * http://philipwalton.github.io/solved-by-flexbox/demos/grids/ + * By default each .col within a .row will evenly take up + * available width, and the height of each .col with take + * up the height of the tallest .col in the same .row. + */ + .row { + display: -webkit-box; + display: -webkit-flex; + display: -moz-box; + display: -moz-flex; + display: -ms-flexbox; + display: flex; + padding: 5px; + width: 100%; } + + .row-wrap { + -webkit-flex-wrap: wrap; + -moz-flex-wrap: wrap; + -ms-flex-wrap: wrap; + flex-wrap: wrap; } + + .row-no-padding { + padding: 0; } + .row-no-padding > .col { + padding: 0; } + + .row + .row { + margin-top: -5px; + padding-top: 0; } + + .col { + -webkit-box-flex: 1; + -webkit-flex: 1; + -moz-box-flex: 1; + -moz-flex: 1; + -ms-flex: 1; + flex: 1; + display: block; + padding: 5px; + width: 100%; } + + /* Vertically Align Columns */ + /* .row-* vertically aligns every .col in the .row */ + .row-top { + -webkit-box-align: start; + -ms-flex-align: start; + -webkit-align-items: flex-start; + -moz-align-items: flex-start; + align-items: flex-start; } + + .row-bottom { + -webkit-box-align: end; + -ms-flex-align: end; + -webkit-align-items: flex-end; + -moz-align-items: flex-end; + align-items: flex-end; } + + .row-center { + -webkit-box-align: center; + -ms-flex-align: center; + -webkit-align-items: center; + -moz-align-items: center; + align-items: center; } + + .row-stretch { + -webkit-box-align: stretch; + -ms-flex-align: stretch; + -webkit-align-items: stretch; + -moz-align-items: stretch; + align-items: stretch; } + + .row-baseline { + -webkit-box-align: baseline; + -ms-flex-align: baseline; + -webkit-align-items: baseline; + -moz-align-items: baseline; + align-items: baseline; } + + /* .col-* vertically aligns an individual .col */ + .col-top { + -webkit-align-self: flex-start; + -moz-align-self: flex-start; + -ms-flex-item-align: start; + align-self: flex-start; } + + .col-bottom { + -webkit-align-self: flex-end; + -moz-align-self: flex-end; + -ms-flex-item-align: end; + align-self: flex-end; } + + .col-center { + -webkit-align-self: center; + -moz-align-self: center; + -ms-flex-item-align: center; + align-self: center; } + + /* Column Offsets */ + .col-offset-10 { + margin-left: 10%; } + + .col-offset-20 { + margin-left: 20%; } + + .col-offset-25 { + margin-left: 25%; } + + .col-offset-33, .col-offset-34 { + margin-left: 33.3333%; } + + .col-offset-50 { + margin-left: 50%; } + + .col-offset-66, .col-offset-67 { + margin-left: 66.6666%; } + + .col-offset-75 { + margin-left: 75%; } + + .col-offset-80 { + margin-left: 80%; } + + .col-offset-90 { + margin-left: 90%; } + + /* Explicit Column Percent Sizes */ + /* By default each grid column will evenly distribute */ + /* across the grid. However, you can specify individual */ + /* columns to take up a certain size of the available area */ + .col-10 { + -webkit-box-flex: 0; + -webkit-flex: 0 0 10%; + -moz-box-flex: 0; + -moz-flex: 0 0 10%; + -ms-flex: 0 0 10%; + flex: 0 0 10%; + max-width: 10%; } + + .col-20 { + -webkit-box-flex: 0; + -webkit-flex: 0 0 20%; + -moz-box-flex: 0; + -moz-flex: 0 0 20%; + -ms-flex: 0 0 20%; + flex: 0 0 20%; + max-width: 20%; } + + .col-25 { + -webkit-box-flex: 0; + -webkit-flex: 0 0 25%; + -moz-box-flex: 0; + -moz-flex: 0 0 25%; + -ms-flex: 0 0 25%; + flex: 0 0 25%; + max-width: 25%; } + + .col-33, .col-34 { + -webkit-box-flex: 0; + -webkit-flex: 0 0 33.3333%; + -moz-box-flex: 0; + -moz-flex: 0 0 33.3333%; + -ms-flex: 0 0 33.3333%; + flex: 0 0 33.3333%; + max-width: 33.3333%; } + + .col-40 { + -webkit-box-flex: 0; + -webkit-flex: 0 0 40%; + -moz-box-flex: 0; + -moz-flex: 0 0 40%; + -ms-flex: 0 0 40%; + flex: 0 0 40%; + max-width: 40%; } + + .col-50 { + -webkit-box-flex: 0; + -webkit-flex: 0 0 50%; + -moz-box-flex: 0; + -moz-flex: 0 0 50%; + -ms-flex: 0 0 50%; + flex: 0 0 50%; + max-width: 50%; } + + .col-60 { + -webkit-box-flex: 0; + -webkit-flex: 0 0 60%; + -moz-box-flex: 0; + -moz-flex: 0 0 60%; + -ms-flex: 0 0 60%; + flex: 0 0 60%; + max-width: 60%; } + + .col-66, .col-67 { + -webkit-box-flex: 0; + -webkit-flex: 0 0 66.6666%; + -moz-box-flex: 0; + -moz-flex: 0 0 66.6666%; + -ms-flex: 0 0 66.6666%; + flex: 0 0 66.6666%; + max-width: 66.6666%; } + + .col-75 { + -webkit-box-flex: 0; + -webkit-flex: 0 0 75%; + -moz-box-flex: 0; + -moz-flex: 0 0 75%; + -ms-flex: 0 0 75%; + flex: 0 0 75%; + max-width: 75%; } + + .col-80 { + -webkit-box-flex: 0; + -webkit-flex: 0 0 80%; + -moz-box-flex: 0; + -moz-flex: 0 0 80%; + -ms-flex: 0 0 80%; + flex: 0 0 80%; + max-width: 80%; } + + .col-90 { + -webkit-box-flex: 0; + -webkit-flex: 0 0 90%; + -moz-box-flex: 0; + -moz-flex: 0 0 90%; + -ms-flex: 0 0 90%; + flex: 0 0 90%; + max-width: 90%; } + + /* Responsive Grid Classes */ + /* Adding a class of responsive-X to a row */ + /* will trigger the flex-direction to */ + /* change to column and add some margin */ + /* to any columns in the row for clearity */ + @media (max-width: 567px) { + .responsive-sm { + -webkit-box-direction: normal; + -moz-box-direction: normal; + -webkit-box-orient: vertical; + -moz-box-orient: vertical; + -webkit-flex-direction: column; + -ms-flex-direction: column; + flex-direction: column; } + .responsive-sm .col, .responsive-sm .col-10, .responsive-sm .col-20, .responsive-sm .col-25, .responsive-sm .col-33, .responsive-sm .col-34, .responsive-sm .col-50, .responsive-sm .col-66, .responsive-sm .col-67, .responsive-sm .col-75, .responsive-sm .col-80, .responsive-sm .col-90 { + -webkit-box-flex: 1; + -webkit-flex: 1; + -moz-box-flex: 1; + -moz-flex: 1; + -ms-flex: 1; + flex: 1; + margin-bottom: 15px; + margin-left: 0; + max-width: 100%; + width: 100%; } } + + @media (max-width: 767px) { + .responsive-md { + -webkit-box-direction: normal; + -moz-box-direction: normal; + -webkit-box-orient: vertical; + -moz-box-orient: vertical; + -webkit-flex-direction: column; + -ms-flex-direction: column; + flex-direction: column; } + .responsive-md .col, .responsive-md .col-10, .responsive-md .col-20, .responsive-md .col-25, .responsive-md .col-33, .responsive-md .col-34, .responsive-md .col-50, .responsive-md .col-66, .responsive-md .col-67, .responsive-md .col-75, .responsive-md .col-80, .responsive-md .col-90 { + -webkit-box-flex: 1; + -webkit-flex: 1; + -moz-box-flex: 1; + -moz-flex: 1; + -ms-flex: 1; + flex: 1; + margin-bottom: 15px; + margin-left: 0; + max-width: 100%; + width: 100%; } } + + @media (max-width: 1023px) { + .responsive-lg { + -webkit-box-direction: normal; + -moz-box-direction: normal; + -webkit-box-orient: vertical; + -moz-box-orient: vertical; + -webkit-flex-direction: column; + -ms-flex-direction: column; + flex-direction: column; } + .responsive-lg .col, .responsive-lg .col-10, .responsive-lg .col-20, .responsive-lg .col-25, .responsive-lg .col-33, .responsive-lg .col-34, .responsive-lg .col-50, .responsive-lg .col-66, .responsive-lg .col-67, .responsive-lg .col-75, .responsive-lg .col-80, .responsive-lg .col-90 { + -webkit-box-flex: 1; + -webkit-flex: 1; + -moz-box-flex: 1; + -moz-flex: 1; + -ms-flex: 1; + flex: 1; + margin-bottom: 15px; + margin-left: 0; + max-width: 100%; + width: 100%; } } + + /** + * Utility Classes + * -------------------------------------------------- + */ + .hide { + display: none; } + + .opacity-hide { + opacity: 0; } + + .grade-b .opacity-hide, + .grade-c .opacity-hide { + opacity: 1; + display: none; } + + .show { + display: block; } + + .opacity-show { + opacity: 1; } + + .invisible { + visibility: hidden; } + + .keyboard-open .hide-on-keyboard-open { + display: none; } + + .keyboard-open .tabs.hide-on-keyboard-open + .pane .has-tabs, + .keyboard-open .bar-footer.hide-on-keyboard-open + .pane .has-footer { + bottom: 0; } + + .inline { + display: inline-block; } + + .disable-pointer-events { + pointer-events: none; } + + .enable-pointer-events { + pointer-events: auto; } + + .disable-user-behavior { + -webkit-user-select: none; + -moz-user-select: none; + -ms-user-select: none; + user-select: none; + -webkit-touch-callout: none; + -webkit-tap-highlight-color: transparent; + -webkit-tap-highlight-color: transparent; + -webkit-user-drag: none; + -ms-touch-action: none; + -ms-content-zooming: none; } + + .click-block { + position: absolute; + top: 0; + right: 0; + bottom: 0; + left: 0; + opacity: 0; + z-index: 99999; + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); + overflow: hidden; } + + .click-block-hide { + -webkit-transform: translate3d(-9999px, 0, 0); + transform: translate3d(-9999px, 0, 0); } + + .no-resize { + resize: none; } + + .block { + display: block; + clear: both; } + .block:after { + display: block; + visibility: hidden; + clear: both; + height: 0; + content: "."; } + + .full-image { + width: 100%; } + + .clearfix { + *zoom: 1; } + .clearfix:before, .clearfix:after { + display: table; + content: ""; + line-height: 0; } + .clearfix:after { + clear: both; } + + /** + * Content Padding + * -------------------------------------------------- + */ + .padding, .item.large-button-bar { + padding: 10px; } + + .padding-top, + .padding-vertical { + padding-top: 10px; } + + .padding-right, .popover-share .bar-footer .button-close, + .padding-horizontal { + padding-right: 10px; } + + .padding-bottom, .popover-share .bar-footer .button-close, + .padding-vertical { + padding-bottom: 10px; } + + .padding-left, + .padding-horizontal { + padding-left: 10px; } + + /** + * Scrollable iFrames + * -------------------------------------------------- + */ + .iframe-wrapper { + position: fixed; + -webkit-overflow-scrolling: touch; + overflow: scroll; } + .iframe-wrapper iframe { + height: 100%; + width: 100%; } + + /** + * Rounded + * -------------------------------------------------- + */ + .rounded { + border-radius: 4px; } + + /** + * Utility Colors + * -------------------------------------------------- + * Utility colors are added to help set a naming convention. You'll + * notice we purposely do not use words like "red" or "blue", but + * instead have colors which represent an emotion or generic theme. + */ + .light, a.light { + color: #fff; } + + .light-bg { + background-color: #fff; } + + .light-border { + border-color: #ddd; } + + .stable, a.stable { + color: #f8f8f8; } + + .stable-bg { + background-color: #f8f8f8; } + + .stable-border { + border-color: #b2b2b2; } + + .positive, .icon-help, a.positive, a.icon-help { + color: #387ef5; } + + .positive-bg { + background-color: #387ef5; } + + .positive-border { + border-color: #0c60ee; } + + .calm, #menu .footer .icon-help, a.calm, #menu .footer a.icon-help { + color: #11c1f3; } + + .calm-bg { + background-color: #11c1f3; } + + .calm-border { + border-color: #0a9dc7; } + + .assertive, .icon-alert, a.assertive, a.icon-alert { + color: #ef473a; } + + .assertive-bg { + background-color: #ef473a; } + + .assertive-border { + border-color: #e42112; } + + .balanced, a.balanced { + color: #33cd5f; } + + .balanced-bg { + background-color: #33cd5f; } + + .balanced-border { + border-color: #28a54c; } + + .energized, a.energized { + color: #ffc900; } + + .energized-bg { + background-color: #ffc900; } + + .energized-border { + border-color: #e6b500; } + + .royal, a.royal { + color: #886aea; } + + .royal-bg { + background-color: #886aea; } + + .royal-border { + border-color: #6b46e5; } + + .dark, a.dark { + color: #444; } + + .dark-bg { + background-color: #444; } + + .dark-border { + border-color: #111; } + + [collection-repeat] { + /* Position is set by transforms */ + left: 0 !important; + top: 0 !important; + position: absolute !important; + z-index: 1; } + + .collection-repeat-container { + position: relative; + z-index: 1; } + + .collection-repeat-after-container { + z-index: 0; + display: block; + /* when scrolling horizontally, make sure the after container doesn't take up 100% width */ } + .collection-repeat-after-container.horizontal { + display: inline-block; } + + [ng\:cloak], [ng-cloak], [data-ng-cloak], [x-ng-cloak], .ng-cloak, + .x-ng-cloak, .ng-hide:not(.ng-hide-animate) { + display: none !important; } + + /** + * Platform + * -------------------------------------------------- + * Platform specific tweaks + */ + .platform-ios.platform-cordova:not(.fullscreen) .bar-header:not(.bar-subheader) { + height: 64px; } + .platform-ios.platform-cordova:not(.fullscreen) .bar-header:not(.bar-subheader).item-input-inset .item-input-wrapper { + margin-top: 19px !important; } + .platform-ios.platform-cordova:not(.fullscreen) .bar-header:not(.bar-subheader) > * { + margin-top: 20px; } + + .platform-ios.platform-cordova:not(.fullscreen) .tabs-top > .tabs, + .platform-ios.platform-cordova:not(.fullscreen) .tabs.tabs-top { + top: 64px; } + + .platform-ios.platform-cordova:not(.fullscreen) .has-header, + .platform-ios.platform-cordova:not(.fullscreen) .bar-subheader { + top: 64px; } + + .platform-ios.platform-cordova:not(.fullscreen) .has-subheader { + top: 108px; } + + .platform-ios.platform-cordova:not(.fullscreen) .has-header.has-tabs-top { + top: 113px; } + + .platform-ios.platform-cordova:not(.fullscreen) .has-header.has-subheader.has-tabs-top { + top: 157px; } + + .platform-ios.platform-cordova .popover .bar-header:not(.bar-subheader) { + height: 44px; } + .platform-ios.platform-cordova .popover .bar-header:not(.bar-subheader).item-input-inset .item-input-wrapper { + margin-top: -1px; } + .platform-ios.platform-cordova .popover .bar-header:not(.bar-subheader) > * { + margin-top: 0; } + + .platform-ios.platform-cordova .popover .has-header, + .platform-ios.platform-cordova .popover .bar-subheader { + top: 44px; } + + .platform-ios.platform-cordova .popover .has-subheader { + top: 88px; } + + .platform-ios.platform-cordova.status-bar-hide { + margin-bottom: 20px; } + + @media (orientation: landscape) { + .platform-ios.platform-browser.platform-ipad { + position: fixed; } } + + .platform-c:not(.enable-transitions) * { + -webkit-transition: none !important; + transition: none !important; } + + .slide-in-up { + -webkit-transform: translate3d(0, 100%, 0); + transform: translate3d(0, 100%, 0); } + + .slide-in-up.ng-enter, + .slide-in-up > .ng-enter { + -webkit-transition: all cubic-bezier(0.1, 0.7, 0.1, 1) 400ms; + transition: all cubic-bezier(0.1, 0.7, 0.1, 1) 400ms; } + + .slide-in-up.ng-enter-active, + .slide-in-up > .ng-enter-active { + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); } + + .slide-in-up.ng-leave, + .slide-in-up > .ng-leave { + -webkit-transition: all ease-in-out 250ms; + transition: all ease-in-out 250ms; } + + @-webkit-keyframes scaleOut { + from { + -webkit-transform: scale(1); + opacity: 1; } + to { + -webkit-transform: scale(0.8); + opacity: 0; } } + + @keyframes scaleOut { + from { + transform: scale(1); + opacity: 1; } + to { + transform: scale(0.8); + opacity: 0; } } + + @-webkit-keyframes superScaleIn { + from { + -webkit-transform: scale(1.2); + opacity: 0; } + to { + -webkit-transform: scale(1); + opacity: 1; } } + + @keyframes superScaleIn { + from { + transform: scale(1.2); + opacity: 0; } + to { + transform: scale(1); + opacity: 1; } } + + [nav-view-transition="ios"] [nav-view="entering"], + [nav-view-transition="ios"] [nav-view="leaving"] { + -webkit-transition-duration: 500ms; + transition-duration: 500ms; + -webkit-transition-timing-function: cubic-bezier(0.36, 0.66, 0.04, 1); + transition-timing-function: cubic-bezier(0.36, 0.66, 0.04, 1); + -webkit-transition-property: opacity, -webkit-transform, box-shadow; + transition-property: opacity, transform, box-shadow; } + + [nav-view-transition="ios"][nav-view-direction="forward"], [nav-view-transition="ios"][nav-view-direction="back"] { + background-color: #000; } + + [nav-view-transition="ios"] [nav-view="active"], + [nav-view-transition="ios"][nav-view-direction="forward"] [nav-view="entering"], + [nav-view-transition="ios"][nav-view-direction="back"] [nav-view="leaving"] { + z-index: 3; } + + [nav-view-transition="ios"][nav-view-direction="back"] [nav-view="entering"], + [nav-view-transition="ios"][nav-view-direction="forward"] [nav-view="leaving"] { + z-index: 2; } + + [nav-bar-transition="ios"] .title, + [nav-bar-transition="ios"] .buttons, + [nav-bar-transition="ios"] .back-text { + -webkit-transition-duration: 500ms; + transition-duration: 500ms; + -webkit-transition-timing-function: cubic-bezier(0.36, 0.66, 0.04, 1); + transition-timing-function: cubic-bezier(0.36, 0.66, 0.04, 1); + -webkit-transition-property: opacity, -webkit-transform; + transition-property: opacity, transform; } + + [nav-bar-transition="ios"] [nav-bar="active"], + [nav-bar-transition="ios"] [nav-bar="entering"] { + z-index: 10; } + [nav-bar-transition="ios"] [nav-bar="active"] .bar, + [nav-bar-transition="ios"] [nav-bar="entering"] .bar { + background: transparent; } + + [nav-bar-transition="ios"] [nav-bar="cached"] { + display: block; } + [nav-bar-transition="ios"] [nav-bar="cached"] .header-item { + display: none; } + + [nav-view-transition="android"] [nav-view="entering"], + [nav-view-transition="android"] [nav-view="leaving"] { + -webkit-transition-duration: 200ms; + transition-duration: 200ms; + -webkit-transition-timing-function: cubic-bezier(0.4, 0.6, 0.2, 1); + transition-timing-function: cubic-bezier(0.4, 0.6, 0.2, 1); + -webkit-transition-property: -webkit-transform; + transition-property: transform; } + + [nav-view-transition="android"] [nav-view="active"], + [nav-view-transition="android"][nav-view-direction="forward"] [nav-view="entering"], + [nav-view-transition="android"][nav-view-direction="back"] [nav-view="leaving"] { + z-index: 3; } + + [nav-view-transition="android"][nav-view-direction="back"] [nav-view="entering"], + [nav-view-transition="android"][nav-view-direction="forward"] [nav-view="leaving"] { + z-index: 2; } + + [nav-bar-transition="android"] .title, + [nav-bar-transition="android"] .buttons { + -webkit-transition-duration: 200ms; + transition-duration: 200ms; + -webkit-transition-timing-function: cubic-bezier(0.4, 0.6, 0.2, 1); + transition-timing-function: cubic-bezier(0.4, 0.6, 0.2, 1); + -webkit-transition-property: opacity; + transition-property: opacity; } + + [nav-bar-transition="android"] [nav-bar="active"], + [nav-bar-transition="android"] [nav-bar="entering"] { + z-index: 10; } + [nav-bar-transition="android"] [nav-bar="active"] .bar, + [nav-bar-transition="android"] [nav-bar="entering"] .bar { + background: transparent; } + + [nav-bar-transition="android"] [nav-bar="cached"] { + display: block; } + [nav-bar-transition="android"] [nav-bar="cached"] .header-item { + display: none; } + + [nav-swipe="fast"] [nav-view], + [nav-swipe="fast"] .title, + [nav-swipe="fast"] .buttons, + [nav-swipe="fast"] .back-text { + -webkit-transition-duration: 50ms; + transition-duration: 50ms; + -webkit-transition-timing-function: linear; + transition-timing-function: linear; } + + [nav-swipe="slow"] [nav-view], + [nav-swipe="slow"] .title, + [nav-swipe="slow"] .buttons, + [nav-swipe="slow"] .back-text { + -webkit-transition-duration: 160ms; + transition-duration: 160ms; + -webkit-transition-timing-function: linear; + transition-timing-function: linear; } + + [nav-view="cached"], + [nav-bar="cached"] { + display: none; } + + [nav-view="stage"] { + opacity: 0; + -webkit-transition-duration: 0; + transition-duration: 0; } + + [nav-bar="stage"] .title, + [nav-bar="stage"] .buttons, + [nav-bar="stage"] .back-text { + position: absolute; + opacity: 0; + -webkit-transition-duration: 0s; + transition-duration: 0s; } + + /* BEGIN Thin */ + @font-face { + font-family: RobotoDraft; + src: url("../lib/ionic/fonts/robotdraft/Thin/RobotoDraft-Thin.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Thin/RobotoDraft-Thin.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Thin/RobotoDraft-Thin.ttf?v=1.1.0") format("truetype"); + font-weight: 100; + font-style: normal; } + + /* END Thin */ + /* BEGIN Light */ + @font-face { + font-family: RobotoDraft; + src: url("../lib/ionic/fonts/robotdraft/Light/RobotoDraft-Light.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Light/RobotoDraft-Light.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Light/RobotoDraft-Light.ttf?v=1.1.0") format("truetype"); + font-weight: 300; + font-style: normal; } + + /* END Light */ + /* BEGIN Regular */ + @font-face { + font-family: RobotoDraft; + src: url("../lib/ionic/fonts/robotdraft/Regular/RobotoDraft-Regular.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Regular/RobotoDraft-Regular.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Regular/RobotoDraft-Regular.ttf?v=1.1.0") format("truetype"); + font-weight: 400; + font-style: normal; } + + @font-face { + font-family: RobotoDraft; + src: url("../lib/ionic/fonts/robotdraft/Regular/RobotoDraft-Regular.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Regular/RobotoDraft-Regular.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Regular/RobotoDraft-Regular.ttf?v=1.1.0") format("truetype"); + font-weight: normal; + font-style: normal; } + + /* END Regular */ + /* BEGIN Italic */ + @font-face { + font-family: RobotoDraft; + src: url("../lib/ionic/fonts/robotdraft/Italic/RobotoDraft-Italic.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Italic/RobotoDraft-Italic.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Italic/RobotoDraft-Italic.ttf?v=1.1.0") format("truetype"); + font-weight: 400; + font-style: italic; } + + @font-face { + font-family: RobotoDraft; + src: url("../lib/ionic/fonts/robotdraft/Italic/RobotoDraft-Italic.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Italic/RobotoDraft-Italic.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Italic/RobotoDraft-Italic.ttf?v=1.1.0") format("truetype"); + font-weight: normal; + font-style: italic; } + + /* END Italic */ + /* BEGIN Medium */ + @font-face { + font-family: RobotoDraft; + src: url("../lib/ionic/fonts/robotdraft/Medium/RobotoDraft-Medium.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Medium/RobotoDraft-Medium.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Medium/RobotoDraft-Medium.ttf?v=1.1.0") format("truetype"); + font-weight: 500; + font-style: normal; } + + /* END Medium */ + /* BEGIN Bold */ + @font-face { + font-family: RobotoDraft; + src: url("../lib/ionic/fonts/robotdraft/Bold/RobotoDraft-Bold.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Bold/RobotoDraft-Bold.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Bold/RobotoDraft-Bold.ttf?v=1.1.0") format("truetype"); + font-weight: 700; + font-style: normal; } + + @font-face { + font-family: RobotoDraft; + src: url("../lib/ionic/fonts/robotdraft/Bold/RobotoDraft-Bold.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Bold/RobotoDraft-Bold.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Bold/RobotoDraft-Bold.ttf?v=1.1.0") format("truetype"); + font-weight: bold; + font-style: normal; } + + /* END Bold */ + /* BEGIN Bold Italic */ + @font-face { + font-family: RobotoDraft; + src: url("../lib/ionic/fonts/robotdraft/BoldItalic/RobotoDraft-BoldItalic.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/BoldItalic/RobotoDraft-BoldItalic.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/BoldItalic/RobotoDraft-BoldItalic.ttf?v=1.1.0") format("truetype"); + font-weight: 700; + font-style: italic; } + + @font-face { + font-family: RobotoDraft; + src: url("../lib/ionic/fonts/robotdraft/BoldItalic/RobotoDraft-BoldItalic.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/BoldItalic/RobotoDraft-BoldItalic.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/BoldItalic/RobotoDraft-BoldItalic.ttf?v=1.1.0") format("truetype"); + font-weight: bold; + font-style: italic; } + + /* END Bold Italic */ + /* BEGIN Black */ + @font-face { + font-family: RobotoDraft; + src: url("../lib/ionic/fonts/robotdraft/Black/RobotoDraft-Black.woff2?v=1.1.0") format("woff2"), url("../lib/ionic/fonts/robotdraft/Black/RobotoDraft-Black.woff?v=1.1.0") format("woff"), url("../lib/ionic/fonts/robotdraft/Black/RobotoDraft-Black.ttf?v=1.1.0") format("truetype"); + font-weight: 900; + font-style: normal; } + + /* END Black */ + /* Directives : MD Label + ==================================*/ + .item-md-label { + display: block; + background: transparent; + box-shadow: none; + margin-left: 12px; + margin-right: 12px; + padding: 30px 0 0; } + + .item-md-label .input-label { + position: absolute; + padding: 5px 0 0; + z-index: 2; + -webkit-transform: translate3d(0, -30px, 0) scale(1); + transform: translate3d(0, -30px, 0) scale(1); + -webkit-transition: all 0.2s ease; + transition: all 0.2s ease; + color: #fff; + opacity: 0.5; + filter: alpha(opacity=50); + -webkit-transform-origin: 0; + -ms-transform-origin: 0; + transform-origin: 0; } + + .item-md-label input { + background-color: rgba(0, 0, 0, 0.6); + bottom: 0; + color: #fff; + letter-spacing: 0.25rem; + padding: 20px 10px; + position: relative; + z-index: 1; } + + .item-md-label .highlight { + position: absolute; + bottom: 0; + height: 2px; + left: 0; + width: 100%; + -webkit-transform: translate3d(-100%, 0, 0); + transform: translate3d(-100%, 0, 0); + -webkit-transition: all 0.15s ease; + transition: all 0.15s ease; + z-index: 1; } + + .item-md-label .highlight-light { + background: #fff; } + + .item-md-label .highlight-stable { + background: #f8f8f8; } + + .item-md-label .highlight-positive { + background: #387ef5; } + + .item-md-label .highlight-calm { + background: #11c1f3; } + + .item-md-label .highlight-balanced { + background: #33cd5f; } + + .item-md-label .highlight-energized { + background: #ffc900; } + + .item-md-label .highlight-assertive { + background: #ef473a; } + + .item-md-label .highlight-royal { + background: #886aea; } + + .item-md-label .highlight-dark { + background: #444; } + + .item-md-label .input-label { + letter-spacing: 0.25rem; + padding: 0 10px; } + + .item-md-label input:focus ~ .input-label, .item-md-label input.used ~ .input-label { + font-weight: bold; + opacity: 0.7; + filter: alpha(opacity=70); + padding: 0; + text-transform: uppercase; + -webkit-transform: translate3d(0, -60px, 0) scale(0.9); + transform: translate3d(0, -60px, 0) scale(0.9); } + + .item-md-label input:focus ~ .highlight { + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); } + + /* Bar - Header - Expanded + ==================================*/ + .expanded .bar.bar-header, + .bar.bar-header.expanded { + height: 75px; } + + .expanded.bar.bar-header .title, + .bar.bar-header.expanded .title { + bottom: 0; + top: initial; + padding-left: 16px; } + + .expanded .bar.bar-header .title.fab-left, + .bar.bar-header.expanded .title.fab-left { + bottom: 0; + left: 90px; + position: absolute; + right: initial; + top: initial; } + + .expanded .bar.bar-header .title.fab-right, + .bar.bar-header.expanded .title.fab-right { + bottom: 0; + left: 4px; + position: absolute; + top: initial; + right: initial; } + + .expanded .bar.bar-header + .button-fab, + .bar.bar-header.expanded + .button-fab { + top: 50px; } + + .expanded .bar.bar-header.push-down, + .bar.bar-header.expanded.push-down { + height: 44px; + overflow: hidden; } + + .expanded .bar.bar-header, + .bar.bar-header.expanded { + -webkit-transition: height 1s cubic-bezier(0.55, 0, 0.1, 1); + transition: height 1s cubic-bezier(0.55, 0, 0.1, 1); + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); } + + .expanded .bar.bar-header + .button-fab, + .bar.bar-header.expanded + .button-fab { + -webkit-transition: all 1.1s cubic-bezier(0.55, 0, 0.1, 1); + transition: all 1.1s cubic-bezier(0.55, 0, 0.1, 1); + -webkit-transform: translate3d(0, 0, 0) scale(1); + transform: translate3d(0, 0, 0) scale(1); } + + .expanded .bar.bar-header.push-down + .button-fab, + .bar.bar-header.expanded.push-down + .button-fab { + top: 0; + -webkit-transform: translate3d(-100px, -100px, 0) scale(2.5); + transform: translate3d(-100px, -100px, 0) scale(2.5); } + + .expanded .bar.bar-header.push-down .title, + .bar.bar-header.expanded.push-down .title { + opacity: 0; + filter: alpha(opacity=0); + left: initial; + right: initial; } + + .expanded .bar.bar-header .title, + .bar.bar-header.expanded .title { + opacity: 1; + filter: alpha(opacity=100); + -webkit-transition: all 2s cubic-bezier(0.55, 0, 0.1, 1); + transition: all 2s cubic-bezier(0.55, 0, 0.1, 1); } + + .expanded .bar.bar-header .title, .bar.bar-header.expanded .title { + bottom: 0; + left: 42px !important; + top: initial; } + + .expanded.has-header-fab-left .bar.bar-header .title, .bar.bar-header.expanded.has-header-fab-left .title { + left: 76px !important; } + + /* Bar + ==================================*/ + .bar { + z-index: 2; + font-size: 1.3em; + width: 100%; + box-shadow: 0px 2px 5px 0 rgba(0, 0, 0, 0.26); } + + .bar .button { + min-width: 38px; + z-index: 3; } + + .bar .no-text span.back-text { + display: none; } + + .bar .title sup { + opacity: 0.7; } + + .bar.bar-header .button + .title { + text-align: left; + left: 35px; + line-height: 46px; } + + /* Button Bar + ==================================*/ + .button-bar { + box-shadow: 0px 1px 3px rgba(0, 0, 0, 0.15); } + + .button-bar > .button { + box-shadow: none; + /* line-height: initial; */ } + + .button-bar > .button .icon:before, .button-bar > .button .icon-help:before, .button-bar > .button .icon-alert:before, .button-bar > .button #menu .footer .icon-help:before, #menu .footer .button-bar > .button .icon-help:before, + .button-bar > .button:before { + line-height: initial; } + + .bar-footer .button-fab { + position: absolute; + top: -26px; + bottom: initial; } + + .bar-footer .buttons-left .button-fab { + left: 8px; } + + .bar-footer .buttons-right .button-fab { + right: 8px; } + + .bar .button.button-clear, .bar .button.button-text { + box-shadow: none; } + + .left-buttons .button-fab { + left: 8px; + top: 16px; } + + .right-buttons .button-fab { + right: 8px; + top: 16px; } + + .fab-left.title-left, + .fab-left.title.title-left { + left: 68px; } + + /* Button : FAB + ==================================*/ + .button.button-fab, + .bar .button.button-fab { + box-shadow: 0 2px 5px 0 rgba(0, 0, 0, 0.16), 0 2px 10px 0 rgba(0, 0, 0, 0.12); + z-index: 9999; + width: 56px; + height: 56px; + max-height: initial; + max-width: initial; + border-radius: 50%; + border-radius: 50%; + overflow: hidden; + padding: 0; + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); + -webkit-transition: 0.3s fade-in-out; + transition: 0.3s fade-in-out; + -webkit-transition-property: -webkit-transform, box-shadow; + transition-property: transform, box-shadow; } + + .button.button-fab.button-fab-bottom-right, + .bar .button.button-fab.button-fab-bottom-right { + top: auto; + right: 16px; + bottom: 16px; + left: auto; + position: absolute; } + + .button.button-fab.button-fab-bottom-left, + .bar .button.button-fab.button-fab-bottom-left { + top: auto; + right: auto; + bottom: 16px; + left: 16px; + position: absolute; } + + .button.button-fab.button-fab-top-right, + .bar .button.button-fab.button-fab-top-right { + top: 32px; + right: 16px; + bottom: auto; + left: auto; + position: absolute; } + + .button.button-fab.button-fab-top-left, + .bar .button.button-fab.button-fab-top-left { + top: 32px; + right: auto; + bottom: auto; + left: 16px; + position: absolute; } + + .button.button-fab.button-fab-top-left.expanded, + .button.button-fab.button-fab-top-right.expanded, + .bar .button.button-fab.button-fab-top-left.expanded, + .bar .button.button-fab.button-fab-top-right.expanded { + top: 48px; } + + .button.button-fab i, + .bar .button.button-fab i { + font-size: 2.5rem; + margin-top: 0; } + + .button.button-fab.mini, + .bar .button.button-fab.mini { + width: 40px; + height: 40px; } + + .button.button-fab.mini i, + .bar .button.button-fab.mini i { + font-size: 2rem; } + + /* Motion */ + .motion { + -webkit-transition: all 0.5s ease-out; + transition: all 0.5s ease-out; } + + .fade { + opacity: 0; + filter: alpha(opacity=0); + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transition: all 0.1s ease-out !important; + transition: all 0.1s ease-out !important; } + + .spin-back { + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transform: translateZ(0) rotate(360deg) scale(0) !important; + transform: translateZ(0) rotate(360deg) scale(0) !important; + -webkit-transition: all 0.1s ease-out !important; + transition: all 0.1s ease-out !important; } + + .spiral { + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transform: translateZ(0) rotate(-360deg) scale(0) translate(-120px) !important; + transform: translateZ(0) rotate(-360deg) scale(0) translate(-120px) !important; + -webkit-transition: all 0.1s ease-out !important; + transition: all 0.1s ease-out !important; } + + .spiral-back { + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transform: translateZ(0) rotate(360deg) scale(0) translate(120px) !important; + transform: translateZ(0) rotate(360deg) scale(0) translate(120px) !important; + -webkit-transition: all 0.1s ease-out !important; + transition: all 0.1s ease-out !important; } + + .menu-open .avatar { + opacity: 1; + filter: alpha(opacity=100); + -webkit-transform: translateZ(0) rotate(0) scale(1) !important; + transform: translateZ(0) rotate(0) scale(1) !important; + -webkit-transition: all 0.3s ease-out !important; + transition: all 0.3s ease-out !important; } + + .button.button-fab.button-fab-top-left.motion { + -webkit-backface-visibility: hidden; + backface-visibility: hidden; + -webkit-transform: translate3d(-120px, 60px, 0); + transform: translate3d(-120px, 60px, 0); + -webkit-transition: all 0.1s ease-out; + transition: all 0.1s ease-out; } + + .button.button-fab.button-fab-top-right.motion { + -webkit-backface-visibility: hidden; + backface-visibility: hidden; + -webkit-transform: translate3d(120px, 60px, 0); + transform: translate3d(120px, 60px, 0); + -webkit-transition: all 0.1s ease-out; + transition: all 0.1s ease-out; } + + .button.button-fab.button-fab-bottom-left.motion { + -webkit-backface-visibility: hidden; + backface-visibility: hidden; + -webkit-transform: translate3d(-120px, 60px, 0); + transform: translate3d(-120px, 60px, 0); + -webkit-transition: all 0.1s ease-out; + transition: all 0.1s ease-out; } + + .button.button-fab.button-fab-bottom-right.motion { + -webkit-backface-visibility: hidden; + backface-visibility: hidden; + -webkit-transform: translate3d(120px, 60px, 0); + transform: translate3d(120px, 60px, 0); + -webkit-transition: all 0.1s ease-out; + transition: all 0.1s ease-out; } + + .spin { + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transform: translateZ(0) rotate(0) scale(0) !important; + transform: translateZ(0) rotate(0) scale(0) !important; + -webkit-transition: all 0.3s ease-out !important; + transition: all 0.3s ease-out !important; } + + .spin.on { + -webkit-transform: translateZ(0) rotate(-360deg) scale(1) !important; + transform: translateZ(0) rotate(-360deg) scale(1) !important; } + + .flap { + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transform: translateZ(0) rotateX(0) scale(0) translate(-120px) !important; + transform: translateZ(0) rotateX(0) scale(0) translate(-120px) !important; + -webkit-transition: all 0.5s ease-out !important; + transition: all 0.5s ease-out !important; } + + .flap.on { + -webkit-transform: translateZ(0) rotateX(-720deg) scale(1) translate(0) !important; + transform: translateZ(0) rotateX(-720deg) scale(1) translate(0) !important; + -webkit-transition: all 0.5s ease-out !important; + transition: all 0.5s ease-out !important; } + + .drop { + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transform: translateZ(0) scale(3) !important; + transform: translateZ(0) scale(3) !important; + -webkit-transition: all 0.5s ease-out !important; + transition: all 0.5s ease-out !important; } + + .drop.on { + -webkit-transform: translateZ(0) scale(1) !important; + transform: translateZ(0) scale(1) !important; + -webkit-transition: all 0.5s ease-out !important; + transition: all 0.5s ease-out !important; } + + .flip { + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transform: translateZ(0) rotateY(0) scale(0) !important; + transform: translateZ(0) rotateY(0) scale(0) !important; + -webkit-transition: all 0.5s ease-out !important; + transition: all 0.5s ease-out !important; } + + .flip.on { + -webkit-transform: translateZ(0) rotateY(-720deg) scale(1) !important; + transform: translateZ(0) rotateY(-720deg) scale(1) !important; + -webkit-transition: all 0.5s ease-out !important; + transition: all 0.5s ease-out !important; } + + /* Button : Floating + ==================================*/ + .button.button-floating, .bar .button.button-floating { + display: inline-block; + color: #FFF; + position: relative; + z-index: 1; + width: 37px; + height: 37px; + line-height: 37px; + padding: 0; + border-radius: 50%; + background-clip: padding-box; + -webkit-transition: 0.3s; + transition: 0.3s; + cursor: pointer; } + + .button.button-floating i, .bar .button.button-floating i { + width: inherit; + display: inline-block; + text-align: center; + color: #FFF; + font-size: 1.6rem; + line-height: 37px; } + + .button.button-floating.button-large, .bar .button.button-floating.button-large { + width: 55.5px; + height: 55.5px; } + + .button.button-floating.button-large i, .bar .button.button-floating.button-large i { + line-height: 55.5px; } + + /* Button + ==================================*/ + .button, + .button.button-large, + .button.button-flat, + .bar .button, + .bar .button.button-large, + .bar .button.button-flat { + box-shadow: 0 2px 5px 0 rgba(0, 0, 0, 0.16), 0 2px 10px 0 rgba(0, 0, 0, 0.12); + display: inline-block; + height: 36px; + padding: 0 2rem; + border-radius: 2px; + background-clip: padding-box; + text-transform: uppercase; + border: none; + outline: 0; + -webkit-tap-highlight-color: transparent; } + + .button.disabled, + .button.disabled.button-large, + .button.button-floating.disabled, + .button.button-large.disabled, + .button.button:disabled, + .button.button-large:disabled, + .button.button-large:disabled, + .button.button-floating:disabled, + .bar .button.disabled, + .bar .button.disabled.button-large, + .bar .button.button-floating.disabled, + .bar .button.button-large.disabled, + .bar .button.button:disabled, + .bar .button.button-large:disabled, + .bar .button.button-large:disabled, + .bar .button.button-floating:disabled { + background-color: #DFDFDF; + box-shadow: none; + color: #9F9F9F; } + + .button.disabled:hover, + .button.disabled.button-large:hover, + .button.button-floating.disabled:hover, + .button.button-large.disabled:hover, + .button.button:disabled:hover, + .button.button-large:disabled:hover, + .button.button-large:disabled:hover, + .button.button-floating:disabled:hover, + .bar .button.disabled:hover, + .bar .button.disabled.button-large:hover, + .bar .button.button-floating.disabled:hover, + .bar .button.button-large.disabled:hover, + .bar .button.button:disabled:hover, + .bar .button.button-large:disabled:hover, + .bar .button.button-large:disabled:hover, + .bar .button.button-floating:disabled:hover { + background-color: #DFDFDF; + color: #9F9F9F; } + + .button i, + .button.button-large i, + .button.button-floating i, + .button.button-large i, + .button.button-flat i, + .bar .button i, + .bar .button.button-large i, + .bar .button.button-floating i, + .bar .button.button-large i, + .bar .button.button-flat i { + font-size: 1.3rem; } + + .button-bar .button { + border-radius: 0; } + + .button, + .button-large, + .bar .button, + .bar .button-large { + text-decoration: none; + text-align: center; + letter-spacing: 0.5px; + -webkit-transition: 0.2s ease-out; + transition: 0.2s ease-out; + cursor: pointer; } + + .button { + -webkit-user-select: none; + -moz-user-select: none; + -ms-user-select: none; + user-select: none; + position: relative; + outline: none; + margin: 0; + /* background: transparent; */ + white-space: nowrap; + text-align: center; + text-transform: uppercase; + font-weight: 500; + font-style: inherit; + font-variant: inherit; + font-size: inherit; + text-decoration: none; + cursor: pointer; + overflow: hidden; + -webkit-transition: box-shadow 0.4s cubic-bezier(0.25, 0.8, 0.25, 1), background-color 0.4s cubic-bezier(0.25, 0.8, 0.25, 1), -webkit-transform 0.4s cubic-bezier(0.25, 0.8, 0.25, 1); + transition: box-shadow 0.4s cubic-bezier(0.25, 0.8, 0.25, 1), background-color 0.4s cubic-bezier(0.25, 0.8, 0.25, 1), transform 0.4s cubic-bezier(0.25, 0.8, 0.25, 1); } + + .button:focus { + outline: none; } + + .button.ng-hide { + -webkit-transition: none; + transition: none; } + + .button.cornered { + border-radius: 0; } + + .button.raised { + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); } + + .button-outline, + .button-outline:hover, + .button-outline:active { + border-style: solid; + border-width: 1px; } + + .button.button-outline.button-assertive, + .button.button-outline.button-balanced, + .button.button-outline.button-calm, + .button.button-outline.button-dark, + .button.button-outline.button-energized, + .button.button-outline.button-light, + .button.button-outline.button-positive, + .button.button-outline.button-text, + .button.button-outline.button-royal, + .button.button-outline.button-stable, + .button.button-outline { + border-color: rgba(0, 0, 0, 0.1); } + + .button-flat, + .bar .button-flat { + box-shadow: none; + background-color: transparent; + color: #343434; + cursor: pointer; } + + .button.button-flat.disabled, + .bar .button.button-flat.disabled { + color: #b3b3b3; } + + .button.button-large i, + .bar .button.button-large i { + font-size: 1.6rem; } + + .button-pin-header.button-floating { + position: absolute; + z-index: 1000; } + + .button-pin-header.button-pin-left { + left: 24px; + top: -24px; } + + .button-pin-header.button-pin-right { + right: 24px; + top: -24px; } + + .button:not([disabled]).raised:focus, + .button:not([disabled]).raised:hover, + .button:not([disabled]).floating:focus, + .button:not([disabled]).floating:hover { + -webkit-transform: translate3d(0, -1px, 0); + transform: translate3d(0, -1px, 0); } + + .button.button-flat { + box-shadow: none; + /* background: transparent; */ + color: inherit; } + + .button.button-flat:hover { + color: inherit; } + + .button.button-flat, + .button.button-flat:hover, + .button.button-flat:active { + color: #fff; } + + .button.button-clear, .button.button-text, + .button.button-clear:hover, + .button.button-text:hover, + .button.button-clear:active, + .button.button-text:active { + background: transparent; } + + .button-full.ink, + .button-block.ink { + display: block; } + + /* Card + ==================================*/ + .card-item.item { + border: none; + padding-bottom: 4px; + padding-top: 4px; } + + .card-item.item:first-child { + padding-top: 16px; } + + .card { + box-shadow: 0px 2px 5px 0 rgba(0, 0, 0, 0.26); + display: block; + margin: 8px; + padding: 0; + position: relative; } + + .card .image { + display: block; + margin-top: 10px; + margin-bottom: 5px; } + + .card img { + box-shadow: 0px 2px 5px 0 rgba(0, 0, 0, 0.26); + display: block; + max-width: 100%; + max-height: initial; + position: static; } + + .card.card-gallery img { + border: none; + box-shadow: none; + display: block; } + + .card .card-footer { + font-size: 90%; + opacity: 0.8; + filter: alpha(opacity=80); + padding-top: 10px; } + + .card > .item { + border: none; } + + .card.card-gallery > .item { + background: inherit; } + + .card .icon + .icon, .card .icon-help + .icon, .card .icon-alert + .icon, .card #menu .footer .icon-help + .icon, #menu .footer .card .icon-help + .icon, .card .icon + .icon-help, .card .icon-help + .icon-help, .card .icon-alert + .icon-help, .card .icon + .icon-alert, .card .icon-help + .icon-alert, .card .icon-alert + .icon-alert, .card #menu .footer .icon-help + .icon-alert, #menu .footer .card .icon-help + .icon-alert, .card #menu .footer .icon + .icon-help, #menu .footer .card .icon + .icon-help, .card #menu .footer .icon-alert + .icon-help, #menu .footer .card .icon-alert + .icon-help, .card #menu .footer .icon-help + .icon-help, #menu .footer .card .icon-help + .icon-help { + padding-left: 1rem; } + + .card.animate-fade-in { + opacity: 0; + filter: alpha(opacity=0); + -webkit-transform: translate3d(-30px, 1px, 0); + -webkit-transition: all 1s ease-in-out; } + + .card.animate-fade-in.done, .animate-fade-slide-in .expanded .card.animate-fade-in.item, + .animate-fade-slide-in .card.animate-fade-in.expanded.item, + .animate-fade-slide-in-right .expanded .card.animate-fade-in.item, + .animate-fade-slide-in-right .card.animate-fade-in.expanded.item, + .animate-ripple .expanded .card.animate-fade-in.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.animate-fade-in.card-comment { + opacity: 1; + filter: alpha(opacity=100); + -webkit-transform: translate3d(0, 0, 0); } + + .card .item.item-avatar { + min-height: 88px; + padding-left: 88px; } + + /* Hero + ==================================*/ + .hero { + background-size: cover; + box-shadow: 0px 2px 5px 0 rgba(0, 0, 0, 0.26); + color: #fff; + height: 200px; + position: relative; + text-align: center; + -webkit-transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); + transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); + width: 100%; } + + .hero > * { + -webkit-transition: opacity 2.5s cubic-bezier(0.55, 0, 0.1, 1); + transition: opacity 2.5s cubic-bezier(0.55, 0, 0.1, 1); + opacity: 1; + filter: alpha(opacity=100); } + + .hero + .mid-bar { + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); + -webkit-transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); + transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); + height: initial; + opacity: 1; + filter: alpha(opacity=100); } + + .hero .hero-icon { + box-shadow: 0px 0 2px 0 rgba(0, 0, 0, 0.26); + border-radius: 50%; + display: inline-block; + font-size: 65px; + height: 150px; + padding: 10px 30px; + line-height: 136px; + width: 150px; } + + .hero.no-header { + height: 244px; } + + .hero > .content { + bottom: 0; + position: absolute; + text-align: center; + width: 100%; + z-index: 1; } + + .hero > .content > .avatar { + background-position: center; + background-size: cover; + border: solid 1px rgba(255, 255, 255, 0.8); + border-radius: 50%; + display: inline-block; + height: 88px; + left: auto; + margin-bottom: 10px; + position: relative; + width: 88px; } + + .hero h1 .hero h2, .hero h3, .hero h4, .hero h5, .hero h6 { + color: #fff; + margin: 0; } + + .hero h4 { + color: rgba(255, 255, 255, 0.7); + margin: 3px 0 16px; } + + .hero h1 > a, .hero h2 > a, .hero h3 > a, .hero h4 > a, .hero h5 > a, .hero h6 > a { + text-decoration: none; } + + .hero + .button-bar { + border-radius: 0; + margin-top: 0; } + + .hero + .button-bar > .button:first-child, .hero + .button-bar > .button:last-child { + border-radius: 0; } + + .hero .hero-icon { + color: #fff; + font-size: 96px; } + + .hero .hero-icon + h1 { + color: white; + letter-spacing: 0.15rem; } + + .hero .button, .hero .button.button-large, .hero .button.button-flat { + margin: 0; } + + .hero h1.title { + color: #fff; + font-size: 23px; + margin: 0; + text-align: left; + padding-left: 80px; + line-height: 59px; } + + .hero + .mid-bar { + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); + -webkit-transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); + transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); + height: initial; + opacity: 1; + filter: alpha(opacity=100); } + + .hero > * { + -webkit-transition: opacity 2.5s cubic-bezier(0.55, 0, 0.1, 1); + transition: opacity 2.5s cubic-bezier(0.55, 0, 0.1, 1); + opacity: 1; + filter: alpha(opacity=100); } + + /* Item + ==================================*/ + .item { + font-size: 14px; + width: 100%; } + + .item-icon-left .icon, .item-icon-left .icon-help, .item-icon-left .icon-alert, .item-icon-left #menu .footer .icon-help, #menu .footer .item-icon-left .icon-help { + left: 16px; } + + .item-icon-right .icon, .item-icon-right .icon-help, .item-icon-right .icon-alert, .item-icon-right #menu .footer .icon-help, #menu .footer .item-icon-right .icon-help { + right: 16px; } + + /* + .list .item.item-icon-right { + padding-right: 60px; + } + */ + .item-thumbnail-left > img:first-child, .item-thumbnail-left .item-image, .item-thumbnail-left .item-content > img:first-child, .item-thumbnail-left .item-content .item-image { + border-radius: 50%; } + + .tab-item.activated { + height: calc(100% + 3px); + /* Stretch */ } + + /* List + ==================================*/ + .content + .list { + padding-top: 0; } + + .list .item { + border: none; + /* + padding-left: 16px; + padding-right: 16px; + */ + min-height: 48px; + text-align: left; } + + .list .item.tabs { + padding: initial; } + + .list .item.item-bg-image { + max-height: 150px; + min-height: 150px; } + + .list .item.item-bg-image > img { + height: 100%; + left: 0; + max-width: initial; + opacity: 0.65; + filter: alpha(opacity=65); + position: absolute; + top: 0; + width: 100%; + z-index: 0; } + + .list a.item { + opacity: 1; + filter: alpha(opacity=100); } + + .list .item.item-bg-image h1, .list .item.item-bg-image h2, .list .item.item-bg-image h3, .list .item.item-bg-image h4, .list .item.item-bg-image h5, .list .item.item-bg-image h6 { + color: #fff; + font-weight: bold; + position: relative; + text-shadow: 0 0 3px rgba(0, 0, 0, 0.95); + z-index: 1; } + + .list .item.item-bg-image h2 { + font-size: 24px; } + + .list .item.item-bg-image h2 { + font-size: 24px; } + + .list .item.item-bg-image p { + color: white; + font-size: 17px; + position: relative; + text-shadow: 0 0 4px rgba(0, 0, 0, 0.95); + z-index: 1; } + + .item-avatar, .item-avatar .item-content, .item-avatar-left, .item-avatar-left .item-content { + min-height: 80px; } + + /* List: Thumbnails + ==================================*/ + .item-thumbnail-left, .card > .item.item-thumbnail-left, .item-thumbnail-left .item-content { + padding-left: 106px; } + + .item-thumbnail-right, .card > .item.item-thumbnail-right, .item-thumbnail-right .item-content { + padding-right: 106px; } + + /* List: Avatar + ==================================*/ + .item-avatar > img:first-child, .item-avatar .item-image, .item-avatar .item-content > img:first-child, .item-avatar .item-content .item-image, .item-avatar-left > img:first-child, .item-avatar-left .item-image, .item-avatar-left .item-content > img:first-child, .item-avatar-left .item-content .item-image { + border-radius: 50%; + left: 16px; + max-height: 40px; + max-width: 40px; } + + /* + .item-avatar, .list .item-avatar { + padding-left: 100px; + } + */ + .avatar, .item-avatar .avatar { + background-position: center; + background-size: cover; + border-radius: 50%; + display: inline-block; + height: 56px; + left: 16px; + position: absolute; + width: 56px; } + + /* List: Gallery + ==================================*/ + .list.half { + display: inline-block; + float: left; + margin: 0; + padding: 0; + width: 50%; } + + .list.half:first-child { + padding: 16px 8px 16px 16px; } + + .list.half:last-child { + padding: 16px 16px 16px 8px; } + + .list.half:first-child .card.card-gallery { + margin-left: 0; + margin-right: 0; } + + .list.half:last-child .card.card-gallery { + margin-left: 0; + margin-right: 0; } + + .list.condensed-space > .card, .list.condensed-space > .item { + margin: 0px 0px 2px; } + + .list .card.card-gallery { + display: block; + float: left; + margin: 0 0 0 13px; + padding: 0; + width: auto; } + + .list.half .item { + width: 100%; } + + .list.half .item.card { + margin-bottom: 16px; } + + .list .card.card-gallery.item h2 { + padding: 12px; } + + .list .item.item-gallery img { + width: 100%; } + + .item.item-divider { + border-top: solid 1px rgba(0, 0, 0, 0.12); + font-size: 14px; + font-weight: bold; + height: 48px; + line-height: 48px; + color: rgba(0, 0, 0, 0.54); } + .item.item-divider:first-child { + border: none; } + + .item-avatar, .item-avatar .item-content, .item-avatar-left, .item-avatar-left .item-content, .card > .item-avatar { + padding-left: 72px; } + + .item.active, .item.activated, .item-complex.active .item-content, .item-complex.activated .item-content, .item .item-content.active, .item .item-content.activated { + background-color: transparent; } + + .list-inset { + margin: 20px 30px; + border-left: solid 1px #ccc; + border-radius: 0; + background-color: #fff; } + + .list .item.item-floating-label, + .item-floating-label { + border-bottom: solid 1px #ccc; } + + .loader { + position: relative; + margin: 0px auto; + width: 100px; + height: 100px; + zoom: 1.7; } + + .circular { + -webkit-animation: rotate 2s linear infinite; + animation: rotate 2s linear infinite; + height: 100px; + position: relative; + width: 100px; } + + .path { + stroke-dasharray: 1,200; + stroke-dashoffset: 0; + -webkit-animation: dash 1.5s ease-in-out infinite, color 6s ease-in-out infinite; + animation: dash 1.5s ease-in-out infinite, color 6s ease-in-out infinite; + stroke-linecap: round; } + + @-webkit-keyframes rotate { + 100% { + -webkit-transform: rotate(360deg); + transform: rotate(360deg); } } + + @keyframes rotate { + 100% { + -webkit-transform: rotate(360deg); + transform: rotate(360deg); } } + + @-webkit-keyframes dash { + 0% { + stroke-dasharray: 1,200; + stroke-dashoffset: 0; } + 50% { + stroke-dasharray: 89,200; + stroke-dashoffset: -35; } + 100% { + stroke-dasharray: 89,200; + stroke-dashoffset: -124; } } + + @keyframes dash { + 0% { + stroke-dasharray: 1,200; + stroke-dashoffset: 0; } + 50% { + stroke-dasharray: 89,200; + stroke-dashoffset: -35; } + 100% { + stroke-dasharray: 89,200; + stroke-dashoffset: -124; } } + + @-webkit-keyframes color { + 100%, 0% { + stroke: #d62d20; } + 40% { + stroke: #0057e7; } + 66% { + stroke: #008744; } + 80%, 90% { + stroke: #ffa700; } } + + @keyframes color { + 100%, 0% { + stroke: #d62d20; } + 40% { + stroke: #0057e7; } + 66% { + stroke: #008744; } + 80%, 90% { + stroke: #ffa700; } } + + /* Layouts: Login + ==================================*/ + .login { + background-position: 25% 25%; + background-size: 180% 180%; + height: 100%; + -webkit-transition: all 1.5s ease-in-out; + transition: all 1.5s ease-in-out; } + + .login .item { + margin: 0 12px; + padding-left: 0; + padding-right: 0; + width: initial; } + + .login .button-bar { + bottom: 0; + margin: 28px 12px 0; + width: initial; } + + .login .light-bg { + background-color: #fff; } + + .icon.hero-icon:before, .hero-icon.icon-help:before, .hero-icon.icon-alert:before, #menu .footer .hero-icon.icon-help:before { + line-height: 130px; } + + /* Mask + ==================================*/ + .hero.has-mask:after, .item.has-mask:after, .card.has-mask:after { + content: ''; + background: -webkit-linear-gradient(top, transparent 0%, rgba(0, 0, 0, 0.6) 100%); + height: 100%; + left: 0; + position: absolute; + top: 0; + z-index: 0; + width: 100%; } + + .hero.has-mask-reverse:after, .item.has-mask-reverse:after, .card.has-mask-reverse:after { + content: ''; + background: -webkit-linear-gradient(top, rgba(0, 0, 0, 0.6) 0%, transparent 100%); + height: 100%; + left: 0; + position: absolute; + top: 0; + z-index: 0; + width: 100%; } + + /* Menu */ + .menu-bottom { + bottom: 16px; + left: 16px; + right: 16px; + position: absolute; } + + .menu-top { + top: 16px; + left: 16px; + right: 16px; + position: absolute; } + + .menu .avatar { + top: 16px; + left: 16px; + height: 65px; + width: 65px; } + + .menu .bar.bar-header.expanded { + box-shadow: none; + min-height: 150px; + color: #fff; } + + .menu-open .bar.bar-header.expanded { + background-position: 0; + background-size: 100%; } + + .has-expanded-header { + top: 150px !important; } + + .motion { + -webkit-transition: all 0.5s ease-out; + transition: all 0.5s ease-out; } + + .fade { + opacity: 0; + filter: alpha(opacity=0); + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transition: all 0.1s ease-out !important; + transition: all 0.1s ease-out !important; } + + .spin-back { + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transform: translateZ(0) rotate(360deg) scale(0) !important; + transform: translateZ(0) rotate(360deg) scale(0) !important; + -webkit-transition: all 0.1s ease-out !important; + transition: all 0.1s ease-out !important; } + + .spiral { + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transform: translateZ(0) rotate(-360deg) scale(0) translate(-120px) !important; + transform: translateZ(0) rotate(-360deg) scale(0) translate(-120px) !important; + -webkit-transition: all 0.1s ease-out !important; + transition: all 0.1s ease-out !important; } + + .spiral-back { + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transform: translateZ(0) rotate(360deg) scale(0) translate(120px) !important; + transform: translateZ(0) rotate(360deg) scale(0) translate(120px) !important; + -webkit-transition: all 0.1s ease-out !important; + transition: all 0.1s ease-out !important; } + + .menu-open .avatar { + opacity: 1; + filter: alpha(opacity=100); + -webkit-transform: translateZ(0) rotate(0) scale(1) !important; + transform: translateZ(0) rotate(0) scale(1) !important; + -webkit-transition: all 0.3s ease-out !important; + transition: all 0.3s ease-out !important; } + + .spin { + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transform: translateZ(0) rotate(0) scale(0) !important; + transform: translateZ(0) rotate(0) scale(0) !important; + -webkit-transition: all 0.3s ease-out !important; + transition: all 0.3s ease-out !important; } + + .spin.on { + -webkit-transform: translateZ(0) rotate(-360deg) scale(1) !important; + transform: translateZ(0) rotate(-360deg) scale(1) !important; } + + .flap { + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transform: translateZ(0) rotateX(0) scale(0) translate(-120px) !important; + transform: translateZ(0) rotateX(0) scale(0) translate(-120px) !important; + -webkit-transition: all 0.5s ease-out !important; + transition: all 0.5s ease-out !important; } + + .flap.on { + -webkit-transform: translateZ(0) rotateX(-720deg) scale(1) translate(0) !important; + transform: translateZ(0) rotateX(-720deg) scale(1) translate(0) !important; + -webkit-transition: all 0.5s ease-out !important; + transition: all 0.5s ease-out !important; } + + .drop { + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transform: translateZ(0) scale(3) !important; + transform: translateZ(0) scale(3) !important; + -webkit-transition: all 0.5s ease-out !important; + transition: all 0.5s ease-out !important; } + + .drop.on { + -webkit-transform: translateZ(0) scale(1) !important; + transform: translateZ(0) scale(1) !important; + -webkit-transition: all 0.5s ease-out !important; + transition: all 0.5s ease-out !important; } + + .flip { + -webkit-backface-visibility: hidden !important; + backface-visibility: hidden !important; + -webkit-transform: translateZ(0) rotateY(0) scale(0) !important; + transform: translateZ(0) rotateY(0) scale(0) !important; + -webkit-transition: all 0.5s ease-out !important; + transition: all 0.5s ease-out !important; } + + .flip.on { + -webkit-transform: translateZ(0) rotateY(-720deg) scale(1) !important; + transform: translateZ(0) rotateY(-720deg) scale(1) !important; + -webkit-transition: all 0.5s ease-out !important; + transition: all 0.5s ease-out !important; } + + /* Utilities + ==================================*/ + .bold { + font-weight: bold; } + + .static { + position: static; } + + .pull-left, .popover-helptip .icon.icon-left, .popover-helptip .icon-left.icon-help, .popover-helptip .icon-left.icon-alert, .popover-helptip #menu .footer .icon-left.icon-help, #menu .footer .popover-helptip .icon-left.icon-help, .popover-helptip .icon.icon-bottom-left, .popover-helptip .icon-bottom-left.icon-help, .popover-helptip .icon-bottom-left.icon-alert, .popover-helptip #menu .footer .icon-bottom-left.icon-help, #menu .footer .popover-helptip .icon-bottom-left.icon-help { + float: left; } + + .pull-right, .popover-helptip .icon.icon-right, .popover-helptip .icon-right.icon-help, .popover-helptip .icon-right.icon-alert, .popover-helptip #menu .footer .icon-right.icon-help, #menu .footer .popover-helptip .icon-right.icon-help, .popover-helptip .icon.icon-center, .popover-helptip .icon-center.icon-help, .popover-helptip .icon-center.icon-alert, .popover-helptip #menu .footer .icon-center.icon-help, #menu .footer .popover-helptip .icon-center.icon-help, .popover-helptip .icon.icon-bottom-right, .popover-helptip .icon-bottom-right.icon-help, .popover-helptip .icon-bottom-right.icon-alert, .popover-helptip #menu .footer .icon-bottom-right.icon-help, #menu .footer .popover-helptip .icon-bottom-right.icon-help, .popover-helptip .icon.icon-bottom-center, .popover-helptip .icon-bottom-center.icon-help, .popover-helptip .icon-bottom-center.icon-alert, .popover-helptip #menu .footer .icon-bottom-center.icon-help, #menu .footer .popover-helptip .icon-bottom-center.icon-help { + float: right; } + + .double-padding, .ionic-content.double-padding { + padding: 16px; } + + .double-padding-x { + padding-left: 16px; + padding-right: 16px; } + + .double-padding-y { + padding-top: 16px; + padding-bottom: 16px; } + + .outline { + border-style: solid; + border-width: 1px; } + + .border-top { + border-top: solid 1px #ccc; + padding-top: 30px; } + + .no-border { + border: none; } + + .circle { + border-radius: 50%; } + + .no-padding, .list.no-padding, .bar.no-padding, .button-bar.no-padding, .card.no-padding, .button.no-padding, .item.no-padding { + padding: 0; } + + .flat, .flat.tabs, .flat.button, .flat.button.icon, .flat.button.icon-help, .flat.button.icon-alert, #menu .footer .flat.button.icon-help, .flat.hero { + box-shadow: none; + -webkit-box-shadow: none; } + + /* Utilities : Padding + ==================================*/ + .im-wrapper, .padding, .item.large-button-bar { + padding: 16px !important; } + + .padding-bottom, .popover-share .bar-footer .button-close { + padding-bottom: 16px !important; } + + .padding-top { + padding-top: 16px !important; } + + .padding-left { + padding-left: 16px !important; } + + .padding-right, .popover-share .bar-footer .button-close { + padding-right: 16px !important; } + + .no-padding-bottom { + padding-bottom: 0 !important; } + + .no-padding-top { + padding-top: 0 !important; } + + .no-padding-left { + padding-left: 0 !important; } + + .no-padding-right { + padding-right: 0 !important; } + + /* Utilities : Depth + ==================================*/ + .z1 { + box-shadow: 0px 2px 5px 0 rgba(0, 0, 0, 0.26); } + + /* Utilities : Color + ==================================*/ + .bar.bar-positive.darker { + background-color: #164FAB; } + + /* TODO: Expand to other colors */ + .bar.bar-positive.dark-positive-bg { + background-color: #2C5CAD; } + + /* TODO: Expand to other colors */ + .muted { + color: #C3C3C3; } + + .clear-bg { + background: transparent; } + + /* Motion: Blinds + ==================================*/ + .animate-blinds .item, + .animate-blinds .item { + visibility: hidden; } + + .animate-blinds .item, + .animate-blinds .item { + -ms-transform: scale3d(0.8, 0, 1); + -webkit-transform: scale3d(0.8, 0, 1); + transform: scale3d(0.8, 0, 1); + -webkit-transition: -webkit-transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); + transition: transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); } + + .animate-blinds .item-bg-image > img.background, + .animate-blinds .item-bg-image > img.background { + box-shadow: none; + -ms-transform: scale3d(1, 1, 1); + -webkit-transform: scale3d(1, 1, 1); + transform: scale3d(1, 1, 1); } + + .animate-blinds .in, .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded .animate-blinds .item, .animate-blinds + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-blinds .expanded.item, .animate-blinds + .animate-fade-slide-in-right .expanded .item, + .animate-fade-slide-in-right .expanded .animate-blinds .item, .animate-blinds + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-blinds .expanded.item, .animate-blinds + .animate-ripple .expanded .item, + .animate-ripple .expanded .animate-blinds .item, .animate-blinds + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds .card.card-comment, + .animate-blinds.done > *, .animate-fade-slide-in .expanded .animate-blinds.item > *, + .animate-fade-slide-in .animate-blinds.expanded.item > *, + .animate-fade-slide-in-right .expanded .animate-blinds.item > *, + .animate-fade-slide-in-right .animate-blinds.expanded.item > *, + .animate-ripple .expanded .animate-blinds.item > *, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment > *, + .animate-blinds .in, + .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded + .animate-blinds .item, + .animate-blinds + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in + .animate-blinds .expanded.item, + .animate-blinds + .animate-fade-slide-in-right .expanded .item, + .animate-fade-slide-in-right .expanded + .animate-blinds .item, + .animate-blinds + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right + .animate-blinds .expanded.item, + .animate-blinds + .animate-ripple .expanded .item, + .animate-ripple .expanded + .animate-blinds .item, + .animate-blinds + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded + .animate-blinds .card.card-comment, + .animate-blinds.done > *, .animate-fade-slide-in .expanded .animate-blinds.item > *, + .animate-fade-slide-in .animate-blinds.expanded.item > *, + .animate-fade-slide-in-right .expanded .animate-blinds.item > *, + .animate-fade-slide-in-right .animate-blinds.expanded.item > *, + .animate-ripple .expanded .animate-blinds.item > *, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment > * { + -ms-transform: translate3d(0, 0, 0); + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); } + + .animate-blinds .in, .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded .animate-blinds .item, .animate-blinds + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-blinds .expanded.item, .animate-blinds + .animate-fade-slide-in-right .expanded .item, + .animate-fade-slide-in-right .expanded .animate-blinds .item, .animate-blinds + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-blinds .expanded.item, .animate-blinds + .animate-ripple .expanded .item, + .animate-ripple .expanded .animate-blinds .item, .animate-blinds + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds .card.card-comment, + .animate-blinds.done .item, .animate-fade-slide-in .expanded .animate-blinds.item .item, + .animate-fade-slide-in .animate-blinds.expanded.item .item, + .animate-fade-slide-in-right .expanded .animate-blinds.item .item, + .animate-fade-slide-in-right .animate-blinds.expanded.item .item, + .animate-ripple .expanded .animate-blinds.item .item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .item, + .animate-blinds .in, + .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded + .animate-blinds .item, + .animate-blinds + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in + .animate-blinds .expanded.item, + .animate-blinds + .animate-fade-slide-in-right .expanded .item, + .animate-fade-slide-in-right .expanded + .animate-blinds .item, + .animate-blinds + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right + .animate-blinds .expanded.item, + .animate-blinds + .animate-ripple .expanded .item, + .animate-ripple .expanded + .animate-blinds .item, + .animate-blinds + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded + .animate-blinds .card.card-comment, + .animate-blinds.done .item, .animate-fade-slide-in .expanded .animate-blinds.item .item, + .animate-fade-slide-in .animate-blinds.expanded.item .item, + .animate-fade-slide-in-right .expanded .animate-blinds.item .item, + .animate-fade-slide-in-right .animate-blinds.expanded.item .item, + .animate-ripple .expanded .animate-blinds.item .item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .item { + visibility: visible; } + + .animate-blinds .item, + .animate-blinds .item { + visibility: hidden; } + + .animate-blinds .item, + .animate-blinds .item { + opacity: 0; + filter: alpha(opacity=0); } + + .animate-blinds .in, .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded .animate-blinds .item, .animate-blinds + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-blinds .expanded.item, .animate-blinds + .animate-fade-slide-in-right .expanded .item, + .animate-fade-slide-in-right .expanded .animate-blinds .item, .animate-blinds + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-blinds .expanded.item, .animate-blinds + .animate-ripple .expanded .item, + .animate-ripple .expanded .animate-blinds .item, .animate-blinds + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds .card.card-comment, + .animate-blinds.done, + .animate-fade-slide-in .expanded .animate-blinds.item, + .animate-fade-slide-in .animate-blinds.expanded.item, + .animate-fade-slide-in-right .expanded .animate-blinds.item, + .animate-fade-slide-in-right .animate-blinds.expanded.item, + .animate-ripple .expanded .animate-blinds.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment, + .animate-blinds .in, + .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded + .animate-blinds .item, + .animate-blinds + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in + .animate-blinds .expanded.item, + .animate-blinds + .animate-fade-slide-in-right .expanded .item, + .animate-fade-slide-in-right .expanded + .animate-blinds .item, + .animate-blinds + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right + .animate-blinds .expanded.item, + .animate-blinds + .animate-ripple .expanded .item, + .animate-ripple .expanded + .animate-blinds .item, + .animate-blinds + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded + .animate-blinds .card.card-comment, + .animate-blinds.done, + .animate-fade-slide-in .expanded .animate-blinds.item, + .animate-fade-slide-in .animate-blinds.expanded.item, + .animate-fade-slide-in-right .expanded .animate-blinds.item, + .animate-fade-slide-in-right .animate-blinds.expanded.item, + .animate-ripple .expanded .animate-blinds.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment { + -ms-transform: scale3d(1, 1, 1); + -webkit-transform: scale3d(1, 1, 1); + transform: scale3d(1, 1, 1); + -webkit-transition: all 0.3s ease-in-out; + transition: all 0.3s ease-in-out; + opacity: 1; + filter: alpha(opacity=100); } + + .animate-blinds .in, .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded .animate-blinds .item, .animate-blinds + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-blinds .expanded.item, .animate-blinds + .animate-fade-slide-in-right .expanded .item, + .animate-fade-slide-in-right .expanded .animate-blinds .item, .animate-blinds + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-blinds .expanded.item, .animate-blinds + .animate-ripple .expanded .item, + .animate-ripple .expanded .animate-blinds .item, .animate-blinds + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds .card.card-comment, + .animate-blinds.done, + .animate-fade-slide-in .expanded .animate-blinds.item, + .animate-fade-slide-in .animate-blinds.expanded.item, + .animate-fade-slide-in-right .expanded .animate-blinds.item, + .animate-fade-slide-in-right .animate-blinds.expanded.item, + .animate-ripple .expanded .animate-blinds.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment, + .animate-blinds .in, + .animate-blinds .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded + .animate-blinds .item, + .animate-blinds + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in + .animate-blinds .expanded.item, + .animate-blinds + .animate-fade-slide-in-right .expanded .item, + .animate-fade-slide-in-right .expanded + .animate-blinds .item, + .animate-blinds + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right + .animate-blinds .expanded.item, + .animate-blinds + .animate-ripple .expanded .item, + .animate-ripple .expanded + .animate-blinds .item, + .animate-blinds + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded + .animate-blinds .card.card-comment, + .animate-blinds.done, + .animate-fade-slide-in .expanded .animate-blinds.item, + .animate-fade-slide-in .animate-blinds.expanded.item, + .animate-fade-slide-in-right .expanded .animate-blinds.item, + .animate-fade-slide-in-right .animate-blinds.expanded.item, + .animate-ripple .expanded .animate-blinds.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment { + visibility: visible; } + + .animate-blinds.done .in, .animate-fade-slide-in .expanded .animate-blinds.item .in, + .animate-fade-slide-in .animate-blinds.expanded.item .in, + .animate-fade-slide-in-right .expanded .animate-blinds.item .in, + .animate-fade-slide-in-right .animate-blinds.expanded.item .in, + .animate-ripple .expanded .animate-blinds.item .in, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .in, .animate-blinds.done .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded .animate-blinds.done .item, .animate-fade-slide-in .expanded .animate-blinds.item .item, .animate-fade-slide-in .animate-blinds.expanded.item .item, .animate-blinds.done + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-blinds.done .expanded.item, .animate-blinds.done + .animate-fade-slide-in-right .expanded .item, + .animate-fade-slide-in-right .expanded .animate-blinds.done .item, .animate-fade-slide-in-right .expanded .animate-blinds.item .item, .animate-fade-slide-in-right .animate-blinds.expanded.item .item, .animate-blinds.done + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-blinds.done .expanded.item, .animate-blinds.done + .animate-ripple .expanded .item, + .animate-ripple .expanded .animate-blinds.done .item, .animate-ripple .expanded .animate-blinds.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .item, .animate-blinds.done + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.done .card.card-comment, + .animate-fade-slide-in + .animate-ripple .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple + .animate-fade-slide-in .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-fade-slide-in-right + .animate-ripple .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple + .animate-fade-slide-in-right .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-blinds.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .card.card-comment, + .animate-blinds.done .in, .animate-fade-slide-in .expanded .animate-blinds.item .in, + .animate-fade-slide-in .animate-blinds.expanded.item .in, + .animate-fade-slide-in-right .expanded .animate-blinds.item .in, + .animate-fade-slide-in-right .animate-blinds.expanded.item .in, + .animate-ripple .expanded .animate-blinds.item .in, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .in, + .animate-blinds.done .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded + .animate-blinds.done .item, .animate-fade-slide-in .expanded .animate-blinds.item .item, .animate-fade-slide-in .animate-blinds.expanded.item .item, + .animate-blinds.done + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in + .animate-blinds.done .expanded.item, + .animate-blinds.done + .animate-fade-slide-in-right .expanded .item, + .animate-fade-slide-in-right .expanded + .animate-blinds.done .item, .animate-fade-slide-in-right .expanded .animate-blinds.item .item, .animate-fade-slide-in-right .animate-blinds.expanded.item .item, + .animate-blinds.done + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right + .animate-blinds.done .expanded.item, + .animate-blinds.done + .animate-ripple .expanded .item, + .animate-ripple .expanded + .animate-blinds.done .item, .animate-ripple .expanded .animate-blinds.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .item, + .animate-blinds.done + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded + .animate-blinds.done .card.card-comment, + .animate-fade-slide-in + .animate-ripple .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple + .animate-fade-slide-in .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-fade-slide-in-right + .animate-ripple .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple + .animate-fade-slide-in-right .animate-blinds.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-blinds.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .card.card-comment { + opacity: 1; + filter: alpha(opacity=100); } + + .animate-blinds .has-mask-reverse:after, + .animate-blinds .has-mask-reverse:after { + opacity: 0; + filter: alpha(opacity=0); + -webkit-transition: all 0.3s ease-in-out; + transition: all 0.3s ease-in-out; } + + .animate-blinds.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-blinds.item .has-mask-reverse:after, + .animate-fade-slide-in .animate-blinds.expanded.item .has-mask-reverse:after, + .animate-fade-slide-in-right .expanded .animate-blinds.item .has-mask-reverse:after, + .animate-fade-slide-in-right .animate-blinds.expanded.item .has-mask-reverse:after, + .animate-ripple .expanded .animate-blinds.item .has-mask-reverse:after, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .has-mask-reverse:after, + .animate-blinds.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-blinds.item .has-mask-reverse:after, + .animate-fade-slide-in .animate-blinds.expanded.item .has-mask-reverse:after, + .animate-fade-slide-in-right .expanded .animate-blinds.item .has-mask-reverse:after, + .animate-fade-slide-in-right .animate-blinds.expanded.item .has-mask-reverse:after, + .animate-ripple .expanded .animate-blinds.item .has-mask-reverse:after, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-blinds.card.card-comment .has-mask-reverse:after { + opacity: 1; + filter: alpha(opacity=100); } + + .animate-blinds .out, + .animate-blinds .out { + -ms-transform: scale3d(0, 0, 1); + -webkit-transform: scale3d(0, 0, 1); + transform: scale3d(0, 0, 1); } + + /* Motion: Pan In Left + ==================================*/ + .animate-pan-in-left, + .animate-pan-in-left { + background-position: 0% 0%; } + + /* Motion: Ripple + ==================================*/ + .animate-ripple .done, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-ripple .expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .done, .animate-fade-slide-in .expanded + .animate-ripple .item, + .animate-ripple + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in + .animate-ripple .expanded.item, + .animate-fade-slide-in-right .expanded + .animate-ripple .item, + .animate-ripple + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right + .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment { + visibility: hidden; } + + .animate-ripple .done, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-ripple .expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .done, .animate-fade-slide-in .expanded + .animate-ripple .item, + .animate-ripple + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in + .animate-ripple .expanded.item, + .animate-fade-slide-in-right .expanded + .animate-ripple .item, + .animate-ripple + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right + .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment { + -ms-transform: scale3d(0.8, 0, 1); + -webkit-transform: scale3d(0.8, 0, 1); + transform: scale3d(0.8, 0, 1); + -webkit-transition: -webkit-transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); + transition: transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); } + + .animate-ripple .item-bg-image img.background, + .animate-ripple .item-bg-image img.background { + box-shadow: none; + -ms-transform: scale3d(1, 1, 1); + -webkit-transform: scale3d(1, 1, 1); + transform: scale3d(1, 1, 1); } + + .animate-ripple .in, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-ripple .expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, + .animate-fade-slide-in .animate-ripple.expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple.item, + .animate-fade-slide-in-right .animate-ripple.expanded.item, + .animate-ripple .expanded .animate-ripple.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment, + .animate-ripple .in, .animate-fade-slide-in .expanded + .animate-ripple .item, + .animate-ripple + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in + .animate-ripple .expanded.item, + .animate-fade-slide-in-right .expanded + .animate-ripple .item, + .animate-ripple + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right + .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, + .animate-fade-slide-in .animate-ripple.expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple.item, + .animate-fade-slide-in-right .animate-ripple.expanded.item, + .animate-ripple .expanded .animate-ripple.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment { + -ms-transform: scale3d(1, 1, 1); + -webkit-transform: scale3d(1, 1, 1); + transform: scale3d(1, 1, 1); } + + .animate-ripple .in, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-ripple .expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, + .animate-fade-slide-in .animate-ripple.expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple.item, + .animate-fade-slide-in-right .animate-ripple.expanded.item, + .animate-ripple .expanded .animate-ripple.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment, + .animate-ripple .in, .animate-fade-slide-in .expanded + .animate-ripple .item, + .animate-ripple + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in + .animate-ripple .expanded.item, + .animate-fade-slide-in-right .expanded + .animate-ripple .item, + .animate-ripple + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right + .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, + .animate-fade-slide-in .animate-ripple.expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple.item, + .animate-fade-slide-in-right .animate-ripple.expanded.item, + .animate-ripple .expanded .animate-ripple.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment { + visibility: visible; } + + .animate-ripple .item { + -ms-transform: scale3d(0, 0, 1); + -webkit-transform: scale3d(0, 0, 1); + transform: scale3d(0, 0, 1); + opacity: 0; + filter: alpha(opacity=0); } + + .animate-ripple .item.in, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in .item.expanded, + .animate-fade-slide-in .animate-ripple .item.expanded, + .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in-right .item.expanded, + .animate-fade-slide-in-right .animate-ripple .item.expanded, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .item.card.card-comment { + opacity: 1; + filter: alpha(opacity=100); } + + .animate-ripple .done, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-ripple .expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment { + visibility: hidden; } + + .animate-ripple .done, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-ripple .expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .done, .animate-fade-slide-in .expanded + .animate-ripple .item, + .animate-ripple + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in + .animate-ripple .expanded.item, + .animate-fade-slide-in-right .expanded + .animate-ripple .item, + .animate-ripple + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right + .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment { + -ms-transform: scale3d(0.8, 0, 1); + -webkit-transform: scale3d(0.8, 0, 1); + transform: scale3d(0.8, 0, 1); + -webkit-transition: -webkit-transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); + transition: transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); } + + /* Uncomment if you want images to fade in after the card + + .animate-ripple .in .item-bg-image img:last-child, + .animate-ripple .in .item-bg-image img:last-child { + opacity: 0; + } + + .animate-ripple.done .item-bg-image img:last-child, + .animate-ripple.done .item-bg-image img:last-child { + opacity: 1; + -moz-transition: all 1s ease-in-out; + -o-transition: all 1s ease-in-out; + -webkit-transition: all 1s ease-in-out; + transition: all 1s ease-in-out; + } + + .animate-ripple .item-bg-image img:last-child, + .animate-ripple .item-bg-image img:last-child { + box-shadow: none; + -moz-transform: scale3d(1, 1, 1); + -ms-transform: scale3d(1, 1, 1); + -webkit-transform: scale3d(1, 1, 1); + transform: scale3d(1, 1, 1); + } + .animate-ripple .in .item-bg-image img:last-child, + .animate-ripple .in .item-bg-image img:last-child { + opacity: 0; + } + + .animate-ripple.done .item-bg-image img:last-child, + .animate-ripple.done .item-bg-image img:last-child { + opacity: 1; + -moz-transition: all 0.3s ease-in-out; + -o-transition: all 0.3s ease-in-out; + -webkit-transition: all 0.3s ease-in-out; + transition: all 0.3s ease-in-out; + } + + .animate-ripple .in, + .animate-ripple .in { + opacity: 0.6; + } + */ + .animate-ripple .in, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-ripple .expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, + .animate-fade-slide-in .animate-ripple.expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple.item, + .animate-fade-slide-in-right .animate-ripple.expanded.item, + .animate-ripple .expanded .animate-ripple.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment, .animate-ripple .in, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-ripple .expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, + .animate-fade-slide-in .animate-ripple.expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple.item, + .animate-fade-slide-in-right .animate-ripple.expanded.item, + .animate-ripple .expanded .animate-ripple.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment { + -ms-transform: scale3d(1, 1, 1); + -webkit-transform: scale3d(1, 1, 1); + transform: scale3d(1, 1, 1); + -webkit-transition: all 0.3s ease-in-out; + transition: all 0.3s ease-in-out; } + + .animate-ripple .in, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-ripple .expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, + .animate-fade-slide-in .animate-ripple.expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple.item, + .animate-fade-slide-in-right .animate-ripple.expanded.item, + .animate-ripple .expanded .animate-ripple.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment, .animate-ripple .in, .animate-fade-slide-in .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-ripple .expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple .item, .animate-ripple + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-ripple .expanded.item, .animate-ripple .expanded .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple.done, .animate-fade-slide-in .expanded .animate-ripple.item, + .animate-fade-slide-in .animate-ripple.expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple.item, + .animate-fade-slide-in-right .animate-ripple.expanded.item, + .animate-ripple .expanded .animate-ripple.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment { + visibility: visible; } + + .animate-ripple.done .in, .animate-fade-slide-in .expanded .animate-ripple.item .in, + .animate-fade-slide-in .animate-ripple.expanded.item .in, + .animate-fade-slide-in-right .expanded .animate-ripple.item .in, + .animate-fade-slide-in-right .animate-ripple.expanded.item .in, + .animate-ripple .expanded .animate-ripple.item .in, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .in, .animate-fade-slide-in .expanded .animate-ripple.done .item, .animate-fade-slide-in .expanded .animate-ripple.item .item, .animate-fade-slide-in .animate-ripple.expanded.item .item, .animate-ripple.done + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-ripple.done .expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple.done .item, .animate-fade-slide-in-right .expanded .animate-ripple.item .item, .animate-fade-slide-in-right .animate-ripple.expanded.item .item, .animate-ripple.done + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-ripple.done .expanded.item, .animate-ripple.done .expanded .item, .animate-ripple .expanded .animate-ripple.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .item, .animate-ripple.done .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-fade-slide-in .animate-ripple.expanded.item .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-fade-slide-in-right .animate-ripple.expanded.item .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-ripple.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .card.card-comment, .animate-ripple.done .in, .animate-fade-slide-in .expanded .animate-ripple.item .in, + .animate-fade-slide-in .animate-ripple.expanded.item .in, + .animate-fade-slide-in-right .expanded .animate-ripple.item .in, + .animate-fade-slide-in-right .animate-ripple.expanded.item .in, + .animate-ripple .expanded .animate-ripple.item .in, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .in, .animate-fade-slide-in .expanded .animate-ripple.done .item, .animate-fade-slide-in .expanded .animate-ripple.item .item, .animate-fade-slide-in .animate-ripple.expanded.item .item, .animate-ripple.done + .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in .animate-ripple.done .expanded.item, + .animate-fade-slide-in-right .expanded .animate-ripple.done .item, .animate-fade-slide-in-right .expanded .animate-ripple.item .item, .animate-fade-slide-in-right .animate-ripple.expanded.item .item, .animate-ripple.done + .animate-fade-slide-in-right .expanded.item, + .animate-fade-slide-in-right .animate-ripple.done .expanded.item, .animate-ripple.done .expanded .item, .animate-ripple .expanded .animate-ripple.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .item, .animate-ripple.done .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-fade-slide-in .animate-ripple.expanded.item .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-fade-slide-in-right .animate-ripple.expanded.item .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-ripple.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .card.card-comment { + opacity: 1; + filter: alpha(opacity=100); } + + .animate-ripple .has-mask-reverse:after, .animate-ripple .has-mask-reverse:after { + opacity: 0; + filter: alpha(opacity=0); + -webkit-transition: all 0.3s ease-in-out; + transition: all 0.3s ease-in-out; } + + .animate-ripple.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-ripple.item .has-mask-reverse:after, + .animate-fade-slide-in .animate-ripple.expanded.item .has-mask-reverse:after, + .animate-fade-slide-in-right .expanded .animate-ripple.item .has-mask-reverse:after, + .animate-fade-slide-in-right .animate-ripple.expanded.item .has-mask-reverse:after, + .animate-ripple .expanded .animate-ripple.item .has-mask-reverse:after, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .has-mask-reverse:after, .animate-ripple.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-ripple.item .has-mask-reverse:after, + .animate-fade-slide-in .animate-ripple.expanded.item .has-mask-reverse:after, + .animate-fade-slide-in-right .expanded .animate-ripple.item .has-mask-reverse:after, + .animate-fade-slide-in-right .animate-ripple.expanded.item .has-mask-reverse:after, + .animate-ripple .expanded .animate-ripple.item .has-mask-reverse:after, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-ripple.card.card-comment .has-mask-reverse:after { + opacity: 1; + filter: alpha(opacity=100); } + + .animate-ripple .out, .animate-ripple .out { + -ms-transform: scale3d(0, 0, 1); + -webkit-transform: scale3d(0, 0, 1); + transform: scale3d(0, 0, 1); } + + /* Motion: Slide / Fade In + ==================================*/ + .animate-fade-slide-in .item, + .animate-fade-slide-in .item { + visibility: hidden; } + + .animate-fade-slide-in .item, + .animate-fade-slide-in .item { + -ms-transform: scale3d(0.8, 0, 1); + -webkit-transform: scale3d(0.8, 0, 1); + transform: scale3d(0.8, 0, 1); + -webkit-transition: -webkit-transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); + transition: transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); } + + .animate-fade-slide-in .item-bg-image img.background, + .animate-fade-slide-in .item-bg-image img.background { + box-shadow: none; + -ms-transform: scale3d(1, 1, 1); + -webkit-transform: scale3d(1, 1, 1); + transform: scale3d(1, 1, 1); } + + .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in .item, + .animate-ripple .expanded .animate-fade-slide-in .item, .animate-fade-slide-in + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in .card.card-comment, + .animate-fade-slide-in.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .item, + .animate-fade-slide-in .animate-fade-slide-in.expanded.item .item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .item, + .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .item, + .animate-ripple .expanded .animate-fade-slide-in.item .item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .item, + .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in-right .expanded + .animate-fade-slide-in .item, + .animate-ripple .expanded + .animate-fade-slide-in .item, + .animate-fade-slide-in + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded + .animate-fade-slide-in .card.card-comment, + .animate-fade-slide-in.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .item, + .animate-fade-slide-in .animate-fade-slide-in.expanded.item .item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .item, + .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .item, + .animate-ripple .expanded .animate-fade-slide-in.item .item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .item { + -ms-transform: translate3d(0, 0, 0); + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); } + + .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in .item, + .animate-ripple .expanded .animate-fade-slide-in .item, .animate-fade-slide-in + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in .card.card-comment, + .animate-fade-slide-in.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .item, + .animate-fade-slide-in .animate-fade-slide-in.expanded.item .item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .item, + .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .item, + .animate-ripple .expanded .animate-fade-slide-in.item .item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .item, + .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in-right .expanded + .animate-fade-slide-in .item, + .animate-ripple .expanded + .animate-fade-slide-in .item, + .animate-fade-slide-in + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded + .animate-fade-slide-in .card.card-comment, + .animate-fade-slide-in.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .item, + .animate-fade-slide-in .animate-fade-slide-in.expanded.item .item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .item, + .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .item, + .animate-ripple .expanded .animate-fade-slide-in.item .item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .item { + visibility: visible; } + + .list .item.item-bg-image, + .list .item.item-bg-image { + max-height: 150px; } + + .animate-fade-slide-in .item, + .animate-fade-slide-in .item { + visibility: hidden; } + + .animate-fade-slide-in .item, + .animate-fade-slide-in .item { + -ms-transform: translate3d(-250px, 250px, 0); + -webkit-transform: translate3d(-250px, 250px, 0); + transform: translate3d(-250px, 250px, 0); + -webkit-transition: -webkit-transform 0.5s cubic-bezier(0.55, 0, 0.1, 1); + transition: transform 0.5s cubic-bezier(0.55, 0, 0.1, 1); + opacity: 0; + filter: alpha(opacity=0); } + + .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in .item, + .animate-ripple .expanded .animate-fade-slide-in .item, .animate-fade-slide-in + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in .card.card-comment, + .animate-fade-slide-in.done, + .animate-fade-slide-in .expanded .animate-fade-slide-in.item, + .animate-fade-slide-in .animate-fade-slide-in.expanded.item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item, + .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item, + .animate-ripple .expanded .animate-fade-slide-in.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment, + .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in-right .expanded + .animate-fade-slide-in .item, + .animate-ripple .expanded + .animate-fade-slide-in .item, + .animate-fade-slide-in + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded + .animate-fade-slide-in .card.card-comment, + .animate-fade-slide-in.done, + .animate-fade-slide-in .expanded .animate-fade-slide-in.item, + .animate-fade-slide-in .animate-fade-slide-in.expanded.item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item, + .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item, + .animate-ripple .expanded .animate-fade-slide-in.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment { + -ms-transform: scale3d(1, 1, 1); + -webkit-transform: scale3d(1, 1, 1); + transform: scale3d(1, 1, 1); + -webkit-transition: all 0.5s ease-in-out; + transition: all 0.5s ease-in-out; + opacity: 1; + filter: alpha(opacity=100); } + + .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in .item, + .animate-ripple .expanded .animate-fade-slide-in .item, .animate-fade-slide-in + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in .card.card-comment, + .animate-fade-slide-in.done, + .animate-fade-slide-in .expanded .animate-fade-slide-in.item, + .animate-fade-slide-in .animate-fade-slide-in.expanded.item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item, + .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item, + .animate-ripple .expanded .animate-fade-slide-in.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment, + .animate-fade-slide-in .in, .animate-fade-slide-in .expanded .item, .animate-fade-slide-in .expanded.item, + .animate-fade-slide-in-right .expanded + .animate-fade-slide-in .item, + .animate-ripple .expanded + .animate-fade-slide-in .item, + .animate-fade-slide-in + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded + .animate-fade-slide-in .card.card-comment, + .animate-fade-slide-in.done, + .animate-fade-slide-in .expanded .animate-fade-slide-in.item, + .animate-fade-slide-in .animate-fade-slide-in.expanded.item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item, + .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item, + .animate-ripple .expanded .animate-fade-slide-in.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment { + visibility: visible; } + + .animate-fade-slide-in.done .in, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .in, + .animate-fade-slide-in .animate-fade-slide-in.expanded.item .in, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .in, + .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .in, + .animate-ripple .expanded .animate-fade-slide-in.item .in, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .in, .animate-fade-slide-in.done .expanded .item, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .item, .animate-fade-slide-in .animate-fade-slide-in.expanded.item .item, .animate-fade-slide-in.done .expanded.item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in.done .item, .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .item, .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .item, + .animate-ripple .expanded .animate-fade-slide-in.done .item, .animate-ripple .expanded .animate-fade-slide-in.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .item, .animate-fade-slide-in.done + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.done .card.card-comment, + .animate-ripple .animate-fade-slide-in.expanded.item .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.expanded.item .card.card-comment, + .animate-fade-slide-in-right + .animate-ripple .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple + .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-fade-slide-in.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .card.card-comment, + .animate-fade-slide-in.done .in, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .in, + .animate-fade-slide-in .animate-fade-slide-in.expanded.item .in, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .in, + .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .in, + .animate-ripple .expanded .animate-fade-slide-in.item .in, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .in, .animate-fade-slide-in.done .expanded .item, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .item, .animate-fade-slide-in .animate-fade-slide-in.expanded.item .item, .animate-fade-slide-in.done .expanded.item, + .animate-fade-slide-in-right .expanded + .animate-fade-slide-in.done .item, .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .item, .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .item, + .animate-ripple .expanded + .animate-fade-slide-in.done .item, .animate-ripple .expanded .animate-fade-slide-in.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .item, + .animate-fade-slide-in.done + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded + .animate-fade-slide-in.done .card.card-comment, + .animate-ripple .animate-fade-slide-in.expanded.item .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.expanded.item .card.card-comment, + .animate-fade-slide-in-right + .animate-ripple .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple + .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-fade-slide-in.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .card.card-comment { + opacity: 1; + filter: alpha(opacity=100); } + + .animate-fade-slide-in .has-mask-reverse:after, + .animate-fade-slide-in .has-mask-reverse:after { + opacity: 0; + filter: alpha(opacity=0); + -webkit-transition: all 0.3s ease-in-out; + transition: all 0.3s ease-in-out; } + + .animate-fade-slide-in.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .has-mask-reverse:after, + .animate-fade-slide-in .animate-fade-slide-in.expanded.item .has-mask-reverse:after, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .has-mask-reverse:after, + .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .has-mask-reverse:after, + .animate-ripple .expanded .animate-fade-slide-in.item .has-mask-reverse:after, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .has-mask-reverse:after, + .animate-fade-slide-in.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-fade-slide-in.item .has-mask-reverse:after, + .animate-fade-slide-in .animate-fade-slide-in.expanded.item .has-mask-reverse:after, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in.item .has-mask-reverse:after, + .animate-fade-slide-in-right .animate-fade-slide-in.expanded.item .has-mask-reverse:after, + .animate-ripple .expanded .animate-fade-slide-in.item .has-mask-reverse:after, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in.card.card-comment .has-mask-reverse:after { + opacity: 1; + filter: alpha(opacity=100); } + + .animate-fade-slide-in .out, + .animate-fade-slide-in .out { + -ms-transform: scale3d(0, 0, 1); + -webkit-transform: scale3d(0, 0, 1); + transform: scale3d(0, 0, 1); } + + /* Motion: Slide In Right + ==================================*/ + .animate-fade-slide-in-right .item, + .animate-fade-slide-in-right .item { + visibility: hidden; } + + .animate-fade-slide-in-right .item, + .animate-fade-slide-in-right .item { + -ms-transform: scale3d(0.8, 0, 1); + -webkit-transform: scale3d(0.8, 0, 1); + transform: scale3d(0.8, 0, 1); + -webkit-transition: -webkit-transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); + transition: transform 0.3s cubic-bezier(0.55, 0, 0.1, 1); } + + .animate-fade-slide-in-right .item-bg-image > img.background, + .animate-fade-slide-in-right .item-bg-image > img.background { + box-shadow: none; + -ms-transform: scale3d(1, 1, 1); + -webkit-transform: scale3d(1, 1, 1); + transform: scale3d(1, 1, 1); } + + .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, + .animate-ripple .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right .card.card-comment, + .animate-fade-slide-in-right.done > *, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item > *, + .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item > *, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item > *, + .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item > *, + .animate-ripple .expanded .animate-fade-slide-in-right.item > *, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment > *, + .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded + .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, + .animate-ripple .expanded + .animate-fade-slide-in-right .item, + .animate-fade-slide-in-right + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded + .animate-fade-slide-in-right .card.card-comment, + .animate-fade-slide-in-right.done > *, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item > *, + .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item > *, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item > *, + .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item > *, + .animate-ripple .expanded .animate-fade-slide-in-right.item > *, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment > * { + -ms-transform: translate3d(0, 0, 0); + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); } + + .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, + .animate-ripple .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right .card.card-comment, + .animate-fade-slide-in-right.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .item, + .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .item, + .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .item, + .animate-ripple .expanded .animate-fade-slide-in-right.item .item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .item, + .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded + .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, + .animate-ripple .expanded + .animate-fade-slide-in-right .item, + .animate-fade-slide-in-right + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded + .animate-fade-slide-in-right .card.card-comment, + .animate-fade-slide-in-right.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .item, + .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .item, + .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .item, + .animate-ripple .expanded .animate-fade-slide-in-right.item .item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .item { + visibility: visible; } + + .animate-fade-slide-in-right .item, + .animate-fade-slide-in-right .item { + visibility: hidden; } + + .animate-fade-slide-in-right .item, + .animate-fade-slide-in-right .item { + -ms-transform: translate3d(250px, 250px, 0); + -webkit-transform: translate3d(250px, 250px, 0); + transform: translate3d(250px, 250px, 0); + -webkit-transition: -webkit-transform 0.5s cubic-bezier(0.55, 0, 0.1, 1); + transition: transform 0.5s cubic-bezier(0.55, 0, 0.1, 1); + opacity: 0; + filter: alpha(opacity=0); } + + .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, + .animate-ripple .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right .card.card-comment, + .animate-fade-slide-in-right.done, + .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item, + .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item, + .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item, + .animate-ripple .expanded .animate-fade-slide-in-right.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment, + .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded + .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, + .animate-ripple .expanded + .animate-fade-slide-in-right .item, + .animate-fade-slide-in-right + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded + .animate-fade-slide-in-right .card.card-comment, + .animate-fade-slide-in-right.done, + .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item, + .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item, + .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item, + .animate-ripple .expanded .animate-fade-slide-in-right.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment { + -ms-transform: scale3d(1, 1, 1); + -webkit-transform: scale3d(1, 1, 1); + transform: scale3d(1, 1, 1); + -webkit-transition: all 0.3s ease-in-out; + transition: all 0.3s ease-in-out; + opacity: 1; + filter: alpha(opacity=100); } + + .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, + .animate-ripple .expanded .animate-fade-slide-in-right .item, .animate-fade-slide-in-right + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right .card.card-comment, + .animate-fade-slide-in-right.done, + .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item, + .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item, + .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item, + .animate-ripple .expanded .animate-fade-slide-in-right.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment, + .animate-fade-slide-in-right .in, .animate-fade-slide-in .expanded + .animate-fade-slide-in-right .item, .animate-fade-slide-in-right .expanded .item, .animate-fade-slide-in-right .expanded.item, + .animate-ripple .expanded + .animate-fade-slide-in-right .item, + .animate-fade-slide-in-right + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded + .animate-fade-slide-in-right .card.card-comment, + .animate-fade-slide-in-right.done, + .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item, + .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item, + .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item, + .animate-ripple .expanded .animate-fade-slide-in-right.item, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment { + visibility: visible; } + + .animate-fade-slide-in-right.done .in, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .in, + .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .in, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .in, + .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .in, + .animate-ripple .expanded .animate-fade-slide-in-right.item .in, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .in, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .item, .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .item, .animate-fade-slide-in-right.done .expanded .item, .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .item, .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .item, .animate-fade-slide-in-right.done .expanded.item, + .animate-ripple .expanded .animate-fade-slide-in-right.done .item, .animate-ripple .expanded .animate-fade-slide-in-right.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .item, .animate-fade-slide-in-right.done + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.done .card.card-comment, + .animate-fade-slide-in + .animate-ripple .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple + .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .animate-fade-slide-in-right.expanded.item .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.expanded.item .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-fade-slide-in-right.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .card.card-comment, + .animate-fade-slide-in-right.done .in, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .in, + .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .in, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .in, + .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .in, + .animate-ripple .expanded .animate-fade-slide-in-right.item .in, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .in, .animate-fade-slide-in .expanded + .animate-fade-slide-in-right.done .item, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .item, .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .item, .animate-fade-slide-in-right.done .expanded .item, .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .item, .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .item, .animate-fade-slide-in-right.done .expanded.item, + .animate-ripple .expanded + .animate-fade-slide-in-right.done .item, .animate-ripple .expanded .animate-fade-slide-in-right.item .item, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .item, + .animate-fade-slide-in-right.done + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded + .animate-fade-slide-in-right.done .card.card-comment, + .animate-fade-slide-in + .animate-ripple .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple + .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .animate-fade-slide-in-right.expanded.item .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, + .animate-ripple .animate-fade-slide-in-right.expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.expanded.item .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .animate-fade-slide-in-right.item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .card.card-comment, .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .card.card-comment { + opacity: 1; + filter: alpha(opacity=100); } + + .animate-fade-slide-in-right .has-mask-reverse:after, + .animate-fade-slide-in-right .has-mask-reverse:after { + opacity: 0; + filter: alpha(opacity=0); + -webkit-transition: all 0.3s ease-in-out; + transition: all 0.3s ease-in-out; } + + .animate-fade-slide-in-right.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .has-mask-reverse:after, + .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .has-mask-reverse:after, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .has-mask-reverse:after, + .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .has-mask-reverse:after, + .animate-ripple .expanded .animate-fade-slide-in-right.item .has-mask-reverse:after, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .has-mask-reverse:after, + .animate-fade-slide-in-right.done .has-mask-reverse:after, .animate-fade-slide-in .expanded .animate-fade-slide-in-right.item .has-mask-reverse:after, + .animate-fade-slide-in .animate-fade-slide-in-right.expanded.item .has-mask-reverse:after, + .animate-fade-slide-in-right .expanded .animate-fade-slide-in-right.item .has-mask-reverse:after, + .animate-fade-slide-in-right .animate-fade-slide-in-right.expanded.item .has-mask-reverse:after, + .animate-ripple .expanded .animate-fade-slide-in-right.item .has-mask-reverse:after, + .animate-ripple .expanded.item .card-avatar-small.expanded .item .card-avatar-small .expanded .item .card-avatar-small.expanded .card.card-comment .card-avatar-small .expanded .animate-fade-slide-in-right.card.card-comment .has-mask-reverse:after { + opacity: 1; + filter: alpha(opacity=100); } + + .animate-fade-slide-in-right .out, + .animate-fade-slide-in-right .out { + -ms-transform: scale3d(0, 0, 1); + -webkit-transform: scale3d(0, 0, 1); + transform: scale3d(0, 0, 1); } + + /* Motion: Slide Up + ==================================*/ + .slide-up, + .slide-up, + .hero.slide-up { + height: 100%; + overflow: hidden; + text-align: center; } + + .slide-up { + -webkit-transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); + transition: all 1s cubic-bezier(0.55, 0, 0.1, 1); + -webkit-transform: translate3d(0, 0, 0); + transform: translate3d(0, 0, 0); } + + .slide-up *, + .slide-up *, + .hero.slide-up * { + opacity: 0; + filter: alpha(opacity=0); } + + .hero.slide-up + .mid-bar, + .slide-up + .mid-bar, + .slide-up + .mid-bar { + height: 100%; + opacity: 0.7; + filter: alpha(opacity=70); + -webkit-transform: translate3d(100%, -240px, 0); + transform: translate3d(100%, -240px, 0); } + + /*! + * Waves v0.5.4 + * http://fian.my.id/Waves + * + * Copyright 2014 Alfiana E. Sibuea and other contributors + * Forked by Zach Fitzgerald and other contributors for Ionic Material + * + * Released under the MIT license + * https://github.com/fians/Waves/blob/master/LICENSE + * + */ + .ink, .button-fab, .button-flat, .button-raised, .button-clear, .button-text, .popup .button { + position: relative; + cursor: pointer; + /*display: inline-block;*/ + overflow: hidden; + -webkit-user-select: none; + -moz-user-select: none; + -ms-user-select: none; + user-select: none; + -webkit-tap-highlight-color: transparent; + -webkit-transition: all 0.3s ease-out; + -moz-transition: all 0.3s ease-out; + -o-transition: all 0.3s ease-out; + transition: all 0.3s ease-out; } + + .ink-ripple { + position: absolute; + border-radius: 50%; + width: 100px; + height: 100px; + margin-top: -50px; + margin-left: -50px; + opacity: 0; + background-color: rgba(255, 255, 255, 0.4); + -webkit-transition: all 0.5s ease-out; + -moz-transition: all 0.5s ease-out; + -o-transition: all 0.5s ease-out; + transition: all 0.5s ease-out; + -webkit-transition-property: -webkit-transform, opacity; + -moz-transition-property: -moz-transform, opacity; + -o-transition-property: -o-transform, opacity; + transition-property: transform, opacity; + -webkit-transform: scale(0); + -moz-transform: scale(0); + -ms-transform: scale(0); + -o-transform: scale(0); + transform: scale(0); + pointer-events: none; } + + .ink-notransition { + -webkit-transition: none !important; + -moz-transition: none !important; + -o-transition: none !important; + transition: none !important; } + + .button-fab, + .button-flat, + .button-clear, + .button-text, + .button-raised, + .ink-button, + .ink-circle { + -webkit-transform: translateZ(0); + -moz-transform: translateZ(0); + -ms-transform: translateZ(0); + -o-transform: translateZ(0); + transform: translateZ(0); } + + .button-fab.activated, + .button-flat.activated, + .button-raised.activated, + .button-clear.activated, + .activated.button-text, + .ink-button.activated, + .ink.activated, + .ink-circle.activated, + .popup .button.activated, + .button-fab:active, + .button-flat:active, + .button-raised:active, + .button-clear:active, + .button-text:active, + .ink-button:active, + .ink:active, + .ink-circle:active, + .popup .button:active { + -webkit-mask-image: -webkit-radial-gradient(circle, #ffffff 100%, #000000 100%); } + + .ink-button, + .ink-button:visited, + .ink-button:link, + .button-fab, + .button-fab:visited, + .button-fab:link, + .button-flat, + .button-flat:visited, + .button-flat:link, + .button-raised, + .button-raised:visited, + .button-raised:link, + .button-clear, + .button-text, + .button-clear:visited, + .button-text:visited, + .button-clear:link, + .button-text:link { + white-space: nowrap; + vertical-align: middle; + cursor: pointer; + border: none; + outline: none; + /* color: inherit; */ + /* background-color: rgba(0, 0, 0, 0); */ + font-size: 14px; + text-align: center; + text-decoration: none; + z-index: 1; } + + + /* Ionic Overrides + ==================================*/ + * { + font-family: "RobotoDraft","Roboto","Helvetica Neue", "Segoe UI", sans-serif; } + + .rounded { + border-radius: 4px; } + + a { + cursor: pointer; } + + .has-header.expanded { + /* Expanded modifier */ + top: 76px; } + + /* Bar Overrides + ==================================*/ + .bar { + border-bottom: none; + padding: 0; } + + .bar .button { + min-height: 44px; + min-width: 44px; + max-width: 48px; + margin-bottom: 0; + max-height: 44px; + width: 48px; } + + .bar .title + .buttons.buttons-right { + right: 0; + top: 0; } + + /* Title Overrides + ==================================*/ + .title-left, + .title.title-left { + left: 48px; } + + .title-right, + .title.title-right { + left: 48px; } + + /* Background Colors + ==================================*/ + .positive-bg, + .button-positive, + .button-text, + .bar .button-positive, + .bar .button-text, + .header-positive, + .button-bar-positive, + .bar-positive, + .positive-border, + .positive-bg:hover, + .bar .button-positive:hover, + .bar .button-text:hover, + .button-positive:hover, + .button-text:hover, + .header-positive:hover, + .button-bar-positive:hover, + .bar-positive:hover, + .positive-border:hover, + .positive-bg:active, + .bar .button-positive:active, + .bar .button-text:active, + .button-positive:active, + .button-text:active, + .header-positive:active, + .button-bar-positive:active, + .bar-positive:active, + .positive-border:active, + .positive-bg.activated, + .bar .button-positive.activated, + .bar .activated.button-text, + .button-positive.activated, + .activated.button-text, + .header-positive.activated, + .button-bar-positive.activated, + .bar-positive.activated, + .positive-border.activated { + background-color: #3F51B5; + color: #fff; } + + .positive-900-bg, + .button-positive-900, + .bar .button-positive-900, + .header-positive-900, + .button-bar-positive-900, + .bar-positive-900, + .positive-900-border, + .positive-900-bg:hover, + .button-positive-900:hover, + .bar .button-positive-900:hover, + .header-positive-900:hover, + .button-bar-positive-900:hover, + .bar-positive-900:hover, + .positive-900-border:hover, + .positive-900-bg:active, + .bar .button-positive-900:active, + .button-positive-900:active, + .header-positive-900:active, + .button-bar-positive-900:active, + .bar-positive-900:active, + .positive-900-border:active, + .positive-900-bg.activated, + .button-positive-900.activated, + .bar .button-positive-900.activated, + .header-positive-900.activated, + .button-bar-positive-900.activated, + .bar-positive-900.activated, + .positive-900-border.activated { + background-color: #1A237E; + color: #fff; } + + .positive-100-bg, + .button-positive-100, + .bar .button-positive-100, + .header-positive-100, + .button-bar-positive-100, + .bar-positive-100, + .positive-100-border, + .positive-100-bg:hover, + .button-positive-100:hover, + .bar .button-positive-100:hover, + .header-positive-100:hover, + .button-bar-positive-100:hover, + .bar-positive-100:hover, + .positive-100-border:hover, + .positive-100-bg:active, + .button-positive-100:active, + .bar .button-positive-100:active, + .header-positive-100:active, + .button-bar-positive-100:active, + .bar-positive-100:active, + .positive-100-border:active, + .positive-100-bg.activated, + .button-positive-100.activated, + .bar .button-positive-100.activated, + .header-positive-100.activated, + .button-bar-positive-100.activated, + .bar-positive-100.activated, + .positive-100-border.activated { + background-color: #C5CAE9; + color: #fff; } + + .calm-bg, + .button-calm, + .bar .button-calm, + .header-calm, + .button-bar-calm, + .bar-calm, + .calm-border, + .calm-bg:hover, + .button-calm:hover, + .bar .button-calm:hover, + .header-calm:hover, + .button-bar-calm:hover, + .bar-calm:hover, + .calm-border:hover, + .calm-bg:active, + .button-calm:active, + .bar .button-calm:active, + .header-calm:active, + .button-bar-calm:active, + .bar-calm:active, + .calm-border:active, + .calm-bg.activated, + .button-calm.activated, + .bar .button-calm.activated, + .header-calm.activated, + .button-bar-calm.activated, + .bar-calm.activated, + .calm-border.activated { + background-color: #2196F3; + color: #fff; } + + .calm-900-bg, + .button-calm-900, + .bar .button-calm-900, + .header-calm-900, + .button-bar-calm-900, + .bar-calm-900, + .calm-900-border, + .calm-900-bg:hover, + .button-calm-900:hover, + .bar .button-calm-900:hover, + .header-calm-900:hover, + .button-bar-calm-900:hover, + .bar-calm-900:hover, + .calm-900-border:hover, + .calm-900-bg:active, + .button-calm-900:active, + .bar .button-calm-900:active, + .header-calm-900:active, + .button-bar-calm-900:active, + .bar-calm-900:active, + .calm-900-border:active, + .calm-900-bg.activated, + .button-calm-900.activated, + .bar .button-calm-900.activated, + .header-calm-900.activated, + .button-bar-calm-900.activated, + .bar-calm-900.activated, + .calm-900-border.activated { + background-color: #0D47A1; + color: #fff; } + + .calm-100-bg, + .button-calm-100, + .bar .button-calm-100, + .header-calm-100, + .button-bar-calm-100, + .bar-calm-100, + .calm-100-border, + .calm-100-bg:hover, + .button-calm-100:hover, + .bar .button-calm-100:hover, + .header-calm-100:hover, + .button-bar-calm-100:hover, + .bar-calm-100:hover, + .calm-100-border:hover, + .calm-100-bg:active, + .button-calm-100:active, + .bar .button-calm-100:active, + .header-calm-100:active, + .button-bar-calm-100:active, + .bar-calm-100:active, + .calm-100-border:active, + .calm-100-bg.activated, + .button-calm-100.activated, + .bar .button-calm-100.activated, + .header-calm-100.activated, + .button-bar-calm-100.activated, + .bar-calm-100.activated, + .calm-100-border.activated { + background-color: #BBDEFB; + color: #fff; } + + .royal-bg, + .button-royal, + .bar .button-royal, + .header-royal, + .button-bar-royal, + .bar-royal, + .royal-border, + .royal-bg:hover, + .button-royal:hover, + .bar .button-royal:hover, + .header-royal:hover, + .button-bar-royal:hover, + .bar-royal:hover, + .royal-border:hover, + .royal-bg:active, + .button-royal:active, + .bar .button-royal:active, + .header-royal:active, + .button-bar-royal:active, + .bar-royal:active, + .royal-border:active, + .royal-bg.activated, + .button-royal.activated, + .bar .button-royal.activated, + .header-royal.activated, + .button-bar-royal.activated, + .bar-royal.activated, + .royal-border.activated { + background-color: #673AB7; + color: #fff; } + + .royal-900-bg, + .button-royal-900, + .bar .button-royal-900, + .header-royal-900, + .button-bar-royal-900, + .bar-royal-900, + .royal-900-border, + .royal-900-bg:hover, + .button-royal-900:hover, + .bar .button-royal-900:hover, + .header-royal-900:hover, + .button-bar-royal-900:hover, + .bar-royal-900:hover, + .royal-900-border:hover, + .royal-900-bg:active, + .button-royal-900:active, + .bar .button-royal-900:active, + .header-royal-900:active, + .button-bar-royal-900:active, + .bar-royal-900:active, + .royal-900-border:active, + .royal-900-bg.activated, + .button-royal-900.activated, + .bar .button-royal-900.activated, + .header-royal-900.activated, + .button-bar-royal-900.activated, + .bar-royal-900.activated, + .royal-900-border.activated { + background-color: #311B92; + color: #fff; } + + .royal-100-bg, + .button-royal-100, + .bar .button-royal-100, + .header-royal-100, + .button-bar-royal-100, + .bar-royal-100, + .royal-100-border, + .royal-100-bg:hover, + .button-royal-100:hover, + .bar .button-royal-100:hover, + .header-royal-100:hover, + .button-bar-royal-100:hover, + .bar-royal-100:hover, + .royal-100-border:hover, + .royal-100-bg:active, + .button-royal-100:active, + .bar .button-royal-100:active, + .header-royal-100:active, + .button-bar-royal-100:active, + .bar-royal-100:active, + .royal-100-border:active, + .royal-100-bg.activated, + .button-royal-100.activated, + .bar .button-royal-100.activated, + .header-royal-100.activated, + .button-bar-royal-100.activated, + .bar-royal-100.activated, + .royal-100-border.activated { + background-color: #D1C4E9; + color: #fff; } + + .balanced-bg, + .button-balanced, + .bar .button-balanced, + .header-balanced, + .button-bar-balanced, + .bar-balanced, + .balanced-border, + .balanced-bg:hover, + .button-balanced:hover, + .bar .button-balanced:hover, + .header-balanced:hover, + .button-bar-balanced:hover, + .bar-balanced:hover, + .balanced-border:hover, + .balanced-bg:active, + .button-balanced:active, + .bar .button-balanced:active, + .header-balanced:active, + .button-bar-balanced:active, + .bar-balanced:active, + .balanced-border:active, + .balanced-bg.activated, + .button-balanced.activated, + .bar .button-balanced.activated, + .header-balanced.activated, + .button-bar-balanced.activated, + .bar-balanced.activated, + .balanced-border.activated { + background-color: #4CAF50; + color: #fff; } + + .balanced-900-bg, + .button-balanced-900, + .bar .button-balanced-900, + .header-balanced-900, + .button-bar-balanced-900, + .bar-balanced-900, + .balanced-900-border, + .balanced-900-bg:hover, + .button-balanced-900:hover, + .bar .button-balanced-900:hover, + .header-balanced-900:hover, + .button-bar-balanced-900:hover, + .bar-balanced-900:hover, + .balanced-900-border:hover, + .balanced-900-bg:active, + .button-balanced-900:active, + .bar .button-balanced-900:active, + .header-balanced-900:active, + .button-bar-balanced-900:active, + .bar-balanced-900:active, + .balanced-900-border:active, + .balanced-900-bg.activated, + .button-balanced-900.activated, + .bar .button-balanced-900.activated, + .header-balanced-900.activated, + .button-bar-balanced-900.activated, + .bar-balanced-900.activated, + .balanced-900-border.activated { + background-color: #1B5E20; + color: #fff; } + + .balanced-100-bg, + .button-balanced-100, + .bar .button-balanced-100, + .header-balanced-100, + .button-bar-balanced-100, + .bar-balanced-100, + .balanced-100-border, + .balanced-100-bg:hover, + .button-balanced-100:hover, + .bar .balanced-100-bg:hover, + .header-balanced-100:hover, + .button-bar-balanced-100:hover, + .bar-balanced-100:hover, + .balanced-100-border:hover, + .balanced-100-bg:active, + .button-balanced-100:active, + .bar .button-balanced-100:active, + .header-balanced-100:active, + .button-bar-balanced-100:active, + .bar-balanced-100:active, + .balanced-100-border:active, + .balanced-100-bg.activated, + .button-balanced-100.activated, + .bar .button-balanced-100.activated, + .header-balanced-100.activated, + .button-bar-balanced-100.activated, + .bar-balanced-100.activated, + .balanced-100-border.activated { + background-color: #C8E6C9; + color: #fff; } + + .energized-bg, + .button-energized, + .bar .button-energized, + .header-energized, + .button-bar-energized, + .bar-energized, + .energized-border, + .energized-bg:hover, + .button-energized:hover, + .bar .button-energized:hover, + .header-energized:hover, + .button-bar-energized:hover, + .bar-energized:hover, + .energized-border:hover, + .energized-bg:active, + .button-energized:active, + .bar .button-energized:active, + .header-energized:active, + .button-bar-energized:active, + .bar-energized:active, + .energized-border:active, + .energized-bg.activated, + .button-energized.activated, + .bar .button-energized.activated, + .header-energized.activated, + .button-bar-energized.activated, + .bar-energized.activated, + .energized-border.activated { + background-color: #FF9800; + color: #fff; } + + .energized-900-bg, + .button-energized-900, + .bar .button-energized-900, + .header-energized-900, + .button-bar-energized-900, + .bar-energized-900, + .energized-900-border, + .energized-900-bg:hover, + .button-energized-900:hover, + .bar .button-energized-900:hover, + .header-energized-900:hover, + .button-bar-energized-900:hover, + .bar-energized-900:hover, + .energized-900-border:hover, + .energized-900-bg:active, + .button-energized-900:active, + .bar .button-energized-900:active, + .header-energized-900:active, + .button-bar-energized-900:active, + .bar-energized-900:active, + .energized-900-border:active, + .energized-900-bg.activated, + .button-energized-900.activated, + .bar .button-energized-900.activated, + .header-energized-900.activated, + .button-bar-energized-900.activated, + .bar-energized-900.activated, + .energized-900-border.activated { + background-color: #E65100; + color: #fff; } + + .energized-100-bg, + .button-energized-100, + .bar .button-energized-100, + .header-energized-100, + .button-bar-energized-100, + .bar-energized-100, + .energized-100-border, + .energized-100-bg:hover, + .button-energized-100:hover, + .bar .button-energized-100:hover, + .header-energized-100:hover, + .button-bar-energized-100:hover, + .bar-energized-100:hover, + .energized-100-border:hover, + .energized-100-bg:active, + .button-energized-100:active, + .bar .button-energized-100:active, + .header-energized-100:active, + .button-bar-energized-100:active, + .bar-energized-100:active, + .energized-100-border:active, + .energized-100-bg.activated, + .button-energized-100.activated, + .bar .button-energized-100.activated, + .header-energized-100.activated, + .button-bar-energized-100.activated, + .bar-energized-100.activated, + .energized-100-border.activated { + background-color: #FFE0B2; } + + .assertive-bg, + .button-assertive, + .bar .button-assertive, + .header-assertive, + .button-bar-assertive, + .bar-assertive, + .assertive-border, + .assertive-bg:hover, + .button-assertive:hover, + .bar .button-assertive:hover, + .header-assertive:hover, + .button-bar-assertive:hover, + .bar-assertive:hover, + .assertive-border:hover, + .assertive-bg:active, + .button-assertive:active, + .bar .button-assertive:active, + .header-assertive:active, + .button-bar-assertive:active, + .bar-assertive:active, + .assertive-border:active, + .assertive-bg.activated, + .button-assertive.activated, + .bar .button-assertive.activated, + .header-assertive.activated, + .button-bar-assertive.activated, + .bar-assertive.activated, + .assertive-border.activated { + background-color: #F44336; + color: #fff; } + + .assertive-900-bg, + .button-assertive-900, + .bar .button-assertive-900, + .header-assertive-900, + .button-bar-assertive-900, + .bar-assertive-900, + .assertive-900-border, + .assertive-900-bg:hover, + .button-assertive-900:hover, + .bar .button-assertive-900:hover, + .header-assertive-900:hover, + .button-bar-assertive-900:hover, + .bar-assertive-900:hover, + .assertive-900-border:hover, + .assertive-900-bg:active, + .button-assertive-900:active, + .bar .button-assertive-900:active, + .header-assertive-900:active, + .button-bar-assertive-900:active, + .bar-assertive-900:active, + .assertive-900-border:active, + .assertive-900-bg.activated, + .button-assertive-900.activated, + .bar .button-assertive-900.activated, + .header-assertive-900.activated, + .button-bar-assertive-900.activated, + .bar-assertive-900.activated, + .assertive-900-border.activated { + background-color: #B71C1C; + color: #fff; } + + .assertive-100-bg, + .button-assertive-100, + .bar .button-assertive-100, + .header-assertive-100, + .button-bar-assertive-100, + .bar-assertive-100, + .assertive-100-border, + .assertive-100-bg:hover, + .button-assertive-100:hover, + .bar .button-assertive-100:hover, + .header-assertive-100:hover, + .button-bar-assertive-100:hover, + .bar-assertive-100:hover, + .assertive-100-border:hover, + .assertive-100-bg:active, + .button-assertive-100:active, + .bar .button-assertive-100:active, + .header-assertive-100:active, + .button-bar-assertive-100:active, + .bar-assertive-100:active, + .assertive-100-border:active, + .assertive-100-bg.activated, + .bar .button-assertive-100.activated, + .button-assertive-100.activated, + .header-assertive-100.activated, + .button-bar-assertive-100.activated, + .bar-assertive-100.activated, + .assertive-100-border.activated { + background-color: #FFCDD2; + color: #fff; } + + .stable-bg, + .button-stable, + .bar .button-stable, + .header-stable, + .button-bar-stable, + .bar-stable, + .stable-border, + .stable-bg:hover, + .button-stable:hover, + .bar .button-stable:hover, + .header-stable:hover, + .button-bar-stable:hover, + .bar-stable:hover, + .stable-border:hover, + .stable-bg:active, + .button-stable:active, + .bar .button-stable:active, + .header-stable:active, + .button-bar-stable:active, + .bar-stable:active, + .stable-border:active, + .stable-bg.activated, + .button-stable.activated, + .bar .button-stable.activated, + .header-stable.activated, + .button-bar-stable.activated, + .bar-stable.activated, + .stable-border.activated { + background-color: #E0E0E0; + color: #fff; } + + /* Text Colors + ==================================*/ + .positive, .icon-help, + .positive *, .icon-help *, + *.positive, + *.icon-help, + .positive:hover, + .icon-help:hover, + .positive:hover *, .icon-help:hover *, + *.positive:hover, + *.icon-help:hover, + .positive:active, + .icon-help:active, + .positive:active *, .icon-help:active *, + *.positive:active, + *.icon-help:active { + color: #3F51B5; } + + .positive-900, + .positive-900 *, + *.positive-900, + .positive-900:hover, + .positive-900:hover *, + *.positive-900:hover, + .positive-900:active, + .positive-900:active *, + *.positive-900:active { + color: #3F51B5; } + + .positive-100, + .positive-100 *, + *.positive-100, + .positive-100:hover, + .positive-100:hover *, + *.positive-100:hover, + .positive-100:active, + .positive-100:active *, + *.positive-100:active { + color: #C5CAE9; } + + .calm-100, + .calm-100 *, + *.calm-100, + .calm-100:hover, + .calm-100:hover *, + *.calm-100:hover, + .calm-100:active, + .calm-100:active *, + *.calm-100:active { + color: #2196F3; } + + .calm-900, + .calm-900 *, + *.calm-900, + .calm-900:hover, + .calm-900:hover *, + *.calm-900:hover, + .calm-900:active, + .calm-900:active *, + *.calm-900:active { + color: #0D47A1; } + + .calm-100, + .calm-100 *, + *.calm-100, + .calm-100:hover, + .calm-100:hover *, + *.calm-100:hover, + .calm-100:active, + .calm-100:active *, + *.calm-100:active { + color: #BBDEFB; } + + .royal, + .royal *, + *.royal, + .royal:hover, + .royal:hover *, + *.royal:hover, + .royal:active, + .royal:active *, + *.royal:active { + color: #673AB7; } + + .royal-900, + .royal-900 *, + *.royal-900, + .royal-900:hover, + .royal-900:hover *, + *.royal-900:hover, + .royal-900:active, + .royal-900:active *, + *.royal-900:active { + color: #311B92; } + + .royal-100, + .royal-100 *, + *.royal-100, + .royal-100:hover, + .royal-100:hover *, + *.royal-100:hover, + .royal-100:active, + .royal-100:active *, + *.royal-100:active { + color: #D1C4E9; } + + .balanced, + .balanced *, + *.balanced, + .balanced:hover, + .balanced:hover *, + *.balanced:hover, + .balanced:active, + .balanced:active *, + *.balanced:active { + color: #4CAF50; } + + .balanced-900, + .balanced-900 *, + *.balanced-900, + .balanced-900:hover, + .balanced-900:hover *, + *.balanced-900:hover, + .balanced-900:active, + .balanced-900:active *, + *.balanced-900:active { + color: #1B5E20; } + + .balanced-100, + .balanced-100 *, + *.balanced-100, + .balanced-100:hover, + .balanced-100:hover *, + *.balanced-100:hover, + .balanced-100:active, + .balanced-100:active *, + *.balanced-100:active { + color: #C8E6C9; } + + .energized, + .energized *, + *.energized, + .energized:hover, + .energized:hover *, + *.energized:hover, + .energized:active, + .energized:active *, + *.energized:active { + color: #FF9800; } + + .energized-900, + .energized-900 *, + *.energized-900, + .energized-900:hover, + .energized-900:hover *, + *.energized-900:hover, + .energized-900:active, + .energized-900:active *, + *.energized-900:active { + color: #E65100; } + + .energized-100, + .energized-100 *, + *.energized-100, + .energized-100:hover, + .energized-100:hover *, + *.energized-100:hover, + .energized-100:active, + .energized-100:active *, + *.energized-100:active { + color: #FFE0B2; } + + .assertive, .icon-alert, + .assertive *, .icon-alert *, + *.assertive, + *.icon-alert, + .assertive:hover, + .icon-alert:hover, + .assertive:hover *, .icon-alert:hover *, + *.assertive:hover, + *.icon-alert:hover, + .assertive:active, + .icon-alert:active, + .assertive:active *, .icon-alert:active *, + *.assertive:active, + *.icon-alert:active { + color: #F44336; } + + .assertive-900, + .assertive-900 *, + *.assertive-900, + .assertive-900:hover, + .assertive-900:hover *, + *.assertive-900:hover, + .assertive-900:active, + .assertive-900:active *, + *.assertive-900:active { + color: #B71C1C; } + + .assertive-100, + .assertive-100 *, + *.assertive-100, + .assertive-100:hover, + .assertive-100:hover *, + *.assertive-100:hover, + .assertive-100:active, + .assertive-100:active *, + *.assertive-100:active { + color: #FFCDD2; } + + .stable, + .stable *, + *.stable, + .stable:hover, + .stable:hover *, + *.stable:hover, + .stable:active, + .stable:active *, + *.stable:active { + color: #E0E0E0; } + + .light, + .light *, + *.light, + .light:hover, + .light:hover *, + *.light:hover, + .light:active, + .light:active *, + *.light:active { + color: #fff; } + + .dark, + .dark *, + *.dark, + .dark:hover, + .dark:hover *, + *.dark:hover, + .dark:active, + .dark:active *, + *.dark:active { + color: #444; } + + .light-border { + border-color: #ddd; } + + .navbar-default .navbar-nav > li > a { + margin: 0; + padding-right: 26px; + padding-left: 26px; + border-top: 3px solid transparent; + color: #BFD5C9; + opacity: 1; } + + /* Mid-Bar + ==================================*/ + .mid-bar { + padding: 16px; } + + .mid-bar h1, + .mid-bar h2, + .mid-bar h3, + .mid-bar h4, + .mid-bar h5, + .mid-bar h6 { + color: #fff; + margin-bottom: 5px; } + + .mid-bar p { + color: rgba(255, 255, 255, 0.5); + margin-bottom: 0; } + + /* Item + ==================================*/ + .item-avatar, + .item-avatar .item-content, + .item-avatar-left, + .item-avatar-left .item-content, + .card > .item-avatar { + padding-left: 95px; } + + .item, + .item-complex .item-content, + .item-radio .item-content { + background-color: transparent; } + + .dark-bg h2, + .item.dark-bg h2 { + color: #fff; } + + .tabs-striped .tabs { + box-shadow: 0px 2px 5px 0 rgba(0, 0, 0, 0.26); } + + .bar .button.button-clear, .bar .button.button-text { + color: #fff; } + + .bar .button.button-icon .icon:before, .bar .button.button-icon .icon-help:before, .bar .button.button-icon .icon-alert:before, .bar .button.button-icon #menu .footer .icon-help:before, #menu .footer .bar .button.button-icon .icon-help:before, + .bar .button.button-icon.icon-left:before, + .bar .button.button-icon.icon-right:before, + .bar .button.button-icon:before { + vertical-align: top; + font-size: 24px; } + + .button-icon.button.active, + .button-icon.button.activated { + opacity: initial; } + + /* Button + ==================================*/ + .button { + overflow: hidden !important; } + + @font-face { + font-family: "Cesiumicons"; + src: url("../fonts/cesiumicons.eot?v=1.2"); + src: url("../fonts/cesiumicons.eot?v=1.2#iefix") format("embedded-opentype"), url("../fonts/cesiumicons.ttf?v=1.2") format("truetype"), url("../fonts/cesiumicons.woff?v=1.2") format("woff"), url("../fonts/cesiumicons.woff") format("woff"), url() format("svg"); + font-weight: normal; + font-style: normal; } + + .cion, .cesiumicons, + .ion-social-duniter:before, + .ion-social-diaspora:before, + .ion-office:before, + .ion-library:before { + display: inline-block; + font-family: "Cesiumicons"; + speak: none; + font-style: normal; + font-weight: normal; + font-variant: normal; + text-transform: none; + text-rendering: auto; + line-height: 1; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; } + + .ion-social-duniter:before { + content: ""; } + + .ion-social-diaspora:before { + content: ""; } + + .ion-office:before { + content: ""; } + + .ion-library:before { + content: ""; } + + @media screen and (max-width: 400px) { + @-ms-viewport { + width: 320px; } + .item .badge { + right: 16px; } } + + @media screen and (max-width: 767px) { + .hidden-xs { + display: none !important; + visibility: hidden !important; } + .badge { + text-overflow: ellipsis !important; + white-space: nowrap; + overflow: hidden !important; + max-width: 300px !important; + display: block !important; } + .badge:empty { + display: none !important; } + .item .badge { + right: 16px; } + .padding-top-xs { + padding-top: 10px; } } + + @media screen and (min-width: 768px) { + .hidden-xs { + display: inherit; + visibility: visible; } + .row.hidden-xs { + display: flex !important; } + .button.hidden-xs { + display: inline-block; } + .item-toggle .toggle { + right: 32px; } } + + @media screen and (max-width: 767px) { + .visible-xs { + display: inherit !important; + visibility: visible !important; } } + + @media screen and (min-width: 768px) { + .visible-xs { + display: none !important; + visibility: hidden !important; } } + + @media screen and (max-width: 767px) { + .padding-xs { + padding: 16px !important; } } + + @media screen and (min-width: 768px) { + .padding-xs { + padding: inherit; } } + + @media screen and (max-width: 767px) { + .no-padding-xs { + padding: 0px !important; } } + + @media screen and (min-width: 768px) { + .no-padding-xs { + padding: inherit; } } + + @media screen and (max-width: 767px) { + .no-margin-xs { + margin: 0px !important; } } + + @media screen and (min-width: 768px) { + .no-margin-xs { + margin: inherit; } } + + @media screen and (max-width: 991px) and (min-width: 768px) { + .hidden-sm, .row-header.hidden-sm { + display: none !important; + visibility: hidden !important; } + .badge { + text-overflow: ellipsis !important; + white-space: nowrap; + overflow: hidden !important; + max-width: 400px !important; + display: block !important; } + .badge:empty { + display: none !important; } } + + @media screen and (min-width: 992px) { + .hidden-sm { + display: inherit; + visibility: visible; } + .row.hidden-sm { + display: flex !important; } + .button.hidden-sm { + display: inline-block; } } + + @media screen and (max-width: 767px) { + .hidden-sm { + display: inherit; + visibility: visible; } } + + @media screen and (max-width: 991px) { + .visible-sm { + display: inherit !important; + visibility: visible !important; } } + + @media screen and (min-width: 992px) { + .visible-sm { + display: none; + visibility: hidden; } } + + @media screen and (max-width: 767px) { + .visible-sm { + display: none; + visibility: hidden; } } + + @media screen and (max-width: 991px) { + body { + cursor: url(), auto; } } + + @media screen and (min-width: 992px) { + body { + cursor: inherit; } } + + @media screen and (min-width: 992px) and (max-width: 1199px) { + .hidden-md { + display: none !important; + visibility: hidden !important; } + .badge { + text-overflow: ellipsis !important; + white-space: nowrap; + overflow: hidden !important; + max-width: 400px !important; + display: block !important; } + .badge:empty { + display: none !important; } + /* + see issue # + html{ + -webkit-user-selectuser-select: all !important; + -moz-user-select: all !important; + -ms-user-select: all !important; + user-select: all !important; + }*/ } + + @media screen and (min-width: 1200px) { + .hidden-md { + display: inherit; + visibility: visible; } } + + @media screen and (max-width: 991px) { + .hidden-md { + display: inherit; + visibility: visible; } } + + @media screen and (min-width: 992px) and (max-width: 1199px) { + .visible-md { + display: inherit !important; + visibility: visible !important; } } + + @media screen and (min-width: 1200px) { + .visible-md { + display: none; + visibility: hidden; } } + + @media screen and (max-width: 991px) { + .visible-md { + display: none; + visibility: hidden; } } + + @media screen and (min-width: 1200px) { + .hidden-lg { + display: none !important; + visibility: hidden !important; } + .visible-lg { + display: inherit !important; + visibility: visible !important; } + .badge { + text-overflow: ellipsis !important; + white-space: nowrap; + overflow: hidden !important; + max-width: 450px !important; + display: block !important; } + .badge:empty { + display: none !important; } + /*html{ + -webkit-user-select: all !important; + -moz-user-select: all !important; + -ms-user-select: all !important; + user-select: all !important; + }*/ } + + @media screen and (max-width: 1199px) { + .hidden-lg { + display: inherit; + visibility: visible; } } + + @media screen and (max-width: 1199px) { + .visible-lg { + display: none; + visibility: hidden; } } + + @media screen and (max-width: 768px) { + .no-padding-xs { + padding: inherit; } } + + @media screen and (max-width: 767px) { + .no-margin-xs { + margin: 0px !important; } } + + /********** + Notifications view + **********/ + .view-notification .item.unread { + background-color: #ecf0f7 !important; + border-color: #dddfe2 !important; } + + .view-notification ion-item h4 i.icon, .view-notification ion-item h4 i.icon-help, .view-notification ion-item h4 i.icon-alert, .view-notification ion-item h4 #menu .footer i.icon-help, #menu .footer .view-notification ion-item h4 i.icon-help { + font-size: 18px !important; + line-height: 12px !important; + vertical-align: middle !important; } + + /* ============ + Buttons + =============== */ + .bar.bar-header .button.button-clear.button-icon.ion-android-more-vertical, .bar.bar-header .button.button-icon.ion-android-more-vertical.button-text, + .bar.bar-header .button.button-clear.button-icon i.ion-android-more-vertical, + .bar.bar-header .button.button-icon.button-text i.ion-android-more-vertical { + padding-left: 8px; } + + .bar.bar-header .buttons .secondary-buttons > + .button.button-clear.button-icon.ion-android-more-vertical:first-child, .bar.bar-header .buttons .secondary-buttons > + .button.button-icon.ion-android-more-vertical.button-text:first-child { + padding-left: 0px !important; } + + .bar .buttons.pull-right, .bar .popover-helptip .buttons.icon.icon-right, .popover-helptip .bar .buttons.icon.icon-right, .bar .popover-helptip .buttons.icon-right.icon-help, .popover-helptip .bar .buttons.icon-right.icon-help, .bar .popover-helptip .buttons.icon-right.icon-alert, .popover-helptip .bar .buttons.icon-right.icon-alert, .bar .popover-helptip #menu .footer .buttons.icon-right.icon-help, .popover-helptip #menu .footer .bar .buttons.icon-right.icon-help, .bar #menu .footer .popover-helptip .buttons.icon-right.icon-help, #menu .footer .popover-helptip .bar .buttons.icon-right.icon-help, .bar .popover-helptip .buttons.icon.icon-center, .popover-helptip .bar .buttons.icon.icon-center, .bar .popover-helptip .buttons.icon-center.icon-help, .popover-helptip .bar .buttons.icon-center.icon-help, .bar .popover-helptip .buttons.icon-center.icon-alert, .popover-helptip .bar .buttons.icon-center.icon-alert, .bar .popover-helptip #menu .footer .buttons.icon-center.icon-help, .popover-helptip #menu .footer .bar .buttons.icon-center.icon-help, .bar #menu .footer .popover-helptip .buttons.icon-center.icon-help, #menu .footer .popover-helptip .bar .buttons.icon-center.icon-help, .bar .popover-helptip .buttons.icon.icon-bottom-right, .popover-helptip .bar .buttons.icon.icon-bottom-right, .bar .popover-helptip .buttons.icon-bottom-right.icon-help, .popover-helptip .bar .buttons.icon-bottom-right.icon-help, .bar .popover-helptip .buttons.icon-bottom-right.icon-alert, .popover-helptip .bar .buttons.icon-bottom-right.icon-alert, .bar .popover-helptip #menu .footer .buttons.icon-bottom-right.icon-help, .popover-helptip #menu .footer .bar .buttons.icon-bottom-right.icon-help, .bar #menu .footer .popover-helptip .buttons.icon-bottom-right.icon-help, #menu .footer .popover-helptip .bar .buttons.icon-bottom-right.icon-help, .bar .popover-helptip .buttons.icon.icon-bottom-center, .popover-helptip .bar .buttons.icon.icon-bottom-center, .bar .popover-helptip .buttons.icon-bottom-center.icon-help, .popover-helptip .bar .buttons.icon-bottom-center.icon-help, .bar .popover-helptip .buttons.icon-bottom-center.icon-alert, .popover-helptip .bar .buttons.icon-bottom-center.icon-alert, .bar .popover-helptip #menu .footer .buttons.icon-bottom-center.icon-help, .popover-helptip #menu .footer .bar .buttons.icon-bottom-center.icon-help, .bar #menu .footer .popover-helptip .buttons.icon-bottom-center.icon-help, #menu .footer .popover-helptip .bar .buttons.icon-bottom-center.icon-help, .bar .title + .button:last-child, .bar .title + .buttons, .bar > .button + .button:last-child, .bar > .button.pull-right, .popover-helptip .bar > .button.icon.icon-right, .popover-helptip .bar > .button.icon-right.icon-help, .popover-helptip .bar > .button.icon-right.icon-alert, .popover-helptip #menu .footer .bar > .button.icon-right.icon-help, #menu .footer .popover-helptip .bar > .button.icon-right.icon-help, .popover-helptip .bar > .button.icon.icon-center, .popover-helptip .bar > .button.icon-center.icon-help, .popover-helptip .bar > .button.icon-center.icon-alert, .popover-helptip #menu .footer .bar > .button.icon-center.icon-help, #menu .footer .popover-helptip .bar > .button.icon-center.icon-help, .popover-helptip .bar > .button.icon.icon-bottom-right, .popover-helptip .bar > .button.icon-bottom-right.icon-help, .popover-helptip .bar > .button.icon-bottom-right.icon-alert, .popover-helptip #menu .footer .bar > .button.icon-bottom-right.icon-help, #menu .footer .popover-helptip .bar > .button.icon-bottom-right.icon-help, .popover-helptip .bar > .button.icon.icon-bottom-center, .popover-helptip .bar > .button.icon-bottom-center.icon-help, .popover-helptip .bar > .button.icon-bottom-center.icon-alert, .popover-helptip #menu .footer .bar > .button.icon-bottom-center.icon-help, #menu .footer .popover-helptip .bar > .button.icon-bottom-center.icon-help { + top: 0px !important; } + + .bar.bar-header { + padding-right: 5px !important; } + .bar.bar-header .buttons-right span { + margin-left: 0px !important; } + + .bar .title + .buttons.buttons-right { + right: 5px; } + + .button-icon { + border-color: transparent; + box-shadow: none !important; } + + .button-small-padding { + padding: 0 7px !important; } + + .button-text { + color: grey !important; + font-size: 12px; } + + .button-text.button-small { + padding: 5px 2px; + font-size: 12px !important; } + + .button-text-positive { + color: #387ef5 !important; } + + .button-text-stable { + color: #b2b2b2 !important; } + + + .gray, .popover-share .bar-header span, .popover-share .bar-footer .button-close, .popover-helptip .button-close { + color: grey !important; } + .gray b, .popover-share .bar-header span b, .popover-share .bar-footer .button-close b, .popover-helptip .button-close b { + color: grey !important; } + + .gray a, .popover-share .bar-header span a, .popover-share .bar-footer .button-close a, .popover-helptip .button-close a, .positive a, .icon-help a { + color: inherit; } + + .gray a:hover, .popover-share .bar-header span a:hover, .popover-share .bar-footer .button-close a:hover, .popover-helptip .button-close a:hover, .positive a:hover, .icon-help a:hover { + color: inherit; } + + .gray a:visited, .popover-share .bar-header span a:visited, .popover-share .bar-footer .button-close a:visited, .popover-helptip .button-close a:visited, .positive a:visited, .icon-help a:visited { + color: inherit; } + + .item a { + text-decoration: none; } + + .no-padding { + padding: 0px !important; } + + .avatar-member { + background-image: url(); } + + .avatar.disable { + opacity: 0.7; } + + .avatar-wallet { + background-image: url(); } + + .item.item-icon-left > i.avatar:first-child { + position: absolute; + display: flex; + height: 100%; + align-items: center; + font-size: 16px; + left: 16px; + top: 8px; + max-height: 32px; + max-width: 32px; } + + .item.item-checkbox.item-avatar > i.avatar:first-child, + .item.item-checkbox.item-avatar > i.item-image:first-child, + .item.item-checkbox.item-avatar .item-content > i.avatar:first-child, + .item.item-checkbox.item-avatar .item-content > i.item-image:first-child, + .item.item-checkbox.item-avatar * > i.avatar:first-child, + .item.item-checkbox.item-avatar * > i.item-image:first-child, + .item.item-checkbox.item-avatar * .item-content > i.avatar:first-child, + .item.item-checkbox.item-avatar * .item-content > i.item-image:first-child { + left: 65px; } + + .item.item-checkbox.item-avatar .item-content, + .item.item-checkbox .item-content .item-avatar { + padding-left: 65px; } + + .card .card-header, + .card .card-header { + font-size: 90%; + opacity: 0.8; + filter: alpha(opacity=80); } + + .card.stable-900-bg, + .card .stable-900-bg, + .item.stable-900-bg, + .item .stable-900-bg, + .item-complex .item-content .stable-900-bg, + .item-radio .item-content .stable-900-bg { + background-color: #e0e0e0 !important; } + + .card .item { + background: inherit; } + + .card.stable-bg, + .card .stable-bg, + .item.stable-bg, + .item .stable-bg, + .item-complex .item-content .stable-bg, + .item-radio .item-content .stable-bg { + background-color: #f8f8f8 !important; } + + .card .card-header { + padding-top: 5px !important; + padding-bottom: 0px !important; + min-height: 25px; } + + .card .item .card-footer { + margin-bottom: 5px; } + + .card .card-avatar .avatar, + .card.card-avatar .avatar { + box-shadow: 0px 3px 4px 0 rgba(0, 0, 0, 0.26); + top: 7px; + background-color: #D9D9D9; } + + .card .card-avatar img.avatar, + .card.card-avatar img.avatar { + border: 0; + min-height: 54px; + min-width: 54px; } + + .card .card-avatar .item.item-avatar, + .card.card-avatar .item.item-avatar { + padding-top: 10px; + padding-bottom: 2px; + min-height: 45px !important; } + + .card .card-avatar .card-footer, + .card.card-avatar .card-footer { + padding-left: 88px; } + .card .card-avatar .card-footer .pull-right a, .card .card-avatar .card-footer .popover-helptip .icon.icon-right a, .popover-helptip .card .card-avatar .card-footer .icon.icon-right a, .card .card-avatar .card-footer .popover-helptip .icon-right.icon-help a, .popover-helptip .card .card-avatar .card-footer .icon-right.icon-help a, .card .card-avatar .card-footer .popover-helptip .icon-right.icon-alert a, .popover-helptip .card .card-avatar .card-footer .icon-right.icon-alert a, .card .card-avatar .card-footer .popover-helptip #menu .footer .icon-right.icon-help a, .popover-helptip #menu .footer .card .card-avatar .card-footer .icon-right.icon-help a, .card .card-avatar .card-footer #menu .footer .popover-helptip .icon-right.icon-help a, #menu .footer .popover-helptip .card .card-avatar .card-footer .icon-right.icon-help a, .card .card-avatar .card-footer .popover-helptip .icon.icon-center a, .popover-helptip .card .card-avatar .card-footer .icon.icon-center a, .card .card-avatar .card-footer .popover-helptip .icon-center.icon-help a, .popover-helptip .card .card-avatar .card-footer .icon-center.icon-help a, .card .card-avatar .card-footer .popover-helptip .icon-center.icon-alert a, .popover-helptip .card .card-avatar .card-footer .icon-center.icon-alert a, .card .card-avatar .card-footer .popover-helptip #menu .footer .icon-center.icon-help a, .popover-helptip #menu .footer .card .card-avatar .card-footer .icon-center.icon-help a, .card .card-avatar .card-footer #menu .footer .popover-helptip .icon-center.icon-help a, #menu .footer .popover-helptip .card .card-avatar .card-footer .icon-center.icon-help a, .card .card-avatar .card-footer .popover-helptip .icon.icon-bottom-right a, .popover-helptip .card .card-avatar .card-footer .icon.icon-bottom-right a, .card .card-avatar .card-footer .popover-helptip .icon-bottom-right.icon-help a, .popover-helptip .card .card-avatar .card-footer .icon-bottom-right.icon-help a, .card .card-avatar .card-footer .popover-helptip .icon-bottom-right.icon-alert a, .popover-helptip .card .card-avatar .card-footer .icon-bottom-right.icon-alert a, .card .card-avatar .card-footer .popover-helptip #menu .footer .icon-bottom-right.icon-help a, .popover-helptip #menu .footer .card .card-avatar .card-footer .icon-bottom-right.icon-help a, .card .card-avatar .card-footer #menu .footer .popover-helptip .icon-bottom-right.icon-help a, #menu .footer .popover-helptip .card .card-avatar .card-footer .icon-bottom-right.icon-help a, .card .card-avatar .card-footer .popover-helptip .icon.icon-bottom-center a, .popover-helptip .card .card-avatar .card-footer .icon.icon-bottom-center a, .card .card-avatar .card-footer .popover-helptip .icon-bottom-center.icon-help a, .popover-helptip .card .card-avatar .card-footer .icon-bottom-center.icon-help a, .card .card-avatar .card-footer .popover-helptip .icon-bottom-center.icon-alert a, .popover-helptip .card .card-avatar .card-footer .icon-bottom-center.icon-alert a, .card .card-avatar .card-footer .popover-helptip #menu .footer .icon-bottom-center.icon-help a, .popover-helptip #menu .footer .card .card-avatar .card-footer .icon-bottom-center.icon-help a, .card .card-avatar .card-footer #menu .footer .popover-helptip .icon-bottom-center.icon-help a, #menu .footer .popover-helptip .card .card-avatar .card-footer .icon-bottom-center.icon-help a, + .card.card-avatar .card-footer .pull-right a, + .card.card-avatar .card-footer .popover-helptip .icon.icon-right a, .popover-helptip + .card.card-avatar .card-footer .icon.icon-right a, + .card.card-avatar .card-footer .popover-helptip .icon-right.icon-help a, .popover-helptip + .card.card-avatar .card-footer .icon-right.icon-help a, + .card.card-avatar .card-footer .popover-helptip .icon-right.icon-alert a, .popover-helptip + .card.card-avatar .card-footer .icon-right.icon-alert a, + .card.card-avatar .card-footer .popover-helptip #menu .footer .icon-right.icon-help a, .popover-helptip #menu .footer + .card.card-avatar .card-footer .icon-right.icon-help a, + .card.card-avatar .card-footer #menu .footer .popover-helptip .icon-right.icon-help a, #menu .footer .popover-helptip + .card.card-avatar .card-footer .icon-right.icon-help a, + .card.card-avatar .card-footer .popover-helptip .icon.icon-center a, .popover-helptip + .card.card-avatar .card-footer .icon.icon-center a, + .card.card-avatar .card-footer .popover-helptip .icon-center.icon-help a, .popover-helptip + .card.card-avatar .card-footer .icon-center.icon-help a, + .card.card-avatar .card-footer .popover-helptip .icon-center.icon-alert a, .popover-helptip + .card.card-avatar .card-footer .icon-center.icon-alert a, + .card.card-avatar .card-footer .popover-helptip #menu .footer .icon-center.icon-help a, .popover-helptip #menu .footer + .card.card-avatar .card-footer .icon-center.icon-help a, + .card.card-avatar .card-footer #menu .footer .popover-helptip .icon-center.icon-help a, #menu .footer .popover-helptip + .card.card-avatar .card-footer .icon-center.icon-help a, + .card.card-avatar .card-footer .popover-helptip .icon.icon-bottom-right a, .popover-helptip + .card.card-avatar .card-footer .icon.icon-bottom-right a, + .card.card-avatar .card-footer .popover-helptip .icon-bottom-right.icon-help a, .popover-helptip + .card.card-avatar .card-footer .icon-bottom-right.icon-help a, + .card.card-avatar .card-footer .popover-helptip .icon-bottom-right.icon-alert a, .popover-helptip + .card.card-avatar .card-footer .icon-bottom-right.icon-alert a, + .card.card-avatar .card-footer .popover-helptip #menu .footer .icon-bottom-right.icon-help a, .popover-helptip #menu .footer + .card.card-avatar .card-footer .icon-bottom-right.icon-help a, + .card.card-avatar .card-footer #menu .footer .popover-helptip .icon-bottom-right.icon-help a, #menu .footer .popover-helptip + .card.card-avatar .card-footer .icon-bottom-right.icon-help a, + .card.card-avatar .card-footer .popover-helptip .icon.icon-bottom-center a, .popover-helptip + .card.card-avatar .card-footer .icon.icon-bottom-center a, + .card.card-avatar .card-footer .popover-helptip .icon-bottom-center.icon-help a, .popover-helptip + .card.card-avatar .card-footer .icon-bottom-center.icon-help a, + .card.card-avatar .card-footer .popover-helptip .icon-bottom-center.icon-alert a, .popover-helptip + .card.card-avatar .card-footer .icon-bottom-center.icon-alert a, + .card.card-avatar .card-footer .popover-helptip #menu .footer .icon-bottom-center.icon-help a, .popover-helptip #menu .footer + .card.card-avatar .card-footer .icon-bottom-center.icon-help a, + .card.card-avatar .card-footer #menu .footer .popover-helptip .icon-bottom-center.icon-help a, #menu .footer .popover-helptip + .card.card-avatar .card-footer .icon-bottom-center.icon-help a { + margin-right: 8px; } + + a.underline:focus, + .underline a:focus, + .a.underline:active, + .underline a:active, + a.underline:hover, + .underline a:hover { + outline: 1px !important; + text-decoration: underline !important; } + + .card-avatar-small.card, + .card-avatar-small .card, + .card-avatar-small .card.card-avatar, + .card-avatar-small .card .card-avatar { + min-height: 45px; } + .card-avatar-small.card .avatar, + .card-avatar-small.card .item-avatar .avatar, + .card-avatar-small .card .avatar, + .card-avatar-small .card .item-avatar .avatar, + .card-avatar-small .card.card-avatar .avatar, + .card-avatar-small .card.card-avatar .item-avatar .avatar, + .card-avatar-small .card .card-avatar .avatar, + .card-avatar-small .card .card-avatar .item-avatar .avatar { + box-shadow: 0px 2px 2px 0 rgba(0, 0, 0, 0.26); + height: 30px !important; + width: 30px !important; + left: 5px !important; } + .card-avatar-small.card .item.item-avatar, + .card-avatar-small .card .item.item-avatar, + .card-avatar-small .card.card-avatar .item.item-avatar, + .card-avatar-small .card .card-avatar .item.item-avatar { + min-height: 25px !important; + padding-left: 42px !important; } + .card-avatar-small.card .card-footer, + .card-avatar-small .card .card-footer, + .card-avatar-small .card.card-avatar .card-footer, + .card-avatar-small .card .card-avatar .card-footer { + padding-top: 0px; + padding-left: 42px !important; } + + .list .item.text-left { + text-align: left !important; } + + .list .item.text-center, .list .item.large-button-bar { + text-align: center !important; } + + .list .item.text-right { + text-align: right !important; } + + .list .item-divider.item-divider-top-border { + border-top: solid 1px rgba(0, 0, 0, 0.12); } + + /********** + Item > Avatar + **********/ + .item-avatar { + min-height: 80px !important; } + + .item-avatar > i:first-child, + .item-avatar > img:first-child, + .item-avatar i.item-image:first-child, + .item-avatar img.item-image:first-child, + .item-avatar .item-content > i:first-child, + .item-avatar .item-content > img:first-child, + .item-avatar .item-content i.item-image:first-child, + .item-avatar .item-content img.item-image:first-child, + .item-avatar-left > i:first-child, + .item-avatar-left > img:first-child, + .item-avatar-left i.item-image:first-child, + .item-avatar-left img.item-image:first-child, + .item-avatar-left .item-content > i:first-child, + .item-avatar-left .item-content > img:first-child, + .item-avatar-left .item-content i.item-image:first-child, + .item-avatar-left .item-content img.item-image:first-child { + color: #D9D9D9; + background-color: #f8f8f8; + border: solid 1px #D9D9D9; + overflow: hidden !important; + font-size: 45px !important; + /*padding-left: 0px; + padding-top: 0px; + text-align: center; + vertical-align: middle;*/ + line-height: 56px; + width: 100% !important; + /*display: block !important;*/ + max-height: 56px !important; + max-width: 56px !important; + top: 12px !important; } + + .item-avatar > .icon:first-child:before, .item-avatar > .icon-help:first-child:before, .item-avatar > .icon-alert:first-child:before, #menu .footer .item-avatar > .icon-help:first-child:before, + .item-avatar .icon.item-image:first-child:before, + .item-avatar .item-image.icon-help:first-child:before, + .item-avatar .item-image.icon-alert:first-child:before, + .item-avatar #menu .footer .item-image.icon-help:first-child:before, #menu .footer + .item-avatar .item-image.icon-help:first-child:before, + .item-avatar .item-content > .icon:first-child:before, + .item-avatar .item-content > .icon-help:first-child:before, + .item-avatar .item-content > .icon-alert:first-child:before, + .item-avatar #menu .footer .item-content > .icon-help:first-child:before, #menu .footer + .item-avatar .item-content > .icon-help:first-child:before, + .item-avatar.item-icon-right .icon:first-child:before, + .item-avatar.item-icon-right .icon-help:first-child:before, + .item-avatar.item-icon-right .icon-alert:first-child:before, + .item-avatar.item-icon-right #menu .footer .icon-help:first-child:before, #menu .footer + .item-avatar.item-icon-right .icon-help:first-child:before, + .item-avatar.item-icon-right .icon-help:first-child:before, + .item-avatar.item-icon-right .icon-alert:first-child:before, + .item-avatar.item-icon-right #menu .footer .icon-help:first-child:before, + #menu .footer .item-avatar.item-icon-right .icon-help:first-child:before { + width: 56px !important; } + + .item-avatar.item-icon-right .icon:last-child, .item-avatar.item-icon-right .icon-help:last-child, .item-avatar.item-icon-right .icon-alert:last-child, .item-avatar.item-icon-right #menu .footer .icon-help:last-child, #menu .footer .item-avatar.item-icon-right .icon-help:last-child, + .item-avatar.item-icon-right .icon-help:last-child, + .item-avatar.item-icon-right .icon-alert:last-child, + .item-avatar.item-icon-right #menu .footer .icon-help:last-child, + #menu .footer .item-avatar.item-icon-right .icon-help:last-child { + left: auto; } + + /********** + Item > Other + **********/ + .item em { + font-weight: bold !important; } + + @media screen and (min-width: 992px) { + .list .item.item-border-large { + border-bottom: solid 1px #ccc !important; } + .list.item-border-large .item { + border-bottom: solid 1px #ccc !important; + margin: 0px; } + .list.item-border-large .item-divider { + border-top: 0; } } + + .list .item.item-border { + border-bottom: solid 1px #ccc !important; } + + .list .item.item-small-height { + padding-top: 2px; + padding-bottom: 0px; + min-height: 24px; } + .list .item.item-small-height .badge { + padding-top: 0px !important; + top: inherit; } + .list .item.item-small-height .badge.badge-balanced, + .list .item.item-small-height .badge.badge-positive, + .list .item.item-small-height .badge.badge-assertive, + .list .item.item-small-height .badge.badge-editable:hover, + .list .item.item-small-height .badge.badge-royal, + .list .item.item-small-height .badge.badge-calm, + .list .item.item-small-height .badge.badge-energized { + top: 1px !important; + padding-top: 3px !important; + padding-bottom: 2px !important; } + + /* + * Badge + */ + .item.item-icon-right .badge, + .item.item-button-right .badge { + right: 43px; } + + .badge-editable:hover { + cursor: pointer; } + + .badge-editable:hover:before { + content: " "; } + + .badge-button { + position: absolute !important; + margin: 0 !important; + padding: 0px 4px !important; + font-size: 10px; + top: 5px !important; + right: 3px !important; } + + /********** + Item buttons + **********/ + .item-button-right > .button, + .item-button-right .item-content > .button, + .item-button-right > .buttons, + .item-button-right .item-content > .buttons { + top: 16px; } + + .item-button-right > .button.button-small, + .item-button-right .item-content > .button.button-small, + .item-button-right > .buttons .button-small, + .item-button-right .item-content > .buttons .button-small { + font-size: 14px; } + + .item.large-button-bar { + margin-bottom: 10px; } + + /********** + Item avatar + **********/ + .item-avatar-left-padding { + padding-left: 95px; } + + /********** + Item thumbnail + **********/ + .item-thumbnail-left-padding { + padding-left: 106px; } + + .item.item-thumbnail-left, .item-thumbnail-left { + min-height: 100px !important; } + + .item-thumbnail-left > i:first-child, + .item-thumbnail-left i.item-image, + .item-thumbnail-left .item-content > i:first-child, + .item-thumbnail-left .item-content i.item-image { + color: #D9D9D9; + background-color: #f8f8f8; + overflow: hidden !important; + font-size: 50px !important; + line-height: 80px; + padding: 0 15px; + background-position: center; + background-size: cover; + display: inline-block; } + + @media screen and (max-width: 400px) { + .card > .item.item-thumbnail-left, + .item-thumbnail-left, + .item-thumbnail-left .item-content { + padding-left: 84px !important; } + .item-thumbnail-left > img:first-child, + .item-thumbnail-left img.item-image, + .item-thumbnail-left .item-content > img:first-child, + .item-thumbnail-left .item-content img.item-image { + max-width: 70px; + max-height: 70px; } + .item h2 { + font-size: 13px !important; } } + + /********** + Item icons + **********/ + .item-icon-left-padding { + padding-left: 40px; } + + .item-icon-right-padding { + padding-right: 40px; } + + .text-keep-lines { + white-space: pre-line !important; } + + .text-italic { + font-style: italic !important; } + + /********** + Help + **********/ + .icon-help { + font-size: 38px; + vertical-align: middle; } + + .icon-alert { + font-size: 38px; + vertical-align: middle; } + + .bar-header .button-icon .avatar { + height: 35px; + width: 35px; + position: relative; + left: 0px; + top: 4px; + border: solid 1px #D9D9D9; } + + .bar-header .button-icon .avatar.active { + background-color: #e0e0e0; } + + .bar-header .button-icon .avatar { + height: 31px; + width: 31px; + position: relative; + left: 0px; + top: 6px; } + + .badge.badge-secondary, + .badge .badge-secondary { + font-size: 12px; + font-style: italic; + top: 37px !important; + font-weight: normal !important; + margin-right: 0px; + padding-right: 0px; } + + .ion-es-user-api:before { + content: url(); } + + .row-header { + border-bottom: solid 1px #ccc !important; + margin: 0px; + min-height: 28px !important; } + + .col-header { + text-align: center; + display: block !important; } + + .col-15 { + -webkit-box-flex: 0; + -webkit-flex: 0 0 15%; + -moz-box-flex: 0; + -moz-flex: 0 0 15%; + -ms-flex: 0 0 15%; + flex: 0 0 15%; + max-width: 15%; } + + .col-border-left { + border-left: solid 1px #ccc !important; } + + .col-border-right { + border-right: solid 1px #ccc !important; } + + .text-no-transform { + text-transform: inherit; } + +>> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/templates/css_logo.st b/duniter4j-es-subscription/src/main/resources/templates/css_logo.st new file mode 100644 index 0000000000000000000000000000000000000000..8a554e95b6ad236c0bf36e70d93f8accb5d2329f --- /dev/null +++ b/duniter4j-es-subscription/src/main/resources/templates/css_logo.st @@ -0,0 +1,22 @@ +css_logo() ::= << + @media screen and (max-width: 767px) { + .logo { + height: 96px; + background-image: url(); + background-size: 96px 96px; } + } + + @media screen and (min-width: 768px) and (max-width: 991px) { + .logo { + height: 144px; + background-image: url(); + background-size: 144px 144px; } + } + + @media screen and (min-width: 992px) { + .logo { + height: 200px; + background-image: url(); + background-size: 200px 200px; } + } +>> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/templates/event_item.st b/duniter4j-es-subscription/src/main/resources/templates/event_item.st new file mode 100644 index 0000000000000000000000000000000000000000..9fdd774d40b7996695d7dd737d86957b09ba73f7 --- /dev/null +++ b/duniter4j-es-subscription/src/main/resources/templates/event_item.st @@ -0,0 +1,6 @@ +event_item(e) ::= << + <div class="item"> + <h3>$e.description$</h3> + <h4 class="gray">$e.time; format="short"$</h4> + </div> +>> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/templates/html.st b/duniter4j-es-subscription/src/main/resources/templates/html.st new file mode 100644 index 0000000000000000000000000000000000000000..443bbb59028ce00347a89e06254956dab3b9b3cc --- /dev/null +++ b/duniter4j-es-subscription/src/main/resources/templates/html.st @@ -0,0 +1,19 @@ +html(content, useCss) ::= << + <!DOCTYPE html> + <head> + <meta charset="UTF-8"> + <title>Cesium+</title> + $if(useCss)$ + <style> + $css()$ + $css_logo()$ + </style> + $endif$ + </head> + <body class="platform-browser platform-linux platform-ready"> + <ion-content> + $content$ + </ion-content> + </body> +</html> +>> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/templates/html_email_content.st b/duniter4j-es-subscription/src/main/resources/templates/html_email_content.st new file mode 100644 index 0000000000000000000000000000000000000000..629515c48cf4f706e10884fa0fd77255752f698d --- /dev/null +++ b/duniter4j-es-subscription/src/main/resources/templates/html_email_content.st @@ -0,0 +1,33 @@ +html_email_content(issuer, senderPubkey, senderName, events, url) ::= << + <div class="row no-padding"> + <div class="col col-20 hidden-xs hidden-sm text-center" id="home"> + <div class="logo"></div> + </div> + <div class="col"> + <div class="padding padding-bottom row responsive-sm"> + <div class="col"> + $length(events):{count | $i18n_args("duniter4j.es.subscription.email.header", [issuer, count])$}$ + </div> + <div class="col"> + <a class="button button-positive pull-right" + href="$url$">$i18n("duniter4j.es.subscription.email.openCesium")$ >></a> + </div> + </div> + <div class="list item-border-large"> + <div class="item item-divider stable-bg"> + $i18n("duniter4j.es.subscription.email.notificationsDivider")$ + </div> + $events:{e|$event_item(e)$}$ + </div> + <div class="center padding text-center"> + <i class="ion-es-user-api"></i> + $i18n_args("duniter4j.es.subscription.email.footer.sendBy", [{$[url, "/#/app/wot/", senderPubkey, "/"]; separator=""$}, senderName])$ + <br/> + <small> + $i18n_args("duniter4j.es.subscription.email.footer.disableHelp", {$[url, "/#/app/wallet/subscriptions"]; separator=""$})$ + </small> + </div> + </div> + <div class="col col-20 hidden-xs hidden-sm"> </div> + </div> +>> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/templates/i18n.st b/duniter4j-es-subscription/src/main/resources/templates/i18n.st new file mode 100644 index 0000000000000000000000000000000000000000..d5d042c66c9f715169af75d7d19936237ecc45ce --- /dev/null +++ b/duniter4j-es-subscription/src/main/resources/templates/i18n.st @@ -0,0 +1,2 @@ +i18n(key) ::= "$key; format=\"i18n\"$" + diff --git a/duniter4j-es-subscription/src/main/resources/templates/i18n_args.st b/duniter4j-es-subscription/src/main/resources/templates/i18n_args.st new file mode 100644 index 0000000000000000000000000000000000000000..120d7753ab0370967f43ec23feb1afc2da84e3dd --- /dev/null +++ b/duniter4j-es-subscription/src/main/resources/templates/i18n_args.st @@ -0,0 +1,2 @@ +i18n_args(key, params) ::= "$key; format={i18n:$params; separator=\",\"$}$" + diff --git a/duniter4j-es-subscription/src/main/resources/templates/text_email.st b/duniter4j-es-subscription/src/main/resources/templates/text_email.st new file mode 100644 index 0000000000000000000000000000000000000000..b982a67a324dff1e29931fc4eb79f203bac6c048 --- /dev/null +++ b/duniter4j-es-subscription/src/main/resources/templates/text_email.st @@ -0,0 +1,13 @@ +text_email(issuer, senderPubkey, senderName, events, url) ::= << +$length(events):{count | $i18n_args("duniter4j.es.subscription.email.header.text", [issuer, count])$}$ + +$i18n("duniter4j.es.subscription.email.notificationsDivider")$ +$events:{e|$text_event_item(e)$}$ + +$i18n("duniter4j.es.subscription.email.openCesium")$ : $url$ + +----------------------------------------------- +$i18n_args("duniter4j.es.subscription.email.footer.sendBy.text", [{$[url, "/#/app/wot/", senderPubkey, "/"]; separator=""$}, senderName])$ +$i18n_args("duniter4j.es.subscription.email.footer.disableHelp.text", {$[url, "/#/app/wallet/subscriptions"]; separator=""$})$ + +>> \ No newline at end of file diff --git a/duniter4j-es-subscription/src/main/resources/templates/text_event_item.st b/duniter4j-es-subscription/src/main/resources/templates/text_event_item.st new file mode 100644 index 0000000000000000000000000000000000000000..51cd56fa96effebcbdd2fc57e8e4bc74b9a6064d --- /dev/null +++ b/duniter4j-es-subscription/src/main/resources/templates/text_event_item.st @@ -0,0 +1,3 @@ +text_event_item(e) ::= << + - $e.time; format="short"$ | $e.description$ +>> \ No newline at end of file diff --git a/duniter4j-es-gchange/src/test/es-home/config/elasticsearch.yml b/duniter4j-es-subscription/src/test/es-home/config/elasticsearch.yml similarity index 90% rename from duniter4j-es-gchange/src/test/es-home/config/elasticsearch.yml rename to duniter4j-es-subscription/src/test/es-home/config/elasticsearch.yml index 828ddbd04543ef810f380ec8e0dbf229e6c2c48c..c0d0fd118f5894a353c0a84149a37a1f5b69332c 100644 --- a/duniter4j-es-gchange/src/test/es-home/config/elasticsearch.yml +++ b/duniter4j-es-subscription/src/test/es-home/config/elasticsearch.yml @@ -15,7 +15,7 @@ # Use a descriptive name for your cluster: # # cluster.name: my-application -cluster.name: duniter4j-elasticsearch-TEST +cluster.name: duniter4j-subscription-TEST # # ------------------------------------ Node ------------------------------------ # @@ -153,12 +153,15 @@ duniter.data.sync.enable: false #duniter.data.sync.host: data.duniter.fr #duniter.data.sync.port: 80 -# ---------------------------------- Duniter4j SMTP server ------------------------- +# ---------------------------------- Duniter4j SMTP server ---------------------- # # SMTP server configuration (host and port) # +duniter.mail.enable: true #duniter.mail.smtp.host: localhost #duniter.mail.smtp.port: 25 +#duniter.mail.smtp.host: smtp.gmail.com +#duniter.mail.smtp.port: 25 # # Mail 'from' address # @@ -169,10 +172,21 @@ duniter.mail.from: root@EIS-DEV # #duniter.mail.admin: user@domain.com duniter.mail.admin: blavenie@EIS-DEV +#duniter.mail.admin: benoit.lavenier@e-is.pro # # Mail subject prefix # #duniter.mail.subject.prefix: [Duniter4j ES] -duniter.changes.listenSource: */block +duniter.changes.listenSource: '*/block' duniter.ws.port: 9400 + +# ---------------------------------- Duniter4j Subscription services ------------ +# +# Enable subscription services ? (default: true) +# +duniter.subscription.enable: true +# +# Time interval (millisecond) to send email ? (default: 3600000 = 1h) +# +duniter.subscription.email.interval: 10000 \ No newline at end of file diff --git a/duniter4j-es-gchange/src/test/es-home/config/logging.yml b/duniter4j-es-subscription/src/test/es-home/config/logging.yml similarity index 100% rename from duniter4j-es-gchange/src/test/es-home/config/logging.yml rename to duniter4j-es-subscription/src/test/es-home/config/logging.yml diff --git a/duniter4j-es-subscription/src/test/es-home/plugins/mapper-attachments/plugin-descriptor.properties b/duniter4j-es-subscription/src/test/es-home/plugins/mapper-attachments/plugin-descriptor.properties new file mode 100644 index 0000000000000000000000000000000000000000..b2ab35861924ba3083a3a6626bfa64c7dfef06e0 --- /dev/null +++ b/duniter4j-es-subscription/src/test/es-home/plugins/mapper-attachments/plugin-descriptor.properties @@ -0,0 +1,80 @@ +# Elasticsearch plugin descriptor file +# This file must exist as 'plugin-descriptor.properties' at +# the root directory of all plugins. +# +# A plugin can be 'site', 'jvm', or both. +# +### example site plugin for "foo": +# +# foo.zip <-- zip file for the plugin, with this structure: +# _site/ <-- the contents that will be served +# plugin-descriptor.properties <-- example contents below: +# +# site=true +# description=My cool plugin +# version=1.0 +# +### example jvm plugin for "foo" +# +# foo.zip <-- zip file for the plugin, with this structure: +# <arbitrary name1>.jar <-- classes, resources, dependencies +# <arbitrary nameN>.jar <-- any number of jars +# plugin-descriptor.properties <-- example contents below: +# +# jvm=true +# classname=foo.bar.BazPlugin +# description=My cool plugin +# version=2.0.0-rc1 +# elasticsearch.version=2.0 +# java.version=1.7 +# +### mandatory elements for all plugins: +# +# 'description': simple summary of the plugin +description=The mapper attachments plugin adds the attachment type to Elasticsearch using Apache Tika. +# +# 'version': plugin's version +version=2.3.3 +# +# 'name': the plugin name +name=mapper-attachments + +### mandatory elements for site plugins: +# +# 'site': set to true to indicate contents of the _site/ +# directory in the root of the plugin should be served. +site=false +# +### mandatory elements for jvm plugins : +# +# 'jvm': true if the 'classname' class should be loaded +# from jar files in the root directory of the plugin. +# Note that only jar files in the root directory are +# added to the classpath for the plugin! If you need +# other resources, package them into a resources jar. +jvm=true +# +# 'classname': the name of the class to load, fully-qualified. +classname=org.elasticsearch.mapper.attachments.MapperAttachmentsPlugin +# +# 'java.version' version of java the code is built against +# use the system property java.specification.version +# version string must be a sequence of nonnegative decimal integers +# separated by "."'s and may have leading zeros +java.version=1.7 +# +# 'elasticsearch.version' version of elasticsearch compiled against +# You will have to release a new version of the plugin for each new +# elasticsearch release. This version is checked when the plugin +# is loaded so Elasticsearch will refuse to start in the presence of +# plugins with the incorrect elasticsearch.version. +elasticsearch.version=2.3.3 +# +### deprecated elements for jvm plugins : +# +# 'isolated': true if the plugin should have its own classloader. +# passing false is deprecated, and only intended to support plugins +# that have hard dependencies against each other. If this is +# not specified, then the plugin is isolated by default. +isolated=true +# diff --git a/duniter4j-es-subscription/src/test/es-home/plugins/mapper-attachments/plugin-security.policy b/duniter4j-es-subscription/src/test/es-home/plugins/mapper-attachments/plugin-security.policy new file mode 100644 index 0000000000000000000000000000000000000000..3264766682269e55baae64ca70d1170101260abf --- /dev/null +++ b/duniter4j-es-subscription/src/test/es-home/plugins/mapper-attachments/plugin-security.policy @@ -0,0 +1,34 @@ +/* + * Licensed to Elasticsearch under one or more contributor + * license agreements. See the NOTICE file distributed with + * this work for additional information regarding copyright + * ownership. Elasticsearch licenses this file to you under + * the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + */ + +// NOTE: when modifying this file, look at restrictions in TikaImpl too +grant { + // needed to apply additional sandboxing to tika parsing + permission java.security.SecurityPermission "createAccessControlContext"; + + // TODO: fix PDFBox not to actually install bouncy castle like this + permission java.security.SecurityPermission "putProviderProperty.BC"; + permission java.security.SecurityPermission "insertProvider"; + // needed only on java 7 + permission java.security.SecurityPermission "insertProvider.BC"; + // TODO: fix POI XWPF to not do this: https://bz.apache.org/bugzilla/show_bug.cgi?id=58597 + permission java.lang.reflect.ReflectPermission "suppressAccessChecks"; + // needed by xmlbeans, as part of POI for MS xml docs + permission java.lang.RuntimePermission "getClassLoader"; +}; diff --git a/duniter4j-es-gchange/src/test/java/org/duniter/elasticsearch/TestFixtures.java b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/TestFixtures.java similarity index 87% rename from duniter4j-es-gchange/src/test/java/org/duniter/elasticsearch/TestFixtures.java rename to duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/TestFixtures.java index 36f093303027109c68ee24aa3e8d1b763c865daf..43ba7fe6f35db9544894b8432ebb5f9bce1e30df 100644 --- a/duniter4j-es-gchange/src/test/java/org/duniter/elasticsearch/TestFixtures.java +++ b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/TestFixtures.java @@ -1,4 +1,4 @@ -package org.duniter.elasticsearch; +package org.duniter.elasticsearch.subscription; /* * #%L @@ -25,4 +25,7 @@ package org.duniter.elasticsearch; public class TestFixtures extends org.duniter.core.test.TestFixtures { + public String getEmail() { + return "contact@e-is.pro"; + } } diff --git a/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/TestResource.java b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/TestResource.java new file mode 100644 index 0000000000000000000000000000000000000000..048047751cf00742d91aee40d2fff4fc7028c29c --- /dev/null +++ b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/TestResource.java @@ -0,0 +1,109 @@ +package org.duniter.elasticsearch.subscription; + +/* + * #%L + * UCoin Java Client :: Core API + * %% + * Copyright (C) 2014 - 2015 EIS + * %% + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as + * published by the Free Software Foundation, either version 3 of the + * License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public + * License along with this program. If not, see + * <http://www.gnu.org/licenses/gpl-3.0.html>. + * #L% + */ + + +import org.apache.commons.io.FileUtils; +import org.elasticsearch.bootstrap.Elasticsearch; +import org.junit.runner.Description; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.File; + +public class TestResource extends org.duniter.core.test.TestResource { + + private static final Logger log = LoggerFactory.getLogger(TestResource.class); + + + public static TestResource create() { + return new TestResource(null, true); + } + + public static TestResource createNotStartEs() { + return new TestResource(null, false); + } + + public static TestResource create(boolean startES) { + return new TestResource(null, startES); + } + + public static TestResource create(String configName) { + return new TestResource(configName, true); + } + + public static TestResource create(String configName, boolean startES) { + return new TestResource(configName, startES); + } + + private TestFixtures fixtures = new TestFixtures(); + private final boolean startESNode; + + protected TestResource(String configName, boolean startESNode) { + super(configName); + this.startESNode = startESNode; + } + + public TestFixtures getFixtures() { + return fixtures; + } + + public PluginSettings getPluginSettings() { + return PluginSettings.instance(); + } + + protected void before(Description description) throws Throwable { + super.before(description); + + // Prepare ES home + File esHomeDir = getResourceDirectory("es-home"); + + System.setProperty("es.path.home", esHomeDir.getCanonicalPath()); + + FileUtils.copyDirectory(new File("src/test/es-home"), esHomeDir); + FileUtils.copyDirectory(new File("target/classes"), new File(esHomeDir, "plugins/duniter4j-es-subscription")); + + // Copy dependencies plugins + FileUtils.copyDirectory(new File("../duniter4j-es-core/target/classes"), new File(esHomeDir, "plugins/duniter4j-es-core")); + FileUtils.copyDirectory(new File("../duniter4j-es-user/target/classes"), new File(esHomeDir, "plugins/duniter4j-es-user")); + + if (startESNode) { + Elasticsearch.main(new String[]{"start"}); + } + + /*while(true) { + Thread.sleep(10000); + }*/ + } + + /** + * Return configuration files prefix (i.e. 'allegro-test') + * Could be override by external project + * + * @return the prefix to use to retrieve configuration files + */ + protected String getConfigFilesPrefix() { + return "duniter4j-es-subscription-test"; + } + +} diff --git a/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionServiceTest.java b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionServiceTest.java new file mode 100644 index 0000000000000000000000000000000000000000..81b5babe93438854ac8b42b60fd5afcc71052c19 --- /dev/null +++ b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionServiceTest.java @@ -0,0 +1,168 @@ +package org.duniter.elasticsearch.subscription.service; + +/* + * #%L + * UCoin Java Client :: ElasticSearch Indexer + * %% + * Copyright (C) 2014 - 2016 EIS + * %% + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as + * published by the Free Software Foundation, either version 3 of the + * License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public + * License along with this program. If not, see + * <http://www.gnu.org/licenses/gpl-3.0.html>. + * #L% + */ + +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.google.common.collect.ImmutableList; +import org.duniter.core.client.model.ModelUtils; +import org.duniter.core.client.model.bma.jackson.JacksonUtils; +import org.duniter.core.client.model.elasticsearch.Record; +import org.duniter.core.client.model.local.Wallet; +import org.duniter.core.client.service.ServiceLocator; +import org.duniter.core.exception.TechnicalException; +import org.duniter.core.service.CryptoService; +import org.duniter.core.util.StringUtils; +import org.duniter.core.util.crypto.CryptoUtils; +import org.duniter.core.util.url.URLs; +import org.duniter.elasticsearch.subscription.TestResource; +import org.duniter.elasticsearch.subscription.model.email.EmailSubscription; +import org.duniter.elasticsearch.user.model.UserEvent; +import org.duniter.elasticsearch.user.model.UserEventCodes; +import org.duniter.elasticsearch.user.service.UserEventService; +import org.junit.*; +import org.nuiton.i18n.I18n; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.stringtemplate.v4.ST; +import org.stringtemplate.v4.STGroupFile; + +import java.util.Date; +import java.util.List; +import java.util.Locale; +import java.util.Map; + +/** + * Created by Benoit on 06/05/2015. + */ +public class SubscriptionServiceTest { + private static final Logger log = LoggerFactory.getLogger(SubscriptionServiceTest.class); + + @ClassRule + public static final TestResource resource = TestResource.create(); + + private SubscriptionService service; + private UserEventService userEventService; + private CryptoService cryptoService; + private ObjectMapper objectMapper; + + @Before + public void setUp() throws Exception { + service = ServiceLocator.instance().getBean(SubscriptionService.class); + cryptoService = ServiceLocator.instance().getCryptoService(); + userEventService = ServiceLocator.instance().getBean(UserEventService.class); + objectMapper = JacksonUtils.newObjectMapper(); + } + + @Test + public void create() throws JsonProcessingException { + Wallet wallet = createTestWallet(); + + createAndIndexSubscription(wallet); + + } + + @Test + public void executeEmailSubscriptions() throws Exception{ + Wallet wallet = createTestWallet(); + try { + createAndIndexSubscription(wallet); + } catch(Exception e) { + Assume.assumeNoException(e); + } + + userEventService.indexEvent(Locale.getDefault(), + UserEvent.newBuilder( + UserEvent.EventType.INFO, + UserEventCodes.MEMBER_JOIN.name()) + .setRecipient(wallet.getPubKeyHash()) + .build()) + .get(); + + // wait 10s + Thread.sleep(10000); + + service.executeEmailSubscriptions(); + + // wait 10s + Thread.sleep(10000); + } + + /* -- internal methods -- */ + + protected Wallet createTestWallet() { + Wallet wallet = new Wallet( + resource.getFixtures().getCurrency(), + resource.getFixtures().getUid(), + CryptoUtils.decodeBase58(resource.getFixtures().getUserPublicKey()), + CryptoUtils.decodeBase58(resource.getFixtures().getUserSecretKey())); + + return wallet; + } + + protected EmailSubscription createAndIndexSubscription(Wallet wallet) throws JsonProcessingException { + + EmailSubscription subscription = createEmailSubscription(wallet); + + // Compute full JSON (with hash + signature) + String json = objectMapper.writeValueAsString(subscription); + + String id = service.create(json); + Assert.assertNotNull(id); + + subscription.setId(id); + return subscription; + } + + protected EmailSubscription createEmailSubscription(Wallet wallet) throws JsonProcessingException { + + EmailSubscription subscription = new EmailSubscription(); + subscription.setIssuer(wallet.getPubKeyHash()); + subscription.setTime(System.currentTimeMillis()/1000); + subscription.setRecipient(resource.getPluginSettings().getNodePubkey()); + + // Encrypt email then fill + String email = resource.getPluginSettings().getMailAdmin(); + byte[] nonce = cryptoService.getBoxRandomNonce(); + + EmailSubscription.Content content = EmailSubscription.newContent(); + content.setEmail(email); + String jsonContent = objectMapper.writeValueAsString(content); + + String cypherContent = cryptoService.box(jsonContent, nonce, wallet.getSecKey(), wallet.getPubKey()); + subscription.setRecipientContent(cypherContent); + subscription.setNonce(CryptoUtils.encodeBase58(nonce)); + + // Fill hash + signature + String json = objectMapper.writeValueAsString(subscription); + + json = JacksonUtils.removeAttribute(json, Record.PROPERTY_SIGNATURE); + json = JacksonUtils.removeAttribute(json, Record.PROPERTY_HASH); + + subscription.setHash(cryptoService.hash(json)); + subscription.setSignature(cryptoService.sign(json, wallet.getSecKey())); + + return subscription; + } +} + diff --git a/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionTemplateTest.java b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionTemplateTest.java new file mode 100644 index 0000000000000000000000000000000000000000..c526a6a26dd73dca99d343f6b28ab6f9d2e10fe9 --- /dev/null +++ b/duniter4j-es-subscription/src/test/java/org/duniter/elasticsearch/subscription/service/SubscriptionTemplateTest.java @@ -0,0 +1,129 @@ +package org.duniter.elasticsearch.subscription.service; + +/* + * #%L + * UCoin Java Client :: ElasticSearch Indexer + * %% + * Copyright (C) 2014 - 2016 EIS + * %% + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as + * published by the Free Software Foundation, either version 3 of the + * License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public + * License along with this program. If not, see + * <http://www.gnu.org/licenses/gpl-3.0.html>. + * #L% + */ + +import org.duniter.core.client.model.ModelUtils; +import org.duniter.core.exception.TechnicalException; +import org.duniter.core.test.TestResource; +import org.duniter.elasticsearch.subscription.util.stringtemplate.DateRenderer; +import org.duniter.elasticsearch.subscription.util.stringtemplate.I18nRenderer; +import org.junit.Test; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.stringtemplate.v4.ST; +import org.stringtemplate.v4.STGroup; +import org.stringtemplate.v4.STGroupDir; +import org.stringtemplate.v4.StringRenderer; + +import java.io.File; +import java.io.FileWriter; +import java.util.Date; + +import static org.junit.Assert.assertNotNull; + +/** + * Created by Benoit on 06/05/2015. + */ +public class SubscriptionTemplateTest { + private static final Logger log = LoggerFactory.getLogger(SubscriptionTemplateTest.class); + + private static final boolean verbose = true; + + //@ClassRule + public static final TestResource resource = TestResource.create(); + + @Test + public void testHtmlEmail() throws Exception{ + + try { + STGroup group = new STGroupDir("templates", '$', '$'); + + group.registerRenderer(Date.class, new DateRenderer()); + group.registerRenderer(String.class, new StringRenderer()); + group.registerRenderer(String.class, new I18nRenderer()); + + ST contentEmail = group.getInstanceOf("html_email_content"); + contentEmail.add("issuer", "MyIssuerName"); + contentEmail.add("url", "https://g1.duniter.fr"); + contentEmail.add("senderPubkey", "G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU"); + contentEmail.add("senderName", ModelUtils.minifyPubkey("G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU")); + contentEmail.addAggr("events.{description, time}", new Object[]{"My event description", new Date()}); + assertNotNull(contentEmail); + + ST css_logo = group.getInstanceOf("css_logo"); + assertNotNull(css_logo); + + ST htmlTpl = group.getInstanceOf("html"); + assertNotNull(htmlTpl); + htmlTpl.add("content", contentEmail.render()); + htmlTpl.add("useCss", "true"); + String html = htmlTpl.render(); + + if (verbose) { + System.out.println(html); + } + + //FileWriter fw = new FileWriter(new File(resource.getResourceDirectory("out"), "page.html")); + FileWriter fw = new FileWriter(new File("/home/blavenie/git/duniter4j/duniter4j-es-subscription/src/test/resources/test2.html")); + fw.write(html); + fw.flush(); + fw.close(); + + + } + catch (Exception e) { + throw new TechnicalException(e); + } + } + + @Test + public void testTextEmail() throws Exception{ + + try { + STGroup group = new STGroupDir("templates", '$', '$'); + + group.registerRenderer(Date.class, new DateRenderer()); + group.registerRenderer(String.class, new StringRenderer()); + group.registerRenderer(String.class, new I18nRenderer()); + + ST tpl = group.getInstanceOf("text_email"); + tpl.add("issuer", "MyIssuerName"); + tpl.add("url", "https://g1.duniter.fr"); + tpl.add("senderPubkey", "G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU"); + tpl.add("senderName", ModelUtils.minifyPubkey("G2CBgZBPLe6FSFUgpx2Jf1Aqsgta6iib3vmDRA1yLiqU")); + tpl.addAggr("events.{description, time}", new Object[]{"My event description", new Date()}); + assertNotNull(tpl); + + String text = tpl.render(); + + if (verbose) { + System.out.println(text); + } + + } + catch (Exception e) { + throw new TechnicalException(e); + } + } +} + diff --git a/duniter4j-es-subscription/src/test/resources/META-INF/services/org.duniter.core.beans.Bean b/duniter4j-es-subscription/src/test/resources/META-INF/services/org.duniter.core.beans.Bean new file mode 100644 index 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 diff --git a/duniter4j-es-gchange/src/test/resources/curl_test.sh b/duniter4j-es-subscription/src/test/resources/curl_test.sh similarity index 100% rename from duniter4j-es-gchange/src/test/resources/curl_test.sh rename to duniter4j-es-subscription/src/test/resources/curl_test.sh diff --git a/duniter4j-es-subscription/src/test/resources/duniter4j-es-subscription-test.properties b/duniter4j-es-subscription/src/test/resources/duniter4j-es-subscription-test.properties new file mode 100644 index 0000000000000000000000000000000000000000..c58166c5f88111dc375775b9077638967d1499e6 --- /dev/null +++ b/duniter4j-es-subscription/src/test/resources/duniter4j-es-subscription-test.properties @@ -0,0 +1,2 @@ +# Empty test file +# (need for inherited TestResource). See files 'src/test/es-home/config' \ No newline at end of file diff --git a/duniter4j-es-gchange/src/test/resources/log4j.properties b/duniter4j-es-subscription/src/test/resources/log4j.properties similarity index 68% rename from duniter4j-es-gchange/src/test/resources/log4j.properties rename to duniter4j-es-subscription/src/test/resources/log4j.properties index 2712b72e0f06c247e8b96a4b1265f95105fda739..29c857ca161ac4bb1d91bc0576252fc828c44911 100644 --- a/duniter4j-es-gchange/src/test/resources/log4j.properties +++ b/duniter4j-es-subscription/src/test/resources/log4j.properties @@ -9,10 +9,9 @@ log4j.appender.stdout.layout.ConversionPattern=%d{ISO8601} %5p (%c:%L) - [%t] %m # duniter4j levels log4j.logger.org.duniter=INFO -#log4j.logger.org.duniter=DEBUG -log4j.logger.org.duniter.core=WARN -log4j.logger.org.duniter.elasticsearch=DEBUG +log4j.logger.org.duniter.elasticsearch.subscription=DEBUG # Other frameworks levels -log4j.logger.org.elasticsearch=INFO - +log4j.logger.org.elasticsearch=WARN +log4j.logger.org.apache=ERROR +log4j.logger.org.nuiton.converter=ERROR \ No newline at end of file diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginInit.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginInit.java index d06b1f759ca28f5ad87282e82ba066d9cc50e1c6..0e554a0d8d79146bf6adf2ad1dc2facc7166697d 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginInit.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginInit.java @@ -22,6 +22,9 @@ package org.duniter.elasticsearch.user; * #L% */ +import org.duniter.core.util.StringUtils; +import org.duniter.core.util.crypto.CryptoUtils; +import org.duniter.core.util.crypto.KeyPair; import org.duniter.elasticsearch.PluginSettings; import org.duniter.elasticsearch.threadpool.ThreadPool; import org.duniter.elasticsearch.user.model.UserEvent; @@ -61,18 +64,9 @@ public class PluginInit extends AbstractLifecycleComponent<PluginInit> { threadPool.scheduleOnClusterHealthStatus(() -> { createIndices(); - // Waiting cluster back to GREEN or YELLOW state, before synchronize - threadPool.scheduleOnClusterHealthStatus(() -> { - synchronize(); - - // Notify admin - injector.getInstance(UserEventService.class) - .notifyAdmin(new UserEvent( - UserEvent.EventType.INFO, - UserEventCodes.NODE_STARTED.name(), - I18n.n("duniter.event.NODE_STARTED"), - clusterName)); - }, ClusterHealthStatus.YELLOW, ClusterHealthStatus.GREEN); + // Waiting cluster back to GREEN or YELLOW state, before doAfterStart + threadPool.scheduleOnClusterHealthStatus(this::doAfterStart, ClusterHealthStatus.YELLOW, ClusterHealthStatus.GREEN); + }, ClusterHealthStatus.YELLOW, ClusterHealthStatus.GREEN); } @@ -130,10 +124,20 @@ public class PluginInit extends AbstractLifecycleComponent<PluginInit> { } } - protected void synchronize() { + protected void doAfterStart() { if (pluginSettings.enableDataSync()) { // Synchronize injector.getInstance(SynchroService.class).synchronize(); } + + // Notify admin + injector.getInstance(AdminService.class) + .notifyAdmin(new UserEvent( + UserEvent.EventType.INFO, + UserEventCodes.NODE_STARTED.name(), + I18n.n("duniter.user.event.NODE_STARTED"), + clusterName)); } + + } diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginSettings.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginSettings.java index b727961d8ca0c2c453f8a3f3b5df6dee6792ab91..4ff4731a08566904f3d7d3b345bb3672d126ae07 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginSettings.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/PluginSettings.java @@ -23,11 +23,18 @@ package org.duniter.elasticsearch.user; */ +import org.duniter.core.service.CryptoService; +import org.duniter.core.util.StringUtils; +import org.duniter.core.util.crypto.CryptoUtils; +import org.duniter.core.util.crypto.KeyPair; import org.elasticsearch.common.component.AbstractLifecycleComponent; import org.elasticsearch.common.component.LifecycleListener; import org.elasticsearch.common.inject.Inject; +import org.elasticsearch.common.inject.Injector; import org.elasticsearch.common.settings.Settings; +import java.util.Locale; + /** * Access to configuration options * @author Benoit Lavenier <benoit.lavenier@e-is.pro> @@ -36,11 +43,19 @@ import org.elasticsearch.common.settings.Settings; public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> { private org.duniter.elasticsearch.PluginSettings delegate; + private CryptoService cryptoService; + + private KeyPair nodeKeyPair; + private boolean isRandomNodeKeyPair; + private String nodePubkey; @Inject - public PluginSettings(Settings settings, org.duniter.elasticsearch.PluginSettings delegate) { + public PluginSettings(Settings settings, + org.duniter.elasticsearch.PluginSettings delegate, + CryptoService cryptoService) { super(settings); this.delegate = delegate; + this.cryptoService = cryptoService; // Add i18n bundle name delegate.addI18nBundleName(getI18nBundleName()); @@ -97,6 +112,14 @@ public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> { return settings.get("duniter.mail.smtp.password"); } + public boolean isMailSmtpStartTLS() { + return settings.getAsBoolean("duniter.mail.smtp.starttle", false); + } + + public boolean isMailSmtpUseSSL() { + return settings.getAsBoolean("duniter.mail.smtp.ssl", false); + } + public String getMailAdmin() { return settings.get("duniter.mail.admin"); } @@ -106,7 +129,7 @@ public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> { } public String getMailSubjectPrefix() { - return settings.get("duniter.mail.subject.prefix", "[Duniter4j ES]"); + return settings.get("duniter.mail.subject.prefix", "[Cesium+]"); } /* -- delegate methods -- */ @@ -151,6 +174,25 @@ public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> { delegate.addI18nBundleName(bundleName); } + public Locale getI18nLocale() { + return delegate.getI18nLocale(); + } + + public KeyPair getNodeKeypair() { + initNodeKeyring(); + return this.nodeKeyPair; + } + + public boolean isRandomNodeKeypair() { + initNodeKeyring(); + return this.isRandomNodeKeyPair; + } + + public String getNodePubkey() { + initNodeKeyring(); + return this.nodePubkey; + } + /* -- protected methods -- */ @@ -158,5 +200,26 @@ public class PluginSettings extends AbstractLifecycleComponent<PluginSettings> { return "duniter4j-es-user-i18n"; } + protected void initNodeKeyring() { + if (this.nodeKeyPair != null) return; + if (StringUtils.isNotBlank(getKeyringSalt()) && + StringUtils.isNotBlank(getKeyringPassword())) { + this.nodeKeyPair = cryptoService.getKeyPair(getKeyringSalt(), getKeyringPassword()); + this.nodePubkey = CryptoUtils.encodeBase58(this.nodeKeyPair.getPubKey()); + this.isRandomNodeKeyPair = false; + } + else { + // Use a ramdom keypair + this.nodeKeyPair = cryptoService.getRandomKeypair(); + this.nodePubkey = CryptoUtils.encodeBase58(this.nodeKeyPair.getPubKey()); + this.isRandomNodeKeyPair = true; + + logger.warn(String.format("No keyring in config. salt/password (or keyring) is need to signed user event documents. Will use a generated key [%s]", this.nodePubkey)); + if (logger.isDebugEnabled()) { + logger.debug(String.format(" salt: " + getKeyringSalt().replaceAll(".", "*"))); + logger.debug(String.format("password: " + getKeyringPassword().replaceAll(".", "*"))); + } + } + } } diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/AdminService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/AdminService.java new file mode 100644 index 0000000000000000000000000000000000000000..e014f885d74999974f92d7ed4ef1c7cf300f4846 --- /dev/null +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/AdminService.java @@ -0,0 +1,105 @@ +package org.duniter.elasticsearch.user.service; + +/* + * #%L + * UCoin Java Client :: Core API + * %% + * Copyright (C) 2014 - 2015 EIS + * %% + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as + * published by the Free Software Foundation, either version 3 of the + * License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public + * License along with this program. If not, see + * <http://www.gnu.org/licenses/gpl-3.0.html>. + * #L% + */ + + +import org.duniter.core.service.CryptoService; +import org.duniter.core.util.Preconditions; +import org.duniter.core.util.StringUtils; +import org.duniter.elasticsearch.client.Duniter4jClient; +import org.duniter.elasticsearch.user.PluginSettings; +import org.duniter.elasticsearch.user.model.UserEvent; +import org.duniter.elasticsearch.user.model.UserProfile; +import org.elasticsearch.common.inject.Inject; +import org.nuiton.i18n.I18n; + +import java.util.Locale; + +/** + * Created by Benoit on 30/03/2015. + */ +public class AdminService extends AbstractService { + + private final UserEventService userEventService; + private final MailService mailService; + + @Inject + public AdminService(final Duniter4jClient client, + final PluginSettings pluginSettings, + final CryptoService cryptoService, + final UserEventService userEventService, + final MailService mailService) { + super("duniter.admin", client, pluginSettings, cryptoService); + this.userEventService = userEventService; + this.mailService = mailService; + } + + /** + * Notify cluster admin + */ + public void notifyAdmin(UserEvent event) { + Preconditions.checkNotNull(event); + + String nodePubkey = pluginSettings.getNodePubkey(); + + UserProfile adminProfile; + if (StringUtils.isNotBlank(nodePubkey) && !pluginSettings.isRandomNodeKeypair()) { + adminProfile = getUserProfile(nodePubkey, UserProfile.PROPERTY_EMAIL, UserProfile.PROPERTY_LOCALE); + } + else { + adminProfile = new UserProfile(); + } + + // Add new event to index + Locale locale = StringUtils.isNotBlank(adminProfile.getLocale()) ? + new Locale(adminProfile.getLocale()) : + I18n.getDefaultLocale(); + if (StringUtils.isNotBlank(nodePubkey)) { + event.setRecipient(nodePubkey); + userEventService.indexEvent(locale, event); + } + + // Send email to admin + String adminEmail = StringUtils.isNotBlank(adminProfile.getEmail()) ? + adminProfile.getEmail() : + pluginSettings.getMailAdmin(); + if (StringUtils.isNotBlank(adminEmail)) { + String subjectPrefix = pluginSettings.getMailSubjectPrefix(); + mailService.sendTextEmail( + I18n.l(locale, "duniter4j.event.subject."+event.getType().name(), subjectPrefix), + event.getLocalizedMessage(locale), + adminEmail); + } + } + + /* -- Internal methods -- */ + + private UserProfile getUserProfile(String pubkey, String... fieldnames) { + UserProfile result = client.getSourceByIdOrNull(UserService.INDEX, UserService.PROFILE_TYPE, pubkey, UserProfile.class, fieldnames); + if (result == null) result = new UserProfile(); + return result; + } + + + +} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/BlockchainUserEventService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/BlockchainUserEventService.java index 2c1a3ccf526deb508f5062490739143a1be595b5..3c2e5fe8d62850bf018b989573cf0ee0c51678ac 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/BlockchainUserEventService.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/BlockchainUserEventService.java @@ -63,6 +63,8 @@ public class BlockchainUserEventService extends AbstractService implements Chang public final UserEventService userEventService; + public final AdminService adminService; + public final ObjectMapper objectMapper; @@ -72,9 +74,11 @@ public class BlockchainUserEventService extends AbstractService implements Chang public BlockchainUserEventService(Duniter4jClient client, PluginSettings settings, CryptoService cryptoService, BlockchainService blockchainService, UserService userService, + AdminService adminService, UserEventService userEventService) { super("duniter.user.event.blockchain", client, settings, cryptoService); this.userService = userService; + this.adminService = adminService; this.userEventService = userEventService; this.objectMapper = JacksonUtils.newObjectMapper(); ChangeService.registerListener(this); @@ -150,8 +154,8 @@ public class BlockchainUserEventService extends AbstractService implements Chang // Send notify on reconnection if (errorNotified) { errorNotified = false; - userEventService.notifyAdmin(UserEvent.newBuilder(UserEvent.EventType.INFO, UserEventCodes.NODE_BMA_UP.name()) - .setMessage(I18n.n("duniter.event.NODE_BMA_UP"), + adminService.notifyAdmin(UserEvent.newBuilder(UserEvent.EventType.INFO, UserEventCodes.NODE_BMA_UP.name()) + .setMessage(I18n.n("duniter.user.event.NODE_BMA_UP"), pluginSettings.getNodeBmaHost(), String.valueOf(pluginSettings.getNodeBmaPort()), pluginSettings.getClusterName()) @@ -168,8 +172,8 @@ public class BlockchainUserEventService extends AbstractService implements Chang boolean wait = now - lastTimeUp < 60; if (!wait) { errorNotified = true; - userEventService.notifyAdmin(UserEvent.newBuilder(UserEvent.EventType.ERROR, UserEventCodes.NODE_BMA_DOWN.name()) - .setMessage(I18n.n("duniter.event.NODE_BMA_DOWN"), + adminService.notifyAdmin(UserEvent.newBuilder(UserEvent.EventType.ERROR, UserEventCodes.NODE_BMA_DOWN.name()) + .setMessage(I18n.n("duniter.user.event.NODE_BMA_DOWN"), pluginSettings.getNodeBmaHost(), String.valueOf(pluginSettings.getNodeBmaPort()), pluginSettings.getClusterName(), @@ -184,21 +188,21 @@ public class BlockchainUserEventService extends AbstractService implements Chang // Joiners if (CollectionUtils.isNotEmpty(block.getJoiners())) { for (BlockchainBlock.Joiner joiner: block.getJoiners()) { - notifyUserEvent(block, joiner.getPublicKey(), UserEventCodes.MEMBER_JOIN, I18n.n("duniter.user.event.ms.join"), block.getCurrency()); + notifyUserEvent(block, joiner.getPublicKey(), UserEventCodes.MEMBER_JOIN, I18n.n("duniter.user.event.MEMBER_JOIN"), block.getCurrency()); } } // Leavers if (CollectionUtils.isNotEmpty(block.getLeavers())) { for (BlockchainBlock.Joiner leaver: block.getJoiners()) { - notifyUserEvent(block, leaver.getPublicKey(), UserEventCodes.MEMBER_LEAVE, I18n.n("duniter.user.event.ms.leave"), block.getCurrency()); + notifyUserEvent(block, leaver.getPublicKey(), UserEventCodes.MEMBER_LEAVE, I18n.n("duniter.user.event.MEMBER_LEAVE"), block.getCurrency()); } } // Actives if (CollectionUtils.isNotEmpty(block.getActives())) { for (BlockchainBlock.Joiner active: block.getActives()) { - notifyUserEvent(block, active.getPublicKey(), UserEventCodes.MEMBER_ACTIVE, I18n.n("duniter.user.event.ms.active"), block.getCurrency()); + notifyUserEvent(block, active.getPublicKey(), UserEventCodes.MEMBER_ACTIVE, I18n.n("duniter.user.event.MEMBER_ACTIVE"), block.getCurrency()); } } @@ -239,7 +243,7 @@ public class BlockchainUserEventService extends AbstractService implements Chang if (parts.length >= 3 && parts[2].startsWith("SIG(")) { String receiver = parts[2].substring(4, parts[2].length() - 1); if (!senders.contains(receiver) && !receivers.contains(receiver)) { - notifyUserEvent(block, receiver, UserEventCodes.TX_RECEIVED, I18n.n("duniter.user.event.tx.received"), sendersPubkeys, senderNames); + notifyUserEvent(block, receiver, UserEventCodes.TX_RECEIVED, I18n.n("duniter.user.event.TX_RECEIVED"), sendersPubkeys, senderNames); receivers.add(receiver); } } @@ -250,7 +254,7 @@ public class BlockchainUserEventService extends AbstractService implements Chang String receiverNames = userService.joinNamesFromPubkeys(receivers, DEFAULT_PUBKEYS_SEPARATOR, true); String receiverPubkeys = ModelUtils.joinPubkeys(receivers, DEFAULT_PUBKEYS_SEPARATOR, false); for (String sender : senders) { - notifyUserEvent(block, sender, UserEventCodes.TX_SENT, I18n.n("duniter.user.event.tx.sent"), receiverPubkeys, receiverNames); + notifyUserEvent(block, sender, UserEventCodes.TX_SENT, I18n.n("duniter.user.event.TX_SENT"), receiverPubkeys, receiverNames); } } @@ -266,14 +270,14 @@ public class BlockchainUserEventService extends AbstractService implements Chang if (senderName == null) { senderName = ModelUtils.minifyPubkey(sender); } - notifyUserEvent(block, receiver, UserEventCodes.CERT_RECEIVED, I18n.n("duniter.user.event.cert.received"), sender, senderName); + notifyUserEvent(block, receiver, UserEventCodes.CERT_RECEIVED, I18n.n("duniter.user.event.CERT_RECEIVED"), sender, senderName); // Sent String receiverName = userService.getProfileTitle(receiver); if (receiverName == null) { receiverName = ModelUtils.minifyPubkey(receiver); } - notifyUserEvent(block, sender, UserEventCodes.CERT_SENT, I18n.n("duniter.user.event.cert.sent"), receiver, receiverName); + notifyUserEvent(block, sender, UserEventCodes.CERT_SENT, I18n.n("duniter.user.event.CERT_SENT"), receiver, receiverName); } private void notifyUserEvent(BlockchainBlock block, String pubkey, UserEventCodes code, String message, String... params) { diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/GroupService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/GroupService.java index 51e6a05802f55746fd1d6de6e3253b21fa2344fd..53c0604b4a5b7f3674045711ee97c36a9279556c 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/GroupService.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/GroupService.java @@ -64,7 +64,7 @@ public class GroupService extends AbstractService { } /** - * Create index need for blockchain registry, if need + * Create index need for blockchain mail, if need */ public GroupService createIndexIfNotExists() { try { @@ -79,7 +79,7 @@ public class GroupService extends AbstractService { } /** - * Create index for registry + * Create index for mail * @throws JsonProcessingException */ public GroupService createIndex() throws JsonProcessingException { diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/HistoryService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/HistoryService.java index 7df495965f4ff5270db75af9aa67eca70672cfa9..e2004b8a5a11d5bf93e8db584b19a7b6420c1c66 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/HistoryService.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/HistoryService.java @@ -59,7 +59,7 @@ public class HistoryService extends AbstractService { @Inject public HistoryService(Duniter4jClient client, PluginSettings settings, CryptoService cryptoService) { - super("gchange." + INDEX, client, settings, cryptoService); + super("subscription." + INDEX, client, settings, cryptoService); } /** @@ -77,7 +77,7 @@ public class HistoryService extends AbstractService { } /** - * Create index need for blockchain registry, if need + * Create index need for blockchain mail, if need */ public HistoryService createIndexIfNotExists() { try { @@ -93,7 +93,7 @@ public class HistoryService extends AbstractService { } /** - * Create index need for category registry + * Create index need for category mail * @throws JsonProcessingException */ public HistoryService createIndex() throws JsonProcessingException { @@ -117,9 +117,9 @@ public class HistoryService extends AbstractService { JsonNode actualObj = readAndVerifyIssuerSignature(recordJson); String issuer = actualObj.get(DeleteRecord.PROPERTY_ISSUER).asText(); - String index = actualObj.get(DeleteRecord.PROPERTY_INDEX).asText(); - String type = actualObj.get(DeleteRecord.PROPERTY_TYPE).asText(); - String id = actualObj.get(DeleteRecord.PROPERTY_ID).asText(); + String index = getMandatoryField(actualObj, DeleteRecord.PROPERTY_INDEX).asText(); + String type = getMandatoryField(actualObj,DeleteRecord.PROPERTY_TYPE).asText(); + String id = getMandatoryField(actualObj,DeleteRecord.PROPERTY_ID).asText(); if (!client.existsIndex(index)) { throw new NotFoundException(String.format("Index [%s] not exists.", index)); diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/MailService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/MailService.java new file mode 100644 index 0000000000000000000000000000000000000000..9b259dcb24c06522ad83da3905efb97626926cd5 --- /dev/null +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/MailService.java @@ -0,0 +1,139 @@ +package org.duniter.elasticsearch.user.service; + +/* + * #%L + * UCoin Java Client :: Core API + * %% + * Copyright (C) 2014 - 2015 EIS + * %% + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as + * published by the Free Software Foundation, either version 3 of the + * License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public + * License along with this program. If not, see + * <http://www.gnu.org/licenses/gpl-3.0.html>. + * #L% + */ + + +import org.duniter.core.exception.TechnicalException; +import org.duniter.core.model.SmtpConfig; +import org.duniter.core.service.CryptoService; +import org.duniter.core.util.Preconditions; +import org.duniter.core.util.StringUtils; +import org.duniter.core.util.crypto.CryptoUtils; +import org.duniter.core.util.crypto.KeyPair; +import org.duniter.elasticsearch.client.Duniter4jClient; +import org.duniter.elasticsearch.service.changes.ChangeService; +import org.duniter.elasticsearch.user.PluginSettings; +import org.duniter.elasticsearch.user.model.UserEvent; +import org.duniter.elasticsearch.user.model.UserProfile; +import org.elasticsearch.common.inject.Inject; +import org.nuiton.i18n.I18n; + +import javax.activation.CommandMap; +import javax.activation.MailcapCommandMap; +import java.util.Locale; + +/** + * Created by Benoit on 30/03/2015. + */ +public class MailService extends AbstractService { + + private final org.duniter.core.service.MailService delegate; + + private final boolean enable; + + @Inject + public MailService(final Duniter4jClient client, + final PluginSettings pluginSettings, + final CryptoService cryptoService, + final org.duniter.core.service.MailService delegate) { + super("duniter.mail", client, pluginSettings, cryptoService); + this.delegate = delegate; + this.enable = pluginSettings.getMailEnable(); + // Init delegated service + if (this.enable) { + delegate.setSmtpConfig(createConfig(pluginSettings)); + } + } + + /** + * Send email + */ + public void sendTextEmail(String subject, String textContent, String... recipients) { + if (!this.enable) return; + + try { + delegate.sendTextEmail(subject, textContent, recipients); + } + catch(TechnicalException e) { + if (logger.isDebugEnabled()) { + logger.error(e.getMessage(), e); + } + else { + logger.error(e.getMessage()); + } + } + } + + /** + * Send email + */ + public void sendHtmlEmail(String subject, String htmlContent, String... recipients) { + if (!this.enable) return; + + try { + delegate.sendHtmlEmail(subject, htmlContent, recipients); + } + catch(TechnicalException e) { + if (logger.isDebugEnabled()) { + logger.error(e.getMessage(), e); + } + else { + logger.error(e.getMessage()); + } + } + } + + /** + * Send email + */ + public void sendHtmlEmailWithText(String subject, String textContent, String htmlContent, String... recipients) { + if (!this.enable) return; + + try { + delegate.sendHtmlEmailWithText(subject, textContent, htmlContent, recipients); + } + catch(TechnicalException e) { + if (logger.isDebugEnabled()) { + logger.error(e.getMessage(), e); + } + else { + logger.error(e.getMessage()); + } + } + } + + /* -- internal methods -- */ + + protected SmtpConfig createConfig(PluginSettings pluginSettings) { + SmtpConfig config = new SmtpConfig(); + config.setSmtpHost(pluginSettings.getMailSmtpHost()); + config.setSmtpPort(pluginSettings.getMailSmtpPort()); + config.setSmtpUsername(pluginSettings.getMailSmtpUsername()); + config.setSmtpPassword(pluginSettings.getMailSmtpPassword()); + config.setSenderAddress(pluginSettings.getMailFrom()); + config.setStartTLS(pluginSettings.isMailSmtpStartTLS()); + config.setUseSsl(pluginSettings.isMailSmtpUseSSL()); + return config; + } + +} diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/MessageService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/MessageService.java index 5a4a2a3835e61a0fd9fd06915c79023833da29dc..b1424cd27bf610afc33cf9b2aef92dbf8946d3b5 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/MessageService.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/MessageService.java @@ -84,7 +84,7 @@ public class MessageService extends AbstractService { } /** - * Create index need for blockchain registry, if need + * Create index need for blockchain mail, if need */ public MessageService createIndexIfNotExists() { try { @@ -100,7 +100,7 @@ public class MessageService extends AbstractService { } /** - * Create index need for category registry + * Create index need for category mail * @throws JsonProcessingException */ public MessageService createIndex() throws JsonProcessingException { @@ -231,6 +231,18 @@ public class MessageService extends AbstractService { .field("index", "not_analyzed") .endObject() + // hash + .startObject("hash") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + + // signature + .startObject("signature") + .field("type", "string") + .field("index", "not_analyzed") + .endObject() + // read_signature .startObject("read_signature") .field("type", "string") diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/ServiceModule.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/ServiceModule.java index e1986861b8cfa378e7adcfd6eb8dc2c632bf4393..af46406ab34de48a845fa5720d8ad5a303830111 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/ServiceModule.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/ServiceModule.java @@ -35,6 +35,8 @@ public class ServiceModule extends AbstractModule implements Module { bind(UserService.class).asEagerSingleton(); bind(GroupService.class).asEagerSingleton(); + bind(AdminService.class).asEagerSingleton(); + bind(MailService.class).asEagerSingleton(); bind(UserEventService.class).asEagerSingleton(); bind(UserInvitationService.class).asEagerSingleton(); diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserEventService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserEventService.java index eee64c1b4c3be56fde488512f82416e18710f831..656790b332ae4fbb8b45b9da16a3ee2ea3ca0af7 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserEventService.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserEventService.java @@ -26,13 +26,14 @@ package org.duniter.elasticsearch.user.service; import com.fasterxml.jackson.core.JsonProcessingException; import com.fasterxml.jackson.databind.JsonNode; import com.google.common.collect.ImmutableList; +import org.duniter.core.client.model.bma.jackson.JacksonUtils; +import org.duniter.core.client.model.elasticsearch.Record; import org.duniter.core.exception.TechnicalException; import org.duniter.core.service.CryptoService; import org.duniter.core.service.MailService; +import org.duniter.core.util.CollectionUtils; import org.duniter.core.util.Preconditions; import org.duniter.core.util.StringUtils; -import org.duniter.core.util.crypto.CryptoUtils; -import org.duniter.core.util.crypto.KeyPair; import org.duniter.elasticsearch.client.Duniter4jClient; import org.duniter.elasticsearch.exception.InvalidSignatureException; import org.duniter.elasticsearch.service.changes.ChangeEvent; @@ -51,17 +52,17 @@ import org.elasticsearch.action.search.SearchRequestBuilder; import org.elasticsearch.action.search.SearchResponse; import org.elasticsearch.action.search.SearchType; import org.elasticsearch.action.update.UpdateResponse; -import org.elasticsearch.client.Client; import org.elasticsearch.common.inject.Inject; import org.elasticsearch.common.xcontent.XContentBuilder; import org.elasticsearch.common.xcontent.XContentFactory; import org.elasticsearch.index.query.BoolQueryBuilder; +import org.elasticsearch.index.query.QueryBuilder; import org.elasticsearch.index.query.QueryBuilders; import org.elasticsearch.search.SearchHit; -import org.nuiton.i18n.I18n; import java.io.IOException; import java.util.*; +import java.util.stream.Collectors; /** * Created by Benoit on 30/03/2015. @@ -96,9 +97,6 @@ public class UserEventService extends AbstractService implements ChangeService.C private final MailService mailService; private final ThreadPool threadPool; - public final KeyPair nodeKeyPair; - public final String nodePubkey; - public final boolean mailEnable; public final boolean trace; @Inject @@ -110,53 +108,12 @@ public class UserEventService extends AbstractService implements ChangeService.C super("duniter.user.event", client, pluginSettings, cryptoService); this.mailService = mailService; this.threadPool = threadPool; - this.nodeKeyPair = getNodeKeyPairOrNull(pluginSettings); - this.nodePubkey = getNodePubKey(nodeKeyPair); - this.mailEnable = pluginSettings.getMailEnable(); this.trace = logger.isTraceEnabled(); - if (!this.mailEnable && this.trace) { - logger.trace("Mail disable"); - } ChangeService.registerListener(this); } - /** - * Notify cluster admin - */ - public void notifyAdmin(UserEvent event) { - Preconditions.checkNotNull(event); - - UserProfile adminProfile; - if (StringUtils.isNotBlank(nodePubkey)) { - adminProfile = getUserProfile(nodePubkey, UserProfile.PROPERTY_EMAIL, UserProfile.PROPERTY_LOCALE); - } - else { - adminProfile = new UserProfile(); - } - - // Add new event to index - Locale locale = StringUtils.isNotBlank(adminProfile.getLocale()) ? - new Locale(adminProfile.getLocale()) : - I18n.getDefaultLocale(); - if (StringUtils.isNotBlank(nodePubkey)) { - event.setRecipient(nodePubkey); - indexEvent(locale, event); - } - - // Send email to admin - String adminEmail = StringUtils.isNotBlank(adminProfile.getEmail()) ? - adminProfile.getEmail() : - pluginSettings.getMailAdmin(); - if (StringUtils.isNotBlank(adminEmail)) { - String subjectPrefix = pluginSettings.getMailSubjectPrefix(); - sendEmail(adminEmail, - I18n.l(locale, "duniter4j.event.subject."+event.getType().name(), subjectPrefix), - event.getLocalizedMessage(locale)); - } - } - /** * Notify a user */ @@ -179,6 +136,8 @@ public class UserEventService extends AbstractService implements ChangeService.C Preconditions.checkNotNull(event.getType()); Preconditions.checkNotNull(event.getCode()); + String nodePubkey = pluginSettings.getNodePubkey(); + // Generate json String eventJson; if (StringUtils.isNotBlank(nodePubkey)) { @@ -186,13 +145,18 @@ public class UserEventService extends AbstractService implements ChangeService.C signedEvent.setMessage(event.getLocalizedMessage(locale)); // set issuer, hash, signature signedEvent.setIssuer(nodePubkey); - String hash = cryptoService.hash(toJson(signedEvent)); + + // Add hash + String hash = cryptoService.hash(toJson(signedEvent, true)); signedEvent.setHash(hash); - String signature = cryptoService.sign(toJson(signedEvent), nodeKeyPair.getSecKey()); + + // Add signature + String signature = cryptoService.sign(toJson(signedEvent, true), pluginSettings.getNodeKeypair().getSecKey()); signedEvent.setSignature(signature); + eventJson = toJson(signedEvent); } else { - logger.debug("Could not generate hash for new user event (no keyring)"); + logger.warn("Could not generate hash for new user event (no keyring)"); // Node has not keyring: do NOT sign it eventJson = event.toJson(locale); } @@ -254,8 +218,40 @@ public class UserEventService extends AbstractService implements ChangeService.C .execute(); } + + + public List<UserEvent> getUserEvents(String pubkey, Long lastTime, String[] includesCodes, String[] excludesCodes) { + + BoolQueryBuilder query = QueryBuilders.boolQuery() + .must(QueryBuilders.termQuery(UserEvent.PROPERTY_RECIPIENT, pubkey)); + if (lastTime != null) { + query.must(QueryBuilders.rangeQuery(UserEvent.PROPERTY_TIME).gt(lastTime)); + } + + if (CollectionUtils.isNotEmpty(includesCodes)) { + query.must(QueryBuilders.termsQuery(UserEvent.PROPERTY_CODE, includesCodes)); + } + if (CollectionUtils.isNotEmpty(excludesCodes)) { + query.mustNot(QueryBuilders.termsQuery(UserEvent.PROPERTY_CODE, excludesCodes)); + } + + SearchResponse response = client.prepareSearch(INDEX) + .setTypes(EVENT_TYPE) + .setSearchType(SearchType.DFS_QUERY_THEN_FETCH) + .setFetchSource(true) + .setQuery(query) + .get(); + + + return Arrays.asList(response.getHits().getHits()).stream() + .map(searchHit -> client.readSourceOrNull(searchHit, UserEvent.class)) + .filter(Objects::nonNull) + .collect(Collectors.toList()); + } + /* -- Internal methods -- */ + public static XContentBuilder createEventType() { try { XContentBuilder mapping = XContentFactory.jsonBuilder().startObject().startObject(EVENT_TYPE) @@ -356,54 +352,12 @@ public class UserEventService extends AbstractService implements ChangeService.C } } - - /** - * Send email - */ - private void sendEmail(String recipients, String subject, String textContent) { - if (!this.mailEnable) return; - - String smtpHost = pluginSettings.getMailSmtpHost(); - int smtpPort = pluginSettings.getMailSmtpPort(); - String smtpUsername = pluginSettings.getMailSmtpUsername(); - String smtpPassword = pluginSettings.getMailSmtpPassword(); - String from = pluginSettings.getMailFrom(); - - try { - mailService.sendTextEmail(smtpHost, smtpPort, smtpUsername, smtpPassword, from, recipients, subject, textContent); - } - catch(TechnicalException e) { - logger.error(String.format("Could not send email: %s", e.getMessage())/*, e*/); - } - } - - private KeyPair getNodeKeyPairOrNull(PluginSettings pluginSettings) { - KeyPair result; - if (StringUtils.isNotBlank(pluginSettings.getKeyringSalt()) && - StringUtils.isNotBlank(pluginSettings.getKeyringPassword())) { - result = cryptoService.getKeyPair(pluginSettings.getKeyringSalt(), - pluginSettings.getKeyringPassword()); - } - else { - // Use a ramdom keypair - result = cryptoService.getRandomKeypair(); - logger.warn(String.format("No keyring in config. salt/password (or keyring) is need to signed user event documents. Will use a generated key [%s]", getNodePubKey(result))); - if (logger.isDebugEnabled()) { - logger.debug(String.format(" salt: " + pluginSettings.getKeyringSalt().replaceAll(".", "*"))); - logger.debug(String.format("password: " + pluginSettings.getKeyringPassword().replaceAll(".", "*"))); - } - } - + private UserProfile getUserProfile(String pubkey, String... fieldnames) { + UserProfile result = client.getSourceByIdOrNull(UserService.INDEX, UserService.PROFILE_TYPE, pubkey, UserProfile.class, fieldnames); + if (result == null) result = new UserProfile(); return result; } - private String getNodePubKey(KeyPair nodeKeyPair) { - if (nodeKeyPair == null) return null; - return CryptoUtils.encodeBase58(nodeKeyPair.getPubKey()); - } - - - private void doDeleteEventsByReference(final UserEvent.Reference reference) { // Prepare search request @@ -461,19 +415,23 @@ public class UserEventService extends AbstractService implements ChangeService.C } } - private UserProfile getUserProfile(String pubkey, String... fieldnames) { - UserProfile result = client.getSourceByIdOrNull(UserService.INDEX, UserService.PROFILE_TYPE, pubkey, UserProfile.class, fieldnames); - if (result == null) result = new UserProfile(); - return result; - } private UserProfile getUserProfileOrNull(String pubkey, String... fieldnames) { return client.getSourceByIdOrNull(UserService.INDEX, UserService.PROFILE_TYPE, pubkey, UserProfile.class, fieldnames); } private String toJson(UserEvent userEvent) { + return toJson(userEvent, false); + } + + private String toJson(UserEvent userEvent, boolean cleanHashAndSignature) { try { - return objectMapper.writeValueAsString(userEvent); + String json = objectMapper.writeValueAsString(userEvent); + if (cleanHashAndSignature) { + json = JacksonUtils.removeAttribute(json, Record.PROPERTY_SIGNATURE); + json = JacksonUtils.removeAttribute(json, Record.PROPERTY_HASH); + } + return json; } catch(JsonProcessingException e) { throw new TechnicalException("Unable to serialize UserEvent object", e); } diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserInvitationService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserInvitationService.java index 441b7e3106d96bef0ca17b45e13b3088a255d134..f5e393b8cce5b1f585d40aaf430bf6e63dd32eca 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserInvitationService.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserInvitationService.java @@ -71,7 +71,7 @@ public class UserInvitationService extends AbstractService { } /** - * Create index need for blockchain registry, if need + * Create index need for blockchain mail, if need */ public UserInvitationService createIndexIfNotExists() { try { @@ -87,7 +87,7 @@ public class UserInvitationService extends AbstractService { } /** - * Create index need for category registry + * Create index need for category mail * @throws JsonProcessingException */ public UserInvitationService createIndex() throws JsonProcessingException { @@ -126,7 +126,7 @@ public class UserInvitationService extends AbstractService { // Notify recipient userEventService.notifyUser(UserEvent.newBuilder(UserEvent.EventType.INFO, UserEventCodes.INVITATION_TO_CERTIFY.name()) .setRecipient(recipient) - .setMessage(I18n.n("duniter.invitation.cert.received"), issuer, ModelUtils.minifyPubkey(issuer)) + .setMessage(I18n.n("duniter.user.event.INVITATION_TO_CERTIFY"), issuer, ModelUtils.minifyPubkey(issuer)) .setTime(time) .setReference(INDEX, CERTIFICATION_TYPE, invitationId) .build()); diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserService.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserService.java index 11f62c689b835b425cdbd8eb72d62152e24d64f9..f827dd289b0772da960ff1b8006649d4f141d0e0 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserService.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/service/UserService.java @@ -67,7 +67,7 @@ public class UserService extends AbstractService { } /** - * Create index need for blockchain registry, if need + * Create index need for blockchain mail, if need */ public UserService createIndexIfNotExists() { try { @@ -82,7 +82,7 @@ public class UserService extends AbstractService { } /** - * Create index for registry + * Create index for mail * @throws JsonProcessingException */ public UserService createIndex() throws JsonProcessingException { diff --git a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/websocket/WebsocketUserEventEndPoint.java b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/websocket/WebsocketUserEventEndPoint.java index 5037ac3ea5c9eba991b8c63917fb0c3e2346d857..552300ca04f7ba29539115c77c4190efa96d2174 100644 --- a/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/websocket/WebsocketUserEventEndPoint.java +++ b/duniter4j-es-user/src/main/java/org/duniter/elasticsearch/user/websocket/WebsocketUserEventEndPoint.java @@ -40,6 +40,7 @@ package org.duniter.elasticsearch.user.websocket; import org.duniter.core.client.model.bma.Constants; import org.duniter.core.util.StringUtils; +import org.duniter.elasticsearch.user.PluginSettings; import org.duniter.elasticsearch.user.model.UserEvent; import org.duniter.elasticsearch.user.service.UserEventService; import org.duniter.elasticsearch.websocket.WebSocketServer; @@ -57,11 +58,15 @@ import java.util.regex.Pattern; @ServerEndpoint(value = "/event/user/{pubkey}/{locale}") public class WebsocketUserEventEndPoint implements UserEventService.UserEventListener { + public static Locale defaultLocale; + public static class Init { @Inject - public Init(WebSocketServer webSocketServer) { + public Init(WebSocketServer webSocketServer, PluginSettings pluginSettings) { webSocketServer.addEndPoint(WebsocketUserEventEndPoint.class); + defaultLocale = pluginSettings.getI18nLocale(); + if (defaultLocale == null) defaultLocale = new Locale("en", "GB"); } } @@ -78,7 +83,7 @@ public class WebsocketUserEventEndPoint implements UserEventService.UserEventLis public void onOpen(Session session) { this.session = session; this.pubkey = session.getPathParameters() != null ? session.getPathParameters().get(PATH_PARAM_PUBKEY) : null; - this.locale = new Locale(session.getPathParameters() != null ? session.getPathParameters().get(PATH_PARAM_LOCALE) : "fr"); + this.locale = session.getPathParameters() != null ? new Locale(session.getPathParameters().get(PATH_PARAM_LOCALE)) : defaultLocale; if (StringUtils.isBlank(pubkey) || !PUBKEY_PATTERN.matcher(pubkey).matches()) { try { diff --git a/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_en_GB.properties b/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_en_GB.properties index e149f6a60fe85719287665c08940e908ef3a9a62..829d6574229eae4835e5813427d91df1bc57c467 100644 --- a/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_en_GB.properties +++ b/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_en_GB.properties @@ -1,7 +1,15 @@ -duniter.event.NODE_BMA_DOWN=Duniter node [%1$s\:%2$s] is DOWN\: no access from ES node [%3$s]. Last connexion at %4$d. Blockchain indexation waiting. -duniter.event.NODE_BMA_UP=Duniter node [%1$s\:%2$s] is UP again. -duniter.event.NODE_STARTED=Node started on cluster Duniter4j ES [%s] duniter.invitation.cert.received= +duniter.user.event.CERT_RECEIVED= +duniter.user.event.CERT_SENT= +duniter.user.event.INVITATION_TO_CERTIFY= +duniter.user.event.MEMBER_ACTIVE= +duniter.user.event.MEMBER_JOIN= +duniter.user.event.MEMBER_LEAVE= +duniter.user.event.NODE_BMA_DOWN=Duniter node [%1$s\:%2$s] is DOWN\: no access from ES node [%3$s]. Last connexion at %4$d. Blockchain indexation waiting. +duniter.user.event.NODE_BMA_UP=Duniter node [%1$s\:%2$s] is UP again. +duniter.user.event.NODE_STARTED=Node started on cluster Duniter4j ES [%s] +duniter.user.event.TX_RECEIVED= +duniter.user.event.TX_SENT= duniter.user.event.active= duniter.user.event.cert.received= duniter.user.event.cert.sent= diff --git a/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_fr_FR.properties b/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_fr_FR.properties index ab9df8238fc6befdd3f88d3a7ab63e4798bb1048..74e0b613a31e5aa4ccdf82ef78c9e16d52723c2b 100644 --- a/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_fr_FR.properties +++ b/duniter4j-es-user/src/main/resources/i18n/duniter4j-es-user_fr_FR.properties @@ -1,15 +1,15 @@ -duniter.event.NODE_BMA_DOWN=Noeud Duniter [%1$s\:%2$s] non joignable, depuis le noeud ES API [%3$s]. Dernière connexion à %4$d. Indexation de blockchain en attente. -duniter.event.NODE_BMA_UP=Noeud Duniter [%1$s\:%2$s] à nouveau accessible. -duniter.event.NODE_STARTED=Noeud ES API démarré sur le cluster Duniter [%1$s] -duniter.invitation.cert.received=%2$s vous invite à certifier une identité. -duniter.user.event.cert.received=%2$s vous a certifié (certification prise en compte). -duniter.user.event.cert.sent=Votre certification de %2$s a été pris en compte. +duniter.user.event.CERT_RECEIVED=%2$s vous a certifié (certification prise en compte). +duniter.user.event.CERT_SENT=Votre certification de %2$s a été pris en compte. +duniter.user.event.INVITATION_TO_CERTIFY=%2$s vous invite à certifier une identité. +duniter.user.event.MEMBER_ACTIVE=Votre adhésion comme membre a bien été renouvellée +duniter.user.event.MEMBER_JOIN=Vous êtes maintenant membre de la monnaie +duniter.user.event.MEMBER_LEAVE=Votre adhésion comme membre à expirée +duniter.user.event.NODE_BMA_DOWN=Noeud Duniter [%1$s\:%2$s] non joignable, depuis le noeud ES API [%3$s]. Dernière connexion à %4$d. Indexation de blockchain en attente. +duniter.user.event.NODE_BMA_UP=Noeud Duniter [%1$s\:%2$s] à nouveau accessible. +duniter.user.event.NODE_STARTED=Noeud ES API démarré sur le cluster Duniter [%1$s] +duniter.user.event.TX_RECEIVED=Vous avez recu un paiement de %2$s +duniter.user.event.TX_SENT=Votre paiement à %2$s a bien été executé duniter.user.event.message.received=Vous avez reçu un message de %2$s -duniter.user.event.ms.active=Votre adhésion comme membre a bien été renouvellée -duniter.user.event.ms.join=Vous êtes maintenant membre de la monnaie -duniter.user.event.ms.leave=Votre adhésion comme membre à expirée -duniter.user.event.tx.received=Vous avez recu un paiement de %2$s -duniter.user.event.tx.sent=Votre paiement à %2$s a bien été executé duniter4j.event.subject.ERROR=%s Message d'erreur duniter4j.event.subject.INFO=%s Message d'information duniter4j.event.subject.WARN=%s Message d'avertissement diff --git a/pom.xml b/pom.xml index 5baa771d3bc6176e3c38c0e2ec4a6d54c88fbcd1..f2f19db90ac2f8b41baed46680f9f62da8b55951 100644 --- a/pom.xml +++ b/pom.xml @@ -18,7 +18,7 @@ <slf4j.version>1.7.5</slf4j.version> <guava.version>18.0</guava.version> <xml-apis.version>2.0.2</xml-apis.version> - <kalium.version>0.5.0</kalium.version> + <kalium.version>0.5.1-SNAPSHOT</kalium.version> <scrypt.version>1.4.0</scrypt.version> <elasticsearch.version>2.3.3</elasticsearch.version> <jna.version>4.1.0</jna.version> @@ -107,7 +107,7 @@ <module>duniter4j-client</module> <module>duniter4j-es-core</module> <module>duniter4j-es-user</module> - <module>duniter4j-es-gchange</module> + <module>duniter4j-es-subscription</module> <module>duniter4j-es-assembly</module> </modules> @@ -150,6 +150,11 @@ <artifactId>commons-lang3</artifactId> <version>3.1</version> </dependency> + <dependency> + <groupId>org.antlr</groupId> + <artifactId>stringtemplate</artifactId> + <version>4.0.2</version> + </dependency> <dependency> <groupId>commons-logging</groupId> <artifactId>commons-logging</artifactId>