discoveryResults = inbox.stream()
+ .filter(discoveryResult -> willConnectVia(discoveryResult, bridge)).collect(Collectors.toList());
+
+ return templateGenerator.createBridgeAndThingConfigurationTemplate(bridge, pairedThings, discoveryResults);
+ }
+
+ private boolean isConnectedVia(Thing thing, Bridge bridge) {
+ return bridge.getUID().equals(thing.getBridgeUID());
+ }
+
+ private boolean willConnectVia(DiscoveryResult discoveryResult, Bridge bridge) {
+ return bridge.getUID().equals(discoveryResult.getBridgeUID());
+ }
+
+ private boolean isMieleCloudBridge(Thing thing) {
+ return MieleCloudBindingConstants.THING_TYPE_BRIDGE.equals(thing.getThingTypeUID());
+ }
+
+ private String renderSslWarning(HttpServletRequest request, String skeleton) {
+ if (!request.isSecure()) {
+ return skeleton.replace(NO_SSL_WARNING_PLACEHOLDER, "\n"
+ + " Warning: We strongly advice to proceed only with SSL enabled for a secure data exchange.\n"
+ + " See
Securing access to openHAB for details.\n"
+ + "
");
+ } else {
+ return skeleton.replace(NO_SSL_WARNING_PLACEHOLDER, "");
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/CreateBridgeServlet.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/CreateBridgeServlet.java
new file mode 100644
index 00000000000..3b667ce183d
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/CreateBridgeServlet.java
@@ -0,0 +1,217 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.config.servlet;
+
+import java.util.concurrent.TimeUnit;
+import java.util.function.BooleanSupplier;
+
+import javax.servlet.http.HttpServletRequest;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants;
+import org.openhab.binding.mielecloud.internal.config.exception.BridgeCreationFailedException;
+import org.openhab.binding.mielecloud.internal.config.exception.BridgeReconfigurationFailedException;
+import org.openhab.binding.mielecloud.internal.handler.MieleBridgeHandler;
+import org.openhab.binding.mielecloud.internal.util.EmailValidator;
+import org.openhab.binding.mielecloud.internal.util.LocaleValidator;
+import org.openhab.core.config.discovery.DiscoveryResult;
+import org.openhab.core.config.discovery.DiscoveryResultBuilder;
+import org.openhab.core.config.discovery.inbox.Inbox;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.thing.ThingRegistry;
+import org.openhab.core.thing.ThingStatus;
+import org.openhab.core.thing.ThingUID;
+import org.openhab.core.thing.binding.ThingHandler;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * Servlet that automatically creates a bridge and then redirects the browser to the account overview page.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class CreateBridgeServlet extends AbstractRedirectionServlet {
+ private static final String MIELE_CLOUD_BRIDGE_NAME = "Cloud Connector";
+ private static final String MIELE_CLOUD_BRIDGE_LABEL = "Miele@home Account";
+
+ private static final String LOCALE_PARAMETER_NAME = "locale";
+ public static final String BRIDGE_UID_PARAMETER_NAME = "bridgeUid";
+ public static final String EMAIL_PARAMETER_NAME = "email";
+
+ private static final long serialVersionUID = -2912042079128722887L;
+
+ private static final String DEFAULT_LOCALE = "en";
+
+ private static final long ONLINE_WAIT_TIMEOUT_IN_MILLISECONDS = 5000;
+ private static final long DISCOVERY_COMPLETION_TIMEOUT_IN_MILLISECONDS = 5000;
+ private static final long CHECK_INTERVAL_IN_MILLISECONDS = 100;
+
+ private final Logger logger = LoggerFactory.getLogger(CreateBridgeServlet.class);
+
+ private final Inbox inbox;
+ private final ThingRegistry thingRegistry;
+
+ /**
+ * Creates a new {@link CreateBridgeServlet}.
+ *
+ * @param inbox openHAB inbox for discovery results.
+ * @param thingRegistry openHAB thing registry.
+ */
+ public CreateBridgeServlet(Inbox inbox, ThingRegistry thingRegistry) {
+ this.inbox = inbox;
+ this.thingRegistry = thingRegistry;
+ }
+
+ @Override
+ protected String getRedirectionDestination(HttpServletRequest request) {
+ String bridgeUidString = request.getParameter(BRIDGE_UID_PARAMETER_NAME);
+ if (bridgeUidString == null || bridgeUidString.isEmpty()) {
+ logger.warn("Cannot create bridge: Bridge UID is missing.");
+ return "/mielecloud/failure?" + FailureServlet.MISSING_BRIDGE_UID_PARAMETER_NAME + "=true";
+ }
+
+ String email = request.getParameter(EMAIL_PARAMETER_NAME);
+ if (email == null || email.isEmpty()) {
+ logger.warn("Cannot create bridge: E-mail address is missing.");
+ return "/mielecloud/failure?" + FailureServlet.MISSING_EMAIL_PARAMETER_NAME + "=true";
+ }
+
+ ThingUID bridgeUid = null;
+ try {
+ bridgeUid = new ThingUID(bridgeUidString);
+ } catch (IllegalArgumentException e) {
+ logger.warn("Cannot create bridge: Bridge UID '{}' is malformed.", bridgeUid);
+ return "/mielecloud/failure?" + FailureServlet.MALFORMED_BRIDGE_UID_PARAMETER_NAME + "=true";
+ }
+
+ if (!EmailValidator.isValid(email)) {
+ logger.warn("Cannot create bridge: E-mail address '{}' is malformed.", email);
+ return "/mielecloud/failure?" + FailureServlet.MALFORMED_EMAIL_PARAMETER_NAME + "=true";
+ }
+
+ String locale = getValidLocale(request.getParameter(LOCALE_PARAMETER_NAME));
+
+ logger.debug("Auto configuring Miele account using locale '{}' (requested locale was '{}')", locale,
+ request.getParameter(LOCALE_PARAMETER_NAME));
+ try {
+ Thing bridge = pairOrReconfigureBridge(locale, bridgeUid, email);
+ waitForBridgeToComeOnline(bridge);
+ return "/mielecloud";
+ } catch (BridgeReconfigurationFailedException e) {
+ logger.warn("{}", e.getMessage());
+ return "/mielecloud/success?" + SuccessServlet.BRIDGE_RECONFIGURATION_FAILED_PARAMETER_NAME + "=true&"
+ + SuccessServlet.BRIDGE_UID_PARAMETER_NAME + "=" + bridgeUidString + "&"
+ + SuccessServlet.EMAIL_PARAMETER_NAME + "=" + email;
+ } catch (BridgeCreationFailedException e) {
+ logger.warn("Thing creation failed because there was no binding available that supports the thing.");
+ return "/mielecloud/success?" + SuccessServlet.BRIDGE_CREATION_FAILED_PARAMETER_NAME + "=true&"
+ + SuccessServlet.BRIDGE_UID_PARAMETER_NAME + "=" + bridgeUidString + "&"
+ + SuccessServlet.EMAIL_PARAMETER_NAME + "=" + email;
+ }
+ }
+
+ private Thing pairOrReconfigureBridge(String locale, ThingUID bridgeUid, String email) {
+ DiscoveryResult result = DiscoveryResultBuilder.create(bridgeUid)
+ .withRepresentationProperty(Thing.PROPERTY_MODEL_ID).withLabel(MIELE_CLOUD_BRIDGE_LABEL)
+ .withProperty(Thing.PROPERTY_MODEL_ID, MIELE_CLOUD_BRIDGE_NAME)
+ .withProperty(MieleCloudBindingConstants.CONFIG_PARAM_LOCALE, locale)
+ .withProperty(MieleCloudBindingConstants.CONFIG_PARAM_EMAIL, email).build();
+ if (inbox.add(result)) {
+ return pairBridge(bridgeUid);
+ } else {
+ return reconfigureBridge(bridgeUid, locale, email);
+ }
+ }
+
+ private Thing pairBridge(ThingUID thingUid) {
+ Thing thing = inbox.approve(thingUid, MIELE_CLOUD_BRIDGE_LABEL, null);
+ if (thing == null) {
+ throw new BridgeCreationFailedException();
+ }
+
+ logger.debug("Successfully created bridge {}", thingUid);
+ return thing;
+ }
+
+ private Thing reconfigureBridge(ThingUID thingUid, String locale, String email) {
+ logger.debug("Thing already exists. Modifying configuration.");
+ Thing thing = thingRegistry.get(thingUid);
+ if (thing == null) {
+ throw new BridgeReconfigurationFailedException(
+ "Cannot modify non existing bridge: Could neither add bridge via inbox nor find existing bridge.");
+ }
+
+ ThingHandler handler = thing.getHandler();
+ if (handler == null) {
+ throw new BridgeReconfigurationFailedException("Bridge exists but has no handler.");
+ }
+ if (!(handler instanceof MieleBridgeHandler)) {
+ throw new BridgeReconfigurationFailedException("Bridge handler is of wrong type, expected '"
+ + MieleBridgeHandler.class.getSimpleName() + "' but got '" + handler.getClass().getName() + "'.");
+ }
+
+ MieleBridgeHandler bridgeHandler = (MieleBridgeHandler) handler;
+ bridgeHandler.disposeWebservice();
+ bridgeHandler.initializeWebservice();
+
+ return thing;
+ }
+
+ private String getValidLocale(@Nullable String localeParameterValue) {
+ if (localeParameterValue == null || localeParameterValue.isEmpty()
+ || !LocaleValidator.isValidLanguage(localeParameterValue)) {
+ return DEFAULT_LOCALE;
+ } else {
+ return localeParameterValue;
+ }
+ }
+
+ private void waitForBridgeToComeOnline(Thing bridge) {
+ try {
+ waitForConditionWithTimeout(() -> bridge.getStatus() == ThingStatus.ONLINE,
+ ONLINE_WAIT_TIMEOUT_IN_MILLISECONDS);
+ waitForConditionWithTimeout(new DiscoveryResultCountDoesNotChangeCondition(),
+ DISCOVERY_COMPLETION_TIMEOUT_IN_MILLISECONDS);
+ } catch (InterruptedException e) {
+ Thread.currentThread().interrupt();
+ }
+ }
+
+ private void waitForConditionWithTimeout(BooleanSupplier condition, long timeoutInMilliseconds)
+ throws InterruptedException {
+ long remainingWaitTime = timeoutInMilliseconds;
+ while (!condition.getAsBoolean() && remainingWaitTime > 0) {
+ TimeUnit.MILLISECONDS.sleep(CHECK_INTERVAL_IN_MILLISECONDS);
+ remainingWaitTime -= CHECK_INTERVAL_IN_MILLISECONDS;
+ }
+ }
+
+ private class DiscoveryResultCountDoesNotChangeCondition implements BooleanSupplier {
+ private long previousDiscoveryResultCount = 0;
+
+ @Override
+ public boolean getAsBoolean() {
+ var discoveryResultCount = countOwnDiscoveryResults();
+ var discoveryResultCountUnchanged = previousDiscoveryResultCount == discoveryResultCount;
+ previousDiscoveryResultCount = discoveryResultCount;
+ return discoveryResultCountUnchanged;
+ }
+
+ private long countOwnDiscoveryResults() {
+ return inbox.stream().map(DiscoveryResult::getBindingId)
+ .filter(MieleCloudBindingConstants.BINDING_ID::equals).count();
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/FailureServlet.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/FailureServlet.java
new file mode 100644
index 00000000000..a24802b3b29
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/FailureServlet.java
@@ -0,0 +1,116 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.config.servlet;
+
+import java.io.IOException;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Servlet showing a failure page.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public class FailureServlet extends AbstractShowPageServlet {
+ private static final long serialVersionUID = -5195984256535664942L;
+
+ public static final String OAUTH2_ERROR_PARAMETER_NAME = "oauth2Error";
+ public static final String ILLEGAL_RESPONSE_PARAMETER_NAME = "illegalResponse";
+ public static final String NO_ONGOING_AUTHORIZATION_PARAMETER_NAME = "noOngoingAuthorization";
+ public static final String FAILED_TO_COMPLETE_AUTHORIZATION_PARAMETER_NAME = "failedToCompleteAuthorization";
+ public static final String MISSING_BRIDGE_UID_PARAMETER_NAME = "missingBridgeUid";
+ public static final String MISSING_EMAIL_PARAMETER_NAME = "missingEmail";
+ public static final String MALFORMED_BRIDGE_UID_PARAMETER_NAME = "malformedBridgeUid";
+ public static final String MALFORMED_EMAIL_PARAMETER_NAME = "malformedEmail";
+ public static final String MISSING_REQUEST_URL_PARAMETER_NAME = "missingRequestUrl";
+
+ public static final String OAUTH2_ERROR_ACCESS_DENIED = "access_denied";
+ public static final String OAUTH2_ERROR_INVALID_REQUEST = "invalid_request";
+ public static final String OAUTH2_ERROR_UNAUTHORIZED_CLIENT = "unauthorized_client";
+ public static final String OAUTH2_ERROR_UNSUPPORTED_RESPONSE_TYPE = "unsupported_response_type";
+ public static final String OAUTH2_ERROR_INVALID_SCOPE = "invalid_scope";
+ public static final String OAUTH2_ERROR_SERVER_ERROR = "server_error";
+ public static final String OAUTH2_ERROR_TEMPORARY_UNAVAILABLE = "temporarily_unavailable";
+
+ private static final String ERROR_MESSAGE_TEXT_PLACEHOLDER = "";
+
+ /**
+ * Creates a new {@link FailureServlet}.
+ *
+ * @param resourceLoader Loader to use for resources.
+ */
+ public FailureServlet(ResourceLoader resourceLoader) {
+ super(resourceLoader);
+ }
+
+ @Override
+ protected String handleGetRequest(HttpServletRequest request, HttpServletResponse response)
+ throws MieleHttpException, IOException {
+ return getResourceLoader().loadResourceAsString("failure.html").replace(ERROR_MESSAGE_TEXT_PLACEHOLDER,
+ getErrorMessage(request));
+ }
+
+ private String getErrorMessage(HttpServletRequest request) {
+ String oauth2Error = request.getParameter(OAUTH2_ERROR_PARAMETER_NAME);
+ if (oauth2Error != null) {
+ return getOAuth2ErrorMessage(oauth2Error);
+ } else if (ServletUtil.isParameterEnabled(request, ILLEGAL_RESPONSE_PARAMETER_NAME)) {
+ return "Miele cloud service returned an illegal response.";
+ } else if (ServletUtil.isParameterEnabled(request, NO_ONGOING_AUTHORIZATION_PARAMETER_NAME)) {
+ return "There is no ongoing authorization. Please start an authorization first.";
+ } else if (ServletUtil.isParameterEnabled(request, FAILED_TO_COMPLETE_AUTHORIZATION_PARAMETER_NAME)) {
+ return "Completing the final authorization request failed. Please try the config flow again.";
+ } else if (ServletUtil.isParameterEnabled(request, MISSING_BRIDGE_UID_PARAMETER_NAME)) {
+ return "Missing bridge UID.";
+ } else if (ServletUtil.isParameterEnabled(request, MISSING_EMAIL_PARAMETER_NAME)) {
+ return "Missing e-mail address.";
+ } else if (ServletUtil.isParameterEnabled(request, MALFORMED_BRIDGE_UID_PARAMETER_NAME)) {
+ return "Malformed bridge UID.";
+ } else if (ServletUtil.isParameterEnabled(request, MALFORMED_EMAIL_PARAMETER_NAME)) {
+ return "Malformed e-mail address.";
+ } else if (ServletUtil.isParameterEnabled(request, MISSING_REQUEST_URL_PARAMETER_NAME)) {
+ return "Missing request URL. Please try the config flow again.";
+ } else {
+ return "Unknown error.";
+ }
+ }
+
+ private String getOAuth2ErrorMessage(String oauth2Error) {
+ return "OAuth2 authentication with Miele cloud service failed: " + getOAuth2ErrorDetailMessage(oauth2Error);
+ }
+
+ private String getOAuth2ErrorDetailMessage(String oauth2Error) {
+ switch (oauth2Error) {
+ case OAUTH2_ERROR_ACCESS_DENIED:
+ return "Access denied.";
+ case OAUTH2_ERROR_INVALID_REQUEST:
+ return "Malformed request.";
+ case OAUTH2_ERROR_UNAUTHORIZED_CLIENT:
+ return "Account not authorized to request authorization code.";
+ case OAUTH2_ERROR_UNSUPPORTED_RESPONSE_TYPE:
+ return "Obtaining an authorization code is not supported.";
+ case OAUTH2_ERROR_INVALID_SCOPE:
+ return "Invalid scope.";
+ case OAUTH2_ERROR_SERVER_ERROR:
+ return "Unexpected server error.";
+ case OAUTH2_ERROR_TEMPORARY_UNAVAILABLE:
+ return "Authorization server temporarily unavailable.";
+ default:
+ return "Unknown error code \"" + oauth2Error + "\".";
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/ForwardToLoginServlet.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/ForwardToLoginServlet.java
new file mode 100644
index 00000000000..e817463adf8
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/ForwardToLoginServlet.java
@@ -0,0 +1,148 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.config.servlet;
+
+import java.time.LocalDateTime;
+import java.time.temporal.ChronoUnit;
+
+import javax.servlet.http.HttpServletRequest;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants;
+import org.openhab.binding.mielecloud.internal.auth.OAuthException;
+import org.openhab.binding.mielecloud.internal.config.OAuthAuthorizationHandler;
+import org.openhab.binding.mielecloud.internal.config.exception.NoOngoingAuthorizationException;
+import org.openhab.binding.mielecloud.internal.config.exception.OngoingAuthorizationException;
+import org.openhab.binding.mielecloud.internal.util.EmailValidator;
+import org.openhab.core.thing.ThingUID;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * Servlet gathers and processes required information to perform an authorization with the Miele cloud service
+ * and create a bridge afterwards. Required parameters are the client ID, client secret, an ID for the bridge and an
+ * e-mail address. If the given parameters are valid, the browser is redirected to the Miele service login. Otherwise,
+ * the browser is redirected to the previous page with an according error message.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class ForwardToLoginServlet extends AbstractRedirectionServlet {
+ private static final long serialVersionUID = -9094642228439994183L;
+
+ public static final String CLIENT_ID_PARAMETER_NAME = "clientId";
+ public static final String CLIENT_SECRET_PARAMETER_NAME = "clientSecret";
+ public static final String BRIDGE_ID_PARAMETER_NAME = "bridgeId";
+ public static final String EMAIL_PARAMETER_NAME = "email";
+
+ private final Logger logger = LoggerFactory.getLogger(ForwardToLoginServlet.class);
+
+ private final OAuthAuthorizationHandler authorizationHandler;
+
+ /**
+ * Creates a new {@link ForwardToLoginServlet}.
+ *
+ * @param authorizationHandler Handler implementing the OAuth authorization process.
+ */
+ public ForwardToLoginServlet(OAuthAuthorizationHandler authorizationHandler) {
+ this.authorizationHandler = authorizationHandler;
+ }
+
+ @Override
+ protected String getRedirectionDestination(HttpServletRequest request) {
+ String clientId = request.getParameter(CLIENT_ID_PARAMETER_NAME);
+ String clientSecret = request.getParameter(CLIENT_SECRET_PARAMETER_NAME);
+ String bridgeId = request.getParameter(BRIDGE_ID_PARAMETER_NAME);
+ String email = request.getParameter(EMAIL_PARAMETER_NAME);
+
+ if (clientId == null || clientId.isEmpty()) {
+ logger.warn("Request is missing client ID.");
+ return getErrorRedirectionUrl(PairAccountServlet.MISSING_CLIENT_ID_PARAMETER_NAME);
+ }
+ if (clientSecret == null || clientSecret.isEmpty()) {
+ logger.warn("Request is missing client secret.");
+ return getErrorRedirectionUrl(PairAccountServlet.MISSING_CLIENT_SECRET_PARAMETER_NAME);
+ }
+ if (bridgeId == null || bridgeId.isEmpty()) {
+ logger.warn("Request is missing bridge ID.");
+ return getErrorRedirectionUrl(PairAccountServlet.MISSING_BRIDGE_ID_PARAMETER_NAME);
+ }
+ if (email == null || email.isEmpty()) {
+ logger.warn("Request is missing e-mail address.");
+ return getErrorRedirectionUrl(PairAccountServlet.MISSING_EMAIL_PARAMETER_NAME);
+ }
+
+ ThingUID bridgeUid = null;
+ try {
+ bridgeUid = new ThingUID(MieleCloudBindingConstants.THING_TYPE_BRIDGE, bridgeId);
+ } catch (IllegalArgumentException e) {
+ logger.warn("Passed bridge ID '{}' is invalid.", bridgeId);
+ return getErrorRedirectionUrl(PairAccountServlet.MALFORMED_BRIDGE_ID_PARAMETER_NAME);
+ }
+
+ if (!EmailValidator.isValid(email)) {
+ logger.warn("Passed e-mail address '{}' is invalid.", email);
+ return getErrorRedirectionUrl(PairAccountServlet.MALFORMED_EMAIL_PARAMETER_NAME);
+ }
+
+ try {
+ authorizationHandler.beginAuthorization(clientId, clientSecret, bridgeUid, email);
+ } catch (OngoingAuthorizationException e) {
+ logger.warn("Cannot begin new authorization process while another one is still running.");
+ return getErrorRedirectUrlWithExpiryTime(e.getOngoingAuthorizationExpiryTimestamp());
+ }
+
+ StringBuffer requestUrl = request.getRequestURL();
+ if (requestUrl == null) {
+ return getErrorRedirectionUrl(PairAccountServlet.MISSING_REQUEST_URL_PARAMETER_NAME);
+ }
+
+ try {
+ return authorizationHandler.getAuthorizationUrl(deriveRedirectUri(requestUrl.toString()));
+ } catch (NoOngoingAuthorizationException e) {
+ logger.warn(
+ "Failed to create authorization URL: There was no ongoing authorization although we just started one.");
+ return getErrorRedirectionUrl(PairAccountServlet.NO_ONGOING_AUTHORIZATION_IN_STEP2_PARAMETER_NAME);
+ } catch (OAuthException e) {
+ logger.warn("Failed to create authorization URL.", e);
+ return getErrorRedirectionUrl(PairAccountServlet.FAILED_TO_DERIVE_REDIRECT_URL_PARAMETER_NAME);
+ }
+ }
+
+ private String getErrorRedirectUrlWithExpiryTime(@Nullable LocalDateTime ongoingAuthorizationExpiryTimestamp) {
+ if (ongoingAuthorizationExpiryTimestamp == null) {
+ return getErrorRedirectionUrl(
+ PairAccountServlet.ONGOING_AUTHORIZATION_IN_STEP1_EXPIRES_IN_MINUTES_PARAMETER_NAME,
+ PairAccountServlet.ONGOING_AUTHORIZATION_UNKNOWN_EXPIRY_TIME);
+ }
+
+ long minutesUntilExpiry = ChronoUnit.MINUTES.between(LocalDateTime.now(), ongoingAuthorizationExpiryTimestamp)
+ + 1;
+ return getErrorRedirectionUrl(
+ PairAccountServlet.ONGOING_AUTHORIZATION_IN_STEP1_EXPIRES_IN_MINUTES_PARAMETER_NAME,
+ Long.toString(minutesUntilExpiry));
+ }
+
+ private String getErrorRedirectionUrl(String errorCode) {
+ return getErrorRedirectionUrl(errorCode, "true");
+ }
+
+ private String getErrorRedirectionUrl(String errorCode, String parameterValue) {
+ return "/mielecloud/pair?" + errorCode + "=" + parameterValue;
+ }
+
+ private String deriveRedirectUri(String requestUrl) {
+ return requestUrl + "/../result";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/MieleHttpException.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/MieleHttpException.java
new file mode 100644
index 00000000000..c5eff7bc669
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/MieleHttpException.java
@@ -0,0 +1,35 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.config.servlet;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Exception wrapping a HTTP error code for further processing.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class MieleHttpException extends Exception {
+ private static final long serialVersionUID = 1825214275413952809L;
+
+ private final int httpErrorCode;
+
+ public MieleHttpException(int httpErrorCode) {
+ this.httpErrorCode = httpErrorCode;
+ }
+
+ public int getHttpErrorCode() {
+ return httpErrorCode;
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/PairAccountServlet.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/PairAccountServlet.java
new file mode 100644
index 00000000000..79d872a7caf
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/PairAccountServlet.java
@@ -0,0 +1,124 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.config.servlet;
+
+import java.io.IOException;
+import java.util.Optional;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Servlet showing the pair account page.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class PairAccountServlet extends AbstractShowPageServlet {
+ private static final long serialVersionUID = 6565378471951635420L;
+
+ public static final String CLIENT_ID_PARAMETER_NAME = "clientId";
+ public static final String CLIENT_SECRET_PARAMETER_NAME = "clientSecret";
+
+ public static final String MISSING_CLIENT_ID_PARAMETER_NAME = "missingClientId";
+ public static final String MISSING_CLIENT_SECRET_PARAMETER_NAME = "missingClientSecret";
+ public static final String MISSING_BRIDGE_ID_PARAMETER_NAME = "missingBridgeId";
+ public static final String MISSING_EMAIL_PARAMETER_NAME = "missingEmail";
+ public static final String MALFORMED_BRIDGE_ID_PARAMETER_NAME = "malformedBridgeId";
+ public static final String MALFORMED_EMAIL_PARAMETER_NAME = "malformedEmail";
+ public static final String FAILED_TO_DERIVE_REDIRECT_URL_PARAMETER_NAME = "failedToDeriveRedirectUrl";
+ public static final String ONGOING_AUTHORIZATION_IN_STEP1_EXPIRES_IN_MINUTES_PARAMETER_NAME = "ongoingAuthorizationInStep1ExpiresInMinutes";
+ public static final String ONGOING_AUTHORIZATION_UNKNOWN_EXPIRY_TIME = "unknown";
+ public static final String NO_ONGOING_AUTHORIZATION_IN_STEP2_PARAMETER_NAME = "noOngoingAuthorizationInStep2";
+ public static final String MISSING_REQUEST_URL_PARAMETER_NAME = "missingRequestUrl";
+
+ private static final String PAIR_ACCOUNT_SKELETON = "pairing.html";
+
+ private static final String CLIENT_ID_PLACEHOLDER = "";
+ private static final String CLIENT_SECRET_PLACEHOLDER = "";
+ private static final String ERROR_MESSAGE_PLACEHOLDER = "";
+
+ /**
+ * Creates a new {@link PairAccountServlet}.
+ *
+ * @param resourceLoader Loader for resources.
+ */
+ public PairAccountServlet(ResourceLoader resourceLoader) {
+ super(resourceLoader);
+ }
+
+ @Override
+ protected String handleGetRequest(HttpServletRequest request, HttpServletResponse response)
+ throws MieleHttpException, IOException {
+ String skeleton = getResourceLoader().loadResourceAsString(PAIR_ACCOUNT_SKELETON);
+ skeleton = renderClientIdAndClientSecret(request, skeleton);
+ skeleton = renderErrorMessage(request, skeleton);
+ return skeleton;
+ }
+
+ private String renderClientIdAndClientSecret(HttpServletRequest request, String skeleton) {
+ String prefilledClientId = Optional.ofNullable(request.getParameter(CLIENT_ID_PARAMETER_NAME)).orElse("");
+ String prefilledClientSecret = Optional.ofNullable(request.getParameter(CLIENT_SECRET_PARAMETER_NAME))
+ .orElse("");
+ return skeleton.replace(CLIENT_ID_PLACEHOLDER, prefilledClientId).replace(CLIENT_SECRET_PLACEHOLDER,
+ prefilledClientSecret);
+ }
+
+ private String renderErrorMessage(HttpServletRequest request, String skeleton) {
+ if (ServletUtil.isParameterEnabled(request, MISSING_CLIENT_ID_PARAMETER_NAME)) {
+ return skeleton.replace(ERROR_MESSAGE_PLACEHOLDER,
+ "Missing client ID.
");
+ } else if (ServletUtil.isParameterEnabled(request, MISSING_CLIENT_SECRET_PARAMETER_NAME)) {
+ return skeleton.replace(ERROR_MESSAGE_PLACEHOLDER,
+
+ "Missing client secret.
");
+ } else if (ServletUtil.isParameterEnabled(request, MISSING_BRIDGE_ID_PARAMETER_NAME)) {
+ return skeleton.replace(ERROR_MESSAGE_PLACEHOLDER,
+ "Missing bridge ID.
");
+ } else if (ServletUtil.isParameterEnabled(request, MISSING_EMAIL_PARAMETER_NAME)) {
+ return skeleton.replace(ERROR_MESSAGE_PLACEHOLDER,
+ "Missing e-mail address.
");
+ } else if (ServletUtil.isParameterEnabled(request, MALFORMED_BRIDGE_ID_PARAMETER_NAME)) {
+ return skeleton.replace(ERROR_MESSAGE_PLACEHOLDER,
+ "Malformed bridge ID. A bridge ID may only contain letters, numbers, '-' and '_'!
");
+ } else if (ServletUtil.isParameterEnabled(request, MALFORMED_EMAIL_PARAMETER_NAME)) {
+ return skeleton.replace(ERROR_MESSAGE_PLACEHOLDER,
+ "Malformed e-mail address.
");
+ } else if (ServletUtil.isParameterEnabled(request, FAILED_TO_DERIVE_REDIRECT_URL_PARAMETER_NAME)) {
+ return skeleton.replace(ERROR_MESSAGE_PLACEHOLDER,
+ "Failed to derive redirect URL.
");
+ } else if (ServletUtil.isParameterPresent(request,
+ ONGOING_AUTHORIZATION_IN_STEP1_EXPIRES_IN_MINUTES_PARAMETER_NAME)) {
+ String minutesUntilExpiry = request
+ .getParameter(ONGOING_AUTHORIZATION_IN_STEP1_EXPIRES_IN_MINUTES_PARAMETER_NAME);
+ if (ONGOING_AUTHORIZATION_UNKNOWN_EXPIRY_TIME.equals(minutesUntilExpiry)) {
+ return skeleton.replace(ERROR_MESSAGE_PLACEHOLDER,
+ "There is an authorization ongoing at the moment. Please complete that authorization prior to starting a new one or try again later.
");
+ } else {
+ return skeleton.replace(ERROR_MESSAGE_PLACEHOLDER,
+ "There is an authorization ongoing at the moment. Please complete that authorization prior to starting a new one or try again in "
+ + minutesUntilExpiry + " minutes.
");
+ }
+ } else if (ServletUtil.isParameterEnabled(request, NO_ONGOING_AUTHORIZATION_IN_STEP2_PARAMETER_NAME)) {
+ return skeleton.replace(ERROR_MESSAGE_PLACEHOLDER,
+ "Failed to start auhtorization process. Are you trying to perform multiple authorizations at the same time?
");
+ } else if (ServletUtil.isParameterEnabled(request, MISSING_REQUEST_URL_PARAMETER_NAME)) {
+ return skeleton.replace(ERROR_MESSAGE_PLACEHOLDER,
+ "Missing request URL. Please try again.
");
+ } else {
+ return skeleton.replace(ERROR_MESSAGE_PLACEHOLDER, "");
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/ResourceLoader.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/ResourceLoader.java
new file mode 100644
index 00000000000..d93a3c9999f
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/ResourceLoader.java
@@ -0,0 +1,86 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.config.servlet;
+
+import java.io.FileNotFoundException;
+import java.io.IOException;
+import java.io.InputStream;
+import java.net.URL;
+import java.nio.charset.StandardCharsets;
+import java.util.Scanner;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.osgi.framework.BundleContext;
+
+/**
+ * Provides access to resource files for servlets.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class ResourceLoader {
+ private static final String BEGINNING_OF_INPUT = "\\A";
+
+ private final String basePath;
+ private final BundleContext bundleContext;
+
+ /**
+ * Creates a new {@link ResourceLoader}.
+ *
+ * @param basePath The base path to use for loading. A trailing {@code "/"} is removed.
+ * @param bundleContext {@link BundleContext} to load from.
+ */
+ public ResourceLoader(String basePath, BundleContext bundleContext) {
+ this.basePath = removeTrailingSlashes(basePath);
+ this.bundleContext = bundleContext;
+ }
+
+ private String removeTrailingSlashes(String value) {
+ String ret = value;
+ while (ret.endsWith("/")) {
+ ret = ret.substring(0, ret.length() - 1);
+ }
+ return ret;
+ }
+
+ /**
+ * Opens a resource relative to the base path.
+ *
+ * @param filename The filename of the resource to load.
+ * @return A stream reading from the resource file.
+ * @throws FileNotFoundException If the requested resource file cannot be found.
+ * @throws IOException If an error occurs while opening a stream to the resource.
+ */
+ public InputStream openResource(String filename) throws IOException {
+ URL url = bundleContext.getBundle().getEntry(basePath + "/" + filename);
+ if (url == null) {
+ throw new FileNotFoundException("Cannot find '" + filename + "' relative to '" + basePath + "'");
+ }
+
+ return url.openStream();
+ }
+
+ /**
+ * Loads the contents of a resource file as UTF-8 encoded {@link String}.
+ *
+ * @param filename The filename of the resource to load.
+ * @return The contents of the file.
+ * @throws FileNotFoundException If the requested resource file cannot be found.
+ * @throws IOException If an error occurs while opening a stream to the resource or reading from it.
+ */
+ public String loadResourceAsString(String filename) throws IOException {
+ try (Scanner scanner = new Scanner(openResource(filename), StandardCharsets.UTF_8.name())) {
+ return scanner.useDelimiter(BEGINNING_OF_INPUT).next();
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/ResultServlet.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/ResultServlet.java
new file mode 100644
index 00000000000..5a5db090909
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/ResultServlet.java
@@ -0,0 +1,96 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.config.servlet;
+
+import javax.servlet.http.HttpServletRequest;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.auth.OAuthException;
+import org.openhab.binding.mielecloud.internal.config.OAuthAuthorizationHandler;
+import org.openhab.binding.mielecloud.internal.config.exception.NoOngoingAuthorizationException;
+import org.openhab.core.thing.ThingUID;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * Servlet processing the response by the Miele service after a login. This servlet is called as a result of a
+ * completed login to the Miele service and assumes that the OAuth 2 parameters are passed. Depending on the parameters
+ * and whether the token response can be fetched either the browser is redirected to the success or the failure page.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class ResultServlet extends AbstractRedirectionServlet {
+ private static final long serialVersionUID = 2157912755568949550L;
+
+ public static final String CODE_PARAMETER_NAME = "code";
+ public static final String STATE_PARAMETER_NAME = "state";
+ public static final String ERROR_PARAMETER_NAME = "error";
+
+ private final Logger logger = LoggerFactory.getLogger(ResultServlet.class);
+
+ private final OAuthAuthorizationHandler authorizationHandler;
+
+ /**
+ * Creates a new {@link ResultServlet}.
+ *
+ * @param authorizationHandler Handler implementing the OAuth authorization.
+ */
+ public ResultServlet(OAuthAuthorizationHandler authorizationHandler) {
+ this.authorizationHandler = authorizationHandler;
+ }
+
+ @Override
+ protected String getRedirectionDestination(HttpServletRequest request) {
+ String error = request.getParameter(ERROR_PARAMETER_NAME);
+ if (error != null) {
+ logger.warn("Received error response: {}", error);
+ return "/mielecloud/failure?" + FailureServlet.OAUTH2_ERROR_PARAMETER_NAME + "=" + error;
+ }
+
+ String code = request.getParameter(CODE_PARAMETER_NAME);
+ if (code == null) {
+ logger.warn("Code is null");
+ return "/mielecloud/failure?" + FailureServlet.ILLEGAL_RESPONSE_PARAMETER_NAME + "=true";
+ }
+ String state = request.getParameter(STATE_PARAMETER_NAME);
+ if (state == null) {
+ logger.warn("State is null");
+ return "/mielecloud/failure?" + FailureServlet.ILLEGAL_RESPONSE_PARAMETER_NAME + "=true";
+ }
+
+ try {
+ ThingUID bridgeId = authorizationHandler.getBridgeUid();
+ String email = authorizationHandler.getEmail();
+
+ StringBuffer requestUrl = request.getRequestURL();
+ if (requestUrl == null) {
+ return "/mielecloud/failure?" + FailureServlet.MISSING_REQUEST_URL_PARAMETER_NAME + "=true";
+ }
+
+ try {
+ authorizationHandler.completeAuthorization(requestUrl.toString() + "?" + request.getQueryString());
+ } catch (OAuthException e) {
+ logger.warn("Failed to complete authorization.", e);
+ return "/mielecloud/failure?" + FailureServlet.FAILED_TO_COMPLETE_AUTHORIZATION_PARAMETER_NAME
+ + "=true";
+ }
+
+ return "/mielecloud/success?" + SuccessServlet.BRIDGE_UID_PARAMETER_NAME + "=" + bridgeId.getAsString()
+ + "&" + SuccessServlet.EMAIL_PARAMETER_NAME + "=" + email;
+ } catch (NoOngoingAuthorizationException e) {
+ logger.warn("Failed to complete authorization: There is no ongoing authorization or it timed out");
+ return "/mielecloud/failure?" + FailureServlet.NO_ONGOING_AUTHORIZATION_PARAMETER_NAME + "=true";
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/ServletUtil.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/ServletUtil.java
new file mode 100644
index 00000000000..4441aca3d8d
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/ServletUtil.java
@@ -0,0 +1,57 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.config.servlet;
+
+import javax.servlet.http.HttpServletRequest;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Utility class for common servlet tasks.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class ServletUtil {
+ private ServletUtil() {
+ throw new UnsupportedOperationException();
+ }
+
+ /**
+ * Gets the value of a request parameter or returns a default if the parameter is not present.
+ */
+ public static String getParameterValueOrDefault(HttpServletRequest request, String parameterName,
+ String defaultValue) {
+ String parameterValue = request.getParameter(parameterName);
+ if (parameterValue == null) {
+ return defaultValue;
+ } else {
+ return parameterValue;
+ }
+ }
+
+ /**
+ * Checks whether a request parameter is enabled.
+ */
+ public static boolean isParameterEnabled(HttpServletRequest request, String parameterName) {
+ return "true".equalsIgnoreCase(getParameterValueOrDefault(request, parameterName, "false"));
+ }
+
+ /**
+ * Checks whether a parameter is present in a request.
+ */
+ public static boolean isParameterPresent(HttpServletRequest request, String parameterName) {
+ String parameterValue = request.getParameter(parameterName);
+ return parameterValue != null && !parameterValue.trim().isEmpty();
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/SuccessServlet.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/SuccessServlet.java
new file mode 100644
index 00000000000..d240f215ee7
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/config/servlet/SuccessServlet.java
@@ -0,0 +1,212 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.config.servlet;
+
+import java.io.IOException;
+import java.util.Locale;
+import java.util.Optional;
+import java.util.Set;
+import java.util.stream.Collectors;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.config.ThingsTemplateGenerator;
+import org.openhab.binding.mielecloud.internal.util.EmailValidator;
+import org.openhab.binding.mielecloud.internal.webservice.language.LanguageProvider;
+import org.openhab.core.thing.ThingUID;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * Servlet showing the success page.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public class SuccessServlet extends AbstractShowPageServlet {
+ private static final long serialVersionUID = 7013060161686096950L;
+
+ public static final String BRIDGE_UID_PARAMETER_NAME = "bridgeUid";
+ public static final String EMAIL_PARAMETER_NAME = "email";
+
+ public static final String BRIDGE_CREATION_FAILED_PARAMETER_NAME = "bridgeCreationFailed";
+ public static final String BRIDGE_RECONFIGURATION_FAILED_PARAMETER_NAME = "bridgeReconfigurationFailed";
+
+ private static final String ERROR_MESSAGE_TEXT_PLACEHOLDER = "";
+ private static final String BRIDGE_UID_PLACEHOLDER = "";
+ private static final String EMAIL_PLACEHOLDER = "";
+ private static final String THINGS_TEMPLATE_CODE_PLACEHOLDER = "";
+
+ private static final String LOCALE_OPTIONS_PLACEHOLDER = "";
+
+ private static final String DEFAULT_LANGUAGE = "en";
+ private static final Set SUPPORTED_LANGUAGES = Set.of("da", "nl", "en", "fr", "de", "it", "nb", "es");
+
+ private final Logger logger = LoggerFactory.getLogger(SuccessServlet.class);
+
+ private final LanguageProvider languageProvider;
+ private final ThingsTemplateGenerator templateGenerator;
+
+ /**
+ * Creates a new {@link SuccessServlet}.
+ *
+ * @param resourceLoader Loader for resources.
+ * @param languageProvider Provider for the language to use as default selection.
+ */
+ public SuccessServlet(ResourceLoader resourceLoader, LanguageProvider languageProvider) {
+ super(resourceLoader);
+ this.languageProvider = languageProvider;
+ this.templateGenerator = new ThingsTemplateGenerator();
+ }
+
+ @Override
+ protected String handleGetRequest(HttpServletRequest request, HttpServletResponse response)
+ throws MieleHttpException, IOException {
+ String bridgeUidString = request.getParameter(BRIDGE_UID_PARAMETER_NAME);
+ if (bridgeUidString == null || bridgeUidString.isEmpty()) {
+ logger.warn("Success page is missing bridge UID.");
+ return getResourceLoader().loadResourceAsString("failure.html").replace(ERROR_MESSAGE_TEXT_PLACEHOLDER,
+ "Missing bridge UID.");
+ }
+
+ String email = request.getParameter(EMAIL_PARAMETER_NAME);
+ if (email == null || email.isEmpty()) {
+ logger.warn("Success page is missing e-mail address.");
+ return getResourceLoader().loadResourceAsString("failure.html").replace(ERROR_MESSAGE_TEXT_PLACEHOLDER,
+ "Missing e-mail address.");
+ }
+
+ ThingUID bridgeUid = null;
+ try {
+ bridgeUid = new ThingUID(bridgeUidString);
+ } catch (IllegalArgumentException e) {
+ logger.warn("Success page received malformed bridge UID '{}'.", bridgeUidString);
+ return getResourceLoader().loadResourceAsString("failure.html").replace(ERROR_MESSAGE_TEXT_PLACEHOLDER,
+ "Malformed bridge UID.");
+ }
+
+ if (!EmailValidator.isValid(email)) {
+ logger.warn("Success page received malformed e-mail address '{}'.", email);
+ return getResourceLoader().loadResourceAsString("failure.html").replace(ERROR_MESSAGE_TEXT_PLACEHOLDER,
+ "Malformed e-mail address.");
+ }
+
+ String skeleton = getResourceLoader().loadResourceAsString("success.html");
+ skeleton = renderErrorMessage(request, skeleton);
+ skeleton = renderBridgeUid(skeleton, bridgeUid);
+ skeleton = renderEmail(skeleton, email);
+ skeleton = renderLocaleSelection(skeleton);
+ skeleton = renderBridgeConfigurationTemplate(skeleton, bridgeUid, email);
+ return skeleton;
+ }
+
+ private String renderErrorMessage(HttpServletRequest request, String skeleton) {
+ if (ServletUtil.isParameterEnabled(request, BRIDGE_CREATION_FAILED_PARAMETER_NAME)) {
+ return skeleton.replace(ERROR_MESSAGE_TEXT_PLACEHOLDER,
+ "Could not auto configure the bridge. Failed to approve the bridge from the inbox. Please try the configuration flow again.
");
+ } else if (ServletUtil.isParameterEnabled(request, BRIDGE_RECONFIGURATION_FAILED_PARAMETER_NAME)) {
+ return skeleton.replace(ERROR_MESSAGE_TEXT_PLACEHOLDER,
+ "Could not auto reconfigure the bridge. Bridge thing or thing handler is not available. Please try the configuration flow again.
");
+ } else {
+ return skeleton.replace(ERROR_MESSAGE_TEXT_PLACEHOLDER, "");
+ }
+ }
+
+ private String renderBridgeUid(String skeleton, ThingUID bridgeUid) {
+ return skeleton.replace(BRIDGE_UID_PLACEHOLDER, bridgeUid.getAsString());
+ }
+
+ private String renderEmail(String skeleton, String email) {
+ return skeleton.replace(EMAIL_PLACEHOLDER, email);
+ }
+
+ private String renderLocaleSelection(String skeleton) {
+ String preSelectedLanguage = languageProvider.getLanguage().filter(SUPPORTED_LANGUAGES::contains)
+ .orElse(DEFAULT_LANGUAGE);
+
+ return skeleton.replace(LOCALE_OPTIONS_PLACEHOLDER,
+ SUPPORTED_LANGUAGES.stream().map(Language::fromCode).filter(Optional::isPresent).map(Optional::get)
+ .sorted()
+ .map(language -> createOptionTag(language, preSelectedLanguage.equals(language.getCode())))
+ .collect(Collectors.joining("\n")));
+ }
+
+ private String createOptionTag(Language language, boolean selected) {
+ String firstPart = " ";
+ if (selected) {
+ return firstPart + " selected=\"selected\"" + secondPart;
+ } else {
+ return firstPart + secondPart;
+ }
+ }
+
+ private String renderBridgeConfigurationTemplate(String skeleton, ThingUID bridgeUid, String email) {
+ String bridgeTemplate = templateGenerator.createBridgeConfigurationTemplate(bridgeUid.getId(), email,
+ languageProvider.getLanguage().orElse("en"));
+ return skeleton.replace(THINGS_TEMPLATE_CODE_PLACEHOLDER, bridgeTemplate);
+ }
+
+ /**
+ * A language representation for user display.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+ private static final class Language implements Comparable {
+ private final String code;
+ private final String name;
+
+ private Language(String code, String name) {
+ this.code = code;
+ this.name = name;
+ }
+
+ /**
+ * Gets the 2-letter language code for accessing the Miele Cloud service.
+ */
+ public String getCode() {
+ return code;
+ }
+
+ /**
+ * Formats the language for displaying.
+ */
+ public String format() {
+ return name + " - " + code;
+ }
+
+ @Override
+ public int compareTo(Language other) {
+ return name.toUpperCase().compareTo(other.name.toUpperCase());
+ }
+
+ /**
+ * Constructs a {@link Language} from a 2-letter language code.
+ *
+ * @param code 2-letter language code.
+ * @return An {@link Optional} wrapping the {@link Language} or an empty {@link Optional} if there is no
+ * representation for the given language code.
+ */
+ public static Optional fromCode(String code) {
+ Locale locale = new Locale(code);
+ String name = locale.getDisplayLanguage(locale);
+ if (name.isEmpty()) {
+ return Optional.empty();
+ } else {
+ return Optional.of(new Language(code, name));
+ }
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/discovery/ThingDiscoveryService.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/discovery/ThingDiscoveryService.java
new file mode 100644
index 00000000000..08c8187d62c
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/discovery/ThingDiscoveryService.java
@@ -0,0 +1,214 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.discovery;
+
+import static org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.*;
+import static org.openhab.binding.mielecloud.internal.handler.MieleHandlerFactory.SUPPORTED_THING_TYPES;
+
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.mielecloud.internal.handler.MieleBridgeHandler;
+import org.openhab.binding.mielecloud.internal.webservice.api.DeviceState;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.DeviceType;
+import org.openhab.core.config.discovery.AbstractDiscoveryService;
+import org.openhab.core.config.discovery.DiscoveryResult;
+import org.openhab.core.config.discovery.DiscoveryResultBuilder;
+import org.openhab.core.config.discovery.DiscoveryService;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.thing.ThingTypeUID;
+import org.openhab.core.thing.ThingUID;
+import org.openhab.core.thing.binding.ThingHandler;
+import org.openhab.core.thing.binding.ThingHandlerService;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * Discovery service for things linked to a Miele cloud account.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Do not directly listen to webservice events
+ */
+@NonNullByDefault
+public class ThingDiscoveryService extends AbstractDiscoveryService implements DiscoveryService, ThingHandlerService {
+ private static final int BACKGROUND_DISCOVERY_TIMEOUT_IN_SECONDS = 5;
+
+ @Nullable
+ private MieleBridgeHandler bridgeHandler;
+
+ private final Logger logger = LoggerFactory.getLogger(this.getClass());
+
+ private boolean discoveringDevices = false;
+
+ /**
+ * Creates a new {@link ThingDiscoveryService}.
+ */
+ public ThingDiscoveryService() {
+ super(SUPPORTED_THING_TYPES, BACKGROUND_DISCOVERY_TIMEOUT_IN_SECONDS);
+ }
+
+ @Nullable
+ private ThingUID getBridgeUid() {
+ var bridgeHandler = this.bridgeHandler;
+ if (bridgeHandler == null) {
+ return null;
+ } else {
+ return bridgeHandler.getThing().getUID();
+ }
+ }
+
+ @Override
+ protected void startScan() {
+ }
+
+ @Override
+ public void activate() {
+ startBackgroundDiscovery();
+ }
+
+ @Override
+ public void deactivate() {
+ stopBackgroundDiscovery();
+ removeOlderResults(System.currentTimeMillis(), getBridgeUid());
+ }
+
+ /**
+ * Invoked when a device state update is received from the Miele cloud.
+ */
+ public void onDeviceStateUpdated(DeviceState deviceState) {
+ if (!discoveringDevices) {
+ return;
+ }
+
+ Optional thingTypeUid = getThingTypeUID(deviceState);
+ if (thingTypeUid.isPresent()) {
+ createDiscoveryResult(deviceState, thingTypeUid.get());
+ } else {
+ logger.debug("Unsupported Miele device type: {}", deviceState.getType().orElse(""));
+ }
+ }
+
+ private void createDiscoveryResult(DeviceState deviceState, ThingTypeUID thingTypeUid) {
+ MieleBridgeHandler bridgeHandler = this.bridgeHandler;
+ if (bridgeHandler == null) {
+ return;
+ }
+
+ ThingUID thingUid = new ThingUID(thingTypeUid, bridgeHandler.getThing().getUID(),
+ deviceState.getDeviceIdentifier());
+
+ DiscoveryResultBuilder discoveryResultBuilder = DiscoveryResultBuilder.create(thingUid)
+ .withBridge(bridgeHandler.getThing().getUID()).withRepresentationProperty(Thing.PROPERTY_SERIAL_NUMBER)
+ .withLabel(getLabel(deviceState));
+
+ ThingInformationExtractor.extractProperties(thingTypeUid, deviceState).entrySet()
+ .forEach(entry -> discoveryResultBuilder.withProperty(entry.getKey(), entry.getValue()));
+
+ DiscoveryResult result = discoveryResultBuilder.build();
+
+ thingDiscovered(result);
+ }
+
+ private Optional getThingTypeUID(DeviceState deviceState) {
+ switch (deviceState.getRawType()) {
+ case COFFEE_SYSTEM:
+ return Optional.of(THING_TYPE_COFFEE_SYSTEM);
+ case TUMBLE_DRYER:
+ return Optional.of(THING_TYPE_DRYER);
+ case WASHING_MACHINE:
+ return Optional.of(THING_TYPE_WASHING_MACHINE);
+ case WASHER_DRYER:
+ return Optional.of(THING_TYPE_WASHER_DRYER);
+ case FREEZER:
+ return Optional.of(THING_TYPE_FREEZER);
+ case FRIDGE:
+ return Optional.of(THING_TYPE_FRIDGE);
+ case FRIDGE_FREEZER_COMBINATION:
+ return Optional.of(THING_TYPE_FRIDGE_FREEZER);
+ case HOB_INDUCTION:
+ case HOB_HIGHLIGHT:
+ return Optional.of(THING_TYPE_HOB);
+ case DISHWASHER:
+ return Optional.of(THING_TYPE_DISHWASHER);
+ case OVEN:
+ case OVEN_MICROWAVE:
+ case STEAM_OVEN:
+ case STEAM_OVEN_COMBINATION:
+ case STEAM_OVEN_MICROWAVE_COMBINATION:
+ case DIALOGOVEN:
+ return Optional.of(THING_TYPE_OVEN);
+ case WINE_CABINET:
+ case WINE_STORAGE_CONDITIONING_UNIT:
+ case WINE_CONDITIONING_UNIT:
+ case WINE_CABINET_FREEZER_COMBINATION:
+ return Optional.of(THING_TYPE_WINE_STORAGE);
+ case HOOD:
+ return Optional.of(THING_TYPE_HOOD);
+ case DISH_WARMER:
+ return Optional.of(THING_TYPE_DISH_WARMER);
+ case VACUUM_CLEANER:
+ return Optional.of(THING_TYPE_ROBOTIC_VACUUM_CLEANER);
+
+ default:
+ if (deviceState.getRawType() != DeviceType.UNKNOWN) {
+ logger.warn("Found no matching thing type for device type {}", deviceState.getRawType());
+ }
+ return Optional.empty();
+ }
+ }
+
+ @Override
+ protected void startBackgroundDiscovery() {
+ logger.debug("Starting background discovery");
+
+ removeOlderResults(System.currentTimeMillis(), getBridgeUid());
+ discoveringDevices = true;
+ }
+
+ @Override
+ protected void stopBackgroundDiscovery() {
+ logger.debug("Stopping background discovery");
+ discoveringDevices = false;
+ }
+
+ /**
+ * Invoked when a device is removed from the Miele cloud.
+ */
+ public void onDeviceRemoved(String deviceIdentifier) {
+ removeOlderResults(System.currentTimeMillis(), getBridgeUid());
+ }
+
+ private String getLabel(DeviceState deviceState) {
+ Optional deviceName = deviceState.getDeviceName();
+ if (deviceName.isPresent()) {
+ return deviceName.get();
+ }
+
+ return ThingInformationExtractor.getDeviceAndTechType(deviceState).orElse("Miele Device");
+ }
+
+ @Override
+ public void setThingHandler(ThingHandler handler) {
+ if (handler instanceof MieleBridgeHandler) {
+ var bridgeHandler = (MieleBridgeHandler) handler;
+ bridgeHandler.setDiscoveryService(this);
+ this.bridgeHandler = bridgeHandler;
+ }
+ }
+
+ @Override
+ public @Nullable ThingHandler getThingHandler() {
+ return bridgeHandler;
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/discovery/ThingInformationExtractor.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/discovery/ThingInformationExtractor.java
new file mode 100644
index 00000000000..52d93b0c955
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/discovery/ThingInformationExtractor.java
@@ -0,0 +1,93 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.discovery;
+
+import java.util.HashMap;
+import java.util.Map;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants;
+import org.openhab.binding.mielecloud.internal.webservice.api.DeviceState;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.thing.ThingTypeUID;
+
+/**
+ * Helper class extracting information related to things from {@link DeviceState}s received from the Miele cloud.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public final class ThingInformationExtractor {
+ private ThingInformationExtractor() {
+ throw new IllegalStateException(getClass().getName() + " cannot be instantiated");
+ }
+
+ /**
+ * Extracts thing properties from a {@link DeviceState}.
+ *
+ * The returned properties always contain {@link Thing#PROPERTY_SERIAL_NUMBER} and {@link Thing#PROPERTY_MODEL_ID}.
+ * More might be present depending on the type of device.
+ *
+ * @param thingTypeUid {@link ThingTypeUID} of the thing to extract properties for.
+ * @param deviceState {@link DeviceState} received from the Miele cloud.
+ * @return A {@link Map} holding the properties as key-value pairs.
+ */
+ public static Map extractProperties(ThingTypeUID thingTypeUid, DeviceState deviceState) {
+ var propertyMap = new HashMap();
+ propertyMap.put(Thing.PROPERTY_SERIAL_NUMBER, getSerialNumber(deviceState));
+ propertyMap.put(Thing.PROPERTY_MODEL_ID, getModelId(deviceState));
+ propertyMap.put(MieleCloudBindingConstants.CONFIG_PARAM_DEVICE_IDENTIFIER, deviceState.getDeviceIdentifier());
+
+ if (MieleCloudBindingConstants.THING_TYPE_HOB.equals(thingTypeUid)) {
+ deviceState.getPlateStepCount().ifPresent(plateCount -> propertyMap
+ .put(MieleCloudBindingConstants.PROPERTY_PLATE_COUNT, plateCount.toString()));
+ }
+
+ return propertyMap;
+ }
+
+ private static String getSerialNumber(DeviceState deviceState) {
+ return deviceState.getFabNumber().orElse(deviceState.getDeviceIdentifier());
+ }
+
+ private static String getModelId(DeviceState deviceState) {
+ return getDeviceAndTechType(deviceState).orElse("Unknown");
+ }
+
+ /**
+ * Formats device type and tech type from the given {@link DeviceState} for the purpose of displaying then to the
+ * user.
+ *
+ * If either of device or tech type is missing then it will be omitted. If both are missing then an empty
+ * {@link Optional} will be returned.
+ *
+ * @param deviceState {@link DeviceState} obtained from the Miele cloud.
+ * @return An {@link Optional} holding the formatted value or an empty {@link Optional} if neither device type nor
+ * tech type were present.
+ */
+ static Optional getDeviceAndTechType(DeviceState deviceState) {
+ Optional deviceType = deviceState.getType();
+ Optional techType = deviceState.getTechType();
+ if (deviceType.isPresent() && techType.isPresent()) {
+ return Optional.of(deviceType.get() + " " + techType.get());
+ }
+ if (!deviceType.isPresent() && techType.isPresent()) {
+ return techType;
+ }
+ if (deviceType.isPresent() && !techType.isPresent()) {
+ return deviceType;
+ }
+ return Optional.empty();
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/AbstractMieleThingHandler.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/AbstractMieleThingHandler.java
new file mode 100644
index 00000000000..b1b94a0ec1b
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/AbstractMieleThingHandler.java
@@ -0,0 +1,293 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler;
+
+import static org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.Channels.*;
+import static org.openhab.binding.mielecloud.internal.webservice.api.PowerStatus.*;
+import static org.openhab.binding.mielecloud.internal.webservice.api.ProgramStatus.*;
+import static org.openhab.binding.mielecloud.internal.webservice.api.json.ProcessAction.*;
+
+import java.util.Optional;
+import java.util.function.Consumer;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants;
+import org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.I18NKeys;
+import org.openhab.binding.mielecloud.internal.discovery.ThingInformationExtractor;
+import org.openhab.binding.mielecloud.internal.handler.channel.ActionsChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.DeviceChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.TransitionChannelState;
+import org.openhab.binding.mielecloud.internal.webservice.ActionStateFetcher;
+import org.openhab.binding.mielecloud.internal.webservice.MieleWebservice;
+import org.openhab.binding.mielecloud.internal.webservice.UnavailableMieleWebservice;
+import org.openhab.binding.mielecloud.internal.webservice.api.ActionsState;
+import org.openhab.binding.mielecloud.internal.webservice.api.DeviceState;
+import org.openhab.binding.mielecloud.internal.webservice.api.PowerStatus;
+import org.openhab.binding.mielecloud.internal.webservice.api.ProgramStatus;
+import org.openhab.binding.mielecloud.internal.webservice.api.TransitionState;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.ProcessAction;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.StateType;
+import org.openhab.binding.mielecloud.internal.webservice.exception.TooManyRequestsException;
+import org.openhab.core.library.types.OnOffType;
+import org.openhab.core.thing.Bridge;
+import org.openhab.core.thing.ChannelUID;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.thing.ThingStatus;
+import org.openhab.core.thing.ThingStatusDetail;
+import org.openhab.core.thing.binding.BaseThingHandler;
+import org.openhab.core.thing.binding.BridgeHandler;
+import org.openhab.core.types.Command;
+import org.openhab.core.types.RefreshType;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * Abstract base class for all Miele thing handlers.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Add channel state wrappers
+ */
+@NonNullByDefault
+public abstract class AbstractMieleThingHandler extends BaseThingHandler {
+ protected final ActionStateFetcher actionFetcher;
+ protected DeviceState latestDeviceState = new DeviceState(getDeviceId(), null);
+ protected TransitionState latestTransitionState = new TransitionState(null, latestDeviceState);
+ protected ActionsState latestActionsState = new ActionsState(getDeviceId(), null);
+
+ private final Logger logger = LoggerFactory.getLogger(this.getClass());
+
+ /**
+ * Creates a new {@link AbstractMieleThingHandler}.
+ *
+ * @param thing The thing to handle.
+ */
+ public AbstractMieleThingHandler(Thing thing) {
+ super(thing);
+ this.actionFetcher = new ActionStateFetcher(this::getWebservice, scheduler);
+ }
+
+ private Optional getMieleBridgeHandler() {
+ Bridge bridge = getBridge();
+ if (bridge == null) {
+ return Optional.empty();
+ }
+
+ BridgeHandler handler = bridge.getHandler();
+ if (handler == null || !(handler instanceof MieleBridgeHandler)) {
+ return Optional.empty();
+ }
+
+ return Optional.of((MieleBridgeHandler) handler);
+ }
+
+ protected MieleWebservice getWebservice() {
+ return getMieleBridgeHandler().map(MieleBridgeHandler::getWebservice)
+ .orElse(UnavailableMieleWebservice.INSTANCE);
+ }
+
+ @Override
+ public void initialize() {
+ getWebservice().dispatchDeviceState(getDeviceId());
+
+ // If no device state update was received so far, set the device to OFFLINE.
+ if (getThing().getStatus() == ThingStatus.INITIALIZING) {
+ updateStatus(ThingStatus.OFFLINE);
+ }
+ }
+
+ @Override
+ public void handleCommand(ChannelUID channelUID, Command command) {
+ if (RefreshType.REFRESH.equals(command)) {
+ updateDeviceState(new DeviceChannelState(latestDeviceState));
+ updateTransitionState(new TransitionChannelState(latestTransitionState));
+ updateActionState(new ActionsChannelState(latestActionsState));
+ }
+
+ switch (channelUID.getId()) {
+ case PROGRAM_START_STOP:
+ if (PROGRAM_STARTED.matches(command.toString())) {
+ triggerProcessAction(START);
+ } else if (PROGRAM_STOPPED.matches(command.toString())) {
+ triggerProcessAction(STOP);
+ }
+ break;
+
+ case PROGRAM_START_STOP_PAUSE:
+ if (PROGRAM_STARTED.matches(command.toString())) {
+ triggerProcessAction(START);
+ } else if (PROGRAM_STOPPED.matches(command.toString())) {
+ triggerProcessAction(STOP);
+ } else if (PROGRAM_PAUSED.matches(command.toString())) {
+ triggerProcessAction(PAUSE);
+ }
+ break;
+
+ case LIGHT_SWITCH:
+ if (command instanceof OnOffType) {
+ triggerLight(OnOffType.ON.equals(command));
+ }
+ break;
+
+ case POWER_ON_OFF:
+ if (POWER_ON.matches(command.toString()) || POWER_OFF.matches(command.toString())) {
+ triggerPowerState(OnOffType.ON.equals(OnOffType.from(command.toString())));
+ }
+ break;
+ }
+ }
+
+ @Override
+ public void dispose() {
+ }
+
+ /**
+ * Invoked when an update of the available actions for the device managed by this handler is received from the Miele
+ * cloud.
+ */
+ public final void onProcessActionUpdated(ActionsState actionState) {
+ latestActionsState = actionState;
+ updateActionState(new ActionsChannelState(latestActionsState));
+ }
+
+ /**
+ * Invoked when the device managed by this handler was removed from the Miele cloud.
+ */
+ public final void onDeviceRemoved() {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.GONE, I18NKeys.THING_STATUS_DESCRIPTION_REMOVED);
+ }
+
+ /**
+ * Invoked when a device state update for the device managed by this handler is received from the Miele cloud.
+ */
+ public final void onDeviceStateUpdated(DeviceState deviceState) {
+ actionFetcher.onDeviceStateUpdated(deviceState);
+
+ latestTransitionState = new TransitionState(latestTransitionState, deviceState);
+ latestDeviceState = deviceState;
+
+ updateThingProperties(deviceState);
+ updateDeviceState(new DeviceChannelState(latestDeviceState));
+ updateTransitionState(new TransitionChannelState(latestTransitionState));
+ updateThingStatus(latestDeviceState);
+ }
+
+ protected void triggerProcessAction(final ProcessAction processAction) {
+ performPutAction(() -> getWebservice().putProcessAction(getDeviceId(), processAction),
+ t -> logger.warn("Failed to perform '{}' operation for device '{}'.", processAction, getDeviceId(), t));
+ }
+
+ protected void triggerLight(final boolean on) {
+ performPutAction(() -> getWebservice().putLight(getDeviceId(), on),
+ t -> logger.warn("Failed to set light state to '{}' for device '{}'.", on, getDeviceId(), t));
+ }
+
+ protected void triggerPowerState(final boolean on) {
+ performPutAction(() -> getWebservice().putPowerState(getDeviceId(), on),
+ t -> logger.warn("Failed to set the power state to '{}' for device '{}'.", on, getDeviceId(), t));
+ }
+
+ protected void triggerProgram(final long programId) {
+ performPutAction(() -> getWebservice().putProgram(getDeviceId(), programId), t -> logger
+ .warn("Failed to activate program with ID '{}' for device '{}'.", programId, getDeviceId(), t));
+ }
+
+ private void performPutAction(Runnable action, Consumer onError) {
+ scheduler.execute(() -> {
+ try {
+ action.run();
+ } catch (TooManyRequestsException e) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR,
+ I18NKeys.THING_STATUS_DESCRIPTION_RATELIMIT);
+ onError.accept(e);
+ } catch (Exception e) {
+ onError.accept(e);
+ }
+ });
+ }
+
+ protected final String getDeviceId() {
+ return getConfig().get(MieleCloudBindingConstants.CONFIG_PARAM_DEVICE_IDENTIFIER).toString();
+ }
+
+ /**
+ * Creates a {@link ChannelUID} from the given name.
+ *
+ * @param name channel name
+ * @return {@link ChannelUID}
+ */
+ protected ChannelUID channel(String name) {
+ return new ChannelUID(getThing().getUID(), name);
+ }
+
+ /**
+ * Updates the thing status depending on whether the managed device is connected and reachable.
+ */
+ private void updateThingStatus(DeviceState deviceState) {
+ if (deviceState.isInState(StateType.NOT_CONNECTED)) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR,
+ I18NKeys.THING_STATUS_DESCRIPTION_DISCONNECTED);
+ } else {
+ updateStatus(ThingStatus.ONLINE);
+ }
+ }
+
+ /**
+ * Determines the status of the currently selected program.
+ */
+ protected ProgramStatus getProgramStatus(StateType rawStatus) {
+ if (rawStatus.equals(StateType.RUNNING)) {
+ return PROGRAM_STARTED;
+ }
+ return PROGRAM_STOPPED;
+ }
+
+ /**
+ * Determines the power status of the managed device.
+ */
+ protected PowerStatus getPowerStatus(StateType rawStatus) {
+ if (rawStatus.equals(StateType.OFF) || rawStatus.equals(StateType.NOT_CONNECTED)) {
+ return POWER_OFF;
+ }
+ return POWER_ON;
+ }
+
+ /**
+ * Updates the thing properties. This is necessary if properties have not been set during discovery.
+ */
+ private void updateThingProperties(DeviceState deviceState) {
+ var properties = editProperties();
+ properties.putAll(ThingInformationExtractor.extractProperties(getThing().getThingTypeUID(), deviceState));
+ updateProperties(properties);
+ }
+
+ /**
+ * Updates the device state channels.
+ *
+ * @param device The {@link DeviceChannelState} information to update the device channel states with.
+ */
+ protected abstract void updateDeviceState(DeviceChannelState device);
+
+ /**
+ * Updates the transition state channels.
+ *
+ * @param transition The {@link TransitionChannelState} information to update the transition channel states with.
+ */
+ protected abstract void updateTransitionState(TransitionChannelState transition);
+
+ /**
+ * Updates the device action state channels.
+ *
+ * @param action The {@link ActionsChannelState} information to update the action channel states with.
+ */
+ protected abstract void updateActionState(ActionsChannelState actions);
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/CoffeeSystemThingHandler.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/CoffeeSystemThingHandler.java
new file mode 100644
index 00000000000..ca4557c2c60
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/CoffeeSystemThingHandler.java
@@ -0,0 +1,75 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler;
+
+import static org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.Channels.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.handler.channel.ActionsChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.DeviceChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.TransitionChannelState;
+import org.openhab.core.library.types.OnOffType;
+import org.openhab.core.thing.Thing;
+
+/**
+ * ThingHandler implementation for the Miele coffee devices.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Switch from polling to SSE, add channel state wrappers
+ * @author Benjamin Bolte - Add info state channel and map signal flags from API
+ * @author Björn Lange - Add elapsed time channel
+ */
+@NonNullByDefault
+public class CoffeeSystemThingHandler extends AbstractMieleThingHandler {
+ /**
+ * Creates a new {@link CoffeeSystemThingHandler}.
+ *
+ * @param thing The thing to handle.
+ */
+ public CoffeeSystemThingHandler(Thing thing) {
+ super(thing);
+
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STARTED), OnOffType.OFF);
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STOPPED), OnOffType.OFF);
+ }
+
+ @Override
+ protected void updateDeviceState(DeviceChannelState device) {
+ updateState(channel(PROGRAM_ACTIVE), device.getProgramActive());
+ updateState(channel(PROGRAM_ACTIVE_RAW), device.getProgramActiveRaw());
+ updateState(channel(PROGRAM_PHASE), device.getProgramPhase());
+ updateState(channel(PROGRAM_PHASE_RAW), device.getProgramPhaseRaw());
+ updateState(channel(OPERATION_STATE), device.getOperationState());
+ updateState(channel(OPERATION_STATE_RAW), device.getOperationStateRaw());
+ updateState(channel(PROGRAM_ELAPSED_TIME), device.getProgramElapsedTime());
+ updateState(channel(POWER_ON_OFF), device.getPowerOnOff());
+ updateState(channel(ERROR_STATE), device.getErrorState());
+ updateState(channel(INFO_STATE), device.getInfoState());
+ updateState(channel(LIGHT_SWITCH), device.getLightSwitch());
+ }
+
+ @Override
+ protected void updateTransitionState(TransitionChannelState transition) {
+ updateState(channel(PROGRAM_REMAINING_TIME), transition.getProgramRemainingTime());
+ if (transition.hasFinishedChanged()) {
+ updateState(channel(FINISH_STATE), transition.getFinishState());
+ }
+ }
+
+ @Override
+ protected void updateActionState(ActionsChannelState actions) {
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_ON), actions.getRemoteControlCanBeSwitchedOn());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_OFF), actions.getRemoteControlCanBeSwitchedOff());
+ updateState(channel(LIGHT_CAN_BE_CONTROLLED), actions.getLightCanBeControlled());
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/CoolingDeviceThingHandler.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/CoolingDeviceThingHandler.java
new file mode 100644
index 00000000000..03539f11b4d
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/CoolingDeviceThingHandler.java
@@ -0,0 +1,91 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler;
+
+import static org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.Channels.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.handler.channel.ActionsChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.DeviceChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.TransitionChannelState;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.ProcessAction;
+import org.openhab.core.library.types.OnOffType;
+import org.openhab.core.thing.ChannelUID;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.types.Command;
+
+/**
+ * ThingHandler implementation for the Miele cooling devices.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Add channel state wrappers
+ * @author Benjamin Bolte - Add door state and door alarm, add info state channel and map signal flags from API
+ */
+@NonNullByDefault
+public class CoolingDeviceThingHandler extends AbstractMieleThingHandler {
+ /**
+ * Creates a new {@link CoolingDeviceThingHandler}.
+ *
+ * @param thing The thing to handle.
+ */
+ public CoolingDeviceThingHandler(Thing thing) {
+ super(thing);
+ }
+
+ @Override
+ public void handleCommand(ChannelUID channelUID, Command command) {
+ super.handleCommand(channelUID, command);
+
+ if (!OnOffType.ON.equals(command) && !OnOffType.OFF.equals(command)) {
+ return;
+ }
+
+ switch (channelUID.getId()) {
+ case FRIDGE_SUPER_COOL:
+ triggerProcessAction(OnOffType.ON.equals(command) ? ProcessAction.START_SUPERCOOLING
+ : ProcessAction.STOP_SUPERCOOLING);
+ break;
+
+ case FREEZER_SUPER_FREEZE:
+ triggerProcessAction(OnOffType.ON.equals(command) ? ProcessAction.START_SUPERFREEZING
+ : ProcessAction.STOP_SUPERFREEZING);
+ break;
+ }
+ }
+
+ @Override
+ protected void updateDeviceState(DeviceChannelState device) {
+ updateState(channel(OPERATION_STATE), device.getOperationState());
+ updateState(channel(OPERATION_STATE_RAW), device.getOperationStateRaw());
+ updateState(channel(FRIDGE_SUPER_COOL), device.getFridgeSuperCool());
+ updateState(channel(FREEZER_SUPER_FREEZE), device.getFreezerSuperFreeze());
+ updateState(channel(FRIDGE_TEMPERATURE_TARGET), device.getFridgeTemperatureTarget());
+ updateState(channel(FREEZER_TEMPERATURE_TARGET), device.getFreezerTemperatureTarget());
+ updateState(channel(FRIDGE_TEMPERATURE_CURRENT), device.getFridgeTemperatureCurrent());
+ updateState(channel(FREEZER_TEMPERATURE_CURRENT), device.getFreezerTemperatureCurrent());
+ updateState(channel(ERROR_STATE), device.getErrorState());
+ updateState(channel(INFO_STATE), device.getInfoState());
+ updateState(channel(DOOR_STATE), device.getDoorState());
+ updateState(channel(DOOR_ALARM), device.getDoorAlarm());
+ }
+
+ @Override
+ protected void updateTransitionState(TransitionChannelState transition) {
+ }
+
+ @Override
+ protected void updateActionState(ActionsChannelState actions) {
+ updateState(channel(SUPER_COOL_CAN_BE_CONTROLLED), actions.getSuperCoolCanBeControlled());
+ updateState(channel(SUPER_FREEZE_CAN_BE_CONTROLLED), actions.getSuperFreezeCanBeControlled());
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/DishWarmerDeviceThingHandler.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/DishWarmerDeviceThingHandler.java
new file mode 100644
index 00000000000..62da3a1c515
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/DishWarmerDeviceThingHandler.java
@@ -0,0 +1,88 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler;
+
+import static org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.Channels.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.handler.channel.ActionsChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.DeviceChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.TransitionChannelState;
+import org.openhab.core.library.types.StringType;
+import org.openhab.core.thing.ChannelUID;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.types.Command;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * ThingHandler implementation for Miele dish warmers.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class DishWarmerDeviceThingHandler extends AbstractMieleThingHandler {
+ private final Logger logger = LoggerFactory.getLogger(this.getClass());
+
+ /**
+ * Creates a new {@link DishWarmerDeviceThingHandler}.
+ *
+ * @param thing The thing to handle.
+ */
+ public DishWarmerDeviceThingHandler(Thing thing) {
+ super(thing);
+ }
+
+ @Override
+ public void handleCommand(ChannelUID channelUID, Command command) {
+ super.handleCommand(channelUID, command);
+
+ if (DISH_WARMER_PROGRAM_ACTIVE.equals(channelUID.getId()) && command instanceof StringType) {
+ try {
+ triggerProgram(Long.parseLong(command.toString()));
+ } catch (NumberFormatException e) {
+ logger.warn("Failed to activate program: '{}' is not a valid program ID", command.toString());
+ }
+ }
+ }
+
+ @Override
+ protected void updateDeviceState(DeviceChannelState device) {
+ updateState(channel(DISH_WARMER_PROGRAM_ACTIVE), device.getProgramActiveId());
+ updateState(channel(PROGRAM_ACTIVE_RAW), device.getProgramActiveRaw());
+ updateState(channel(OPERATION_STATE), device.getOperationState());
+ updateState(channel(OPERATION_STATE_RAW), device.getOperationStateRaw());
+ updateState(channel(POWER_ON_OFF), device.getPowerOnOff());
+ updateState(channel(PROGRAM_ELAPSED_TIME), device.getProgramElapsedTime());
+ updateState(channel(TEMPERATURE_TARGET), device.getTemperatureTarget());
+ updateState(channel(TEMPERATURE_CURRENT), device.getTemperatureCurrent());
+ updateState(channel(ERROR_STATE), device.getErrorState());
+ updateState(channel(INFO_STATE), device.getInfoState());
+ updateState(channel(DOOR_STATE), device.getDoorState());
+ }
+
+ @Override
+ protected void updateTransitionState(TransitionChannelState transition) {
+ updateState(channel(PROGRAM_REMAINING_TIME), transition.getProgramRemainingTime());
+ updateState(channel(PROGRAM_PROGRESS), transition.getProgramProgress());
+ if (transition.hasFinishedChanged()) {
+ updateState(channel(FINISH_STATE), transition.getFinishState());
+ }
+ }
+
+ @Override
+ protected void updateActionState(ActionsChannelState actions) {
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_ON), actions.getRemoteControlCanBeSwitchedOn());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_OFF), actions.getRemoteControlCanBeSwitchedOff());
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/DishwasherDeviceThingHandler.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/DishwasherDeviceThingHandler.java
new file mode 100644
index 00000000000..5c48e78a1f9
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/DishwasherDeviceThingHandler.java
@@ -0,0 +1,75 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler;
+
+import static org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.Channels.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.handler.channel.ActionsChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.DeviceChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.TransitionChannelState;
+import org.openhab.core.thing.Thing;
+
+/**
+ * ThingHandler implementation for the Miele dishwasher devices.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Add channel state wrappers
+ * @author Benjamin Bolte - Add info state channel and map signal flags from API
+ * @author Björn Lange - Add elapsed time channel
+ */
+@NonNullByDefault
+public class DishwasherDeviceThingHandler extends AbstractMieleThingHandler {
+ /**
+ * Creates a new {@link DishwasherDeviceThingHandler}.
+ *
+ * @param thing The thing to handle.
+ */
+ public DishwasherDeviceThingHandler(Thing thing) {
+ super(thing);
+ }
+
+ @Override
+ protected void updateDeviceState(DeviceChannelState device) {
+ updateState(channel(PROGRAM_ACTIVE), device.getProgramActive());
+ updateState(channel(PROGRAM_ACTIVE_RAW), device.getProgramActiveRaw());
+ updateState(channel(PROGRAM_PHASE), device.getProgramPhase());
+ updateState(channel(PROGRAM_PHASE_RAW), device.getProgramPhaseRaw());
+ updateState(channel(OPERATION_STATE), device.getOperationState());
+ updateState(channel(OPERATION_STATE_RAW), device.getOperationStateRaw());
+ updateState(channel(PROGRAM_START_STOP), device.getProgramStartStop());
+ updateState(channel(POWER_ON_OFF), device.getPowerOnOff());
+ updateState(channel(DELAYED_START_TIME), device.getDelayedStartTime());
+ updateState(channel(PROGRAM_ELAPSED_TIME), device.getProgramElapsedTime());
+ updateState(channel(ERROR_STATE), device.getErrorState());
+ updateState(channel(INFO_STATE), device.getInfoState());
+ updateState(channel(DOOR_STATE), device.getDoorState());
+ }
+
+ @Override
+ protected void updateTransitionState(TransitionChannelState transition) {
+ updateState(channel(PROGRAM_REMAINING_TIME), transition.getProgramRemainingTime());
+ updateState(channel(PROGRAM_PROGRESS), transition.getProgramProgress());
+ if (transition.hasFinishedChanged()) {
+ updateState(channel(FINISH_STATE), transition.getFinishState());
+ }
+ }
+
+ @Override
+ protected void updateActionState(ActionsChannelState actions) {
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STARTED), actions.getRemoteControlCanBeStarted());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STOPPED), actions.getRemoteControlCanBeStopped());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_ON), actions.getRemoteControlCanBeSwitchedOn());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_OFF), actions.getRemoteControlCanBeSwitchedOff());
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/DryerDeviceThingHandler.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/DryerDeviceThingHandler.java
new file mode 100644
index 00000000000..7c5354163c8
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/DryerDeviceThingHandler.java
@@ -0,0 +1,79 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler;
+
+import static org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.Channels.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.handler.channel.ActionsChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.DeviceChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.TransitionChannelState;
+import org.openhab.core.thing.Thing;
+
+/**
+ * ThingHandler implementation for the Miele dryer and washingDryer devices.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Add channel state wrappers
+ * @author Benjamin Bolte - Add info state channel and map signal flags from API
+ * @author Björn Lange - Add elapsed time channel
+ */
+@NonNullByDefault
+public class DryerDeviceThingHandler extends AbstractMieleThingHandler {
+ /**
+ * Creates a new {@link DryerDeviceThingHandler}.
+ *
+ * @param thing The thing to handle.
+ */
+ public DryerDeviceThingHandler(Thing thing) {
+ super(thing);
+ }
+
+ @Override
+ protected void updateDeviceState(DeviceChannelState device) {
+ updateState(channel(PROGRAM_ACTIVE), device.getProgramActive());
+ updateState(channel(PROGRAM_ACTIVE_RAW), device.getProgramActiveRaw());
+ updateState(channel(PROGRAM_PHASE), device.getProgramPhase());
+ updateState(channel(PROGRAM_PHASE_RAW), device.getProgramPhaseRaw());
+ updateState(channel(OPERATION_STATE), device.getOperationState());
+ updateState(channel(OPERATION_STATE_RAW), device.getOperationStateRaw());
+ updateState(channel(PROGRAM_START_STOP), device.getProgramStartStop());
+ updateState(channel(DELAYED_START_TIME), device.getDelayedStartTime());
+ updateState(channel(PROGRAM_ELAPSED_TIME), device.getProgramElapsedTime());
+ updateState(channel(DRYING_TARGET), device.getDryingTarget());
+ updateState(channel(DRYING_TARGET_RAW), device.getDryingTargetRaw());
+ updateState(channel(POWER_ON_OFF), device.getPowerOnOff());
+ updateState(channel(ERROR_STATE), device.getErrorState());
+ updateState(channel(INFO_STATE), device.getInfoState());
+ updateState(channel(LIGHT_SWITCH), device.getLightSwitch());
+ updateState(channel(DOOR_STATE), device.getDoorState());
+ }
+
+ @Override
+ protected void updateTransitionState(TransitionChannelState transition) {
+ updateState(channel(PROGRAM_REMAINING_TIME), transition.getProgramRemainingTime());
+ updateState(channel(PROGRAM_PROGRESS), transition.getProgramProgress());
+ if (transition.hasFinishedChanged()) {
+ updateState(channel(FINISH_STATE), transition.getFinishState());
+ }
+ }
+
+ @Override
+ protected void updateActionState(ActionsChannelState actions) {
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STARTED), actions.getRemoteControlCanBeStarted());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STOPPED), actions.getRemoteControlCanBeStopped());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_ON), actions.getRemoteControlCanBeSwitchedOn());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_OFF), actions.getRemoteControlCanBeSwitchedOff());
+ updateState(channel(LIGHT_CAN_BE_CONTROLLED), actions.getLightCanBeControlled());
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/HobDeviceThingHandler.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/HobDeviceThingHandler.java
new file mode 100644
index 00000000000..0c3739c42b8
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/HobDeviceThingHandler.java
@@ -0,0 +1,70 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler;
+
+import static org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.Channels.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.handler.channel.ActionsChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.DeviceChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.TransitionChannelState;
+import org.openhab.core.thing.Thing;
+
+/**
+ * ThingHandler implementation for the Miele hob devices.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Add channel state wrappers
+ * @author Benjamin Bolte - Add plate step, add info state channel and map signal flags from API
+ */
+@NonNullByDefault
+public class HobDeviceThingHandler extends AbstractMieleThingHandler {
+ /**
+ * Creates a new {@link HobDeviceThingHandler}.
+ *
+ * @param thing The thing to handle.
+ */
+ public HobDeviceThingHandler(Thing thing) {
+ super(thing);
+ }
+
+ @Override
+ protected void updateDeviceState(DeviceChannelState device) {
+ updateState(channel(OPERATION_STATE), device.getOperationState());
+ updateState(channel(OPERATION_STATE_RAW), device.getOperationStateRaw());
+ updateState(channel(ERROR_STATE), device.getErrorState());
+ updateState(channel(INFO_STATE), device.getInfoState());
+ updateState(channel(PLATE_1_POWER_STEP), device.getPlateStep(0));
+ updateState(channel(PLATE_1_POWER_STEP_RAW), device.getPlateStepRaw(0));
+ updateState(channel(PLATE_2_POWER_STEP), device.getPlateStep(1));
+ updateState(channel(PLATE_2_POWER_STEP_RAW), device.getPlateStepRaw(1));
+ updateState(channel(PLATE_3_POWER_STEP), device.getPlateStep(2));
+ updateState(channel(PLATE_3_POWER_STEP_RAW), device.getPlateStepRaw(2));
+ updateState(channel(PLATE_4_POWER_STEP), device.getPlateStep(3));
+ updateState(channel(PLATE_4_POWER_STEP_RAW), device.getPlateStepRaw(3));
+ updateState(channel(PLATE_5_POWER_STEP), device.getPlateStep(4));
+ updateState(channel(PLATE_5_POWER_STEP_RAW), device.getPlateStepRaw(4));
+ updateState(channel(PLATE_6_POWER_STEP), device.getPlateStep(5));
+ updateState(channel(PLATE_6_POWER_STEP_RAW), device.getPlateStepRaw(5));
+ }
+
+ @Override
+ protected void updateTransitionState(TransitionChannelState transition) {
+ // No state transition required
+ }
+
+ @Override
+ protected void updateActionState(ActionsChannelState actions) {
+ // The Hob device has no trigger actions
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/HoodDeviceThingHandler.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/HoodDeviceThingHandler.java
new file mode 100644
index 00000000000..631f999a772
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/HoodDeviceThingHandler.java
@@ -0,0 +1,73 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler;
+
+import static org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.Channels.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.handler.channel.ActionsChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.DeviceChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.TransitionChannelState;
+import org.openhab.core.library.types.OnOffType;
+import org.openhab.core.thing.Thing;
+
+/**
+ * ThingHandler implementation for the Miele Hood devices.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Add channel state wrappers
+ * @author Benjamin Bolte - Add info state channel and map signal flags from API
+ */
+@NonNullByDefault
+public class HoodDeviceThingHandler extends AbstractMieleThingHandler {
+ /**
+ * Creates a new {@link HoodDeviceThingHandler}.
+ *
+ * @param thing The thing to handle.
+ */
+ public HoodDeviceThingHandler(Thing thing) {
+ super(thing);
+ }
+
+ @Override
+ public void initialize() {
+ super.initialize();
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STARTED), OnOffType.OFF);
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STOPPED), OnOffType.OFF);
+ }
+
+ @Override
+ protected void updateDeviceState(DeviceChannelState device) {
+ updateState(channel(PROGRAM_PHASE), device.getProgramPhase());
+ updateState(channel(PROGRAM_PHASE_RAW), device.getProgramPhaseRaw());
+ updateState(channel(OPERATION_STATE), device.getOperationState());
+ updateState(channel(OPERATION_STATE_RAW), device.getOperationStateRaw());
+ updateState(channel(POWER_ON_OFF), device.getPowerOnOff());
+ updateState(channel(VENTILATION_POWER), device.getVentilationPower());
+ updateState(channel(VENTILATION_POWER_RAW), device.getVentilationPowerRaw());
+ updateState(channel(ERROR_STATE), device.getErrorState());
+ updateState(channel(INFO_STATE), device.getInfoState());
+ updateState(channel(LIGHT_SWITCH), device.getLightSwitch());
+ }
+
+ @Override
+ protected void updateTransitionState(TransitionChannelState transition) {
+ }
+
+ @Override
+ protected void updateActionState(ActionsChannelState actions) {
+ updateState(channel(LIGHT_CAN_BE_CONTROLLED), actions.getLightCanBeControlled());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_ON), actions.getRemoteControlCanBeSwitchedOn());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_OFF), actions.getRemoteControlCanBeSwitchedOff());
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/MieleBridgeHandler.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/MieleBridgeHandler.java
new file mode 100644
index 00000000000..03d93ee28c3
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/MieleBridgeHandler.java
@@ -0,0 +1,362 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler;
+
+import java.util.Collection;
+import java.util.Collections;
+import java.util.Optional;
+import java.util.concurrent.CompletableFuture;
+import java.util.concurrent.ExecutionException;
+import java.util.concurrent.ScheduledExecutorService;
+import java.util.function.Consumer;
+import java.util.function.Function;
+import java.util.function.Supplier;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants;
+import org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.I18NKeys;
+import org.openhab.binding.mielecloud.internal.auth.OAuthException;
+import org.openhab.binding.mielecloud.internal.auth.OAuthTokenRefreshListener;
+import org.openhab.binding.mielecloud.internal.auth.OAuthTokenRefresher;
+import org.openhab.binding.mielecloud.internal.discovery.ThingDiscoveryService;
+import org.openhab.binding.mielecloud.internal.util.EmailValidator;
+import org.openhab.binding.mielecloud.internal.util.LocaleValidator;
+import org.openhab.binding.mielecloud.internal.webservice.ConnectionError;
+import org.openhab.binding.mielecloud.internal.webservice.ConnectionStatusListener;
+import org.openhab.binding.mielecloud.internal.webservice.DeviceStateListener;
+import org.openhab.binding.mielecloud.internal.webservice.MieleWebservice;
+import org.openhab.binding.mielecloud.internal.webservice.UnavailableMieleWebservice;
+import org.openhab.binding.mielecloud.internal.webservice.api.ActionsState;
+import org.openhab.binding.mielecloud.internal.webservice.api.DeviceState;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceInitializationException;
+import org.openhab.binding.mielecloud.internal.webservice.language.CombiningLanguageProvider;
+import org.openhab.binding.mielecloud.internal.webservice.language.LanguageProvider;
+import org.openhab.core.thing.Bridge;
+import org.openhab.core.thing.ChannelUID;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.thing.ThingStatus;
+import org.openhab.core.thing.ThingStatusDetail;
+import org.openhab.core.thing.binding.BaseBridgeHandler;
+import org.openhab.core.thing.binding.ThingHandlerService;
+import org.openhab.core.types.Command;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * BridgeHandler implementation for the Miele cloud account.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Introduced CombiningLanguageProvider field and interactions, added LanguageProvider super
+ * interface, switched from polling to SSE, added support for multiple bridges
+ */
+@NonNullByDefault
+public class MieleBridgeHandler extends BaseBridgeHandler
+ implements OAuthTokenRefreshListener, LanguageProvider, ConnectionStatusListener, DeviceStateListener {
+ private static final int NUMBER_OF_SSE_RECONNECTION_ATTEMPTS_BEFORE_STATUS_IS_UPDATED = 6;
+
+ private final Supplier webserviceFactory;
+
+ private final OAuthTokenRefresher tokenRefresher;
+ private final CombiningLanguageProvider languageProvider;
+ private final Logger logger = LoggerFactory.getLogger(this.getClass());
+
+ private @Nullable CompletableFuture<@Nullable Void> logoutFuture;
+ private @Nullable MieleWebservice webService;
+ private @Nullable ThingDiscoveryService discoveryService;
+
+ /**
+ * Creates a new {@link MieleBridgeHandler}.
+ *
+ * @param bridge The bridge to handle.
+ * @param webserviceFactory Factory for creating {@link MieleWebservice} instances.
+ * @param tokenRefresher Token refresher.
+ * @param languageProvider Language provider.
+ */
+ public MieleBridgeHandler(Bridge bridge, Function webserviceFactory,
+ OAuthTokenRefresher tokenRefresher, CombiningLanguageProvider languageProvider) {
+ super(bridge);
+ this.webserviceFactory = () -> webserviceFactory.apply(scheduler);
+ this.tokenRefresher = tokenRefresher;
+ this.languageProvider = languageProvider;
+ }
+
+ public void setDiscoveryService(@Nullable ThingDiscoveryService discoveryService) {
+ this.discoveryService = discoveryService;
+ }
+
+ /**
+ * Gets the current webservice instance for communication with the Miele service.
+ *
+ * This function may return an {@link UnavailableMieleWebservice} in case no webservice is available at the moment.
+ */
+ public MieleWebservice getWebservice() {
+ MieleWebservice webservice = webService;
+ if (webservice != null) {
+ return webservice;
+ } else {
+ return UnavailableMieleWebservice.INSTANCE;
+ }
+ }
+
+ private String getOAuthServiceHandle() {
+ return getConfig().get(MieleCloudBindingConstants.CONFIG_PARAM_EMAIL).toString();
+ }
+
+ @Override
+ public void initialize() {
+ // It is required to set a status in this method as stated in the Javadoc of ThingHandler.initialize
+ updateStatus(ThingStatus.UNKNOWN);
+
+ initializeWebservice();
+ }
+
+ public void initializeWebservice() {
+ if (!EmailValidator.isValid(getConfig().get(MieleCloudBindingConstants.CONFIG_PARAM_EMAIL).toString())) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
+ I18NKeys.BRIDGE_STATUS_DESCRIPTION_INVALID_EMAIL);
+ // When the e-mail configuration is changed a new initialization will be triggered. Therefore, we can leave
+ // the bridge in this state.
+ return;
+ }
+
+ try {
+ webService = webserviceFactory.get();
+ } catch (MieleWebserviceInitializationException e) {
+ logger.warn("Failed to initialize webservice.", e);
+ updateStatus(ThingStatus.OFFLINE);
+ return;
+ }
+
+ try {
+ tokenRefresher.setRefreshListener(this, getOAuthServiceHandle());
+ } catch (OAuthException e) {
+ logger.debug("Could not initialize Miele Cloud bridge.", e);
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
+ I18NKeys.BRIDGE_STATUS_DESCRIPTION_ACCOUNT_NOT_AUTHORIZED);
+ // When the authorization takes place a new initialization will be triggered. Therefore, we can leave the
+ // bridge in this state.
+ return;
+ }
+ languageProvider.setPrioritizedLanguageProvider(this);
+ tryInitializeWebservice();
+
+ MieleWebservice webservice = getWebservice();
+ webservice.addConnectionStatusListener(this);
+ webservice.addDeviceStateListener(this);
+ if (webservice.hasAccessToken()) {
+ webservice.connectSse();
+ }
+ }
+
+ @Override
+ public void handleRemoval() {
+ performLogout();
+ tokenRefresher.removeTokensFromStorage(getOAuthServiceHandle());
+ super.handleRemoval();
+ }
+
+ @Override
+ public void dispose() {
+ logger.debug("Disposing {}", this.getClass().getName());
+ disposeWebservice();
+ }
+
+ public void disposeWebservice() {
+ getWebservice().removeConnectionStatusListener(this);
+ getWebservice().removeDeviceStateListener(this);
+ getWebservice().disconnectSse();
+ languageProvider.unsetPrioritizedLanguageProvider();
+ tokenRefresher.unsetRefreshListener(getOAuthServiceHandle());
+
+ stopWebservice();
+ }
+
+ private void stopWebservice() {
+ final MieleWebservice webService = this.webService;
+ this.webService = null;
+ if (webService == null) {
+ return;
+ }
+
+ scheduler.submit(() -> {
+ CompletableFuture<@Nullable Void> logoutFuture = this.logoutFuture;
+ if (logoutFuture != null) {
+ try {
+ logoutFuture.get();
+ } catch (InterruptedException e) {
+ logger.warn("Interrupted while waiting for logout!");
+ } catch (ExecutionException e) {
+ logger.warn("Failed to wait for logout.", e);
+ }
+ }
+
+ try {
+ webService.close();
+ } catch (Exception e) {
+ logger.warn("Failed to close webservice.", e);
+ }
+ });
+ }
+
+ @Override
+ public void onNewAccessToken(String accessToken) {
+ logger.debug("Setting new access token for webservice access.");
+ updateProperty(MieleCloudBindingConstants.PROPERTY_ACCESS_TOKEN, accessToken);
+
+ // Without this the retry would fail causing the thing to go OFFLINE
+ getWebservice().setAccessToken(accessToken);
+
+ // If there was no access token during initialization then the SSE connection was not established.
+ getWebservice().connectSse();
+ }
+
+ @Override
+ public void handleCommand(ChannelUID channelUID, Command command) {
+ }
+
+ private void performLogout() {
+ logoutFuture = new CompletableFuture<>();
+ scheduler.execute(() -> {
+ try {
+ getWebservice().logout();
+ } catch (Exception e) {
+ logger.warn("Failed to logout from Miele cloud.", e);
+ }
+ Optional.ofNullable(logoutFuture).map(future -> future.complete(null));
+ });
+ }
+
+ private void tryInitializeWebservice() {
+ Optional accessToken = tokenRefresher.getAccessTokenFromStorage(getOAuthServiceHandle());
+ if (!accessToken.isPresent()) {
+ logger.debug("No OAuth2 access token available. Retrying later.");
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_PENDING,
+ I18NKeys.BRIDGE_STATUS_DESCRIPTION_ACCESS_TOKEN_NOT_CONFIGURED);
+ return;
+ }
+ getWebservice().setAccessToken(accessToken.get());
+ updateProperty(MieleCloudBindingConstants.PROPERTY_ACCESS_TOKEN, accessToken.get());
+ }
+
+ @Override
+ public void onConnectionAlive() {
+ updateStatus(ThingStatus.ONLINE);
+ }
+
+ @Override
+ public void onConnectionError(ConnectionError connectionError, int failedReconnectionAttempts) {
+ if (connectionError == ConnectionError.AUTHORIZATION_FAILED) {
+ tryToRefreshAccessToken();
+ return;
+ }
+
+ if (failedReconnectionAttempts <= NUMBER_OF_SSE_RECONNECTION_ATTEMPTS_BEFORE_STATUS_IS_UPDATED
+ && getThing().getStatus() != ThingStatus.UNKNOWN) {
+ return;
+ }
+
+ if (getThing().getStatus() == ThingStatus.UNKNOWN && connectionError == ConnectionError.REQUEST_INTERRUPTED
+ && failedReconnectionAttempts <= NUMBER_OF_SSE_RECONNECTION_ATTEMPTS_BEFORE_STATUS_IS_UPDATED) {
+ return;
+ }
+
+ switch (connectionError) {
+ case AUTHORIZATION_FAILED:
+ // Handled above.
+ break;
+
+ case REQUEST_EXECUTION_FAILED:
+ case SERVICE_UNAVAILABLE:
+ case RESPONSE_MALFORMED:
+ case TIMEOUT:
+ case TOO_MANY_RERQUESTS:
+ case SSE_STREAM_ENDED:
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR);
+ break;
+
+ case SERVER_ERROR:
+ case REQUEST_INTERRUPTED:
+ case OTHER_HTTP_ERROR:
+ default:
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.NONE,
+ I18NKeys.BRIDGE_STATUS_DESCRIPTION_TRANSIENT_HTTP_ERROR);
+ break;
+ }
+ }
+
+ private void tryToRefreshAccessToken() {
+ try {
+ tokenRefresher.refreshToken(getOAuthServiceHandle());
+ getWebservice().connectSse();
+ } catch (OAuthException e) {
+ logger.debug("Failed to refresh OAuth token!", e);
+ getWebservice().disconnectSse();
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
+ I18NKeys.BRIDGE_STATUS_DESCRIPTION_ACCESS_TOKEN_REFRESH_FAILED);
+ }
+ }
+
+ @Override
+ public Optional getLanguage() {
+ Object languageObject = thing.getConfiguration().get(MieleCloudBindingConstants.CONFIG_PARAM_LOCALE);
+ if (languageObject instanceof String) {
+ String language = (String) languageObject;
+ if (language.isEmpty() || !LocaleValidator.isValidLanguage(language)) {
+ return Optional.empty();
+ } else {
+ return Optional.of(language);
+ }
+ } else {
+ return Optional.empty();
+ }
+ }
+
+ @Override
+ public void onDeviceStateUpdated(DeviceState deviceState) {
+ ThingDiscoveryService discoveryService = this.discoveryService;
+ if (discoveryService != null) {
+ discoveryService.onDeviceStateUpdated(deviceState);
+ }
+
+ invokeOnThingHandlers(deviceState.getDeviceIdentifier(), handler -> handler.onDeviceStateUpdated(deviceState));
+ }
+
+ @Override
+ public void onProcessActionUpdated(ActionsState actionState) {
+ invokeOnThingHandlers(actionState.getDeviceIdentifier(),
+ handler -> handler.onProcessActionUpdated(actionState));
+ }
+
+ @Override
+ public void onDeviceRemoved(String deviceIdentifier) {
+ ThingDiscoveryService discoveryService = this.discoveryService;
+ if (discoveryService != null) {
+ discoveryService.onDeviceRemoved(deviceIdentifier);
+ }
+
+ invokeOnThingHandlers(deviceIdentifier, handler -> handler.onDeviceRemoved());
+ }
+
+ private void invokeOnThingHandlers(String deviceIdentifier, Consumer action) {
+ getThing().getThings().stream().map(Thing::getHandler)
+ .filter(handler -> handler instanceof AbstractMieleThingHandler)
+ .map(handler -> (AbstractMieleThingHandler) handler)
+ .filter(handler -> deviceIdentifier.equals(handler.getDeviceId())).forEach(action);
+ }
+
+ @Override
+ public Collection> getServices() {
+ return Collections.singleton(ThingDiscoveryService.class);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/MieleHandlerFactory.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/MieleHandlerFactory.java
new file mode 100644
index 00000000000..e9f64385ebf
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/MieleHandlerFactory.java
@@ -0,0 +1,123 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler;
+
+import static org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.*;
+
+import java.util.Set;
+import java.util.concurrent.ScheduledExecutorService;
+import java.util.function.Function;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.mielecloud.internal.auth.OAuthTokenRefresher;
+import org.openhab.binding.mielecloud.internal.webservice.DefaultMieleWebserviceFactory;
+import org.openhab.binding.mielecloud.internal.webservice.MieleWebservice;
+import org.openhab.binding.mielecloud.internal.webservice.MieleWebserviceConfiguration;
+import org.openhab.binding.mielecloud.internal.webservice.MieleWebserviceFactory;
+import org.openhab.binding.mielecloud.internal.webservice.language.CombiningLanguageProvider;
+import org.openhab.binding.mielecloud.internal.webservice.language.OpenHabLanguageProvider;
+import org.openhab.core.i18n.LocaleProvider;
+import org.openhab.core.io.net.http.HttpClientFactory;
+import org.openhab.core.thing.Bridge;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.thing.ThingTypeUID;
+import org.openhab.core.thing.binding.BaseThingHandlerFactory;
+import org.openhab.core.thing.binding.ThingHandler;
+import org.openhab.core.thing.binding.ThingHandlerFactory;
+import org.osgi.service.component.annotations.Activate;
+import org.osgi.service.component.annotations.Component;
+import org.osgi.service.component.annotations.Reference;
+
+/**
+ * Factory producing the {@link ThingHandler}s for all things supported by this binding.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Added language provider, added support for multiple bridges
+ */
+@NonNullByDefault
+@Component(service = ThingHandlerFactory.class, configurationPid = "binding.mielecloud")
+public class MieleHandlerFactory extends BaseThingHandlerFactory {
+ public static final Set SUPPORTED_THING_TYPES = Set.of(THING_TYPE_BRIDGE, THING_TYPE_WASHING_MACHINE,
+ THING_TYPE_WASHER_DRYER, THING_TYPE_COFFEE_SYSTEM, THING_TYPE_FRIDGE_FREEZER, THING_TYPE_FRIDGE,
+ THING_TYPE_FREEZER, THING_TYPE_OVEN, THING_TYPE_WINE_STORAGE, THING_TYPE_HOB, THING_TYPE_DRYER,
+ THING_TYPE_DISHWASHER, THING_TYPE_HOOD, THING_TYPE_DISH_WARMER, THING_TYPE_ROBOTIC_VACUUM_CLEANER);
+
+ private final HttpClientFactory httpClientFactory;
+ private final OAuthTokenRefresher tokenRefresher;
+ private final LocaleProvider localeProvider;
+
+ private final MieleWebserviceFactory webserviceFactory = new DefaultMieleWebserviceFactory();
+
+ @Activate
+ public MieleHandlerFactory(@Reference HttpClientFactory httpClientFactory,
+ @Reference OAuthTokenRefresher tokenRefresher, @Reference LocaleProvider localeProvider) {
+ this.httpClientFactory = httpClientFactory;
+ this.tokenRefresher = tokenRefresher;
+ this.localeProvider = localeProvider;
+ }
+
+ @Override
+ public boolean supportsThingType(ThingTypeUID thingTypeUID) {
+ return SUPPORTED_THING_TYPES.contains(thingTypeUID);
+ }
+
+ @Override
+ @Nullable
+ protected ThingHandler createHandler(Thing thing) {
+ ThingTypeUID thingTypeUID = thing.getThingTypeUID();
+
+ if (thingTypeUID.equals(THING_TYPE_BRIDGE)) {
+ return createBridgeHandler(thing);
+ } else if (thingTypeUID.equals(THING_TYPE_WASHING_MACHINE) || thingTypeUID.equals(THING_TYPE_WASHER_DRYER)) {
+ return new WashingDeviceThingHandler(thing);
+ } else if (thingTypeUID.equals(THING_TYPE_COFFEE_SYSTEM)) {
+ return new CoffeeSystemThingHandler(thing);
+ } else if (thingTypeUID.equals(THING_TYPE_FRIDGE_FREEZER) || thingTypeUID.equals(THING_TYPE_FRIDGE)
+ || thingTypeUID.equals(THING_TYPE_FREEZER)) {
+ return new CoolingDeviceThingHandler(thing);
+ } else if (thingTypeUID.equals(THING_TYPE_WINE_STORAGE)) {
+ return new WineStorageDeviceThingHandler(thing);
+ } else if (thingTypeUID.equals(THING_TYPE_OVEN)) {
+ return new OvenDeviceThingHandler(thing);
+ } else if (thingTypeUID.equals(THING_TYPE_HOB)) {
+ return new HobDeviceThingHandler(thing);
+ } else if (thingTypeUID.equals(THING_TYPE_DISHWASHER)) {
+ return new DishwasherDeviceThingHandler(thing);
+ } else if (thingTypeUID.equals(THING_TYPE_DRYER)) {
+ return new DryerDeviceThingHandler(thing);
+ } else if (thingTypeUID.equals(THING_TYPE_HOOD)) {
+ return new HoodDeviceThingHandler(thing);
+ } else if (thingTypeUID.equals(THING_TYPE_DISH_WARMER)) {
+ return new DishWarmerDeviceThingHandler(thing);
+ } else if (thingTypeUID.equals(THING_TYPE_ROBOTIC_VACUUM_CLEANER)) {
+ return new RoboticVacuumCleanerDeviceThingHandler(thing);
+ }
+
+ return null;
+ }
+
+ private ThingHandler createBridgeHandler(Thing thing) {
+ CombiningLanguageProvider languageProvider = getLanguageProvider();
+ Function webserviceFactoryFunction = scheduler -> webserviceFactory
+ .create(MieleWebserviceConfiguration.builder().withHttpClientFactory(httpClientFactory)
+ .withLanguageProvider(languageProvider).withTokenRefresher(tokenRefresher)
+ .withServiceHandle(thing.getUID().getAsString()).withScheduler(scheduler).build());
+
+ return new MieleBridgeHandler((Bridge) thing, webserviceFactoryFunction, tokenRefresher, languageProvider);
+ }
+
+ private CombiningLanguageProvider getLanguageProvider() {
+ return new CombiningLanguageProvider(null, new OpenHabLanguageProvider(localeProvider));
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/OvenDeviceThingHandler.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/OvenDeviceThingHandler.java
new file mode 100644
index 00000000000..6e6f7656d06
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/OvenDeviceThingHandler.java
@@ -0,0 +1,80 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler;
+
+import static org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.Channels.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.handler.channel.ActionsChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.DeviceChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.TransitionChannelState;
+import org.openhab.core.thing.Thing;
+
+/**
+ * ThingHandler implementation for the Miele oven devices.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Add channel state wrappers
+ * @author Benjamin Bolte - Add pre-heat finished channel, add info state channel and map signal flags from API
+ * @author Björn Lange - Add elapsed time channel
+ */
+@NonNullByDefault
+public class OvenDeviceThingHandler extends AbstractMieleThingHandler {
+ /**
+ * Creates a new {@link OvenDeviceThingHandler}.
+ *
+ * @param thing The thing to handle.
+ */
+ public OvenDeviceThingHandler(Thing thing) {
+ super(thing);
+ }
+
+ @Override
+ protected void updateDeviceState(DeviceChannelState device) {
+ updateState(channel(PROGRAM_ACTIVE), device.getProgramActive());
+ updateState(channel(PROGRAM_ACTIVE_RAW), device.getProgramActiveRaw());
+ updateState(channel(PROGRAM_PHASE), device.getProgramPhase());
+ updateState(channel(PROGRAM_PHASE_RAW), device.getProgramPhaseRaw());
+ updateState(channel(OPERATION_STATE), device.getOperationState());
+ updateState(channel(OPERATION_STATE_RAW), device.getOperationStateRaw());
+ updateState(channel(PROGRAM_START_STOP), device.getProgramStartStop());
+ updateState(channel(DELAYED_START_TIME), device.getDelayedStartTime());
+ updateState(channel(PROGRAM_ELAPSED_TIME), device.getProgramElapsedTime());
+ updateState(channel(PRE_HEAT_FINISHED), device.hasPreHeatFinished());
+ updateState(channel(TEMPERATURE_TARGET), device.getTemperatureTarget());
+ updateState(channel(TEMPERATURE_CURRENT), device.getTemperatureCurrent());
+ updateState(channel(POWER_ON_OFF), device.getPowerOnOff());
+ updateState(channel(ERROR_STATE), device.getErrorState());
+ updateState(channel(INFO_STATE), device.getInfoState());
+ updateState(channel(LIGHT_SWITCH), device.getLightSwitch());
+ updateState(channel(DOOR_STATE), device.getDoorState());
+ }
+
+ @Override
+ protected void updateTransitionState(TransitionChannelState transition) {
+ updateState(channel(PROGRAM_REMAINING_TIME), transition.getProgramRemainingTime());
+ updateState(channel(PROGRAM_PROGRESS), transition.getProgramProgress());
+ if (transition.hasFinishedChanged()) {
+ updateState(channel(FINISH_STATE), transition.getFinishState());
+ }
+ }
+
+ @Override
+ protected void updateActionState(ActionsChannelState actions) {
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STARTED), actions.getRemoteControlCanBeStarted());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STOPPED), actions.getRemoteControlCanBeStopped());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_ON), actions.getRemoteControlCanBeSwitchedOn());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_OFF), actions.getRemoteControlCanBeSwitchedOff());
+ updateState(channel(LIGHT_CAN_BE_CONTROLLED), actions.getLightCanBeControlled());
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/RoboticVacuumCleanerDeviceThingHandler.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/RoboticVacuumCleanerDeviceThingHandler.java
new file mode 100644
index 00000000000..3b36a0f10cc
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/RoboticVacuumCleanerDeviceThingHandler.java
@@ -0,0 +1,86 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler;
+
+import static org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.Channels.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.handler.channel.ActionsChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.DeviceChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.TransitionChannelState;
+import org.openhab.core.library.types.StringType;
+import org.openhab.core.thing.ChannelUID;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.types.Command;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * ThingHandler implementation for Miele robotic vacuum cleaners.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class RoboticVacuumCleanerDeviceThingHandler extends AbstractMieleThingHandler {
+ private final Logger logger = LoggerFactory.getLogger(this.getClass());
+
+ /**
+ * Creates a new {@link RoboticVacuumCleanerDeviceThingHandler}.
+ *
+ * @param thing The thing to handle.
+ */
+ public RoboticVacuumCleanerDeviceThingHandler(Thing thing) {
+ super(thing);
+ }
+
+ @Override
+ public void handleCommand(ChannelUID channelUID, Command command) {
+ super.handleCommand(channelUID, command);
+
+ if (VACUUM_CLEANER_PROGRAM_ACTIVE.equals(channelUID.getId()) && command instanceof StringType) {
+ try {
+ triggerProgram(Long.parseLong(command.toString()));
+ } catch (NumberFormatException e) {
+ logger.warn("Failed to activate program: '{}' is not a valid program ID", command.toString());
+ }
+ }
+ }
+
+ @Override
+ protected void updateDeviceState(DeviceChannelState device) {
+ updateState(channel(VACUUM_CLEANER_PROGRAM_ACTIVE), device.getProgramActiveId());
+ updateState(channel(PROGRAM_ACTIVE_RAW), device.getProgramActiveRaw());
+ updateState(channel(OPERATION_STATE), device.getOperationState());
+ updateState(channel(OPERATION_STATE_RAW), device.getOperationStateRaw());
+ updateState(channel(PROGRAM_START_STOP_PAUSE), device.getProgramStartStopPause());
+ updateState(channel(POWER_ON_OFF), device.getPowerOnOff());
+ updateState(channel(ERROR_STATE), device.getErrorState());
+ updateState(channel(INFO_STATE), device.getInfoState());
+ updateState(channel(BATTERY_LEVEL), device.getBatteryLevel());
+ }
+
+ @Override
+ protected void updateTransitionState(TransitionChannelState transition) {
+ if (transition.hasFinishedChanged()) {
+ updateState(channel(FINISH_STATE), transition.getFinishState());
+ }
+ }
+
+ @Override
+ protected void updateActionState(ActionsChannelState actions) {
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STARTED), actions.getRemoteControlCanBeStarted());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STOPPED), actions.getRemoteControlCanBeStopped());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_PAUSED), actions.getRemoteControlCanBePaused());
+ updateState(channel(REMOTE_CONTROL_CAN_SET_PROGRAM_ACTIVE), actions.getRemoteControlCanSetProgramActive());
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/WashingDeviceThingHandler.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/WashingDeviceThingHandler.java
new file mode 100644
index 00000000000..dff24484918
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/WashingDeviceThingHandler.java
@@ -0,0 +1,80 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler;
+
+import static org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.Channels.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.handler.channel.ActionsChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.DeviceChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.TransitionChannelState;
+import org.openhab.core.thing.Thing;
+
+/**
+ * ThingHandler implementation for the Miele washing devices.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Add channel state wrappers
+ * @author Benjamin Bolte - Add info state channel and map signal flags from API
+ * @author Björn Lange - Add elapsed time channel
+ */
+@NonNullByDefault
+public class WashingDeviceThingHandler extends AbstractMieleThingHandler {
+ /**
+ * Creates a new {@link WashingDeviceThingHandler}.
+ *
+ * @param thing The thing to handle.
+ */
+ public WashingDeviceThingHandler(Thing thing) {
+ super(thing);
+ }
+
+ @Override
+ protected void updateDeviceState(DeviceChannelState device) {
+ updateState(channel(SPINNING_SPEED), device.getSpinningSpeed());
+ updateState(channel(SPINNING_SPEED_RAW), device.getSpinningSpeedRaw());
+ updateState(channel(PROGRAM_ACTIVE), device.getProgramActive());
+ updateState(channel(PROGRAM_ACTIVE_RAW), device.getProgramActiveRaw());
+ updateState(channel(PROGRAM_PHASE), device.getProgramPhase());
+ updateState(channel(PROGRAM_PHASE_RAW), device.getProgramPhaseRaw());
+ updateState(channel(OPERATION_STATE), device.getOperationState());
+ updateState(channel(OPERATION_STATE_RAW), device.getOperationStateRaw());
+ updateState(channel(PROGRAM_START_STOP), device.getProgramStartStop());
+ updateState(channel(DELAYED_START_TIME), device.getDelayedStartTime());
+ updateState(channel(PROGRAM_ELAPSED_TIME), device.getProgramElapsedTime());
+ updateState(channel(TEMPERATURE_TARGET), device.getTemperatureTarget());
+ updateState(channel(POWER_ON_OFF), device.getPowerOnOff());
+ updateState(channel(ERROR_STATE), device.getErrorState());
+ updateState(channel(INFO_STATE), device.getInfoState());
+ updateState(channel(LIGHT_SWITCH), device.getLightSwitch());
+ updateState(channel(DOOR_STATE), device.getDoorState());
+ }
+
+ @Override
+ protected void updateTransitionState(TransitionChannelState transition) {
+ updateState(channel(PROGRAM_REMAINING_TIME), transition.getProgramRemainingTime());
+ updateState(channel(PROGRAM_PROGRESS), transition.getProgramProgress());
+ if (transition.hasFinishedChanged()) {
+ updateState(channel(FINISH_STATE), transition.getFinishState());
+ }
+ }
+
+ @Override
+ protected void updateActionState(ActionsChannelState actions) {
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STARTED), actions.getRemoteControlCanBeStarted());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STOPPED), actions.getRemoteControlCanBeStopped());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_ON), actions.getRemoteControlCanBeSwitchedOn());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_OFF), actions.getRemoteControlCanBeSwitchedOff());
+ updateState(channel(LIGHT_CAN_BE_CONTROLLED), actions.getLightCanBeControlled());
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/WineStorageDeviceThingHandler.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/WineStorageDeviceThingHandler.java
new file mode 100644
index 00000000000..e883e45cf8a
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/WineStorageDeviceThingHandler.java
@@ -0,0 +1,75 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler;
+
+import static org.openhab.binding.mielecloud.internal.MieleCloudBindingConstants.Channels.*;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.handler.channel.ActionsChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.DeviceChannelState;
+import org.openhab.binding.mielecloud.internal.handler.channel.TransitionChannelState;
+import org.openhab.core.library.types.OnOffType;
+import org.openhab.core.thing.Thing;
+
+/**
+ * ThingHandler implementation for the Miele wine storage devices.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Add channel state wrappers
+ * @author Benjamin Bolte - Add info state channel and map signal flags from API
+ */
+@NonNullByDefault
+public class WineStorageDeviceThingHandler extends AbstractMieleThingHandler {
+ /**
+ * Creates a new {@link WineStorageDeviceThingHandler}.
+ *
+ * @param thing The thing to handle.
+ */
+ public WineStorageDeviceThingHandler(Thing thing) {
+ super(thing);
+ }
+
+ @Override
+ public void initialize() {
+ super.initialize();
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STARTED), OnOffType.OFF);
+ updateState(channel(REMOTE_CONTROL_CAN_BE_STOPPED), OnOffType.OFF);
+ }
+
+ @Override
+ protected void updateDeviceState(DeviceChannelState device) {
+ updateState(channel(OPERATION_STATE), device.getOperationState());
+ updateState(channel(OPERATION_STATE_RAW), device.getOperationStateRaw());
+ updateState(channel(TEMPERATURE_TARGET), device.getWineTemperatureTarget());
+ updateState(channel(TEMPERATURE_CURRENT), device.getWineTemperatureCurrent());
+ updateState(channel(TOP_TEMPERATURE_TARGET), device.getWineTopTemperatureTarget());
+ updateState(channel(TOP_TEMPERATURE_CURRENT), device.getWineTopTemperatureCurrent());
+ updateState(channel(MIDDLE_TEMPERATURE_TARGET), device.getWineMiddleTemperatureTarget());
+ updateState(channel(MIDDLE_TEMPERATURE_CURRENT), device.getWineMiddleTemperatureCurrent());
+ updateState(channel(BOTTOM_TEMPERATURE_TARGET), device.getWineBottomTemperatureTarget());
+ updateState(channel(BOTTOM_TEMPERATURE_CURRENT), device.getWineBottomTemperatureCurrent());
+ updateState(channel(POWER_ON_OFF), device.getPowerOnOff());
+ updateState(channel(ERROR_STATE), device.getErrorState());
+ updateState(channel(INFO_STATE), device.getInfoState());
+ }
+
+ @Override
+ protected void updateTransitionState(TransitionChannelState transition) {
+ }
+
+ @Override
+ protected void updateActionState(ActionsChannelState actions) {
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_ON), actions.getRemoteControlCanBeSwitchedOn());
+ updateState(channel(REMOTE_CONTROL_CAN_BE_SWITCHED_OFF), actions.getRemoteControlCanBeSwitchedOff());
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/channel/ActionsChannelState.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/channel/ActionsChannelState.java
new file mode 100644
index 00000000000..9e4c108d3b4
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/channel/ActionsChannelState.java
@@ -0,0 +1,68 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler.channel;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.webservice.api.ActionsState;
+import org.openhab.core.library.types.OnOffType;
+import org.openhab.core.types.State;
+
+/**
+ * Wrapper for {@link ActionsState} handling the type conversion to {@link State} for directly filling channels.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class ActionsChannelState {
+ private final ActionsState actions;
+
+ public ActionsChannelState(ActionsState actions) {
+ this.actions = actions;
+ }
+
+ public State getRemoteControlCanBeSwitchedOn() {
+ return OnOffType.from(actions.canBeSwitchedOn());
+ }
+
+ public State getRemoteControlCanBeSwitchedOff() {
+ return OnOffType.from(actions.canBeSwitchedOff());
+ }
+
+ public State getLightCanBeControlled() {
+ return OnOffType.from(actions.canControlLight());
+ }
+
+ public State getSuperCoolCanBeControlled() {
+ return OnOffType.from(actions.canContolSupercooling());
+ }
+
+ public State getSuperFreezeCanBeControlled() {
+ return OnOffType.from(actions.canControlSuperfreezing());
+ }
+
+ public State getRemoteControlCanBeStarted() {
+ return OnOffType.from(actions.canBeStarted());
+ }
+
+ public State getRemoteControlCanBeStopped() {
+ return OnOffType.from(actions.canBeStopped());
+ }
+
+ public State getRemoteControlCanBePaused() {
+ return OnOffType.from(actions.canBePaused());
+ }
+
+ public State getRemoteControlCanSetProgramActive() {
+ return OnOffType.from(actions.canSetActiveProgramId());
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/channel/ChannelTypeUtil.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/channel/ChannelTypeUtil.java
new file mode 100644
index 00000000000..6a9b47f0a18
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/channel/ChannelTypeUtil.java
@@ -0,0 +1,73 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler.channel;
+
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.core.library.types.DecimalType;
+import org.openhab.core.library.types.OnOffType;
+import org.openhab.core.library.types.QuantityType;
+import org.openhab.core.library.types.StringType;
+import org.openhab.core.library.unit.SIUnits;
+import org.openhab.core.types.State;
+import org.openhab.core.types.UnDefType;
+
+/**
+ * Utility class handling type conversions from Java types to channel types.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class ChannelTypeUtil {
+ private ChannelTypeUtil() {
+ throw new IllegalStateException("ChannelTypeUtil cannot be instantiated.");
+ }
+
+ /**
+ * Converts an {@link Optional} of {@link String} to {@link State}.
+ */
+ public static State stringToState(Optional value) {
+ return value.filter(v -> !v.isEmpty()).filter(v -> !v.equals("null")).map(v -> (State) new StringType(v))
+ .orElse(UnDefType.UNDEF);
+ }
+
+ /**
+ * Converts an {@link Optional} of {@link Boolean} to {@link State}.
+ */
+ public static State booleanToState(Optional value) {
+ return value.map(v -> (State) OnOffType.from(v)).orElse(UnDefType.UNDEF);
+ }
+
+ /**
+ * Converts an {@link Optional} of {@link Integer} to {@link State}.
+ */
+ public static State intToState(Optional value) {
+ return value.map(v -> (State) new DecimalType(v)).orElse(UnDefType.UNDEF);
+ }
+
+ /**
+ * Converts an {@link Optional} of {@link Long} to {@link State}.
+ */
+ public static State longToState(Optional value) {
+ return value.map(v -> (State) new DecimalType(v)).orElse(UnDefType.UNDEF);
+ }
+
+ /**
+ * Converts an {@link Optional} of {@link Integer} to {@link State} representing a temperature.
+ */
+ public static State intToTemperatureState(Optional value) {
+ // The Miele 3rd Party API always provides temperatures in °C (even if the device uses another unit).
+ return value.map(v -> (State) new QuantityType<>(v, SIUnits.CELSIUS)).orElse(UnDefType.UNDEF);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/channel/DeviceChannelState.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/channel/DeviceChannelState.java
new file mode 100644
index 00000000000..e075ced52a5
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/channel/DeviceChannelState.java
@@ -0,0 +1,269 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler.channel;
+
+import static org.openhab.binding.mielecloud.internal.webservice.api.PowerStatus.*;
+import static org.openhab.binding.mielecloud.internal.webservice.api.ProgramStatus.*;
+
+import java.util.Arrays;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.webservice.api.CoolingDeviceTemperatureState;
+import org.openhab.binding.mielecloud.internal.webservice.api.DeviceState;
+import org.openhab.binding.mielecloud.internal.webservice.api.PowerStatus;
+import org.openhab.binding.mielecloud.internal.webservice.api.ProgramStatus;
+import org.openhab.binding.mielecloud.internal.webservice.api.WineStorageDeviceTemperatureState;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.StateType;
+import org.openhab.core.library.types.OnOffType;
+import org.openhab.core.library.types.StringType;
+import org.openhab.core.types.State;
+
+/**
+ * Wrapper for {@link DeviceState} handling the type conversion to {@link State} for directly filling channels.
+ *
+ * @author Björn Lange - Initial contribution
+ * @author Benjamin Bolte - Add pre-heat finished, plate step, door state, door alarm and info state channel and map
+ * signal flags from API
+ * @author Björn Lange - Add elapsed time channel, dish warmer and robotic vacuum cleaner thing
+ */
+@NonNullByDefault
+public final class DeviceChannelState {
+ private final DeviceState device;
+ private final CoolingDeviceTemperatureState coolingTemperature;
+ private final WineStorageDeviceTemperatureState wineTemperature;
+
+ public DeviceChannelState(DeviceState device) {
+ this.device = device;
+ this.coolingTemperature = new CoolingDeviceTemperatureState(device);
+ this.wineTemperature = new WineStorageDeviceTemperatureState(device);
+ }
+
+ public State getLightSwitch() {
+ return ChannelTypeUtil.booleanToState(device.getLightState());
+ }
+
+ public State getDoorState() {
+ return ChannelTypeUtil.booleanToState(device.getDoorState());
+ }
+
+ public State getDoorAlarm() {
+ return ChannelTypeUtil.booleanToState(device.getDoorAlarm());
+ }
+
+ public State getErrorState() {
+ return OnOffType.from(device.hasError());
+ }
+
+ public State getInfoState() {
+ return OnOffType.from(device.hasInfo());
+ }
+
+ public State getPowerOnOff() {
+ return new StringType(getPowerStatus().getState());
+ }
+
+ public State getProgramElapsedTime() {
+ return ChannelTypeUtil.intToState(device.getElapsedTime());
+ }
+
+ public State getOperationState() {
+ return ChannelTypeUtil.stringToState(device.getStatus());
+ }
+
+ public State getOperationStateRaw() {
+ return ChannelTypeUtil.intToState(device.getStatusRaw());
+ }
+
+ public State getProgramPhase() {
+ return ChannelTypeUtil.stringToState(device.getProgramPhase());
+ }
+
+ public State getProgramPhaseRaw() {
+ return ChannelTypeUtil.intToState(device.getProgramPhaseRaw());
+ }
+
+ public State getProgramActive() {
+ return ChannelTypeUtil.stringToState(device.getSelectedProgram());
+ }
+
+ public State getProgramActiveRaw() {
+ return ChannelTypeUtil.longToState(device.getSelectedProgramId());
+ }
+
+ public State getProgramActiveId() {
+ return ChannelTypeUtil.stringToState(device.getSelectedProgramId().map(Object::toString));
+ }
+
+ public State getFridgeSuperCool() {
+ return ChannelTypeUtil.booleanToState(isInState(StateType.SUPERCOOLING, StateType.SUPERCOOLING_SUPERFREEZING));
+ }
+
+ public State getFreezerSuperFreeze() {
+ return ChannelTypeUtil.booleanToState(isInState(StateType.SUPERFREEZING, StateType.SUPERCOOLING_SUPERFREEZING));
+ }
+
+ public State getFridgeTemperatureTarget() {
+ return ChannelTypeUtil.intToTemperatureState(coolingTemperature.getFridgeTargetTemperature());
+ }
+
+ public State getFreezerTemperatureTarget() {
+ return ChannelTypeUtil.intToTemperatureState(coolingTemperature.getFreezerTargetTemperature());
+ }
+
+ public State getFridgeTemperatureCurrent() {
+ return ChannelTypeUtil.intToTemperatureState(coolingTemperature.getFridgeTemperature());
+ }
+
+ public State getFreezerTemperatureCurrent() {
+ return ChannelTypeUtil.intToTemperatureState(coolingTemperature.getFreezerTemperature());
+ }
+
+ public State getProgramStartStop() {
+ return new StringType(getProgramStartStopStatus().getState());
+ }
+
+ public State getProgramStartStopPause() {
+ return new StringType(getProgramStartStopPauseStatus().getState());
+ }
+
+ public State getDelayedStartTime() {
+ return ChannelTypeUtil.intToState(device.getStartTime());
+ }
+
+ public State getDryingTarget() {
+ return ChannelTypeUtil.stringToState(device.getDryingTarget());
+ }
+
+ public State getDryingTargetRaw() {
+ return ChannelTypeUtil.intToState(device.getDryingTargetRaw());
+ }
+
+ public State hasPreHeatFinished() {
+ return ChannelTypeUtil.booleanToState(device.hasPreHeatFinished());
+ }
+
+ public State getTemperatureTarget() {
+ return ChannelTypeUtil.intToTemperatureState(device.getTargetTemperature(0));
+ }
+
+ public State getVentilationPower() {
+ return ChannelTypeUtil.stringToState(device.getVentilationStep());
+ }
+
+ public State getVentilationPowerRaw() {
+ return ChannelTypeUtil.intToState(device.getVentilationStepRaw());
+ }
+
+ public State getPlateStep(int index) {
+ return ChannelTypeUtil.stringToState(device.getPlateStep(index));
+ }
+
+ public State getPlateStepRaw(int index) {
+ return ChannelTypeUtil.intToState(device.getPlateStepRaw(index));
+ }
+
+ public State getTemperatureCurrent() {
+ return ChannelTypeUtil.intToTemperatureState(device.getTemperature(0));
+ }
+
+ public State getSpinningSpeed() {
+ return ChannelTypeUtil.stringToState(device.getSpinningSpeed());
+ }
+
+ public State getSpinningSpeedRaw() {
+ return ChannelTypeUtil.intToState(device.getSpinningSpeedRaw());
+ }
+
+ public State getBatteryLevel() {
+ return ChannelTypeUtil.intToState(device.getBatteryLevel());
+ }
+
+ public State getWineTemperatureTarget() {
+ return ChannelTypeUtil.intToState(wineTemperature.getTargetTemperature());
+ }
+
+ public State getWineTemperatureCurrent() {
+ return ChannelTypeUtil.intToTemperatureState(wineTemperature.getTemperature());
+ }
+
+ public State getWineTopTemperatureTarget() {
+ return ChannelTypeUtil.intToTemperatureState(wineTemperature.getTopTargetTemperature());
+ }
+
+ public State getWineTopTemperatureCurrent() {
+ return ChannelTypeUtil.intToTemperatureState(wineTemperature.getTopTemperature());
+ }
+
+ public State getWineMiddleTemperatureTarget() {
+ return ChannelTypeUtil.intToTemperatureState(wineTemperature.getMiddleTargetTemperature());
+ }
+
+ public State getWineMiddleTemperatureCurrent() {
+ return ChannelTypeUtil.intToTemperatureState(wineTemperature.getMiddleTemperature());
+ }
+
+ public State getWineBottomTemperatureTarget() {
+ return ChannelTypeUtil.intToTemperatureState(wineTemperature.getBottomTargetTemperature());
+ }
+
+ public State getWineBottomTemperatureCurrent() {
+ return ChannelTypeUtil.intToTemperatureState(wineTemperature.getBottomTemperature());
+ }
+
+ /**
+ * Determines the status of the currently selected program.
+ */
+ private PowerStatus getPowerStatus() {
+ if (device.isInState(StateType.OFF) || device.isInState(StateType.NOT_CONNECTED)) {
+ return POWER_OFF;
+ } else {
+ return POWER_ON;
+ }
+ }
+
+ /**
+ * Determines the status of the currently selected program respecting the possibilities started and stopped.
+ */
+ protected ProgramStatus getProgramStartStopStatus() {
+ if (device.isInState(StateType.RUNNING)) {
+ return PROGRAM_STARTED;
+ } else {
+ return PROGRAM_STOPPED;
+ }
+ }
+
+ /**
+ * Determines the status of the currently selected program respecting the possibilities started, stopped and paused.
+ */
+ protected ProgramStatus getProgramStartStopPauseStatus() {
+ if (device.isInState(StateType.RUNNING)) {
+ return PROGRAM_STARTED;
+ } else if (device.isInState(StateType.PAUSE)) {
+ return PROGRAM_PAUSED;
+ } else {
+ return PROGRAM_STOPPED;
+ }
+ }
+
+ /**
+ * Gets whether the device is in one of the given states.
+ *
+ * @param stateType The states to check.
+ * @return An empty {@link Optional} if the raw status is unknown, otherwise an {@link Optional} with a value
+ * indicating whether the device is in one of the given states.
+ */
+ private Optional isInState(StateType... stateType) {
+ return device.getStateType().map(it -> Arrays.asList(stateType).contains(it));
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/channel/TransitionChannelState.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/channel/TransitionChannelState.java
new file mode 100644
index 00000000000..ceb430b3070
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/handler/channel/TransitionChannelState.java
@@ -0,0 +1,47 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.handler.channel;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.webservice.api.TransitionState;
+import org.openhab.core.types.State;
+
+/**
+ * Wrapper for {@link TransitionState} handling the type conversion to {@link State} for directly filling channels.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class TransitionChannelState {
+ private final TransitionState transition;
+
+ public TransitionChannelState(TransitionState transition) {
+ this.transition = transition;
+ }
+
+ public boolean hasFinishedChanged() {
+ return transition.hasFinishedChanged();
+ }
+
+ public State getFinishState() {
+ return ChannelTypeUtil.booleanToState(transition.isFinished());
+ }
+
+ public State getProgramRemainingTime() {
+ return ChannelTypeUtil.intToState(transition.getRemainingTime());
+ }
+
+ public State getProgramProgress() {
+ return ChannelTypeUtil.intToState(transition.getProgress());
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/util/EmailValidator.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/util/EmailValidator.java
new file mode 100644
index 00000000000..4e4083f8596
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/util/EmailValidator.java
@@ -0,0 +1,35 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.util;
+
+import java.util.regex.Pattern;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Utility for validating e-mail addresses.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class EmailValidator {
+ private static final Pattern EMAIL_PATTERN = Pattern.compile("^[\\w-_\\.+]*[\\w-_\\.]\\@([\\w]+\\.)+[\\w]+[\\w]$");
+
+ private EmailValidator() {
+ throw new UnsupportedOperationException();
+ }
+
+ public static boolean isValid(String emailAddress) {
+ return EMAIL_PATTERN.matcher(emailAddress).matches();
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/util/LocaleValidator.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/util/LocaleValidator.java
new file mode 100644
index 00000000000..6d213d5de2b
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/util/LocaleValidator.java
@@ -0,0 +1,45 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.util;
+
+import java.util.Locale;
+import java.util.MissingResourceException;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Utility for validating locales.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class LocaleValidator {
+ private LocaleValidator() {
+ throw new UnsupportedOperationException();
+ }
+
+ /**
+ * Checks whether the given string is a valid two letter language code.
+ *
+ * @param language The string to check.
+ * @return Whether it is a valid language.
+ */
+ public static boolean isValidLanguage(String language) {
+ try {
+ String iso3Language = new Locale(language).getISO3Language();
+ return iso3Language != null && !iso3Language.isEmpty();
+ } catch (MissingResourceException e) {
+ return false;
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/ActionStateFetcher.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/ActionStateFetcher.java
new file mode 100644
index 00000000000..47c851a8bc7
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/ActionStateFetcher.java
@@ -0,0 +1,81 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice;
+
+import java.util.Optional;
+import java.util.concurrent.ScheduledExecutorService;
+import java.util.function.Supplier;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.webservice.api.DeviceState;
+import org.openhab.binding.mielecloud.internal.webservice.exception.AuthorizationFailedException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.TooManyRequestsException;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * The {@link ActionStateFetcher} fetches the updated actions state for a device from the {@link MieleWebservice} if
+ * the state of that device changed.
+ *
+ * Note that an instance of this class is required for each device.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Make calls to webservice asynchronous
+ */
+@NonNullByDefault
+public class ActionStateFetcher {
+ private Optional lastDeviceState = Optional.empty();
+ private final Supplier webserviceSupplier;
+ private final ScheduledExecutorService scheduler;
+
+ private final Logger logger = LoggerFactory.getLogger(ActionStateFetcher.class);
+
+ /**
+ * Creates a new {@link ActionStateFetcher}.
+ *
+ * @param webserviceSupplier Getter function for access to the {@link MieleWebservice}.
+ * @param scheduler System-wide scheduler.
+ */
+ public ActionStateFetcher(Supplier webserviceSupplier, ScheduledExecutorService scheduler) {
+ this.webserviceSupplier = webserviceSupplier;
+ this.scheduler = scheduler;
+ }
+
+ /**
+ * Invoked when the state of a device was updated.
+ */
+ public void onDeviceStateUpdated(DeviceState deviceState) {
+ if (hasDeviceStatusChanged(deviceState)) {
+ scheduler.submit(() -> fetchActions(deviceState));
+ }
+ lastDeviceState = Optional.of(deviceState);
+ }
+
+ private boolean hasDeviceStatusChanged(DeviceState newDeviceState) {
+ return lastDeviceState.map(DeviceState::getStateType)
+ .map(rawStatus -> !newDeviceState.getStateType().equals(rawStatus)).orElse(true);
+ }
+
+ private void fetchActions(DeviceState deviceState) {
+ try {
+ webserviceSupplier.get().fetchActions(deviceState.getDeviceIdentifier());
+ } catch (MieleWebserviceException e) {
+ logger.warn("Failed to fetch action state for device {}: {} - {}", deviceState.getDeviceIdentifier(),
+ e.getConnectionError(), e.getMessage());
+ } catch (AuthorizationFailedException | TooManyRequestsException e) {
+ logger.warn("Failed to fetch action state for device {}: {}", deviceState.getDeviceIdentifier(),
+ e.getMessage());
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/ConnectionError.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/ConnectionError.java
new file mode 100644
index 00000000000..471f6cd382e
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/ConnectionError.java
@@ -0,0 +1,35 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * The {@link ConnectionError} enumeration represents the error state of a connection to the Miele cloud.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public enum ConnectionError {
+ SERVER_ERROR,
+ SERVICE_UNAVAILABLE,
+ OTHER_HTTP_ERROR,
+ REQUEST_INTERRUPTED,
+ TIMEOUT,
+ REQUEST_EXECUTION_FAILED,
+ RESPONSE_MALFORMED,
+ AUTHORIZATION_FAILED,
+ TOO_MANY_RERQUESTS,
+ SSE_STREAM_ENDED,
+ UNKNOWN,
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/ConnectionStatusListener.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/ConnectionStatusListener.java
new file mode 100644
index 00000000000..444252dbb2d
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/ConnectionStatusListener.java
@@ -0,0 +1,36 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Listener for the connection status.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public interface ConnectionStatusListener {
+ /**
+ * Called regularly while the connection is up and running.
+ */
+ void onConnectionAlive();
+
+ /**
+ * Called when a connection error is encountered.
+ *
+ * @param connectionError The error.
+ * @param failedReconnectAttempts The number of failed attempts to reconnect.
+ */
+ void onConnectionError(ConnectionError connectionError, int failedReconnectAttempts);
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/DefaultMieleWebservice.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/DefaultMieleWebservice.java
new file mode 100644
index 00000000000..a3ab848e23a
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/DefaultMieleWebservice.java
@@ -0,0 +1,355 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice;
+
+import java.net.URI;
+import java.util.ArrayList;
+import java.util.List;
+import java.util.Optional;
+import java.util.concurrent.ExecutionException;
+import java.util.concurrent.ScheduledExecutorService;
+import java.util.concurrent.TimeoutException;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.eclipse.jetty.client.api.ContentResponse;
+import org.eclipse.jetty.client.api.Request;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.Actions;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.DeviceCollection;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.Light;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.MieleSyntaxException;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.ProcessAction;
+import org.openhab.binding.mielecloud.internal.webservice.exception.AuthorizationFailedException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceInitializationException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceTransientException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.TooManyRequestsException;
+import org.openhab.binding.mielecloud.internal.webservice.request.RequestFactory;
+import org.openhab.binding.mielecloud.internal.webservice.request.RequestFactoryImpl;
+import org.openhab.binding.mielecloud.internal.webservice.retry.AuthorizationFailedRetryStrategy;
+import org.openhab.binding.mielecloud.internal.webservice.retry.NTimesRetryStrategy;
+import org.openhab.binding.mielecloud.internal.webservice.retry.RetryStrategy;
+import org.openhab.binding.mielecloud.internal.webservice.retry.RetryStrategyCombiner;
+import org.openhab.binding.mielecloud.internal.webservice.sse.ServerSentEvent;
+import org.openhab.binding.mielecloud.internal.webservice.sse.SseConnection;
+import org.openhab.binding.mielecloud.internal.webservice.sse.SseListener;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import com.google.gson.Gson;
+import com.google.gson.JsonSyntaxException;
+
+/**
+ * Default implementation of the {@link MieleWebservice}.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public final class DefaultMieleWebservice implements MieleWebservice, SseListener {
+ private static final String SERVER_ADDRESS = "https://api.mcs3.miele.com";
+ public static final String THIRD_PARTY_ENDPOINTS_BASENAME = SERVER_ADDRESS + "/thirdparty";
+ private static final String ENDPOINT_DEVICES = SERVER_ADDRESS + "/v1/devices/";
+ private static final String ENDPOINT_ACTIONS = ENDPOINT_DEVICES + "%s" + "/actions";
+ private static final String ENDPOINT_LOGOUT = THIRD_PARTY_ENDPOINTS_BASENAME + "/logout";
+ private static final String ENDPOINT_ALL_SSE_EVENTS = ENDPOINT_DEVICES + "all/events";
+
+ private static final String SSE_EVENT_TYPE_DEVICES = "devices";
+
+ private static final Gson GSON = new Gson();
+
+ private final Logger logger = LoggerFactory.getLogger(DefaultMieleWebservice.class);
+
+ private Optional accessToken = Optional.empty();
+ private final RequestFactory requestFactory;
+
+ private final DeviceStateDispatcher deviceStateDispatcher;
+ private final List connectionStatusListeners = new ArrayList<>();
+
+ private final RetryStrategy retryStrategy;
+
+ private final SseConnection sseConnection;
+
+ /**
+ * Creates a new {@link DefaultMieleWebservice} with default retry configuration which is to retry failed operations
+ * once on a transient error. In case an authorization error occurs, a new access token is requested and a retry of
+ * the failed request is executed.
+ *
+ * @param configuration The configuration holding all parameters for constructing the instance.
+ * @throws MieleWebserviceInitializationException if initializing the HTTP client fails.
+ */
+ public DefaultMieleWebservice(MieleWebserviceConfiguration configuration) {
+ this(new RequestFactoryImpl(configuration.getHttpClientFactory(), configuration.getLanguageProvider()),
+ new RetryStrategyCombiner(new NTimesRetryStrategy(1),
+ new AuthorizationFailedRetryStrategy(configuration.getTokenRefresher(),
+ configuration.getServiceHandle())),
+ new DeviceStateDispatcher(), configuration.getScheduler());
+ }
+
+ /**
+ * This constructor only exists for testing.
+ */
+ DefaultMieleWebservice(RequestFactory requestFactory, RetryStrategy retryStrategy,
+ DeviceStateDispatcher deviceStateDispatcher, ScheduledExecutorService scheduler) {
+ this.requestFactory = requestFactory;
+ this.retryStrategy = retryStrategy;
+ this.deviceStateDispatcher = deviceStateDispatcher;
+ this.sseConnection = new SseConnection(ENDPOINT_ALL_SSE_EVENTS, this::createSseRequest, scheduler);
+ this.sseConnection.addSseListener(this);
+ }
+
+ @Override
+ public void setAccessToken(String accessToken) {
+ this.accessToken = Optional.of(accessToken);
+ }
+
+ @Override
+ public boolean hasAccessToken() {
+ return accessToken.isPresent();
+ }
+
+ @Override
+ public synchronized void connectSse() {
+ sseConnection.connect();
+ }
+
+ @Override
+ public synchronized void disconnectSse() {
+ sseConnection.disconnect();
+ }
+
+ @Nullable
+ private Request createSseRequest(String endpoint) {
+ Optional accessToken = this.accessToken;
+ if (!accessToken.isPresent()) {
+ logger.warn("No access token present.");
+ return null;
+ }
+
+ return requestFactory.createSseRequest(endpoint, accessToken.get());
+ }
+
+ @Override
+ public void onServerSentEvent(ServerSentEvent event) {
+ fireConnectionAlive();
+
+ if (!SSE_EVENT_TYPE_DEVICES.equals(event.getEvent())) {
+ return;
+ }
+
+ try {
+ deviceStateDispatcher.dispatchDeviceStateUpdates(DeviceCollection.fromJson(event.getData()));
+ } catch (MieleSyntaxException e) {
+ logger.warn("SSE payload is not valid Json: {}", event.getData());
+ }
+ }
+
+ private void fireConnectionAlive() {
+ connectionStatusListeners.forEach(ConnectionStatusListener::onConnectionAlive);
+ }
+
+ @Override
+ public void onConnectionError(ConnectionError connectionError, int failedReconnectAttempts) {
+ connectionStatusListeners.forEach(l -> l.onConnectionError(connectionError, failedReconnectAttempts));
+ }
+
+ @Override
+ public void fetchActions(String deviceId) {
+ Actions actions = retryStrategy.performRetryableOperation(() -> getActions(deviceId),
+ e -> logger.warn("Cannot poll action state: {}. Retrying...", e.getMessage()));
+ if (actions != null) {
+ deviceStateDispatcher.dispatchActionStateUpdates(deviceId, actions);
+ } else {
+ logger.warn("Cannot poll action state. Response is missing actions.");
+ }
+ }
+
+ @Override
+ public void putProcessAction(String deviceId, ProcessAction processAction) {
+ if (processAction.equals(ProcessAction.UNKNOWN)) {
+ throw new IllegalArgumentException("Process action must not be UNKNOWN.");
+ }
+
+ String formattedProcessAction = GSON.toJson(processAction, ProcessAction.class);
+ formattedProcessAction = formattedProcessAction.substring(1, formattedProcessAction.length() - 1);
+ String json = "{\"processAction\":" + formattedProcessAction + "}";
+
+ logger.debug("Activate process action {} of Miele device {}", processAction.toString(), deviceId);
+ putActions(deviceId, json);
+ }
+
+ @Override
+ public void putLight(String deviceId, boolean enabled) {
+ Light light = enabled ? Light.ENABLE : Light.DISABLE;
+ String json = "{\"light\":" + light.format() + "}";
+
+ logger.debug("Set light of Miele device {} to {}", deviceId, enabled);
+ putActions(deviceId, json);
+ }
+
+ @Override
+ public void putPowerState(String deviceId, boolean enabled) {
+ String action = enabled ? "powerOn" : "powerOff";
+ String json = "{\"" + action + "\":true}";
+
+ logger.debug("Set power state of Miele device {} to {}", deviceId, action);
+ putActions(deviceId, json);
+ }
+
+ @Override
+ public void putProgram(String deviceId, long programId) {
+ String json = "{\"programId\":" + programId + "}";
+
+ logger.debug("Activate program with ID {} of Miele device {}", programId, deviceId);
+ putActions(deviceId, json);
+ }
+
+ @Override
+ public void logout() {
+ Optional accessToken = this.accessToken;
+ if (!accessToken.isPresent()) {
+ logger.debug("No access token present.");
+ return;
+ }
+
+ try {
+ logger.debug("Invalidating Miele webservice access token.");
+ Request request = requestFactory.createPostRequest(ENDPOINT_LOGOUT, accessToken.get());
+ this.accessToken = Optional.empty();
+ sendRequest(request);
+ } catch (MieleWebserviceTransientException e) {
+ throw new MieleWebserviceException("Transient error occurred during logout.", e, e.getConnectionError());
+ }
+ }
+
+ /**
+ * Sends the given request and wraps the possible exceptions in Miele exception types.
+ *
+ * @param request The {@link Request} to send.
+ * @return The obtained {@link ContentResponse}.
+ * @throws MieleWebserviceException if an irrecoverable error occurred.
+ * @throws MieleWebserviceTransientException if a recoverable error occurred.
+ */
+ private ContentResponse sendRequest(Request request) {
+ try {
+ if (logger.isDebugEnabled()) {
+ logger.debug("Send {} request to Miele webservice on uri {}",
+ Optional.ofNullable(request).map(Request::getMethod).orElse("null"),
+ Optional.ofNullable(request).map(Request::getURI).map(URI::toString).orElse("null"));
+ }
+
+ ContentResponse response = request.send();
+ logger.debug("Received response with status code {}", response.getStatus());
+ return response;
+ } catch (InterruptedException e) {
+ Thread.currentThread().interrupt();
+ throw new MieleWebserviceException("Interrupted.", e, ConnectionError.REQUEST_INTERRUPTED);
+ } catch (TimeoutException e) {
+ throw new MieleWebserviceTransientException("Request timed out.", e, ConnectionError.TIMEOUT);
+ } catch (ExecutionException e) {
+ throw new MieleWebserviceException("Request execution failed.", e,
+ ConnectionError.REQUEST_EXECUTION_FAILED);
+ }
+ }
+
+ /**
+ * Gets all available device actions.
+ *
+ * @param deviceId The unique device ID.
+ *
+ * @throws MieleWebserviceException if an error occurs during webservice requests or content parsing.
+ * @throws MieleWebserviceTransientException if an error occurs during webservice requests or content parsing that
+ * is recoverable by retrying the operation.
+ * @throws AuthorizationFailedException if the authorization against the webservice failed.
+ * @throws TooManyRequestsException if too many requests have been made against the webservice recently.
+ */
+ private Actions getActions(String deviceId) {
+ Optional accessToken = this.accessToken;
+ if (!accessToken.isPresent()) {
+ throw new MieleWebserviceException("Missing access token.", ConnectionError.AUTHORIZATION_FAILED);
+ }
+
+ try {
+ logger.debug("Fetch action state description for Miele device {}", deviceId);
+ Request request = requestFactory.createGetRequest(String.format(ENDPOINT_ACTIONS, deviceId),
+ accessToken.get());
+ ContentResponse response = sendRequest(request);
+ HttpUtil.checkHttpSuccess(response);
+ Actions actions = GSON.fromJson(response.getContentAsString(), Actions.class);
+ if (actions == null) {
+ throw new MieleWebserviceTransientException("Failed to parse response message.",
+ ConnectionError.RESPONSE_MALFORMED);
+ }
+ return actions;
+ } catch (JsonSyntaxException e) {
+ throw new MieleWebserviceTransientException("Failed to parse response message.", e,
+ ConnectionError.RESPONSE_MALFORMED);
+ }
+ }
+
+ /**
+ * Performs a PUT request to the actions endpoint for the specified device.
+ *
+ * @param deviceId The ID of the device to PUT for.
+ * @param json The Json body to send with the request.
+ * @throws MieleWebserviceException if an error occurs during webservice requests or content parsing.
+ * @throws MieleWebserviceTransientException if an error occurs during webservice requests or content parsing that
+ * is recoverable by retrying the operation.
+ * @throws AuthorizationFailedException if the authorization against the webservice failed.
+ * @throws TooManyRequestsException if too many requests have been made against the webservice recently.
+ */
+ private void putActions(String deviceId, String json) {
+ retryStrategy.performRetryableOperation(() -> {
+ Optional accessToken = this.accessToken;
+ if (!accessToken.isPresent()) {
+ throw new MieleWebserviceException("Missing access token.", ConnectionError.AUTHORIZATION_FAILED);
+ }
+
+ Request request = requestFactory.createPutRequest(String.format(ENDPOINT_ACTIONS, deviceId),
+ accessToken.get(), json);
+ ContentResponse response = sendRequest(request);
+ HttpUtil.checkHttpSuccess(response);
+ }, e -> {
+ logger.warn("Failed to perform PUT request: {}. Retrying...", e.getMessage());
+ });
+ }
+
+ @Override
+ public void dispatchDeviceState(String deviceIdentifier) {
+ deviceStateDispatcher.dispatchDeviceState(deviceIdentifier);
+ }
+
+ @Override
+ public void addDeviceStateListener(DeviceStateListener listener) {
+ deviceStateDispatcher.addListener(listener);
+ }
+
+ @Override
+ public void removeDeviceStateListener(DeviceStateListener listener) {
+ deviceStateDispatcher.removeListener(listener);
+ }
+
+ @Override
+ public void addConnectionStatusListener(ConnectionStatusListener listener) {
+ connectionStatusListeners.add(listener);
+ }
+
+ @Override
+ public void removeConnectionStatusListener(ConnectionStatusListener listener) {
+ connectionStatusListeners.remove(listener);
+ }
+
+ @Override
+ public void close() throws Exception {
+ requestFactory.close();
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/DefaultMieleWebserviceFactory.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/DefaultMieleWebserviceFactory.java
new file mode 100644
index 00000000000..57e3fb135f1
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/DefaultMieleWebserviceFactory.java
@@ -0,0 +1,28 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Factory creating {@link DefaultMieleWebservice} instances.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class DefaultMieleWebserviceFactory implements MieleWebserviceFactory {
+ @Override
+ public MieleWebservice create(MieleWebserviceConfiguration configuration) {
+ return new DefaultMieleWebservice(configuration);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/DeviceCache.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/DeviceCache.java
new file mode 100644
index 00000000000..d0ac8cd9951
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/DeviceCache.java
@@ -0,0 +1,49 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice;
+
+import java.util.HashMap;
+import java.util.Map;
+import java.util.Optional;
+import java.util.Set;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.Device;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.DeviceCollection;
+
+/**
+ * A cache for {@link Device} objects associated with unique identifiers.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+class DeviceCache {
+ private final Map entries = new HashMap<>();
+
+ public void replaceAllDevices(DeviceCollection deviceCollection) {
+ clear();
+ deviceCollection.getDeviceIdentifiers().stream().forEach(i -> entries.put(i, deviceCollection.getDevice(i)));
+ }
+
+ public void clear() {
+ entries.clear();
+ }
+
+ public Set getDeviceIds() {
+ return entries.keySet();
+ }
+
+ public Optional getDevice(String deviceIdentifier) {
+ return Optional.ofNullable(entries.get(deviceIdentifier));
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/DeviceStateDispatcher.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/DeviceStateDispatcher.java
new file mode 100644
index 00000000000..02950bca8ad
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/DeviceStateDispatcher.java
@@ -0,0 +1,109 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice;
+
+import java.util.HashSet;
+import java.util.List;
+import java.util.Set;
+import java.util.concurrent.CopyOnWriteArrayList;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.webservice.api.ActionsState;
+import org.openhab.binding.mielecloud.internal.webservice.api.DeviceState;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.Actions;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.DeviceCollection;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * Handles event dispatching to {@link DeviceStateListener}s.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class DeviceStateDispatcher {
+ private final Logger logger = LoggerFactory.getLogger(this.getClass());
+
+ private final List listeners = new CopyOnWriteArrayList<>();
+ private Set previousDeviceIdentifiers = new HashSet<>();
+ private final DeviceCache cache = new DeviceCache();
+
+ /**
+ * Adds a listener. The listener will be immediately invoked with the current status of all known devices.
+ *
+ * @param listener The listener to add.
+ */
+ public void addListener(DeviceStateListener listener) {
+ if (listeners.contains(listener)) {
+ logger.warn("Listener '{}' was registered multiple times.", listener);
+ }
+ listeners.add(listener);
+
+ cache.getDeviceIds().forEach(deviceIdentifier -> cache.getDevice(deviceIdentifier)
+ .ifPresent(device -> listener.onDeviceStateUpdated(new DeviceState(deviceIdentifier, device))));
+ }
+
+ /**
+ * Removes a listener.
+ */
+ public void removeListener(DeviceStateListener listener) {
+ listeners.remove(listener);
+ }
+
+ /**
+ * Clears the internal device state cache.
+ */
+ public void clearCache() {
+ cache.clear();
+ }
+
+ /**
+ * Dispatches device status updates to all registered {@link DeviceStateListener}. This includes device removal.
+ *
+ * @param devices {@link DeviceCollection} which contains the state information to dispatch.
+ */
+ public void dispatchDeviceStateUpdates(DeviceCollection devices) {
+ cache.replaceAllDevices(devices);
+ dispatchDevicesRemoved(devices);
+ cache.getDeviceIds().forEach(this::dispatchDeviceState);
+ }
+
+ /**
+ * Dispatches the cached state of the device identified by the given device identifier.
+ */
+ public void dispatchDeviceState(String deviceIdentifier) {
+ cache.getDevice(deviceIdentifier).ifPresent(device -> listeners
+ .forEach(listener -> listener.onDeviceStateUpdated(new DeviceState(deviceIdentifier, device))));
+ }
+
+ /**
+ * Dispatches device action updates to all registered {@link DeviceStateListener}.
+ *
+ * @param deviceId ID of the device to dispatch the {@link Actions} for.
+ * @param actions {@link Actions} to dispatch.
+ */
+ public void dispatchActionStateUpdates(String deviceId, Actions actions) {
+ listeners.forEach(listener -> listener.onProcessActionUpdated(new ActionsState(deviceId, actions)));
+ }
+
+ private void dispatchDevicesRemoved(DeviceCollection devices) {
+ Set presentDeviceIdentifiers = devices.getDeviceIdentifiers();
+ Set removedDeviceIdentifiers = previousDeviceIdentifiers;
+ removedDeviceIdentifiers.removeAll(presentDeviceIdentifiers);
+
+ previousDeviceIdentifiers = devices.getDeviceIdentifiers();
+
+ removedDeviceIdentifiers
+ .forEach(deviceIdentifier -> listeners.forEach(listener -> listener.onDeviceRemoved(deviceIdentifier)));
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/DeviceStateListener.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/DeviceStateListener.java
new file mode 100644
index 00000000000..b7400c6314d
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/DeviceStateListener.java
@@ -0,0 +1,46 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.webservice.api.ActionsState;
+import org.openhab.binding.mielecloud.internal.webservice.api.DeviceState;
+
+/**
+ * Listener for the device states.
+ *
+ * @author Björn Lange and Roland Edelhoff - Initial contribution
+ */
+@NonNullByDefault
+public interface DeviceStateListener {
+ /**
+ * Invoked when new status information is available for a device.
+ *
+ * @param deviceState The device state information.
+ */
+ void onDeviceStateUpdated(DeviceState deviceState);
+
+ /**
+ * Invoked when a new process action is available for a device.
+ *
+ * @param ActionsState The action state information.
+ */
+ void onProcessActionUpdated(ActionsState actionState);
+
+ /**
+ * Invoked when a device got removed from the Miele cloud and no information is available about it.
+ *
+ * @param deviceIdentifier The identifier of the removed device.
+ */
+ void onDeviceRemoved(String deviceIdentifier);
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/HttpUtil.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/HttpUtil.java
new file mode 100644
index 00000000000..e8ccae3176e
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/HttpUtil.java
@@ -0,0 +1,94 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.api.ContentResponse;
+import org.eclipse.jetty.client.api.Response;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.ErrorMessage;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.MieleSyntaxException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.AuthorizationFailedException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceTransientException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.TooManyRequestsException;
+
+/**
+ * Holds utility functions for working with HTTP.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class HttpUtil {
+ private static final String RETRY_AFTER_HEADER_FIELD_NAME = "Retry-After";
+
+ private HttpUtil() {
+ throw new IllegalStateException("This class must not be instantiated");
+ }
+
+ /**
+ * Checks whether the HTTP status given in {@code response} is a success state. In case an error state is obtained,
+ * exceptions are thrown.
+ *
+ * @param response The response to check.
+ * @throws MieleWebserviceTransientException if the status indicates a transient HTTP error.
+ * @throws MieleWebserviceException if the status indicates another HTTP error.
+ * @throws AuthorizationFailedException if the status indicates an authorization failure.
+ * @throws TooManyRequestsException if the status indicates that too many requests have been made against the remote
+ * endpoint.
+ */
+ public static void checkHttpSuccess(Response response) {
+ if (isHttpSuccessStatus(response.getStatus())) {
+ return;
+ }
+
+ String exceptionMessage = getHttpErrorMessageFromCloudResponse(response);
+
+ switch (response.getStatus()) {
+ case 401:
+ throw new AuthorizationFailedException(exceptionMessage);
+ case 429:
+ String retryAfter = null;
+ if (response.getHeaders().containsKey(RETRY_AFTER_HEADER_FIELD_NAME)) {
+ retryAfter = response.getHeaders().get(RETRY_AFTER_HEADER_FIELD_NAME);
+ }
+ throw new TooManyRequestsException(exceptionMessage, retryAfter);
+ case 500:
+ throw new MieleWebserviceTransientException(exceptionMessage, ConnectionError.SERVER_ERROR);
+ case 503:
+ throw new MieleWebserviceTransientException(exceptionMessage, ConnectionError.SERVICE_UNAVAILABLE);
+ default:
+ throw new MieleWebserviceException(exceptionMessage, ConnectionError.OTHER_HTTP_ERROR);
+ }
+ }
+
+ /**
+ * Gets whether {@code httpStatus} is a HTTP error code from the 200 range (success).
+ */
+ private static boolean isHttpSuccessStatus(int httpStatus) {
+ return httpStatus / 100 == 2;
+ }
+
+ private static String getHttpErrorMessageFromCloudResponse(Response response) {
+ String exceptionMessage = "HTTP error " + response.getStatus() + ": " + response.getReason();
+
+ if (response instanceof ContentResponse) {
+ try {
+ ErrorMessage errorMessage = ErrorMessage.fromJson(((ContentResponse) response).getContentAsString());
+ exceptionMessage += "\nCloud returned message: " + errorMessage.getMessage();
+ } catch (MieleSyntaxException e) {
+ exceptionMessage += "\nCloud returned invalid message.";
+ }
+ }
+ return exceptionMessage;
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/MieleWebservice.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/MieleWebservice.java
new file mode 100644
index 00000000000..a25027b7f70
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/MieleWebservice.java
@@ -0,0 +1,142 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.ProcessAction;
+import org.openhab.binding.mielecloud.internal.webservice.exception.AuthorizationFailedException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.TooManyRequestsException;
+
+/**
+ * The {@link MieleWebservice} serves as an interface to the Miele REST API and wraps all calls to it.
+ *
+ * @author Björn Lange and Roland Edelhoff - Initial contribution
+ */
+@NonNullByDefault
+public interface MieleWebservice extends AutoCloseable {
+ /**
+ * Sets the OAuth2 access token to use.
+ */
+ void setAccessToken(String accessToken);
+
+ /**
+ * Returns whether an access token is available.
+ */
+ boolean hasAccessToken();
+
+ /**
+ * Connects to the Miele webservice SSE endpoint and starts receiving events.
+ */
+ void connectSse();
+
+ /**
+ * Disconnects a running connection from the Miele SSE endpoint.
+ */
+ void disconnectSse();
+
+ /**
+ * Fetches the available actions for the device with the given {@code deviceId}.
+ *
+ * @param deviceId The unique ID of the device to fetch the available actions for.
+ * @throws MieleWebserviceException if an error occurs during webservice requests or content parsing.
+ * @throws AuthorizationFailedException if the authorization against the webservice failed.
+ * @throws TooManyRequestsException if too many requests have been made against the webservice recently.
+ */
+ void fetchActions(String deviceId);
+
+ /**
+ * Performs a PUT operation with the given {@code processAction}.
+ *
+ * @param deviceId ID of the device to trigger the action for.
+ * @param processAction The action to perform.
+ * @throws MieleWebserviceException if an error occurs during webservice requests or content parsing.
+ * @throws AuthorizationFailedException if the authorization against the webservice failed.
+ * @throws TooManyRequestsException if too many requests have been made against the webservice recently.
+ */
+ void putProcessAction(String deviceId, ProcessAction processAction);
+
+ /**
+ * Performs a PUT operation enabling or disabling the device's light.
+ *
+ * @param deviceId ID of the device to trigger the action for.
+ * @param enabled {@code true} to enable or {@code false} to disable the light.
+ * @throws MieleWebserviceException if an error occurs during webservice requests or content parsing.
+ * @throws AuthorizationFailedException if the authorization against the webservice failed.
+ * @throws TooManyRequestsException if too many requests have been made against the webservice recently.
+ */
+ void putLight(String deviceId, boolean enabled);
+
+ /**
+ * Performs a PUT operation switching the device on or off.
+ *
+ * @param deviceId ID of the device to trigger the action for.
+ * @param enabled {@code true} to switch on or {@code false} to switch off the device.
+ * @throws MieleWebserviceException if an error occurs during webservice requests or content parsing.
+ * @throws AuthorizationFailedException if the authorization against the webservice failed.
+ * @throws TooManyRequestsException if too many requests have been made against the webservice recently.
+ */
+ void putPowerState(String deviceId, boolean enabled);
+
+ /**
+ * Performs a PUT operation setting the active program.
+ *
+ * @param deviceId ID of the device to trigger the action for.
+ * @param program The program to activate.
+ * @throws MieleWebserviceException if an error occurs during webservice requests or content parsing.
+ * @throws AuthorizationFailedException if the authorization against the webservice failed.
+ * @throws TooManyRequestsException if too many requests have been made against the webservice recently.
+ */
+ void putProgram(String deviceId, long programId);
+
+ /**
+ * Performs a logout and invalidates the current OAuth2 token. This operation is assumed to work on the first try
+ * and is never retried. HTTP errors are ignored.
+ *
+ * @throws MieleWebserviceException if the request operation fails.
+ */
+ void logout();
+
+ /**
+ * Dispatches the cached state of the device identified by the given device identifier.
+ */
+ void dispatchDeviceState(String deviceIdentifier);
+
+ /**
+ * Adds a {@link DeviceStateListener}.
+ *
+ * @param listener The listener to add.
+ */
+ void addDeviceStateListener(DeviceStateListener listener);
+
+ /**
+ * Removes a {@link DeviceStateListener}.
+ *
+ * @param listener The listener to remove.
+ */
+ void removeDeviceStateListener(DeviceStateListener listener);
+
+ /**
+ * Adds a {@link ConnectionStatusListener}.
+ *
+ * @param listener The listener to add.
+ */
+ void addConnectionStatusListener(ConnectionStatusListener listener);
+
+ /**
+ * Removes a {@link ConnectionStatusListener}.
+ *
+ * @param listener The listener to remove.
+ */
+ void removeConnectionStatusListener(ConnectionStatusListener listener);
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/MieleWebserviceConfiguration.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/MieleWebserviceConfiguration.java
new file mode 100644
index 00000000000..08b23d36ce4
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/MieleWebserviceConfiguration.java
@@ -0,0 +1,134 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice;
+
+import java.util.concurrent.ScheduledExecutorService;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.mielecloud.internal.auth.OAuthTokenRefresher;
+import org.openhab.binding.mielecloud.internal.webservice.language.LanguageProvider;
+import org.openhab.core.io.net.http.HttpClientFactory;
+
+/**
+ * Represents a webservice configuration.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public final class MieleWebserviceConfiguration {
+ private final HttpClientFactory httpClientFactory;
+ private final LanguageProvider languageProvider;
+ private final OAuthTokenRefresher tokenRefresher;
+ private final String serviceHandle;
+ private final ScheduledExecutorService scheduler;
+
+ private MieleWebserviceConfiguration(MieleWebserviceConfigurationBuilder builder) {
+ this.httpClientFactory = getOrThrow(builder.httpClientFactory, "httpClientFactory");
+ this.languageProvider = getOrThrow(builder.languageProvider, "languageProvider");
+ this.tokenRefresher = getOrThrow(builder.tokenRefresher, "tokenRefresher");
+ this.serviceHandle = getOrThrow(builder.serviceHandle, "serviceHandle");
+ this.scheduler = getOrThrow(builder.scheduler, "scheduler");
+ }
+
+ private static T getOrThrow(@Nullable T object, String objectName) {
+ if (object == null) {
+ throw new IllegalArgumentException(objectName + " must not be null");
+ }
+ return object;
+ }
+
+ /**
+ * Gets the factory to use for HttpClient construction.
+ */
+ public HttpClientFactory getHttpClientFactory() {
+ return httpClientFactory;
+ }
+
+ /**
+ * Gets the provider for the language to use when making requests to the API.
+ */
+ public LanguageProvider getLanguageProvider() {
+ return languageProvider;
+ }
+
+ /**
+ * Gets the refresher for OAuth tokens.
+ */
+ public OAuthTokenRefresher getTokenRefresher() {
+ return tokenRefresher;
+ }
+
+ /**
+ * Gets the handle referring to the OAuth tokens in the framework's persistent storage.
+ */
+ public String getServiceHandle() {
+ return serviceHandle;
+ }
+
+ /**
+ * Gets the system wide scheduler.
+ */
+ public ScheduledExecutorService getScheduler() {
+ return scheduler;
+ }
+
+ public static MieleWebserviceConfigurationBuilder builder() {
+ return new MieleWebserviceConfigurationBuilder();
+ }
+
+ public static final class MieleWebserviceConfigurationBuilder {
+ @Nullable
+ private HttpClientFactory httpClientFactory;
+ @Nullable
+ private LanguageProvider languageProvider;
+ @Nullable
+ private OAuthTokenRefresher tokenRefresher;
+ @Nullable
+ private String serviceHandle;
+ @Nullable
+ private ScheduledExecutorService scheduler;
+
+ private MieleWebserviceConfigurationBuilder() {
+ }
+
+ public MieleWebserviceConfigurationBuilder withHttpClientFactory(HttpClientFactory httpClientFactory) {
+ this.httpClientFactory = httpClientFactory;
+ return this;
+ }
+
+ public MieleWebserviceConfigurationBuilder withLanguageProvider(LanguageProvider languageProvider) {
+ this.languageProvider = languageProvider;
+ return this;
+ }
+
+ public MieleWebserviceConfigurationBuilder withTokenRefresher(OAuthTokenRefresher tokenRefresher) {
+ this.tokenRefresher = tokenRefresher;
+ return this;
+ }
+
+ public MieleWebserviceConfigurationBuilder withServiceHandle(String serviceHandle) {
+ this.serviceHandle = serviceHandle;
+ return this;
+ }
+
+ public MieleWebserviceConfigurationBuilder withScheduler(ScheduledExecutorService scheduler) {
+ this.scheduler = scheduler;
+ return this;
+ }
+
+ public MieleWebserviceConfiguration build() {
+ return new MieleWebserviceConfiguration(this);
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/MieleWebserviceFactory.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/MieleWebserviceFactory.java
new file mode 100644
index 00000000000..576c3d826a3
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/MieleWebserviceFactory.java
@@ -0,0 +1,31 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Factory for creating {@link MieleWebservice} instances.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public interface MieleWebserviceFactory {
+ /**
+ * Creates a new {@link MieleWebservice}.
+ *
+ * @param configuration The configuration holding all required parameters to construct the instance.
+ * @return A new {@link MieleWebservice}.
+ */
+ public MieleWebservice create(MieleWebserviceConfiguration configuration);
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/UnavailableMieleWebservice.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/UnavailableMieleWebservice.java
new file mode 100644
index 00000000000..71d7df38687
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/UnavailableMieleWebservice.java
@@ -0,0 +1,118 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.ProcessAction;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * Implementation of {@link MieleWebservice} that serves as a replacement when no webservice is available.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class UnavailableMieleWebservice implements MieleWebservice {
+ public static final UnavailableMieleWebservice INSTANCE = new UnavailableMieleWebservice();
+
+ private final Logger logger = LoggerFactory.getLogger(this.getClass());
+
+ private UnavailableMieleWebservice() {
+ }
+
+ @Override
+ public void setAccessToken(String accessToken) {
+ logger.warn("Cannot set access token: The Miele cloud service is not available.");
+ }
+
+ @Override
+ public boolean hasAccessToken() {
+ logger.warn("There is no access token: The Miele cloud service is not available.");
+ return false;
+ }
+
+ @Override
+ public void connectSse() {
+ logger.warn("Cannot connect to SSE stream: The Miele cloud service is not available.");
+ }
+
+ @Override
+ public void disconnectSse() {
+ logger.warn("Cannot disconnect from SSE stream: The Miele cloud service is not available.");
+ }
+
+ @Override
+ public void fetchActions(String deviceId) {
+ logger.warn("Cannot fetch actions for device '{}': The Miele cloud service is not available.", deviceId);
+ }
+
+ @Override
+ public void putProcessAction(String deviceId, ProcessAction processAction) {
+ logger.warn("Cannot perform '{}' operation for device '{}': The Miele cloud service is not available.",
+ processAction, deviceId);
+ }
+
+ @Override
+ public void putLight(String deviceId, boolean enabled) {
+ logger.warn("Cannot set light state to '{}' for device '{}': The Miele cloud service is not available.",
+ enabled ? "ON" : "OFF", deviceId);
+ }
+
+ @Override
+ public void putPowerState(String deviceId, boolean enabled) {
+ logger.warn("Cannot set power state to '{}' for device '{}': The Miele cloud service is not available.",
+ enabled ? "ON" : "OFF", deviceId);
+ }
+
+ @Override
+ public void putProgram(String deviceId, long programId) {
+ logger.warn("Cannot activate program with ID '{}' for device '{}': The Miele cloud service is not available.",
+ programId, deviceId);
+ }
+
+ @Override
+ public void logout() {
+ logger.warn("Cannot logout: The Miele cloud service is not available.");
+ }
+
+ @Override
+ public void dispatchDeviceState(String deviceIdentifier) {
+ logger.warn("Cannot re-emit device state for device '{}': The Miele cloud service is not available.",
+ deviceIdentifier);
+ }
+
+ @Override
+ public void addDeviceStateListener(DeviceStateListener listener) {
+ logger.warn("Cannot add listener for all devices: The Miele cloud service is not available.");
+ }
+
+ @Override
+ public void removeDeviceStateListener(DeviceStateListener listener) {
+ logger.warn("Cannot remove listener: The Miele cloud service is not available.");
+ }
+
+ @Override
+ public void addConnectionStatusListener(ConnectionStatusListener listener) {
+ logger.warn("Cannot add connection error listener: The Miele cloud service is not available.");
+ }
+
+ @Override
+ public void removeConnectionStatusListener(ConnectionStatusListener listener) {
+ logger.warn("Cannot remove listener: The Miele cloud service is not available.");
+ }
+
+ @Override
+ public void close() throws Exception {
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/ActionsState.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/ActionsState.java
new file mode 100644
index 00000000000..e29be3c1885
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/ActionsState.java
@@ -0,0 +1,176 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api;
+
+import java.util.List;
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.Actions;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.Light;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.ProcessAction;
+
+/**
+ * Provides convenient access to the list of actions that can be performed with a device.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ */
+@NonNullByDefault
+public class ActionsState {
+
+ private final String deviceIdentifier;
+ private final Optional actions;
+
+ public ActionsState(String deviceIdentifier, @Nullable Actions actions) {
+ this.deviceIdentifier = deviceIdentifier;
+ this.actions = Optional.ofNullable(actions);
+ }
+
+ /**
+ * Gets the unique identifier of the device to which this state refers.
+ */
+ public String getDeviceIdentifier() {
+ return deviceIdentifier;
+ }
+
+ /**
+ * Gets whether the device can be started.
+ */
+ public boolean canBeStarted() {
+ return actions.map(Actions::getProcessAction).map(a -> a.contains(ProcessAction.START)).orElse(false);
+ }
+
+ /**
+ * Gets whether the device can be stopped.
+ */
+ public boolean canBeStopped() {
+ return actions.map(Actions::getProcessAction).map(a -> a.contains(ProcessAction.STOP)).orElse(false);
+ }
+
+ /**
+ * Gets whether the device can be paused.
+ */
+ public boolean canBePaused() {
+ return actions.map(Actions::getProcessAction).map(a -> a.contains(ProcessAction.PAUSE)).orElse(false);
+ }
+
+ /**
+ * Gets whether supercooling can be controlled.
+ */
+ public boolean canContolSupercooling() {
+ return canStartSupercooling() || canStopSupercooling();
+ }
+
+ /**
+ * Gets whether supercooling can be started.
+ */
+ public boolean canStartSupercooling() {
+ return actions.map(Actions::getProcessAction).map(a -> a.contains(ProcessAction.START_SUPERCOOLING))
+ .orElse(false);
+ }
+
+ /**
+ * Gets whether supercooling can be stopped.
+ */
+ public boolean canStopSupercooling() {
+ return actions.map(Actions::getProcessAction).map(a -> a.contains(ProcessAction.STOP_SUPERCOOLING))
+ .orElse(false);
+ }
+
+ /**
+ * Gets whether superfreezing can be controlled.
+ */
+ public boolean canControlSuperfreezing() {
+ return canStartSuperfreezing() || canStopSuperfreezing();
+ }
+
+ /**
+ * Gets whether superfreezing can be started.
+ */
+ public boolean canStartSuperfreezing() {
+ return actions.map(Actions::getProcessAction).map(a -> a.contains(ProcessAction.START_SUPERFREEZING))
+ .orElse(false);
+ }
+
+ /**
+ * Gets whether superfreezing can be stopped.
+ */
+ public boolean canStopSuperfreezing() {
+ return actions.map(Actions::getProcessAction).map(a -> a.contains(ProcessAction.STOP_SUPERFREEZING))
+ .orElse(false);
+ }
+
+ /**
+ * Gets whether light can be enabled.
+ */
+ public boolean canEnableLight() {
+ return actions.map(Actions::getLight).map(a -> a.contains(Light.ENABLE)).orElse(false);
+ }
+
+ /**
+ * Gets whether light can be disabled.
+ */
+ public boolean canDisableLight() {
+ return actions.map(Actions::getLight).map(a -> a.contains(Light.DISABLE)).orElse(false);
+ }
+
+ /**
+ * Gets whether the device can be switched on.
+ */
+ public boolean canBeSwitchedOn() {
+ return actions.flatMap(Actions::getPowerOn).map(Boolean.TRUE::equals).orElse(false);
+ }
+
+ /**
+ * Gets whether the device can be switched off.
+ */
+ public boolean canBeSwitchedOff() {
+ return actions.flatMap(Actions::getPowerOff).map(Boolean.TRUE::equals).orElse(false);
+ }
+
+ /**
+ * Gets whether the light can be controlled.
+ */
+ public boolean canControlLight() {
+ return canEnableLight() || canDisableLight();
+ }
+
+ /**
+ * Gets whether the active program can be set.
+ */
+ public boolean canSetActiveProgramId() {
+ return !actions.map(Actions::getProgramId).map(List::isEmpty).orElse(true);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(actions, deviceIdentifier);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ ActionsState other = (ActionsState) obj;
+ return Objects.equals(actions, other.actions) && Objects.equals(deviceIdentifier, other.deviceIdentifier);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/CoolingDeviceTemperatureState.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/CoolingDeviceTemperatureState.java
new file mode 100644
index 00000000000..2e22ac2a3da
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/CoolingDeviceTemperatureState.java
@@ -0,0 +1,103 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api;
+
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Provides easy access to temperature values mapped for cooling devices.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class CoolingDeviceTemperatureState {
+ private final DeviceState deviceState;
+
+ public CoolingDeviceTemperatureState(DeviceState deviceState) {
+ this.deviceState = deviceState;
+ }
+
+ /**
+ * Gets the current temperature of the fridge part of the device.
+ *
+ * @return The current temperature of the fridge part of the device.
+ */
+ public Optional getFridgeTemperature() {
+ switch (deviceState.getRawType()) {
+ case FRIDGE:
+ return deviceState.getTemperature(0);
+
+ case FRIDGE_FREEZER_COMBINATION:
+ return deviceState.getTemperature(0);
+
+ default:
+ return Optional.empty();
+ }
+ }
+
+ /**
+ * Gets the target temperature of the fridge part of the device.
+ *
+ * @return The target temperature of the fridge part of the device.
+ */
+ public Optional getFridgeTargetTemperature() {
+ switch (deviceState.getRawType()) {
+ case FRIDGE:
+ return deviceState.getTargetTemperature(0);
+
+ case FRIDGE_FREEZER_COMBINATION:
+ return deviceState.getTargetTemperature(0);
+
+ default:
+ return Optional.empty();
+ }
+ }
+
+ /**
+ * Gets the current temperature of the freezer part of the device.
+ *
+ * @return The current temperature of the freezer part of the device.
+ */
+ public Optional getFreezerTemperature() {
+ switch (deviceState.getRawType()) {
+ case FREEZER:
+ return deviceState.getTemperature(0);
+
+ case FRIDGE_FREEZER_COMBINATION:
+ return deviceState.getTemperature(1);
+
+ default:
+ return Optional.empty();
+ }
+ }
+
+ /**
+ * Gets the target temperature of the freezer part of the device.
+ *
+ * @return The target temperature of the freezer part of the device.
+ */
+ public Optional getFreezerTargetTemperature() {
+ switch (deviceState.getRawType()) {
+ case FREEZER:
+ return deviceState.getTargetTemperature(0);
+
+ case FRIDGE_FREEZER_COMBINATION:
+ return deviceState.getTargetTemperature(1);
+
+ default:
+ return Optional.empty();
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/DeviceState.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/DeviceState.java
new file mode 100644
index 00000000000..c4ab6d28bc2
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/DeviceState.java
@@ -0,0 +1,558 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api;
+
+import java.util.List;
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.Device;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.DeviceIdentLabel;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.DeviceType;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.DryingStep;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.Ident;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.Light;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.PlateStep;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.ProgramId;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.ProgramPhase;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.RemoteEnable;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.SpinningSpeed;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.State;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.StateType;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.Status;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.Temperature;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.Type;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.VentilationStep;
+
+/**
+ * This immutable class provides methods to extract the device state information in a comfortable way.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Introduced null handling
+ * @author Benjamin Bolte - Add pre-heat finished, plate step, door state, door alarm, info state channel and map signal
+ * flags from API
+ * @author Björn Lange - Add elapsed time channel, dish warmer and robotic vacuum cleaner things
+ */
+@NonNullByDefault
+public class DeviceState {
+
+ private final String deviceIdentifier;
+
+ private final Optional device;
+
+ public DeviceState(String deviceIdentifier, @Nullable Device device) {
+ this.deviceIdentifier = deviceIdentifier;
+ this.device = Optional.ofNullable(device);
+ }
+
+ /**
+ * Gets the unique identifier for this device.
+ *
+ * @return The unique identifier for this device.
+ */
+ public String getDeviceIdentifier() {
+ return deviceIdentifier;
+ }
+
+ /**
+ * Gets the main operation status of the device.
+ *
+ * @return The main operation status of the device.
+ */
+ public Optional getStatus() {
+ return device.flatMap(Device::getState).flatMap(State::getStatus).flatMap(Status::getValueLocalized);
+ }
+
+ /**
+ * Gets the raw main operation status of the device.
+ *
+ * @return The raw main operation status of the device.
+ */
+ public Optional getStatusRaw() {
+ return device.flatMap(Device::getState).flatMap(State::getStatus).flatMap(Status::getValueRaw);
+ }
+
+ /**
+ * Gets the raw operation status of the device parsed to a {@link StateType}.
+ *
+ * @return The raw operation status of the device parsed to a {@link StateType}.
+ */
+ public Optional getStateType() {
+ return device.flatMap(Device::getState).flatMap(State::getStatus).flatMap(Status::getValueRaw)
+ .flatMap(StateType::fromCode);
+ }
+
+ /**
+ * Gets the currently selected program type of the device.
+ *
+ * @return The currently selected program type of the device.
+ */
+ public Optional getSelectedProgram() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).flatMap(State::getProgramId).flatMap(ProgramId::getValueLocalized);
+ }
+
+ /**
+ * Gets the selected program ID.
+ *
+ * @return The selected program ID.
+ */
+ public Optional getSelectedProgramId() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).flatMap(State::getProgramId).flatMap(ProgramId::getValueRaw);
+ }
+
+ /**
+ * Gets the currently active phase of the active program.
+ *
+ * @return The currently active phase of the active program.
+ */
+ public Optional getProgramPhase() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).flatMap(State::getProgramPhase)
+ .flatMap(ProgramPhase::getValueLocalized);
+ }
+
+ /**
+ * Gets the currently active raw phase of the active program.
+ *
+ * @return The currently active raw phase of the active program.
+ */
+ public Optional getProgramPhaseRaw() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).flatMap(State::getProgramPhase).flatMap(ProgramPhase::getValueRaw);
+ }
+
+ /**
+ * Gets the currently selected drying step.
+ *
+ * @return The currently selected drying step.
+ */
+ public Optional getDryingTarget() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).flatMap(State::getDryingStep).flatMap(DryingStep::getValueLocalized);
+ }
+
+ /**
+ * Gets the currently selected raw drying step.
+ *
+ * @return The currently selected raw drying step.
+ */
+ public Optional getDryingTargetRaw() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).flatMap(State::getDryingStep).flatMap(DryingStep::getValueRaw);
+ }
+
+ /**
+ * Calculates if pre-heating the oven has finished.
+ *
+ * @return Whether pre-heating the oven has finished.
+ */
+ public Optional hasPreHeatFinished() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+
+ Optional targetTemperature = getTargetTemperature(0);
+ Optional currentTemperature = getTemperature(0);
+
+ if (!targetTemperature.isPresent() || !currentTemperature.isPresent()) {
+ return Optional.empty();
+ }
+
+ return Optional.of(isInState(StateType.RUNNING) && currentTemperature.get() >= targetTemperature.get());
+ }
+
+ /**
+ * Gets the target temperature with the given index.
+ *
+ * @return The target temperature with the given index.
+ */
+ public Optional getTargetTemperature(int index) {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).map(State::getTargetTemperature).flatMap(l -> getOrNull(l, index))
+ .flatMap(Temperature::getValueLocalized);
+ }
+
+ /**
+ * Gets the current temperature of the device for the given index.
+ *
+ * @param index The index of the device zone for which the temperature shall be obtained.
+ * @return The target temperature if available.
+ */
+ public Optional getTemperature(int index) {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+
+ return device.flatMap(Device::getState).map(State::getTemperature).flatMap(l -> getOrNull(l, index))
+ .flatMap(Temperature::getValueLocalized);
+ }
+
+ /**
+ * Gets the remaining time of the active program.
+ *
+ * @return The remaining time in seconds.
+ */
+ public Optional getRemainingTime() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).flatMap(State::getRemainingTime).flatMap(this::toSeconds);
+ }
+
+ /**
+ * Gets the elapsed time of the active program.
+ *
+ * @return The elapsed time in seconds.
+ */
+ public Optional getElapsedTime() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).flatMap(State::getElapsedTime).flatMap(this::toSeconds);
+ }
+
+ /**
+ * Gets the relative start time of the active program.
+ *
+ * @return The delayed start time in seconds.
+ */
+ public Optional getStartTime() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).flatMap(State::getStartTime).flatMap(this::toSeconds);
+ }
+
+ /**
+ * Gets the "fullRemoteControl" state information of the device. If this flag is true ALL remote control actions
+ * of the device can be triggered.
+ *
+ * @return Whether the device can be remote controlled.
+ */
+ public Optional isRemoteControlEnabled() {
+ return device.flatMap(Device::getState).flatMap(State::getRemoteEnable)
+ .flatMap(RemoteEnable::getFullRemoteControl);
+ }
+
+ /**
+ * Calculates the program process.
+ *
+ * @return The progress of the active program in percent.
+ */
+ public Optional getProgress() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+
+ Optional elapsedTime = device.flatMap(Device::getState).flatMap(State::getElapsedTime)
+ .flatMap(this::toSeconds).map(Integer::doubleValue);
+ Optional remainingTime = device.flatMap(Device::getState).flatMap(State::getRemainingTime)
+ .flatMap(this::toSeconds).map(Integer::doubleValue);
+
+ if (elapsedTime.isPresent() && remainingTime.isPresent()
+ && (elapsedTime.get() != 0 || remainingTime.get() != 0)) {
+ return Optional.of((int) ((elapsedTime.get() / (elapsedTime.get() + remainingTime.get())) * 100.0));
+ } else {
+ return Optional.empty();
+ }
+ }
+
+ private Optional toSeconds(List time) {
+ if (time.size() != 2) {
+ return Optional.empty();
+ }
+ return Optional.of((time.get(0) * 60 + time.get(1)) * 60);
+ }
+
+ /**
+ * Gets the spinning speed.
+ *
+ * @return The spinning speed.
+ */
+ public Optional getSpinningSpeed() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).flatMap(State::getSpinningSpeed).flatMap(SpinningSpeed::getValueRaw)
+ .map(String::valueOf);
+ }
+
+ /**
+ * Gets the raw spinning speed.
+ *
+ * @return The raw spinning speed.
+ */
+ public Optional getSpinningSpeedRaw() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).flatMap(State::getSpinningSpeed).flatMap(SpinningSpeed::getValueRaw);
+ }
+
+ /**
+ * Gets the ventilation step.
+ *
+ * @return The ventilation step.
+ */
+ public Optional getVentilationStep() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).flatMap(State::getVentilationStep)
+ .flatMap(VentilationStep::getValueLocalized).map(Object::toString);
+ }
+
+ /**
+ * Gets the raw ventilation step.
+ *
+ * @return The raw ventilation step.
+ */
+ public Optional getVentilationStepRaw() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).flatMap(State::getVentilationStep)
+ .flatMap(VentilationStep::getValueRaw);
+ }
+
+ /**
+ * Gets the plate power step of the device for the given index.
+ *
+ * @param index The index of the device plate for which the power step shall be obtained.
+ * @return The plate power step if available.
+ */
+ public Optional getPlateStep(int index) {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).map(State::getPlateStep).flatMap(l -> getOrNull(l, index))
+ .flatMap(PlateStep::getValueLocalized);
+ }
+
+ /**
+ * Gets the raw plate power step of the device for the given index.
+ *
+ * @param index The index of the device plate for which the power step shall be obtained.
+ * @return The raw plate power step if available.
+ */
+ public Optional getPlateStepRaw(int index) {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+ return device.flatMap(Device::getState).map(State::getPlateStep).flatMap(l -> getOrNull(l, index))
+ .flatMap(PlateStep::getValueRaw);
+ }
+
+ /**
+ * Gets the number of available plate steps.
+ *
+ * @return The number of available plate steps.
+ */
+ public Optional getPlateStepCount() {
+ return device.flatMap(Device::getState).map(State::getPlateStep).map(List::size);
+ }
+
+ /**
+ * Indicates if the device has an error that requires a user action.
+ *
+ * @return Whether the device has an error that requires a user action.
+ */
+ public boolean hasError() {
+ return isInState(StateType.FAILURE)
+ || device.flatMap(Device::getState).flatMap(State::getSignalFailure).orElse(false);
+ }
+
+ /**
+ * Indicates if the device has a user information.
+ *
+ * @return Whether the device has a user information.
+ */
+ public boolean hasInfo() {
+ if (deviceIsInOffState()) {
+ return false;
+ }
+ return device.flatMap(Device::getState).flatMap(State::getSignalInfo).orElse(false);
+ }
+
+ /**
+ * Gets the state of the light attached to the device.
+ *
+ * @return An {@link Optional} with value {@code true} if the light is turned on, {@code false} if the light is
+ * turned off or an empty {@link Optional} if light is not supported or no state is available.
+ */
+ public Optional getLightState() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+
+ Optional light = device.flatMap(Device::getState).map(State::getLight);
+ if (light.isPresent()) {
+ if (light.get().equals(Light.ENABLE)) {
+ return Optional.of(true);
+ } else if (light.get().equals(Light.DISABLE)) {
+ return Optional.of(false);
+ }
+ }
+
+ return Optional.empty();
+ }
+
+ /**
+ * Gets the state of the door attached to the device.
+ *
+ * @return Whether the device door is open.
+ */
+ public Optional getDoorState() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+
+ return device.flatMap(Device::getState).flatMap(State::getSignalDoor);
+ }
+
+ /**
+ * Gets the state of the device's door alarm.
+ *
+ * @return Whether the device door alarm was triggered.
+ */
+ public Optional getDoorAlarm() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+
+ Optional doorState = getDoorState();
+ Optional failure = device.flatMap(Device::getState).flatMap(State::getSignalFailure);
+
+ if (!doorState.isPresent() || !failure.isPresent()) {
+ return Optional.empty();
+ }
+
+ return Optional.of(doorState.get() && failure.get());
+ }
+
+ /**
+ * Gets the battery level.
+ *
+ * @return The battery level.
+ */
+ public Optional getBatteryLevel() {
+ if (deviceIsInOffState()) {
+ return Optional.empty();
+ }
+
+ return device.flatMap(Device::getState).flatMap(State::getBatteryLevel);
+ }
+
+ /**
+ * Gets the device type.
+ *
+ * @return The device type as human readable value.
+ */
+ public Optional getType() {
+ return device.flatMap(Device::getIdent).flatMap(Ident::getType).flatMap(Type::getValueLocalized)
+ .filter(type -> !type.isEmpty());
+ }
+
+ /**
+ * Gets the raw device type.
+ *
+ * @return The raw device type.
+ */
+ public DeviceType getRawType() {
+ return device.flatMap(Device::getIdent).flatMap(Ident::getType).map(Type::getValueRaw)
+ .orElse(DeviceType.UNKNOWN);
+ }
+
+ /**
+ * Gets the user-defined name of the device.
+ *
+ * @return The user-defined name of the device.
+ */
+ public Optional getDeviceName() {
+ return device.flatMap(Device::getIdent).flatMap(Ident::getDeviceName).filter(name -> !name.isEmpty());
+ }
+
+ /**
+ * Gets the fabrication (=serial) number of the device.
+ *
+ * @return The serial number of the device.
+ */
+ public Optional getFabNumber() {
+ return device.flatMap(Device::getIdent).flatMap(Ident::getDeviceIdentLabel)
+ .flatMap(DeviceIdentLabel::getFabNumber).filter(fabNumber -> !fabNumber.isEmpty());
+ }
+
+ /**
+ * Gets the tech type of the device.
+ *
+ * @return The tech type of the device.
+ */
+ public Optional getTechType() {
+ return device.flatMap(Device::getIdent).flatMap(Ident::getDeviceIdentLabel)
+ .flatMap(DeviceIdentLabel::getTechType).filter(techType -> !techType.isEmpty());
+ }
+
+ private Optional getOrNull(List list, int index) {
+ if (index < 0 || index >= list.size()) {
+ return Optional.empty();
+ }
+
+ return Optional.ofNullable(list.get(index));
+ }
+
+ private boolean deviceIsInOffState() {
+ return getStateType().map(StateType.OFF::equals).orElse(true);
+ }
+
+ public boolean isInState(StateType stateType) {
+ return getStateType().map(stateType::equals).orElse(false);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(device, deviceIdentifier);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ DeviceState other = (DeviceState) obj;
+ return Objects.equals(device, other.device) && Objects.equals(deviceIdentifier, other.deviceIdentifier);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/PowerStatus.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/PowerStatus.java
new file mode 100644
index 00000000000..a68fce9273c
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/PowerStatus.java
@@ -0,0 +1,50 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Represents the power status of the device, i.e. whether it is powered on, off or in standby.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public enum PowerStatus {
+ POWER_ON("on"),
+ POWER_OFF("off"),
+ STANDBY("standby");
+
+ /**
+ * Corresponding state of the ChannelTypeDefinition
+ */
+ private String state;
+
+ PowerStatus(String value) {
+ this.state = value;
+ }
+
+ /**
+ * Checks whether the given value is the raw state represented by this enum instance.
+ */
+ public boolean matches(String passedValue) {
+ return state.equalsIgnoreCase(passedValue);
+ }
+
+ /**
+ * Gets the raw state.
+ */
+ public String getState() {
+ return state;
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/ProgramStatus.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/ProgramStatus.java
new file mode 100644
index 00000000000..bb73af48639
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/ProgramStatus.java
@@ -0,0 +1,50 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Represents the status of a program.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public enum ProgramStatus {
+ PROGRAM_STARTED("start"),
+ PROGRAM_STOPPED("stop"),
+ PROGRAM_PAUSED("pause");
+
+ /**
+ * Corresponding state of the ChannelTypeDefinition
+ */
+ private String state;
+
+ ProgramStatus(String value) {
+ this.state = value;
+ }
+
+ /**
+ * Checks whether the given value is the raw state represented by this enum instance.
+ */
+ public boolean matches(String passedValue) {
+ return state.equalsIgnoreCase(passedValue);
+ }
+
+ /**
+ * Gets the raw state.
+ */
+ public String getState() {
+ return state;
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/TransitionState.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/TransitionState.java
new file mode 100644
index 00000000000..d2dd05c3c4d
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/TransitionState.java
@@ -0,0 +1,147 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api;
+
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.StateType;
+
+/**
+ * This immutable class provides methods to extract the state information related to state transitions in a comfortable
+ * way.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class TransitionState {
+ private final boolean remainingTimeWasSetInCurrentProgram;
+ private final Optional previousState;
+ private final DeviceState nextState;
+
+ /**
+ * Creates a new {@link TransitionState}.
+ *
+ * Note: {@code previousState} must not be saved in a field in this class as this will create a linked list
+ * and cause memory issues. The constructor only serves the purpose of unpacking state that must be carried on.
+ *
+ * @param previousTransitionState The previous transition state if it exists.
+ * @param nextState The device state which the device is transitioning to.
+ */
+ public TransitionState(@Nullable TransitionState previousTransitionState, DeviceState nextState) {
+ this.remainingTimeWasSetInCurrentProgram = wasRemainingTimeSetInCurrentProgram(previousTransitionState,
+ nextState);
+ this.previousState = Optional.ofNullable(previousTransitionState).map(it -> it.nextState);
+ this.nextState = nextState;
+ }
+
+ /**
+ * Gets whether the finish state changed due to the transition form the previous to the current state.
+ *
+ * @return Whether the finish state changed due to the transition form the previous to the current state.
+ */
+ public boolean hasFinishedChanged() {
+ return previousState.map(this::hasFinishedChangedFromPreviousState).orElse(true);
+ }
+
+ private boolean hasFinishedChangedFromPreviousState(DeviceState previous) {
+ if (previous.getStateType().equals(nextState.getStateType())) {
+ return false;
+ }
+
+ if (isInRunningState(previous) && nextState.isInState(StateType.FAILURE)) {
+ return false;
+ }
+
+ if (isInRunningState(previous) != isInRunningState(nextState)) {
+ return true;
+ }
+
+ if (nextState.isInState(StateType.OFF)) {
+ return true;
+ }
+
+ return false;
+ }
+
+ /**
+ * Gets whether a program finished.
+ *
+ * @return Whether a program finished.
+ */
+ public Optional isFinished() {
+ return previousState.flatMap(this::hasFinishedFromPreviousState);
+ }
+
+ private Optional hasFinishedFromPreviousState(DeviceState prevState) {
+ if (!prevState.getStateType().isPresent()) {
+ return Optional.empty();
+ }
+
+ if (nextState.isInState(StateType.OFF)) {
+ return Optional.of(false);
+ }
+
+ if (nextState.isInState(StateType.FAILURE)) {
+ return Optional.of(false);
+ }
+
+ return Optional.of(!isInRunningState(nextState));
+ }
+
+ /**
+ * Gets the remaining time of the active program.
+ *
+ * Note: Tracking changes in the remaining time is a workaround for the Miele API not properly distinguishing
+ * between "there is no remaining time set" and "the remaining time is zero". If the remaining time is zero when a
+ * program is started then we assume that no timer was set / program with remaining time is active. This may be
+ * changed later by the user which is detected by the remaining time changing from 0 to some larger value.
+ *
+ * @return The remaining time in seconds.
+ */
+ public Optional getRemainingTime() {
+ if (!remainingTimeWasSetInCurrentProgram && isInRunningState(nextState)) {
+ return nextState.getRemainingTime().filter(it -> it != 0);
+ } else {
+ return nextState.getRemainingTime();
+ }
+ }
+
+ /**
+ * Gets the program progress.
+ *
+ * @return The progress of the active program in percent.
+ */
+ public Optional getProgress() {
+ if (getRemainingTime().isPresent()) {
+ return nextState.getProgress();
+ } else {
+ return Optional.empty();
+ }
+ }
+
+ private static boolean wasRemainingTimeSetInCurrentProgram(@Nullable TransitionState previousTransitionState,
+ DeviceState nextState) {
+ if (previousTransitionState != null && isInRunningState(previousTransitionState.nextState)) {
+ return previousTransitionState.remainingTimeWasSetInCurrentProgram
+ || previousTransitionState.getRemainingTime().isPresent();
+ } else {
+ return false;
+ }
+ }
+
+ private static boolean isInRunningState(DeviceState device) {
+ return device.isInState(StateType.RUNNING) || device.isInState(StateType.PAUSE);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/WineStorageDeviceTemperatureState.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/WineStorageDeviceTemperatureState.java
new file mode 100644
index 00000000000..5b8915ea8ee
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/WineStorageDeviceTemperatureState.java
@@ -0,0 +1,206 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api;
+
+import java.util.Arrays;
+import java.util.List;
+import java.util.Optional;
+import java.util.Set;
+import java.util.stream.Collectors;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.webservice.api.json.DeviceType;
+
+/**
+ * Provides easy access to temperature values mapped for wine storage devices.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class WineStorageDeviceTemperatureState {
+ private static final Set ALL_WINE_STORAGES = Set.of(DeviceType.WINE_CABINET,
+ DeviceType.WINE_CABINET_FREEZER_COMBINATION, DeviceType.WINE_CONDITIONING_UNIT,
+ DeviceType.WINE_STORAGE_CONDITIONING_UNIT);
+
+ private final DeviceState deviceState;
+ private final List effectiveTemperatures;
+ private final List effectiveTargetTemperatures;
+
+ /**
+ * Creates a new {@link WineStorageDeviceTemperatureState}.
+ *
+ * @param deviceState Device state to query extended state information from.
+ */
+ public WineStorageDeviceTemperatureState(DeviceState deviceState) {
+ this.deviceState = deviceState;
+ effectiveTemperatures = getEffectiveTemperatures();
+ effectiveTargetTemperatures = getEffectiveTargetTemperatures();
+ }
+
+ private List getEffectiveTemperatures() {
+ return Arrays
+ .asList(deviceState.getTemperature(0), deviceState.getTemperature(1), deviceState.getTemperature(2))
+ .stream().filter(Optional::isPresent).map(Optional::get).collect(Collectors.toList());
+ }
+
+ private List getEffectiveTargetTemperatures() {
+ return Arrays
+ .asList(deviceState.getTargetTemperature(0), deviceState.getTargetTemperature(1),
+ deviceState.getTargetTemperature(2))
+ .stream().filter(Optional::isPresent).map(Optional::get).collect(Collectors.toList());
+ }
+
+ /**
+ * Gets the current main temperature of the wine storage.
+ *
+ * @return The current main temperature of the wine storage.
+ */
+ public Optional getTemperature() {
+ if (!ALL_WINE_STORAGES.contains(deviceState.getRawType())) {
+ return Optional.empty();
+ }
+
+ return getTemperatureFromList(effectiveTemperatures);
+ }
+
+ /**
+ * Gets the target main temperature of the wine storage.
+ *
+ * @return The target main temperature of the wine storage.
+ */
+ public Optional getTargetTemperature() {
+ if (!ALL_WINE_STORAGES.contains(deviceState.getRawType())) {
+ return Optional.empty();
+ }
+
+ return getTemperatureFromList(effectiveTargetTemperatures);
+ }
+
+ private Optional getTemperatureFromList(List temperatures) {
+ if (temperatures.isEmpty()) {
+ return Optional.empty();
+ }
+
+ if (temperatures.size() > 1) {
+ return Optional.empty();
+ }
+
+ return Optional.of(temperatures.get(0));
+ }
+
+ /**
+ * Gets the current top temperature of the wine storage.
+ *
+ * @return The current top temperature of the wine storage.
+ */
+ public Optional getTopTemperature() {
+ if (!ALL_WINE_STORAGES.contains(deviceState.getRawType())) {
+ return Optional.empty();
+ }
+
+ return getTopTemperatureFromList(effectiveTemperatures);
+ }
+
+ /**
+ * Gets the target top temperature of the wine storage.
+ *
+ * @return The target top temperature of the wine storage.
+ */
+ public Optional getTopTargetTemperature() {
+ if (!ALL_WINE_STORAGES.contains(deviceState.getRawType())) {
+ return Optional.empty();
+ }
+
+ return getTopTemperatureFromList(effectiveTargetTemperatures);
+ }
+
+ private Optional getTopTemperatureFromList(List temperatures) {
+ if (temperatures.size() <= 1) {
+ return Optional.empty();
+ }
+
+ return Optional.of(temperatures.get(0));
+ }
+
+ /**
+ * Gets the current middle temperature of the wine storage.
+ *
+ * @return The current middle temperature of the wine storage.
+ */
+ public Optional getMiddleTemperature() {
+ if (!ALL_WINE_STORAGES.contains(deviceState.getRawType())) {
+ return Optional.empty();
+ }
+
+ return getMiddleTemperatureFromList(effectiveTemperatures);
+ }
+
+ /**
+ * Gets the target middle temperature of the wine storage.
+ *
+ * @return The target middle temperature of the wine storage.
+ */
+ public Optional getMiddleTargetTemperature() {
+ if (!ALL_WINE_STORAGES.contains(deviceState.getRawType())) {
+ return Optional.empty();
+ }
+
+ return getMiddleTemperatureFromList(effectiveTargetTemperatures);
+ }
+
+ private Optional getMiddleTemperatureFromList(List temperatures) {
+ if (temperatures.size() != 3) {
+ return Optional.empty();
+ }
+
+ return Optional.of(temperatures.get(1));
+ }
+
+ /**
+ * Gets the current bottom temperature of the wine storage.
+ *
+ * @return The current bottom temperature of the wine storage.
+ */
+ public Optional getBottomTemperature() {
+ if (!ALL_WINE_STORAGES.contains(deviceState.getRawType())) {
+ return Optional.empty();
+ }
+
+ return getBottomTemperatureFromList(effectiveTemperatures);
+ }
+
+ /**
+ * Gets the target bottom temperature of the wine storage.
+ *
+ * @return The target bottom temperature of the wine storage.
+ */
+ public Optional getBottomTargetTemperature() {
+ if (!ALL_WINE_STORAGES.contains(deviceState.getRawType())) {
+ return Optional.empty();
+ }
+
+ return getBottomTemperatureFromList(effectiveTargetTemperatures);
+ }
+
+ private Optional getBottomTemperatureFromList(List temperatures) {
+ if (temperatures.size() == 3) {
+ return Optional.of(temperatures.get(2));
+ }
+
+ if (temperatures.size() == 2) {
+ return Optional.of(temperatures.get(1));
+ }
+
+ return Optional.empty();
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Actions.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Actions.java
new file mode 100644
index 00000000000..835b0fdde0a
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Actions.java
@@ -0,0 +1,137 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Collections;
+import java.util.List;
+import java.util.Objects;
+import java.util.Optional;
+import java.util.stream.Collectors;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * Immutable POJO representing the device actions queried from the Miele REST API.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ */
+@NonNullByDefault
+public class Actions {
+ @SerializedName("processAction")
+ @Nullable
+ private final List processAction = null;
+ @SerializedName("light")
+ @Nullable
+ private final List light = null;
+ @SerializedName("startTime")
+ @Nullable
+ private final List> startTime = null;
+ @SerializedName("programId")
+ @Nullable
+ private final List programId = null;
+ @SerializedName("deviceName")
+ @Nullable
+ private String deviceName;
+ @SerializedName("powerOff")
+ @Nullable
+ private Boolean powerOff;
+ @SerializedName("powerOn")
+ @Nullable
+ private Boolean powerOn;
+
+ public List getProcessAction() {
+ if (processAction == null) {
+ return Collections.emptyList();
+ }
+
+ return Collections.unmodifiableList(processAction);
+ }
+
+ public List getLight() {
+ final List lightRefCopy = light;
+ if (lightRefCopy == null) {
+ return Collections.emptyList();
+ }
+
+ return Collections.unmodifiableList(lightRefCopy.stream().map(Light::fromId).collect(Collectors.toList()));
+ }
+
+ /**
+ * Gets the start time encoded as {@link List} of {@link List} of {@link Integer} values.
+ * The first list entry defines the lower time constraint for setting the delayed start time. The second list
+ * entry defines the upper time constraint. The time constraints are defined as a list of integers with the full
+ * hour as first and minutes as second element.
+ *
+ * @return The possible start time interval encoded as described above.
+ */
+ public Optional>> getStartTime() {
+ if (startTime == null) {
+ return Optional.empty();
+ }
+
+ return Optional.of(Collections.unmodifiableList(startTime));
+ }
+
+ public List getProgramId() {
+ if (programId == null) {
+ return Collections.emptyList();
+ }
+
+ return Collections.unmodifiableList(programId);
+ }
+
+ public Optional getDeviceName() {
+ return Optional.ofNullable(deviceName);
+ }
+
+ public Optional getPowerOn() {
+ return Optional.ofNullable(powerOn);
+ }
+
+ public Optional getPowerOff() {
+ return Optional.ofNullable(powerOff);
+ }
+
+ @Override
+ public String toString() {
+ return "ActionState [processAction=" + processAction + ", light=" + light + ", startTime=" + startTime
+ + ", programId=" + programId + ", deviceName=" + deviceName + ", powerOff=" + powerOff + ", powerOn="
+ + powerOn + "]";
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(deviceName, light, powerOn, powerOff, processAction, startTime, programId);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ Actions other = (Actions) obj;
+ return Objects.equals(deviceName, other.deviceName) && Objects.equals(light, other.light)
+ && Objects.equals(powerOn, other.powerOn) && Objects.equals(powerOff, other.powerOff)
+ && Objects.equals(processAction, other.processAction) && Objects.equals(startTime, other.startTime)
+ && Objects.equals(programId, other.programId);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Device.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Device.java
new file mode 100644
index 00000000000..4fcf8013203
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Device.java
@@ -0,0 +1,65 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * Immutable POJO representing a device queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class Device {
+ @Nullable
+ private Ident ident;
+ @Nullable
+ private State state;
+
+ public Optional getIdent() {
+ return Optional.ofNullable(ident);
+ }
+
+ public Optional getState() {
+ return Optional.ofNullable(state);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(ident, state);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ Device other = (Device) obj;
+ return Objects.equals(ident, other.ident) && Objects.equals(state, other.state);
+ }
+
+ @Override
+ public String toString() {
+ return "Device [ident=" + ident + ", state=" + state + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/DeviceCollection.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/DeviceCollection.java
new file mode 100644
index 00000000000..da56e6d6bc8
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/DeviceCollection.java
@@ -0,0 +1,97 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Map;
+import java.util.Objects;
+import java.util.Set;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.Gson;
+import com.google.gson.JsonSyntaxException;
+import com.google.gson.reflect.TypeToken;
+
+/**
+ * Immutable POJO representing a collection of devices queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class DeviceCollection {
+ private static final java.lang.reflect.Type STRING_DEVICE_MAP_TYPE = new TypeToken>() {
+ }.getType();
+
+ private final Map devices;
+
+ DeviceCollection(Map devices) {
+ this.devices = devices;
+ }
+
+ /**
+ * Creates a new {@link DeviceCollection} from the given Json text.
+ *
+ * @param json The Json text.
+ * @return The created {@link DeviceCollection}.
+ * @throws MieleSyntaxException if parsing the data from {@code json} fails.
+ */
+ public static DeviceCollection fromJson(String json) {
+ try {
+ Map devices = new Gson().fromJson(json, STRING_DEVICE_MAP_TYPE);
+ if (devices == null) {
+ throw new MieleSyntaxException("Failed to parse Json.");
+ }
+ return new DeviceCollection(devices);
+ } catch (JsonSyntaxException e) {
+ throw new MieleSyntaxException("Failed to parse Json.", e);
+ }
+ }
+
+ public Set getDeviceIdentifiers() {
+ return devices.keySet();
+ }
+
+ public Device getDevice(String identifier) {
+ Device device = devices.get(identifier);
+ if (device == null) {
+ throw new IllegalArgumentException("There is no device for identifier " + identifier);
+ }
+ return device;
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(devices);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ DeviceCollection other = (DeviceCollection) obj;
+ return Objects.equals(devices, other.devices);
+ }
+
+ @Override
+ public String toString() {
+ return "DeviceCollection [devices=" + devices + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/DeviceIdentLabel.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/DeviceIdentLabel.java
new file mode 100644
index 00000000000..26857fe11e1
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/DeviceIdentLabel.java
@@ -0,0 +1,92 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Collections;
+import java.util.List;
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * Immutable POJO representing the full device identification queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class DeviceIdentLabel {
+ @Nullable
+ private String fabNumber;
+ @Nullable
+ private String fabIndex;
+ @Nullable
+ private String techType;
+ @Nullable
+ private String matNumber;
+ @Nullable
+ private final List swids = null;
+
+ public Optional getFabNumber() {
+ return Optional.ofNullable(fabNumber);
+ }
+
+ public Optional getFabIndex() {
+ return Optional.ofNullable(fabIndex);
+ }
+
+ public Optional getTechType() {
+ return Optional.ofNullable(techType);
+ }
+
+ public Optional getMatNumber() {
+ return Optional.ofNullable(matNumber);
+ }
+
+ public List getSwids() {
+ if (swids == null) {
+ return Collections.emptyList();
+ }
+
+ return Collections.unmodifiableList(swids);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(fabIndex, fabNumber, matNumber, swids, techType);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ DeviceIdentLabel other = (DeviceIdentLabel) obj;
+ return Objects.equals(fabIndex, other.fabIndex) && Objects.equals(fabNumber, other.fabNumber)
+ && Objects.equals(matNumber, other.matNumber) && Objects.equals(swids, other.swids)
+ && Objects.equals(techType, other.techType);
+ }
+
+ @Override
+ public String toString() {
+ return "DeviceIdentLabel [fabNumber=" + fabNumber + ", fabIndex=" + fabIndex + ", techType=" + techType
+ + ", matNumber=" + matNumber + ", swids=" + swids + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/DeviceType.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/DeviceType.java
new file mode 100644
index 00000000000..d25f9ad9ec8
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/DeviceType.java
@@ -0,0 +1,129 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * Represents the Miele device type.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public enum DeviceType {
+ /**
+ * {@link DeviceType} for unknown devices.
+ */
+ UNKNOWN,
+
+ @SerializedName("1")
+ WASHING_MACHINE,
+
+ @SerializedName("2")
+ TUMBLE_DRYER,
+
+ @SerializedName("7")
+ DISHWASHER,
+
+ @SerializedName("8")
+ DISHWASHER_SEMI_PROF,
+
+ @SerializedName("12")
+ OVEN,
+
+ @SerializedName("13")
+ OVEN_MICROWAVE,
+
+ @SerializedName("14")
+ HOB_HIGHLIGHT,
+
+ @SerializedName("15")
+ STEAM_OVEN,
+
+ @SerializedName("16")
+ MICROWAVE,
+
+ @SerializedName("17")
+ COFFEE_SYSTEM,
+
+ @SerializedName("18")
+ HOOD,
+
+ @SerializedName("19")
+ FRIDGE,
+
+ @SerializedName("20")
+ FREEZER,
+
+ @SerializedName("21")
+ FRIDGE_FREEZER_COMBINATION,
+
+ /**
+ * Might also be AUTOMATIC ROBOTIC VACUUM CLEANER.
+ */
+ @SerializedName("23")
+ VACUUM_CLEANER,
+
+ @SerializedName("24")
+ WASHER_DRYER,
+
+ @SerializedName("25")
+ DISH_WARMER,
+
+ @SerializedName("27")
+ HOB_INDUCTION,
+
+ @SerializedName("28")
+ HOB_GAS,
+
+ @SerializedName("31")
+ STEAM_OVEN_COMBINATION,
+
+ @SerializedName("32")
+ WINE_CABINET,
+
+ @SerializedName("33")
+ WINE_CONDITIONING_UNIT,
+
+ @SerializedName("34")
+ WINE_STORAGE_CONDITIONING_UNIT,
+
+ @SerializedName("39")
+ DOUBLE_OVEN,
+
+ @SerializedName("40")
+ DOUBLE_STEAM_OVEN,
+
+ @SerializedName("41")
+ DOUBLE_STEAM_OVEN_COMBINATION,
+
+ @SerializedName("42")
+ DOUBLE_MICROWAVE,
+
+ @SerializedName("43")
+ DOUBLE_MICROWAVE_OVEN,
+
+ @SerializedName("45")
+ STEAM_OVEN_MICROWAVE_COMBINATION,
+
+ @SerializedName("48")
+ VACUUM_DRAWER,
+
+ @SerializedName("67")
+ DIALOGOVEN,
+
+ @SerializedName("68")
+ WINE_CABINET_FREEZER_COMBINATION,
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/DryingStep.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/DryingStep.java
new file mode 100644
index 00000000000..e7c55bb7503
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/DryingStep.java
@@ -0,0 +1,78 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * Immutable POJO representing the current drying step, queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class DryingStep {
+ @SerializedName("value_raw")
+ @Nullable
+ private Integer valueRaw;
+ @SerializedName("value_localized")
+ @Nullable
+ private String valueLocalized;
+ @SerializedName("key_localized")
+ @Nullable
+ private String keyLocalized;
+
+ public Optional getValueRaw() {
+ return Optional.ofNullable(valueRaw);
+ }
+
+ public Optional getValueLocalized() {
+ return Optional.ofNullable(valueLocalized);
+ }
+
+ public Optional getKeyLocalized() {
+ return Optional.ofNullable(keyLocalized);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(keyLocalized, valueLocalized, valueRaw);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ DryingStep other = (DryingStep) obj;
+ return Objects.equals(keyLocalized, other.keyLocalized) && Objects.equals(valueLocalized, other.valueLocalized)
+ && Objects.equals(valueRaw, other.valueRaw);
+ }
+
+ @Override
+ public String toString() {
+ return "DryingStep [valueRaw=" + valueRaw + ", valueLocalized=" + valueLocalized + ", keyLocalized="
+ + keyLocalized + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/ErrorMessage.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/ErrorMessage.java
new file mode 100644
index 00000000000..c72bb80ed6b
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/ErrorMessage.java
@@ -0,0 +1,76 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.Gson;
+import com.google.gson.JsonSyntaxException;
+
+/**
+ * Immutable POJO representing an error message. Queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class ErrorMessage {
+ @Nullable
+ private String message;
+
+ /**
+ * Creates a new {@link ErrorMessage} from the given Json text.
+ *
+ * @param json The Json text.
+ * @return The created {@link ErrorMessage}.
+ * @throws MieleSyntaxException if parsing the data from {@code json} fails.
+ */
+ public static ErrorMessage fromJson(String json) {
+ try {
+ ErrorMessage errorMessage = new Gson().fromJson(json, ErrorMessage.class);
+ if (errorMessage == null) {
+ throw new MieleSyntaxException("Failed to parse Json.");
+ }
+ return errorMessage;
+ } catch (JsonSyntaxException e) {
+ throw new MieleSyntaxException("Failed to parse Json.", e);
+ }
+ }
+
+ public Optional getMessage() {
+ return Optional.ofNullable(message);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(message);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ ErrorMessage other = (ErrorMessage) obj;
+ return Objects.equals(message, other.message);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Ident.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Ident.java
new file mode 100644
index 00000000000..dbbc7637686
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Ident.java
@@ -0,0 +1,79 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * Immutable POJO representing the device identification queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class Ident {
+ @Nullable
+ private Type type;
+ @Nullable
+ private String deviceName;
+ @Nullable
+ private DeviceIdentLabel deviceIdentLabel;
+ @Nullable
+ private XkmIdentLabel xkmIdentLabel;
+
+ public Optional getType() {
+ return Optional.ofNullable(type);
+ }
+
+ public Optional getDeviceName() {
+ return Optional.ofNullable(deviceName);
+ }
+
+ public Optional getDeviceIdentLabel() {
+ return Optional.ofNullable(deviceIdentLabel);
+ }
+
+ public Optional getXkmIdentLabel() {
+ return Optional.ofNullable(xkmIdentLabel);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(deviceIdentLabel, deviceName, type, xkmIdentLabel);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ Ident other = (Ident) obj;
+ return Objects.equals(deviceIdentLabel, other.deviceIdentLabel) && Objects.equals(deviceName, other.deviceName)
+ && Objects.equals(type, other.type) && Objects.equals(xkmIdentLabel, other.xkmIdentLabel);
+ }
+
+ @Override
+ public String toString() {
+ return "Ident [type=" + type + ", deviceName=" + deviceName + ", deviceIdentLabel=" + deviceIdentLabel
+ + ", xkmIdentLabel=" + xkmIdentLabel + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Light.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Light.java
new file mode 100644
index 00000000000..db911714e80
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Light.java
@@ -0,0 +1,74 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Arrays;
+import java.util.Collections;
+import java.util.List;
+import java.util.stream.Collectors;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * Represents the state of a light on a Miele device.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ * @author Björn Lange - Added NOT_SUPPORTED entry
+ */
+@NonNullByDefault
+public enum Light {
+ /**
+ * {Light} for unknown states.
+ */
+ UNKNOWN(),
+
+ ENABLE(1),
+
+ DISABLE(2),
+
+ NOT_SUPPORTED(0, 255);
+
+ private List ids;
+
+ Light(int... ids) {
+ this.ids = Collections.unmodifiableList(Arrays.stream(ids).boxed().collect(Collectors.toList()));
+ }
+
+ /**
+ * Gets the {@link Light} state matching the given ID.
+ *
+ * @param id The ID.
+ * @return The matching {@link Light} or {@code UNKNOWN} if no ID matches.
+ */
+ public static Light fromId(@Nullable Integer id) {
+ for (Light light : Light.values()) {
+ if (light.ids.contains(id)) {
+ return light;
+ }
+ }
+
+ return Light.UNKNOWN;
+ }
+
+ /**
+ * Formats this instance for interaction with the Miele webservice.
+ */
+ public String format() {
+ if (ids.isEmpty()) {
+ return "";
+ } else {
+ return Integer.toString(ids.get(0));
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/MieleSyntaxException.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/MieleSyntaxException.java
new file mode 100644
index 00000000000..b6f563e5c05
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/MieleSyntaxException.java
@@ -0,0 +1,34 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * {@link RuntimeException} thrown when the syntax of a message received from the Miele REST API does not match and
+ * cannot be interpreted as the expected syntax (e.g. by ignoring entries).
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class MieleSyntaxException extends RuntimeException {
+ private static final long serialVersionUID = 8253804935427566729L;
+
+ public MieleSyntaxException(String message) {
+ super(message);
+ }
+
+ public MieleSyntaxException(String message, Throwable cause) {
+ super(message, cause);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/PlateStep.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/PlateStep.java
new file mode 100644
index 00000000000..03926618d48
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/PlateStep.java
@@ -0,0 +1,78 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * Immutable POJO representing a plate power state. Queried from the Miele REST API.
+ *
+ * @author Benjamin Bolte - Initial contribution
+ */
+@NonNullByDefault
+public class PlateStep {
+ @SerializedName("value_raw")
+ @Nullable
+ private Integer valueRaw;
+ @SerializedName("value_localized")
+ @Nullable
+ private String valueLocalized;
+ @SerializedName("key_localized")
+ @Nullable
+ private String keyLocalized;
+
+ public Optional getValueRaw() {
+ return Optional.ofNullable(valueRaw);
+ }
+
+ public Optional getValueLocalized() {
+ return Optional.ofNullable(valueLocalized);
+ }
+
+ public Optional getKeyLocalized() {
+ return Optional.ofNullable(keyLocalized);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(keyLocalized, valueLocalized, valueRaw);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ PlateStep other = (PlateStep) obj;
+ return Objects.equals(keyLocalized, other.keyLocalized) && Objects.equals(valueLocalized, other.valueLocalized)
+ && Objects.equals(valueRaw, other.valueRaw);
+ }
+
+ @Override
+ public String toString() {
+ return "PlateStep [valueRaw=" + valueRaw + ", valueLocalized=" + valueLocalized + ", key_localized="
+ + keyLocalized + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/ProcessAction.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/ProcessAction.java
new file mode 100644
index 00000000000..80c08bf0dda
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/ProcessAction.java
@@ -0,0 +1,51 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * Represents a process action.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ */
+@NonNullByDefault
+public enum ProcessAction {
+ /**
+ * {@StateType} for unknown states.
+ */
+ UNKNOWN,
+
+ @SerializedName("1")
+ START,
+
+ @SerializedName("2")
+ STOP,
+
+ @SerializedName("3")
+ PAUSE,
+
+ @SerializedName("4")
+ START_SUPERFREEZING,
+
+ @SerializedName("5")
+ STOP_SUPERFREEZING,
+
+ @SerializedName("6")
+ START_SUPERCOOLING,
+
+ @SerializedName("7")
+ STOP_SUPERCOOLING,
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/ProgramId.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/ProgramId.java
new file mode 100644
index 00000000000..b9d665da091
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/ProgramId.java
@@ -0,0 +1,78 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * Immutable POJO representing the program type that is currently running. Queried from the Miele REST API.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ */
+@NonNullByDefault
+public class ProgramId {
+ @SerializedName("value_raw")
+ @Nullable
+ private Long valueRaw;
+ @SerializedName("value_localized")
+ @Nullable
+ private String valueLocalized;
+ @SerializedName("key_localized")
+ @Nullable
+ private String keyLocalized;
+
+ public Optional getValueRaw() {
+ return Optional.ofNullable(valueRaw);
+ }
+
+ public Optional getValueLocalized() {
+ return Optional.ofNullable(valueLocalized);
+ }
+
+ public Optional getKeyLocalized() {
+ return Optional.ofNullable(keyLocalized);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(keyLocalized, valueLocalized, valueRaw);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ ProgramId other = (ProgramId) obj;
+ return Objects.equals(keyLocalized, other.keyLocalized) && Objects.equals(valueLocalized, other.valueLocalized)
+ && Objects.equals(valueRaw, other.valueRaw);
+ }
+
+ @Override
+ public String toString() {
+ return "ProgramType [valueRaw=" + valueRaw + ", valueLocalized=" + valueLocalized + ", keyLocalized="
+ + keyLocalized + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/ProgramPhase.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/ProgramPhase.java
new file mode 100644
index 00000000000..1b3d9316385
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/ProgramPhase.java
@@ -0,0 +1,78 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * Immutable POJO representing the current program's phase. Queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class ProgramPhase {
+ @SerializedName("value_raw")
+ @Nullable
+ private Integer valueRaw;
+ @SerializedName("value_localized")
+ @Nullable
+ private String valueLocalized;
+ @SerializedName("key_localized")
+ @Nullable
+ private String keyLocalized;
+
+ public Optional getValueRaw() {
+ return Optional.ofNullable(valueRaw);
+ }
+
+ public Optional getValueLocalized() {
+ return Optional.ofNullable(valueLocalized);
+ }
+
+ public Optional getKeyLocalized() {
+ return Optional.ofNullable(keyLocalized);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(keyLocalized, valueLocalized, valueRaw);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ ProgramPhase other = (ProgramPhase) obj;
+ return Objects.equals(keyLocalized, other.keyLocalized) && Objects.equals(valueLocalized, other.valueLocalized)
+ && Objects.equals(valueRaw, other.valueRaw);
+ }
+
+ @Override
+ public String toString() {
+ return "ProgramPhase [valueRaw=" + valueRaw + ", valueLocalized=" + valueLocalized + ", keyLocalized="
+ + keyLocalized + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/ProgramType.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/ProgramType.java
new file mode 100644
index 00000000000..2dac3fd2b19
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/ProgramType.java
@@ -0,0 +1,78 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * Immutable POJO representing the type of program currently running. Queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class ProgramType {
+ @SerializedName("value_raw")
+ @Nullable
+ private Integer valueRaw;
+ @SerializedName("value_localized")
+ @Nullable
+ private String valueLocalized;
+ @SerializedName("key_localized")
+ @Nullable
+ private String keyLocalized;
+
+ public Optional getValueRaw() {
+ return Optional.ofNullable(valueRaw);
+ }
+
+ public Optional getValueLocalized() {
+ return Optional.ofNullable(valueLocalized);
+ }
+
+ public Optional getKeyLocalized() {
+ return Optional.ofNullable(keyLocalized);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(keyLocalized, valueLocalized, valueRaw);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ ProgramType other = (ProgramType) obj;
+ return Objects.equals(keyLocalized, other.keyLocalized) && Objects.equals(valueLocalized, other.valueLocalized)
+ && Objects.equals(valueRaw, other.valueRaw);
+ }
+
+ @Override
+ public String toString() {
+ return "ProgramType [valueRaw=" + valueRaw + ", valueLocalized=" + valueLocalized + ", keyLocalized="
+ + keyLocalized + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/RemoteEnable.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/RemoteEnable.java
new file mode 100644
index 00000000000..ffa34fa14d7
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/RemoteEnable.java
@@ -0,0 +1,65 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * Immutable POJO representing the remote control capabilities of a device. Queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class RemoteEnable {
+ @Nullable
+ private Boolean fullRemoteControl;
+ @Nullable
+ private Boolean smartGrid;
+
+ public Optional getFullRemoteControl() {
+ return Optional.ofNullable(fullRemoteControl);
+ }
+
+ public Optional getSmartGrid() {
+ return Optional.ofNullable(smartGrid);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(fullRemoteControl, smartGrid);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ RemoteEnable other = (RemoteEnable) obj;
+ return Objects.equals(fullRemoteControl, other.fullRemoteControl) && Objects.equals(smartGrid, other.smartGrid);
+ }
+
+ @Override
+ public String toString() {
+ return "RemoteEnable [fullRemoteControl=" + fullRemoteControl + ", smartGrid=" + smartGrid + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/SpinningSpeed.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/SpinningSpeed.java
new file mode 100644
index 00000000000..bdaca12ffe9
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/SpinningSpeed.java
@@ -0,0 +1,77 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * Immutable POJO representing the current spinning speed, queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class SpinningSpeed {
+ @SerializedName("value_raw")
+ @Nullable
+ private Integer valueRaw;
+ @SerializedName("value_localized")
+ @Nullable
+ private String valueLocalized;
+ @SerializedName("unit")
+ @Nullable
+ private String unit;
+
+ public Optional getValueRaw() {
+ return Optional.ofNullable(valueRaw);
+ }
+
+ public Optional getValueLocalized() {
+ return Optional.ofNullable(valueLocalized);
+ }
+
+ public Optional getUnit() {
+ return Optional.ofNullable(unit);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(unit, valueLocalized, valueRaw);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ SpinningSpeed other = (SpinningSpeed) obj;
+ return Objects.equals(unit, other.unit) && Objects.equals(valueLocalized, other.valueLocalized)
+ && Objects.equals(valueRaw, other.valueRaw);
+ }
+
+ @Override
+ public String toString() {
+ return "SpinningSpeed [valueRaw=" + valueRaw + ", valueLocalized=" + valueLocalized + ", unit=" + unit + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/State.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/State.java
new file mode 100644
index 00000000000..c9f6d111b50
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/State.java
@@ -0,0 +1,238 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Collections;
+import java.util.List;
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * Immutable POJO representing the state of a device. Queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ * @author Benjamin Bolte - Add plate step
+ * @author Björn Lange - Add elapsed time channel
+ */
+@NonNullByDefault
+public class State {
+ @Nullable
+ private Status status;
+ /**
+ * Currently used by Miele webservice.
+ */
+ @Nullable
+ private ProgramId ProgramID;
+ /**
+ * Planned to be used in the future.
+ */
+ @Nullable
+ private ProgramId programId;
+ @Nullable
+ private ProgramType programType;
+ @Nullable
+ private ProgramPhase programPhase;
+ @Nullable
+ private final List remainingTime = null;
+ @Nullable
+ private final List startTime = null;
+ @Nullable
+ private final List targetTemperature = null;
+ @Nullable
+ private final List temperature = null;
+ @Nullable
+ private Boolean signalInfo;
+ @Nullable
+ private Boolean signalFailure;
+ @Nullable
+ private Boolean signalDoor;
+ @Nullable
+ private RemoteEnable remoteEnable;
+ @Nullable
+ private Integer light;
+ @Nullable
+ private final List elapsedTime = null;
+ @Nullable
+ private SpinningSpeed spinningSpeed;
+ @Nullable
+ private DryingStep dryingStep;
+ @Nullable
+ private VentilationStep ventilationStep;
+ @Nullable
+ private final List plateStep = null;
+ @Nullable
+ private Integer batteryLevel;
+
+ public Optional getStatus() {
+ return Optional.ofNullable(status);
+ }
+
+ public Optional getProgramId() {
+ // There is a typo for the program ID in the Miele Cloud API, which will be corrected in the future.
+ // For the sake of robustness, we currently support both upper and lower case.
+ return Optional.ofNullable(programId != null ? programId : ProgramID);
+ }
+
+ public Optional getProgramType() {
+ return Optional.ofNullable(programType);
+ }
+
+ public Optional getProgramPhase() {
+ return Optional.ofNullable(programPhase);
+ }
+
+ /**
+ * Gets the remaining time encoded as {@link List} of {@link Integer} values.
+ *
+ * @return The remaining time encoded as {@link List} of {@link Integer} values.
+ */
+ public Optional> getRemainingTime() {
+ if (remainingTime == null) {
+ return Optional.empty();
+ }
+
+ return Optional.ofNullable(Collections.unmodifiableList(remainingTime));
+ }
+
+ /**
+ * Gets the start time encoded as {@link List} of {@link Integer} values.
+ *
+ * @return The start time encoded as {@link List} of {@link Integer} values.
+ */
+ public Optional> getStartTime() {
+ if (startTime == null) {
+ return Optional.empty();
+ }
+
+ return Optional.ofNullable(Collections.unmodifiableList(startTime));
+ }
+
+ public List getTargetTemperature() {
+ if (targetTemperature == null) {
+ return Collections.emptyList();
+ }
+
+ return Collections.unmodifiableList(targetTemperature);
+ }
+
+ public List getTemperature() {
+ if (temperature == null) {
+ return Collections.emptyList();
+ }
+
+ return Collections.unmodifiableList(temperature);
+ }
+
+ public Optional getSignalInfo() {
+ return Optional.ofNullable(signalInfo);
+ }
+
+ public Optional getSignalFailure() {
+ return Optional.ofNullable(signalFailure);
+ }
+
+ public Optional getSignalDoor() {
+ return Optional.ofNullable(signalDoor);
+ }
+
+ public Optional getRemoteEnable() {
+ return Optional.ofNullable(remoteEnable);
+ }
+
+ public Light getLight() {
+ return Light.fromId(light);
+ }
+
+ /**
+ * Gets the elapsed time encoded as {@link List} of {@link Integer} values.
+ *
+ * @return The elapsed time encoded as {@link List} of {@link Integer} values.
+ */
+ public Optional> getElapsedTime() {
+ if (elapsedTime == null) {
+ return Optional.empty();
+ }
+
+ return Optional.ofNullable(Collections.unmodifiableList(elapsedTime));
+ }
+
+ public Optional getSpinningSpeed() {
+ return Optional.ofNullable(spinningSpeed);
+ }
+
+ public Optional getDryingStep() {
+ return Optional.ofNullable(dryingStep);
+ }
+
+ public Optional getVentilationStep() {
+ return Optional.ofNullable(ventilationStep);
+ }
+
+ public List getPlateStep() {
+ if (plateStep == null) {
+ return Collections.emptyList();
+ }
+
+ return Collections.unmodifiableList(plateStep);
+ }
+
+ public Optional getBatteryLevel() {
+ return Optional.ofNullable(batteryLevel);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(dryingStep, elapsedTime, light, programPhase, ProgramID, programId, programType,
+ remainingTime, remoteEnable, signalDoor, signalFailure, signalInfo, startTime, status,
+ targetTemperature, temperature, ventilationStep, plateStep, batteryLevel);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ State other = (State) obj;
+ return Objects.equals(dryingStep, other.dryingStep) && Objects.equals(elapsedTime, other.elapsedTime)
+ && Objects.equals(light, other.light) && Objects.equals(programPhase, other.programPhase)
+ && Objects.equals(ProgramID, other.ProgramID) && Objects.equals(programId, other.programId)
+ && Objects.equals(programType, other.programType) && Objects.equals(remainingTime, other.remainingTime)
+ && Objects.equals(remoteEnable, other.remoteEnable) && Objects.equals(signalDoor, other.signalDoor)
+ && Objects.equals(signalFailure, other.signalFailure) && Objects.equals(signalInfo, other.signalInfo)
+ && Objects.equals(startTime, other.startTime) && Objects.equals(status, other.status)
+ && Objects.equals(targetTemperature, other.targetTemperature)
+ && Objects.equals(temperature, other.temperature)
+ && Objects.equals(ventilationStep, other.ventilationStep) && Objects.equals(plateStep, other.plateStep)
+ && Objects.equals(batteryLevel, other.batteryLevel);
+ }
+
+ @Override
+ public String toString() {
+ return "State [status=" + status + ", programId=" + getProgramId() + ", programType=" + programType
+ + ", programPhase=" + programPhase + ", remainingTime=" + remainingTime + ", startTime=" + startTime
+ + ", targetTemperature=" + targetTemperature + ", temperature=" + temperature + ", signalInfo="
+ + signalInfo + ", signalFailure=" + signalFailure + ", signalDoor=" + signalDoor + ", remoteEnable="
+ + remoteEnable + ", light=" + light + ", elapsedTime=" + elapsedTime + ", dryingStep=" + dryingStep
+ + ", ventilationStep=" + ventilationStep + ", plateStep=" + plateStep + ", batteryLevel=" + batteryLevel
+ + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/StateType.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/StateType.java
new file mode 100644
index 00000000000..a3058e01ff1
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/StateType.java
@@ -0,0 +1,70 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Collections;
+import java.util.HashMap;
+import java.util.Map;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Represents the Miele device state.
+ *
+ * @author Roland Edelhoff - Initial contribution
+ */
+@NonNullByDefault
+public enum StateType {
+ OFF(1),
+ ON(2),
+ PROGRAMMED(3),
+ PROGRAMMED_WAITING_TO_START(4),
+ RUNNING(5),
+ PAUSE(6),
+ END_PROGRAMMED(7),
+ FAILURE(8),
+ PROGRAMME_INTERRUPTED(9),
+ IDLE(10),
+ RINSE_HOLD(11),
+ SERVICE(12),
+ SUPERFREEZING(13),
+ SUPERCOOLING(14),
+ SUPERHEATING(15),
+ SUPERCOOLING_SUPERFREEZING(146),
+ NOT_CONNECTED(255);
+
+ private static final Map STATE_TYPE_BY_CODE;
+
+ static {
+ Map stateTypeByCode = new HashMap<>();
+ for (StateType stateType : values()) {
+ stateTypeByCode.put(stateType.code, stateType);
+ }
+ STATE_TYPE_BY_CODE = Collections.unmodifiableMap(stateTypeByCode);
+ }
+
+ private final int code;
+
+ private StateType(int code) {
+ this.code = code;
+ }
+
+ public int getCode() {
+ return code;
+ }
+
+ public static Optional fromCode(int code) {
+ return Optional.ofNullable(STATE_TYPE_BY_CODE.get(code));
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Status.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Status.java
new file mode 100644
index 00000000000..f5c5c6e82e1
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Status.java
@@ -0,0 +1,78 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * Immutable POJO representing the actual status of a device. Queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class Status {
+ @SerializedName("value_raw")
+ @Nullable
+ private Integer valueRaw;
+ @SerializedName("value_localized")
+ @Nullable
+ private String valueLocalized;
+ @SerializedName("key_localized")
+ @Nullable
+ private String keyLocalized;
+
+ public Optional getValueRaw() {
+ return Optional.ofNullable(valueRaw);
+ }
+
+ public Optional getValueLocalized() {
+ return Optional.ofNullable(valueLocalized);
+ }
+
+ public Optional getKeyLocalized() {
+ return Optional.ofNullable(keyLocalized);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(keyLocalized, valueLocalized, valueRaw);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ Status other = (Status) obj;
+ return Objects.equals(keyLocalized, other.keyLocalized) && Objects.equals(valueLocalized, other.valueLocalized)
+ && Objects.equals(valueRaw, other.valueRaw);
+ }
+
+ @Override
+ public String toString() {
+ return "Status [valueRaw=" + valueRaw + ", valueLocalized=" + valueLocalized + ", keyLocalized=" + keyLocalized
+ + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Temperature.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Temperature.java
new file mode 100644
index 00000000000..8ddae8fe695
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Temperature.java
@@ -0,0 +1,77 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * Immutable POJO representing a temperature value. Queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class Temperature {
+ @SerializedName("value_raw")
+ @Nullable
+ private Integer valueRaw;
+ @SerializedName("value_localized")
+ @Nullable
+ private Double valueLocalized;
+ @SerializedName("unit")
+ @Nullable
+ private String unit;
+
+ public Optional getValueRaw() {
+ return Optional.ofNullable(valueRaw);
+ }
+
+ public Optional getValueLocalized() {
+ return Optional.ofNullable(valueLocalized).map(Double::intValue);
+ }
+
+ public Optional getUnit() {
+ return Optional.ofNullable(unit);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(unit, valueLocalized, valueRaw);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ Temperature other = (Temperature) obj;
+ return Objects.equals(unit, other.unit) && Objects.equals(valueLocalized, other.valueLocalized)
+ && Objects.equals(valueRaw, other.valueRaw);
+ }
+
+ @Override
+ public String toString() {
+ return "Temperature [valueRaw=" + valueRaw + ", valueLocalized=" + valueLocalized + ", unit=" + unit + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Type.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Type.java
new file mode 100644
index 00000000000..0ed689173aa
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/Type.java
@@ -0,0 +1,78 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * Immutable POJO representing the type of a device. Queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class Type {
+ @SerializedName("key_localized")
+ @Nullable
+ private String keyLocalized;
+ @SerializedName("value_raw")
+ @Nullable
+ private DeviceType valueRaw;
+ @SerializedName("value_localized")
+ @Nullable
+ private String valueLocalized;
+
+ public Optional getKeyLocalized() {
+ return Optional.ofNullable(keyLocalized);
+ }
+
+ public DeviceType getValueRaw() {
+ return Optional.ofNullable(valueRaw).orElse(DeviceType.UNKNOWN);
+ }
+
+ public Optional getValueLocalized() {
+ return Optional.ofNullable(valueLocalized);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(keyLocalized, valueLocalized, valueRaw);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ Type other = (Type) obj;
+ return Objects.equals(keyLocalized, other.keyLocalized) && Objects.equals(valueLocalized, other.valueLocalized)
+ && valueRaw == other.valueRaw;
+ }
+
+ @Override
+ public String toString() {
+ return "Type [keyLocalized=" + keyLocalized + ", valueRaw=" + valueRaw + ", valueLocalized=" + valueLocalized
+ + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/VentilationStep.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/VentilationStep.java
new file mode 100644
index 00000000000..ea7a4b9ffda
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/VentilationStep.java
@@ -0,0 +1,78 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+import com.google.gson.annotations.SerializedName;
+
+/**
+ * Immutable POJO representing the current ventilation step. Queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class VentilationStep {
+ @SerializedName("value_raw")
+ @Nullable
+ private Integer valueRaw;
+ @SerializedName("value_localized")
+ @Nullable
+ private String valueLocalized;
+ @SerializedName("key_localized")
+ @Nullable
+ private String keyLocalized;
+
+ public Optional getValueRaw() {
+ return Optional.ofNullable(valueRaw);
+ }
+
+ public Optional getValueLocalized() {
+ return Optional.ofNullable(valueLocalized);
+ }
+
+ public Optional getKeyLocalized() {
+ return Optional.ofNullable(keyLocalized);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(keyLocalized, valueLocalized, valueRaw);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ VentilationStep other = (VentilationStep) obj;
+ return Objects.equals(keyLocalized, other.keyLocalized) && Objects.equals(valueLocalized, other.valueLocalized)
+ && Objects.equals(valueRaw, other.valueRaw);
+ }
+
+ @Override
+ public String toString() {
+ return "VentilationStep [valueRaw=" + valueRaw + ", valueLocalized=" + valueLocalized + ", keyLocalized="
+ + keyLocalized + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/XkmIdentLabel.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/XkmIdentLabel.java
new file mode 100644
index 00000000000..5b126eacd60
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/api/json/XkmIdentLabel.java
@@ -0,0 +1,65 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.api.json;
+
+import java.util.Objects;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * Immutable POJO representing the XKM (Miele communication module) identification. Queried from the Miele REST API.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class XkmIdentLabel {
+ @Nullable
+ private String techType;
+ @Nullable
+ private String releaseVersion;
+
+ public Optional getTechType() {
+ return Optional.ofNullable(techType);
+ }
+
+ public Optional getReleaseVersion() {
+ return Optional.ofNullable(releaseVersion);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(releaseVersion, techType);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ XkmIdentLabel other = (XkmIdentLabel) obj;
+ return Objects.equals(releaseVersion, other.releaseVersion) && Objects.equals(techType, other.techType);
+ }
+
+ @Override
+ public String toString() {
+ return "XkmIdentLabel [techType=" + techType + ", releaseVersion=" + releaseVersion + "]";
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/AuthorizationFailedException.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/AuthorizationFailedException.java
new file mode 100644
index 00000000000..f958ec17afc
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/AuthorizationFailedException.java
@@ -0,0 +1,29 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.exception;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * This {@link RuntimeException} is thrown if an error occurred due to authorization failure.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class AuthorizationFailedException extends RuntimeException {
+ private static final long serialVersionUID = 963609531804668970L;
+
+ public AuthorizationFailedException(final String message) {
+ super(message);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/MieleWebserviceDisconnectSseException.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/MieleWebserviceDisconnectSseException.java
new file mode 100644
index 00000000000..5d42428a7b4
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/MieleWebserviceDisconnectSseException.java
@@ -0,0 +1,25 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.exception;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Used as a notification to close SSE connections.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public class MieleWebserviceDisconnectSseException extends RuntimeException {
+ private static final long serialVersionUID = 607435177026345387L;
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/MieleWebserviceException.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/MieleWebserviceException.java
new file mode 100644
index 00000000000..4523f6faf65
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/MieleWebserviceException.java
@@ -0,0 +1,45 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.exception;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.mielecloud.internal.webservice.ConnectionError;
+
+/**
+ * {@link RuntimeException} thrown if the Miele service is not available or unable to handle requests.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class MieleWebserviceException extends RuntimeException {
+
+ private static final long serialVersionUID = 6268725866086530042L;
+
+ private final ConnectionError connectionError;
+
+ public MieleWebserviceException(final String message, final ConnectionError connectionError) {
+ super(message);
+ this.connectionError = connectionError;
+ }
+
+ public MieleWebserviceException(final String message, @Nullable final Throwable cause,
+ final ConnectionError connectionError) {
+ super(message, cause);
+ this.connectionError = connectionError;
+ }
+
+ public ConnectionError getConnectionError() {
+ return connectionError;
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/MieleWebserviceInitializationException.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/MieleWebserviceInitializationException.java
new file mode 100644
index 00000000000..c9de5eaebc0
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/MieleWebserviceInitializationException.java
@@ -0,0 +1,29 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.exception;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Exception thrown when the Miele webservice fails to initialize.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class MieleWebserviceInitializationException extends RuntimeException {
+ private static final long serialVersionUID = -3778846331483843234L;
+
+ public MieleWebserviceInitializationException(String message, Throwable cause) {
+ super(message, cause);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/MieleWebserviceTransientException.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/MieleWebserviceTransientException.java
new file mode 100644
index 00000000000..ce00575676e
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/MieleWebserviceTransientException.java
@@ -0,0 +1,43 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.exception;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.webservice.ConnectionError;
+
+/**
+ * {@link RuntimeException} thrown if a transient error occurred which the binding can recover from by retrying.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class MieleWebserviceTransientException extends RuntimeException {
+ private static final long serialVersionUID = -1863609233382694104L;
+
+ private final ConnectionError connectionError;
+
+ public MieleWebserviceTransientException(final String message, final ConnectionError connectionError) {
+ super(message);
+ this.connectionError = connectionError;
+ }
+
+ public MieleWebserviceTransientException(final String message, final Throwable cause,
+ final ConnectionError connectionError) {
+ super(message, cause);
+ this.connectionError = connectionError;
+ }
+
+ public ConnectionError getConnectionError() {
+ return connectionError;
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/TooManyRequestsException.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/TooManyRequestsException.java
new file mode 100644
index 00000000000..afb8e7f57e7
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/exception/TooManyRequestsException.java
@@ -0,0 +1,90 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.exception;
+
+import java.time.Duration;
+import java.time.LocalDateTime;
+import java.time.format.DateTimeFormatter;
+import java.time.format.DateTimeParseException;
+import java.util.Locale;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * {@link RuntimeException} indicating that too many requests have been made against the cloud service.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class TooManyRequestsException extends RuntimeException {
+ private static final long serialVersionUID = 3393292912418862566L;
+
+ @Nullable
+ private final String retryAfter;
+
+ private final Logger logger = LoggerFactory.getLogger(this.getClass());
+
+ public TooManyRequestsException(String message, @Nullable String retryAfter) {
+ super(message);
+ this.retryAfter = retryAfter;
+ }
+
+ /**
+ * Gets whether a hint on when to retry the operation is available.
+ *
+ * @return Whether a hint on when to retry the operation is available.
+ */
+ public boolean hasRetryAfterHint() {
+ return retryAfter != null;
+ }
+
+ /**
+ * Gets the number of seconds until the operation may be retried.
+ *
+ * @return The number of seconds until the operation may be retried. This will return -1 if no Retry-After header
+ * was present or parsing the data from the header fails.
+ */
+ public long getSecondsUntilRetry() {
+ String retryAfter = this.retryAfter;
+ if (retryAfter == null) {
+ logger.debug("Received no Retry-After header.");
+ return -1;
+ }
+
+ logger.debug("Received Retry-After header: {}", retryAfter);
+ try {
+ long seconds = Long.parseLong(retryAfter);
+ logger.debug("Interpreted Retry-After header value: {} seconds", seconds);
+ return seconds;
+ } catch (NumberFormatException e) {
+ DateTimeFormatter formatter = DateTimeFormatter.ofPattern("ccc, d MMM yyyy HH:mm:ss z", Locale.US);
+
+ try {
+ LocalDateTime dateTime = LocalDateTime.parse(retryAfter, formatter);
+ logger.debug("Interpreted Retry-After header value: {}", dateTime);
+
+ Duration duration = Duration.between(LocalDateTime.now(), dateTime);
+
+ long seconds = Math.max(0, duration.toMillis() / 1000);
+ logger.debug("Interpreted Retry-After header value: {} seconds.", seconds);
+ return seconds;
+ } catch (DateTimeParseException dateTimeParseException) {
+ logger.warn("Unable to parse Retry-After header: {}", retryAfter);
+ return -1;
+ }
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/language/CombiningLanguageProvider.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/language/CombiningLanguageProvider.java
new file mode 100644
index 00000000000..e470e0f82b9
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/language/CombiningLanguageProvider.java
@@ -0,0 +1,71 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.language;
+
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * {@link LanguageProvider} combining two {@link LanguageProvider}s, a prioritized and a fallback provider.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class CombiningLanguageProvider implements LanguageProvider {
+ private @Nullable LanguageProvider prioritizedLanguageProvider;
+ private @Nullable LanguageProvider fallbackLanguageProvider;
+
+ /**
+ * Creates a new instance.
+ *
+ * @param prioritizedLanguageProvider Primary {@link LanguageProvider} to use. May be {@code null}, in that case the
+ * {@code fallbackLanguageProvider} will be used.
+ * @param fallbackLanguageProvider {@link LanguageProvider} to fall back to if the
+ * {@code prioritizedLanguageProvider} is {@code null} or provides no language. May be
+ * {@code null}, in case the fallback is used and returns no language then no language will be returned.
+ */
+ public CombiningLanguageProvider(@Nullable LanguageProvider prioritizedLanguageProvider,
+ @Nullable LanguageProvider fallbackLanguageProvider) {
+ this.prioritizedLanguageProvider = prioritizedLanguageProvider;
+ this.fallbackLanguageProvider = fallbackLanguageProvider;
+ }
+
+ public void setPrioritizedLanguageProvider(LanguageProvider prioritizedLanguageProvider) {
+ this.prioritizedLanguageProvider = prioritizedLanguageProvider;
+ }
+
+ public void unsetPrioritizedLanguageProvider() {
+ this.prioritizedLanguageProvider = null;
+ }
+
+ public void setFallbackLanguageProvider(LanguageProvider fallbackLanguageProvider) {
+ this.fallbackLanguageProvider = fallbackLanguageProvider;
+ }
+
+ public void unsetFallbackLanguageProvider() {
+ this.fallbackLanguageProvider = null;
+ }
+
+ @Override
+ public Optional getLanguage() {
+ Optional prioritizedLanguage = Optional.ofNullable(prioritizedLanguageProvider)
+ .flatMap(LanguageProvider::getLanguage);
+ if (prioritizedLanguage.isPresent()) {
+ return prioritizedLanguage;
+ } else {
+ return Optional.ofNullable(fallbackLanguageProvider).flatMap(LanguageProvider::getLanguage);
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/language/JvmLanguageProvider.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/language/JvmLanguageProvider.java
new file mode 100644
index 00000000000..c16c0707a5e
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/language/JvmLanguageProvider.java
@@ -0,0 +1,31 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.language;
+
+import java.util.Locale;
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * {@link LanguageProvider} returning the default JVM language.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class JvmLanguageProvider implements LanguageProvider {
+ @Override
+ public Optional getLanguage() {
+ return Optional.ofNullable(Locale.getDefault()).map(Locale::getLanguage).filter(l -> !l.isEmpty());
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/language/LanguageProvider.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/language/LanguageProvider.java
new file mode 100644
index 00000000000..86e1259d5f7
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/language/LanguageProvider.java
@@ -0,0 +1,32 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.language;
+
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * Interface for providing language code information.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public interface LanguageProvider {
+ /**
+ * Gets a language represented as 2-letter language code.
+ *
+ * @return The language represented as 2-letter language code.
+ */
+ Optional getLanguage();
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/language/OpenHabLanguageProvider.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/language/OpenHabLanguageProvider.java
new file mode 100644
index 00000000000..7c7436d3551
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/language/OpenHabLanguageProvider.java
@@ -0,0 +1,37 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.language;
+
+import java.util.Optional;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.core.i18n.LocaleProvider;
+
+/**
+ * Language provider relying on the openHAB runtime to provide a locale which is converted to a language.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public class OpenHabLanguageProvider implements LanguageProvider {
+ private final LocaleProvider localeProvider;
+
+ public OpenHabLanguageProvider(LocaleProvider localeProvider) {
+ this.localeProvider = localeProvider;
+ }
+
+ @Override
+ public Optional getLanguage() {
+ return Optional.of(localeProvider.getLocale().getLanguage());
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/request/RequestFactory.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/request/RequestFactory.java
new file mode 100644
index 00000000000..f1c8d5ed923
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/request/RequestFactory.java
@@ -0,0 +1,64 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.request;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.api.Request;
+
+/**
+ * Factory for {@link Request} objects.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public interface RequestFactory extends AutoCloseable {
+ /**
+ * Creates a GET {@link Request} for the given URL decorated with all required headers to interact with the Miele
+ * cloud.
+ *
+ * @param url The URL to GET.
+ * @param accessToken The OAuth2 access token for bearer authentication.
+ * @return The {@link Request}.
+ */
+ Request createGetRequest(String url, String accessToken);
+
+ /**
+ * Creates a PUT {@link Request} for the given URL decorated with all required headers to interact with the Miele
+ * cloud.
+ *
+ * @param url The URL to PUT.
+ * @param accessToken The OAuth2 access token for bearer authentication.
+ * @param jsonContent Json content to send in the body of the request.
+ * @return The {@link Request}.
+ */
+ Request createPutRequest(String url, String accessToken, String jsonContent);
+
+ /**
+ * Creates a POST {@link Request} for the given URL decorated with all required headers to interact with the Miele
+ * cloud.
+ *
+ * @param url The URL to POST.
+ * @param accessToken The OAuth2 access token for bearer authentication.
+ * @return The {@link Request}.
+ */
+ Request createPostRequest(String url, String accessToken);
+
+ /**
+ * Creates a GET request prepared for HTTP event stream data (also referred to as Server Sent Events, SSE).
+ *
+ * @param url The URL to subscribe to.
+ * @param accessToken The OAuth2 access token for bearer authentication.
+ * @return The {@link Request}.
+ */
+ Request createSseRequest(String url, String accessToken);
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/request/RequestFactoryImpl.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/request/RequestFactoryImpl.java
new file mode 100644
index 00000000000..2f557ecb259
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/request/RequestFactoryImpl.java
@@ -0,0 +1,113 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.request;
+
+import java.nio.charset.StandardCharsets;
+import java.util.Optional;
+import java.util.concurrent.TimeUnit;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jetty.client.HttpClient;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.client.util.StringContentProvider;
+import org.eclipse.jetty.http.HttpMethod;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceInitializationException;
+import org.openhab.binding.mielecloud.internal.webservice.language.LanguageProvider;
+import org.openhab.core.io.net.http.HttpClientFactory;
+
+/**
+ * Default implementation of {@link RequestFactory}.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class RequestFactoryImpl implements RequestFactory {
+ private static final long REQUEST_TIMEOUT = 5;
+ private static final long EXTENDED_REQUEST_TIMEOUT = 10;
+ private static final TimeUnit REQUEST_TIMEOUT_UNIT = TimeUnit.SECONDS;
+
+ private final HttpClient httpClient;
+ private final LanguageProvider languageProvider;
+
+ /**
+ * Creates a new {@link RequestFactoryImpl}.
+ *
+ * @param httpClientFactory Factory for obtaining a {@link HttpClient}.
+ * @param languageProvider Provider for the language to use for new requests.
+ * @throws MieleWebserviceInitializationException if creating and starting a new {@link HttpClient} fails.
+ */
+ public RequestFactoryImpl(HttpClientFactory httpClientFactory, LanguageProvider languageProvider) {
+ this.httpClient = httpClientFactory.createHttpClient("mielecloud");
+ try {
+ this.httpClient.start();
+ } catch (Exception e) {
+ throw new MieleWebserviceInitializationException("Failed to start HttpClient", e);
+ }
+ this.languageProvider = languageProvider;
+ }
+
+ private Request createRequestWithDefaultHeaders(String url, String accessToken) {
+ return httpClient.newRequest(url).header("Content-type", "application/json").header("Authorization",
+ "Bearer " + accessToken);
+ }
+
+ private Request decorateWithLanguageParameter(Request request) {
+ Optional language = languageProvider.getLanguage();
+ if (language.isPresent() && !language.get().isEmpty()) {
+ return request.param("language", language.get());
+ } else {
+ return request;
+ }
+ }
+
+ private Request decorateWithAcceptLanguageHeader(Request request) {
+ Optional language = languageProvider.getLanguage();
+ if (language.isPresent() && !language.get().isEmpty()) {
+ return request.header("Accept-Language", language.get());
+ } else {
+ return request;
+ }
+ }
+
+ private Request createDefaultHttpRequest(String url, String accessToken, long timeout) {
+ return decorateWithLanguageParameter(createRequestWithDefaultHeaders(url, accessToken)).header("Accept", "*/*")
+ .timeout(timeout, REQUEST_TIMEOUT_UNIT);
+ }
+
+ @Override
+ public Request createGetRequest(String url, String accessToken) {
+ return createDefaultHttpRequest(url, accessToken, REQUEST_TIMEOUT).method(HttpMethod.GET);
+ }
+
+ @Override
+ public Request createPutRequest(String url, String accessToken, String jsonContent) {
+ return createDefaultHttpRequest(url, accessToken, EXTENDED_REQUEST_TIMEOUT).method(HttpMethod.PUT)
+ .content(new StringContentProvider("application/json", jsonContent, StandardCharsets.UTF_8));
+ }
+
+ @Override
+ public Request createPostRequest(String url, String accessToken) {
+ return createDefaultHttpRequest(url, accessToken, REQUEST_TIMEOUT).method(HttpMethod.POST);
+ }
+
+ @Override
+ public Request createSseRequest(String url, String accessToken) {
+ return decorateWithAcceptLanguageHeader(createRequestWithDefaultHeaders(url, accessToken)).header("Accept",
+ "text/event-stream");
+ }
+
+ @Override
+ public void close() throws Exception {
+ httpClient.stop();
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/retry/AuthorizationFailedRetryStrategy.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/retry/AuthorizationFailedRetryStrategy.java
new file mode 100644
index 00000000000..aa56865883b
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/retry/AuthorizationFailedRetryStrategy.java
@@ -0,0 +1,93 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.retry;
+
+import java.util.concurrent.ExecutionException;
+import java.util.function.Consumer;
+import java.util.function.Supplier;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.mielecloud.internal.auth.OAuthException;
+import org.openhab.binding.mielecloud.internal.auth.OAuthTokenRefresher;
+import org.openhab.binding.mielecloud.internal.webservice.ConnectionError;
+import org.openhab.binding.mielecloud.internal.webservice.exception.AuthorizationFailedException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceException;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * The {@link AuthorizationFailedRetryStrategy} retries an operation after refreshing the access token in case of an
+ * authorization failure.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class AuthorizationFailedRetryStrategy implements RetryStrategy {
+ /**
+ * Message of exception thrown by the Jetty client in case of unmatching header fields and body content. E.g.
+ * application/json header with HTML body content. Mostly thrown when an invalid 401 response is received.
+ */
+ public static final String JETTY_401_HEADER_BODY_MISMATCH_EXCEPTION_MESSAGE = "org.eclipse.jetty.client.HttpResponseException: HTTP protocol violation: Authentication challenge without WWW-Authenticate header";
+
+ private final Logger logger = LoggerFactory.getLogger(this.getClass());
+
+ private final OAuthTokenRefresher tokenRefresher;
+ private final String serviceHandle;
+
+ public AuthorizationFailedRetryStrategy(OAuthTokenRefresher tokenRefresher, String serviceHandle) {
+ this.tokenRefresher = tokenRefresher;
+ this.serviceHandle = serviceHandle;
+ }
+
+ private void refreshToken() {
+ try {
+ logger.debug("Refreshing Miele OAuth access token.");
+ tokenRefresher.refreshToken(serviceHandle);
+ logger.debug("Miele OAuth access token has successfully been refreshed.");
+ } catch (OAuthException e) {
+ throw new MieleWebserviceException("Failed to refresh access token.", e,
+ ConnectionError.AUTHORIZATION_FAILED);
+ }
+ }
+
+ @Override
+ public <@Nullable T> T performRetryableOperation(Supplier operation, Consumer onException) {
+ try {
+ return operation.get();
+ } catch (AuthorizationFailedException e) {
+ onException.accept(e);
+ refreshToken();
+ } catch (MieleWebserviceException e) {
+ // Workaround for HTML response from cloud in case of a 401 HTTP error.
+ var cause = e.getCause();
+ if (cause == null || !(cause instanceof ExecutionException)) {
+ throw e;
+ }
+
+ if (!JETTY_401_HEADER_BODY_MISMATCH_EXCEPTION_MESSAGE.equals(cause.getMessage())) {
+ throw e;
+ }
+
+ onException.accept(e);
+ refreshToken();
+ }
+
+ try {
+ return operation.get();
+ } catch (AuthorizationFailedException e) {
+ throw new MieleWebserviceException("Request failed after access token renewal.", e,
+ ConnectionError.AUTHORIZATION_FAILED);
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/retry/NTimesRetryStrategy.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/retry/NTimesRetryStrategy.java
new file mode 100644
index 00000000000..9e957c86e6e
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/retry/NTimesRetryStrategy.java
@@ -0,0 +1,72 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.retry;
+
+import java.util.function.Consumer;
+import java.util.function.Supplier;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.mielecloud.internal.webservice.ConnectionError;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceTransientException;
+
+/**
+ * {@link RetryStrategy} retrying a failing operation for a number of times.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public class NTimesRetryStrategy implements RetryStrategy {
+ private final int numberOfRetries;
+
+ /**
+ * Creates a new {@link NTimesRetryStrategy}.
+ *
+ * @param numberOfRetries The number of retries to make.
+ * @throws IllegalArgumentException if {@code numberOfRetries} is smaller than zero.
+ */
+ public NTimesRetryStrategy(int numberOfRetries) {
+ if (numberOfRetries < 0) {
+ throw new IllegalArgumentException("Number of retries must not be negative.");
+ }
+
+ this.numberOfRetries = numberOfRetries;
+ }
+
+ @Override
+ public <@Nullable T> T performRetryableOperation(Supplier operation, Consumer onException) {
+ boolean obtainedReturnValue = false;
+ T returnValue = null;
+ MieleWebserviceTransientException lastException = null;
+ for (int i = 0; !obtainedReturnValue && i < numberOfRetries + 1; i++) {
+ try {
+ returnValue = operation.get();
+ obtainedReturnValue = true;
+ } catch (MieleWebserviceTransientException e) {
+ lastException = e;
+ if (i < numberOfRetries) {
+ onException.accept(e);
+ }
+ }
+ }
+
+ if (!obtainedReturnValue) {
+ throw new MieleWebserviceException(
+ "Unable to perform operation. Operation failed " + (numberOfRetries + 1) + " times.", lastException,
+ lastException == null ? ConnectionError.UNKNOWN : lastException.getConnectionError());
+ } else {
+ return returnValue;
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/retry/RetryStrategy.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/retry/RetryStrategy.java
new file mode 100644
index 00000000000..3aae0dc6d03
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/retry/RetryStrategy.java
@@ -0,0 +1,62 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.retry;
+
+import java.util.function.Consumer;
+import java.util.function.Supplier;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * Interface for strategies implementing the retry behavior of requests against the Miele cloud.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+public interface RetryStrategy {
+ /**
+ * Performs an operation which may be retried several times.
+ *
+ * If retrying fails or a critical error occurred, this method may throw {@link Exception}s of any type.
+ *
+ * @param operation The operation to perform. To signal that an error can be resolved by retrying this operation it
+ * should throw an {@link Exception}. Whether the operation is retried is up to the {@link RetryStrategy}
+ * implementation.
+ * @param onException Handler to invoke when an {@link Exception} is handled by retrying the {@code operation}. This
+ * handler should at least log a message. It must not throw any exception.
+ * @return The object returned by {@code operation} if it completed successfully.
+ */
+ <@Nullable T> T performRetryableOperation(Supplier operation, Consumer onException);
+
+ /**
+ * Performs an operation which may be retried several times.
+ *
+ * If retrying fails or a critical error occurred, this method may throw {@link Exception}s of any type.
+ *
+ * @param operation The operation to perform. To signal that an error can be resolved by retrying this operation it
+ * should throw an {@link Exception}. Whether the operation is retried is up to the {@link RetryStrategy}
+ * implementation
+ * @param onException Handler to invoke when an {@link Exception} is handled by retrying the {@code operation}. This
+ * handler should at least log a message. It may not throw any exception.
+ */
+ default void performRetryableOperation(Runnable operation, Consumer onException) {
+ performRetryableOperation(new Supplier<@Nullable Void>() {
+ @Override
+ public @Nullable Void get() {
+ operation.run();
+ return null;
+ }
+ }, onException);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/retry/RetryStrategyCombiner.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/retry/RetryStrategyCombiner.java
new file mode 100644
index 00000000000..374e11bbc6d
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/retry/RetryStrategyCombiner.java
@@ -0,0 +1,47 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.retry;
+
+import java.util.function.Consumer;
+import java.util.function.Supplier;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * {@link RetryStrategy} implementation wrapping the consecutive execution of two retry strategies.
+ *
+ * @author Björn Lange and Roland Edelhoff - Initial contribution
+ */
+@NonNullByDefault
+public class RetryStrategyCombiner implements RetryStrategy {
+ private final RetryStrategy first;
+ private final RetryStrategy second;
+
+ /**
+ * Creates a new {@link RetryStrategy} combining the given ones.
+ *
+ * @param first First strategy to execute.
+ * @param second Strategy to execute in each execution of {@code first}.
+ */
+ public RetryStrategyCombiner(RetryStrategy first, RetryStrategy second) {
+ this.first = first;
+ this.second = second;
+ }
+
+ @Override
+ public <@Nullable T> T performRetryableOperation(Supplier operation, Consumer onException) {
+ return first.performRetryableOperation(() -> second.performRetryableOperation(operation, onException),
+ onException);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/BackoffStrategy.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/BackoffStrategy.java
new file mode 100644
index 00000000000..dace1fe8b0c
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/BackoffStrategy.java
@@ -0,0 +1,50 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.sse;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * A strategy computing the wait time between multiple connection attempts.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+interface BackoffStrategy {
+ /**
+ * Gets the minimal number of seconds to wait until retrying an operation. This is the lower bound of the value
+ * returned by {@link #getSecondsUntilRetry(int)}.
+ *
+ * @return The minimal number of seconds to wait until retrying an operation. Always larger or equal to zero, always
+ * smaller than {@link #getMaximumSecondsUntilRetry()}.
+ */
+ long getMinimumSecondsUntilRetry();
+
+ /**
+ * Gets the maximal number of seconds to wait until retrying an operation. This is the upper bound of the value
+ * returned by {@link #getSecondsUntilRetry(int)}.
+ *
+ * @return The maximal number of seconds to wait until retrying an operation. Always larger or equal to zero, always
+ * larger than {@link #getMinimumSecondsUntilRetry()}.
+ */
+ long getMaximumSecondsUntilRetry();
+
+ /**
+ * Gets the number of seconds until a retryable operation is performed. The value returned by this method is within
+ * the interval defined by {@link #getMinimumSecondsUntilRetry()} and {@link #getMaximumSecondsUntilRetry()}.
+ *
+ * @param failedConnectionAttempts The number of failed attempts.
+ * @return The number of seconds to wait before making the next attempt.
+ */
+ long getSecondsUntilRetry(int failedAttempts);
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/ExponentialBackoffWithJitter.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/ExponentialBackoffWithJitter.java
new file mode 100644
index 00000000000..200ff4b6185
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/ExponentialBackoffWithJitter.java
@@ -0,0 +1,101 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.sse;
+
+import java.util.Random;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * Implements the exponential backoff with jitter backoff strategy.
+ *
+ * @author Björn Lange - Initial contribution
+ */
+@NonNullByDefault
+class ExponentialBackoffWithJitter implements BackoffStrategy {
+ private static final long INITIAL_RECONNECT_ATTEMPT_WAIT_TIME_IN_SECONDS = 5;
+ private static final long MAXIMUM_RECONNECT_ATTEMPT_WAIT_TIME_IN_SECONDS = 3600;
+
+ private final long minimumWaitTimeInSeconds;
+ private final long maximumWaitTimeInSeconds;
+ private final long retryIntervalInSeconds;
+ private final Random random;
+
+ private final Logger logger = LoggerFactory.getLogger(ExponentialBackoffWithJitter.class);
+
+ /**
+ * Creates a new {@link ExponentialBackoffWithJitter}.
+ */
+ public ExponentialBackoffWithJitter() {
+ this(INITIAL_RECONNECT_ATTEMPT_WAIT_TIME_IN_SECONDS, MAXIMUM_RECONNECT_ATTEMPT_WAIT_TIME_IN_SECONDS,
+ INITIAL_RECONNECT_ATTEMPT_WAIT_TIME_IN_SECONDS);
+ }
+
+ ExponentialBackoffWithJitter(long minimumWaitTimeInSeconds, long maximumWaitTimeInSeconds,
+ long retryIntervalInSeconds) {
+ this(minimumWaitTimeInSeconds, maximumWaitTimeInSeconds, retryIntervalInSeconds, new Random());
+ }
+
+ ExponentialBackoffWithJitter(long minimumWaitTimeInSeconds, long maximumWaitTimeInSeconds,
+ long retryIntervalInSeconds, Random random) {
+ if (minimumWaitTimeInSeconds < 0) {
+ throw new IllegalArgumentException("minimumWaitTimeInSeconds must not be smaller than zero");
+ }
+ if (maximumWaitTimeInSeconds < 0) {
+ throw new IllegalArgumentException("maximumWaitTimeInSeconds must not be smaller than zero");
+ }
+ if (retryIntervalInSeconds < 0) {
+ throw new IllegalArgumentException("retryIntervalInSeconds must not be smaller than zero");
+ }
+ if (maximumWaitTimeInSeconds < minimumWaitTimeInSeconds) {
+ throw new IllegalArgumentException(
+ "maximumWaitTimeInSeconds must not be smaller than minimumWaitTimeInSeconds");
+ }
+ if (maximumWaitTimeInSeconds < retryIntervalInSeconds) {
+ throw new IllegalArgumentException(
+ "maximumWaitTimeInSeconds must not be smaller than retryIntervalInSeconds");
+ }
+
+ this.minimumWaitTimeInSeconds = minimumWaitTimeInSeconds;
+ this.maximumWaitTimeInSeconds = maximumWaitTimeInSeconds;
+ this.retryIntervalInSeconds = retryIntervalInSeconds;
+ this.random = random;
+ }
+
+ @Override
+ public long getMinimumSecondsUntilRetry() {
+ return minimumWaitTimeInSeconds;
+ }
+
+ @Override
+ public long getMaximumSecondsUntilRetry() {
+ return maximumWaitTimeInSeconds;
+ }
+
+ @Override
+ public long getSecondsUntilRetry(int failedAttempts) {
+ if (failedAttempts < 0) {
+ logger.warn("The number of failed attempts must not be smaller than zero, was {}.", failedAttempts);
+ }
+
+ return minimumWaitTimeInSeconds
+ + getRandomLongWithUpperLimit(Math.min(maximumWaitTimeInSeconds - minimumWaitTimeInSeconds,
+ retryIntervalInSeconds * (long) Math.pow(2, Math.max(0, failedAttempts))));
+ }
+
+ private long getRandomLongWithUpperLimit(long upperLimit) {
+ return Math.abs(random.nextLong()) % (upperLimit + 1);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/ServerSentEvent.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/ServerSentEvent.java
new file mode 100644
index 00000000000..3848abf6741
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/ServerSentEvent.java
@@ -0,0 +1,62 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.sse;
+
+import java.util.Objects;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+
+/**
+ * An event emitted by an SSE connection.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class ServerSentEvent {
+ private final String event;
+ private final String data;
+
+ ServerSentEvent(String event, String data) {
+ this.event = event;
+ this.data = data;
+ }
+
+ public String getEvent() {
+ return event;
+ }
+
+ public String getData() {
+ return data;
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(event, data);
+ }
+
+ @Override
+ public boolean equals(@Nullable Object obj) {
+ if (this == obj) {
+ return true;
+ }
+ if (obj == null) {
+ return false;
+ }
+ if (getClass() != obj.getClass()) {
+ return false;
+ }
+ ServerSentEvent other = (ServerSentEvent) obj;
+ return Objects.equals(event, other.event) && Objects.equals(data, other.data);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/SseConnection.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/SseConnection.java
new file mode 100644
index 00000000000..c250c94497b
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/SseConnection.java
@@ -0,0 +1,240 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.sse;
+
+import java.util.ArrayList;
+import java.util.List;
+import java.util.concurrent.ScheduledExecutorService;
+import java.util.concurrent.TimeUnit;
+import java.util.concurrent.TimeoutException;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.eclipse.jetty.client.api.Request;
+import org.eclipse.jetty.client.api.Response;
+import org.eclipse.jetty.client.api.Result;
+import org.eclipse.jetty.client.util.InputStreamResponseListener;
+import org.openhab.binding.mielecloud.internal.webservice.ConnectionError;
+import org.openhab.binding.mielecloud.internal.webservice.HttpUtil;
+import org.openhab.binding.mielecloud.internal.webservice.exception.AuthorizationFailedException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceDisconnectSseException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceTransientException;
+import org.openhab.binding.mielecloud.internal.webservice.exception.TooManyRequestsException;
+import org.openhab.binding.mielecloud.internal.webservice.retry.AuthorizationFailedRetryStrategy;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * An active or inactive SSE connection emitting a stream of events.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public final class SseConnection {
+ private static final long CONNECTION_TIMEOUT = 30;
+ private static final TimeUnit CONNECTION_TIMEOUT_UNIT = TimeUnit.SECONDS;
+
+ private final Logger logger = LoggerFactory.getLogger(SseConnection.class);
+
+ private final String endpoint;
+ private final SseRequestFactory requestFactory;
+ private final ScheduledExecutorService scheduler;
+ private final BackoffStrategy backoffStrategy;
+
+ private final List listeners = new ArrayList<>();
+
+ private boolean active = false;
+
+ private int failedConnectionAttempts = 0;
+
+ @Nullable
+ private Request sseRequest;
+
+ /**
+ * Creates a new {@link SseConnection} to the given endpoint.
+ *
+ * Note: It is required to call {@link #connect()} in order to open the connection and start receiving events.
+ *
+ * @param endpoint The endpoint to connect to.
+ * @param requestFactory Factory for creating requests.
+ * @param scheduler Scheduler to run scheduled and concurrent tasks on.
+ */
+ public SseConnection(String endpoint, SseRequestFactory requestFactory, ScheduledExecutorService scheduler) {
+ this(endpoint, requestFactory, scheduler, new ExponentialBackoffWithJitter());
+ }
+
+ /**
+ * Creates a new {@link SseConnection} to the given endpoint.
+ *
+ * Note: It is required to call {@link #connect()} in order to open the connection and start receiving events.
+ *
+ * @param endpoint The endpoint to connect to.
+ * @param requestFactory Factory for creating requests.
+ * @param scheduler Scheduler to run scheduled and concurrent tasks on.
+ * @param backoffStrategy Strategy for deriving the wait time between connection attempts.
+ */
+ SseConnection(String endpoint, SseRequestFactory requestFactory, ScheduledExecutorService scheduler,
+ BackoffStrategy backoffStrategy) {
+ this.endpoint = endpoint;
+ this.requestFactory = requestFactory;
+ this.scheduler = scheduler;
+ this.backoffStrategy = backoffStrategy;
+ }
+
+ public synchronized void connect() {
+ active = true;
+ connectInternal();
+ }
+
+ private synchronized void connectInternal() {
+ if (!active) {
+ return;
+ }
+
+ Request runningRequest = this.sseRequest;
+ if (runningRequest != null) {
+ return;
+ }
+
+ logger.debug("Opening SSE connection...");
+ Request sseRequest = createRequest();
+ if (sseRequest == null) {
+ logger.warn("Could not create SSE request, not opening SSE connection.");
+ return;
+ }
+
+ final InputStreamResponseListener stream = new InputStreamResponseListener();
+ SseStreamParser eventStreamParser = new SseStreamParser(stream.getInputStream(), this::onServerSentEvent,
+ this::onSseStreamClosed);
+
+ sseRequest = sseRequest
+ .onResponseHeaders(
+ response -> scheduler.schedule(eventStreamParser::parseAndDispatchEvents, 0, TimeUnit.SECONDS))
+ .onComplete(result -> onConnectionComplete(result));
+ sseRequest.send(stream);
+ this.sseRequest = sseRequest;
+ }
+
+ @Nullable
+ private Request createRequest() {
+ Request sseRequest = requestFactory.createSseRequest(endpoint);
+ if (sseRequest == null) {
+ return null;
+ }
+
+ return sseRequest.timeout(0, TimeUnit.SECONDS).idleTimeout(CONNECTION_TIMEOUT, CONNECTION_TIMEOUT_UNIT);
+ }
+
+ private synchronized void onSseStreamClosed(@Nullable Throwable exception) {
+ if (exception != null && AuthorizationFailedRetryStrategy.JETTY_401_HEADER_BODY_MISMATCH_EXCEPTION_MESSAGE
+ .equals(exception.getMessage())) {
+ onConnectionError(ConnectionError.AUTHORIZATION_FAILED);
+ } else if (exception instanceof TimeoutException) {
+ onConnectionError(ConnectionError.TIMEOUT);
+ } else {
+ onConnectionError(ConnectionError.SSE_STREAM_ENDED);
+ }
+ }
+
+ private synchronized void onConnectionComplete(@Nullable Result result) {
+ sseRequest = null;
+
+ if (result == null) {
+ logger.warn("SSE stream was closed but there was no result delivered.");
+ onConnectionError(ConnectionError.SSE_STREAM_ENDED);
+ return;
+ }
+
+ Response response = result.getResponse();
+ if (response == null) {
+ logger.warn("SSE stream was closed without response.");
+ onConnectionError(ConnectionError.SSE_STREAM_ENDED);
+ return;
+ }
+
+ onConnectionClosed(response);
+ }
+
+ private void onConnectionClosed(Response response) {
+ try {
+ HttpUtil.checkHttpSuccess(response);
+ onConnectionError(ConnectionError.SSE_STREAM_ENDED);
+ } catch (AuthorizationFailedException e) {
+ onConnectionError(ConnectionError.AUTHORIZATION_FAILED);
+ } catch (TooManyRequestsException e) {
+ long secondsUntilRetry = e.getSecondsUntilRetry();
+ if (secondsUntilRetry < 0) {
+ onConnectionError(ConnectionError.TOO_MANY_RERQUESTS);
+ } else {
+ onConnectionError(ConnectionError.TOO_MANY_RERQUESTS, secondsUntilRetry);
+ }
+ } catch (MieleWebserviceTransientException e) {
+ onConnectionError(e.getConnectionError(), 0);
+ } catch (MieleWebserviceException e) {
+ onConnectionError(e.getConnectionError());
+ }
+ }
+
+ private void onConnectionError(ConnectionError connectionError) {
+ onConnectionError(connectionError, backoffStrategy.getSecondsUntilRetry(failedConnectionAttempts));
+ }
+
+ private synchronized void onConnectionError(ConnectionError connectionError, long secondsUntilRetry) {
+ if (!active) {
+ return;
+ }
+
+ if (connectionError != ConnectionError.AUTHORIZATION_FAILED) {
+ scheduleReconnect(secondsUntilRetry);
+ }
+
+ fireConnectionError(connectionError);
+ failedConnectionAttempts++;
+ }
+
+ private void scheduleReconnect(long secondsUntilRetry) {
+ long retryInSeconds = Math.max(backoffStrategy.getMinimumSecondsUntilRetry(),
+ Math.min(secondsUntilRetry, backoffStrategy.getMaximumSecondsUntilRetry()));
+ scheduler.schedule(this::connectInternal, retryInSeconds, TimeUnit.SECONDS);
+ logger.debug("Scheduled reconnect attempt for Miele webservice to take place in {} seconds", retryInSeconds);
+ }
+
+ public synchronized void disconnect() {
+ active = false;
+
+ Request runningRequest = sseRequest;
+ if (runningRequest == null) {
+ logger.debug("SSE connection is not established, skipping SSE disconnect.");
+ return;
+ }
+
+ logger.debug("Disconnecting SSE");
+ runningRequest.abort(new MieleWebserviceDisconnectSseException());
+ sseRequest = null;
+ logger.debug("Disconnected");
+ }
+
+ private void onServerSentEvent(ServerSentEvent event) {
+ failedConnectionAttempts = 0;
+ listeners.forEach(l -> l.onServerSentEvent(event));
+ }
+
+ private void fireConnectionError(ConnectionError connectionError) {
+ listeners.forEach(l -> l.onConnectionError(connectionError, failedConnectionAttempts));
+ }
+
+ public void addSseListener(SseListener listener) {
+ listeners.add(listener);
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/SseListener.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/SseListener.java
new file mode 100644
index 00000000000..cb39a47c85a
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/SseListener.java
@@ -0,0 +1,39 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.sse;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.mielecloud.internal.webservice.ConnectionError;
+
+/**
+ * Listens to events received via a SSE connection and errors concerning that connection.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+public interface SseListener {
+ /**
+ * Called when an event is received via a SSE connection.
+ *
+ * @param event The received event.
+ */
+ void onServerSentEvent(ServerSentEvent event);
+
+ /**
+ * Called when an error occurs that is related to the connection and cannot be handled automatically.
+ *
+ * @param connectionError The connection error.
+ * @param failedReconnectAttempts The number of attempts that were made to reconnect to the event stream.
+ */
+ void onConnectionError(ConnectionError connectionError, int failedReconnectAttempts);
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/SseRequestFactory.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/SseRequestFactory.java
new file mode 100644
index 00000000000..b509e9e7571
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/SseRequestFactory.java
@@ -0,0 +1,36 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.sse;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.eclipse.jetty.client.api.Request;
+
+/**
+ * Factory that produces configured {@link Request} instances for usage with SSE.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+@FunctionalInterface
+public interface SseRequestFactory {
+ /**
+ * Produces a {@link Request} which is decorated with all required headers.
+ *
+ * @param endpoint The endpoint to connect to.
+ * @return The created {@link Request} or {@code null} if no request can be created due to lacking request
+ * information. If this method returns {@code null} then all connection attempts will be cancelled.
+ */
+ @Nullable
+ Request createSseRequest(String endpoint);
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/SseStreamParser.java b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/SseStreamParser.java
new file mode 100644
index 00000000000..92607589bff
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/java/org/openhab/binding/mielecloud/internal/webservice/sse/SseStreamParser.java
@@ -0,0 +1,102 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.mielecloud.internal.webservice.sse;
+
+import java.io.BufferedReader;
+import java.io.IOException;
+import java.io.InputStream;
+import java.io.InputStreamReader;
+import java.nio.charset.StandardCharsets;
+import java.util.function.Consumer;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.mielecloud.internal.webservice.exception.MieleWebserviceDisconnectSseException;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * Parses events from the SSE event stream and emits them via the given dispatcher.
+ *
+ * @author Björn Lange - Initial Contribution
+ */
+@NonNullByDefault
+class SseStreamParser {
+ private static final String SSE_KEY_EVENT = "event:";
+ private static final String SSE_KEY_DATA = "data:";
+
+ private final Logger logger = LoggerFactory.getLogger(SseStreamParser.class);
+
+ private final BufferedReader reader;
+ private final Consumer onServerSentEventCallback;
+ private final Consumer<@Nullable Throwable> onStreamClosedCallback;
+
+ private @Nullable String event;
+
+ SseStreamParser(InputStream inputStream, Consumer onServerSentEventCallback,
+ Consumer<@Nullable Throwable> onStreamClosedCallback) {
+ this.reader = new BufferedReader(new InputStreamReader(inputStream, StandardCharsets.UTF_8));
+ this.onServerSentEventCallback = onServerSentEventCallback;
+ this.onStreamClosedCallback = onStreamClosedCallback;
+ }
+
+ void parseAndDispatchEvents() {
+ try {
+ String line = null;
+ while ((line = reader.readLine()) != null) {
+ onLineReceived(line);
+ }
+
+ silentlyCloseReader();
+ logger.debug("SSE stream ended. Closing stream.");
+ onStreamClosedCallback.accept(null);
+ } catch (IOException exception) {
+ silentlyCloseReader();
+
+ if (!(exception.getCause() instanceof MieleWebserviceDisconnectSseException)) {
+ logger.warn("SSE connection failed unexpectedly: {}", exception.getMessage());
+ onStreamClosedCallback.accept(exception.getCause());
+ }
+ }
+ logger.debug("SSE stream closed.");
+ }
+
+ private void silentlyCloseReader() {
+ try {
+ reader.close();
+ } catch (IOException e) {
+ logger.warn("Failed to clean up SSE connection resources!", e);
+ }
+ }
+
+ private void onLineReceived(String line) {
+ if (line.isEmpty()) {
+ return;
+ }
+
+ if (line.startsWith(SSE_KEY_EVENT)) {
+ event = line.substring(SSE_KEY_EVENT.length()).trim();
+ } else if (line.startsWith(SSE_KEY_DATA)) {
+ String event = this.event;
+ String data = line.substring(SSE_KEY_DATA.length()).trim();
+
+ if (event == null) {
+ logger.warn("Received data payload without prior event payload.");
+ } else {
+ onServerSentEventCallback.accept(new ServerSentEvent(event, data));
+ }
+ } else {
+ logger.warn("Unable to parse line from SSE stream: {}", line);
+ }
+ }
+}
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/binding/binding.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/binding/binding.xml
new file mode 100644
index 00000000000..a9dc15f89c5
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/binding/binding.xml
@@ -0,0 +1,8 @@
+
+
+
+ @text/binding.mielecloud.name
+ @text/binding.mielecloud.description
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/config/configDescription.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/config/configDescription.xml
new file mode 100644
index 00000000000..e2f9d5dc177
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/config/configDescription.xml
@@ -0,0 +1,13 @@
+
+
+
+
+
+ @text/thing-type.config.mielecloud.device.deviceIdentifier.label
+ @text/thing-type.config.mielecloud.device.deviceIdentifier.description
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/i18n/mielecloud.properties b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/i18n/mielecloud.properties
new file mode 100644
index 00000000000..745298699d5
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/i18n/mielecloud.properties
@@ -0,0 +1,253 @@
+# Binding related texts
+binding.mielecloud.name=Miele@home Cloud Binding
+binding.mielecloud.description=This is the cloud-based Miele@home binding.
+
+# Thing related texts
+thing-type.mielecloud.account.label=Miele@home Account
+thing-type.mielecloud.account.description=The Miele@home Account is used to access linked Miele Conn@ct smart home devices.
+
+thing-type.config.mielecloud.account.locale.label=E-mail
+thing-type.config.mielecloud.account.locale.description=E-mail address associated with the Miele Cloud account.
+
+thing-type.config.mielecloud.account.locale.label=Locale
+thing-type.config.mielecloud.account.locale.description=Locale to be used for API calls.
+
+thing-type.config.mielecloud.device.deviceIdentifier.label=Device identifier
+thing-type.config.mielecloud.device.deviceIdentifier.description=Technical device identifier used to identify the Miele device.
+
+thing-type.mielecloud.coffee_system.label=Coffee System
+thing-type.mielecloud.coffee_system.description=The generic thing type for all Miele coffee systems.
+
+thing-type.mielecloud.dishwasher.label=Dishwasher
+thing-type.mielecloud.dishwasher.description=The generic thing type for all Miele dish washing devices.
+
+thing-type.mielecloud.dish_warmer.label=Dish Warmer
+thing-type.mielecloud.dish_warmer.description=The generic thing type for all Miele dish warmer devices.
+
+thing-type.mielecloud.dryer.label=Tumble Dryer
+thing-type.mielecloud.dryer.description=The generic thing type for all Miele drying devices.
+
+thing-type.mielecloud.freezer.label=Freezer
+thing-type.mielecloud.freezer.description=The generic thing type for all Miele freezer devices.
+
+thing-type.mielecloud.fridge.label=Fridge
+thing-type.mielecloud.fridge.description=The generic thing type for all Miele fridge devices.
+
+thing-type.mielecloud.fridge_freezer.label=Fridge Freezer
+thing-type.mielecloud.fridge_freezer.description=The generic thing type for all Miele fridge freezer devices.
+
+thing-type.mielecloud.hob.label=Hob
+thing-type.mielecloud.hob.description=The generic thing type for all Miele hob devices.
+
+thing-type.mielecloud.hood.label=Hood
+thing-type.mielecloud.hood.description=The generic thing type for all Miele hood devices.
+
+thing-type.mielecloud.oven.label=Oven
+thing-type.mielecloud.oven.description=The generic thing type for all Miele oven devices. Includes also Steam Ovens and Dialog Oven.
+
+thing-type.mielecloud.robotic_vacuum_cleaner.label=Robotic Vacuum Cleaner
+thing-type.mielecloud.robotic_vacuum_cleaner.description=The generic thing type for all Miele robotic vacuum cleaner devices.
+
+thing-type.mielecloud.washer_dryer.label=Washer Dryer
+thing-type.mielecloud.washer_dryer.description=The generic thing type for all Miele washer dryer devices.
+
+thing-type.mielecloud.washing_machine.label=Washing Machine
+thing-type.mielecloud.washing_machine.description=The generic thing type for all Miele washing devices.
+
+thing-type.mielecloud.wine_storage.label=Wine Storage
+thing-type.mielecloud.wine_storage.description=The generic thing type for all Miele wine storage devices.
+
+# Channel related texts
+channel-type.mielecloud.remote_control_can_be_started.label=Can Be Started
+channel-type.mielecloud.remote_control_can_be_started.description=Indicates if this device can be started remotely.
+
+channel-type.mielecloud.remote_control_can_be_stopped.label=Can Be Stopped
+channel-type.mielecloud.remote_control_can_be_stopped.description=Indicates if this device can be stopped remotely.
+
+channel-type.mielecloud.remote_control_can_be_paused.label=Can Be Paused
+channel-type.mielecloud.remote_control_can_be_paused.description=Indicates if this device can be paused remotely.
+
+channel-type.mielecloud.remote_control_can_be_switched_on.label=Can Be Switched On
+channel-type.mielecloud.remote_control_can_be_switched_on.description=Indicates if the device can be switched on remotely.
+
+channel-type.mielecloud.remote_control_can_be_switched_off.label=Can Be Switched Off
+channel-type.mielecloud.remote_control_can_be_switched_off.description=Indicates if the device can be switched off remotely.
+
+channel-type.mielecloud.remote_control_can_set_program_active.label=Can Set Active Program
+channel-type.mielecloud.remote_control_can_set_program_active.description=Indicates if the active program of the device can be set remotely.
+
+channel-type.mielecloud.spinning_speed.label=Spinning Speed
+channel-type.mielecloud.spinning_speed.description=The spinning speed of the active program.
+
+channel-type.mielecloud.spinning_speed_raw.label=Raw Spinning Speed
+channel-type.mielecloud.spinning_speed_raw.description=The raw spinning speed of the active program.
+
+channel-type.mielecloud.program_active.label=Active Program
+channel-type.mielecloud.program_active.description=The active program of the device.
+
+channel-type.mielecloud.program_active_raw.label=Raw Active Program
+channel-type.mielecloud.program_active_raw.description=The raw active program of the device.
+
+channel-type.mielecloud.dish_warmer_program_active.label=Active Program
+channel-type.mielecloud.dish_warmer_program_active.description=The active program of the device.
+channel-option.mielecloud.dish_warmer_program_active.warming_cups_glasses=Warming cups/glasses
+channel-option.mielecloud.dish_warmer_program_active.warming_dishes_plates=Warming dishes/plates
+channel-option.mielecloud.dish_warmer_program_active.keeping_food_warm=Keeping food warm
+channel-option.mielecloud.dish_warmer_program_active.low_temperature_cooking=Low temperature cooking
+
+channel-type.mielecloud.vacuum_cleaner_program_active.label=Active Program
+channel-type.mielecloud.vacuum_cleaner_program_active.description=The active program of the device.
+channel-option.mielecloud.vacuum_cleaner_program_active.auto=Auto
+channel-option.mielecloud.vacuum_cleaner_program_active.spot=Spot
+channel-option.mielecloud.vacuum_cleaner_program_active.turbo=Turbo
+channel-option.mielecloud.vacuum_cleaner_program_active.silent=Silent
+
+channel-type.mielecloud.program_phase.label=Program Phase
+channel-type.mielecloud.program_phase.description=The phase of the active program.
+
+channel-type.mielecloud.program_phase_raw.label=Raw Program Phase
+channel-type.mielecloud.program_phase_raw.description=The raw phase of the active program.
+
+channel-type.mielecloud.operation_state.label=Operation State
+channel-type.mielecloud.operation_state.description=The operation state of the device.
+
+channel-type.mielecloud.operation_state_raw.label=Raw Operation State
+channel-type.mielecloud.operation_state_raw.description=The raw operation state of the device.
+
+channel-type.mielecloud.program_start.label=Start
+channel-type.mielecloud.program_start.description=Starts the currently selected program.
+
+channel-type.mielecloud.program_stop.label=Stop
+channel-type.mielecloud.program_stop.description=Stops the currently selected program.
+
+channel-type.mielecloud.program_start_stop.label=Start Stop
+channel-type.mielecloud.program_start_stop.description=Starts or stops the currently selected program.
+channel-option.mielecloud.program_start_stop.start=Start
+channel-option.mielecloud.program_start_stop.stop=Stop
+
+channel-type.mielecloud.program_start_stop_pause.label=Start Stop Pause
+channel-type.mielecloud.program_start_stop_pause.description=Starts, stops or pauses the currently selected program.
+channel-option.mielecloud.program_start_stop_pause.start=Start
+channel-option.mielecloud.program_start_stop_pause.stop=Stop
+channel-option.mielecloud.program_start_stop_pause.pause=Pause
+
+channel-type.mielecloud.power_state_on_off.label=Power
+channel-type.mielecloud.power_state_on_off.description=Switches the device On or Off.
+channel-option.mielecloud.power_state_on_off.on=On
+channel-option.mielecloud.power_state_on_off.off=Off
+
+channel-type.mielecloud.finish_state.label=Finished
+channel-type.mielecloud.finish_state.description=Indicates whether the most recent program finished.
+
+channel-type.mielecloud.delayed_start_time.label=Delayed Start Time
+channel-type.mielecloud.delayed_start_time.description=The delayed start time of the selected program.
+
+channel-type.mielecloud.program_remaining_time.label=Program Remaining Time
+channel-type.mielecloud.program_remaining_time.description=The remaining time of the active program.
+
+channel-type.mielecloud.program_elapsed_time.label=Program Elapsed Time
+channel-type.mielecloud.program_elapsed_time.description=The elapsed time of the active program.
+
+channel-type.mielecloud.program_progress.label=Program Progress
+channel-type.mielecloud.program_progress.description=The progress of the active program.
+
+channel-type.mielecloud.drying_target.label=Drying Target
+channel-type.mielecloud.drying_target.description=The target drying step of the laundry.
+
+channel-type.mielecloud.drying_target_raw.label=Raw Drying Target
+channel-type.mielecloud.drying_target_raw.description=The raw target drying step of the laundry.
+
+channel-type.mielecloud.pre_heat_finished.label=Pre-heat Finished
+channel-type.mielecloud.pre_heat_finished.description=Indicates whether the pre-heating finished.
+
+channel-type.mielecloud.temperature_target.label=Target Temperature
+channel-type.mielecloud.temperature_target.description=The target temperature of the device.
+
+channel-type.mielecloud.temperature_current.label=Current Temperature
+channel-type.mielecloud.temperature_current.description=The currently measured temperature of the device.
+
+channel-type.mielecloud.ventilation_power.label=Ventilation Power
+channel-type.mielecloud.ventilation_power.description=The current ventilation power of the hood.
+
+channel-type.mielecloud.ventilation_power_raw.label=Raw Ventilation Power
+channel-type.mielecloud.ventilation_power_raw.description=The current raw ventilation power of the hood.
+
+channel-type.mielecloud.error_state.label=Error
+channel-type.mielecloud.error_state.description=Indication flag which signals an error state for the device.
+
+channel-type.mielecloud.info_state.label=Info
+channel-type.mielecloud.info_state.description=Indication flag which signals an information of the device.
+
+channel-type.mielecloud.fridge_super_cool.label=Supercool
+channel-type.mielecloud.fridge_super_cool.description=Start the super cooling mode of the fridge.
+
+channel-type.mielecloud.freezer_super_freeze.label=Superfreeze
+channel-type.mielecloud.freezer_super_freeze.description=Start the super freezing mode of the freezer.
+
+channel-type.mielecloud.super_cool_can_be_controlled.label=Can Control Supercool
+channel-type.mielecloud.super_cool_can_be_controlled.description=Indicates if super cooling can be toggled.
+
+channel-type.mielecloud.super_freeze_can_be_controlled.label=Can Control Superfreeze
+channel-type.mielecloud.super_freeze_can_be_controlled.description=Indicates if super freezing can be toggled
+
+channel-type.mielecloud.fridge_temperature_target.label=Fridge Target Temperature
+channel-type.mielecloud.fridge_temperature_target.description=The target temperature of the fridge.
+
+channel-type.mielecloud.fridge_temperature_current.label=Current Fridge Temperature
+channel-type.mielecloud.fridge_temperature_current.description=The currently measured temperature of the fridge.
+
+channel-type.mielecloud.freezer_temperature_target.label=Freezer Target Temperature
+channel-type.mielecloud.freezer_temperature_target.description=The target temperature of the freezer.
+
+channel-type.mielecloud.freezer_temperature_current.label=Current Freezer Temperature
+channel-type.mielecloud.freezer_temperature_current.description=The currently measured temperature of the freezer.
+
+channel-type.mielecloud.top_temperature_target.label=Top Target Temperature
+channel-type.mielecloud.top_temperature_target.description=The target temperature of the top area.
+
+channel-type.mielecloud.top_temperature_current.label=Current Top Temperature
+channel-type.mielecloud.top_temperature_current.description=The currently measured temperature of the top area.
+
+channel-type.mielecloud.middle_temperature_target.label=Middle Target Temperature
+channel-type.mielecloud.middle_temperature_target.description=The target temperature of the middle area.
+
+channel-type.mielecloud.middle_temperature_current.label=Current Middle Temperature
+channel-type.mielecloud.middle_temperature_current.description=The currently measured temperature of the middle area.
+
+channel-type.mielecloud.bottom_temperature_target.label=Bottom Target Temperature
+channel-type.mielecloud.bottom_temperature_target.description=The target temperature of the bottom area.
+
+channel-type.mielecloud.bottom_temperature_current.label=Current Bottom Temperature
+channel-type.mielecloud.bottom_temperature_current.description=The currently measured temperature of the bottom area.
+
+channel-type.mielecloud.light_switch.label=Light Enabled
+channel-type.mielecloud.light_switch.description=Indicates if the light of the device is enabled.
+
+channel-type.mielecloud.light_can_be_controlled.label=Can Control Light
+channel-type.mielecloud.light_can_be_controlled.description=Indicates if the light of the device can be controlled.
+
+channel-type.mielecloud.plate_power_step.label=Plate Power Step
+channel-type.mielecloud.plate_power_step.description=The power level of the heating plate.
+
+channel-type.mielecloud.plate_power_step_raw.label=Raw Plate Power Step
+channel-type.mielecloud.plate_power_step_raw.description=The raw power level of the heating plate.
+
+channel-type.mielecloud.door_state.label=Door Signal
+channel-type.mielecloud.door_state.description=Indicates if the door of the device is open.
+
+channel-type.mielecloud.door_alarm.label=Door Alarm
+channel-type.mielecloud.door_alarm.description=Indicates if the door alarm of the device is active.
+
+channel-type.mielecloud.battery_level.label=Battery Level
+channel-type.mielecloud.battery_level.description=The battery level of the robotic vacuum cleaner.
+
+# Error message texts
+mielecloud.bridge.status.access.token.not.configured=The OAuth2 access token is not configured.
+mielecloud.bridge.status.account.not.authorized=The account has not been authorized. Please consult the documentation on how to do that.
+mielecloud.bridge.status.access.token.refresh.failed=Failed to refresh the OAuth2 access token. Please authorize the account again.
+mielecloud.bridge.status.invalid.email=The configured e-mail address has an invalid format.
+mielecloud.bridge.status.transient.http.error=An unexpected HTTP error occurred. Check the logs if this error persists.
+mielecloud.thing.status.webservice.missing=The Miele webservice cannot be accessed over the bridge. Check the bridge configuration.
+mielecloud.thing.status.removed=This Miele device has been removed from the Miele@home account.
+mielecloud.thing.status.ratelimit=The rate limit of the Miele cloud has been exceeded.
+mielecloud.thing.status.disconnected=This Miele device is not connected to the internet.
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/bridge.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/bridge.xml
new file mode 100644
index 00000000000..9da9db339e3
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/bridge.xml
@@ -0,0 +1,32 @@
+
+
+
+
+
+ @text/thing-type.mielecloud.account.label
+ @text/thing-type.mielecloud.account.description
+ WebService
+
+
+ Miele
+ Cloud Connector
+ INTERNET
+
+
+
+
+
+ email
+ @text/thing-type.config.mielecloud.account.email.label
+ @text/thing-type.config.mielecloud.account.email.description
+
+
+ @text/thing-type.config.mielecloud.account.locale.label
+ @text/thing-type.config.mielecloud.account.locale.description
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/channelTypes.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/channelTypes.xml
new file mode 100644
index 00000000000..e1cb16cca92
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/channelTypes.xml
@@ -0,0 +1,448 @@
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.remote_control_can_be_started.label
+ @text/channel-type.mielecloud.remote_control_can_be_started.description
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.remote_control_can_be_stopped.label
+ @text/channel-type.mielecloud.remote_control_can_be_stopped.description
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.remote_control_can_be_paused.label
+ @text/channel-type.mielecloud.remote_control_can_be_paused.description
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.remote_control_can_be_switched_on.label
+ @text/channel-type.mielecloud.remote_control_can_be_switched_on.description
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.remote_control_can_be_switched_off.label
+ @text/channel-type.mielecloud.remote_control_can_be_switched_off.description
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.remote_control_can_set_program_active.label
+ @text/channel-type.mielecloud.remote_control_can_set_program_active.description
+
+
+
+
+ String
+ @text/channel-type.mielecloud.spinning_speed.label
+ @text/channel-type.mielecloud.spinning_speed.description
+
+
+
+
+ Number
+ @text/channel-type.mielecloud.spinning_speed_raw.label
+ @text/channel-type.mielecloud.spinning_speed_raw.description
+
+
+
+
+ String
+ @text/channel-type.mielecloud.program_active.label
+ @text/channel-type.mielecloud.program_active.description
+
+
+
+
+ Number
+ @text/channel-type.mielecloud.program_active_raw.label
+ @text/channel-type.mielecloud.program_active_raw.description
+
+
+
+
+ String
+ @text/channel-type.mielecloud.dish_warmer_program_active.label
+ @text/channel-type.mielecloud.dish_warmer_program_active.description
+
+
+ @text/channel-option.mielecloud.dish_warmer_program_active.warming_cups_glasses
+ @text/channel-option.mielecloud.dish_warmer_program_active.warming_dishes_plates
+ @text/channel-option.mielecloud.dish_warmer_program_active.keeping_food_warm
+ @text/channel-option.mielecloud.dish_warmer_program_active.low_temperature_cooking
+
+
+
+
+
+ String
+ @text/channel-type.mielecloud.vacuum_cleaner_program_active.label
+ @text/channel-type.mielecloud.vacuum_cleaner_program_active.description
+
+
+ @text/channel-option.mielecloud.vacuum_cleaner_program_active.auto
+ @text/channel-option.mielecloud.vacuum_cleaner_program_active.spot
+ @text/channel-option.mielecloud.vacuum_cleaner_program_active.turbo
+ @text/channel-option.mielecloud.vacuum_cleaner_program_active.silent
+
+
+
+
+
+ String
+ @text/channel-type.mielecloud.program_phase.label
+ @text/channel-type.mielecloud.program_phase.description
+
+
+
+
+ Number
+ @text/channel-type.mielecloud.program_phase_raw.label
+ @text/channel-type.mielecloud.program_phase_raw.description
+
+
+
+
+ String
+ @text/channel-type.mielecloud.operation_state.label
+ @text/channel-type.mielecloud.operation_state.description
+
+
+
+
+ Number
+ @text/channel-type.mielecloud.operation_state_raw.label
+ @text/channel-type.mielecloud.operation_state_raw.description
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.program_start.label
+ @text/channel-type.mielecloud.program_start.description
+
+
+
+ Switch
+ @text/channel-type.mielecloud.program_stop.label
+ @text/channel-type.mielecloud.program_stop.description
+
+
+
+ String
+ @text/channel-type.mielecloud.program_start_stop.label
+ @text/channel-type.mielecloud.program_start_stop.description
+
+
+ @text/channel-option.mielecloud.program_start_stop.start
+ @text/channel-option.mielecloud.program_start_stop.stop
+
+
+
+
+
+ String
+ @text/channel-type.mielecloud.program_start_stop_pause.label
+ @text/channel-type.mielecloud.program_start_stop_pause.description
+
+
+ @text/channel-option.mielecloud.program_start_stop_pause.start
+ @text/channel-option.mielecloud.program_start_stop_pause.stop
+ @text/channel-option.mielecloud.program_start_stop_pause.pause
+
+
+
+
+
+ String
+ @text/channel-type.mielecloud.power_state_on_off.label
+ @text/channel-type.mielecloud.power_state_on_off.description
+
+
+ @text/channel-option.mielecloud.power_state_on_off.on
+ @text/channel-option.mielecloud.power_state_on_off.off
+
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.finish_state.label
+ @text/channel-type.mielecloud.finish_state.description
+ Alarm
+
+
+
+
+ Number
+ @text/channel-type.mielecloud.delayed_start_time.label
+ @text/channel-type.mielecloud.delayed_start_time.description
+ Number
+
+
+
+
+ Number
+ @text/channel-type.mielecloud.program_remaining_time.label
+ @text/channel-type.mielecloud.program_remaining_time.description
+ Number
+
+
+
+
+ Number
+ @text/channel-type.mielecloud.program_elapsed_time.label
+ @text/channel-type.mielecloud.program_elapsed_time.description
+ Number
+
+
+
+
+ Number
+ @text/channel-type.mielecloud.program_progress.label
+ @text/channel-type.mielecloud.program_progress.description
+ Number
+
+
+
+
+ String
+ @text/channel-type.mielecloud.drying_target.label
+ @text/channel-type.mielecloud.drying_target.description
+
+
+
+
+ Number
+ @text/channel-type.mielecloud.drying_target_raw.label
+ @text/channel-type.mielecloud.drying_target_raw.description
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.pre_heat_finished.label
+ @text/channel-type.mielecloud.pre_heat_finished.description
+ Alarm
+
+
+
+
+ Number:Temperature
+ @text/channel-type.mielecloud.temperature_target.label
+ @text/channel-type.mielecloud.temperature_target.description
+ Number:Temperature
+
+
+
+
+ Number:Temperature
+ @text/channel-type.mielecloud.temperature_current.label
+ @text/channel-type.mielecloud.temperature_current.description
+ Number:Temperature
+
+
+
+
+ String
+ @text/channel-type.mielecloud.ventilation_power.label
+ @text/channel-type.mielecloud.ventilation_power.description
+
+
+
+
+ Number
+ @text/channel-type.mielecloud.ventilation_power_raw.label
+ @text/channel-type.mielecloud.ventilation_power_raw.description
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.error_state.label
+ @text/channel-type.mielecloud.error_state.description
+ Alarm
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.info_state.label
+ @text/channel-type.mielecloud.info_state.description
+ Alarm
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.fridge_super_cool.label
+ @text/channel-type.mielecloud.fridge_super_cool.description
+ Switch
+
+
+
+ Switch
+ @text/channel-type.mielecloud.freezer_super_freeze.label
+ @text/channel-type.mielecloud.freezer_super_freeze.description
+ Switch
+
+
+
+ Switch
+ @text/channel-type.mielecloud.super_cool_can_be_controlled.label
+ @text/channel-type.mielecloud.super_cool_can_be_controlled.description
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.super_freeze_can_be_controlled.label
+ @text/channel-type.mielecloud.super_freeze_can_be_controlled.description
+
+
+
+
+ Number:Temperature
+ @text/channel-type.mielecloud.fridge_temperature_target.label
+ @text/channel-type.mielecloud.fridge_temperature_target.description
+ Number:Temperature
+
+
+
+
+ Number:Temperature
+ @text/channel-type.mielecloud.fridge_temperature_current.label
+ @text/channel-type.mielecloud.fridge_temperature_current.description
+ Number:Temperature
+
+
+
+
+ Number:Temperature
+ @text/channel-type.mielecloud.freezer_temperature_target.label
+ @text/channel-type.mielecloud.freezer_temperature_target.description
+ Number:Temperature
+
+
+
+
+ Number:Temperature
+ @text/channel-type.mielecloud.freezer_temperature_current.label
+ @text/channel-type.mielecloud.freezer_temperature_current.description
+ Number:Temperature
+
+
+
+
+ Number:Temperature
+ @text/channel-type.mielecloud.top_temperature_target.label
+ @text/channel-type.mielecloud.top_temperature_target.description
+ Number:Temperature
+
+
+
+
+ Number:Temperature
+ @text/channel-type.mielecloud.top_temperature_current.label
+ @text/channel-type.mielecloud.top_temperature_current.description
+ Number:Temperature
+
+
+
+
+ Number:Temperature
+ @text/channel-type.mielecloud.middle_temperature_target.label
+ @text/channel-type.mielecloud.middle_temperature_target.description
+ Number:Temperature
+
+
+
+
+ Number:Temperature
+ @text/channel-type.mielecloud.middle_temperature_current.label
+ @text/channel-type.mielecloud.middle_temperature_current.description
+ Number:Temperature
+
+
+
+
+ Number:Temperature
+ @text/channel-type.mielecloud.bottom_temperature_target.label
+ @text/channel-type.mielecloud.bottom_temperature_target.description
+ Number:Temperature
+
+
+
+
+ Number:Temperature
+ @text/channel-type.mielecloud.bottom_temperature_current.label
+ @text/channel-type.mielecloud.bottom_temperature_current.description
+ Number:Temperature
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.light_switch.label
+ @text/channel-type.mielecloud.light_switch.description
+ Switch
+
+
+
+ Switch
+ @text/channel-type.mielecloud.light_can_be_controlled.label
+ @text/channel-type.mielecloud.light_can_be_controlled.description
+
+
+
+
+ String
+ @text/channel-type.mielecloud.plate_power_step.label
+ @text/channel-type.mielecloud.plate_power_step.description
+
+
+
+
+ Number
+ @text/channel-type.mielecloud.plate_power_step_raw.label
+ @text/channel-type.mielecloud.plate_power_step_raw.description
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.door_state.label
+ @text/channel-type.mielecloud.door_state.description
+
+
+
+
+ Switch
+ @text/channel-type.mielecloud.door_alarm.label
+ @text/channel-type.mielecloud.door_alarm.description
+ Alarm
+
+
+
+
+ Number
+ @text/channel-type.mielecloud.battery_level.label
+ @text/channel-type.mielecloud.battery_level.description
+ Battery
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/coffeeSystem.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/coffeeSystem.xml
new file mode 100644
index 00000000000..dae30e6ed9a
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/coffeeSystem.xml
@@ -0,0 +1,44 @@
+
+
+
+
+
+
+
+
+ @text/thing-type.mielecloud.coffee_system.label
+ @text/thing-type.mielecloud.coffee_system.description
+ WhiteGood
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Miele
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/dishWarmerDevice.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/dishWarmerDevice.xml
new file mode 100644
index 00000000000..7ddf1e37c5b
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/dishWarmerDevice.xml
@@ -0,0 +1,42 @@
+
+
+
+
+
+
+
+
+ @text/thing-type.mielecloud.dish_warmer.label
+ @text/thing-type.mielecloud.dish_warmer.description
+ WhiteGood
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Miele
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/dishwasherDevice.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/dishwasherDevice.xml
new file mode 100644
index 00000000000..2d97427f21b
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/dishwasherDevice.xml
@@ -0,0 +1,46 @@
+
+
+
+
+
+
+
+
+ @text/thing-type.mielecloud.dishwasher.label
+ @text/thing-type.mielecloud.dishwasher.description
+ WhiteGood
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Miele
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/dryerDevice.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/dryerDevice.xml
new file mode 100644
index 00000000000..0522f002100
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/dryerDevice.xml
@@ -0,0 +1,50 @@
+
+
+
+
+
+
+
+
+ @text/thing-type.mielecloud.dryer.label
+ @text/thing-type.mielecloud.dryer.description
+ WhiteGood
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Miele
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/freezer.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/freezer.xml
new file mode 100644
index 00000000000..6f331fca0b6
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/freezer.xml
@@ -0,0 +1,36 @@
+
+
+
+
+
+
+
+
+ @text/thing-type.mielecloud.freezer.label
+ @text/thing-type.mielecloud.freezer.description
+ WhiteGood
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Miele
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/fridge.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/fridge.xml
new file mode 100644
index 00000000000..6fe4f85bcf6
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/fridge.xml
@@ -0,0 +1,36 @@
+
+
+
+
+
+
+
+
+ @text/thing-type.mielecloud.fridge.label
+ @text/thing-type.mielecloud.fridge.description
+ WhiteGood
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Miele
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/fridgeFreezer.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/fridgeFreezer.xml
new file mode 100644
index 00000000000..2e8d56ba860
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/fridgeFreezer.xml
@@ -0,0 +1,40 @@
+
+
+
+
+
+
+
+
+ @text/thing-type.mielecloud.fridge_freezer.label
+ @text/thing-type.mielecloud.fridge_freezer.description
+ WhiteGood
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Miele
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/hobDevice.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/hobDevice.xml
new file mode 100644
index 00000000000..83db9fc9c15
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/hobDevice.xml
@@ -0,0 +1,42 @@
+
+
+
+
+
+
+
+
+ @text/thing-type.mielecloud.hob.label
+ @text/thing-type.mielecloud.hob.description
+ WhiteGood
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Miele
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/hoodDevice.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/hoodDevice.xml
new file mode 100644
index 00000000000..78c36f18a28
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/hoodDevice.xml
@@ -0,0 +1,40 @@
+
+
+
+
+
+
+
+
+ @text/thing-type.mielecloud.hood.label
+ @text/thing-type.mielecloud.hood.description
+ WhiteGood
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Miele
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/ovenDevice.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/ovenDevice.xml
new file mode 100644
index 00000000000..abedb9f50c6
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/ovenDevice.xml
@@ -0,0 +1,51 @@
+
+
+
+
+
+
+
+
+ @text/thing-type.mielecloud.oven.label
+ @text/thing-type.mielecloud.oven.description
+ WhiteGood
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Miele
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/roboticVacuumCleanerDevice.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/roboticVacuumCleanerDevice.xml
new file mode 100644
index 00000000000..ec01510b542
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/roboticVacuumCleanerDevice.xml
@@ -0,0 +1,40 @@
+
+
+
+
+
+
+
+
+ @text/thing-type.mielecloud.robotic_vacuum_cleaner.label
+ @text/thing-type.mielecloud.robotic_vacuum_cleaner.description
+ WhiteGood
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Miele
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/washerDryer.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/washerDryer.xml
new file mode 100644
index 00000000000..21f21cb0e66
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/washerDryer.xml
@@ -0,0 +1,53 @@
+
+
+
+
+
+
+
+
+ @text/thing-type.mielecloud.washer_dryer.label
+ @text/thing-type.mielecloud.washer_dryer.description
+ WhiteGood
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Miele
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/washingMachine.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/washingMachine.xml
new file mode 100644
index 00000000000..8143ebb70f0
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/washingMachine.xml
@@ -0,0 +1,51 @@
+
+
+
+
+
+
+
+
+ @text/thing-type.mielecloud.washing_machine.label
+ @text/thing-type.mielecloud.washing_machine.description
+ WhiteGood
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Miele
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/wineStorageDevice.xml b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/wineStorageDevice.xml
new file mode 100644
index 00000000000..cf9896025b0
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/OH-INF/thing/wineStorageDevice.xml
@@ -0,0 +1,43 @@
+
+
+
+
+
+
+
+
+ @text/thing-type.mielecloud.wine_storage.label
+ @text/thing-type.mielecloud.wine_storage.description
+ WhiteGood
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Miele
+
+
+
+
+
+
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/css/main.css b/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/css/main.css
new file mode 100755
index 00000000000..e51da11735a
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/css/main.css
@@ -0,0 +1,15023 @@
+/*! normalize.css v5.0.0 | MIT License | github.com/necolas/normalize.css */
+/**
+ * 1. Change the default font family in all browsers (opinionated).
+ * 2. Correct the line height in all browsers.
+ * 3. Prevent adjustments of font size after orientation changes in
+ * IE on Windows Phone and in iOS.
+ */
+/* Document
+ ========================================================================== */
+/* line 13, src/assets/scss/vendors/_normalize.scss */
+html {
+ font-family: sans-serif;
+ /* 1 */
+ line-height: 1.15;
+ /* 2 */
+ -ms-text-size-adjust: 100%;
+ /* 3 */
+ -webkit-text-size-adjust: 100%;
+ /* 3 */
+}
+
+/* Sections
+ ========================================================================== */
+/**
+ * Remove the margin in all browsers (opinionated).
+ */
+/* line 27, src/assets/scss/vendors/_normalize.scss */
+body {
+ margin: 0;
+}
+
+/**
+ * Add the correct display in IE 9-.
+ */
+/* line 35, src/assets/scss/vendors/_normalize.scss */
+article,
+aside,
+footer,
+header,
+nav,
+section {
+ display: block;
+}
+
+/**
+ * Correct the font size and margin on `h1` elements within `section` and
+ * `article` contexts in Chrome, Firefox, and Safari.
+ */
+/* line 49, src/assets/scss/vendors/_normalize.scss */
+h1 {
+ font-size: 2em;
+ margin: 0.67em 0;
+}
+
+/* Grouping content
+ ========================================================================== */
+/**
+ * Add the correct display in IE 9-.
+ * 1. Add the correct display in IE.
+ */
+/* line 62, src/assets/scss/vendors/_normalize.scss */
+figcaption,
+figure,
+main {
+ /* 1 */
+ display: block;
+}
+
+/**
+ * Add the correct margin in IE 8.
+ */
+/* line 72, src/assets/scss/vendors/_normalize.scss */
+figure {
+ margin: 1em 40px;
+}
+
+/**
+ * 1. Add the correct box sizing in Firefox.
+ * 2. Show the overflow in Edge and IE.
+ */
+/* line 81, src/assets/scss/vendors/_normalize.scss */
+hr {
+ -webkit-box-sizing: content-box;
+ box-sizing: content-box;
+ /* 1 */
+ height: 0;
+ /* 1 */
+ overflow: visible;
+ /* 2 */
+}
+
+/**
+ * 1. Correct the inheritance and scaling of font size in all browsers.
+ * 2. Correct the odd `em` font sizing in all browsers.
+ */
+/* line 92, src/assets/scss/vendors/_normalize.scss */
+pre {
+ font-family: monospace, monospace;
+ /* 1 */
+ font-size: 1em;
+ /* 2 */
+}
+
+/* Text-level semantics
+ ========================================================================== */
+/**
+ * 1. Remove the gray background on active links in IE 10.
+ * 2. Remove gaps in links underline in iOS 8+ and Safari 8+.
+ */
+/* line 105, src/assets/scss/vendors/_normalize.scss */
+a {
+ background-color: transparent;
+ /* 1 */
+ -webkit-text-decoration-skip: objects;
+ /* 2 */
+}
+
+/**
+ * Remove the outline on focused links when they are also active or hovered
+ * in all browsers (opinionated).
+ */
+/* line 115, src/assets/scss/vendors/_normalize.scss */
+a:active,
+a:hover {
+ outline-width: 0;
+}
+
+/**
+ * 1. Remove the bottom border in Firefox 39-.
+ * 2. Add the correct text decoration in Chrome, Edge, IE, Opera, and Safari.
+ */
+/* line 125, src/assets/scss/vendors/_normalize.scss */
+abbr[title] {
+ border-bottom: none;
+ /* 1 */
+ text-decoration: underline;
+ /* 2 */
+ -webkit-text-decoration: underline dotted;
+ text-decoration: underline dotted;
+ /* 2 */
+}
+
+/**
+ * Prevent the duplicate application of `bolder` by the next rule in Safari 6.
+ */
+/* line 135, src/assets/scss/vendors/_normalize.scss */
+b,
+strong {
+ font-weight: inherit;
+}
+
+/**
+ * Add the correct font weight in Chrome, Edge, and Safari.
+ */
+/* line 144, src/assets/scss/vendors/_normalize.scss */
+b,
+strong {
+ font-weight: bolder;
+}
+
+/**
+ * 1. Correct the inheritance and scaling of font size in all browsers.
+ * 2. Correct the odd `em` font sizing in all browsers.
+ */
+/* line 154, src/assets/scss/vendors/_normalize.scss */
+code,
+kbd,
+samp {
+ font-family: monospace, monospace;
+ /* 1 */
+ font-size: 1em;
+ /* 2 */
+}
+
+/**
+ * Add the correct font style in Android 4.3-.
+ */
+/* line 165, src/assets/scss/vendors/_normalize.scss */
+dfn {
+ font-style: italic;
+}
+
+/**
+ * Add the correct background and color in IE 9-.
+ */
+/* line 173, src/assets/scss/vendors/_normalize.scss */
+mark {
+ background-color: #ff0;
+ color: #000;
+}
+
+/**
+ * Add the correct font size in all browsers.
+ */
+/* line 182, src/assets/scss/vendors/_normalize.scss */
+small {
+ font-size: 80%;
+}
+
+/**
+ * Prevent `sub` and `sup` elements from affecting the line height in
+ * all browsers.
+ */
+/* line 191, src/assets/scss/vendors/_normalize.scss */
+sub,
+sup {
+ font-size: 75%;
+ line-height: 0;
+ position: relative;
+ vertical-align: baseline;
+}
+
+/* line 199, src/assets/scss/vendors/_normalize.scss */
+sub {
+ bottom: -0.25em;
+}
+
+/* line 203, src/assets/scss/vendors/_normalize.scss */
+sup {
+ top: -0.5em;
+}
+
+/* Embedded content
+ ========================================================================== */
+/**
+ * Add the correct display in IE 9-.
+ */
+/* line 214, src/assets/scss/vendors/_normalize.scss */
+audio,
+video {
+ display: inline-block;
+}
+
+/**
+ * Add the correct display in iOS 4-7.
+ */
+/* line 223, src/assets/scss/vendors/_normalize.scss */
+audio:not([controls]) {
+ display: none;
+ height: 0;
+}
+
+/**
+ * Remove the border on images inside links in IE 10-.
+ */
+/* line 232, src/assets/scss/vendors/_normalize.scss */
+img {
+ border-style: none;
+}
+
+/**
+ * Hide the overflow in IE.
+ */
+/* line 240, src/assets/scss/vendors/_normalize.scss */
+svg:not(:root) {
+ overflow: hidden;
+}
+
+/* Forms
+ ========================================================================== */
+/**
+ * 1. Change the font styles in all browsers (opinionated).
+ * 2. Remove the margin in Firefox and Safari.
+ */
+/* line 252, src/assets/scss/vendors/_normalize.scss */
+button,
+input,
+optgroup,
+select,
+textarea {
+ font-family: sans-serif;
+ /* 1 */
+ font-size: 100%;
+ /* 1 */
+ line-height: 1.15;
+ /* 1 */
+ margin: 0;
+ /* 2 */
+}
+
+/**
+ * Show the overflow in IE.
+ * 1. Show the overflow in Edge.
+ */
+/* line 268, src/assets/scss/vendors/_normalize.scss */
+button,
+input {
+ /* 1 */
+ overflow: visible;
+}
+
+/**
+ * Remove the inheritance of text transform in Edge, Firefox, and IE.
+ * 1. Remove the inheritance of text transform in Firefox.
+ */
+/* line 278, src/assets/scss/vendors/_normalize.scss */
+button,
+select {
+ /* 1 */
+ text-transform: none;
+}
+
+/**
+ * 1. Prevent a WebKit bug where (2) destroys native `audio` and `video`
+ * controls in Android 4.
+ * 2. Correct the inability to style clickable types in iOS and Safari.
+ */
+/* line 289, src/assets/scss/vendors/_normalize.scss */
+button,
+html [type="button"],
+[type="reset"],
+[type="submit"] {
+ -webkit-appearance: button;
+ /* 2 */
+}
+
+/**
+ * Remove the inner border and padding in Firefox.
+ */
+/* line 300, src/assets/scss/vendors/_normalize.scss */
+button::-moz-focus-inner,
+[type="button"]::-moz-focus-inner,
+[type="reset"]::-moz-focus-inner,
+[type="submit"]::-moz-focus-inner {
+ border-style: none;
+ padding: 0;
+}
+
+/**
+ * Restore the focus styles unset by the previous rule.
+ */
+/* line 312, src/assets/scss/vendors/_normalize.scss */
+button:-moz-focusring,
+[type="button"]:-moz-focusring,
+[type="reset"]:-moz-focusring,
+[type="submit"]:-moz-focusring {
+ outline: 1px dotted ButtonText;
+}
+
+/**
+ * Change the border, margin, and padding in all browsers (opinionated).
+ */
+/* line 323, src/assets/scss/vendors/_normalize.scss */
+fieldset {
+ border: 1px solid #c0c0c0;
+ margin: 0 2px;
+ padding: 0.35em 0.625em 0.75em;
+}
+
+/**
+ * 1. Correct the text wrapping in Edge and IE.
+ * 2. Correct the color inheritance from `fieldset` elements in IE.
+ * 3. Remove the padding so developers are not caught out when they zero out
+ * `fieldset` elements in all browsers.
+ */
+/* line 336, src/assets/scss/vendors/_normalize.scss */
+legend {
+ -webkit-box-sizing: border-box;
+ box-sizing: border-box;
+ /* 1 */
+ color: inherit;
+ /* 2 */
+ display: table;
+ /* 1 */
+ max-width: 100%;
+ /* 1 */
+ padding: 0;
+ /* 3 */
+ white-space: normal;
+ /* 1 */
+}
+
+/**
+ * 1. Add the correct display in IE 9-.
+ * 2. Add the correct vertical alignment in Chrome, Firefox, and Opera.
+ */
+/* line 350, src/assets/scss/vendors/_normalize.scss */
+progress {
+ display: inline-block;
+ /* 1 */
+ vertical-align: baseline;
+ /* 2 */
+}
+
+/**
+ * Remove the default vertical scrollbar in IE.
+ */
+/* line 359, src/assets/scss/vendors/_normalize.scss */
+textarea {
+ overflow: auto;
+}
+
+/**
+ * 1. Add the correct box sizing in IE 10-.
+ * 2. Remove the padding in IE 10-.
+ */
+/* line 368, src/assets/scss/vendors/_normalize.scss */
+[type="checkbox"],
+[type="radio"] {
+ -webkit-box-sizing: border-box;
+ box-sizing: border-box;
+ /* 1 */
+ padding: 0;
+ /* 2 */
+}
+
+/**
+ * Correct the cursor style of increment and decrement buttons in Chrome.
+ */
+/* line 378, src/assets/scss/vendors/_normalize.scss */
+[type="number"]::-webkit-inner-spin-button,
+[type="number"]::-webkit-outer-spin-button {
+ height: auto;
+}
+
+/**
+ * 1. Correct the odd appearance in Chrome and Safari.
+ * 2. Correct the outline style in Safari.
+ */
+/* line 388, src/assets/scss/vendors/_normalize.scss */
+[type="search"] {
+ -webkit-appearance: textfield;
+ /* 1 */
+ outline-offset: -2px;
+ /* 2 */
+}
+
+/**
+ * Remove the inner padding and cancel buttons in Chrome and Safari on macOS.
+ */
+/* line 397, src/assets/scss/vendors/_normalize.scss */
+[type="search"]::-webkit-search-cancel-button,
+[type="search"]::-webkit-search-decoration {
+ -webkit-appearance: none;
+}
+
+/**
+ * 1. Correct the inability to style clickable types in iOS and Safari.
+ * 2. Change font properties to `inherit` in Safari.
+ */
+/* line 407, src/assets/scss/vendors/_normalize.scss */
+::-webkit-file-upload-button {
+ -webkit-appearance: button;
+ /* 1 */
+ font: inherit;
+ /* 2 */
+}
+
+/* Interactive
+ ========================================================================== */
+/*
+ * Add the correct display in IE 9-.
+ * 1. Add the correct display in Edge, IE, and Firefox.
+ */
+/* line 420, src/assets/scss/vendors/_normalize.scss */
+details,
+menu {
+ display: block;
+}
+
+/*
+ * Add the correct display in all browsers.
+ */
+/* line 429, src/assets/scss/vendors/_normalize.scss */
+summary {
+ display: list-item;
+}
+
+/* Scripting
+ ========================================================================== */
+/**
+ * Add the correct display in IE 9-.
+ */
+/* line 440, src/assets/scss/vendors/_normalize.scss */
+canvas {
+ display: inline-block;
+}
+
+/**
+ * Add the correct display in IE.
+ */
+/* line 448, src/assets/scss/vendors/_normalize.scss */
+template {
+ display: none;
+}
+
+/* Hidden
+ ========================================================================== */
+/**
+ * Add the correct display in IE 10-.
+ */
+/* line 459, src/assets/scss/vendors/_normalize.scss */
+[hidden] {
+ display: none;
+}
+
+/*!
+ * Bootstrap v4.5.2 (https://getbootstrap.com/)
+ * Copyright 2011-2020 The Bootstrap Authors
+ * Copyright 2011-2020 Twitter, Inc.
+ * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)
+ */
+/* line 2, node_modules/bootstrap/scss/_root.scss */
+:root {
+ --blue: #6fa7fd;
+ --indigo: #6610f2;
+ --purple: #6f42c1;
+ --pink: #e83e8c;
+ --red: #e54a19;
+ --orange: #fd7e14;
+ --yellow: #ffc107;
+ --green: #28a745;
+ --teal: #20c997;
+ --cyan: #17a2b8;
+ --white: #ffffff;
+ --gray: #6c757d;
+ --gray-dark: #343a40;
+ --primary: #464746;
+ --secondary: #f0ebe3;
+ --success: #f0ebe3;
+ --info: #464746;
+ --warning: #464746;
+ --danger: #e54a19;
+ --light: #f7f7f7;
+ --dark: #343a40;
+ --breakpoint-xs: 0;
+ --breakpoint-sm: 576px;
+ --breakpoint-md: 768px;
+ --breakpoint-lg: 1024px;
+ --breakpoint-xl: 1280px;
+ --breakpoint-xxl: 1440px;
+ --font-family-sans-serif: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";
+ --font-family-monospace: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;
+}
+
+/* line 19, node_modules/bootstrap/scss/_reboot.scss */
+*,
+*::before,
+*::after {
+ -webkit-box-sizing: border-box;
+ box-sizing: border-box;
+}
+
+/* line 25, node_modules/bootstrap/scss/_reboot.scss */
+html {
+ font-family: sans-serif;
+ line-height: 1.15;
+ -webkit-text-size-adjust: 100%;
+ -webkit-tap-highlight-color: rgba(0, 0, 0, 0);
+}
+
+/* line 35, node_modules/bootstrap/scss/_reboot.scss */
+article, aside, figcaption, figure, footer, header, hgroup, main, nav, section {
+ display: block;
+}
+
+/* line 46, node_modules/bootstrap/scss/_reboot.scss */
+body {
+ margin: 0;
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";
+ font-size: 1rem;
+ font-weight: 300;
+ line-height: 1.5;
+ color: #464746;
+ text-align: left;
+ background-color: #ffffff;
+}
+
+/* line 66, node_modules/bootstrap/scss/_reboot.scss */
+[tabindex="-1"]:focus:not(:focus-visible) {
+ outline: 0 !important;
+}
+
+/* line 76, node_modules/bootstrap/scss/_reboot.scss */
+hr {
+ -webkit-box-sizing: content-box;
+ box-sizing: content-box;
+ height: 0;
+ overflow: visible;
+}
+
+/* line 92, node_modules/bootstrap/scss/_reboot.scss */
+h1, h2, h3, h4, h5, h6 {
+ margin-top: 0;
+ margin-bottom: 0.5rem;
+}
+
+/* line 101, node_modules/bootstrap/scss/_reboot.scss */
+p {
+ margin-top: 0;
+ margin-bottom: 1rem;
+}
+
+/* line 114, node_modules/bootstrap/scss/_reboot.scss */
+abbr[title],
+abbr[data-original-title] {
+ text-decoration: underline;
+ -webkit-text-decoration: underline dotted;
+ text-decoration: underline dotted;
+ cursor: help;
+ border-bottom: 0;
+ -webkit-text-decoration-skip-ink: none;
+ text-decoration-skip-ink: none;
+}
+
+/* line 123, node_modules/bootstrap/scss/_reboot.scss */
+address {
+ margin-bottom: 1rem;
+ font-style: normal;
+ line-height: inherit;
+}
+
+/* line 129, node_modules/bootstrap/scss/_reboot.scss */
+ol,
+ul,
+dl {
+ margin-top: 0;
+ margin-bottom: 1rem;
+}
+
+/* line 136, node_modules/bootstrap/scss/_reboot.scss */
+ol ol,
+ul ul,
+ol ul,
+ul ol {
+ margin-bottom: 0;
+}
+
+/* line 143, node_modules/bootstrap/scss/_reboot.scss */
+dt {
+ font-weight: 700;
+}
+
+/* line 147, node_modules/bootstrap/scss/_reboot.scss */
+dd {
+ margin-bottom: .5rem;
+ margin-left: 0;
+}
+
+/* line 152, node_modules/bootstrap/scss/_reboot.scss */
+blockquote {
+ margin: 0 0 1rem;
+}
+
+/* line 156, node_modules/bootstrap/scss/_reboot.scss */
+b,
+strong {
+ font-weight: 800;
+}
+
+/* line 161, node_modules/bootstrap/scss/_reboot.scss */
+small {
+ font-size: 80%;
+}
+
+/* line 170, node_modules/bootstrap/scss/_reboot.scss */
+sub,
+sup {
+ position: relative;
+ font-size: 75%;
+ line-height: 0;
+ vertical-align: baseline;
+}
+
+/* line 178, node_modules/bootstrap/scss/_reboot.scss */
+sub {
+ bottom: -.25em;
+}
+
+/* line 179, node_modules/bootstrap/scss/_reboot.scss */
+sup {
+ top: -.5em;
+}
+
+/* line 186, node_modules/bootstrap/scss/_reboot.scss */
+a {
+ color: #464746;
+ text-decoration: none;
+ background-color: transparent;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+a:hover {
+ color: #202020;
+ text-decoration: underline;
+}
+
+/* line 202, node_modules/bootstrap/scss/_reboot.scss */
+a:not([href]):not([class]) {
+ color: inherit;
+ text-decoration: none;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+a:not([href]):not([class]):hover {
+ color: inherit;
+ text-decoration: none;
+}
+
+/* line 217, node_modules/bootstrap/scss/_reboot.scss */
+pre,
+code,
+kbd,
+samp {
+ font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;
+ font-size: 1em;
+}
+
+/* line 225, node_modules/bootstrap/scss/_reboot.scss */
+pre {
+ margin-top: 0;
+ margin-bottom: 1rem;
+ overflow: auto;
+ -ms-overflow-style: scrollbar;
+}
+
+/* line 242, node_modules/bootstrap/scss/_reboot.scss */
+figure {
+ margin: 0 0 1rem;
+}
+
+/* line 252, node_modules/bootstrap/scss/_reboot.scss */
+img {
+ vertical-align: middle;
+ border-style: none;
+}
+
+/* line 257, node_modules/bootstrap/scss/_reboot.scss */
+svg {
+ overflow: hidden;
+ vertical-align: middle;
+}
+
+/* line 269, node_modules/bootstrap/scss/_reboot.scss */
+table {
+ border-collapse: collapse;
+}
+
+/* line 273, node_modules/bootstrap/scss/_reboot.scss */
+caption {
+ padding-top: 0.75rem;
+ padding-bottom: 0.75rem;
+ color: #6c757d;
+ text-align: left;
+ caption-side: bottom;
+}
+
+/* line 281, node_modules/bootstrap/scss/_reboot.scss */
+th {
+ text-align: inherit;
+}
+
+/* line 292, node_modules/bootstrap/scss/_reboot.scss */
+label {
+ display: inline-block;
+ margin-bottom: 0.5rem;
+}
+
+/* line 301, node_modules/bootstrap/scss/_reboot.scss */
+button {
+ border-radius: 0;
+}
+
+/* line 310, node_modules/bootstrap/scss/_reboot.scss */
+button:focus {
+ outline: 1px dotted;
+ outline: 5px auto -webkit-focus-ring-color;
+}
+
+/* line 315, node_modules/bootstrap/scss/_reboot.scss */
+input,
+button,
+select,
+optgroup,
+textarea {
+ margin: 0;
+ font-family: inherit;
+ font-size: inherit;
+ line-height: inherit;
+}
+
+/* line 326, node_modules/bootstrap/scss/_reboot.scss */
+button,
+input {
+ overflow: visible;
+}
+
+/* line 331, node_modules/bootstrap/scss/_reboot.scss */
+button,
+select {
+ text-transform: none;
+}
+
+/* line 339, node_modules/bootstrap/scss/_reboot.scss */
+[role="button"] {
+ cursor: pointer;
+}
+
+/* line 346, node_modules/bootstrap/scss/_reboot.scss */
+select {
+ word-wrap: normal;
+}
+
+/* line 354, node_modules/bootstrap/scss/_reboot.scss */
+button,
+[type="button"],
+[type="reset"],
+[type="submit"] {
+ -webkit-appearance: button;
+}
+
+/* line 367, node_modules/bootstrap/scss/_reboot.scss */
+button:not(:disabled),
+[type="button"]:not(:disabled),
+[type="reset"]:not(:disabled),
+[type="submit"]:not(:disabled) {
+ cursor: pointer;
+}
+
+/* line 374, node_modules/bootstrap/scss/_reboot.scss */
+button::-moz-focus-inner,
+[type="button"]::-moz-focus-inner,
+[type="reset"]::-moz-focus-inner,
+[type="submit"]::-moz-focus-inner {
+ padding: 0;
+ border-style: none;
+}
+
+/* line 382, node_modules/bootstrap/scss/_reboot.scss */
+input[type="radio"],
+input[type="checkbox"] {
+ -webkit-box-sizing: border-box;
+ box-sizing: border-box;
+ padding: 0;
+}
+
+/* line 389, node_modules/bootstrap/scss/_reboot.scss */
+textarea {
+ overflow: auto;
+ resize: vertical;
+}
+
+/* line 395, node_modules/bootstrap/scss/_reboot.scss */
+fieldset {
+ min-width: 0;
+ padding: 0;
+ margin: 0;
+ border: 0;
+}
+
+/* line 410, node_modules/bootstrap/scss/_reboot.scss */
+legend {
+ display: block;
+ width: 100%;
+ max-width: 100%;
+ padding: 0;
+ margin-bottom: .5rem;
+ font-size: 1.5rem;
+ line-height: inherit;
+ color: inherit;
+ white-space: normal;
+}
+
+/* line 422, node_modules/bootstrap/scss/_reboot.scss */
+progress {
+ vertical-align: baseline;
+}
+
+/* line 427, node_modules/bootstrap/scss/_reboot.scss */
+[type="number"]::-webkit-inner-spin-button,
+[type="number"]::-webkit-outer-spin-button {
+ height: auto;
+}
+
+/* line 432, node_modules/bootstrap/scss/_reboot.scss */
+[type="search"] {
+ outline-offset: -2px;
+ -webkit-appearance: none;
+}
+
+/* line 445, node_modules/bootstrap/scss/_reboot.scss */
+[type="search"]::-webkit-search-decoration {
+ -webkit-appearance: none;
+}
+
+/* line 454, node_modules/bootstrap/scss/_reboot.scss */
+::-webkit-file-upload-button {
+ font: inherit;
+ -webkit-appearance: button;
+}
+
+/* line 463, node_modules/bootstrap/scss/_reboot.scss */
+output {
+ display: inline-block;
+}
+
+/* line 467, node_modules/bootstrap/scss/_reboot.scss */
+summary {
+ display: list-item;
+ cursor: pointer;
+}
+
+/* line 472, node_modules/bootstrap/scss/_reboot.scss */
+template {
+ display: none;
+}
+
+/* line 478, node_modules/bootstrap/scss/_reboot.scss */
+[hidden] {
+ display: none !important;
+}
+
+/* line 7, node_modules/bootstrap/scss/_type.scss */
+h1, h2, h3, h4, h5, h6,
+.h1, .h2, .h3, .h4, .h5, .h6 {
+ margin-bottom: 0.5rem;
+ font-weight: 500;
+ line-height: 1.2;
+}
+
+/* line 16, node_modules/bootstrap/scss/_type.scss */
+h1, .h1 {
+ font-size: 4rem;
+}
+
+/* line 17, node_modules/bootstrap/scss/_type.scss */
+h2, .h2 {
+ font-size: 2.4375rem;
+}
+
+/* line 18, node_modules/bootstrap/scss/_type.scss */
+h3, .h3 {
+ font-size: 1.5rem;
+}
+
+/* line 19, node_modules/bootstrap/scss/_type.scss */
+h4, .h4 {
+ font-size: 0.9375rem;
+}
+
+/* line 20, node_modules/bootstrap/scss/_type.scss */
+h5, .h5 {
+ font-size: 0.75rem;
+}
+
+/* line 21, node_modules/bootstrap/scss/_type.scss */
+h6, .h6 {
+ font-size: 0.6875rem;
+}
+
+/* line 23, node_modules/bootstrap/scss/_type.scss */
+.lead {
+ font-size: 1.25rem;
+ font-weight: 300;
+}
+
+/* line 29, node_modules/bootstrap/scss/_type.scss */
+.display-1 {
+ font-size: 6rem;
+ font-weight: 300;
+ line-height: 1.2;
+}
+
+/* line 34, node_modules/bootstrap/scss/_type.scss */
+.display-2 {
+ font-size: 5.5rem;
+ font-weight: 300;
+ line-height: 1.2;
+}
+
+/* line 39, node_modules/bootstrap/scss/_type.scss */
+.display-3 {
+ font-size: 4.5rem;
+ font-weight: 300;
+ line-height: 1.2;
+}
+
+/* line 44, node_modules/bootstrap/scss/_type.scss */
+.display-4 {
+ font-size: 3.5rem;
+ font-weight: 300;
+ line-height: 1.2;
+}
+
+/* line 55, node_modules/bootstrap/scss/_type.scss */
+hr {
+ margin-top: 1rem;
+ margin-bottom: 1rem;
+ border: 0;
+ border-top: 1px solid rgba(0, 0, 0, 0.1);
+}
+
+/* line 67, node_modules/bootstrap/scss/_type.scss */
+small,
+.small {
+ font-size: 80%;
+ font-weight: 400;
+}
+
+/* line 73, node_modules/bootstrap/scss/_type.scss */
+mark,
+.mark {
+ padding: 0.2em;
+ background-color: #fcf8e3;
+}
+
+/* line 84, node_modules/bootstrap/scss/_type.scss */
+.list-unstyled {
+ padding-left: 0;
+ list-style: none;
+}
+
+/* line 89, node_modules/bootstrap/scss/_type.scss */
+.list-inline {
+ padding-left: 0;
+ list-style: none;
+}
+
+/* line 92, node_modules/bootstrap/scss/_type.scss */
+.list-inline-item {
+ display: inline-block;
+}
+
+/* line 95, node_modules/bootstrap/scss/_type.scss */
+.list-inline-item:not(:last-child) {
+ margin-right: 0.5rem;
+}
+
+/* line 106, node_modules/bootstrap/scss/_type.scss */
+.initialism {
+ font-size: 90%;
+ text-transform: uppercase;
+}
+
+/* line 112, node_modules/bootstrap/scss/_type.scss */
+.blockquote {
+ margin-bottom: 1rem;
+ font-size: 1.25rem;
+}
+
+/* line 117, node_modules/bootstrap/scss/_type.scss */
+.blockquote-footer {
+ display: block;
+ font-size: 80%;
+ color: #6c757d;
+}
+
+/* line 122, node_modules/bootstrap/scss/_type.scss */
+.blockquote-footer::before {
+ content: "\2014\00A0";
+}
+
+/* line 8, node_modules/bootstrap/scss/_images.scss */
+.img-fluid {
+ max-width: 100%;
+ height: auto;
+}
+
+/* line 14, node_modules/bootstrap/scss/_images.scss */
+.img-thumbnail {
+ padding: 0.25rem;
+ background-color: #ffffff;
+ border: 1px solid #dee2e6;
+ border-radius: 0.25rem;
+ max-width: 100%;
+ height: auto;
+}
+
+/* line 29, node_modules/bootstrap/scss/_images.scss */
+.figure {
+ display: inline-block;
+}
+
+/* line 34, node_modules/bootstrap/scss/_images.scss */
+.figure-img {
+ margin-bottom: 0.5rem;
+ line-height: 1;
+}
+
+/* line 39, node_modules/bootstrap/scss/_images.scss */
+.figure-caption {
+ font-size: 90%;
+ color: #6c757d;
+}
+
+/* line 2, node_modules/bootstrap/scss/_code.scss */
+code {
+ font-size: 87.5%;
+ color: #e83e8c;
+ word-wrap: break-word;
+}
+
+/* line 8, node_modules/bootstrap/scss/_code.scss */
+a > code {
+ color: inherit;
+}
+
+/* line 14, node_modules/bootstrap/scss/_code.scss */
+kbd {
+ padding: 0.2rem 0.4rem;
+ font-size: 87.5%;
+ color: #ffffff;
+ background-color: #464746;
+ border-radius: 0.2rem;
+}
+
+/* line 22, node_modules/bootstrap/scss/_code.scss */
+kbd kbd {
+ padding: 0;
+ font-size: 100%;
+ font-weight: 700;
+}
+
+/* line 31, node_modules/bootstrap/scss/_code.scss */
+pre {
+ display: block;
+ font-size: 87.5%;
+ color: #464746;
+}
+
+/* line 37, node_modules/bootstrap/scss/_code.scss */
+pre code {
+ font-size: inherit;
+ color: inherit;
+ word-break: normal;
+}
+
+/* line 45, node_modules/bootstrap/scss/_code.scss */
+.pre-scrollable {
+ max-height: 340px;
+ overflow-y: scroll;
+}
+
+/* line 7, node_modules/bootstrap/scss/_grid.scss */
+.container,
+.container-fluid,
+.container-sm,
+.container-md,
+.container-lg,
+.container-xl,
+.container-xxl {
+ width: 100%;
+ padding-right: 15px;
+ padding-left: 15px;
+ margin-right: auto;
+ margin-left: auto;
+}
+
+@media (min-width: 576px) {
+ /* line 20, node_modules/bootstrap/scss/_grid.scss */
+ .container, .container-sm {
+ max-width: 100%;
+ }
+}
+
+@media (min-width: 768px) {
+ /* line 20, node_modules/bootstrap/scss/_grid.scss */
+ .container, .container-sm, .container-md {
+ max-width: 100%;
+ }
+}
+
+@media (min-width: 1024px) {
+ /* line 20, node_modules/bootstrap/scss/_grid.scss */
+ .container, .container-sm, .container-md, .container-lg {
+ max-width: 100%;
+ }
+}
+
+@media (min-width: 1280px) {
+ /* line 20, node_modules/bootstrap/scss/_grid.scss */
+ .container, .container-sm, .container-md, .container-lg, .container-xl {
+ max-width: 1150px;
+ }
+}
+
+@media (min-width: 1440px) {
+ /* line 20, node_modules/bootstrap/scss/_grid.scss */
+ .container, .container-sm, .container-md, .container-lg, .container-xl, .container-xxl {
+ max-width: 1310px;
+ }
+}
+
+/* line 49, node_modules/bootstrap/scss/_grid.scss */
+.row {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -ms-flex-wrap: wrap;
+ flex-wrap: wrap;
+ margin-right: -15px;
+ margin-left: -15px;
+}
+
+/* line 55, node_modules/bootstrap/scss/_grid.scss */
+.no-gutters {
+ margin-right: 0;
+ margin-left: 0;
+}
+
+/* line 59, node_modules/bootstrap/scss/_grid.scss */
+.no-gutters > .col,
+.no-gutters > [class*="col-"] {
+ padding-right: 0;
+ padding-left: 0;
+}
+
+/* line 8, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.col-1, .col-2, .col-3, .col-4, .col-5, .col-6, .col-7, .col-8, .col-9, .col-10, .col-11, .col-12, .col,
+.col-auto, .col-sm-1, .col-sm-2, .col-sm-3, .col-sm-4, .col-sm-5, .col-sm-6, .col-sm-7, .col-sm-8, .col-sm-9, .col-sm-10, .col-sm-11, .col-sm-12, .col-sm,
+.col-sm-auto, .col-md-1, .col-md-2, .col-md-3, .col-md-4, .col-md-5, .col-md-6, .col-md-7, .col-md-8, .col-md-9, .col-md-10, .col-md-11, .col-md-12, .col-md,
+.col-md-auto, .col-lg-1, .col-lg-2, .col-lg-3, .col-lg-4, .col-lg-5, .col-lg-6, .col-lg-7, .col-lg-8, .col-lg-9, .col-lg-10, .col-lg-11, .col-lg-12, .col-lg,
+.col-lg-auto, .col-xl-1, .col-xl-2, .col-xl-3, .col-xl-4, .col-xl-5, .col-xl-6, .col-xl-7, .col-xl-8, .col-xl-9, .col-xl-10, .col-xl-11, .col-xl-12, .col-xl,
+.col-xl-auto, .col-xxl-1, .col-xxl-2, .col-xxl-3, .col-xxl-4, .col-xxl-5, .col-xxl-6, .col-xxl-7, .col-xxl-8, .col-xxl-9, .col-xxl-10, .col-xxl-11, .col-xxl-12, .col-xxl,
+.col-xxl-auto {
+ position: relative;
+ width: 100%;
+ padding-right: 15px;
+ padding-left: 15px;
+}
+
+/* line 34, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.col {
+ -ms-flex-preferred-size: 0;
+ flex-basis: 0;
+ -webkit-box-flex: 1;
+ -ms-flex-positive: 1;
+ flex-grow: 1;
+ max-width: 100%;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+.row-cols-1 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 100%;
+ flex: 0 0 100%;
+ max-width: 100%;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+.row-cols-2 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 50%;
+ flex: 0 0 50%;
+ max-width: 50%;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+.row-cols-3 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 33.33333%;
+ flex: 0 0 33.33333%;
+ max-width: 33.33333%;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+.row-cols-4 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 25%;
+ flex: 0 0 25%;
+ max-width: 25%;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+.row-cols-5 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 20%;
+ flex: 0 0 20%;
+ max-width: 20%;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+.row-cols-6 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 16.66667%;
+ flex: 0 0 16.66667%;
+ max-width: 16.66667%;
+}
+
+/* line 48, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.col-auto {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 auto;
+ flex: 0 0 auto;
+ width: auto;
+ max-width: 100%;
+}
+
+/* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.col-1 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 8.33333%;
+ flex: 0 0 8.33333%;
+ max-width: 8.33333%;
+}
+
+/* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.col-2 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 16.66667%;
+ flex: 0 0 16.66667%;
+ max-width: 16.66667%;
+}
+
+/* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.col-3 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 25%;
+ flex: 0 0 25%;
+ max-width: 25%;
+}
+
+/* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.col-4 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 33.33333%;
+ flex: 0 0 33.33333%;
+ max-width: 33.33333%;
+}
+
+/* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.col-5 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 41.66667%;
+ flex: 0 0 41.66667%;
+ max-width: 41.66667%;
+}
+
+/* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.col-6 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 50%;
+ flex: 0 0 50%;
+ max-width: 50%;
+}
+
+/* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.col-7 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 58.33333%;
+ flex: 0 0 58.33333%;
+ max-width: 58.33333%;
+}
+
+/* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.col-8 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 66.66667%;
+ flex: 0 0 66.66667%;
+ max-width: 66.66667%;
+}
+
+/* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.col-9 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 75%;
+ flex: 0 0 75%;
+ max-width: 75%;
+}
+
+/* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.col-10 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 83.33333%;
+ flex: 0 0 83.33333%;
+ max-width: 83.33333%;
+}
+
+/* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.col-11 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 91.66667%;
+ flex: 0 0 91.66667%;
+ max-width: 91.66667%;
+}
+
+/* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.col-12 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 100%;
+ flex: 0 0 100%;
+ max-width: 100%;
+}
+
+/* line 60, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.order-first {
+ -webkit-box-ordinal-group: 0;
+ -ms-flex-order: -1;
+ order: -1;
+}
+
+/* line 62, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.order-last {
+ -webkit-box-ordinal-group: 14;
+ -ms-flex-order: 13;
+ order: 13;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.order-0 {
+ -webkit-box-ordinal-group: 1;
+ -ms-flex-order: 0;
+ order: 0;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.order-1 {
+ -webkit-box-ordinal-group: 2;
+ -ms-flex-order: 1;
+ order: 1;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.order-2 {
+ -webkit-box-ordinal-group: 3;
+ -ms-flex-order: 2;
+ order: 2;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.order-3 {
+ -webkit-box-ordinal-group: 4;
+ -ms-flex-order: 3;
+ order: 3;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.order-4 {
+ -webkit-box-ordinal-group: 5;
+ -ms-flex-order: 4;
+ order: 4;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.order-5 {
+ -webkit-box-ordinal-group: 6;
+ -ms-flex-order: 5;
+ order: 5;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.order-6 {
+ -webkit-box-ordinal-group: 7;
+ -ms-flex-order: 6;
+ order: 6;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.order-7 {
+ -webkit-box-ordinal-group: 8;
+ -ms-flex-order: 7;
+ order: 7;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.order-8 {
+ -webkit-box-ordinal-group: 9;
+ -ms-flex-order: 8;
+ order: 8;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.order-9 {
+ -webkit-box-ordinal-group: 10;
+ -ms-flex-order: 9;
+ order: 9;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.order-10 {
+ -webkit-box-ordinal-group: 11;
+ -ms-flex-order: 10;
+ order: 10;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.order-11 {
+ -webkit-box-ordinal-group: 12;
+ -ms-flex-order: 11;
+ order: 11;
+}
+
+/* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.order-12 {
+ -webkit-box-ordinal-group: 13;
+ -ms-flex-order: 12;
+ order: 12;
+}
+
+/* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.offset-1 {
+ margin-left: 8.33333%;
+}
+
+/* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.offset-2 {
+ margin-left: 16.66667%;
+}
+
+/* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.offset-3 {
+ margin-left: 25%;
+}
+
+/* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.offset-4 {
+ margin-left: 33.33333%;
+}
+
+/* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.offset-5 {
+ margin-left: 41.66667%;
+}
+
+/* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.offset-6 {
+ margin-left: 50%;
+}
+
+/* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.offset-7 {
+ margin-left: 58.33333%;
+}
+
+/* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.offset-8 {
+ margin-left: 66.66667%;
+}
+
+/* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.offset-9 {
+ margin-left: 75%;
+}
+
+/* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.offset-10 {
+ margin-left: 83.33333%;
+}
+
+/* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+.offset-11 {
+ margin-left: 91.66667%;
+}
+
+@media (min-width: 576px) {
+ /* line 34, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-sm {
+ -ms-flex-preferred-size: 0;
+ flex-basis: 0;
+ -webkit-box-flex: 1;
+ -ms-flex-positive: 1;
+ flex-grow: 1;
+ max-width: 100%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-sm-1 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 100%;
+ flex: 0 0 100%;
+ max-width: 100%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-sm-2 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 50%;
+ flex: 0 0 50%;
+ max-width: 50%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-sm-3 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 33.33333%;
+ flex: 0 0 33.33333%;
+ max-width: 33.33333%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-sm-4 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 25%;
+ flex: 0 0 25%;
+ max-width: 25%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-sm-5 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 20%;
+ flex: 0 0 20%;
+ max-width: 20%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-sm-6 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 16.66667%;
+ flex: 0 0 16.66667%;
+ max-width: 16.66667%;
+ }
+ /* line 48, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-sm-auto {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 auto;
+ flex: 0 0 auto;
+ width: auto;
+ max-width: 100%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-sm-1 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 8.33333%;
+ flex: 0 0 8.33333%;
+ max-width: 8.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-sm-2 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 16.66667%;
+ flex: 0 0 16.66667%;
+ max-width: 16.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-sm-3 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 25%;
+ flex: 0 0 25%;
+ max-width: 25%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-sm-4 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 33.33333%;
+ flex: 0 0 33.33333%;
+ max-width: 33.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-sm-5 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 41.66667%;
+ flex: 0 0 41.66667%;
+ max-width: 41.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-sm-6 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 50%;
+ flex: 0 0 50%;
+ max-width: 50%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-sm-7 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 58.33333%;
+ flex: 0 0 58.33333%;
+ max-width: 58.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-sm-8 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 66.66667%;
+ flex: 0 0 66.66667%;
+ max-width: 66.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-sm-9 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 75%;
+ flex: 0 0 75%;
+ max-width: 75%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-sm-10 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 83.33333%;
+ flex: 0 0 83.33333%;
+ max-width: 83.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-sm-11 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 91.66667%;
+ flex: 0 0 91.66667%;
+ max-width: 91.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-sm-12 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 100%;
+ flex: 0 0 100%;
+ max-width: 100%;
+ }
+ /* line 60, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-sm-first {
+ -webkit-box-ordinal-group: 0;
+ -ms-flex-order: -1;
+ order: -1;
+ }
+ /* line 62, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-sm-last {
+ -webkit-box-ordinal-group: 14;
+ -ms-flex-order: 13;
+ order: 13;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-sm-0 {
+ -webkit-box-ordinal-group: 1;
+ -ms-flex-order: 0;
+ order: 0;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-sm-1 {
+ -webkit-box-ordinal-group: 2;
+ -ms-flex-order: 1;
+ order: 1;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-sm-2 {
+ -webkit-box-ordinal-group: 3;
+ -ms-flex-order: 2;
+ order: 2;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-sm-3 {
+ -webkit-box-ordinal-group: 4;
+ -ms-flex-order: 3;
+ order: 3;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-sm-4 {
+ -webkit-box-ordinal-group: 5;
+ -ms-flex-order: 4;
+ order: 4;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-sm-5 {
+ -webkit-box-ordinal-group: 6;
+ -ms-flex-order: 5;
+ order: 5;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-sm-6 {
+ -webkit-box-ordinal-group: 7;
+ -ms-flex-order: 6;
+ order: 6;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-sm-7 {
+ -webkit-box-ordinal-group: 8;
+ -ms-flex-order: 7;
+ order: 7;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-sm-8 {
+ -webkit-box-ordinal-group: 9;
+ -ms-flex-order: 8;
+ order: 8;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-sm-9 {
+ -webkit-box-ordinal-group: 10;
+ -ms-flex-order: 9;
+ order: 9;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-sm-10 {
+ -webkit-box-ordinal-group: 11;
+ -ms-flex-order: 10;
+ order: 10;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-sm-11 {
+ -webkit-box-ordinal-group: 12;
+ -ms-flex-order: 11;
+ order: 11;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-sm-12 {
+ -webkit-box-ordinal-group: 13;
+ -ms-flex-order: 12;
+ order: 12;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-sm-0 {
+ margin-left: 0;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-sm-1 {
+ margin-left: 8.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-sm-2 {
+ margin-left: 16.66667%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-sm-3 {
+ margin-left: 25%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-sm-4 {
+ margin-left: 33.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-sm-5 {
+ margin-left: 41.66667%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-sm-6 {
+ margin-left: 50%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-sm-7 {
+ margin-left: 58.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-sm-8 {
+ margin-left: 66.66667%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-sm-9 {
+ margin-left: 75%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-sm-10 {
+ margin-left: 83.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-sm-11 {
+ margin-left: 91.66667%;
+ }
+}
+
+@media (min-width: 768px) {
+ /* line 34, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-md {
+ -ms-flex-preferred-size: 0;
+ flex-basis: 0;
+ -webkit-box-flex: 1;
+ -ms-flex-positive: 1;
+ flex-grow: 1;
+ max-width: 100%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-md-1 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 100%;
+ flex: 0 0 100%;
+ max-width: 100%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-md-2 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 50%;
+ flex: 0 0 50%;
+ max-width: 50%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-md-3 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 33.33333%;
+ flex: 0 0 33.33333%;
+ max-width: 33.33333%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-md-4 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 25%;
+ flex: 0 0 25%;
+ max-width: 25%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-md-5 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 20%;
+ flex: 0 0 20%;
+ max-width: 20%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-md-6 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 16.66667%;
+ flex: 0 0 16.66667%;
+ max-width: 16.66667%;
+ }
+ /* line 48, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-md-auto {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 auto;
+ flex: 0 0 auto;
+ width: auto;
+ max-width: 100%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-md-1 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 8.33333%;
+ flex: 0 0 8.33333%;
+ max-width: 8.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-md-2 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 16.66667%;
+ flex: 0 0 16.66667%;
+ max-width: 16.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-md-3 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 25%;
+ flex: 0 0 25%;
+ max-width: 25%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-md-4 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 33.33333%;
+ flex: 0 0 33.33333%;
+ max-width: 33.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-md-5 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 41.66667%;
+ flex: 0 0 41.66667%;
+ max-width: 41.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-md-6 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 50%;
+ flex: 0 0 50%;
+ max-width: 50%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-md-7 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 58.33333%;
+ flex: 0 0 58.33333%;
+ max-width: 58.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-md-8 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 66.66667%;
+ flex: 0 0 66.66667%;
+ max-width: 66.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-md-9 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 75%;
+ flex: 0 0 75%;
+ max-width: 75%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-md-10 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 83.33333%;
+ flex: 0 0 83.33333%;
+ max-width: 83.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-md-11 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 91.66667%;
+ flex: 0 0 91.66667%;
+ max-width: 91.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-md-12 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 100%;
+ flex: 0 0 100%;
+ max-width: 100%;
+ }
+ /* line 60, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-md-first {
+ -webkit-box-ordinal-group: 0;
+ -ms-flex-order: -1;
+ order: -1;
+ }
+ /* line 62, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-md-last {
+ -webkit-box-ordinal-group: 14;
+ -ms-flex-order: 13;
+ order: 13;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-md-0 {
+ -webkit-box-ordinal-group: 1;
+ -ms-flex-order: 0;
+ order: 0;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-md-1 {
+ -webkit-box-ordinal-group: 2;
+ -ms-flex-order: 1;
+ order: 1;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-md-2 {
+ -webkit-box-ordinal-group: 3;
+ -ms-flex-order: 2;
+ order: 2;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-md-3 {
+ -webkit-box-ordinal-group: 4;
+ -ms-flex-order: 3;
+ order: 3;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-md-4 {
+ -webkit-box-ordinal-group: 5;
+ -ms-flex-order: 4;
+ order: 4;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-md-5 {
+ -webkit-box-ordinal-group: 6;
+ -ms-flex-order: 5;
+ order: 5;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-md-6 {
+ -webkit-box-ordinal-group: 7;
+ -ms-flex-order: 6;
+ order: 6;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-md-7 {
+ -webkit-box-ordinal-group: 8;
+ -ms-flex-order: 7;
+ order: 7;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-md-8 {
+ -webkit-box-ordinal-group: 9;
+ -ms-flex-order: 8;
+ order: 8;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-md-9 {
+ -webkit-box-ordinal-group: 10;
+ -ms-flex-order: 9;
+ order: 9;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-md-10 {
+ -webkit-box-ordinal-group: 11;
+ -ms-flex-order: 10;
+ order: 10;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-md-11 {
+ -webkit-box-ordinal-group: 12;
+ -ms-flex-order: 11;
+ order: 11;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-md-12 {
+ -webkit-box-ordinal-group: 13;
+ -ms-flex-order: 12;
+ order: 12;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-md-0 {
+ margin-left: 0;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-md-1 {
+ margin-left: 8.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-md-2 {
+ margin-left: 16.66667%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-md-3 {
+ margin-left: 25%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-md-4 {
+ margin-left: 33.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-md-5 {
+ margin-left: 41.66667%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-md-6 {
+ margin-left: 50%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-md-7 {
+ margin-left: 58.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-md-8 {
+ margin-left: 66.66667%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-md-9 {
+ margin-left: 75%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-md-10 {
+ margin-left: 83.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-md-11 {
+ margin-left: 91.66667%;
+ }
+}
+
+@media (min-width: 1024px) {
+ /* line 34, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-lg {
+ -ms-flex-preferred-size: 0;
+ flex-basis: 0;
+ -webkit-box-flex: 1;
+ -ms-flex-positive: 1;
+ flex-grow: 1;
+ max-width: 100%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-lg-1 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 100%;
+ flex: 0 0 100%;
+ max-width: 100%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-lg-2 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 50%;
+ flex: 0 0 50%;
+ max-width: 50%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-lg-3 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 33.33333%;
+ flex: 0 0 33.33333%;
+ max-width: 33.33333%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-lg-4 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 25%;
+ flex: 0 0 25%;
+ max-width: 25%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-lg-5 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 20%;
+ flex: 0 0 20%;
+ max-width: 20%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-lg-6 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 16.66667%;
+ flex: 0 0 16.66667%;
+ max-width: 16.66667%;
+ }
+ /* line 48, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-lg-auto {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 auto;
+ flex: 0 0 auto;
+ width: auto;
+ max-width: 100%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-lg-1 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 8.33333%;
+ flex: 0 0 8.33333%;
+ max-width: 8.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-lg-2 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 16.66667%;
+ flex: 0 0 16.66667%;
+ max-width: 16.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-lg-3 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 25%;
+ flex: 0 0 25%;
+ max-width: 25%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-lg-4 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 33.33333%;
+ flex: 0 0 33.33333%;
+ max-width: 33.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-lg-5 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 41.66667%;
+ flex: 0 0 41.66667%;
+ max-width: 41.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-lg-6 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 50%;
+ flex: 0 0 50%;
+ max-width: 50%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-lg-7 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 58.33333%;
+ flex: 0 0 58.33333%;
+ max-width: 58.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-lg-8 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 66.66667%;
+ flex: 0 0 66.66667%;
+ max-width: 66.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-lg-9 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 75%;
+ flex: 0 0 75%;
+ max-width: 75%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-lg-10 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 83.33333%;
+ flex: 0 0 83.33333%;
+ max-width: 83.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-lg-11 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 91.66667%;
+ flex: 0 0 91.66667%;
+ max-width: 91.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-lg-12 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 100%;
+ flex: 0 0 100%;
+ max-width: 100%;
+ }
+ /* line 60, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-lg-first {
+ -webkit-box-ordinal-group: 0;
+ -ms-flex-order: -1;
+ order: -1;
+ }
+ /* line 62, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-lg-last {
+ -webkit-box-ordinal-group: 14;
+ -ms-flex-order: 13;
+ order: 13;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-lg-0 {
+ -webkit-box-ordinal-group: 1;
+ -ms-flex-order: 0;
+ order: 0;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-lg-1 {
+ -webkit-box-ordinal-group: 2;
+ -ms-flex-order: 1;
+ order: 1;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-lg-2 {
+ -webkit-box-ordinal-group: 3;
+ -ms-flex-order: 2;
+ order: 2;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-lg-3 {
+ -webkit-box-ordinal-group: 4;
+ -ms-flex-order: 3;
+ order: 3;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-lg-4 {
+ -webkit-box-ordinal-group: 5;
+ -ms-flex-order: 4;
+ order: 4;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-lg-5 {
+ -webkit-box-ordinal-group: 6;
+ -ms-flex-order: 5;
+ order: 5;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-lg-6 {
+ -webkit-box-ordinal-group: 7;
+ -ms-flex-order: 6;
+ order: 6;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-lg-7 {
+ -webkit-box-ordinal-group: 8;
+ -ms-flex-order: 7;
+ order: 7;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-lg-8 {
+ -webkit-box-ordinal-group: 9;
+ -ms-flex-order: 8;
+ order: 8;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-lg-9 {
+ -webkit-box-ordinal-group: 10;
+ -ms-flex-order: 9;
+ order: 9;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-lg-10 {
+ -webkit-box-ordinal-group: 11;
+ -ms-flex-order: 10;
+ order: 10;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-lg-11 {
+ -webkit-box-ordinal-group: 12;
+ -ms-flex-order: 11;
+ order: 11;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-lg-12 {
+ -webkit-box-ordinal-group: 13;
+ -ms-flex-order: 12;
+ order: 12;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-lg-0 {
+ margin-left: 0;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-lg-1 {
+ margin-left: 8.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-lg-2 {
+ margin-left: 16.66667%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-lg-3 {
+ margin-left: 25%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-lg-4 {
+ margin-left: 33.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-lg-5 {
+ margin-left: 41.66667%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-lg-6 {
+ margin-left: 50%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-lg-7 {
+ margin-left: 58.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-lg-8 {
+ margin-left: 66.66667%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-lg-9 {
+ margin-left: 75%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-lg-10 {
+ margin-left: 83.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-lg-11 {
+ margin-left: 91.66667%;
+ }
+}
+
+@media (min-width: 1280px) {
+ /* line 34, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xl {
+ -ms-flex-preferred-size: 0;
+ flex-basis: 0;
+ -webkit-box-flex: 1;
+ -ms-flex-positive: 1;
+ flex-grow: 1;
+ max-width: 100%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-xl-1 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 100%;
+ flex: 0 0 100%;
+ max-width: 100%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-xl-2 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 50%;
+ flex: 0 0 50%;
+ max-width: 50%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-xl-3 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 33.33333%;
+ flex: 0 0 33.33333%;
+ max-width: 33.33333%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-xl-4 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 25%;
+ flex: 0 0 25%;
+ max-width: 25%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-xl-5 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 20%;
+ flex: 0 0 20%;
+ max-width: 20%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-xl-6 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 16.66667%;
+ flex: 0 0 16.66667%;
+ max-width: 16.66667%;
+ }
+ /* line 48, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xl-auto {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 auto;
+ flex: 0 0 auto;
+ width: auto;
+ max-width: 100%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xl-1 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 8.33333%;
+ flex: 0 0 8.33333%;
+ max-width: 8.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xl-2 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 16.66667%;
+ flex: 0 0 16.66667%;
+ max-width: 16.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xl-3 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 25%;
+ flex: 0 0 25%;
+ max-width: 25%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xl-4 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 33.33333%;
+ flex: 0 0 33.33333%;
+ max-width: 33.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xl-5 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 41.66667%;
+ flex: 0 0 41.66667%;
+ max-width: 41.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xl-6 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 50%;
+ flex: 0 0 50%;
+ max-width: 50%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xl-7 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 58.33333%;
+ flex: 0 0 58.33333%;
+ max-width: 58.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xl-8 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 66.66667%;
+ flex: 0 0 66.66667%;
+ max-width: 66.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xl-9 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 75%;
+ flex: 0 0 75%;
+ max-width: 75%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xl-10 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 83.33333%;
+ flex: 0 0 83.33333%;
+ max-width: 83.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xl-11 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 91.66667%;
+ flex: 0 0 91.66667%;
+ max-width: 91.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xl-12 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 100%;
+ flex: 0 0 100%;
+ max-width: 100%;
+ }
+ /* line 60, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xl-first {
+ -webkit-box-ordinal-group: 0;
+ -ms-flex-order: -1;
+ order: -1;
+ }
+ /* line 62, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xl-last {
+ -webkit-box-ordinal-group: 14;
+ -ms-flex-order: 13;
+ order: 13;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xl-0 {
+ -webkit-box-ordinal-group: 1;
+ -ms-flex-order: 0;
+ order: 0;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xl-1 {
+ -webkit-box-ordinal-group: 2;
+ -ms-flex-order: 1;
+ order: 1;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xl-2 {
+ -webkit-box-ordinal-group: 3;
+ -ms-flex-order: 2;
+ order: 2;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xl-3 {
+ -webkit-box-ordinal-group: 4;
+ -ms-flex-order: 3;
+ order: 3;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xl-4 {
+ -webkit-box-ordinal-group: 5;
+ -ms-flex-order: 4;
+ order: 4;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xl-5 {
+ -webkit-box-ordinal-group: 6;
+ -ms-flex-order: 5;
+ order: 5;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xl-6 {
+ -webkit-box-ordinal-group: 7;
+ -ms-flex-order: 6;
+ order: 6;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xl-7 {
+ -webkit-box-ordinal-group: 8;
+ -ms-flex-order: 7;
+ order: 7;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xl-8 {
+ -webkit-box-ordinal-group: 9;
+ -ms-flex-order: 8;
+ order: 8;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xl-9 {
+ -webkit-box-ordinal-group: 10;
+ -ms-flex-order: 9;
+ order: 9;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xl-10 {
+ -webkit-box-ordinal-group: 11;
+ -ms-flex-order: 10;
+ order: 10;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xl-11 {
+ -webkit-box-ordinal-group: 12;
+ -ms-flex-order: 11;
+ order: 11;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xl-12 {
+ -webkit-box-ordinal-group: 13;
+ -ms-flex-order: 12;
+ order: 12;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xl-0 {
+ margin-left: 0;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xl-1 {
+ margin-left: 8.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xl-2 {
+ margin-left: 16.66667%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xl-3 {
+ margin-left: 25%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xl-4 {
+ margin-left: 33.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xl-5 {
+ margin-left: 41.66667%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xl-6 {
+ margin-left: 50%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xl-7 {
+ margin-left: 58.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xl-8 {
+ margin-left: 66.66667%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xl-9 {
+ margin-left: 75%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xl-10 {
+ margin-left: 83.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xl-11 {
+ margin-left: 91.66667%;
+ }
+}
+
+@media (min-width: 1440px) {
+ /* line 34, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xxl {
+ -ms-flex-preferred-size: 0;
+ flex-basis: 0;
+ -webkit-box-flex: 1;
+ -ms-flex-positive: 1;
+ flex-grow: 1;
+ max-width: 100%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-xxl-1 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 100%;
+ flex: 0 0 100%;
+ max-width: 100%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-xxl-2 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 50%;
+ flex: 0 0 50%;
+ max-width: 50%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-xxl-3 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 33.33333%;
+ flex: 0 0 33.33333%;
+ max-width: 33.33333%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-xxl-4 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 25%;
+ flex: 0 0 25%;
+ max-width: 25%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-xxl-5 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 20%;
+ flex: 0 0 20%;
+ max-width: 20%;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid.scss */
+ .row-cols-xxl-6 > * {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 16.66667%;
+ flex: 0 0 16.66667%;
+ max-width: 16.66667%;
+ }
+ /* line 48, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xxl-auto {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 auto;
+ flex: 0 0 auto;
+ width: auto;
+ max-width: 100%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xxl-1 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 8.33333%;
+ flex: 0 0 8.33333%;
+ max-width: 8.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xxl-2 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 16.66667%;
+ flex: 0 0 16.66667%;
+ max-width: 16.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xxl-3 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 25%;
+ flex: 0 0 25%;
+ max-width: 25%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xxl-4 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 33.33333%;
+ flex: 0 0 33.33333%;
+ max-width: 33.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xxl-5 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 41.66667%;
+ flex: 0 0 41.66667%;
+ max-width: 41.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xxl-6 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 50%;
+ flex: 0 0 50%;
+ max-width: 50%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xxl-7 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 58.33333%;
+ flex: 0 0 58.33333%;
+ max-width: 58.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xxl-8 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 66.66667%;
+ flex: 0 0 66.66667%;
+ max-width: 66.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xxl-9 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 75%;
+ flex: 0 0 75%;
+ max-width: 75%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xxl-10 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 83.33333%;
+ flex: 0 0 83.33333%;
+ max-width: 83.33333%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xxl-11 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 91.66667%;
+ flex: 0 0 91.66667%;
+ max-width: 91.66667%;
+ }
+ /* line 54, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .col-xxl-12 {
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 100%;
+ flex: 0 0 100%;
+ max-width: 100%;
+ }
+ /* line 60, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xxl-first {
+ -webkit-box-ordinal-group: 0;
+ -ms-flex-order: -1;
+ order: -1;
+ }
+ /* line 62, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xxl-last {
+ -webkit-box-ordinal-group: 14;
+ -ms-flex-order: 13;
+ order: 13;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xxl-0 {
+ -webkit-box-ordinal-group: 1;
+ -ms-flex-order: 0;
+ order: 0;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xxl-1 {
+ -webkit-box-ordinal-group: 2;
+ -ms-flex-order: 1;
+ order: 1;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xxl-2 {
+ -webkit-box-ordinal-group: 3;
+ -ms-flex-order: 2;
+ order: 2;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xxl-3 {
+ -webkit-box-ordinal-group: 4;
+ -ms-flex-order: 3;
+ order: 3;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xxl-4 {
+ -webkit-box-ordinal-group: 5;
+ -ms-flex-order: 4;
+ order: 4;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xxl-5 {
+ -webkit-box-ordinal-group: 6;
+ -ms-flex-order: 5;
+ order: 5;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xxl-6 {
+ -webkit-box-ordinal-group: 7;
+ -ms-flex-order: 6;
+ order: 6;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xxl-7 {
+ -webkit-box-ordinal-group: 8;
+ -ms-flex-order: 7;
+ order: 7;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xxl-8 {
+ -webkit-box-ordinal-group: 9;
+ -ms-flex-order: 8;
+ order: 8;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xxl-9 {
+ -webkit-box-ordinal-group: 10;
+ -ms-flex-order: 9;
+ order: 9;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xxl-10 {
+ -webkit-box-ordinal-group: 11;
+ -ms-flex-order: 10;
+ order: 10;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xxl-11 {
+ -webkit-box-ordinal-group: 12;
+ -ms-flex-order: 11;
+ order: 11;
+ }
+ /* line 65, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .order-xxl-12 {
+ -webkit-box-ordinal-group: 13;
+ -ms-flex-order: 12;
+ order: 12;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xxl-0 {
+ margin-left: 0;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xxl-1 {
+ margin-left: 8.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xxl-2 {
+ margin-left: 16.66667%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xxl-3 {
+ margin-left: 25%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xxl-4 {
+ margin-left: 33.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xxl-5 {
+ margin-left: 41.66667%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xxl-6 {
+ margin-left: 50%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xxl-7 {
+ margin-left: 58.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xxl-8 {
+ margin-left: 66.66667%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xxl-9 {
+ margin-left: 75%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xxl-10 {
+ margin-left: 83.33333%;
+ }
+ /* line 72, node_modules/bootstrap/scss/mixins/_grid-framework.scss */
+ .offset-xxl-11 {
+ margin-left: 91.66667%;
+ }
+}
+
+/* line 5, node_modules/bootstrap/scss/_tables.scss */
+.table {
+ width: 100%;
+ margin-bottom: 1rem;
+ color: #464746;
+}
+
+/* line 11, node_modules/bootstrap/scss/_tables.scss */
+.table th,
+.table td {
+ padding: 0.75rem;
+ vertical-align: top;
+ border-top: 1px solid #dee2e6;
+}
+
+/* line 18, node_modules/bootstrap/scss/_tables.scss */
+.table thead th {
+ vertical-align: bottom;
+ border-bottom: 2px solid #dee2e6;
+}
+
+/* line 23, node_modules/bootstrap/scss/_tables.scss */
+.table tbody + tbody {
+ border-top: 2px solid #dee2e6;
+}
+
+/* line 34, node_modules/bootstrap/scss/_tables.scss */
+.table-sm th,
+.table-sm td {
+ padding: 0.3rem;
+}
+
+/* line 45, node_modules/bootstrap/scss/_tables.scss */
+.table-bordered {
+ border: 1px solid #dee2e6;
+}
+
+/* line 48, node_modules/bootstrap/scss/_tables.scss */
+.table-bordered th,
+.table-bordered td {
+ border: 1px solid #dee2e6;
+}
+
+/* line 54, node_modules/bootstrap/scss/_tables.scss */
+.table-bordered thead th,
+.table-bordered thead td {
+ border-bottom-width: 2px;
+}
+
+/* line 62, node_modules/bootstrap/scss/_tables.scss */
+.table-borderless th,
+.table-borderless td,
+.table-borderless thead th,
+.table-borderless tbody + tbody {
+ border: 0;
+}
+
+/* line 75, node_modules/bootstrap/scss/_tables.scss */
+.table-striped tbody tr:nth-of-type(odd) {
+ background-color: rgba(0, 0, 0, 0.05);
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.table-hover tbody tr:hover {
+ color: #464746;
+ background-color: rgba(0, 0, 0, 0.075);
+}
+
+/* line 7, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-primary,
+.table-primary > th,
+.table-primary > td {
+ background-color: #cbcbcb;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-primary th,
+.table-primary td,
+.table-primary thead th,
+.table-primary tbody + tbody {
+ border-color: #9f9f9f;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.table-hover .table-primary:hover {
+ background-color: #bebebe;
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-hover .table-primary:hover > td,
+.table-hover .table-primary:hover > th {
+ background-color: #bebebe;
+}
+
+/* line 7, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-secondary,
+.table-secondary > th,
+.table-secondary > td {
+ background-color: #fbf9f7;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-secondary th,
+.table-secondary td,
+.table-secondary thead th,
+.table-secondary tbody + tbody {
+ border-color: #f7f5f0;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.table-hover .table-secondary:hover {
+ background-color: #f3ece6;
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-hover .table-secondary:hover > td,
+.table-hover .table-secondary:hover > th {
+ background-color: #f3ece6;
+}
+
+/* line 7, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-success,
+.table-success > th,
+.table-success > td {
+ background-color: #fbf9f7;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-success th,
+.table-success td,
+.table-success thead th,
+.table-success tbody + tbody {
+ border-color: #f7f5f0;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.table-hover .table-success:hover {
+ background-color: #f3ece6;
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-hover .table-success:hover > td,
+.table-hover .table-success:hover > th {
+ background-color: #f3ece6;
+}
+
+/* line 7, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-info,
+.table-info > th,
+.table-info > td {
+ background-color: #cbcbcb;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-info th,
+.table-info td,
+.table-info thead th,
+.table-info tbody + tbody {
+ border-color: #9f9f9f;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.table-hover .table-info:hover {
+ background-color: #bebebe;
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-hover .table-info:hover > td,
+.table-hover .table-info:hover > th {
+ background-color: #bebebe;
+}
+
+/* line 7, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-warning,
+.table-warning > th,
+.table-warning > td {
+ background-color: #cbcbcb;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-warning th,
+.table-warning td,
+.table-warning thead th,
+.table-warning tbody + tbody {
+ border-color: #9f9f9f;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.table-hover .table-warning:hover {
+ background-color: #bebebe;
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-hover .table-warning:hover > td,
+.table-hover .table-warning:hover > th {
+ background-color: #bebebe;
+}
+
+/* line 7, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-danger,
+.table-danger > th,
+.table-danger > td {
+ background-color: #f8ccbf;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-danger th,
+.table-danger td,
+.table-danger thead th,
+.table-danger tbody + tbody {
+ border-color: #f1a187;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.table-hover .table-danger:hover {
+ background-color: #f5baa8;
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-hover .table-danger:hover > td,
+.table-hover .table-danger:hover > th {
+ background-color: #f5baa8;
+}
+
+/* line 7, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-light,
+.table-light > th,
+.table-light > td {
+ background-color: #fdfdfd;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-light th,
+.table-light td,
+.table-light thead th,
+.table-light tbody + tbody {
+ border-color: #fbfbfb;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.table-hover .table-light:hover {
+ background-color: #f0f0f0;
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-hover .table-light:hover > td,
+.table-hover .table-light:hover > th {
+ background-color: #f0f0f0;
+}
+
+/* line 7, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-dark,
+.table-dark > th,
+.table-dark > td {
+ background-color: #c6c8ca;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-dark th,
+.table-dark td,
+.table-dark thead th,
+.table-dark tbody + tbody {
+ border-color: #95999c;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.table-hover .table-dark:hover {
+ background-color: #b9bbbe;
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-hover .table-dark:hover > td,
+.table-hover .table-dark:hover > th {
+ background-color: #b9bbbe;
+}
+
+/* line 7, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-active,
+.table-active > th,
+.table-active > td {
+ background-color: rgba(0, 0, 0, 0.075);
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.table-hover .table-active:hover {
+ background-color: rgba(0, 0, 0, 0.075);
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_table-row.scss */
+.table-hover .table-active:hover > td,
+.table-hover .table-active:hover > th {
+ background-color: rgba(0, 0, 0, 0.075);
+}
+
+/* line 114, node_modules/bootstrap/scss/_tables.scss */
+.table .thead-dark th {
+ color: #ffffff;
+ background-color: #343a40;
+ border-color: #454d55;
+}
+
+/* line 122, node_modules/bootstrap/scss/_tables.scss */
+.table .thead-light th {
+ color: #495057;
+ background-color: #eaebea;
+ border-color: #dee2e6;
+}
+
+/* line 130, node_modules/bootstrap/scss/_tables.scss */
+.table-dark {
+ color: #ffffff;
+ background-color: #343a40;
+}
+
+/* line 134, node_modules/bootstrap/scss/_tables.scss */
+.table-dark th,
+.table-dark td,
+.table-dark thead th {
+ border-color: #454d55;
+}
+
+/* line 140, node_modules/bootstrap/scss/_tables.scss */
+.table-dark.table-bordered {
+ border: 0;
+}
+
+/* line 145, node_modules/bootstrap/scss/_tables.scss */
+.table-dark.table-striped tbody tr:nth-of-type(odd) {
+ background-color: rgba(255, 255, 255, 0.05);
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.table-dark.table-hover tbody tr:hover {
+ color: #ffffff;
+ background-color: rgba(255, 255, 255, 0.075);
+}
+
+@media (max-width: 575.98px) {
+ /* line 171, node_modules/bootstrap/scss/_tables.scss */
+ .table-responsive-sm {
+ display: block;
+ width: 100%;
+ overflow-x: auto;
+ -webkit-overflow-scrolling: touch;
+ }
+ /* line 179, node_modules/bootstrap/scss/_tables.scss */
+ .table-responsive-sm > .table-bordered {
+ border: 0;
+ }
+}
+
+@media (max-width: 767.98px) {
+ /* line 171, node_modules/bootstrap/scss/_tables.scss */
+ .table-responsive-md {
+ display: block;
+ width: 100%;
+ overflow-x: auto;
+ -webkit-overflow-scrolling: touch;
+ }
+ /* line 179, node_modules/bootstrap/scss/_tables.scss */
+ .table-responsive-md > .table-bordered {
+ border: 0;
+ }
+}
+
+@media (max-width: 1023.98px) {
+ /* line 171, node_modules/bootstrap/scss/_tables.scss */
+ .table-responsive-lg {
+ display: block;
+ width: 100%;
+ overflow-x: auto;
+ -webkit-overflow-scrolling: touch;
+ }
+ /* line 179, node_modules/bootstrap/scss/_tables.scss */
+ .table-responsive-lg > .table-bordered {
+ border: 0;
+ }
+}
+
+@media (max-width: 1279.98px) {
+ /* line 171, node_modules/bootstrap/scss/_tables.scss */
+ .table-responsive-xl {
+ display: block;
+ width: 100%;
+ overflow-x: auto;
+ -webkit-overflow-scrolling: touch;
+ }
+ /* line 179, node_modules/bootstrap/scss/_tables.scss */
+ .table-responsive-xl > .table-bordered {
+ border: 0;
+ }
+}
+
+@media (max-width: 1439.98px) {
+ /* line 171, node_modules/bootstrap/scss/_tables.scss */
+ .table-responsive-xxl {
+ display: block;
+ width: 100%;
+ overflow-x: auto;
+ -webkit-overflow-scrolling: touch;
+ }
+ /* line 179, node_modules/bootstrap/scss/_tables.scss */
+ .table-responsive-xxl > .table-bordered {
+ border: 0;
+ }
+}
+
+/* line 171, node_modules/bootstrap/scss/_tables.scss */
+.table-responsive {
+ display: block;
+ width: 100%;
+ overflow-x: auto;
+ -webkit-overflow-scrolling: touch;
+}
+
+/* line 179, node_modules/bootstrap/scss/_tables.scss */
+.table-responsive > .table-bordered {
+ border: 0;
+}
+
+/* line 7, node_modules/bootstrap/scss/_forms.scss */
+.form-control {
+ display: block;
+ width: 100%;
+ height: calc(1.5em + 0.75rem + 2px);
+ padding: 0.375rem 0.75rem;
+ font-size: 1rem;
+ font-weight: 300;
+ line-height: 1.5;
+ color: #495057;
+ background-color: #ffffff;
+ background-clip: padding-box;
+ border: 1px solid #ced4da;
+ border-radius: 0.25rem;
+ -webkit-transition: border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+ transition: border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+ transition: border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;
+ transition: border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 7, node_modules/bootstrap/scss/_forms.scss */
+ .form-control {
+ -webkit-transition: none;
+ transition: none;
+ }
+}
+
+/* line 28, node_modules/bootstrap/scss/_forms.scss */
+.form-control::-ms-expand {
+ background-color: transparent;
+ border: 0;
+}
+
+/* line 34, node_modules/bootstrap/scss/_forms.scss */
+.form-control:-moz-focusring {
+ color: transparent;
+ text-shadow: 0 0 0 #495057;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_forms.scss */
+.form-control:focus {
+ color: #495057;
+ background-color: #ffffff;
+ border-color: #858785;
+ outline: 0;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+ box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+}
+
+/* line 43, node_modules/bootstrap/scss/_forms.scss */
+.form-control::-webkit-input-placeholder {
+ color: #6c757d;
+ opacity: 1;
+}
+.form-control::-moz-placeholder {
+ color: #6c757d;
+ opacity: 1;
+}
+.form-control:-ms-input-placeholder {
+ color: #6c757d;
+ opacity: 1;
+}
+.form-control::-ms-input-placeholder {
+ color: #6c757d;
+ opacity: 1;
+}
+.form-control::placeholder {
+ color: #6c757d;
+ opacity: 1;
+}
+
+/* line 54, node_modules/bootstrap/scss/_forms.scss */
+.form-control:disabled, .form-control[readonly] {
+ background-color: #eaebea;
+ opacity: 1;
+}
+
+/* line 66, node_modules/bootstrap/scss/_forms.scss */
+input[type="date"].form-control,
+input[type="time"].form-control,
+input[type="datetime-local"].form-control,
+input[type="month"].form-control {
+ -webkit-appearance: none;
+ -moz-appearance: none;
+ appearance: none;
+}
+
+/* line 72, node_modules/bootstrap/scss/_forms.scss */
+select.form-control:focus::-ms-value {
+ color: #495057;
+ background-color: #ffffff;
+}
+
+/* line 84, node_modules/bootstrap/scss/_forms.scss */
+.form-control-file,
+.form-control-range {
+ display: block;
+ width: 100%;
+}
+
+/* line 97, node_modules/bootstrap/scss/_forms.scss */
+.col-form-label {
+ padding-top: calc(0.375rem + 1px);
+ padding-bottom: calc(0.375rem + 1px);
+ margin-bottom: 0;
+ font-size: inherit;
+ line-height: 1.5;
+}
+
+/* line 105, node_modules/bootstrap/scss/_forms.scss */
+.col-form-label-lg {
+ padding-top: calc(0.5rem + 1px);
+ padding-bottom: calc(0.5rem + 1px);
+ font-size: 1.25rem;
+ line-height: 1.5;
+}
+
+/* line 112, node_modules/bootstrap/scss/_forms.scss */
+.col-form-label-sm {
+ padding-top: calc(0.25rem + 1px);
+ padding-bottom: calc(0.25rem + 1px);
+ font-size: 0.875rem;
+ line-height: 1.5;
+}
+
+/* line 125, node_modules/bootstrap/scss/_forms.scss */
+.form-control-plaintext {
+ display: block;
+ width: 100%;
+ padding: 0.375rem 0;
+ margin-bottom: 0;
+ font-size: 1rem;
+ line-height: 1.5;
+ color: #464746;
+ background-color: transparent;
+ border: solid transparent;
+ border-width: 1px 0;
+}
+
+/* line 137, node_modules/bootstrap/scss/_forms.scss */
+.form-control-plaintext.form-control-sm, .form-control-plaintext.form-control-lg {
+ padding-right: 0;
+ padding-left: 0;
+}
+
+/* line 152, node_modules/bootstrap/scss/_forms.scss */
+.form-control-sm {
+ height: calc(1.5em + 0.5rem + 2px);
+ padding: 0.25rem 0.5rem;
+ font-size: 0.875rem;
+ line-height: 1.5;
+ border-radius: 0.2rem;
+}
+
+/* line 160, node_modules/bootstrap/scss/_forms.scss */
+.form-control-lg {
+ height: calc(1.5em + 1rem + 2px);
+ padding: 0.5rem 1rem;
+ font-size: 1.25rem;
+ line-height: 1.5;
+ border-radius: 0.3rem;
+}
+
+/* line 170, node_modules/bootstrap/scss/_forms.scss */
+select.form-control[size], select.form-control[multiple] {
+ height: auto;
+}
+
+/* line 176, node_modules/bootstrap/scss/_forms.scss */
+textarea.form-control {
+ height: auto;
+}
+
+/* line 185, node_modules/bootstrap/scss/_forms.scss */
+.form-group {
+ margin-bottom: 1rem;
+}
+
+/* line 189, node_modules/bootstrap/scss/_forms.scss */
+.form-text {
+ display: block;
+ margin-top: 0.25rem;
+}
+
+/* line 199, node_modules/bootstrap/scss/_forms.scss */
+.form-row {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -ms-flex-wrap: wrap;
+ flex-wrap: wrap;
+ margin-right: -5px;
+ margin-left: -5px;
+}
+
+/* line 205, node_modules/bootstrap/scss/_forms.scss */
+.form-row > .col,
+.form-row > [class*="col-"] {
+ padding-right: 5px;
+ padding-left: 5px;
+}
+
+/* line 217, node_modules/bootstrap/scss/_forms.scss */
+.form-check {
+ position: relative;
+ display: block;
+ padding-left: 1.25rem;
+}
+
+/* line 223, node_modules/bootstrap/scss/_forms.scss */
+.form-check-input {
+ position: absolute;
+ margin-top: 0.3rem;
+ margin-left: -1.25rem;
+}
+
+/* line 229, node_modules/bootstrap/scss/_forms.scss */
+.form-check-input[disabled] ~ .form-check-label,
+.form-check-input:disabled ~ .form-check-label {
+ color: #6c757d;
+}
+
+/* line 235, node_modules/bootstrap/scss/_forms.scss */
+.form-check-label {
+ margin-bottom: 0;
+}
+
+/* line 239, node_modules/bootstrap/scss/_forms.scss */
+.form-check-inline {
+ display: -webkit-inline-box;
+ display: -ms-inline-flexbox;
+ display: inline-flex;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ padding-left: 0;
+ margin-right: 0.75rem;
+}
+
+/* line 246, node_modules/bootstrap/scss/_forms.scss */
+.form-check-inline .form-check-input {
+ position: static;
+ margin-top: 0;
+ margin-right: 0.3125rem;
+ margin-left: 0;
+}
+
+/* line 45, node_modules/bootstrap/scss/mixins/_forms.scss */
+.valid-feedback {
+ display: none;
+ width: 100%;
+ margin-top: 0.25rem;
+ font-size: 80%;
+ color: #f0ebe3;
+}
+
+/* line 53, node_modules/bootstrap/scss/mixins/_forms.scss */
+.valid-tooltip {
+ position: absolute;
+ top: 100%;
+ left: 0;
+ z-index: 5;
+ display: none;
+ max-width: 100%;
+ padding: 0.25rem 0.5rem;
+ margin-top: .1rem;
+ font-size: 0.875rem;
+ line-height: 1.5;
+ color: #464746;
+ background-color: rgba(240, 235, 227, 0.9);
+ border-radius: 0.25rem;
+}
+
+/* line 70, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated :valid ~ .valid-feedback,
+.was-validated :valid ~ .valid-tooltip,
+.is-valid ~ .valid-feedback,
+.is-valid ~ .valid-tooltip {
+ display: block;
+}
+
+/* line 33, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .form-control:valid, .form-control.is-valid {
+ border-color: #f0ebe3;
+ padding-right: calc(1.5em + 0.75rem);
+ background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='8' height='8' viewBox='0 0 8 8'%3e%3cpath fill='%23f0ebe3' d='M2.3 6.73L.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e");
+ background-repeat: no-repeat;
+ background-position: right calc(0.375em + 0.1875rem) center;
+ background-size: calc(0.75em + 0.375rem) calc(0.75em + 0.375rem);
+}
+
+/* line 88, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .form-control:valid:focus, .form-control.is-valid:focus {
+ border-color: #f0ebe3;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.25);
+ box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.25);
+}
+
+/* line 33, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated textarea.form-control:valid, textarea.form-control.is-valid {
+ padding-right: calc(1.5em + 0.75rem);
+ background-position: top calc(0.375em + 0.1875rem) right calc(0.375em + 0.1875rem);
+}
+
+/* line 33, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-select:valid, .custom-select.is-valid {
+ border-color: #f0ebe3;
+ padding-right: calc(0.75em + 2.3125rem);
+ background: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='4' height='5' viewBox='0 0 4 5'%3e%3cpath fill='%23343a40' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") no-repeat right 0.75rem center/8px 10px, url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='8' height='8' viewBox='0 0 8 8'%3e%3cpath fill='%23f0ebe3' d='M2.3 6.73L.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e") #ffffff no-repeat center right 1.75rem/calc(0.75em + 0.375rem) calc(0.75em + 0.375rem);
+}
+
+/* line 114, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-select:valid:focus, .custom-select.is-valid:focus {
+ border-color: #f0ebe3;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.25);
+ box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.25);
+}
+
+/* line 123, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .form-check-input:valid ~ .form-check-label, .form-check-input.is-valid ~ .form-check-label {
+ color: #f0ebe3;
+}
+
+/* line 127, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .form-check-input:valid ~ .valid-feedback,
+.was-validated .form-check-input:valid ~ .valid-tooltip, .form-check-input.is-valid ~ .valid-feedback,
+.form-check-input.is-valid ~ .valid-tooltip {
+ display: block;
+}
+
+/* line 136, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-control-input:valid ~ .custom-control-label, .custom-control-input.is-valid ~ .custom-control-label {
+ color: #f0ebe3;
+}
+
+/* line 139, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-control-input:valid ~ .custom-control-label::before, .custom-control-input.is-valid ~ .custom-control-label::before {
+ border-color: #f0ebe3;
+}
+
+/* line 145, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-control-input:valid:checked ~ .custom-control-label::before, .custom-control-input.is-valid:checked ~ .custom-control-label::before {
+ border-color: white;
+ background-color: white;
+}
+
+/* line 152, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-control-input:valid:focus ~ .custom-control-label::before, .custom-control-input.is-valid:focus ~ .custom-control-label::before {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.25);
+ box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.25);
+}
+
+/* line 156, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-control-input:valid:focus:not(:checked) ~ .custom-control-label::before, .custom-control-input.is-valid:focus:not(:checked) ~ .custom-control-label::before {
+ border-color: #f0ebe3;
+}
+
+/* line 166, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-file-input:valid ~ .custom-file-label, .custom-file-input.is-valid ~ .custom-file-label {
+ border-color: #f0ebe3;
+}
+
+/* line 171, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-file-input:valid:focus ~ .custom-file-label, .custom-file-input.is-valid:focus ~ .custom-file-label {
+ border-color: #f0ebe3;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.25);
+ box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.25);
+}
+
+/* line 45, node_modules/bootstrap/scss/mixins/_forms.scss */
+.invalid-feedback {
+ display: none;
+ width: 100%;
+ margin-top: 0.25rem;
+ font-size: 80%;
+ color: #e54a19;
+}
+
+/* line 53, node_modules/bootstrap/scss/mixins/_forms.scss */
+.invalid-tooltip {
+ position: absolute;
+ top: 100%;
+ left: 0;
+ z-index: 5;
+ display: none;
+ max-width: 100%;
+ padding: 0.25rem 0.5rem;
+ margin-top: .1rem;
+ font-size: 0.875rem;
+ line-height: 1.5;
+ color: #ffffff;
+ background-color: rgba(229, 74, 25, 0.9);
+ border-radius: 0.25rem;
+}
+
+/* line 70, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated :invalid ~ .invalid-feedback,
+.was-validated :invalid ~ .invalid-tooltip,
+.is-invalid ~ .invalid-feedback,
+.is-invalid ~ .invalid-tooltip {
+ display: block;
+}
+
+/* line 33, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .form-control:invalid, .form-control.is-invalid {
+ border-color: #e54a19;
+ padding-right: calc(1.5em + 0.75rem);
+ background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' fill='none' stroke='%23e54a19' viewBox='0 0 12 12'%3e%3ccircle cx='6' cy='6' r='4.5'/%3e%3cpath stroke-linejoin='round' d='M5.8 3.6h.4L6 6.5z'/%3e%3ccircle cx='6' cy='8.2' r='.6' fill='%23e54a19' stroke='none'/%3e%3c/svg%3e");
+ background-repeat: no-repeat;
+ background-position: right calc(0.375em + 0.1875rem) center;
+ background-size: calc(0.75em + 0.375rem) calc(0.75em + 0.375rem);
+}
+
+/* line 88, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .form-control:invalid:focus, .form-control.is-invalid:focus {
+ border-color: #e54a19;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(229, 74, 25, 0.25);
+ box-shadow: 0 0 0 0.2rem rgba(229, 74, 25, 0.25);
+}
+
+/* line 33, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated textarea.form-control:invalid, textarea.form-control.is-invalid {
+ padding-right: calc(1.5em + 0.75rem);
+ background-position: top calc(0.375em + 0.1875rem) right calc(0.375em + 0.1875rem);
+}
+
+/* line 33, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-select:invalid, .custom-select.is-invalid {
+ border-color: #e54a19;
+ padding-right: calc(0.75em + 2.3125rem);
+ background: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='4' height='5' viewBox='0 0 4 5'%3e%3cpath fill='%23343a40' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") no-repeat right 0.75rem center/8px 10px, url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' fill='none' stroke='%23e54a19' viewBox='0 0 12 12'%3e%3ccircle cx='6' cy='6' r='4.5'/%3e%3cpath stroke-linejoin='round' d='M5.8 3.6h.4L6 6.5z'/%3e%3ccircle cx='6' cy='8.2' r='.6' fill='%23e54a19' stroke='none'/%3e%3c/svg%3e") #ffffff no-repeat center right 1.75rem/calc(0.75em + 0.375rem) calc(0.75em + 0.375rem);
+}
+
+/* line 114, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-select:invalid:focus, .custom-select.is-invalid:focus {
+ border-color: #e54a19;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(229, 74, 25, 0.25);
+ box-shadow: 0 0 0 0.2rem rgba(229, 74, 25, 0.25);
+}
+
+/* line 123, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .form-check-input:invalid ~ .form-check-label, .form-check-input.is-invalid ~ .form-check-label {
+ color: #e54a19;
+}
+
+/* line 127, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .form-check-input:invalid ~ .invalid-feedback,
+.was-validated .form-check-input:invalid ~ .invalid-tooltip, .form-check-input.is-invalid ~ .invalid-feedback,
+.form-check-input.is-invalid ~ .invalid-tooltip {
+ display: block;
+}
+
+/* line 136, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-control-input:invalid ~ .custom-control-label, .custom-control-input.is-invalid ~ .custom-control-label {
+ color: #e54a19;
+}
+
+/* line 139, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-control-input:invalid ~ .custom-control-label::before, .custom-control-input.is-invalid ~ .custom-control-label::before {
+ border-color: #e54a19;
+}
+
+/* line 145, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-control-input:invalid:checked ~ .custom-control-label::before, .custom-control-input.is-invalid:checked ~ .custom-control-label::before {
+ border-color: #eb6e46;
+ background-color: #eb6e46;
+}
+
+/* line 152, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-control-input:invalid:focus ~ .custom-control-label::before, .custom-control-input.is-invalid:focus ~ .custom-control-label::before {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(229, 74, 25, 0.25);
+ box-shadow: 0 0 0 0.2rem rgba(229, 74, 25, 0.25);
+}
+
+/* line 156, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-control-input:invalid:focus:not(:checked) ~ .custom-control-label::before, .custom-control-input.is-invalid:focus:not(:checked) ~ .custom-control-label::before {
+ border-color: #e54a19;
+}
+
+/* line 166, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-file-input:invalid ~ .custom-file-label, .custom-file-input.is-invalid ~ .custom-file-label {
+ border-color: #e54a19;
+}
+
+/* line 171, node_modules/bootstrap/scss/mixins/_forms.scss */
+.was-validated .custom-file-input:invalid:focus ~ .custom-file-label, .custom-file-input.is-invalid:focus ~ .custom-file-label {
+ border-color: #e54a19;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(229, 74, 25, 0.25);
+ box-shadow: 0 0 0 0.2rem rgba(229, 74, 25, 0.25);
+}
+
+/* line 275, node_modules/bootstrap/scss/_forms.scss */
+.form-inline {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-flow: row wrap;
+ flex-flow: row wrap;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+}
+
+/* line 283, node_modules/bootstrap/scss/_forms.scss */
+.form-inline .form-check {
+ width: 100%;
+}
+
+@media (min-width: 576px) {
+ /* line 289, node_modules/bootstrap/scss/_forms.scss */
+ .form-inline label {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ -webkit-box-pack: center;
+ -ms-flex-pack: center;
+ justify-content: center;
+ margin-bottom: 0;
+ }
+ /* line 297, node_modules/bootstrap/scss/_forms.scss */
+ .form-inline .form-group {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-flex: 0;
+ -ms-flex: 0 0 auto;
+ flex: 0 0 auto;
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-flow: row wrap;
+ flex-flow: row wrap;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ margin-bottom: 0;
+ }
+ /* line 306, node_modules/bootstrap/scss/_forms.scss */
+ .form-inline .form-control {
+ display: inline-block;
+ width: auto;
+ vertical-align: middle;
+ }
+ /* line 313, node_modules/bootstrap/scss/_forms.scss */
+ .form-inline .form-control-plaintext {
+ display: inline-block;
+ }
+ /* line 317, node_modules/bootstrap/scss/_forms.scss */
+ .form-inline .input-group,
+ .form-inline .custom-select {
+ width: auto;
+ }
+ /* line 324, node_modules/bootstrap/scss/_forms.scss */
+ .form-inline .form-check {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ -webkit-box-pack: center;
+ -ms-flex-pack: center;
+ justify-content: center;
+ width: auto;
+ padding-left: 0;
+ }
+ /* line 331, node_modules/bootstrap/scss/_forms.scss */
+ .form-inline .form-check-input {
+ position: relative;
+ -ms-flex-negative: 0;
+ flex-shrink: 0;
+ margin-top: 0;
+ margin-right: 0.25rem;
+ margin-left: 0;
+ }
+ /* line 339, node_modules/bootstrap/scss/_forms.scss */
+ .form-inline .custom-control {
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ -webkit-box-pack: center;
+ -ms-flex-pack: center;
+ justify-content: center;
+ }
+ /* line 343, node_modules/bootstrap/scss/_forms.scss */
+ .form-inline .custom-control-label {
+ margin-bottom: 0;
+ }
+}
+
+/* line 7, node_modules/bootstrap/scss/_buttons.scss */
+.btn {
+ display: inline-block;
+ font-weight: 400;
+ color: #464746;
+ text-align: center;
+ vertical-align: middle;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+ background-color: transparent;
+ border: 1px solid transparent;
+ padding: 0.375rem 0.75rem;
+ font-size: 1rem;
+ line-height: 1.5;
+ border-radius: 0.25rem;
+ -webkit-transition: color 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+ transition: color 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+ transition: color 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;
+ transition: color 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 7, node_modules/bootstrap/scss/_buttons.scss */
+ .btn {
+ -webkit-transition: none;
+ transition: none;
+ }
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn:hover {
+ color: #464746;
+ text-decoration: none;
+}
+
+/* line 27, node_modules/bootstrap/scss/_buttons.scss */
+.btn:focus, .btn.focus {
+ outline: 0;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+ box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+}
+
+/* line 34, node_modules/bootstrap/scss/_buttons.scss */
+.btn.disabled, .btn:disabled {
+ opacity: 0.65;
+}
+
+/* line 40, node_modules/bootstrap/scss/_buttons.scss */
+.btn:not(:disabled):not(.disabled) {
+ cursor: pointer;
+}
+
+/* line 55, node_modules/bootstrap/scss/_buttons.scss */
+a.btn.disabled,
+fieldset:disabled a.btn {
+ pointer-events: none;
+}
+
+/* line 66, node_modules/bootstrap/scss/_buttons.scss */
+.btn-primary {
+ color: #ffffff;
+ background-color: #464746;
+ border-color: #464746;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-primary:hover {
+ color: #ffffff;
+ background-color: #333433;
+ border-color: #2d2d2d;
+}
+
+/* line 18, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-primary:focus, .btn-primary.focus {
+ color: #ffffff;
+ background-color: #333433;
+ border-color: #2d2d2d;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(98, 99, 98, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(98, 99, 98, 0.5);
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-primary.disabled, .btn-primary:disabled {
+ color: #ffffff;
+ background-color: #464746;
+ border-color: #464746;
+}
+
+/* line 43, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-primary:not(:disabled):not(.disabled):active, .btn-primary:not(:disabled):not(.disabled).active,
+.show > .btn-primary.dropdown-toggle {
+ color: #ffffff;
+ background-color: #2d2d2d;
+ border-color: #262726;
+}
+
+/* line 53, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-primary:not(:disabled):not(.disabled):active:focus, .btn-primary:not(:disabled):not(.disabled).active:focus,
+.show > .btn-primary.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(98, 99, 98, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(98, 99, 98, 0.5);
+}
+
+/* line 66, node_modules/bootstrap/scss/_buttons.scss */
+.btn-secondary {
+ color: #464746;
+ background-color: #f0ebe3;
+ border-color: #f0ebe3;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-secondary:hover {
+ color: #464746;
+ background-color: #e3d9ca;
+ border-color: #ded3c2;
+}
+
+/* line 18, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-secondary:focus, .btn-secondary.focus {
+ color: #464746;
+ background-color: #e3d9ca;
+ border-color: #ded3c2;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(215, 210, 203, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(215, 210, 203, 0.5);
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-secondary.disabled, .btn-secondary:disabled {
+ color: #464746;
+ background-color: #f0ebe3;
+ border-color: #f0ebe3;
+}
+
+/* line 43, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-secondary:not(:disabled):not(.disabled):active, .btn-secondary:not(:disabled):not(.disabled).active,
+.show > .btn-secondary.dropdown-toggle {
+ color: #464746;
+ background-color: #ded3c2;
+ border-color: #dacdb9;
+}
+
+/* line 53, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-secondary:not(:disabled):not(.disabled):active:focus, .btn-secondary:not(:disabled):not(.disabled).active:focus,
+.show > .btn-secondary.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(215, 210, 203, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(215, 210, 203, 0.5);
+}
+
+/* line 66, node_modules/bootstrap/scss/_buttons.scss */
+.btn-success {
+ color: #464746;
+ background-color: #f0ebe3;
+ border-color: #f0ebe3;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-success:hover {
+ color: #464746;
+ background-color: #e3d9ca;
+ border-color: #ded3c2;
+}
+
+/* line 18, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-success:focus, .btn-success.focus {
+ color: #464746;
+ background-color: #e3d9ca;
+ border-color: #ded3c2;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(215, 210, 203, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(215, 210, 203, 0.5);
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-success.disabled, .btn-success:disabled {
+ color: #464746;
+ background-color: #f0ebe3;
+ border-color: #f0ebe3;
+}
+
+/* line 43, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-success:not(:disabled):not(.disabled):active, .btn-success:not(:disabled):not(.disabled).active,
+.show > .btn-success.dropdown-toggle {
+ color: #464746;
+ background-color: #ded3c2;
+ border-color: #dacdb9;
+}
+
+/* line 53, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-success:not(:disabled):not(.disabled):active:focus, .btn-success:not(:disabled):not(.disabled).active:focus,
+.show > .btn-success.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(215, 210, 203, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(215, 210, 203, 0.5);
+}
+
+/* line 66, node_modules/bootstrap/scss/_buttons.scss */
+.btn-info {
+ color: #ffffff;
+ background-color: #464746;
+ border-color: #464746;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-info:hover {
+ color: #ffffff;
+ background-color: #333433;
+ border-color: #2d2d2d;
+}
+
+/* line 18, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-info:focus, .btn-info.focus {
+ color: #ffffff;
+ background-color: #333433;
+ border-color: #2d2d2d;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(98, 99, 98, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(98, 99, 98, 0.5);
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-info.disabled, .btn-info:disabled {
+ color: #ffffff;
+ background-color: #464746;
+ border-color: #464746;
+}
+
+/* line 43, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-info:not(:disabled):not(.disabled):active, .btn-info:not(:disabled):not(.disabled).active,
+.show > .btn-info.dropdown-toggle {
+ color: #ffffff;
+ background-color: #2d2d2d;
+ border-color: #262726;
+}
+
+/* line 53, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-info:not(:disabled):not(.disabled):active:focus, .btn-info:not(:disabled):not(.disabled).active:focus,
+.show > .btn-info.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(98, 99, 98, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(98, 99, 98, 0.5);
+}
+
+/* line 66, node_modules/bootstrap/scss/_buttons.scss */
+.btn-warning {
+ color: #ffffff;
+ background-color: #464746;
+ border-color: #464746;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-warning:hover {
+ color: #ffffff;
+ background-color: #333433;
+ border-color: #2d2d2d;
+}
+
+/* line 18, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-warning:focus, .btn-warning.focus {
+ color: #ffffff;
+ background-color: #333433;
+ border-color: #2d2d2d;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(98, 99, 98, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(98, 99, 98, 0.5);
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-warning.disabled, .btn-warning:disabled {
+ color: #ffffff;
+ background-color: #464746;
+ border-color: #464746;
+}
+
+/* line 43, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-warning:not(:disabled):not(.disabled):active, .btn-warning:not(:disabled):not(.disabled).active,
+.show > .btn-warning.dropdown-toggle {
+ color: #ffffff;
+ background-color: #2d2d2d;
+ border-color: #262726;
+}
+
+/* line 53, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-warning:not(:disabled):not(.disabled):active:focus, .btn-warning:not(:disabled):not(.disabled).active:focus,
+.show > .btn-warning.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(98, 99, 98, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(98, 99, 98, 0.5);
+}
+
+/* line 66, node_modules/bootstrap/scss/_buttons.scss */
+.btn-danger {
+ color: #ffffff;
+ background-color: #e54a19;
+ border-color: #e54a19;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-danger:hover {
+ color: #ffffff;
+ background-color: #c33f15;
+ border-color: #b73b14;
+}
+
+/* line 18, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-danger:focus, .btn-danger.focus {
+ color: #ffffff;
+ background-color: #c33f15;
+ border-color: #b73b14;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(233, 101, 60, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(233, 101, 60, 0.5);
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-danger.disabled, .btn-danger:disabled {
+ color: #ffffff;
+ background-color: #e54a19;
+ border-color: #e54a19;
+}
+
+/* line 43, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-danger:not(:disabled):not(.disabled):active, .btn-danger:not(:disabled):not(.disabled).active,
+.show > .btn-danger.dropdown-toggle {
+ color: #ffffff;
+ background-color: #b73b14;
+ border-color: #ac3713;
+}
+
+/* line 53, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-danger:not(:disabled):not(.disabled):active:focus, .btn-danger:not(:disabled):not(.disabled).active:focus,
+.show > .btn-danger.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(233, 101, 60, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(233, 101, 60, 0.5);
+}
+
+/* line 66, node_modules/bootstrap/scss/_buttons.scss */
+.btn-light {
+ color: #464746;
+ background-color: #f7f7f7;
+ border-color: #f7f7f7;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-light:hover {
+ color: #464746;
+ background-color: #e4e4e4;
+ border-color: #dedede;
+}
+
+/* line 18, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-light:focus, .btn-light.focus {
+ color: #464746;
+ background-color: #e4e4e4;
+ border-color: #dedede;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(220, 221, 220, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(220, 221, 220, 0.5);
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-light.disabled, .btn-light:disabled {
+ color: #464746;
+ background-color: #f7f7f7;
+ border-color: #f7f7f7;
+}
+
+/* line 43, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-light:not(:disabled):not(.disabled):active, .btn-light:not(:disabled):not(.disabled).active,
+.show > .btn-light.dropdown-toggle {
+ color: #464746;
+ background-color: #dedede;
+ border-color: #d7d7d7;
+}
+
+/* line 53, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-light:not(:disabled):not(.disabled):active:focus, .btn-light:not(:disabled):not(.disabled).active:focus,
+.show > .btn-light.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(220, 221, 220, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(220, 221, 220, 0.5);
+}
+
+/* line 66, node_modules/bootstrap/scss/_buttons.scss */
+.btn-dark {
+ color: #ffffff;
+ background-color: #343a40;
+ border-color: #343a40;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-dark:hover {
+ color: #ffffff;
+ background-color: #23272b;
+ border-color: #1d2124;
+}
+
+/* line 18, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-dark:focus, .btn-dark.focus {
+ color: #ffffff;
+ background-color: #23272b;
+ border-color: #1d2124;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(82, 88, 93, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(82, 88, 93, 0.5);
+}
+
+/* line 32, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-dark.disabled, .btn-dark:disabled {
+ color: #ffffff;
+ background-color: #343a40;
+ border-color: #343a40;
+}
+
+/* line 43, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-dark:not(:disabled):not(.disabled):active, .btn-dark:not(:disabled):not(.disabled).active,
+.show > .btn-dark.dropdown-toggle {
+ color: #ffffff;
+ background-color: #1d2124;
+ border-color: #171a1d;
+}
+
+/* line 53, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-dark:not(:disabled):not(.disabled):active:focus, .btn-dark:not(:disabled):not(.disabled).active:focus,
+.show > .btn-dark.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(82, 88, 93, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(82, 88, 93, 0.5);
+}
+
+/* line 72, node_modules/bootstrap/scss/_buttons.scss */
+.btn-outline-primary {
+ color: #464746;
+ border-color: #464746;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-outline-primary:hover {
+ color: #ffffff;
+ background-color: #464746;
+ border-color: #464746;
+}
+
+/* line 74, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-primary:focus, .btn-outline-primary.focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+}
+
+/* line 79, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-primary.disabled, .btn-outline-primary:disabled {
+ color: #464746;
+ background-color: transparent;
+}
+
+/* line 85, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-primary:not(:disabled):not(.disabled):active, .btn-outline-primary:not(:disabled):not(.disabled).active,
+.show > .btn-outline-primary.dropdown-toggle {
+ color: #ffffff;
+ background-color: #464746;
+ border-color: #464746;
+}
+
+/* line 92, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-primary:not(:disabled):not(.disabled):active:focus, .btn-outline-primary:not(:disabled):not(.disabled).active:focus,
+.show > .btn-outline-primary.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+}
+
+/* line 72, node_modules/bootstrap/scss/_buttons.scss */
+.btn-outline-secondary {
+ color: #f0ebe3;
+ border-color: #f0ebe3;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-outline-secondary:hover {
+ color: #464746;
+ background-color: #f0ebe3;
+ border-color: #f0ebe3;
+}
+
+/* line 74, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-secondary:focus, .btn-outline-secondary.focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.5);
+}
+
+/* line 79, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-secondary.disabled, .btn-outline-secondary:disabled {
+ color: #f0ebe3;
+ background-color: transparent;
+}
+
+/* line 85, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-secondary:not(:disabled):not(.disabled):active, .btn-outline-secondary:not(:disabled):not(.disabled).active,
+.show > .btn-outline-secondary.dropdown-toggle {
+ color: #464746;
+ background-color: #f0ebe3;
+ border-color: #f0ebe3;
+}
+
+/* line 92, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-secondary:not(:disabled):not(.disabled):active:focus, .btn-outline-secondary:not(:disabled):not(.disabled).active:focus,
+.show > .btn-outline-secondary.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.5);
+}
+
+/* line 72, node_modules/bootstrap/scss/_buttons.scss */
+.btn-outline-success {
+ color: #f0ebe3;
+ border-color: #f0ebe3;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-outline-success:hover {
+ color: #464746;
+ background-color: #f0ebe3;
+ border-color: #f0ebe3;
+}
+
+/* line 74, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-success:focus, .btn-outline-success.focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.5);
+}
+
+/* line 79, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-success.disabled, .btn-outline-success:disabled {
+ color: #f0ebe3;
+ background-color: transparent;
+}
+
+/* line 85, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-success:not(:disabled):not(.disabled):active, .btn-outline-success:not(:disabled):not(.disabled).active,
+.show > .btn-outline-success.dropdown-toggle {
+ color: #464746;
+ background-color: #f0ebe3;
+ border-color: #f0ebe3;
+}
+
+/* line 92, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-success:not(:disabled):not(.disabled):active:focus, .btn-outline-success:not(:disabled):not(.disabled).active:focus,
+.show > .btn-outline-success.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.5);
+}
+
+/* line 72, node_modules/bootstrap/scss/_buttons.scss */
+.btn-outline-info {
+ color: #464746;
+ border-color: #464746;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-outline-info:hover {
+ color: #ffffff;
+ background-color: #464746;
+ border-color: #464746;
+}
+
+/* line 74, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-info:focus, .btn-outline-info.focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+}
+
+/* line 79, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-info.disabled, .btn-outline-info:disabled {
+ color: #464746;
+ background-color: transparent;
+}
+
+/* line 85, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-info:not(:disabled):not(.disabled):active, .btn-outline-info:not(:disabled):not(.disabled).active,
+.show > .btn-outline-info.dropdown-toggle {
+ color: #ffffff;
+ background-color: #464746;
+ border-color: #464746;
+}
+
+/* line 92, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-info:not(:disabled):not(.disabled):active:focus, .btn-outline-info:not(:disabled):not(.disabled).active:focus,
+.show > .btn-outline-info.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+}
+
+/* line 72, node_modules/bootstrap/scss/_buttons.scss */
+.btn-outline-warning {
+ color: #464746;
+ border-color: #464746;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-outline-warning:hover {
+ color: #ffffff;
+ background-color: #464746;
+ border-color: #464746;
+}
+
+/* line 74, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-warning:focus, .btn-outline-warning.focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+}
+
+/* line 79, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-warning.disabled, .btn-outline-warning:disabled {
+ color: #464746;
+ background-color: transparent;
+}
+
+/* line 85, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-warning:not(:disabled):not(.disabled):active, .btn-outline-warning:not(:disabled):not(.disabled).active,
+.show > .btn-outline-warning.dropdown-toggle {
+ color: #ffffff;
+ background-color: #464746;
+ border-color: #464746;
+}
+
+/* line 92, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-warning:not(:disabled):not(.disabled):active:focus, .btn-outline-warning:not(:disabled):not(.disabled).active:focus,
+.show > .btn-outline-warning.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+}
+
+/* line 72, node_modules/bootstrap/scss/_buttons.scss */
+.btn-outline-danger {
+ color: #e54a19;
+ border-color: #e54a19;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-outline-danger:hover {
+ color: #ffffff;
+ background-color: #e54a19;
+ border-color: #e54a19;
+}
+
+/* line 74, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-danger:focus, .btn-outline-danger.focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(229, 74, 25, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(229, 74, 25, 0.5);
+}
+
+/* line 79, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-danger.disabled, .btn-outline-danger:disabled {
+ color: #e54a19;
+ background-color: transparent;
+}
+
+/* line 85, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-danger:not(:disabled):not(.disabled):active, .btn-outline-danger:not(:disabled):not(.disabled).active,
+.show > .btn-outline-danger.dropdown-toggle {
+ color: #ffffff;
+ background-color: #e54a19;
+ border-color: #e54a19;
+}
+
+/* line 92, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-danger:not(:disabled):not(.disabled):active:focus, .btn-outline-danger:not(:disabled):not(.disabled).active:focus,
+.show > .btn-outline-danger.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(229, 74, 25, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(229, 74, 25, 0.5);
+}
+
+/* line 72, node_modules/bootstrap/scss/_buttons.scss */
+.btn-outline-light {
+ color: #f7f7f7;
+ border-color: #f7f7f7;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-outline-light:hover {
+ color: #464746;
+ background-color: #f7f7f7;
+ border-color: #f7f7f7;
+}
+
+/* line 74, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-light:focus, .btn-outline-light.focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(247, 247, 247, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(247, 247, 247, 0.5);
+}
+
+/* line 79, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-light.disabled, .btn-outline-light:disabled {
+ color: #f7f7f7;
+ background-color: transparent;
+}
+
+/* line 85, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-light:not(:disabled):not(.disabled):active, .btn-outline-light:not(:disabled):not(.disabled).active,
+.show > .btn-outline-light.dropdown-toggle {
+ color: #464746;
+ background-color: #f7f7f7;
+ border-color: #f7f7f7;
+}
+
+/* line 92, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-light:not(:disabled):not(.disabled):active:focus, .btn-outline-light:not(:disabled):not(.disabled).active:focus,
+.show > .btn-outline-light.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(247, 247, 247, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(247, 247, 247, 0.5);
+}
+
+/* line 72, node_modules/bootstrap/scss/_buttons.scss */
+.btn-outline-dark {
+ color: #343a40;
+ border-color: #343a40;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-outline-dark:hover {
+ color: #ffffff;
+ background-color: #343a40;
+ border-color: #343a40;
+}
+
+/* line 74, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-dark:focus, .btn-outline-dark.focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(52, 58, 64, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(52, 58, 64, 0.5);
+}
+
+/* line 79, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-dark.disabled, .btn-outline-dark:disabled {
+ color: #343a40;
+ background-color: transparent;
+}
+
+/* line 85, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-dark:not(:disabled):not(.disabled):active, .btn-outline-dark:not(:disabled):not(.disabled).active,
+.show > .btn-outline-dark.dropdown-toggle {
+ color: #ffffff;
+ background-color: #343a40;
+ border-color: #343a40;
+}
+
+/* line 92, node_modules/bootstrap/scss/mixins/_buttons.scss */
+.btn-outline-dark:not(:disabled):not(.disabled):active:focus, .btn-outline-dark:not(:disabled):not(.disabled).active:focus,
+.show > .btn-outline-dark.dropdown-toggle:focus {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(52, 58, 64, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(52, 58, 64, 0.5);
+}
+
+/* line 83, node_modules/bootstrap/scss/_buttons.scss */
+.btn-link {
+ font-weight: 400;
+ color: #464746;
+ text-decoration: none;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-link:hover {
+ color: #202020;
+ text-decoration: underline;
+}
+
+/* line 93, node_modules/bootstrap/scss/_buttons.scss */
+.btn-link:focus, .btn-link.focus {
+ text-decoration: underline;
+}
+
+/* line 98, node_modules/bootstrap/scss/_buttons.scss */
+.btn-link:disabled, .btn-link.disabled {
+ color: #6c757d;
+ pointer-events: none;
+}
+
+/* line 112, node_modules/bootstrap/scss/_buttons.scss */
+.btn-lg, .btn-group-lg > .btn {
+ padding: 0.5rem 1rem;
+ font-size: 1.25rem;
+ line-height: 1.5;
+ border-radius: 0.3rem;
+}
+
+/* line 116, node_modules/bootstrap/scss/_buttons.scss */
+.btn-sm, .btn-group-sm > .btn {
+ padding: 0.25rem 0.5rem;
+ font-size: 0.875rem;
+ line-height: 1.5;
+ border-radius: 0.2rem;
+}
+
+/* line 125, node_modules/bootstrap/scss/_buttons.scss */
+.btn-block {
+ display: block;
+ width: 100%;
+}
+
+/* line 130, node_modules/bootstrap/scss/_buttons.scss */
+.btn-block + .btn-block {
+ margin-top: 0.5rem;
+}
+
+/* line 139, node_modules/bootstrap/scss/_buttons.scss */
+input[type="submit"].btn-block,
+input[type="reset"].btn-block,
+input[type="button"].btn-block {
+ width: 100%;
+}
+
+/* line 1, node_modules/bootstrap/scss/_transitions.scss */
+.fade {
+ -webkit-transition: opacity 0.15s linear;
+ transition: opacity 0.15s linear;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 1, node_modules/bootstrap/scss/_transitions.scss */
+ .fade {
+ -webkit-transition: none;
+ transition: none;
+ }
+}
+
+/* line 4, node_modules/bootstrap/scss/_transitions.scss */
+.fade:not(.show) {
+ opacity: 0;
+}
+
+/* line 10, node_modules/bootstrap/scss/_transitions.scss */
+.collapse:not(.show) {
+ display: none;
+}
+
+/* line 15, node_modules/bootstrap/scss/_transitions.scss */
+.collapsing {
+ position: relative;
+ height: 0;
+ overflow: hidden;
+ -webkit-transition: height 0.35s ease;
+ transition: height 0.35s ease;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 15, node_modules/bootstrap/scss/_transitions.scss */
+ .collapsing {
+ -webkit-transition: none;
+ transition: none;
+ }
+}
+
+/* line 2, node_modules/bootstrap/scss/_dropdown.scss */
+.dropup,
+.dropright,
+.dropdown,
+.dropleft {
+ position: relative;
+}
+
+/* line 9, node_modules/bootstrap/scss/_dropdown.scss */
+.dropdown-toggle {
+ white-space: nowrap;
+}
+
+/* line 30, node_modules/bootstrap/scss/mixins/_caret.scss */
+.dropdown-toggle::after {
+ display: inline-block;
+ margin-left: 0.255em;
+ vertical-align: 0.255em;
+ content: "";
+ border-top: 0.3em solid;
+ border-right: 0.3em solid transparent;
+ border-bottom: 0;
+ border-left: 0.3em solid transparent;
+}
+
+/* line 58, node_modules/bootstrap/scss/mixins/_caret.scss */
+.dropdown-toggle:empty::after {
+ margin-left: 0;
+}
+
+/* line 17, node_modules/bootstrap/scss/_dropdown.scss */
+.dropdown-menu {
+ position: absolute;
+ top: 100%;
+ left: 0;
+ z-index: 1000;
+ display: none;
+ float: left;
+ min-width: 10rem;
+ padding: 0.5rem 0;
+ margin: 0.125rem 0 0;
+ font-size: 1rem;
+ color: #464746;
+ text-align: left;
+ list-style: none;
+ background-color: #ffffff;
+ background-clip: padding-box;
+ border: 1px solid rgba(0, 0, 0, 0.15);
+ border-radius: 0.25rem;
+}
+
+/* line 42, node_modules/bootstrap/scss/_dropdown.scss */
+.dropdown-menu-left {
+ right: auto;
+ left: 0;
+}
+
+/* line 47, node_modules/bootstrap/scss/_dropdown.scss */
+.dropdown-menu-right {
+ right: 0;
+ left: auto;
+}
+
+@media (min-width: 576px) {
+ /* line 42, node_modules/bootstrap/scss/_dropdown.scss */
+ .dropdown-menu-sm-left {
+ right: auto;
+ left: 0;
+ }
+ /* line 47, node_modules/bootstrap/scss/_dropdown.scss */
+ .dropdown-menu-sm-right {
+ right: 0;
+ left: auto;
+ }
+}
+
+@media (min-width: 768px) {
+ /* line 42, node_modules/bootstrap/scss/_dropdown.scss */
+ .dropdown-menu-md-left {
+ right: auto;
+ left: 0;
+ }
+ /* line 47, node_modules/bootstrap/scss/_dropdown.scss */
+ .dropdown-menu-md-right {
+ right: 0;
+ left: auto;
+ }
+}
+
+@media (min-width: 1024px) {
+ /* line 42, node_modules/bootstrap/scss/_dropdown.scss */
+ .dropdown-menu-lg-left {
+ right: auto;
+ left: 0;
+ }
+ /* line 47, node_modules/bootstrap/scss/_dropdown.scss */
+ .dropdown-menu-lg-right {
+ right: 0;
+ left: auto;
+ }
+}
+
+@media (min-width: 1280px) {
+ /* line 42, node_modules/bootstrap/scss/_dropdown.scss */
+ .dropdown-menu-xl-left {
+ right: auto;
+ left: 0;
+ }
+ /* line 47, node_modules/bootstrap/scss/_dropdown.scss */
+ .dropdown-menu-xl-right {
+ right: 0;
+ left: auto;
+ }
+}
+
+@media (min-width: 1440px) {
+ /* line 42, node_modules/bootstrap/scss/_dropdown.scss */
+ .dropdown-menu-xxl-left {
+ right: auto;
+ left: 0;
+ }
+ /* line 47, node_modules/bootstrap/scss/_dropdown.scss */
+ .dropdown-menu-xxl-right {
+ right: 0;
+ left: auto;
+ }
+}
+
+/* line 57, node_modules/bootstrap/scss/_dropdown.scss */
+.dropup .dropdown-menu {
+ top: auto;
+ bottom: 100%;
+ margin-top: 0;
+ margin-bottom: 0.125rem;
+}
+
+/* line 30, node_modules/bootstrap/scss/mixins/_caret.scss */
+.dropup .dropdown-toggle::after {
+ display: inline-block;
+ margin-left: 0.255em;
+ vertical-align: 0.255em;
+ content: "";
+ border-top: 0;
+ border-right: 0.3em solid transparent;
+ border-bottom: 0.3em solid;
+ border-left: 0.3em solid transparent;
+}
+
+/* line 58, node_modules/bootstrap/scss/mixins/_caret.scss */
+.dropup .dropdown-toggle:empty::after {
+ margin-left: 0;
+}
+
+/* line 70, node_modules/bootstrap/scss/_dropdown.scss */
+.dropright .dropdown-menu {
+ top: 0;
+ right: auto;
+ left: 100%;
+ margin-top: 0;
+ margin-left: 0.125rem;
+}
+
+/* line 30, node_modules/bootstrap/scss/mixins/_caret.scss */
+.dropright .dropdown-toggle::after {
+ display: inline-block;
+ margin-left: 0.255em;
+ vertical-align: 0.255em;
+ content: "";
+ border-top: 0.3em solid transparent;
+ border-right: 0;
+ border-bottom: 0.3em solid transparent;
+ border-left: 0.3em solid;
+}
+
+/* line 58, node_modules/bootstrap/scss/mixins/_caret.scss */
+.dropright .dropdown-toggle:empty::after {
+ margin-left: 0;
+}
+
+/* line 80, node_modules/bootstrap/scss/_dropdown.scss */
+.dropright .dropdown-toggle::after {
+ vertical-align: 0;
+}
+
+/* line 87, node_modules/bootstrap/scss/_dropdown.scss */
+.dropleft .dropdown-menu {
+ top: 0;
+ right: 100%;
+ left: auto;
+ margin-top: 0;
+ margin-right: 0.125rem;
+}
+
+/* line 30, node_modules/bootstrap/scss/mixins/_caret.scss */
+.dropleft .dropdown-toggle::after {
+ display: inline-block;
+ margin-left: 0.255em;
+ vertical-align: 0.255em;
+ content: "";
+}
+
+/* line 45, node_modules/bootstrap/scss/mixins/_caret.scss */
+.dropleft .dropdown-toggle::after {
+ display: none;
+}
+
+/* line 49, node_modules/bootstrap/scss/mixins/_caret.scss */
+.dropleft .dropdown-toggle::before {
+ display: inline-block;
+ margin-right: 0.255em;
+ vertical-align: 0.255em;
+ content: "";
+ border-top: 0.3em solid transparent;
+ border-right: 0.3em solid;
+ border-bottom: 0.3em solid transparent;
+}
+
+/* line 58, node_modules/bootstrap/scss/mixins/_caret.scss */
+.dropleft .dropdown-toggle:empty::after {
+ margin-left: 0;
+}
+
+/* line 97, node_modules/bootstrap/scss/_dropdown.scss */
+.dropleft .dropdown-toggle::before {
+ vertical-align: 0;
+}
+
+/* line 106, node_modules/bootstrap/scss/_dropdown.scss */
+.dropdown-menu[x-placement^="top"], .dropdown-menu[x-placement^="right"], .dropdown-menu[x-placement^="bottom"], .dropdown-menu[x-placement^="left"] {
+ right: auto;
+ bottom: auto;
+}
+
+/* line 116, node_modules/bootstrap/scss/_dropdown.scss */
+.dropdown-divider {
+ height: 0;
+ margin: 0.5rem 0;
+ overflow: hidden;
+ border-top: 1px solid #eaebea;
+}
+
+/* line 123, node_modules/bootstrap/scss/_dropdown.scss */
+.dropdown-item {
+ display: block;
+ width: 100%;
+ padding: 0.25rem 1.5rem;
+ clear: both;
+ font-weight: 400;
+ color: #464746;
+ text-align: inherit;
+ white-space: nowrap;
+ background-color: transparent;
+ border: 0;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.dropdown-item:hover, .dropdown-item:focus {
+ color: #393a39;
+ text-decoration: none;
+ background-color: #f7f7f7;
+}
+
+/* line 154, node_modules/bootstrap/scss/_dropdown.scss */
+.dropdown-item.active, .dropdown-item:active {
+ color: #ffffff;
+ text-decoration: none;
+ background-color: #464746;
+}
+
+/* line 161, node_modules/bootstrap/scss/_dropdown.scss */
+.dropdown-item.disabled, .dropdown-item:disabled {
+ color: #6c757d;
+ pointer-events: none;
+ background-color: transparent;
+}
+
+/* line 173, node_modules/bootstrap/scss/_dropdown.scss */
+.dropdown-menu.show {
+ display: block;
+}
+
+/* line 178, node_modules/bootstrap/scss/_dropdown.scss */
+.dropdown-header {
+ display: block;
+ padding: 0.5rem 1.5rem;
+ margin-bottom: 0;
+ font-size: 0.875rem;
+ color: #6c757d;
+ white-space: nowrap;
+}
+
+/* line 188, node_modules/bootstrap/scss/_dropdown.scss */
+.dropdown-item-text {
+ display: block;
+ padding: 0.25rem 1.5rem;
+ color: #464746;
+}
+
+/* line 4, node_modules/bootstrap/scss/_button-group.scss */
+.btn-group,
+.btn-group-vertical {
+ position: relative;
+ display: -webkit-inline-box;
+ display: -ms-inline-flexbox;
+ display: inline-flex;
+ vertical-align: middle;
+}
+
+/* line 10, node_modules/bootstrap/scss/_button-group.scss */
+.btn-group > .btn,
+.btn-group-vertical > .btn {
+ position: relative;
+ -webkit-box-flex: 1;
+ -ms-flex: 1 1 auto;
+ flex: 1 1 auto;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.btn-group > .btn:hover,
+.btn-group-vertical > .btn:hover {
+ z-index: 1;
+}
+
+/* line 19, node_modules/bootstrap/scss/_button-group.scss */
+.btn-group > .btn:focus, .btn-group > .btn:active, .btn-group > .btn.active,
+.btn-group-vertical > .btn:focus,
+.btn-group-vertical > .btn:active,
+.btn-group-vertical > .btn.active {
+ z-index: 1;
+}
+
+/* line 28, node_modules/bootstrap/scss/_button-group.scss */
+.btn-toolbar {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -ms-flex-wrap: wrap;
+ flex-wrap: wrap;
+ -webkit-box-pack: start;
+ -ms-flex-pack: start;
+ justify-content: flex-start;
+}
+
+/* line 33, node_modules/bootstrap/scss/_button-group.scss */
+.btn-toolbar .input-group {
+ width: auto;
+}
+
+/* line 40, node_modules/bootstrap/scss/_button-group.scss */
+.btn-group > .btn:not(:first-child),
+.btn-group > .btn-group:not(:first-child) {
+ margin-left: -1px;
+}
+
+/* line 46, node_modules/bootstrap/scss/_button-group.scss */
+.btn-group > .btn:not(:last-child):not(.dropdown-toggle),
+.btn-group > .btn-group:not(:last-child) > .btn {
+ border-top-right-radius: 0;
+ border-bottom-right-radius: 0;
+}
+
+/* line 51, node_modules/bootstrap/scss/_button-group.scss */
+.btn-group > .btn:not(:first-child),
+.btn-group > .btn-group:not(:first-child) > .btn {
+ border-top-left-radius: 0;
+ border-bottom-left-radius: 0;
+}
+
+/* line 69, node_modules/bootstrap/scss/_button-group.scss */
+.dropdown-toggle-split {
+ padding-right: 0.5625rem;
+ padding-left: 0.5625rem;
+}
+
+/* line 73, node_modules/bootstrap/scss/_button-group.scss */
+.dropdown-toggle-split::after,
+.dropup .dropdown-toggle-split::after,
+.dropright .dropdown-toggle-split::after {
+ margin-left: 0;
+}
+
+/* line 79, node_modules/bootstrap/scss/_button-group.scss */
+.dropleft .dropdown-toggle-split::before {
+ margin-right: 0;
+}
+
+/* line 84, node_modules/bootstrap/scss/_button-group.scss */
+.btn-sm + .dropdown-toggle-split, .btn-group-sm > .btn + .dropdown-toggle-split {
+ padding-right: 0.375rem;
+ padding-left: 0.375rem;
+}
+
+/* line 89, node_modules/bootstrap/scss/_button-group.scss */
+.btn-lg + .dropdown-toggle-split, .btn-group-lg > .btn + .dropdown-toggle-split {
+ padding-right: 0.75rem;
+ padding-left: 0.75rem;
+}
+
+/* line 111, node_modules/bootstrap/scss/_button-group.scss */
+.btn-group-vertical {
+ -webkit-box-orient: vertical;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: column;
+ flex-direction: column;
+ -webkit-box-align: start;
+ -ms-flex-align: start;
+ align-items: flex-start;
+ -webkit-box-pack: center;
+ -ms-flex-pack: center;
+ justify-content: center;
+}
+
+/* line 116, node_modules/bootstrap/scss/_button-group.scss */
+.btn-group-vertical > .btn,
+.btn-group-vertical > .btn-group {
+ width: 100%;
+}
+
+/* line 121, node_modules/bootstrap/scss/_button-group.scss */
+.btn-group-vertical > .btn:not(:first-child),
+.btn-group-vertical > .btn-group:not(:first-child) {
+ margin-top: -1px;
+}
+
+/* line 127, node_modules/bootstrap/scss/_button-group.scss */
+.btn-group-vertical > .btn:not(:last-child):not(.dropdown-toggle),
+.btn-group-vertical > .btn-group:not(:last-child) > .btn {
+ border-bottom-right-radius: 0;
+ border-bottom-left-radius: 0;
+}
+
+/* line 132, node_modules/bootstrap/scss/_button-group.scss */
+.btn-group-vertical > .btn:not(:first-child),
+.btn-group-vertical > .btn-group:not(:first-child) > .btn {
+ border-top-left-radius: 0;
+ border-top-right-radius: 0;
+}
+
+/* line 152, node_modules/bootstrap/scss/_button-group.scss */
+.btn-group-toggle > .btn,
+.btn-group-toggle > .btn-group > .btn {
+ margin-bottom: 0;
+}
+
+/* line 156, node_modules/bootstrap/scss/_button-group.scss */
+.btn-group-toggle > .btn input[type="radio"],
+.btn-group-toggle > .btn input[type="checkbox"],
+.btn-group-toggle > .btn-group > .btn input[type="radio"],
+.btn-group-toggle > .btn-group > .btn input[type="checkbox"] {
+ position: absolute;
+ clip: rect(0, 0, 0, 0);
+ pointer-events: none;
+}
+
+/* line 7, node_modules/bootstrap/scss/_input-group.scss */
+.input-group {
+ position: relative;
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -ms-flex-wrap: wrap;
+ flex-wrap: wrap;
+ -webkit-box-align: stretch;
+ -ms-flex-align: stretch;
+ align-items: stretch;
+ width: 100%;
+}
+
+/* line 14, node_modules/bootstrap/scss/_input-group.scss */
+.input-group > .form-control,
+.input-group > .form-control-plaintext,
+.input-group > .custom-select,
+.input-group > .custom-file {
+ position: relative;
+ -webkit-box-flex: 1;
+ -ms-flex: 1 1 auto;
+ flex: 1 1 auto;
+ width: 1%;
+ min-width: 0;
+ margin-bottom: 0;
+}
+
+/* line 24, node_modules/bootstrap/scss/_input-group.scss */
+.input-group > .form-control + .form-control,
+.input-group > .form-control + .custom-select,
+.input-group > .form-control + .custom-file,
+.input-group > .form-control-plaintext + .form-control,
+.input-group > .form-control-plaintext + .custom-select,
+.input-group > .form-control-plaintext + .custom-file,
+.input-group > .custom-select + .form-control,
+.input-group > .custom-select + .custom-select,
+.input-group > .custom-select + .custom-file,
+.input-group > .custom-file + .form-control,
+.input-group > .custom-file + .custom-select,
+.input-group > .custom-file + .custom-file {
+ margin-left: -1px;
+}
+
+/* line 32, node_modules/bootstrap/scss/_input-group.scss */
+.input-group > .form-control:focus,
+.input-group > .custom-select:focus,
+.input-group > .custom-file .custom-file-input:focus ~ .custom-file-label {
+ z-index: 3;
+}
+
+/* line 39, node_modules/bootstrap/scss/_input-group.scss */
+.input-group > .custom-file .custom-file-input:focus {
+ z-index: 4;
+}
+
+/* line 45, node_modules/bootstrap/scss/_input-group.scss */
+.input-group > .form-control:not(:last-child),
+.input-group > .custom-select:not(:last-child) {
+ border-top-right-radius: 0;
+ border-bottom-right-radius: 0;
+}
+
+/* line 46, node_modules/bootstrap/scss/_input-group.scss */
+.input-group > .form-control:not(:first-child),
+.input-group > .custom-select:not(:first-child) {
+ border-top-left-radius: 0;
+ border-bottom-left-radius: 0;
+}
+
+/* line 51, node_modules/bootstrap/scss/_input-group.scss */
+.input-group > .custom-file {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+}
+
+/* line 55, node_modules/bootstrap/scss/_input-group.scss */
+.input-group > .custom-file:not(:last-child) .custom-file-label,
+.input-group > .custom-file:not(:last-child) .custom-file-label::after {
+ border-top-right-radius: 0;
+ border-bottom-right-radius: 0;
+}
+
+/* line 57, node_modules/bootstrap/scss/_input-group.scss */
+.input-group > .custom-file:not(:first-child) .custom-file-label {
+ border-top-left-radius: 0;
+ border-bottom-left-radius: 0;
+}
+
+/* line 68, node_modules/bootstrap/scss/_input-group.scss */
+.input-group-prepend,
+.input-group-append {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+}
+
+/* line 75, node_modules/bootstrap/scss/_input-group.scss */
+.input-group-prepend .btn,
+.input-group-append .btn {
+ position: relative;
+ z-index: 2;
+}
+
+/* line 79, node_modules/bootstrap/scss/_input-group.scss */
+.input-group-prepend .btn:focus,
+.input-group-append .btn:focus {
+ z-index: 3;
+}
+
+/* line 84, node_modules/bootstrap/scss/_input-group.scss */
+.input-group-prepend .btn + .btn,
+.input-group-prepend .btn + .input-group-text,
+.input-group-prepend .input-group-text + .input-group-text,
+.input-group-prepend .input-group-text + .btn,
+.input-group-append .btn + .btn,
+.input-group-append .btn + .input-group-text,
+.input-group-append .input-group-text + .input-group-text,
+.input-group-append .input-group-text + .btn {
+ margin-left: -1px;
+}
+
+/* line 92, node_modules/bootstrap/scss/_input-group.scss */
+.input-group-prepend {
+ margin-right: -1px;
+}
+
+/* line 93, node_modules/bootstrap/scss/_input-group.scss */
+.input-group-append {
+ margin-left: -1px;
+}
+
+/* line 101, node_modules/bootstrap/scss/_input-group.scss */
+.input-group-text {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ padding: 0.375rem 0.75rem;
+ margin-bottom: 0;
+ font-size: 1rem;
+ font-weight: 400;
+ line-height: 1.5;
+ color: #495057;
+ text-align: center;
+ white-space: nowrap;
+ background-color: #eaebea;
+ border: 1px solid #ced4da;
+ border-radius: 0.25rem;
+}
+
+/* line 117, node_modules/bootstrap/scss/_input-group.scss */
+.input-group-text input[type="radio"],
+.input-group-text input[type="checkbox"] {
+ margin-top: 0;
+}
+
+/* line 129, node_modules/bootstrap/scss/_input-group.scss */
+.input-group-lg > .form-control:not(textarea),
+.input-group-lg > .custom-select {
+ height: calc(1.5em + 1rem + 2px);
+}
+
+/* line 134, node_modules/bootstrap/scss/_input-group.scss */
+.input-group-lg > .form-control,
+.input-group-lg > .custom-select,
+.input-group-lg > .input-group-prepend > .input-group-text,
+.input-group-lg > .input-group-append > .input-group-text,
+.input-group-lg > .input-group-prepend > .btn,
+.input-group-lg > .input-group-append > .btn {
+ padding: 0.5rem 1rem;
+ font-size: 1.25rem;
+ line-height: 1.5;
+ border-radius: 0.3rem;
+}
+
+/* line 146, node_modules/bootstrap/scss/_input-group.scss */
+.input-group-sm > .form-control:not(textarea),
+.input-group-sm > .custom-select {
+ height: calc(1.5em + 0.5rem + 2px);
+}
+
+/* line 151, node_modules/bootstrap/scss/_input-group.scss */
+.input-group-sm > .form-control,
+.input-group-sm > .custom-select,
+.input-group-sm > .input-group-prepend > .input-group-text,
+.input-group-sm > .input-group-append > .input-group-text,
+.input-group-sm > .input-group-prepend > .btn,
+.input-group-sm > .input-group-append > .btn {
+ padding: 0.25rem 0.5rem;
+ font-size: 0.875rem;
+ line-height: 1.5;
+ border-radius: 0.2rem;
+}
+
+/* line 163, node_modules/bootstrap/scss/_input-group.scss */
+.input-group-lg > .custom-select,
+.input-group-sm > .custom-select {
+ padding-right: 1.75rem;
+}
+
+/* line 176, node_modules/bootstrap/scss/_input-group.scss */
+.input-group > .input-group-prepend > .btn,
+.input-group > .input-group-prepend > .input-group-text,
+.input-group > .input-group-append:not(:last-child) > .btn,
+.input-group > .input-group-append:not(:last-child) > .input-group-text,
+.input-group > .input-group-append:last-child > .btn:not(:last-child):not(.dropdown-toggle),
+.input-group > .input-group-append:last-child > .input-group-text:not(:last-child) {
+ border-top-right-radius: 0;
+ border-bottom-right-radius: 0;
+}
+
+/* line 185, node_modules/bootstrap/scss/_input-group.scss */
+.input-group > .input-group-append > .btn,
+.input-group > .input-group-append > .input-group-text,
+.input-group > .input-group-prepend:not(:first-child) > .btn,
+.input-group > .input-group-prepend:not(:first-child) > .input-group-text,
+.input-group > .input-group-prepend:first-child > .btn:not(:first-child),
+.input-group > .input-group-prepend:first-child > .input-group-text:not(:first-child) {
+ border-top-left-radius: 0;
+ border-bottom-left-radius: 0;
+}
+
+/* line 10, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-control {
+ position: relative;
+ z-index: 1;
+ display: block;
+ min-height: 1.5rem;
+ padding-left: 1.5rem;
+}
+
+/* line 18, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-control-inline {
+ display: -webkit-inline-box;
+ display: -ms-inline-flexbox;
+ display: inline-flex;
+ margin-right: 1rem;
+}
+
+/* line 23, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-control-input {
+ position: absolute;
+ left: 0;
+ z-index: -1;
+ width: 1rem;
+ height: 1.25rem;
+ opacity: 0;
+}
+
+/* line 31, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-control-input:checked ~ .custom-control-label::before {
+ color: #ffffff;
+ border-color: #464746;
+ background-color: #464746;
+}
+
+/* line 38, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-control-input:focus ~ .custom-control-label::before {
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+ box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+}
+
+/* line 47, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-control-input:focus:not(:checked) ~ .custom-control-label::before {
+ border-color: #858785;
+}
+
+/* line 51, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-control-input:not(:disabled):active ~ .custom-control-label::before {
+ color: #ffffff;
+ background-color: #9fa09f;
+ border-color: #9fa09f;
+}
+
+/* line 61, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-control-input[disabled] ~ .custom-control-label, .custom-control-input:disabled ~ .custom-control-label {
+ color: #6c757d;
+}
+
+/* line 64, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-control-input[disabled] ~ .custom-control-label::before, .custom-control-input:disabled ~ .custom-control-label::before {
+ background-color: #eaebea;
+}
+
+/* line 75, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-control-label {
+ position: relative;
+ margin-bottom: 0;
+ vertical-align: top;
+}
+
+/* line 83, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-control-label::before {
+ position: absolute;
+ top: 0.25rem;
+ left: -1.5rem;
+ display: block;
+ width: 1rem;
+ height: 1rem;
+ pointer-events: none;
+ content: "";
+ background-color: #ffffff;
+ border: #adb5bd solid 1px;
+}
+
+/* line 98, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-control-label::after {
+ position: absolute;
+ top: 0.25rem;
+ left: -1.5rem;
+ display: block;
+ width: 1rem;
+ height: 1rem;
+ content: "";
+ background: no-repeat 50% / 50% 50%;
+}
+
+/* line 116, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-checkbox .custom-control-label::before {
+ border-radius: 0.25rem;
+}
+
+/* line 121, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-checkbox .custom-control-input:checked ~ .custom-control-label::after {
+ background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='8' height='8' viewBox='0 0 8 8'%3e%3cpath fill='%23ffffff' d='M6.564.75l-3.59 3.612-1.538-1.55L0 4.26l2.974 2.99L8 2.193z'/%3e%3c/svg%3e");
+}
+
+/* line 127, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-checkbox .custom-control-input:indeterminate ~ .custom-control-label::before {
+ border-color: #464746;
+ background-color: #464746;
+}
+
+/* line 132, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-checkbox .custom-control-input:indeterminate ~ .custom-control-label::after {
+ background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='4' height='4' viewBox='0 0 4 4'%3e%3cpath stroke='%23ffffff' d='M0 2h4'/%3e%3c/svg%3e");
+}
+
+/* line 138, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-checkbox .custom-control-input:disabled:checked ~ .custom-control-label::before {
+ background-color: rgba(70, 71, 70, 0.5);
+}
+
+/* line 141, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-checkbox .custom-control-input:disabled:indeterminate ~ .custom-control-label::before {
+ background-color: rgba(70, 71, 70, 0.5);
+}
+
+/* line 152, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-radio .custom-control-label::before {
+ border-radius: 50%;
+}
+
+/* line 158, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-radio .custom-control-input:checked ~ .custom-control-label::after {
+ background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='%23ffffff'/%3e%3c/svg%3e");
+}
+
+/* line 164, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-radio .custom-control-input:disabled:checked ~ .custom-control-label::before {
+ background-color: rgba(70, 71, 70, 0.5);
+}
+
+/* line 175, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-switch {
+ padding-left: 2.25rem;
+}
+
+/* line 179, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-switch .custom-control-label::before {
+ left: -2.25rem;
+ width: 1.75rem;
+ pointer-events: all;
+ border-radius: 0.5rem;
+}
+
+/* line 187, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-switch .custom-control-label::after {
+ top: calc(0.25rem + 2px);
+ left: calc(-2.25rem + 2px);
+ width: calc(1rem - 4px);
+ height: calc(1rem - 4px);
+ background-color: #adb5bd;
+ border-radius: 0.5rem;
+ -webkit-transition: background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-transform 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+ transition: background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-transform 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+ transition: transform 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;
+ transition: transform 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out, -webkit-transform 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 187, node_modules/bootstrap/scss/_custom-forms.scss */
+ .custom-switch .custom-control-label::after {
+ -webkit-transition: none;
+ transition: none;
+ }
+}
+
+/* line 200, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-switch .custom-control-input:checked ~ .custom-control-label::after {
+ background-color: #ffffff;
+ -webkit-transform: translateX(0.75rem);
+ transform: translateX(0.75rem);
+}
+
+/* line 207, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-switch .custom-control-input:disabled:checked ~ .custom-control-label::before {
+ background-color: rgba(70, 71, 70, 0.5);
+}
+
+/* line 220, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-select {
+ display: inline-block;
+ width: 100%;
+ height: calc(1.5em + 0.75rem + 2px);
+ padding: 0.375rem 1.75rem 0.375rem 0.75rem;
+ font-size: 1rem;
+ font-weight: 300;
+ line-height: 1.5;
+ color: #495057;
+ vertical-align: middle;
+ background: #ffffff url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='4' height='5' viewBox='0 0 4 5'%3e%3cpath fill='%23343a40' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") no-repeat right 0.75rem center/8px 10px;
+ border: 1px solid #ced4da;
+ border-radius: 0.25rem;
+ -webkit-appearance: none;
+ -moz-appearance: none;
+ appearance: none;
+}
+
+/* line 237, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-select:focus {
+ border-color: #858785;
+ outline: 0;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+ box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+}
+
+/* line 247, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-select:focus::-ms-value {
+ color: #495057;
+ background-color: #ffffff;
+}
+
+/* line 258, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-select[multiple], .custom-select[size]:not([size="1"]) {
+ height: auto;
+ padding-right: 0.75rem;
+ background-image: none;
+}
+
+/* line 265, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-select:disabled {
+ color: #6c757d;
+ background-color: #eaebea;
+}
+
+/* line 271, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-select::-ms-expand {
+ display: none;
+}
+
+/* line 276, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-select:-moz-focusring {
+ color: transparent;
+ text-shadow: 0 0 0 #495057;
+}
+
+/* line 282, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-select-sm {
+ height: calc(1.5em + 0.5rem + 2px);
+ padding-top: 0.25rem;
+ padding-bottom: 0.25rem;
+ padding-left: 0.5rem;
+ font-size: 0.875rem;
+}
+
+/* line 290, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-select-lg {
+ height: calc(1.5em + 1rem + 2px);
+ padding-top: 0.5rem;
+ padding-bottom: 0.5rem;
+ padding-left: 1rem;
+ font-size: 1.25rem;
+}
+
+/* line 303, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-file {
+ position: relative;
+ display: inline-block;
+ width: 100%;
+ height: calc(1.5em + 0.75rem + 2px);
+ margin-bottom: 0;
+}
+
+/* line 311, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-file-input {
+ position: relative;
+ z-index: 2;
+ width: 100%;
+ height: calc(1.5em + 0.75rem + 2px);
+ margin: 0;
+ opacity: 0;
+}
+
+/* line 319, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-file-input:focus ~ .custom-file-label {
+ border-color: #858785;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+ box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+}
+
+/* line 325, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-file-input[disabled] ~ .custom-file-label,
+.custom-file-input:disabled ~ .custom-file-label {
+ background-color: #eaebea;
+}
+
+/* line 331, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-file-input:lang(en) ~ .custom-file-label::after {
+ content: "Browse";
+}
+
+/* line 336, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-file-input ~ .custom-file-label[data-browse]::after {
+ content: attr(data-browse);
+}
+
+/* line 341, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-file-label {
+ position: absolute;
+ top: 0;
+ right: 0;
+ left: 0;
+ z-index: 1;
+ height: calc(1.5em + 0.75rem + 2px);
+ padding: 0.375rem 0.75rem;
+ font-weight: 300;
+ line-height: 1.5;
+ color: #495057;
+ background-color: #ffffff;
+ border: 1px solid #ced4da;
+ border-radius: 0.25rem;
+}
+
+/* line 358, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-file-label::after {
+ position: absolute;
+ top: 0;
+ right: 0;
+ bottom: 0;
+ z-index: 3;
+ display: block;
+ height: calc(1.5em + 0.75rem);
+ padding: 0.375rem 0.75rem;
+ line-height: 1.5;
+ color: #495057;
+ content: "Browse";
+ background-color: #eaebea;
+ border-left: inherit;
+ border-radius: 0 0.25rem 0.25rem 0;
+}
+
+/* line 382, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range {
+ width: 100%;
+ height: 1.4rem;
+ padding: 0;
+ background-color: transparent;
+ -webkit-appearance: none;
+ -moz-appearance: none;
+ appearance: none;
+}
+
+/* line 389, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range:focus {
+ outline: none;
+}
+
+/* line 394, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range:focus::-webkit-slider-thumb {
+ -webkit-box-shadow: 0 0 0 1px #ffffff, 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+ box-shadow: 0 0 0 1px #ffffff, 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+}
+
+/* line 395, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range:focus::-moz-range-thumb {
+ box-shadow: 0 0 0 1px #ffffff, 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+}
+
+/* line 396, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range:focus::-ms-thumb {
+ box-shadow: 0 0 0 1px #ffffff, 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+}
+
+/* line 399, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range::-moz-focus-outer {
+ border: 0;
+}
+
+/* line 403, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range::-webkit-slider-thumb {
+ width: 1rem;
+ height: 1rem;
+ margin-top: -0.25rem;
+ background-color: #464746;
+ border: 0;
+ border-radius: 1rem;
+ -webkit-transition: background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+ transition: background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+ transition: background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;
+ transition: background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+ -webkit-appearance: none;
+ appearance: none;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 403, node_modules/bootstrap/scss/_custom-forms.scss */
+ .custom-range::-webkit-slider-thumb {
+ -webkit-transition: none;
+ transition: none;
+ }
+}
+
+/* line 414, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range::-webkit-slider-thumb:active {
+ background-color: #9fa09f;
+}
+
+/* line 419, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range::-webkit-slider-runnable-track {
+ width: 100%;
+ height: 0.5rem;
+ color: transparent;
+ cursor: pointer;
+ background-color: #dee2e6;
+ border-color: transparent;
+ border-radius: 1rem;
+}
+
+/* line 430, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range::-moz-range-thumb {
+ width: 1rem;
+ height: 1rem;
+ background-color: #464746;
+ border: 0;
+ border-radius: 1rem;
+ -moz-transition: background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;
+ transition: background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;
+ -moz-appearance: none;
+ appearance: none;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 430, node_modules/bootstrap/scss/_custom-forms.scss */
+ .custom-range::-moz-range-thumb {
+ -moz-transition: none;
+ transition: none;
+ }
+}
+
+/* line 440, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range::-moz-range-thumb:active {
+ background-color: #9fa09f;
+}
+
+/* line 445, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range::-moz-range-track {
+ width: 100%;
+ height: 0.5rem;
+ color: transparent;
+ cursor: pointer;
+ background-color: #dee2e6;
+ border-color: transparent;
+ border-radius: 1rem;
+}
+
+/* line 456, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range::-ms-thumb {
+ width: 1rem;
+ height: 1rem;
+ margin-top: 0;
+ margin-right: 0.2rem;
+ margin-left: 0.2rem;
+ background-color: #464746;
+ border: 0;
+ border-radius: 1rem;
+ -ms-transition: background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;
+ transition: background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;
+ appearance: none;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 456, node_modules/bootstrap/scss/_custom-forms.scss */
+ .custom-range::-ms-thumb {
+ -ms-transition: none;
+ transition: none;
+ }
+}
+
+/* line 469, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range::-ms-thumb:active {
+ background-color: #9fa09f;
+}
+
+/* line 474, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range::-ms-track {
+ width: 100%;
+ height: 0.5rem;
+ color: transparent;
+ cursor: pointer;
+ background-color: transparent;
+ border-color: transparent;
+ border-width: 0.5rem;
+}
+
+/* line 485, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range::-ms-fill-lower {
+ background-color: #dee2e6;
+ border-radius: 1rem;
+}
+
+/* line 490, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range::-ms-fill-upper {
+ margin-right: 15px;
+ background-color: #dee2e6;
+ border-radius: 1rem;
+}
+
+/* line 497, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range:disabled::-webkit-slider-thumb {
+ background-color: #adb5bd;
+}
+
+/* line 501, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range:disabled::-webkit-slider-runnable-track {
+ cursor: default;
+}
+
+/* line 505, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range:disabled::-moz-range-thumb {
+ background-color: #adb5bd;
+}
+
+/* line 509, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range:disabled::-moz-range-track {
+ cursor: default;
+}
+
+/* line 513, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-range:disabled::-ms-thumb {
+ background-color: #adb5bd;
+}
+
+/* line 519, node_modules/bootstrap/scss/_custom-forms.scss */
+.custom-control-label::before,
+.custom-file-label,
+.custom-select {
+ -webkit-transition: background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+ transition: background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+ transition: background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;
+ transition: background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 519, node_modules/bootstrap/scss/_custom-forms.scss */
+ .custom-control-label::before,
+ .custom-file-label,
+ .custom-select {
+ -webkit-transition: none;
+ transition: none;
+ }
+}
+
+/* line 6, node_modules/bootstrap/scss/_nav.scss */
+.nav {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -ms-flex-wrap: wrap;
+ flex-wrap: wrap;
+ padding-left: 0;
+ margin-bottom: 0;
+ list-style: none;
+}
+
+/* line 14, node_modules/bootstrap/scss/_nav.scss */
+.nav-link {
+ display: block;
+ padding: 0.5rem 1rem;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.nav-link:hover, .nav-link:focus {
+ text-decoration: none;
+}
+
+/* line 24, node_modules/bootstrap/scss/_nav.scss */
+.nav-link.disabled {
+ color: #6c757d;
+ pointer-events: none;
+ cursor: default;
+}
+
+/* line 35, node_modules/bootstrap/scss/_nav.scss */
+.nav-tabs {
+ border-bottom: 1px solid #dee2e6;
+}
+
+/* line 38, node_modules/bootstrap/scss/_nav.scss */
+.nav-tabs .nav-item {
+ margin-bottom: -1px;
+}
+
+/* line 42, node_modules/bootstrap/scss/_nav.scss */
+.nav-tabs .nav-link {
+ border: 1px solid transparent;
+ border-top-left-radius: 0.25rem;
+ border-top-right-radius: 0.25rem;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.nav-tabs .nav-link:hover, .nav-tabs .nav-link:focus {
+ border-color: #eaebea #eaebea #dee2e6;
+}
+
+/* line 50, node_modules/bootstrap/scss/_nav.scss */
+.nav-tabs .nav-link.disabled {
+ color: #6c757d;
+ background-color: transparent;
+ border-color: transparent;
+}
+
+/* line 57, node_modules/bootstrap/scss/_nav.scss */
+.nav-tabs .nav-link.active,
+.nav-tabs .nav-item.show .nav-link {
+ color: #495057;
+ background-color: #ffffff;
+ border-color: #dee2e6 #dee2e6 #ffffff;
+}
+
+/* line 64, node_modules/bootstrap/scss/_nav.scss */
+.nav-tabs .dropdown-menu {
+ margin-top: -1px;
+ border-top-left-radius: 0;
+ border-top-right-radius: 0;
+}
+
+/* line 78, node_modules/bootstrap/scss/_nav.scss */
+.nav-pills .nav-link {
+ border-radius: 0.25rem;
+}
+
+/* line 82, node_modules/bootstrap/scss/_nav.scss */
+.nav-pills .nav-link.active,
+.nav-pills .show > .nav-link {
+ color: #ffffff;
+ background-color: #464746;
+}
+
+/* line 95, node_modules/bootstrap/scss/_nav.scss */
+.nav-fill > .nav-link,
+.nav-fill .nav-item {
+ -webkit-box-flex: 1;
+ -ms-flex: 1 1 auto;
+ flex: 1 1 auto;
+ text-align: center;
+}
+
+/* line 103, node_modules/bootstrap/scss/_nav.scss */
+.nav-justified > .nav-link,
+.nav-justified .nav-item {
+ -ms-flex-preferred-size: 0;
+ flex-basis: 0;
+ -webkit-box-flex: 1;
+ -ms-flex-positive: 1;
+ flex-grow: 1;
+ text-align: center;
+}
+
+/* line 117, node_modules/bootstrap/scss/_nav.scss */
+.tab-content > .tab-pane {
+ display: none;
+}
+
+/* line 120, node_modules/bootstrap/scss/_nav.scss */
+.tab-content > .active {
+ display: block;
+}
+
+/* line 18, node_modules/bootstrap/scss/_navbar.scss */
+.navbar {
+ position: relative;
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -ms-flex-wrap: wrap;
+ flex-wrap: wrap;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ -webkit-box-pack: justify;
+ -ms-flex-pack: justify;
+ justify-content: space-between;
+ padding: 0.5rem 1rem;
+}
+
+/* line 28, node_modules/bootstrap/scss/_navbar.scss */
+.navbar .container,
+.navbar .container-fluid, .navbar .container-sm, .navbar .container-md, .navbar .container-lg, .navbar .container-xl, .navbar .container-xxl {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -ms-flex-wrap: wrap;
+ flex-wrap: wrap;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ -webkit-box-pack: justify;
+ -ms-flex-pack: justify;
+ justify-content: space-between;
+}
+
+/* line 52, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-brand {
+ display: inline-block;
+ padding-top: 0.3125rem;
+ padding-bottom: 0.3125rem;
+ margin-right: 1rem;
+ font-size: 1.25rem;
+ line-height: inherit;
+ white-space: nowrap;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.navbar-brand:hover, .navbar-brand:focus {
+ text-decoration: none;
+}
+
+/* line 71, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-nav {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-orient: vertical;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: column;
+ flex-direction: column;
+ padding-left: 0;
+ margin-bottom: 0;
+ list-style: none;
+}
+
+/* line 78, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-nav .nav-link {
+ padding-right: 0;
+ padding-left: 0;
+}
+
+/* line 83, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-nav .dropdown-menu {
+ position: static;
+ float: none;
+}
+
+/* line 94, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-text {
+ display: inline-block;
+ padding-top: 0.5rem;
+ padding-bottom: 0.5rem;
+}
+
+/* line 109, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-collapse {
+ -ms-flex-preferred-size: 100%;
+ flex-basis: 100%;
+ -webkit-box-flex: 1;
+ -ms-flex-positive: 1;
+ flex-grow: 1;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+}
+
+/* line 118, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-toggler {
+ padding: 0.25rem 0.75rem;
+ font-size: 1.25rem;
+ line-height: 1;
+ background-color: transparent;
+ border: 1px solid transparent;
+ border-radius: 0.25rem;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.navbar-toggler:hover, .navbar-toggler:focus {
+ text-decoration: none;
+}
+
+/* line 133, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-toggler-icon {
+ display: inline-block;
+ width: 1.5em;
+ height: 1.5em;
+ vertical-align: middle;
+ content: "";
+ background: no-repeat center center;
+ background-size: 100% 100%;
+}
+
+@media (max-width: 575.98px) {
+ /* line 152, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-sm > .container,
+ .navbar-expand-sm > .container-fluid, .navbar-expand-sm > .container-sm, .navbar-expand-sm > .container-md, .navbar-expand-sm > .container-lg, .navbar-expand-sm > .container-xl, .navbar-expand-sm > .container-xxl {
+ padding-right: 0;
+ padding-left: 0;
+ }
+}
+
+@media (min-width: 576px) {
+ /* line 150, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-sm {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-flow: row nowrap;
+ flex-flow: row nowrap;
+ -webkit-box-pack: start;
+ -ms-flex-pack: start;
+ justify-content: flex-start;
+ }
+ /* line 173, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-sm .navbar-nav {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ }
+ /* line 176, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-sm .navbar-nav .dropdown-menu {
+ position: absolute;
+ }
+ /* line 180, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-sm .navbar-nav .nav-link {
+ padding-right: 0.5rem;
+ padding-left: 0.5rem;
+ }
+ /* line 187, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-sm > .container,
+ .navbar-expand-sm > .container-fluid, .navbar-expand-sm > .container-sm, .navbar-expand-sm > .container-md, .navbar-expand-sm > .container-lg, .navbar-expand-sm > .container-xl, .navbar-expand-sm > .container-xxl {
+ -ms-flex-wrap: nowrap;
+ flex-wrap: nowrap;
+ }
+ /* line 202, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-sm .navbar-collapse {
+ display: -webkit-box !important;
+ display: -ms-flexbox !important;
+ display: flex !important;
+ -ms-flex-preferred-size: auto;
+ flex-basis: auto;
+ }
+ /* line 209, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-sm .navbar-toggler {
+ display: none;
+ }
+}
+
+@media (max-width: 767.98px) {
+ /* line 152, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-md > .container,
+ .navbar-expand-md > .container-fluid, .navbar-expand-md > .container-sm, .navbar-expand-md > .container-md, .navbar-expand-md > .container-lg, .navbar-expand-md > .container-xl, .navbar-expand-md > .container-xxl {
+ padding-right: 0;
+ padding-left: 0;
+ }
+}
+
+@media (min-width: 768px) {
+ /* line 150, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-md {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-flow: row nowrap;
+ flex-flow: row nowrap;
+ -webkit-box-pack: start;
+ -ms-flex-pack: start;
+ justify-content: flex-start;
+ }
+ /* line 173, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-md .navbar-nav {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ }
+ /* line 176, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-md .navbar-nav .dropdown-menu {
+ position: absolute;
+ }
+ /* line 180, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-md .navbar-nav .nav-link {
+ padding-right: 0.5rem;
+ padding-left: 0.5rem;
+ }
+ /* line 187, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-md > .container,
+ .navbar-expand-md > .container-fluid, .navbar-expand-md > .container-sm, .navbar-expand-md > .container-md, .navbar-expand-md > .container-lg, .navbar-expand-md > .container-xl, .navbar-expand-md > .container-xxl {
+ -ms-flex-wrap: nowrap;
+ flex-wrap: nowrap;
+ }
+ /* line 202, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-md .navbar-collapse {
+ display: -webkit-box !important;
+ display: -ms-flexbox !important;
+ display: flex !important;
+ -ms-flex-preferred-size: auto;
+ flex-basis: auto;
+ }
+ /* line 209, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-md .navbar-toggler {
+ display: none;
+ }
+}
+
+@media (max-width: 1023.98px) {
+ /* line 152, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-lg > .container,
+ .navbar-expand-lg > .container-fluid, .navbar-expand-lg > .container-sm, .navbar-expand-lg > .container-md, .navbar-expand-lg > .container-lg, .navbar-expand-lg > .container-xl, .navbar-expand-lg > .container-xxl {
+ padding-right: 0;
+ padding-left: 0;
+ }
+}
+
+@media (min-width: 1024px) {
+ /* line 150, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-lg {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-flow: row nowrap;
+ flex-flow: row nowrap;
+ -webkit-box-pack: start;
+ -ms-flex-pack: start;
+ justify-content: flex-start;
+ }
+ /* line 173, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-lg .navbar-nav {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ }
+ /* line 176, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-lg .navbar-nav .dropdown-menu {
+ position: absolute;
+ }
+ /* line 180, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-lg .navbar-nav .nav-link {
+ padding-right: 0.5rem;
+ padding-left: 0.5rem;
+ }
+ /* line 187, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-lg > .container,
+ .navbar-expand-lg > .container-fluid, .navbar-expand-lg > .container-sm, .navbar-expand-lg > .container-md, .navbar-expand-lg > .container-lg, .navbar-expand-lg > .container-xl, .navbar-expand-lg > .container-xxl {
+ -ms-flex-wrap: nowrap;
+ flex-wrap: nowrap;
+ }
+ /* line 202, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-lg .navbar-collapse {
+ display: -webkit-box !important;
+ display: -ms-flexbox !important;
+ display: flex !important;
+ -ms-flex-preferred-size: auto;
+ flex-basis: auto;
+ }
+ /* line 209, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-lg .navbar-toggler {
+ display: none;
+ }
+}
+
+@media (max-width: 1279.98px) {
+ /* line 152, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xl > .container,
+ .navbar-expand-xl > .container-fluid, .navbar-expand-xl > .container-sm, .navbar-expand-xl > .container-md, .navbar-expand-xl > .container-lg, .navbar-expand-xl > .container-xl, .navbar-expand-xl > .container-xxl {
+ padding-right: 0;
+ padding-left: 0;
+ }
+}
+
+@media (min-width: 1280px) {
+ /* line 150, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xl {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-flow: row nowrap;
+ flex-flow: row nowrap;
+ -webkit-box-pack: start;
+ -ms-flex-pack: start;
+ justify-content: flex-start;
+ }
+ /* line 173, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xl .navbar-nav {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ }
+ /* line 176, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xl .navbar-nav .dropdown-menu {
+ position: absolute;
+ }
+ /* line 180, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xl .navbar-nav .nav-link {
+ padding-right: 0.5rem;
+ padding-left: 0.5rem;
+ }
+ /* line 187, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xl > .container,
+ .navbar-expand-xl > .container-fluid, .navbar-expand-xl > .container-sm, .navbar-expand-xl > .container-md, .navbar-expand-xl > .container-lg, .navbar-expand-xl > .container-xl, .navbar-expand-xl > .container-xxl {
+ -ms-flex-wrap: nowrap;
+ flex-wrap: nowrap;
+ }
+ /* line 202, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xl .navbar-collapse {
+ display: -webkit-box !important;
+ display: -ms-flexbox !important;
+ display: flex !important;
+ -ms-flex-preferred-size: auto;
+ flex-basis: auto;
+ }
+ /* line 209, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xl .navbar-toggler {
+ display: none;
+ }
+}
+
+@media (max-width: 1439.98px) {
+ /* line 152, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xxl > .container,
+ .navbar-expand-xxl > .container-fluid, .navbar-expand-xxl > .container-sm, .navbar-expand-xxl > .container-md, .navbar-expand-xxl > .container-lg, .navbar-expand-xxl > .container-xl, .navbar-expand-xxl > .container-xxl {
+ padding-right: 0;
+ padding-left: 0;
+ }
+}
+
+@media (min-width: 1440px) {
+ /* line 150, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xxl {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-flow: row nowrap;
+ flex-flow: row nowrap;
+ -webkit-box-pack: start;
+ -ms-flex-pack: start;
+ justify-content: flex-start;
+ }
+ /* line 173, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xxl .navbar-nav {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ }
+ /* line 176, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xxl .navbar-nav .dropdown-menu {
+ position: absolute;
+ }
+ /* line 180, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xxl .navbar-nav .nav-link {
+ padding-right: 0.5rem;
+ padding-left: 0.5rem;
+ }
+ /* line 187, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xxl > .container,
+ .navbar-expand-xxl > .container-fluid, .navbar-expand-xxl > .container-sm, .navbar-expand-xxl > .container-md, .navbar-expand-xxl > .container-lg, .navbar-expand-xxl > .container-xl, .navbar-expand-xxl > .container-xxl {
+ -ms-flex-wrap: nowrap;
+ flex-wrap: nowrap;
+ }
+ /* line 202, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xxl .navbar-collapse {
+ display: -webkit-box !important;
+ display: -ms-flexbox !important;
+ display: flex !important;
+ -ms-flex-preferred-size: auto;
+ flex-basis: auto;
+ }
+ /* line 209, node_modules/bootstrap/scss/_navbar.scss */
+ .navbar-expand-xxl .navbar-toggler {
+ display: none;
+ }
+}
+
+/* line 150, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-expand {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-flow: row nowrap;
+ flex-flow: row nowrap;
+ -webkit-box-pack: start;
+ -ms-flex-pack: start;
+ justify-content: flex-start;
+}
+
+/* line 152, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-expand > .container,
+.navbar-expand > .container-fluid, .navbar-expand > .container-sm, .navbar-expand > .container-md, .navbar-expand > .container-lg, .navbar-expand > .container-xl, .navbar-expand > .container-xxl {
+ padding-right: 0;
+ padding-left: 0;
+}
+
+/* line 173, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-expand .navbar-nav {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: row;
+ flex-direction: row;
+}
+
+/* line 176, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-expand .navbar-nav .dropdown-menu {
+ position: absolute;
+}
+
+/* line 180, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-expand .navbar-nav .nav-link {
+ padding-right: 0.5rem;
+ padding-left: 0.5rem;
+}
+
+/* line 187, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-expand > .container,
+.navbar-expand > .container-fluid, .navbar-expand > .container-sm, .navbar-expand > .container-md, .navbar-expand > .container-lg, .navbar-expand > .container-xl, .navbar-expand > .container-xxl {
+ -ms-flex-wrap: nowrap;
+ flex-wrap: nowrap;
+}
+
+/* line 202, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-expand .navbar-collapse {
+ display: -webkit-box !important;
+ display: -ms-flexbox !important;
+ display: flex !important;
+ -ms-flex-preferred-size: auto;
+ flex-basis: auto;
+}
+
+/* line 209, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-expand .navbar-toggler {
+ display: none;
+}
+
+/* line 224, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-light .navbar-brand {
+ color: rgba(0, 0, 0, 0.9);
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.navbar-light .navbar-brand:hover, .navbar-light .navbar-brand:focus {
+ color: rgba(0, 0, 0, 0.9);
+}
+
+/* line 233, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-light .navbar-nav .nav-link {
+ color: rgba(0, 0, 0, 0.5);
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.navbar-light .navbar-nav .nav-link:hover, .navbar-light .navbar-nav .nav-link:focus {
+ color: rgba(0, 0, 0, 0.7);
+}
+
+/* line 240, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-light .navbar-nav .nav-link.disabled {
+ color: rgba(0, 0, 0, 0.3);
+}
+
+/* line 245, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-light .navbar-nav .show > .nav-link,
+.navbar-light .navbar-nav .active > .nav-link,
+.navbar-light .navbar-nav .nav-link.show,
+.navbar-light .navbar-nav .nav-link.active {
+ color: rgba(0, 0, 0, 0.9);
+}
+
+/* line 253, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-light .navbar-toggler {
+ color: rgba(0, 0, 0, 0.5);
+ border-color: rgba(0, 0, 0, 0.1);
+}
+
+/* line 258, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-light .navbar-toggler-icon {
+ background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='30' height='30' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%280, 0, 0, 0.5%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e");
+}
+
+/* line 262, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-light .navbar-text {
+ color: rgba(0, 0, 0, 0.5);
+}
+
+/* line 264, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-light .navbar-text a {
+ color: rgba(0, 0, 0, 0.9);
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.navbar-light .navbar-text a:hover, .navbar-light .navbar-text a:focus {
+ color: rgba(0, 0, 0, 0.9);
+}
+
+/* line 276, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-dark .navbar-brand {
+ color: #ffffff;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.navbar-dark .navbar-brand:hover, .navbar-dark .navbar-brand:focus {
+ color: #ffffff;
+}
+
+/* line 285, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-dark .navbar-nav .nav-link {
+ color: rgba(255, 255, 255, 0.5);
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.navbar-dark .navbar-nav .nav-link:hover, .navbar-dark .navbar-nav .nav-link:focus {
+ color: rgba(255, 255, 255, 0.75);
+}
+
+/* line 292, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-dark .navbar-nav .nav-link.disabled {
+ color: rgba(255, 255, 255, 0.25);
+}
+
+/* line 297, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-dark .navbar-nav .show > .nav-link,
+.navbar-dark .navbar-nav .active > .nav-link,
+.navbar-dark .navbar-nav .nav-link.show,
+.navbar-dark .navbar-nav .nav-link.active {
+ color: #ffffff;
+}
+
+/* line 305, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-dark .navbar-toggler {
+ color: rgba(255, 255, 255, 0.5);
+ border-color: rgba(255, 255, 255, 0.1);
+}
+
+/* line 310, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-dark .navbar-toggler-icon {
+ background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='30' height='30' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%28255, 255, 255, 0.5%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e");
+}
+
+/* line 314, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-dark .navbar-text {
+ color: rgba(255, 255, 255, 0.5);
+}
+
+/* line 316, node_modules/bootstrap/scss/_navbar.scss */
+.navbar-dark .navbar-text a {
+ color: #ffffff;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.navbar-dark .navbar-text a:hover, .navbar-dark .navbar-text a:focus {
+ color: #ffffff;
+}
+
+/* line 5, node_modules/bootstrap/scss/_card.scss */
+.card {
+ position: relative;
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-orient: vertical;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: column;
+ flex-direction: column;
+ min-width: 0;
+ word-wrap: break-word;
+ background-color: #ffffff;
+ background-clip: border-box;
+ border: 1px solid rgba(0, 0, 0, 0.125);
+ border-radius: 0.25rem;
+}
+
+/* line 17, node_modules/bootstrap/scss/_card.scss */
+.card > hr {
+ margin-right: 0;
+ margin-left: 0;
+}
+
+/* line 22, node_modules/bootstrap/scss/_card.scss */
+.card > .list-group {
+ border-top: inherit;
+ border-bottom: inherit;
+}
+
+/* line 26, node_modules/bootstrap/scss/_card.scss */
+.card > .list-group:first-child {
+ border-top-width: 0;
+ border-top-left-radius: calc(0.25rem - 1px);
+ border-top-right-radius: calc(0.25rem - 1px);
+}
+
+/* line 31, node_modules/bootstrap/scss/_card.scss */
+.card > .list-group:last-child {
+ border-bottom-width: 0;
+ border-bottom-right-radius: calc(0.25rem - 1px);
+ border-bottom-left-radius: calc(0.25rem - 1px);
+}
+
+/* line 39, node_modules/bootstrap/scss/_card.scss */
+.card > .card-header + .list-group,
+.card > .list-group + .card-footer {
+ border-top: 0;
+}
+
+/* line 45, node_modules/bootstrap/scss/_card.scss */
+.card-body {
+ -webkit-box-flex: 1;
+ -ms-flex: 1 1 auto;
+ flex: 1 1 auto;
+ min-height: 1px;
+ padding: 1.25rem;
+}
+
+/* line 56, node_modules/bootstrap/scss/_card.scss */
+.card-title {
+ margin-bottom: 0.75rem;
+}
+
+/* line 60, node_modules/bootstrap/scss/_card.scss */
+.card-subtitle {
+ margin-top: -0.375rem;
+ margin-bottom: 0;
+}
+
+/* line 65, node_modules/bootstrap/scss/_card.scss */
+.card-text:last-child {
+ margin-bottom: 0;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.card-link:hover {
+ text-decoration: none;
+}
+
+/* line 74, node_modules/bootstrap/scss/_card.scss */
+.card-link + .card-link {
+ margin-left: 1.25rem;
+}
+
+/* line 83, node_modules/bootstrap/scss/_card.scss */
+.card-header {
+ padding: 0.75rem 1.25rem;
+ margin-bottom: 0;
+ background-color: rgba(0, 0, 0, 0.03);
+ border-bottom: 1px solid rgba(0, 0, 0, 0.125);
+}
+
+/* line 90, node_modules/bootstrap/scss/_card.scss */
+.card-header:first-child {
+ border-radius: calc(0.25rem - 1px) calc(0.25rem - 1px) 0 0;
+}
+
+/* line 95, node_modules/bootstrap/scss/_card.scss */
+.card-footer {
+ padding: 0.75rem 1.25rem;
+ background-color: rgba(0, 0, 0, 0.03);
+ border-top: 1px solid rgba(0, 0, 0, 0.125);
+}
+
+/* line 101, node_modules/bootstrap/scss/_card.scss */
+.card-footer:last-child {
+ border-radius: 0 0 calc(0.25rem - 1px) calc(0.25rem - 1px);
+}
+
+/* line 111, node_modules/bootstrap/scss/_card.scss */
+.card-header-tabs {
+ margin-right: -0.625rem;
+ margin-bottom: -0.75rem;
+ margin-left: -0.625rem;
+ border-bottom: 0;
+}
+
+/* line 118, node_modules/bootstrap/scss/_card.scss */
+.card-header-pills {
+ margin-right: -0.625rem;
+ margin-left: -0.625rem;
+}
+
+/* line 124, node_modules/bootstrap/scss/_card.scss */
+.card-img-overlay {
+ position: absolute;
+ top: 0;
+ right: 0;
+ bottom: 0;
+ left: 0;
+ padding: 1.25rem;
+ border-radius: calc(0.25rem - 1px);
+}
+
+/* line 134, node_modules/bootstrap/scss/_card.scss */
+.card-img,
+.card-img-top,
+.card-img-bottom {
+ -ms-flex-negative: 0;
+ flex-shrink: 0;
+ width: 100%;
+}
+
+/* line 141, node_modules/bootstrap/scss/_card.scss */
+.card-img,
+.card-img-top {
+ border-top-left-radius: calc(0.25rem - 1px);
+ border-top-right-radius: calc(0.25rem - 1px);
+}
+
+/* line 146, node_modules/bootstrap/scss/_card.scss */
+.card-img,
+.card-img-bottom {
+ border-bottom-right-radius: calc(0.25rem - 1px);
+ border-bottom-left-radius: calc(0.25rem - 1px);
+}
+
+/* line 155, node_modules/bootstrap/scss/_card.scss */
+.card-deck .card {
+ margin-bottom: 15px;
+}
+
+@media (min-width: 576px) {
+ /* line 154, node_modules/bootstrap/scss/_card.scss */
+ .card-deck {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-flow: row wrap;
+ flex-flow: row wrap;
+ margin-right: -15px;
+ margin-left: -15px;
+ }
+ /* line 165, node_modules/bootstrap/scss/_card.scss */
+ .card-deck .card {
+ -webkit-box-flex: 1;
+ -ms-flex: 1 0 0%;
+ flex: 1 0 0%;
+ margin-right: 15px;
+ margin-bottom: 0;
+ margin-left: 15px;
+ }
+}
+
+/* line 183, node_modules/bootstrap/scss/_card.scss */
+.card-group > .card {
+ margin-bottom: 15px;
+}
+
+@media (min-width: 576px) {
+ /* line 180, node_modules/bootstrap/scss/_card.scss */
+ .card-group {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-flow: row wrap;
+ flex-flow: row wrap;
+ }
+ /* line 192, node_modules/bootstrap/scss/_card.scss */
+ .card-group > .card {
+ -webkit-box-flex: 1;
+ -ms-flex: 1 0 0%;
+ flex: 1 0 0%;
+ margin-bottom: 0;
+ }
+ /* line 197, node_modules/bootstrap/scss/_card.scss */
+ .card-group > .card + .card {
+ margin-left: 0;
+ border-left: 0;
+ }
+ /* line 204, node_modules/bootstrap/scss/_card.scss */
+ .card-group > .card:not(:last-child) {
+ border-top-right-radius: 0;
+ border-bottom-right-radius: 0;
+ }
+ /* line 207, node_modules/bootstrap/scss/_card.scss */
+ .card-group > .card:not(:last-child) .card-img-top,
+ .card-group > .card:not(:last-child) .card-header {
+ border-top-right-radius: 0;
+ }
+ /* line 212, node_modules/bootstrap/scss/_card.scss */
+ .card-group > .card:not(:last-child) .card-img-bottom,
+ .card-group > .card:not(:last-child) .card-footer {
+ border-bottom-right-radius: 0;
+ }
+ /* line 219, node_modules/bootstrap/scss/_card.scss */
+ .card-group > .card:not(:first-child) {
+ border-top-left-radius: 0;
+ border-bottom-left-radius: 0;
+ }
+ /* line 222, node_modules/bootstrap/scss/_card.scss */
+ .card-group > .card:not(:first-child) .card-img-top,
+ .card-group > .card:not(:first-child) .card-header {
+ border-top-left-radius: 0;
+ }
+ /* line 227, node_modules/bootstrap/scss/_card.scss */
+ .card-group > .card:not(:first-child) .card-img-bottom,
+ .card-group > .card:not(:first-child) .card-footer {
+ border-bottom-left-radius: 0;
+ }
+}
+
+/* line 244, node_modules/bootstrap/scss/_card.scss */
+.card-columns .card {
+ margin-bottom: 0.75rem;
+}
+
+@media (min-width: 576px) {
+ /* line 243, node_modules/bootstrap/scss/_card.scss */
+ .card-columns {
+ -webkit-column-count: 3;
+ -moz-column-count: 3;
+ column-count: 3;
+ -webkit-column-gap: 1.25rem;
+ -moz-column-gap: 1.25rem;
+ column-gap: 1.25rem;
+ orphans: 1;
+ widows: 1;
+ }
+ /* line 254, node_modules/bootstrap/scss/_card.scss */
+ .card-columns .card {
+ display: inline-block;
+ width: 100%;
+ }
+}
+
+/* line 266, node_modules/bootstrap/scss/_card.scss */
+.accordion {
+ overflow-anchor: none;
+}
+
+/* line 269, node_modules/bootstrap/scss/_card.scss */
+.accordion > .card {
+ overflow: hidden;
+}
+
+/* line 272, node_modules/bootstrap/scss/_card.scss */
+.accordion > .card:not(:last-of-type) {
+ border-bottom: 0;
+ border-bottom-right-radius: 0;
+ border-bottom-left-radius: 0;
+}
+
+/* line 277, node_modules/bootstrap/scss/_card.scss */
+.accordion > .card:not(:first-of-type) {
+ border-top-left-radius: 0;
+ border-top-right-radius: 0;
+}
+
+/* line 281, node_modules/bootstrap/scss/_card.scss */
+.accordion > .card > .card-header {
+ border-radius: 0;
+ margin-bottom: -1px;
+}
+
+/* line 1, node_modules/bootstrap/scss/_breadcrumb.scss */
+.breadcrumb {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -ms-flex-wrap: wrap;
+ flex-wrap: wrap;
+ padding: 0.75rem 1rem;
+ margin-bottom: 1rem;
+ list-style: none;
+ background-color: #eaebea;
+ border-radius: 0.25rem;
+}
+
+/* line 12, node_modules/bootstrap/scss/_breadcrumb.scss */
+.breadcrumb-item {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+}
+
+/* line 16, node_modules/bootstrap/scss/_breadcrumb.scss */
+.breadcrumb-item + .breadcrumb-item {
+ padding-left: 0.5rem;
+}
+
+/* line 19, node_modules/bootstrap/scss/_breadcrumb.scss */
+.breadcrumb-item + .breadcrumb-item::before {
+ display: inline-block;
+ padding-right: 0.5rem;
+ color: #6c757d;
+ content: "/";
+}
+
+/* line 33, node_modules/bootstrap/scss/_breadcrumb.scss */
+.breadcrumb-item + .breadcrumb-item:hover::before {
+ text-decoration: underline;
+}
+
+/* line 37, node_modules/bootstrap/scss/_breadcrumb.scss */
+.breadcrumb-item + .breadcrumb-item:hover::before {
+ text-decoration: none;
+}
+
+/* line 41, node_modules/bootstrap/scss/_breadcrumb.scss */
+.breadcrumb-item.active {
+ color: #6c757d;
+}
+
+/* line 1, node_modules/bootstrap/scss/_pagination.scss */
+.pagination {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ padding-left: 0;
+ list-style: none;
+ border-radius: 0.25rem;
+}
+
+/* line 7, node_modules/bootstrap/scss/_pagination.scss */
+.page-link {
+ position: relative;
+ display: block;
+ padding: 0.5rem 0.75rem;
+ margin-left: -1px;
+ line-height: 1.25;
+ color: #464746;
+ background-color: #ffffff;
+ border: 1px solid #dee2e6;
+}
+
+/* line 18, node_modules/bootstrap/scss/_pagination.scss */
+.page-link:hover {
+ z-index: 2;
+ color: #202020;
+ text-decoration: none;
+ background-color: #eaebea;
+ border-color: #dee2e6;
+}
+
+/* line 26, node_modules/bootstrap/scss/_pagination.scss */
+.page-link:focus {
+ z-index: 3;
+ outline: 0;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+ box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.25);
+}
+
+/* line 35, node_modules/bootstrap/scss/_pagination.scss */
+.page-item:first-child .page-link {
+ margin-left: 0;
+ border-top-left-radius: 0.25rem;
+ border-bottom-left-radius: 0.25rem;
+}
+
+/* line 41, node_modules/bootstrap/scss/_pagination.scss */
+.page-item:last-child .page-link {
+ border-top-right-radius: 0.25rem;
+ border-bottom-right-radius: 0.25rem;
+}
+
+/* line 46, node_modules/bootstrap/scss/_pagination.scss */
+.page-item.active .page-link {
+ z-index: 3;
+ color: #ffffff;
+ background-color: #464746;
+ border-color: #464746;
+}
+
+/* line 53, node_modules/bootstrap/scss/_pagination.scss */
+.page-item.disabled .page-link {
+ color: #6c757d;
+ pointer-events: none;
+ cursor: auto;
+ background-color: #ffffff;
+ border-color: #dee2e6;
+}
+
+/* line 4, node_modules/bootstrap/scss/mixins/_pagination.scss */
+.pagination-lg .page-link {
+ padding: 0.75rem 1.5rem;
+ font-size: 1.25rem;
+ line-height: 1.5;
+}
+
+/* line 12, node_modules/bootstrap/scss/mixins/_pagination.scss */
+.pagination-lg .page-item:first-child .page-link {
+ border-top-left-radius: 0.3rem;
+ border-bottom-left-radius: 0.3rem;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_pagination.scss */
+.pagination-lg .page-item:last-child .page-link {
+ border-top-right-radius: 0.3rem;
+ border-bottom-right-radius: 0.3rem;
+}
+
+/* line 4, node_modules/bootstrap/scss/mixins/_pagination.scss */
+.pagination-sm .page-link {
+ padding: 0.25rem 0.5rem;
+ font-size: 0.875rem;
+ line-height: 1.5;
+}
+
+/* line 12, node_modules/bootstrap/scss/mixins/_pagination.scss */
+.pagination-sm .page-item:first-child .page-link {
+ border-top-left-radius: 0.2rem;
+ border-bottom-left-radius: 0.2rem;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_pagination.scss */
+.pagination-sm .page-item:last-child .page-link {
+ border-top-right-radius: 0.2rem;
+ border-bottom-right-radius: 0.2rem;
+}
+
+/* line 6, node_modules/bootstrap/scss/_badge.scss */
+.badge {
+ display: inline-block;
+ padding: 0.25em 0.4em;
+ font-size: 75%;
+ font-weight: 700;
+ line-height: 1;
+ text-align: center;
+ white-space: nowrap;
+ vertical-align: baseline;
+ border-radius: 0.25rem;
+ -webkit-transition: color 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+ transition: color 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+ transition: color 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;
+ transition: color 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 6, node_modules/bootstrap/scss/_badge.scss */
+ .badge {
+ -webkit-transition: none;
+ transition: none;
+ }
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.badge:hover, a.badge:focus {
+ text-decoration: none;
+}
+
+/* line 25, node_modules/bootstrap/scss/_badge.scss */
+.badge:empty {
+ display: none;
+}
+
+/* line 31, node_modules/bootstrap/scss/_badge.scss */
+.btn .badge {
+ position: relative;
+ top: -1px;
+}
+
+/* line 40, node_modules/bootstrap/scss/_badge.scss */
+.badge-pill {
+ padding-right: 0.6em;
+ padding-left: 0.6em;
+ border-radius: 10rem;
+}
+
+/* line 51, node_modules/bootstrap/scss/_badge.scss */
+.badge-primary {
+ color: #ffffff;
+ background-color: #464746;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.badge-primary:hover, a.badge-primary:focus {
+ color: #ffffff;
+ background-color: #2d2d2d;
+}
+
+/* line 11, node_modules/bootstrap/scss/mixins/_badge.scss */
+a.badge-primary:focus, a.badge-primary.focus {
+ outline: 0;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+}
+
+/* line 51, node_modules/bootstrap/scss/_badge.scss */
+.badge-secondary {
+ color: #464746;
+ background-color: #f0ebe3;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.badge-secondary:hover, a.badge-secondary:focus {
+ color: #464746;
+ background-color: #ded3c2;
+}
+
+/* line 11, node_modules/bootstrap/scss/mixins/_badge.scss */
+a.badge-secondary:focus, a.badge-secondary.focus {
+ outline: 0;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.5);
+}
+
+/* line 51, node_modules/bootstrap/scss/_badge.scss */
+.badge-success {
+ color: #464746;
+ background-color: #f0ebe3;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.badge-success:hover, a.badge-success:focus {
+ color: #464746;
+ background-color: #ded3c2;
+}
+
+/* line 11, node_modules/bootstrap/scss/mixins/_badge.scss */
+a.badge-success:focus, a.badge-success.focus {
+ outline: 0;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(240, 235, 227, 0.5);
+}
+
+/* line 51, node_modules/bootstrap/scss/_badge.scss */
+.badge-info {
+ color: #ffffff;
+ background-color: #464746;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.badge-info:hover, a.badge-info:focus {
+ color: #ffffff;
+ background-color: #2d2d2d;
+}
+
+/* line 11, node_modules/bootstrap/scss/mixins/_badge.scss */
+a.badge-info:focus, a.badge-info.focus {
+ outline: 0;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+}
+
+/* line 51, node_modules/bootstrap/scss/_badge.scss */
+.badge-warning {
+ color: #ffffff;
+ background-color: #464746;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.badge-warning:hover, a.badge-warning:focus {
+ color: #ffffff;
+ background-color: #2d2d2d;
+}
+
+/* line 11, node_modules/bootstrap/scss/mixins/_badge.scss */
+a.badge-warning:focus, a.badge-warning.focus {
+ outline: 0;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(70, 71, 70, 0.5);
+}
+
+/* line 51, node_modules/bootstrap/scss/_badge.scss */
+.badge-danger {
+ color: #ffffff;
+ background-color: #e54a19;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.badge-danger:hover, a.badge-danger:focus {
+ color: #ffffff;
+ background-color: #b73b14;
+}
+
+/* line 11, node_modules/bootstrap/scss/mixins/_badge.scss */
+a.badge-danger:focus, a.badge-danger.focus {
+ outline: 0;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(229, 74, 25, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(229, 74, 25, 0.5);
+}
+
+/* line 51, node_modules/bootstrap/scss/_badge.scss */
+.badge-light {
+ color: #464746;
+ background-color: #f7f7f7;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.badge-light:hover, a.badge-light:focus {
+ color: #464746;
+ background-color: #dedede;
+}
+
+/* line 11, node_modules/bootstrap/scss/mixins/_badge.scss */
+a.badge-light:focus, a.badge-light.focus {
+ outline: 0;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(247, 247, 247, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(247, 247, 247, 0.5);
+}
+
+/* line 51, node_modules/bootstrap/scss/_badge.scss */
+.badge-dark {
+ color: #ffffff;
+ background-color: #343a40;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.badge-dark:hover, a.badge-dark:focus {
+ color: #ffffff;
+ background-color: #1d2124;
+}
+
+/* line 11, node_modules/bootstrap/scss/mixins/_badge.scss */
+a.badge-dark:focus, a.badge-dark.focus {
+ outline: 0;
+ -webkit-box-shadow: 0 0 0 0.2rem rgba(52, 58, 64, 0.5);
+ box-shadow: 0 0 0 0.2rem rgba(52, 58, 64, 0.5);
+}
+
+/* line 1, node_modules/bootstrap/scss/_jumbotron.scss */
+.jumbotron {
+ padding: 2rem 1rem;
+ margin-bottom: 2rem;
+ background-color: #eaebea;
+ border-radius: 0.3rem;
+}
+
+@media (min-width: 576px) {
+ /* line 1, node_modules/bootstrap/scss/_jumbotron.scss */
+ .jumbotron {
+ padding: 4rem 2rem;
+ }
+}
+
+/* line 13, node_modules/bootstrap/scss/_jumbotron.scss */
+.jumbotron-fluid {
+ padding-right: 0;
+ padding-left: 0;
+ border-radius: 0;
+}
+
+/* line 5, node_modules/bootstrap/scss/_alert.scss */
+.alert {
+ position: relative;
+ padding: 0.75rem 1.25rem;
+ margin-bottom: 1rem;
+ border: 1px solid transparent;
+ border-radius: 0.25rem;
+}
+
+/* line 14, node_modules/bootstrap/scss/_alert.scss */
+.alert-heading {
+ color: inherit;
+}
+
+/* line 20, node_modules/bootstrap/scss/_alert.scss */
+.alert-link {
+ font-weight: 700;
+}
+
+/* line 29, node_modules/bootstrap/scss/_alert.scss */
+.alert-dismissible {
+ padding-right: 4rem;
+}
+
+/* line 33, node_modules/bootstrap/scss/_alert.scss */
+.alert-dismissible .close {
+ position: absolute;
+ top: 0;
+ right: 0;
+ padding: 0.75rem 1.25rem;
+ color: inherit;
+}
+
+/* line 48, node_modules/bootstrap/scss/_alert.scss */
+.alert-primary {
+ color: #242524;
+ background-color: #dadada;
+ border-color: #cbcbcb;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-primary hr {
+ border-top-color: #bebebe;
+}
+
+/* line 10, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-primary .alert-link {
+ color: #0b0b0b;
+}
+
+/* line 48, node_modules/bootstrap/scss/_alert.scss */
+.alert-secondary {
+ color: #7d7a76;
+ background-color: #fcfbf9;
+ border-color: #fbf9f7;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-secondary hr {
+ border-top-color: #f3ece6;
+}
+
+/* line 10, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-secondary .alert-link {
+ color: #63605d;
+}
+
+/* line 48, node_modules/bootstrap/scss/_alert.scss */
+.alert-success {
+ color: #7d7a76;
+ background-color: #fcfbf9;
+ border-color: #fbf9f7;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-success hr {
+ border-top-color: #f3ece6;
+}
+
+/* line 10, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-success .alert-link {
+ color: #63605d;
+}
+
+/* line 48, node_modules/bootstrap/scss/_alert.scss */
+.alert-info {
+ color: #242524;
+ background-color: #dadada;
+ border-color: #cbcbcb;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-info hr {
+ border-top-color: #bebebe;
+}
+
+/* line 10, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-info .alert-link {
+ color: #0b0b0b;
+}
+
+/* line 48, node_modules/bootstrap/scss/_alert.scss */
+.alert-warning {
+ color: #242524;
+ background-color: #dadada;
+ border-color: #cbcbcb;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-warning hr {
+ border-top-color: #bebebe;
+}
+
+/* line 10, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-warning .alert-link {
+ color: #0b0b0b;
+}
+
+/* line 48, node_modules/bootstrap/scss/_alert.scss */
+.alert-danger {
+ color: #77260d;
+ background-color: #fadbd1;
+ border-color: #f8ccbf;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-danger hr {
+ border-top-color: #f5baa8;
+}
+
+/* line 10, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-danger .alert-link {
+ color: #491708;
+}
+
+/* line 48, node_modules/bootstrap/scss/_alert.scss */
+.alert-light {
+ color: gray;
+ background-color: #fdfdfd;
+ border-color: #fdfdfd;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-light hr {
+ border-top-color: #f0f0f0;
+}
+
+/* line 10, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-light .alert-link {
+ color: #676767;
+}
+
+/* line 48, node_modules/bootstrap/scss/_alert.scss */
+.alert-dark {
+ color: #1b1e21;
+ background-color: #d6d8d9;
+ border-color: #c6c8ca;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-dark hr {
+ border-top-color: #b9bbbe;
+}
+
+/* line 10, node_modules/bootstrap/scss/mixins/_alert.scss */
+.alert-dark .alert-link {
+ color: #040505;
+}
+
+@-webkit-keyframes progress-bar-stripes {
+ from {
+ background-position: 1rem 0;
+ }
+ to {
+ background-position: 0 0;
+ }
+}
+
+@keyframes progress-bar-stripes {
+ from {
+ background-position: 1rem 0;
+ }
+ to {
+ background-position: 0 0;
+ }
+}
+
+/* line 9, node_modules/bootstrap/scss/_progress.scss */
+.progress {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ height: 1rem;
+ overflow: hidden;
+ line-height: 0;
+ font-size: 0.75rem;
+ background-color: #eaebea;
+ border-radius: 0.25rem;
+}
+
+/* line 20, node_modules/bootstrap/scss/_progress.scss */
+.progress-bar {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-orient: vertical;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: column;
+ flex-direction: column;
+ -webkit-box-pack: center;
+ -ms-flex-pack: center;
+ justify-content: center;
+ overflow: hidden;
+ color: #ffffff;
+ text-align: center;
+ white-space: nowrap;
+ background-color: #464746;
+ -webkit-transition: width 0.6s ease;
+ transition: width 0.6s ease;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 20, node_modules/bootstrap/scss/_progress.scss */
+ .progress-bar {
+ -webkit-transition: none;
+ transition: none;
+ }
+}
+
+/* line 32, node_modules/bootstrap/scss/_progress.scss */
+.progress-bar-striped {
+ background-image: linear-gradient(45deg, rgba(255, 255, 255, 0.15) 25%, transparent 25%, transparent 50%, rgba(255, 255, 255, 0.15) 50%, rgba(255, 255, 255, 0.15) 75%, transparent 75%, transparent);
+ background-size: 1rem 1rem;
+}
+
+/* line 38, node_modules/bootstrap/scss/_progress.scss */
+.progress-bar-animated {
+ -webkit-animation: progress-bar-stripes 1s linear infinite;
+ animation: progress-bar-stripes 1s linear infinite;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 38, node_modules/bootstrap/scss/_progress.scss */
+ .progress-bar-animated {
+ -webkit-animation: none;
+ animation: none;
+ }
+}
+
+/* line 1, node_modules/bootstrap/scss/_media.scss */
+.media {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-align: start;
+ -ms-flex-align: start;
+ align-items: flex-start;
+}
+
+/* line 6, node_modules/bootstrap/scss/_media.scss */
+.media-body {
+ -webkit-box-flex: 1;
+ -ms-flex: 1;
+ flex: 1;
+}
+
+/* line 5, node_modules/bootstrap/scss/_list-group.scss */
+.list-group {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-orient: vertical;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: column;
+ flex-direction: column;
+ padding-left: 0;
+ margin-bottom: 0;
+ border-radius: 0.25rem;
+}
+
+/* line 21, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-item-action {
+ width: 100%;
+ color: #495057;
+ text-align: inherit;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.list-group-item-action:hover, .list-group-item-action:focus {
+ z-index: 1;
+ color: #495057;
+ text-decoration: none;
+ background-color: #f7f7f7;
+}
+
+/* line 34, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-item-action:active {
+ color: #464746;
+ background-color: #eaebea;
+}
+
+/* line 45, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-item {
+ position: relative;
+ display: block;
+ padding: 0.75rem 1.25rem;
+ background-color: #ffffff;
+ border: 1px solid rgba(0, 0, 0, 0.125);
+}
+
+/* line 54, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-item:first-child {
+ border-top-left-radius: inherit;
+ border-top-right-radius: inherit;
+}
+
+/* line 58, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-item:last-child {
+ border-bottom-right-radius: inherit;
+ border-bottom-left-radius: inherit;
+}
+
+/* line 62, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-item.disabled, .list-group-item:disabled {
+ color: #6c757d;
+ pointer-events: none;
+ background-color: #ffffff;
+}
+
+/* line 70, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-item.active {
+ z-index: 2;
+ color: #ffffff;
+ background-color: #464746;
+ border-color: #464746;
+}
+
+/* line 77, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-item + .list-group-item {
+ border-top-width: 0;
+}
+
+/* line 80, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-item + .list-group-item.active {
+ margin-top: -1px;
+ border-top-width: 1px;
+}
+
+/* line 96, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-horizontal {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: row;
+ flex-direction: row;
+}
+
+/* line 100, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-horizontal > .list-group-item:first-child {
+ border-bottom-left-radius: 0.25rem;
+ border-top-right-radius: 0;
+}
+
+/* line 105, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-horizontal > .list-group-item:last-child {
+ border-top-right-radius: 0.25rem;
+ border-bottom-left-radius: 0;
+}
+
+/* line 110, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-horizontal > .list-group-item.active {
+ margin-top: 0;
+}
+
+/* line 114, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-horizontal > .list-group-item + .list-group-item {
+ border-top-width: 1px;
+ border-left-width: 0;
+}
+
+/* line 118, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-horizontal > .list-group-item + .list-group-item.active {
+ margin-left: -1px;
+ border-left-width: 1px;
+}
+
+@media (min-width: 576px) {
+ /* line 96, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-sm {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ }
+ /* line 100, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-sm > .list-group-item:first-child {
+ border-bottom-left-radius: 0.25rem;
+ border-top-right-radius: 0;
+ }
+ /* line 105, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-sm > .list-group-item:last-child {
+ border-top-right-radius: 0.25rem;
+ border-bottom-left-radius: 0;
+ }
+ /* line 110, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-sm > .list-group-item.active {
+ margin-top: 0;
+ }
+ /* line 114, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-sm > .list-group-item + .list-group-item {
+ border-top-width: 1px;
+ border-left-width: 0;
+ }
+ /* line 118, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-sm > .list-group-item + .list-group-item.active {
+ margin-left: -1px;
+ border-left-width: 1px;
+ }
+}
+
+@media (min-width: 768px) {
+ /* line 96, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-md {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ }
+ /* line 100, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-md > .list-group-item:first-child {
+ border-bottom-left-radius: 0.25rem;
+ border-top-right-radius: 0;
+ }
+ /* line 105, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-md > .list-group-item:last-child {
+ border-top-right-radius: 0.25rem;
+ border-bottom-left-radius: 0;
+ }
+ /* line 110, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-md > .list-group-item.active {
+ margin-top: 0;
+ }
+ /* line 114, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-md > .list-group-item + .list-group-item {
+ border-top-width: 1px;
+ border-left-width: 0;
+ }
+ /* line 118, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-md > .list-group-item + .list-group-item.active {
+ margin-left: -1px;
+ border-left-width: 1px;
+ }
+}
+
+@media (min-width: 1024px) {
+ /* line 96, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-lg {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ }
+ /* line 100, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-lg > .list-group-item:first-child {
+ border-bottom-left-radius: 0.25rem;
+ border-top-right-radius: 0;
+ }
+ /* line 105, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-lg > .list-group-item:last-child {
+ border-top-right-radius: 0.25rem;
+ border-bottom-left-radius: 0;
+ }
+ /* line 110, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-lg > .list-group-item.active {
+ margin-top: 0;
+ }
+ /* line 114, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-lg > .list-group-item + .list-group-item {
+ border-top-width: 1px;
+ border-left-width: 0;
+ }
+ /* line 118, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-lg > .list-group-item + .list-group-item.active {
+ margin-left: -1px;
+ border-left-width: 1px;
+ }
+}
+
+@media (min-width: 1280px) {
+ /* line 96, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-xl {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ }
+ /* line 100, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-xl > .list-group-item:first-child {
+ border-bottom-left-radius: 0.25rem;
+ border-top-right-radius: 0;
+ }
+ /* line 105, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-xl > .list-group-item:last-child {
+ border-top-right-radius: 0.25rem;
+ border-bottom-left-radius: 0;
+ }
+ /* line 110, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-xl > .list-group-item.active {
+ margin-top: 0;
+ }
+ /* line 114, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-xl > .list-group-item + .list-group-item {
+ border-top-width: 1px;
+ border-left-width: 0;
+ }
+ /* line 118, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-xl > .list-group-item + .list-group-item.active {
+ margin-left: -1px;
+ border-left-width: 1px;
+ }
+}
+
+@media (min-width: 1440px) {
+ /* line 96, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-xxl {
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ }
+ /* line 100, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-xxl > .list-group-item:first-child {
+ border-bottom-left-radius: 0.25rem;
+ border-top-right-radius: 0;
+ }
+ /* line 105, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-xxl > .list-group-item:last-child {
+ border-top-right-radius: 0.25rem;
+ border-bottom-left-radius: 0;
+ }
+ /* line 110, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-xxl > .list-group-item.active {
+ margin-top: 0;
+ }
+ /* line 114, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-xxl > .list-group-item + .list-group-item {
+ border-top-width: 1px;
+ border-left-width: 0;
+ }
+ /* line 118, node_modules/bootstrap/scss/_list-group.scss */
+ .list-group-horizontal-xxl > .list-group-item + .list-group-item.active {
+ margin-left: -1px;
+ border-left-width: 1px;
+ }
+}
+
+/* line 134, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-flush {
+ border-radius: 0;
+}
+
+/* line 137, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-flush > .list-group-item {
+ border-width: 0 0 1px;
+}
+
+/* line 140, node_modules/bootstrap/scss/_list-group.scss */
+.list-group-flush > .list-group-item:last-child {
+ border-bottom-width: 0;
+}
+
+/* line 4, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-primary {
+ color: #242524;
+ background-color: #cbcbcb;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.list-group-item-primary.list-group-item-action:hover, .list-group-item-primary.list-group-item-action:focus {
+ color: #242524;
+ background-color: #bebebe;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-primary.list-group-item-action.active {
+ color: #ffffff;
+ background-color: #242524;
+ border-color: #242524;
+}
+
+/* line 4, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-secondary {
+ color: #7d7a76;
+ background-color: #fbf9f7;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.list-group-item-secondary.list-group-item-action:hover, .list-group-item-secondary.list-group-item-action:focus {
+ color: #7d7a76;
+ background-color: #f3ece6;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-secondary.list-group-item-action.active {
+ color: #ffffff;
+ background-color: #7d7a76;
+ border-color: #7d7a76;
+}
+
+/* line 4, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-success {
+ color: #7d7a76;
+ background-color: #fbf9f7;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.list-group-item-success.list-group-item-action:hover, .list-group-item-success.list-group-item-action:focus {
+ color: #7d7a76;
+ background-color: #f3ece6;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-success.list-group-item-action.active {
+ color: #ffffff;
+ background-color: #7d7a76;
+ border-color: #7d7a76;
+}
+
+/* line 4, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-info {
+ color: #242524;
+ background-color: #cbcbcb;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.list-group-item-info.list-group-item-action:hover, .list-group-item-info.list-group-item-action:focus {
+ color: #242524;
+ background-color: #bebebe;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-info.list-group-item-action.active {
+ color: #ffffff;
+ background-color: #242524;
+ border-color: #242524;
+}
+
+/* line 4, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-warning {
+ color: #242524;
+ background-color: #cbcbcb;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.list-group-item-warning.list-group-item-action:hover, .list-group-item-warning.list-group-item-action:focus {
+ color: #242524;
+ background-color: #bebebe;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-warning.list-group-item-action.active {
+ color: #ffffff;
+ background-color: #242524;
+ border-color: #242524;
+}
+
+/* line 4, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-danger {
+ color: #77260d;
+ background-color: #f8ccbf;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.list-group-item-danger.list-group-item-action:hover, .list-group-item-danger.list-group-item-action:focus {
+ color: #77260d;
+ background-color: #f5baa8;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-danger.list-group-item-action.active {
+ color: #ffffff;
+ background-color: #77260d;
+ border-color: #77260d;
+}
+
+/* line 4, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-light {
+ color: gray;
+ background-color: #fdfdfd;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.list-group-item-light.list-group-item-action:hover, .list-group-item-light.list-group-item-action:focus {
+ color: gray;
+ background-color: #f0f0f0;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-light.list-group-item-action.active {
+ color: #ffffff;
+ background-color: gray;
+ border-color: gray;
+}
+
+/* line 4, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-dark {
+ color: #1b1e21;
+ background-color: #c6c8ca;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.list-group-item-dark.list-group-item-action:hover, .list-group-item-dark.list-group-item-action:focus {
+ color: #1b1e21;
+ background-color: #b9bbbe;
+}
+
+/* line 14, node_modules/bootstrap/scss/mixins/_list-group.scss */
+.list-group-item-dark.list-group-item-action.active {
+ color: #ffffff;
+ background-color: #1b1e21;
+ border-color: #1b1e21;
+}
+
+/* line 1, node_modules/bootstrap/scss/_close.scss */
+.close {
+ float: right;
+ font-size: 1.5rem;
+ font-weight: 700;
+ line-height: 1;
+ color: #000000;
+ text-shadow: 0 1px 0 #ffffff;
+ opacity: .5;
+}
+
+/* line 13, node_modules/bootstrap/scss/mixins/_hover.scss */
+.close:hover {
+ color: #000000;
+ text-decoration: none;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.close:not(:disabled):not(.disabled):hover, .close:not(:disabled):not(.disabled):focus {
+ opacity: .75;
+}
+
+/* line 29, node_modules/bootstrap/scss/_close.scss */
+button.close {
+ padding: 0;
+ background-color: transparent;
+ border: 0;
+}
+
+/* line 38, node_modules/bootstrap/scss/_close.scss */
+a.close.disabled {
+ pointer-events: none;
+}
+
+/* line 1, node_modules/bootstrap/scss/_toasts.scss */
+.toast {
+ -ms-flex-preferred-size: 350px;
+ flex-basis: 350px;
+ max-width: 350px;
+ font-size: 0.875rem;
+ background-color: rgba(255, 255, 255, 0.85);
+ background-clip: padding-box;
+ border: 1px solid rgba(0, 0, 0, 0.1);
+ -webkit-box-shadow: 0 0.25rem 0.75rem rgba(0, 0, 0, 0.1);
+ box-shadow: 0 0.25rem 0.75rem rgba(0, 0, 0, 0.1);
+ opacity: 0;
+ border-radius: 0.25rem;
+}
+
+/* line 15, node_modules/bootstrap/scss/_toasts.scss */
+.toast:not(:last-child) {
+ margin-bottom: 0.75rem;
+}
+
+/* line 19, node_modules/bootstrap/scss/_toasts.scss */
+.toast.showing {
+ opacity: 1;
+}
+
+/* line 23, node_modules/bootstrap/scss/_toasts.scss */
+.toast.show {
+ display: block;
+ opacity: 1;
+}
+
+/* line 28, node_modules/bootstrap/scss/_toasts.scss */
+.toast.hide {
+ display: none;
+}
+
+/* line 33, node_modules/bootstrap/scss/_toasts.scss */
+.toast-header {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ padding: 0.25rem 0.75rem;
+ color: #6c757d;
+ background-color: rgba(255, 255, 255, 0.85);
+ background-clip: padding-box;
+ border-bottom: 1px solid rgba(0, 0, 0, 0.05);
+ border-top-left-radius: calc(0.25rem - 1px);
+ border-top-right-radius: calc(0.25rem - 1px);
+}
+
+/* line 44, node_modules/bootstrap/scss/_toasts.scss */
+.toast-body {
+ padding: 0.75rem;
+}
+
+/* line 7, node_modules/bootstrap/scss/_modal.scss */
+.modal-open {
+ overflow: hidden;
+}
+
+/* line 11, node_modules/bootstrap/scss/_modal.scss */
+.modal-open .modal {
+ overflow-x: hidden;
+ overflow-y: auto;
+}
+
+/* line 18, node_modules/bootstrap/scss/_modal.scss */
+.modal {
+ position: fixed;
+ top: 0;
+ left: 0;
+ z-index: 1050;
+ display: none;
+ width: 100%;
+ height: 100%;
+ overflow: hidden;
+ outline: 0;
+}
+
+/* line 36, node_modules/bootstrap/scss/_modal.scss */
+.modal-dialog {
+ position: relative;
+ width: auto;
+ margin: 0.5rem;
+ pointer-events: none;
+}
+
+/* line 44, node_modules/bootstrap/scss/_modal.scss */
+.modal.fade .modal-dialog {
+ -webkit-transition: -webkit-transform 0.3s ease-out;
+ transition: -webkit-transform 0.3s ease-out;
+ transition: transform 0.3s ease-out;
+ transition: transform 0.3s ease-out, -webkit-transform 0.3s ease-out;
+ -webkit-transform: translate(0, -50px);
+ transform: translate(0, -50px);
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 44, node_modules/bootstrap/scss/_modal.scss */
+ .modal.fade .modal-dialog {
+ -webkit-transition: none;
+ transition: none;
+ }
+}
+
+/* line 48, node_modules/bootstrap/scss/_modal.scss */
+.modal.show .modal-dialog {
+ -webkit-transform: none;
+ transform: none;
+}
+
+/* line 53, node_modules/bootstrap/scss/_modal.scss */
+.modal.modal-static .modal-dialog {
+ -webkit-transform: scale(1.02);
+ transform: scale(1.02);
+}
+
+/* line 58, node_modules/bootstrap/scss/_modal.scss */
+.modal-dialog-scrollable {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ max-height: calc(100% - 1rem);
+}
+
+/* line 62, node_modules/bootstrap/scss/_modal.scss */
+.modal-dialog-scrollable .modal-content {
+ max-height: calc(100vh - 1rem);
+ overflow: hidden;
+}
+
+/* line 67, node_modules/bootstrap/scss/_modal.scss */
+.modal-dialog-scrollable .modal-header,
+.modal-dialog-scrollable .modal-footer {
+ -ms-flex-negative: 0;
+ flex-shrink: 0;
+}
+
+/* line 72, node_modules/bootstrap/scss/_modal.scss */
+.modal-dialog-scrollable .modal-body {
+ overflow-y: auto;
+}
+
+/* line 77, node_modules/bootstrap/scss/_modal.scss */
+.modal-dialog-centered {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ min-height: calc(100% - 1rem);
+}
+
+/* line 83, node_modules/bootstrap/scss/_modal.scss */
+.modal-dialog-centered::before {
+ display: block;
+ height: calc(100vh - 1rem);
+ height: -webkit-min-content;
+ height: -moz-min-content;
+ height: min-content;
+ content: "";
+}
+
+/* line 91, node_modules/bootstrap/scss/_modal.scss */
+.modal-dialog-centered.modal-dialog-scrollable {
+ -webkit-box-orient: vertical;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: column;
+ flex-direction: column;
+ -webkit-box-pack: center;
+ -ms-flex-pack: center;
+ justify-content: center;
+ height: 100%;
+}
+
+/* line 96, node_modules/bootstrap/scss/_modal.scss */
+.modal-dialog-centered.modal-dialog-scrollable .modal-content {
+ max-height: none;
+}
+
+/* line 100, node_modules/bootstrap/scss/_modal.scss */
+.modal-dialog-centered.modal-dialog-scrollable::before {
+ content: none;
+}
+
+/* line 107, node_modules/bootstrap/scss/_modal.scss */
+.modal-content {
+ position: relative;
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-orient: vertical;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: column;
+ flex-direction: column;
+ width: 100%;
+ pointer-events: auto;
+ background-color: #ffffff;
+ background-clip: padding-box;
+ border: 1px solid rgba(0, 0, 0, 0.2);
+ border-radius: 0.3rem;
+ outline: 0;
+}
+
+/* line 125, node_modules/bootstrap/scss/_modal.scss */
+.modal-backdrop {
+ position: fixed;
+ top: 0;
+ left: 0;
+ z-index: 1040;
+ width: 100vw;
+ height: 100vh;
+ background-color: #000000;
+}
+
+/* line 135, node_modules/bootstrap/scss/_modal.scss */
+.modal-backdrop.fade {
+ opacity: 0;
+}
+
+/* line 136, node_modules/bootstrap/scss/_modal.scss */
+.modal-backdrop.show {
+ opacity: 0.5;
+}
+
+/* line 141, node_modules/bootstrap/scss/_modal.scss */
+.modal-header {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-align: start;
+ -ms-flex-align: start;
+ align-items: flex-start;
+ -webkit-box-pack: justify;
+ -ms-flex-pack: justify;
+ justify-content: space-between;
+ padding: 1rem 1rem;
+ border-bottom: 1px solid #dee2e6;
+ border-top-left-radius: calc(0.3rem - 1px);
+ border-top-right-radius: calc(0.3rem - 1px);
+}
+
+/* line 149, node_modules/bootstrap/scss/_modal.scss */
+.modal-header .close {
+ padding: 1rem 1rem;
+ margin: -1rem -1rem -1rem auto;
+}
+
+/* line 157, node_modules/bootstrap/scss/_modal.scss */
+.modal-title {
+ margin-bottom: 0;
+ line-height: 1.5;
+}
+
+/* line 164, node_modules/bootstrap/scss/_modal.scss */
+.modal-body {
+ position: relative;
+ -webkit-box-flex: 1;
+ -ms-flex: 1 1 auto;
+ flex: 1 1 auto;
+ padding: 1rem;
+}
+
+/* line 173, node_modules/bootstrap/scss/_modal.scss */
+.modal-footer {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -ms-flex-wrap: wrap;
+ flex-wrap: wrap;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ -webkit-box-pack: end;
+ -ms-flex-pack: end;
+ justify-content: flex-end;
+ padding: 0.75rem;
+ border-top: 1px solid #dee2e6;
+ border-bottom-right-radius: calc(0.3rem - 1px);
+ border-bottom-left-radius: calc(0.3rem - 1px);
+}
+
+/* line 185, node_modules/bootstrap/scss/_modal.scss */
+.modal-footer > * {
+ margin: 0.25rem;
+}
+
+/* line 191, node_modules/bootstrap/scss/_modal.scss */
+.modal-scrollbar-measure {
+ position: absolute;
+ top: -9999px;
+ width: 50px;
+ height: 50px;
+ overflow: scroll;
+}
+
+@media (min-width: 576px) {
+ /* line 202, node_modules/bootstrap/scss/_modal.scss */
+ .modal-dialog {
+ max-width: 500px;
+ margin: 1.75rem auto;
+ }
+ /* line 207, node_modules/bootstrap/scss/_modal.scss */
+ .modal-dialog-scrollable {
+ max-height: calc(100% - 3.5rem);
+ }
+ /* line 210, node_modules/bootstrap/scss/_modal.scss */
+ .modal-dialog-scrollable .modal-content {
+ max-height: calc(100vh - 3.5rem);
+ }
+ /* line 215, node_modules/bootstrap/scss/_modal.scss */
+ .modal-dialog-centered {
+ min-height: calc(100% - 3.5rem);
+ }
+ /* line 218, node_modules/bootstrap/scss/_modal.scss */
+ .modal-dialog-centered::before {
+ height: calc(100vh - 3.5rem);
+ height: -webkit-min-content;
+ height: -moz-min-content;
+ height: min-content;
+ }
+ /* line 228, node_modules/bootstrap/scss/_modal.scss */
+ .modal-sm {
+ max-width: 300px;
+ }
+}
+
+@media (min-width: 1024px) {
+ /* line 232, node_modules/bootstrap/scss/_modal.scss */
+ .modal-lg,
+ .modal-xl {
+ max-width: 800px;
+ }
+}
+
+@media (min-width: 1280px) {
+ /* line 239, node_modules/bootstrap/scss/_modal.scss */
+ .modal-xl {
+ max-width: 1140px;
+ }
+}
+
+/* line 2, node_modules/bootstrap/scss/_tooltip.scss */
+.tooltip {
+ position: absolute;
+ z-index: 1070;
+ display: block;
+ margin: 0;
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";
+ font-style: normal;
+ font-weight: 400;
+ line-height: 1.5;
+ text-align: left;
+ text-align: start;
+ text-decoration: none;
+ text-shadow: none;
+ text-transform: none;
+ letter-spacing: normal;
+ word-break: normal;
+ word-spacing: normal;
+ white-space: normal;
+ line-break: auto;
+ font-size: 0.875rem;
+ word-wrap: break-word;
+ opacity: 0;
+}
+
+/* line 15, node_modules/bootstrap/scss/_tooltip.scss */
+.tooltip.show {
+ opacity: 0.9;
+}
+
+/* line 17, node_modules/bootstrap/scss/_tooltip.scss */
+.tooltip .arrow {
+ position: absolute;
+ display: block;
+ width: 0.8rem;
+ height: 0.4rem;
+}
+
+/* line 23, node_modules/bootstrap/scss/_tooltip.scss */
+.tooltip .arrow::before {
+ position: absolute;
+ content: "";
+ border-color: transparent;
+ border-style: solid;
+}
+
+/* line 32, node_modules/bootstrap/scss/_tooltip.scss */
+.bs-tooltip-top, .bs-tooltip-auto[x-placement^="top"] {
+ padding: 0.4rem 0;
+}
+
+/* line 35, node_modules/bootstrap/scss/_tooltip.scss */
+.bs-tooltip-top .arrow, .bs-tooltip-auto[x-placement^="top"] .arrow {
+ bottom: 0;
+}
+
+/* line 38, node_modules/bootstrap/scss/_tooltip.scss */
+.bs-tooltip-top .arrow::before, .bs-tooltip-auto[x-placement^="top"] .arrow::before {
+ top: 0;
+ border-width: 0.4rem 0.4rem 0;
+ border-top-color: #000000;
+}
+
+/* line 46, node_modules/bootstrap/scss/_tooltip.scss */
+.bs-tooltip-right, .bs-tooltip-auto[x-placement^="right"] {
+ padding: 0 0.4rem;
+}
+
+/* line 49, node_modules/bootstrap/scss/_tooltip.scss */
+.bs-tooltip-right .arrow, .bs-tooltip-auto[x-placement^="right"] .arrow {
+ left: 0;
+ width: 0.4rem;
+ height: 0.8rem;
+}
+
+/* line 54, node_modules/bootstrap/scss/_tooltip.scss */
+.bs-tooltip-right .arrow::before, .bs-tooltip-auto[x-placement^="right"] .arrow::before {
+ right: 0;
+ border-width: 0.4rem 0.4rem 0.4rem 0;
+ border-right-color: #000000;
+}
+
+/* line 62, node_modules/bootstrap/scss/_tooltip.scss */
+.bs-tooltip-bottom, .bs-tooltip-auto[x-placement^="bottom"] {
+ padding: 0.4rem 0;
+}
+
+/* line 65, node_modules/bootstrap/scss/_tooltip.scss */
+.bs-tooltip-bottom .arrow, .bs-tooltip-auto[x-placement^="bottom"] .arrow {
+ top: 0;
+}
+
+/* line 68, node_modules/bootstrap/scss/_tooltip.scss */
+.bs-tooltip-bottom .arrow::before, .bs-tooltip-auto[x-placement^="bottom"] .arrow::before {
+ bottom: 0;
+ border-width: 0 0.4rem 0.4rem;
+ border-bottom-color: #000000;
+}
+
+/* line 76, node_modules/bootstrap/scss/_tooltip.scss */
+.bs-tooltip-left, .bs-tooltip-auto[x-placement^="left"] {
+ padding: 0 0.4rem;
+}
+
+/* line 79, node_modules/bootstrap/scss/_tooltip.scss */
+.bs-tooltip-left .arrow, .bs-tooltip-auto[x-placement^="left"] .arrow {
+ right: 0;
+ width: 0.4rem;
+ height: 0.8rem;
+}
+
+/* line 84, node_modules/bootstrap/scss/_tooltip.scss */
+.bs-tooltip-left .arrow::before, .bs-tooltip-auto[x-placement^="left"] .arrow::before {
+ left: 0;
+ border-width: 0.4rem 0 0.4rem 0.4rem;
+ border-left-color: #000000;
+}
+
+/* line 108, node_modules/bootstrap/scss/_tooltip.scss */
+.tooltip-inner {
+ max-width: 200px;
+ padding: 0.25rem 0.5rem;
+ color: #ffffff;
+ text-align: center;
+ background-color: #000000;
+ border-radius: 0.25rem;
+}
+
+/* line 1, node_modules/bootstrap/scss/_popover.scss */
+.popover {
+ position: absolute;
+ top: 0;
+ left: 0;
+ z-index: 1060;
+ display: block;
+ max-width: 276px;
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";
+ font-style: normal;
+ font-weight: 400;
+ line-height: 1.5;
+ text-align: left;
+ text-align: start;
+ text-decoration: none;
+ text-shadow: none;
+ text-transform: none;
+ letter-spacing: normal;
+ word-break: normal;
+ word-spacing: normal;
+ white-space: normal;
+ line-break: auto;
+ font-size: 0.875rem;
+ word-wrap: break-word;
+ background-color: #ffffff;
+ background-clip: padding-box;
+ border: 1px solid rgba(0, 0, 0, 0.2);
+ border-radius: 0.3rem;
+}
+
+/* line 20, node_modules/bootstrap/scss/_popover.scss */
+.popover .arrow {
+ position: absolute;
+ display: block;
+ width: 1rem;
+ height: 0.5rem;
+ margin: 0 0.3rem;
+}
+
+/* line 27, node_modules/bootstrap/scss/_popover.scss */
+.popover .arrow::before, .popover .arrow::after {
+ position: absolute;
+ display: block;
+ content: "";
+ border-color: transparent;
+ border-style: solid;
+}
+
+/* line 38, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-top, .bs-popover-auto[x-placement^="top"] {
+ margin-bottom: 0.5rem;
+}
+
+/* line 41, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-top > .arrow, .bs-popover-auto[x-placement^="top"] > .arrow {
+ bottom: calc(-0.5rem - 1px);
+}
+
+/* line 44, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-top > .arrow::before, .bs-popover-auto[x-placement^="top"] > .arrow::before {
+ bottom: 0;
+ border-width: 0.5rem 0.5rem 0;
+ border-top-color: rgba(0, 0, 0, 0.25);
+}
+
+/* line 50, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-top > .arrow::after, .bs-popover-auto[x-placement^="top"] > .arrow::after {
+ bottom: 1px;
+ border-width: 0.5rem 0.5rem 0;
+ border-top-color: #ffffff;
+}
+
+/* line 58, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-right, .bs-popover-auto[x-placement^="right"] {
+ margin-left: 0.5rem;
+}
+
+/* line 61, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-right > .arrow, .bs-popover-auto[x-placement^="right"] > .arrow {
+ left: calc(-0.5rem - 1px);
+ width: 0.5rem;
+ height: 1rem;
+ margin: 0.3rem 0;
+}
+
+/* line 67, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-right > .arrow::before, .bs-popover-auto[x-placement^="right"] > .arrow::before {
+ left: 0;
+ border-width: 0.5rem 0.5rem 0.5rem 0;
+ border-right-color: rgba(0, 0, 0, 0.25);
+}
+
+/* line 73, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-right > .arrow::after, .bs-popover-auto[x-placement^="right"] > .arrow::after {
+ left: 1px;
+ border-width: 0.5rem 0.5rem 0.5rem 0;
+ border-right-color: #ffffff;
+}
+
+/* line 81, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-bottom, .bs-popover-auto[x-placement^="bottom"] {
+ margin-top: 0.5rem;
+}
+
+/* line 84, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-bottom > .arrow, .bs-popover-auto[x-placement^="bottom"] > .arrow {
+ top: calc(-0.5rem - 1px);
+}
+
+/* line 87, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-bottom > .arrow::before, .bs-popover-auto[x-placement^="bottom"] > .arrow::before {
+ top: 0;
+ border-width: 0 0.5rem 0.5rem 0.5rem;
+ border-bottom-color: rgba(0, 0, 0, 0.25);
+}
+
+/* line 93, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-bottom > .arrow::after, .bs-popover-auto[x-placement^="bottom"] > .arrow::after {
+ top: 1px;
+ border-width: 0 0.5rem 0.5rem 0.5rem;
+ border-bottom-color: #ffffff;
+}
+
+/* line 101, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-bottom .popover-header::before, .bs-popover-auto[x-placement^="bottom"] .popover-header::before {
+ position: absolute;
+ top: 0;
+ left: 50%;
+ display: block;
+ width: 1rem;
+ margin-left: -0.5rem;
+ content: "";
+ border-bottom: 1px solid #f7f7f7;
+}
+
+/* line 113, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-left, .bs-popover-auto[x-placement^="left"] {
+ margin-right: 0.5rem;
+}
+
+/* line 116, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-left > .arrow, .bs-popover-auto[x-placement^="left"] > .arrow {
+ right: calc(-0.5rem - 1px);
+ width: 0.5rem;
+ height: 1rem;
+ margin: 0.3rem 0;
+}
+
+/* line 122, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-left > .arrow::before, .bs-popover-auto[x-placement^="left"] > .arrow::before {
+ right: 0;
+ border-width: 0.5rem 0 0.5rem 0.5rem;
+ border-left-color: rgba(0, 0, 0, 0.25);
+}
+
+/* line 128, node_modules/bootstrap/scss/_popover.scss */
+.bs-popover-left > .arrow::after, .bs-popover-auto[x-placement^="left"] > .arrow::after {
+ right: 1px;
+ border-width: 0.5rem 0 0.5rem 0.5rem;
+ border-left-color: #ffffff;
+}
+
+/* line 153, node_modules/bootstrap/scss/_popover.scss */
+.popover-header {
+ padding: 0.5rem 0.75rem;
+ margin-bottom: 0;
+ font-size: 1rem;
+ background-color: #f7f7f7;
+ border-bottom: 1px solid #ebebeb;
+ border-top-left-radius: calc(0.3rem - 1px);
+ border-top-right-radius: calc(0.3rem - 1px);
+}
+
+/* line 162, node_modules/bootstrap/scss/_popover.scss */
+.popover-header:empty {
+ display: none;
+}
+
+/* line 167, node_modules/bootstrap/scss/_popover.scss */
+.popover-body {
+ padding: 0.5rem 0.75rem;
+ color: #464746;
+}
+
+/* line 14, node_modules/bootstrap/scss/_carousel.scss */
+.carousel {
+ position: relative;
+}
+
+/* line 18, node_modules/bootstrap/scss/_carousel.scss */
+.carousel.pointer-event {
+ -ms-touch-action: pan-y;
+ touch-action: pan-y;
+}
+
+/* line 22, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-inner {
+ position: relative;
+ width: 100%;
+ overflow: hidden;
+}
+
+/* line 2, node_modules/bootstrap/scss/mixins/_clearfix.scss */
+.carousel-inner::after {
+ display: block;
+ clear: both;
+ content: "";
+}
+
+/* line 29, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-item {
+ position: relative;
+ display: none;
+ float: left;
+ width: 100%;
+ margin-right: -100%;
+ -webkit-backface-visibility: hidden;
+ backface-visibility: hidden;
+ -webkit-transition: -webkit-transform 0.6s ease-in-out;
+ transition: -webkit-transform 0.6s ease-in-out;
+ transition: transform 0.6s ease-in-out;
+ transition: transform 0.6s ease-in-out, -webkit-transform 0.6s ease-in-out;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 29, node_modules/bootstrap/scss/_carousel.scss */
+ .carousel-item {
+ -webkit-transition: none;
+ transition: none;
+ }
+}
+
+/* line 39, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-item.active,
+.carousel-item-next,
+.carousel-item-prev {
+ display: block;
+}
+
+/* line 45, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-item-next:not(.carousel-item-left),
+.active.carousel-item-right {
+ -webkit-transform: translateX(100%);
+ transform: translateX(100%);
+}
+
+/* line 50, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-item-prev:not(.carousel-item-right),
+.active.carousel-item-left {
+ -webkit-transform: translateX(-100%);
+ transform: translateX(-100%);
+}
+
+/* line 61, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-fade .carousel-item {
+ opacity: 0;
+ -webkit-transition-property: opacity;
+ transition-property: opacity;
+ -webkit-transform: none;
+ transform: none;
+}
+
+/* line 67, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-fade .carousel-item.active,
+.carousel-fade .carousel-item-next.carousel-item-left,
+.carousel-fade .carousel-item-prev.carousel-item-right {
+ z-index: 1;
+ opacity: 1;
+}
+
+/* line 74, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-fade .active.carousel-item-left,
+.carousel-fade .active.carousel-item-right {
+ z-index: 0;
+ opacity: 0;
+ -webkit-transition: opacity 0s 0.6s;
+ transition: opacity 0s 0.6s;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 74, node_modules/bootstrap/scss/_carousel.scss */
+ .carousel-fade .active.carousel-item-left,
+ .carousel-fade .active.carousel-item-right {
+ -webkit-transition: none;
+ transition: none;
+ }
+}
+
+/* line 87, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-control-prev,
+.carousel-control-next {
+ position: absolute;
+ top: 0;
+ bottom: 0;
+ z-index: 1;
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ -webkit-box-pack: center;
+ -ms-flex-pack: center;
+ justify-content: center;
+ width: 15%;
+ color: #ffffff;
+ text-align: center;
+ opacity: 0.5;
+ -webkit-transition: opacity 0.15s ease;
+ transition: opacity 0.15s ease;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 87, node_modules/bootstrap/scss/_carousel.scss */
+ .carousel-control-prev,
+ .carousel-control-next {
+ -webkit-transition: none;
+ transition: none;
+ }
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+.carousel-control-prev:hover, .carousel-control-prev:focus,
+.carousel-control-next:hover,
+.carousel-control-next:focus {
+ color: #ffffff;
+ text-decoration: none;
+ outline: 0;
+ opacity: 0.9;
+}
+
+/* line 111, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-control-prev {
+ left: 0;
+}
+
+/* line 117, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-control-next {
+ right: 0;
+}
+
+/* line 125, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-control-prev-icon,
+.carousel-control-next-icon {
+ display: inline-block;
+ width: 20px;
+ height: 20px;
+ background: no-repeat 50% / 100% 100%;
+}
+
+/* line 132, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-control-prev-icon {
+ background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='%23ffffff' width='8' height='8' viewBox='0 0 8 8'%3e%3cpath d='M5.25 0l-4 4 4 4 1.5-1.5L4.25 4l2.5-2.5L5.25 0z'/%3e%3c/svg%3e");
+}
+
+/* line 135, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-control-next-icon {
+ background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='%23ffffff' width='8' height='8' viewBox='0 0 8 8'%3e%3cpath d='M2.75 0l-1.5 1.5L3.75 4l-2.5 2.5L2.75 8l4-4-4-4z'/%3e%3c/svg%3e");
+}
+
+/* line 145, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-indicators {
+ position: absolute;
+ right: 0;
+ bottom: 0;
+ left: 0;
+ z-index: 15;
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-pack: center;
+ -ms-flex-pack: center;
+ justify-content: center;
+ padding-left: 0;
+ margin-right: 15%;
+ margin-left: 15%;
+ list-style: none;
+}
+
+/* line 159, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-indicators li {
+ -webkit-box-sizing: content-box;
+ box-sizing: content-box;
+ -webkit-box-flex: 0;
+ -ms-flex: 0 1 auto;
+ flex: 0 1 auto;
+ width: 30px;
+ height: 3px;
+ margin-right: 3px;
+ margin-left: 3px;
+ text-indent: -999px;
+ cursor: pointer;
+ background-color: #ffffff;
+ background-clip: padding-box;
+ border-top: 10px solid transparent;
+ border-bottom: 10px solid transparent;
+ opacity: .5;
+ -webkit-transition: opacity 0.6s ease;
+ transition: opacity 0.6s ease;
+}
+
+@media (prefers-reduced-motion: reduce) {
+ /* line 159, node_modules/bootstrap/scss/_carousel.scss */
+ .carousel-indicators li {
+ -webkit-transition: none;
+ transition: none;
+ }
+}
+
+/* line 177, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-indicators .active {
+ opacity: 1;
+}
+
+/* line 187, node_modules/bootstrap/scss/_carousel.scss */
+.carousel-caption {
+ position: absolute;
+ right: 15%;
+ bottom: 20px;
+ left: 15%;
+ z-index: 10;
+ padding-top: 20px;
+ padding-bottom: 20px;
+ color: #ffffff;
+ text-align: center;
+}
+
+@-webkit-keyframes spinner-border {
+ to {
+ -webkit-transform: rotate(360deg);
+ transform: rotate(360deg);
+ }
+}
+
+@keyframes spinner-border {
+ to {
+ -webkit-transform: rotate(360deg);
+ transform: rotate(360deg);
+ }
+}
+
+/* line 9, node_modules/bootstrap/scss/_spinners.scss */
+.spinner-border {
+ display: inline-block;
+ width: 2rem;
+ height: 2rem;
+ vertical-align: text-bottom;
+ border: 0.25em solid currentColor;
+ border-right-color: transparent;
+ border-radius: 50%;
+ -webkit-animation: spinner-border .75s linear infinite;
+ animation: spinner-border .75s linear infinite;
+}
+
+/* line 21, node_modules/bootstrap/scss/_spinners.scss */
+.spinner-border-sm {
+ width: 1rem;
+ height: 1rem;
+ border-width: 0.2em;
+}
+
+@-webkit-keyframes spinner-grow {
+ 0% {
+ -webkit-transform: scale(0);
+ transform: scale(0);
+ }
+ 50% {
+ opacity: 1;
+ -webkit-transform: none;
+ transform: none;
+ }
+}
+
+@keyframes spinner-grow {
+ 0% {
+ -webkit-transform: scale(0);
+ transform: scale(0);
+ }
+ 50% {
+ opacity: 1;
+ -webkit-transform: none;
+ transform: none;
+ }
+}
+
+/* line 41, node_modules/bootstrap/scss/_spinners.scss */
+.spinner-grow {
+ display: inline-block;
+ width: 2rem;
+ height: 2rem;
+ vertical-align: text-bottom;
+ background-color: currentColor;
+ border-radius: 50%;
+ opacity: 0;
+ -webkit-animation: spinner-grow .75s linear infinite;
+ animation: spinner-grow .75s linear infinite;
+}
+
+/* line 53, node_modules/bootstrap/scss/_spinners.scss */
+.spinner-grow-sm {
+ width: 1rem;
+ height: 1rem;
+}
+
+/* line 3, node_modules/bootstrap/scss/utilities/_align.scss */
+.align-baseline {
+ vertical-align: baseline !important;
+}
+
+/* line 4, node_modules/bootstrap/scss/utilities/_align.scss */
+.align-top {
+ vertical-align: top !important;
+}
+
+/* line 5, node_modules/bootstrap/scss/utilities/_align.scss */
+.align-middle {
+ vertical-align: middle !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/utilities/_align.scss */
+.align-bottom {
+ vertical-align: bottom !important;
+}
+
+/* line 7, node_modules/bootstrap/scss/utilities/_align.scss */
+.align-text-bottom {
+ vertical-align: text-bottom !important;
+}
+
+/* line 8, node_modules/bootstrap/scss/utilities/_align.scss */
+.align-text-top {
+ vertical-align: text-top !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_background-variant.scss */
+.bg-primary {
+ background-color: #464746 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.bg-primary:hover, a.bg-primary:focus,
+button.bg-primary:hover,
+button.bg-primary:focus {
+ background-color: #2d2d2d !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_background-variant.scss */
+.bg-secondary {
+ background-color: #f0ebe3 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.bg-secondary:hover, a.bg-secondary:focus,
+button.bg-secondary:hover,
+button.bg-secondary:focus {
+ background-color: #ded3c2 !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_background-variant.scss */
+.bg-success {
+ background-color: #f0ebe3 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.bg-success:hover, a.bg-success:focus,
+button.bg-success:hover,
+button.bg-success:focus {
+ background-color: #ded3c2 !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_background-variant.scss */
+.bg-info {
+ background-color: #464746 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.bg-info:hover, a.bg-info:focus,
+button.bg-info:hover,
+button.bg-info:focus {
+ background-color: #2d2d2d !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_background-variant.scss */
+.bg-warning {
+ background-color: #464746 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.bg-warning:hover, a.bg-warning:focus,
+button.bg-warning:hover,
+button.bg-warning:focus {
+ background-color: #2d2d2d !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_background-variant.scss */
+.bg-danger {
+ background-color: #e54a19 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.bg-danger:hover, a.bg-danger:focus,
+button.bg-danger:hover,
+button.bg-danger:focus {
+ background-color: #b73b14 !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_background-variant.scss */
+.bg-light {
+ background-color: #f7f7f7 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.bg-light:hover, a.bg-light:focus,
+button.bg-light:hover,
+button.bg-light:focus {
+ background-color: #dedede !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_background-variant.scss */
+.bg-dark {
+ background-color: #343a40 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.bg-dark:hover, a.bg-dark:focus,
+button.bg-dark:hover,
+button.bg-dark:focus {
+ background-color: #1d2124 !important;
+}
+
+/* line 13, node_modules/bootstrap/scss/utilities/_background.scss */
+.bg-white {
+ background-color: #ffffff !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/utilities/_background.scss */
+.bg-transparent {
+ background-color: transparent !important;
+}
+
+/* line 7, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border {
+ border: 1px solid #dee2e6 !important;
+}
+
+/* line 8, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-top {
+ border-top: 1px solid #dee2e6 !important;
+}
+
+/* line 9, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-right {
+ border-right: 1px solid #dee2e6 !important;
+}
+
+/* line 10, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-bottom {
+ border-bottom: 1px solid #dee2e6 !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-left {
+ border-left: 1px solid #dee2e6 !important;
+}
+
+/* line 13, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-0 {
+ border: 0 !important;
+}
+
+/* line 14, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-top-0 {
+ border-top: 0 !important;
+}
+
+/* line 15, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-right-0 {
+ border-right: 0 !important;
+}
+
+/* line 16, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-bottom-0 {
+ border-bottom: 0 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-left-0 {
+ border-left: 0 !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-primary {
+ border-color: #464746 !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-secondary {
+ border-color: #f0ebe3 !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-success {
+ border-color: #f0ebe3 !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-info {
+ border-color: #464746 !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-warning {
+ border-color: #464746 !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-danger {
+ border-color: #e54a19 !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-light {
+ border-color: #f7f7f7 !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-dark {
+ border-color: #343a40 !important;
+}
+
+/* line 25, node_modules/bootstrap/scss/utilities/_borders.scss */
+.border-white {
+ border-color: #ffffff !important;
+}
+
+/* line 33, node_modules/bootstrap/scss/utilities/_borders.scss */
+.rounded-sm {
+ border-radius: 0.2rem !important;
+}
+
+/* line 37, node_modules/bootstrap/scss/utilities/_borders.scss */
+.rounded {
+ border-radius: 0.25rem !important;
+}
+
+/* line 41, node_modules/bootstrap/scss/utilities/_borders.scss */
+.rounded-top {
+ border-top-left-radius: 0.25rem !important;
+ border-top-right-radius: 0.25rem !important;
+}
+
+/* line 46, node_modules/bootstrap/scss/utilities/_borders.scss */
+.rounded-right {
+ border-top-right-radius: 0.25rem !important;
+ border-bottom-right-radius: 0.25rem !important;
+}
+
+/* line 51, node_modules/bootstrap/scss/utilities/_borders.scss */
+.rounded-bottom {
+ border-bottom-right-radius: 0.25rem !important;
+ border-bottom-left-radius: 0.25rem !important;
+}
+
+/* line 56, node_modules/bootstrap/scss/utilities/_borders.scss */
+.rounded-left {
+ border-top-left-radius: 0.25rem !important;
+ border-bottom-left-radius: 0.25rem !important;
+}
+
+/* line 61, node_modules/bootstrap/scss/utilities/_borders.scss */
+.rounded-lg {
+ border-radius: 0.3rem !important;
+}
+
+/* line 65, node_modules/bootstrap/scss/utilities/_borders.scss */
+.rounded-circle {
+ border-radius: 50% !important;
+}
+
+/* line 69, node_modules/bootstrap/scss/utilities/_borders.scss */
+.rounded-pill {
+ border-radius: 50rem !important;
+}
+
+/* line 73, node_modules/bootstrap/scss/utilities/_borders.scss */
+.rounded-0 {
+ border-radius: 0 !important;
+}
+
+/* line 2, node_modules/bootstrap/scss/mixins/_clearfix.scss */
+.clearfix::after {
+ display: block;
+ clear: both;
+ content: "";
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+.d-none {
+ display: none !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+.d-inline {
+ display: inline !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+.d-inline-block {
+ display: inline-block !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+.d-block {
+ display: block !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+.d-table {
+ display: table !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+.d-table-row {
+ display: table-row !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+.d-table-cell {
+ display: table-cell !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+.d-flex {
+ display: -webkit-box !important;
+ display: -ms-flexbox !important;
+ display: flex !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+.d-inline-flex {
+ display: -webkit-inline-box !important;
+ display: -ms-inline-flexbox !important;
+ display: inline-flex !important;
+}
+
+@media (min-width: 576px) {
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-sm-none {
+ display: none !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-sm-inline {
+ display: inline !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-sm-inline-block {
+ display: inline-block !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-sm-block {
+ display: block !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-sm-table {
+ display: table !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-sm-table-row {
+ display: table-row !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-sm-table-cell {
+ display: table-cell !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-sm-flex {
+ display: -webkit-box !important;
+ display: -ms-flexbox !important;
+ display: flex !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-sm-inline-flex {
+ display: -webkit-inline-box !important;
+ display: -ms-inline-flexbox !important;
+ display: inline-flex !important;
+ }
+}
+
+@media (min-width: 768px) {
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-md-none {
+ display: none !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-md-inline {
+ display: inline !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-md-inline-block {
+ display: inline-block !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-md-block {
+ display: block !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-md-table {
+ display: table !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-md-table-row {
+ display: table-row !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-md-table-cell {
+ display: table-cell !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-md-flex {
+ display: -webkit-box !important;
+ display: -ms-flexbox !important;
+ display: flex !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-md-inline-flex {
+ display: -webkit-inline-box !important;
+ display: -ms-inline-flexbox !important;
+ display: inline-flex !important;
+ }
+}
+
+@media (min-width: 1024px) {
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-lg-none {
+ display: none !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-lg-inline {
+ display: inline !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-lg-inline-block {
+ display: inline-block !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-lg-block {
+ display: block !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-lg-table {
+ display: table !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-lg-table-row {
+ display: table-row !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-lg-table-cell {
+ display: table-cell !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-lg-flex {
+ display: -webkit-box !important;
+ display: -ms-flexbox !important;
+ display: flex !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-lg-inline-flex {
+ display: -webkit-inline-box !important;
+ display: -ms-inline-flexbox !important;
+ display: inline-flex !important;
+ }
+}
+
+@media (min-width: 1280px) {
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xl-none {
+ display: none !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xl-inline {
+ display: inline !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xl-inline-block {
+ display: inline-block !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xl-block {
+ display: block !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xl-table {
+ display: table !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xl-table-row {
+ display: table-row !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xl-table-cell {
+ display: table-cell !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xl-flex {
+ display: -webkit-box !important;
+ display: -ms-flexbox !important;
+ display: flex !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xl-inline-flex {
+ display: -webkit-inline-box !important;
+ display: -ms-inline-flexbox !important;
+ display: inline-flex !important;
+ }
+}
+
+@media (min-width: 1440px) {
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xxl-none {
+ display: none !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xxl-inline {
+ display: inline !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xxl-inline-block {
+ display: inline-block !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xxl-block {
+ display: block !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xxl-table {
+ display: table !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xxl-table-row {
+ display: table-row !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xxl-table-cell {
+ display: table-cell !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xxl-flex {
+ display: -webkit-box !important;
+ display: -ms-flexbox !important;
+ display: flex !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-xxl-inline-flex {
+ display: -webkit-inline-box !important;
+ display: -ms-inline-flexbox !important;
+ display: inline-flex !important;
+ }
+}
+
+@media print {
+ /* line 24, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-print-none {
+ display: none !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-print-inline {
+ display: inline !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-print-inline-block {
+ display: inline-block !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-print-block {
+ display: block !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-print-table {
+ display: table !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-print-table-row {
+ display: table-row !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-print-table-cell {
+ display: table-cell !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-print-flex {
+ display: -webkit-box !important;
+ display: -ms-flexbox !important;
+ display: flex !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_display.scss */
+ .d-print-inline-flex {
+ display: -webkit-inline-box !important;
+ display: -ms-inline-flexbox !important;
+ display: inline-flex !important;
+ }
+}
+
+/* line 3, node_modules/bootstrap/scss/utilities/_embed.scss */
+.embed-responsive {
+ position: relative;
+ display: block;
+ width: 100%;
+ padding: 0;
+ overflow: hidden;
+}
+
+/* line 10, node_modules/bootstrap/scss/utilities/_embed.scss */
+.embed-responsive::before {
+ display: block;
+ content: "";
+}
+
+/* line 15, node_modules/bootstrap/scss/utilities/_embed.scss */
+.embed-responsive .embed-responsive-item,
+.embed-responsive iframe,
+.embed-responsive embed,
+.embed-responsive object,
+.embed-responsive video {
+ position: absolute;
+ top: 0;
+ bottom: 0;
+ left: 0;
+ width: 100%;
+ height: 100%;
+ border: 0;
+}
+
+/* line 35, node_modules/bootstrap/scss/utilities/_embed.scss */
+.embed-responsive-21by9::before {
+ padding-top: 42.85714%;
+}
+
+/* line 35, node_modules/bootstrap/scss/utilities/_embed.scss */
+.embed-responsive-16by9::before {
+ padding-top: 56.25%;
+}
+
+/* line 35, node_modules/bootstrap/scss/utilities/_embed.scss */
+.embed-responsive-4by3::before {
+ padding-top: 75%;
+}
+
+/* line 35, node_modules/bootstrap/scss/utilities/_embed.scss */
+.embed-responsive-1by1::before {
+ padding-top: 100%;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_flex.scss */
+.flex-row {
+ -webkit-box-orient: horizontal !important;
+ -webkit-box-direction: normal !important;
+ -ms-flex-direction: row !important;
+ flex-direction: row !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_flex.scss */
+.flex-column {
+ -webkit-box-orient: vertical !important;
+ -webkit-box-direction: normal !important;
+ -ms-flex-direction: column !important;
+ flex-direction: column !important;
+}
+
+/* line 13, node_modules/bootstrap/scss/utilities/_flex.scss */
+.flex-row-reverse {
+ -webkit-box-orient: horizontal !important;
+ -webkit-box-direction: reverse !important;
+ -ms-flex-direction: row-reverse !important;
+ flex-direction: row-reverse !important;
+}
+
+/* line 14, node_modules/bootstrap/scss/utilities/_flex.scss */
+.flex-column-reverse {
+ -webkit-box-orient: vertical !important;
+ -webkit-box-direction: reverse !important;
+ -ms-flex-direction: column-reverse !important;
+ flex-direction: column-reverse !important;
+}
+
+/* line 16, node_modules/bootstrap/scss/utilities/_flex.scss */
+.flex-wrap {
+ -ms-flex-wrap: wrap !important;
+ flex-wrap: wrap !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/utilities/_flex.scss */
+.flex-nowrap {
+ -ms-flex-wrap: nowrap !important;
+ flex-wrap: nowrap !important;
+}
+
+/* line 18, node_modules/bootstrap/scss/utilities/_flex.scss */
+.flex-wrap-reverse {
+ -ms-flex-wrap: wrap-reverse !important;
+ flex-wrap: wrap-reverse !important;
+}
+
+/* line 19, node_modules/bootstrap/scss/utilities/_flex.scss */
+.flex-fill {
+ -webkit-box-flex: 1 !important;
+ -ms-flex: 1 1 auto !important;
+ flex: 1 1 auto !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_flex.scss */
+.flex-grow-0 {
+ -webkit-box-flex: 0 !important;
+ -ms-flex-positive: 0 !important;
+ flex-grow: 0 !important;
+}
+
+/* line 21, node_modules/bootstrap/scss/utilities/_flex.scss */
+.flex-grow-1 {
+ -webkit-box-flex: 1 !important;
+ -ms-flex-positive: 1 !important;
+ flex-grow: 1 !important;
+}
+
+/* line 22, node_modules/bootstrap/scss/utilities/_flex.scss */
+.flex-shrink-0 {
+ -ms-flex-negative: 0 !important;
+ flex-shrink: 0 !important;
+}
+
+/* line 23, node_modules/bootstrap/scss/utilities/_flex.scss */
+.flex-shrink-1 {
+ -ms-flex-negative: 1 !important;
+ flex-shrink: 1 !important;
+}
+
+/* line 25, node_modules/bootstrap/scss/utilities/_flex.scss */
+.justify-content-start {
+ -webkit-box-pack: start !important;
+ -ms-flex-pack: start !important;
+ justify-content: flex-start !important;
+}
+
+/* line 26, node_modules/bootstrap/scss/utilities/_flex.scss */
+.justify-content-end {
+ -webkit-box-pack: end !important;
+ -ms-flex-pack: end !important;
+ justify-content: flex-end !important;
+}
+
+/* line 27, node_modules/bootstrap/scss/utilities/_flex.scss */
+.justify-content-center {
+ -webkit-box-pack: center !important;
+ -ms-flex-pack: center !important;
+ justify-content: center !important;
+}
+
+/* line 28, node_modules/bootstrap/scss/utilities/_flex.scss */
+.justify-content-between {
+ -webkit-box-pack: justify !important;
+ -ms-flex-pack: justify !important;
+ justify-content: space-between !important;
+}
+
+/* line 29, node_modules/bootstrap/scss/utilities/_flex.scss */
+.justify-content-around {
+ -ms-flex-pack: distribute !important;
+ justify-content: space-around !important;
+}
+
+/* line 31, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-items-start {
+ -webkit-box-align: start !important;
+ -ms-flex-align: start !important;
+ align-items: flex-start !important;
+}
+
+/* line 32, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-items-end {
+ -webkit-box-align: end !important;
+ -ms-flex-align: end !important;
+ align-items: flex-end !important;
+}
+
+/* line 33, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-items-center {
+ -webkit-box-align: center !important;
+ -ms-flex-align: center !important;
+ align-items: center !important;
+}
+
+/* line 34, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-items-baseline {
+ -webkit-box-align: baseline !important;
+ -ms-flex-align: baseline !important;
+ align-items: baseline !important;
+}
+
+/* line 35, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-items-stretch {
+ -webkit-box-align: stretch !important;
+ -ms-flex-align: stretch !important;
+ align-items: stretch !important;
+}
+
+/* line 37, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-content-start {
+ -ms-flex-line-pack: start !important;
+ align-content: flex-start !important;
+}
+
+/* line 38, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-content-end {
+ -ms-flex-line-pack: end !important;
+ align-content: flex-end !important;
+}
+
+/* line 39, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-content-center {
+ -ms-flex-line-pack: center !important;
+ align-content: center !important;
+}
+
+/* line 40, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-content-between {
+ -ms-flex-line-pack: justify !important;
+ align-content: space-between !important;
+}
+
+/* line 41, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-content-around {
+ -ms-flex-line-pack: distribute !important;
+ align-content: space-around !important;
+}
+
+/* line 42, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-content-stretch {
+ -ms-flex-line-pack: stretch !important;
+ align-content: stretch !important;
+}
+
+/* line 44, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-self-auto {
+ -ms-flex-item-align: auto !important;
+ align-self: auto !important;
+}
+
+/* line 45, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-self-start {
+ -ms-flex-item-align: start !important;
+ align-self: flex-start !important;
+}
+
+/* line 46, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-self-end {
+ -ms-flex-item-align: end !important;
+ align-self: flex-end !important;
+}
+
+/* line 47, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-self-center {
+ -ms-flex-item-align: center !important;
+ align-self: center !important;
+}
+
+/* line 48, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-self-baseline {
+ -ms-flex-item-align: baseline !important;
+ align-self: baseline !important;
+}
+
+/* line 49, node_modules/bootstrap/scss/utilities/_flex.scss */
+.align-self-stretch {
+ -ms-flex-item-align: stretch !important;
+ align-self: stretch !important;
+}
+
+@media (min-width: 576px) {
+ /* line 11, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-sm-row {
+ -webkit-box-orient: horizontal !important;
+ -webkit-box-direction: normal !important;
+ -ms-flex-direction: row !important;
+ flex-direction: row !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-sm-column {
+ -webkit-box-orient: vertical !important;
+ -webkit-box-direction: normal !important;
+ -ms-flex-direction: column !important;
+ flex-direction: column !important;
+ }
+ /* line 13, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-sm-row-reverse {
+ -webkit-box-orient: horizontal !important;
+ -webkit-box-direction: reverse !important;
+ -ms-flex-direction: row-reverse !important;
+ flex-direction: row-reverse !important;
+ }
+ /* line 14, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-sm-column-reverse {
+ -webkit-box-orient: vertical !important;
+ -webkit-box-direction: reverse !important;
+ -ms-flex-direction: column-reverse !important;
+ flex-direction: column-reverse !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-sm-wrap {
+ -ms-flex-wrap: wrap !important;
+ flex-wrap: wrap !important;
+ }
+ /* line 17, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-sm-nowrap {
+ -ms-flex-wrap: nowrap !important;
+ flex-wrap: nowrap !important;
+ }
+ /* line 18, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-sm-wrap-reverse {
+ -ms-flex-wrap: wrap-reverse !important;
+ flex-wrap: wrap-reverse !important;
+ }
+ /* line 19, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-sm-fill {
+ -webkit-box-flex: 1 !important;
+ -ms-flex: 1 1 auto !important;
+ flex: 1 1 auto !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-sm-grow-0 {
+ -webkit-box-flex: 0 !important;
+ -ms-flex-positive: 0 !important;
+ flex-grow: 0 !important;
+ }
+ /* line 21, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-sm-grow-1 {
+ -webkit-box-flex: 1 !important;
+ -ms-flex-positive: 1 !important;
+ flex-grow: 1 !important;
+ }
+ /* line 22, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-sm-shrink-0 {
+ -ms-flex-negative: 0 !important;
+ flex-shrink: 0 !important;
+ }
+ /* line 23, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-sm-shrink-1 {
+ -ms-flex-negative: 1 !important;
+ flex-shrink: 1 !important;
+ }
+ /* line 25, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-sm-start {
+ -webkit-box-pack: start !important;
+ -ms-flex-pack: start !important;
+ justify-content: flex-start !important;
+ }
+ /* line 26, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-sm-end {
+ -webkit-box-pack: end !important;
+ -ms-flex-pack: end !important;
+ justify-content: flex-end !important;
+ }
+ /* line 27, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-sm-center {
+ -webkit-box-pack: center !important;
+ -ms-flex-pack: center !important;
+ justify-content: center !important;
+ }
+ /* line 28, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-sm-between {
+ -webkit-box-pack: justify !important;
+ -ms-flex-pack: justify !important;
+ justify-content: space-between !important;
+ }
+ /* line 29, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-sm-around {
+ -ms-flex-pack: distribute !important;
+ justify-content: space-around !important;
+ }
+ /* line 31, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-sm-start {
+ -webkit-box-align: start !important;
+ -ms-flex-align: start !important;
+ align-items: flex-start !important;
+ }
+ /* line 32, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-sm-end {
+ -webkit-box-align: end !important;
+ -ms-flex-align: end !important;
+ align-items: flex-end !important;
+ }
+ /* line 33, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-sm-center {
+ -webkit-box-align: center !important;
+ -ms-flex-align: center !important;
+ align-items: center !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-sm-baseline {
+ -webkit-box-align: baseline !important;
+ -ms-flex-align: baseline !important;
+ align-items: baseline !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-sm-stretch {
+ -webkit-box-align: stretch !important;
+ -ms-flex-align: stretch !important;
+ align-items: stretch !important;
+ }
+ /* line 37, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-sm-start {
+ -ms-flex-line-pack: start !important;
+ align-content: flex-start !important;
+ }
+ /* line 38, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-sm-end {
+ -ms-flex-line-pack: end !important;
+ align-content: flex-end !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-sm-center {
+ -ms-flex-line-pack: center !important;
+ align-content: center !important;
+ }
+ /* line 40, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-sm-between {
+ -ms-flex-line-pack: justify !important;
+ align-content: space-between !important;
+ }
+ /* line 41, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-sm-around {
+ -ms-flex-line-pack: distribute !important;
+ align-content: space-around !important;
+ }
+ /* line 42, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-sm-stretch {
+ -ms-flex-line-pack: stretch !important;
+ align-content: stretch !important;
+ }
+ /* line 44, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-sm-auto {
+ -ms-flex-item-align: auto !important;
+ align-self: auto !important;
+ }
+ /* line 45, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-sm-start {
+ -ms-flex-item-align: start !important;
+ align-self: flex-start !important;
+ }
+ /* line 46, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-sm-end {
+ -ms-flex-item-align: end !important;
+ align-self: flex-end !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-sm-center {
+ -ms-flex-item-align: center !important;
+ align-self: center !important;
+ }
+ /* line 48, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-sm-baseline {
+ -ms-flex-item-align: baseline !important;
+ align-self: baseline !important;
+ }
+ /* line 49, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-sm-stretch {
+ -ms-flex-item-align: stretch !important;
+ align-self: stretch !important;
+ }
+}
+
+@media (min-width: 768px) {
+ /* line 11, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-md-row {
+ -webkit-box-orient: horizontal !important;
+ -webkit-box-direction: normal !important;
+ -ms-flex-direction: row !important;
+ flex-direction: row !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-md-column {
+ -webkit-box-orient: vertical !important;
+ -webkit-box-direction: normal !important;
+ -ms-flex-direction: column !important;
+ flex-direction: column !important;
+ }
+ /* line 13, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-md-row-reverse {
+ -webkit-box-orient: horizontal !important;
+ -webkit-box-direction: reverse !important;
+ -ms-flex-direction: row-reverse !important;
+ flex-direction: row-reverse !important;
+ }
+ /* line 14, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-md-column-reverse {
+ -webkit-box-orient: vertical !important;
+ -webkit-box-direction: reverse !important;
+ -ms-flex-direction: column-reverse !important;
+ flex-direction: column-reverse !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-md-wrap {
+ -ms-flex-wrap: wrap !important;
+ flex-wrap: wrap !important;
+ }
+ /* line 17, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-md-nowrap {
+ -ms-flex-wrap: nowrap !important;
+ flex-wrap: nowrap !important;
+ }
+ /* line 18, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-md-wrap-reverse {
+ -ms-flex-wrap: wrap-reverse !important;
+ flex-wrap: wrap-reverse !important;
+ }
+ /* line 19, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-md-fill {
+ -webkit-box-flex: 1 !important;
+ -ms-flex: 1 1 auto !important;
+ flex: 1 1 auto !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-md-grow-0 {
+ -webkit-box-flex: 0 !important;
+ -ms-flex-positive: 0 !important;
+ flex-grow: 0 !important;
+ }
+ /* line 21, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-md-grow-1 {
+ -webkit-box-flex: 1 !important;
+ -ms-flex-positive: 1 !important;
+ flex-grow: 1 !important;
+ }
+ /* line 22, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-md-shrink-0 {
+ -ms-flex-negative: 0 !important;
+ flex-shrink: 0 !important;
+ }
+ /* line 23, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-md-shrink-1 {
+ -ms-flex-negative: 1 !important;
+ flex-shrink: 1 !important;
+ }
+ /* line 25, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-md-start {
+ -webkit-box-pack: start !important;
+ -ms-flex-pack: start !important;
+ justify-content: flex-start !important;
+ }
+ /* line 26, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-md-end {
+ -webkit-box-pack: end !important;
+ -ms-flex-pack: end !important;
+ justify-content: flex-end !important;
+ }
+ /* line 27, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-md-center {
+ -webkit-box-pack: center !important;
+ -ms-flex-pack: center !important;
+ justify-content: center !important;
+ }
+ /* line 28, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-md-between {
+ -webkit-box-pack: justify !important;
+ -ms-flex-pack: justify !important;
+ justify-content: space-between !important;
+ }
+ /* line 29, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-md-around {
+ -ms-flex-pack: distribute !important;
+ justify-content: space-around !important;
+ }
+ /* line 31, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-md-start {
+ -webkit-box-align: start !important;
+ -ms-flex-align: start !important;
+ align-items: flex-start !important;
+ }
+ /* line 32, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-md-end {
+ -webkit-box-align: end !important;
+ -ms-flex-align: end !important;
+ align-items: flex-end !important;
+ }
+ /* line 33, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-md-center {
+ -webkit-box-align: center !important;
+ -ms-flex-align: center !important;
+ align-items: center !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-md-baseline {
+ -webkit-box-align: baseline !important;
+ -ms-flex-align: baseline !important;
+ align-items: baseline !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-md-stretch {
+ -webkit-box-align: stretch !important;
+ -ms-flex-align: stretch !important;
+ align-items: stretch !important;
+ }
+ /* line 37, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-md-start {
+ -ms-flex-line-pack: start !important;
+ align-content: flex-start !important;
+ }
+ /* line 38, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-md-end {
+ -ms-flex-line-pack: end !important;
+ align-content: flex-end !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-md-center {
+ -ms-flex-line-pack: center !important;
+ align-content: center !important;
+ }
+ /* line 40, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-md-between {
+ -ms-flex-line-pack: justify !important;
+ align-content: space-between !important;
+ }
+ /* line 41, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-md-around {
+ -ms-flex-line-pack: distribute !important;
+ align-content: space-around !important;
+ }
+ /* line 42, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-md-stretch {
+ -ms-flex-line-pack: stretch !important;
+ align-content: stretch !important;
+ }
+ /* line 44, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-md-auto {
+ -ms-flex-item-align: auto !important;
+ align-self: auto !important;
+ }
+ /* line 45, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-md-start {
+ -ms-flex-item-align: start !important;
+ align-self: flex-start !important;
+ }
+ /* line 46, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-md-end {
+ -ms-flex-item-align: end !important;
+ align-self: flex-end !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-md-center {
+ -ms-flex-item-align: center !important;
+ align-self: center !important;
+ }
+ /* line 48, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-md-baseline {
+ -ms-flex-item-align: baseline !important;
+ align-self: baseline !important;
+ }
+ /* line 49, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-md-stretch {
+ -ms-flex-item-align: stretch !important;
+ align-self: stretch !important;
+ }
+}
+
+@media (min-width: 1024px) {
+ /* line 11, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-lg-row {
+ -webkit-box-orient: horizontal !important;
+ -webkit-box-direction: normal !important;
+ -ms-flex-direction: row !important;
+ flex-direction: row !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-lg-column {
+ -webkit-box-orient: vertical !important;
+ -webkit-box-direction: normal !important;
+ -ms-flex-direction: column !important;
+ flex-direction: column !important;
+ }
+ /* line 13, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-lg-row-reverse {
+ -webkit-box-orient: horizontal !important;
+ -webkit-box-direction: reverse !important;
+ -ms-flex-direction: row-reverse !important;
+ flex-direction: row-reverse !important;
+ }
+ /* line 14, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-lg-column-reverse {
+ -webkit-box-orient: vertical !important;
+ -webkit-box-direction: reverse !important;
+ -ms-flex-direction: column-reverse !important;
+ flex-direction: column-reverse !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-lg-wrap {
+ -ms-flex-wrap: wrap !important;
+ flex-wrap: wrap !important;
+ }
+ /* line 17, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-lg-nowrap {
+ -ms-flex-wrap: nowrap !important;
+ flex-wrap: nowrap !important;
+ }
+ /* line 18, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-lg-wrap-reverse {
+ -ms-flex-wrap: wrap-reverse !important;
+ flex-wrap: wrap-reverse !important;
+ }
+ /* line 19, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-lg-fill {
+ -webkit-box-flex: 1 !important;
+ -ms-flex: 1 1 auto !important;
+ flex: 1 1 auto !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-lg-grow-0 {
+ -webkit-box-flex: 0 !important;
+ -ms-flex-positive: 0 !important;
+ flex-grow: 0 !important;
+ }
+ /* line 21, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-lg-grow-1 {
+ -webkit-box-flex: 1 !important;
+ -ms-flex-positive: 1 !important;
+ flex-grow: 1 !important;
+ }
+ /* line 22, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-lg-shrink-0 {
+ -ms-flex-negative: 0 !important;
+ flex-shrink: 0 !important;
+ }
+ /* line 23, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-lg-shrink-1 {
+ -ms-flex-negative: 1 !important;
+ flex-shrink: 1 !important;
+ }
+ /* line 25, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-lg-start {
+ -webkit-box-pack: start !important;
+ -ms-flex-pack: start !important;
+ justify-content: flex-start !important;
+ }
+ /* line 26, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-lg-end {
+ -webkit-box-pack: end !important;
+ -ms-flex-pack: end !important;
+ justify-content: flex-end !important;
+ }
+ /* line 27, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-lg-center {
+ -webkit-box-pack: center !important;
+ -ms-flex-pack: center !important;
+ justify-content: center !important;
+ }
+ /* line 28, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-lg-between {
+ -webkit-box-pack: justify !important;
+ -ms-flex-pack: justify !important;
+ justify-content: space-between !important;
+ }
+ /* line 29, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-lg-around {
+ -ms-flex-pack: distribute !important;
+ justify-content: space-around !important;
+ }
+ /* line 31, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-lg-start {
+ -webkit-box-align: start !important;
+ -ms-flex-align: start !important;
+ align-items: flex-start !important;
+ }
+ /* line 32, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-lg-end {
+ -webkit-box-align: end !important;
+ -ms-flex-align: end !important;
+ align-items: flex-end !important;
+ }
+ /* line 33, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-lg-center {
+ -webkit-box-align: center !important;
+ -ms-flex-align: center !important;
+ align-items: center !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-lg-baseline {
+ -webkit-box-align: baseline !important;
+ -ms-flex-align: baseline !important;
+ align-items: baseline !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-lg-stretch {
+ -webkit-box-align: stretch !important;
+ -ms-flex-align: stretch !important;
+ align-items: stretch !important;
+ }
+ /* line 37, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-lg-start {
+ -ms-flex-line-pack: start !important;
+ align-content: flex-start !important;
+ }
+ /* line 38, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-lg-end {
+ -ms-flex-line-pack: end !important;
+ align-content: flex-end !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-lg-center {
+ -ms-flex-line-pack: center !important;
+ align-content: center !important;
+ }
+ /* line 40, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-lg-between {
+ -ms-flex-line-pack: justify !important;
+ align-content: space-between !important;
+ }
+ /* line 41, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-lg-around {
+ -ms-flex-line-pack: distribute !important;
+ align-content: space-around !important;
+ }
+ /* line 42, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-lg-stretch {
+ -ms-flex-line-pack: stretch !important;
+ align-content: stretch !important;
+ }
+ /* line 44, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-lg-auto {
+ -ms-flex-item-align: auto !important;
+ align-self: auto !important;
+ }
+ /* line 45, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-lg-start {
+ -ms-flex-item-align: start !important;
+ align-self: flex-start !important;
+ }
+ /* line 46, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-lg-end {
+ -ms-flex-item-align: end !important;
+ align-self: flex-end !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-lg-center {
+ -ms-flex-item-align: center !important;
+ align-self: center !important;
+ }
+ /* line 48, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-lg-baseline {
+ -ms-flex-item-align: baseline !important;
+ align-self: baseline !important;
+ }
+ /* line 49, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-lg-stretch {
+ -ms-flex-item-align: stretch !important;
+ align-self: stretch !important;
+ }
+}
+
+@media (min-width: 1280px) {
+ /* line 11, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xl-row {
+ -webkit-box-orient: horizontal !important;
+ -webkit-box-direction: normal !important;
+ -ms-flex-direction: row !important;
+ flex-direction: row !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xl-column {
+ -webkit-box-orient: vertical !important;
+ -webkit-box-direction: normal !important;
+ -ms-flex-direction: column !important;
+ flex-direction: column !important;
+ }
+ /* line 13, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xl-row-reverse {
+ -webkit-box-orient: horizontal !important;
+ -webkit-box-direction: reverse !important;
+ -ms-flex-direction: row-reverse !important;
+ flex-direction: row-reverse !important;
+ }
+ /* line 14, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xl-column-reverse {
+ -webkit-box-orient: vertical !important;
+ -webkit-box-direction: reverse !important;
+ -ms-flex-direction: column-reverse !important;
+ flex-direction: column-reverse !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xl-wrap {
+ -ms-flex-wrap: wrap !important;
+ flex-wrap: wrap !important;
+ }
+ /* line 17, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xl-nowrap {
+ -ms-flex-wrap: nowrap !important;
+ flex-wrap: nowrap !important;
+ }
+ /* line 18, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xl-wrap-reverse {
+ -ms-flex-wrap: wrap-reverse !important;
+ flex-wrap: wrap-reverse !important;
+ }
+ /* line 19, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xl-fill {
+ -webkit-box-flex: 1 !important;
+ -ms-flex: 1 1 auto !important;
+ flex: 1 1 auto !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xl-grow-0 {
+ -webkit-box-flex: 0 !important;
+ -ms-flex-positive: 0 !important;
+ flex-grow: 0 !important;
+ }
+ /* line 21, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xl-grow-1 {
+ -webkit-box-flex: 1 !important;
+ -ms-flex-positive: 1 !important;
+ flex-grow: 1 !important;
+ }
+ /* line 22, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xl-shrink-0 {
+ -ms-flex-negative: 0 !important;
+ flex-shrink: 0 !important;
+ }
+ /* line 23, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xl-shrink-1 {
+ -ms-flex-negative: 1 !important;
+ flex-shrink: 1 !important;
+ }
+ /* line 25, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-xl-start {
+ -webkit-box-pack: start !important;
+ -ms-flex-pack: start !important;
+ justify-content: flex-start !important;
+ }
+ /* line 26, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-xl-end {
+ -webkit-box-pack: end !important;
+ -ms-flex-pack: end !important;
+ justify-content: flex-end !important;
+ }
+ /* line 27, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-xl-center {
+ -webkit-box-pack: center !important;
+ -ms-flex-pack: center !important;
+ justify-content: center !important;
+ }
+ /* line 28, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-xl-between {
+ -webkit-box-pack: justify !important;
+ -ms-flex-pack: justify !important;
+ justify-content: space-between !important;
+ }
+ /* line 29, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-xl-around {
+ -ms-flex-pack: distribute !important;
+ justify-content: space-around !important;
+ }
+ /* line 31, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-xl-start {
+ -webkit-box-align: start !important;
+ -ms-flex-align: start !important;
+ align-items: flex-start !important;
+ }
+ /* line 32, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-xl-end {
+ -webkit-box-align: end !important;
+ -ms-flex-align: end !important;
+ align-items: flex-end !important;
+ }
+ /* line 33, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-xl-center {
+ -webkit-box-align: center !important;
+ -ms-flex-align: center !important;
+ align-items: center !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-xl-baseline {
+ -webkit-box-align: baseline !important;
+ -ms-flex-align: baseline !important;
+ align-items: baseline !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-xl-stretch {
+ -webkit-box-align: stretch !important;
+ -ms-flex-align: stretch !important;
+ align-items: stretch !important;
+ }
+ /* line 37, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-xl-start {
+ -ms-flex-line-pack: start !important;
+ align-content: flex-start !important;
+ }
+ /* line 38, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-xl-end {
+ -ms-flex-line-pack: end !important;
+ align-content: flex-end !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-xl-center {
+ -ms-flex-line-pack: center !important;
+ align-content: center !important;
+ }
+ /* line 40, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-xl-between {
+ -ms-flex-line-pack: justify !important;
+ align-content: space-between !important;
+ }
+ /* line 41, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-xl-around {
+ -ms-flex-line-pack: distribute !important;
+ align-content: space-around !important;
+ }
+ /* line 42, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-xl-stretch {
+ -ms-flex-line-pack: stretch !important;
+ align-content: stretch !important;
+ }
+ /* line 44, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-xl-auto {
+ -ms-flex-item-align: auto !important;
+ align-self: auto !important;
+ }
+ /* line 45, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-xl-start {
+ -ms-flex-item-align: start !important;
+ align-self: flex-start !important;
+ }
+ /* line 46, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-xl-end {
+ -ms-flex-item-align: end !important;
+ align-self: flex-end !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-xl-center {
+ -ms-flex-item-align: center !important;
+ align-self: center !important;
+ }
+ /* line 48, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-xl-baseline {
+ -ms-flex-item-align: baseline !important;
+ align-self: baseline !important;
+ }
+ /* line 49, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-xl-stretch {
+ -ms-flex-item-align: stretch !important;
+ align-self: stretch !important;
+ }
+}
+
+@media (min-width: 1440px) {
+ /* line 11, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xxl-row {
+ -webkit-box-orient: horizontal !important;
+ -webkit-box-direction: normal !important;
+ -ms-flex-direction: row !important;
+ flex-direction: row !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xxl-column {
+ -webkit-box-orient: vertical !important;
+ -webkit-box-direction: normal !important;
+ -ms-flex-direction: column !important;
+ flex-direction: column !important;
+ }
+ /* line 13, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xxl-row-reverse {
+ -webkit-box-orient: horizontal !important;
+ -webkit-box-direction: reverse !important;
+ -ms-flex-direction: row-reverse !important;
+ flex-direction: row-reverse !important;
+ }
+ /* line 14, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xxl-column-reverse {
+ -webkit-box-orient: vertical !important;
+ -webkit-box-direction: reverse !important;
+ -ms-flex-direction: column-reverse !important;
+ flex-direction: column-reverse !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xxl-wrap {
+ -ms-flex-wrap: wrap !important;
+ flex-wrap: wrap !important;
+ }
+ /* line 17, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xxl-nowrap {
+ -ms-flex-wrap: nowrap !important;
+ flex-wrap: nowrap !important;
+ }
+ /* line 18, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xxl-wrap-reverse {
+ -ms-flex-wrap: wrap-reverse !important;
+ flex-wrap: wrap-reverse !important;
+ }
+ /* line 19, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xxl-fill {
+ -webkit-box-flex: 1 !important;
+ -ms-flex: 1 1 auto !important;
+ flex: 1 1 auto !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xxl-grow-0 {
+ -webkit-box-flex: 0 !important;
+ -ms-flex-positive: 0 !important;
+ flex-grow: 0 !important;
+ }
+ /* line 21, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xxl-grow-1 {
+ -webkit-box-flex: 1 !important;
+ -ms-flex-positive: 1 !important;
+ flex-grow: 1 !important;
+ }
+ /* line 22, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xxl-shrink-0 {
+ -ms-flex-negative: 0 !important;
+ flex-shrink: 0 !important;
+ }
+ /* line 23, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .flex-xxl-shrink-1 {
+ -ms-flex-negative: 1 !important;
+ flex-shrink: 1 !important;
+ }
+ /* line 25, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-xxl-start {
+ -webkit-box-pack: start !important;
+ -ms-flex-pack: start !important;
+ justify-content: flex-start !important;
+ }
+ /* line 26, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-xxl-end {
+ -webkit-box-pack: end !important;
+ -ms-flex-pack: end !important;
+ justify-content: flex-end !important;
+ }
+ /* line 27, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-xxl-center {
+ -webkit-box-pack: center !important;
+ -ms-flex-pack: center !important;
+ justify-content: center !important;
+ }
+ /* line 28, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-xxl-between {
+ -webkit-box-pack: justify !important;
+ -ms-flex-pack: justify !important;
+ justify-content: space-between !important;
+ }
+ /* line 29, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .justify-content-xxl-around {
+ -ms-flex-pack: distribute !important;
+ justify-content: space-around !important;
+ }
+ /* line 31, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-xxl-start {
+ -webkit-box-align: start !important;
+ -ms-flex-align: start !important;
+ align-items: flex-start !important;
+ }
+ /* line 32, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-xxl-end {
+ -webkit-box-align: end !important;
+ -ms-flex-align: end !important;
+ align-items: flex-end !important;
+ }
+ /* line 33, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-xxl-center {
+ -webkit-box-align: center !important;
+ -ms-flex-align: center !important;
+ align-items: center !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-xxl-baseline {
+ -webkit-box-align: baseline !important;
+ -ms-flex-align: baseline !important;
+ align-items: baseline !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-items-xxl-stretch {
+ -webkit-box-align: stretch !important;
+ -ms-flex-align: stretch !important;
+ align-items: stretch !important;
+ }
+ /* line 37, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-xxl-start {
+ -ms-flex-line-pack: start !important;
+ align-content: flex-start !important;
+ }
+ /* line 38, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-xxl-end {
+ -ms-flex-line-pack: end !important;
+ align-content: flex-end !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-xxl-center {
+ -ms-flex-line-pack: center !important;
+ align-content: center !important;
+ }
+ /* line 40, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-xxl-between {
+ -ms-flex-line-pack: justify !important;
+ align-content: space-between !important;
+ }
+ /* line 41, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-xxl-around {
+ -ms-flex-line-pack: distribute !important;
+ align-content: space-around !important;
+ }
+ /* line 42, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-content-xxl-stretch {
+ -ms-flex-line-pack: stretch !important;
+ align-content: stretch !important;
+ }
+ /* line 44, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-xxl-auto {
+ -ms-flex-item-align: auto !important;
+ align-self: auto !important;
+ }
+ /* line 45, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-xxl-start {
+ -ms-flex-item-align: start !important;
+ align-self: flex-start !important;
+ }
+ /* line 46, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-xxl-end {
+ -ms-flex-item-align: end !important;
+ align-self: flex-end !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-xxl-center {
+ -ms-flex-item-align: center !important;
+ align-self: center !important;
+ }
+ /* line 48, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-xxl-baseline {
+ -ms-flex-item-align: baseline !important;
+ align-self: baseline !important;
+ }
+ /* line 49, node_modules/bootstrap/scss/utilities/_flex.scss */
+ .align-self-xxl-stretch {
+ -ms-flex-item-align: stretch !important;
+ align-self: stretch !important;
+ }
+}
+
+/* line 7, node_modules/bootstrap/scss/utilities/_float.scss */
+.float-left {
+ float: left !important;
+}
+
+/* line 8, node_modules/bootstrap/scss/utilities/_float.scss */
+.float-right {
+ float: right !important;
+}
+
+/* line 9, node_modules/bootstrap/scss/utilities/_float.scss */
+.float-none {
+ float: none !important;
+}
+
+@media (min-width: 576px) {
+ /* line 7, node_modules/bootstrap/scss/utilities/_float.scss */
+ .float-sm-left {
+ float: left !important;
+ }
+ /* line 8, node_modules/bootstrap/scss/utilities/_float.scss */
+ .float-sm-right {
+ float: right !important;
+ }
+ /* line 9, node_modules/bootstrap/scss/utilities/_float.scss */
+ .float-sm-none {
+ float: none !important;
+ }
+}
+
+@media (min-width: 768px) {
+ /* line 7, node_modules/bootstrap/scss/utilities/_float.scss */
+ .float-md-left {
+ float: left !important;
+ }
+ /* line 8, node_modules/bootstrap/scss/utilities/_float.scss */
+ .float-md-right {
+ float: right !important;
+ }
+ /* line 9, node_modules/bootstrap/scss/utilities/_float.scss */
+ .float-md-none {
+ float: none !important;
+ }
+}
+
+@media (min-width: 1024px) {
+ /* line 7, node_modules/bootstrap/scss/utilities/_float.scss */
+ .float-lg-left {
+ float: left !important;
+ }
+ /* line 8, node_modules/bootstrap/scss/utilities/_float.scss */
+ .float-lg-right {
+ float: right !important;
+ }
+ /* line 9, node_modules/bootstrap/scss/utilities/_float.scss */
+ .float-lg-none {
+ float: none !important;
+ }
+}
+
+@media (min-width: 1280px) {
+ /* line 7, node_modules/bootstrap/scss/utilities/_float.scss */
+ .float-xl-left {
+ float: left !important;
+ }
+ /* line 8, node_modules/bootstrap/scss/utilities/_float.scss */
+ .float-xl-right {
+ float: right !important;
+ }
+ /* line 9, node_modules/bootstrap/scss/utilities/_float.scss */
+ .float-xl-none {
+ float: none !important;
+ }
+}
+
+@media (min-width: 1440px) {
+ /* line 7, node_modules/bootstrap/scss/utilities/_float.scss */
+ .float-xxl-left {
+ float: left !important;
+ }
+ /* line 8, node_modules/bootstrap/scss/utilities/_float.scss */
+ .float-xxl-right {
+ float: right !important;
+ }
+ /* line 9, node_modules/bootstrap/scss/utilities/_float.scss */
+ .float-xxl-none {
+ float: none !important;
+ }
+}
+
+/* line 4, node_modules/bootstrap/scss/utilities/_interactions.scss */
+.user-select-all {
+ -webkit-user-select: all !important;
+ -moz-user-select: all !important;
+ -ms-user-select: all !important;
+ user-select: all !important;
+}
+
+/* line 4, node_modules/bootstrap/scss/utilities/_interactions.scss */
+.user-select-auto {
+ -webkit-user-select: auto !important;
+ -moz-user-select: auto !important;
+ -ms-user-select: auto !important;
+ user-select: auto !important;
+}
+
+/* line 4, node_modules/bootstrap/scss/utilities/_interactions.scss */
+.user-select-none {
+ -webkit-user-select: none !important;
+ -moz-user-select: none !important;
+ -ms-user-select: none !important;
+ user-select: none !important;
+}
+
+/* line 4, node_modules/bootstrap/scss/utilities/_overflow.scss */
+.overflow-auto {
+ overflow: auto !important;
+}
+
+/* line 4, node_modules/bootstrap/scss/utilities/_overflow.scss */
+.overflow-hidden {
+ overflow: hidden !important;
+}
+
+/* line 5, node_modules/bootstrap/scss/utilities/_position.scss */
+.position-static {
+ position: static !important;
+}
+
+/* line 5, node_modules/bootstrap/scss/utilities/_position.scss */
+.position-relative {
+ position: relative !important;
+}
+
+/* line 5, node_modules/bootstrap/scss/utilities/_position.scss */
+.position-absolute {
+ position: absolute !important;
+}
+
+/* line 5, node_modules/bootstrap/scss/utilities/_position.scss */
+.position-fixed {
+ position: fixed !important;
+}
+
+/* line 5, node_modules/bootstrap/scss/utilities/_position.scss */
+.position-sticky {
+ position: sticky !important;
+}
+
+/* line 10, node_modules/bootstrap/scss/utilities/_position.scss */
+.fixed-top {
+ position: fixed;
+ top: 0;
+ right: 0;
+ left: 0;
+ z-index: 1030;
+}
+
+/* line 18, node_modules/bootstrap/scss/utilities/_position.scss */
+.fixed-bottom {
+ position: fixed;
+ right: 0;
+ bottom: 0;
+ left: 0;
+ z-index: 1030;
+}
+
+@supports (position: sticky) {
+ /* line 26, node_modules/bootstrap/scss/utilities/_position.scss */
+ .sticky-top {
+ position: sticky;
+ top: 0;
+ z-index: 1020;
+ }
+}
+
+/* line 5, node_modules/bootstrap/scss/utilities/_screenreaders.scss */
+.sr-only {
+ position: absolute;
+ width: 1px;
+ height: 1px;
+ padding: 0;
+ margin: -1px;
+ overflow: hidden;
+ clip: rect(0, 0, 0, 0);
+ white-space: nowrap;
+ border: 0;
+}
+
+/* line 25, node_modules/bootstrap/scss/mixins/_screen-reader.scss */
+.sr-only-focusable:active, .sr-only-focusable:focus {
+ position: static;
+ width: auto;
+ height: auto;
+ overflow: visible;
+ clip: auto;
+ white-space: normal;
+}
+
+/* line 3, node_modules/bootstrap/scss/utilities/_shadows.scss */
+.shadow-sm {
+ -webkit-box-shadow: 0 0.125rem 0.25rem rgba(0, 0, 0, 0.075) !important;
+ box-shadow: 0 0.125rem 0.25rem rgba(0, 0, 0, 0.075) !important;
+}
+
+/* line 4, node_modules/bootstrap/scss/utilities/_shadows.scss */
+.shadow {
+ -webkit-box-shadow: 0 0.5rem 1rem rgba(0, 0, 0, 0.15) !important;
+ box-shadow: 0 0.5rem 1rem rgba(0, 0, 0, 0.15) !important;
+}
+
+/* line 5, node_modules/bootstrap/scss/utilities/_shadows.scss */
+.shadow-lg {
+ -webkit-box-shadow: 0 1rem 3rem rgba(0, 0, 0, 0.175) !important;
+ box-shadow: 0 1rem 3rem rgba(0, 0, 0, 0.175) !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/utilities/_shadows.scss */
+.shadow-none {
+ -webkit-box-shadow: none !important;
+ box-shadow: none !important;
+}
+
+/* line 7, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.w-25 {
+ width: 25% !important;
+}
+
+/* line 7, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.w-50 {
+ width: 50% !important;
+}
+
+/* line 7, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.w-75 {
+ width: 75% !important;
+}
+
+/* line 7, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.w-100 {
+ width: 100% !important;
+}
+
+/* line 7, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.w-auto {
+ width: auto !important;
+}
+
+/* line 7, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.h-25 {
+ height: 25% !important;
+}
+
+/* line 7, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.h-50 {
+ height: 50% !important;
+}
+
+/* line 7, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.h-75 {
+ height: 75% !important;
+}
+
+/* line 7, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.h-100 {
+ height: 100% !important;
+}
+
+/* line 7, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.h-auto {
+ height: auto !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.mw-100 {
+ max-width: 100% !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.mh-100 {
+ max-height: 100% !important;
+}
+
+/* line 16, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.min-vw-100 {
+ min-width: 100vw !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.min-vh-100 {
+ min-height: 100vh !important;
+}
+
+/* line 19, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.vw-100 {
+ width: 100vw !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_sizing.scss */
+.vh-100 {
+ height: 100vh !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.m-0 {
+ margin: 0 !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mt-0,
+.my-0 {
+ margin-top: 0 !important;
+}
+
+/* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mr-0,
+.mx-0 {
+ margin-right: 0 !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mb-0,
+.my-0 {
+ margin-bottom: 0 !important;
+}
+
+/* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.ml-0,
+.mx-0 {
+ margin-left: 0 !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.m-1 {
+ margin: 0.25rem !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mt-1,
+.my-1 {
+ margin-top: 0.25rem !important;
+}
+
+/* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mr-1,
+.mx-1 {
+ margin-right: 0.25rem !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mb-1,
+.my-1 {
+ margin-bottom: 0.25rem !important;
+}
+
+/* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.ml-1,
+.mx-1 {
+ margin-left: 0.25rem !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.m-2 {
+ margin: 0.5rem !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mt-2,
+.my-2 {
+ margin-top: 0.5rem !important;
+}
+
+/* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mr-2,
+.mx-2 {
+ margin-right: 0.5rem !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mb-2,
+.my-2 {
+ margin-bottom: 0.5rem !important;
+}
+
+/* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.ml-2,
+.mx-2 {
+ margin-left: 0.5rem !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.m-3 {
+ margin: 1rem !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mt-3,
+.my-3 {
+ margin-top: 1rem !important;
+}
+
+/* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mr-3,
+.mx-3 {
+ margin-right: 1rem !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mb-3,
+.my-3 {
+ margin-bottom: 1rem !important;
+}
+
+/* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.ml-3,
+.mx-3 {
+ margin-left: 1rem !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.m-4 {
+ margin: 1.5rem !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mt-4,
+.my-4 {
+ margin-top: 1.5rem !important;
+}
+
+/* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mr-4,
+.mx-4 {
+ margin-right: 1.5rem !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mb-4,
+.my-4 {
+ margin-bottom: 1.5rem !important;
+}
+
+/* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.ml-4,
+.mx-4 {
+ margin-left: 1.5rem !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.m-5 {
+ margin: 3rem !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mt-5,
+.my-5 {
+ margin-top: 3rem !important;
+}
+
+/* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mr-5,
+.mx-5 {
+ margin-right: 3rem !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mb-5,
+.my-5 {
+ margin-bottom: 3rem !important;
+}
+
+/* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.ml-5,
+.mx-5 {
+ margin-left: 3rem !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.p-0 {
+ padding: 0 !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pt-0,
+.py-0 {
+ padding-top: 0 !important;
+}
+
+/* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pr-0,
+.px-0 {
+ padding-right: 0 !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pb-0,
+.py-0 {
+ padding-bottom: 0 !important;
+}
+
+/* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pl-0,
+.px-0 {
+ padding-left: 0 !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.p-1 {
+ padding: 0.25rem !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pt-1,
+.py-1 {
+ padding-top: 0.25rem !important;
+}
+
+/* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pr-1,
+.px-1 {
+ padding-right: 0.25rem !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pb-1,
+.py-1 {
+ padding-bottom: 0.25rem !important;
+}
+
+/* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pl-1,
+.px-1 {
+ padding-left: 0.25rem !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.p-2 {
+ padding: 0.5rem !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pt-2,
+.py-2 {
+ padding-top: 0.5rem !important;
+}
+
+/* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pr-2,
+.px-2 {
+ padding-right: 0.5rem !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pb-2,
+.py-2 {
+ padding-bottom: 0.5rem !important;
+}
+
+/* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pl-2,
+.px-2 {
+ padding-left: 0.5rem !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.p-3 {
+ padding: 1rem !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pt-3,
+.py-3 {
+ padding-top: 1rem !important;
+}
+
+/* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pr-3,
+.px-3 {
+ padding-right: 1rem !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pb-3,
+.py-3 {
+ padding-bottom: 1rem !important;
+}
+
+/* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pl-3,
+.px-3 {
+ padding-left: 1rem !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.p-4 {
+ padding: 1.5rem !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pt-4,
+.py-4 {
+ padding-top: 1.5rem !important;
+}
+
+/* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pr-4,
+.px-4 {
+ padding-right: 1.5rem !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pb-4,
+.py-4 {
+ padding-bottom: 1.5rem !important;
+}
+
+/* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pl-4,
+.px-4 {
+ padding-left: 1.5rem !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.p-5 {
+ padding: 3rem !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pt-5,
+.py-5 {
+ padding-top: 3rem !important;
+}
+
+/* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pr-5,
+.px-5 {
+ padding-right: 3rem !important;
+}
+
+/* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pb-5,
+.py-5 {
+ padding-bottom: 3rem !important;
+}
+
+/* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.pl-5,
+.px-5 {
+ padding-left: 3rem !important;
+}
+
+/* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.m-n1 {
+ margin: -0.25rem !important;
+}
+
+/* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mt-n1,
+.my-n1 {
+ margin-top: -0.25rem !important;
+}
+
+/* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mr-n1,
+.mx-n1 {
+ margin-right: -0.25rem !important;
+}
+
+/* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mb-n1,
+.my-n1 {
+ margin-bottom: -0.25rem !important;
+}
+
+/* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.ml-n1,
+.mx-n1 {
+ margin-left: -0.25rem !important;
+}
+
+/* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.m-n2 {
+ margin: -0.5rem !important;
+}
+
+/* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mt-n2,
+.my-n2 {
+ margin-top: -0.5rem !important;
+}
+
+/* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mr-n2,
+.mx-n2 {
+ margin-right: -0.5rem !important;
+}
+
+/* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mb-n2,
+.my-n2 {
+ margin-bottom: -0.5rem !important;
+}
+
+/* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.ml-n2,
+.mx-n2 {
+ margin-left: -0.5rem !important;
+}
+
+/* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.m-n3 {
+ margin: -1rem !important;
+}
+
+/* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mt-n3,
+.my-n3 {
+ margin-top: -1rem !important;
+}
+
+/* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mr-n3,
+.mx-n3 {
+ margin-right: -1rem !important;
+}
+
+/* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mb-n3,
+.my-n3 {
+ margin-bottom: -1rem !important;
+}
+
+/* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.ml-n3,
+.mx-n3 {
+ margin-left: -1rem !important;
+}
+
+/* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.m-n4 {
+ margin: -1.5rem !important;
+}
+
+/* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mt-n4,
+.my-n4 {
+ margin-top: -1.5rem !important;
+}
+
+/* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mr-n4,
+.mx-n4 {
+ margin-right: -1.5rem !important;
+}
+
+/* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mb-n4,
+.my-n4 {
+ margin-bottom: -1.5rem !important;
+}
+
+/* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.ml-n4,
+.mx-n4 {
+ margin-left: -1.5rem !important;
+}
+
+/* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.m-n5 {
+ margin: -3rem !important;
+}
+
+/* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mt-n5,
+.my-n5 {
+ margin-top: -3rem !important;
+}
+
+/* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mr-n5,
+.mx-n5 {
+ margin-right: -3rem !important;
+}
+
+/* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mb-n5,
+.my-n5 {
+ margin-bottom: -3rem !important;
+}
+
+/* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.ml-n5,
+.mx-n5 {
+ margin-left: -3rem !important;
+}
+
+/* line 55, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.m-auto {
+ margin: auto !important;
+}
+
+/* line 56, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mt-auto,
+.my-auto {
+ margin-top: auto !important;
+}
+
+/* line 60, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mr-auto,
+.mx-auto {
+ margin-right: auto !important;
+}
+
+/* line 64, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.mb-auto,
+.my-auto {
+ margin-bottom: auto !important;
+}
+
+/* line 68, node_modules/bootstrap/scss/utilities/_spacing.scss */
+.ml-auto,
+.mx-auto {
+ margin-left: auto !important;
+}
+
+@media (min-width: 576px) {
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-sm-0 {
+ margin: 0 !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-sm-0,
+ .my-sm-0 {
+ margin-top: 0 !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-sm-0,
+ .mx-sm-0 {
+ margin-right: 0 !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-sm-0,
+ .my-sm-0 {
+ margin-bottom: 0 !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-sm-0,
+ .mx-sm-0 {
+ margin-left: 0 !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-sm-1 {
+ margin: 0.25rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-sm-1,
+ .my-sm-1 {
+ margin-top: 0.25rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-sm-1,
+ .mx-sm-1 {
+ margin-right: 0.25rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-sm-1,
+ .my-sm-1 {
+ margin-bottom: 0.25rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-sm-1,
+ .mx-sm-1 {
+ margin-left: 0.25rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-sm-2 {
+ margin: 0.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-sm-2,
+ .my-sm-2 {
+ margin-top: 0.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-sm-2,
+ .mx-sm-2 {
+ margin-right: 0.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-sm-2,
+ .my-sm-2 {
+ margin-bottom: 0.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-sm-2,
+ .mx-sm-2 {
+ margin-left: 0.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-sm-3 {
+ margin: 1rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-sm-3,
+ .my-sm-3 {
+ margin-top: 1rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-sm-3,
+ .mx-sm-3 {
+ margin-right: 1rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-sm-3,
+ .my-sm-3 {
+ margin-bottom: 1rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-sm-3,
+ .mx-sm-3 {
+ margin-left: 1rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-sm-4 {
+ margin: 1.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-sm-4,
+ .my-sm-4 {
+ margin-top: 1.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-sm-4,
+ .mx-sm-4 {
+ margin-right: 1.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-sm-4,
+ .my-sm-4 {
+ margin-bottom: 1.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-sm-4,
+ .mx-sm-4 {
+ margin-left: 1.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-sm-5 {
+ margin: 3rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-sm-5,
+ .my-sm-5 {
+ margin-top: 3rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-sm-5,
+ .mx-sm-5 {
+ margin-right: 3rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-sm-5,
+ .my-sm-5 {
+ margin-bottom: 3rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-sm-5,
+ .mx-sm-5 {
+ margin-left: 3rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-sm-0 {
+ padding: 0 !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-sm-0,
+ .py-sm-0 {
+ padding-top: 0 !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-sm-0,
+ .px-sm-0 {
+ padding-right: 0 !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-sm-0,
+ .py-sm-0 {
+ padding-bottom: 0 !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-sm-0,
+ .px-sm-0 {
+ padding-left: 0 !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-sm-1 {
+ padding: 0.25rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-sm-1,
+ .py-sm-1 {
+ padding-top: 0.25rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-sm-1,
+ .px-sm-1 {
+ padding-right: 0.25rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-sm-1,
+ .py-sm-1 {
+ padding-bottom: 0.25rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-sm-1,
+ .px-sm-1 {
+ padding-left: 0.25rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-sm-2 {
+ padding: 0.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-sm-2,
+ .py-sm-2 {
+ padding-top: 0.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-sm-2,
+ .px-sm-2 {
+ padding-right: 0.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-sm-2,
+ .py-sm-2 {
+ padding-bottom: 0.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-sm-2,
+ .px-sm-2 {
+ padding-left: 0.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-sm-3 {
+ padding: 1rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-sm-3,
+ .py-sm-3 {
+ padding-top: 1rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-sm-3,
+ .px-sm-3 {
+ padding-right: 1rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-sm-3,
+ .py-sm-3 {
+ padding-bottom: 1rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-sm-3,
+ .px-sm-3 {
+ padding-left: 1rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-sm-4 {
+ padding: 1.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-sm-4,
+ .py-sm-4 {
+ padding-top: 1.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-sm-4,
+ .px-sm-4 {
+ padding-right: 1.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-sm-4,
+ .py-sm-4 {
+ padding-bottom: 1.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-sm-4,
+ .px-sm-4 {
+ padding-left: 1.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-sm-5 {
+ padding: 3rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-sm-5,
+ .py-sm-5 {
+ padding-top: 3rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-sm-5,
+ .px-sm-5 {
+ padding-right: 3rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-sm-5,
+ .py-sm-5 {
+ padding-bottom: 3rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-sm-5,
+ .px-sm-5 {
+ padding-left: 3rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-sm-n1 {
+ margin: -0.25rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-sm-n1,
+ .my-sm-n1 {
+ margin-top: -0.25rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-sm-n1,
+ .mx-sm-n1 {
+ margin-right: -0.25rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-sm-n1,
+ .my-sm-n1 {
+ margin-bottom: -0.25rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-sm-n1,
+ .mx-sm-n1 {
+ margin-left: -0.25rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-sm-n2 {
+ margin: -0.5rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-sm-n2,
+ .my-sm-n2 {
+ margin-top: -0.5rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-sm-n2,
+ .mx-sm-n2 {
+ margin-right: -0.5rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-sm-n2,
+ .my-sm-n2 {
+ margin-bottom: -0.5rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-sm-n2,
+ .mx-sm-n2 {
+ margin-left: -0.5rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-sm-n3 {
+ margin: -1rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-sm-n3,
+ .my-sm-n3 {
+ margin-top: -1rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-sm-n3,
+ .mx-sm-n3 {
+ margin-right: -1rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-sm-n3,
+ .my-sm-n3 {
+ margin-bottom: -1rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-sm-n3,
+ .mx-sm-n3 {
+ margin-left: -1rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-sm-n4 {
+ margin: -1.5rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-sm-n4,
+ .my-sm-n4 {
+ margin-top: -1.5rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-sm-n4,
+ .mx-sm-n4 {
+ margin-right: -1.5rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-sm-n4,
+ .my-sm-n4 {
+ margin-bottom: -1.5rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-sm-n4,
+ .mx-sm-n4 {
+ margin-left: -1.5rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-sm-n5 {
+ margin: -3rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-sm-n5,
+ .my-sm-n5 {
+ margin-top: -3rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-sm-n5,
+ .mx-sm-n5 {
+ margin-right: -3rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-sm-n5,
+ .my-sm-n5 {
+ margin-bottom: -3rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-sm-n5,
+ .mx-sm-n5 {
+ margin-left: -3rem !important;
+ }
+ /* line 55, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-sm-auto {
+ margin: auto !important;
+ }
+ /* line 56, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-sm-auto,
+ .my-sm-auto {
+ margin-top: auto !important;
+ }
+ /* line 60, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-sm-auto,
+ .mx-sm-auto {
+ margin-right: auto !important;
+ }
+ /* line 64, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-sm-auto,
+ .my-sm-auto {
+ margin-bottom: auto !important;
+ }
+ /* line 68, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-sm-auto,
+ .mx-sm-auto {
+ margin-left: auto !important;
+ }
+}
+
+@media (min-width: 768px) {
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-md-0 {
+ margin: 0 !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-md-0,
+ .my-md-0 {
+ margin-top: 0 !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-md-0,
+ .mx-md-0 {
+ margin-right: 0 !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-md-0,
+ .my-md-0 {
+ margin-bottom: 0 !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-md-0,
+ .mx-md-0 {
+ margin-left: 0 !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-md-1 {
+ margin: 0.25rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-md-1,
+ .my-md-1 {
+ margin-top: 0.25rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-md-1,
+ .mx-md-1 {
+ margin-right: 0.25rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-md-1,
+ .my-md-1 {
+ margin-bottom: 0.25rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-md-1,
+ .mx-md-1 {
+ margin-left: 0.25rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-md-2 {
+ margin: 0.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-md-2,
+ .my-md-2 {
+ margin-top: 0.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-md-2,
+ .mx-md-2 {
+ margin-right: 0.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-md-2,
+ .my-md-2 {
+ margin-bottom: 0.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-md-2,
+ .mx-md-2 {
+ margin-left: 0.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-md-3 {
+ margin: 1rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-md-3,
+ .my-md-3 {
+ margin-top: 1rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-md-3,
+ .mx-md-3 {
+ margin-right: 1rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-md-3,
+ .my-md-3 {
+ margin-bottom: 1rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-md-3,
+ .mx-md-3 {
+ margin-left: 1rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-md-4 {
+ margin: 1.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-md-4,
+ .my-md-4 {
+ margin-top: 1.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-md-4,
+ .mx-md-4 {
+ margin-right: 1.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-md-4,
+ .my-md-4 {
+ margin-bottom: 1.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-md-4,
+ .mx-md-4 {
+ margin-left: 1.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-md-5 {
+ margin: 3rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-md-5,
+ .my-md-5 {
+ margin-top: 3rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-md-5,
+ .mx-md-5 {
+ margin-right: 3rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-md-5,
+ .my-md-5 {
+ margin-bottom: 3rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-md-5,
+ .mx-md-5 {
+ margin-left: 3rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-md-0 {
+ padding: 0 !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-md-0,
+ .py-md-0 {
+ padding-top: 0 !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-md-0,
+ .px-md-0 {
+ padding-right: 0 !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-md-0,
+ .py-md-0 {
+ padding-bottom: 0 !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-md-0,
+ .px-md-0 {
+ padding-left: 0 !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-md-1 {
+ padding: 0.25rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-md-1,
+ .py-md-1 {
+ padding-top: 0.25rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-md-1,
+ .px-md-1 {
+ padding-right: 0.25rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-md-1,
+ .py-md-1 {
+ padding-bottom: 0.25rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-md-1,
+ .px-md-1 {
+ padding-left: 0.25rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-md-2 {
+ padding: 0.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-md-2,
+ .py-md-2 {
+ padding-top: 0.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-md-2,
+ .px-md-2 {
+ padding-right: 0.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-md-2,
+ .py-md-2 {
+ padding-bottom: 0.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-md-2,
+ .px-md-2 {
+ padding-left: 0.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-md-3 {
+ padding: 1rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-md-3,
+ .py-md-3 {
+ padding-top: 1rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-md-3,
+ .px-md-3 {
+ padding-right: 1rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-md-3,
+ .py-md-3 {
+ padding-bottom: 1rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-md-3,
+ .px-md-3 {
+ padding-left: 1rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-md-4 {
+ padding: 1.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-md-4,
+ .py-md-4 {
+ padding-top: 1.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-md-4,
+ .px-md-4 {
+ padding-right: 1.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-md-4,
+ .py-md-4 {
+ padding-bottom: 1.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-md-4,
+ .px-md-4 {
+ padding-left: 1.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-md-5 {
+ padding: 3rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-md-5,
+ .py-md-5 {
+ padding-top: 3rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-md-5,
+ .px-md-5 {
+ padding-right: 3rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-md-5,
+ .py-md-5 {
+ padding-bottom: 3rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-md-5,
+ .px-md-5 {
+ padding-left: 3rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-md-n1 {
+ margin: -0.25rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-md-n1,
+ .my-md-n1 {
+ margin-top: -0.25rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-md-n1,
+ .mx-md-n1 {
+ margin-right: -0.25rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-md-n1,
+ .my-md-n1 {
+ margin-bottom: -0.25rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-md-n1,
+ .mx-md-n1 {
+ margin-left: -0.25rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-md-n2 {
+ margin: -0.5rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-md-n2,
+ .my-md-n2 {
+ margin-top: -0.5rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-md-n2,
+ .mx-md-n2 {
+ margin-right: -0.5rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-md-n2,
+ .my-md-n2 {
+ margin-bottom: -0.5rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-md-n2,
+ .mx-md-n2 {
+ margin-left: -0.5rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-md-n3 {
+ margin: -1rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-md-n3,
+ .my-md-n3 {
+ margin-top: -1rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-md-n3,
+ .mx-md-n3 {
+ margin-right: -1rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-md-n3,
+ .my-md-n3 {
+ margin-bottom: -1rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-md-n3,
+ .mx-md-n3 {
+ margin-left: -1rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-md-n4 {
+ margin: -1.5rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-md-n4,
+ .my-md-n4 {
+ margin-top: -1.5rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-md-n4,
+ .mx-md-n4 {
+ margin-right: -1.5rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-md-n4,
+ .my-md-n4 {
+ margin-bottom: -1.5rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-md-n4,
+ .mx-md-n4 {
+ margin-left: -1.5rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-md-n5 {
+ margin: -3rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-md-n5,
+ .my-md-n5 {
+ margin-top: -3rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-md-n5,
+ .mx-md-n5 {
+ margin-right: -3rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-md-n5,
+ .my-md-n5 {
+ margin-bottom: -3rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-md-n5,
+ .mx-md-n5 {
+ margin-left: -3rem !important;
+ }
+ /* line 55, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-md-auto {
+ margin: auto !important;
+ }
+ /* line 56, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-md-auto,
+ .my-md-auto {
+ margin-top: auto !important;
+ }
+ /* line 60, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-md-auto,
+ .mx-md-auto {
+ margin-right: auto !important;
+ }
+ /* line 64, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-md-auto,
+ .my-md-auto {
+ margin-bottom: auto !important;
+ }
+ /* line 68, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-md-auto,
+ .mx-md-auto {
+ margin-left: auto !important;
+ }
+}
+
+@media (min-width: 1024px) {
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-lg-0 {
+ margin: 0 !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-lg-0,
+ .my-lg-0 {
+ margin-top: 0 !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-lg-0,
+ .mx-lg-0 {
+ margin-right: 0 !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-lg-0,
+ .my-lg-0 {
+ margin-bottom: 0 !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-lg-0,
+ .mx-lg-0 {
+ margin-left: 0 !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-lg-1 {
+ margin: 0.25rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-lg-1,
+ .my-lg-1 {
+ margin-top: 0.25rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-lg-1,
+ .mx-lg-1 {
+ margin-right: 0.25rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-lg-1,
+ .my-lg-1 {
+ margin-bottom: 0.25rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-lg-1,
+ .mx-lg-1 {
+ margin-left: 0.25rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-lg-2 {
+ margin: 0.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-lg-2,
+ .my-lg-2 {
+ margin-top: 0.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-lg-2,
+ .mx-lg-2 {
+ margin-right: 0.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-lg-2,
+ .my-lg-2 {
+ margin-bottom: 0.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-lg-2,
+ .mx-lg-2 {
+ margin-left: 0.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-lg-3 {
+ margin: 1rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-lg-3,
+ .my-lg-3 {
+ margin-top: 1rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-lg-3,
+ .mx-lg-3 {
+ margin-right: 1rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-lg-3,
+ .my-lg-3 {
+ margin-bottom: 1rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-lg-3,
+ .mx-lg-3 {
+ margin-left: 1rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-lg-4 {
+ margin: 1.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-lg-4,
+ .my-lg-4 {
+ margin-top: 1.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-lg-4,
+ .mx-lg-4 {
+ margin-right: 1.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-lg-4,
+ .my-lg-4 {
+ margin-bottom: 1.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-lg-4,
+ .mx-lg-4 {
+ margin-left: 1.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-lg-5 {
+ margin: 3rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-lg-5,
+ .my-lg-5 {
+ margin-top: 3rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-lg-5,
+ .mx-lg-5 {
+ margin-right: 3rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-lg-5,
+ .my-lg-5 {
+ margin-bottom: 3rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-lg-5,
+ .mx-lg-5 {
+ margin-left: 3rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-lg-0 {
+ padding: 0 !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-lg-0,
+ .py-lg-0 {
+ padding-top: 0 !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-lg-0,
+ .px-lg-0 {
+ padding-right: 0 !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-lg-0,
+ .py-lg-0 {
+ padding-bottom: 0 !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-lg-0,
+ .px-lg-0 {
+ padding-left: 0 !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-lg-1 {
+ padding: 0.25rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-lg-1,
+ .py-lg-1 {
+ padding-top: 0.25rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-lg-1,
+ .px-lg-1 {
+ padding-right: 0.25rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-lg-1,
+ .py-lg-1 {
+ padding-bottom: 0.25rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-lg-1,
+ .px-lg-1 {
+ padding-left: 0.25rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-lg-2 {
+ padding: 0.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-lg-2,
+ .py-lg-2 {
+ padding-top: 0.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-lg-2,
+ .px-lg-2 {
+ padding-right: 0.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-lg-2,
+ .py-lg-2 {
+ padding-bottom: 0.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-lg-2,
+ .px-lg-2 {
+ padding-left: 0.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-lg-3 {
+ padding: 1rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-lg-3,
+ .py-lg-3 {
+ padding-top: 1rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-lg-3,
+ .px-lg-3 {
+ padding-right: 1rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-lg-3,
+ .py-lg-3 {
+ padding-bottom: 1rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-lg-3,
+ .px-lg-3 {
+ padding-left: 1rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-lg-4 {
+ padding: 1.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-lg-4,
+ .py-lg-4 {
+ padding-top: 1.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-lg-4,
+ .px-lg-4 {
+ padding-right: 1.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-lg-4,
+ .py-lg-4 {
+ padding-bottom: 1.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-lg-4,
+ .px-lg-4 {
+ padding-left: 1.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-lg-5 {
+ padding: 3rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-lg-5,
+ .py-lg-5 {
+ padding-top: 3rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-lg-5,
+ .px-lg-5 {
+ padding-right: 3rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-lg-5,
+ .py-lg-5 {
+ padding-bottom: 3rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-lg-5,
+ .px-lg-5 {
+ padding-left: 3rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-lg-n1 {
+ margin: -0.25rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-lg-n1,
+ .my-lg-n1 {
+ margin-top: -0.25rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-lg-n1,
+ .mx-lg-n1 {
+ margin-right: -0.25rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-lg-n1,
+ .my-lg-n1 {
+ margin-bottom: -0.25rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-lg-n1,
+ .mx-lg-n1 {
+ margin-left: -0.25rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-lg-n2 {
+ margin: -0.5rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-lg-n2,
+ .my-lg-n2 {
+ margin-top: -0.5rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-lg-n2,
+ .mx-lg-n2 {
+ margin-right: -0.5rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-lg-n2,
+ .my-lg-n2 {
+ margin-bottom: -0.5rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-lg-n2,
+ .mx-lg-n2 {
+ margin-left: -0.5rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-lg-n3 {
+ margin: -1rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-lg-n3,
+ .my-lg-n3 {
+ margin-top: -1rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-lg-n3,
+ .mx-lg-n3 {
+ margin-right: -1rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-lg-n3,
+ .my-lg-n3 {
+ margin-bottom: -1rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-lg-n3,
+ .mx-lg-n3 {
+ margin-left: -1rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-lg-n4 {
+ margin: -1.5rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-lg-n4,
+ .my-lg-n4 {
+ margin-top: -1.5rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-lg-n4,
+ .mx-lg-n4 {
+ margin-right: -1.5rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-lg-n4,
+ .my-lg-n4 {
+ margin-bottom: -1.5rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-lg-n4,
+ .mx-lg-n4 {
+ margin-left: -1.5rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-lg-n5 {
+ margin: -3rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-lg-n5,
+ .my-lg-n5 {
+ margin-top: -3rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-lg-n5,
+ .mx-lg-n5 {
+ margin-right: -3rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-lg-n5,
+ .my-lg-n5 {
+ margin-bottom: -3rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-lg-n5,
+ .mx-lg-n5 {
+ margin-left: -3rem !important;
+ }
+ /* line 55, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-lg-auto {
+ margin: auto !important;
+ }
+ /* line 56, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-lg-auto,
+ .my-lg-auto {
+ margin-top: auto !important;
+ }
+ /* line 60, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-lg-auto,
+ .mx-lg-auto {
+ margin-right: auto !important;
+ }
+ /* line 64, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-lg-auto,
+ .my-lg-auto {
+ margin-bottom: auto !important;
+ }
+ /* line 68, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-lg-auto,
+ .mx-lg-auto {
+ margin-left: auto !important;
+ }
+}
+
+@media (min-width: 1280px) {
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xl-0 {
+ margin: 0 !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xl-0,
+ .my-xl-0 {
+ margin-top: 0 !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xl-0,
+ .mx-xl-0 {
+ margin-right: 0 !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xl-0,
+ .my-xl-0 {
+ margin-bottom: 0 !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xl-0,
+ .mx-xl-0 {
+ margin-left: 0 !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xl-1 {
+ margin: 0.25rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xl-1,
+ .my-xl-1 {
+ margin-top: 0.25rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xl-1,
+ .mx-xl-1 {
+ margin-right: 0.25rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xl-1,
+ .my-xl-1 {
+ margin-bottom: 0.25rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xl-1,
+ .mx-xl-1 {
+ margin-left: 0.25rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xl-2 {
+ margin: 0.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xl-2,
+ .my-xl-2 {
+ margin-top: 0.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xl-2,
+ .mx-xl-2 {
+ margin-right: 0.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xl-2,
+ .my-xl-2 {
+ margin-bottom: 0.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xl-2,
+ .mx-xl-2 {
+ margin-left: 0.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xl-3 {
+ margin: 1rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xl-3,
+ .my-xl-3 {
+ margin-top: 1rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xl-3,
+ .mx-xl-3 {
+ margin-right: 1rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xl-3,
+ .my-xl-3 {
+ margin-bottom: 1rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xl-3,
+ .mx-xl-3 {
+ margin-left: 1rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xl-4 {
+ margin: 1.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xl-4,
+ .my-xl-4 {
+ margin-top: 1.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xl-4,
+ .mx-xl-4 {
+ margin-right: 1.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xl-4,
+ .my-xl-4 {
+ margin-bottom: 1.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xl-4,
+ .mx-xl-4 {
+ margin-left: 1.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xl-5 {
+ margin: 3rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xl-5,
+ .my-xl-5 {
+ margin-top: 3rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xl-5,
+ .mx-xl-5 {
+ margin-right: 3rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xl-5,
+ .my-xl-5 {
+ margin-bottom: 3rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xl-5,
+ .mx-xl-5 {
+ margin-left: 3rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-xl-0 {
+ padding: 0 !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-xl-0,
+ .py-xl-0 {
+ padding-top: 0 !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-xl-0,
+ .px-xl-0 {
+ padding-right: 0 !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-xl-0,
+ .py-xl-0 {
+ padding-bottom: 0 !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-xl-0,
+ .px-xl-0 {
+ padding-left: 0 !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-xl-1 {
+ padding: 0.25rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-xl-1,
+ .py-xl-1 {
+ padding-top: 0.25rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-xl-1,
+ .px-xl-1 {
+ padding-right: 0.25rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-xl-1,
+ .py-xl-1 {
+ padding-bottom: 0.25rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-xl-1,
+ .px-xl-1 {
+ padding-left: 0.25rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-xl-2 {
+ padding: 0.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-xl-2,
+ .py-xl-2 {
+ padding-top: 0.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-xl-2,
+ .px-xl-2 {
+ padding-right: 0.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-xl-2,
+ .py-xl-2 {
+ padding-bottom: 0.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-xl-2,
+ .px-xl-2 {
+ padding-left: 0.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-xl-3 {
+ padding: 1rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-xl-3,
+ .py-xl-3 {
+ padding-top: 1rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-xl-3,
+ .px-xl-3 {
+ padding-right: 1rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-xl-3,
+ .py-xl-3 {
+ padding-bottom: 1rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-xl-3,
+ .px-xl-3 {
+ padding-left: 1rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-xl-4 {
+ padding: 1.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-xl-4,
+ .py-xl-4 {
+ padding-top: 1.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-xl-4,
+ .px-xl-4 {
+ padding-right: 1.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-xl-4,
+ .py-xl-4 {
+ padding-bottom: 1.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-xl-4,
+ .px-xl-4 {
+ padding-left: 1.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-xl-5 {
+ padding: 3rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-xl-5,
+ .py-xl-5 {
+ padding-top: 3rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-xl-5,
+ .px-xl-5 {
+ padding-right: 3rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-xl-5,
+ .py-xl-5 {
+ padding-bottom: 3rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-xl-5,
+ .px-xl-5 {
+ padding-left: 3rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xl-n1 {
+ margin: -0.25rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xl-n1,
+ .my-xl-n1 {
+ margin-top: -0.25rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xl-n1,
+ .mx-xl-n1 {
+ margin-right: -0.25rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xl-n1,
+ .my-xl-n1 {
+ margin-bottom: -0.25rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xl-n1,
+ .mx-xl-n1 {
+ margin-left: -0.25rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xl-n2 {
+ margin: -0.5rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xl-n2,
+ .my-xl-n2 {
+ margin-top: -0.5rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xl-n2,
+ .mx-xl-n2 {
+ margin-right: -0.5rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xl-n2,
+ .my-xl-n2 {
+ margin-bottom: -0.5rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xl-n2,
+ .mx-xl-n2 {
+ margin-left: -0.5rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xl-n3 {
+ margin: -1rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xl-n3,
+ .my-xl-n3 {
+ margin-top: -1rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xl-n3,
+ .mx-xl-n3 {
+ margin-right: -1rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xl-n3,
+ .my-xl-n3 {
+ margin-bottom: -1rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xl-n3,
+ .mx-xl-n3 {
+ margin-left: -1rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xl-n4 {
+ margin: -1.5rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xl-n4,
+ .my-xl-n4 {
+ margin-top: -1.5rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xl-n4,
+ .mx-xl-n4 {
+ margin-right: -1.5rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xl-n4,
+ .my-xl-n4 {
+ margin-bottom: -1.5rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xl-n4,
+ .mx-xl-n4 {
+ margin-left: -1.5rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xl-n5 {
+ margin: -3rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xl-n5,
+ .my-xl-n5 {
+ margin-top: -3rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xl-n5,
+ .mx-xl-n5 {
+ margin-right: -3rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xl-n5,
+ .my-xl-n5 {
+ margin-bottom: -3rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xl-n5,
+ .mx-xl-n5 {
+ margin-left: -3rem !important;
+ }
+ /* line 55, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xl-auto {
+ margin: auto !important;
+ }
+ /* line 56, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xl-auto,
+ .my-xl-auto {
+ margin-top: auto !important;
+ }
+ /* line 60, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xl-auto,
+ .mx-xl-auto {
+ margin-right: auto !important;
+ }
+ /* line 64, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xl-auto,
+ .my-xl-auto {
+ margin-bottom: auto !important;
+ }
+ /* line 68, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xl-auto,
+ .mx-xl-auto {
+ margin-left: auto !important;
+ }
+}
+
+@media (min-width: 1440px) {
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xxl-0 {
+ margin: 0 !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xxl-0,
+ .my-xxl-0 {
+ margin-top: 0 !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xxl-0,
+ .mx-xxl-0 {
+ margin-right: 0 !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xxl-0,
+ .my-xxl-0 {
+ margin-bottom: 0 !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xxl-0,
+ .mx-xxl-0 {
+ margin-left: 0 !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xxl-1 {
+ margin: 0.25rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xxl-1,
+ .my-xxl-1 {
+ margin-top: 0.25rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xxl-1,
+ .mx-xxl-1 {
+ margin-right: 0.25rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xxl-1,
+ .my-xxl-1 {
+ margin-bottom: 0.25rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xxl-1,
+ .mx-xxl-1 {
+ margin-left: 0.25rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xxl-2 {
+ margin: 0.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xxl-2,
+ .my-xxl-2 {
+ margin-top: 0.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xxl-2,
+ .mx-xxl-2 {
+ margin-right: 0.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xxl-2,
+ .my-xxl-2 {
+ margin-bottom: 0.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xxl-2,
+ .mx-xxl-2 {
+ margin-left: 0.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xxl-3 {
+ margin: 1rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xxl-3,
+ .my-xxl-3 {
+ margin-top: 1rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xxl-3,
+ .mx-xxl-3 {
+ margin-right: 1rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xxl-3,
+ .my-xxl-3 {
+ margin-bottom: 1rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xxl-3,
+ .mx-xxl-3 {
+ margin-left: 1rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xxl-4 {
+ margin: 1.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xxl-4,
+ .my-xxl-4 {
+ margin-top: 1.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xxl-4,
+ .mx-xxl-4 {
+ margin-right: 1.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xxl-4,
+ .my-xxl-4 {
+ margin-bottom: 1.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xxl-4,
+ .mx-xxl-4 {
+ margin-left: 1.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xxl-5 {
+ margin: 3rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xxl-5,
+ .my-xxl-5 {
+ margin-top: 3rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xxl-5,
+ .mx-xxl-5 {
+ margin-right: 3rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xxl-5,
+ .my-xxl-5 {
+ margin-bottom: 3rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xxl-5,
+ .mx-xxl-5 {
+ margin-left: 3rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-xxl-0 {
+ padding: 0 !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-xxl-0,
+ .py-xxl-0 {
+ padding-top: 0 !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-xxl-0,
+ .px-xxl-0 {
+ padding-right: 0 !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-xxl-0,
+ .py-xxl-0 {
+ padding-bottom: 0 !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-xxl-0,
+ .px-xxl-0 {
+ padding-left: 0 !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-xxl-1 {
+ padding: 0.25rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-xxl-1,
+ .py-xxl-1 {
+ padding-top: 0.25rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-xxl-1,
+ .px-xxl-1 {
+ padding-right: 0.25rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-xxl-1,
+ .py-xxl-1 {
+ padding-bottom: 0.25rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-xxl-1,
+ .px-xxl-1 {
+ padding-left: 0.25rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-xxl-2 {
+ padding: 0.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-xxl-2,
+ .py-xxl-2 {
+ padding-top: 0.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-xxl-2,
+ .px-xxl-2 {
+ padding-right: 0.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-xxl-2,
+ .py-xxl-2 {
+ padding-bottom: 0.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-xxl-2,
+ .px-xxl-2 {
+ padding-left: 0.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-xxl-3 {
+ padding: 1rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-xxl-3,
+ .py-xxl-3 {
+ padding-top: 1rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-xxl-3,
+ .px-xxl-3 {
+ padding-right: 1rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-xxl-3,
+ .py-xxl-3 {
+ padding-bottom: 1rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-xxl-3,
+ .px-xxl-3 {
+ padding-left: 1rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-xxl-4 {
+ padding: 1.5rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-xxl-4,
+ .py-xxl-4 {
+ padding-top: 1.5rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-xxl-4,
+ .px-xxl-4 {
+ padding-right: 1.5rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-xxl-4,
+ .py-xxl-4 {
+ padding-bottom: 1.5rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-xxl-4,
+ .px-xxl-4 {
+ padding-left: 1.5rem !important;
+ }
+ /* line 11, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .p-xxl-5 {
+ padding: 3rem !important;
+ }
+ /* line 12, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pt-xxl-5,
+ .py-xxl-5 {
+ padding-top: 3rem !important;
+ }
+ /* line 16, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pr-xxl-5,
+ .px-xxl-5 {
+ padding-right: 3rem !important;
+ }
+ /* line 20, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pb-xxl-5,
+ .py-xxl-5 {
+ padding-bottom: 3rem !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .pl-xxl-5,
+ .px-xxl-5 {
+ padding-left: 3rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xxl-n1 {
+ margin: -0.25rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xxl-n1,
+ .my-xxl-n1 {
+ margin-top: -0.25rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xxl-n1,
+ .mx-xxl-n1 {
+ margin-right: -0.25rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xxl-n1,
+ .my-xxl-n1 {
+ margin-bottom: -0.25rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xxl-n1,
+ .mx-xxl-n1 {
+ margin-left: -0.25rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xxl-n2 {
+ margin: -0.5rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xxl-n2,
+ .my-xxl-n2 {
+ margin-top: -0.5rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xxl-n2,
+ .mx-xxl-n2 {
+ margin-right: -0.5rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xxl-n2,
+ .my-xxl-n2 {
+ margin-bottom: -0.5rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xxl-n2,
+ .mx-xxl-n2 {
+ margin-left: -0.5rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xxl-n3 {
+ margin: -1rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xxl-n3,
+ .my-xxl-n3 {
+ margin-top: -1rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xxl-n3,
+ .mx-xxl-n3 {
+ margin-right: -1rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xxl-n3,
+ .my-xxl-n3 {
+ margin-bottom: -1rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xxl-n3,
+ .mx-xxl-n3 {
+ margin-left: -1rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xxl-n4 {
+ margin: -1.5rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xxl-n4,
+ .my-xxl-n4 {
+ margin-top: -1.5rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xxl-n4,
+ .mx-xxl-n4 {
+ margin-right: -1.5rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xxl-n4,
+ .my-xxl-n4 {
+ margin-bottom: -1.5rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xxl-n4,
+ .mx-xxl-n4 {
+ margin-left: -1.5rem !important;
+ }
+ /* line 34, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xxl-n5 {
+ margin: -3rem !important;
+ }
+ /* line 35, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xxl-n5,
+ .my-xxl-n5 {
+ margin-top: -3rem !important;
+ }
+ /* line 39, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xxl-n5,
+ .mx-xxl-n5 {
+ margin-right: -3rem !important;
+ }
+ /* line 43, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xxl-n5,
+ .my-xxl-n5 {
+ margin-bottom: -3rem !important;
+ }
+ /* line 47, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xxl-n5,
+ .mx-xxl-n5 {
+ margin-left: -3rem !important;
+ }
+ /* line 55, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .m-xxl-auto {
+ margin: auto !important;
+ }
+ /* line 56, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mt-xxl-auto,
+ .my-xxl-auto {
+ margin-top: auto !important;
+ }
+ /* line 60, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mr-xxl-auto,
+ .mx-xxl-auto {
+ margin-right: auto !important;
+ }
+ /* line 64, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .mb-xxl-auto,
+ .my-xxl-auto {
+ margin-bottom: auto !important;
+ }
+ /* line 68, node_modules/bootstrap/scss/utilities/_spacing.scss */
+ .ml-xxl-auto,
+ .mx-xxl-auto {
+ margin-left: auto !important;
+ }
+}
+
+/* line 6, node_modules/bootstrap/scss/utilities/_stretched-link.scss */
+.stretched-link::after {
+ position: absolute;
+ top: 0;
+ right: 0;
+ bottom: 0;
+ left: 0;
+ z-index: 1;
+ pointer-events: auto;
+ content: "";
+ background-color: rgba(0, 0, 0, 0);
+}
+
+/* line 7, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-monospace {
+ font-family: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-justify {
+ text-align: justify !important;
+}
+
+/* line 12, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-wrap {
+ white-space: normal !important;
+}
+
+/* line 13, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-nowrap {
+ white-space: nowrap !important;
+}
+
+/* line 14, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-truncate {
+ overflow: hidden;
+ text-overflow: ellipsis;
+ white-space: nowrap;
+}
+
+/* line 22, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-left {
+ text-align: left !important;
+}
+
+/* line 23, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-right {
+ text-align: right !important;
+}
+
+/* line 24, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-center {
+ text-align: center !important;
+}
+
+@media (min-width: 576px) {
+ /* line 22, node_modules/bootstrap/scss/utilities/_text.scss */
+ .text-sm-left {
+ text-align: left !important;
+ }
+ /* line 23, node_modules/bootstrap/scss/utilities/_text.scss */
+ .text-sm-right {
+ text-align: right !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_text.scss */
+ .text-sm-center {
+ text-align: center !important;
+ }
+}
+
+@media (min-width: 768px) {
+ /* line 22, node_modules/bootstrap/scss/utilities/_text.scss */
+ .text-md-left {
+ text-align: left !important;
+ }
+ /* line 23, node_modules/bootstrap/scss/utilities/_text.scss */
+ .text-md-right {
+ text-align: right !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_text.scss */
+ .text-md-center {
+ text-align: center !important;
+ }
+}
+
+@media (min-width: 1024px) {
+ /* line 22, node_modules/bootstrap/scss/utilities/_text.scss */
+ .text-lg-left {
+ text-align: left !important;
+ }
+ /* line 23, node_modules/bootstrap/scss/utilities/_text.scss */
+ .text-lg-right {
+ text-align: right !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_text.scss */
+ .text-lg-center {
+ text-align: center !important;
+ }
+}
+
+@media (min-width: 1280px) {
+ /* line 22, node_modules/bootstrap/scss/utilities/_text.scss */
+ .text-xl-left {
+ text-align: left !important;
+ }
+ /* line 23, node_modules/bootstrap/scss/utilities/_text.scss */
+ .text-xl-right {
+ text-align: right !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_text.scss */
+ .text-xl-center {
+ text-align: center !important;
+ }
+}
+
+@media (min-width: 1440px) {
+ /* line 22, node_modules/bootstrap/scss/utilities/_text.scss */
+ .text-xxl-left {
+ text-align: left !important;
+ }
+ /* line 23, node_modules/bootstrap/scss/utilities/_text.scss */
+ .text-xxl-right {
+ text-align: right !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/utilities/_text.scss */
+ .text-xxl-center {
+ text-align: center !important;
+ }
+}
+
+/* line 30, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-lowercase {
+ text-transform: lowercase !important;
+}
+
+/* line 31, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-uppercase {
+ text-transform: uppercase !important;
+}
+
+/* line 32, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-capitalize {
+ text-transform: capitalize !important;
+}
+
+/* line 36, node_modules/bootstrap/scss/utilities/_text.scss */
+.font-weight-light {
+ font-weight: 300 !important;
+}
+
+/* line 37, node_modules/bootstrap/scss/utilities/_text.scss */
+.font-weight-lighter {
+ font-weight: lighter !important;
+}
+
+/* line 38, node_modules/bootstrap/scss/utilities/_text.scss */
+.font-weight-normal {
+ font-weight: 400 !important;
+}
+
+/* line 39, node_modules/bootstrap/scss/utilities/_text.scss */
+.font-weight-bold {
+ font-weight: 700 !important;
+}
+
+/* line 40, node_modules/bootstrap/scss/utilities/_text.scss */
+.font-weight-bolder {
+ font-weight: 800 !important;
+}
+
+/* line 41, node_modules/bootstrap/scss/utilities/_text.scss */
+.font-italic {
+ font-style: italic !important;
+}
+
+/* line 45, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-white {
+ color: #ffffff !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_text-emphasis.scss */
+.text-primary {
+ color: #464746 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.text-primary:hover, a.text-primary:focus {
+ color: #202020 !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_text-emphasis.scss */
+.text-secondary {
+ color: #f0ebe3 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.text-secondary:hover, a.text-secondary:focus {
+ color: #d5c7b1 !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_text-emphasis.scss */
+.text-success {
+ color: #f0ebe3 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.text-success:hover, a.text-success:focus {
+ color: #d5c7b1 !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_text-emphasis.scss */
+.text-info {
+ color: #464746 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.text-info:hover, a.text-info:focus {
+ color: #202020 !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_text-emphasis.scss */
+.text-warning {
+ color: #464746 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.text-warning:hover, a.text-warning:focus {
+ color: #202020 !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_text-emphasis.scss */
+.text-danger {
+ color: #e54a19 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.text-danger:hover, a.text-danger:focus {
+ color: #a03411 !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_text-emphasis.scss */
+.text-light {
+ color: #f7f7f7 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.text-light:hover, a.text-light:focus {
+ color: #d1d1d1 !important;
+}
+
+/* line 6, node_modules/bootstrap/scss/mixins/_text-emphasis.scss */
+.text-dark {
+ color: #343a40 !important;
+}
+
+/* line 17, node_modules/bootstrap/scss/mixins/_hover.scss */
+a.text-dark:hover, a.text-dark:focus {
+ color: #121416 !important;
+}
+
+/* line 51, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-body {
+ color: #464746 !important;
+}
+
+/* line 52, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-muted {
+ color: #6c757d !important;
+}
+
+/* line 54, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-black-50 {
+ color: rgba(0, 0, 0, 0.5) !important;
+}
+
+/* line 55, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-white-50 {
+ color: rgba(255, 255, 255, 0.5) !important;
+}
+
+/* line 59, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-hide {
+ font: 0/0 a;
+ color: transparent;
+ text-shadow: none;
+ background-color: transparent;
+ border: 0;
+}
+
+/* line 63, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-decoration-none {
+ text-decoration: none !important;
+}
+
+/* line 65, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-break {
+ word-break: break-word !important;
+ overflow-wrap: break-word !important;
+}
+
+/* line 72, node_modules/bootstrap/scss/utilities/_text.scss */
+.text-reset {
+ color: inherit !important;
+}
+
+/* line 7, node_modules/bootstrap/scss/utilities/_visibility.scss */
+.visible {
+ visibility: visible !important;
+}
+
+/* line 11, node_modules/bootstrap/scss/utilities/_visibility.scss */
+.invisible {
+ visibility: hidden !important;
+}
+
+@media print {
+ /* line 13, node_modules/bootstrap/scss/_print.scss */
+ *,
+ *::before,
+ *::after {
+ text-shadow: none !important;
+ -webkit-box-shadow: none !important;
+ box-shadow: none !important;
+ }
+ /* line 24, node_modules/bootstrap/scss/_print.scss */
+ a:not(.btn) {
+ text-decoration: underline;
+ }
+ /* line 34, node_modules/bootstrap/scss/_print.scss */
+ abbr[title]::after {
+ content: " (" attr(title) ")";
+ }
+ /* line 49, node_modules/bootstrap/scss/_print.scss */
+ pre {
+ white-space: pre-wrap !important;
+ }
+ /* line 52, node_modules/bootstrap/scss/_print.scss */
+ pre,
+ blockquote {
+ border: 1px solid #adb5bd;
+ page-break-inside: avoid;
+ }
+ /* line 63, node_modules/bootstrap/scss/_print.scss */
+ thead {
+ display: table-header-group;
+ }
+ /* line 67, node_modules/bootstrap/scss/_print.scss */
+ tr,
+ img {
+ page-break-inside: avoid;
+ }
+ /* line 72, node_modules/bootstrap/scss/_print.scss */
+ p,
+ h2,
+ h3 {
+ orphans: 3;
+ widows: 3;
+ }
+ /* line 79, node_modules/bootstrap/scss/_print.scss */
+ h2,
+ h3 {
+ page-break-after: avoid;
+ }
+ @page {
+ size: a3;
+ }
+ /* line 92, node_modules/bootstrap/scss/_print.scss */
+ body {
+ min-width: 1024px !important;
+ }
+ /* line 95, node_modules/bootstrap/scss/_print.scss */
+ .container {
+ min-width: 1024px !important;
+ }
+ /* line 100, node_modules/bootstrap/scss/_print.scss */
+ .navbar {
+ display: none;
+ }
+ /* line 103, node_modules/bootstrap/scss/_print.scss */
+ .badge {
+ border: 1px solid #000000;
+ }
+ /* line 107, node_modules/bootstrap/scss/_print.scss */
+ .table {
+ border-collapse: collapse !important;
+ }
+ /* line 110, node_modules/bootstrap/scss/_print.scss */
+ .table td,
+ .table th {
+ background-color: #ffffff !important;
+ }
+ /* line 117, node_modules/bootstrap/scss/_print.scss */
+ .table-bordered th,
+ .table-bordered td {
+ border: 1px solid #dee2e6 !important;
+ }
+ /* line 123, node_modules/bootstrap/scss/_print.scss */
+ .table-dark {
+ color: inherit;
+ }
+ /* line 126, node_modules/bootstrap/scss/_print.scss */
+ .table-dark th,
+ .table-dark td,
+ .table-dark thead th,
+ .table-dark tbody + tbody {
+ border-color: #dee2e6;
+ }
+ /* line 134, node_modules/bootstrap/scss/_print.scss */
+ .table .thead-dark th {
+ color: inherit;
+ border-color: #dee2e6;
+ }
+}
+
+/**
+ * Set up a decent box model on the root element
+ */
+/* line 8, src/assets/scss/base/_base.scss */
+html {
+ -webkit-box-sizing: border-box;
+ box-sizing: border-box;
+}
+
+/**
+ * Make all elements from the DOM inherit from the parent box-sizing
+ * Since `*` has a specificity of 0, it does not override the `html` value
+ * making all elements inheriting from the root box-sizing value
+ * See: https://css-tricks.com/inheriting-box-sizing-probably-slightly-better-best-practice/
+ */
+/* line 18, src/assets/scss/base/_base.scss */
+*,
+*::before,
+*::after {
+ -webkit-box-sizing: inherit;
+ box-sizing: inherit;
+}
+
+/**
+ * Basic styles for links
+ */
+/* line 27, src/assets/scss/base/_base.scss */
+a {
+ color: #e54a19;
+ text-decoration: none;
+}
+
+/**
+ * Basic typography style for copy text
+ A solution for this problem is percentage. Usually default font-size of the browser is 16px.
+ Setting font-size: 100% will make 1rem = 16px. But it will make calculations a little difficult.
+ A better way is to set font-size: 62.5%. Because 62.5% of 16px is 10px. Which makes 1rem = 10px.
+ CALCULATION: Element font size in rem x 16px;
+ */
+/* line 16, src/assets/scss/base/_typography.scss */
+body {
+ font-size: 0.875rem;
+ font-weight: 300;
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
+ letter-spacing: 0.4px;
+ line-height: 1.5rem;
+ color: #464746;
+}
+
+@media (min-width: 1024px) {
+ /* line 16, src/assets/scss/base/_typography.scss */
+ body {
+ font-size: 1rem;
+ }
+}
+
+/* line 30, src/assets/scss/base/_typography.scss */
+h1, h2, h3, h4, h5, h6,
+.h1, .h2, .h3, .h4, .h5, .h6 {
+ margin-bottom: 28px;
+}
+
+@media (min-width: 1024px) {
+ /* line 30, src/assets/scss/base/_typography.scss */
+ h1, h2, h3, h4, h5, h6,
+ .h1, .h2, .h3, .h4, .h5, .h6 {
+ margin-bottom: 36px;
+ }
+}
+
+/* line 49, src/assets/scss/base/_typography.scss */
+ol,
+ul,
+p,
+blockquote,
+.preamble {
+ margin-bottom: 28px;
+}
+
+@media (min-width: 1024px) {
+ /* line 49, src/assets/scss/base/_typography.scss */
+ ol,
+ ul,
+ p,
+ blockquote,
+ .preamble {
+ margin-bottom: 36px;
+ }
+}
+
+/* line 63, src/assets/scss/base/_typography.scss */
+h1,
+h2,
+h3,
+h4,
+.h1,
+.h2,
+.h3,
+.h4 {
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
+}
+
+/* line 74, src/assets/scss/base/_typography.scss */
+h5,
+h6,
+.h5,
+.h6 {
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
+}
+
+/* line 81, src/assets/scss/base/_typography.scss */
+h1, .h1 {
+ color: #111111;
+ font-size: 2.25rem;
+ font-weight: 400;
+ line-height: 2.5rem;
+ letter-spacing: 0.13px;
+ text-transform: uppercase;
+}
+
+@media (min-width: 1024px) {
+ /* line 81, src/assets/scss/base/_typography.scss */
+ h1, .h1 {
+ font-size: 4rem;
+ letter-spacing: 0.22px;
+ line-height: 4.5rem;
+ }
+}
+
+/* line 97, src/assets/scss/base/_typography.scss */
+h2, .h2 {
+ color: #111111;
+ font-size: 1.5rem;
+ font-weight: 400;
+ letter-spacing: 0.08px;
+ line-height: 2.25rem;
+ text-transform: initial;
+}
+
+@media (min-width: 1024px) {
+ /* line 97, src/assets/scss/base/_typography.scss */
+ h2, .h2 {
+ font-size: 2.4375rem;
+ letter-spacing: 0.14px;
+ line-height: 3rem;
+ }
+}
+
+/* line 113, src/assets/scss/base/_typography.scss */
+h3, .h3 {
+ color: #111111;
+ font-size: 1.0625rem;
+ font-weight: 400;
+ letter-spacing: 0.06px;
+ line-height: 1.5rem;
+ text-transform: initial;
+}
+
+@media (min-width: 1024px) {
+ /* line 113, src/assets/scss/base/_typography.scss */
+ h3, .h3 {
+ font-size: 1.5rem;
+ letter-spacing: 0.08px;
+ line-height: 2.25rem;
+ }
+}
+
+/* line 129, src/assets/scss/base/_typography.scss */
+h4, .h4 {
+ color: #111111;
+ font-size: 0.875rem;
+ font-weight: 400;
+ letter-spacing: 0.05px;
+ line-height: 1.5rem;
+ text-transform: uppercase;
+}
+
+@media (min-width: 1024px) {
+ /* line 129, src/assets/scss/base/_typography.scss */
+ h4, .h4 {
+ font-size: 0.9375rem;
+ line-height: 1.5rem;
+ }
+}
+
+/* line 144, src/assets/scss/base/_typography.scss */
+h5, .h5 {
+ color: #111111;
+ font-size: 0.75rem;
+ font-weight: 700;
+ letter-spacing: normal;
+ line-height: 1.5rem;
+ text-transform: initial;
+}
+
+@media (min-width: 1024px) {
+ /* line 144, src/assets/scss/base/_typography.scss */
+ h5, .h5 {
+ font-size: 0.75rem;
+ line-height: 1.5rem;
+ }
+}
+
+/* line 159, src/assets/scss/base/_typography.scss */
+h6, .h6 {
+ color: #111111;
+ font-size: 0.6875rem;
+ font-weight: 700;
+ letter-spacing: normal;
+ line-height: 1.5rem;
+ text-transform: initial;
+}
+
+@media (min-width: 1024px) {
+ /* line 159, src/assets/scss/base/_typography.scss */
+ h6, .h6 {
+ font-size: 0.6875rem;
+ line-height: 1.5rem;
+ }
+}
+
+/**
+ * Clear inner floats
+ */
+/* line 8, src/assets/scss/base/_helpers.scss */
+.clearfix::after {
+ clear: both;
+ content: '';
+ display: table;
+}
+
+/**
+ * Hide text while making it readable for screen readers
+ * 1. Needed in WebKit-based browsers because of an implementation bug;
+ * See: https://code.google.com/p/chromium/issues/detail?id=457146
+ */
+/* line 19, src/assets/scss/base/_helpers.scss */
+.hide-text {
+ overflow: hidden;
+ padding: 0;
+ /* 1 */
+ text-indent: 101%;
+ white-space: nowrap;
+}
+
+/**
+ * Hide element while making it readable for screen readers
+ * Shamelessly borrowed from HTML5Boilerplate:
+ * https://github.com/h5bp/html5-boilerplate/blob/master/src/css/main.css#L119-L133
+ */
+/* line 31, src/assets/scss/base/_helpers.scss */
+.visually-hidden {
+ border: 0;
+ clip: rect(0 0 0 0);
+ height: 1px;
+ margin: -1px;
+ overflow: hidden;
+ padding: 0;
+ position: absolute;
+ width: 1px;
+}
+
+/* line 4, src/assets/scss/layout/_header.scss */
+.logo-container {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ -webkit-box-pack: justify;
+ -ms-flex-pack: justify;
+ justify-content: space-between;
+ padding: 20px 0;
+ margin: 10px 0 40px;
+ border-bottom: 1px solid #eaebea;
+}
+
+/* line 12, src/assets/scss/layout/_header.scss */
+.logo-container img {
+ display: block;
+ max-width: 50%;
+ height: 100%;
+}
+
+/* line 17, src/assets/scss/layout/_header.scss */
+.logo-container img:last-child {
+ max-width: 35%;
+}
+
+/* line 4, src/assets/scss/layout/_footer.scss */
+.footer {
+ border-top: 1px solid #f7f7f7;
+}
+
+/* line 1, src/assets/scss/components/_alert.scss */
+.alert {
+ border-radius: 0;
+ margin-top: 10px;
+ background: none;
+ border-width: 2px;
+ padding: 15px;
+ line-height: 1.4;
+}
+
+/* line 9, src/assets/scss/components/_alert.scss */
+.alert-danger {
+ color: #e54a19;
+}
+
+/* line 5, src/assets/scss/components/_button.scss */
+.btn {
+ border-radius: 0;
+}
+
+/* line 8, src/assets/scss/components/_button.scss */
+.btn.btn-lg, .btn-group-lg > .btn {
+ font-size: 21px;
+}
+
+/* line 2, src/assets/scss/components/_form.scss */
+form .form-group {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ margin-bottom: 8px;
+}
+
+/* line 6, src/assets/scss/components/_form.scss */
+form .form-group label {
+ width: 180px;
+ height: calc(1.5em + 0.75rem + 2px);
+ line-height: 38px;
+}
+
+/* line 12, src/assets/scss/components/_form.scss */
+form .form-group input,
+form .form-group textarea,
+form .form-group select {
+ border-radius: 0;
+}
+
+/* line 19, src/assets/scss/components/_form.scss */
+form button {
+ margin-top: 30px;
+}
+
+/* line 1, src/assets/scss/components/_statusbar.scss */
+.statusbar {
+ display: -webkit-box;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-orient: horizontal;
+ -webkit-box-direction: normal;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ list-style-type: none;
+ margin: 0 0 50px;
+ padding: 0;
+}
+
+/* line 8, src/assets/scss/components/_statusbar.scss */
+.statusbar li {
+ position: relative;
+ width: 33.33%;
+ padding: 5px 10px;
+ background: #eaebea;
+ color: #989998;
+ text-align: center;
+ text-transform: uppercase;
+ font-size: 18px;
+}
+
+/* line 18, src/assets/scss/components/_statusbar.scss */
+.statusbar li:not(:first-of-type)::before {
+ content: '';
+ border: 17px solid transparent;
+ width: 0;
+ height: 0;
+ border-top-color: #eaebea;
+ border-bottom-color: #eaebea;
+ border-right-width: 0;
+ line-height: 0;
+ position: absolute;
+ left: -17px;
+ top: 0;
+}
+
+/* line 32, src/assets/scss/components/_statusbar.scss */
+.statusbar li:not(:last-of-type) {
+ margin-right: 23px;
+}
+
+/* line 35, src/assets/scss/components/_statusbar.scss */
+.statusbar li:not(:last-of-type)::after {
+ content: '';
+ border: 17px solid transparent;
+ width: 0;
+ height: 0;
+ border-left-color: #eaebea;
+ border-right-width: 0;
+ line-height: 0;
+ position: absolute;
+ right: -17px;
+ top: 0;
+}
+
+/* line 49, src/assets/scss/components/_statusbar.scss */
+.statusbar li.active {
+ background: #e54a19;
+ color: #ffffff;
+}
+
+/* line 53, src/assets/scss/components/_statusbar.scss */
+.statusbar li.active::before {
+ border-top-color: #e54a19;
+ border-bottom-color: #e54a19;
+}
+
+/* line 58, src/assets/scss/components/_statusbar.scss */
+.statusbar li.active::after {
+ border-left-color: #e54a19;
+}
+
+/* line 5, src/assets/scss/pages/_home.scss */
+.container {
+ max-width: 800px;
+}
+
+/* line 9, src/assets/scss/pages/_home.scss */
+h2 {
+ font-size: 40px;
+ line-height: 1.2;
+}
+
+/* line 13, src/assets/scss/pages/_home.scss */
+h3 {
+ font-size: 26px;
+}
+
+/* line 16, src/assets/scss/pages/_home.scss */
+h2,
+h3 {
+ color: #464746;
+}
+
+/* line 21, src/assets/scss/pages/_home.scss */
+a {
+ color: #6fa7fd;
+}
+
+/* line 25, src/assets/scss/pages/_home.scss */
+p {
+ margin-bottom: 23px;
+}
+
+/* line 29, src/assets/scss/pages/_home.scss */
+.accounts {
+ list-style-type: none;
+ margin: 0 0 60px;
+ padding: 0;
+}
+
+/* line 34, src/assets/scss/pages/_home.scss */
+.accounts li {
+ margin-bottom: 30px;
+ color: #e54a19;
+ font-size: 21px;
+}
+
+/* line 39, src/assets/scss/pages/_home.scss */
+.accounts li .status {
+ background: #eaebea;
+ color: #111111;
+ padding: 3px 5px;
+ margin-left: 10px;
+ font-size: 16px;
+}
+
+/* line 46, src/assets/scss/pages/_home.scss */
+.accounts li .status.online {
+ background: #e54a19;
+ color: #ffffff;
+}
+
+/* line 51, src/assets/scss/pages/_home.scss */
+.accounts li label {
+ position: relative;
+ background: #eaebea;
+ color: #eaebea;
+ border-color: #464746;
+ font-size: 16px;
+ padding: 4px 5px;
+ margin: 0;
+ line-height: 1;
+ cursor: pointer;
+}
+
+/* line 62, src/assets/scss/pages/_home.scss */
+.accounts li label::before, .accounts li label::after {
+ content: '';
+ position: absolute;
+ border: 2px solid transparent;
+ border-top-color: inherit;
+ border-left-color: inherit;
+ height: 10px;
+ width: 10px;
+ top: 50%;
+ margin-top: -5px;
+}
+
+/* line 74, src/assets/scss/pages/_home.scss */
+.accounts li label::before {
+ -webkit-transform: rotate(-45deg);
+ transform: rotate(-45deg);
+ left: 8px;
+}
+
+/* line 78, src/assets/scss/pages/_home.scss */
+.accounts li label::after {
+ -webkit-transform: rotate(135deg);
+ transform: rotate(135deg);
+ right: 8px;
+}
+
+/* line 83, src/assets/scss/pages/_home.scss */
+.accounts li .trigger {
+ display: none;
+}
+
+/* line 86, src/assets/scss/pages/_home.scss */
+.accounts li .trigger:checked + label {
+ background: #464746;
+ color: #464746;
+ border-color: #ffffff;
+}
+
+/* line 91, src/assets/scss/pages/_home.scss */
+.accounts li .trigger:checked + label + .things {
+ display: block;
+}
+
+/* line 96, src/assets/scss/pages/_home.scss */
+.accounts li .things {
+ display: none;
+}
+
+/* line 102, src/assets/scss/pages/_home.scss */
+.things {
+ margin-top: 25px;
+}
+
+/* line 105, src/assets/scss/pages/_home.scss */
+.things .legend {
+ display: block;
+ color: #111111;
+ margin: 5px 0 10px;
+ font-size: 16px;
+}
+
+/* line 111, src/assets/scss/pages/_home.scss */
+.things .code-container {
+ position: relative;
+}
+
+/* line 114, src/assets/scss/pages/_home.scss */
+.things .code-container textarea {
+ background: #eaebea;
+ color: #464746;
+ border: none;
+ padding: 25px 20px;
+ width: 100%;
+ height: 200px;
+ font-family: SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;
+ font-size: .8rem;
+ white-space: pre;
+}
+
+/* line 126, src/assets/scss/pages/_home.scss */
+.things .code-container .copy {
+ position: absolute;
+ top: 5px;
+ right: 5px;
+ border: 0;
+ color: #989998;
+}
+
+/* line 133, src/assets/scss/pages/_home.scss */
+.things .code-container .copy:hover, .things .code-container .copy:focus, .things .code-container .copy:active {
+ color: #fff;
+}
+
+/* line 143, src/assets/scss/pages/_home.scss */
+.accounts + .alert-danger::after {
+ content: '';
+ position: absolute;
+ left: 65px;
+ bottom: -8px;
+ height: 16px;
+ width: 16px;
+ background: white;
+ border: inherit;
+ border-top-color: transparent;
+ border-left-color: transparent;
+ -webkit-transform: rotate(45deg);
+ transform: rotate(45deg);
+}
+
+/* line 158, src/assets/scss/pages/_home.scss */
+.controls {
+ margin-top: 25px;
+}
+
+/*# sourceMappingURL=../../../scss */
\ No newline at end of file
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/css/rtl.css b/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/css/rtl.css
new file mode 100644
index 00000000000..f1e048e4430
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/css/rtl.css
@@ -0,0 +1,10 @@
+/* line 5, src/assets/scss/rtl.scss */
+.jumbotron {
+ direction: ltr;
+ text-align: left;
+ margin: 0 2em 0 1em;
+ padding-right: 1em;
+ margin: 0 !important;
+}
+
+/*# sourceMappingURL=../../../scss */
\ No newline at end of file
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/img/OpenHAB_logo.svg b/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/img/OpenHAB_logo.svg
new file mode 100644
index 00000000000..0c8c6eb8ae1
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/img/OpenHAB_logo.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/img/favicon.ico b/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/img/favicon.ico
new file mode 100644
index 00000000000..070fbb11c94
Binary files /dev/null and b/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/img/favicon.ico differ
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/img/miele.png b/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/img/miele.png
new file mode 100644
index 00000000000..e53337966e6
Binary files /dev/null and b/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/img/miele.png differ
diff --git a/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/js/main.js b/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/js/main.js
new file mode 100644
index 00000000000..df9dac3c6cb
--- /dev/null
+++ b/bundles/org.openhab.binding.mielecloud/src/main/resources/org/openhab/binding/mielecloud/internal/config/assets/js/main.js
@@ -0,0 +1,17768 @@
+/*!
+ * jQuery JavaScript Library v3.4.1
+ * https://jquery.com/
+ *
+ * Includes Sizzle.js
+ * https://sizzlejs.com/
+ *
+ * Copyright JS Foundation and other contributors
+ * Released under the MIT license
+ * https://jquery.org/license
+ *
+ * Date: 2019-05-01T21:04Z
+ */
+( function( global, factory ) {
+
+ "use strict";
+
+ if ( typeof module === "object" && typeof module.exports === "object" ) {
+
+ // For CommonJS and CommonJS-like environments where a proper `window`
+ // is present, execute the factory and get jQuery.
+ // For environments that do not have a `window` with a `document`
+ // (such as Node.js), expose a factory as module.exports.
+ // This accentuates the need for the creation of a real `window`.
+ // e.g. var jQuery = require("jquery")(window);
+ // See ticket #14549 for more info.
+ module.exports = global.document ?
+ factory( global, true ) :
+ function( w ) {
+ if ( !w.document ) {
+ throw new Error( "jQuery requires a window with a document" );
+ }
+ return factory( w );
+ };
+ } else {
+ factory( global );
+ }
+
+// Pass this if window is not defined yet
+} )( typeof window !== "undefined" ? window : this, function( window, noGlobal ) {
+
+// Edge <= 12 - 13+, Firefox <=18 - 45+, IE 10 - 11, Safari 5.1 - 9+, iOS 6 - 9.1
+// throw exceptions when non-strict code (e.g., ASP.NET 4.5) accesses strict mode
+// arguments.callee.caller (trac-13335). But as of jQuery 3.0 (2016), strict mode should be common
+// enough that all such attempts are guarded in a try block.
+"use strict";
+
+var arr = [];
+
+var document = window.document;
+
+var getProto = Object.getPrototypeOf;
+
+var slice = arr.slice;
+
+var concat = arr.concat;
+
+var push = arr.push;
+
+var indexOf = arr.indexOf;
+
+var class2type = {};
+
+var toString = class2type.toString;
+
+var hasOwn = class2type.hasOwnProperty;
+
+var fnToString = hasOwn.toString;
+
+var ObjectFunctionString = fnToString.call( Object );
+
+var support = {};
+
+var isFunction = function isFunction( obj ) {
+
+ // Support: Chrome <=57, Firefox <=52
+ // In some browsers, typeof returns "function" for HTML elements
+ // (i.e., `typeof document.createElement( "object" ) === "function"`).
+ // We don't want to classify *any* DOM node as a function.
+ return typeof obj === "function" && typeof obj.nodeType !== "number";
+ };
+
+
+var isWindow = function isWindow( obj ) {
+ return obj != null && obj === obj.window;
+ };
+
+
+
+
+ var preservedScriptAttributes = {
+ type: true,
+ src: true,
+ nonce: true,
+ noModule: true
+ };
+
+ function DOMEval( code, node, doc ) {
+ doc = doc || document;
+
+ var i, val,
+ script = doc.createElement( "script" );
+
+ script.text = code;
+ if ( node ) {
+ for ( i in preservedScriptAttributes ) {
+
+ // Support: Firefox 64+, Edge 18+
+ // Some browsers don't support the "nonce" property on scripts.
+ // On the other hand, just using `getAttribute` is not enough as
+ // the `nonce` attribute is reset to an empty string whenever it
+ // becomes browsing-context connected.
+ // See https://github.com/whatwg/html/issues/2369
+ // See https://html.spec.whatwg.org/#nonce-attributes
+ // The `node.getAttribute` check was added for the sake of
+ // `jQuery.globalEval` so that it can fake a nonce-containing node
+ // via an object.
+ val = node[ i ] || node.getAttribute && node.getAttribute( i );
+ if ( val ) {
+ script.setAttribute( i, val );
+ }
+ }
+ }
+ doc.head.appendChild( script ).parentNode.removeChild( script );
+ }
+
+
+function toType( obj ) {
+ if ( obj == null ) {
+ return obj + "";
+ }
+
+ // Support: Android <=2.3 only (functionish RegExp)
+ return typeof obj === "object" || typeof obj === "function" ?
+ class2type[ toString.call( obj ) ] || "object" :
+ typeof obj;
+}
+/* global Symbol */
+// Defining this global in .eslintrc.json would create a danger of using the global
+// unguarded in another place, it seems safer to define global only for this module
+
+
+
+var
+ version = "3.4.1",
+
+ // Define a local copy of jQuery
+ jQuery = function( selector, context ) {
+
+ // The jQuery object is actually just the init constructor 'enhanced'
+ // Need init if jQuery is called (just allow error to be thrown if not included)
+ return new jQuery.fn.init( selector, context );
+ },
+
+ // Support: Android <=4.0 only
+ // Make sure we trim BOM and NBSP
+ rtrim = /^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g;
+
+jQuery.fn = jQuery.prototype = {
+
+ // The current version of jQuery being used
+ jquery: version,
+
+ constructor: jQuery,
+
+ // The default length of a jQuery object is 0
+ length: 0,
+
+ toArray: function() {
+ return slice.call( this );
+ },
+
+ // Get the Nth element in the matched element set OR
+ // Get the whole matched element set as a clean array
+ get: function( num ) {
+
+ // Return all the elements in a clean array
+ if ( num == null ) {
+ return slice.call( this );
+ }
+
+ // Return just the one element from the set
+ return num < 0 ? this[ num + this.length ] : this[ num ];
+ },
+
+ // Take an array of elements and push it onto the stack
+ // (returning the new matched element set)
+ pushStack: function( elems ) {
+
+ // Build a new jQuery matched element set
+ var ret = jQuery.merge( this.constructor(), elems );
+
+ // Add the old object onto the stack (as a reference)
+ ret.prevObject = this;
+
+ // Return the newly-formed element set
+ return ret;
+ },
+
+ // Execute a callback for every element in the matched set.
+ each: function( callback ) {
+ return jQuery.each( this, callback );
+ },
+
+ map: function( callback ) {
+ return this.pushStack( jQuery.map( this, function( elem, i ) {
+ return callback.call( elem, i, elem );
+ } ) );
+ },
+
+ slice: function() {
+ return this.pushStack( slice.apply( this, arguments ) );
+ },
+
+ first: function() {
+ return this.eq( 0 );
+ },
+
+ last: function() {
+ return this.eq( -1 );
+ },
+
+ eq: function( i ) {
+ var len = this.length,
+ j = +i + ( i < 0 ? len : 0 );
+ return this.pushStack( j >= 0 && j < len ? [ this[ j ] ] : [] );
+ },
+
+ end: function() {
+ return this.prevObject || this.constructor();
+ },
+
+ // For internal use only.
+ // Behaves like an Array's method, not like a jQuery method.
+ push: push,
+ sort: arr.sort,
+ splice: arr.splice
+};
+
+jQuery.extend = jQuery.fn.extend = function() {
+ var options, name, src, copy, copyIsArray, clone,
+ target = arguments[ 0 ] || {},
+ i = 1,
+ length = arguments.length,
+ deep = false;
+
+ // Handle a deep copy situation
+ if ( typeof target === "boolean" ) {
+ deep = target;
+
+ // Skip the boolean and the target
+ target = arguments[ i ] || {};
+ i++;
+ }
+
+ // Handle case when target is a string or something (possible in deep copy)
+ if ( typeof target !== "object" && !isFunction( target ) ) {
+ target = {};
+ }
+
+ // Extend jQuery itself if only one argument is passed
+ if ( i === length ) {
+ target = this;
+ i--;
+ }
+
+ for ( ; i < length; i++ ) {
+
+ // Only deal with non-null/undefined values
+ if ( ( options = arguments[ i ] ) != null ) {
+
+ // Extend the base object
+ for ( name in options ) {
+ copy = options[ name ];
+
+ // Prevent Object.prototype pollution
+ // Prevent never-ending loop
+ if ( name === "__proto__" || target === copy ) {
+ continue;
+ }
+
+ // Recurse if we're merging plain objects or arrays
+ if ( deep && copy && ( jQuery.isPlainObject( copy ) ||
+ ( copyIsArray = Array.isArray( copy ) ) ) ) {
+ src = target[ name ];
+
+ // Ensure proper type for the source value
+ if ( copyIsArray && !Array.isArray( src ) ) {
+ clone = [];
+ } else if ( !copyIsArray && !jQuery.isPlainObject( src ) ) {
+ clone = {};
+ } else {
+ clone = src;
+ }
+ copyIsArray = false;
+
+ // Never move original objects, clone them
+ target[ name ] = jQuery.extend( deep, clone, copy );
+
+ // Don't bring in undefined values
+ } else if ( copy !== undefined ) {
+ target[ name ] = copy;
+ }
+ }
+ }
+ }
+
+ // Return the modified object
+ return target;
+};
+
+jQuery.extend( {
+
+ // Unique for each copy of jQuery on the page
+ expando: "jQuery" + ( version + Math.random() ).replace( /\D/g, "" ),
+
+ // Assume jQuery is ready without the ready module
+ isReady: true,
+
+ error: function( msg ) {
+ throw new Error( msg );
+ },
+
+ noop: function() {},
+
+ isPlainObject: function( obj ) {
+ var proto, Ctor;
+
+ // Detect obvious negatives
+ // Use toString instead of jQuery.type to catch host objects
+ if ( !obj || toString.call( obj ) !== "[object Object]" ) {
+ return false;
+ }
+
+ proto = getProto( obj );
+
+ // Objects with no prototype (e.g., `Object.create( null )`) are plain
+ if ( !proto ) {
+ return true;
+ }
+
+ // Objects with prototype are plain iff they were constructed by a global Object function
+ Ctor = hasOwn.call( proto, "constructor" ) && proto.constructor;
+ return typeof Ctor === "function" && fnToString.call( Ctor ) === ObjectFunctionString;
+ },
+
+ isEmptyObject: function( obj ) {
+ var name;
+
+ for ( name in obj ) {
+ return false;
+ }
+ return true;
+ },
+
+ // Evaluates a script in a global context
+ globalEval: function( code, options ) {
+ DOMEval( code, { nonce: options && options.nonce } );
+ },
+
+ each: function( obj, callback ) {
+ var length, i = 0;
+
+ if ( isArrayLike( obj ) ) {
+ length = obj.length;
+ for ( ; i < length; i++ ) {
+ if ( callback.call( obj[ i ], i, obj[ i ] ) === false ) {
+ break;
+ }
+ }
+ } else {
+ for ( i in obj ) {
+ if ( callback.call( obj[ i ], i, obj[ i ] ) === false ) {
+ break;
+ }
+ }
+ }
+
+ return obj;
+ },
+
+ // Support: Android <=4.0 only
+ trim: function( text ) {
+ return text == null ?
+ "" :
+ ( text + "" ).replace( rtrim, "" );
+ },
+
+ // results is for internal usage only
+ makeArray: function( arr, results ) {
+ var ret = results || [];
+
+ if ( arr != null ) {
+ if ( isArrayLike( Object( arr ) ) ) {
+ jQuery.merge( ret,
+ typeof arr === "string" ?
+ [ arr ] : arr
+ );
+ } else {
+ push.call( ret, arr );
+ }
+ }
+
+ return ret;
+ },
+
+ inArray: function( elem, arr, i ) {
+ return arr == null ? -1 : indexOf.call( arr, elem, i );
+ },
+
+ // Support: Android <=4.0 only, PhantomJS 1 only
+ // push.apply(_, arraylike) throws on ancient WebKit
+ merge: function( first, second ) {
+ var len = +second.length,
+ j = 0,
+ i = first.length;
+
+ for ( ; j < len; j++ ) {
+ first[ i++ ] = second[ j ];
+ }
+
+ first.length = i;
+
+ return first;
+ },
+
+ grep: function( elems, callback, invert ) {
+ var callbackInverse,
+ matches = [],
+ i = 0,
+ length = elems.length,
+ callbackExpect = !invert;
+
+ // Go through the array, only saving the items
+ // that pass the validator function
+ for ( ; i < length; i++ ) {
+ callbackInverse = !callback( elems[ i ], i );
+ if ( callbackInverse !== callbackExpect ) {
+ matches.push( elems[ i ] );
+ }
+ }
+
+ return matches;
+ },
+
+ // arg is for internal usage only
+ map: function( elems, callback, arg ) {
+ var length, value,
+ i = 0,
+ ret = [];
+
+ // Go through the array, translating each of the items to their new values
+ if ( isArrayLike( elems ) ) {
+ length = elems.length;
+ for ( ; i < length; i++ ) {
+ value = callback( elems[ i ], i, arg );
+
+ if ( value != null ) {
+ ret.push( value );
+ }
+ }
+
+ // Go through every key on the object,
+ } else {
+ for ( i in elems ) {
+ value = callback( elems[ i ], i, arg );
+
+ if ( value != null ) {
+ ret.push( value );
+ }
+ }
+ }
+
+ // Flatten any nested arrays
+ return concat.apply( [], ret );
+ },
+
+ // A global GUID counter for objects
+ guid: 1,
+
+ // jQuery.support is not used in Core but other projects attach their
+ // properties to it so it needs to exist.
+ support: support
+} );
+
+if ( typeof Symbol === "function" ) {
+ jQuery.fn[ Symbol.iterator ] = arr[ Symbol.iterator ];
+}
+
+// Populate the class2type map
+jQuery.each( "Boolean Number String Function Array Date RegExp Object Error Symbol".split( " " ),
+function( i, name ) {
+ class2type[ "[object " + name + "]" ] = name.toLowerCase();
+} );
+
+function isArrayLike( obj ) {
+
+ // Support: real iOS 8.2 only (not reproducible in simulator)
+ // `in` check used to prevent JIT error (gh-2145)
+ // hasOwn isn't used here due to false negatives
+ // regarding Nodelist length in IE
+ var length = !!obj && "length" in obj && obj.length,
+ type = toType( obj );
+
+ if ( isFunction( obj ) || isWindow( obj ) ) {
+ return false;
+ }
+
+ return type === "array" || length === 0 ||
+ typeof length === "number" && length > 0 && ( length - 1 ) in obj;
+}
+var Sizzle =
+/*!
+ * Sizzle CSS Selector Engine v2.3.4
+ * https://sizzlejs.com/
+ *
+ * Copyright JS Foundation and other contributors
+ * Released under the MIT license
+ * https://js.foundation/
+ *
+ * Date: 2019-04-08
+ */
+(function( window ) {
+
+var i,
+ support,
+ Expr,
+ getText,
+ isXML,
+ tokenize,
+ compile,
+ select,
+ outermostContext,
+ sortInput,
+ hasDuplicate,
+
+ // Local document vars
+ setDocument,
+ document,
+ docElem,
+ documentIsHTML,
+ rbuggyQSA,
+ rbuggyMatches,
+ matches,
+ contains,
+
+ // Instance-specific data
+ expando = "sizzle" + 1 * new Date(),
+ preferredDoc = window.document,
+ dirruns = 0,
+ done = 0,
+ classCache = createCache(),
+ tokenCache = createCache(),
+ compilerCache = createCache(),
+ nonnativeSelectorCache = createCache(),
+ sortOrder = function( a, b ) {
+ if ( a === b ) {
+ hasDuplicate = true;
+ }
+ return 0;
+ },
+
+ // Instance methods
+ hasOwn = ({}).hasOwnProperty,
+ arr = [],
+ pop = arr.pop,
+ push_native = arr.push,
+ push = arr.push,
+ slice = arr.slice,
+ // Use a stripped-down indexOf as it's faster than native
+ // https://jsperf.com/thor-indexof-vs-for/5
+ indexOf = function( list, elem ) {
+ var i = 0,
+ len = list.length;
+ for ( ; i < len; i++ ) {
+ if ( list[i] === elem ) {
+ return i;
+ }
+ }
+ return -1;
+ },
+
+ booleans = "checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped",
+
+ // Regular expressions
+
+ // http://www.w3.org/TR/css3-selectors/#whitespace
+ whitespace = "[\\x20\\t\\r\\n\\f]",
+
+ // http://www.w3.org/TR/CSS21/syndata.html#value-def-identifier
+ identifier = "(?:\\\\.|[\\w-]|[^\0-\\xa0])+",
+
+ // Attribute selectors: http://www.w3.org/TR/selectors/#attribute-selectors
+ attributes = "\\[" + whitespace + "*(" + identifier + ")(?:" + whitespace +
+ // Operator (capture 2)
+ "*([*^$|!~]?=)" + whitespace +
+ // "Attribute values must be CSS identifiers [capture 5] or strings [capture 3 or capture 4]"
+ "*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|(" + identifier + "))|)" + whitespace +
+ "*\\]",
+
+ pseudos = ":(" + identifier + ")(?:\\((" +
+ // To reduce the number of selectors needing tokenize in the preFilter, prefer arguments:
+ // 1. quoted (capture 3; capture 4 or capture 5)
+ "('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|" +
+ // 2. simple (capture 6)
+ "((?:\\\\.|[^\\\\()[\\]]|" + attributes + ")*)|" +
+ // 3. anything else (capture 2)
+ ".*" +
+ ")\\)|)",
+
+ // Leading and non-escaped trailing whitespace, capturing some non-whitespace characters preceding the latter
+ rwhitespace = new RegExp( whitespace + "+", "g" ),
+ rtrim = new RegExp( "^" + whitespace + "+|((?:^|[^\\\\])(?:\\\\.)*)" + whitespace + "+$", "g" ),
+
+ rcomma = new RegExp( "^" + whitespace + "*," + whitespace + "*" ),
+ rcombinators = new RegExp( "^" + whitespace + "*([>+~]|" + whitespace + ")" + whitespace + "*" ),
+ rdescend = new RegExp( whitespace + "|>" ),
+
+ rpseudo = new RegExp( pseudos ),
+ ridentifier = new RegExp( "^" + identifier + "$" ),
+
+ matchExpr = {
+ "ID": new RegExp( "^#(" + identifier + ")" ),
+ "CLASS": new RegExp( "^\\.(" + identifier + ")" ),
+ "TAG": new RegExp( "^(" + identifier + "|[*])" ),
+ "ATTR": new RegExp( "^" + attributes ),
+ "PSEUDO": new RegExp( "^" + pseudos ),
+ "CHILD": new RegExp( "^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\(" + whitespace +
+ "*(even|odd|(([+-]|)(\\d*)n|)" + whitespace + "*(?:([+-]|)" + whitespace +
+ "*(\\d+)|))" + whitespace + "*\\)|)", "i" ),
+ "bool": new RegExp( "^(?:" + booleans + ")$", "i" ),
+ // For use in libraries implementing .is()
+ // We use this for POS matching in `select`
+ "needsContext": new RegExp( "^" + whitespace + "*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\(" +
+ whitespace + "*((?:-\\d)?\\d*)" + whitespace + "*\\)|)(?=[^-]|$)", "i" )
+ },
+
+ rhtml = /HTML$/i,
+ rinputs = /^(?:input|select|textarea|button)$/i,
+ rheader = /^h\d$/i,
+
+ rnative = /^[^{]+\{\s*\[native \w/,
+
+ // Easily-parseable/retrievable ID or TAG or CLASS selectors
+ rquickExpr = /^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,
+
+ rsibling = /[+~]/,
+
+ // CSS escapes
+ // http://www.w3.org/TR/CSS21/syndata.html#escaped-characters
+ runescape = new RegExp( "\\\\([\\da-f]{1,6}" + whitespace + "?|(" + whitespace + ")|.)", "ig" ),
+ funescape = function( _, escaped, escapedWhitespace ) {
+ var high = "0x" + escaped - 0x10000;
+ // NaN means non-codepoint
+ // Support: Firefox<24
+ // Workaround erroneous numeric interpretation of +"0x"
+ return high !== high || escapedWhitespace ?
+ escaped :
+ high < 0 ?
+ // BMP codepoint
+ String.fromCharCode( high + 0x10000 ) :
+ // Supplemental Plane codepoint (surrogate pair)
+ String.fromCharCode( high >> 10 | 0xD800, high & 0x3FF | 0xDC00 );
+ },
+
+ // CSS string/identifier serialization
+ // https://drafts.csswg.org/cssom/#common-serializing-idioms
+ rcssescape = /([\0-\x1f\x7f]|^-?\d)|^-$|[^\0-\x1f\x7f-\uFFFF\w-]/g,
+ fcssescape = function( ch, asCodePoint ) {
+ if ( asCodePoint ) {
+
+ // U+0000 NULL becomes U+FFFD REPLACEMENT CHARACTER
+ if ( ch === "\0" ) {
+ return "\uFFFD";
+ }
+
+ // Control characters and (dependent upon position) numbers get escaped as code points
+ return ch.slice( 0, -1 ) + "\\" + ch.charCodeAt( ch.length - 1 ).toString( 16 ) + " ";
+ }
+
+ // Other potentially-special ASCII characters get backslash-escaped
+ return "\\" + ch;
+ },
+
+ // Used for iframes
+ // See setDocument()
+ // Removing the function wrapper causes a "Permission Denied"
+ // error in IE
+ unloadHandler = function() {
+ setDocument();
+ },
+
+ inDisabledFieldset = addCombinator(
+ function( elem ) {
+ return elem.disabled === true && elem.nodeName.toLowerCase() === "fieldset";
+ },
+ { dir: "parentNode", next: "legend" }
+ );
+
+// Optimize for push.apply( _, NodeList )
+try {
+ push.apply(
+ (arr = slice.call( preferredDoc.childNodes )),
+ preferredDoc.childNodes
+ );
+ // Support: Android<4.0
+ // Detect silently failing push.apply
+ arr[ preferredDoc.childNodes.length ].nodeType;
+} catch ( e ) {
+ push = { apply: arr.length ?
+
+ // Leverage slice if possible
+ function( target, els ) {
+ push_native.apply( target, slice.call(els) );
+ } :
+
+ // Support: IE<9
+ // Otherwise append directly
+ function( target, els ) {
+ var j = target.length,
+ i = 0;
+ // Can't trust NodeList.length
+ while ( (target[j++] = els[i++]) ) {}
+ target.length = j - 1;
+ }
+ };
+}
+
+function Sizzle( selector, context, results, seed ) {
+ var m, i, elem, nid, match, groups, newSelector,
+ newContext = context && context.ownerDocument,
+
+ // nodeType defaults to 9, since context defaults to document
+ nodeType = context ? context.nodeType : 9;
+
+ results = results || [];
+
+ // Return early from calls with invalid selector or context
+ if ( typeof selector !== "string" || !selector ||
+ nodeType !== 1 && nodeType !== 9 && nodeType !== 11 ) {
+
+ return results;
+ }
+
+ // Try to shortcut find operations (as opposed to filters) in HTML documents
+ if ( !seed ) {
+
+ if ( ( context ? context.ownerDocument || context : preferredDoc ) !== document ) {
+ setDocument( context );
+ }
+ context = context || document;
+
+ if ( documentIsHTML ) {
+
+ // If the selector is sufficiently simple, try using a "get*By*" DOM method
+ // (excepting DocumentFragment context, where the methods don't exist)
+ if ( nodeType !== 11 && (match = rquickExpr.exec( selector )) ) {
+
+ // ID selector
+ if ( (m = match[1]) ) {
+
+ // Document context
+ if ( nodeType === 9 ) {
+ if ( (elem = context.getElementById( m )) ) {
+
+ // Support: IE, Opera, Webkit
+ // TODO: identify versions
+ // getElementById can match elements by name instead of ID
+ if ( elem.id === m ) {
+ results.push( elem );
+ return results;
+ }
+ } else {
+ return results;
+ }
+
+ // Element context
+ } else {
+
+ // Support: IE, Opera, Webkit
+ // TODO: identify versions
+ // getElementById can match elements by name instead of ID
+ if ( newContext && (elem = newContext.getElementById( m )) &&
+ contains( context, elem ) &&
+ elem.id === m ) {
+
+ results.push( elem );
+ return results;
+ }
+ }
+
+ // Type selector
+ } else if ( match[2] ) {
+ push.apply( results, context.getElementsByTagName( selector ) );
+ return results;
+
+ // Class selector
+ } else if ( (m = match[3]) && support.getElementsByClassName &&
+ context.getElementsByClassName ) {
+
+ push.apply( results, context.getElementsByClassName( m ) );
+ return results;
+ }
+ }
+
+ // Take advantage of querySelectorAll
+ if ( support.qsa &&
+ !nonnativeSelectorCache[ selector + " " ] &&
+ (!rbuggyQSA || !rbuggyQSA.test( selector )) &&
+
+ // Support: IE 8 only
+ // Exclude object elements
+ (nodeType !== 1 || context.nodeName.toLowerCase() !== "object") ) {
+
+ newSelector = selector;
+ newContext = context;
+
+ // qSA considers elements outside a scoping root when evaluating child or
+ // descendant combinators, which is not what we want.
+ // In such cases, we work around the behavior by prefixing every selector in the
+ // list with an ID selector referencing the scope context.
+ // Thanks to Andrew Dupont for this technique.
+ if ( nodeType === 1 && rdescend.test( selector ) ) {
+
+ // Capture the context ID, setting it first if necessary
+ if ( (nid = context.getAttribute( "id" )) ) {
+ nid = nid.replace( rcssescape, fcssescape );
+ } else {
+ context.setAttribute( "id", (nid = expando) );
+ }
+
+ // Prefix every selector in the list
+ groups = tokenize( selector );
+ i = groups.length;
+ while ( i-- ) {
+ groups[i] = "#" + nid + " " + toSelector( groups[i] );
+ }
+ newSelector = groups.join( "," );
+
+ // Expand context for sibling selectors
+ newContext = rsibling.test( selector ) && testContext( context.parentNode ) ||
+ context;
+ }
+
+ try {
+ push.apply( results,
+ newContext.querySelectorAll( newSelector )
+ );
+ return results;
+ } catch ( qsaError ) {
+ nonnativeSelectorCache( selector, true );
+ } finally {
+ if ( nid === expando ) {
+ context.removeAttribute( "id" );
+ }
+ }
+ }
+ }
+ }
+
+ // All others
+ return select( selector.replace( rtrim, "$1" ), context, results, seed );
+}
+
+/**
+ * Create key-value caches of limited size
+ * @returns {function(string, object)} Returns the Object data after storing it on itself with
+ * property name the (space-suffixed) string and (if the cache is larger than Expr.cacheLength)
+ * deleting the oldest entry
+ */
+function createCache() {
+ var keys = [];
+
+ function cache( key, value ) {
+ // Use (key + " ") to avoid collision with native prototype properties (see Issue #157)
+ if ( keys.push( key + " " ) > Expr.cacheLength ) {
+ // Only keep the most recent entries
+ delete cache[ keys.shift() ];
+ }
+ return (cache[ key + " " ] = value);
+ }
+ return cache;
+}
+
+/**
+ * Mark a function for special use by Sizzle
+ * @param {Function} fn The function to mark
+ */
+function markFunction( fn ) {
+ fn[ expando ] = true;
+ return fn;
+}
+
+/**
+ * Support testing using an element
+ * @param {Function} fn Passed the created element and returns a boolean result
+ */
+function assert( fn ) {
+ var el = document.createElement("fieldset");
+
+ try {
+ return !!fn( el );
+ } catch (e) {
+ return false;
+ } finally {
+ // Remove from its parent by default
+ if ( el.parentNode ) {
+ el.parentNode.removeChild( el );
+ }
+ // release memory in IE
+ el = null;
+ }
+}
+
+/**
+ * Adds the same handler for all of the specified attrs
+ * @param {String} attrs Pipe-separated list of attributes
+ * @param {Function} handler The method that will be applied
+ */
+function addHandle( attrs, handler ) {
+ var arr = attrs.split("|"),
+ i = arr.length;
+
+ while ( i-- ) {
+ Expr.attrHandle[ arr[i] ] = handler;
+ }
+}
+
+/**
+ * Checks document order of two siblings
+ * @param {Element} a
+ * @param {Element} b
+ * @returns {Number} Returns less than 0 if a precedes b, greater than 0 if a follows b
+ */
+function siblingCheck( a, b ) {
+ var cur = b && a,
+ diff = cur && a.nodeType === 1 && b.nodeType === 1 &&
+ a.sourceIndex - b.sourceIndex;
+
+ // Use IE sourceIndex if available on both nodes
+ if ( diff ) {
+ return diff;
+ }
+
+ // Check if b follows a
+ if ( cur ) {
+ while ( (cur = cur.nextSibling) ) {
+ if ( cur === b ) {
+ return -1;
+ }
+ }
+ }
+
+ return a ? 1 : -1;
+}
+
+/**
+ * Returns a function to use in pseudos for input types
+ * @param {String} type
+ */
+function createInputPseudo( type ) {
+ return function( elem ) {
+ var name = elem.nodeName.toLowerCase();
+ return name === "input" && elem.type === type;
+ };
+}
+
+/**
+ * Returns a function to use in pseudos for buttons
+ * @param {String} type
+ */
+function createButtonPseudo( type ) {
+ return function( elem ) {
+ var name = elem.nodeName.toLowerCase();
+ return (name === "input" || name === "button") && elem.type === type;
+ };
+}
+
+/**
+ * Returns a function to use in pseudos for :enabled/:disabled
+ * @param {Boolean} disabled true for :disabled; false for :enabled
+ */
+function createDisabledPseudo( disabled ) {
+
+ // Known :disabled false positives: fieldset[disabled] > legend:nth-of-type(n+2) :can-disable
+ return function( elem ) {
+
+ // Only certain elements can match :enabled or :disabled
+ // https://html.spec.whatwg.org/multipage/scripting.html#selector-enabled
+ // https://html.spec.whatwg.org/multipage/scripting.html#selector-disabled
+ if ( "form" in elem ) {
+
+ // Check for inherited disabledness on relevant non-disabled elements:
+ // * listed form-associated elements in a disabled fieldset
+ // https://html.spec.whatwg.org/multipage/forms.html#category-listed
+ // https://html.spec.whatwg.org/multipage/forms.html#concept-fe-disabled
+ // * option elements in a disabled optgroup
+ // https://html.spec.whatwg.org/multipage/forms.html#concept-option-disabled
+ // All such elements have a "form" property.
+ if ( elem.parentNode && elem.disabled === false ) {
+
+ // Option elements defer to a parent optgroup if present
+ if ( "label" in elem ) {
+ if ( "label" in elem.parentNode ) {
+ return elem.parentNode.disabled === disabled;
+ } else {
+ return elem.disabled === disabled;
+ }
+ }
+
+ // Support: IE 6 - 11
+ // Use the isDisabled shortcut property to check for disabled fieldset ancestors
+ return elem.isDisabled === disabled ||
+
+ // Where there is no isDisabled, check manually
+ /* jshint -W018 */
+ elem.isDisabled !== !disabled &&
+ inDisabledFieldset( elem ) === disabled;
+ }
+
+ return elem.disabled === disabled;
+
+ // Try to winnow out elements that can't be disabled before trusting the disabled property.
+ // Some victims get caught in our net (label, legend, menu, track), but it shouldn't
+ // even exist on them, let alone have a boolean value.
+ } else if ( "label" in elem ) {
+ return elem.disabled === disabled;
+ }
+
+ // Remaining elements are neither :enabled nor :disabled
+ return false;
+ };
+}
+
+/**
+ * Returns a function to use in pseudos for positionals
+ * @param {Function} fn
+ */
+function createPositionalPseudo( fn ) {
+ return markFunction(function( argument ) {
+ argument = +argument;
+ return markFunction(function( seed, matches ) {
+ var j,
+ matchIndexes = fn( [], seed.length, argument ),
+ i = matchIndexes.length;
+
+ // Match elements found at the specified indexes
+ while ( i-- ) {
+ if ( seed[ (j = matchIndexes[i]) ] ) {
+ seed[j] = !(matches[j] = seed[j]);
+ }
+ }
+ });
+ });
+}
+
+/**
+ * Checks a node for validity as a Sizzle context
+ * @param {Element|Object=} context
+ * @returns {Element|Object|Boolean} The input node if acceptable, otherwise a falsy value
+ */
+function testContext( context ) {
+ return context && typeof context.getElementsByTagName !== "undefined" && context;
+}
+
+// Expose support vars for convenience
+support = Sizzle.support = {};
+
+/**
+ * Detects XML nodes
+ * @param {Element|Object} elem An element or a document
+ * @returns {Boolean} True iff elem is a non-HTML XML node
+ */
+isXML = Sizzle.isXML = function( elem ) {
+ var namespace = elem.namespaceURI,
+ docElem = (elem.ownerDocument || elem).documentElement;
+
+ // Support: IE <=8
+ // Assume HTML when documentElement doesn't yet exist, such as inside loading iframes
+ // https://bugs.jquery.com/ticket/4833
+ return !rhtml.test( namespace || docElem && docElem.nodeName || "HTML" );
+};
+
+/**
+ * Sets document-related variables once based on the current document
+ * @param {Element|Object} [doc] An element or document object to use to set the document
+ * @returns {Object} Returns the current document
+ */
+setDocument = Sizzle.setDocument = function( node ) {
+ var hasCompare, subWindow,
+ doc = node ? node.ownerDocument || node : preferredDoc;
+
+ // Return early if doc is invalid or already selected
+ if ( doc === document || doc.nodeType !== 9 || !doc.documentElement ) {
+ return document;
+ }
+
+ // Update global variables
+ document = doc;
+ docElem = document.documentElement;
+ documentIsHTML = !isXML( document );
+
+ // Support: IE 9-11, Edge
+ // Accessing iframe documents after unload throws "permission denied" errors (jQuery #13936)
+ if ( preferredDoc !== document &&
+ (subWindow = document.defaultView) && subWindow.top !== subWindow ) {
+
+ // Support: IE 11, Edge
+ if ( subWindow.addEventListener ) {
+ subWindow.addEventListener( "unload", unloadHandler, false );
+
+ // Support: IE 9 - 10 only
+ } else if ( subWindow.attachEvent ) {
+ subWindow.attachEvent( "onunload", unloadHandler );
+ }
+ }
+
+ /* Attributes
+ ---------------------------------------------------------------------- */
+
+ // Support: IE<8
+ // Verify that getAttribute really returns attributes and not properties
+ // (excepting IE8 booleans)
+ support.attributes = assert(function( el ) {
+ el.className = "i";
+ return !el.getAttribute("className");
+ });
+
+ /* getElement(s)By*
+ ---------------------------------------------------------------------- */
+
+ // Check if getElementsByTagName("*") returns only elements
+ support.getElementsByTagName = assert(function( el ) {
+ el.appendChild( document.createComment("") );
+ return !el.getElementsByTagName("*").length;
+ });
+
+ // Support: IE<9
+ support.getElementsByClassName = rnative.test( document.getElementsByClassName );
+
+ // Support: IE<10
+ // Check if getElementById returns elements by name
+ // The broken getElementById methods don't pick up programmatically-set names,
+ // so use a roundabout getElementsByName test
+ support.getById = assert(function( el ) {
+ docElem.appendChild( el ).id = expando;
+ return !document.getElementsByName || !document.getElementsByName( expando ).length;
+ });
+
+ // ID filter and find
+ if ( support.getById ) {
+ Expr.filter["ID"] = function( id ) {
+ var attrId = id.replace( runescape, funescape );
+ return function( elem ) {
+ return elem.getAttribute("id") === attrId;
+ };
+ };
+ Expr.find["ID"] = function( id, context ) {
+ if ( typeof context.getElementById !== "undefined" && documentIsHTML ) {
+ var elem = context.getElementById( id );
+ return elem ? [ elem ] : [];
+ }
+ };
+ } else {
+ Expr.filter["ID"] = function( id ) {
+ var attrId = id.replace( runescape, funescape );
+ return function( elem ) {
+ var node = typeof elem.getAttributeNode !== "undefined" &&
+ elem.getAttributeNode("id");
+ return node && node.value === attrId;
+ };
+ };
+
+ // Support: IE 6 - 7 only
+ // getElementById is not reliable as a find shortcut
+ Expr.find["ID"] = function( id, context ) {
+ if ( typeof context.getElementById !== "undefined" && documentIsHTML ) {
+ var node, i, elems,
+ elem = context.getElementById( id );
+
+ if ( elem ) {
+
+ // Verify the id attribute
+ node = elem.getAttributeNode("id");
+ if ( node && node.value === id ) {
+ return [ elem ];
+ }
+
+ // Fall back on getElementsByName
+ elems = context.getElementsByName( id );
+ i = 0;
+ while ( (elem = elems[i++]) ) {
+ node = elem.getAttributeNode("id");
+ if ( node && node.value === id ) {
+ return [ elem ];
+ }
+ }
+ }
+
+ return [];
+ }
+ };
+ }
+
+ // Tag
+ Expr.find["TAG"] = support.getElementsByTagName ?
+ function( tag, context ) {
+ if ( typeof context.getElementsByTagName !== "undefined" ) {
+ return context.getElementsByTagName( tag );
+
+ // DocumentFragment nodes don't have gEBTN
+ } else if ( support.qsa ) {
+ return context.querySelectorAll( tag );
+ }
+ } :
+
+ function( tag, context ) {
+ var elem,
+ tmp = [],
+ i = 0,
+ // By happy coincidence, a (broken) gEBTN appears on DocumentFragment nodes too
+ results = context.getElementsByTagName( tag );
+
+ // Filter out possible comments
+ if ( tag === "*" ) {
+ while ( (elem = results[i++]) ) {
+ if ( elem.nodeType === 1 ) {
+ tmp.push( elem );
+ }
+ }
+
+ return tmp;
+ }
+ return results;
+ };
+
+ // Class
+ Expr.find["CLASS"] = support.getElementsByClassName && function( className, context ) {
+ if ( typeof context.getElementsByClassName !== "undefined" && documentIsHTML ) {
+ return context.getElementsByClassName( className );
+ }
+ };
+
+ /* QSA/matchesSelector
+ ---------------------------------------------------------------------- */
+
+ // QSA and matchesSelector support
+
+ // matchesSelector(:active) reports false when true (IE9/Opera 11.5)
+ rbuggyMatches = [];
+
+ // qSa(:focus) reports false when true (Chrome 21)
+ // We allow this because of a bug in IE8/9 that throws an error
+ // whenever `document.activeElement` is accessed on an iframe
+ // So, we allow :focus to pass through QSA all the time to avoid the IE error
+ // See https://bugs.jquery.com/ticket/13378
+ rbuggyQSA = [];
+
+ if ( (support.qsa = rnative.test( document.querySelectorAll )) ) {
+ // Build QSA regex
+ // Regex strategy adopted from Diego Perini
+ assert(function( el ) {
+ // Select is set to empty string on purpose
+ // This is to test IE's treatment of not explicitly
+ // setting a boolean content attribute,
+ // since its presence should be enough
+ // https://bugs.jquery.com/ticket/12359
+ docElem.appendChild( el ).innerHTML = " " +
+ "" +
+ " ";
+
+ // Support: IE8, Opera 11-12.16
+ // Nothing should be selected when empty strings follow ^= or $= or *=
+ // The test attribute must be unknown in Opera but "safe" for WinRT
+ // https://msdn.microsoft.com/en-us/library/ie/hh465388.aspx#attribute_section
+ if ( el.querySelectorAll("[msallowcapture^='']").length ) {
+ rbuggyQSA.push( "[*^$]=" + whitespace + "*(?:''|\"\")" );
+ }
+
+ // Support: IE8
+ // Boolean attributes and "value" are not treated correctly
+ if ( !el.querySelectorAll("[selected]").length ) {
+ rbuggyQSA.push( "\\[" + whitespace + "*(?:value|" + booleans + ")" );
+ }
+
+ // Support: Chrome<29, Android<4.4, Safari<7.0+, iOS<7.0+, PhantomJS<1.9.8+
+ if ( !el.querySelectorAll( "[id~=" + expando + "-]" ).length ) {
+ rbuggyQSA.push("~=");
+ }
+
+ // Webkit/Opera - :checked should return selected option elements
+ // http://www.w3.org/TR/2011/REC-css3-selectors-20110929/#checked
+ // IE8 throws error here and will not see later tests
+ if ( !el.querySelectorAll(":checked").length ) {
+ rbuggyQSA.push(":checked");
+ }
+
+ // Support: Safari 8+, iOS 8+
+ // https://bugs.webkit.org/show_bug.cgi?id=136851
+ // In-page `selector#id sibling-combinator selector` fails
+ if ( !el.querySelectorAll( "a#" + expando + "+*" ).length ) {
+ rbuggyQSA.push(".#.+[+~]");
+ }
+ });
+
+ assert(function( el ) {
+ el.innerHTML = " " +
+ " ";
+
+ // Support: Windows 8 Native Apps
+ // The type and name attributes are restricted during .innerHTML assignment
+ var input = document.createElement("input");
+ input.setAttribute( "type", "hidden" );
+ el.appendChild( input ).setAttribute( "name", "D" );
+
+ // Support: IE8
+ // Enforce case-sensitivity of name attribute
+ if ( el.querySelectorAll("[name=d]").length ) {
+ rbuggyQSA.push( "name" + whitespace + "*[*^$|!~]?=" );
+ }
+
+ // FF 3.5 - :enabled/:disabled and hidden elements (hidden elements are still enabled)
+ // IE8 throws error here and will not see later tests
+ if ( el.querySelectorAll(":enabled").length !== 2 ) {
+ rbuggyQSA.push( ":enabled", ":disabled" );
+ }
+
+ // Support: IE9-11+
+ // IE's :disabled selector does not pick up the children of disabled fieldsets
+ docElem.appendChild( el ).disabled = true;
+ if ( el.querySelectorAll(":disabled").length !== 2 ) {
+ rbuggyQSA.push( ":enabled", ":disabled" );
+ }
+
+ // Opera 10-11 does not throw on post-comma invalid pseudos
+ el.querySelectorAll("*,:x");
+ rbuggyQSA.push(",.*:");
+ });
+ }
+
+ if ( (support.matchesSelector = rnative.test( (matches = docElem.matches ||
+ docElem.webkitMatchesSelector ||
+ docElem.mozMatchesSelector ||
+ docElem.oMatchesSelector ||
+ docElem.msMatchesSelector) )) ) {
+
+ assert(function( el ) {
+ // Check to see if it's possible to do matchesSelector
+ // on a disconnected node (IE 9)
+ support.disconnectedMatch = matches.call( el, "*" );
+
+ // This should fail with an exception
+ // Gecko does not error, returns false instead
+ matches.call( el, "[s!='']:x" );
+ rbuggyMatches.push( "!=", pseudos );
+ });
+ }
+
+ rbuggyQSA = rbuggyQSA.length && new RegExp( rbuggyQSA.join("|") );
+ rbuggyMatches = rbuggyMatches.length && new RegExp( rbuggyMatches.join("|") );
+
+ /* Contains
+ ---------------------------------------------------------------------- */
+ hasCompare = rnative.test( docElem.compareDocumentPosition );
+
+ // Element contains another
+ // Purposefully self-exclusive
+ // As in, an element does not contain itself
+ contains = hasCompare || rnative.test( docElem.contains ) ?
+ function( a, b ) {
+ var adown = a.nodeType === 9 ? a.documentElement : a,
+ bup = b && b.parentNode;
+ return a === bup || !!( bup && bup.nodeType === 1 && (
+ adown.contains ?
+ adown.contains( bup ) :
+ a.compareDocumentPosition && a.compareDocumentPosition( bup ) & 16
+ ));
+ } :
+ function( a, b ) {
+ if ( b ) {
+ while ( (b = b.parentNode) ) {
+ if ( b === a ) {
+ return true;
+ }
+ }
+ }
+ return false;
+ };
+
+ /* Sorting
+ ---------------------------------------------------------------------- */
+
+ // Document order sorting
+ sortOrder = hasCompare ?
+ function( a, b ) {
+
+ // Flag for duplicate removal
+ if ( a === b ) {
+ hasDuplicate = true;
+ return 0;
+ }
+
+ // Sort on method existence if only one input has compareDocumentPosition
+ var compare = !a.compareDocumentPosition - !b.compareDocumentPosition;
+ if ( compare ) {
+ return compare;
+ }
+
+ // Calculate position if both inputs belong to the same document
+ compare = ( a.ownerDocument || a ) === ( b.ownerDocument || b ) ?
+ a.compareDocumentPosition( b ) :
+
+ // Otherwise we know they are disconnected
+ 1;
+
+ // Disconnected nodes
+ if ( compare & 1 ||
+ (!support.sortDetached && b.compareDocumentPosition( a ) === compare) ) {
+
+ // Choose the first element that is related to our preferred document
+ if ( a === document || a.ownerDocument === preferredDoc && contains(preferredDoc, a) ) {
+ return -1;
+ }
+ if ( b === document || b.ownerDocument === preferredDoc && contains(preferredDoc, b) ) {
+ return 1;
+ }
+
+ // Maintain original order
+ return sortInput ?
+ ( indexOf( sortInput, a ) - indexOf( sortInput, b ) ) :
+ 0;
+ }
+
+ return compare & 4 ? -1 : 1;
+ } :
+ function( a, b ) {
+ // Exit early if the nodes are identical
+ if ( a === b ) {
+ hasDuplicate = true;
+ return 0;
+ }
+
+ var cur,
+ i = 0,
+ aup = a.parentNode,
+ bup = b.parentNode,
+ ap = [ a ],
+ bp = [ b ];
+
+ // Parentless nodes are either documents or disconnected
+ if ( !aup || !bup ) {
+ return a === document ? -1 :
+ b === document ? 1 :
+ aup ? -1 :
+ bup ? 1 :
+ sortInput ?
+ ( indexOf( sortInput, a ) - indexOf( sortInput, b ) ) :
+ 0;
+
+ // If the nodes are siblings, we can do a quick check
+ } else if ( aup === bup ) {
+ return siblingCheck( a, b );
+ }
+
+ // Otherwise we need full lists of their ancestors for comparison
+ cur = a;
+ while ( (cur = cur.parentNode) ) {
+ ap.unshift( cur );
+ }
+ cur = b;
+ while ( (cur = cur.parentNode) ) {
+ bp.unshift( cur );
+ }
+
+ // Walk down the tree looking for a discrepancy
+ while ( ap[i] === bp[i] ) {
+ i++;
+ }
+
+ return i ?
+ // Do a sibling check if the nodes have a common ancestor
+ siblingCheck( ap[i], bp[i] ) :
+
+ // Otherwise nodes in our document sort first
+ ap[i] === preferredDoc ? -1 :
+ bp[i] === preferredDoc ? 1 :
+ 0;
+ };
+
+ return document;
+};
+
+Sizzle.matches = function( expr, elements ) {
+ return Sizzle( expr, null, null, elements );
+};
+
+Sizzle.matchesSelector = function( elem, expr ) {
+ // Set document vars if needed
+ if ( ( elem.ownerDocument || elem ) !== document ) {
+ setDocument( elem );
+ }
+
+ if ( support.matchesSelector && documentIsHTML &&
+ !nonnativeSelectorCache[ expr + " " ] &&
+ ( !rbuggyMatches || !rbuggyMatches.test( expr ) ) &&
+ ( !rbuggyQSA || !rbuggyQSA.test( expr ) ) ) {
+
+ try {
+ var ret = matches.call( elem, expr );
+
+ // IE 9's matchesSelector returns false on disconnected nodes
+ if ( ret || support.disconnectedMatch ||
+ // As well, disconnected nodes are said to be in a document
+ // fragment in IE 9
+ elem.document && elem.document.nodeType !== 11 ) {
+ return ret;
+ }
+ } catch (e) {
+ nonnativeSelectorCache( expr, true );
+ }
+ }
+
+ return Sizzle( expr, document, null, [ elem ] ).length > 0;
+};
+
+Sizzle.contains = function( context, elem ) {
+ // Set document vars if needed
+ if ( ( context.ownerDocument || context ) !== document ) {
+ setDocument( context );
+ }
+ return contains( context, elem );
+};
+
+Sizzle.attr = function( elem, name ) {
+ // Set document vars if needed
+ if ( ( elem.ownerDocument || elem ) !== document ) {
+ setDocument( elem );
+ }
+
+ var fn = Expr.attrHandle[ name.toLowerCase() ],
+ // Don't get fooled by Object.prototype properties (jQuery #13807)
+ val = fn && hasOwn.call( Expr.attrHandle, name.toLowerCase() ) ?
+ fn( elem, name, !documentIsHTML ) :
+ undefined;
+
+ return val !== undefined ?
+ val :
+ support.attributes || !documentIsHTML ?
+ elem.getAttribute( name ) :
+ (val = elem.getAttributeNode(name)) && val.specified ?
+ val.value :
+ null;
+};
+
+Sizzle.escape = function( sel ) {
+ return (sel + "").replace( rcssescape, fcssescape );
+};
+
+Sizzle.error = function( msg ) {
+ throw new Error( "Syntax error, unrecognized expression: " + msg );
+};
+
+/**
+ * Document sorting and removing duplicates
+ * @param {ArrayLike} results
+ */
+Sizzle.uniqueSort = function( results ) {
+ var elem,
+ duplicates = [],
+ j = 0,
+ i = 0;
+
+ // Unless we *know* we can detect duplicates, assume their presence
+ hasDuplicate = !support.detectDuplicates;
+ sortInput = !support.sortStable && results.slice( 0 );
+ results.sort( sortOrder );
+
+ if ( hasDuplicate ) {
+ while ( (elem = results[i++]) ) {
+ if ( elem === results[ i ] ) {
+ j = duplicates.push( i );
+ }
+ }
+ while ( j-- ) {
+ results.splice( duplicates[ j ], 1 );
+ }
+ }
+
+ // Clear input after sorting to release objects
+ // See https://github.com/jquery/sizzle/pull/225
+ sortInput = null;
+
+ return results;
+};
+
+/**
+ * Utility function for retrieving the text value of an array of DOM nodes
+ * @param {Array|Element} elem
+ */
+getText = Sizzle.getText = function( elem ) {
+ var node,
+ ret = "",
+ i = 0,
+ nodeType = elem.nodeType;
+
+ if ( !nodeType ) {
+ // If no nodeType, this is expected to be an array
+ while ( (node = elem[i++]) ) {
+ // Do not traverse comment nodes
+ ret += getText( node );
+ }
+ } else if ( nodeType === 1 || nodeType === 9 || nodeType === 11 ) {
+ // Use textContent for elements
+ // innerText usage removed for consistency of new lines (jQuery #11153)
+ if ( typeof elem.textContent === "string" ) {
+ return elem.textContent;
+ } else {
+ // Traverse its children
+ for ( elem = elem.firstChild; elem; elem = elem.nextSibling ) {
+ ret += getText( elem );
+ }
+ }
+ } else if ( nodeType === 3 || nodeType === 4 ) {
+ return elem.nodeValue;
+ }
+ // Do not include comment or processing instruction nodes
+
+ return ret;
+};
+
+Expr = Sizzle.selectors = {
+
+ // Can be adjusted by the user
+ cacheLength: 50,
+
+ createPseudo: markFunction,
+
+ match: matchExpr,
+
+ attrHandle: {},
+
+ find: {},
+
+ relative: {
+ ">": { dir: "parentNode", first: true },
+ " ": { dir: "parentNode" },
+ "+": { dir: "previousSibling", first: true },
+ "~": { dir: "previousSibling" }
+ },
+
+ preFilter: {
+ "ATTR": function( match ) {
+ match[1] = match[1].replace( runescape, funescape );
+
+ // Move the given value to match[3] whether quoted or unquoted
+ match[3] = ( match[3] || match[4] || match[5] || "" ).replace( runescape, funescape );
+
+ if ( match[2] === "~=" ) {
+ match[3] = " " + match[3] + " ";
+ }
+
+ return match.slice( 0, 4 );
+ },
+
+ "CHILD": function( match ) {
+ /* matches from matchExpr["CHILD"]
+ 1 type (only|nth|...)
+ 2 what (child|of-type)
+ 3 argument (even|odd|\d*|\d*n([+-]\d+)?|...)
+ 4 xn-component of xn+y argument ([+-]?\d*n|)
+ 5 sign of xn-component
+ 6 x of xn-component
+ 7 sign of y-component
+ 8 y of y-component
+ */
+ match[1] = match[1].toLowerCase();
+
+ if ( match[1].slice( 0, 3 ) === "nth" ) {
+ // nth-* requires argument
+ if ( !match[3] ) {
+ Sizzle.error( match[0] );
+ }
+
+ // numeric x and y parameters for Expr.filter.CHILD
+ // remember that false/true cast respectively to 0/1
+ match[4] = +( match[4] ? match[5] + (match[6] || 1) : 2 * ( match[3] === "even" || match[3] === "odd" ) );
+ match[5] = +( ( match[7] + match[8] ) || match[3] === "odd" );
+
+ // other types prohibit arguments
+ } else if ( match[3] ) {
+ Sizzle.error( match[0] );
+ }
+
+ return match;
+ },
+
+ "PSEUDO": function( match ) {
+ var excess,
+ unquoted = !match[6] && match[2];
+
+ if ( matchExpr["CHILD"].test( match[0] ) ) {
+ return null;
+ }
+
+ // Accept quoted arguments as-is
+ if ( match[3] ) {
+ match[2] = match[4] || match[5] || "";
+
+ // Strip excess characters from unquoted arguments
+ } else if ( unquoted && rpseudo.test( unquoted ) &&
+ // Get excess from tokenize (recursively)
+ (excess = tokenize( unquoted, true )) &&
+ // advance to the next closing parenthesis
+ (excess = unquoted.indexOf( ")", unquoted.length - excess ) - unquoted.length) ) {
+
+ // excess is a negative index
+ match[0] = match[0].slice( 0, excess );
+ match[2] = unquoted.slice( 0, excess );
+ }
+
+ // Return only captures needed by the pseudo filter method (type and argument)
+ return match.slice( 0, 3 );
+ }
+ },
+
+ filter: {
+
+ "TAG": function( nodeNameSelector ) {
+ var nodeName = nodeNameSelector.replace( runescape, funescape ).toLowerCase();
+ return nodeNameSelector === "*" ?
+ function() { return true; } :
+ function( elem ) {
+ return elem.nodeName && elem.nodeName.toLowerCase() === nodeName;
+ };
+ },
+
+ "CLASS": function( className ) {
+ var pattern = classCache[ className + " " ];
+
+ return pattern ||
+ (pattern = new RegExp( "(^|" + whitespace + ")" + className + "(" + whitespace + "|$)" )) &&
+ classCache( className, function( elem ) {
+ return pattern.test( typeof elem.className === "string" && elem.className || typeof elem.getAttribute !== "undefined" && elem.getAttribute("class") || "" );
+ });
+ },
+
+ "ATTR": function( name, operator, check ) {
+ return function( elem ) {
+ var result = Sizzle.attr( elem, name );
+
+ if ( result == null ) {
+ return operator === "!=";
+ }
+ if ( !operator ) {
+ return true;
+ }
+
+ result += "";
+
+ return operator === "=" ? result === check :
+ operator === "!=" ? result !== check :
+ operator === "^=" ? check && result.indexOf( check ) === 0 :
+ operator === "*=" ? check && result.indexOf( check ) > -1 :
+ operator === "$=" ? check && result.slice( -check.length ) === check :
+ operator === "~=" ? ( " " + result.replace( rwhitespace, " " ) + " " ).indexOf( check ) > -1 :
+ operator === "|=" ? result === check || result.slice( 0, check.length + 1 ) === check + "-" :
+ false;
+ };
+ },
+
+ "CHILD": function( type, what, argument, first, last ) {
+ var simple = type.slice( 0, 3 ) !== "nth",
+ forward = type.slice( -4 ) !== "last",
+ ofType = what === "of-type";
+
+ return first === 1 && last === 0 ?
+
+ // Shortcut for :nth-*(n)
+ function( elem ) {
+ return !!elem.parentNode;
+ } :
+
+ function( elem, context, xml ) {
+ var cache, uniqueCache, outerCache, node, nodeIndex, start,
+ dir = simple !== forward ? "nextSibling" : "previousSibling",
+ parent = elem.parentNode,
+ name = ofType && elem.nodeName.toLowerCase(),
+ useCache = !xml && !ofType,
+ diff = false;
+
+ if ( parent ) {
+
+ // :(first|last|only)-(child|of-type)
+ if ( simple ) {
+ while ( dir ) {
+ node = elem;
+ while ( (node = node[ dir ]) ) {
+ if ( ofType ?
+ node.nodeName.toLowerCase() === name :
+ node.nodeType === 1 ) {
+
+ return false;
+ }
+ }
+ // Reverse direction for :only-* (if we haven't yet done so)
+ start = dir = type === "only" && !start && "nextSibling";
+ }
+ return true;
+ }
+
+ start = [ forward ? parent.firstChild : parent.lastChild ];
+
+ // non-xml :nth-child(...) stores cache data on `parent`
+ if ( forward && useCache ) {
+
+ // Seek `elem` from a previously-cached index
+
+ // ...in a gzip-friendly way
+ node = parent;
+ outerCache = node[ expando ] || (node[ expando ] = {});
+
+ // Support: IE <9 only
+ // Defend against cloned attroperties (jQuery gh-1709)
+ uniqueCache = outerCache[ node.uniqueID ] ||
+ (outerCache[ node.uniqueID ] = {});
+
+ cache = uniqueCache[ type ] || [];
+ nodeIndex = cache[ 0 ] === dirruns && cache[ 1 ];
+ diff = nodeIndex && cache[ 2 ];
+ node = nodeIndex && parent.childNodes[ nodeIndex ];
+
+ while ( (node = ++nodeIndex && node && node[ dir ] ||
+
+ // Fallback to seeking `elem` from the start
+ (diff = nodeIndex = 0) || start.pop()) ) {
+
+ // When found, cache indexes on `parent` and break
+ if ( node.nodeType === 1 && ++diff && node === elem ) {
+ uniqueCache[ type ] = [ dirruns, nodeIndex, diff ];
+ break;
+ }
+ }
+
+ } else {
+ // Use previously-cached element index if available
+ if ( useCache ) {
+ // ...in a gzip-friendly way
+ node = elem;
+ outerCache = node[ expando ] || (node[ expando ] = {});
+
+ // Support: IE <9 only
+ // Defend against cloned attroperties (jQuery gh-1709)
+ uniqueCache = outerCache[ node.uniqueID ] ||
+ (outerCache[ node.uniqueID ] = {});
+
+ cache = uniqueCache[ type ] || [];
+ nodeIndex = cache[ 0 ] === dirruns && cache[ 1 ];
+ diff = nodeIndex;
+ }
+
+ // xml :nth-child(...)
+ // or :nth-last-child(...) or :nth(-last)?-of-type(...)
+ if ( diff === false ) {
+ // Use the same loop as above to seek `elem` from the start
+ while ( (node = ++nodeIndex && node && node[ dir ] ||
+ (diff = nodeIndex = 0) || start.pop()) ) {
+
+ if ( ( ofType ?
+ node.nodeName.toLowerCase() === name :
+ node.nodeType === 1 ) &&
+ ++diff ) {
+
+ // Cache the index of each encountered element
+ if ( useCache ) {
+ outerCache = node[ expando ] || (node[ expando ] = {});
+
+ // Support: IE <9 only
+ // Defend against cloned attroperties (jQuery gh-1709)
+ uniqueCache = outerCache[ node.uniqueID ] ||
+ (outerCache[ node.uniqueID ] = {});
+
+ uniqueCache[ type ] = [ dirruns, diff ];
+ }
+
+ if ( node === elem ) {
+ break;
+ }
+ }
+ }
+ }
+ }
+
+ // Incorporate the offset, then check against cycle size
+ diff -= last;
+ return diff === first || ( diff % first === 0 && diff / first >= 0 );
+ }
+ };
+ },
+
+ "PSEUDO": function( pseudo, argument ) {
+ // pseudo-class names are case-insensitive
+ // http://www.w3.org/TR/selectors/#pseudo-classes
+ // Prioritize by case sensitivity in case custom pseudos are added with uppercase letters
+ // Remember that setFilters inherits from pseudos
+ var args,
+ fn = Expr.pseudos[ pseudo ] || Expr.setFilters[ pseudo.toLowerCase() ] ||
+ Sizzle.error( "unsupported pseudo: " + pseudo );
+
+ // The user may use createPseudo to indicate that
+ // arguments are needed to create the filter function
+ // just as Sizzle does
+ if ( fn[ expando ] ) {
+ return fn( argument );
+ }
+
+ // But maintain support for old signatures
+ if ( fn.length > 1 ) {
+ args = [ pseudo, pseudo, "", argument ];
+ return Expr.setFilters.hasOwnProperty( pseudo.toLowerCase() ) ?
+ markFunction(function( seed, matches ) {
+ var idx,
+ matched = fn( seed, argument ),
+ i = matched.length;
+ while ( i-- ) {
+ idx = indexOf( seed, matched[i] );
+ seed[ idx ] = !( matches[ idx ] = matched[i] );
+ }
+ }) :
+ function( elem ) {
+ return fn( elem, 0, args );
+ };
+ }
+
+ return fn;
+ }
+ },
+
+ pseudos: {
+ // Potentially complex pseudos
+ "not": markFunction(function( selector ) {
+ // Trim the selector passed to compile
+ // to avoid treating leading and trailing
+ // spaces as combinators
+ var input = [],
+ results = [],
+ matcher = compile( selector.replace( rtrim, "$1" ) );
+
+ return matcher[ expando ] ?
+ markFunction(function( seed, matches, context, xml ) {
+ var elem,
+ unmatched = matcher( seed, null, xml, [] ),
+ i = seed.length;
+
+ // Match elements unmatched by `matcher`
+ while ( i-- ) {
+ if ( (elem = unmatched[i]) ) {
+ seed[i] = !(matches[i] = elem);
+ }
+ }
+ }) :
+ function( elem, context, xml ) {
+ input[0] = elem;
+ matcher( input, null, xml, results );
+ // Don't keep the element (issue #299)
+ input[0] = null;
+ return !results.pop();
+ };
+ }),
+
+ "has": markFunction(function( selector ) {
+ return function( elem ) {
+ return Sizzle( selector, elem ).length > 0;
+ };
+ }),
+
+ "contains": markFunction(function( text ) {
+ text = text.replace( runescape, funescape );
+ return function( elem ) {
+ return ( elem.textContent || getText( elem ) ).indexOf( text ) > -1;
+ };
+ }),
+
+ // "Whether an element is represented by a :lang() selector
+ // is based solely on the element's language value
+ // being equal to the identifier C,
+ // or beginning with the identifier C immediately followed by "-".
+ // The matching of C against the element's language value is performed case-insensitively.
+ // The identifier C does not have to be a valid language name."
+ // http://www.w3.org/TR/selectors/#lang-pseudo
+ "lang": markFunction( function( lang ) {
+ // lang value must be a valid identifier
+ if ( !ridentifier.test(lang || "") ) {
+ Sizzle.error( "unsupported lang: " + lang );
+ }
+ lang = lang.replace( runescape, funescape ).toLowerCase();
+ return function( elem ) {
+ var elemLang;
+ do {
+ if ( (elemLang = documentIsHTML ?
+ elem.lang :
+ elem.getAttribute("xml:lang") || elem.getAttribute("lang")) ) {
+
+ elemLang = elemLang.toLowerCase();
+ return elemLang === lang || elemLang.indexOf( lang + "-" ) === 0;
+ }
+ } while ( (elem = elem.parentNode) && elem.nodeType === 1 );
+ return false;
+ };
+ }),
+
+ // Miscellaneous
+ "target": function( elem ) {
+ var hash = window.location && window.location.hash;
+ return hash && hash.slice( 1 ) === elem.id;
+ },
+
+ "root": function( elem ) {
+ return elem === docElem;
+ },
+
+ "focus": function( elem ) {
+ return elem === document.activeElement && (!document.hasFocus || document.hasFocus()) && !!(elem.type || elem.href || ~elem.tabIndex);
+ },
+
+ // Boolean properties
+ "enabled": createDisabledPseudo( false ),
+ "disabled": createDisabledPseudo( true ),
+
+ "checked": function( elem ) {
+ // In CSS3, :checked should return both checked and selected elements
+ // http://www.w3.org/TR/2011/REC-css3-selectors-20110929/#checked
+ var nodeName = elem.nodeName.toLowerCase();
+ return (nodeName === "input" && !!elem.checked) || (nodeName === "option" && !!elem.selected);
+ },
+
+ "selected": function( elem ) {
+ // Accessing this property makes selected-by-default
+ // options in Safari work properly
+ if ( elem.parentNode ) {
+ elem.parentNode.selectedIndex;
+ }
+
+ return elem.selected === true;
+ },
+
+ // Contents
+ "empty": function( elem ) {
+ // http://www.w3.org/TR/selectors/#empty-pseudo
+ // :empty is negated by element (1) or content nodes (text: 3; cdata: 4; entity ref: 5),
+ // but not by others (comment: 8; processing instruction: 7; etc.)
+ // nodeType < 6 works because attributes (2) do not appear as children
+ for ( elem = elem.firstChild; elem; elem = elem.nextSibling ) {
+ if ( elem.nodeType < 6 ) {
+ return false;
+ }
+ }
+ return true;
+ },
+
+ "parent": function( elem ) {
+ return !Expr.pseudos["empty"]( elem );
+ },
+
+ // Element/input types
+ "header": function( elem ) {
+ return rheader.test( elem.nodeName );
+ },
+
+ "input": function( elem ) {
+ return rinputs.test( elem.nodeName );
+ },
+
+ "button": function( elem ) {
+ var name = elem.nodeName.toLowerCase();
+ return name === "input" && elem.type === "button" || name === "button";
+ },
+
+ "text": function( elem ) {
+ var attr;
+ return elem.nodeName.toLowerCase() === "input" &&
+ elem.type === "text" &&
+
+ // Support: IE<8
+ // New HTML5 attribute values (e.g., "search") appear with elem.type === "text"
+ ( (attr = elem.getAttribute("type")) == null || attr.toLowerCase() === "text" );
+ },
+
+ // Position-in-collection
+ "first": createPositionalPseudo(function() {
+ return [ 0 ];
+ }),
+
+ "last": createPositionalPseudo(function( matchIndexes, length ) {
+ return [ length - 1 ];
+ }),
+
+ "eq": createPositionalPseudo(function( matchIndexes, length, argument ) {
+ return [ argument < 0 ? argument + length : argument ];
+ }),
+
+ "even": createPositionalPseudo(function( matchIndexes, length ) {
+ var i = 0;
+ for ( ; i < length; i += 2 ) {
+ matchIndexes.push( i );
+ }
+ return matchIndexes;
+ }),
+
+ "odd": createPositionalPseudo(function( matchIndexes, length ) {
+ var i = 1;
+ for ( ; i < length; i += 2 ) {
+ matchIndexes.push( i );
+ }
+ return matchIndexes;
+ }),
+
+ "lt": createPositionalPseudo(function( matchIndexes, length, argument ) {
+ var i = argument < 0 ?
+ argument + length :
+ argument > length ?
+ length :
+ argument;
+ for ( ; --i >= 0; ) {
+ matchIndexes.push( i );
+ }
+ return matchIndexes;
+ }),
+
+ "gt": createPositionalPseudo(function( matchIndexes, length, argument ) {
+ var i = argument < 0 ? argument + length : argument;
+ for ( ; ++i < length; ) {
+ matchIndexes.push( i );
+ }
+ return matchIndexes;
+ })
+ }
+};
+
+Expr.pseudos["nth"] = Expr.pseudos["eq"];
+
+// Add button/input type pseudos
+for ( i in { radio: true, checkbox: true, file: true, password: true, image: true } ) {
+ Expr.pseudos[ i ] = createInputPseudo( i );
+}
+for ( i in { submit: true, reset: true } ) {
+ Expr.pseudos[ i ] = createButtonPseudo( i );
+}
+
+// Easy API for creating new setFilters
+function setFilters() {}
+setFilters.prototype = Expr.filters = Expr.pseudos;
+Expr.setFilters = new setFilters();
+
+tokenize = Sizzle.tokenize = function( selector, parseOnly ) {
+ var matched, match, tokens, type,
+ soFar, groups, preFilters,
+ cached = tokenCache[ selector + " " ];
+
+ if ( cached ) {
+ return parseOnly ? 0 : cached.slice( 0 );
+ }
+
+ soFar = selector;
+ groups = [];
+ preFilters = Expr.preFilter;
+
+ while ( soFar ) {
+
+ // Comma and first run
+ if ( !matched || (match = rcomma.exec( soFar )) ) {
+ if ( match ) {
+ // Don't consume trailing commas as valid
+ soFar = soFar.slice( match[0].length ) || soFar;
+ }
+ groups.push( (tokens = []) );
+ }
+
+ matched = false;
+
+ // Combinators
+ if ( (match = rcombinators.exec( soFar )) ) {
+ matched = match.shift();
+ tokens.push({
+ value: matched,
+ // Cast descendant combinators to space
+ type: match[0].replace( rtrim, " " )
+ });
+ soFar = soFar.slice( matched.length );
+ }
+
+ // Filters
+ for ( type in Expr.filter ) {
+ if ( (match = matchExpr[ type ].exec( soFar )) && (!preFilters[ type ] ||
+ (match = preFilters[ type ]( match ))) ) {
+ matched = match.shift();
+ tokens.push({
+ value: matched,
+ type: type,
+ matches: match
+ });
+ soFar = soFar.slice( matched.length );
+ }
+ }
+
+ if ( !matched ) {
+ break;
+ }
+ }
+
+ // Return the length of the invalid excess
+ // if we're just parsing
+ // Otherwise, throw an error or return tokens
+ return parseOnly ?
+ soFar.length :
+ soFar ?
+ Sizzle.error( selector ) :
+ // Cache the tokens
+ tokenCache( selector, groups ).slice( 0 );
+};
+
+function toSelector( tokens ) {
+ var i = 0,
+ len = tokens.length,
+ selector = "";
+ for ( ; i < len; i++ ) {
+ selector += tokens[i].value;
+ }
+ return selector;
+}
+
+function addCombinator( matcher, combinator, base ) {
+ var dir = combinator.dir,
+ skip = combinator.next,
+ key = skip || dir,
+ checkNonElements = base && key === "parentNode",
+ doneName = done++;
+
+ return combinator.first ?
+ // Check against closest ancestor/preceding element
+ function( elem, context, xml ) {
+ while ( (elem = elem[ dir ]) ) {
+ if ( elem.nodeType === 1 || checkNonElements ) {
+ return matcher( elem, context, xml );
+ }
+ }
+ return false;
+ } :
+
+ // Check against all ancestor/preceding elements
+ function( elem, context, xml ) {
+ var oldCache, uniqueCache, outerCache,
+ newCache = [ dirruns, doneName ];
+
+ // We can't set arbitrary data on XML nodes, so they don't benefit from combinator caching
+ if ( xml ) {
+ while ( (elem = elem[ dir ]) ) {
+ if ( elem.nodeType === 1 || checkNonElements ) {
+ if ( matcher( elem, context, xml ) ) {
+ return true;
+ }
+ }
+ }
+ } else {
+ while ( (elem = elem[ dir ]) ) {
+ if ( elem.nodeType === 1 || checkNonElements ) {
+ outerCache = elem[ expando ] || (elem[ expando ] = {});
+
+ // Support: IE <9 only
+ // Defend against cloned attroperties (jQuery gh-1709)
+ uniqueCache = outerCache[ elem.uniqueID ] || (outerCache[ elem.uniqueID ] = {});
+
+ if ( skip && skip === elem.nodeName.toLowerCase() ) {
+ elem = elem[ dir ] || elem;
+ } else if ( (oldCache = uniqueCache[ key ]) &&
+ oldCache[ 0 ] === dirruns && oldCache[ 1 ] === doneName ) {
+
+ // Assign to newCache so results back-propagate to previous elements
+ return (newCache[ 2 ] = oldCache[ 2 ]);
+ } else {
+ // Reuse newcache so results back-propagate to previous elements
+ uniqueCache[ key ] = newCache;
+
+ // A match means we're done; a fail means we have to keep checking
+ if ( (newCache[ 2 ] = matcher( elem, context, xml )) ) {
+ return true;
+ }
+ }
+ }
+ }
+ }
+ return false;
+ };
+}
+
+function elementMatcher( matchers ) {
+ return matchers.length > 1 ?
+ function( elem, context, xml ) {
+ var i = matchers.length;
+ while ( i-- ) {
+ if ( !matchers[i]( elem, context, xml ) ) {
+ return false;
+ }
+ }
+ return true;
+ } :
+ matchers[0];
+}
+
+function multipleContexts( selector, contexts, results ) {
+ var i = 0,
+ len = contexts.length;
+ for ( ; i < len; i++ ) {
+ Sizzle( selector, contexts[i], results );
+ }
+ return results;
+}
+
+function condense( unmatched, map, filter, context, xml ) {
+ var elem,
+ newUnmatched = [],
+ i = 0,
+ len = unmatched.length,
+ mapped = map != null;
+
+ for ( ; i < len; i++ ) {
+ if ( (elem = unmatched[i]) ) {
+ if ( !filter || filter( elem, context, xml ) ) {
+ newUnmatched.push( elem );
+ if ( mapped ) {
+ map.push( i );
+ }
+ }
+ }
+ }
+
+ return newUnmatched;
+}
+
+function setMatcher( preFilter, selector, matcher, postFilter, postFinder, postSelector ) {
+ if ( postFilter && !postFilter[ expando ] ) {
+ postFilter = setMatcher( postFilter );
+ }
+ if ( postFinder && !postFinder[ expando ] ) {
+ postFinder = setMatcher( postFinder, postSelector );
+ }
+ return markFunction(function( seed, results, context, xml ) {
+ var temp, i, elem,
+ preMap = [],
+ postMap = [],
+ preexisting = results.length,
+
+ // Get initial elements from seed or context
+ elems = seed || multipleContexts( selector || "*", context.nodeType ? [ context ] : context, [] ),
+
+ // Prefilter to get matcher input, preserving a map for seed-results synchronization
+ matcherIn = preFilter && ( seed || !selector ) ?
+ condense( elems, preMap, preFilter, context, xml ) :
+ elems,
+
+ matcherOut = matcher ?
+ // If we have a postFinder, or filtered seed, or non-seed postFilter or preexisting results,
+ postFinder || ( seed ? preFilter : preexisting || postFilter ) ?
+
+ // ...intermediate processing is necessary
+ [] :
+
+ // ...otherwise use results directly
+ results :
+ matcherIn;
+
+ // Find primary matches
+ if ( matcher ) {
+ matcher( matcherIn, matcherOut, context, xml );
+ }
+
+ // Apply postFilter
+ if ( postFilter ) {
+ temp = condense( matcherOut, postMap );
+ postFilter( temp, [], context, xml );
+
+ // Un-match failing elements by moving them back to matcherIn
+ i = temp.length;
+ while ( i-- ) {
+ if ( (elem = temp[i]) ) {
+ matcherOut[ postMap[i] ] = !(matcherIn[ postMap[i] ] = elem);
+ }
+ }
+ }
+
+ if ( seed ) {
+ if ( postFinder || preFilter ) {
+ if ( postFinder ) {
+ // Get the final matcherOut by condensing this intermediate into postFinder contexts
+ temp = [];
+ i = matcherOut.length;
+ while ( i-- ) {
+ if ( (elem = matcherOut[i]) ) {
+ // Restore matcherIn since elem is not yet a final match
+ temp.push( (matcherIn[i] = elem) );
+ }
+ }
+ postFinder( null, (matcherOut = []), temp, xml );
+ }
+
+ // Move matched elements from seed to results to keep them synchronized
+ i = matcherOut.length;
+ while ( i-- ) {
+ if ( (elem = matcherOut[i]) &&
+ (temp = postFinder ? indexOf( seed, elem ) : preMap[i]) > -1 ) {
+
+ seed[temp] = !(results[temp] = elem);
+ }
+ }
+ }
+
+ // Add elements to results, through postFinder if defined
+ } else {
+ matcherOut = condense(
+ matcherOut === results ?
+ matcherOut.splice( preexisting, matcherOut.length ) :
+ matcherOut
+ );
+ if ( postFinder ) {
+ postFinder( null, results, matcherOut, xml );
+ } else {
+ push.apply( results, matcherOut );
+ }
+ }
+ });
+}
+
+function matcherFromTokens( tokens ) {
+ var checkContext, matcher, j,
+ len = tokens.length,
+ leadingRelative = Expr.relative[ tokens[0].type ],
+ implicitRelative = leadingRelative || Expr.relative[" "],
+ i = leadingRelative ? 1 : 0,
+
+ // The foundational matcher ensures that elements are reachable from top-level context(s)
+ matchContext = addCombinator( function( elem ) {
+ return elem === checkContext;
+ }, implicitRelative, true ),
+ matchAnyContext = addCombinator( function( elem ) {
+ return indexOf( checkContext, elem ) > -1;
+ }, implicitRelative, true ),
+ matchers = [ function( elem, context, xml ) {
+ var ret = ( !leadingRelative && ( xml || context !== outermostContext ) ) || (
+ (checkContext = context).nodeType ?
+ matchContext( elem, context, xml ) :
+ matchAnyContext( elem, context, xml ) );
+ // Avoid hanging onto element (issue #299)
+ checkContext = null;
+ return ret;
+ } ];
+
+ for ( ; i < len; i++ ) {
+ if ( (matcher = Expr.relative[ tokens[i].type ]) ) {
+ matchers = [ addCombinator(elementMatcher( matchers ), matcher) ];
+ } else {
+ matcher = Expr.filter[ tokens[i].type ].apply( null, tokens[i].matches );
+
+ // Return special upon seeing a positional matcher
+ if ( matcher[ expando ] ) {
+ // Find the next relative operator (if any) for proper handling
+ j = ++i;
+ for ( ; j < len; j++ ) {
+ if ( Expr.relative[ tokens[j].type ] ) {
+ break;
+ }
+ }
+ return setMatcher(
+ i > 1 && elementMatcher( matchers ),
+ i > 1 && toSelector(
+ // If the preceding token was a descendant combinator, insert an implicit any-element `*`
+ tokens.slice( 0, i - 1 ).concat({ value: tokens[ i - 2 ].type === " " ? "*" : "" })
+ ).replace( rtrim, "$1" ),
+ matcher,
+ i < j && matcherFromTokens( tokens.slice( i, j ) ),
+ j < len && matcherFromTokens( (tokens = tokens.slice( j )) ),
+ j < len && toSelector( tokens )
+ );
+ }
+ matchers.push( matcher );
+ }
+ }
+
+ return elementMatcher( matchers );
+}
+
+function matcherFromGroupMatchers( elementMatchers, setMatchers ) {
+ var bySet = setMatchers.length > 0,
+ byElement = elementMatchers.length > 0,
+ superMatcher = function( seed, context, xml, results, outermost ) {
+ var elem, j, matcher,
+ matchedCount = 0,
+ i = "0",
+ unmatched = seed && [],
+ setMatched = [],
+ contextBackup = outermostContext,
+ // We must always have either seed elements or outermost context
+ elems = seed || byElement && Expr.find["TAG"]( "*", outermost ),
+ // Use integer dirruns iff this is the outermost matcher
+ dirrunsUnique = (dirruns += contextBackup == null ? 1 : Math.random() || 0.1),
+ len = elems.length;
+
+ if ( outermost ) {
+ outermostContext = context === document || context || outermost;
+ }
+
+ // Add elements passing elementMatchers directly to results
+ // Support: IE<9, Safari
+ // Tolerate NodeList properties (IE: "length"; Safari: ) matching elements by id
+ for ( ; i !== len && (elem = elems[i]) != null; i++ ) {
+ if ( byElement && elem ) {
+ j = 0;
+ if ( !context && elem.ownerDocument !== document ) {
+ setDocument( elem );
+ xml = !documentIsHTML;
+ }
+ while ( (matcher = elementMatchers[j++]) ) {
+ if ( matcher( elem, context || document, xml) ) {
+ results.push( elem );
+ break;
+ }
+ }
+ if ( outermost ) {
+ dirruns = dirrunsUnique;
+ }
+ }
+
+ // Track unmatched elements for set filters
+ if ( bySet ) {
+ // They will have gone through all possible matchers
+ if ( (elem = !matcher && elem) ) {
+ matchedCount--;
+ }
+
+ // Lengthen the array for every element, matched or not
+ if ( seed ) {
+ unmatched.push( elem );
+ }
+ }
+ }
+
+ // `i` is now the count of elements visited above, and adding it to `matchedCount`
+ // makes the latter nonnegative.
+ matchedCount += i;
+
+ // Apply set filters to unmatched elements
+ // NOTE: This can be skipped if there are no unmatched elements (i.e., `matchedCount`
+ // equals `i`), unless we didn't visit _any_ elements in the above loop because we have
+ // no element matchers and no seed.
+ // Incrementing an initially-string "0" `i` allows `i` to remain a string only in that
+ // case, which will result in a "00" `matchedCount` that differs from `i` but is also
+ // numerically zero.
+ if ( bySet && i !== matchedCount ) {
+ j = 0;
+ while ( (matcher = setMatchers[j++]) ) {
+ matcher( unmatched, setMatched, context, xml );
+ }
+
+ if ( seed ) {
+ // Reintegrate element matches to eliminate the need for sorting
+ if ( matchedCount > 0 ) {
+ while ( i-- ) {
+ if ( !(unmatched[i] || setMatched[i]) ) {
+ setMatched[i] = pop.call( results );
+ }
+ }
+ }
+
+ // Discard index placeholder values to get only actual matches
+ setMatched = condense( setMatched );
+ }
+
+ // Add matches to results
+ push.apply( results, setMatched );
+
+ // Seedless set matches succeeding multiple successful matchers stipulate sorting
+ if ( outermost && !seed && setMatched.length > 0 &&
+ ( matchedCount + setMatchers.length ) > 1 ) {
+
+ Sizzle.uniqueSort( results );
+ }
+ }
+
+ // Override manipulation of globals by nested matchers
+ if ( outermost ) {
+ dirruns = dirrunsUnique;
+ outermostContext = contextBackup;
+ }
+
+ return unmatched;
+ };
+
+ return bySet ?
+ markFunction( superMatcher ) :
+ superMatcher;
+}
+
+compile = Sizzle.compile = function( selector, match /* Internal Use Only */ ) {
+ var i,
+ setMatchers = [],
+ elementMatchers = [],
+ cached = compilerCache[ selector + " " ];
+
+ if ( !cached ) {
+ // Generate a function of recursive functions that can be used to check each element
+ if ( !match ) {
+ match = tokenize( selector );
+ }
+ i = match.length;
+ while ( i-- ) {
+ cached = matcherFromTokens( match[i] );
+ if ( cached[ expando ] ) {
+ setMatchers.push( cached );
+ } else {
+ elementMatchers.push( cached );
+ }
+ }
+
+ // Cache the compiled function
+ cached = compilerCache( selector, matcherFromGroupMatchers( elementMatchers, setMatchers ) );
+
+ // Save selector and tokenization
+ cached.selector = selector;
+ }
+ return cached;
+};
+
+/**
+ * A low-level selection function that works with Sizzle's compiled
+ * selector functions
+ * @param {String|Function} selector A selector or a pre-compiled
+ * selector function built with Sizzle.compile
+ * @param {Element} context
+ * @param {Array} [results]
+ * @param {Array} [seed] A set of elements to match against
+ */
+select = Sizzle.select = function( selector, context, results, seed ) {
+ var i, tokens, token, type, find,
+ compiled = typeof selector === "function" && selector,
+ match = !seed && tokenize( (selector = compiled.selector || selector) );
+
+ results = results || [];
+
+ // Try to minimize operations if there is only one selector in the list and no seed
+ // (the latter of which guarantees us context)
+ if ( match.length === 1 ) {
+
+ // Reduce context if the leading compound selector is an ID
+ tokens = match[0] = match[0].slice( 0 );
+ if ( tokens.length > 2 && (token = tokens[0]).type === "ID" &&
+ context.nodeType === 9 && documentIsHTML && Expr.relative[ tokens[1].type ] ) {
+
+ context = ( Expr.find["ID"]( token.matches[0].replace(runescape, funescape), context ) || [] )[0];
+ if ( !context ) {
+ return results;
+
+ // Precompiled matchers will still verify ancestry, so step up a level
+ } else if ( compiled ) {
+ context = context.parentNode;
+ }
+
+ selector = selector.slice( tokens.shift().value.length );
+ }
+
+ // Fetch a seed set for right-to-left matching
+ i = matchExpr["needsContext"].test( selector ) ? 0 : tokens.length;
+ while ( i-- ) {
+ token = tokens[i];
+
+ // Abort if we hit a combinator
+ if ( Expr.relative[ (type = token.type) ] ) {
+ break;
+ }
+ if ( (find = Expr.find[ type ]) ) {
+ // Search, expanding context for leading sibling combinators
+ if ( (seed = find(
+ token.matches[0].replace( runescape, funescape ),
+ rsibling.test( tokens[0].type ) && testContext( context.parentNode ) || context
+ )) ) {
+
+ // If seed is empty or no tokens remain, we can return early
+ tokens.splice( i, 1 );
+ selector = seed.length && toSelector( tokens );
+ if ( !selector ) {
+ push.apply( results, seed );
+ return results;
+ }
+
+ break;
+ }
+ }
+ }
+ }
+
+ // Compile and execute a filtering function if one is not provided
+ // Provide `match` to avoid retokenization if we modified the selector above
+ ( compiled || compile( selector, match ) )(
+ seed,
+ context,
+ !documentIsHTML,
+ results,
+ !context || rsibling.test( selector ) && testContext( context.parentNode ) || context
+ );
+ return results;
+};
+
+// One-time assignments
+
+// Sort stability
+support.sortStable = expando.split("").sort( sortOrder ).join("") === expando;
+
+// Support: Chrome 14-35+
+// Always assume duplicates if they aren't passed to the comparison function
+support.detectDuplicates = !!hasDuplicate;
+
+// Initialize against the default document
+setDocument();
+
+// Support: Webkit<537.32 - Safari 6.0.3/Chrome 25 (fixed in Chrome 27)
+// Detached nodes confoundingly follow *each other*
+support.sortDetached = assert(function( el ) {
+ // Should return 1, but returns 4 (following)
+ return el.compareDocumentPosition( document.createElement("fieldset") ) & 1;
+});
+
+// Support: IE<8
+// Prevent attribute/property "interpolation"
+// https://msdn.microsoft.com/en-us/library/ms536429%28VS.85%29.aspx
+if ( !assert(function( el ) {
+ el.innerHTML = " ";
+ return el.firstChild.getAttribute("href") === "#" ;
+}) ) {
+ addHandle( "type|href|height|width", function( elem, name, isXML ) {
+ if ( !isXML ) {
+ return elem.getAttribute( name, name.toLowerCase() === "type" ? 1 : 2 );
+ }
+ });
+}
+
+// Support: IE<9
+// Use defaultValue in place of getAttribute("value")
+if ( !support.attributes || !assert(function( el ) {
+ el.innerHTML = " ";
+ el.firstChild.setAttribute( "value", "" );
+ return el.firstChild.getAttribute( "value" ) === "";
+}) ) {
+ addHandle( "value", function( elem, name, isXML ) {
+ if ( !isXML && elem.nodeName.toLowerCase() === "input" ) {
+ return elem.defaultValue;
+ }
+ });
+}
+
+// Support: IE<9
+// Use getAttributeNode to fetch booleans when getAttribute lies
+if ( !assert(function( el ) {
+ return el.getAttribute("disabled") == null;
+}) ) {
+ addHandle( booleans, function( elem, name, isXML ) {
+ var val;
+ if ( !isXML ) {
+ return elem[ name ] === true ? name.toLowerCase() :
+ (val = elem.getAttributeNode( name )) && val.specified ?
+ val.value :
+ null;
+ }
+ });
+}
+
+return Sizzle;
+
+})( window );
+
+
+
+jQuery.find = Sizzle;
+jQuery.expr = Sizzle.selectors;
+
+// Deprecated
+jQuery.expr[ ":" ] = jQuery.expr.pseudos;
+jQuery.uniqueSort = jQuery.unique = Sizzle.uniqueSort;
+jQuery.text = Sizzle.getText;
+jQuery.isXMLDoc = Sizzle.isXML;
+jQuery.contains = Sizzle.contains;
+jQuery.escapeSelector = Sizzle.escape;
+
+
+
+
+var dir = function( elem, dir, until ) {
+ var matched = [],
+ truncate = until !== undefined;
+
+ while ( ( elem = elem[ dir ] ) && elem.nodeType !== 9 ) {
+ if ( elem.nodeType === 1 ) {
+ if ( truncate && jQuery( elem ).is( until ) ) {
+ break;
+ }
+ matched.push( elem );
+ }
+ }
+ return matched;
+};
+
+
+var siblings = function( n, elem ) {
+ var matched = [];
+
+ for ( ; n; n = n.nextSibling ) {
+ if ( n.nodeType === 1 && n !== elem ) {
+ matched.push( n );
+ }
+ }
+
+ return matched;
+};
+
+
+var rneedsContext = jQuery.expr.match.needsContext;
+
+
+
+function nodeName( elem, name ) {
+
+ return elem.nodeName && elem.nodeName.toLowerCase() === name.toLowerCase();
+
+};
+var rsingleTag = ( /^<([a-z][^\/\0>:\x20\t\r\n\f]*)[\x20\t\r\n\f]*\/?>(?:<\/\1>|)$/i );
+
+
+
+// Implement the identical functionality for filter and not
+function winnow( elements, qualifier, not ) {
+ if ( isFunction( qualifier ) ) {
+ return jQuery.grep( elements, function( elem, i ) {
+ return !!qualifier.call( elem, i, elem ) !== not;
+ } );
+ }
+
+ // Single element
+ if ( qualifier.nodeType ) {
+ return jQuery.grep( elements, function( elem ) {
+ return ( elem === qualifier ) !== not;
+ } );
+ }
+
+ // Arraylike of elements (jQuery, arguments, Array)
+ if ( typeof qualifier !== "string" ) {
+ return jQuery.grep( elements, function( elem ) {
+ return ( indexOf.call( qualifier, elem ) > -1 ) !== not;
+ } );
+ }
+
+ // Filtered directly for both simple and complex selectors
+ return jQuery.filter( qualifier, elements, not );
+}
+
+jQuery.filter = function( expr, elems, not ) {
+ var elem = elems[ 0 ];
+
+ if ( not ) {
+ expr = ":not(" + expr + ")";
+ }
+
+ if ( elems.length === 1 && elem.nodeType === 1 ) {
+ return jQuery.find.matchesSelector( elem, expr ) ? [ elem ] : [];
+ }
+
+ return jQuery.find.matches( expr, jQuery.grep( elems, function( elem ) {
+ return elem.nodeType === 1;
+ } ) );
+};
+
+jQuery.fn.extend( {
+ find: function( selector ) {
+ var i, ret,
+ len = this.length,
+ self = this;
+
+ if ( typeof selector !== "string" ) {
+ return this.pushStack( jQuery( selector ).filter( function() {
+ for ( i = 0; i < len; i++ ) {
+ if ( jQuery.contains( self[ i ], this ) ) {
+ return true;
+ }
+ }
+ } ) );
+ }
+
+ ret = this.pushStack( [] );
+
+ for ( i = 0; i < len; i++ ) {
+ jQuery.find( selector, self[ i ], ret );
+ }
+
+ return len > 1 ? jQuery.uniqueSort( ret ) : ret;
+ },
+ filter: function( selector ) {
+ return this.pushStack( winnow( this, selector || [], false ) );
+ },
+ not: function( selector ) {
+ return this.pushStack( winnow( this, selector || [], true ) );
+ },
+ is: function( selector ) {
+ return !!winnow(
+ this,
+
+ // If this is a positional/relative selector, check membership in the returned set
+ // so $("p:first").is("p:last") won't return true for a doc with two "p".
+ typeof selector === "string" && rneedsContext.test( selector ) ?
+ jQuery( selector ) :
+ selector || [],
+ false
+ ).length;
+ }
+} );
+
+
+// Initialize a jQuery object
+
+
+// A central reference to the root jQuery(document)
+var rootjQuery,
+
+ // A simple way to check for HTML strings
+ // Prioritize #id over to avoid XSS via location.hash (#9521)
+ // Strict HTML recognition (#11290: must start with <)
+ // Shortcut simple #id case for speed
+ rquickExpr = /^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]+))$/,
+
+ init = jQuery.fn.init = function( selector, context, root ) {
+ var match, elem;
+
+ // HANDLE: $(""), $(null), $(undefined), $(false)
+ if ( !selector ) {
+ return this;
+ }
+
+ // Method init() accepts an alternate rootjQuery
+ // so migrate can support jQuery.sub (gh-2101)
+ root = root || rootjQuery;
+
+ // Handle HTML strings
+ if ( typeof selector === "string" ) {
+ if ( selector[ 0 ] === "<" &&
+ selector[ selector.length - 1 ] === ">" &&
+ selector.length >= 3 ) {
+
+ // Assume that strings that start and end with <> are HTML and skip the regex check
+ match = [ null, selector, null ];
+
+ } else {
+ match = rquickExpr.exec( selector );
+ }
+
+ // Match html or make sure no context is specified for #id
+ if ( match && ( match[ 1 ] || !context ) ) {
+
+ // HANDLE: $(html) -> $(array)
+ if ( match[ 1 ] ) {
+ context = context instanceof jQuery ? context[ 0 ] : context;
+
+ // Option to run scripts is true for back-compat
+ // Intentionally let the error be thrown if parseHTML is not present
+ jQuery.merge( this, jQuery.parseHTML(
+ match[ 1 ],
+ context && context.nodeType ? context.ownerDocument || context : document,
+ true
+ ) );
+
+ // HANDLE: $(html, props)
+ if ( rsingleTag.test( match[ 1 ] ) && jQuery.isPlainObject( context ) ) {
+ for ( match in context ) {
+
+ // Properties of context are called as methods if possible
+ if ( isFunction( this[ match ] ) ) {
+ this[ match ]( context[ match ] );
+
+ // ...and otherwise set as attributes
+ } else {
+ this.attr( match, context[ match ] );
+ }
+ }
+ }
+
+ return this;
+
+ // HANDLE: $(#id)
+ } else {
+ elem = document.getElementById( match[ 2 ] );
+
+ if ( elem ) {
+
+ // Inject the element directly into the jQuery object
+ this[ 0 ] = elem;
+ this.length = 1;
+ }
+ return this;
+ }
+
+ // HANDLE: $(expr, $(...))
+ } else if ( !context || context.jquery ) {
+ return ( context || root ).find( selector );
+
+ // HANDLE: $(expr, context)
+ // (which is just equivalent to: $(context).find(expr)
+ } else {
+ return this.constructor( context ).find( selector );
+ }
+
+ // HANDLE: $(DOMElement)
+ } else if ( selector.nodeType ) {
+ this[ 0 ] = selector;
+ this.length = 1;
+ return this;
+
+ // HANDLE: $(function)
+ // Shortcut for document ready
+ } else if ( isFunction( selector ) ) {
+ return root.ready !== undefined ?
+ root.ready( selector ) :
+
+ // Execute immediately if ready is not present
+ selector( jQuery );
+ }
+
+ return jQuery.makeArray( selector, this );
+ };
+
+// Give the init function the jQuery prototype for later instantiation
+init.prototype = jQuery.fn;
+
+// Initialize central reference
+rootjQuery = jQuery( document );
+
+
+var rparentsprev = /^(?:parents|prev(?:Until|All))/,
+
+ // Methods guaranteed to produce a unique set when starting from a unique set
+ guaranteedUnique = {
+ children: true,
+ contents: true,
+ next: true,
+ prev: true
+ };
+
+jQuery.fn.extend( {
+ has: function( target ) {
+ var targets = jQuery( target, this ),
+ l = targets.length;
+
+ return this.filter( function() {
+ var i = 0;
+ for ( ; i < l; i++ ) {
+ if ( jQuery.contains( this, targets[ i ] ) ) {
+ return true;
+ }
+ }
+ } );
+ },
+
+ closest: function( selectors, context ) {
+ var cur,
+ i = 0,
+ l = this.length,
+ matched = [],
+ targets = typeof selectors !== "string" && jQuery( selectors );
+
+ // Positional selectors never match, since there's no _selection_ context
+ if ( !rneedsContext.test( selectors ) ) {
+ for ( ; i < l; i++ ) {
+ for ( cur = this[ i ]; cur && cur !== context; cur = cur.parentNode ) {
+
+ // Always skip document fragments
+ if ( cur.nodeType < 11 && ( targets ?
+ targets.index( cur ) > -1 :
+
+ // Don't pass non-elements to Sizzle
+ cur.nodeType === 1 &&
+ jQuery.find.matchesSelector( cur, selectors ) ) ) {
+
+ matched.push( cur );
+ break;
+ }
+ }
+ }
+ }
+
+ return this.pushStack( matched.length > 1 ? jQuery.uniqueSort( matched ) : matched );
+ },
+
+ // Determine the position of an element within the set
+ index: function( elem ) {
+
+ // No argument, return index in parent
+ if ( !elem ) {
+ return ( this[ 0 ] && this[ 0 ].parentNode ) ? this.first().prevAll().length : -1;
+ }
+
+ // Index in selector
+ if ( typeof elem === "string" ) {
+ return indexOf.call( jQuery( elem ), this[ 0 ] );
+ }
+
+ // Locate the position of the desired element
+ return indexOf.call( this,
+
+ // If it receives a jQuery object, the first element is used
+ elem.jquery ? elem[ 0 ] : elem
+ );
+ },
+
+ add: function( selector, context ) {
+ return this.pushStack(
+ jQuery.uniqueSort(
+ jQuery.merge( this.get(), jQuery( selector, context ) )
+ )
+ );
+ },
+
+ addBack: function( selector ) {
+ return this.add( selector == null ?
+ this.prevObject : this.prevObject.filter( selector )
+ );
+ }
+} );
+
+function sibling( cur, dir ) {
+ while ( ( cur = cur[ dir ] ) && cur.nodeType !== 1 ) {}
+ return cur;
+}
+
+jQuery.each( {
+ parent: function( elem ) {
+ var parent = elem.parentNode;
+ return parent && parent.nodeType !== 11 ? parent : null;
+ },
+ parents: function( elem ) {
+ return dir( elem, "parentNode" );
+ },
+ parentsUntil: function( elem, i, until ) {
+ return dir( elem, "parentNode", until );
+ },
+ next: function( elem ) {
+ return sibling( elem, "nextSibling" );
+ },
+ prev: function( elem ) {
+ return sibling( elem, "previousSibling" );
+ },
+ nextAll: function( elem ) {
+ return dir( elem, "nextSibling" );
+ },
+ prevAll: function( elem ) {
+ return dir( elem, "previousSibling" );
+ },
+ nextUntil: function( elem, i, until ) {
+ return dir( elem, "nextSibling", until );
+ },
+ prevUntil: function( elem, i, until ) {
+ return dir( elem, "previousSibling", until );
+ },
+ siblings: function( elem ) {
+ return siblings( ( elem.parentNode || {} ).firstChild, elem );
+ },
+ children: function( elem ) {
+ return siblings( elem.firstChild );
+ },
+ contents: function( elem ) {
+ if ( typeof elem.contentDocument !== "undefined" ) {
+ return elem.contentDocument;
+ }
+
+ // Support: IE 9 - 11 only, iOS 7 only, Android Browser <=4.3 only
+ // Treat the template element as a regular one in browsers that
+ // don't support it.
+ if ( nodeName( elem, "template" ) ) {
+ elem = elem.content || elem;
+ }
+
+ return jQuery.merge( [], elem.childNodes );
+ }
+}, function( name, fn ) {
+ jQuery.fn[ name ] = function( until, selector ) {
+ var matched = jQuery.map( this, fn, until );
+
+ if ( name.slice( -5 ) !== "Until" ) {
+ selector = until;
+ }
+
+ if ( selector && typeof selector === "string" ) {
+ matched = jQuery.filter( selector, matched );
+ }
+
+ if ( this.length > 1 ) {
+
+ // Remove duplicates
+ if ( !guaranteedUnique[ name ] ) {
+ jQuery.uniqueSort( matched );
+ }
+
+ // Reverse order for parents* and prev-derivatives
+ if ( rparentsprev.test( name ) ) {
+ matched.reverse();
+ }
+ }
+
+ return this.pushStack( matched );
+ };
+} );
+var rnothtmlwhite = ( /[^\x20\t\r\n\f]+/g );
+
+
+
+// Convert String-formatted options into Object-formatted ones
+function createOptions( options ) {
+ var object = {};
+ jQuery.each( options.match( rnothtmlwhite ) || [], function( _, flag ) {
+ object[ flag ] = true;
+ } );
+ return object;
+}
+
+/*
+ * Create a callback list using the following parameters:
+ *
+ * options: an optional list of space-separated options that will change how
+ * the callback list behaves or a more traditional option object
+ *
+ * By default a callback list will act like an event callback list and can be
+ * "fired" multiple times.
+ *
+ * Possible options:
+ *
+ * once: will ensure the callback list can only be fired once (like a Deferred)
+ *
+ * memory: will keep track of previous values and will call any callback added
+ * after the list has been fired right away with the latest "memorized"
+ * values (like a Deferred)
+ *
+ * unique: will ensure a callback can only be added once (no duplicate in the list)
+ *
+ * stopOnFalse: interrupt callings when a callback returns false
+ *
+ */
+jQuery.Callbacks = function( options ) {
+
+ // Convert options from String-formatted to Object-formatted if needed
+ // (we check in cache first)
+ options = typeof options === "string" ?
+ createOptions( options ) :
+ jQuery.extend( {}, options );
+
+ var // Flag to know if list is currently firing
+ firing,
+
+ // Last fire value for non-forgettable lists
+ memory,
+
+ // Flag to know if list was already fired
+ fired,
+
+ // Flag to prevent firing
+ locked,
+
+ // Actual callback list
+ list = [],
+
+ // Queue of execution data for repeatable lists
+ queue = [],
+
+ // Index of currently firing callback (modified by add/remove as needed)
+ firingIndex = -1,
+
+ // Fire callbacks
+ fire = function() {
+
+ // Enforce single-firing
+ locked = locked || options.once;
+
+ // Execute callbacks for all pending executions,
+ // respecting firingIndex overrides and runtime changes
+ fired = firing = true;
+ for ( ; queue.length; firingIndex = -1 ) {
+ memory = queue.shift();
+ while ( ++firingIndex < list.length ) {
+
+ // Run callback and check for early termination
+ if ( list[ firingIndex ].apply( memory[ 0 ], memory[ 1 ] ) === false &&
+ options.stopOnFalse ) {
+
+ // Jump to end and forget the data so .add doesn't re-fire
+ firingIndex = list.length;
+ memory = false;
+ }
+ }
+ }
+
+ // Forget the data if we're done with it
+ if ( !options.memory ) {
+ memory = false;
+ }
+
+ firing = false;
+
+ // Clean up if we're done firing for good
+ if ( locked ) {
+
+ // Keep an empty list if we have data for future add calls
+ if ( memory ) {
+ list = [];
+
+ // Otherwise, this object is spent
+ } else {
+ list = "";
+ }
+ }
+ },
+
+ // Actual Callbacks object
+ self = {
+
+ // Add a callback or a collection of callbacks to the list
+ add: function() {
+ if ( list ) {
+
+ // If we have memory from a past run, we should fire after adding
+ if ( memory && !firing ) {
+ firingIndex = list.length - 1;
+ queue.push( memory );
+ }
+
+ ( function add( args ) {
+ jQuery.each( args, function( _, arg ) {
+ if ( isFunction( arg ) ) {
+ if ( !options.unique || !self.has( arg ) ) {
+ list.push( arg );
+ }
+ } else if ( arg && arg.length && toType( arg ) !== "string" ) {
+
+ // Inspect recursively
+ add( arg );
+ }
+ } );
+ } )( arguments );
+
+ if ( memory && !firing ) {
+ fire();
+ }
+ }
+ return this;
+ },
+
+ // Remove a callback from the list
+ remove: function() {
+ jQuery.each( arguments, function( _, arg ) {
+ var index;
+ while ( ( index = jQuery.inArray( arg, list, index ) ) > -1 ) {
+ list.splice( index, 1 );
+
+ // Handle firing indexes
+ if ( index <= firingIndex ) {
+ firingIndex--;
+ }
+ }
+ } );
+ return this;
+ },
+
+ // Check if a given callback is in the list.
+ // If no argument is given, return whether or not list has callbacks attached.
+ has: function( fn ) {
+ return fn ?
+ jQuery.inArray( fn, list ) > -1 :
+ list.length > 0;
+ },
+
+ // Remove all callbacks from the list
+ empty: function() {
+ if ( list ) {
+ list = [];
+ }
+ return this;
+ },
+
+ // Disable .fire and .add
+ // Abort any current/pending executions
+ // Clear all callbacks and values
+ disable: function() {
+ locked = queue = [];
+ list = memory = "";
+ return this;
+ },
+ disabled: function() {
+ return !list;
+ },
+
+ // Disable .fire
+ // Also disable .add unless we have memory (since it would have no effect)
+ // Abort any pending executions
+ lock: function() {
+ locked = queue = [];
+ if ( !memory && !firing ) {
+ list = memory = "";
+ }
+ return this;
+ },
+ locked: function() {
+ return !!locked;
+ },
+
+ // Call all callbacks with the given context and arguments
+ fireWith: function( context, args ) {
+ if ( !locked ) {
+ args = args || [];
+ args = [ context, args.slice ? args.slice() : args ];
+ queue.push( args );
+ if ( !firing ) {
+ fire();
+ }
+ }
+ return this;
+ },
+
+ // Call all the callbacks with the given arguments
+ fire: function() {
+ self.fireWith( this, arguments );
+ return this;
+ },
+
+ // To know if the callbacks have already been called at least once
+ fired: function() {
+ return !!fired;
+ }
+ };
+
+ return self;
+};
+
+
+function Identity( v ) {
+ return v;
+}
+function Thrower( ex ) {
+ throw ex;
+}
+
+function adoptValue( value, resolve, reject, noValue ) {
+ var method;
+
+ try {
+
+ // Check for promise aspect first to privilege synchronous behavior
+ if ( value && isFunction( ( method = value.promise ) ) ) {
+ method.call( value ).done( resolve ).fail( reject );
+
+ // Other thenables
+ } else if ( value && isFunction( ( method = value.then ) ) ) {
+ method.call( value, resolve, reject );
+
+ // Other non-thenables
+ } else {
+
+ // Control `resolve` arguments by letting Array#slice cast boolean `noValue` to integer:
+ // * false: [ value ].slice( 0 ) => resolve( value )
+ // * true: [ value ].slice( 1 ) => resolve()
+ resolve.apply( undefined, [ value ].slice( noValue ) );
+ }
+
+ // For Promises/A+, convert exceptions into rejections
+ // Since jQuery.when doesn't unwrap thenables, we can skip the extra checks appearing in
+ // Deferred#then to conditionally suppress rejection.
+ } catch ( value ) {
+
+ // Support: Android 4.0 only
+ // Strict mode functions invoked without .call/.apply get global-object context
+ reject.apply( undefined, [ value ] );
+ }
+}
+
+jQuery.extend( {
+
+ Deferred: function( func ) {
+ var tuples = [
+
+ // action, add listener, callbacks,
+ // ... .then handlers, argument index, [final state]
+ [ "notify", "progress", jQuery.Callbacks( "memory" ),
+ jQuery.Callbacks( "memory" ), 2 ],
+ [ "resolve", "done", jQuery.Callbacks( "once memory" ),
+ jQuery.Callbacks( "once memory" ), 0, "resolved" ],
+ [ "reject", "fail", jQuery.Callbacks( "once memory" ),
+ jQuery.Callbacks( "once memory" ), 1, "rejected" ]
+ ],
+ state = "pending",
+ promise = {
+ state: function() {
+ return state;
+ },
+ always: function() {
+ deferred.done( arguments ).fail( arguments );
+ return this;
+ },
+ "catch": function( fn ) {
+ return promise.then( null, fn );
+ },
+
+ // Keep pipe for back-compat
+ pipe: function( /* fnDone, fnFail, fnProgress */ ) {
+ var fns = arguments;
+
+ return jQuery.Deferred( function( newDefer ) {
+ jQuery.each( tuples, function( i, tuple ) {
+
+ // Map tuples (progress, done, fail) to arguments (done, fail, progress)
+ var fn = isFunction( fns[ tuple[ 4 ] ] ) && fns[ tuple[ 4 ] ];
+
+ // deferred.progress(function() { bind to newDefer or newDefer.notify })
+ // deferred.done(function() { bind to newDefer or newDefer.resolve })
+ // deferred.fail(function() { bind to newDefer or newDefer.reject })
+ deferred[ tuple[ 1 ] ]( function() {
+ var returned = fn && fn.apply( this, arguments );
+ if ( returned && isFunction( returned.promise ) ) {
+ returned.promise()
+ .progress( newDefer.notify )
+ .done( newDefer.resolve )
+ .fail( newDefer.reject );
+ } else {
+ newDefer[ tuple[ 0 ] + "With" ](
+ this,
+ fn ? [ returned ] : arguments
+ );
+ }
+ } );
+ } );
+ fns = null;
+ } ).promise();
+ },
+ then: function( onFulfilled, onRejected, onProgress ) {
+ var maxDepth = 0;
+ function resolve( depth, deferred, handler, special ) {
+ return function() {
+ var that = this,
+ args = arguments,
+ mightThrow = function() {
+ var returned, then;
+
+ // Support: Promises/A+ section 2.3.3.3.3
+ // https://promisesaplus.com/#point-59
+ // Ignore double-resolution attempts
+ if ( depth < maxDepth ) {
+ return;
+ }
+
+ returned = handler.apply( that, args );
+
+ // Support: Promises/A+ section 2.3.1
+ // https://promisesaplus.com/#point-48
+ if ( returned === deferred.promise() ) {
+ throw new TypeError( "Thenable self-resolution" );
+ }
+
+ // Support: Promises/A+ sections 2.3.3.1, 3.5
+ // https://promisesaplus.com/#point-54
+ // https://promisesaplus.com/#point-75
+ // Retrieve `then` only once
+ then = returned &&
+
+ // Support: Promises/A+ section 2.3.4
+ // https://promisesaplus.com/#point-64
+ // Only check objects and functions for thenability
+ ( typeof returned === "object" ||
+ typeof returned === "function" ) &&
+ returned.then;
+
+ // Handle a returned thenable
+ if ( isFunction( then ) ) {
+
+ // Special processors (notify) just wait for resolution
+ if ( special ) {
+ then.call(
+ returned,
+ resolve( maxDepth, deferred, Identity, special ),
+ resolve( maxDepth, deferred, Thrower, special )
+ );
+
+ // Normal processors (resolve) also hook into progress
+ } else {
+
+ // ...and disregard older resolution values
+ maxDepth++;
+
+ then.call(
+ returned,
+ resolve( maxDepth, deferred, Identity, special ),
+ resolve( maxDepth, deferred, Thrower, special ),
+ resolve( maxDepth, deferred, Identity,
+ deferred.notifyWith )
+ );
+ }
+
+ // Handle all other returned values
+ } else {
+
+ // Only substitute handlers pass on context
+ // and multiple values (non-spec behavior)
+ if ( handler !== Identity ) {
+ that = undefined;
+ args = [ returned ];
+ }
+
+ // Process the value(s)
+ // Default process is resolve
+ ( special || deferred.resolveWith )( that, args );
+ }
+ },
+
+ // Only normal processors (resolve) catch and reject exceptions
+ process = special ?
+ mightThrow :
+ function() {
+ try {
+ mightThrow();
+ } catch ( e ) {
+
+ if ( jQuery.Deferred.exceptionHook ) {
+ jQuery.Deferred.exceptionHook( e,
+ process.stackTrace );
+ }
+
+ // Support: Promises/A+ section 2.3.3.3.4.1
+ // https://promisesaplus.com/#point-61
+ // Ignore post-resolution exceptions
+ if ( depth + 1 >= maxDepth ) {
+
+ // Only substitute handlers pass on context
+ // and multiple values (non-spec behavior)
+ if ( handler !== Thrower ) {
+ that = undefined;
+ args = [ e ];
+ }
+
+ deferred.rejectWith( that, args );
+ }
+ }
+ };
+
+ // Support: Promises/A+ section 2.3.3.3.1
+ // https://promisesaplus.com/#point-57
+ // Re-resolve promises immediately to dodge false rejection from
+ // subsequent errors
+ if ( depth ) {
+ process();
+ } else {
+
+ // Call an optional hook to record the stack, in case of exception
+ // since it's otherwise lost when execution goes async
+ if ( jQuery.Deferred.getStackHook ) {
+ process.stackTrace = jQuery.Deferred.getStackHook();
+ }
+ window.setTimeout( process );
+ }
+ };
+ }
+
+ return jQuery.Deferred( function( newDefer ) {
+
+ // progress_handlers.add( ... )
+ tuples[ 0 ][ 3 ].add(
+ resolve(
+ 0,
+ newDefer,
+ isFunction( onProgress ) ?
+ onProgress :
+ Identity,
+ newDefer.notifyWith
+ )
+ );
+
+ // fulfilled_handlers.add( ... )
+ tuples[ 1 ][ 3 ].add(
+ resolve(
+ 0,
+ newDefer,
+ isFunction( onFulfilled ) ?
+ onFulfilled :
+ Identity
+ )
+ );
+
+ // rejected_handlers.add( ... )
+ tuples[ 2 ][ 3 ].add(
+ resolve(
+ 0,
+ newDefer,
+ isFunction( onRejected ) ?
+ onRejected :
+ Thrower
+ )
+ );
+ } ).promise();
+ },
+
+ // Get a promise for this deferred
+ // If obj is provided, the promise aspect is added to the object
+ promise: function( obj ) {
+ return obj != null ? jQuery.extend( obj, promise ) : promise;
+ }
+ },
+ deferred = {};
+
+ // Add list-specific methods
+ jQuery.each( tuples, function( i, tuple ) {
+ var list = tuple[ 2 ],
+ stateString = tuple[ 5 ];
+
+ // promise.progress = list.add
+ // promise.done = list.add
+ // promise.fail = list.add
+ promise[ tuple[ 1 ] ] = list.add;
+
+ // Handle state
+ if ( stateString ) {
+ list.add(
+ function() {
+
+ // state = "resolved" (i.e., fulfilled)
+ // state = "rejected"
+ state = stateString;
+ },
+
+ // rejected_callbacks.disable
+ // fulfilled_callbacks.disable
+ tuples[ 3 - i ][ 2 ].disable,
+
+ // rejected_handlers.disable
+ // fulfilled_handlers.disable
+ tuples[ 3 - i ][ 3 ].disable,
+
+ // progress_callbacks.lock
+ tuples[ 0 ][ 2 ].lock,
+
+ // progress_handlers.lock
+ tuples[ 0 ][ 3 ].lock
+ );
+ }
+
+ // progress_handlers.fire
+ // fulfilled_handlers.fire
+ // rejected_handlers.fire
+ list.add( tuple[ 3 ].fire );
+
+ // deferred.notify = function() { deferred.notifyWith(...) }
+ // deferred.resolve = function() { deferred.resolveWith(...) }
+ // deferred.reject = function() { deferred.rejectWith(...) }
+ deferred[ tuple[ 0 ] ] = function() {
+ deferred[ tuple[ 0 ] + "With" ]( this === deferred ? undefined : this, arguments );
+ return this;
+ };
+
+ // deferred.notifyWith = list.fireWith
+ // deferred.resolveWith = list.fireWith
+ // deferred.rejectWith = list.fireWith
+ deferred[ tuple[ 0 ] + "With" ] = list.fireWith;
+ } );
+
+ // Make the deferred a promise
+ promise.promise( deferred );
+
+ // Call given func if any
+ if ( func ) {
+ func.call( deferred, deferred );
+ }
+
+ // All done!
+ return deferred;
+ },
+
+ // Deferred helper
+ when: function( singleValue ) {
+ var
+
+ // count of uncompleted subordinates
+ remaining = arguments.length,
+
+ // count of unprocessed arguments
+ i = remaining,
+
+ // subordinate fulfillment data
+ resolveContexts = Array( i ),
+ resolveValues = slice.call( arguments ),
+
+ // the master Deferred
+ master = jQuery.Deferred(),
+
+ // subordinate callback factory
+ updateFunc = function( i ) {
+ return function( value ) {
+ resolveContexts[ i ] = this;
+ resolveValues[ i ] = arguments.length > 1 ? slice.call( arguments ) : value;
+ if ( !( --remaining ) ) {
+ master.resolveWith( resolveContexts, resolveValues );
+ }
+ };
+ };
+
+ // Single- and empty arguments are adopted like Promise.resolve
+ if ( remaining <= 1 ) {
+ adoptValue( singleValue, master.done( updateFunc( i ) ).resolve, master.reject,
+ !remaining );
+
+ // Use .then() to unwrap secondary thenables (cf. gh-3000)
+ if ( master.state() === "pending" ||
+ isFunction( resolveValues[ i ] && resolveValues[ i ].then ) ) {
+
+ return master.then();
+ }
+ }
+
+ // Multiple arguments are aggregated like Promise.all array elements
+ while ( i-- ) {
+ adoptValue( resolveValues[ i ], updateFunc( i ), master.reject );
+ }
+
+ return master.promise();
+ }
+} );
+
+
+// These usually indicate a programmer mistake during development,
+// warn about them ASAP rather than swallowing them by default.
+var rerrorNames = /^(Eval|Internal|Range|Reference|Syntax|Type|URI)Error$/;
+
+jQuery.Deferred.exceptionHook = function( error, stack ) {
+
+ // Support: IE 8 - 9 only
+ // Console exists when dev tools are open, which can happen at any time
+ if ( window.console && window.console.warn && error && rerrorNames.test( error.name ) ) {
+ window.console.warn( "jQuery.Deferred exception: " + error.message, error.stack, stack );
+ }
+};
+
+
+
+
+jQuery.readyException = function( error ) {
+ window.setTimeout( function() {
+ throw error;
+ } );
+};
+
+
+
+
+// The deferred used on DOM ready
+var readyList = jQuery.Deferred();
+
+jQuery.fn.ready = function( fn ) {
+
+ readyList
+ .then( fn )
+
+ // Wrap jQuery.readyException in a function so that the lookup
+ // happens at the time of error handling instead of callback
+ // registration.
+ .catch( function( error ) {
+ jQuery.readyException( error );
+ } );
+
+ return this;
+};
+
+jQuery.extend( {
+
+ // Is the DOM ready to be used? Set to true once it occurs.
+ isReady: false,
+
+ // A counter to track how many items to wait for before
+ // the ready event fires. See #6781
+ readyWait: 1,
+
+ // Handle when the DOM is ready
+ ready: function( wait ) {
+
+ // Abort if there are pending holds or we're already ready
+ if ( wait === true ? --jQuery.readyWait : jQuery.isReady ) {
+ return;
+ }
+
+ // Remember that the DOM is ready
+ jQuery.isReady = true;
+
+ // If a normal DOM Ready event fired, decrement, and wait if need be
+ if ( wait !== true && --jQuery.readyWait > 0 ) {
+ return;
+ }
+
+ // If there are functions bound, to execute
+ readyList.resolveWith( document, [ jQuery ] );
+ }
+} );
+
+jQuery.ready.then = readyList.then;
+
+// The ready event handler and self cleanup method
+function completed() {
+ document.removeEventListener( "DOMContentLoaded", completed );
+ window.removeEventListener( "load", completed );
+ jQuery.ready();
+}
+
+// Catch cases where $(document).ready() is called
+// after the browser event has already occurred.
+// Support: IE <=9 - 10 only
+// Older IE sometimes signals "interactive" too soon
+if ( document.readyState === "complete" ||
+ ( document.readyState !== "loading" && !document.documentElement.doScroll ) ) {
+
+ // Handle it asynchronously to allow scripts the opportunity to delay ready
+ window.setTimeout( jQuery.ready );
+
+} else {
+
+ // Use the handy event callback
+ document.addEventListener( "DOMContentLoaded", completed );
+
+ // A fallback to window.onload, that will always work
+ window.addEventListener( "load", completed );
+}
+
+
+
+
+// Multifunctional method to get and set values of a collection
+// The value/s can optionally be executed if it's a function
+var access = function( elems, fn, key, value, chainable, emptyGet, raw ) {
+ var i = 0,
+ len = elems.length,
+ bulk = key == null;
+
+ // Sets many values
+ if ( toType( key ) === "object" ) {
+ chainable = true;
+ for ( i in key ) {
+ access( elems, fn, i, key[ i ], true, emptyGet, raw );
+ }
+
+ // Sets one value
+ } else if ( value !== undefined ) {
+ chainable = true;
+
+ if ( !isFunction( value ) ) {
+ raw = true;
+ }
+
+ if ( bulk ) {
+
+ // Bulk operations run against the entire set
+ if ( raw ) {
+ fn.call( elems, value );
+ fn = null;
+
+ // ...except when executing function values
+ } else {
+ bulk = fn;
+ fn = function( elem, key, value ) {
+ return bulk.call( jQuery( elem ), value );
+ };
+ }
+ }
+
+ if ( fn ) {
+ for ( ; i < len; i++ ) {
+ fn(
+ elems[ i ], key, raw ?
+ value :
+ value.call( elems[ i ], i, fn( elems[ i ], key ) )
+ );
+ }
+ }
+ }
+
+ if ( chainable ) {
+ return elems;
+ }
+
+ // Gets
+ if ( bulk ) {
+ return fn.call( elems );
+ }
+
+ return len ? fn( elems[ 0 ], key ) : emptyGet;
+};
+
+
+// Matches dashed string for camelizing
+var rmsPrefix = /^-ms-/,
+ rdashAlpha = /-([a-z])/g;
+
+// Used by camelCase as callback to replace()
+function fcamelCase( all, letter ) {
+ return letter.toUpperCase();
+}
+
+// Convert dashed to camelCase; used by the css and data modules
+// Support: IE <=9 - 11, Edge 12 - 15
+// Microsoft forgot to hump their vendor prefix (#9572)
+function camelCase( string ) {
+ return string.replace( rmsPrefix, "ms-" ).replace( rdashAlpha, fcamelCase );
+}
+var acceptData = function( owner ) {
+
+ // Accepts only:
+ // - Node
+ // - Node.ELEMENT_NODE
+ // - Node.DOCUMENT_NODE
+ // - Object
+ // - Any
+ return owner.nodeType === 1 || owner.nodeType === 9 || !( +owner.nodeType );
+};
+
+
+
+
+function Data() {
+ this.expando = jQuery.expando + Data.uid++;
+}
+
+Data.uid = 1;
+
+Data.prototype = {
+
+ cache: function( owner ) {
+
+ // Check if the owner object already has a cache
+ var value = owner[ this.expando ];
+
+ // If not, create one
+ if ( !value ) {
+ value = {};
+
+ // We can accept data for non-element nodes in modern browsers,
+ // but we should not, see #8335.
+ // Always return an empty object.
+ if ( acceptData( owner ) ) {
+
+ // If it is a node unlikely to be stringify-ed or looped over
+ // use plain assignment
+ if ( owner.nodeType ) {
+ owner[ this.expando ] = value;
+
+ // Otherwise secure it in a non-enumerable property
+ // configurable must be true to allow the property to be
+ // deleted when data is removed
+ } else {
+ Object.defineProperty( owner, this.expando, {
+ value: value,
+ configurable: true
+ } );
+ }
+ }
+ }
+
+ return value;
+ },
+ set: function( owner, data, value ) {
+ var prop,
+ cache = this.cache( owner );
+
+ // Handle: [ owner, key, value ] args
+ // Always use camelCase key (gh-2257)
+ if ( typeof data === "string" ) {
+ cache[ camelCase( data ) ] = value;
+
+ // Handle: [ owner, { properties } ] args
+ } else {
+
+ // Copy the properties one-by-one to the cache object
+ for ( prop in data ) {
+ cache[ camelCase( prop ) ] = data[ prop ];
+ }
+ }
+ return cache;
+ },
+ get: function( owner, key ) {
+ return key === undefined ?
+ this.cache( owner ) :
+
+ // Always use camelCase key (gh-2257)
+ owner[ this.expando ] && owner[ this.expando ][ camelCase( key ) ];
+ },
+ access: function( owner, key, value ) {
+
+ // In cases where either:
+ //
+ // 1. No key was specified
+ // 2. A string key was specified, but no value provided
+ //
+ // Take the "read" path and allow the get method to determine
+ // which value to return, respectively either:
+ //
+ // 1. The entire cache object
+ // 2. The data stored at the key
+ //
+ if ( key === undefined ||
+ ( ( key && typeof key === "string" ) && value === undefined ) ) {
+
+ return this.get( owner, key );
+ }
+
+ // When the key is not a string, or both a key and value
+ // are specified, set or extend (existing objects) with either:
+ //
+ // 1. An object of properties
+ // 2. A key and value
+ //
+ this.set( owner, key, value );
+
+ // Since the "set" path can have two possible entry points
+ // return the expected data based on which path was taken[*]
+ return value !== undefined ? value : key;
+ },
+ remove: function( owner, key ) {
+ var i,
+ cache = owner[ this.expando ];
+
+ if ( cache === undefined ) {
+ return;
+ }
+
+ if ( key !== undefined ) {
+
+ // Support array or space separated string of keys
+ if ( Array.isArray( key ) ) {
+
+ // If key is an array of keys...
+ // We always set camelCase keys, so remove that.
+ key = key.map( camelCase );
+ } else {
+ key = camelCase( key );
+
+ // If a key with the spaces exists, use it.
+ // Otherwise, create an array by matching non-whitespace
+ key = key in cache ?
+ [ key ] :
+ ( key.match( rnothtmlwhite ) || [] );
+ }
+
+ i = key.length;
+
+ while ( i-- ) {
+ delete cache[ key[ i ] ];
+ }
+ }
+
+ // Remove the expando if there's no more data
+ if ( key === undefined || jQuery.isEmptyObject( cache ) ) {
+
+ // Support: Chrome <=35 - 45
+ // Webkit & Blink performance suffers when deleting properties
+ // from DOM nodes, so set to undefined instead
+ // https://bugs.chromium.org/p/chromium/issues/detail?id=378607 (bug restricted)
+ if ( owner.nodeType ) {
+ owner[ this.expando ] = undefined;
+ } else {
+ delete owner[ this.expando ];
+ }
+ }
+ },
+ hasData: function( owner ) {
+ var cache = owner[ this.expando ];
+ return cache !== undefined && !jQuery.isEmptyObject( cache );
+ }
+};
+var dataPriv = new Data();
+
+var dataUser = new Data();
+
+
+
+// Implementation Summary
+//
+// 1. Enforce API surface and semantic compatibility with 1.9.x branch
+// 2. Improve the module's maintainability by reducing the storage
+// paths to a single mechanism.
+// 3. Use the same single mechanism to support "private" and "user" data.
+// 4. _Never_ expose "private" data to user code (TODO: Drop _data, _removeData)
+// 5. Avoid exposing implementation details on user objects (eg. expando properties)
+// 6. Provide a clear path for implementation upgrade to WeakMap in 2014
+
+var rbrace = /^(?:\{[\w\W]*\}|\[[\w\W]*\])$/,
+ rmultiDash = /[A-Z]/g;
+
+function getData( data ) {
+ if ( data === "true" ) {
+ return true;
+ }
+
+ if ( data === "false" ) {
+ return false;
+ }
+
+ if ( data === "null" ) {
+ return null;
+ }
+
+ // Only convert to a number if it doesn't change the string
+ if ( data === +data + "" ) {
+ return +data;
+ }
+
+ if ( rbrace.test( data ) ) {
+ return JSON.parse( data );
+ }
+
+ return data;
+}
+
+function dataAttr( elem, key, data ) {
+ var name;
+
+ // If nothing was found internally, try to fetch any
+ // data from the HTML5 data-* attribute
+ if ( data === undefined && elem.nodeType === 1 ) {
+ name = "data-" + key.replace( rmultiDash, "-$&" ).toLowerCase();
+ data = elem.getAttribute( name );
+
+ if ( typeof data === "string" ) {
+ try {
+ data = getData( data );
+ } catch ( e ) {}
+
+ // Make sure we set the data so it isn't changed later
+ dataUser.set( elem, key, data );
+ } else {
+ data = undefined;
+ }
+ }
+ return data;
+}
+
+jQuery.extend( {
+ hasData: function( elem ) {
+ return dataUser.hasData( elem ) || dataPriv.hasData( elem );
+ },
+
+ data: function( elem, name, data ) {
+ return dataUser.access( elem, name, data );
+ },
+
+ removeData: function( elem, name ) {
+ dataUser.remove( elem, name );
+ },
+
+ // TODO: Now that all calls to _data and _removeData have been replaced
+ // with direct calls to dataPriv methods, these can be deprecated.
+ _data: function( elem, name, data ) {
+ return dataPriv.access( elem, name, data );
+ },
+
+ _removeData: function( elem, name ) {
+ dataPriv.remove( elem, name );
+ }
+} );
+
+jQuery.fn.extend( {
+ data: function( key, value ) {
+ var i, name, data,
+ elem = this[ 0 ],
+ attrs = elem && elem.attributes;
+
+ // Gets all values
+ if ( key === undefined ) {
+ if ( this.length ) {
+ data = dataUser.get( elem );
+
+ if ( elem.nodeType === 1 && !dataPriv.get( elem, "hasDataAttrs" ) ) {
+ i = attrs.length;
+ while ( i-- ) {
+
+ // Support: IE 11 only
+ // The attrs elements can be null (#14894)
+ if ( attrs[ i ] ) {
+ name = attrs[ i ].name;
+ if ( name.indexOf( "data-" ) === 0 ) {
+ name = camelCase( name.slice( 5 ) );
+ dataAttr( elem, name, data[ name ] );
+ }
+ }
+ }
+ dataPriv.set( elem, "hasDataAttrs", true );
+ }
+ }
+
+ return data;
+ }
+
+ // Sets multiple values
+ if ( typeof key === "object" ) {
+ return this.each( function() {
+ dataUser.set( this, key );
+ } );
+ }
+
+ return access( this, function( value ) {
+ var data;
+
+ // The calling jQuery object (element matches) is not empty
+ // (and therefore has an element appears at this[ 0 ]) and the
+ // `value` parameter was not undefined. An empty jQuery object
+ // will result in `undefined` for elem = this[ 0 ] which will
+ // throw an exception if an attempt to read a data cache is made.
+ if ( elem && value === undefined ) {
+
+ // Attempt to get data from the cache
+ // The key will always be camelCased in Data
+ data = dataUser.get( elem, key );
+ if ( data !== undefined ) {
+ return data;
+ }
+
+ // Attempt to "discover" the data in
+ // HTML5 custom data-* attrs
+ data = dataAttr( elem, key );
+ if ( data !== undefined ) {
+ return data;
+ }
+
+ // We tried really hard, but the data doesn't exist.
+ return;
+ }
+
+ // Set the data...
+ this.each( function() {
+
+ // We always store the camelCased key
+ dataUser.set( this, key, value );
+ } );
+ }, null, value, arguments.length > 1, null, true );
+ },
+
+ removeData: function( key ) {
+ return this.each( function() {
+ dataUser.remove( this, key );
+ } );
+ }
+} );
+
+
+jQuery.extend( {
+ queue: function( elem, type, data ) {
+ var queue;
+
+ if ( elem ) {
+ type = ( type || "fx" ) + "queue";
+ queue = dataPriv.get( elem, type );
+
+ // Speed up dequeue by getting out quickly if this is just a lookup
+ if ( data ) {
+ if ( !queue || Array.isArray( data ) ) {
+ queue = dataPriv.access( elem, type, jQuery.makeArray( data ) );
+ } else {
+ queue.push( data );
+ }
+ }
+ return queue || [];
+ }
+ },
+
+ dequeue: function( elem, type ) {
+ type = type || "fx";
+
+ var queue = jQuery.queue( elem, type ),
+ startLength = queue.length,
+ fn = queue.shift(),
+ hooks = jQuery._queueHooks( elem, type ),
+ next = function() {
+ jQuery.dequeue( elem, type );
+ };
+
+ // If the fx queue is dequeued, always remove the progress sentinel
+ if ( fn === "inprogress" ) {
+ fn = queue.shift();
+ startLength--;
+ }
+
+ if ( fn ) {
+
+ // Add a progress sentinel to prevent the fx queue from being
+ // automatically dequeued
+ if ( type === "fx" ) {
+ queue.unshift( "inprogress" );
+ }
+
+ // Clear up the last queue stop function
+ delete hooks.stop;
+ fn.call( elem, next, hooks );
+ }
+
+ if ( !startLength && hooks ) {
+ hooks.empty.fire();
+ }
+ },
+
+ // Not public - generate a queueHooks object, or return the current one
+ _queueHooks: function( elem, type ) {
+ var key = type + "queueHooks";
+ return dataPriv.get( elem, key ) || dataPriv.access( elem, key, {
+ empty: jQuery.Callbacks( "once memory" ).add( function() {
+ dataPriv.remove( elem, [ type + "queue", key ] );
+ } )
+ } );
+ }
+} );
+
+jQuery.fn.extend( {
+ queue: function( type, data ) {
+ var setter = 2;
+
+ if ( typeof type !== "string" ) {
+ data = type;
+ type = "fx";
+ setter--;
+ }
+
+ if ( arguments.length < setter ) {
+ return jQuery.queue( this[ 0 ], type );
+ }
+
+ return data === undefined ?
+ this :
+ this.each( function() {
+ var queue = jQuery.queue( this, type, data );
+
+ // Ensure a hooks for this queue
+ jQuery._queueHooks( this, type );
+
+ if ( type === "fx" && queue[ 0 ] !== "inprogress" ) {
+ jQuery.dequeue( this, type );
+ }
+ } );
+ },
+ dequeue: function( type ) {
+ return this.each( function() {
+ jQuery.dequeue( this, type );
+ } );
+ },
+ clearQueue: function( type ) {
+ return this.queue( type || "fx", [] );
+ },
+
+ // Get a promise resolved when queues of a certain type
+ // are emptied (fx is the type by default)
+ promise: function( type, obj ) {
+ var tmp,
+ count = 1,
+ defer = jQuery.Deferred(),
+ elements = this,
+ i = this.length,
+ resolve = function() {
+ if ( !( --count ) ) {
+ defer.resolveWith( elements, [ elements ] );
+ }
+ };
+
+ if ( typeof type !== "string" ) {
+ obj = type;
+ type = undefined;
+ }
+ type = type || "fx";
+
+ while ( i-- ) {
+ tmp = dataPriv.get( elements[ i ], type + "queueHooks" );
+ if ( tmp && tmp.empty ) {
+ count++;
+ tmp.empty.add( resolve );
+ }
+ }
+ resolve();
+ return defer.promise( obj );
+ }
+} );
+var pnum = ( /[+-]?(?:\d*\.|)\d+(?:[eE][+-]?\d+|)/ ).source;
+
+var rcssNum = new RegExp( "^(?:([+-])=|)(" + pnum + ")([a-z%]*)$", "i" );
+
+
+var cssExpand = [ "Top", "Right", "Bottom", "Left" ];
+
+var documentElement = document.documentElement;
+
+
+
+ var isAttached = function( elem ) {
+ return jQuery.contains( elem.ownerDocument, elem );
+ },
+ composed = { composed: true };
+
+ // Support: IE 9 - 11+, Edge 12 - 18+, iOS 10.0 - 10.2 only
+ // Check attachment across shadow DOM boundaries when possible (gh-3504)
+ // Support: iOS 10.0-10.2 only
+ // Early iOS 10 versions support `attachShadow` but not `getRootNode`,
+ // leading to errors. We need to check for `getRootNode`.
+ if ( documentElement.getRootNode ) {
+ isAttached = function( elem ) {
+ return jQuery.contains( elem.ownerDocument, elem ) ||
+ elem.getRootNode( composed ) === elem.ownerDocument;
+ };
+ }
+var isHiddenWithinTree = function( elem, el ) {
+
+ // isHiddenWithinTree might be called from jQuery#filter function;
+ // in that case, element will be second argument
+ elem = el || elem;
+
+ // Inline style trumps all
+ return elem.style.display === "none" ||
+ elem.style.display === "" &&
+
+ // Otherwise, check computed style
+ // Support: Firefox <=43 - 45
+ // Disconnected elements can have computed display: none, so first confirm that elem is
+ // in the document.
+ isAttached( elem ) &&
+
+ jQuery.css( elem, "display" ) === "none";
+ };
+
+var swap = function( elem, options, callback, args ) {
+ var ret, name,
+ old = {};
+
+ // Remember the old values, and insert the new ones
+ for ( name in options ) {
+ old[ name ] = elem.style[ name ];
+ elem.style[ name ] = options[ name ];
+ }
+
+ ret = callback.apply( elem, args || [] );
+
+ // Revert the old values
+ for ( name in options ) {
+ elem.style[ name ] = old[ name ];
+ }
+
+ return ret;
+};
+
+
+
+
+function adjustCSS( elem, prop, valueParts, tween ) {
+ var adjusted, scale,
+ maxIterations = 20,
+ currentValue = tween ?
+ function() {
+ return tween.cur();
+ } :
+ function() {
+ return jQuery.css( elem, prop, "" );
+ },
+ initial = currentValue(),
+ unit = valueParts && valueParts[ 3 ] || ( jQuery.cssNumber[ prop ] ? "" : "px" ),
+
+ // Starting value computation is required for potential unit mismatches
+ initialInUnit = elem.nodeType &&
+ ( jQuery.cssNumber[ prop ] || unit !== "px" && +initial ) &&
+ rcssNum.exec( jQuery.css( elem, prop ) );
+
+ if ( initialInUnit && initialInUnit[ 3 ] !== unit ) {
+
+ // Support: Firefox <=54
+ // Halve the iteration target value to prevent interference from CSS upper bounds (gh-2144)
+ initial = initial / 2;
+
+ // Trust units reported by jQuery.css
+ unit = unit || initialInUnit[ 3 ];
+
+ // Iteratively approximate from a nonzero starting point
+ initialInUnit = +initial || 1;
+
+ while ( maxIterations-- ) {
+
+ // Evaluate and update our best guess (doubling guesses that zero out).
+ // Finish if the scale equals or crosses 1 (making the old*new product non-positive).
+ jQuery.style( elem, prop, initialInUnit + unit );
+ if ( ( 1 - scale ) * ( 1 - ( scale = currentValue() / initial || 0.5 ) ) <= 0 ) {
+ maxIterations = 0;
+ }
+ initialInUnit = initialInUnit / scale;
+
+ }
+
+ initialInUnit = initialInUnit * 2;
+ jQuery.style( elem, prop, initialInUnit + unit );
+
+ // Make sure we update the tween properties later on
+ valueParts = valueParts || [];
+ }
+
+ if ( valueParts ) {
+ initialInUnit = +initialInUnit || +initial || 0;
+
+ // Apply relative offset (+=/-=) if specified
+ adjusted = valueParts[ 1 ] ?
+ initialInUnit + ( valueParts[ 1 ] + 1 ) * valueParts[ 2 ] :
+ +valueParts[ 2 ];
+ if ( tween ) {
+ tween.unit = unit;
+ tween.start = initialInUnit;
+ tween.end = adjusted;
+ }
+ }
+ return adjusted;
+}
+
+
+var defaultDisplayMap = {};
+
+function getDefaultDisplay( elem ) {
+ var temp,
+ doc = elem.ownerDocument,
+ nodeName = elem.nodeName,
+ display = defaultDisplayMap[ nodeName ];
+
+ if ( display ) {
+ return display;
+ }
+
+ temp = doc.body.appendChild( doc.createElement( nodeName ) );
+ display = jQuery.css( temp, "display" );
+
+ temp.parentNode.removeChild( temp );
+
+ if ( display === "none" ) {
+ display = "block";
+ }
+ defaultDisplayMap[ nodeName ] = display;
+
+ return display;
+}
+
+function showHide( elements, show ) {
+ var display, elem,
+ values = [],
+ index = 0,
+ length = elements.length;
+
+ // Determine new display value for elements that need to change
+ for ( ; index < length; index++ ) {
+ elem = elements[ index ];
+ if ( !elem.style ) {
+ continue;
+ }
+
+ display = elem.style.display;
+ if ( show ) {
+
+ // Since we force visibility upon cascade-hidden elements, an immediate (and slow)
+ // check is required in this first loop unless we have a nonempty display value (either
+ // inline or about-to-be-restored)
+ if ( display === "none" ) {
+ values[ index ] = dataPriv.get( elem, "display" ) || null;
+ if ( !values[ index ] ) {
+ elem.style.display = "";
+ }
+ }
+ if ( elem.style.display === "" && isHiddenWithinTree( elem ) ) {
+ values[ index ] = getDefaultDisplay( elem );
+ }
+ } else {
+ if ( display !== "none" ) {
+ values[ index ] = "none";
+
+ // Remember what we're overwriting
+ dataPriv.set( elem, "display", display );
+ }
+ }
+ }
+
+ // Set the display of the elements in a second loop to avoid constant reflow
+ for ( index = 0; index < length; index++ ) {
+ if ( values[ index ] != null ) {
+ elements[ index ].style.display = values[ index ];
+ }
+ }
+
+ return elements;
+}
+
+jQuery.fn.extend( {
+ show: function() {
+ return showHide( this, true );
+ },
+ hide: function() {
+ return showHide( this );
+ },
+ toggle: function( state ) {
+ if ( typeof state === "boolean" ) {
+ return state ? this.show() : this.hide();
+ }
+
+ return this.each( function() {
+ if ( isHiddenWithinTree( this ) ) {
+ jQuery( this ).show();
+ } else {
+ jQuery( this ).hide();
+ }
+ } );
+ }
+} );
+var rcheckableType = ( /^(?:checkbox|radio)$/i );
+
+var rtagName = ( /<([a-z][^\/\0>\x20\t\r\n\f]*)/i );
+
+var rscriptType = ( /^$|^module$|\/(?:java|ecma)script/i );
+
+
+
+// We have to close these tags to support XHTML (#13200)
+var wrapMap = {
+
+ // Support: IE <=9 only
+ option: [ 1, "", " " ],
+
+ // XHTML parsers do not magically insert elements in the
+ // same way that tag soup parsers do. So we cannot shorten
+ // this by omitting or other required elements.
+ thead: [ 1, "" ],
+ col: [ 2, "" ],
+ tr: [ 2, "" ],
+ td: [ 3, "" ],
+
+ _default: [ 0, "", "" ]
+};
+
+// Support: IE <=9 only
+wrapMap.optgroup = wrapMap.option;
+
+wrapMap.tbody = wrapMap.tfoot = wrapMap.colgroup = wrapMap.caption = wrapMap.thead;
+wrapMap.th = wrapMap.td;
+
+
+function getAll( context, tag ) {
+
+ // Support: IE <=9 - 11 only
+ // Use typeof to avoid zero-argument method invocation on host objects (#15151)
+ var ret;
+
+ if ( typeof context.getElementsByTagName !== "undefined" ) {
+ ret = context.getElementsByTagName( tag || "*" );
+
+ } else if ( typeof context.querySelectorAll !== "undefined" ) {
+ ret = context.querySelectorAll( tag || "*" );
+
+ } else {
+ ret = [];
+ }
+
+ if ( tag === undefined || tag && nodeName( context, tag ) ) {
+ return jQuery.merge( [ context ], ret );
+ }
+
+ return ret;
+}
+
+
+// Mark scripts as having already been evaluated
+function setGlobalEval( elems, refElements ) {
+ var i = 0,
+ l = elems.length;
+
+ for ( ; i < l; i++ ) {
+ dataPriv.set(
+ elems[ i ],
+ "globalEval",
+ !refElements || dataPriv.get( refElements[ i ], "globalEval" )
+ );
+ }
+}
+
+
+var rhtml = /<|?\w+;/;
+
+function buildFragment( elems, context, scripts, selection, ignored ) {
+ var elem, tmp, tag, wrap, attached, j,
+ fragment = context.createDocumentFragment(),
+ nodes = [],
+ i = 0,
+ l = elems.length;
+
+ for ( ; i < l; i++ ) {
+ elem = elems[ i ];
+
+ if ( elem || elem === 0 ) {
+
+ // Add nodes directly
+ if ( toType( elem ) === "object" ) {
+
+ // Support: Android <=4.0 only, PhantomJS 1 only
+ // push.apply(_, arraylike) throws on ancient WebKit
+ jQuery.merge( nodes, elem.nodeType ? [ elem ] : elem );
+
+ // Convert non-html into a text node
+ } else if ( !rhtml.test( elem ) ) {
+ nodes.push( context.createTextNode( elem ) );
+
+ // Convert html into DOM nodes
+ } else {
+ tmp = tmp || fragment.appendChild( context.createElement( "div" ) );
+
+ // Deserialize a standard representation
+ tag = ( rtagName.exec( elem ) || [ "", "" ] )[ 1 ].toLowerCase();
+ wrap = wrapMap[ tag ] || wrapMap._default;
+ tmp.innerHTML = wrap[ 1 ] + jQuery.htmlPrefilter( elem ) + wrap[ 2 ];
+
+ // Descend through wrappers to the right content
+ j = wrap[ 0 ];
+ while ( j-- ) {
+ tmp = tmp.lastChild;
+ }
+
+ // Support: Android <=4.0 only, PhantomJS 1 only
+ // push.apply(_, arraylike) throws on ancient WebKit
+ jQuery.merge( nodes, tmp.childNodes );
+
+ // Remember the top-level container
+ tmp = fragment.firstChild;
+
+ // Ensure the created nodes are orphaned (#12392)
+ tmp.textContent = "";
+ }
+ }
+ }
+
+ // Remove wrapper from fragment
+ fragment.textContent = "";
+
+ i = 0;
+ while ( ( elem = nodes[ i++ ] ) ) {
+
+ // Skip elements already in the context collection (trac-4087)
+ if ( selection && jQuery.inArray( elem, selection ) > -1 ) {
+ if ( ignored ) {
+ ignored.push( elem );
+ }
+ continue;
+ }
+
+ attached = isAttached( elem );
+
+ // Append to fragment
+ tmp = getAll( fragment.appendChild( elem ), "script" );
+
+ // Preserve script evaluation history
+ if ( attached ) {
+ setGlobalEval( tmp );
+ }
+
+ // Capture executables
+ if ( scripts ) {
+ j = 0;
+ while ( ( elem = tmp[ j++ ] ) ) {
+ if ( rscriptType.test( elem.type || "" ) ) {
+ scripts.push( elem );
+ }
+ }
+ }
+ }
+
+ return fragment;
+}
+
+
+( function() {
+ var fragment = document.createDocumentFragment(),
+ div = fragment.appendChild( document.createElement( "div" ) ),
+ input = document.createElement( "input" );
+
+ // Support: Android 4.0 - 4.3 only
+ // Check state lost if the name is set (#11217)
+ // Support: Windows Web Apps (WWA)
+ // `name` and `type` must use .setAttribute for WWA (#14901)
+ input.setAttribute( "type", "radio" );
+ input.setAttribute( "checked", "checked" );
+ input.setAttribute( "name", "t" );
+
+ div.appendChild( input );
+
+ // Support: Android <=4.1 only
+ // Older WebKit doesn't clone checked state correctly in fragments
+ support.checkClone = div.cloneNode( true ).cloneNode( true ).lastChild.checked;
+
+ // Support: IE <=11 only
+ // Make sure textarea (and checkbox) defaultValue is properly cloned
+ div.innerHTML = "";
+ support.noCloneChecked = !!div.cloneNode( true ).lastChild.defaultValue;
+} )();
+
+
+var
+ rkeyEvent = /^key/,
+ rmouseEvent = /^(?:mouse|pointer|contextmenu|drag|drop)|click/,
+ rtypenamespace = /^([^.]*)(?:\.(.+)|)/;
+
+function returnTrue() {
+ return true;
+}
+
+function returnFalse() {
+ return false;
+}
+
+// Support: IE <=9 - 11+
+// focus() and blur() are asynchronous, except when they are no-op.
+// So expect focus to be synchronous when the element is already active,
+// and blur to be synchronous when the element is not already active.
+// (focus and blur are always synchronous in other supported browsers,
+// this just defines when we can count on it).
+function expectSync( elem, type ) {
+ return ( elem === safeActiveElement() ) === ( type === "focus" );
+}
+
+// Support: IE <=9 only
+// Accessing document.activeElement can throw unexpectedly
+// https://bugs.jquery.com/ticket/13393
+function safeActiveElement() {
+ try {
+ return document.activeElement;
+ } catch ( err ) { }
+}
+
+function on( elem, types, selector, data, fn, one ) {
+ var origFn, type;
+
+ // Types can be a map of types/handlers
+ if ( typeof types === "object" ) {
+
+ // ( types-Object, selector, data )
+ if ( typeof selector !== "string" ) {
+
+ // ( types-Object, data )
+ data = data || selector;
+ selector = undefined;
+ }
+ for ( type in types ) {
+ on( elem, type, selector, data, types[ type ], one );
+ }
+ return elem;
+ }
+
+ if ( data == null && fn == null ) {
+
+ // ( types, fn )
+ fn = selector;
+ data = selector = undefined;
+ } else if ( fn == null ) {
+ if ( typeof selector === "string" ) {
+
+ // ( types, selector, fn )
+ fn = data;
+ data = undefined;
+ } else {
+
+ // ( types, data, fn )
+ fn = data;
+ data = selector;
+ selector = undefined;
+ }
+ }
+ if ( fn === false ) {
+ fn = returnFalse;
+ } else if ( !fn ) {
+ return elem;
+ }
+
+ if ( one === 1 ) {
+ origFn = fn;
+ fn = function( event ) {
+
+ // Can use an empty set, since event contains the info
+ jQuery().off( event );
+ return origFn.apply( this, arguments );
+ };
+
+ // Use same guid so caller can remove using origFn
+ fn.guid = origFn.guid || ( origFn.guid = jQuery.guid++ );
+ }
+ return elem.each( function() {
+ jQuery.event.add( this, types, fn, data, selector );
+ } );
+}
+
+/*
+ * Helper functions for managing events -- not part of the public interface.
+ * Props to Dean Edwards' addEvent library for many of the ideas.
+ */
+jQuery.event = {
+
+ global: {},
+
+ add: function( elem, types, handler, data, selector ) {
+
+ var handleObjIn, eventHandle, tmp,
+ events, t, handleObj,
+ special, handlers, type, namespaces, origType,
+ elemData = dataPriv.get( elem );
+
+ // Don't attach events to noData or text/comment nodes (but allow plain objects)
+ if ( !elemData ) {
+ return;
+ }
+
+ // Caller can pass in an object of custom data in lieu of the handler
+ if ( handler.handler ) {
+ handleObjIn = handler;
+ handler = handleObjIn.handler;
+ selector = handleObjIn.selector;
+ }
+
+ // Ensure that invalid selectors throw exceptions at attach time
+ // Evaluate against documentElement in case elem is a non-element node (e.g., document)
+ if ( selector ) {
+ jQuery.find.matchesSelector( documentElement, selector );
+ }
+
+ // Make sure that the handler has a unique ID, used to find/remove it later
+ if ( !handler.guid ) {
+ handler.guid = jQuery.guid++;
+ }
+
+ // Init the element's event structure and main handler, if this is the first
+ if ( !( events = elemData.events ) ) {
+ events = elemData.events = {};
+ }
+ if ( !( eventHandle = elemData.handle ) ) {
+ eventHandle = elemData.handle = function( e ) {
+
+ // Discard the second event of a jQuery.event.trigger() and
+ // when an event is called after a page has unloaded
+ return typeof jQuery !== "undefined" && jQuery.event.triggered !== e.type ?
+ jQuery.event.dispatch.apply( elem, arguments ) : undefined;
+ };
+ }
+
+ // Handle multiple events separated by a space
+ types = ( types || "" ).match( rnothtmlwhite ) || [ "" ];
+ t = types.length;
+ while ( t-- ) {
+ tmp = rtypenamespace.exec( types[ t ] ) || [];
+ type = origType = tmp[ 1 ];
+ namespaces = ( tmp[ 2 ] || "" ).split( "." ).sort();
+
+ // There *must* be a type, no attaching namespace-only handlers
+ if ( !type ) {
+ continue;
+ }
+
+ // If event changes its type, use the special event handlers for the changed type
+ special = jQuery.event.special[ type ] || {};
+
+ // If selector defined, determine special event api type, otherwise given type
+ type = ( selector ? special.delegateType : special.bindType ) || type;
+
+ // Update special based on newly reset type
+ special = jQuery.event.special[ type ] || {};
+
+ // handleObj is passed to all event handlers
+ handleObj = jQuery.extend( {
+ type: type,
+ origType: origType,
+ data: data,
+ handler: handler,
+ guid: handler.guid,
+ selector: selector,
+ needsContext: selector && jQuery.expr.match.needsContext.test( selector ),
+ namespace: namespaces.join( "." )
+ }, handleObjIn );
+
+ // Init the event handler queue if we're the first
+ if ( !( handlers = events[ type ] ) ) {
+ handlers = events[ type ] = [];
+ handlers.delegateCount = 0;
+
+ // Only use addEventListener if the special events handler returns false
+ if ( !special.setup ||
+ special.setup.call( elem, data, namespaces, eventHandle ) === false ) {
+
+ if ( elem.addEventListener ) {
+ elem.addEventListener( type, eventHandle );
+ }
+ }
+ }
+
+ if ( special.add ) {
+ special.add.call( elem, handleObj );
+
+ if ( !handleObj.handler.guid ) {
+ handleObj.handler.guid = handler.guid;
+ }
+ }
+
+ // Add to the element's handler list, delegates in front
+ if ( selector ) {
+ handlers.splice( handlers.delegateCount++, 0, handleObj );
+ } else {
+ handlers.push( handleObj );
+ }
+
+ // Keep track of which events have ever been used, for event optimization
+ jQuery.event.global[ type ] = true;
+ }
+
+ },
+
+ // Detach an event or set of events from an element
+ remove: function( elem, types, handler, selector, mappedTypes ) {
+
+ var j, origCount, tmp,
+ events, t, handleObj,
+ special, handlers, type, namespaces, origType,
+ elemData = dataPriv.hasData( elem ) && dataPriv.get( elem );
+
+ if ( !elemData || !( events = elemData.events ) ) {
+ return;
+ }
+
+ // Once for each type.namespace in types; type may be omitted
+ types = ( types || "" ).match( rnothtmlwhite ) || [ "" ];
+ t = types.length;
+ while ( t-- ) {
+ tmp = rtypenamespace.exec( types[ t ] ) || [];
+ type = origType = tmp[ 1 ];
+ namespaces = ( tmp[ 2 ] || "" ).split( "." ).sort();
+
+ // Unbind all events (on this namespace, if provided) for the element
+ if ( !type ) {
+ for ( type in events ) {
+ jQuery.event.remove( elem, type + types[ t ], handler, selector, true );
+ }
+ continue;
+ }
+
+ special = jQuery.event.special[ type ] || {};
+ type = ( selector ? special.delegateType : special.bindType ) || type;
+ handlers = events[ type ] || [];
+ tmp = tmp[ 2 ] &&
+ new RegExp( "(^|\\.)" + namespaces.join( "\\.(?:.*\\.|)" ) + "(\\.|$)" );
+
+ // Remove matching events
+ origCount = j = handlers.length;
+ while ( j-- ) {
+ handleObj = handlers[ j ];
+
+ if ( ( mappedTypes || origType === handleObj.origType ) &&
+ ( !handler || handler.guid === handleObj.guid ) &&
+ ( !tmp || tmp.test( handleObj.namespace ) ) &&
+ ( !selector || selector === handleObj.selector ||
+ selector === "**" && handleObj.selector ) ) {
+ handlers.splice( j, 1 );
+
+ if ( handleObj.selector ) {
+ handlers.delegateCount--;
+ }
+ if ( special.remove ) {
+ special.remove.call( elem, handleObj );
+ }
+ }
+ }
+
+ // Remove generic event handler if we removed something and no more handlers exist
+ // (avoids potential for endless recursion during removal of special event handlers)
+ if ( origCount && !handlers.length ) {
+ if ( !special.teardown ||
+ special.teardown.call( elem, namespaces, elemData.handle ) === false ) {
+
+ jQuery.removeEvent( elem, type, elemData.handle );
+ }
+
+ delete events[ type ];
+ }
+ }
+
+ // Remove data and the expando if it's no longer used
+ if ( jQuery.isEmptyObject( events ) ) {
+ dataPriv.remove( elem, "handle events" );
+ }
+ },
+
+ dispatch: function( nativeEvent ) {
+
+ // Make a writable jQuery.Event from the native event object
+ var event = jQuery.event.fix( nativeEvent );
+
+ var i, j, ret, matched, handleObj, handlerQueue,
+ args = new Array( arguments.length ),
+ handlers = ( dataPriv.get( this, "events" ) || {} )[ event.type ] || [],
+ special = jQuery.event.special[ event.type ] || {};
+
+ // Use the fix-ed jQuery.Event rather than the (read-only) native event
+ args[ 0 ] = event;
+
+ for ( i = 1; i < arguments.length; i++ ) {
+ args[ i ] = arguments[ i ];
+ }
+
+ event.delegateTarget = this;
+
+ // Call the preDispatch hook for the mapped type, and let it bail if desired
+ if ( special.preDispatch && special.preDispatch.call( this, event ) === false ) {
+ return;
+ }
+
+ // Determine handlers
+ handlerQueue = jQuery.event.handlers.call( this, event, handlers );
+
+ // Run delegates first; they may want to stop propagation beneath us
+ i = 0;
+ while ( ( matched = handlerQueue[ i++ ] ) && !event.isPropagationStopped() ) {
+ event.currentTarget = matched.elem;
+
+ j = 0;
+ while ( ( handleObj = matched.handlers[ j++ ] ) &&
+ !event.isImmediatePropagationStopped() ) {
+
+ // If the event is namespaced, then each handler is only invoked if it is
+ // specially universal or its namespaces are a superset of the event's.
+ if ( !event.rnamespace || handleObj.namespace === false ||
+ event.rnamespace.test( handleObj.namespace ) ) {
+
+ event.handleObj = handleObj;
+ event.data = handleObj.data;
+
+ ret = ( ( jQuery.event.special[ handleObj.origType ] || {} ).handle ||
+ handleObj.handler ).apply( matched.elem, args );
+
+ if ( ret !== undefined ) {
+ if ( ( event.result = ret ) === false ) {
+ event.preventDefault();
+ event.stopPropagation();
+ }
+ }
+ }
+ }
+ }
+
+ // Call the postDispatch hook for the mapped type
+ if ( special.postDispatch ) {
+ special.postDispatch.call( this, event );
+ }
+
+ return event.result;
+ },
+
+ handlers: function( event, handlers ) {
+ var i, handleObj, sel, matchedHandlers, matchedSelectors,
+ handlerQueue = [],
+ delegateCount = handlers.delegateCount,
+ cur = event.target;
+
+ // Find delegate handlers
+ if ( delegateCount &&
+
+ // Support: IE <=9
+ // Black-hole SVG instance trees (trac-13180)
+ cur.nodeType &&
+
+ // Support: Firefox <=42
+ // Suppress spec-violating clicks indicating a non-primary pointer button (trac-3861)
+ // https://www.w3.org/TR/DOM-Level-3-Events/#event-type-click
+ // Support: IE 11 only
+ // ...but not arrow key "clicks" of radio inputs, which can have `button` -1 (gh-2343)
+ !( event.type === "click" && event.button >= 1 ) ) {
+
+ for ( ; cur !== this; cur = cur.parentNode || this ) {
+
+ // Don't check non-elements (#13208)
+ // Don't process clicks on disabled elements (#6911, #8165, #11382, #11764)
+ if ( cur.nodeType === 1 && !( event.type === "click" && cur.disabled === true ) ) {
+ matchedHandlers = [];
+ matchedSelectors = {};
+ for ( i = 0; i < delegateCount; i++ ) {
+ handleObj = handlers[ i ];
+
+ // Don't conflict with Object.prototype properties (#13203)
+ sel = handleObj.selector + " ";
+
+ if ( matchedSelectors[ sel ] === undefined ) {
+ matchedSelectors[ sel ] = handleObj.needsContext ?
+ jQuery( sel, this ).index( cur ) > -1 :
+ jQuery.find( sel, this, null, [ cur ] ).length;
+ }
+ if ( matchedSelectors[ sel ] ) {
+ matchedHandlers.push( handleObj );
+ }
+ }
+ if ( matchedHandlers.length ) {
+ handlerQueue.push( { elem: cur, handlers: matchedHandlers } );
+ }
+ }
+ }
+ }
+
+ // Add the remaining (directly-bound) handlers
+ cur = this;
+ if ( delegateCount < handlers.length ) {
+ handlerQueue.push( { elem: cur, handlers: handlers.slice( delegateCount ) } );
+ }
+
+ return handlerQueue;
+ },
+
+ addProp: function( name, hook ) {
+ Object.defineProperty( jQuery.Event.prototype, name, {
+ enumerable: true,
+ configurable: true,
+
+ get: isFunction( hook ) ?
+ function() {
+ if ( this.originalEvent ) {
+ return hook( this.originalEvent );
+ }
+ } :
+ function() {
+ if ( this.originalEvent ) {
+ return this.originalEvent[ name ];
+ }
+ },
+
+ set: function( value ) {
+ Object.defineProperty( this, name, {
+ enumerable: true,
+ configurable: true,
+ writable: true,
+ value: value
+ } );
+ }
+ } );
+ },
+
+ fix: function( originalEvent ) {
+ return originalEvent[ jQuery.expando ] ?
+ originalEvent :
+ new jQuery.Event( originalEvent );
+ },
+
+ special: {
+ load: {
+
+ // Prevent triggered image.load events from bubbling to window.load
+ noBubble: true
+ },
+ click: {
+
+ // Utilize native event to ensure correct state for checkable inputs
+ setup: function( data ) {
+
+ // For mutual compressibility with _default, replace `this` access with a local var.
+ // `|| data` is dead code meant only to preserve the variable through minification.
+ var el = this || data;
+
+ // Claim the first handler
+ if ( rcheckableType.test( el.type ) &&
+ el.click && nodeName( el, "input" ) ) {
+
+ // dataPriv.set( el, "click", ... )
+ leverageNative( el, "click", returnTrue );
+ }
+
+ // Return false to allow normal processing in the caller
+ return false;
+ },
+ trigger: function( data ) {
+
+ // For mutual compressibility with _default, replace `this` access with a local var.
+ // `|| data` is dead code meant only to preserve the variable through minification.
+ var el = this || data;
+
+ // Force setup before triggering a click
+ if ( rcheckableType.test( el.type ) &&
+ el.click && nodeName( el, "input" ) ) {
+
+ leverageNative( el, "click" );
+ }
+
+ // Return non-false to allow normal event-path propagation
+ return true;
+ },
+
+ // For cross-browser consistency, suppress native .click() on links
+ // Also prevent it if we're currently inside a leveraged native-event stack
+ _default: function( event ) {
+ var target = event.target;
+ return rcheckableType.test( target.type ) &&
+ target.click && nodeName( target, "input" ) &&
+ dataPriv.get( target, "click" ) ||
+ nodeName( target, "a" );
+ }
+ },
+
+ beforeunload: {
+ postDispatch: function( event ) {
+
+ // Support: Firefox 20+
+ // Firefox doesn't alert if the returnValue field is not set.
+ if ( event.result !== undefined && event.originalEvent ) {
+ event.originalEvent.returnValue = event.result;
+ }
+ }
+ }
+ }
+};
+
+// Ensure the presence of an event listener that handles manually-triggered
+// synthetic events by interrupting progress until reinvoked in response to
+// *native* events that it fires directly, ensuring that state changes have
+// already occurred before other listeners are invoked.
+function leverageNative( el, type, expectSync ) {
+
+ // Missing expectSync indicates a trigger call, which must force setup through jQuery.event.add
+ if ( !expectSync ) {
+ if ( dataPriv.get( el, type ) === undefined ) {
+ jQuery.event.add( el, type, returnTrue );
+ }
+ return;
+ }
+
+ // Register the controller as a special universal handler for all event namespaces
+ dataPriv.set( el, type, false );
+ jQuery.event.add( el, type, {
+ namespace: false,
+ handler: function( event ) {
+ var notAsync, result,
+ saved = dataPriv.get( this, type );
+
+ if ( ( event.isTrigger & 1 ) && this[ type ] ) {
+
+ // Interrupt processing of the outer synthetic .trigger()ed event
+ // Saved data should be false in such cases, but might be a leftover capture object
+ // from an async native handler (gh-4350)
+ if ( !saved.length ) {
+
+ // Store arguments for use when handling the inner native event
+ // There will always be at least one argument (an event object), so this array
+ // will not be confused with a leftover capture object.
+ saved = slice.call( arguments );
+ dataPriv.set( this, type, saved );
+
+ // Trigger the native event and capture its result
+ // Support: IE <=9 - 11+
+ // focus() and blur() are asynchronous
+ notAsync = expectSync( this, type );
+ this[ type ]();
+ result = dataPriv.get( this, type );
+ if ( saved !== result || notAsync ) {
+ dataPriv.set( this, type, false );
+ } else {
+ result = {};
+ }
+ if ( saved !== result ) {
+
+ // Cancel the outer synthetic event
+ event.stopImmediatePropagation();
+ event.preventDefault();
+ return result.value;
+ }
+
+ // If this is an inner synthetic event for an event with a bubbling surrogate
+ // (focus or blur), assume that the surrogate already propagated from triggering the
+ // native event and prevent that from happening again here.
+ // This technically gets the ordering wrong w.r.t. to `.trigger()` (in which the
+ // bubbling surrogate propagates *after* the non-bubbling base), but that seems
+ // less bad than duplication.
+ } else if ( ( jQuery.event.special[ type ] || {} ).delegateType ) {
+ event.stopPropagation();
+ }
+
+ // If this is a native event triggered above, everything is now in order
+ // Fire an inner synthetic event with the original arguments
+ } else if ( saved.length ) {
+
+ // ...and capture the result
+ dataPriv.set( this, type, {
+ value: jQuery.event.trigger(
+
+ // Support: IE <=9 - 11+
+ // Extend with the prototype to reset the above stopImmediatePropagation()
+ jQuery.extend( saved[ 0 ], jQuery.Event.prototype ),
+ saved.slice( 1 ),
+ this
+ )
+ } );
+
+ // Abort handling of the native event
+ event.stopImmediatePropagation();
+ }
+ }
+ } );
+}
+
+jQuery.removeEvent = function( elem, type, handle ) {
+
+ // This "if" is needed for plain objects
+ if ( elem.removeEventListener ) {
+ elem.removeEventListener( type, handle );
+ }
+};
+
+jQuery.Event = function( src, props ) {
+
+ // Allow instantiation without the 'new' keyword
+ if ( !( this instanceof jQuery.Event ) ) {
+ return new jQuery.Event( src, props );
+ }
+
+ // Event object
+ if ( src && src.type ) {
+ this.originalEvent = src;
+ this.type = src.type;
+
+ // Events bubbling up the document may have been marked as prevented
+ // by a handler lower down the tree; reflect the correct value.
+ this.isDefaultPrevented = src.defaultPrevented ||
+ src.defaultPrevented === undefined &&
+
+ // Support: Android <=2.3 only
+ src.returnValue === false ?
+ returnTrue :
+ returnFalse;
+
+ // Create target properties
+ // Support: Safari <=6 - 7 only
+ // Target should not be a text node (#504, #13143)
+ this.target = ( src.target && src.target.nodeType === 3 ) ?
+ src.target.parentNode :
+ src.target;
+
+ this.currentTarget = src.currentTarget;
+ this.relatedTarget = src.relatedTarget;
+
+ // Event type
+ } else {
+ this.type = src;
+ }
+
+ // Put explicitly provided properties onto the event object
+ if ( props ) {
+ jQuery.extend( this, props );
+ }
+
+ // Create a timestamp if incoming event doesn't have one
+ this.timeStamp = src && src.timeStamp || Date.now();
+
+ // Mark it as fixed
+ this[ jQuery.expando ] = true;
+};
+
+// jQuery.Event is based on DOM3 Events as specified by the ECMAScript Language Binding
+// https://www.w3.org/TR/2003/WD-DOM-Level-3-Events-20030331/ecma-script-binding.html
+jQuery.Event.prototype = {
+ constructor: jQuery.Event,
+ isDefaultPrevented: returnFalse,
+ isPropagationStopped: returnFalse,
+ isImmediatePropagationStopped: returnFalse,
+ isSimulated: false,
+
+ preventDefault: function() {
+ var e = this.originalEvent;
+
+ this.isDefaultPrevented = returnTrue;
+
+ if ( e && !this.isSimulated ) {
+ e.preventDefault();
+ }
+ },
+ stopPropagation: function() {
+ var e = this.originalEvent;
+
+ this.isPropagationStopped = returnTrue;
+
+ if ( e && !this.isSimulated ) {
+ e.stopPropagation();
+ }
+ },
+ stopImmediatePropagation: function() {
+ var e = this.originalEvent;
+
+ this.isImmediatePropagationStopped = returnTrue;
+
+ if ( e && !this.isSimulated ) {
+ e.stopImmediatePropagation();
+ }
+
+ this.stopPropagation();
+ }
+};
+
+// Includes all common event props including KeyEvent and MouseEvent specific props
+jQuery.each( {
+ altKey: true,
+ bubbles: true,
+ cancelable: true,
+ changedTouches: true,
+ ctrlKey: true,
+ detail: true,
+ eventPhase: true,
+ metaKey: true,
+ pageX: true,
+ pageY: true,
+ shiftKey: true,
+ view: true,
+ "char": true,
+ code: true,
+ charCode: true,
+ key: true,
+ keyCode: true,
+ button: true,
+ buttons: true,
+ clientX: true,
+ clientY: true,
+ offsetX: true,
+ offsetY: true,
+ pointerId: true,
+ pointerType: true,
+ screenX: true,
+ screenY: true,
+ targetTouches: true,
+ toElement: true,
+ touches: true,
+
+ which: function( event ) {
+ var button = event.button;
+
+ // Add which for key events
+ if ( event.which == null && rkeyEvent.test( event.type ) ) {
+ return event.charCode != null ? event.charCode : event.keyCode;
+ }
+
+ // Add which for click: 1 === left; 2 === middle; 3 === right
+ if ( !event.which && button !== undefined && rmouseEvent.test( event.type ) ) {
+ if ( button & 1 ) {
+ return 1;
+ }
+
+ if ( button & 2 ) {
+ return 3;
+ }
+
+ if ( button & 4 ) {
+ return 2;
+ }
+
+ return 0;
+ }
+
+ return event.which;
+ }
+}, jQuery.event.addProp );
+
+jQuery.each( { focus: "focusin", blur: "focusout" }, function( type, delegateType ) {
+ jQuery.event.special[ type ] = {
+
+ // Utilize native event if possible so blur/focus sequence is correct
+ setup: function() {
+
+ // Claim the first handler
+ // dataPriv.set( this, "focus", ... )
+ // dataPriv.set( this, "blur", ... )
+ leverageNative( this, type, expectSync );
+
+ // Return false to allow normal processing in the caller
+ return false;
+ },
+ trigger: function() {
+
+ // Force setup before trigger
+ leverageNative( this, type );
+
+ // Return non-false to allow normal event-path propagation
+ return true;
+ },
+
+ delegateType: delegateType
+ };
+} );
+
+// Create mouseenter/leave events using mouseover/out and event-time checks
+// so that event delegation works in jQuery.
+// Do the same for pointerenter/pointerleave and pointerover/pointerout
+//
+// Support: Safari 7 only
+// Safari sends mouseenter too often; see:
+// https://bugs.chromium.org/p/chromium/issues/detail?id=470258
+// for the description of the bug (it existed in older Chrome versions as well).
+jQuery.each( {
+ mouseenter: "mouseover",
+ mouseleave: "mouseout",
+ pointerenter: "pointerover",
+ pointerleave: "pointerout"
+}, function( orig, fix ) {
+ jQuery.event.special[ orig ] = {
+ delegateType: fix,
+ bindType: fix,
+
+ handle: function( event ) {
+ var ret,
+ target = this,
+ related = event.relatedTarget,
+ handleObj = event.handleObj;
+
+ // For mouseenter/leave call the handler if related is outside the target.
+ // NB: No relatedTarget if the mouse left/entered the browser window
+ if ( !related || ( related !== target && !jQuery.contains( target, related ) ) ) {
+ event.type = handleObj.origType;
+ ret = handleObj.handler.apply( this, arguments );
+ event.type = fix;
+ }
+ return ret;
+ }
+ };
+} );
+
+jQuery.fn.extend( {
+
+ on: function( types, selector, data, fn ) {
+ return on( this, types, selector, data, fn );
+ },
+ one: function( types, selector, data, fn ) {
+ return on( this, types, selector, data, fn, 1 );
+ },
+ off: function( types, selector, fn ) {
+ var handleObj, type;
+ if ( types && types.preventDefault && types.handleObj ) {
+
+ // ( event ) dispatched jQuery.Event
+ handleObj = types.handleObj;
+ jQuery( types.delegateTarget ).off(
+ handleObj.namespace ?
+ handleObj.origType + "." + handleObj.namespace :
+ handleObj.origType,
+ handleObj.selector,
+ handleObj.handler
+ );
+ return this;
+ }
+ if ( typeof types === "object" ) {
+
+ // ( types-object [, selector] )
+ for ( type in types ) {
+ this.off( type, selector, types[ type ] );
+ }
+ return this;
+ }
+ if ( selector === false || typeof selector === "function" ) {
+
+ // ( types [, fn] )
+ fn = selector;
+ selector = undefined;
+ }
+ if ( fn === false ) {
+ fn = returnFalse;
+ }
+ return this.each( function() {
+ jQuery.event.remove( this, types, fn, selector );
+ } );
+ }
+} );
+
+
+var
+
+ /* eslint-disable max-len */
+
+ // See https://github.com/eslint/eslint/issues/3229
+ rxhtmlTag = /<(?!area|br|col|embed|hr|img|input|link|meta|param)(([a-z][^\/\0>\x20\t\r\n\f]*)[^>]*)\/>/gi,
+
+ /* eslint-enable */
+
+ // Support: IE <=10 - 11, Edge 12 - 13 only
+ // In IE/Edge using regex groups here causes severe slowdowns.
+ // See https://connect.microsoft.com/IE/feedback/details/1736512/
+ rnoInnerhtml = /
+
+