2018-01-22 13:01:38 -08:00
<!-- Modal install app -->
2019-07-31 08:09:36 +02:00
< div class = "modal fade appstore-install" id = "appInstallModal" tabindex = "-1" role = "dialog" >
< div class = "modal-dialog" >
< div class = "modal-content" >
< div class = "modal-header" >
2020-11-13 21:58:09 +01:00
< img ng-src = "{{appInstall.app.iconUrl}}" onerror = "this.onerror=null; this.src='img/appicon_fallback.png'" class = "app-icon" / >
< h3 class = "appstore-install-title" > {{ appInstall.app.manifest.title }}< / h3 >
2019-07-31 08:09:36 +02:00
< br / >
< span class = "appstore-install-meta" > < a href = "{{ appInstall.app.manifest.website }}" target = "_blank" > {{ appInstall.app.manifest.author }}< / a > < / span >
< br / >
2020-11-13 21:58:09 +01:00
< span class = "appstore-install-meta" > {{ 'appstore.installDialog.lastUpdated' | tr:{ date: (appInstall.app.creationDate | prettyDate) } }}< / span >
2019-07-31 08:09:36 +02:00
< br / >
2020-11-13 21:58:09 +01:00
< span class = "appstore-install-meta" > {{ 'appstore.installDialog.memoryRequirement' | tr:{ size: (appInstall.app.manifest.memoryLimit | prettyByteSize:'256 MB') } }}< / span >
2019-07-31 08:09:36 +02:00
< / div >
< div class = "modal-body" >
< div class = "collapse" id = "collapseInstallForm" data-toggle = "false" >
< form role = "form" name = "appInstallForm" ng-submit = "appInstall.submit()" autocomplete = "off" >
< div class = "has-error text-center" ng-show = "appInstall.error.other" ng-bind-html = "appInstall.error.other" > < / div >
< div class = "form-group" ng-class = "{ 'has-error': (appInstallForm.location.$dirty && appInstallForm.location.$invalid) || (!appInstallForm.location.$dirty && appInstall.error.location) }" >
2020-11-13 21:58:09 +01:00
< label class = "control-label" for = "appInstallLocationInput" > {{ 'appstore.installDialog.location' | tr }}< / label >
2019-07-31 08:09:36 +02:00
< div class = "input-group form-inline" >
2020-11-13 21:58:09 +01:00
< input type = "text" class = "form-control" ng-model = "appInstall.location" id = "appInstallLocationInput" name = "location" placeholder = "{{ 'appstore.installDialog.locationPlaceholder' | tr }}" autofocus >
2019-07-31 08:09:36 +02:00
< div class = "input-group-btn" >
< button type = "button" class = "btn btn-default dropdown-toggle" data-toggle = "dropdown" >
2020-08-15 18:35:51 -07:00
< span > {{ (appInstall.location ? '.' : '') + appInstall.domain.domain }}< / span >
2019-07-31 08:09:36 +02:00
< span class = "caret" > < / span >
< / button >
< ul class = "dropdown-menu dropdown-menu-right" role = "menu" >
< li ng-repeat = "domain in domains" >
< a href = "" ng-click = "appInstall.domain = domain" > {{ domain.domain }}< / a >
< / li >
< / ul >
< / div >
< / div >
2020-11-13 21:58:09 +01:00
< div ng-show = "appInstall.error.location" class = "text-small" > {{ appInstall.error.location }}< / div >
2018-01-22 13:01:38 -08:00
< / div >
2021-09-20 11:10:37 -07:00
< p class = "text-small text-warning" ng-show = "appInstall.domain.provider === 'noop' || appInstall.domain.provider === 'manual'" ng-bind-html = "'appstore.installDialog.manualWarning' | tr:{ location: ((appInstall.location ? appInstall.location + '.' : '') + appInstall.domain.domain) }" > < / p >
2018-01-22 13:01:38 -08:00
2019-07-31 08:09:36 +02:00
< div class = "has-error text-center" ng-show = "appInstall.error.port" > {{ appInstall.error.port }}< / div >
< div ng-repeat = "(env, info) in appInstall.portBindingsInfo" >
< ng-form name = "portInfo_form" >
< div class = "form-group" ng-class = "{ 'has-error': (!appInstallForm.itemName{{$index}}.$dirty && appInstall.error.port) || (portInfo_form.itemName{{$index}}.$dirty && portInfo_form.itemName{{$index}}.$invalid) }" >
< label class = "control-label" for = "inputPortInfo{{env}}" > < input type = "checkbox" ng-model = "appInstall.portBindingsEnabled[env]" >
{{ info.title }}
< sup >
< a popover-placement = "top-right" popover-trigger = "outsideClick" uib-popover = "{{info.description}} ({{ HOST_PORT_MIN }} - {{ HOST_PORT_MAX }})" > < i class = "fa fa-question-circle" > < / i > < / a >
< / sup >
< / label >
< input type = "number" class = "form-control" ng-model = "appInstall.portBindings[env]" ng-disabled = "!appInstall.portBindingsEnabled[env]" id = "inputPortInfo{{env}}" later-name = "itemName{{$index}}" min = "{{hostPortMin}}" max = "{{hostPortMax}}" required >
2021-09-20 11:24:08 -07:00
< p class = "text-small text-warning text-bold" ng-show = "appInstall.domain.provider === 'cloudflare'" > {{ 'appstore.installDialog.cloudflarePortWarning' | tr }} < / p >
2019-07-31 08:09:36 +02:00
< / div >
< / ng-form >
< / div >
2018-01-22 13:01:38 -08:00
2020-02-25 13:00:36 +01:00
< div class = "form-group" ng-show = "appInstall.app.manifest.addons.email" >
2020-11-13 21:58:09 +01:00
< label class = "control-label" > {{ 'appstore.installDialog.userManagement' | tr }}< / label >
2021-03-17 14:11:08 -07:00
< p > {{ 'appstore.installDialog.userManagementMailbox' | tr }}
< span ng-bind-html = "'appstore.installDialog.configuredForCloudronEmail' | tr:{ emailDocsLink: 'https://docs.cloudron.io/email/' }" >
< / p >
2019-07-31 08:09:36 +02:00
< / div >
2018-01-22 13:01:38 -08:00
2021-03-01 20:16:05 +01:00
< div class = "form-group" >
2021-03-17 14:11:08 -07:00
< label class = "control-label" ng-show = "!appInstall.customAuth && !appInstall.app.manifest.addons.email" > {{ 'appstore.installDialog.userManagement' | tr }} < sup > < a ng-href = "https://docs.cloudron.io/apps/#access-restriction" class = "help" target = "_blank" > < i class = "fa fa-question-circle" > < / i > < / a > < / sup > < / label >
2021-03-02 20:49:47 -08:00
< label class = "control-label" ng-show = "appInstall.customAuth || appInstall.app.manifest.addons.email" > {{ 'app.accessControl.userManagement.dashboardVisibility' | tr }} < sup > < a ng-href = "https://docs.cloudron.io/apps/#dashboard-visibility" class = "help" target = "_blank" > < i class = "fa fa-question-circle" > < / i > < / a > < / sup > < / label >
2021-03-01 18:52:10 +01:00
2019-07-31 08:09:36 +02:00
< div class = "radio" ng-show = "appInstall.optionalSso" >
< label >
2020-11-13 21:58:09 +01:00
< input type = "radio" ng-model = "appInstall.accessRestrictionOption" value = "nosso" > {{ 'appstore.installDialog.userManagementLeaveToApp' | tr }}
2019-07-31 08:09:36 +02:00
< / label >
< / div >
< div class = "radio" >
< label >
2021-03-01 18:52:10 +01:00
< input type = "radio" ng-model = "appInstall.accessRestrictionOption" value = "any" >
< span ng-show = "!appInstall.customAuth" > {{ 'appstore.installDialog.userManagementAllUsers' | tr }}< / span >
< span ng-show = "appInstall.customAuth" > {{ 'app.accessControl.userManagement.visibleForAllUsers' | tr }}< / span >
2019-07-31 08:09:36 +02:00
< / label >
< / div >
< div class = "radio" >
< label >
2021-03-01 18:52:10 +01:00
< input type = "radio" ng-model = "appInstall.accessRestrictionOption" value = "groups" >
< span ng-show = "!appInstall.customAuth" > {{ 'appstore.installDialog.userManagementSelectUsers' | tr }}< / span >
< span ng-show = "appInstall.customAuth" > {{ 'app.accessControl.userManagement.visibleForSelected' | tr }}< / span >
2020-11-13 21:58:09 +01:00
< span class = "label label-danger" ng-show = "appInstall.accessRestrictionOption === 'groups' && !appInstall.isAccessRestrictionValid()" > {{ 'appstore.installDialog.errorUserManagementSelectAtLeastOne' | tr }}< / span >
2019-07-31 08:09:36 +02:00
< / label >
< / div >
< div >
< div style = "margin-left: 20px;" >
< div class = "col-md-5" >
2020-11-13 21:58:09 +01:00
{{ 'appstore.installDialog.users' | tr }}:
2019-11-05 19:45:59 +01:00
< multiselect ng-model = "appInstall.accessRestriction.users" ng-disabled = "appInstall.accessRestrictionOption !== 'groups'" options = "user.username for user in users" data-multiple = "true" filter-after-rows = "5" scroll-after-rows = "10" > < / multiselect >
2019-07-31 08:09:36 +02:00
< / div >
2018-01-22 13:01:38 -08:00
2019-07-31 08:09:36 +02:00
< div class = "col-md-5" >
2020-11-13 21:58:09 +01:00
{{ 'appstore.installDialog.groups' | tr }}:
2019-11-05 19:45:59 +01:00
< multiselect ng-model = "appInstall.accessRestriction.groups" ng-disabled = "appInstall.accessRestrictionOption !== 'groups'" options = "group.name for group in groups" data-multiple = "true" filter-after-rows = "5" scroll-after-rows = "10" > < / multiselect >
2019-07-31 08:09:36 +02:00
< / div >
< / div >
< / div >
2018-01-22 13:01:38 -08:00
2019-07-31 08:09:36 +02:00
< br / >
< br / >
< / div >
2018-01-22 13:01:38 -08:00
2019-07-31 08:09:36 +02:00
< div class = "hide" >
2021-01-12 19:48:48 -08:00
< label class = "control-label" for = "appInstallCertificateInput" > Certificate (optional)< / label >
< div class = "has-error text-center" ng-show = "appInstall.error.cert" > {{ appInstall.error.cert }}< / div >
< div class = "form-group" ng-class = "{ 'has-error': !appInstallForm.certificate.$dirty && appInstall.error.cert }" >
2019-07-31 08:09:36 +02:00
< div class = "input-group" >
< input type = "file" id = "appInstallCertificateFileInput" style = "display:none" / >
< input type = "text" class = "form-control" placeholder = "Certificate" ng-model = "appInstall.certificateFileName" id = "appInstallCertificateInput" name = "certificate" onclick = "getElementById('appInstallCertificateFileInput').click();" style = "cursor: pointer;" ng-required = "appInstall.keyFileName" >
< span class = "input-group-addon" >
< i class = "fa fa-upload" onclick = "getElementById('appInstallCertificateFileInput').click();" > < / i >
< / span >
2018-01-22 13:01:38 -08:00
< / div >
2019-07-31 08:09:36 +02:00
< / div >
2021-01-12 19:48:48 -08:00
< div class = "form-group" ng-class = "{ 'has-error': !appInstallForm.key.$dirty && appInstall.error.cert }" >
2019-07-31 08:09:36 +02:00
< div class = "input-group" >
< input type = "file" id = "appInstallKeyFileInput" style = "display:none" / >
< input type = "text" class = "form-control" placeholder = "Key" ng-model = "appInstall.keyFileName" id = "appInstallKeyInput" name = "key" onclick = "getElementById('appInstallKeyFileInput').click();" style = "cursor: pointer;" ng-required = "appInstall.certificateFileName" >
< span class = "input-group-addon" >
< i class = "fa fa-upload" onclick = "getElementById('appInstallKeyFileInput').click();" > < / i >
< / span >
2018-01-22 13:01:38 -08:00
< / div >
2019-07-31 08:09:36 +02:00
< / div >
2018-01-22 13:01:38 -08:00
< / div >
2019-07-31 08:09:36 +02:00
2021-02-17 17:09:46 +01:00
< input class = "ng-hide" type = "submit" ng-disabled = "(appInstall.accessRestrictionOption === 'groups' && !appInstall.isAccessRestrictionValid()) || !appInstall.accessRestrictionOption || appInstallForm.$invalid || busy" / >
2019-07-31 08:09:36 +02:00
< / form >
< / div >
< div class = "collapse" id = "collapseMediaLinksCarousel" data-toggle = "false" >
< div ng-repeat = "mediaLink in appInstall.mediaLinks" class = "slick-item" style = "background-image: url('{{mediaLink}}');" ng-show = "appInstall.mediaLinks.length == 1" > < / div >
< slick init-onload = "true" current-index = "0" autoplay = "true" arrows = "false" autoplay-speed = "2000" data = "appInstall.mediaLinks" ng-show = "appInstall.mediaLinks.length > 1" >
< div ng-repeat = "mediaLink in appInstall.mediaLinks" class = "slick-item" style = "background-image: url('{{mediaLink}}');" > < / div >
< / slick >
2021-01-22 12:35:09 -08:00
< br / >
2019-07-31 08:09:36 +02:00
< div class = "appstore-install-description" >
< div ng-bind-html = "appInstall.app.manifest.description | markdown2html" > < / div >
< / div >
< / div >
< div class = "collapse" id = "collapseResourceConstraint" data-toggle = "false" >
2020-11-13 21:58:09 +01:00
< h4 class = "text-danger" > {{ 'appstore.installDialog.lowOnResources' | tr }}< / h4 >
< p > {{ 'appstore.installDialog.pleaseUpgradeServer' | tr }}< / p >
2019-07-31 08:09:36 +02:00
< / div >
2019-08-14 16:38:02 +02:00
< div class = "collapse" id = "collapseSubscriptionRequired" data-toggle = "false" >
2020-11-13 21:58:09 +01:00
< p > {{ 'appstore.installDialog.subscriptionRequired' | tr }}< / p >
2018-01-22 13:01:38 -08:00
< / div >
2019-07-31 08:09:36 +02:00
< / div >
< div class = "modal-footer" >
2020-11-13 21:58:09 +01:00
< button type = "button" class = "btn btn-success pull-left" ng-click = "openSubscriptionSetup()" ng-show = "appInstall.state === 'subscriptionRequired'" > {{ 'appstore.installDialog.setupSubscriptionAction' | tr }}< / button >
< button type = "button" class = "btn btn-default" data-dismiss = "modal" > {{ 'main.dialog.close' | tr }}< / button >
< button type = "button" class = "btn btn-danger" ng-show = "appInstall.state === 'resourceConstraint'" ng-click = "appInstall.showForm(true)" > {{ 'appstore.installDialog.installAnywayAction' | tr }}< / button >
< button type = "button" class = "btn btn-success" ng-show = "appInstall.state === 'appInfo'" ng-click = "appInstall.showForm()" > {{ 'appstore.installDialog.installAction' | tr }}< / button >
2021-02-17 17:09:46 +01:00
< button type = "button" class = "btn btn-success" ng-show = "appInstall.state === 'installForm'" ng-click = "appInstall.submit()" ng-disabled = "(appInstall.accessRestrictionOption === 'groups' && !appInstall.isAccessRestrictionValid()) || !appInstall.accessRestrictionOption || appInstallForm.$invalid || appInstall.busy" > < i class = "fa fa-circle-notch fa-spin" ng-show = "appInstall.busy" > < / i > {{ 'appstore.installDialog.doInstallAction' | tr:{ dnsOverwrite: appInstall.needsOverwrite } }}< / button >
2019-07-31 08:09:36 +02:00
< / div >
2018-01-22 13:01:38 -08:00
< / div >
2019-07-31 08:09:36 +02:00
< / div >
2018-01-22 13:01:38 -08:00
< / div >
<!-- Modal app not found -->
< div class = "modal fade" id = "appNotFoundModal" tabindex = "-1" role = "dialog" >
2020-11-13 21:58:09 +01:00
< div class = "modal-dialog" >
< div class = "modal-content" >
< div class = "modal-header" >
< h4 class = "modal-title" > {{ 'appstore.appNotFoundDialog.title' | tr }}< / h4 >
< / div >
< div class = "modal-body" ng-bind-html = "'appstore.appNotFoundDialog.description' | tr:{ appId: appNotFound.appId, version: appNotFound.version }" > < / div >
< div class = "modal-footer" >
< button type = "button" class = "btn btn-primary" data-dismiss = "modal" > {{ 'main.dialog.close' | tr }}< / button >
< / div >
2018-01-22 13:01:38 -08:00
< / div >
2020-11-13 21:58:09 +01:00
< / div >
2018-01-22 13:01:38 -08:00
< / div >
< div ng-show = "!ready" class = "loading-banner" >
2020-11-13 21:58:09 +01:00
< h1 > < i class = "fa fa-circle-notch fa-spin" > < / i > < / h1 >
2018-01-22 13:01:38 -08:00
< / div >
<!-- appstore login -->
2019-05-04 21:57:42 -07:00
< div ng-show = "ready && !validSubscription" class = "container card card-small appstore-login ng-cloak" >
2020-11-13 21:58:09 +01:00
< div class = "col-md-12 text-center" >
< h1 ng-show = "appstoreLogin.register" > {{ 'appstore.accountDialog.titleSignUp' | tr }}< / h1 >
< h1 ng-hide = "appstoreLogin.register" > {{ 'appstore.accountDialog.titleLogin' | tr }}< / h1 >
< / div >
< div class = "col-md-12 text-center" >
< p > {{ 'appstore.accountDialog.description' | tr }}< / p >
< / div >
< div class = "col-md-12" style = "margin-bottom: 10px;" >
< small class = "text-danger" ng-show = "appstoreLogin.error.generic" > {{ appstoreLogin.error.generic }}< / small >
< / div >
< div class = "col-md-12" >
< br / >
2018-03-13 10:59:15 -07:00
2020-11-13 21:58:09 +01:00
< form name = "appstoreLoginForm" role = "form" novalidate ng-submit = "appstoreLogin.submit()" autocomplete = "off" >
< input type = "password" style = "display: none;" >
2018-01-22 13:01:38 -08:00
2020-11-13 21:58:09 +01:00
< div class = "form-group" ng-class = "{ 'has-error': (appstoreLoginForm.email.$dirty && appstoreLoginForm.email.$invalid) || appstoreLogin.error.generic }" >
< label class = "control-label" > {{ 'appstore.accountDialog.email' | tr }}< / label >
< input type = "email" class = "form-control" ng-model = "appstoreLogin.email" id = "inputAppstoreLoginEmail" name = "email" required autofocus >
< div class = "control-label" ng-show = "(!appstoreLoginForm.email.$dirty && appstoreLogin.error.email) || (appstoreLoginForm.email.$dirty && appstoreLoginForm.email.$invalid) || appstoreLogin.error.email" >
< small class = "text-danger" ng-show = "appstoreLogin.error.email" > {{ appstoreLogin.error.email }}< / small >
< / div >
< / div >
2018-01-22 13:01:38 -08:00
2020-11-13 21:58:09 +01:00
< div class = "form-group" ng-class = "{ 'has-error': (!appstoreLoginForm.password.$dirty && appstoreLogin.error.password) || (appstoreLoginForm.password.$dirty && appstoreLoginForm.password.$invalid) || appstoreLogin.error.generic }" >
< label class = "control-label" > {{ 'appstore.accountDialog.password' | tr }}< / label >
< input type = "password" class = "form-control" ng-model = "appstoreLogin.password" id = "inputAppstoreLoginPassword" name = "password" required >
< div class = "control-label" ng-show = "(!appstoreLoginForm.password.$dirty && appstoreLogin.error.password) || (appstoreLoginForm.password.$dirty && appstoreLoginForm.password.$invalid)" >
< small ng-show = "!appstoreLoginForm.password.$dirty && appstoreLogin.error.password" > {{ 'appstore.accountDialog.errorWrongPassword' | tr }}< / small >
< / div >
< / div >
2018-01-22 13:01:38 -08:00
2020-11-13 21:58:09 +01:00
< div class = "form-group" ng-hide = "appstoreLogin.register" ng-class = "{ 'has-error': appstoreLogin.error.totpToken }" >
< label class = "control-label" > {{ 'appstore.accountDialog.2faToken' | tr }}< / label >
< input type = "text" class = "form-control" ng-model = "appstoreLogin.totpToken" id = "inputAppstoreLoginTotpToken" name = "totpToken" >
< div class = "control-label" ng-show = "appstoreLogin.error.totpToken" >
< small ng-show = "appstoreLogin.error.totpToken" > {{ appstoreLogin.error.totpToken }}< / small >
< / div >
< / div >
2018-04-22 18:52:37 +02:00
2020-11-13 21:58:09 +01:00
< div class = "checkbox" >
< label >
2020-11-18 09:35:20 +01:00
< input type = "checkbox" ng-model = "appstoreLogin.termsAccepted" > < span ng-bind-html = "'appstore.accountDialog.licenseCheckbox' | tr:{ licenseLink: 'https://cloudron.io/legal/license.html' }" > < / span >
2020-11-13 21:58:09 +01:00
< / label >
< / div >
2018-01-22 13:01:38 -08:00
2020-11-13 21:58:09 +01:00
< br / >
2018-01-22 13:01:38 -08:00
2020-11-13 21:58:09 +01:00
< center >
< button type = "submit" class = "btn btn-lg btn-success" ng-disabled = "appstoreLoginForm.$invalid || appstoreLogin.busy || !appstoreLogin.termsAccepted" >
< i class = "fa fa-circle-notch fa-spin" ng-show = "appstoreLogin.busy" > < / i > < span ng-hide = "appstoreLogin.register" > {{ 'appstore.accountDialog.loginAction' | tr }}< / span > < span ng-show = "appstoreLogin.register" > {{ 'appstore.accountDialog.createAccountAction' | tr }}< / span >
< / button >
2018-03-13 10:21:25 +01:00
2020-11-13 21:58:09 +01:00
< br / >
< br / >
2018-03-13 10:21:25 +01:00
2020-11-13 21:58:09 +01:00
< a href = "" ng-click = "appstoreLogin.register = true" ng-hide = "appstoreLogin.register" > {{ 'appstore.accountDialog.switchToSignUpAction' | tr }}< / a >
< a href = "" ng-click = "appstoreLogin.register = false" ng-show = "appstoreLogin.register" > {{ 'appstore.accountDialog.switchToLoginAction' | tr }}< / a >
< / center >
2018-01-22 13:01:38 -08:00
2020-11-13 21:58:09 +01:00
< / form >
< / div >
2018-01-22 13:01:38 -08:00
< / div >
2018-03-14 09:04:05 -07:00
2018-04-05 22:03:40 +02:00
<!-- give more vertical spacing so the login form does not appear clipped -->
2019-05-04 21:57:42 -07:00
< div ng-show = "ready && !validSubscription" >
2020-06-12 15:02:41 +02:00
< br / >
< br / >
2018-04-05 22:03:40 +02:00
< / div >
2020-06-12 15:02:41 +02:00
< div ng-show = "ready && validSubscription" class = "ng-cloak appstore-toolbar" >
< div class = "appstore-toolbar-content" >
2020-11-13 21:58:09 +01:00
< button class = "btn" type = "button" ng-click = "showCategory('');" ng-class = "{ 'btn-primary': '' === category }" > {{ 'appstore.category.all' | tr }}< / button >
< button class = "btn" type = "button" ng-click = "showCategory('new');" ng-class = "{ 'btn-primary': 'new' === category }" > {{ 'appstore.category.newApps' | tr }}< / button >
2020-06-12 15:02:41 +02:00
< div class = "dropdown" >
2021-03-18 14:28:22 +01:00
< button class = "btn dropdown-toggle" type = "button" data-toggle = "dropdown" ng-class = "{ 'btn-primary': '' !== category && 'recent' !== category && 'new' !== category }" >
2020-06-13 22:56:24 +02:00
{{ categoryButtonLabel(category) }}
2020-06-12 15:02:41 +02:00
< span class = "caret" > < / span >
< / button >
< ul class = "dropdown-menu" >
2021-03-31 14:03:49 +02:00
< li ng-repeat = "category in categories | orderBy:'label'" > < a href = "" ng-click = "showCategory(category.id);" > < i class = "{{ category.icon }} fa-fw" > < / i > {{ category.label }}< / a > < / li >
2020-06-12 15:02:41 +02:00
< / ul >
2018-03-14 09:04:05 -07:00
< / div >
2020-11-13 21:58:09 +01:00
< input type = "text" id = "appstoreSearch" class = "form-control" placeholder = "{{ 'appstore.searchPlaceholder' | tr }}" ng-model = "searchString" ng-change = "search()" autofocus >
2020-06-12 15:02:41 +02:00
< / div >
< / div >
2020-06-16 11:37:10 +02:00
< div ng-show = "ready && validSubscription" class = "ng-cloak appstore-grid" >
2020-06-12 15:02:41 +02:00
< div class = "row" >
2020-06-16 11:20:28 +02:00
< div class = "col-md-12 text-center" ng-hide = "apps.length" >
2020-06-12 15:02:41 +02:00
< br / >
< br / >
2020-06-16 11:20:28 +02:00
< br / >
2020-11-13 21:58:09 +01:00
< h3 class = "text-muted" > {{ 'appstore.noAppsFound' | tr }}< / h3 >
2020-06-16 11:20:28 +02:00
< br / >
2020-11-13 21:58:09 +01:00
< a href = "https://forum.cloudron.io/category/5/app-requests" target = "_blank" > {{ 'appstore.appMissing' | tr }}< / a >
2020-06-12 15:02:41 +02:00
< / div >
2021-03-18 14:14:22 +01:00
< div class = "col-md-12" ng-show = "category === '' && popularApps.length" >
< div class = "row-no-margin" >
< div class = "col-sm-12" >
< h2 > {{ 'appstore.category.popular' | tr }}< / h2 >
< / div >
< / div >
< div class = "row-no-margin" >
< div class = "col-sm-1 appstore-item" ng-repeat = "app in popularApps" >
< div class = "appstore-item-content highlight" ng-click = "gotoApp(app)" ng-class = "{ 'appstore-item-content-testing': app.releaseState === 'unstable' }" >
< span class = "badge badge-danger appstore-item-badge-testing" ng-show = "app.releaseState === 'unstable'" > {{ 'appstore.unstable' | tr }}< / span >
< div class = "appstore-item-content-icon col-same-height" >
< img ng-src = "{{app.iconUrl}}" onerror = "this.onerror=null;this.src='img/appicon_fallback.png'" class = "app-icon" / >
< / div >
< div class = "appstore-item-content-description col-same-height" >
< h4 class = "appstore-item-content-title" > {{ app.manifest.title }}< / h4 >
< div class = "appstore-item-content-tagline text-muted" > {{ app.manifest.tagline }}< / div >
< / div >
< / div >
< / div >
< / div >
< / div >
2020-06-12 15:02:41 +02:00
< div class = "col-md-12" ng-show = "apps.length" >
2021-03-18 14:28:22 +01:00
< div class = "row-no-margin" ng-show = "!category && !searchString" >
2021-03-18 14:14:22 +01:00
< div class = "col-sm-12" >
< h2 > {{ 'appstore.category.all' | tr }}< / h2 >
< / div >
< / div >
2020-06-12 15:02:41 +02:00
< div class = "row-no-margin" >
2020-07-17 14:19:25 -07:00
< div class = "col-sm-1 appstore-item" ng-repeat = "app in apps" >
2020-06-12 15:02:41 +02:00
< div class = "appstore-item-content highlight" ng-click = "gotoApp(app)" ng-class = "{ 'appstore-item-content-testing': app.releaseState === 'unstable' }" >
2020-11-13 21:58:09 +01:00
< span class = "badge badge-danger appstore-item-badge-testing" ng-show = "app.releaseState === 'unstable'" > {{ 'appstore.unstable' | tr }}< / span >
2020-06-12 15:02:41 +02:00
< div class = "appstore-item-content-icon col-same-height" >
< img ng-src = "{{app.iconUrl}}" onerror = "this.onerror=null;this.src='img/appicon_fallback.png'" class = "app-icon" / >
2018-03-14 09:04:05 -07:00
< / div >
2020-06-12 15:02:41 +02:00
< div class = "appstore-item-content-description col-same-height" >
< h4 class = "appstore-item-content-title" > {{ app.manifest.title }}< / h4 >
< div class = "appstore-item-content-tagline text-muted" > {{ app.manifest.tagline }}< / div >
< / div >
< / div >
2018-03-14 09:04:05 -07:00
< / div >
2020-06-12 15:02:41 +02:00
< / div >
2018-03-14 09:04:05 -07:00
< / div >
2020-06-12 15:02:41 +02:00
< / div >
2018-03-14 09:04:05 -07:00
< / div >