From 66a9a566c0f6ae2529c1528aa9e1c2356ec12e9e Mon Sep 17 00:00:00 2001 From: Franzz Date: Mon, 6 Apr 2020 17:53:49 +0200 Subject: [PATCH] Replace mail() with PHPMailer + new logo --- inc/PHPMailer/Exception.php | 39 + inc/PHPMailer/OAuth.php | 138 + inc/PHPMailer/PHPMailer.php | 4820 +++++++++++++++++++++++++++++++++++ inc/PHPMailer/POP3.php | 421 +++ inc/PHPMailer/SMTP.php | 1371 ++++++++++ inc/email.php | 59 +- masks/project.html | 1 + settings-sample.php | 3 + style/_mask_project.scss | 25 +- style/spot.css | 2 +- style/spot.css.map | 2 +- 11 files changed, 6875 insertions(+), 6 deletions(-) create mode 100644 inc/PHPMailer/Exception.php create mode 100644 inc/PHPMailer/OAuth.php create mode 100644 inc/PHPMailer/PHPMailer.php create mode 100644 inc/PHPMailer/POP3.php create mode 100644 inc/PHPMailer/SMTP.php diff --git a/inc/PHPMailer/Exception.php b/inc/PHPMailer/Exception.php new file mode 100644 index 0000000..b1e552f --- /dev/null +++ b/inc/PHPMailer/Exception.php @@ -0,0 +1,39 @@ + + * @author Jim Jagielski (jimjag) + * @author Andy Prevost (codeworxtech) + * @author Brent R. Matzelle (original founder) + * @copyright 2012 - 2017 Marcus Bointon + * @copyright 2010 - 2012 Jim Jagielski + * @copyright 2004 - 2009 Andy Prevost + * @license http://www.gnu.org/copyleft/lesser.html GNU Lesser General Public License + * @note This program is distributed in the hope that it will be useful - WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. + */ + +namespace PHPMailer\PHPMailer; + +/** + * PHPMailer exception handler. + * + * @author Marcus Bointon + */ +class Exception extends \Exception +{ + /** + * Prettify error message output. + * + * @return string + */ + public function errorMessage() + { + return '' . htmlspecialchars($this->getMessage()) . "
\n"; + } +} diff --git a/inc/PHPMailer/OAuth.php b/inc/PHPMailer/OAuth.php new file mode 100644 index 0000000..0271963 --- /dev/null +++ b/inc/PHPMailer/OAuth.php @@ -0,0 +1,138 @@ + + * @author Jim Jagielski (jimjag) + * @author Andy Prevost (codeworxtech) + * @author Brent R. Matzelle (original founder) + * @copyright 2012 - 2015 Marcus Bointon + * @copyright 2010 - 2012 Jim Jagielski + * @copyright 2004 - 2009 Andy Prevost + * @license http://www.gnu.org/copyleft/lesser.html GNU Lesser General Public License + * @note This program is distributed in the hope that it will be useful - WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. + */ + +namespace PHPMailer\PHPMailer; + +use League\OAuth2\Client\Grant\RefreshToken; +use League\OAuth2\Client\Provider\AbstractProvider; +use League\OAuth2\Client\Token\AccessToken; + +/** + * OAuth - OAuth2 authentication wrapper class. + * Uses the oauth2-client package from the League of Extraordinary Packages. + * + * @see http://oauth2-client.thephpleague.com + * + * @author Marcus Bointon (Synchro/coolbru) + */ +class OAuth +{ + /** + * An instance of the League OAuth Client Provider. + * + * @var AbstractProvider + */ + protected $provider; + + /** + * The current OAuth access token. + * + * @var AccessToken + */ + protected $oauthToken; + + /** + * The user's email address, usually used as the login ID + * and also the from address when sending email. + * + * @var string + */ + protected $oauthUserEmail = ''; + + /** + * The client secret, generated in the app definition of the service you're connecting to. + * + * @var string + */ + protected $oauthClientSecret = ''; + + /** + * The client ID, generated in the app definition of the service you're connecting to. + * + * @var string + */ + protected $oauthClientId = ''; + + /** + * The refresh token, used to obtain new AccessTokens. + * + * @var string + */ + protected $oauthRefreshToken = ''; + + /** + * OAuth constructor. + * + * @param array $options Associative array containing + * `provider`, `userName`, `clientSecret`, `clientId` and `refreshToken` elements + */ + public function __construct($options) + { + $this->provider = $options['provider']; + $this->oauthUserEmail = $options['userName']; + $this->oauthClientSecret = $options['clientSecret']; + $this->oauthClientId = $options['clientId']; + $this->oauthRefreshToken = $options['refreshToken']; + } + + /** + * Get a new RefreshToken. + * + * @return RefreshToken + */ + protected function getGrant() + { + return new RefreshToken(); + } + + /** + * Get a new AccessToken. + * + * @return AccessToken + */ + protected function getToken() + { + return $this->provider->getAccessToken( + $this->getGrant(), + ['refresh_token' => $this->oauthRefreshToken] + ); + } + + /** + * Generate a base64-encoded OAuth token. + * + * @return string + */ + public function getOauth64() + { + // Get a new token if it's not available or has expired + if (null === $this->oauthToken || $this->oauthToken->hasExpired()) { + $this->oauthToken = $this->getToken(); + } + + return base64_encode( + 'user=' . + $this->oauthUserEmail . + "\001auth=Bearer " . + $this->oauthToken . + "\001\001" + ); + } +} diff --git a/inc/PHPMailer/PHPMailer.php b/inc/PHPMailer/PHPMailer.php new file mode 100644 index 0000000..fddad40 --- /dev/null +++ b/inc/PHPMailer/PHPMailer.php @@ -0,0 +1,4820 @@ + + * @author Jim Jagielski (jimjag) + * @author Andy Prevost (codeworxtech) + * @author Brent R. Matzelle (original founder) + * @copyright 2012 - 2019 Marcus Bointon + * @copyright 2010 - 2012 Jim Jagielski + * @copyright 2004 - 2009 Andy Prevost + * @license http://www.gnu.org/copyleft/lesser.html GNU Lesser General Public License + * @note This program is distributed in the hope that it will be useful - WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. + */ + +namespace PHPMailer\PHPMailer; + +/** + * PHPMailer - PHP email creation and transport class. + * + * @author Marcus Bointon (Synchro/coolbru) + * @author Jim Jagielski (jimjag) + * @author Andy Prevost (codeworxtech) + * @author Brent R. Matzelle (original founder) + */ +class PHPMailer +{ + const CHARSET_ASCII = 'us-ascii'; + const CHARSET_ISO88591 = 'iso-8859-1'; + const CHARSET_UTF8 = 'utf-8'; + + const CONTENT_TYPE_PLAINTEXT = 'text/plain'; + const CONTENT_TYPE_TEXT_CALENDAR = 'text/calendar'; + const CONTENT_TYPE_TEXT_HTML = 'text/html'; + const CONTENT_TYPE_MULTIPART_ALTERNATIVE = 'multipart/alternative'; + const CONTENT_TYPE_MULTIPART_MIXED = 'multipart/mixed'; + const CONTENT_TYPE_MULTIPART_RELATED = 'multipart/related'; + + const ENCODING_7BIT = '7bit'; + const ENCODING_8BIT = '8bit'; + const ENCODING_BASE64 = 'base64'; + const ENCODING_BINARY = 'binary'; + const ENCODING_QUOTED_PRINTABLE = 'quoted-printable'; + + const ENCRYPTION_STARTTLS = 'tls'; + const ENCRYPTION_SMTPS = 'ssl'; + + const ICAL_METHOD_REQUEST = 'REQUEST'; + const ICAL_METHOD_PUBLISH = 'PUBLISH'; + const ICAL_METHOD_REPLY = 'REPLY'; + const ICAL_METHOD_ADD = 'ADD'; + const ICAL_METHOD_CANCEL = 'CANCEL'; + const ICAL_METHOD_REFRESH = 'REFRESH'; + const ICAL_METHOD_COUNTER = 'COUNTER'; + const ICAL_METHOD_DECLINECOUNTER = 'DECLINECOUNTER'; + + /** + * Email priority. + * Options: null (default), 1 = High, 3 = Normal, 5 = low. + * When null, the header is not set at all. + * + * @var int|null + */ + public $Priority; + + /** + * The character set of the message. + * + * @var string + */ + public $CharSet = self::CHARSET_ISO88591; + + /** + * The MIME Content-type of the message. + * + * @var string + */ + public $ContentType = self::CONTENT_TYPE_PLAINTEXT; + + /** + * The message encoding. + * Options: "8bit", "7bit", "binary", "base64", and "quoted-printable". + * + * @var string + */ + public $Encoding = self::ENCODING_8BIT; + + /** + * Holds the most recent mailer error message. + * + * @var string + */ + public $ErrorInfo = ''; + + /** + * The From email address for the message. + * + * @var string + */ + public $From = 'root@localhost'; + + /** + * The From name of the message. + * + * @var string + */ + public $FromName = 'Root User'; + + /** + * The envelope sender of the message. + * This will usually be turned into a Return-Path header by the receiver, + * and is the address that bounces will be sent to. + * If not empty, will be passed via `-f` to sendmail or as the 'MAIL FROM' value over SMTP. + * + * @var string + */ + public $Sender = ''; + + /** + * The Subject of the message. + * + * @var string + */ + public $Subject = ''; + + /** + * An HTML or plain text message body. + * If HTML then call isHTML(true). + * + * @var string + */ + public $Body = ''; + + /** + * The plain-text message body. + * This body can be read by mail clients that do not have HTML email + * capability such as mutt & Eudora. + * Clients that can read HTML will view the normal Body. + * + * @var string + */ + public $AltBody = ''; + + /** + * An iCal message part body. + * Only supported in simple alt or alt_inline message types + * To generate iCal event structures, use classes like EasyPeasyICS or iCalcreator. + * + * @see http://sprain.ch/blog/downloads/php-class-easypeasyics-create-ical-files-with-php/ + * @see http://kigkonsult.se/iCalcreator/ + * + * @var string + */ + public $Ical = ''; + + /** + * Value-array of "method" in Contenttype header "text/calendar" + * + * @var string[] + */ + protected static $IcalMethods = [ + self::ICAL_METHOD_REQUEST, + self::ICAL_METHOD_PUBLISH, + self::ICAL_METHOD_REPLY, + self::ICAL_METHOD_ADD, + self::ICAL_METHOD_CANCEL, + self::ICAL_METHOD_REFRESH, + self::ICAL_METHOD_COUNTER, + self::ICAL_METHOD_DECLINECOUNTER, + ]; + + /** + * The complete compiled MIME message body. + * + * @var string + */ + protected $MIMEBody = ''; + + /** + * The complete compiled MIME message headers. + * + * @var string + */ + protected $MIMEHeader = ''; + + /** + * Extra headers that createHeader() doesn't fold in. + * + * @var string + */ + protected $mailHeader = ''; + + /** + * Word-wrap the message body to this number of chars. + * Set to 0 to not wrap. A useful value here is 78, for RFC2822 section 2.1.1 compliance. + * + * @see static::STD_LINE_LENGTH + * + * @var int + */ + public $WordWrap = 0; + + /** + * Which method to use to send mail. + * Options: "mail", "sendmail", or "smtp". + * + * @var string + */ + public $Mailer = 'mail'; + + /** + * The path to the sendmail program. + * + * @var string + */ + public $Sendmail = '/usr/sbin/sendmail'; + + /** + * Whether mail() uses a fully sendmail-compatible MTA. + * One which supports sendmail's "-oi -f" options. + * + * @var bool + */ + public $UseSendmailOptions = true; + + /** + * The email address that a reading confirmation should be sent to, also known as read receipt. + * + * @var string + */ + public $ConfirmReadingTo = ''; + + /** + * The hostname to use in the Message-ID header and as default HELO string. + * If empty, PHPMailer attempts to find one with, in order, + * $_SERVER['SERVER_NAME'], gethostname(), php_uname('n'), or the value + * 'localhost.localdomain'. + * + * @see PHPMailer::$Helo + * + * @var string + */ + public $Hostname = ''; + + /** + * An ID to be used in the Message-ID header. + * If empty, a unique id will be generated. + * You can set your own, but it must be in the format "", + * as defined in RFC5322 section 3.6.4 or it will be ignored. + * + * @see https://tools.ietf.org/html/rfc5322#section-3.6.4 + * + * @var string + */ + public $MessageID = ''; + + /** + * The message Date to be used in the Date header. + * If empty, the current date will be added. + * + * @var string + */ + public $MessageDate = ''; + + /** + * SMTP hosts. + * Either a single hostname or multiple semicolon-delimited hostnames. + * You can also specify a different port + * for each host by using this format: [hostname:port] + * (e.g. "smtp1.example.com:25;smtp2.example.com"). + * You can also specify encryption type, for example: + * (e.g. "tls://smtp1.example.com:587;ssl://smtp2.example.com:465"). + * Hosts will be tried in order. + * + * @var string + */ + public $Host = 'localhost'; + + /** + * The default SMTP server port. + * + * @var int + */ + public $Port = 25; + + /** + * The SMTP HELO/EHLO name used for the SMTP connection. + * Default is $Hostname. If $Hostname is empty, PHPMailer attempts to find + * one with the same method described above for $Hostname. + * + * @see PHPMailer::$Hostname + * + * @var string + */ + public $Helo = ''; + + /** + * What kind of encryption to use on the SMTP connection. + * Options: '', static::ENCRYPTION_STARTTLS, or static::ENCRYPTION_SMTPS. + * + * @var string + */ + public $SMTPSecure = ''; + + /** + * Whether to enable TLS encryption automatically if a server supports it, + * even if `SMTPSecure` is not set to 'tls'. + * Be aware that in PHP >= 5.6 this requires that the server's certificates are valid. + * + * @var bool + */ + public $SMTPAutoTLS = true; + + /** + * Whether to use SMTP authentication. + * Uses the Username and Password properties. + * + * @see PHPMailer::$Username + * @see PHPMailer::$Password + * + * @var bool + */ + public $SMTPAuth = false; + + /** + * Options array passed to stream_context_create when connecting via SMTP. + * + * @var array + */ + public $SMTPOptions = []; + + /** + * SMTP username. + * + * @var string + */ + public $Username = ''; + + /** + * SMTP password. + * + * @var string + */ + public $Password = ''; + + /** + * SMTP auth type. + * Options are CRAM-MD5, LOGIN, PLAIN, XOAUTH2, attempted in that order if not specified. + * + * @var string + */ + public $AuthType = ''; + + /** + * An instance of the PHPMailer OAuth class. + * + * @var OAuth + */ + protected $oauth; + + /** + * The SMTP server timeout in seconds. + * Default of 5 minutes (300sec) is from RFC2821 section 4.5.3.2. + * + * @var int + */ + public $Timeout = 300; + + /** + * Comma separated list of DSN notifications + * 'NEVER' under no circumstances a DSN must be returned to the sender. + * If you use NEVER all other notifications will be ignored. + * 'SUCCESS' will notify you when your mail has arrived at its destination. + * 'FAILURE' will arrive if an error occurred during delivery. + * 'DELAY' will notify you if there is an unusual delay in delivery, but the actual + * delivery's outcome (success or failure) is not yet decided. + * + * @see https://tools.ietf.org/html/rfc3461 See section 4.1 for more information about NOTIFY + */ + public $dsn = ''; + + /** + * SMTP class debug output mode. + * Debug output level. + * Options: + * * SMTP::DEBUG_OFF: No output + * * SMTP::DEBUG_CLIENT: Client messages + * * SMTP::DEBUG_SERVER: Client and server messages + * * SMTP::DEBUG_CONNECTION: As SERVER plus connection status + * * SMTP::DEBUG_LOWLEVEL: Noisy, low-level data output, rarely needed + * + * @see SMTP::$do_debug + * + * @var int + */ + public $SMTPDebug = 0; + + /** + * How to handle debug output. + * Options: + * * `echo` Output plain-text as-is, appropriate for CLI + * * `html` Output escaped, line breaks converted to `
`, appropriate for browser output + * * `error_log` Output to error log as configured in php.ini + * By default PHPMailer will use `echo` if run from a `cli` or `cli-server` SAPI, `html` otherwise. + * Alternatively, you can provide a callable expecting two params: a message string and the debug level: + * + * ```php + * $mail->Debugoutput = function($str, $level) {echo "debug level $level; message: $str";}; + * ``` + * + * Alternatively, you can pass in an instance of a PSR-3 compatible logger, though only `debug` + * level output is used: + * + * ```php + * $mail->Debugoutput = new myPsr3Logger; + * ``` + * + * @see SMTP::$Debugoutput + * + * @var string|callable|\Psr\Log\LoggerInterface + */ + public $Debugoutput = 'echo'; + + /** + * Whether to keep SMTP connection open after each message. + * If this is set to true then to close the connection + * requires an explicit call to smtpClose(). + * + * @var bool + */ + public $SMTPKeepAlive = false; + + /** + * Whether to split multiple to addresses into multiple messages + * or send them all in one message. + * Only supported in `mail` and `sendmail` transports, not in SMTP. + * + * @var bool + */ + public $SingleTo = false; + + /** + * Storage for addresses when SingleTo is enabled. + * + * @var array + */ + protected $SingleToArray = []; + + /** + * Whether to generate VERP addresses on send. + * Only applicable when sending via SMTP. + * + * @see https://en.wikipedia.org/wiki/Variable_envelope_return_path + * @see http://www.postfix.org/VERP_README.html Postfix VERP info + * + * @var bool + */ + public $do_verp = false; + + /** + * Whether to allow sending messages with an empty body. + * + * @var bool + */ + public $AllowEmpty = false; + + /** + * DKIM selector. + * + * @var string + */ + public $DKIM_selector = ''; + + /** + * DKIM Identity. + * Usually the email address used as the source of the email. + * + * @var string + */ + public $DKIM_identity = ''; + + /** + * DKIM passphrase. + * Used if your key is encrypted. + * + * @var string + */ + public $DKIM_passphrase = ''; + + /** + * DKIM signing domain name. + * + * @example 'example.com' + * + * @var string + */ + public $DKIM_domain = ''; + + /** + * DKIM Copy header field values for diagnostic use. + * + * @var bool + */ + public $DKIM_copyHeaderFields = true; + + /** + * DKIM Extra signing headers. + * + * @example ['List-Unsubscribe', 'List-Help'] + * + * @var array + */ + public $DKIM_extraHeaders = []; + + /** + * DKIM private key file path. + * + * @var string + */ + public $DKIM_private = ''; + + /** + * DKIM private key string. + * + * If set, takes precedence over `$DKIM_private`. + * + * @var string + */ + public $DKIM_private_string = ''; + + /** + * Callback Action function name. + * + * The function that handles the result of the send email action. + * It is called out by send() for each email sent. + * + * Value can be any php callable: http://www.php.net/is_callable + * + * Parameters: + * bool $result result of the send action + * array $to email addresses of the recipients + * array $cc cc email addresses + * array $bcc bcc email addresses + * string $subject the subject + * string $body the email body + * string $from email address of sender + * string $extra extra information of possible use + * "smtp_transaction_id' => last smtp transaction id + * + * @var string + */ + public $action_function = ''; + + /** + * What to put in the X-Mailer header. + * Options: An empty string for PHPMailer default, whitespace/null for none, or a string to use. + * + * @var string|null + */ + public $XMailer = ''; + + /** + * Which validator to use by default when validating email addresses. + * May be a callable to inject your own validator, but there are several built-in validators. + * The default validator uses PHP's FILTER_VALIDATE_EMAIL filter_var option. + * + * @see PHPMailer::validateAddress() + * + * @var string|callable + */ + public static $validator = 'php'; + + /** + * An instance of the SMTP sender class. + * + * @var SMTP + */ + protected $smtp; + + /** + * The array of 'to' names and addresses. + * + * @var array + */ + protected $to = []; + + /** + * The array of 'cc' names and addresses. + * + * @var array + */ + protected $cc = []; + + /** + * The array of 'bcc' names and addresses. + * + * @var array + */ + protected $bcc = []; + + /** + * The array of reply-to names and addresses. + * + * @var array + */ + protected $ReplyTo = []; + + /** + * An array of all kinds of addresses. + * Includes all of $to, $cc, $bcc. + * + * @see PHPMailer::$to + * @see PHPMailer::$cc + * @see PHPMailer::$bcc + * + * @var array + */ + protected $all_recipients = []; + + /** + * An array of names and addresses queued for validation. + * In send(), valid and non duplicate entries are moved to $all_recipients + * and one of $to, $cc, or $bcc. + * This array is used only for addresses with IDN. + * + * @see PHPMailer::$to + * @see PHPMailer::$cc + * @see PHPMailer::$bcc + * @see PHPMailer::$all_recipients + * + * @var array + */ + protected $RecipientsQueue = []; + + /** + * An array of reply-to names and addresses queued for validation. + * In send(), valid and non duplicate entries are moved to $ReplyTo. + * This array is used only for addresses with IDN. + * + * @see PHPMailer::$ReplyTo + * + * @var array + */ + protected $ReplyToQueue = []; + + /** + * The array of attachments. + * + * @var array + */ + protected $attachment = []; + + /** + * The array of custom headers. + * + * @var array + */ + protected $CustomHeader = []; + + /** + * The most recent Message-ID (including angular brackets). + * + * @var string + */ + protected $lastMessageID = ''; + + /** + * The message's MIME type. + * + * @var string + */ + protected $message_type = ''; + + /** + * The array of MIME boundary strings. + * + * @var array + */ + protected $boundary = []; + + /** + * The array of available languages. + * + * @var array + */ + protected $language = []; + + /** + * The number of errors encountered. + * + * @var int + */ + protected $error_count = 0; + + /** + * The S/MIME certificate file path. + * + * @var string + */ + protected $sign_cert_file = ''; + + /** + * The S/MIME key file path. + * + * @var string + */ + protected $sign_key_file = ''; + + /** + * The optional S/MIME extra certificates ("CA Chain") file path. + * + * @var string + */ + protected $sign_extracerts_file = ''; + + /** + * The S/MIME password for the key. + * Used only if the key is encrypted. + * + * @var string + */ + protected $sign_key_pass = ''; + + /** + * Whether to throw exceptions for errors. + * + * @var bool + */ + protected $exceptions = false; + + /** + * Unique ID used for message ID and boundaries. + * + * @var string + */ + protected $uniqueid = ''; + + /** + * The PHPMailer Version number. + * + * @var string + */ + const VERSION = '6.1.5'; + + /** + * Error severity: message only, continue processing. + * + * @var int + */ + const STOP_MESSAGE = 0; + + /** + * Error severity: message, likely ok to continue processing. + * + * @var int + */ + const STOP_CONTINUE = 1; + + /** + * Error severity: message, plus full stop, critical error reached. + * + * @var int + */ + const STOP_CRITICAL = 2; + + /** + * The SMTP standard CRLF line break. + * If you want to change line break format, change static::$LE, not this. + */ + const CRLF = "\r\n"; + + /** + * "Folding White Space" a white space string used for line folding. + */ + const FWS = ' '; + + /** + * SMTP RFC standard line ending; Carriage Return, Line Feed. + * + * @var string + */ + protected static $LE = self::CRLF; + + /** + * The maximum line length supported by mail(). + * + * Background: mail() will sometimes corrupt messages + * with headers headers longer than 65 chars, see #818. + * + * @var int + */ + const MAIL_MAX_LINE_LENGTH = 63; + + /** + * The maximum line length allowed by RFC 2822 section 2.1.1. + * + * @var int + */ + const MAX_LINE_LENGTH = 998; + + /** + * The lower maximum line length allowed by RFC 2822 section 2.1.1. + * This length does NOT include the line break + * 76 means that lines will be 77 or 78 chars depending on whether + * the line break format is LF or CRLF; both are valid. + * + * @var int + */ + const STD_LINE_LENGTH = 76; + + /** + * Constructor. + * + * @param bool $exceptions Should we throw external exceptions? + */ + public function __construct($exceptions = null) + { + if (null !== $exceptions) { + $this->exceptions = (bool) $exceptions; + } + //Pick an appropriate debug output format automatically + $this->Debugoutput = (strpos(PHP_SAPI, 'cli') !== false ? 'echo' : 'html'); + } + + /** + * Destructor. + */ + public function __destruct() + { + //Close any open SMTP connection nicely + $this->smtpClose(); + } + + /** + * Call mail() in a safe_mode-aware fashion. + * Also, unless sendmail_path points to sendmail (or something that + * claims to be sendmail), don't pass params (not a perfect fix, + * but it will do). + * + * @param string $to To + * @param string $subject Subject + * @param string $body Message Body + * @param string $header Additional Header(s) + * @param string|null $params Params + * + * @return bool + */ + private function mailPassthru($to, $subject, $body, $header, $params) + { + //Check overloading of mail function to avoid double-encoding + if (ini_get('mbstring.func_overload') & 1) { + $subject = $this->secureHeader($subject); + } else { + $subject = $this->encodeHeader($this->secureHeader($subject)); + } + //Calling mail() with null params breaks + if (!$this->UseSendmailOptions || null === $params) { + $result = @mail($to, $subject, $body, $header); + } else { + $result = @mail($to, $subject, $body, $header, $params); + } + + return $result; + } + + /** + * Output debugging info via user-defined method. + * Only generates output if SMTP debug output is enabled (@see SMTP::$do_debug). + * + * @see PHPMailer::$Debugoutput + * @see PHPMailer::$SMTPDebug + * + * @param string $str + */ + protected function edebug($str) + { + if ($this->SMTPDebug <= 0) { + return; + } + //Is this a PSR-3 logger? + if ($this->Debugoutput instanceof \Psr\Log\LoggerInterface) { + $this->Debugoutput->debug($str); + + return; + } + //Avoid clash with built-in function names + if (is_callable($this->Debugoutput) && !in_array($this->Debugoutput, ['error_log', 'html', 'echo'])) { + call_user_func($this->Debugoutput, $str, $this->SMTPDebug); + + return; + } + switch ($this->Debugoutput) { + case 'error_log': + //Don't output, just log + error_log($str); + break; + case 'html': + //Cleans up output a bit for a better looking, HTML-safe output + echo htmlentities( + preg_replace('/[\r\n]+/', '', $str), + ENT_QUOTES, + 'UTF-8' + ), "
\n"; + break; + case 'echo': + default: + //Normalize line breaks + $str = preg_replace('/\r\n|\r/m', "\n", $str); + echo gmdate('Y-m-d H:i:s'), + "\t", + //Trim trailing space + trim( + //Indent for readability, except for trailing break + str_replace( + "\n", + "\n \t ", + trim($str) + ) + ), + "\n"; + } + } + + /** + * Sets message type to HTML or plain. + * + * @param bool $isHtml True for HTML mode + */ + public function isHTML($isHtml = true) + { + if ($isHtml) { + $this->ContentType = static::CONTENT_TYPE_TEXT_HTML; + } else { + $this->ContentType = static::CONTENT_TYPE_PLAINTEXT; + } + } + + /** + * Send messages using SMTP. + */ + public function isSMTP() + { + $this->Mailer = 'smtp'; + } + + /** + * Send messages using PHP's mail() function. + */ + public function isMail() + { + $this->Mailer = 'mail'; + } + + /** + * Send messages using $Sendmail. + */ + public function isSendmail() + { + $ini_sendmail_path = ini_get('sendmail_path'); + + if (false === stripos($ini_sendmail_path, 'sendmail')) { + $this->Sendmail = '/usr/sbin/sendmail'; + } else { + $this->Sendmail = $ini_sendmail_path; + } + $this->Mailer = 'sendmail'; + } + + /** + * Send messages using qmail. + */ + public function isQmail() + { + $ini_sendmail_path = ini_get('sendmail_path'); + + if (false === stripos($ini_sendmail_path, 'qmail')) { + $this->Sendmail = '/var/qmail/bin/qmail-inject'; + } else { + $this->Sendmail = $ini_sendmail_path; + } + $this->Mailer = 'qmail'; + } + + /** + * Add a "To" address. + * + * @param string $address The email address to send to + * @param string $name + * + * @throws Exception + * + * @return bool true on success, false if address already used or invalid in some way + */ + public function addAddress($address, $name = '') + { + return $this->addOrEnqueueAnAddress('to', $address, $name); + } + + /** + * Add a "CC" address. + * + * @param string $address The email address to send to + * @param string $name + * + * @throws Exception + * + * @return bool true on success, false if address already used or invalid in some way + */ + public function addCC($address, $name = '') + { + return $this->addOrEnqueueAnAddress('cc', $address, $name); + } + + /** + * Add a "BCC" address. + * + * @param string $address The email address to send to + * @param string $name + * + * @throws Exception + * + * @return bool true on success, false if address already used or invalid in some way + */ + public function addBCC($address, $name = '') + { + return $this->addOrEnqueueAnAddress('bcc', $address, $name); + } + + /** + * Add a "Reply-To" address. + * + * @param string $address The email address to reply to + * @param string $name + * + * @throws Exception + * + * @return bool true on success, false if address already used or invalid in some way + */ + public function addReplyTo($address, $name = '') + { + return $this->addOrEnqueueAnAddress('Reply-To', $address, $name); + } + + /** + * Add an address to one of the recipient arrays or to the ReplyTo array. Because PHPMailer + * can't validate addresses with an IDN without knowing the PHPMailer::$CharSet (that can still + * be modified after calling this function), addition of such addresses is delayed until send(). + * Addresses that have been added already return false, but do not throw exceptions. + * + * @param string $kind One of 'to', 'cc', 'bcc', or 'ReplyTo' + * @param string $address The email address to send, resp. to reply to + * @param string $name + * + * @throws Exception + * + * @return bool true on success, false if address already used or invalid in some way + */ + protected function addOrEnqueueAnAddress($kind, $address, $name) + { + $address = trim($address); + $name = trim(preg_replace('/[\r\n]+/', '', $name)); //Strip breaks and trim + $pos = strrpos($address, '@'); + if (false === $pos) { + // At-sign is missing. + $error_message = sprintf( + '%s (%s): %s', + $this->lang('invalid_address'), + $kind, + $address + ); + $this->setError($error_message); + $this->edebug($error_message); + if ($this->exceptions) { + throw new Exception($error_message); + } + + return false; + } + $params = [$kind, $address, $name]; + // Enqueue addresses with IDN until we know the PHPMailer::$CharSet. + if (static::idnSupported() && $this->has8bitChars(substr($address, ++$pos))) { + if ('Reply-To' !== $kind) { + if (!array_key_exists($address, $this->RecipientsQueue)) { + $this->RecipientsQueue[$address] = $params; + + return true; + } + } elseif (!array_key_exists($address, $this->ReplyToQueue)) { + $this->ReplyToQueue[$address] = $params; + + return true; + } + + return false; + } + + // Immediately add standard addresses without IDN. + return call_user_func_array([$this, 'addAnAddress'], $params); + } + + /** + * Add an address to one of the recipient arrays or to the ReplyTo array. + * Addresses that have been added already return false, but do not throw exceptions. + * + * @param string $kind One of 'to', 'cc', 'bcc', or 'ReplyTo' + * @param string $address The email address to send, resp. to reply to + * @param string $name + * + * @throws Exception + * + * @return bool true on success, false if address already used or invalid in some way + */ + protected function addAnAddress($kind, $address, $name = '') + { + if (!in_array($kind, ['to', 'cc', 'bcc', 'Reply-To'])) { + $error_message = sprintf( + '%s: %s', + $this->lang('Invalid recipient kind'), + $kind + ); + $this->setError($error_message); + $this->edebug($error_message); + if ($this->exceptions) { + throw new Exception($error_message); + } + + return false; + } + if (!static::validateAddress($address)) { + $error_message = sprintf( + '%s (%s): %s', + $this->lang('invalid_address'), + $kind, + $address + ); + $this->setError($error_message); + $this->edebug($error_message); + if ($this->exceptions) { + throw new Exception($error_message); + } + + return false; + } + if ('Reply-To' !== $kind) { + if (!array_key_exists(strtolower($address), $this->all_recipients)) { + $this->{$kind}[] = [$address, $name]; + $this->all_recipients[strtolower($address)] = true; + + return true; + } + } elseif (!array_key_exists(strtolower($address), $this->ReplyTo)) { + $this->ReplyTo[strtolower($address)] = [$address, $name]; + + return true; + } + + return false; + } + + /** + * Parse and validate a string containing one or more RFC822-style comma-separated email addresses + * of the form "display name
" into an array of name/address pairs. + * Uses the imap_rfc822_parse_adrlist function if the IMAP extension is available. + * Note that quotes in the name part are removed. + * + * @see http://www.andrew.cmu.edu/user/agreen1/testing/mrbs/web/Mail/RFC822.php A more careful implementation + * + * @param string $addrstr The address list string + * @param bool $useimap Whether to use the IMAP extension to parse the list + * + * @return array + */ + public static function parseAddresses($addrstr, $useimap = true) + { + $addresses = []; + if ($useimap && function_exists('imap_rfc822_parse_adrlist')) { + //Use this built-in parser if it's available + $list = imap_rfc822_parse_adrlist($addrstr, ''); + foreach ($list as $address) { + if (('.SYNTAX-ERROR.' !== $address->host) && static::validateAddress( + $address->mailbox . '@' . $address->host + )) { + $addresses[] = [ + 'name' => (property_exists($address, 'personal') ? $address->personal : ''), + 'address' => $address->mailbox . '@' . $address->host, + ]; + } + } + } else { + //Use this simpler parser + $list = explode(',', $addrstr); + foreach ($list as $address) { + $address = trim($address); + //Is there a separate name part? + if (strpos($address, '<') === false) { + //No separate name, just use the whole thing + if (static::validateAddress($address)) { + $addresses[] = [ + 'name' => '', + 'address' => $address, + ]; + } + } else { + list($name, $email) = explode('<', $address); + $email = trim(str_replace('>', '', $email)); + if (static::validateAddress($email)) { + $addresses[] = [ + 'name' => trim(str_replace(['"', "'"], '', $name)), + 'address' => $email, + ]; + } + } + } + } + + return $addresses; + } + + /** + * Set the From and FromName properties. + * + * @param string $address + * @param string $name + * @param bool $auto Whether to also set the Sender address, defaults to true + * + * @throws Exception + * + * @return bool + */ + public function setFrom($address, $name = '', $auto = true) + { + $address = trim($address); + $name = trim(preg_replace('/[\r\n]+/', '', $name)); //Strip breaks and trim + // Don't validate now addresses with IDN. Will be done in send(). + $pos = strrpos($address, '@'); + if ((false === $pos) + || ((!$this->has8bitChars(substr($address, ++$pos)) || !static::idnSupported()) + && !static::validateAddress($address)) + ) { + $error_message = sprintf( + '%s (From): %s', + $this->lang('invalid_address'), + $address + ); + $this->setError($error_message); + $this->edebug($error_message); + if ($this->exceptions) { + throw new Exception($error_message); + } + + return false; + } + $this->From = $address; + $this->FromName = $name; + if ($auto && empty($this->Sender)) { + $this->Sender = $address; + } + + return true; + } + + /** + * Return the Message-ID header of the last email. + * Technically this is the value from the last time the headers were created, + * but it's also the message ID of the last sent message except in + * pathological cases. + * + * @return string + */ + public function getLastMessageID() + { + return $this->lastMessageID; + } + + /** + * Check that a string looks like an email address. + * Validation patterns supported: + * * `auto` Pick best pattern automatically; + * * `pcre8` Use the squiloople.com pattern, requires PCRE > 8.0; + * * `pcre` Use old PCRE implementation; + * * `php` Use PHP built-in FILTER_VALIDATE_EMAIL; + * * `html5` Use the pattern given by the HTML5 spec for 'email' type form input elements. + * * `noregex` Don't use a regex: super fast, really dumb. + * Alternatively you may pass in a callable to inject your own validator, for example: + * + * ```php + * PHPMailer::validateAddress('user@example.com', function($address) { + * return (strpos($address, '@') !== false); + * }); + * ``` + * + * You can also set the PHPMailer::$validator static to a callable, allowing built-in methods to use your validator. + * + * @param string $address The email address to check + * @param string|callable $patternselect Which pattern to use + * + * @return bool + */ + public static function validateAddress($address, $patternselect = null) + { + if (null === $patternselect) { + $patternselect = static::$validator; + } + if (is_callable($patternselect)) { + return $patternselect($address); + } + //Reject line breaks in addresses; it's valid RFC5322, but not RFC5321 + if (strpos($address, "\n") !== false || strpos($address, "\r") !== false) { + return false; + } + switch ($patternselect) { + case 'pcre': //Kept for BC + case 'pcre8': + /* + * A more complex and more permissive version of the RFC5322 regex on which FILTER_VALIDATE_EMAIL + * is based. + * In addition to the addresses allowed by filter_var, also permits: + * * dotless domains: `a@b` + * * comments: `1234 @ local(blah) .machine .example` + * * quoted elements: `'"test blah"@example.org'` + * * numeric TLDs: `a@b.123` + * * unbracketed IPv4 literals: `a@192.168.0.1` + * * IPv6 literals: 'first.last@[IPv6:a1::]' + * Not all of these will necessarily work for sending! + * + * @see http://squiloople.com/2009/12/20/email-address-validation/ + * @copyright 2009-2010 Michael Rushton + * Feel free to use and redistribute this code. But please keep this copyright notice. + */ + return (bool) preg_match( + '/^(?!(?>(?1)"?(?>\\\[ -~]|[^"])"?(?1)){255,})(?!(?>(?1)"?(?>\\\[ -~]|[^"])"?(?1)){65,}@)' . + '((?>(?>(?>((?>(?>(?>\x0D\x0A)?[\t ])+|(?>[\t ]*\x0D\x0A)?[\t ]+)?)(\((?>(?2)' . + '(?>[\x01-\x08\x0B\x0C\x0E-\'*-\[\]-\x7F]|\\\[\x00-\x7F]|(?3)))*(?2)\)))+(?2))|(?2))?)' . + '([!#-\'*+\/-9=?^-~-]+|"(?>(?2)(?>[\x01-\x08\x0B\x0C\x0E-!#-\[\]-\x7F]|\\\[\x00-\x7F]))*' . + '(?2)")(?>(?1)\.(?1)(?4))*(?1)@(?!(?1)[a-z0-9-]{64,})(?1)(?>([a-z0-9](?>[a-z0-9-]*[a-z0-9])?)' . + '(?>(?1)\.(?!(?1)[a-z0-9-]{64,})(?1)(?5)){0,126}|\[(?:(?>IPv6:(?>([a-f0-9]{1,4})(?>:(?6)){7}' . + '|(?!(?:.*[a-f0-9][:\]]){8,})((?6)(?>:(?6)){0,6})?::(?7)?))|(?>(?>IPv6:(?>(?6)(?>:(?6)){5}:' . + '|(?!(?:.*[a-f0-9]:){6,})(?8)?::(?>((?6)(?>:(?6)){0,4}):)?))?(25[0-5]|2[0-4][0-9]|1[0-9]{2}' . + '|[1-9]?[0-9])(?>\.(?9)){3}))\])(?1)$/isD', + $address + ); + case 'html5': + /* + * This is the pattern used in the HTML5 spec for validation of 'email' type form input elements. + * + * @see http://www.whatwg.org/specs/web-apps/current-work/#e-mail-state-(type=email) + */ + return (bool) preg_match( + '/^[a-zA-Z0-9.!#$%&\'*+\/=?^_`{|}~-]+@[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}' . + '[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)*$/sD', + $address + ); + case 'php': + default: + return filter_var($address, FILTER_VALIDATE_EMAIL) !== false; + } + } + + /** + * Tells whether IDNs (Internationalized Domain Names) are supported or not. This requires the + * `intl` and `mbstring` PHP extensions. + * + * @return bool `true` if required functions for IDN support are present + */ + public static function idnSupported() + { + return function_exists('idn_to_ascii') && function_exists('mb_convert_encoding'); + } + + /** + * Converts IDN in given email address to its ASCII form, also known as punycode, if possible. + * Important: Address must be passed in same encoding as currently set in PHPMailer::$CharSet. + * This function silently returns unmodified address if: + * - No conversion is necessary (i.e. domain name is not an IDN, or is already in ASCII form) + * - Conversion to punycode is impossible (e.g. required PHP functions are not available) + * or fails for any reason (e.g. domain contains characters not allowed in an IDN). + * + * @see PHPMailer::$CharSet + * + * @param string $address The email address to convert + * + * @return string The encoded address in ASCII form + */ + public function punyencodeAddress($address) + { + // Verify we have required functions, CharSet, and at-sign. + $pos = strrpos($address, '@'); + if (!empty($this->CharSet) && + false !== $pos && + static::idnSupported() + ) { + $domain = substr($address, ++$pos); + // Verify CharSet string is a valid one, and domain properly encoded in this CharSet. + if ($this->has8bitChars($domain) && @mb_check_encoding($domain, $this->CharSet)) { + $domain = mb_convert_encoding($domain, 'UTF-8', $this->CharSet); + //Ignore IDE complaints about this line - method signature changed in PHP 5.4 + $errorcode = 0; + if (defined('INTL_IDNA_VARIANT_UTS46')) { + $punycode = idn_to_ascii($domain, $errorcode, INTL_IDNA_VARIANT_UTS46); + } elseif (defined('INTL_IDNA_VARIANT_2003')) { + $punycode = idn_to_ascii($domain, $errorcode, INTL_IDNA_VARIANT_2003); + } else { + $punycode = idn_to_ascii($domain, $errorcode); + } + if (false !== $punycode) { + return substr($address, 0, $pos) . $punycode; + } + } + } + + return $address; + } + + /** + * Create a message and send it. + * Uses the sending method specified by $Mailer. + * + * @throws Exception + * + * @return bool false on error - See the ErrorInfo property for details of the error + */ + public function send() + { + try { + if (!$this->preSend()) { + return false; + } + + return $this->postSend(); + } catch (Exception $exc) { + $this->mailHeader = ''; + $this->setError($exc->getMessage()); + if ($this->exceptions) { + throw $exc; + } + + return false; + } + } + + /** + * Prepare a message for sending. + * + * @throws Exception + * + * @return bool + */ + public function preSend() + { + if ('smtp' === $this->Mailer + || ('mail' === $this->Mailer && stripos(PHP_OS, 'WIN') === 0) + ) { + //SMTP mandates RFC-compliant line endings + //and it's also used with mail() on Windows + static::setLE(self::CRLF); + } else { + //Maintain backward compatibility with legacy Linux command line mailers + static::setLE(PHP_EOL); + } + //Check for buggy PHP versions that add a header with an incorrect line break + if ('mail' === $this->Mailer + && ((PHP_VERSION_ID >= 70000 && PHP_VERSION_ID < 70017) + || (PHP_VERSION_ID >= 70100 && PHP_VERSION_ID < 70103)) + && ini_get('mail.add_x_header') === '1' + && stripos(PHP_OS, 'WIN') === 0 + ) { + trigger_error( + 'Your version of PHP is affected by a bug that may result in corrupted messages.' . + ' To fix it, switch to sending using SMTP, disable the mail.add_x_header option in' . + ' your php.ini, switch to MacOS or Linux, or upgrade your PHP to version 7.0.17+ or 7.1.3+.', + E_USER_WARNING + ); + } + + try { + $this->error_count = 0; // Reset errors + $this->mailHeader = ''; + + // Dequeue recipient and Reply-To addresses with IDN + foreach (array_merge($this->RecipientsQueue, $this->ReplyToQueue) as $params) { + $params[1] = $this->punyencodeAddress($params[1]); + call_user_func_array([$this, 'addAnAddress'], $params); + } + if (count($this->to) + count($this->cc) + count($this->bcc) < 1) { + throw new Exception($this->lang('provide_address'), self::STOP_CRITICAL); + } + + // Validate From, Sender, and ConfirmReadingTo addresses + foreach (['From', 'Sender', 'ConfirmReadingTo'] as $address_kind) { + $this->$address_kind = trim($this->$address_kind); + if (empty($this->$address_kind)) { + continue; + } + $this->$address_kind = $this->punyencodeAddress($this->$address_kind); + if (!static::validateAddress($this->$address_kind)) { + $error_message = sprintf( + '%s (%s): %s', + $this->lang('invalid_address'), + $address_kind, + $this->$address_kind + ); + $this->setError($error_message); + $this->edebug($error_message); + if ($this->exceptions) { + throw new Exception($error_message); + } + + return false; + } + } + + // Set whether the message is multipart/alternative + if ($this->alternativeExists()) { + $this->ContentType = static::CONTENT_TYPE_MULTIPART_ALTERNATIVE; + } + + $this->setMessageType(); + // Refuse to send an empty message unless we are specifically allowing it + if (!$this->AllowEmpty && empty($this->Body)) { + throw new Exception($this->lang('empty_message'), self::STOP_CRITICAL); + } + + //Trim subject consistently + $this->Subject = trim($this->Subject); + // Create body before headers in case body makes changes to headers (e.g. altering transfer encoding) + $this->MIMEHeader = ''; + $this->MIMEBody = $this->createBody(); + // createBody may have added some headers, so retain them + $tempheaders = $this->MIMEHeader; + $this->MIMEHeader = $this->createHeader(); + $this->MIMEHeader .= $tempheaders; + + // To capture the complete message when using mail(), create + // an extra header list which createHeader() doesn't fold in + if ('mail' === $this->Mailer) { + if (count($this->to) > 0) { + $this->mailHeader .= $this->addrAppend('To', $this->to); + } else { + $this->mailHeader .= $this->headerLine('To', 'undisclosed-recipients:;'); + } + $this->mailHeader .= $this->headerLine( + 'Subject', + $this->encodeHeader($this->secureHeader($this->Subject)) + ); + } + + // Sign with DKIM if enabled + if (!empty($this->DKIM_domain) + && !empty($this->DKIM_selector) + && (!empty($this->DKIM_private_string) + || (!empty($this->DKIM_private) + && static::isPermittedPath($this->DKIM_private) + && file_exists($this->DKIM_private) + ) + ) + ) { + $header_dkim = $this->DKIM_Add( + $this->MIMEHeader . $this->mailHeader, + $this->encodeHeader($this->secureHeader($this->Subject)), + $this->MIMEBody + ); + $this->MIMEHeader = static::stripTrailingWSP($this->MIMEHeader) . static::$LE . + static::normalizeBreaks($header_dkim) . static::$LE; + } + + return true; + } catch (Exception $exc) { + $this->setError($exc->getMessage()); + if ($this->exceptions) { + throw $exc; + } + + return false; + } + } + + /** + * Actually send a message via the selected mechanism. + * + * @throws Exception + * + * @return bool + */ + public function postSend() + { + try { + // Choose the mailer and send through it + switch ($this->Mailer) { + case 'sendmail': + case 'qmail': + return $this->sendmailSend($this->MIMEHeader, $this->MIMEBody); + case 'smtp': + return $this->smtpSend($this->MIMEHeader, $this->MIMEBody); + case 'mail': + return $this->mailSend($this->MIMEHeader, $this->MIMEBody); + default: + $sendMethod = $this->Mailer . 'Send'; + if (method_exists($this, $sendMethod)) { + return $this->$sendMethod($this->MIMEHeader, $this->MIMEBody); + } + + return $this->mailSend($this->MIMEHeader, $this->MIMEBody); + } + } catch (Exception $exc) { + $this->setError($exc->getMessage()); + $this->edebug($exc->getMessage()); + if ($this->exceptions) { + throw $exc; + } + } + + return false; + } + + /** + * Send mail using the $Sendmail program. + * + * @see PHPMailer::$Sendmail + * + * @param string $header The message headers + * @param string $body The message body + * + * @throws Exception + * + * @return bool + */ + protected function sendmailSend($header, $body) + { + $header = static::stripTrailingWSP($header) . static::$LE . static::$LE; + + // CVE-2016-10033, CVE-2016-10045: Don't pass -f if characters will be escaped. + if (!empty($this->Sender) && self::isShellSafe($this->Sender)) { + if ('qmail' === $this->Mailer) { + $sendmailFmt = '%s -f%s'; + } else { + $sendmailFmt = '%s -oi -f%s -t'; + } + } elseif ('qmail' === $this->Mailer) { + $sendmailFmt = '%s'; + } else { + $sendmailFmt = '%s -oi -t'; + } + + $sendmail = sprintf($sendmailFmt, escapeshellcmd($this->Sendmail), $this->Sender); + + if ($this->SingleTo) { + foreach ($this->SingleToArray as $toAddr) { + $mail = @popen($sendmail, 'w'); + if (!$mail) { + throw new Exception($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL); + } + fwrite($mail, 'To: ' . $toAddr . "\n"); + fwrite($mail, $header); + fwrite($mail, $body); + $result = pclose($mail); + $this->doCallback( + ($result === 0), + [$toAddr], + $this->cc, + $this->bcc, + $this->Subject, + $body, + $this->From, + [] + ); + if (0 !== $result) { + throw new Exception($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL); + } + } + } else { + $mail = @popen($sendmail, 'w'); + if (!$mail) { + throw new Exception($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL); + } + fwrite($mail, $header); + fwrite($mail, $body); + $result = pclose($mail); + $this->doCallback( + ($result === 0), + $this->to, + $this->cc, + $this->bcc, + $this->Subject, + $body, + $this->From, + [] + ); + if (0 !== $result) { + throw new Exception($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL); + } + } + + return true; + } + + /** + * Fix CVE-2016-10033 and CVE-2016-10045 by disallowing potentially unsafe shell characters. + * Note that escapeshellarg and escapeshellcmd are inadequate for our purposes, especially on Windows. + * + * @see https://github.com/PHPMailer/PHPMailer/issues/924 CVE-2016-10045 bug report + * + * @param string $string The string to be validated + * + * @return bool + */ + protected static function isShellSafe($string) + { + // Future-proof + if (escapeshellcmd($string) !== $string + || !in_array(escapeshellarg($string), ["'$string'", "\"$string\""]) + ) { + return false; + } + + $length = strlen($string); + + for ($i = 0; $i < $length; ++$i) { + $c = $string[$i]; + + // All other characters have a special meaning in at least one common shell, including = and +. + // Full stop (.) has a special meaning in cmd.exe, but its impact should be negligible here. + // Note that this does permit non-Latin alphanumeric characters based on the current locale. + if (!ctype_alnum($c) && strpos('@_-.', $c) === false) { + return false; + } + } + + return true; + } + + /** + * Check whether a file path is of a permitted type. + * Used to reject URLs and phar files from functions that access local file paths, + * such as addAttachment. + * + * @param string $path A relative or absolute path to a file + * + * @return bool + */ + protected static function isPermittedPath($path) + { + return !preg_match('#^[a-z]+://#i', $path); + } + + /** + * Send mail using the PHP mail() function. + * + * @see http://www.php.net/manual/en/book.mail.php + * + * @param string $header The message headers + * @param string $body The message body + * + * @throws Exception + * + * @return bool + */ + protected function mailSend($header, $body) + { + $header = static::stripTrailingWSP($header) . static::$LE . static::$LE; + + $toArr = []; + foreach ($this->to as $toaddr) { + $toArr[] = $this->addrFormat($toaddr); + } + $to = implode(', ', $toArr); + + $params = null; + //This sets the SMTP envelope sender which gets turned into a return-path header by the receiver + //A space after `-f` is optional, but there is a long history of its presence + //causing problems, so we don't use one + //Exim docs: http://www.exim.org/exim-html-current/doc/html/spec_html/ch-the_exim_command_line.html + //Sendmail docs: http://www.sendmail.org/~ca/email/man/sendmail.html + //Qmail docs: http://www.qmail.org/man/man8/qmail-inject.html + //Example problem: https://www.drupal.org/node/1057954 + // CVE-2016-10033, CVE-2016-10045: Don't pass -f if characters will be escaped. + if (!empty($this->Sender) && static::validateAddress($this->Sender) && self::isShellSafe($this->Sender)) { + $params = sprintf('-f%s', $this->Sender); + } + if (!empty($this->Sender) && static::validateAddress($this->Sender)) { + $old_from = ini_get('sendmail_from'); + ini_set('sendmail_from', $this->Sender); + } + $result = false; + if ($this->SingleTo && count($toArr) > 1) { + foreach ($toArr as $toAddr) { + $result = $this->mailPassthru($toAddr, $this->Subject, $body, $header, $params); + $this->doCallback($result, [$toAddr], $this->cc, $this->bcc, $this->Subject, $body, $this->From, []); + } + } else { + $result = $this->mailPassthru($to, $this->Subject, $body, $header, $params); + $this->doCallback($result, $this->to, $this->cc, $this->bcc, $this->Subject, $body, $this->From, []); + } + if (isset($old_from)) { + ini_set('sendmail_from', $old_from); + } + if (!$result) { + throw new Exception($this->lang('instantiate'), self::STOP_CRITICAL); + } + + return true; + } + + /** + * Get an instance to use for SMTP operations. + * Override this function to load your own SMTP implementation, + * or set one with setSMTPInstance. + * + * @return SMTP + */ + public function getSMTPInstance() + { + if (!is_object($this->smtp)) { + $this->smtp = new SMTP(); + } + + return $this->smtp; + } + + /** + * Provide an instance to use for SMTP operations. + * + * @return SMTP + */ + public function setSMTPInstance(SMTP $smtp) + { + $this->smtp = $smtp; + + return $this->smtp; + } + + /** + * Send mail via SMTP. + * Returns false if there is a bad MAIL FROM, RCPT, or DATA input. + * + * @see PHPMailer::setSMTPInstance() to use a different class. + * + * @uses \PHPMailer\PHPMailer\SMTP + * + * @param string $header The message headers + * @param string $body The message body + * + * @throws Exception + * + * @return bool + */ + protected function smtpSend($header, $body) + { + $header = static::stripTrailingWSP($header) . static::$LE . static::$LE; + $bad_rcpt = []; + if (!$this->smtpConnect($this->SMTPOptions)) { + throw new Exception($this->lang('smtp_connect_failed'), self::STOP_CRITICAL); + } + //Sender already validated in preSend() + if ('' === $this->Sender) { + $smtp_from = $this->From; + } else { + $smtp_from = $this->Sender; + } + if (!$this->smtp->mail($smtp_from)) { + $this->setError($this->lang('from_failed') . $smtp_from . ' : ' . implode(',', $this->smtp->getError())); + throw new Exception($this->ErrorInfo, self::STOP_CRITICAL); + } + + $callbacks = []; + // Attempt to send to all recipients + foreach ([$this->to, $this->cc, $this->bcc] as $togroup) { + foreach ($togroup as $to) { + if (!$this->smtp->recipient($to[0], $this->dsn)) { + $error = $this->smtp->getError(); + $bad_rcpt[] = ['to' => $to[0], 'error' => $error['detail']]; + $isSent = false; + } else { + $isSent = true; + } + + $callbacks[] = ['issent'=>$isSent, 'to'=>$to[0]]; + } + } + + // Only send the DATA command if we have viable recipients + if ((count($this->all_recipients) > count($bad_rcpt)) && !$this->smtp->data($header . $body)) { + throw new Exception($this->lang('data_not_accepted'), self::STOP_CRITICAL); + } + + $smtp_transaction_id = $this->smtp->getLastTransactionID(); + + if ($this->SMTPKeepAlive) { + $this->smtp->reset(); + } else { + $this->smtp->quit(); + $this->smtp->close(); + } + + foreach ($callbacks as $cb) { + $this->doCallback( + $cb['issent'], + [$cb['to']], + [], + [], + $this->Subject, + $body, + $this->From, + ['smtp_transaction_id' => $smtp_transaction_id] + ); + } + + //Create error message for any bad addresses + if (count($bad_rcpt) > 0) { + $errstr = ''; + foreach ($bad_rcpt as $bad) { + $errstr .= $bad['to'] . ': ' . $bad['error']; + } + throw new Exception($this->lang('recipients_failed') . $errstr, self::STOP_CONTINUE); + } + + return true; + } + + /** + * Initiate a connection to an SMTP server. + * Returns false if the operation failed. + * + * @param array $options An array of options compatible with stream_context_create() + * + * @throws Exception + * + * @uses \PHPMailer\PHPMailer\SMTP + * + * @return bool + */ + public function smtpConnect($options = null) + { + if (null === $this->smtp) { + $this->smtp = $this->getSMTPInstance(); + } + + //If no options are provided, use whatever is set in the instance + if (null === $options) { + $options = $this->SMTPOptions; + } + + // Already connected? + if ($this->smtp->connected()) { + return true; + } + + $this->smtp->setTimeout($this->Timeout); + $this->smtp->setDebugLevel($this->SMTPDebug); + $this->smtp->setDebugOutput($this->Debugoutput); + $this->smtp->setVerp($this->do_verp); + $hosts = explode(';', $this->Host); + $lastexception = null; + + foreach ($hosts as $hostentry) { + $hostinfo = []; + if (!preg_match( + '/^(?:(ssl|tls):\/\/)?(.+?)(?::(\d+))?$/', + trim($hostentry), + $hostinfo + )) { + $this->edebug($this->lang('invalid_hostentry') . ' ' . trim($hostentry)); + // Not a valid host entry + continue; + } + // $hostinfo[1]: optional ssl or tls prefix + // $hostinfo[2]: the hostname + // $hostinfo[3]: optional port number + // The host string prefix can temporarily override the current setting for SMTPSecure + // If it's not specified, the default value is used + + //Check the host name is a valid name or IP address before trying to use it + if (!static::isValidHost($hostinfo[2])) { + $this->edebug($this->lang('invalid_host') . ' ' . $hostinfo[2]); + continue; + } + $prefix = ''; + $secure = $this->SMTPSecure; + $tls = (static::ENCRYPTION_STARTTLS === $this->SMTPSecure); + if ('ssl' === $hostinfo[1] || ('' === $hostinfo[1] && static::ENCRYPTION_SMTPS === $this->SMTPSecure)) { + $prefix = 'ssl://'; + $tls = false; // Can't have SSL and TLS at the same time + $secure = static::ENCRYPTION_SMTPS; + } elseif ('tls' === $hostinfo[1]) { + $tls = true; + // tls doesn't use a prefix + $secure = static::ENCRYPTION_STARTTLS; + } + //Do we need the OpenSSL extension? + $sslext = defined('OPENSSL_ALGO_SHA256'); + if (static::ENCRYPTION_STARTTLS === $secure || static::ENCRYPTION_SMTPS === $secure) { + //Check for an OpenSSL constant rather than using extension_loaded, which is sometimes disabled + if (!$sslext) { + throw new Exception($this->lang('extension_missing') . 'openssl', self::STOP_CRITICAL); + } + } + $host = $hostinfo[2]; + $port = $this->Port; + if (array_key_exists(3, $hostinfo) && is_numeric($hostinfo[3]) && $hostinfo[3] > 0 && $hostinfo[3] < 65536) { + $port = (int) $hostinfo[3]; + } + if ($this->smtp->connect($prefix . $host, $port, $this->Timeout, $options)) { + try { + if ($this->Helo) { + $hello = $this->Helo; + } else { + $hello = $this->serverHostname(); + } + $this->smtp->hello($hello); + //Automatically enable TLS encryption if: + // * it's not disabled + // * we have openssl extension + // * we are not already using SSL + // * the server offers STARTTLS + if ($this->SMTPAutoTLS && $sslext && 'ssl' !== $secure && $this->smtp->getServerExt('STARTTLS')) { + $tls = true; + } + if ($tls) { + if (!$this->smtp->startTLS()) { + throw new Exception($this->lang('connect_host')); + } + // We must resend EHLO after TLS negotiation + $this->smtp->hello($hello); + } + if ($this->SMTPAuth && !$this->smtp->authenticate( + $this->Username, + $this->Password, + $this->AuthType, + $this->oauth + )) { + throw new Exception($this->lang('authenticate')); + } + + return true; + } catch (Exception $exc) { + $lastexception = $exc; + $this->edebug($exc->getMessage()); + // We must have connected, but then failed TLS or Auth, so close connection nicely + $this->smtp->quit(); + } + } + } + // If we get here, all connection attempts have failed, so close connection hard + $this->smtp->close(); + // As we've caught all exceptions, just report whatever the last one was + if ($this->exceptions && null !== $lastexception) { + throw $lastexception; + } + + return false; + } + + /** + * Close the active SMTP session if one exists. + */ + public function smtpClose() + { + if ((null !== $this->smtp) && $this->smtp->connected()) { + $this->smtp->quit(); + $this->smtp->close(); + } + } + + /** + * Set the language for error messages. + * Returns false if it cannot load the language file. + * The default language is English. + * + * @param string $langcode ISO 639-1 2-character language code (e.g. French is "fr") + * @param string $lang_path Path to the language file directory, with trailing separator (slash) + * + * @return bool + */ + public function setLanguage($langcode = 'en', $lang_path = '') + { + // Backwards compatibility for renamed language codes + $renamed_langcodes = [ + 'br' => 'pt_br', + 'cz' => 'cs', + 'dk' => 'da', + 'no' => 'nb', + 'se' => 'sv', + 'rs' => 'sr', + 'tg' => 'tl', + ]; + + if (isset($renamed_langcodes[$langcode])) { + $langcode = $renamed_langcodes[$langcode]; + } + + // Define full set of translatable strings in English + $PHPMAILER_LANG = [ + 'authenticate' => 'SMTP Error: Could not authenticate.', + 'connect_host' => 'SMTP Error: Could not connect to SMTP host.', + 'data_not_accepted' => 'SMTP Error: data not accepted.', + 'empty_message' => 'Message body empty', + 'encoding' => 'Unknown encoding: ', + 'execute' => 'Could not execute: ', + 'file_access' => 'Could not access file: ', + 'file_open' => 'File Error: Could not open file: ', + 'from_failed' => 'The following From address failed: ', + 'instantiate' => 'Could not instantiate mail function.', + 'invalid_address' => 'Invalid address: ', + 'invalid_hostentry' => 'Invalid hostentry: ', + 'invalid_host' => 'Invalid host: ', + 'mailer_not_supported' => ' mailer is not supported.', + 'provide_address' => 'You must provide at least one recipient email address.', + 'recipients_failed' => 'SMTP Error: The following recipients failed: ', + 'signing' => 'Signing Error: ', + 'smtp_connect_failed' => 'SMTP connect() failed.', + 'smtp_error' => 'SMTP server error: ', + 'variable_set' => 'Cannot set or reset variable: ', + 'extension_missing' => 'Extension missing: ', + ]; + if (empty($lang_path)) { + // Calculate an absolute path so it can work if CWD is not here + $lang_path = dirname(__DIR__) . DIRECTORY_SEPARATOR . 'language' . DIRECTORY_SEPARATOR; + } + //Validate $langcode + if (!preg_match('/^[a-z]{2}(?:_[a-zA-Z]{2})?$/', $langcode)) { + $langcode = 'en'; + } + $foundlang = true; + $lang_file = $lang_path . 'phpmailer.lang-' . $langcode . '.php'; + // There is no English translation file + if ('en' !== $langcode) { + // Make sure language file path is readable + if (!static::isPermittedPath($lang_file) || !file_exists($lang_file)) { + $foundlang = false; + } else { + // Overwrite language-specific strings. + // This way we'll never have missing translation keys. + $foundlang = include $lang_file; + } + } + $this->language = $PHPMAILER_LANG; + + return (bool) $foundlang; // Returns false if language not found + } + + /** + * Get the array of strings for the current language. + * + * @return array + */ + public function getTranslations() + { + return $this->language; + } + + /** + * Create recipient headers. + * + * @param string $type + * @param array $addr An array of recipients, + * where each recipient is a 2-element indexed array with element 0 containing an address + * and element 1 containing a name, like: + * [['joe@example.com', 'Joe User'], ['zoe@example.com', 'Zoe User']] + * + * @return string + */ + public function addrAppend($type, $addr) + { + $addresses = []; + foreach ($addr as $address) { + $addresses[] = $this->addrFormat($address); + } + + return $type . ': ' . implode(', ', $addresses) . static::$LE; + } + + /** + * Format an address for use in a message header. + * + * @param array $addr A 2-element indexed array, element 0 containing an address, element 1 containing a name like + * ['joe@example.com', 'Joe User'] + * + * @return string + */ + public function addrFormat($addr) + { + if (empty($addr[1])) { // No name provided + return $this->secureHeader($addr[0]); + } + + return $this->encodeHeader($this->secureHeader($addr[1]), 'phrase') . + ' <' . $this->secureHeader($addr[0]) . '>'; + } + + /** + * Word-wrap message. + * For use with mailers that do not automatically perform wrapping + * and for quoted-printable encoded messages. + * Original written by philippe. + * + * @param string $message The message to wrap + * @param int $length The line length to wrap to + * @param bool $qp_mode Whether to run in Quoted-Printable mode + * + * @return string + */ + public function wrapText($message, $length, $qp_mode = false) + { + if ($qp_mode) { + $soft_break = sprintf(' =%s', static::$LE); + } else { + $soft_break = static::$LE; + } + // If utf-8 encoding is used, we will need to make sure we don't + // split multibyte characters when we wrap + $is_utf8 = static::CHARSET_UTF8 === strtolower($this->CharSet); + $lelen = strlen(static::$LE); + $crlflen = strlen(static::$LE); + + $message = static::normalizeBreaks($message); + //Remove a trailing line break + if (substr($message, -$lelen) === static::$LE) { + $message = substr($message, 0, -$lelen); + } + + //Split message into lines + $lines = explode(static::$LE, $message); + //Message will be rebuilt in here + $message = ''; + foreach ($lines as $line) { + $words = explode(' ', $line); + $buf = ''; + $firstword = true; + foreach ($words as $word) { + if ($qp_mode && (strlen($word) > $length)) { + $space_left = $length - strlen($buf) - $crlflen; + if (!$firstword) { + if ($space_left > 20) { + $len = $space_left; + if ($is_utf8) { + $len = $this->utf8CharBoundary($word, $len); + } elseif ('=' === substr($word, $len - 1, 1)) { + --$len; + } elseif ('=' === substr($word, $len - 2, 1)) { + $len -= 2; + } + $part = substr($word, 0, $len); + $word = substr($word, $len); + $buf .= ' ' . $part; + $message .= $buf . sprintf('=%s', static::$LE); + } else { + $message .= $buf . $soft_break; + } + $buf = ''; + } + while ($word !== '') { + if ($length <= 0) { + break; + } + $len = $length; + if ($is_utf8) { + $len = $this->utf8CharBoundary($word, $len); + } elseif ('=' === substr($word, $len - 1, 1)) { + --$len; + } elseif ('=' === substr($word, $len - 2, 1)) { + $len -= 2; + } + $part = substr($word, 0, $len); + $word = (string) substr($word, $len); + + if ($word !== '') { + $message .= $part . sprintf('=%s', static::$LE); + } else { + $buf = $part; + } + } + } else { + $buf_o = $buf; + if (!$firstword) { + $buf .= ' '; + } + $buf .= $word; + + if ('' !== $buf_o && strlen($buf) > $length) { + $message .= $buf_o . $soft_break; + $buf = $word; + } + } + $firstword = false; + } + $message .= $buf . static::$LE; + } + + return $message; + } + + /** + * Find the last character boundary prior to $maxLength in a utf-8 + * quoted-printable encoded string. + * Original written by Colin Brown. + * + * @param string $encodedText utf-8 QP text + * @param int $maxLength Find the last character boundary prior to this length + * + * @return int + */ + public function utf8CharBoundary($encodedText, $maxLength) + { + $foundSplitPos = false; + $lookBack = 3; + while (!$foundSplitPos) { + $lastChunk = substr($encodedText, $maxLength - $lookBack, $lookBack); + $encodedCharPos = strpos($lastChunk, '='); + if (false !== $encodedCharPos) { + // Found start of encoded character byte within $lookBack block. + // Check the encoded byte value (the 2 chars after the '=') + $hex = substr($encodedText, $maxLength - $lookBack + $encodedCharPos + 1, 2); + $dec = hexdec($hex); + if ($dec < 128) { + // Single byte character. + // If the encoded char was found at pos 0, it will fit + // otherwise reduce maxLength to start of the encoded char + if ($encodedCharPos > 0) { + $maxLength -= $lookBack - $encodedCharPos; + } + $foundSplitPos = true; + } elseif ($dec >= 192) { + // First byte of a multi byte character + // Reduce maxLength to split at start of character + $maxLength -= $lookBack - $encodedCharPos; + $foundSplitPos = true; + } elseif ($dec < 192) { + // Middle byte of a multi byte character, look further back + $lookBack += 3; + } + } else { + // No encoded character found + $foundSplitPos = true; + } + } + + return $maxLength; + } + + /** + * Apply word wrapping to the message body. + * Wraps the message body to the number of chars set in the WordWrap property. + * You should only do this to plain-text bodies as wrapping HTML tags may break them. + * This is called automatically by createBody(), so you don't need to call it yourself. + */ + public function setWordWrap() + { + if ($this->WordWrap < 1) { + return; + } + + switch ($this->message_type) { + case 'alt': + case 'alt_inline': + case 'alt_attach': + case 'alt_inline_attach': + $this->AltBody = $this->wrapText($this->AltBody, $this->WordWrap); + break; + default: + $this->Body = $this->wrapText($this->Body, $this->WordWrap); + break; + } + } + + /** + * Assemble message headers. + * + * @return string The assembled headers + */ + public function createHeader() + { + $result = ''; + + $result .= $this->headerLine('Date', '' === $this->MessageDate ? self::rfcDate() : $this->MessageDate); + + // To be created automatically by mail() + if ($this->SingleTo) { + if ('mail' !== $this->Mailer) { + foreach ($this->to as $toaddr) { + $this->SingleToArray[] = $this->addrFormat($toaddr); + } + } + } elseif (count($this->to) > 0) { + if ('mail' !== $this->Mailer) { + $result .= $this->addrAppend('To', $this->to); + } + } elseif (count($this->cc) === 0) { + $result .= $this->headerLine('To', 'undisclosed-recipients:;'); + } + + $result .= $this->addrAppend('From', [[trim($this->From), $this->FromName]]); + + // sendmail and mail() extract Cc from the header before sending + if (count($this->cc) > 0) { + $result .= $this->addrAppend('Cc', $this->cc); + } + + // sendmail and mail() extract Bcc from the header before sending + if (( + 'sendmail' === $this->Mailer || 'qmail' === $this->Mailer || 'mail' === $this->Mailer + ) + && count($this->bcc) > 0 + ) { + $result .= $this->addrAppend('Bcc', $this->bcc); + } + + if (count($this->ReplyTo) > 0) { + $result .= $this->addrAppend('Reply-To', $this->ReplyTo); + } + + // mail() sets the subject itself + if ('mail' !== $this->Mailer) { + $result .= $this->headerLine('Subject', $this->encodeHeader($this->secureHeader($this->Subject))); + } + + // Only allow a custom message ID if it conforms to RFC 5322 section 3.6.4 + // https://tools.ietf.org/html/rfc5322#section-3.6.4 + if ('' !== $this->MessageID && preg_match('/^<.*@.*>$/', $this->MessageID)) { + $this->lastMessageID = $this->MessageID; + } else { + $this->lastMessageID = sprintf('<%s@%s>', $this->uniqueid, $this->serverHostname()); + } + $result .= $this->headerLine('Message-ID', $this->lastMessageID); + if (null !== $this->Priority) { + $result .= $this->headerLine('X-Priority', $this->Priority); + } + if ('' === $this->XMailer) { + $result .= $this->headerLine( + 'X-Mailer', + 'PHPMailer ' . self::VERSION . ' (https://github.com/PHPMailer/PHPMailer)' + ); + } else { + $myXmailer = trim($this->XMailer); + if ($myXmailer) { + $result .= $this->headerLine('X-Mailer', $myXmailer); + } + } + + if ('' !== $this->ConfirmReadingTo) { + $result .= $this->headerLine('Disposition-Notification-To', '<' . $this->ConfirmReadingTo . '>'); + } + + // Add custom headers + foreach ($this->CustomHeader as $header) { + $result .= $this->headerLine( + trim($header[0]), + $this->encodeHeader(trim($header[1])) + ); + } + if (!$this->sign_key_file) { + $result .= $this->headerLine('MIME-Version', '1.0'); + $result .= $this->getMailMIME(); + } + + return $result; + } + + /** + * Get the message MIME type headers. + * + * @return string + */ + public function getMailMIME() + { + $result = ''; + $ismultipart = true; + switch ($this->message_type) { + case 'inline': + $result .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_RELATED . ';'); + $result .= $this->textLine(' boundary="' . $this->boundary[1] . '"'); + break; + case 'attach': + case 'inline_attach': + case 'alt_attach': + case 'alt_inline_attach': + $result .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_MIXED . ';'); + $result .= $this->textLine(' boundary="' . $this->boundary[1] . '"'); + break; + case 'alt': + case 'alt_inline': + $result .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_ALTERNATIVE . ';'); + $result .= $this->textLine(' boundary="' . $this->boundary[1] . '"'); + break; + default: + // Catches case 'plain': and case '': + $result .= $this->textLine('Content-Type: ' . $this->ContentType . '; charset=' . $this->CharSet); + $ismultipart = false; + break; + } + // RFC1341 part 5 says 7bit is assumed if not specified + if (static::ENCODING_7BIT !== $this->Encoding) { + // RFC 2045 section 6.4 says multipart MIME parts may only use 7bit, 8bit or binary CTE + if ($ismultipart) { + if (static::ENCODING_8BIT === $this->Encoding) { + $result .= $this->headerLine('Content-Transfer-Encoding', static::ENCODING_8BIT); + } + // The only remaining alternatives are quoted-printable and base64, which are both 7bit compatible + } else { + $result .= $this->headerLine('Content-Transfer-Encoding', $this->Encoding); + } + } + + if ('mail' !== $this->Mailer) { +// $result .= static::$LE; + } + + return $result; + } + + /** + * Returns the whole MIME message. + * Includes complete headers and body. + * Only valid post preSend(). + * + * @see PHPMailer::preSend() + * + * @return string + */ + public function getSentMIMEMessage() + { + return static::stripTrailingWSP($this->MIMEHeader . $this->mailHeader) . + static::$LE . static::$LE . $this->MIMEBody; + } + + /** + * Create a unique ID to use for boundaries. + * + * @return string + */ + protected function generateId() + { + $len = 32; //32 bytes = 256 bits + $bytes = ''; + if (function_exists('random_bytes')) { + try { + $bytes = random_bytes($len); + } catch (\Exception $e) { + //Do nothing + } + } elseif (function_exists('openssl_random_pseudo_bytes')) { + /** @noinspection CryptographicallySecureRandomnessInspection */ + $bytes = openssl_random_pseudo_bytes($len); + } + if ($bytes === '') { + //We failed to produce a proper random string, so make do. + //Use a hash to force the length to the same as the other methods + $bytes = hash('sha256', uniqid((string) mt_rand(), true), true); + } + + //We don't care about messing up base64 format here, just want a random string + return str_replace(['=', '+', '/'], '', base64_encode(hash('sha256', $bytes, true))); + } + + /** + * Assemble the message body. + * Returns an empty string on failure. + * + * @throws Exception + * + * @return string The assembled message body + */ + public function createBody() + { + $body = ''; + //Create unique IDs and preset boundaries + $this->uniqueid = $this->generateId(); + $this->boundary[1] = 'b1_' . $this->uniqueid; + $this->boundary[2] = 'b2_' . $this->uniqueid; + $this->boundary[3] = 'b3_' . $this->uniqueid; + + if ($this->sign_key_file) { + $body .= $this->getMailMIME() . static::$LE; + } + + $this->setWordWrap(); + + $bodyEncoding = $this->Encoding; + $bodyCharSet = $this->CharSet; + //Can we do a 7-bit downgrade? + if (static::ENCODING_8BIT === $bodyEncoding && !$this->has8bitChars($this->Body)) { + $bodyEncoding = static::ENCODING_7BIT; + //All ISO 8859, Windows codepage and UTF-8 charsets are ascii compatible up to 7-bit + $bodyCharSet = static::CHARSET_ASCII; + } + //If lines are too long, and we're not already using an encoding that will shorten them, + //change to quoted-printable transfer encoding for the body part only + if (static::ENCODING_BASE64 !== $this->Encoding && static::hasLineLongerThanMax($this->Body)) { + $bodyEncoding = static::ENCODING_QUOTED_PRINTABLE; + } + + $altBodyEncoding = $this->Encoding; + $altBodyCharSet = $this->CharSet; + //Can we do a 7-bit downgrade? + if (static::ENCODING_8BIT === $altBodyEncoding && !$this->has8bitChars($this->AltBody)) { + $altBodyEncoding = static::ENCODING_7BIT; + //All ISO 8859, Windows codepage and UTF-8 charsets are ascii compatible up to 7-bit + $altBodyCharSet = static::CHARSET_ASCII; + } + //If lines are too long, and we're not already using an encoding that will shorten them, + //change to quoted-printable transfer encoding for the alt body part only + if (static::ENCODING_BASE64 !== $altBodyEncoding && static::hasLineLongerThanMax($this->AltBody)) { + $altBodyEncoding = static::ENCODING_QUOTED_PRINTABLE; + } + //Use this as a preamble in all multipart message types + $mimepre = 'This is a multi-part message in MIME format.' . static::$LE . static::$LE; + switch ($this->message_type) { + case 'inline': + $body .= $mimepre; + $body .= $this->getBoundary($this->boundary[1], $bodyCharSet, '', $bodyEncoding); + $body .= $this->encodeString($this->Body, $bodyEncoding); + $body .= static::$LE; + $body .= $this->attachAll('inline', $this->boundary[1]); + break; + case 'attach': + $body .= $mimepre; + $body .= $this->getBoundary($this->boundary[1], $bodyCharSet, '', $bodyEncoding); + $body .= $this->encodeString($this->Body, $bodyEncoding); + $body .= static::$LE; + $body .= $this->attachAll('attachment', $this->boundary[1]); + break; + case 'inline_attach': + $body .= $mimepre; + $body .= $this->textLine('--' . $this->boundary[1]); + $body .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_RELATED . ';'); + $body .= $this->textLine(' boundary="' . $this->boundary[2] . '";'); + $body .= $this->textLine(' type="' . static::CONTENT_TYPE_TEXT_HTML . '"'); + $body .= static::$LE; + $body .= $this->getBoundary($this->boundary[2], $bodyCharSet, '', $bodyEncoding); + $body .= $this->encodeString($this->Body, $bodyEncoding); + $body .= static::$LE; + $body .= $this->attachAll('inline', $this->boundary[2]); + $body .= static::$LE; + $body .= $this->attachAll('attachment', $this->boundary[1]); + break; + case 'alt': + $body .= $mimepre; + $body .= $this->getBoundary( + $this->boundary[1], + $altBodyCharSet, + static::CONTENT_TYPE_PLAINTEXT, + $altBodyEncoding + ); + $body .= $this->encodeString($this->AltBody, $altBodyEncoding); + $body .= static::$LE; + $body .= $this->getBoundary( + $this->boundary[1], + $bodyCharSet, + static::CONTENT_TYPE_TEXT_HTML, + $bodyEncoding + ); + $body .= $this->encodeString($this->Body, $bodyEncoding); + $body .= static::$LE; + if (!empty($this->Ical)) { + $method = static::ICAL_METHOD_REQUEST; + foreach (static::$IcalMethods as $imethod) { + if (stripos($this->Ical, 'METHOD:' . $imethod) !== false) { + $method = $imethod; + break; + } + } + $body .= $this->getBoundary( + $this->boundary[1], + '', + static::CONTENT_TYPE_TEXT_CALENDAR . '; method=' . $method, + '' + ); + $body .= $this->encodeString($this->Ical, $this->Encoding); + $body .= static::$LE; + } + $body .= $this->endBoundary($this->boundary[1]); + break; + case 'alt_inline': + $body .= $mimepre; + $body .= $this->getBoundary( + $this->boundary[1], + $altBodyCharSet, + static::CONTENT_TYPE_PLAINTEXT, + $altBodyEncoding + ); + $body .= $this->encodeString($this->AltBody, $altBodyEncoding); + $body .= static::$LE; + $body .= $this->textLine('--' . $this->boundary[1]); + $body .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_RELATED . ';'); + $body .= $this->textLine(' boundary="' . $this->boundary[2] . '";'); + $body .= $this->textLine(' type="' . static::CONTENT_TYPE_TEXT_HTML . '"'); + $body .= static::$LE; + $body .= $this->getBoundary( + $this->boundary[2], + $bodyCharSet, + static::CONTENT_TYPE_TEXT_HTML, + $bodyEncoding + ); + $body .= $this->encodeString($this->Body, $bodyEncoding); + $body .= static::$LE; + $body .= $this->attachAll('inline', $this->boundary[2]); + $body .= static::$LE; + $body .= $this->endBoundary($this->boundary[1]); + break; + case 'alt_attach': + $body .= $mimepre; + $body .= $this->textLine('--' . $this->boundary[1]); + $body .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_ALTERNATIVE . ';'); + $body .= $this->textLine(' boundary="' . $this->boundary[2] . '"'); + $body .= static::$LE; + $body .= $this->getBoundary( + $this->boundary[2], + $altBodyCharSet, + static::CONTENT_TYPE_PLAINTEXT, + $altBodyEncoding + ); + $body .= $this->encodeString($this->AltBody, $altBodyEncoding); + $body .= static::$LE; + $body .= $this->getBoundary( + $this->boundary[2], + $bodyCharSet, + static::CONTENT_TYPE_TEXT_HTML, + $bodyEncoding + ); + $body .= $this->encodeString($this->Body, $bodyEncoding); + $body .= static::$LE; + if (!empty($this->Ical)) { + $method = static::ICAL_METHOD_REQUEST; + foreach (static::$IcalMethods as $imethod) { + if (stripos($this->Ical, 'METHOD:' . $imethod) !== false) { + $method = $imethod; + break; + } + } + $body .= $this->getBoundary( + $this->boundary[2], + '', + static::CONTENT_TYPE_TEXT_CALENDAR . '; method=' . $method, + '' + ); + $body .= $this->encodeString($this->Ical, $this->Encoding); + } + $body .= $this->endBoundary($this->boundary[2]); + $body .= static::$LE; + $body .= $this->attachAll('attachment', $this->boundary[1]); + break; + case 'alt_inline_attach': + $body .= $mimepre; + $body .= $this->textLine('--' . $this->boundary[1]); + $body .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_ALTERNATIVE . ';'); + $body .= $this->textLine(' boundary="' . $this->boundary[2] . '"'); + $body .= static::$LE; + $body .= $this->getBoundary( + $this->boundary[2], + $altBodyCharSet, + static::CONTENT_TYPE_PLAINTEXT, + $altBodyEncoding + ); + $body .= $this->encodeString($this->AltBody, $altBodyEncoding); + $body .= static::$LE; + $body .= $this->textLine('--' . $this->boundary[2]); + $body .= $this->headerLine('Content-Type', static::CONTENT_TYPE_MULTIPART_RELATED . ';'); + $body .= $this->textLine(' boundary="' . $this->boundary[3] . '";'); + $body .= $this->textLine(' type="' . static::CONTENT_TYPE_TEXT_HTML . '"'); + $body .= static::$LE; + $body .= $this->getBoundary( + $this->boundary[3], + $bodyCharSet, + static::CONTENT_TYPE_TEXT_HTML, + $bodyEncoding + ); + $body .= $this->encodeString($this->Body, $bodyEncoding); + $body .= static::$LE; + $body .= $this->attachAll('inline', $this->boundary[3]); + $body .= static::$LE; + $body .= $this->endBoundary($this->boundary[2]); + $body .= static::$LE; + $body .= $this->attachAll('attachment', $this->boundary[1]); + break; + default: + // Catch case 'plain' and case '', applies to simple `text/plain` and `text/html` body content types + //Reset the `Encoding` property in case we changed it for line length reasons + $this->Encoding = $bodyEncoding; + $body .= $this->encodeString($this->Body, $this->Encoding); + break; + } + + if ($this->isError()) { + $body = ''; + if ($this->exceptions) { + throw new Exception($this->lang('empty_message'), self::STOP_CRITICAL); + } + } elseif ($this->sign_key_file) { + try { + if (!defined('PKCS7_TEXT')) { + throw new Exception($this->lang('extension_missing') . 'openssl'); + } + + $file = tempnam(sys_get_temp_dir(), 'srcsign'); + $signed = tempnam(sys_get_temp_dir(), 'mailsign'); + file_put_contents($file, $body); + + //Workaround for PHP bug https://bugs.php.net/bug.php?id=69197 + if (empty($this->sign_extracerts_file)) { + $sign = @openssl_pkcs7_sign( + $file, + $signed, + 'file://' . realpath($this->sign_cert_file), + ['file://' . realpath($this->sign_key_file), $this->sign_key_pass], + [] + ); + } else { + $sign = @openssl_pkcs7_sign( + $file, + $signed, + 'file://' . realpath($this->sign_cert_file), + ['file://' . realpath($this->sign_key_file), $this->sign_key_pass], + [], + PKCS7_DETACHED, + $this->sign_extracerts_file + ); + } + + @unlink($file); + if ($sign) { + $body = file_get_contents($signed); + @unlink($signed); + //The message returned by openssl contains both headers and body, so need to split them up + $parts = explode("\n\n", $body, 2); + $this->MIMEHeader .= $parts[0] . static::$LE . static::$LE; + $body = $parts[1]; + } else { + @unlink($signed); + throw new Exception($this->lang('signing') . openssl_error_string()); + } + } catch (Exception $exc) { + $body = ''; + if ($this->exceptions) { + throw $exc; + } + } + } + + return $body; + } + + /** + * Return the start of a message boundary. + * + * @param string $boundary + * @param string $charSet + * @param string $contentType + * @param string $encoding + * + * @return string + */ + protected function getBoundary($boundary, $charSet, $contentType, $encoding) + { + $result = ''; + if ('' === $charSet) { + $charSet = $this->CharSet; + } + if ('' === $contentType) { + $contentType = $this->ContentType; + } + if ('' === $encoding) { + $encoding = $this->Encoding; + } + $result .= $this->textLine('--' . $boundary); + $result .= sprintf('Content-Type: %s; charset=%s', $contentType, $charSet); + $result .= static::$LE; + // RFC1341 part 5 says 7bit is assumed if not specified + if (static::ENCODING_7BIT !== $encoding) { + $result .= $this->headerLine('Content-Transfer-Encoding', $encoding); + } + $result .= static::$LE; + + return $result; + } + + /** + * Return the end of a message boundary. + * + * @param string $boundary + * + * @return string + */ + protected function endBoundary($boundary) + { + return static::$LE . '--' . $boundary . '--' . static::$LE; + } + + /** + * Set the message type. + * PHPMailer only supports some preset message types, not arbitrary MIME structures. + */ + protected function setMessageType() + { + $type = []; + if ($this->alternativeExists()) { + $type[] = 'alt'; + } + if ($this->inlineImageExists()) { + $type[] = 'inline'; + } + if ($this->attachmentExists()) { + $type[] = 'attach'; + } + $this->message_type = implode('_', $type); + if ('' === $this->message_type) { + //The 'plain' message_type refers to the message having a single body element, not that it is plain-text + $this->message_type = 'plain'; + } + } + + /** + * Format a header line. + * + * @param string $name + * @param string|int $value + * + * @return string + */ + public function headerLine($name, $value) + { + return $name . ': ' . $value . static::$LE; + } + + /** + * Return a formatted mail line. + * + * @param string $value + * + * @return string + */ + public function textLine($value) + { + return $value . static::$LE; + } + + /** + * Add an attachment from a path on the filesystem. + * Never use a user-supplied path to a file! + * Returns false if the file could not be found or read. + * Explicitly *does not* support passing URLs; PHPMailer is not an HTTP client. + * If you need to do that, fetch the resource yourself and pass it in via a local file or string. + * + * @param string $path Path to the attachment + * @param string $name Overrides the attachment name + * @param string $encoding File encoding (see $Encoding) + * @param string $type File extension (MIME) type + * @param string $disposition Disposition to use + * + * @throws Exception + * + * @return bool + */ + public function addAttachment( + $path, + $name = '', + $encoding = self::ENCODING_BASE64, + $type = '', + $disposition = 'attachment' + ) { + try { + if (!static::isPermittedPath($path) || !@is_file($path) || !is_readable($path)) { + throw new Exception($this->lang('file_access') . $path, self::STOP_CONTINUE); + } + + // If a MIME type is not specified, try to work it out from the file name + if ('' === $type) { + $type = static::filenameToType($path); + } + + $filename = (string) static::mb_pathinfo($path, PATHINFO_BASENAME); + if ('' === $name) { + $name = $filename; + } + + if (!$this->validateEncoding($encoding)) { + throw new Exception($this->lang('encoding') . $encoding); + } + + $this->attachment[] = [ + 0 => $path, + 1 => $filename, + 2 => $name, + 3 => $encoding, + 4 => $type, + 5 => false, // isStringAttachment + 6 => $disposition, + 7 => $name, + ]; + } catch (Exception $exc) { + $this->setError($exc->getMessage()); + $this->edebug($exc->getMessage()); + if ($this->exceptions) { + throw $exc; + } + + return false; + } + + return true; + } + + /** + * Return the array of attachments. + * + * @return array + */ + public function getAttachments() + { + return $this->attachment; + } + + /** + * Attach all file, string, and binary attachments to the message. + * Returns an empty string on failure. + * + * @param string $disposition_type + * @param string $boundary + * + * @throws Exception + * + * @return string + */ + protected function attachAll($disposition_type, $boundary) + { + // Return text of body + $mime = []; + $cidUniq = []; + $incl = []; + + // Add all attachments + foreach ($this->attachment as $attachment) { + // Check if it is a valid disposition_filter + if ($attachment[6] === $disposition_type) { + // Check for string attachment + $string = ''; + $path = ''; + $bString = $attachment[5]; + if ($bString) { + $string = $attachment[0]; + } else { + $path = $attachment[0]; + } + + $inclhash = hash('sha256', serialize($attachment)); + if (in_array($inclhash, $incl, true)) { + continue; + } + $incl[] = $inclhash; + $name = $attachment[2]; + $encoding = $attachment[3]; + $type = $attachment[4]; + $disposition = $attachment[6]; + $cid = $attachment[7]; + if ('inline' === $disposition && array_key_exists($cid, $cidUniq)) { + continue; + } + $cidUniq[$cid] = true; + + $mime[] = sprintf('--%s%s', $boundary, static::$LE); + //Only include a filename property if we have one + if (!empty($name)) { + $mime[] = sprintf( + 'Content-Type: %s; name="%s"%s', + $type, + $this->encodeHeader($this->secureHeader($name)), + static::$LE + ); + } else { + $mime[] = sprintf( + 'Content-Type: %s%s', + $type, + static::$LE + ); + } + // RFC1341 part 5 says 7bit is assumed if not specified + if (static::ENCODING_7BIT !== $encoding) { + $mime[] = sprintf('Content-Transfer-Encoding: %s%s', $encoding, static::$LE); + } + + //Only set Content-IDs on inline attachments + if ((string) $cid !== '' && $disposition === 'inline') { + $mime[] = 'Content-ID: <' . $this->encodeHeader($this->secureHeader($cid)) . '>' . static::$LE; + } + + // If a filename contains any of these chars, it should be quoted, + // but not otherwise: RFC2183 & RFC2045 5.1 + // Fixes a warning in IETF's msglint MIME checker + // Allow for bypassing the Content-Disposition header totally + if (!empty($disposition)) { + $encoded_name = $this->encodeHeader($this->secureHeader($name)); + if (preg_match('/[ ()<>@,;:"\/\[\]?=]/', $encoded_name)) { + $mime[] = sprintf( + 'Content-Disposition: %s; filename="%s"%s', + $disposition, + $encoded_name, + static::$LE . static::$LE + ); + } elseif (!empty($encoded_name)) { + $mime[] = sprintf( + 'Content-Disposition: %s; filename=%s%s', + $disposition, + $encoded_name, + static::$LE . static::$LE + ); + } else { + $mime[] = sprintf( + 'Content-Disposition: %s%s', + $disposition, + static::$LE . static::$LE + ); + } + } else { + $mime[] = static::$LE; + } + + // Encode as string attachment + if ($bString) { + $mime[] = $this->encodeString($string, $encoding); + } else { + $mime[] = $this->encodeFile($path, $encoding); + } + if ($this->isError()) { + return ''; + } + $mime[] = static::$LE; + } + } + + $mime[] = sprintf('--%s--%s', $boundary, static::$LE); + + return implode('', $mime); + } + + /** + * Encode a file attachment in requested format. + * Returns an empty string on failure. + * + * @param string $path The full path to the file + * @param string $encoding The encoding to use; one of 'base64', '7bit', '8bit', 'binary', 'quoted-printable' + * + * @return string + */ + protected function encodeFile($path, $encoding = self::ENCODING_BASE64) + { + try { + if (!static::isPermittedPath($path) || !file_exists($path) || !is_readable($path)) { + throw new Exception($this->lang('file_open') . $path, self::STOP_CONTINUE); + } + $file_buffer = file_get_contents($path); + if (false === $file_buffer) { + throw new Exception($this->lang('file_open') . $path, self::STOP_CONTINUE); + } + $file_buffer = $this->encodeString($file_buffer, $encoding); + + return $file_buffer; + } catch (Exception $exc) { + $this->setError($exc->getMessage()); + $this->edebug($exc->getMessage()); + if ($this->exceptions) { + throw $exc; + } + return ''; + } + } + + /** + * Encode a string in requested format. + * Returns an empty string on failure. + * + * @param string $str The text to encode + * @param string $encoding The encoding to use; one of 'base64', '7bit', '8bit', 'binary', 'quoted-printable' + * + * @throws Exception + * + * @return string + */ + public function encodeString($str, $encoding = self::ENCODING_BASE64) + { + $encoded = ''; + switch (strtolower($encoding)) { + case static::ENCODING_BASE64: + $encoded = chunk_split( + base64_encode($str), + static::STD_LINE_LENGTH, + static::$LE + ); + break; + case static::ENCODING_7BIT: + case static::ENCODING_8BIT: + $encoded = static::normalizeBreaks($str); + // Make sure it ends with a line break + if (substr($encoded, -(strlen(static::$LE))) !== static::$LE) { + $encoded .= static::$LE; + } + break; + case static::ENCODING_BINARY: + $encoded = $str; + break; + case static::ENCODING_QUOTED_PRINTABLE: + $encoded = $this->encodeQP($str); + break; + default: + $this->setError($this->lang('encoding') . $encoding); + if ($this->exceptions) { + throw new Exception($this->lang('encoding') . $encoding); + } + break; + } + + return $encoded; + } + + /** + * Encode a header value (not including its label) optimally. + * Picks shortest of Q, B, or none. Result includes folding if needed. + * See RFC822 definitions for phrase, comment and text positions. + * + * @param string $str The header value to encode + * @param string $position What context the string will be used in + * + * @return string + */ + public function encodeHeader($str, $position = 'text') + { + $matchcount = 0; + switch (strtolower($position)) { + case 'phrase': + if (!preg_match('/[\200-\377]/', $str)) { + // Can't use addslashes as we don't know the value of magic_quotes_sybase + $encoded = addcslashes($str, "\0..\37\177\\\""); + if (($str === $encoded) && !preg_match('/[^A-Za-z0-9!#$%&\'*+\/=?^_`{|}~ -]/', $str)) { + return $encoded; + } + + return "\"$encoded\""; + } + $matchcount = preg_match_all('/[^\040\041\043-\133\135-\176]/', $str, $matches); + break; + /* @noinspection PhpMissingBreakStatementInspection */ + case 'comment': + $matchcount = preg_match_all('/[()"]/', $str, $matches); + //fallthrough + case 'text': + default: + $matchcount += preg_match_all('/[\000-\010\013\014\016-\037\177-\377]/', $str, $matches); + break; + } + + if ($this->has8bitChars($str)) { + $charset = $this->CharSet; + } else { + $charset = static::CHARSET_ASCII; + } + + // Q/B encoding adds 8 chars and the charset ("` =??[QB]??=`"). + $overhead = 8 + strlen($charset); + + if ('mail' === $this->Mailer) { + $maxlen = static::MAIL_MAX_LINE_LENGTH - $overhead; + } else { + $maxlen = static::MAX_LINE_LENGTH - $overhead; + } + + // Select the encoding that produces the shortest output and/or prevents corruption. + if ($matchcount > strlen($str) / 3) { + // More than 1/3 of the content needs encoding, use B-encode. + $encoding = 'B'; + } elseif ($matchcount > 0) { + // Less than 1/3 of the content needs encoding, use Q-encode. + $encoding = 'Q'; + } elseif (strlen($str) > $maxlen) { + // No encoding needed, but value exceeds max line length, use Q-encode to prevent corruption. + $encoding = 'Q'; + } else { + // No reformatting needed + $encoding = false; + } + + switch ($encoding) { + case 'B': + if ($this->hasMultiBytes($str)) { + // Use a custom function which correctly encodes and wraps long + // multibyte strings without breaking lines within a character + $encoded = $this->base64EncodeWrapMB($str, "\n"); + } else { + $encoded = base64_encode($str); + $maxlen -= $maxlen % 4; + $encoded = trim(chunk_split($encoded, $maxlen, "\n")); + } + $encoded = preg_replace('/^(.*)$/m', ' =?' . $charset . "?$encoding?\\1?=", $encoded); + break; + case 'Q': + $encoded = $this->encodeQ($str, $position); + $encoded = $this->wrapText($encoded, $maxlen, true); + $encoded = str_replace('=' . static::$LE, "\n", trim($encoded)); + $encoded = preg_replace('/^(.*)$/m', ' =?' . $charset . "?$encoding?\\1?=", $encoded); + break; + default: + return $str; + } + + return trim(static::normalizeBreaks($encoded)); + } + + /** + * Check if a string contains multi-byte characters. + * + * @param string $str multi-byte text to wrap encode + * + * @return bool + */ + public function hasMultiBytes($str) + { + if (function_exists('mb_strlen')) { + return strlen($str) > mb_strlen($str, $this->CharSet); + } + + // Assume no multibytes (we can't handle without mbstring functions anyway) + return false; + } + + /** + * Does a string contain any 8-bit chars (in any charset)? + * + * @param string $text + * + * @return bool + */ + public function has8bitChars($text) + { + return (bool) preg_match('/[\x80-\xFF]/', $text); + } + + /** + * Encode and wrap long multibyte strings for mail headers + * without breaking lines within a character. + * Adapted from a function by paravoid. + * + * @see http://www.php.net/manual/en/function.mb-encode-mimeheader.php#60283 + * + * @param string $str multi-byte text to wrap encode + * @param string $linebreak string to use as linefeed/end-of-line + * + * @return string + */ + public function base64EncodeWrapMB($str, $linebreak = null) + { + $start = '=?' . $this->CharSet . '?B?'; + $end = '?='; + $encoded = ''; + if (null === $linebreak) { + $linebreak = static::$LE; + } + + $mb_length = mb_strlen($str, $this->CharSet); + // Each line must have length <= 75, including $start and $end + $length = 75 - strlen($start) - strlen($end); + // Average multi-byte ratio + $ratio = $mb_length / strlen($str); + // Base64 has a 4:3 ratio + $avgLength = floor($length * $ratio * .75); + + $offset = 0; + for ($i = 0; $i < $mb_length; $i += $offset) { + $lookBack = 0; + do { + $offset = $avgLength - $lookBack; + $chunk = mb_substr($str, $i, $offset, $this->CharSet); + $chunk = base64_encode($chunk); + ++$lookBack; + } while (strlen($chunk) > $length); + $encoded .= $chunk . $linebreak; + } + + // Chomp the last linefeed + return substr($encoded, 0, -strlen($linebreak)); + } + + /** + * Encode a string in quoted-printable format. + * According to RFC2045 section 6.7. + * + * @param string $string The text to encode + * + * @return string + */ + public function encodeQP($string) + { + return static::normalizeBreaks(quoted_printable_encode($string)); + } + + /** + * Encode a string using Q encoding. + * + * @see http://tools.ietf.org/html/rfc2047#section-4.2 + * + * @param string $str the text to encode + * @param string $position Where the text is going to be used, see the RFC for what that means + * + * @return string + */ + public function encodeQ($str, $position = 'text') + { + // There should not be any EOL in the string + $pattern = ''; + $encoded = str_replace(["\r", "\n"], '', $str); + switch (strtolower($position)) { + case 'phrase': + // RFC 2047 section 5.3 + $pattern = '^A-Za-z0-9!*+\/ -'; + break; + /* + * RFC 2047 section 5.2. + * Build $pattern without including delimiters and [] + */ + /* @noinspection PhpMissingBreakStatementInspection */ + case 'comment': + $pattern = '\(\)"'; + /* Intentional fall through */ + case 'text': + default: + // RFC 2047 section 5.1 + // Replace every high ascii, control, =, ? and _ characters + $pattern = '\000-\011\013\014\016-\037\075\077\137\177-\377' . $pattern; + break; + } + $matches = []; + if (preg_match_all("/[{$pattern}]/", $encoded, $matches)) { + // If the string contains an '=', make sure it's the first thing we replace + // so as to avoid double-encoding + $eqkey = array_search('=', $matches[0], true); + if (false !== $eqkey) { + unset($matches[0][$eqkey]); + array_unshift($matches[0], '='); + } + foreach (array_unique($matches[0]) as $char) { + $encoded = str_replace($char, '=' . sprintf('%02X', ord($char)), $encoded); + } + } + // Replace spaces with _ (more readable than =20) + // RFC 2047 section 4.2(2) + return str_replace(' ', '_', $encoded); + } + + /** + * Add a string or binary attachment (non-filesystem). + * This method can be used to attach ascii or binary data, + * such as a BLOB record from a database. + * + * @param string $string String attachment data + * @param string $filename Name of the attachment + * @param string $encoding File encoding (see $Encoding) + * @param string $type File extension (MIME) type + * @param string $disposition Disposition to use + * + * @throws Exception + * + * @return bool True on successfully adding an attachment + */ + public function addStringAttachment( + $string, + $filename, + $encoding = self::ENCODING_BASE64, + $type = '', + $disposition = 'attachment' + ) { + try { + // If a MIME type is not specified, try to work it out from the file name + if ('' === $type) { + $type = static::filenameToType($filename); + } + + if (!$this->validateEncoding($encoding)) { + throw new Exception($this->lang('encoding') . $encoding); + } + + // Append to $attachment array + $this->attachment[] = [ + 0 => $string, + 1 => $filename, + 2 => static::mb_pathinfo($filename, PATHINFO_BASENAME), + 3 => $encoding, + 4 => $type, + 5 => true, // isStringAttachment + 6 => $disposition, + 7 => 0, + ]; + } catch (Exception $exc) { + $this->setError($exc->getMessage()); + $this->edebug($exc->getMessage()); + if ($this->exceptions) { + throw $exc; + } + + return false; + } + + return true; + } + + /** + * Add an embedded (inline) attachment from a file. + * This can include images, sounds, and just about any other document type. + * These differ from 'regular' attachments in that they are intended to be + * displayed inline with the message, not just attached for download. + * This is used in HTML messages that embed the images + * the HTML refers to using the $cid value. + * Never use a user-supplied path to a file! + * + * @param string $path Path to the attachment + * @param string $cid Content ID of the attachment; Use this to reference + * the content when using an embedded image in HTML + * @param string $name Overrides the attachment name + * @param string $encoding File encoding (see $Encoding) + * @param string $type File MIME type + * @param string $disposition Disposition to use + * + * @throws Exception + * + * @return bool True on successfully adding an attachment + */ + public function addEmbeddedImage( + $path, + $cid, + $name = '', + $encoding = self::ENCODING_BASE64, + $type = '', + $disposition = 'inline' + ) { + try { + if (!static::isPermittedPath($path) || !@is_file($path) || !is_readable($path)) { + throw new Exception($this->lang('file_access') . $path, self::STOP_CONTINUE); + } + + // If a MIME type is not specified, try to work it out from the file name + if ('' === $type) { + $type = static::filenameToType($path); + } + + if (!$this->validateEncoding($encoding)) { + throw new Exception($this->lang('encoding') . $encoding); + } + + $filename = (string) static::mb_pathinfo($path, PATHINFO_BASENAME); + if ('' === $name) { + $name = $filename; + } + + // Append to $attachment array + $this->attachment[] = [ + 0 => $path, + 1 => $filename, + 2 => $name, + 3 => $encoding, + 4 => $type, + 5 => false, // isStringAttachment + 6 => $disposition, + 7 => $cid, + ]; + } catch (Exception $exc) { + $this->setError($exc->getMessage()); + $this->edebug($exc->getMessage()); + if ($this->exceptions) { + throw $exc; + } + + return false; + } + + return true; + } + + /** + * Add an embedded stringified attachment. + * This can include images, sounds, and just about any other document type. + * If your filename doesn't contain an extension, be sure to set the $type to an appropriate MIME type. + * + * @param string $string The attachment binary data + * @param string $cid Content ID of the attachment; Use this to reference + * the content when using an embedded image in HTML + * @param string $name A filename for the attachment. If this contains an extension, + * PHPMailer will attempt to set a MIME type for the attachment. + * For example 'file.jpg' would get an 'image/jpeg' MIME type. + * @param string $encoding File encoding (see $Encoding), defaults to 'base64' + * @param string $type MIME type - will be used in preference to any automatically derived type + * @param string $disposition Disposition to use + * + * @throws Exception + * + * @return bool True on successfully adding an attachment + */ + public function addStringEmbeddedImage( + $string, + $cid, + $name = '', + $encoding = self::ENCODING_BASE64, + $type = '', + $disposition = 'inline' + ) { + try { + // If a MIME type is not specified, try to work it out from the name + if ('' === $type && !empty($name)) { + $type = static::filenameToType($name); + } + + if (!$this->validateEncoding($encoding)) { + throw new Exception($this->lang('encoding') . $encoding); + } + + // Append to $attachment array + $this->attachment[] = [ + 0 => $string, + 1 => $name, + 2 => $name, + 3 => $encoding, + 4 => $type, + 5 => true, // isStringAttachment + 6 => $disposition, + 7 => $cid, + ]; + } catch (Exception $exc) { + $this->setError($exc->getMessage()); + $this->edebug($exc->getMessage()); + if ($this->exceptions) { + throw $exc; + } + + return false; + } + + return true; + } + + /** + * Validate encodings. + * + * @param string $encoding + * + * @return bool + */ + protected function validateEncoding($encoding) + { + return in_array( + $encoding, + [ + self::ENCODING_7BIT, + self::ENCODING_QUOTED_PRINTABLE, + self::ENCODING_BASE64, + self::ENCODING_8BIT, + self::ENCODING_BINARY, + ], + true + ); + } + + /** + * Check if an embedded attachment is present with this cid. + * + * @param string $cid + * + * @return bool + */ + protected function cidExists($cid) + { + foreach ($this->attachment as $attachment) { + if ('inline' === $attachment[6] && $cid === $attachment[7]) { + return true; + } + } + + return false; + } + + /** + * Check if an inline attachment is present. + * + * @return bool + */ + public function inlineImageExists() + { + foreach ($this->attachment as $attachment) { + if ('inline' === $attachment[6]) { + return true; + } + } + + return false; + } + + /** + * Check if an attachment (non-inline) is present. + * + * @return bool + */ + public function attachmentExists() + { + foreach ($this->attachment as $attachment) { + if ('attachment' === $attachment[6]) { + return true; + } + } + + return false; + } + + /** + * Check if this message has an alternative body set. + * + * @return bool + */ + public function alternativeExists() + { + return !empty($this->AltBody); + } + + /** + * Clear queued addresses of given kind. + * + * @param string $kind 'to', 'cc', or 'bcc' + */ + public function clearQueuedAddresses($kind) + { + $this->RecipientsQueue = array_filter( + $this->RecipientsQueue, + static function ($params) use ($kind) { + return $params[0] !== $kind; + } + ); + } + + /** + * Clear all To recipients. + */ + public function clearAddresses() + { + foreach ($this->to as $to) { + unset($this->all_recipients[strtolower($to[0])]); + } + $this->to = []; + $this->clearQueuedAddresses('to'); + } + + /** + * Clear all CC recipients. + */ + public function clearCCs() + { + foreach ($this->cc as $cc) { + unset($this->all_recipients[strtolower($cc[0])]); + } + $this->cc = []; + $this->clearQueuedAddresses('cc'); + } + + /** + * Clear all BCC recipients. + */ + public function clearBCCs() + { + foreach ($this->bcc as $bcc) { + unset($this->all_recipients[strtolower($bcc[0])]); + } + $this->bcc = []; + $this->clearQueuedAddresses('bcc'); + } + + /** + * Clear all ReplyTo recipients. + */ + public function clearReplyTos() + { + $this->ReplyTo = []; + $this->ReplyToQueue = []; + } + + /** + * Clear all recipient types. + */ + public function clearAllRecipients() + { + $this->to = []; + $this->cc = []; + $this->bcc = []; + $this->all_recipients = []; + $this->RecipientsQueue = []; + } + + /** + * Clear all filesystem, string, and binary attachments. + */ + public function clearAttachments() + { + $this->attachment = []; + } + + /** + * Clear all custom headers. + */ + public function clearCustomHeaders() + { + $this->CustomHeader = []; + } + + /** + * Add an error message to the error container. + * + * @param string $msg + */ + protected function setError($msg) + { + ++$this->error_count; + if ('smtp' === $this->Mailer && null !== $this->smtp) { + $lasterror = $this->smtp->getError(); + if (!empty($lasterror['error'])) { + $msg .= $this->lang('smtp_error') . $lasterror['error']; + if (!empty($lasterror['detail'])) { + $msg .= ' Detail: ' . $lasterror['detail']; + } + if (!empty($lasterror['smtp_code'])) { + $msg .= ' SMTP code: ' . $lasterror['smtp_code']; + } + if (!empty($lasterror['smtp_code_ex'])) { + $msg .= ' Additional SMTP info: ' . $lasterror['smtp_code_ex']; + } + } + } + $this->ErrorInfo = $msg; + } + + /** + * Return an RFC 822 formatted date. + * + * @return string + */ + public static function rfcDate() + { + // Set the time zone to whatever the default is to avoid 500 errors + // Will default to UTC if it's not set properly in php.ini + date_default_timezone_set(@date_default_timezone_get()); + + return date('D, j M Y H:i:s O'); + } + + /** + * Get the server hostname. + * Returns 'localhost.localdomain' if unknown. + * + * @return string + */ + protected function serverHostname() + { + $result = ''; + if (!empty($this->Hostname)) { + $result = $this->Hostname; + } elseif (isset($_SERVER) && array_key_exists('SERVER_NAME', $_SERVER)) { + $result = $_SERVER['SERVER_NAME']; + } elseif (function_exists('gethostname') && gethostname() !== false) { + $result = gethostname(); + } elseif (php_uname('n') !== false) { + $result = php_uname('n'); + } + if (!static::isValidHost($result)) { + return 'localhost.localdomain'; + } + + return $result; + } + + /** + * Validate whether a string contains a valid value to use as a hostname or IP address. + * IPv6 addresses must include [], e.g. `[::1]`, not just `::1`. + * + * @param string $host The host name or IP address to check + * + * @return bool + */ + public static function isValidHost($host) + { + //Simple syntax limits + if (empty($host) + || !is_string($host) + || strlen($host) > 256 + || !preg_match('/^([a-zA-Z\d.-]*|\[[a-fA-F\d:]+])$/', $host) + ) { + return false; + } + //Looks like a bracketed IPv6 address + if (strlen($host) > 2 && substr($host, 0, 1) === '[' && substr($host, -1, 1) === ']') { + return filter_var(substr($host, 1, -1), FILTER_VALIDATE_IP, FILTER_FLAG_IPV6) !== false; + } + //If removing all the dots results in a numeric string, it must be an IPv4 address. + //Need to check this first because otherwise things like `999.0.0.0` are considered valid host names + if (is_numeric(str_replace('.', '', $host))) { + //Is it a valid IPv4 address? + return filter_var($host, FILTER_VALIDATE_IP, FILTER_FLAG_IPV4) !== false; + } + if (filter_var('http://' . $host, FILTER_VALIDATE_URL) !== false) { + //Is it a syntactically valid hostname? + return true; + } + + return false; + } + + /** + * Get an error message in the current language. + * + * @param string $key + * + * @return string + */ + protected function lang($key) + { + if (count($this->language) < 1) { + $this->setLanguage(); // set the default language + } + + if (array_key_exists($key, $this->language)) { + if ('smtp_connect_failed' === $key) { + //Include a link to troubleshooting docs on SMTP connection failure + //this is by far the biggest cause of support questions + //but it's usually not PHPMailer's fault. + return $this->language[$key] . ' https://github.com/PHPMailer/PHPMailer/wiki/Troubleshooting'; + } + + return $this->language[$key]; + } + + //Return the key as a fallback + return $key; + } + + /** + * Check if an error occurred. + * + * @return bool True if an error did occur + */ + public function isError() + { + return $this->error_count > 0; + } + + /** + * Add a custom header. + * $name value can be overloaded to contain + * both header name and value (name:value). + * + * @param string $name Custom header name + * @param string|null $value Header value + * + * @throws Exception + */ + public function addCustomHeader($name, $value = null) + { + if (null === $value && strpos($name, ':') !== false) { + // Value passed in as name:value + list($name, $value) = explode(':', $name, 2); + } + $name = trim($name); + $value = trim($value); + //Ensure name is not empty, and that neither name nor value contain line breaks + if (empty($name) || strpbrk($name . $value, "\r\n") !== false) { + if ($this->exceptions) { + throw new Exception('Invalid header name or value'); + } + + return false; + } + $this->CustomHeader[] = [$name, $value]; + + return true; + } + + /** + * Returns all custom headers. + * + * @return array + */ + public function getCustomHeaders() + { + return $this->CustomHeader; + } + + /** + * Create a message body from an HTML string. + * Automatically inlines images and creates a plain-text version by converting the HTML, + * overwriting any existing values in Body and AltBody. + * Do not source $message content from user input! + * $basedir is prepended when handling relative URLs, e.g. and must not be empty + * will look for an image file in $basedir/images/a.png and convert it to inline. + * If you don't provide a $basedir, relative paths will be left untouched (and thus probably break in email) + * Converts data-uri images into embedded attachments. + * If you don't want to apply these transformations to your HTML, just set Body and AltBody directly. + * + * @param string $message HTML message string + * @param string $basedir Absolute path to a base directory to prepend to relative paths to images + * @param bool|callable $advanced Whether to use the internal HTML to text converter + * or your own custom converter @return string $message The transformed message Body + * + * @throws Exception + * + * @see PHPMailer::html2text() + */ + public function msgHTML($message, $basedir = '', $advanced = false) + { + preg_match_all('/(? 1 && '/' !== substr($basedir, -1)) { + // Ensure $basedir has a trailing / + $basedir .= '/'; + } + foreach ($images[2] as $imgindex => $url) { + // Convert data URIs into embedded images + //e.g. "" + $match = []; + if (preg_match('#^data:(image/(?:jpe?g|gif|png));?(base64)?,(.+)#', $url, $match)) { + if (count($match) === 4 && static::ENCODING_BASE64 === $match[2]) { + $data = base64_decode($match[3]); + } elseif ('' === $match[2]) { + $data = rawurldecode($match[3]); + } else { + //Not recognised so leave it alone + continue; + } + //Hash the decoded data, not the URL, so that the same data-URI image used in multiple places + //will only be embedded once, even if it used a different encoding + $cid = substr(hash('sha256', $data), 0, 32) . '@phpmailer.0'; // RFC2392 S 2 + + if (!$this->cidExists($cid)) { + $this->addStringEmbeddedImage( + $data, + $cid, + 'embed' . $imgindex, + static::ENCODING_BASE64, + $match[1] + ); + } + $message = str_replace( + $images[0][$imgindex], + $images[1][$imgindex] . '="cid:' . $cid . '"', + $message + ); + continue; + } + if (// Only process relative URLs if a basedir is provided (i.e. no absolute local paths) + !empty($basedir) + // Ignore URLs containing parent dir traversal (..) + && (strpos($url, '..') === false) + // Do not change urls that are already inline images + && 0 !== strpos($url, 'cid:') + // Do not change absolute URLs, including anonymous protocol + && !preg_match('#^[a-z][a-z0-9+.-]*:?//#i', $url) + ) { + $filename = static::mb_pathinfo($url, PATHINFO_BASENAME); + $directory = dirname($url); + if ('.' === $directory) { + $directory = ''; + } + // RFC2392 S 2 + $cid = substr(hash('sha256', $url), 0, 32) . '@phpmailer.0'; + if (strlen($basedir) > 1 && '/' !== substr($basedir, -1)) { + $basedir .= '/'; + } + if (strlen($directory) > 1 && '/' !== substr($directory, -1)) { + $directory .= '/'; + } + if ($this->addEmbeddedImage( + $basedir . $directory . $filename, + $cid, + $filename, + static::ENCODING_BASE64, + static::_mime_types((string) static::mb_pathinfo($filename, PATHINFO_EXTENSION)) + ) + ) { + $message = preg_replace( + '/' . $images[1][$imgindex] . '=["\']' . preg_quote($url, '/') . '["\']/Ui', + $images[1][$imgindex] . '="cid:' . $cid . '"', + $message + ); + } + } + } + } + $this->isHTML(); + // Convert all message body line breaks to LE, makes quoted-printable encoding work much better + $this->Body = static::normalizeBreaks($message); + $this->AltBody = static::normalizeBreaks($this->html2text($message, $advanced)); + if (!$this->alternativeExists()) { + $this->AltBody = 'This is an HTML-only message. To view it, activate HTML in your email application.' + . static::$LE; + } + + return $this->Body; + } + + /** + * Convert an HTML string into plain text. + * This is used by msgHTML(). + * Note - older versions of this function used a bundled advanced converter + * which was removed for license reasons in #232. + * Example usage: + * + * ```php + * // Use default conversion + * $plain = $mail->html2text($html); + * // Use your own custom converter + * $plain = $mail->html2text($html, function($html) { + * $converter = new MyHtml2text($html); + * return $converter->get_text(); + * }); + * ``` + * + * @param string $html The HTML text to convert + * @param bool|callable $advanced Any boolean value to use the internal converter, + * or provide your own callable for custom conversion + * + * @return string + */ + public function html2text($html, $advanced = false) + { + if (is_callable($advanced)) { + return $advanced($html); + } + + return html_entity_decode( + trim(strip_tags(preg_replace('/<(head|title|style|script)[^>]*>.*?<\/\\1>/si', '', $html))), + ENT_QUOTES, + $this->CharSet + ); + } + + /** + * Get the MIME type for a file extension. + * + * @param string $ext File extension + * + * @return string MIME type of file + */ + public static function _mime_types($ext = '') + { + $mimes = [ + 'xl' => 'application/excel', + 'js' => 'application/javascript', + 'hqx' => 'application/mac-binhex40', + 'cpt' => 'application/mac-compactpro', + 'bin' => 'application/macbinary', + 'doc' => 'application/msword', + 'word' => 'application/msword', + 'xlsx' => 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet', + 'xltx' => 'application/vnd.openxmlformats-officedocument.spreadsheetml.template', + 'potx' => 'application/vnd.openxmlformats-officedocument.presentationml.template', + 'ppsx' => 'application/vnd.openxmlformats-officedocument.presentationml.slideshow', + 'pptx' => 'application/vnd.openxmlformats-officedocument.presentationml.presentation', + 'sldx' => 'application/vnd.openxmlformats-officedocument.presentationml.slide', + 'docx' => 'application/vnd.openxmlformats-officedocument.wordprocessingml.document', + 'dotx' => 'application/vnd.openxmlformats-officedocument.wordprocessingml.template', + 'xlam' => 'application/vnd.ms-excel.addin.macroEnabled.12', + 'xlsb' => 'application/vnd.ms-excel.sheet.binary.macroEnabled.12', + 'class' => 'application/octet-stream', + 'dll' => 'application/octet-stream', + 'dms' => 'application/octet-stream', + 'exe' => 'application/octet-stream', + 'lha' => 'application/octet-stream', + 'lzh' => 'application/octet-stream', + 'psd' => 'application/octet-stream', + 'sea' => 'application/octet-stream', + 'so' => 'application/octet-stream', + 'oda' => 'application/oda', + 'pdf' => 'application/pdf', + 'ai' => 'application/postscript', + 'eps' => 'application/postscript', + 'ps' => 'application/postscript', + 'smi' => 'application/smil', + 'smil' => 'application/smil', + 'mif' => 'application/vnd.mif', + 'xls' => 'application/vnd.ms-excel', + 'ppt' => 'application/vnd.ms-powerpoint', + 'wbxml' => 'application/vnd.wap.wbxml', + 'wmlc' => 'application/vnd.wap.wmlc', + 'dcr' => 'application/x-director', + 'dir' => 'application/x-director', + 'dxr' => 'application/x-director', + 'dvi' => 'application/x-dvi', + 'gtar' => 'application/x-gtar', + 'php3' => 'application/x-httpd-php', + 'php4' => 'application/x-httpd-php', + 'php' => 'application/x-httpd-php', + 'phtml' => 'application/x-httpd-php', + 'phps' => 'application/x-httpd-php-source', + 'swf' => 'application/x-shockwave-flash', + 'sit' => 'application/x-stuffit', + 'tar' => 'application/x-tar', + 'tgz' => 'application/x-tar', + 'xht' => 'application/xhtml+xml', + 'xhtml' => 'application/xhtml+xml', + 'zip' => 'application/zip', + 'mid' => 'audio/midi', + 'midi' => 'audio/midi', + 'mp2' => 'audio/mpeg', + 'mp3' => 'audio/mpeg', + 'm4a' => 'audio/mp4', + 'mpga' => 'audio/mpeg', + 'aif' => 'audio/x-aiff', + 'aifc' => 'audio/x-aiff', + 'aiff' => 'audio/x-aiff', + 'ram' => 'audio/x-pn-realaudio', + 'rm' => 'audio/x-pn-realaudio', + 'rpm' => 'audio/x-pn-realaudio-plugin', + 'ra' => 'audio/x-realaudio', + 'wav' => 'audio/x-wav', + 'mka' => 'audio/x-matroska', + 'bmp' => 'image/bmp', + 'gif' => 'image/gif', + 'jpeg' => 'image/jpeg', + 'jpe' => 'image/jpeg', + 'jpg' => 'image/jpeg', + 'png' => 'image/png', + 'tiff' => 'image/tiff', + 'tif' => 'image/tiff', + 'webp' => 'image/webp', + 'heif' => 'image/heif', + 'heifs' => 'image/heif-sequence', + 'heic' => 'image/heic', + 'heics' => 'image/heic-sequence', + 'eml' => 'message/rfc822', + 'css' => 'text/css', + 'html' => 'text/html', + 'htm' => 'text/html', + 'shtml' => 'text/html', + 'log' => 'text/plain', + 'text' => 'text/plain', + 'txt' => 'text/plain', + 'rtx' => 'text/richtext', + 'rtf' => 'text/rtf', + 'vcf' => 'text/vcard', + 'vcard' => 'text/vcard', + 'ics' => 'text/calendar', + 'xml' => 'text/xml', + 'xsl' => 'text/xml', + 'wmv' => 'video/x-ms-wmv', + 'mpeg' => 'video/mpeg', + 'mpe' => 'video/mpeg', + 'mpg' => 'video/mpeg', + 'mp4' => 'video/mp4', + 'm4v' => 'video/mp4', + 'mov' => 'video/quicktime', + 'qt' => 'video/quicktime', + 'rv' => 'video/vnd.rn-realvideo', + 'avi' => 'video/x-msvideo', + 'movie' => 'video/x-sgi-movie', + 'webm' => 'video/webm', + 'mkv' => 'video/x-matroska', + ]; + $ext = strtolower($ext); + if (array_key_exists($ext, $mimes)) { + return $mimes[$ext]; + } + + return 'application/octet-stream'; + } + + /** + * Map a file name to a MIME type. + * Defaults to 'application/octet-stream', i.e.. arbitrary binary data. + * + * @param string $filename A file name or full path, does not need to exist as a file + * + * @return string + */ + public static function filenameToType($filename) + { + // In case the path is a URL, strip any query string before getting extension + $qpos = strpos($filename, '?'); + if (false !== $qpos) { + $filename = substr($filename, 0, $qpos); + } + $ext = static::mb_pathinfo($filename, PATHINFO_EXTENSION); + + return static::_mime_types($ext); + } + + /** + * Multi-byte-safe pathinfo replacement. + * Drop-in replacement for pathinfo(), but multibyte- and cross-platform-safe. + * + * @see http://www.php.net/manual/en/function.pathinfo.php#107461 + * + * @param string $path A filename or path, does not need to exist as a file + * @param int|string $options Either a PATHINFO_* constant, + * or a string name to return only the specified piece + * + * @return string|array + */ + public static function mb_pathinfo($path, $options = null) + { + $ret = ['dirname' => '', 'basename' => '', 'extension' => '', 'filename' => '']; + $pathinfo = []; + if (preg_match('#^(.*?)[\\\\/]*(([^/\\\\]*?)(\.([^.\\\\/]+?)|))[\\\\/.]*$#m', $path, $pathinfo)) { + if (array_key_exists(1, $pathinfo)) { + $ret['dirname'] = $pathinfo[1]; + } + if (array_key_exists(2, $pathinfo)) { + $ret['basename'] = $pathinfo[2]; + } + if (array_key_exists(5, $pathinfo)) { + $ret['extension'] = $pathinfo[5]; + } + if (array_key_exists(3, $pathinfo)) { + $ret['filename'] = $pathinfo[3]; + } + } + switch ($options) { + case PATHINFO_DIRNAME: + case 'dirname': + return $ret['dirname']; + case PATHINFO_BASENAME: + case 'basename': + return $ret['basename']; + case PATHINFO_EXTENSION: + case 'extension': + return $ret['extension']; + case PATHINFO_FILENAME: + case 'filename': + return $ret['filename']; + default: + return $ret; + } + } + + /** + * Set or reset instance properties. + * You should avoid this function - it's more verbose, less efficient, more error-prone and + * harder to debug than setting properties directly. + * Usage Example: + * `$mail->set('SMTPSecure', static::ENCRYPTION_STARTTLS);` + * is the same as: + * `$mail->SMTPSecure = static::ENCRYPTION_STARTTLS;`. + * + * @param string $name The property name to set + * @param mixed $value The value to set the property to + * + * @return bool + */ + public function set($name, $value = '') + { + if (property_exists($this, $name)) { + $this->$name = $value; + + return true; + } + $this->setError($this->lang('variable_set') . $name); + + return false; + } + + /** + * Strip newlines to prevent header injection. + * + * @param string $str + * + * @return string + */ + public function secureHeader($str) + { + return trim(str_replace(["\r", "\n"], '', $str)); + } + + /** + * Normalize line breaks in a string. + * Converts UNIX LF, Mac CR and Windows CRLF line breaks into a single line break format. + * Defaults to CRLF (for message bodies) and preserves consecutive breaks. + * + * @param string $text + * @param string $breaktype What kind of line break to use; defaults to static::$LE + * + * @return string + */ + public static function normalizeBreaks($text, $breaktype = null) + { + if (null === $breaktype) { + $breaktype = static::$LE; + } + // Normalise to \n + $text = str_replace([self::CRLF, "\r"], "\n", $text); + // Now convert LE as needed + if ("\n" !== $breaktype) { + $text = str_replace("\n", $breaktype, $text); + } + + return $text; + } + + /** + * Remove trailing breaks from a string. + * + * @param string $text + * + * @return string The text to remove breaks from + */ + public static function stripTrailingWSP($text) + { + return rtrim($text, " \r\n\t"); + } + + /** + * Return the current line break format string. + * + * @return string + */ + public static function getLE() + { + return static::$LE; + } + + /** + * Set the line break format string, e.g. "\r\n". + * + * @param string $le + */ + protected static function setLE($le) + { + static::$LE = $le; + } + + /** + * Set the public and private key files and password for S/MIME signing. + * + * @param string $cert_filename + * @param string $key_filename + * @param string $key_pass Password for private key + * @param string $extracerts_filename Optional path to chain certificate + */ + public function sign($cert_filename, $key_filename, $key_pass, $extracerts_filename = '') + { + $this->sign_cert_file = $cert_filename; + $this->sign_key_file = $key_filename; + $this->sign_key_pass = $key_pass; + $this->sign_extracerts_file = $extracerts_filename; + } + + /** + * Quoted-Printable-encode a DKIM header. + * + * @param string $txt + * + * @return string + */ + public function DKIM_QP($txt) + { + $line = ''; + $len = strlen($txt); + for ($i = 0; $i < $len; ++$i) { + $ord = ord($txt[$i]); + if (((0x21 <= $ord) && ($ord <= 0x3A)) || $ord === 0x3C || ((0x3E <= $ord) && ($ord <= 0x7E))) { + $line .= $txt[$i]; + } else { + $line .= '=' . sprintf('%02X', $ord); + } + } + + return $line; + } + + /** + * Generate a DKIM signature. + * + * @param string $signHeader + * + * @throws Exception + * + * @return string The DKIM signature value + */ + public function DKIM_Sign($signHeader) + { + if (!defined('PKCS7_TEXT')) { + if ($this->exceptions) { + throw new Exception($this->lang('extension_missing') . 'openssl'); + } + + return ''; + } + $privKeyStr = !empty($this->DKIM_private_string) ? + $this->DKIM_private_string : + file_get_contents($this->DKIM_private); + if ('' !== $this->DKIM_passphrase) { + $privKey = openssl_pkey_get_private($privKeyStr, $this->DKIM_passphrase); + } else { + $privKey = openssl_pkey_get_private($privKeyStr); + } + if (openssl_sign($signHeader, $signature, $privKey, 'sha256WithRSAEncryption')) { + openssl_pkey_free($privKey); + + return base64_encode($signature); + } + openssl_pkey_free($privKey); + + return ''; + } + + /** + * Generate a DKIM canonicalization header. + * Uses the 'relaxed' algorithm from RFC6376 section 3.4.2. + * Canonicalized headers should *always* use CRLF, regardless of mailer setting. + * + * @see https://tools.ietf.org/html/rfc6376#section-3.4.2 + * + * @param string $signHeader Header + * + * @return string + */ + public function DKIM_HeaderC($signHeader) + { + //Normalize breaks to CRLF (regardless of the mailer) + $signHeader = static::normalizeBreaks($signHeader, self::CRLF); + //Unfold header lines + //Note PCRE \s is too broad a definition of whitespace; RFC5322 defines it as `[ \t]` + //@see https://tools.ietf.org/html/rfc5322#section-2.2 + //That means this may break if you do something daft like put vertical tabs in your headers. + $signHeader = preg_replace('/\r\n[ \t]+/', ' ', $signHeader); + //Break headers out into an array + $lines = explode(self::CRLF, $signHeader); + foreach ($lines as $key => $line) { + //If the header is missing a :, skip it as it's invalid + //This is likely to happen because the explode() above will also split + //on the trailing LE, leaving an empty line + if (strpos($line, ':') === false) { + continue; + } + list($heading, $value) = explode(':', $line, 2); + //Lower-case header name + $heading = strtolower($heading); + //Collapse white space within the value, also convert WSP to space + $value = preg_replace('/[ \t]+/', ' ', $value); + //RFC6376 is slightly unclear here - it says to delete space at the *end* of each value + //But then says to delete space before and after the colon. + //Net result is the same as trimming both ends of the value. + //By elimination, the same applies to the field name + $lines[$key] = trim($heading, " \t") . ':' . trim($value, " \t"); + } + + return implode(self::CRLF, $lines); + } + + /** + * Generate a DKIM canonicalization body. + * Uses the 'simple' algorithm from RFC6376 section 3.4.3. + * Canonicalized bodies should *always* use CRLF, regardless of mailer setting. + * + * @see https://tools.ietf.org/html/rfc6376#section-3.4.3 + * + * @param string $body Message Body + * + * @return string + */ + public function DKIM_BodyC($body) + { + if (empty($body)) { + return self::CRLF; + } + // Normalize line endings to CRLF + $body = static::normalizeBreaks($body, self::CRLF); + + //Reduce multiple trailing line breaks to a single one + return static::stripTrailingWSP($body) . self::CRLF; + } + + /** + * Create the DKIM header and body in a new message header. + * + * @param string $headers_line Header lines + * @param string $subject Subject + * @param string $body Body + * + * @throws Exception + * + * @return string + */ + public function DKIM_Add($headers_line, $subject, $body) + { + $DKIMsignatureType = 'rsa-sha256'; // Signature & hash algorithms + $DKIMcanonicalization = 'relaxed/simple'; // Canonicalization methods of header & body + $DKIMquery = 'dns/txt'; // Query method + $DKIMtime = time(); + //Always sign these headers without being asked + //Recommended list from https://tools.ietf.org/html/rfc6376#section-5.4.1 + $autoSignHeaders = [ + 'from', + 'to', + 'cc', + 'date', + 'subject', + 'reply-to', + 'message-id', + 'content-type', + 'mime-version', + 'x-mailer', + ]; + if (stripos($headers_line, 'Subject') === false) { + $headers_line .= 'Subject: ' . $subject . static::$LE; + } + $headerLines = explode(static::$LE, $headers_line); + $currentHeaderLabel = ''; + $currentHeaderValue = ''; + $parsedHeaders = []; + $headerLineIndex = 0; + $headerLineCount = count($headerLines); + foreach ($headerLines as $headerLine) { + $matches = []; + if (preg_match('/^([^ \t]*?)(?::[ \t]*)(.*)$/', $headerLine, $matches)) { + if ($currentHeaderLabel !== '') { + //We were previously in another header; This is the start of a new header, so save the previous one + $parsedHeaders[] = ['label' => $currentHeaderLabel, 'value' => $currentHeaderValue]; + } + $currentHeaderLabel = $matches[1]; + $currentHeaderValue = $matches[2]; + } elseif (preg_match('/^[ \t]+(.*)$/', $headerLine, $matches)) { + //This is a folded continuation of the current header, so unfold it + $currentHeaderValue .= ' ' . $matches[1]; + } + ++$headerLineIndex; + if ($headerLineIndex >= $headerLineCount) { + //This was the last line, so finish off this header + $parsedHeaders[] = ['label' => $currentHeaderLabel, 'value' => $currentHeaderValue]; + } + } + $copiedHeaders = []; + $headersToSignKeys = []; + $headersToSign = []; + foreach ($parsedHeaders as $header) { + //Is this header one that must be included in the DKIM signature? + if (in_array(strtolower($header['label']), $autoSignHeaders, true)) { + $headersToSignKeys[] = $header['label']; + $headersToSign[] = $header['label'] . ': ' . $header['value']; + if ($this->DKIM_copyHeaderFields) { + $copiedHeaders[] = $header['label'] . ':' . //Note no space after this, as per RFC + str_replace('|', '=7C', $this->DKIM_QP($header['value'])); + } + continue; + } + //Is this an extra custom header we've been asked to sign? + if (in_array($header['label'], $this->DKIM_extraHeaders, true)) { + //Find its value in custom headers + foreach ($this->CustomHeader as $customHeader) { + if ($customHeader[0] === $header['label']) { + $headersToSignKeys[] = $header['label']; + $headersToSign[] = $header['label'] . ': ' . $header['value']; + if ($this->DKIM_copyHeaderFields) { + $copiedHeaders[] = $header['label'] . ':' . //Note no space after this, as per RFC + str_replace('|', '=7C', $this->DKIM_QP($header['value'])); + } + //Skip straight to the next header + continue 2; + } + } + } + } + $copiedHeaderFields = ''; + if ($this->DKIM_copyHeaderFields && count($copiedHeaders) > 0) { + //Assemble a DKIM 'z' tag + $copiedHeaderFields = ' z='; + $first = true; + foreach ($copiedHeaders as $copiedHeader) { + if (!$first) { + $copiedHeaderFields .= static::$LE . ' |'; + } + //Fold long values + if (strlen($copiedHeader) > self::STD_LINE_LENGTH - 3) { + $copiedHeaderFields .= substr( + chunk_split($copiedHeader, self::STD_LINE_LENGTH - 3, static::$LE . self::FWS), + 0, + -strlen(static::$LE . self::FWS) + ); + } else { + $copiedHeaderFields .= $copiedHeader; + } + $first = false; + } + $copiedHeaderFields .= ';' . static::$LE; + } + $headerKeys = ' h=' . implode(':', $headersToSignKeys) . ';' . static::$LE; + $headerValues = implode(static::$LE, $headersToSign); + $body = $this->DKIM_BodyC($body); + $DKIMb64 = base64_encode(pack('H*', hash('sha256', $body))); // Base64 of packed binary SHA-256 hash of body + $ident = ''; + if ('' !== $this->DKIM_identity) { + $ident = ' i=' . $this->DKIM_identity . ';' . static::$LE; + } + //The DKIM-Signature header is included in the signature *except for* the value of the `b` tag + //which is appended after calculating the signature + //https://tools.ietf.org/html/rfc6376#section-3.5 + $dkimSignatureHeader = 'DKIM-Signature: v=1;' . + ' d=' . $this->DKIM_domain . ';' . + ' s=' . $this->DKIM_selector . ';' . static::$LE . + ' a=' . $DKIMsignatureType . ';' . + ' q=' . $DKIMquery . ';' . + ' t=' . $DKIMtime . ';' . + ' c=' . $DKIMcanonicalization . ';' . static::$LE . + $headerKeys . + $ident . + $copiedHeaderFields . + ' bh=' . $DKIMb64 . ';' . static::$LE . + ' b='; + //Canonicalize the set of headers + $canonicalizedHeaders = $this->DKIM_HeaderC( + $headerValues . static::$LE . $dkimSignatureHeader + ); + $signature = $this->DKIM_Sign($canonicalizedHeaders); + $signature = trim(chunk_split($signature, self::STD_LINE_LENGTH - 3, static::$LE . self::FWS)); + + return static::normalizeBreaks($dkimSignatureHeader . $signature); + } + + /** + * Detect if a string contains a line longer than the maximum line length + * allowed by RFC 2822 section 2.1.1. + * + * @param string $str + * + * @return bool + */ + public static function hasLineLongerThanMax($str) + { + return (bool) preg_match('/^(.{' . (self::MAX_LINE_LENGTH + strlen(static::$LE)) . ',})/m', $str); + } + + /** + * Allows for public read access to 'to' property. + * Before the send() call, queued addresses (i.e. with IDN) are not yet included. + * + * @return array + */ + public function getToAddresses() + { + return $this->to; + } + + /** + * Allows for public read access to 'cc' property. + * Before the send() call, queued addresses (i.e. with IDN) are not yet included. + * + * @return array + */ + public function getCcAddresses() + { + return $this->cc; + } + + /** + * Allows for public read access to 'bcc' property. + * Before the send() call, queued addresses (i.e. with IDN) are not yet included. + * + * @return array + */ + public function getBccAddresses() + { + return $this->bcc; + } + + /** + * Allows for public read access to 'ReplyTo' property. + * Before the send() call, queued addresses (i.e. with IDN) are not yet included. + * + * @return array + */ + public function getReplyToAddresses() + { + return $this->ReplyTo; + } + + /** + * Allows for public read access to 'all_recipients' property. + * Before the send() call, queued addresses (i.e. with IDN) are not yet included. + * + * @return array + */ + public function getAllRecipientAddresses() + { + return $this->all_recipients; + } + + /** + * Perform a callback. + * + * @param bool $isSent + * @param array $to + * @param array $cc + * @param array $bcc + * @param string $subject + * @param string $body + * @param string $from + * @param array $extra + */ + protected function doCallback($isSent, $to, $cc, $bcc, $subject, $body, $from, $extra) + { + if (!empty($this->action_function) && is_callable($this->action_function)) { + call_user_func($this->action_function, $isSent, $to, $cc, $bcc, $subject, $body, $from, $extra); + } + } + + /** + * Get the OAuth instance. + * + * @return OAuth + */ + public function getOAuth() + { + return $this->oauth; + } + + /** + * Set an OAuth instance. + */ + public function setOAuth(OAuth $oauth) + { + $this->oauth = $oauth; + } +} diff --git a/inc/PHPMailer/POP3.php b/inc/PHPMailer/POP3.php new file mode 100644 index 0000000..cd6fc2f --- /dev/null +++ b/inc/PHPMailer/POP3.php @@ -0,0 +1,421 @@ + + * @author Jim Jagielski (jimjag) + * @author Andy Prevost (codeworxtech) + * @author Brent R. Matzelle (original founder) + * @copyright 2012 - 2019 Marcus Bointon + * @copyright 2010 - 2012 Jim Jagielski + * @copyright 2004 - 2009 Andy Prevost + * @license http://www.gnu.org/copyleft/lesser.html GNU Lesser General Public License + * @note This program is distributed in the hope that it will be useful - WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. + */ + +namespace PHPMailer\PHPMailer; + +/** + * PHPMailer POP-Before-SMTP Authentication Class. + * Specifically for PHPMailer to use for RFC1939 POP-before-SMTP authentication. + * 1) This class does not support APOP authentication. + * 2) Opening and closing lots of POP3 connections can be quite slow. If you need + * to send a batch of emails then just perform the authentication once at the start, + * and then loop through your mail sending script. Providing this process doesn't + * take longer than the verification period lasts on your POP3 server, you should be fine. + * 3) This is really ancient technology; you should only need to use it to talk to very old systems. + * 4) This POP3 class is deliberately lightweight and incomplete, implementing just + * enough to do authentication. + * If you want a more complete class there are other POP3 classes for PHP available. + * + * @author Richard Davey (original author) + * @author Marcus Bointon (Synchro/coolbru) + * @author Jim Jagielski (jimjag) + * @author Andy Prevost (codeworxtech) + */ +class POP3 +{ + /** + * The POP3 PHPMailer Version number. + * + * @var string + */ + const VERSION = '6.1.5'; + + /** + * Default POP3 port number. + * + * @var int + */ + const DEFAULT_PORT = 110; + + /** + * Default timeout in seconds. + * + * @var int + */ + const DEFAULT_TIMEOUT = 30; + + /** + * Debug display level. + * Options: 0 = no, 1+ = yes. + * + * @var int + */ + public $do_debug = 0; + + /** + * POP3 mail server hostname. + * + * @var string + */ + public $host; + + /** + * POP3 port number. + * + * @var int + */ + public $port; + + /** + * POP3 Timeout Value in seconds. + * + * @var int + */ + public $tval; + + /** + * POP3 username. + * + * @var string + */ + public $username; + + /** + * POP3 password. + * + * @var string + */ + public $password; + + /** + * Resource handle for the POP3 connection socket. + * + * @var resource + */ + protected $pop_conn; + + /** + * Are we connected? + * + * @var bool + */ + protected $connected = false; + + /** + * Error container. + * + * @var array + */ + protected $errors = []; + + /** + * Line break constant. + */ + const LE = "\r\n"; + + /** + * Simple static wrapper for all-in-one POP before SMTP. + * + * @param string $host The hostname to connect to + * @param int|bool $port The port number to connect to + * @param int|bool $timeout The timeout value + * @param string $username + * @param string $password + * @param int $debug_level + * + * @return bool + */ + public static function popBeforeSmtp( + $host, + $port = false, + $timeout = false, + $username = '', + $password = '', + $debug_level = 0 + ) { + $pop = new self(); + + return $pop->authorise($host, $port, $timeout, $username, $password, $debug_level); + } + + /** + * Authenticate with a POP3 server. + * A connect, login, disconnect sequence + * appropriate for POP-before SMTP authorisation. + * + * @param string $host The hostname to connect to + * @param int|bool $port The port number to connect to + * @param int|bool $timeout The timeout value + * @param string $username + * @param string $password + * @param int $debug_level + * + * @return bool + */ + public function authorise($host, $port = false, $timeout = false, $username = '', $password = '', $debug_level = 0) + { + $this->host = $host; + // If no port value provided, use default + if (false === $port) { + $this->port = static::DEFAULT_PORT; + } else { + $this->port = (int) $port; + } + // If no timeout value provided, use default + if (false === $timeout) { + $this->tval = static::DEFAULT_TIMEOUT; + } else { + $this->tval = (int) $timeout; + } + $this->do_debug = $debug_level; + $this->username = $username; + $this->password = $password; + // Reset the error log + $this->errors = []; + // connect + $result = $this->connect($this->host, $this->port, $this->tval); + if ($result) { + $login_result = $this->login($this->username, $this->password); + if ($login_result) { + $this->disconnect(); + + return true; + } + } + // We need to disconnect regardless of whether the login succeeded + $this->disconnect(); + + return false; + } + + /** + * Connect to a POP3 server. + * + * @param string $host + * @param int|bool $port + * @param int $tval + * + * @return bool + */ + public function connect($host, $port = false, $tval = 30) + { + // Are we already connected? + if ($this->connected) { + return true; + } + + //On Windows this will raise a PHP Warning error if the hostname doesn't exist. + //Rather than suppress it with @fsockopen, capture it cleanly instead + set_error_handler([$this, 'catchWarning']); + + if (false === $port) { + $port = static::DEFAULT_PORT; + } + + // connect to the POP3 server + $errno = 0; + $errstr = ''; + $this->pop_conn = fsockopen( + $host, // POP3 Host + $port, // Port # + $errno, // Error Number + $errstr, // Error Message + $tval + ); // Timeout (seconds) + // Restore the error handler + restore_error_handler(); + + // Did we connect? + if (false === $this->pop_conn) { + // It would appear not... + $this->setError( + "Failed to connect to server $host on port $port. errno: $errno; errstr: $errstr" + ); + + return false; + } + + // Increase the stream time-out + stream_set_timeout($this->pop_conn, $tval, 0); + + // Get the POP3 server response + $pop3_response = $this->getResponse(); + // Check for the +OK + if ($this->checkResponse($pop3_response)) { + // The connection is established and the POP3 server is talking + $this->connected = true; + + return true; + } + + return false; + } + + /** + * Log in to the POP3 server. + * Does not support APOP (RFC 2828, 4949). + * + * @param string $username + * @param string $password + * + * @return bool + */ + public function login($username = '', $password = '') + { + if (!$this->connected) { + $this->setError('Not connected to POP3 server'); + } + if (empty($username)) { + $username = $this->username; + } + if (empty($password)) { + $password = $this->password; + } + + // Send the Username + $this->sendString("USER $username" . static::LE); + $pop3_response = $this->getResponse(); + if ($this->checkResponse($pop3_response)) { + // Send the Password + $this->sendString("PASS $password" . static::LE); + $pop3_response = $this->getResponse(); + if ($this->checkResponse($pop3_response)) { + return true; + } + } + + return false; + } + + /** + * Disconnect from the POP3 server. + */ + public function disconnect() + { + $this->sendString('QUIT'); + //The QUIT command may cause the daemon to exit, which will kill our connection + //So ignore errors here + try { + @fclose($this->pop_conn); + } catch (Exception $e) { + //Do nothing + } + } + + /** + * Get a response from the POP3 server. + * + * @param int $size The maximum number of bytes to retrieve + * + * @return string + */ + protected function getResponse($size = 128) + { + $response = fgets($this->pop_conn, $size); + if ($this->do_debug >= 1) { + echo 'Server -> Client: ', $response; + } + + return $response; + } + + /** + * Send raw data to the POP3 server. + * + * @param string $string + * + * @return int + */ + protected function sendString($string) + { + if ($this->pop_conn) { + if ($this->do_debug >= 2) { //Show client messages when debug >= 2 + echo 'Client -> Server: ', $string; + } + + return fwrite($this->pop_conn, $string, strlen($string)); + } + + return 0; + } + + /** + * Checks the POP3 server response. + * Looks for for +OK or -ERR. + * + * @param string $string + * + * @return bool + */ + protected function checkResponse($string) + { + if (strpos($string, '+OK') !== 0) { + $this->setError("Server reported an error: $string"); + + return false; + } + + return true; + } + + /** + * Add an error to the internal error store. + * Also display debug output if it's enabled. + * + * @param string $error + */ + protected function setError($error) + { + $this->errors[] = $error; + if ($this->do_debug >= 1) { + echo '
';
+            foreach ($this->errors as $e) {
+                print_r($e);
+            }
+            echo '
'; + } + } + + /** + * Get an array of error messages, if any. + * + * @return array + */ + public function getErrors() + { + return $this->errors; + } + + /** + * POP3 connection error handler. + * + * @param int $errno + * @param string $errstr + * @param string $errfile + * @param int $errline + */ + protected function catchWarning($errno, $errstr, $errfile, $errline) + { + $this->setError( + 'Connecting to the POP3 server raised a PHP warning:' . + "errno: $errno errstr: $errstr; errfile: $errfile; errline: $errline" + ); + } +} diff --git a/inc/PHPMailer/SMTP.php b/inc/PHPMailer/SMTP.php new file mode 100644 index 0000000..1e38ba7 --- /dev/null +++ b/inc/PHPMailer/SMTP.php @@ -0,0 +1,1371 @@ + + * @author Jim Jagielski (jimjag) + * @author Andy Prevost (codeworxtech) + * @author Brent R. Matzelle (original founder) + * @copyright 2012 - 2019 Marcus Bointon + * @copyright 2010 - 2012 Jim Jagielski + * @copyright 2004 - 2009 Andy Prevost + * @license http://www.gnu.org/copyleft/lesser.html GNU Lesser General Public License + * @note This program is distributed in the hope that it will be useful - WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. + */ + +namespace PHPMailer\PHPMailer; + +/** + * PHPMailer RFC821 SMTP email transport class. + * Implements RFC 821 SMTP commands and provides some utility methods for sending mail to an SMTP server. + * + * @author Chris Ryan + * @author Marcus Bointon + */ +class SMTP +{ + /** + * The PHPMailer SMTP version number. + * + * @var string + */ + const VERSION = '6.1.5'; + + /** + * SMTP line break constant. + * + * @var string + */ + const LE = "\r\n"; + + /** + * The SMTP port to use if one is not specified. + * + * @var int + */ + const DEFAULT_PORT = 25; + + /** + * The maximum line length allowed by RFC 5321 section 4.5.3.1.6, + * *excluding* a trailing CRLF break. + * + * @see https://tools.ietf.org/html/rfc5321#section-4.5.3.1.6 + * + * @var int + */ + const MAX_LINE_LENGTH = 998; + + /** + * The maximum line length allowed for replies in RFC 5321 section 4.5.3.1.5, + * *including* a trailing CRLF line break. + * + * @see https://tools.ietf.org/html/rfc5321#section-4.5.3.1.5 + * + * @var int + */ + const MAX_REPLY_LENGTH = 512; + + /** + * Debug level for no output. + * + * @var int + */ + const DEBUG_OFF = 0; + + /** + * Debug level to show client -> server messages. + * + * @var int + */ + const DEBUG_CLIENT = 1; + + /** + * Debug level to show client -> server and server -> client messages. + * + * @var int + */ + const DEBUG_SERVER = 2; + + /** + * Debug level to show connection status, client -> server and server -> client messages. + * + * @var int + */ + const DEBUG_CONNECTION = 3; + + /** + * Debug level to show all messages. + * + * @var int + */ + const DEBUG_LOWLEVEL = 4; + + /** + * Debug output level. + * Options: + * * self::DEBUG_OFF (`0`) No debug output, default + * * self::DEBUG_CLIENT (`1`) Client commands + * * self::DEBUG_SERVER (`2`) Client commands and server responses + * * self::DEBUG_CONNECTION (`3`) As DEBUG_SERVER plus connection status + * * self::DEBUG_LOWLEVEL (`4`) Low-level data output, all messages. + * + * @var int + */ + public $do_debug = self::DEBUG_OFF; + + /** + * How to handle debug output. + * Options: + * * `echo` Output plain-text as-is, appropriate for CLI + * * `html` Output escaped, line breaks converted to `
`, appropriate for browser output + * * `error_log` Output to error log as configured in php.ini + * Alternatively, you can provide a callable expecting two params: a message string and the debug level: + * + * ```php + * $smtp->Debugoutput = function($str, $level) {echo "debug level $level; message: $str";}; + * ``` + * + * Alternatively, you can pass in an instance of a PSR-3 compatible logger, though only `debug` + * level output is used: + * + * ```php + * $mail->Debugoutput = new myPsr3Logger; + * ``` + * + * @var string|callable|\Psr\Log\LoggerInterface + */ + public $Debugoutput = 'echo'; + + /** + * Whether to use VERP. + * + * @see http://en.wikipedia.org/wiki/Variable_envelope_return_path + * @see http://www.postfix.org/VERP_README.html Info on VERP + * + * @var bool + */ + public $do_verp = false; + + /** + * The timeout value for connection, in seconds. + * Default of 5 minutes (300sec) is from RFC2821 section 4.5.3.2. + * This needs to be quite high to function correctly with hosts using greetdelay as an anti-spam measure. + * + * @see http://tools.ietf.org/html/rfc2821#section-4.5.3.2 + * + * @var int + */ + public $Timeout = 300; + + /** + * How long to wait for commands to complete, in seconds. + * Default of 5 minutes (300sec) is from RFC2821 section 4.5.3.2. + * + * @var int + */ + public $Timelimit = 300; + + /** + * Patterns to extract an SMTP transaction id from reply to a DATA command. + * The first capture group in each regex will be used as the ID. + * MS ESMTP returns the message ID, which may not be correct for internal tracking. + * + * @var string[] + */ + protected $smtp_transaction_id_patterns = [ + 'exim' => '/[\d]{3} OK id=(.*)/', + 'sendmail' => '/[\d]{3} 2.0.0 (.*) Message/', + 'postfix' => '/[\d]{3} 2.0.0 Ok: queued as (.*)/', + 'Microsoft_ESMTP' => '/[0-9]{3} 2.[\d].0 (.*)@(?:.*) Queued mail for delivery/', + 'Amazon_SES' => '/[\d]{3} Ok (.*)/', + 'SendGrid' => '/[\d]{3} Ok: queued as (.*)/', + 'CampaignMonitor' => '/[\d]{3} 2.0.0 OK:([a-zA-Z\d]{48})/', + ]; + + /** + * The last transaction ID issued in response to a DATA command, + * if one was detected. + * + * @var string|bool|null + */ + protected $last_smtp_transaction_id; + + /** + * The socket for the server connection. + * + * @var ?resource + */ + protected $smtp_conn; + + /** + * Error information, if any, for the last SMTP command. + * + * @var array + */ + protected $error = [ + 'error' => '', + 'detail' => '', + 'smtp_code' => '', + 'smtp_code_ex' => '', + ]; + + /** + * The reply the server sent to us for HELO. + * If null, no HELO string has yet been received. + * + * @var string|null + */ + protected $helo_rply; + + /** + * The set of SMTP extensions sent in reply to EHLO command. + * Indexes of the array are extension names. + * Value at index 'HELO' or 'EHLO' (according to command that was sent) + * represents the server name. In case of HELO it is the only element of the array. + * Other values can be boolean TRUE or an array containing extension options. + * If null, no HELO/EHLO string has yet been received. + * + * @var array|null + */ + protected $server_caps; + + /** + * The most recent reply received from the server. + * + * @var string + */ + protected $last_reply = ''; + + /** + * Output debugging info via a user-selected method. + * + * @param string $str Debug string to output + * @param int $level The debug level of this message; see DEBUG_* constants + * + * @see SMTP::$Debugoutput + * @see SMTP::$do_debug + */ + protected function edebug($str, $level = 0) + { + if ($level > $this->do_debug) { + return; + } + //Is this a PSR-3 logger? + if ($this->Debugoutput instanceof \Psr\Log\LoggerInterface) { + $this->Debugoutput->debug($str); + + return; + } + //Avoid clash with built-in function names + if (is_callable($this->Debugoutput) && !in_array($this->Debugoutput, ['error_log', 'html', 'echo'])) { + call_user_func($this->Debugoutput, $str, $level); + + return; + } + switch ($this->Debugoutput) { + case 'error_log': + //Don't output, just log + error_log($str); + break; + case 'html': + //Cleans up output a bit for a better looking, HTML-safe output + echo gmdate('Y-m-d H:i:s'), ' ', htmlentities( + preg_replace('/[\r\n]+/', '', $str), + ENT_QUOTES, + 'UTF-8' + ), "
\n"; + break; + case 'echo': + default: + //Normalize line breaks + $str = preg_replace('/\r\n|\r/m', "\n", $str); + echo gmdate('Y-m-d H:i:s'), + "\t", + //Trim trailing space + trim( + //Indent for readability, except for trailing break + str_replace( + "\n", + "\n \t ", + trim($str) + ) + ), + "\n"; + } + } + + /** + * Connect to an SMTP server. + * + * @param string $host SMTP server IP or host name + * @param int $port The port number to connect to + * @param int $timeout How long to wait for the connection to open + * @param array $options An array of options for stream_context_create() + * + * @return bool + */ + public function connect($host, $port = null, $timeout = 30, $options = []) + { + static $streamok; + //This is enabled by default since 5.0.0 but some providers disable it + //Check this once and cache the result + if (null === $streamok) { + $streamok = function_exists('stream_socket_client'); + } + // Clear errors to avoid confusion + $this->setError(''); + // Make sure we are __not__ connected + if ($this->connected()) { + // Already connected, generate error + $this->setError('Already connected to a server'); + + return false; + } + if (empty($port)) { + $port = self::DEFAULT_PORT; + } + // Connect to the SMTP server + $this->edebug( + "Connection: opening to $host:$port, timeout=$timeout, options=" . + (count($options) > 0 ? var_export($options, true) : 'array()'), + self::DEBUG_CONNECTION + ); + $errno = 0; + $errstr = ''; + if ($streamok) { + $socket_context = stream_context_create($options); + set_error_handler([$this, 'errorHandler']); + $this->smtp_conn = stream_socket_client( + $host . ':' . $port, + $errno, + $errstr, + $timeout, + STREAM_CLIENT_CONNECT, + $socket_context + ); + restore_error_handler(); + } else { + //Fall back to fsockopen which should work in more places, but is missing some features + $this->edebug( + 'Connection: stream_socket_client not available, falling back to fsockopen', + self::DEBUG_CONNECTION + ); + set_error_handler([$this, 'errorHandler']); + $this->smtp_conn = fsockopen( + $host, + $port, + $errno, + $errstr, + $timeout + ); + restore_error_handler(); + } + // Verify we connected properly + if (!is_resource($this->smtp_conn)) { + $this->setError( + 'Failed to connect to server', + '', + (string) $errno, + $errstr + ); + $this->edebug( + 'SMTP ERROR: ' . $this->error['error'] + . ": $errstr ($errno)", + self::DEBUG_CLIENT + ); + + return false; + } + $this->edebug('Connection: opened', self::DEBUG_CONNECTION); + // SMTP server can take longer to respond, give longer timeout for first read + // Windows does not have support for this timeout function + if (strpos(PHP_OS, 'WIN') !== 0) { + $max = (int) ini_get('max_execution_time'); + // Don't bother if unlimited + if (0 !== $max && $timeout > $max) { + @set_time_limit($timeout); + } + stream_set_timeout($this->smtp_conn, $timeout, 0); + } + // Get any announcement + $announce = $this->get_lines(); + $this->edebug('SERVER -> CLIENT: ' . $announce, self::DEBUG_SERVER); + + return true; + } + + /** + * Initiate a TLS (encrypted) session. + * + * @return bool + */ + public function startTLS() + { + if (!$this->sendCommand('STARTTLS', 'STARTTLS', 220)) { + return false; + } + + //Allow the best TLS version(s) we can + $crypto_method = STREAM_CRYPTO_METHOD_TLS_CLIENT; + + //PHP 5.6.7 dropped inclusion of TLS 1.1 and 1.2 in STREAM_CRYPTO_METHOD_TLS_CLIENT + //so add them back in manually if we can + if (defined('STREAM_CRYPTO_METHOD_TLSv1_2_CLIENT')) { + $crypto_method |= STREAM_CRYPTO_METHOD_TLSv1_2_CLIENT; + $crypto_method |= STREAM_CRYPTO_METHOD_TLSv1_1_CLIENT; + } + + // Begin encrypted connection + set_error_handler([$this, 'errorHandler']); + $crypto_ok = stream_socket_enable_crypto( + $this->smtp_conn, + true, + $crypto_method + ); + restore_error_handler(); + + return (bool) $crypto_ok; + } + + /** + * Perform SMTP authentication. + * Must be run after hello(). + * + * @see hello() + * + * @param string $username The user name + * @param string $password The password + * @param string $authtype The auth type (CRAM-MD5, PLAIN, LOGIN, XOAUTH2) + * @param OAuth $OAuth An optional OAuth instance for XOAUTH2 authentication + * + * @return bool True if successfully authenticated + */ + public function authenticate( + $username, + $password, + $authtype = null, + $OAuth = null + ) { + if (!$this->server_caps) { + $this->setError('Authentication is not allowed before HELO/EHLO'); + + return false; + } + + if (array_key_exists('EHLO', $this->server_caps)) { + // SMTP extensions are available; try to find a proper authentication method + if (!array_key_exists('AUTH', $this->server_caps)) { + $this->setError('Authentication is not allowed at this stage'); + // 'at this stage' means that auth may be allowed after the stage changes + // e.g. after STARTTLS + + return false; + } + + $this->edebug('Auth method requested: ' . ($authtype ?: 'UNSPECIFIED'), self::DEBUG_LOWLEVEL); + $this->edebug( + 'Auth methods available on the server: ' . implode(',', $this->server_caps['AUTH']), + self::DEBUG_LOWLEVEL + ); + + //If we have requested a specific auth type, check the server supports it before trying others + if (null !== $authtype && !in_array($authtype, $this->server_caps['AUTH'], true)) { + $this->edebug('Requested auth method not available: ' . $authtype, self::DEBUG_LOWLEVEL); + $authtype = null; + } + + if (empty($authtype)) { + //If no auth mechanism is specified, attempt to use these, in this order + //Try CRAM-MD5 first as it's more secure than the others + foreach (['CRAM-MD5', 'LOGIN', 'PLAIN', 'XOAUTH2'] as $method) { + if (in_array($method, $this->server_caps['AUTH'], true)) { + $authtype = $method; + break; + } + } + if (empty($authtype)) { + $this->setError('No supported authentication methods found'); + + return false; + } + $this->edebug('Auth method selected: ' . $authtype, self::DEBUG_LOWLEVEL); + } + + if (!in_array($authtype, $this->server_caps['AUTH'], true)) { + $this->setError("The requested authentication method \"$authtype\" is not supported by the server"); + + return false; + } + } elseif (empty($authtype)) { + $authtype = 'LOGIN'; + } + switch ($authtype) { + case 'PLAIN': + // Start authentication + if (!$this->sendCommand('AUTH', 'AUTH PLAIN', 334)) { + return false; + } + // Send encoded username and password + if (!$this->sendCommand( + 'User & Password', + base64_encode("\0" . $username . "\0" . $password), + 235 + ) + ) { + return false; + } + break; + case 'LOGIN': + // Start authentication + if (!$this->sendCommand('AUTH', 'AUTH LOGIN', 334)) { + return false; + } + if (!$this->sendCommand('Username', base64_encode($username), 334)) { + return false; + } + if (!$this->sendCommand('Password', base64_encode($password), 235)) { + return false; + } + break; + case 'CRAM-MD5': + // Start authentication + if (!$this->sendCommand('AUTH CRAM-MD5', 'AUTH CRAM-MD5', 334)) { + return false; + } + // Get the challenge + $challenge = base64_decode(substr($this->last_reply, 4)); + + // Build the response + $response = $username . ' ' . $this->hmac($challenge, $password); + + // send encoded credentials + return $this->sendCommand('Username', base64_encode($response), 235); + case 'XOAUTH2': + //The OAuth instance must be set up prior to requesting auth. + if (null === $OAuth) { + return false; + } + $oauth = $OAuth->getOauth64(); + + // Start authentication + if (!$this->sendCommand('AUTH', 'AUTH XOAUTH2 ' . $oauth, 235)) { + return false; + } + break; + default: + $this->setError("Authentication method \"$authtype\" is not supported"); + + return false; + } + + return true; + } + + /** + * Calculate an MD5 HMAC hash. + * Works like hash_hmac('md5', $data, $key) + * in case that function is not available. + * + * @param string $data The data to hash + * @param string $key The key to hash with + * + * @return string + */ + protected function hmac($data, $key) + { + if (function_exists('hash_hmac')) { + return hash_hmac('md5', $data, $key); + } + + // The following borrowed from + // http://php.net/manual/en/function.mhash.php#27225 + + // RFC 2104 HMAC implementation for php. + // Creates an md5 HMAC. + // Eliminates the need to install mhash to compute a HMAC + // by Lance Rushing + + $bytelen = 64; // byte length for md5 + if (strlen($key) > $bytelen) { + $key = pack('H*', md5($key)); + } + $key = str_pad($key, $bytelen, chr(0x00)); + $ipad = str_pad('', $bytelen, chr(0x36)); + $opad = str_pad('', $bytelen, chr(0x5c)); + $k_ipad = $key ^ $ipad; + $k_opad = $key ^ $opad; + + return md5($k_opad . pack('H*', md5($k_ipad . $data))); + } + + /** + * Check connection state. + * + * @return bool True if connected + */ + public function connected() + { + if (is_resource($this->smtp_conn)) { + $sock_status = stream_get_meta_data($this->smtp_conn); + if ($sock_status['eof']) { + // The socket is valid but we are not connected + $this->edebug( + 'SMTP NOTICE: EOF caught while checking if connected', + self::DEBUG_CLIENT + ); + $this->close(); + + return false; + } + + return true; // everything looks good + } + + return false; + } + + /** + * Close the socket and clean up the state of the class. + * Don't use this function without first trying to use QUIT. + * + * @see quit() + */ + public function close() + { + $this->setError(''); + $this->server_caps = null; + $this->helo_rply = null; + if (is_resource($this->smtp_conn)) { + // close the connection and cleanup + fclose($this->smtp_conn); + $this->smtp_conn = null; //Makes for cleaner serialization + $this->edebug('Connection: closed', self::DEBUG_CONNECTION); + } + } + + /** + * Send an SMTP DATA command. + * Issues a data command and sends the msg_data to the server, + * finializing the mail transaction. $msg_data is the message + * that is to be send with the headers. Each header needs to be + * on a single line followed by a with the message headers + * and the message body being separated by an additional . + * Implements RFC 821: DATA . + * + * @param string $msg_data Message data to send + * + * @return bool + */ + public function data($msg_data) + { + //This will use the standard timelimit + if (!$this->sendCommand('DATA', 'DATA', 354)) { + return false; + } + + /* The server is ready to accept data! + * According to rfc821 we should not send more than 1000 characters on a single line (including the LE) + * so we will break the data up into lines by \r and/or \n then if needed we will break each of those into + * smaller lines to fit within the limit. + * We will also look for lines that start with a '.' and prepend an additional '.'. + * NOTE: this does not count towards line-length limit. + */ + + // Normalize line breaks before exploding + $lines = explode("\n", str_replace(["\r\n", "\r"], "\n", $msg_data)); + + /* To distinguish between a complete RFC822 message and a plain message body, we check if the first field + * of the first line (':' separated) does not contain a space then it _should_ be a header and we will + * process all lines before a blank line as headers. + */ + + $field = substr($lines[0], 0, strpos($lines[0], ':')); + $in_headers = false; + if (!empty($field) && strpos($field, ' ') === false) { + $in_headers = true; + } + + foreach ($lines as $line) { + $lines_out = []; + if ($in_headers && $line === '') { + $in_headers = false; + } + //Break this line up into several smaller lines if it's too long + //Micro-optimisation: isset($str[$len]) is faster than (strlen($str) > $len), + while (isset($line[self::MAX_LINE_LENGTH])) { + //Working backwards, try to find a space within the last MAX_LINE_LENGTH chars of the line to break on + //so as to avoid breaking in the middle of a word + $pos = strrpos(substr($line, 0, self::MAX_LINE_LENGTH), ' '); + //Deliberately matches both false and 0 + if (!$pos) { + //No nice break found, add a hard break + $pos = self::MAX_LINE_LENGTH - 1; + $lines_out[] = substr($line, 0, $pos); + $line = substr($line, $pos); + } else { + //Break at the found point + $lines_out[] = substr($line, 0, $pos); + //Move along by the amount we dealt with + $line = substr($line, $pos + 1); + } + //If processing headers add a LWSP-char to the front of new line RFC822 section 3.1.1 + if ($in_headers) { + $line = "\t" . $line; + } + } + $lines_out[] = $line; + + //Send the lines to the server + foreach ($lines_out as $line_out) { + //RFC2821 section 4.5.2 + if (!empty($line_out) && $line_out[0] === '.') { + $line_out = '.' . $line_out; + } + $this->client_send($line_out . static::LE, 'DATA'); + } + } + + //Message data has been sent, complete the command + //Increase timelimit for end of DATA command + $savetimelimit = $this->Timelimit; + $this->Timelimit *= 2; + $result = $this->sendCommand('DATA END', '.', 250); + $this->recordLastTransactionID(); + //Restore timelimit + $this->Timelimit = $savetimelimit; + + return $result; + } + + /** + * Send an SMTP HELO or EHLO command. + * Used to identify the sending server to the receiving server. + * This makes sure that client and server are in a known state. + * Implements RFC 821: HELO + * and RFC 2821 EHLO. + * + * @param string $host The host name or IP to connect to + * + * @return bool + */ + public function hello($host = '') + { + //Try extended hello first (RFC 2821) + return $this->sendHello('EHLO', $host) or $this->sendHello('HELO', $host); + } + + /** + * Send an SMTP HELO or EHLO command. + * Low-level implementation used by hello(). + * + * @param string $hello The HELO string + * @param string $host The hostname to say we are + * + * @return bool + * + * @see hello() + */ + protected function sendHello($hello, $host) + { + $noerror = $this->sendCommand($hello, $hello . ' ' . $host, 250); + $this->helo_rply = $this->last_reply; + if ($noerror) { + $this->parseHelloFields($hello); + } else { + $this->server_caps = null; + } + + return $noerror; + } + + /** + * Parse a reply to HELO/EHLO command to discover server extensions. + * In case of HELO, the only parameter that can be discovered is a server name. + * + * @param string $type `HELO` or `EHLO` + */ + protected function parseHelloFields($type) + { + $this->server_caps = []; + $lines = explode("\n", $this->helo_rply); + + foreach ($lines as $n => $s) { + //First 4 chars contain response code followed by - or space + $s = trim(substr($s, 4)); + if (empty($s)) { + continue; + } + $fields = explode(' ', $s); + if (!empty($fields)) { + if (!$n) { + $name = $type; + $fields = $fields[0]; + } else { + $name = array_shift($fields); + switch ($name) { + case 'SIZE': + $fields = ($fields ? $fields[0] : 0); + break; + case 'AUTH': + if (!is_array($fields)) { + $fields = []; + } + break; + default: + $fields = true; + } + } + $this->server_caps[$name] = $fields; + } + } + } + + /** + * Send an SMTP MAIL command. + * Starts a mail transaction from the email address specified in + * $from. Returns true if successful or false otherwise. If True + * the mail transaction is started and then one or more recipient + * commands may be called followed by a data command. + * Implements RFC 821: MAIL FROM: . + * + * @param string $from Source address of this message + * + * @return bool + */ + public function mail($from) + { + $useVerp = ($this->do_verp ? ' XVERP' : ''); + + return $this->sendCommand( + 'MAIL FROM', + 'MAIL FROM:<' . $from . '>' . $useVerp, + 250 + ); + } + + /** + * Send an SMTP QUIT command. + * Closes the socket if there is no error or the $close_on_error argument is true. + * Implements from RFC 821: QUIT . + * + * @param bool $close_on_error Should the connection close if an error occurs? + * + * @return bool + */ + public function quit($close_on_error = true) + { + $noerror = $this->sendCommand('QUIT', 'QUIT', 221); + $err = $this->error; //Save any error + if ($noerror || $close_on_error) { + $this->close(); + $this->error = $err; //Restore any error from the quit command + } + + return $noerror; + } + + /** + * Send an SMTP RCPT command. + * Sets the TO argument to $toaddr. + * Returns true if the recipient was accepted false if it was rejected. + * Implements from RFC 821: RCPT TO: . + * + * @param string $address The address the message is being sent to + * @param string $dsn Comma separated list of DSN notifications. NEVER, SUCCESS, FAILURE + * or DELAY. If you specify NEVER all other notifications are ignored. + * + * @return bool + */ + public function recipient($address, $dsn = '') + { + if (empty($dsn)) { + $rcpt = 'RCPT TO:<' . $address . '>'; + } else { + $dsn = strtoupper($dsn); + $notify = []; + + if (strpos($dsn, 'NEVER') !== false) { + $notify[] = 'NEVER'; + } else { + foreach (['SUCCESS', 'FAILURE', 'DELAY'] as $value) { + if (strpos($dsn, $value) !== false) { + $notify[] = $value; + } + } + } + + $rcpt = 'RCPT TO:<' . $address . '> NOTIFY=' . implode(',', $notify); + } + + return $this->sendCommand( + 'RCPT TO', + $rcpt, + [250, 251] + ); + } + + /** + * Send an SMTP RSET command. + * Abort any transaction that is currently in progress. + * Implements RFC 821: RSET . + * + * @return bool True on success + */ + public function reset() + { + return $this->sendCommand('RSET', 'RSET', 250); + } + + /** + * Send a command to an SMTP server and check its return code. + * + * @param string $command The command name - not sent to the server + * @param string $commandstring The actual command to send + * @param int|array $expect One or more expected integer success codes + * + * @return bool True on success + */ + protected function sendCommand($command, $commandstring, $expect) + { + if (!$this->connected()) { + $this->setError("Called $command without being connected"); + + return false; + } + //Reject line breaks in all commands + if ((strpos($commandstring, "\n") !== false) || (strpos($commandstring, "\r") !== false)) { + $this->setError("Command '$command' contained line breaks"); + + return false; + } + $this->client_send($commandstring . static::LE, $command); + + $this->last_reply = $this->get_lines(); + // Fetch SMTP code and possible error code explanation + $matches = []; + if (preg_match('/^([\d]{3})[ -](?:([\d]\\.[\d]\\.[\d]{1,2}) )?/', $this->last_reply, $matches)) { + $code = (int) $matches[1]; + $code_ex = (count($matches) > 2 ? $matches[2] : null); + // Cut off error code from each response line + $detail = preg_replace( + "/{$code}[ -]" . + ($code_ex ? str_replace('.', '\\.', $code_ex) . ' ' : '') . '/m', + '', + $this->last_reply + ); + } else { + // Fall back to simple parsing if regex fails + $code = (int) substr($this->last_reply, 0, 3); + $code_ex = null; + $detail = substr($this->last_reply, 4); + } + + $this->edebug('SERVER -> CLIENT: ' . $this->last_reply, self::DEBUG_SERVER); + + if (!in_array($code, (array) $expect, true)) { + $this->setError( + "$command command failed", + $detail, + $code, + $code_ex + ); + $this->edebug( + 'SMTP ERROR: ' . $this->error['error'] . ': ' . $this->last_reply, + self::DEBUG_CLIENT + ); + + return false; + } + + $this->setError(''); + + return true; + } + + /** + * Send an SMTP SAML command. + * Starts a mail transaction from the email address specified in $from. + * Returns true if successful or false otherwise. If True + * the mail transaction is started and then one or more recipient + * commands may be called followed by a data command. This command + * will send the message to the users terminal if they are logged + * in and send them an email. + * Implements RFC 821: SAML FROM: . + * + * @param string $from The address the message is from + * + * @return bool + */ + public function sendAndMail($from) + { + return $this->sendCommand('SAML', "SAML FROM:$from", 250); + } + + /** + * Send an SMTP VRFY command. + * + * @param string $name The name to verify + * + * @return bool + */ + public function verify($name) + { + return $this->sendCommand('VRFY', "VRFY $name", [250, 251]); + } + + /** + * Send an SMTP NOOP command. + * Used to keep keep-alives alive, doesn't actually do anything. + * + * @return bool + */ + public function noop() + { + return $this->sendCommand('NOOP', 'NOOP', 250); + } + + /** + * Send an SMTP TURN command. + * This is an optional command for SMTP that this class does not support. + * This method is here to make the RFC821 Definition complete for this class + * and _may_ be implemented in future. + * Implements from RFC 821: TURN . + * + * @return bool + */ + public function turn() + { + $this->setError('The SMTP TURN command is not implemented'); + $this->edebug('SMTP NOTICE: ' . $this->error['error'], self::DEBUG_CLIENT); + + return false; + } + + /** + * Send raw data to the server. + * + * @param string $data The data to send + * @param string $command Optionally, the command this is part of, used only for controlling debug output + * + * @return int|bool The number of bytes sent to the server or false on error + */ + public function client_send($data, $command = '') + { + //If SMTP transcripts are left enabled, or debug output is posted online + //it can leak credentials, so hide credentials in all but lowest level + if (self::DEBUG_LOWLEVEL > $this->do_debug && + in_array($command, ['User & Password', 'Username', 'Password'], true)) { + $this->edebug('CLIENT -> SERVER: [credentials hidden]', self::DEBUG_CLIENT); + } else { + $this->edebug('CLIENT -> SERVER: ' . $data, self::DEBUG_CLIENT); + } + set_error_handler([$this, 'errorHandler']); + $result = fwrite($this->smtp_conn, $data); + restore_error_handler(); + + return $result; + } + + /** + * Get the latest error. + * + * @return array + */ + public function getError() + { + return $this->error; + } + + /** + * Get SMTP extensions available on the server. + * + * @return array|null + */ + public function getServerExtList() + { + return $this->server_caps; + } + + /** + * Get metadata about the SMTP server from its HELO/EHLO response. + * The method works in three ways, dependent on argument value and current state: + * 1. HELO/EHLO has not been sent - returns null and populates $this->error. + * 2. HELO has been sent - + * $name == 'HELO': returns server name + * $name == 'EHLO': returns boolean false + * $name == any other string: returns null and populates $this->error + * 3. EHLO has been sent - + * $name == 'HELO'|'EHLO': returns the server name + * $name == any other string: if extension $name exists, returns True + * or its options (e.g. AUTH mechanisms supported). Otherwise returns False. + * + * @param string $name Name of SMTP extension or 'HELO'|'EHLO' + * + * @return string|bool|null + */ + public function getServerExt($name) + { + if (!$this->server_caps) { + $this->setError('No HELO/EHLO was sent'); + + return; + } + + if (!array_key_exists($name, $this->server_caps)) { + if ('HELO' === $name) { + return $this->server_caps['EHLO']; + } + if ('EHLO' === $name || array_key_exists('EHLO', $this->server_caps)) { + return false; + } + $this->setError('HELO handshake was used; No information about server extensions available'); + + return; + } + + return $this->server_caps[$name]; + } + + /** + * Get the last reply from the server. + * + * @return string + */ + public function getLastReply() + { + return $this->last_reply; + } + + /** + * Read the SMTP server's response. + * Either before eof or socket timeout occurs on the operation. + * With SMTP we can tell if we have more lines to read if the + * 4th character is '-' symbol. If it is a space then we don't + * need to read anything else. + * + * @return string + */ + protected function get_lines() + { + // If the connection is bad, give up straight away + if (!is_resource($this->smtp_conn)) { + return ''; + } + $data = ''; + $endtime = 0; + stream_set_timeout($this->smtp_conn, $this->Timeout); + if ($this->Timelimit > 0) { + $endtime = time() + $this->Timelimit; + } + $selR = [$this->smtp_conn]; + $selW = null; + while (is_resource($this->smtp_conn) && !feof($this->smtp_conn)) { + //Must pass vars in here as params are by reference + if (!stream_select($selR, $selW, $selW, $this->Timelimit)) { + $this->edebug( + 'SMTP -> get_lines(): timed-out (' . $this->Timeout . ' sec)', + self::DEBUG_LOWLEVEL + ); + break; + } + //Deliberate noise suppression - errors are handled afterwards + $str = @fgets($this->smtp_conn, self::MAX_REPLY_LENGTH); + $this->edebug('SMTP INBOUND: "' . trim($str) . '"', self::DEBUG_LOWLEVEL); + $data .= $str; + // If response is only 3 chars (not valid, but RFC5321 S4.2 says it must be handled), + // or 4th character is a space or a line break char, we are done reading, break the loop. + // String array access is a significant micro-optimisation over strlen + if (!isset($str[3]) || $str[3] === ' ' || $str[3] === "\r" || $str[3] === "\n") { + break; + } + // Timed-out? Log and break + $info = stream_get_meta_data($this->smtp_conn); + if ($info['timed_out']) { + $this->edebug( + 'SMTP -> get_lines(): timed-out (' . $this->Timeout . ' sec)', + self::DEBUG_LOWLEVEL + ); + break; + } + // Now check if reads took too long + if ($endtime && time() > $endtime) { + $this->edebug( + 'SMTP -> get_lines(): timelimit reached (' . + $this->Timelimit . ' sec)', + self::DEBUG_LOWLEVEL + ); + break; + } + } + + return $data; + } + + /** + * Enable or disable VERP address generation. + * + * @param bool $enabled + */ + public function setVerp($enabled = false) + { + $this->do_verp = $enabled; + } + + /** + * Get VERP address generation mode. + * + * @return bool + */ + public function getVerp() + { + return $this->do_verp; + } + + /** + * Set error messages and codes. + * + * @param string $message The error message + * @param string $detail Further detail on the error + * @param string $smtp_code An associated SMTP error code + * @param string $smtp_code_ex Extended SMTP code + */ + protected function setError($message, $detail = '', $smtp_code = '', $smtp_code_ex = '') + { + $this->error = [ + 'error' => $message, + 'detail' => $detail, + 'smtp_code' => $smtp_code, + 'smtp_code_ex' => $smtp_code_ex, + ]; + } + + /** + * Set debug output method. + * + * @param string|callable $method The name of the mechanism to use for debugging output, or a callable to handle it + */ + public function setDebugOutput($method = 'echo') + { + $this->Debugoutput = $method; + } + + /** + * Get debug output method. + * + * @return string + */ + public function getDebugOutput() + { + return $this->Debugoutput; + } + + /** + * Set debug output level. + * + * @param int $level + */ + public function setDebugLevel($level = 0) + { + $this->do_debug = $level; + } + + /** + * Get debug output level. + * + * @return int + */ + public function getDebugLevel() + { + return $this->do_debug; + } + + /** + * Set SMTP timeout. + * + * @param int $timeout The timeout duration in seconds + */ + public function setTimeout($timeout = 0) + { + $this->Timeout = $timeout; + } + + /** + * Get SMTP timeout. + * + * @return int + */ + public function getTimeout() + { + return $this->Timeout; + } + + /** + * Reports an error number and string. + * + * @param int $errno The error number returned by PHP + * @param string $errmsg The error message returned by PHP + * @param string $errfile The file the error occurred in + * @param int $errline The line number the error occurred on + */ + protected function errorHandler($errno, $errmsg, $errfile = '', $errline = 0) + { + $notice = 'Connection failed.'; + $this->setError( + $notice, + $errmsg, + (string) $errno + ); + $this->edebug( + "$notice Error #$errno: $errmsg [$errfile line $errline]", + self::DEBUG_CONNECTION + ); + } + + /** + * Extract and return the ID of the last SMTP transaction based on + * a list of patterns provided in SMTP::$smtp_transaction_id_patterns. + * Relies on the host providing the ID in response to a DATA command. + * If no reply has been received yet, it will return null. + * If no pattern was matched, it will return false. + * + * @return bool|string|null + */ + protected function recordLastTransactionID() + { + $reply = $this->getLastReply(); + + if (empty($reply)) { + $this->last_smtp_transaction_id = null; + } else { + $this->last_smtp_transaction_id = false; + foreach ($this->smtp_transaction_id_patterns as $smtp_transaction_id_pattern) { + $matches = []; + if (preg_match($smtp_transaction_id_pattern, $reply, $matches)) { + $this->last_smtp_transaction_id = trim($matches[1]); + break; + } + } + } + + return $this->last_smtp_transaction_id; + } + + /** + * Get the queue/transaction ID of the last SMTP transaction + * If no reply has been received yet, it will return null. + * If no pattern was matched, it will return false. + * + * @return bool|string|null + * + * @see recordLastTransactionID() + */ + public function getLastTransactionID() + { + return $this->last_smtp_transaction_id; + } +} diff --git a/inc/email.php b/inc/email.php index 72e99d7..edc6026 100644 --- a/inc/email.php +++ b/inc/email.php @@ -1,4 +1,11 @@ asDests = $asDests; } - public function send() { + /*public function send() { $sEOL = "\r\n"; foreach($this->asDests as $asDest) { //Message @@ -102,5 +109,55 @@ class Email extends PhpObject { //Send if(!mail($asDest['email'], $asTemplate['subject'], $sBody, $sHeaders)) $this->addError('Could not send '.$this->sTemplate.' email to '.$asDest['email']); } + }*/ + + public function send() { + //Instantiation and passing `true` enables exceptions + $oPHPMailer = new PHPMailer(true); + + //Server settings + if(Settings::DEBUG) $oPHPMailer->SMTPDebug = SMTP::DEBUG_SERVER;//Enable verbose debug output + $oPHPMailer->isSMTP(); //Send using SMTP + $oPHPMailer->CharSet = Settings::TEXT_ENC; //Mail Character Set + $oPHPMailer->Encoding = 'base64'; //Base 64 Character Encoding + $oPHPMailer->Host = Settings::MAIL_SERVER; //Set the SMTP server to send through + $oPHPMailer->SMTPAuth = true; //Enable SMTP authentication + $oPHPMailer->Username = Settings::MAIL_USER; //SMTP username + $oPHPMailer->Password = Settings::MAIL_PASS; //SMTP password + $oPHPMailer->SMTPSecure = PHPMailer::ENCRYPTION_STARTTLS; //Enable TLS encryption; `PHPMailer::ENCRYPTION_SMTPS` encouraged + $oPHPMailer->Port = 587; //TCP port to connect to, use 465 for `PHPMailer::ENCRYPTION_SMTPS` above + $oPHPMailer->setFrom(Settings::MAIL_USER, 'Spotty'); + $oPHPMailer->addReplyTo(Settings::MAIL_USER, 'Spotty'); + + foreach($this->asDests as $asDest) { + try { + //Message + $asTemplate = $this->getTemplate($asDest['language']); + $oEmail = $asTemplate['email']; + + //Unsubscribe Link + $sUnsubLink = $this->sServName.'?a=unsubscribe_email&id='.$asDest['id_user']; + $oEmail->setTag('unsubscribe_link', $sUnsubLink); + + //Email Content + $sHtmlMessage = $oEmail->getMask(); + $sPlainMessage = strip_tags(str_replace('
', "\n", $sHtmlMessage)); + + //Recipients + $oPHPMailer->addAddress($asDest['email'], $asDest['name']); + + //Content + $oPHPMailer->isHTML(true); + $oPHPMailer->Subject = $asTemplate['subject']; + $oPHPMailer->Body = $sHtmlMessage; + $oPHPMailer->AltBody = $sPlainMessage; + + $oPHPMailer->send(); + } + catch (Exception $e) { + $this->addError('Message could not be sent to "'.$asDest['email'].'". Mailer Error: '.$oPHPMailer->ErrorInfo); + } + } } + } \ No newline at end of file diff --git a/masks/project.html b/masks/project.html index 65ab1ac..6d7fad8 100644 --- a/masks/project.html +++ b/masks/project.html @@ -5,6 +5,7 @@
+

[#]lang:maps[#]

diff --git a/settings-sample.php b/settings-sample.php index 52a9d6d..4357f80 100755 --- a/settings-sample.php +++ b/settings-sample.php @@ -10,5 +10,8 @@ class Settings const DB_ENC = 'utf8mb4'; const TEXT_ENC = 'UTF-8'; const TIMEZONE = 'Europe/Paris'; + const MAIL_SERVER = ''; + const MAIL_USER = ''; + const MAIL_PASS = ''; const DEBUG = true; } diff --git a/style/_mask_project.scss b/style/_mask_project.scss index f91300d..0dd52d6 100644 --- a/style/_mask_project.scss +++ b/style/_mask_project.scss @@ -476,12 +476,12 @@ $legend-color: $post-color; background: rgba(255, 255, 255, 0.8); .settings-section { + display: inline-block; + width: 100%; + h1 { margin: 1.5rem 0 1rem; } - &:first-child h1 { - margin-top: 0; - } label { margin-bottom: .3em; @@ -489,6 +489,25 @@ $legend-color: $post-color; cursor: pointer; } + &.title { + margin: -1rem; + width: calc(100% + 2rem); + background: rgba(255, 255, 255, 0.4); + text-align: center; + border-radius: 3px 3px 0 0; + + .logo { + background: rgba(255,255,255,.4); + padding: 1.5rem; + + img { + width: calc(100% - 3rem); + max-width: 180px; + transform: translateX(-10%); //Center Text, not logo. logo width (40px) / image width (200px) = 20%. And centering: 20% / 2 = 10% + } + } + } + &.newsletter { input#email { width: calc(100% - 6em); diff --git a/style/spot.css b/style/spot.css index 7003cf0..78a4d19 100644 --- a/style/spot.css +++ b/style/spot.css @@ -1,4 +1,4 @@ @font-face{font-family:"Ubuntu";font-style:normal;font-weight:400;src:local("Ubuntu Regular"),local("Ubuntu-Regular"),url(https://fonts.gstatic.com/s/ubuntu/v13/4iCs6KVjbNBYlgoKcg72j00.woff2) format("woff2");unicode-range:U+0460-052F,U+1C80-1C88,U+20B4,U+2DE0-2DFF,U+A640-A69F,U+FE2E-FE2F}@font-face{font-family:"Ubuntu";font-style:normal;font-weight:400;src:local("Ubuntu Regular"),local("Ubuntu-Regular"),url(https://fonts.gstatic.com/s/ubuntu/v13/4iCs6KVjbNBYlgoKew72j00.woff2) format("woff2");unicode-range:U+0400-045F,U+0490-0491,U+04B0-04B1,U+2116}@font-face{font-family:"Ubuntu";font-style:normal;font-weight:400;src:local("Ubuntu Regular"),local("Ubuntu-Regular"),url(https://fonts.gstatic.com/s/ubuntu/v13/4iCs6KVjbNBYlgoKcw72j00.woff2) format("woff2");unicode-range:U+1F00-1FFF}@font-face{font-family:"Ubuntu";font-style:normal;font-weight:400;src:local("Ubuntu Regular"),local("Ubuntu-Regular"),url(https://fonts.gstatic.com/s/ubuntu/v13/4iCs6KVjbNBYlgoKfA72j00.woff2) format("woff2");unicode-range:U+0370-03FF}@font-face{font-family:"Ubuntu";font-style:normal;font-weight:400;src:local("Ubuntu Regular"),local("Ubuntu-Regular"),url(https://fonts.gstatic.com/s/ubuntu/v13/4iCs6KVjbNBYlgoKcQ72j00.woff2) format("woff2");unicode-range:U+0100-024F,U+0259,U+1E00-1EFF,U+2020,U+20A0-20AB,U+20AD-20CF,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:"Ubuntu";font-style:normal;font-weight:400;src:local("Ubuntu Regular"),local("Ubuntu-Regular"),url(fonts/4iCs6KVjbNBYlgoKfw72.woff2) format("woff2");unicode-range:U+0000-00FF,U+0131,U+0152-0153,U+02BB-02BC,U+02C6,U+02DA,U+02DC,U+2000-206F,U+2074,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@font-face{font-family:"Ubuntu";font-style:normal;font-weight:700;src:local("Ubuntu Bold"),local("Ubuntu-Bold"),url(https://fonts.gstatic.com/s/ubuntu/v13/4iCv6KVjbNBYlgoCxCvjvWyNL4U.woff2) format("woff2");unicode-range:U+0460-052F,U+1C80-1C88,U+20B4,U+2DE0-2DFF,U+A640-A69F,U+FE2E-FE2F}@font-face{font-family:"Ubuntu";font-style:normal;font-weight:700;src:local("Ubuntu Bold"),local("Ubuntu-Bold"),url(https://fonts.gstatic.com/s/ubuntu/v13/4iCv6KVjbNBYlgoCxCvjtGyNL4U.woff2) format("woff2");unicode-range:U+0400-045F,U+0490-0491,U+04B0-04B1,U+2116}@font-face{font-family:"Ubuntu";font-style:normal;font-weight:700;src:local("Ubuntu Bold"),local("Ubuntu-Bold"),url(https://fonts.gstatic.com/s/ubuntu/v13/4iCv6KVjbNBYlgoCxCvjvGyNL4U.woff2) format("woff2");unicode-range:U+1F00-1FFF}@font-face{font-family:"Ubuntu";font-style:normal;font-weight:700;src:local("Ubuntu Bold"),local("Ubuntu-Bold"),url(https://fonts.gstatic.com/s/ubuntu/v13/4iCv6KVjbNBYlgoCxCvjs2yNL4U.woff2) format("woff2");unicode-range:U+0370-03FF}@font-face{font-family:"Ubuntu";font-style:normal;font-weight:700;src:local("Ubuntu Bold"),local("Ubuntu-Bold"),url(https://fonts.gstatic.com/s/ubuntu/v13/4iCv6KVjbNBYlgoCxCvjvmyNL4U.woff2) format("woff2");unicode-range:U+0100-024F,U+0259,U+1E00-1EFF,U+2020,U+20A0-20AB,U+20AD-20CF,U+2113,U+2C60-2C7F,U+A720-A7FF}@font-face{font-family:"Ubuntu";font-style:normal;font-weight:700;src:local("Ubuntu Bold"),local("Ubuntu-Bold"),url(fonts/4iCv6KVjbNBYlgoCxCvjsGyN.woff2) format("woff2");unicode-range:U+0000-00FF,U+0131,U+0152-0153,U+02BB-02BC,U+02C6,U+02DA,U+02DC,U+2000-206F,U+2074,U+20AC,U+2122,U+2191,U+2193,U+2212,U+2215,U+FEFF,U+FFFD}@-webkit-keyframes fadeIn{from{opacity:.3}}@-moz-keyframes fadeIn{from{opacity:.3}}@-ms-keyframes fadeIn{from{opacity:.3}}@-o-keyframes fadeIn{from{opacity:.3}}@keyframes fadeIn{from{opacity:.3}}.flicker,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.loading span,.lightbox .lb-cancel{-webkit-animation:fadeIn .5s infinite alternate;-moz-animation:fadeIn .5s infinite alternate;-ms-animation:fadeIn .5s infinite alternate;-o-animation:fadeIn .5s infinite alternate;animation:fadeIn .5s infinite alternate}body,textarea,input,button{font-size:14px;font-family:"Ubuntu",sans-serif;margin:0}textarea{resize:none}button{cursor:pointer;font-weight:bold}input,textarea,button{border:none;padding:.5em 1em;border-radius:3px}.feedback p{margin:0 0 1em 0}.feedback p.error{color:red}.feedback p.warning{color:orange}.feedback p.success{color:green}/*! * Font Awesome Pro 5.11.2 by @fontawesome - https://fontawesome.com * License - https://fontawesome.com/license (Commercial License) - */@font-face{font-family:"Font Awesome 5 Pro";font-style:normal;font-weight:900;font-display:auto;src:url("fa/fonts/fa-solid-900.eot");src:url("fa/fonts/fa-solid-900.eot?#iefix") format("embedded-opentype"),url("fa/fonts/fa-solid-900.woff2") format("woff2"),url("fa/fonts/fa-solid-900.woff") format("woff"),url("fa/fonts/fa-solid-900.ttf") format("truetype"),url("fa/fonts/fa-solid-900.svg#fontawesome") format("svg")}.fa,.lightbox .lb-cancel,.lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.control-icon,#projects #post-button .fa,#projects #post-button .control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .lb-close,#projects #post-button .lightbox .lb-cancel,.lightbox #projects #post-button .lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .elevation-toggle,#projects #settings-button .fa,#projects #settings-button .control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .lb-close,#projects #settings-button .lightbox .lb-cancel,.lightbox #projects #settings-button .lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .elevation-toggle,#projects #post-button #settings-button .fa,#projects #settings-button #post-button .fa,#projects #post-button #settings-button .control-icon,#projects #settings-button #post-button .control-icon,.spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.fas{font-family:"Font Awesome 5 Pro";font-weight:900}.fa,.lightbox .lb-cancel,.lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.control-icon,#projects #post-button .fa,#projects #post-button .control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .lb-close,#projects #post-button .lightbox .lb-cancel,.lightbox #projects #post-button .lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .elevation-toggle,#projects #settings-button .fa,#projects #settings-button .control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .lb-close,#projects #settings-button .lightbox .lb-cancel,.lightbox #projects #settings-button .lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .elevation-toggle,#projects #post-button #settings-button .fa,#projects #settings-button #post-button .fa,#projects #post-button #settings-button .control-icon,#projects #settings-button #post-button .control-icon,.spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.fas,.far,.fal,.fad,.fab{-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;display:inline-block;font-style:normal;font-variant:normal;text-rendering:auto;line-height:1}.fa-lg{font-size:1.3333333333em;line-height:.75em;vertical-align:-0.0667em}.fa-xs{font-size:.75em}.fa-sm{font-size:.875em}.fa-1x{font-size:1em}.fa-2x{font-size:2em}.fa-3x{font-size:3em}.fa-4x{font-size:4em}.fa-5x{font-size:5em}.fa-6x{font-size:6em}.fa-7x{font-size:7em}.fa-8x{font-size:8em}.fa-9x{font-size:9em}.fa-10x{font-size:10em}.fa-fw{text-align:center;width:1.25em}.fa-ul{list-style-type:none;margin-left:2.5em;padding-left:0}.fa-ul>li{position:relative}.fa-li{left:-2em;position:absolute;text-align:center;width:2em;line-height:inherit}.fa-border{border:solid .08em #eee;border-radius:.1em;padding:.2em .25em .15em}.fa-pull-left{float:left}.fa-pull-right{float:right}.fa.fa-pull-left,.lightbox .fa-pull-left.lb-cancel,.lightbox .lb-dataContainer .lb-data .lb-closeContainer .fa-pull-left.lb-close,.lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-left.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-left.lb-prev,.fa-pull-left.control-icon,#projects #post-button .fa-pull-left.fa,#projects #post-button .fa-pull-left.control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-left.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.fa-pull-left.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-left.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.fa-pull-left.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .fa-pull-left.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .fa-pull-left.lb-close,#projects #post-button .lightbox .fa-pull-left.lb-cancel,.lightbox #projects #post-button .fa-pull-left.lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .fa-pull-left.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .fa-pull-left.elevation-toggle,#projects #settings-button .fa-pull-left.fa,#projects #settings-button .fa-pull-left.control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-left.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.fa-pull-left.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-left.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.fa-pull-left.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .fa-pull-left.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .fa-pull-left.lb-close,#projects #settings-button .lightbox .fa-pull-left.lb-cancel,.lightbox #projects #settings-button .fa-pull-left.lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .fa-pull-left.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .fa-pull-left.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed .fa-pull-left.elevation-toggle,.fas.fa-pull-left,.far.fa-pull-left,.fal.fa-pull-left,.fab.fa-pull-left{margin-right:.3em}.fa.fa-pull-right,.lightbox .fa-pull-right.lb-cancel,.lightbox .lb-dataContainer .lb-data .lb-closeContainer .fa-pull-right.lb-close,.lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-right.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-right.lb-prev,.fa-pull-right.control-icon,#projects #post-button .fa-pull-right.fa,#projects #post-button .fa-pull-right.control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-right.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.fa-pull-right.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-right.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.fa-pull-right.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .fa-pull-right.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .fa-pull-right.lb-close,#projects #post-button .lightbox .fa-pull-right.lb-cancel,.lightbox #projects #post-button .fa-pull-right.lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .fa-pull-right.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .fa-pull-right.elevation-toggle,#projects #settings-button .fa-pull-right.fa,#projects #settings-button .fa-pull-right.control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-right.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.fa-pull-right.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-right.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.fa-pull-right.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .fa-pull-right.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .fa-pull-right.lb-close,#projects #settings-button .lightbox .fa-pull-right.lb-cancel,.lightbox #projects #settings-button .fa-pull-right.lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .fa-pull-right.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .fa-pull-right.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed .fa-pull-right.elevation-toggle,.fas.fa-pull-right,.far.fa-pull-right,.fal.fa-pull-right,.fab.fa-pull-right{margin-left:.3em}.fa-spin{animation:fa-spin 2s infinite linear}.fa-pulse{animation:fa-spin 1s infinite steps(8)}@keyframes fa-spin{0%{transform:rotate(0deg)}100%{transform:rotate(360deg)}}.fa-rotate-90{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=1)";transform:rotate(90deg)}.fa-rotate-180{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=2)";transform:rotate(180deg)}.fa-rotate-270{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=3)";transform:rotate(270deg)}.fa-flip-horizontal{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=0, mirror=1)";transform:scale(-1, 1)}.fa-flip-vertical{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=2, mirror=1)";transform:scale(1, -1)}.fa-flip-both,.fa-flip-horizontal.fa-flip-vertical{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=2, mirror=1)";transform:scale(-1, -1)}:root .fa-rotate-90,:root .fa-rotate-180,:root .fa-rotate-270,:root .fa-flip-horizontal,:root .fa-flip-vertical,:root .fa-flip-both{filter:none}.fa-stack{display:inline-block;height:2em;line-height:2em;position:relative;vertical-align:middle;width:2.5em}.fa-stack-1x,.fa-stack-2x{left:0;position:absolute;text-align:center;width:100%}.fa-stack-1x{line-height:inherit}.fa-stack-2x{font-size:2em}.fa-inverse{color:#fff}.fa.push,.lightbox .push.lb-cancel,.lightbox .lb-dataContainer .lb-data .lb-closeContainer .push.lb-close,.lightbox .lb-outerContainer .lb-container .lb-nav a.push.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav a.push.lb-prev,.push.control-icon,#projects #post-button .push.fa,#projects #post-button .push.control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.push.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.push.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .push.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .push.lb-close,#projects #post-button .lightbox .push.lb-cancel,.lightbox #projects #post-button .push.lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .push.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .push.elevation-toggle,#projects #settings-button .push.fa,#projects #settings-button .push.control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.push.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.push.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .push.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .push.lb-close,#projects #settings-button .lightbox .push.lb-cancel,.lightbox #projects #settings-button .push.lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .push.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .push.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed .push.elevation-toggle{margin-right:.5rem}.fa.push-left,.lightbox .push-left.lb-cancel,.lightbox .lb-dataContainer .lb-data .lb-closeContainer .push-left.lb-close,.lightbox .lb-outerContainer .lb-container .lb-nav a.push-left.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav a.push-left.lb-prev,.push-left.control-icon,#projects #post-button .push-left.fa,#projects #post-button .push-left.control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push-left.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.push-left.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push-left.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.push-left.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .push-left.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .push-left.lb-close,#projects #post-button .lightbox .push-left.lb-cancel,.lightbox #projects #post-button .push-left.lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .push-left.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .push-left.elevation-toggle,#projects #settings-button .push-left.fa,#projects #settings-button .push-left.control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push-left.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.push-left.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push-left.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.push-left.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .push-left.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .push-left.lb-close,#projects #settings-button .lightbox .push-left.lb-cancel,.lightbox #projects #settings-button .push-left.lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .push-left.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .push-left.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed .push-left.elevation-toggle{margin-left:.5rem}.control-icon,#projects #post-button .fa,#projects #post-button .control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .lb-close,#projects #post-button .lightbox .lb-cancel,.lightbox #projects #post-button .lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .elevation-toggle,#projects #settings-button .fa,#projects #settings-button .control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .lb-close,#projects #settings-button .lightbox .lb-cancel,.lightbox #projects #settings-button .lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .elevation-toggle,#projects #post-button #settings-button .fa,#projects #settings-button #post-button .fa,#projects #post-button #settings-button .control-icon,#projects #settings-button #post-button .control-icon,.spot-theme.leaflet-control.elevation-collapsed .elevation-toggle{font-size:28px;text-align:center;line-height:44px;text-decoration:none;color:#999;background:none}.fa-map:before{content:""}.fa-track-off-track:before{content:""}.fa-track-main:before{content:""}.fa-track-hitchhiking:before{content:""}.fa-track-start:before{content:""}.fa-track-end:before{content:""}.fa-layers:before{content:""}.fa-elev-chart:before,.spot-theme.leaflet-control.elevation-collapsed .elevation-toggle:before{content:""}.fa-distance:before{content:""}.fa-elev-drop:before{content:""}.fa-elev-gain:before{content:""}.fa-download:before{content:""}.fa-menu:before,#projects #settings-button .fa:before,#projects #settings-button .control-icon:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-prev:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-next:before,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close:before,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .lb-close:before,#projects #settings-button .lightbox .lb-cancel:before,.lightbox #projects #settings-button .lb-cancel:before,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle:before,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .elevation-toggle:before,#projects #settings-button #post-button .fa:before,#projects #post-button #settings-button .fa:before,#projects #settings-button #post-button .control-icon:before,#projects #post-button #settings-button .control-icon:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-prev:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-next:before,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button .lb-close:before,#projects #post-button .lightbox #settings-button .lb-cancel:before,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings-button .elevation-toggle:before{content:""}.fa-newsletter:before{content:""}.fa-project:before{content:""}.fa-error:before{content:""}.fa-warning:before{content:""}.fa-success:before{content:""}.fa-unsubscribe:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .fa:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .control-icon:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe a.lb-prev:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe a.lb-next:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close:before,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lb-close:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lightbox .lb-cancel:before,.lightbox #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lb-cancel:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle:before,.spot-theme.leaflet-control.elevation-collapsed #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .elevation-toggle:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe a.lb-prev:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe a.lb-next:before,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lb-close:before,#projects #post-button .lightbox #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lb-cancel:before,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .elevation-toggle:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe a.lb-prev:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe a.lb-next:before,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lb-close:before,#projects #settings-button .lightbox #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lb-cancel:before,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .elevation-toggle:before{content:""}.fa-post:before,#projects #post-button .fa:before,#projects #post-button .control-icon:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-prev:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-next:before,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close:before,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .lb-close:before,#projects #post-button .lightbox .lb-cancel:before,.lightbox #projects #post-button .lb-cancel:before,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle:before,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .elevation-toggle:before,#projects #post-button #settings-button .fa:before,#projects #settings-button #post-button .fa:before,#projects #post-button #settings-button .control-icon:before,#projects #settings-button #post-button .control-icon:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-prev:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-next:before,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button .lb-close:before,#projects #settings-button .lightbox #post-button .lb-cancel:before,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #post-button .elevation-toggle:before{content:""}.fa-media:before{content:""}.fa-video:before{content:""}.fa-image:before{content:""}.fa-message:before{content:""}.fa-message-in:before{content:""}.fa-time:before{content:""}.fa-coords:before{content:""}.fa-drill-video:before{content:""}.fa-drill-picture:before{content:""}.fa-drill-message:before,#projects #feed #posts .post.message a.drill:hover .fa-message:before{content:""}.fa-upload:before{content:""}.fa-video-shot:before{content:""}.fa-image-shot:before{content:""}.fa-poster:before{content:""}.fa-send:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .fa:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .control-icon:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe a.lb-prev:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe a.lb-next:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close:before,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lb-close:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lightbox .lb-cancel:before,.lightbox #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lb-cancel:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle:before,.spot-theme.leaflet-control.elevation-collapsed #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .elevation-toggle:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe a.lb-prev:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe a.lb-next:before,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lb-close:before,#projects #post-button .lightbox #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lb-cancel:before,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .elevation-toggle:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe a.lb-prev:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe a.lb-next:before,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lb-close:before,#projects #settings-button .lightbox #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lb-cancel:before,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .elevation-toggle:before{content:""}.fa-cancel:before,.lightbox .lb-cancel:before{content:""}.fa-prev:before,#projects.with-settings #settings-button .fa:before,#projects.with-settings #settings-button .control-icon:before,#projects.with-settings #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-prev:before,#projects.with-settings #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-next:before,#projects.with-settings #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button .lb-close:before,#projects.with-settings #post-button .lightbox #settings-button .lb-cancel:before,#projects.with-settings #post-button .spot-theme.leaflet-control.elevation-collapsed #settings-button .elevation-toggle:before,#projects.with-settings #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects.with-settings #settings-button a.lb-prev:before,#projects.with-settings #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects.with-settings #settings-button a.lb-next:before,#projects.with-settings #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close:before,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects.with-settings #settings-button .lb-close:before,#projects.with-settings #settings-button .lightbox .lb-cancel:before,.lightbox #projects.with-settings #settings-button .lb-cancel:before,#projects.with-settings #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle:before,.spot-theme.leaflet-control.elevation-collapsed #projects.with-settings #settings-button .elevation-toggle:before,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before{content:""}.fa-next:before,#projects.with-feed #post-button .fa:before,#projects.with-feed #post-button .control-icon:before,#projects.with-feed #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects.with-feed #post-button a.lb-prev:before,#projects.with-feed #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects.with-feed #post-button a.lb-next:before,#projects.with-feed #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close:before,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects.with-feed #post-button .lb-close:before,#projects.with-feed #post-button .lightbox .lb-cancel:before,.lightbox #projects.with-feed #post-button .lb-cancel:before,#projects.with-feed #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle:before,.spot-theme.leaflet-control.elevation-collapsed #projects.with-feed #post-button .elevation-toggle:before,#projects.with-feed #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-prev:before,#projects.with-feed #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-next:before,#projects.with-feed #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button .lb-close:before,#projects.with-feed #settings-button .lightbox #post-button .lb-cancel:before,#projects.with-feed #settings-button .spot-theme.leaflet-control.elevation-collapsed #post-button .elevation-toggle:before,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before{content:""}.fa-close:before,.lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close:before{content:""}body.lb-disable-scrolling{overflow:hidden}.lightboxOverlay{position:absolute;top:0;left:0;z-index:9999;background-color:#000;filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=80);opacity:.8;display:none}.lightbox{position:absolute;left:0;width:100%;z-index:10000;text-align:center;line-height:0;font-weight:normal;outline:none}.lightbox .lb-image{display:block;height:auto;max-width:inherit;max-height:none;border-radius:3px;border:4px solid #fff}.lightbox .lb-video{box-sizing:content-box}.lightbox .lb-video-nav .lb-nav{height:calc(100% - 45px)}.lightbox a img{border:none}.lb-outerContainer{position:relative;*zoom:1;width:250px;height:250px;margin:0 auto;border-radius:4px;background-color:#fff}.lb-outerContainer:after{content:"";display:table;clear:both}.lb-loader{position:absolute;top:43%;left:0;height:25%;width:100%;text-align:center;line-height:0}.lb-cancel{display:block;width:32px;height:32px;margin:0 auto;background:url(../images/loading.gif) no-repeat}.lb-nav{position:absolute;top:0;left:0;height:100%;width:100%;z-index:10}.lb-container>.nav{left:0}.lb-nav a{outline:none;background-image:url("")}.lb-prev,.lb-next{height:100%;cursor:pointer;display:block}.lb-nav a.lb-prev{width:34%;left:0;float:left;background:url(../images/prev.png) left 48% no-repeat;filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=0);opacity:0;-webkit-transition:opacity .6s;-moz-transition:opacity .6s;-o-transition:opacity .6s;transition:opacity .6s}.lb-nav a.lb-prev:hover{filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=100);opacity:1}.lb-nav a.lb-next{width:64%;right:0;float:right;background:url(../images/next.png) right 48% no-repeat;filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=0);opacity:0;-webkit-transition:opacity .6s;-moz-transition:opacity .6s;-o-transition:opacity .6s;transition:opacity .6s}.lb-nav a.lb-next:hover{filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=100);opacity:1}.lb-dataContainer{margin:0 auto;padding-top:5px;*zoom:1;width:100%;border-bottom-left-radius:4px;border-bottom-right-radius:4px}.lb-dataContainer:after{content:"";display:table;clear:both}.lb-data{padding:0 4px;color:#ccc}.lb-data .lb-details{width:85%;float:left;text-align:left;line-height:1.1em}.lb-data .lb-caption{font-size:13px;font-weight:bold;line-height:1em}.lb-data .lb-caption a{color:#4ae}.lb-data .lb-number{display:block;clear:left;padding-bottom:1em;font-size:12px;color:#999}.lb-data .lb-close{display:block;float:right;width:30px;height:30px;background:url(../images/close.png) top right no-repeat;text-align:right;outline:none;filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=70);opacity:.7;-webkit-transition:opacity .2s;-moz-transition:opacity .2s;-o-transition:opacity .2s;transition:opacity .2s}.lb-data .lb-close:hover{cursor:pointer;filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=100);opacity:1}.lightboxOverlay{bottom:0;right:0}.lightbox{display:flex;align-items:center;justify-content:center;width:100%;height:100%}.lightbox .lb-outerContainer{margin:0;border-radius:0;background-color:rgba(255,255,255,.5)}.lightbox .lb-outerContainer .lb-container{overflow:hidden}.lightbox .lb-outerContainer .lb-container .lb-image{image-orientation:from-image;border:none;--translate-x: 0;--translate-y: 0;--scale: 1;transform:translateX(var(--translate-x, 0)) translateY(var(--translate-y, 0)) scale(var(--scale, 1))}.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next{color:#fff;text-decoration:none}.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before{position:absolute;top:calc(50% - 1em)}.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev{background:none;font-size:2em}.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before{left:2em}.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next{background:none;font-size:2em}.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before{right:2em}.lightbox .lb-dataContainer{margin:0;padding:0;display:inline-block;vertical-align:top;width:200px;max-width:100%;flex:0 0 auto;overflow:hidden}.lightbox .lb-dataContainer .lb-data{padding:1em 0 0 1em}.lightbox .lb-dataContainer .lb-data .lb-details{float:none}.lightbox .lb-dataContainer .lb-data .lb-details .lb-caption-line{cursor:default;display:block;margin-top:1em}.lightbox .lb-dataContainer .lb-data .lb-details .lb-caption-line:first-child{margin-top:0}.lightbox .lb-dataContainer .lb-data .lb-details .lb-number{padding:0;margin-top:2em}.lightbox .lb-dataContainer .lb-data .lb-closeContainer{display:none !important;margin-top:2em}.lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close{background:none;font-size:2em;float:none}.lightbox .lb-cancel{background:none;font-size:2em;color:#ccc}[data-simplebar]{position:relative;flex-direction:column;flex-wrap:wrap;justify-content:flex-start;align-content:flex-start;align-items:flex-start}.simplebar-wrapper{overflow:hidden;width:inherit;height:inherit;max-width:inherit;max-height:inherit}.simplebar-mask{direction:inherit;position:absolute;overflow:hidden;padding:0;margin:0;left:0;top:0;bottom:0;right:0;width:auto !important;height:auto !important;z-index:0}.simplebar-offset{direction:inherit !important;box-sizing:inherit !important;resize:none !important;position:absolute;top:0;left:0;bottom:0;right:0;padding:0;margin:0;-webkit-overflow-scrolling:touch}.simplebar-content-wrapper{direction:inherit;box-sizing:border-box !important;position:relative;display:block;height:100%;width:auto;visibility:visible;max-width:100%;max-height:100%;scrollbar-width:none;-ms-overflow-style:none}.simplebar-content-wrapper::-webkit-scrollbar,.simplebar-hide-scrollbar::-webkit-scrollbar{width:0;height:0}.simplebar-content:before,.simplebar-content:after{content:" ";display:table}.simplebar-placeholder{max-height:100%;max-width:100%;width:100%;pointer-events:none}.simplebar-height-auto-observer-wrapper{box-sizing:inherit !important;height:100%;width:100%;max-width:1px;position:relative;float:left;max-height:1px;overflow:hidden;z-index:-1;padding:0;margin:0;pointer-events:none;flex-grow:inherit;flex-shrink:0;flex-basis:0}.simplebar-height-auto-observer{box-sizing:inherit;display:block;opacity:0;position:absolute;top:0;left:0;height:1000%;width:1000%;min-height:1px;min-width:1px;overflow:hidden;pointer-events:none;z-index:-1}.simplebar-track{z-index:1;position:absolute;right:0;bottom:0;pointer-events:none;overflow:hidden}[data-simplebar].simplebar-dragging .simplebar-content{pointer-events:none;user-select:none;-webkit-user-select:none}[data-simplebar].simplebar-dragging .simplebar-track{pointer-events:all}.simplebar-scrollbar{position:absolute;right:2px;width:7px;min-height:10px}.simplebar-scrollbar:before{position:absolute;content:"";background:#000;border-radius:7px;left:0;right:0;opacity:0;transition:opacity .2s linear}.simplebar-scrollbar.simplebar-visible:before{opacity:.5;transition:opacity 0s linear}.simplebar-track.simplebar-vertical{top:0;width:11px}.simplebar-track.simplebar-vertical .simplebar-scrollbar:before{top:2px;bottom:2px}.simplebar-track.simplebar-horizontal{left:0;height:11px}.simplebar-track.simplebar-horizontal .simplebar-scrollbar:before{height:100%;left:2px;right:2px}.simplebar-track.simplebar-horizontal .simplebar-scrollbar{right:auto;left:0;top:2px;height:7px;min-height:0;min-width:10px;width:auto}[data-simplebar-direction=rtl] .simplebar-track.simplebar-vertical{right:auto;left:0}.hs-dummy-scrollbar-size{direction:rtl;position:fixed;opacity:0;visibility:hidden;height:500px;width:500px;overflow-y:hidden;overflow-x:scroll}.simplebar-hide-scrollbar{position:fixed;left:0;visibility:hidden;overflow-y:scroll;scrollbar-width:none;-ms-overflow-style:none}.leaflet-pane,.leaflet-tile,.leaflet-marker-icon,.leaflet-marker-shadow,.leaflet-tile-container,.leaflet-pane>svg,.leaflet-pane>canvas,.leaflet-zoom-box,.leaflet-image-layer,.leaflet-layer{position:absolute;left:0;top:0}.leaflet-container{overflow:hidden}.leaflet-tile,.leaflet-marker-icon,.leaflet-marker-shadow{-webkit-user-select:none;-moz-user-select:none;user-select:none;-webkit-user-drag:none}.leaflet-tile::selection{background:transparent}.leaflet-safari .leaflet-tile{image-rendering:-webkit-optimize-contrast}.leaflet-safari .leaflet-tile-container{width:1600px;height:1600px;-webkit-transform-origin:0 0}.leaflet-marker-icon,.leaflet-marker-shadow{display:block}.leaflet-container .leaflet-overlay-pane svg,.leaflet-container .leaflet-marker-pane img,.leaflet-container .leaflet-shadow-pane img,.leaflet-container .leaflet-tile-pane img,.leaflet-container img.leaflet-image-layer,.leaflet-container .leaflet-tile{max-width:none !important;max-height:none !important}.leaflet-container.leaflet-touch-zoom{-ms-touch-action:pan-x pan-y;touch-action:pan-x pan-y}.leaflet-container.leaflet-touch-drag{-ms-touch-action:pinch-zoom;touch-action:none;touch-action:pinch-zoom}.leaflet-container.leaflet-touch-drag.leaflet-touch-zoom{-ms-touch-action:none;touch-action:none}.leaflet-container{-webkit-tap-highlight-color:transparent}.leaflet-container a{-webkit-tap-highlight-color:rgba(51,181,229,.4)}.leaflet-tile{filter:inherit;visibility:hidden}.leaflet-tile-loaded{visibility:inherit}.leaflet-zoom-box{width:0;height:0;-moz-box-sizing:border-box;box-sizing:border-box;z-index:800}.leaflet-overlay-pane svg{-moz-user-select:none}.leaflet-pane{z-index:400}.leaflet-tile-pane{z-index:200}.leaflet-overlay-pane{z-index:400}.leaflet-shadow-pane{z-index:500}.leaflet-marker-pane{z-index:600}.leaflet-tooltip-pane{z-index:650}.leaflet-popup-pane{z-index:700}.leaflet-map-pane canvas{z-index:100}.leaflet-map-pane svg{z-index:200}.leaflet-vml-shape{width:1px;height:1px}.lvml{behavior:url(#default#VML);display:inline-block;position:absolute}.leaflet-control{position:relative;z-index:800;pointer-events:visiblePainted;pointer-events:auto}.leaflet-top,.leaflet-bottom{position:absolute;z-index:1000;pointer-events:none}.leaflet-top{top:0}.leaflet-right{right:0}.leaflet-bottom{bottom:0}.leaflet-left{left:0}.leaflet-control{float:left;clear:both}.leaflet-right .leaflet-control{float:right}.leaflet-top .leaflet-control{margin-top:10px}.leaflet-bottom .leaflet-control{margin-bottom:10px}.leaflet-left .leaflet-control{margin-left:10px}.leaflet-right .leaflet-control{margin-right:10px}.leaflet-fade-anim .leaflet-tile{will-change:opacity}.leaflet-fade-anim .leaflet-popup{opacity:0;-webkit-transition:opacity .2s linear;-moz-transition:opacity .2s linear;transition:opacity .2s linear}.leaflet-fade-anim .leaflet-map-pane .leaflet-popup{opacity:1}.leaflet-zoom-animated{-webkit-transform-origin:0 0;-ms-transform-origin:0 0;transform-origin:0 0}.leaflet-zoom-anim .leaflet-zoom-animated{will-change:transform}.leaflet-zoom-anim .leaflet-zoom-animated{-webkit-transition:-webkit-transform .25s cubic-bezier(0, 0, 0.25, 1);-moz-transition:-moz-transform .25s cubic-bezier(0, 0, 0.25, 1);transition:transform .25s cubic-bezier(0, 0, 0.25, 1)}.leaflet-zoom-anim .leaflet-tile,.leaflet-pan-anim .leaflet-tile{-webkit-transition:none;-moz-transition:none;transition:none}.leaflet-zoom-anim .leaflet-zoom-hide{visibility:hidden}.leaflet-interactive{cursor:pointer}.leaflet-grab{cursor:-webkit-grab;cursor:-moz-grab;cursor:grab}.leaflet-crosshair,.leaflet-crosshair .leaflet-interactive{cursor:crosshair}.leaflet-popup-pane,.leaflet-control{cursor:auto}.leaflet-dragging .leaflet-grab,.leaflet-dragging .leaflet-grab .leaflet-interactive,.leaflet-dragging .leaflet-marker-draggable{cursor:move;cursor:-webkit-grabbing;cursor:-moz-grabbing;cursor:grabbing}.leaflet-marker-icon,.leaflet-marker-shadow,.leaflet-image-layer,.leaflet-pane>svg path,.leaflet-tile-container{pointer-events:none}.leaflet-marker-icon.leaflet-interactive,.leaflet-image-layer.leaflet-interactive,.leaflet-pane>svg path.leaflet-interactive,svg.leaflet-image-layer.leaflet-interactive path{pointer-events:visiblePainted;pointer-events:auto}.leaflet-container{background:#ddd;outline:0}.leaflet-container a{color:#0078a8}.leaflet-container a.leaflet-active{outline:2px solid orange}.leaflet-zoom-box{border:2px dotted #38f;background:rgba(255,255,255,.5)}.leaflet-container{font:12px/1.5 "Helvetica Neue",Arial,Helvetica,sans-serif}.leaflet-bar{box-shadow:0 1px 5px rgba(0,0,0,.65);border-radius:4px}.leaflet-bar a,.leaflet-bar a:hover{background-color:#fff;border-bottom:1px solid #ccc;width:26px;height:26px;line-height:26px;display:block;text-align:center;text-decoration:none;color:#000}.leaflet-bar a,.leaflet-control-layers-toggle{background-position:50% 50%;background-repeat:no-repeat;display:block}.leaflet-bar a:hover{background-color:#f4f4f4}.leaflet-bar a:first-child{border-top-left-radius:4px;border-top-right-radius:4px}.leaflet-bar a:last-child{border-bottom-left-radius:4px;border-bottom-right-radius:4px;border-bottom:none}.leaflet-bar a.leaflet-disabled{cursor:default;background-color:#f4f4f4;color:#bbb}.leaflet-touch .leaflet-bar a{width:30px;height:30px;line-height:30px}.leaflet-touch .leaflet-bar a:first-child{border-top-left-radius:2px;border-top-right-radius:2px}.leaflet-touch .leaflet-bar a:last-child{border-bottom-left-radius:2px;border-bottom-right-radius:2px}.leaflet-control-zoom-in,.leaflet-control-zoom-out{font:bold 18px "Lucida Console",Monaco,monospace;text-indent:1px}.leaflet-touch .leaflet-control-zoom-in,.leaflet-touch .leaflet-control-zoom-out{font-size:22px}.leaflet-control-layers{box-shadow:0 1px 5px rgba(0,0,0,.4);background:#fff;border-radius:5px}.leaflet-control-layers-toggle{background-image:url(images/layers.png);width:36px;height:36px}.leaflet-retina .leaflet-control-layers-toggle{background-image:url(images/layers-2x.png);background-size:26px 26px}.leaflet-touch .leaflet-control-layers-toggle{width:44px;height:44px}.leaflet-control-layers .leaflet-control-layers-list,.leaflet-control-layers-expanded .leaflet-control-layers-toggle{display:none}.leaflet-control-layers-expanded .leaflet-control-layers-list{display:block;position:relative}.leaflet-control-layers-expanded{padding:6px 10px 6px 6px;color:#333;background:#fff}.leaflet-control-layers-scrollbar{overflow-y:scroll;overflow-x:hidden;padding-right:5px}.leaflet-control-layers-selector{margin-top:2px;position:relative;top:1px}.leaflet-control-layers label{display:block}.leaflet-control-layers-separator{height:0;border-top:1px solid #ddd;margin:5px -10px 5px -6px}.leaflet-default-icon-path{background-image:url(images/marker-icon.png)}.leaflet-container .leaflet-control-attribution{background:#fff;background:rgba(255,255,255,.7);margin:0}.leaflet-control-attribution,.leaflet-control-scale-line{padding:0 5px;color:#333}.leaflet-control-attribution a{text-decoration:none}.leaflet-control-attribution a:hover{text-decoration:underline}.leaflet-container .leaflet-control-attribution,.leaflet-container .leaflet-control-scale{font-size:11px}.leaflet-left .leaflet-control-scale{margin-left:5px}.leaflet-bottom .leaflet-control-scale{margin-bottom:5px}.leaflet-control-scale-line{border:2px solid #777;border-top:none;line-height:1.1;padding:2px 5px 1px;font-size:11px;white-space:nowrap;overflow:hidden;-moz-box-sizing:border-box;box-sizing:border-box;background:#fff;background:rgba(255,255,255,.5)}.leaflet-control-scale-line:not(:first-child){border-top:2px solid #777;border-bottom:none;margin-top:-2px}.leaflet-control-scale-line:not(:first-child):not(:last-child){border-bottom:2px solid #777}.leaflet-touch .leaflet-control-attribution,.leaflet-touch .leaflet-control-layers,.leaflet-touch .leaflet-bar{box-shadow:none}.leaflet-touch .leaflet-control-layers,.leaflet-touch .leaflet-bar{border:2px solid rgba(0,0,0,.2);background-clip:padding-box}.leaflet-popup{position:absolute;text-align:center;margin-bottom:20px}.leaflet-popup-content-wrapper{padding:1px;text-align:left;border-radius:12px}.leaflet-popup-content{margin:13px 19px;line-height:1.4}.leaflet-popup-content p{margin:18px 0}.leaflet-popup-tip-container{width:40px;height:20px;position:absolute;left:50%;margin-left:-20px;overflow:hidden;pointer-events:none}.leaflet-popup-tip{width:17px;height:17px;padding:1px;margin:-10px auto 0;-webkit-transform:rotate(45deg);-moz-transform:rotate(45deg);-ms-transform:rotate(45deg);transform:rotate(45deg)}.leaflet-popup-content-wrapper,.leaflet-popup-tip{background:#fff;color:#333;box-shadow:0 3px 14px rgba(0,0,0,.4)}.leaflet-container a.leaflet-popup-close-button{position:absolute;top:0;right:0;padding:4px 4px 0 0;border:none;text-align:center;width:18px;height:14px;font:16px/14px Tahoma,Verdana,sans-serif;color:#c3c3c3;text-decoration:none;font-weight:bold;background:transparent}.leaflet-container a.leaflet-popup-close-button:hover{color:#999}.leaflet-popup-scrolled{overflow:auto;border-bottom:1px solid #ddd;border-top:1px solid #ddd}.leaflet-oldie .leaflet-popup-content-wrapper{zoom:1}.leaflet-oldie .leaflet-popup-tip{width:24px;margin:0 auto;-ms-filter:"progid:DXImageTransform.Microsoft.Matrix(M11=0.70710678, M12=0.70710678, M21=-0.70710678, M22=0.70710678)";filter:progid:DXImageTransform.Microsoft.Matrix(M11=0.70710678, M12=0.70710678, M21=-0.70710678, M22=0.70710678)}.leaflet-oldie .leaflet-popup-tip-container{margin-top:-1px}.leaflet-oldie .leaflet-control-zoom,.leaflet-oldie .leaflet-control-layers,.leaflet-oldie .leaflet-popup-content-wrapper,.leaflet-oldie .leaflet-popup-tip{border:1px solid #999}.leaflet-div-icon{background:#fff;border:1px solid #666}.leaflet-tooltip{position:absolute;padding:6px;background-color:#fff;border:1px solid #fff;border-radius:3px;color:#222;white-space:nowrap;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;pointer-events:none;box-shadow:0 1px 3px rgba(0,0,0,.4)}.leaflet-tooltip.leaflet-clickable{cursor:pointer;pointer-events:auto}.leaflet-tooltip-top:before,.leaflet-tooltip-bottom:before,.leaflet-tooltip-left:before,.leaflet-tooltip-right:before{position:absolute;pointer-events:none;border:6px solid transparent;background:transparent;content:""}.leaflet-tooltip-bottom{margin-top:6px}.leaflet-tooltip-top{margin-top:-6px}.leaflet-tooltip-bottom:before,.leaflet-tooltip-top:before{left:50%;margin-left:-6px}.leaflet-tooltip-top:before{bottom:0;margin-bottom:-12px;border-top-color:#fff}.leaflet-tooltip-bottom:before{top:0;margin-top:-12px;margin-left:-6px;border-bottom-color:#fff}.leaflet-tooltip-left{margin-left:-6px}.leaflet-tooltip-right{margin-left:6px}.leaflet-tooltip-left:before,.leaflet-tooltip-right:before{top:50%;margin-top:-6px}.leaflet-tooltip-left:before{right:0;margin-right:-12px;border-left-color:#fff}.leaflet-tooltip-right:before{left:0;margin-left:-12px;border-right-color:#fff}.spot-theme.leaflet-control.elevation .background{margin:6px 0 -12px}.spot-theme.leaflet-control.elevation .axis path,.spot-theme.leaflet-control.elevation .axis line{fill:none;stroke:#999;stroke-width:2}.spot-theme.leaflet-control.elevation .mouse-focus-label-x{text-anchor:middle}.spot-theme.leaflet-control.elevation .mouse-drag{fill:rgba(255,255,255,.2)}.spot-theme.leaflet-control.elevation .elevation-toggle{cursor:pointer;width:44px;height:44px;color:#ccc;text-shadow:0px 1px 1px rgba(0,0,0,.8)}.spot-theme.leaflet-control.elevation .area{fill:#ccc;filter:drop-shadow(-1px 1px 1px rgba(0, 0, 0, 0.6));-webkit-filter:drop-shadow(-1px 1px 1px rgba(0, 0, 0, 0.6))}.spot-theme.leaflet-control.elevation .mouse-focus-line{pointer-events:none;stroke-width:1;stroke:#666}.spot-theme.height-focus{stroke:#ccc;fill:#ccc}.spot-theme.height-focus.line{pointer-events:none;stroke-width:2}.spot-theme.height-focus-label{text-anchor:middle;fill:#ccc}.leaflet-container{background:none}.spot-theme.height-focus,.spot-theme.height-focus.line,.spot-theme.height-focus-label,.spot-theme.leaflet-control.elevation .area{filter:drop-shadow(-1px 1px 1px rgba(0, 0, 0, 0.6));-webkit-filter:drop-shadow(-1px 1px 1px rgba(0, 0, 0, 0.6))}.spot-theme.leaflet-control.elevation-collapsed .background{display:none}.spot-theme .details text{text-anchor:middle}#projects.with-feed #submap{width:calc(100% - 30%);min-width:calc(100% - 400px + 3 * 1rem)}#projects.with-feed .leaflet-right{width:calc(30%);max-width:calc(400px + 3 * 1rem)}#projects.with-feed #feed{z-index:999;transition:none}#projects:not(.with-feed) #feed #posts{right:-100%}#projects.with-settings #submap{width:calc(100% - 30%);min-width:calc(100% - 400px + 3 * 1rem)}#projects.with-settings .leaflet-left{width:calc(30%);max-width:calc(400px + 3 * 1rem)}#projects.with-settings #settings{z-index:999;transition:none}#projects:not(.with-settings) #settings #settings-sections{left:-100%}#projects.with-feed.with-settings #submap{width:calc(100% - 30% * 2);min-width:calc(100% - 400px + 3 * 1rem * 2)}#projects #submap{position:absolute;left:0;top:0;bottom:0;width:100%}#projects #submap .loader{position:absolute;font-size:3em;top:calc(50% - 0.5em);left:calc(50% - 0.66666em);color:#ccc}#projects #map{position:absolute;left:0;top:0;bottom:0;width:100%}#projects #map .track_tooltip p{margin:0}#projects #map .track_tooltip p.name{font-weight:bold;font-size:1.2em}#projects #map .track_tooltip p.description{font-style:italic}#projects #map .track_tooltip p.detail{margin-top:1em;width:50%;display:inline-block}#projects .leaflet-control{background-color:rgba(255,255,255,.6);font-family:Roboto,Arial,sans-serif;border-radius:3px;border:none;margin:1rem}#projects .leaflet-control+.leaflet-control{margin-top:0}#projects .leaflet-control.leaflet-control-scale{padding:.5em}#projects .leaflet-control.leaflet-control-scale .leaflet-control-scale-line{background:none}#projects .leaflet-right,#projects .leaflet-left{transition:all .5s;width:0;max-width:0}#projects .leaflet-right .leaflet-control{left:-100%}#projects .leaflet-left .leaflet-control{right:-100%}#projects .leaflet-top.leaflet-left .leaflet-control-layers{display:none}#projects #legend .track{white-space:nowrap}#projects #legend .track .line{width:2em;height:4px;display:inline-block;border-radius:2px;vertical-align:middle}#projects #legend .track .line.main{background-color:#00ff78}#projects #legend .track .line.off-track{background-color:blue}#projects #legend .track .line.hitchhiking{background-color:#ff7814}#projects #legend .track .desc{font-size:1em;margin-left:.5em;color:#333}#projects #post-button,#projects #settings-button{cursor:pointer;text-shadow:0px 1px 1px rgba(0,0,0,.8);width:44px;text-align:center}#projects #post-button:hover .fa,#projects #post-button:hover .control-icon,#projects #post-button:hover .fa,#projects #post-button:hover .control-icon,#projects #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover a.lb-prev,#projects #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover a.lb-next,#projects #post-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button:hover .lb-close,#projects #post-button:hover .lightbox .lb-cancel,.lightbox #projects #post-button:hover .lb-cancel,#projects #post-button:hover .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button:hover .elevation-toggle,#projects #post-button:hover #settings-button .fa,#projects #settings-button #post-button:hover .fa,#projects #post-button:hover #settings-button .control-icon,#projects #settings-button #post-button:hover .control-icon,#projects #post-button:hover #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button:hover a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover #settings-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button:hover a.lb-prev,#projects #post-button:hover #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button:hover a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover #settings-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button:hover a.lb-next,#projects #post-button:hover #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button:hover .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button:hover #settings-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button #post-button:hover .lb-close,#projects #post-button:hover #settings-button .lightbox .lb-cancel,#projects #settings-button .lightbox #post-button:hover .lb-cancel,.lightbox #projects #post-button:hover #settings-button .lb-cancel,.lightbox #projects #settings-button #post-button:hover .lb-cancel,#projects #post-button:hover #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #post-button:hover .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button:hover #settings-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button #post-button:hover .elevation-toggle,#projects #post-button:hover #settings-button .fa,#projects #settings-button #post-button:hover .fa,#projects #post-button:hover #settings-button .control-icon,#projects #settings-button #post-button:hover .control-icon,#projects #post-button:hover #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button:hover a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover #settings-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button:hover a.lb-prev,#projects #post-button:hover #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button:hover a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover #settings-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button:hover a.lb-next,#projects #post-button:hover #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button:hover .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button:hover #settings-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button #post-button:hover .lb-close,#projects #post-button:hover #settings-button .lightbox .lb-cancel,#projects #settings-button .lightbox #post-button:hover .lb-cancel,.lightbox #projects #post-button:hover #settings-button .lb-cancel,.lightbox #projects #settings-button #post-button:hover .lb-cancel,#projects #post-button:hover #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #post-button:hover .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button:hover #settings-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button #post-button:hover .elevation-toggle,#projects #settings-button #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-prev,#projects #settings-button #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-next,#projects #settings-button #post-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #post-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button .lb-close,#projects #settings-button #post-button:hover .lightbox .lb-cancel,#projects #post-button:hover .lightbox #settings-button .lb-cancel,#projects #settings-button #post-button:hover .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #post-button:hover .spot-theme.leaflet-control.elevation-collapsed #settings-button .elevation-toggle,#projects #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover a.lb-prev,#projects #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover a.lb-next,#projects #post-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button:hover .lb-close,#projects #post-button:hover .lightbox .lb-cancel,.lightbox #projects #post-button:hover .lb-cancel,#projects #post-button:hover .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button:hover .elevation-toggle,#projects #settings-button:hover .fa,#projects #settings-button:hover .control-icon,#projects #settings-button:hover #post-button .fa,#projects #post-button #settings-button:hover .fa,#projects #settings-button:hover #post-button .control-icon,#projects #post-button #settings-button:hover .control-icon,#projects #settings-button:hover #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button:hover a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover #post-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button:hover a.lb-prev,#projects #settings-button:hover #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button:hover a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover #post-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button:hover a.lb-next,#projects #settings-button:hover #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button:hover .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button:hover #post-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button #settings-button:hover .lb-close,#projects #settings-button:hover #post-button .lightbox .lb-cancel,#projects #post-button .lightbox #settings-button:hover .lb-cancel,.lightbox #projects #settings-button:hover #post-button .lb-cancel,.lightbox #projects #post-button #settings-button:hover .lb-cancel,#projects #settings-button:hover #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings-button:hover .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button:hover #post-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button #settings-button:hover .elevation-toggle,#projects #settings-button:hover .fa,#projects #settings-button:hover .control-icon,#projects #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover a.lb-prev,#projects #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover a.lb-next,#projects #settings-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button:hover .lb-close,#projects #settings-button:hover .lightbox .lb-cancel,.lightbox #projects #settings-button:hover .lb-cancel,#projects #settings-button:hover .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button:hover .elevation-toggle,#projects #post-button #settings-button:hover .fa,#projects #settings-button:hover #post-button .fa,#projects #post-button #settings-button:hover .control-icon,#projects #settings-button:hover #post-button .control-icon,#projects #post-button #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button:hover a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover #post-button a.lb-prev,#projects #post-button #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button:hover a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover #post-button a.lb-next,#projects #post-button #settings-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #settings-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button #settings-button:hover .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button:hover #post-button .lb-close,#projects #post-button #settings-button:hover .lightbox .lb-cancel,#projects #settings-button:hover .lightbox #post-button .lb-cancel,.lightbox #projects #post-button #settings-button:hover .lb-cancel,.lightbox #projects #settings-button:hover #post-button .lb-cancel,#projects #post-button #settings-button:hover .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #settings-button:hover .spot-theme.leaflet-control.elevation-collapsed #post-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button #settings-button:hover .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button:hover #post-button .elevation-toggle,#projects #settings-button:hover #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button:hover a.lb-prev,#projects #settings-button:hover #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button:hover a.lb-next,#projects #settings-button:hover #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button:hover .lb-close,#projects #settings-button:hover #post-button .lightbox .lb-cancel,#projects #post-button .lightbox #settings-button:hover .lb-cancel,#projects #settings-button:hover #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings-button:hover .elevation-toggle,#projects #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover a.lb-prev,#projects #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover a.lb-next,#projects #settings-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button:hover .lb-close,#projects #settings-button:hover .lightbox .lb-cancel,.lightbox #projects #settings-button:hover .lb-cancel,#projects #settings-button:hover .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button:hover .elevation-toggle{color:#fff}#projects #post-button .fa,#projects #post-button .control-icon,#projects #post-button .fa,#projects #post-button .control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .lb-close,#projects #post-button .lightbox .lb-cancel,.lightbox #projects #post-button .lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .elevation-toggle,#projects #post-button #settings-button .fa,#projects #settings-button #post-button .fa,#projects #post-button #settings-button .control-icon,#projects #settings-button #post-button .control-icon,#projects #post-button #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-prev,#projects #post-button #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-next,#projects #post-button #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button #settings-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button #post-button .lb-close,#projects #post-button #settings-button .lightbox .lb-cancel,#projects #settings-button .lightbox #post-button .lb-cancel,.lightbox #projects #post-button #settings-button .lb-cancel,.lightbox #projects #settings-button #post-button .lb-cancel,#projects #post-button #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #post-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button #settings-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button #post-button .elevation-toggle,#projects #post-button #settings-button .fa,#projects #settings-button #post-button .fa,#projects #post-button #settings-button .control-icon,#projects #settings-button #post-button .control-icon,#projects #post-button #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-prev,#projects #post-button #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-next,#projects #post-button #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button #settings-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button #post-button .lb-close,#projects #post-button #settings-button .lightbox .lb-cancel,#projects #settings-button .lightbox #post-button .lb-cancel,.lightbox #projects #post-button #settings-button .lb-cancel,.lightbox #projects #settings-button #post-button .lb-cancel,#projects #post-button #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #post-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button #settings-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button #post-button .elevation-toggle,#projects #settings-button #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-prev,#projects #settings-button #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-next,#projects #settings-button #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button .lb-close,#projects #settings-button #post-button .lightbox .lb-cancel,#projects #post-button .lightbox #settings-button .lb-cancel,#projects #settings-button #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings-button .elevation-toggle,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .lb-close,#projects #post-button .lightbox .lb-cancel,.lightbox #projects #post-button .lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .elevation-toggle,#projects #settings-button .fa,#projects #settings-button .control-icon,#projects #settings-button #post-button .fa,#projects #post-button #settings-button .fa,#projects #settings-button #post-button .control-icon,#projects #post-button #settings-button .control-icon,#projects #settings-button #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-prev,#projects #settings-button #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-next,#projects #settings-button #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button #post-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button #settings-button .lb-close,#projects #settings-button #post-button .lightbox .lb-cancel,#projects #post-button .lightbox #settings-button .lb-cancel,.lightbox #projects #settings-button #post-button .lb-cancel,.lightbox #projects #post-button #settings-button .lb-cancel,#projects #settings-button #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button #post-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button #settings-button .elevation-toggle,#projects #settings-button .fa,#projects #settings-button .control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .lb-close,#projects #settings-button .lightbox .lb-cancel,.lightbox #projects #settings-button .lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .elevation-toggle,#projects #post-button #settings-button .fa,#projects #settings-button #post-button .fa,#projects #post-button #settings-button .control-icon,#projects #settings-button #post-button .control-icon,#projects #post-button #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-prev,#projects #post-button #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-next,#projects #post-button #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button #settings-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button #post-button .lb-close,#projects #post-button #settings-button .lightbox .lb-cancel,#projects #settings-button .lightbox #post-button .lb-cancel,.lightbox #projects #post-button #settings-button .lb-cancel,.lightbox #projects #settings-button #post-button .lb-cancel,#projects #post-button #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #post-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button #settings-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button #post-button .elevation-toggle,#projects #settings-button #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-prev,#projects #settings-button #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-next,#projects #settings-button #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button .lb-close,#projects #settings-button #post-button .lightbox .lb-cancel,#projects #post-button .lightbox #settings-button .lb-cancel,#projects #settings-button #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings-button .elevation-toggle,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .lb-close,#projects #settings-button .lightbox .lb-cancel,.lightbox #projects #settings-button .lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .elevation-toggle{color:#ccc}#projects a.drill{position:relative;overflow:hidden;text-decoration:none;display:inline-block}#projects a.drill .drill-icon{position:absolute;display:inline-block;top:50%;left:50%;transform:translate(-50%, -50%)}#projects a.drill .drill-icon i{transition:all .3s;cursor:pointer}#projects .fa-stack .fa-message{font-size:32px;text-shadow:rgba(0,0,0,.5) 3px 3px 3px;color:#6dff58}#projects .fa-stack .fa-message-in{font-size:13px;color:#326526;top:1px}#projects .fa-stack .fa-track-start,#projects .fa-stack .fa-track-end{color:#326526;font-size:14px;top:1px}#projects .fa-stack .fa-track-end{color:#ff7814}#projects #feed,#projects #settings{position:absolute;top:0;bottom:0;width:calc(30%);max-width:calc(400px + 3 * 1rem);z-index:-1;transition-property:z-index;transition-duration:.1s;transition-delay:.5s;overflow:hidden}#projects #feed input,#projects #feed textarea,#projects #settings input,#projects #settings textarea{background-color:#fff;color:#333}#projects #feed button,#projects #settings button{background-color:#333;color:rgba(255,255,255,.8)}#projects #feed button:hover,#projects #settings button:hover{background-color:#fff;color:#333}#projects #feed{right:0}#projects #feed #posts{position:absolute;transition:all .5s;top:0;bottom:0;right:0;width:100%}#projects #feed #posts #poster textarea#post{margin-bottom:1em;width:calc(100% - 2em)}#projects #feed #posts #poster input#name{width:calc(100% - 6em)}#projects #feed #posts #poster button#submit{margin-left:1em;margin-bottom:.5em}#projects #feed #posts .post{margin-bottom:1rem;background:rgba(255,255,255,.8);color:#333;border-radius:3px;width:calc(100% - 1rem);box-shadow:2px 2px 3px 0px rgba(0,0,0,.5)}#projects #feed #posts .post:first-child{margin-top:1rem}#projects #feed #posts .post .message{margin:.3em 0 0 0}#projects #feed #posts .post .signature{margin:.5em 0 0 0;text-align:right;font-style:italic}#projects #feed #posts .post .header{font-style:italic;font-size:.8em;padding:.5em 1em}#projects #feed #posts .post .header span{display:inline-block;width:50%;cursor:default}#projects #feed #posts .post .header span.index{font-style:normal}#projects #feed #posts .post .header span.time{text-align:right}#projects #feed #posts .post .body{clear:both;padding:0em 1em .5em}#projects #feed #posts .post.headerless .header{display:none}#projects #feed #posts .post.headerless .body{padding-top:.5em}#projects #feed #posts .post.media{background:rgba(255,255,255,.8);color:#333}#projects #feed #posts .post.media a{display:inline-block;width:100%;line-height:0;margin:0}#projects #feed #posts .post.media a.drill{font-size:3em}#projects #feed #posts .post.media a.drill .fa-drill-picture{color:transparent}#projects #feed #posts .post.media a.drill .fa-drill-video{color:rgba(255,255,255,.5)}#projects #feed #posts .post.media a.drill:hover .fa-drill-picture,#projects #feed #posts .post.media a.drill:hover .fa-drill-video{color:rgba(255,255,255,.75)}#projects #feed #posts .post.media a img{width:100%;image-orientation:from-image;outline:none;border-radius:3px}#projects #feed #posts .post.media p{margin:0;text-align:justify}#projects #feed #posts .post.message{background:#6dff58;color:#326526}#projects #feed #posts .post.message p{font-size:.9em;margin:.5em 0}#projects #feed #posts .post.message a{color:#326526}#projects #feed #posts .post.message a.drill .drill-icon{transform:translate(-16px, -32px)}#projects #feed #posts .post.message a.drill .drill-icon .fa-message-in{top:0;left:-1px}#projects #feed #posts .post.message a.drill:hover .fa-message{top:16px}#projects #feed #posts .post.message a.drill:hover .fa-message-in{display:none}#projects #feed #posts .post.message .staticmap{width:100%;border-radius:3px;cursor:pointer}#projects #feed #posts .post.loading .body{text-align:center}#projects #feed #posts .post.loading .body p{display:inline-block;font-size:2em;color:#333}#projects #settings{left:0}#projects #settings #settings-sections{width:calc(100% - 3rem);margin:1rem;padding:1rem;background:#fff;border-radius:3px;box-shadow:2px 2px 3px 0px rgba(0,0,0,.5);position:absolute;transition:all .5s;top:0;bottom:0;left:0;color:#333;background:rgba(255,255,255,.8)}#projects #settings #settings-sections .settings-section h1{margin:1.5rem 0 1rem}#projects #settings #settings-sections .settings-section:first-child h1{margin-top:0}#projects #settings #settings-sections .settings-section label{margin-bottom:.3em;display:block;cursor:pointer}#projects #settings #settings-sections .settings-section.newsletter input#email{width:calc(100% - 6em)}#projects #settings #settings-sections .settings-section.newsletter input#email:disabled{color:#999;background:rgba(255,255,255,.2)}#projects #settings #settings-sections .settings-section.newsletter button#nl_btn{margin-left:1em;margin-bottom:1em}#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.loading{background-color:#326526;color:#fff}#projects #settings #settings-sections .settings-section #settings-projects a.fa-download{color:#333}#projects #settings #settings-sections .settings-section #settings-projects a.fa-download:hover{color:#0078a8}#projects .leaflet-popup-content{margin:0}#projects .leaflet-popup-content .info-window{margin:1rem}#projects .leaflet-popup-content .info-window h1{font-size:1.2em;margin:1em 0 1.2em}#projects .leaflet-popup-content .info-window h1 i{margin-right:.3125em}#projects .leaflet-popup-content .info-window p{font-size:1em;margin:.5em 0 0 0}#projects .leaflet-popup-content .info-window p i{padding-right:.5em}#projects .leaflet-popup-content .info-window p a{color:#333}#projects .leaflet-popup-content .info-window .medias{margin-top:-0.5rem;line-height:0}#projects .leaflet-popup-content .info-window .medias a{display:inline-block;margin-right:1rem;margin-top:1rem}#projects .leaflet-popup-content .info-window .medias a:last-child{margin-right:0}#projects .leaflet-popup-content .info-window .medias a.drill{font-size:2em}#projects .leaflet-popup-content .info-window .medias a.drill .fa-drill-picture{color:transparent}#projects .leaflet-popup-content .info-window .medias a.drill .fa-drill-video{color:rgba(255,255,255,.5)}#projects .leaflet-popup-content .info-window .medias a.drill:hover .fa-drill-video,#projects .leaflet-popup-content .info-window .medias a.drill:hover .fa-drill-picture{color:rgba(255,255,255,.75)}#projects .leaflet-popup-content .info-window .medias a img{max-width:200px;max-height:100px;border-radius:3px;image-orientation:from-image;transition:All .2s}#elems{display:none}#upload{padding:1em}#upload h1{font-size:2em;border-bottom:2px solid #000;margin:0 0 1em 0;padding-bottom:.5em}#upload .bar{height:18px;background:green}#upload .comment{margin-top:1em}#upload .comment .thumb{width:30%;max-width:100px}#upload .comment form{display:inline-block;width:calc(70% - 1em);min-width:calc(100% - 100px - 1em);margin-left:1em;vertical-align:top}#upload .comment form .content{width:100%;box-sizing:border-box;padding:.5em}#upload .comment form .save{margin-top:1em;padding:.5em}#admin{margin:1em}#admin table{margin-bottom:1em;border-collapse:collapse}#admin table tr th{background:#aaa;color:#fff;padding:.2rem .5rem}#admin table tr td{background:#eee;text-align:center;padding:.2rem .5rem}#admin table tr td input[type=number]{width:50px}#admin table tr td input[name=ref_feed_id]{width:300px}@media only screen and (max-width: 800px){.desktop{display:none}#projects.with-feed #submap,#projects.with-settings #submap{width:100%}#projects.with-feed .leaflet-right,#projects.with-feed .leaflet-left,#projects.with-settings .leaflet-right,#projects.with-settings .leaflet-left{width:calc(100% - 44px - 2 * 1rem)}#projects.with-feed .leaflet-control-container .leaflet-bottom.leaflet-right,#projects.with-settings .leaflet-control-container .leaflet-bottom.leaflet-right{display:none}#projects .leaflet-control-container .leaflet-bottom.leaflet-left,#projects .leaflet-control-container .leaflet-bottom.leaflet-right .leaflet-control.elevation{display:none}#projects #feed,#projects #settings{width:calc(100% - 44px - 2 * 1rem)}}@media only screen and (min-width: 801px){.mobile{display:none}}/*# sourceMappingURL=spot.css.map */ + */@font-face{font-family:"Font Awesome 5 Pro";font-style:normal;font-weight:900;font-display:auto;src:url("fa/fonts/fa-solid-900.eot");src:url("fa/fonts/fa-solid-900.eot?#iefix") format("embedded-opentype"),url("fa/fonts/fa-solid-900.woff2") format("woff2"),url("fa/fonts/fa-solid-900.woff") format("woff"),url("fa/fonts/fa-solid-900.ttf") format("truetype"),url("fa/fonts/fa-solid-900.svg#fontawesome") format("svg")}.fa,.lightbox .lb-cancel,.lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.control-icon,#projects #post-button .fa,#projects #post-button .control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .lb-close,#projects #post-button .lightbox .lb-cancel,.lightbox #projects #post-button .lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .elevation-toggle,#projects #settings-button .fa,#projects #settings-button .control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .lb-close,#projects #settings-button .lightbox .lb-cancel,.lightbox #projects #settings-button .lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .elevation-toggle,#projects #post-button #settings-button .fa,#projects #settings-button #post-button .fa,#projects #post-button #settings-button .control-icon,#projects #settings-button #post-button .control-icon,.spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.fas{font-family:"Font Awesome 5 Pro";font-weight:900}.fa,.lightbox .lb-cancel,.lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.control-icon,#projects #post-button .fa,#projects #post-button .control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .lb-close,#projects #post-button .lightbox .lb-cancel,.lightbox #projects #post-button .lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .elevation-toggle,#projects #settings-button .fa,#projects #settings-button .control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .lb-close,#projects #settings-button .lightbox .lb-cancel,.lightbox #projects #settings-button .lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .elevation-toggle,#projects #post-button #settings-button .fa,#projects #settings-button #post-button .fa,#projects #post-button #settings-button .control-icon,#projects #settings-button #post-button .control-icon,.spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.fas,.far,.fal,.fad,.fab{-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;display:inline-block;font-style:normal;font-variant:normal;text-rendering:auto;line-height:1}.fa-lg{font-size:1.3333333333em;line-height:.75em;vertical-align:-0.0667em}.fa-xs{font-size:.75em}.fa-sm{font-size:.875em}.fa-1x{font-size:1em}.fa-2x{font-size:2em}.fa-3x{font-size:3em}.fa-4x{font-size:4em}.fa-5x{font-size:5em}.fa-6x{font-size:6em}.fa-7x{font-size:7em}.fa-8x{font-size:8em}.fa-9x{font-size:9em}.fa-10x{font-size:10em}.fa-fw{text-align:center;width:1.25em}.fa-ul{list-style-type:none;margin-left:2.5em;padding-left:0}.fa-ul>li{position:relative}.fa-li{left:-2em;position:absolute;text-align:center;width:2em;line-height:inherit}.fa-border{border:solid .08em #eee;border-radius:.1em;padding:.2em .25em .15em}.fa-pull-left{float:left}.fa-pull-right{float:right}.fa.fa-pull-left,.lightbox .fa-pull-left.lb-cancel,.lightbox .lb-dataContainer .lb-data .lb-closeContainer .fa-pull-left.lb-close,.lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-left.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-left.lb-prev,.fa-pull-left.control-icon,#projects #post-button .fa-pull-left.fa,#projects #post-button .fa-pull-left.control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-left.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.fa-pull-left.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-left.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.fa-pull-left.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .fa-pull-left.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .fa-pull-left.lb-close,#projects #post-button .lightbox .fa-pull-left.lb-cancel,.lightbox #projects #post-button .fa-pull-left.lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .fa-pull-left.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .fa-pull-left.elevation-toggle,#projects #settings-button .fa-pull-left.fa,#projects #settings-button .fa-pull-left.control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-left.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.fa-pull-left.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-left.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.fa-pull-left.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .fa-pull-left.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .fa-pull-left.lb-close,#projects #settings-button .lightbox .fa-pull-left.lb-cancel,.lightbox #projects #settings-button .fa-pull-left.lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .fa-pull-left.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .fa-pull-left.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed .fa-pull-left.elevation-toggle,.fas.fa-pull-left,.far.fa-pull-left,.fal.fa-pull-left,.fab.fa-pull-left{margin-right:.3em}.fa.fa-pull-right,.lightbox .fa-pull-right.lb-cancel,.lightbox .lb-dataContainer .lb-data .lb-closeContainer .fa-pull-right.lb-close,.lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-right.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-right.lb-prev,.fa-pull-right.control-icon,#projects #post-button .fa-pull-right.fa,#projects #post-button .fa-pull-right.control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-right.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.fa-pull-right.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-right.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.fa-pull-right.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .fa-pull-right.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .fa-pull-right.lb-close,#projects #post-button .lightbox .fa-pull-right.lb-cancel,.lightbox #projects #post-button .fa-pull-right.lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .fa-pull-right.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .fa-pull-right.elevation-toggle,#projects #settings-button .fa-pull-right.fa,#projects #settings-button .fa-pull-right.control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-right.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.fa-pull-right.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.fa-pull-right.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.fa-pull-right.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .fa-pull-right.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .fa-pull-right.lb-close,#projects #settings-button .lightbox .fa-pull-right.lb-cancel,.lightbox #projects #settings-button .fa-pull-right.lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .fa-pull-right.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .fa-pull-right.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed .fa-pull-right.elevation-toggle,.fas.fa-pull-right,.far.fa-pull-right,.fal.fa-pull-right,.fab.fa-pull-right{margin-left:.3em}.fa-spin{animation:fa-spin 2s infinite linear}.fa-pulse{animation:fa-spin 1s infinite steps(8)}@keyframes fa-spin{0%{transform:rotate(0deg)}100%{transform:rotate(360deg)}}.fa-rotate-90{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=1)";transform:rotate(90deg)}.fa-rotate-180{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=2)";transform:rotate(180deg)}.fa-rotate-270{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=3)";transform:rotate(270deg)}.fa-flip-horizontal{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=0, mirror=1)";transform:scale(-1, 1)}.fa-flip-vertical{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=2, mirror=1)";transform:scale(1, -1)}.fa-flip-both,.fa-flip-horizontal.fa-flip-vertical{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=2, mirror=1)";transform:scale(-1, -1)}:root .fa-rotate-90,:root .fa-rotate-180,:root .fa-rotate-270,:root .fa-flip-horizontal,:root .fa-flip-vertical,:root .fa-flip-both{filter:none}.fa-stack{display:inline-block;height:2em;line-height:2em;position:relative;vertical-align:middle;width:2.5em}.fa-stack-1x,.fa-stack-2x{left:0;position:absolute;text-align:center;width:100%}.fa-stack-1x{line-height:inherit}.fa-stack-2x{font-size:2em}.fa-inverse{color:#fff}.fa.push,.lightbox .push.lb-cancel,.lightbox .lb-dataContainer .lb-data .lb-closeContainer .push.lb-close,.lightbox .lb-outerContainer .lb-container .lb-nav a.push.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav a.push.lb-prev,.push.control-icon,#projects #post-button .push.fa,#projects #post-button .push.control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.push.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.push.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .push.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .push.lb-close,#projects #post-button .lightbox .push.lb-cancel,.lightbox #projects #post-button .push.lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .push.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .push.elevation-toggle,#projects #settings-button .push.fa,#projects #settings-button .push.control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.push.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.push.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .push.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .push.lb-close,#projects #settings-button .lightbox .push.lb-cancel,.lightbox #projects #settings-button .push.lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .push.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .push.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed .push.elevation-toggle{margin-right:.5rem}.fa.push-left,.lightbox .push-left.lb-cancel,.lightbox .lb-dataContainer .lb-data .lb-closeContainer .push-left.lb-close,.lightbox .lb-outerContainer .lb-container .lb-nav a.push-left.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav a.push-left.lb-prev,.push-left.control-icon,#projects #post-button .push-left.fa,#projects #post-button .push-left.control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push-left.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.push-left.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push-left.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.push-left.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .push-left.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .push-left.lb-close,#projects #post-button .lightbox .push-left.lb-cancel,.lightbox #projects #post-button .push-left.lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .push-left.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .push-left.elevation-toggle,#projects #settings-button .push-left.fa,#projects #settings-button .push-left.control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push-left.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.push-left.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.push-left.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.push-left.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .push-left.lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .push-left.lb-close,#projects #settings-button .lightbox .push-left.lb-cancel,.lightbox #projects #settings-button .push-left.lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .push-left.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .push-left.elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed .push-left.elevation-toggle{margin-left:.5rem}.control-icon,#projects #post-button .fa,#projects #post-button .control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .lb-close,#projects #post-button .lightbox .lb-cancel,.lightbox #projects #post-button .lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .elevation-toggle,#projects #settings-button .fa,#projects #settings-button .control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .lb-close,#projects #settings-button .lightbox .lb-cancel,.lightbox #projects #settings-button .lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .elevation-toggle,#projects #post-button #settings-button .fa,#projects #settings-button #post-button .fa,#projects #post-button #settings-button .control-icon,#projects #settings-button #post-button .control-icon,.spot-theme.leaflet-control.elevation-collapsed .elevation-toggle{font-size:28px;text-align:center;line-height:44px;text-decoration:none;color:#999;background:none}.fa-map:before{content:""}.fa-track-off-track:before{content:""}.fa-track-main:before{content:""}.fa-track-hitchhiking:before{content:""}.fa-track-start:before{content:""}.fa-track-end:before{content:""}.fa-layers:before{content:""}.fa-elev-chart:before,.spot-theme.leaflet-control.elevation-collapsed .elevation-toggle:before{content:""}.fa-distance:before{content:""}.fa-elev-drop:before{content:""}.fa-elev-gain:before{content:""}.fa-download:before{content:""}.fa-menu:before,#projects #settings-button .fa:before,#projects #settings-button .control-icon:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-prev:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-next:before,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close:before,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .lb-close:before,#projects #settings-button .lightbox .lb-cancel:before,.lightbox #projects #settings-button .lb-cancel:before,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle:before,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .elevation-toggle:before,#projects #settings-button #post-button .fa:before,#projects #post-button #settings-button .fa:before,#projects #settings-button #post-button .control-icon:before,#projects #post-button #settings-button .control-icon:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-prev:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-next:before,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button .lb-close:before,#projects #post-button .lightbox #settings-button .lb-cancel:before,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings-button .elevation-toggle:before{content:""}.fa-newsletter:before{content:""}.fa-project:before{content:""}.fa-error:before{content:""}.fa-warning:before{content:""}.fa-success:before{content:""}.fa-unsubscribe:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .fa:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .control-icon:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe a.lb-prev:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe a.lb-next:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close:before,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lb-close:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lightbox .lb-cancel:before,.lightbox #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lb-cancel:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle:before,.spot-theme.leaflet-control.elevation-collapsed #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .elevation-toggle:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe a.lb-prev:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe a.lb-next:before,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lb-close:before,#projects #post-button .lightbox #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lb-cancel:before,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .elevation-toggle:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe a.lb-prev:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe a.lb-next:before,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lb-close:before,#projects #settings-button .lightbox #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .lb-cancel:before,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #settings #settings-sections .settings-section.newsletter button#nl_btn.unsubscribe .elevation-toggle:before{content:""}.fa-post:before,#projects #post-button .fa:before,#projects #post-button .control-icon:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-prev:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-next:before,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close:before,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .lb-close:before,#projects #post-button .lightbox .lb-cancel:before,.lightbox #projects #post-button .lb-cancel:before,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle:before,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .elevation-toggle:before,#projects #post-button #settings-button .fa:before,#projects #settings-button #post-button .fa:before,#projects #post-button #settings-button .control-icon:before,#projects #settings-button #post-button .control-icon:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-prev:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-next:before,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button .lb-close:before,#projects #settings-button .lightbox #post-button .lb-cancel:before,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #post-button .elevation-toggle:before{content:""}.fa-media:before{content:""}.fa-video:before{content:""}.fa-image:before{content:""}.fa-message:before{content:""}.fa-message-in:before{content:""}.fa-time:before{content:""}.fa-coords:before{content:""}.fa-drill-video:before{content:""}.fa-drill-picture:before{content:""}.fa-drill-message:before,#projects #feed #posts .post.message a.drill:hover .fa-message:before{content:""}.fa-upload:before{content:""}.fa-video-shot:before{content:""}.fa-image-shot:before{content:""}.fa-poster:before{content:""}.fa-send:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .fa:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .control-icon:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe a.lb-prev:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe a.lb-next:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close:before,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lb-close:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lightbox .lb-cancel:before,.lightbox #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lb-cancel:before,#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle:before,.spot-theme.leaflet-control.elevation-collapsed #projects #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .elevation-toggle:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe a.lb-prev:before,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe a.lb-next:before,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lb-close:before,#projects #post-button .lightbox #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lb-cancel:before,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .elevation-toggle:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe a.lb-prev:before,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe a.lb-next:before,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lb-close:before,#projects #settings-button .lightbox #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .lb-cancel:before,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #settings #settings-sections .settings-section.newsletter button#nl_btn.subscribe .elevation-toggle:before{content:""}.fa-cancel:before,.lightbox .lb-cancel:before{content:""}.fa-prev:before,#projects.with-settings #settings-button .fa:before,#projects.with-settings #settings-button .control-icon:before,#projects.with-settings #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-prev:before,#projects.with-settings #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-next:before,#projects.with-settings #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button .lb-close:before,#projects.with-settings #post-button .lightbox #settings-button .lb-cancel:before,#projects.with-settings #post-button .spot-theme.leaflet-control.elevation-collapsed #settings-button .elevation-toggle:before,#projects.with-settings #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects.with-settings #settings-button a.lb-prev:before,#projects.with-settings #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects.with-settings #settings-button a.lb-next:before,#projects.with-settings #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close:before,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects.with-settings #settings-button .lb-close:before,#projects.with-settings #settings-button .lightbox .lb-cancel:before,.lightbox #projects.with-settings #settings-button .lb-cancel:before,#projects.with-settings #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle:before,.spot-theme.leaflet-control.elevation-collapsed #projects.with-settings #settings-button .elevation-toggle:before,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before{content:""}.fa-next:before,#projects.with-feed #post-button .fa:before,#projects.with-feed #post-button .control-icon:before,#projects.with-feed #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects.with-feed #post-button a.lb-prev:before,#projects.with-feed #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before,.lightbox .lb-outerContainer .lb-container .lb-nav #projects.with-feed #post-button a.lb-next:before,#projects.with-feed #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close:before,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects.with-feed #post-button .lb-close:before,#projects.with-feed #post-button .lightbox .lb-cancel:before,.lightbox #projects.with-feed #post-button .lb-cancel:before,#projects.with-feed #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle:before,.spot-theme.leaflet-control.elevation-collapsed #projects.with-feed #post-button .elevation-toggle:before,#projects.with-feed #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-prev:before,#projects.with-feed #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-next:before,#projects.with-feed #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button .lb-close:before,#projects.with-feed #settings-button .lightbox #post-button .lb-cancel:before,#projects.with-feed #settings-button .spot-theme.leaflet-control.elevation-collapsed #post-button .elevation-toggle:before,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before{content:""}.fa-close:before,.lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close:before{content:""}body.lb-disable-scrolling{overflow:hidden}.lightboxOverlay{position:absolute;top:0;left:0;z-index:9999;background-color:#000;filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=80);opacity:.8;display:none}.lightbox{position:absolute;left:0;width:100%;z-index:10000;text-align:center;line-height:0;font-weight:normal;outline:none}.lightbox .lb-image{display:block;height:auto;max-width:inherit;max-height:none;border-radius:3px;border:4px solid #fff}.lightbox .lb-video{box-sizing:content-box}.lightbox .lb-video-nav .lb-nav{height:calc(100% - 45px)}.lightbox a img{border:none}.lb-outerContainer{position:relative;*zoom:1;width:250px;height:250px;margin:0 auto;border-radius:4px;background-color:#fff}.lb-outerContainer:after{content:"";display:table;clear:both}.lb-loader{position:absolute;top:43%;left:0;height:25%;width:100%;text-align:center;line-height:0}.lb-cancel{display:block;width:32px;height:32px;margin:0 auto;background:url(../images/loading.gif) no-repeat}.lb-nav{position:absolute;top:0;left:0;height:100%;width:100%;z-index:10}.lb-container>.nav{left:0}.lb-nav a{outline:none;background-image:url("")}.lb-prev,.lb-next{height:100%;cursor:pointer;display:block}.lb-nav a.lb-prev{width:34%;left:0;float:left;background:url(../images/prev.png) left 48% no-repeat;filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=0);opacity:0;-webkit-transition:opacity .6s;-moz-transition:opacity .6s;-o-transition:opacity .6s;transition:opacity .6s}.lb-nav a.lb-prev:hover{filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=100);opacity:1}.lb-nav a.lb-next{width:64%;right:0;float:right;background:url(../images/next.png) right 48% no-repeat;filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=0);opacity:0;-webkit-transition:opacity .6s;-moz-transition:opacity .6s;-o-transition:opacity .6s;transition:opacity .6s}.lb-nav a.lb-next:hover{filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=100);opacity:1}.lb-dataContainer{margin:0 auto;padding-top:5px;*zoom:1;width:100%;border-bottom-left-radius:4px;border-bottom-right-radius:4px}.lb-dataContainer:after{content:"";display:table;clear:both}.lb-data{padding:0 4px;color:#ccc}.lb-data .lb-details{width:85%;float:left;text-align:left;line-height:1.1em}.lb-data .lb-caption{font-size:13px;font-weight:bold;line-height:1em}.lb-data .lb-caption a{color:#4ae}.lb-data .lb-number{display:block;clear:left;padding-bottom:1em;font-size:12px;color:#999}.lb-data .lb-close{display:block;float:right;width:30px;height:30px;background:url(../images/close.png) top right no-repeat;text-align:right;outline:none;filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=70);opacity:.7;-webkit-transition:opacity .2s;-moz-transition:opacity .2s;-o-transition:opacity .2s;transition:opacity .2s}.lb-data .lb-close:hover{cursor:pointer;filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=100);opacity:1}.lightboxOverlay{bottom:0;right:0}.lightbox{display:flex;align-items:center;justify-content:center;width:100%;height:100%}.lightbox .lb-outerContainer{margin:0;border-radius:0;background-color:rgba(255,255,255,.5)}.lightbox .lb-outerContainer .lb-container{overflow:hidden}.lightbox .lb-outerContainer .lb-container .lb-image{image-orientation:from-image;border:none;--translate-x: 0;--translate-y: 0;--scale: 1;transform:translateX(var(--translate-x, 0)) translateY(var(--translate-y, 0)) scale(var(--scale, 1))}.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next{color:#fff;text-decoration:none}.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before,.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before{position:absolute;top:calc(50% - 1em)}.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev{background:none;font-size:2em}.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev:before{left:2em}.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next{background:none;font-size:2em}.lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next:before{right:2em}.lightbox .lb-dataContainer{margin:0;padding:0;display:inline-block;vertical-align:top;width:200px;max-width:100%;flex:0 0 auto;overflow:hidden}.lightbox .lb-dataContainer .lb-data{padding:1em 0 0 1em}.lightbox .lb-dataContainer .lb-data .lb-details{float:none}.lightbox .lb-dataContainer .lb-data .lb-details .lb-caption-line{cursor:default;display:block;margin-top:1em}.lightbox .lb-dataContainer .lb-data .lb-details .lb-caption-line:first-child{margin-top:0}.lightbox .lb-dataContainer .lb-data .lb-details .lb-number{padding:0;margin-top:2em}.lightbox .lb-dataContainer .lb-data .lb-closeContainer{display:none !important;margin-top:2em}.lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close{background:none;font-size:2em;float:none}.lightbox .lb-cancel{background:none;font-size:2em;color:#ccc}[data-simplebar]{position:relative;flex-direction:column;flex-wrap:wrap;justify-content:flex-start;align-content:flex-start;align-items:flex-start}.simplebar-wrapper{overflow:hidden;width:inherit;height:inherit;max-width:inherit;max-height:inherit}.simplebar-mask{direction:inherit;position:absolute;overflow:hidden;padding:0;margin:0;left:0;top:0;bottom:0;right:0;width:auto !important;height:auto !important;z-index:0}.simplebar-offset{direction:inherit !important;box-sizing:inherit !important;resize:none !important;position:absolute;top:0;left:0;bottom:0;right:0;padding:0;margin:0;-webkit-overflow-scrolling:touch}.simplebar-content-wrapper{direction:inherit;box-sizing:border-box !important;position:relative;display:block;height:100%;width:auto;visibility:visible;max-width:100%;max-height:100%;scrollbar-width:none;-ms-overflow-style:none}.simplebar-content-wrapper::-webkit-scrollbar,.simplebar-hide-scrollbar::-webkit-scrollbar{width:0;height:0}.simplebar-content:before,.simplebar-content:after{content:" ";display:table}.simplebar-placeholder{max-height:100%;max-width:100%;width:100%;pointer-events:none}.simplebar-height-auto-observer-wrapper{box-sizing:inherit !important;height:100%;width:100%;max-width:1px;position:relative;float:left;max-height:1px;overflow:hidden;z-index:-1;padding:0;margin:0;pointer-events:none;flex-grow:inherit;flex-shrink:0;flex-basis:0}.simplebar-height-auto-observer{box-sizing:inherit;display:block;opacity:0;position:absolute;top:0;left:0;height:1000%;width:1000%;min-height:1px;min-width:1px;overflow:hidden;pointer-events:none;z-index:-1}.simplebar-track{z-index:1;position:absolute;right:0;bottom:0;pointer-events:none;overflow:hidden}[data-simplebar].simplebar-dragging .simplebar-content{pointer-events:none;user-select:none;-webkit-user-select:none}[data-simplebar].simplebar-dragging .simplebar-track{pointer-events:all}.simplebar-scrollbar{position:absolute;right:2px;width:7px;min-height:10px}.simplebar-scrollbar:before{position:absolute;content:"";background:#000;border-radius:7px;left:0;right:0;opacity:0;transition:opacity .2s linear}.simplebar-scrollbar.simplebar-visible:before{opacity:.5;transition:opacity 0s linear}.simplebar-track.simplebar-vertical{top:0;width:11px}.simplebar-track.simplebar-vertical .simplebar-scrollbar:before{top:2px;bottom:2px}.simplebar-track.simplebar-horizontal{left:0;height:11px}.simplebar-track.simplebar-horizontal .simplebar-scrollbar:before{height:100%;left:2px;right:2px}.simplebar-track.simplebar-horizontal .simplebar-scrollbar{right:auto;left:0;top:2px;height:7px;min-height:0;min-width:10px;width:auto}[data-simplebar-direction=rtl] .simplebar-track.simplebar-vertical{right:auto;left:0}.hs-dummy-scrollbar-size{direction:rtl;position:fixed;opacity:0;visibility:hidden;height:500px;width:500px;overflow-y:hidden;overflow-x:scroll}.simplebar-hide-scrollbar{position:fixed;left:0;visibility:hidden;overflow-y:scroll;scrollbar-width:none;-ms-overflow-style:none}.leaflet-pane,.leaflet-tile,.leaflet-marker-icon,.leaflet-marker-shadow,.leaflet-tile-container,.leaflet-pane>svg,.leaflet-pane>canvas,.leaflet-zoom-box,.leaflet-image-layer,.leaflet-layer{position:absolute;left:0;top:0}.leaflet-container{overflow:hidden}.leaflet-tile,.leaflet-marker-icon,.leaflet-marker-shadow{-webkit-user-select:none;-moz-user-select:none;user-select:none;-webkit-user-drag:none}.leaflet-tile::selection{background:transparent}.leaflet-safari .leaflet-tile{image-rendering:-webkit-optimize-contrast}.leaflet-safari .leaflet-tile-container{width:1600px;height:1600px;-webkit-transform-origin:0 0}.leaflet-marker-icon,.leaflet-marker-shadow{display:block}.leaflet-container .leaflet-overlay-pane svg,.leaflet-container .leaflet-marker-pane img,.leaflet-container .leaflet-shadow-pane img,.leaflet-container .leaflet-tile-pane img,.leaflet-container img.leaflet-image-layer,.leaflet-container .leaflet-tile{max-width:none !important;max-height:none !important}.leaflet-container.leaflet-touch-zoom{-ms-touch-action:pan-x pan-y;touch-action:pan-x pan-y}.leaflet-container.leaflet-touch-drag{-ms-touch-action:pinch-zoom;touch-action:none;touch-action:pinch-zoom}.leaflet-container.leaflet-touch-drag.leaflet-touch-zoom{-ms-touch-action:none;touch-action:none}.leaflet-container{-webkit-tap-highlight-color:transparent}.leaflet-container a{-webkit-tap-highlight-color:rgba(51,181,229,.4)}.leaflet-tile{filter:inherit;visibility:hidden}.leaflet-tile-loaded{visibility:inherit}.leaflet-zoom-box{width:0;height:0;-moz-box-sizing:border-box;box-sizing:border-box;z-index:800}.leaflet-overlay-pane svg{-moz-user-select:none}.leaflet-pane{z-index:400}.leaflet-tile-pane{z-index:200}.leaflet-overlay-pane{z-index:400}.leaflet-shadow-pane{z-index:500}.leaflet-marker-pane{z-index:600}.leaflet-tooltip-pane{z-index:650}.leaflet-popup-pane{z-index:700}.leaflet-map-pane canvas{z-index:100}.leaflet-map-pane svg{z-index:200}.leaflet-vml-shape{width:1px;height:1px}.lvml{behavior:url(#default#VML);display:inline-block;position:absolute}.leaflet-control{position:relative;z-index:800;pointer-events:visiblePainted;pointer-events:auto}.leaflet-top,.leaflet-bottom{position:absolute;z-index:1000;pointer-events:none}.leaflet-top{top:0}.leaflet-right{right:0}.leaflet-bottom{bottom:0}.leaflet-left{left:0}.leaflet-control{float:left;clear:both}.leaflet-right .leaflet-control{float:right}.leaflet-top .leaflet-control{margin-top:10px}.leaflet-bottom .leaflet-control{margin-bottom:10px}.leaflet-left .leaflet-control{margin-left:10px}.leaflet-right .leaflet-control{margin-right:10px}.leaflet-fade-anim .leaflet-tile{will-change:opacity}.leaflet-fade-anim .leaflet-popup{opacity:0;-webkit-transition:opacity .2s linear;-moz-transition:opacity .2s linear;transition:opacity .2s linear}.leaflet-fade-anim .leaflet-map-pane .leaflet-popup{opacity:1}.leaflet-zoom-animated{-webkit-transform-origin:0 0;-ms-transform-origin:0 0;transform-origin:0 0}.leaflet-zoom-anim .leaflet-zoom-animated{will-change:transform}.leaflet-zoom-anim .leaflet-zoom-animated{-webkit-transition:-webkit-transform .25s cubic-bezier(0, 0, 0.25, 1);-moz-transition:-moz-transform .25s cubic-bezier(0, 0, 0.25, 1);transition:transform .25s cubic-bezier(0, 0, 0.25, 1)}.leaflet-zoom-anim .leaflet-tile,.leaflet-pan-anim .leaflet-tile{-webkit-transition:none;-moz-transition:none;transition:none}.leaflet-zoom-anim .leaflet-zoom-hide{visibility:hidden}.leaflet-interactive{cursor:pointer}.leaflet-grab{cursor:-webkit-grab;cursor:-moz-grab;cursor:grab}.leaflet-crosshair,.leaflet-crosshair .leaflet-interactive{cursor:crosshair}.leaflet-popup-pane,.leaflet-control{cursor:auto}.leaflet-dragging .leaflet-grab,.leaflet-dragging .leaflet-grab .leaflet-interactive,.leaflet-dragging .leaflet-marker-draggable{cursor:move;cursor:-webkit-grabbing;cursor:-moz-grabbing;cursor:grabbing}.leaflet-marker-icon,.leaflet-marker-shadow,.leaflet-image-layer,.leaflet-pane>svg path,.leaflet-tile-container{pointer-events:none}.leaflet-marker-icon.leaflet-interactive,.leaflet-image-layer.leaflet-interactive,.leaflet-pane>svg path.leaflet-interactive,svg.leaflet-image-layer.leaflet-interactive path{pointer-events:visiblePainted;pointer-events:auto}.leaflet-container{background:#ddd;outline:0}.leaflet-container a{color:#0078a8}.leaflet-container a.leaflet-active{outline:2px solid orange}.leaflet-zoom-box{border:2px dotted #38f;background:rgba(255,255,255,.5)}.leaflet-container{font:12px/1.5 "Helvetica Neue",Arial,Helvetica,sans-serif}.leaflet-bar{box-shadow:0 1px 5px rgba(0,0,0,.65);border-radius:4px}.leaflet-bar a,.leaflet-bar a:hover{background-color:#fff;border-bottom:1px solid #ccc;width:26px;height:26px;line-height:26px;display:block;text-align:center;text-decoration:none;color:#000}.leaflet-bar a,.leaflet-control-layers-toggle{background-position:50% 50%;background-repeat:no-repeat;display:block}.leaflet-bar a:hover{background-color:#f4f4f4}.leaflet-bar a:first-child{border-top-left-radius:4px;border-top-right-radius:4px}.leaflet-bar a:last-child{border-bottom-left-radius:4px;border-bottom-right-radius:4px;border-bottom:none}.leaflet-bar a.leaflet-disabled{cursor:default;background-color:#f4f4f4;color:#bbb}.leaflet-touch .leaflet-bar a{width:30px;height:30px;line-height:30px}.leaflet-touch .leaflet-bar a:first-child{border-top-left-radius:2px;border-top-right-radius:2px}.leaflet-touch .leaflet-bar a:last-child{border-bottom-left-radius:2px;border-bottom-right-radius:2px}.leaflet-control-zoom-in,.leaflet-control-zoom-out{font:bold 18px "Lucida Console",Monaco,monospace;text-indent:1px}.leaflet-touch .leaflet-control-zoom-in,.leaflet-touch .leaflet-control-zoom-out{font-size:22px}.leaflet-control-layers{box-shadow:0 1px 5px rgba(0,0,0,.4);background:#fff;border-radius:5px}.leaflet-control-layers-toggle{background-image:url(images/layers.png);width:36px;height:36px}.leaflet-retina .leaflet-control-layers-toggle{background-image:url(images/layers-2x.png);background-size:26px 26px}.leaflet-touch .leaflet-control-layers-toggle{width:44px;height:44px}.leaflet-control-layers .leaflet-control-layers-list,.leaflet-control-layers-expanded .leaflet-control-layers-toggle{display:none}.leaflet-control-layers-expanded .leaflet-control-layers-list{display:block;position:relative}.leaflet-control-layers-expanded{padding:6px 10px 6px 6px;color:#333;background:#fff}.leaflet-control-layers-scrollbar{overflow-y:scroll;overflow-x:hidden;padding-right:5px}.leaflet-control-layers-selector{margin-top:2px;position:relative;top:1px}.leaflet-control-layers label{display:block}.leaflet-control-layers-separator{height:0;border-top:1px solid #ddd;margin:5px -10px 5px -6px}.leaflet-default-icon-path{background-image:url(images/marker-icon.png)}.leaflet-container .leaflet-control-attribution{background:#fff;background:rgba(255,255,255,.7);margin:0}.leaflet-control-attribution,.leaflet-control-scale-line{padding:0 5px;color:#333}.leaflet-control-attribution a{text-decoration:none}.leaflet-control-attribution a:hover{text-decoration:underline}.leaflet-container .leaflet-control-attribution,.leaflet-container .leaflet-control-scale{font-size:11px}.leaflet-left .leaflet-control-scale{margin-left:5px}.leaflet-bottom .leaflet-control-scale{margin-bottom:5px}.leaflet-control-scale-line{border:2px solid #777;border-top:none;line-height:1.1;padding:2px 5px 1px;font-size:11px;white-space:nowrap;overflow:hidden;-moz-box-sizing:border-box;box-sizing:border-box;background:#fff;background:rgba(255,255,255,.5)}.leaflet-control-scale-line:not(:first-child){border-top:2px solid #777;border-bottom:none;margin-top:-2px}.leaflet-control-scale-line:not(:first-child):not(:last-child){border-bottom:2px solid #777}.leaflet-touch .leaflet-control-attribution,.leaflet-touch .leaflet-control-layers,.leaflet-touch .leaflet-bar{box-shadow:none}.leaflet-touch .leaflet-control-layers,.leaflet-touch .leaflet-bar{border:2px solid rgba(0,0,0,.2);background-clip:padding-box}.leaflet-popup{position:absolute;text-align:center;margin-bottom:20px}.leaflet-popup-content-wrapper{padding:1px;text-align:left;border-radius:12px}.leaflet-popup-content{margin:13px 19px;line-height:1.4}.leaflet-popup-content p{margin:18px 0}.leaflet-popup-tip-container{width:40px;height:20px;position:absolute;left:50%;margin-left:-20px;overflow:hidden;pointer-events:none}.leaflet-popup-tip{width:17px;height:17px;padding:1px;margin:-10px auto 0;-webkit-transform:rotate(45deg);-moz-transform:rotate(45deg);-ms-transform:rotate(45deg);transform:rotate(45deg)}.leaflet-popup-content-wrapper,.leaflet-popup-tip{background:#fff;color:#333;box-shadow:0 3px 14px rgba(0,0,0,.4)}.leaflet-container a.leaflet-popup-close-button{position:absolute;top:0;right:0;padding:4px 4px 0 0;border:none;text-align:center;width:18px;height:14px;font:16px/14px Tahoma,Verdana,sans-serif;color:#c3c3c3;text-decoration:none;font-weight:bold;background:transparent}.leaflet-container a.leaflet-popup-close-button:hover{color:#999}.leaflet-popup-scrolled{overflow:auto;border-bottom:1px solid #ddd;border-top:1px solid #ddd}.leaflet-oldie .leaflet-popup-content-wrapper{zoom:1}.leaflet-oldie .leaflet-popup-tip{width:24px;margin:0 auto;-ms-filter:"progid:DXImageTransform.Microsoft.Matrix(M11=0.70710678, M12=0.70710678, M21=-0.70710678, M22=0.70710678)";filter:progid:DXImageTransform.Microsoft.Matrix(M11=0.70710678, M12=0.70710678, M21=-0.70710678, M22=0.70710678)}.leaflet-oldie .leaflet-popup-tip-container{margin-top:-1px}.leaflet-oldie .leaflet-control-zoom,.leaflet-oldie .leaflet-control-layers,.leaflet-oldie .leaflet-popup-content-wrapper,.leaflet-oldie .leaflet-popup-tip{border:1px solid #999}.leaflet-div-icon{background:#fff;border:1px solid #666}.leaflet-tooltip{position:absolute;padding:6px;background-color:#fff;border:1px solid #fff;border-radius:3px;color:#222;white-space:nowrap;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;pointer-events:none;box-shadow:0 1px 3px rgba(0,0,0,.4)}.leaflet-tooltip.leaflet-clickable{cursor:pointer;pointer-events:auto}.leaflet-tooltip-top:before,.leaflet-tooltip-bottom:before,.leaflet-tooltip-left:before,.leaflet-tooltip-right:before{position:absolute;pointer-events:none;border:6px solid transparent;background:transparent;content:""}.leaflet-tooltip-bottom{margin-top:6px}.leaflet-tooltip-top{margin-top:-6px}.leaflet-tooltip-bottom:before,.leaflet-tooltip-top:before{left:50%;margin-left:-6px}.leaflet-tooltip-top:before{bottom:0;margin-bottom:-12px;border-top-color:#fff}.leaflet-tooltip-bottom:before{top:0;margin-top:-12px;margin-left:-6px;border-bottom-color:#fff}.leaflet-tooltip-left{margin-left:-6px}.leaflet-tooltip-right{margin-left:6px}.leaflet-tooltip-left:before,.leaflet-tooltip-right:before{top:50%;margin-top:-6px}.leaflet-tooltip-left:before{right:0;margin-right:-12px;border-left-color:#fff}.leaflet-tooltip-right:before{left:0;margin-left:-12px;border-right-color:#fff}.spot-theme.leaflet-control.elevation .background{margin:6px 0 -12px}.spot-theme.leaflet-control.elevation .axis path,.spot-theme.leaflet-control.elevation .axis line{fill:none;stroke:#999;stroke-width:2}.spot-theme.leaflet-control.elevation .mouse-focus-label-x{text-anchor:middle}.spot-theme.leaflet-control.elevation .mouse-drag{fill:rgba(255,255,255,.2)}.spot-theme.leaflet-control.elevation .elevation-toggle{cursor:pointer;width:44px;height:44px;color:#ccc;text-shadow:0px 1px 1px rgba(0,0,0,.8)}.spot-theme.leaflet-control.elevation .area{fill:#ccc;filter:drop-shadow(-1px 1px 1px rgba(0, 0, 0, 0.6));-webkit-filter:drop-shadow(-1px 1px 1px rgba(0, 0, 0, 0.6))}.spot-theme.leaflet-control.elevation .mouse-focus-line{pointer-events:none;stroke-width:1;stroke:#666}.spot-theme.height-focus{stroke:#ccc;fill:#ccc}.spot-theme.height-focus.line{pointer-events:none;stroke-width:2}.spot-theme.height-focus-label{text-anchor:middle;fill:#ccc}.leaflet-container{background:none}.spot-theme.height-focus,.spot-theme.height-focus.line,.spot-theme.height-focus-label,.spot-theme.leaflet-control.elevation .area{filter:drop-shadow(-1px 1px 1px rgba(0, 0, 0, 0.6));-webkit-filter:drop-shadow(-1px 1px 1px rgba(0, 0, 0, 0.6))}.spot-theme.leaflet-control.elevation-collapsed .background{display:none}.spot-theme .details text{text-anchor:middle}#projects.with-feed #submap{width:calc(100% - 30%);min-width:calc(100% - 400px + 3 * 1rem)}#projects.with-feed .leaflet-right{width:calc(30%);max-width:calc(400px + 3 * 1rem)}#projects.with-feed #feed{z-index:999;transition:none}#projects:not(.with-feed) #feed #posts{right:-100%}#projects.with-settings #submap{width:calc(100% - 30%);min-width:calc(100% - 400px + 3 * 1rem)}#projects.with-settings .leaflet-left{width:calc(30%);max-width:calc(400px + 3 * 1rem)}#projects.with-settings #settings{z-index:999;transition:none}#projects:not(.with-settings) #settings #settings-sections{left:-100%}#projects.with-feed.with-settings #submap{width:calc(100% - 30% * 2);min-width:calc(100% - 400px + 3 * 1rem * 2)}#projects #submap{position:absolute;left:0;top:0;bottom:0;width:100%}#projects #submap .loader{position:absolute;font-size:3em;top:calc(50% - 0.5em);left:calc(50% - 0.66666em);color:#ccc}#projects #map{position:absolute;left:0;top:0;bottom:0;width:100%}#projects #map .track_tooltip p{margin:0}#projects #map .track_tooltip p.name{font-weight:bold;font-size:1.2em}#projects #map .track_tooltip p.description{font-style:italic}#projects #map .track_tooltip p.detail{margin-top:1em;width:50%;display:inline-block}#projects .leaflet-control{background-color:rgba(255,255,255,.6);font-family:Roboto,Arial,sans-serif;border-radius:3px;border:none;margin:1rem}#projects .leaflet-control+.leaflet-control{margin-top:0}#projects .leaflet-control.leaflet-control-scale{padding:.5em}#projects .leaflet-control.leaflet-control-scale .leaflet-control-scale-line{background:none}#projects .leaflet-right,#projects .leaflet-left{transition:all .5s;width:0;max-width:0}#projects .leaflet-right .leaflet-control{left:-100%}#projects .leaflet-left .leaflet-control{right:-100%}#projects .leaflet-top.leaflet-left .leaflet-control-layers{display:none}#projects #legend .track{white-space:nowrap}#projects #legend .track .line{width:2em;height:4px;display:inline-block;border-radius:2px;vertical-align:middle}#projects #legend .track .line.main{background-color:#00ff78}#projects #legend .track .line.off-track{background-color:blue}#projects #legend .track .line.hitchhiking{background-color:#ff7814}#projects #legend .track .desc{font-size:1em;margin-left:.5em;color:#333}#projects #post-button,#projects #settings-button{cursor:pointer;text-shadow:0px 1px 1px rgba(0,0,0,.8);width:44px;text-align:center}#projects #post-button:hover .fa,#projects #post-button:hover .control-icon,#projects #post-button:hover .fa,#projects #post-button:hover .control-icon,#projects #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover a.lb-prev,#projects #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover a.lb-next,#projects #post-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button:hover .lb-close,#projects #post-button:hover .lightbox .lb-cancel,.lightbox #projects #post-button:hover .lb-cancel,#projects #post-button:hover .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button:hover .elevation-toggle,#projects #post-button:hover #settings-button .fa,#projects #settings-button #post-button:hover .fa,#projects #post-button:hover #settings-button .control-icon,#projects #settings-button #post-button:hover .control-icon,#projects #post-button:hover #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button:hover a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover #settings-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button:hover a.lb-prev,#projects #post-button:hover #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button:hover a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover #settings-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button:hover a.lb-next,#projects #post-button:hover #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button:hover .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button:hover #settings-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button #post-button:hover .lb-close,#projects #post-button:hover #settings-button .lightbox .lb-cancel,#projects #settings-button .lightbox #post-button:hover .lb-cancel,.lightbox #projects #post-button:hover #settings-button .lb-cancel,.lightbox #projects #settings-button #post-button:hover .lb-cancel,#projects #post-button:hover #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #post-button:hover .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button:hover #settings-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button #post-button:hover .elevation-toggle,#projects #post-button:hover #settings-button .fa,#projects #settings-button #post-button:hover .fa,#projects #post-button:hover #settings-button .control-icon,#projects #settings-button #post-button:hover .control-icon,#projects #post-button:hover #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button:hover a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover #settings-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button:hover a.lb-prev,#projects #post-button:hover #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button:hover a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover #settings-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button:hover a.lb-next,#projects #post-button:hover #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button:hover .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button:hover #settings-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button #post-button:hover .lb-close,#projects #post-button:hover #settings-button .lightbox .lb-cancel,#projects #settings-button .lightbox #post-button:hover .lb-cancel,.lightbox #projects #post-button:hover #settings-button .lb-cancel,.lightbox #projects #settings-button #post-button:hover .lb-cancel,#projects #post-button:hover #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #post-button:hover .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button:hover #settings-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button #post-button:hover .elevation-toggle,#projects #settings-button #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-prev,#projects #settings-button #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-next,#projects #settings-button #post-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #post-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button .lb-close,#projects #settings-button #post-button:hover .lightbox .lb-cancel,#projects #post-button:hover .lightbox #settings-button .lb-cancel,#projects #settings-button #post-button:hover .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #post-button:hover .spot-theme.leaflet-control.elevation-collapsed #settings-button .elevation-toggle,#projects #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover a.lb-prev,#projects #post-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button:hover a.lb-next,#projects #post-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button:hover .lb-close,#projects #post-button:hover .lightbox .lb-cancel,.lightbox #projects #post-button:hover .lb-cancel,#projects #post-button:hover .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button:hover .elevation-toggle,#projects #settings-button:hover .fa,#projects #settings-button:hover .control-icon,#projects #settings-button:hover #post-button .fa,#projects #post-button #settings-button:hover .fa,#projects #settings-button:hover #post-button .control-icon,#projects #post-button #settings-button:hover .control-icon,#projects #settings-button:hover #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button:hover a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover #post-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button:hover a.lb-prev,#projects #settings-button:hover #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button:hover a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover #post-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button:hover a.lb-next,#projects #settings-button:hover #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button:hover .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button:hover #post-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button #settings-button:hover .lb-close,#projects #settings-button:hover #post-button .lightbox .lb-cancel,#projects #post-button .lightbox #settings-button:hover .lb-cancel,.lightbox #projects #settings-button:hover #post-button .lb-cancel,.lightbox #projects #post-button #settings-button:hover .lb-cancel,#projects #settings-button:hover #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings-button:hover .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button:hover #post-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button #settings-button:hover .elevation-toggle,#projects #settings-button:hover .fa,#projects #settings-button:hover .control-icon,#projects #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover a.lb-prev,#projects #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover a.lb-next,#projects #settings-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button:hover .lb-close,#projects #settings-button:hover .lightbox .lb-cancel,.lightbox #projects #settings-button:hover .lb-cancel,#projects #settings-button:hover .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button:hover .elevation-toggle,#projects #post-button #settings-button:hover .fa,#projects #settings-button:hover #post-button .fa,#projects #post-button #settings-button:hover .control-icon,#projects #settings-button:hover #post-button .control-icon,#projects #post-button #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button:hover a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover #post-button a.lb-prev,#projects #post-button #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button:hover a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover #post-button a.lb-next,#projects #post-button #settings-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #settings-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button #settings-button:hover .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button:hover #post-button .lb-close,#projects #post-button #settings-button:hover .lightbox .lb-cancel,#projects #settings-button:hover .lightbox #post-button .lb-cancel,.lightbox #projects #post-button #settings-button:hover .lb-cancel,.lightbox #projects #settings-button:hover #post-button .lb-cancel,#projects #post-button #settings-button:hover .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #settings-button:hover .spot-theme.leaflet-control.elevation-collapsed #post-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button #settings-button:hover .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button:hover #post-button .elevation-toggle,#projects #settings-button:hover #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button:hover a.lb-prev,#projects #settings-button:hover #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button:hover a.lb-next,#projects #settings-button:hover #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button:hover .lb-close,#projects #settings-button:hover #post-button .lightbox .lb-cancel,#projects #post-button .lightbox #settings-button:hover .lb-cancel,#projects #settings-button:hover #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings-button:hover .elevation-toggle,#projects #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover a.lb-prev,#projects #settings-button:hover .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button:hover a.lb-next,#projects #settings-button:hover .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button:hover .lb-close,#projects #settings-button:hover .lightbox .lb-cancel,.lightbox #projects #settings-button:hover .lb-cancel,#projects #settings-button:hover .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button:hover .elevation-toggle{color:#fff}#projects #post-button .fa,#projects #post-button .control-icon,#projects #post-button .fa,#projects #post-button .control-icon,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .lb-close,#projects #post-button .lightbox .lb-cancel,.lightbox #projects #post-button .lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .elevation-toggle,#projects #post-button #settings-button .fa,#projects #settings-button #post-button .fa,#projects #post-button #settings-button .control-icon,#projects #settings-button #post-button .control-icon,#projects #post-button #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-prev,#projects #post-button #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-next,#projects #post-button #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button #settings-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button #post-button .lb-close,#projects #post-button #settings-button .lightbox .lb-cancel,#projects #settings-button .lightbox #post-button .lb-cancel,.lightbox #projects #post-button #settings-button .lb-cancel,.lightbox #projects #settings-button #post-button .lb-cancel,#projects #post-button #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #post-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button #settings-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button #post-button .elevation-toggle,#projects #post-button #settings-button .fa,#projects #settings-button #post-button .fa,#projects #post-button #settings-button .control-icon,#projects #settings-button #post-button .control-icon,#projects #post-button #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-prev,#projects #post-button #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-next,#projects #post-button #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button #settings-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button #post-button .lb-close,#projects #post-button #settings-button .lightbox .lb-cancel,#projects #settings-button .lightbox #post-button .lb-cancel,.lightbox #projects #post-button #settings-button .lb-cancel,.lightbox #projects #settings-button #post-button .lb-cancel,#projects #post-button #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #post-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button #settings-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button #post-button .elevation-toggle,#projects #settings-button #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-prev,#projects #settings-button #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-next,#projects #settings-button #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button .lb-close,#projects #settings-button #post-button .lightbox .lb-cancel,#projects #post-button .lightbox #settings-button .lb-cancel,#projects #settings-button #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings-button .elevation-toggle,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button a.lb-next,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button .lb-close,#projects #post-button .lightbox .lb-cancel,.lightbox #projects #post-button .lb-cancel,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button .elevation-toggle,#projects #settings-button .fa,#projects #settings-button .control-icon,#projects #settings-button #post-button .fa,#projects #post-button #settings-button .fa,#projects #settings-button #post-button .control-icon,#projects #post-button #settings-button .control-icon,#projects #settings-button #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-prev,#projects #settings-button #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-next,#projects #settings-button #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button #post-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button #settings-button .lb-close,#projects #settings-button #post-button .lightbox .lb-cancel,#projects #post-button .lightbox #settings-button .lb-cancel,.lightbox #projects #settings-button #post-button .lb-cancel,.lightbox #projects #post-button #settings-button .lb-cancel,#projects #settings-button #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button #post-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button #settings-button .elevation-toggle,#projects #settings-button .fa,#projects #settings-button .control-icon,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .lb-close,#projects #settings-button .lightbox .lb-cancel,.lightbox #projects #settings-button .lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .elevation-toggle,#projects #post-button #settings-button .fa,#projects #settings-button #post-button .fa,#projects #post-button #settings-button .control-icon,#projects #settings-button #post-button .control-icon,#projects #post-button #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-prev,#projects #post-button #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav #post-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #post-button #settings-button a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button #post-button a.lb-next,#projects #post-button #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #post-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #post-button #settings-button .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button #post-button .lb-close,#projects #post-button #settings-button .lightbox .lb-cancel,#projects #settings-button .lightbox #post-button .lb-cancel,.lightbox #projects #post-button #settings-button .lb-cancel,.lightbox #projects #settings-button #post-button .lb-cancel,#projects #post-button #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed #post-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #post-button #settings-button .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button #post-button .elevation-toggle,#projects #settings-button #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-prev,#projects #settings-button #post-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,#projects #post-button .lightbox .lb-outerContainer .lb-container .lb-nav #settings-button a.lb-next,#projects #settings-button #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,#projects #post-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer #settings-button .lb-close,#projects #settings-button #post-button .lightbox .lb-cancel,#projects #post-button .lightbox #settings-button .lb-cancel,#projects #settings-button #post-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,#projects #post-button .spot-theme.leaflet-control.elevation-collapsed #settings-button .elevation-toggle,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-prev,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-prev,#projects #settings-button .lightbox .lb-outerContainer .lb-container .lb-nav a.lb-next,.lightbox .lb-outerContainer .lb-container .lb-nav #projects #settings-button a.lb-next,#projects #settings-button .lightbox .lb-dataContainer .lb-data .lb-closeContainer .lb-close,.lightbox .lb-dataContainer .lb-data .lb-closeContainer #projects #settings-button .lb-close,#projects #settings-button .lightbox .lb-cancel,.lightbox #projects #settings-button .lb-cancel,#projects #settings-button .spot-theme.leaflet-control.elevation-collapsed .elevation-toggle,.spot-theme.leaflet-control.elevation-collapsed #projects #settings-button .elevation-toggle{color:#ccc}#projects a.drill{position:relative;overflow:hidden;text-decoration:none;display:inline-block}#projects a.drill .drill-icon{position:absolute;display:inline-block;top:50%;left:50%;transform:translate(-50%, -50%)}#projects a.drill .drill-icon i{transition:all .3s;cursor:pointer}#projects .fa-stack .fa-message{font-size:32px;text-shadow:rgba(0,0,0,.5) 3px 3px 3px;color:#6dff58}#projects .fa-stack .fa-message-in{font-size:13px;color:#326526;top:1px}#projects .fa-stack .fa-track-start,#projects .fa-stack .fa-track-end{color:#326526;font-size:14px;top:1px}#projects .fa-stack .fa-track-end{color:#ff7814}#projects #feed,#projects #settings{position:absolute;top:0;bottom:0;width:calc(30%);max-width:calc(400px + 3 * 1rem);z-index:-1;transition-property:z-index;transition-duration:.1s;transition-delay:.5s;overflow:hidden}#projects #feed input,#projects #feed textarea,#projects #settings input,#projects #settings textarea{background-color:#fff;color:#333}#projects #feed button,#projects #settings button{background-color:#333;color:rgba(255,255,255,.8)}#projects #feed button:hover,#projects #settings button:hover{background-color:#fff;color:#333}#projects #feed{right:0}#projects #feed #posts{position:absolute;transition:all .5s;top:0;bottom:0;right:0;width:100%}#projects #feed #posts #poster textarea#post{margin-bottom:1em;width:calc(100% - 2em)}#projects #feed #posts #poster input#name{width:calc(100% - 6em)}#projects #feed #posts #poster button#submit{margin-left:1em;margin-bottom:.5em}#projects #feed #posts .post{margin-bottom:1rem;background:rgba(255,255,255,.8);color:#333;border-radius:3px;width:calc(100% - 1rem);box-shadow:2px 2px 3px 0px rgba(0,0,0,.5)}#projects #feed #posts .post:first-child{margin-top:1rem}#projects #feed #posts .post .message{margin:.3em 0 0 0}#projects #feed #posts .post .signature{margin:.5em 0 0 0;text-align:right;font-style:italic}#projects #feed #posts .post .header{font-style:italic;font-size:.8em;padding:.5em 1em}#projects #feed #posts .post .header span{display:inline-block;width:50%;cursor:default}#projects #feed #posts .post .header span.index{font-style:normal}#projects #feed #posts .post .header span.time{text-align:right}#projects #feed #posts .post .body{clear:both;padding:0em 1em .5em}#projects #feed #posts .post.headerless .header{display:none}#projects #feed #posts .post.headerless .body{padding-top:.5em}#projects #feed #posts .post.media{background:rgba(255,255,255,.8);color:#333}#projects #feed #posts .post.media a{display:inline-block;width:100%;line-height:0;margin:0}#projects #feed #posts .post.media a.drill{font-size:3em}#projects #feed #posts .post.media a.drill .fa-drill-picture{color:transparent}#projects #feed #posts .post.media a.drill .fa-drill-video{color:rgba(255,255,255,.5)}#projects #feed #posts .post.media a.drill:hover .fa-drill-picture,#projects #feed #posts .post.media a.drill:hover .fa-drill-video{color:rgba(255,255,255,.75)}#projects #feed #posts .post.media a img{width:100%;image-orientation:from-image;outline:none;border-radius:3px}#projects #feed #posts .post.media p{margin:0;text-align:justify}#projects #feed #posts .post.message{background:#6dff58;color:#326526}#projects #feed #posts .post.message p{font-size:.9em;margin:.5em 0}#projects #feed #posts .post.message a{color:#326526}#projects #feed #posts .post.message a.drill .drill-icon{transform:translate(-16px, -32px)}#projects #feed #posts .post.message a.drill .drill-icon .fa-message-in{top:0;left:-1px}#projects #feed #posts .post.message a.drill:hover .fa-message{top:16px}#projects #feed #posts .post.message a.drill:hover .fa-message-in{display:none}#projects #feed #posts .post.message .staticmap{width:100%;border-radius:3px;cursor:pointer}#projects #feed #posts .post.loading .body{text-align:center}#projects #feed #posts .post.loading .body p{display:inline-block;font-size:2em;color:#333}#projects #settings{left:0}#projects #settings #settings-sections{width:calc(100% - 3rem);margin:1rem;padding:1rem;background:#fff;border-radius:3px;box-shadow:2px 2px 3px 0px rgba(0,0,0,.5);position:absolute;transition:all .5s;top:0;bottom:0;left:0;color:#333;background:rgba(255,255,255,.8)}#projects #settings #settings-sections .settings-section{display:inline-block;width:100%}#projects #settings #settings-sections .settings-section h1{margin:1.5rem 0 1rem}#projects #settings #settings-sections .settings-section label{margin-bottom:.3em;display:block;cursor:pointer}#projects #settings #settings-sections .settings-section.title{margin:-1rem;width:calc(100% + 2rem);background:rgba(255,255,255,.4);text-align:center;border-radius:3px 3px 0 0}#projects #settings #settings-sections .settings-section.title .logo{background:rgba(255,255,255,.4);padding:1.5rem}#projects #settings #settings-sections .settings-section.title .logo img{width:calc(100% - 3rem);max-width:180px;transform:translateX(-10%)}#projects #settings #settings-sections .settings-section.newsletter input#email{width:calc(100% - 6em)}#projects #settings #settings-sections .settings-section.newsletter input#email:disabled{color:#999;background:rgba(255,255,255,.2)}#projects #settings #settings-sections .settings-section.newsletter button#nl_btn{margin-left:1em;margin-bottom:1em}#projects #settings #settings-sections .settings-section.newsletter button#nl_btn.loading{background-color:#326526;color:#fff}#projects #settings #settings-sections .settings-section #settings-projects a.fa-download{color:#333}#projects #settings #settings-sections .settings-section #settings-projects a.fa-download:hover{color:#0078a8}#projects .leaflet-popup-content{margin:0}#projects .leaflet-popup-content .info-window{margin:1rem}#projects .leaflet-popup-content .info-window h1{font-size:1.2em;margin:1em 0 1.2em}#projects .leaflet-popup-content .info-window h1 i{margin-right:.3125em}#projects .leaflet-popup-content .info-window p{font-size:1em;margin:.5em 0 0 0}#projects .leaflet-popup-content .info-window p i{padding-right:.5em}#projects .leaflet-popup-content .info-window p a{color:#333}#projects .leaflet-popup-content .info-window .medias{margin-top:-0.5rem;line-height:0}#projects .leaflet-popup-content .info-window .medias a{display:inline-block;margin-right:1rem;margin-top:1rem}#projects .leaflet-popup-content .info-window .medias a:last-child{margin-right:0}#projects .leaflet-popup-content .info-window .medias a.drill{font-size:2em}#projects .leaflet-popup-content .info-window .medias a.drill .fa-drill-picture{color:transparent}#projects .leaflet-popup-content .info-window .medias a.drill .fa-drill-video{color:rgba(255,255,255,.5)}#projects .leaflet-popup-content .info-window .medias a.drill:hover .fa-drill-video,#projects .leaflet-popup-content .info-window .medias a.drill:hover .fa-drill-picture{color:rgba(255,255,255,.75)}#projects .leaflet-popup-content .info-window .medias a img{max-width:200px;max-height:100px;border-radius:3px;image-orientation:from-image;transition:All .2s}#elems{display:none}#upload{padding:1em}#upload h1{font-size:2em;border-bottom:2px solid #000;margin:0 0 1em 0;padding-bottom:.5em}#upload .bar{height:18px;background:green}#upload .comment{margin-top:1em}#upload .comment .thumb{width:30%;max-width:100px}#upload .comment form{display:inline-block;width:calc(70% - 1em);min-width:calc(100% - 100px - 1em);margin-left:1em;vertical-align:top}#upload .comment form .content{width:100%;box-sizing:border-box;padding:.5em}#upload .comment form .save{margin-top:1em;padding:.5em}#admin{margin:1em}#admin table{margin-bottom:1em;border-collapse:collapse}#admin table tr th{background:#aaa;color:#fff;padding:.2rem .5rem}#admin table tr td{background:#eee;text-align:center;padding:.2rem .5rem}#admin table tr td input[type=number]{width:50px}#admin table tr td input[name=ref_feed_id]{width:300px}@media only screen and (max-width: 800px){.desktop{display:none}#projects.with-feed #submap,#projects.with-settings #submap{width:100%}#projects.with-feed .leaflet-right,#projects.with-feed .leaflet-left,#projects.with-settings .leaflet-right,#projects.with-settings .leaflet-left{width:calc(100% - 44px - 2 * 1rem)}#projects.with-feed .leaflet-control-container .leaflet-bottom.leaflet-right,#projects.with-settings .leaflet-control-container .leaflet-bottom.leaflet-right{display:none}#projects .leaflet-control-container .leaflet-bottom.leaflet-left,#projects .leaflet-control-container .leaflet-bottom.leaflet-right .leaflet-control.elevation{display:none}#projects #feed,#projects #settings{width:calc(100% - 44px - 2 * 1rem)}}@media only screen and (min-width: 801px){.mobile{display:none}}/*# sourceMappingURL=spot.css.map */ diff --git a/style/spot.css.map b/style/spot.css.map index 7f9fb80..13c08b9 100644 --- a/style/spot.css.map +++ b/style/spot.css.map @@ -1 +1 @@ -{"version":3,"sourceRoot":"","sources":["_fonts.scss","_common.scss","fa/solid.scss","fa/_variables.scss","fa/_core.scss","fa/_larger.scss","fa/_fixed-width.scss","fa/_list.scss","fa/_bordered-pulled.scss","fa/_animated.scss","fa/_rotated-flipped.scss","fa/_mixins.scss","fa/_stacked.scss","_fa.scss","lightbox/_lightbox.scss","_lightbox.scss","_simplebar.scss","leaflet/_leaflet.scss","leaflet/_leaflet_elevation.scss","_leaflet.scss","_mask_project.scss","_mask_upload.scss","_mask_admin.scss","_mobile.scss"],"names":[],"mappings":"CAGA,WACE,qBACA,kBACA,gBACA,8IACA,iFAGF,WACE,qBACA,kBACA,gBACA,8IACA,yDAGF,WACE,qBACA,kBACA,gBACA,8IACA,0BAGF,WACE,qBACA,kBACA,gBACA,8IACA,0BAGF,WACE,qBACA,kBACA,gBACA,8IACA,2GAGF,WACE,qBACA,kBACA,gBACA,0GACA,yJAGF,WACE,qBACA,kBACA,gBACA,4IACA,iFAGF,WACE,qBACA,kBACA,gBACA,4IACA,yDAGF,WACE,qBACA,kBACA,gBACA,4IACA,0BAGF,WACE,qBACA,kBACA,gBACA,4IACA,0BAGF,WACE,qBACA,kBACA,gBACA,4IACA,2GAGF,WACE,qBACA,kBACA,gBACA,wGACA,yJC9FF,0BACE,iBAEF,uBACE,iBAEF,sBACE,iBAEF,qBACE,iBAEF,kBACE,iBAWF,6HAPC,kBAQoB,8BAPpB,eAOoB,8BANpB,cAMoB,8BALpB,aAKoB,8BAJpB,UAIoB,8BAkBrB,2BACC,eACA,gCACA,SAGD,SACC,YAGD,OACC,eACA,iBAGD,sBACC,YACA,iBACA,kBAMA,YACC,iBACA,kBACC,UAED,oBACC,aAED,oBACC,YC9EH;AAAA;AAAA;AAAA,GAMA,WACE,iCACA,kBACA,gBACA,aCLqB,KDMrB,qCACA,2RAOF,srEAEE,iCACA,gBEnBF,0sEAME,kCACA,mCACA,qBACA,kBACA,oBACA,oBACA,cCXF,OACE,yBACA,kBACA,yBAGF,OACE,gBAGF,OACE,iBAIA,OACE,cADF,OACE,cADF,OACE,cADF,OACE,cADF,OACE,cADF,OACE,cADF,OACE,cADF,OACE,cADF,OACE,cADF,QACE,eClBJ,OACE,kBACA,MHOqB,OIRvB,OACE,qBACA,kBACA,eAEA,4BAGF,OACE,UACA,kBACA,kBACA,MJLqB,IIMrB,oBCbF,WACE,wBACA,mBACA,yBAGF,yBACA,2BAOE,09EACA,4/ECfF,SACE,qCAGF,UACE,uCAGF,mBACE,GACE,uBAGF,KACE,0BCdJ,cCWE,sEACA,wBDXF,eCUE,sEACA,yBDVF,eCSE,sEACA,yBDRF,oBCYE,gFACA,uBDZF,kBCWE,gFACA,uBDXF,mDCUE,gFACA,wBDLA,oIAME,YElBJ,UACE,qBACA,WACA,gBACA,kBACA,sBACA,YAGF,0BAEE,OACA,kBACA,kBACA,WAGF,aACE,oBAGF,aACE,cAGF,YACE,MTpBqB,KUQtB,woEACC,mBAED,myEACC,kBAIF,49DACC,eACA,kBACA,iBACA,qBACA,WACA,gBAKD,2BACA,uCACA,kCACA,yCACA,mCACA,iCACA,8BACA,2GACA,gCACA,iCACA,iCACA,gCAGA,0sDACA,kCACA,+BACA,6BACA,+BACA,+BACA,q8GAGA,0pDACA,6BACA,6BACA,6BACA,+BACA,kCACA,4BACA,8BACA,mCACA,qCACA,2GACA,8BACA,kCACA,kCAGA,8BACA,k5GAGA,0DACA,4xDACA,wqDACA,sGClFA,0BACE,gBAGF,iBACE,kBACA,MACA,OACA,aACA,sBACA,2DACA,WACA,aAGF,UACE,kBACA,OACA,WACA,cACA,kBACA,cACA,mBACA,aAGF,oBACE,cACA,YACA,kBACA,gBACA,kBAGA,sBAIF,oBACE,uBAEF,gCACC,yBAID,gBACE,YAGF,mBACE,kBACA,QACA,YACA,aACA,cACA,kBAIA,sBAGF,yBACE,WACA,cACA,WAGF,WACE,kBACA,QACA,OACA,WACA,WACA,kBACA,cAGF,WACE,cACA,WACA,YACA,cACA,gDAGF,QACE,kBACA,MACA,OACA,YACA,WACA,WAGF,mBACE,OAGF,UACE,aACA,2GAGF,kBACE,YACA,eACA,cAGF,kBACE,UACA,OACA,WACA,sDACA,0DACA,UACA,+BACA,4BACA,0BACA,uBAGF,wBACE,4DACA,UAGF,kBACE,UACA,QACA,YACA,uDACA,0DACA,UACA,+BACA,4BACA,0BACA,uBAGF,wBACE,4DACA,UAGF,kBACE,cACA,gBACA,QACA,WACA,8BACA,+BAGF,wBACE,WACA,cACA,WAGF,SACE,cACA,WAGF,qBACE,UACA,WACA,gBACA,kBAGF,qBACE,eACA,iBACA,gBAGF,uBACE,WAGF,oBACE,cACA,WACA,mBACA,eACA,WAGF,mBACE,cACA,YACA,WACA,YACA,wDACA,iBACA,aACA,2DACA,WACA,+BACA,4BACA,0BACA,uBAGF,yBACE,eACA,4DACA,UC1MF,iBACC,SACA,QAGD,UACC,aACA,mBACA,uBACA,WACA,YAGA,6BACC,SACA,gBACA,sCAEA,2CACC,gBAEA,qDACC,6BACA,YACA,iBACA,iBACA,WAEA,UACC,2FAMD,0HACC,WACA,qBAEA,wIACC,kBACA,oBAIF,6DAnDH,gBACA,cAoDI,oEACC,SAIF,6DA1DH,gBACA,cA2DI,oEACC,UAOL,4BACC,SACA,UACA,qBACA,mBACA,YACA,eACA,cACA,gBAEA,qCACC,oBAEA,iDACC,WAEA,kEACC,eACA,cACA,eAGD,8EACC,aAGD,4DACC,UACA,eAGF,wDACC,wBACA,eAEA,kEAvGH,gBACA,cAwGI,WAMJ,qBA/GA,gBACA,cAiHC,WCrHF,iBACE,kBACA,sBACA,eACA,2BACA,yBACA,uBAGF,mBACE,gBACA,cACA,eACA,kBACA,mBAGF,gBACE,kBACA,kBACA,gBACA,UACA,SACA,OACA,MACA,SACA,QACA,sBACA,uBACA,UAGF,kBACE,6BACA,8BACA,uBACA,kBACA,MACA,OACA,SACA,QACA,UACA,SACA,iCAGF,2BACE,kBACA,iCACA,kBACA,cACA,YACA,WACA,mBACA,eACA,gBACA,qBACA,wBAGF,2FAEE,QACA,SAGF,mDAEE,YACA,cAGF,uBACE,gBACA,eACA,WACA,oBAGF,wCACE,8BACA,YACA,WACA,cACA,kBACA,WACA,eACA,gBACA,WACA,UACA,SACA,oBACA,kBACA,cACA,aAGF,gCACE,mBACA,cACA,UACA,kBACA,MACA,OACA,aACA,YACA,eACA,cACA,gBACA,oBACA,WAGF,iBACE,UACA,kBACA,QACA,SACA,oBACA,gBAGF,uDACE,oBACA,iBACA,yBAGF,qDACE,mBAGF,qBACE,kBACA,UACA,UACA,gBAGF,4BACE,kBACA,WACA,gBACA,kBACA,OACA,QACA,UACA,8BAGF,8CAEE,WACA,6BAGF,oCACE,MACA,WAGF,gEACE,QACA,WAGF,sCACE,OACA,YAGF,kEACE,YACA,SACA,UAGF,2DACE,WACA,OACA,QACA,WACA,aACA,eACA,WAIF,mEACE,WACA,OAGF,yBACE,cACA,eACA,UACA,kBACA,aACA,YACA,kBACA,kBAGF,0BACE,eACA,OACA,kBACA,kBACA,qBACA,wBChNF,6LAUC,kBACA,OACA,MAED,mBACC,gBAED,0DAGC,yBACG,sBACK,iBACN,uBAGH,yBACC,uBAGD,8BACC,0CAGD,wCACC,aACA,cACA,6BAED,4CAEC,cAID,2PAMC,0BACA,2BAGD,sCACC,6BACA,yBAED,sCACC,4BAEA,kBACA,wBAED,yDACC,sBACA,kBAED,mBACC,wCAED,qBACC,gDAED,cACC,eACA,kBAED,qBACC,mBAED,kBACC,QACA,SACA,2BACK,sBACL,YAGD,0BACC,sBAGD,0BAEA,+BACA,kCACA,iCACA,iCACA,kCACA,gCAEA,qCACA,kCAEA,mBACC,UACA,WAED,MACC,2BACA,qBACA,kBAMD,iBACC,kBACA,YACA,8BACA,oBAED,6BAEC,kBACA,aACA,oBAED,aACC,MAED,eACC,QAED,gBACC,SAED,cACC,OAED,iBACC,WACA,WAED,gCACC,YAED,8BACC,gBAED,iCACC,mBAED,+BACC,iBAED,gCACC,kBAMD,iCACC,oBAED,kCACC,UACA,sCACG,mCACK,8BAET,oDACC,UAED,uBACC,6BACI,yBACI,qBAET,0CACC,sBAED,0CACC,sEACG,gEACK,sDAET,iEAEC,wBACG,qBACK,gBAGT,sCACC,kBAMD,qBACC,eAED,cACC,oBACA,iBACA,YAED,2DAEC,iBAED,qCAEC,YAED,iIAGC,YACA,wBACA,qBACA,gBAID,gHAKC,oBAGD,8KAIC,8BACA,oBAKD,mBACC,gBACA,UAED,qBACC,cAED,oCACC,yBAED,kBACC,uBACA,gCAKD,mBACC,0DAMD,aACC,qCACA,kBAED,oCAEC,sBACA,6BACA,WACA,YACA,iBACA,cACA,kBACA,qBACA,WAED,8CAEC,4BACA,4BACA,cAED,qBACC,yBAED,2BACC,2BACA,4BAED,0BACC,8BACA,+BACA,mBAED,gCACC,eACA,yBACA,WAGD,8BACC,WACA,YACA,iBAED,0CACC,2BACA,4BAED,yCACC,8BACA,+BAKD,mDAEC,iDACA,gBAGD,iFACC,eAMD,wBACC,oCACA,gBACA,kBAED,+BACC,wCACA,WACA,YAED,+CACC,2CACA,0BAED,8CACC,WACA,YAED,qHAEC,aAED,8DACC,cACA,kBAED,iCACC,yBACA,WACA,gBAED,kCACC,kBACA,kBACA,kBAED,iCACC,eACA,kBACA,QAED,8BACC,cAED,kCACC,SACA,0BACA,0BAID,2BACC,6CAMD,gDACC,gBACA,gCACA,SAED,yDAEC,cACA,WAED,+BACC,qBAED,qCACC,0BAED,0FAEC,eAED,qCACC,gBAED,uCACC,kBAED,4BACC,sBACA,gBACA,gBACA,oBACA,eACA,mBACA,gBACA,2BACK,sBAEL,gBACA,gCAED,8CACC,0BACA,mBACA,gBAED,+DACC,6BAGD,+GAGC,gBAED,mEAEC,gCACA,4BAMD,eACC,kBACA,kBACA,mBAED,+BACC,YACA,gBACA,mBAED,uBACC,iBACA,gBAED,yBACC,cAED,6BACC,WACA,YACA,kBACA,SACA,kBACA,gBACA,oBAED,mBACC,WACA,YACA,YAEA,oBAEA,gCACG,6BACC,4BACI,wBAET,kDAEC,gBACA,WACA,qCAED,gDACC,kBACA,MACA,QACA,oBACA,YACA,kBACA,WACA,YACA,yCACA,cACA,qBACA,iBACA,uBAED,sDACC,WAED,wBACC,cACA,6BACA,0BAGD,8CACC,OAED,kCACC,WACA,cAEA,uHACA,iHAED,4CACC,gBAGD,4JAIC,sBAMD,kBACC,gBACA,sBAMD,iBACC,kBACA,YACA,sBACA,sBACA,kBACA,WACA,mBACA,yBACA,sBACA,qBACA,iBACA,oBACA,oCAED,mCACC,eACA,oBAED,sHAIC,kBACA,oBACA,6BACA,uBACA,WAKD,wBACC,eAED,qBACC,gBAED,2DAEC,SACA,iBAED,4BACC,SACA,oBACA,sBAED,+BACC,MACA,iBACA,iBACA,yBAED,sBACC,iBAED,uBACC,gBAED,2DAEC,QACA,gBAED,6BACC,QACA,mBACA,uBAED,8BACC,OACA,kBACA,wBC5nBA,kDAGC,mBAGD,kGAEC,UACA,OCNY,KDOZ,aCHmB,EDOpB,2DACC,mBAED,kDACC,KChBY,qBDmBb,wDACC,eACA,WACA,YACA,WACA,uCAGD,4CACC,KC/BY,KlBsCb,oDACA,4DiBJA,wDACC,oBACA,aC/B0B,EDgC1B,OCjCc,KDqChB,yBACC,OC3Ca,KD4Cb,KC5Ca,KD+Cd,8BACC,oBACA,aC1C2B,ED6C5B,+BACC,mBACA,KCtDa,KAcd,mBACC,gBAKA,kIlBkBA,oDACA,4DkBXC,4DACC,aASF,0BACC,mBCjBA,4BACC,uBACA,wCAGD,mCACC,gBACA,iCAGD,0BACC,YACA,gBAUD,uCACC,YAID,gCACC,uBACA,wCAGD,sCACC,gBACA,iCAGD,kCACC,YACA,gBAUD,2DACC,WAID,0CACC,2BACA,4CAIF,kBACC,kBACA,OACA,MACA,SACA,WAEA,0BACC,kBACA,cACA,sBACA,2BACA,WAIF,eACC,kBACA,OACA,MACA,SACA,WAGC,gCACC,SAEA,qCACC,iBACA,gBAED,4CACC,kBAED,uCACC,eACA,UACA,qBAOJ,2BACC,sCACA,oCACA,kBACA,YACA,OAnIc,KAqId,4CACC,aAGD,iDACC,aAEA,6EACC,gBAMH,iDACC,mBACA,QACA,YAED,0CACC,WAED,yCACC,YAID,4DACC,aAIA,yBACC,mBACA,+BACC,UACA,WACA,qBACA,kBACA,sBAEA,oCACC,iBAjKc,QAmKf,yCACC,iBAnKmB,KAqKpB,2CACC,iBArKqB,QAyKvB,+BACC,cACA,iBACA,MAtLU,KA2Lb,kDACC,eACA,uCACA,WACA,kBAEA,o5cACC,WAGD,47aACC,WAaF,kBACC,kBACA,gBACA,qBACA,qBAEA,8BACC,kBACA,qBACA,QACA,SACA,gCAEA,gCACC,mBACA,eAMF,gCACC,eACA,uCACA,MAxOW,QA0OZ,mCACI,eACA,MA7OU,QA8OV,QAEJ,sEACC,MAjPa,QAkPb,eACA,QAED,kCACC,MA9OuB,QAoPzB,oCACC,kBACA,MACA,SACA,gBACA,iCACA,WACA,4BACA,wBACA,qBACA,gBAEA,sGACC,iBA5Qa,KA6Qb,MA5QW,KA+QZ,kDACC,iBAhRW,KAiRX,MAhRS,qBAkRT,8DACC,iBArRY,KAsRZ,MArRU,KAyRb,gBACC,QAEA,uBACC,kBACA,mBACA,MACA,SACA,QACA,WAGC,6CACC,kBACA,uBAGD,0CACC,uBAGD,6CACC,gBACA,mBAIF,6BACC,cA3TY,KA4TZ,WArTQ,qBAsTR,MAvTU,KAwTV,kBACA,wBACA,0CAEA,yCACC,WAnUW,KAsUZ,sCACC,kBAED,wCACC,kBACA,iBACA,kBAED,qCACC,kBACA,eACA,iBAEA,0CACC,qBACA,UACA,eAEA,gDACC,kBAGD,+CACC,iBAIH,mCACC,WACA,qBAGA,gDACC,aAED,8CACC,iBAGF,mCACC,WAnWO,qBAoWP,MArWS,KAuWT,qCACC,qBACA,WACA,cACA,SAEA,2CACC,cAEA,6DACC,kBAED,2DACC,2BAIA,oIACC,4BAKH,yCACC,WACA,6BACA,aACA,kBAGF,qCACC,SACA,mBAGF,qCACC,WA5YS,QA6YT,MA9YW,QAgZX,uCACC,eACA,cAGD,uCACC,MAtZU,QA0ZV,yDACC,kCAEA,wEACC,MACA,UAKD,+DAEC,SAED,kEACC,aAKH,gDACC,WACA,kBACA,eAID,2CACC,kBAEA,6CACC,qBACA,cACA,MA7bO,KAocb,oBACC,OAEA,uCACC,wBACA,YACA,aACA,gBACA,kBACA,0CACA,kBACA,mBACA,MACA,SACA,OACA,MAndW,KAodX,gCAGC,4DACC,qBAED,wEACC,aAGD,+DACC,mBACA,cACA,eAIA,gFACC,uBAEA,yFACC,WACA,gCAGF,kFACC,gBACA,kBAQA,0FACC,iBAtfS,QAufT,WASF,0FACC,MAngBQ,KAqgBR,gGACC,cAUN,iCACC,SAEA,8CACC,YACA,iDACC,gBACA,mBAEA,mDACC,qBAIF,gDACC,cACA,kBAEA,kDACC,mBAGD,kDACC,MAviBS,KA2iBX,sDACC,mBACA,cAEA,wDACC,qBAEA,kBACA,gBACA,mEACC,eAGD,8DACC,cAEA,gFACC,kBAED,8EACC,2BAIA,0KACC,4BAKH,4DACC,gBACA,iBACA,kBACA,6BACA,mBAQN,OACC,aC9lBD,QACC,YAEA,WACC,cACA,6BACA,iBACA,oBAGD,aACC,YACA,iBAGD,iBACC,eAEA,wBACC,UACA,gBAED,sBACC,qBACA,sBACA,mCACA,gBACA,mBAEA,+BACC,WACA,sBACA,aAGD,4BACC,eACA,aCrCJ,OACC,WAEA,aACC,kBACA,yBAEC,mBACC,gBACA,WACA,oBAED,mBACC,gBACA,kBACA,oBAGC,sCACC,WAED,2CACC,YCtBN,0CACC,SACC,aAMC,4DACC,WAGD,kJACC,mCAGD,8JACC,aAKD,gKAEC,aAIF,oCACC,oCAKH,0CACC,QACC","file":"spot.css"} \ No newline at end of file +{"version":3,"sourceRoot":"","sources":["_fonts.scss","_common.scss","fa/solid.scss","fa/_variables.scss","fa/_core.scss","fa/_larger.scss","fa/_fixed-width.scss","fa/_list.scss","fa/_bordered-pulled.scss","fa/_animated.scss","fa/_rotated-flipped.scss","fa/_mixins.scss","fa/_stacked.scss","_fa.scss","lightbox/_lightbox.scss","_lightbox.scss","_simplebar.scss","leaflet/_leaflet.scss","leaflet/_leaflet_elevation.scss","_leaflet.scss","_mask_project.scss","_mask_upload.scss","_mask_admin.scss","_mobile.scss"],"names":[],"mappings":"CAGA,WACE,qBACA,kBACA,gBACA,8IACA,iFAGF,WACE,qBACA,kBACA,gBACA,8IACA,yDAGF,WACE,qBACA,kBACA,gBACA,8IACA,0BAGF,WACE,qBACA,kBACA,gBACA,8IACA,0BAGF,WACE,qBACA,kBACA,gBACA,8IACA,2GAGF,WACE,qBACA,kBACA,gBACA,0GACA,yJAGF,WACE,qBACA,kBACA,gBACA,4IACA,iFAGF,WACE,qBACA,kBACA,gBACA,4IACA,yDAGF,WACE,qBACA,kBACA,gBACA,4IACA,0BAGF,WACE,qBACA,kBACA,gBACA,4IACA,0BAGF,WACE,qBACA,kBACA,gBACA,4IACA,2GAGF,WACE,qBACA,kBACA,gBACA,wGACA,yJC9FF,0BACE,iBAEF,uBACE,iBAEF,sBACE,iBAEF,qBACE,iBAEF,kBACE,iBAWF,6HAPC,kBAQoB,8BAPpB,eAOoB,8BANpB,cAMoB,8BALpB,aAKoB,8BAJpB,UAIoB,8BAkBrB,2BACC,eACA,gCACA,SAGD,SACC,YAGD,OACC,eACA,iBAGD,sBACC,YACA,iBACA,kBAMA,YACC,iBACA,kBACC,UAED,oBACC,aAED,oBACC,YC9EH;AAAA;AAAA;AAAA,GAMA,WACE,iCACA,kBACA,gBACA,aCLqB,KDMrB,qCACA,2RAOF,srEAEE,iCACA,gBEnBF,0sEAME,kCACA,mCACA,qBACA,kBACA,oBACA,oBACA,cCXF,OACE,yBACA,kBACA,yBAGF,OACE,gBAGF,OACE,iBAIA,OACE,cADF,OACE,cADF,OACE,cADF,OACE,cADF,OACE,cADF,OACE,cADF,OACE,cADF,OACE,cADF,OACE,cADF,QACE,eClBJ,OACE,kBACA,MHOqB,OIRvB,OACE,qBACA,kBACA,eAEA,4BAGF,OACE,UACA,kBACA,kBACA,MJLqB,IIMrB,oBCbF,WACE,wBACA,mBACA,yBAGF,yBACA,2BAOE,09EACA,4/ECfF,SACE,qCAGF,UACE,uCAGF,mBACE,GACE,uBAGF,KACE,0BCdJ,cCWE,sEACA,wBDXF,eCUE,sEACA,yBDVF,eCSE,sEACA,yBDRF,oBCYE,gFACA,uBDZF,kBCWE,gFACA,uBDXF,mDCUE,gFACA,wBDLA,oIAME,YElBJ,UACE,qBACA,WACA,gBACA,kBACA,sBACA,YAGF,0BAEE,OACA,kBACA,kBACA,WAGF,aACE,oBAGF,aACE,cAGF,YACE,MTpBqB,KUQtB,woEACC,mBAED,myEACC,kBAIF,49DACC,eACA,kBACA,iBACA,qBACA,WACA,gBAKD,2BACA,uCACA,kCACA,yCACA,mCACA,iCACA,8BACA,2GACA,gCACA,iCACA,iCACA,gCAGA,0sDACA,kCACA,+BACA,6BACA,+BACA,+BACA,q8GAGA,0pDACA,6BACA,6BACA,6BACA,+BACA,kCACA,4BACA,8BACA,mCACA,qCACA,2GACA,8BACA,kCACA,kCAGA,8BACA,k5GAGA,0DACA,4xDACA,wqDACA,sGClFA,0BACE,gBAGF,iBACE,kBACA,MACA,OACA,aACA,sBACA,2DACA,WACA,aAGF,UACE,kBACA,OACA,WACA,cACA,kBACA,cACA,mBACA,aAGF,oBACE,cACA,YACA,kBACA,gBACA,kBAGA,sBAIF,oBACE,uBAEF,gCACC,yBAID,gBACE,YAGF,mBACE,kBACA,QACA,YACA,aACA,cACA,kBAIA,sBAGF,yBACE,WACA,cACA,WAGF,WACE,kBACA,QACA,OACA,WACA,WACA,kBACA,cAGF,WACE,cACA,WACA,YACA,cACA,gDAGF,QACE,kBACA,MACA,OACA,YACA,WACA,WAGF,mBACE,OAGF,UACE,aACA,2GAGF,kBACE,YACA,eACA,cAGF,kBACE,UACA,OACA,WACA,sDACA,0DACA,UACA,+BACA,4BACA,0BACA,uBAGF,wBACE,4DACA,UAGF,kBACE,UACA,QACA,YACA,uDACA,0DACA,UACA,+BACA,4BACA,0BACA,uBAGF,wBACE,4DACA,UAGF,kBACE,cACA,gBACA,QACA,WACA,8BACA,+BAGF,wBACE,WACA,cACA,WAGF,SACE,cACA,WAGF,qBACE,UACA,WACA,gBACA,kBAGF,qBACE,eACA,iBACA,gBAGF,uBACE,WAGF,oBACE,cACA,WACA,mBACA,eACA,WAGF,mBACE,cACA,YACA,WACA,YACA,wDACA,iBACA,aACA,2DACA,WACA,+BACA,4BACA,0BACA,uBAGF,yBACE,eACA,4DACA,UC1MF,iBACC,SACA,QAGD,UACC,aACA,mBACA,uBACA,WACA,YAGA,6BACC,SACA,gBACA,sCAEA,2CACC,gBAEA,qDACC,6BACA,YACA,iBACA,iBACA,WAEA,UACC,2FAMD,0HACC,WACA,qBAEA,wIACC,kBACA,oBAIF,6DAnDH,gBACA,cAoDI,oEACC,SAIF,6DA1DH,gBACA,cA2DI,oEACC,UAOL,4BACC,SACA,UACA,qBACA,mBACA,YACA,eACA,cACA,gBAEA,qCACC,oBAEA,iDACC,WAEA,kEACC,eACA,cACA,eAGD,8EACC,aAGD,4DACC,UACA,eAGF,wDACC,wBACA,eAEA,kEAvGH,gBACA,cAwGI,WAMJ,qBA/GA,gBACA,cAiHC,WCrHF,iBACE,kBACA,sBACA,eACA,2BACA,yBACA,uBAGF,mBACE,gBACA,cACA,eACA,kBACA,mBAGF,gBACE,kBACA,kBACA,gBACA,UACA,SACA,OACA,MACA,SACA,QACA,sBACA,uBACA,UAGF,kBACE,6BACA,8BACA,uBACA,kBACA,MACA,OACA,SACA,QACA,UACA,SACA,iCAGF,2BACE,kBACA,iCACA,kBACA,cACA,YACA,WACA,mBACA,eACA,gBACA,qBACA,wBAGF,2FAEE,QACA,SAGF,mDAEE,YACA,cAGF,uBACE,gBACA,eACA,WACA,oBAGF,wCACE,8BACA,YACA,WACA,cACA,kBACA,WACA,eACA,gBACA,WACA,UACA,SACA,oBACA,kBACA,cACA,aAGF,gCACE,mBACA,cACA,UACA,kBACA,MACA,OACA,aACA,YACA,eACA,cACA,gBACA,oBACA,WAGF,iBACE,UACA,kBACA,QACA,SACA,oBACA,gBAGF,uDACE,oBACA,iBACA,yBAGF,qDACE,mBAGF,qBACE,kBACA,UACA,UACA,gBAGF,4BACE,kBACA,WACA,gBACA,kBACA,OACA,QACA,UACA,8BAGF,8CAEE,WACA,6BAGF,oCACE,MACA,WAGF,gEACE,QACA,WAGF,sCACE,OACA,YAGF,kEACE,YACA,SACA,UAGF,2DACE,WACA,OACA,QACA,WACA,aACA,eACA,WAIF,mEACE,WACA,OAGF,yBACE,cACA,eACA,UACA,kBACA,aACA,YACA,kBACA,kBAGF,0BACE,eACA,OACA,kBACA,kBACA,qBACA,wBChNF,6LAUC,kBACA,OACA,MAED,mBACC,gBAED,0DAGC,yBACG,sBACK,iBACN,uBAGH,yBACC,uBAGD,8BACC,0CAGD,wCACC,aACA,cACA,6BAED,4CAEC,cAID,2PAMC,0BACA,2BAGD,sCACC,6BACA,yBAED,sCACC,4BAEA,kBACA,wBAED,yDACC,sBACA,kBAED,mBACC,wCAED,qBACC,gDAED,cACC,eACA,kBAED,qBACC,mBAED,kBACC,QACA,SACA,2BACK,sBACL,YAGD,0BACC,sBAGD,0BAEA,+BACA,kCACA,iCACA,iCACA,kCACA,gCAEA,qCACA,kCAEA,mBACC,UACA,WAED,MACC,2BACA,qBACA,kBAMD,iBACC,kBACA,YACA,8BACA,oBAED,6BAEC,kBACA,aACA,oBAED,aACC,MAED,eACC,QAED,gBACC,SAED,cACC,OAED,iBACC,WACA,WAED,gCACC,YAED,8BACC,gBAED,iCACC,mBAED,+BACC,iBAED,gCACC,kBAMD,iCACC,oBAED,kCACC,UACA,sCACG,mCACK,8BAET,oDACC,UAED,uBACC,6BACI,yBACI,qBAET,0CACC,sBAED,0CACC,sEACG,gEACK,sDAET,iEAEC,wBACG,qBACK,gBAGT,sCACC,kBAMD,qBACC,eAED,cACC,oBACA,iBACA,YAED,2DAEC,iBAED,qCAEC,YAED,iIAGC,YACA,wBACA,qBACA,gBAID,gHAKC,oBAGD,8KAIC,8BACA,oBAKD,mBACC,gBACA,UAED,qBACC,cAED,oCACC,yBAED,kBACC,uBACA,gCAKD,mBACC,0DAMD,aACC,qCACA,kBAED,oCAEC,sBACA,6BACA,WACA,YACA,iBACA,cACA,kBACA,qBACA,WAED,8CAEC,4BACA,4BACA,cAED,qBACC,yBAED,2BACC,2BACA,4BAED,0BACC,8BACA,+BACA,mBAED,gCACC,eACA,yBACA,WAGD,8BACC,WACA,YACA,iBAED,0CACC,2BACA,4BAED,yCACC,8BACA,+BAKD,mDAEC,iDACA,gBAGD,iFACC,eAMD,wBACC,oCACA,gBACA,kBAED,+BACC,wCACA,WACA,YAED,+CACC,2CACA,0BAED,8CACC,WACA,YAED,qHAEC,aAED,8DACC,cACA,kBAED,iCACC,yBACA,WACA,gBAED,kCACC,kBACA,kBACA,kBAED,iCACC,eACA,kBACA,QAED,8BACC,cAED,kCACC,SACA,0BACA,0BAID,2BACC,6CAMD,gDACC,gBACA,gCACA,SAED,yDAEC,cACA,WAED,+BACC,qBAED,qCACC,0BAED,0FAEC,eAED,qCACC,gBAED,uCACC,kBAED,4BACC,sBACA,gBACA,gBACA,oBACA,eACA,mBACA,gBACA,2BACK,sBAEL,gBACA,gCAED,8CACC,0BACA,mBACA,gBAED,+DACC,6BAGD,+GAGC,gBAED,mEAEC,gCACA,4BAMD,eACC,kBACA,kBACA,mBAED,+BACC,YACA,gBACA,mBAED,uBACC,iBACA,gBAED,yBACC,cAED,6BACC,WACA,YACA,kBACA,SACA,kBACA,gBACA,oBAED,mBACC,WACA,YACA,YAEA,oBAEA,gCACG,6BACC,4BACI,wBAET,kDAEC,gBACA,WACA,qCAED,gDACC,kBACA,MACA,QACA,oBACA,YACA,kBACA,WACA,YACA,yCACA,cACA,qBACA,iBACA,uBAED,sDACC,WAED,wBACC,cACA,6BACA,0BAGD,8CACC,OAED,kCACC,WACA,cAEA,uHACA,iHAED,4CACC,gBAGD,4JAIC,sBAMD,kBACC,gBACA,sBAMD,iBACC,kBACA,YACA,sBACA,sBACA,kBACA,WACA,mBACA,yBACA,sBACA,qBACA,iBACA,oBACA,oCAED,mCACC,eACA,oBAED,sHAIC,kBACA,oBACA,6BACA,uBACA,WAKD,wBACC,eAED,qBACC,gBAED,2DAEC,SACA,iBAED,4BACC,SACA,oBACA,sBAED,+BACC,MACA,iBACA,iBACA,yBAED,sBACC,iBAED,uBACC,gBAED,2DAEC,QACA,gBAED,6BACC,QACA,mBACA,uBAED,8BACC,OACA,kBACA,wBC5nBA,kDAGC,mBAGD,kGAEC,UACA,OCNY,KDOZ,aCHmB,EDOpB,2DACC,mBAED,kDACC,KChBY,qBDmBb,wDACC,eACA,WACA,YACA,WACA,uCAGD,4CACC,KC/BY,KlBsCb,oDACA,4DiBJA,wDACC,oBACA,aC/B0B,EDgC1B,OCjCc,KDqChB,yBACC,OC3Ca,KD4Cb,KC5Ca,KD+Cd,8BACC,oBACA,aC1C2B,ED6C5B,+BACC,mBACA,KCtDa,KAcd,mBACC,gBAKA,kIlBkBA,oDACA,4DkBXC,4DACC,aASF,0BACC,mBCjBA,4BACC,uBACA,wCAGD,mCACC,gBACA,iCAGD,0BACC,YACA,gBAUD,uCACC,YAID,gCACC,uBACA,wCAGD,sCACC,gBACA,iCAGD,kCACC,YACA,gBAUD,2DACC,WAID,0CACC,2BACA,4CAIF,kBACC,kBACA,OACA,MACA,SACA,WAEA,0BACC,kBACA,cACA,sBACA,2BACA,WAIF,eACC,kBACA,OACA,MACA,SACA,WAGC,gCACC,SAEA,qCACC,iBACA,gBAED,4CACC,kBAED,uCACC,eACA,UACA,qBAOJ,2BACC,sCACA,oCACA,kBACA,YACA,OAnIc,KAqId,4CACC,aAGD,iDACC,aAEA,6EACC,gBAMH,iDACC,mBACA,QACA,YAED,0CACC,WAED,yCACC,YAID,4DACC,aAIA,yBACC,mBACA,+BACC,UACA,WACA,qBACA,kBACA,sBAEA,oCACC,iBAjKc,QAmKf,yCACC,iBAnKmB,KAqKpB,2CACC,iBArKqB,QAyKvB,+BACC,cACA,iBACA,MAtLU,KA2Lb,kDACC,eACA,uCACA,WACA,kBAEA,o5cACC,WAGD,47aACC,WAaF,kBACC,kBACA,gBACA,qBACA,qBAEA,8BACC,kBACA,qBACA,QACA,SACA,gCAEA,gCACC,mBACA,eAMF,gCACC,eACA,uCACA,MAxOW,QA0OZ,mCACI,eACA,MA7OU,QA8OV,QAEJ,sEACC,MAjPa,QAkPb,eACA,QAED,kCACC,MA9OuB,QAoPzB,oCACC,kBACA,MACA,SACA,gBACA,iCACA,WACA,4BACA,wBACA,qBACA,gBAEA,sGACC,iBA5Qa,KA6Qb,MA5QW,KA+QZ,kDACC,iBAhRW,KAiRX,MAhRS,qBAkRT,8DACC,iBArRY,KAsRZ,MArRU,KAyRb,gBACC,QAEA,uBACC,kBACA,mBACA,MACA,SACA,QACA,WAGC,6CACC,kBACA,uBAGD,0CACC,uBAGD,6CACC,gBACA,mBAIF,6BACC,cA3TY,KA4TZ,WArTQ,qBAsTR,MAvTU,KAwTV,kBACA,wBACA,0CAEA,yCACC,WAnUW,KAsUZ,sCACC,kBAED,wCACC,kBACA,iBACA,kBAED,qCACC,kBACA,eACA,iBAEA,0CACC,qBACA,UACA,eAEA,gDACC,kBAGD,+CACC,iBAIH,mCACC,WACA,qBAGA,gDACC,aAED,8CACC,iBAGF,mCACC,WAnWO,qBAoWP,MArWS,KAuWT,qCACC,qBACA,WACA,cACA,SAEA,2CACC,cAEA,6DACC,kBAED,2DACC,2BAIA,oIACC,4BAKH,yCACC,WACA,6BACA,aACA,kBAGF,qCACC,SACA,mBAGF,qCACC,WA5YS,QA6YT,MA9YW,QAgZX,uCACC,eACA,cAGD,uCACC,MAtZU,QA0ZV,yDACC,kCAEA,wEACC,MACA,UAKD,+DAEC,SAED,kEACC,aAKH,gDACC,WACA,kBACA,eAID,2CACC,kBAEA,6CACC,qBACA,cACA,MA7bO,KAocb,oBACC,OAEA,uCACC,wBACA,YACA,aACA,gBACA,kBACA,0CACA,kBACA,mBACA,MACA,SACA,OACA,MAndW,KAodX,gCAEA,yDACC,qBACA,WAEA,4DACC,qBAGD,+DACC,mBACA,cACA,eAGD,+DACC,aACA,wBACA,gCACA,kBACA,0BAEA,qEACC,gCACA,eAEA,yEACC,wBACA,gBACA,2BAMF,gFACC,uBAEA,yFACC,WACA,gCAGF,kFACC,gBACA,kBAQA,0FACC,iBAzgBS,QA0gBT,WASF,0FACC,MAthBQ,KAwhBR,gGACC,cAUN,iCACC,SAEA,8CACC,YACA,iDACC,gBACA,mBAEA,mDACC,qBAIF,gDACC,cACA,kBAEA,kDACC,mBAGD,kDACC,MA1jBS,KA8jBX,sDACC,mBACA,cAEA,wDACC,qBAEA,kBACA,gBACA,mEACC,eAGD,8DACC,cAEA,gFACC,kBAED,8EACC,2BAIA,0KACC,4BAKH,4DACC,gBACA,iBACA,kBACA,6BACA,mBAQN,OACC,aCjnBD,QACC,YAEA,WACC,cACA,6BACA,iBACA,oBAGD,aACC,YACA,iBAGD,iBACC,eAEA,wBACC,UACA,gBAED,sBACC,qBACA,sBACA,mCACA,gBACA,mBAEA,+BACC,WACA,sBACA,aAGD,4BACC,eACA,aCrCJ,OACC,WAEA,aACC,kBACA,yBAEC,mBACC,gBACA,WACA,oBAED,mBACC,gBACA,kBACA,oBAGC,sCACC,WAED,2CACC,YCtBN,0CACC,SACC,aAMC,4DACC,WAGD,kJACC,mCAGD,8JACC,aAKD,gKAEC,aAIF,oCACC,oCAKH,0CACC,QACC","file":"spot.css"} \ No newline at end of file