1
0
mirror of https://github.com/bpatrik/pigallery2.git synced 2025-01-12 04:23:09 +02:00

adding random photo url generator

upgrading packages
This commit is contained in:
Patrik J. Braun 2018-10-22 00:24:17 +02:00
parent 151a3782ac
commit 2ea0ea42e3
37 changed files with 757 additions and 147 deletions

View File

@ -23,6 +23,7 @@
"node_modules/ngx-toastr/toastr.css", "node_modules/ngx-toastr/toastr.css",
"node_modules/bootstrap/dist/css/bootstrap.css", "node_modules/bootstrap/dist/css/bootstrap.css",
"node_modules/open-iconic/font/css/open-iconic-bootstrap.css", "node_modules/open-iconic/font/css/open-iconic-bootstrap.css",
"node_modules/ngx-bootstrap/datepicker/bs-datepicker.css",
"frontend/styles.css" "frontend/styles.css"
], ],
"scripts": [] "scripts": []

View File

@ -103,6 +103,28 @@ export class AdminMWs {
} }
public static async updateRandomPhotoSettings(req: Request, res: Response, next: NextFunction) {
if ((typeof req.body === 'undefined') || (typeof req.body.settings === 'undefined')) {
return next(new ErrorDTO(ErrorCodes.INPUT_ERROR, 'settings is needed'));
}
try {
// only updating explicitly set config (not saving config set by the diagnostics)
const original = Config.original();
await ConfigDiagnostics.testRandomPhotoConfig(<ClientConfig.RandomPhotoConfig>req.body.settings, original);
Config.Client.RandomPhoto = <ClientConfig.RandomPhotoConfig>req.body.settings;
original.Client.RandomPhoto = <ClientConfig.RandomPhotoConfig>req.body.settings;
original.save();
await ConfigDiagnostics.runDiagnostics();
Logger.info(LOG_TAG, 'new config:');
Logger.info(LOG_TAG, JSON.stringify(Config, null, '\t'));
return next();
} catch (err) {
return next(new ErrorDTO(ErrorCodes.SETTINGS_ERROR, 'Settings error: ' + JSON.stringify(err, null, ' '), err));
}
}
public static async updateSearchSettings(req: Request, res: Response, next: NextFunction) { public static async updateSearchSettings(req: Request, res: Response, next: NextFunction) {
if ((typeof req.body === 'undefined') || (typeof req.body.settings === 'undefined')) { if ((typeof req.body === 'undefined') || (typeof req.body.settings === 'undefined')) {
return next(new ErrorDTO(ErrorCodes.INPUT_ERROR, 'settings is needed')); return next(new ErrorDTO(ErrorCodes.INPUT_ERROR, 'settings is needed'));

View File

@ -10,6 +10,7 @@ import {PhotoDTO} from '../../common/entities/PhotoDTO';
import {ProjectPath} from '../ProjectPath'; import {ProjectPath} from '../ProjectPath';
import {Config} from '../../common/config/private/Config'; import {Config} from '../../common/config/private/Config';
import {UserDTO} from '../../common/entities/UserDTO'; import {UserDTO} from '../../common/entities/UserDTO';
import {RandomQuery} from '../model/interfaces/IGalleryManager';
const LOG_TAG = '[GalleryMWs]'; const LOG_TAG = '[GalleryMWs]';
@ -79,6 +80,50 @@ export class GalleryMWs {
} }
public static async getRandomImage(req: Request, res: Response, next: NextFunction) {
if (Config.Client.RandomPhoto.enabled === false) {
return next();
}
const query: RandomQuery = {};
if (req.query.directory) {
query.directory = req.query.directory;
}
if (req.query.recursive === 'true') {
query.recursive = true;
}
if (req.query.orientation) {
query.orientation = parseInt(req.query.orientation.toString(), 10);
}
if (req.query.maxResolution) {
query.maxResolution = parseFloat(req.query.maxResolution.toString());
}
if (req.query.minResolution) {
query.minResolution = parseFloat(req.query.minResolution.toString());
}
if (req.query.fromDate) {
query.fromDate = new Date(req.query.fromDate);
}
if (req.query.toDate) {
query.toDate = new Date(req.query.toDate);
}
if (query.minResolution && query.maxResolution && query.maxResolution < query.minResolution) {
return next(new ErrorDTO(ErrorCodes.INPUT_ERROR, 'Input error: min resolution is greater than the max resolution'));
}
if (query.toDate && query.fromDate && query.toDate.getTime() < query.fromDate.getTime()) {
return next(new ErrorDTO(ErrorCodes.INPUT_ERROR, 'Input error: to date is earlier than from date'));
}
const photo = await ObjectManagerRepository.getInstance()
.GalleryManager.getRandomPhoto(query);
if (!photo) {
return next(new ErrorDTO(ErrorCodes.INPUT_ERROR, 'No photo found'));
}
req.params.imagePath = path.join(photo.directory.path, photo.directory.name, photo.name);
return next();
}
public static loadImage(req: Request, res: Response, next: NextFunction) { public static loadImage(req: Request, res: Response, next: NextFunction) {
if (!(req.params.imagePath)) { if (!(req.params.imagePath)) {
return next(); return next();
@ -134,7 +179,7 @@ export class GalleryMWs {
} }
try { try {
const result = await ObjectManagerRepository.getInstance().SearchManager.instantSearch(req.params.text); const result = await ObjectManagerRepository.getInstance().SearchManager.instantSearch(req.params.text);
result.directories.forEach(dir => dir.photos = dir.photos || []); result.directories.forEach(dir => dir.photos = dir.photos || []);
req.resultPipe = new ContentWrapper(null, result); req.resultPipe = new ContentWrapper(null, result);

View File

@ -2,6 +2,7 @@ import {NextFunction, Request, Response} from 'express';
import {CreateSharingDTO, SharingDTO} from '../../common/entities/SharingDTO'; import {CreateSharingDTO, SharingDTO} from '../../common/entities/SharingDTO';
import {ObjectManagerRepository} from '../model/ObjectManagerRepository'; import {ObjectManagerRepository} from '../model/ObjectManagerRepository';
import {ErrorCodes, ErrorDTO} from '../../common/entities/Error'; import {ErrorCodes, ErrorDTO} from '../../common/entities/Error';
import {Config} from '../../common/config/private/Config';
const LOG_TAG = '[SharingMWs]'; const LOG_TAG = '[SharingMWs]';
@ -20,6 +21,9 @@ export class SharingMWs {
public static async getSharing(req: Request, res: Response, next: NextFunction) { public static async getSharing(req: Request, res: Response, next: NextFunction) {
if (Config.Client.Sharing.enabled === false) {
return next();
}
const sharingKey = req.params.sharingKey; const sharingKey = req.params.sharingKey;
try { try {
@ -33,6 +37,9 @@ export class SharingMWs {
} }
public static async createSharing(req: Request, res: Response, next: NextFunction) { public static async createSharing(req: Request, res: Response, next: NextFunction) {
if (Config.Client.Sharing.enabled === false) {
return next();
}
if ((typeof req.body === 'undefined') || (typeof req.body.createSharing === 'undefined')) { if ((typeof req.body === 'undefined') || (typeof req.body.createSharing === 'undefined')) {
return next(new ErrorDTO(ErrorCodes.INPUT_ERROR, 'createSharing filed is missing')); return next(new ErrorDTO(ErrorCodes.INPUT_ERROR, 'createSharing filed is missing'));
} }
@ -75,6 +82,9 @@ export class SharingMWs {
} }
public static async updateSharing(req: Request, res: Response, next: NextFunction) { public static async updateSharing(req: Request, res: Response, next: NextFunction) {
if (Config.Client.Sharing.enabled === false) {
return next();
}
if ((typeof req.body === 'undefined') || (typeof req.body.updateSharing === 'undefined')) { if ((typeof req.body === 'undefined') || (typeof req.body.updateSharing === 'undefined')) {
return next(new ErrorDTO(ErrorCodes.INPUT_ERROR, 'updateSharing filed is missing')); return next(new ErrorDTO(ErrorCodes.INPUT_ERROR, 'updateSharing filed is missing'));
} }

View File

@ -110,6 +110,14 @@ export class ConfigDiagnostics {
} }
} }
static async testRandomPhotoConfig(sharing: ClientConfig.RandomPhotoConfig, config: IPrivateConfig) {
if (sharing.enabled === true &&
config.Server.database.type === DatabaseType.memory) {
throw new Error('Memory Database do not support sharing');
}
}
static async testMapConfig(map: ClientConfig.MapConfig) { static async testMapConfig(map: ClientConfig.MapConfig) {
if (map.enabled === true && (!map.googleApiKey || map.googleApiKey.length === 0)) { if (map.enabled === true && (!map.googleApiKey || map.googleApiKey.length === 0)) {
throw new Error('Maps need a valid google api key'); throw new Error('Maps need a valid google api key');
@ -192,6 +200,17 @@ export class ConfigDiagnostics {
Config.Client.Sharing.enabled = false; Config.Client.Sharing.enabled = false;
} }
try {
await ConfigDiagnostics.testRandomPhotoConfig(Config.Client.Sharing, Config);
} catch (ex) {
const err: Error = ex;
NotificationManager.warning('Random Photo is not supported with these settings. Disabling temporally. ' +
'Please adjust the config properly.', err.toString());
Logger.warn(LOG_TAG, 'Random Photo is not supported with these settings, switching off..', err.toString());
Config.Client.Sharing.enabled = false;
}
try { try {
await ConfigDiagnostics.testMapConfig(Config.Client.Map); await ConfigDiagnostics.testMapConfig(Config.Client.Map);
} catch (ex) { } catch (ex) {

View File

@ -1,8 +1,22 @@
import {DirectoryDTO} from '../../../common/entities/DirectoryDTO'; import {DirectoryDTO} from '../../../common/entities/DirectoryDTO';
import {PhotoDTO} from '../../../common/entities/PhotoDTO';
import {OrientationType, RandomQueryDTO} from '../../../common/entities/RandomQueryDTO';
export interface RandomQuery {
directory?: string;
recursive?: boolean;
orientation?: OrientationType;
fromDate?: Date;
toDate?: Date;
minResolution?: number;
maxResolution?: number;
}
export interface IGalleryManager { export interface IGalleryManager {
listDirectory(relativeDirectoryName: string, listDirectory(relativeDirectoryName: string,
knownLastModified?: number, knownLastModified?: number,
knownLastScanned?: number): Promise<DirectoryDTO>; knownLastScanned?: number): Promise<DirectoryDTO>;
getRandomPhoto(queryFilter: RandomQuery): Promise<PhotoDTO>;
} }

View File

@ -6,6 +6,7 @@ import {DiskManager} from '../DiskManger';
import {ProjectPath} from '../../ProjectPath'; import {ProjectPath} from '../../ProjectPath';
import {Config} from '../../../common/config/private/Config'; import {Config} from '../../../common/config/private/Config';
import {ReIndexingSensitivity} from '../../../common/config/private/IPrivateConfig'; import {ReIndexingSensitivity} from '../../../common/config/private/IPrivateConfig';
import {PhotoDTO} from '../../../common/entities/PhotoDTO';
export class GalleryManager implements IGalleryManager { export class GalleryManager implements IGalleryManager {
@ -23,4 +24,7 @@ export class GalleryManager implements IGalleryManager {
return DiskManager.scanDirectory(relativeDirectoryName); return DiskManager.scanDirectory(relativeDirectoryName);
} }
getRandomPhoto(RandomQuery): Promise<PhotoDTO> {
throw new Error('Random photo is not supported without database');
}
} }

View File

@ -1,4 +1,4 @@
import {IGalleryManager} from '../interfaces/IGalleryManager'; import {IGalleryManager, RandomQuery} from '../interfaces/IGalleryManager';
import {DirectoryDTO} from '../../../common/entities/DirectoryDTO'; import {DirectoryDTO} from '../../../common/entities/DirectoryDTO';
import * as path from 'path'; import * as path from 'path';
import * as fs from 'fs'; import * as fs from 'fs';
@ -11,6 +11,9 @@ import {ProjectPath} from '../../ProjectPath';
import {Config} from '../../../common/config/private/Config'; import {Config} from '../../../common/config/private/Config';
import {ISQLGalleryManager} from './IGalleryManager'; import {ISQLGalleryManager} from './IGalleryManager';
import {ReIndexingSensitivity} from '../../../common/config/private/IPrivateConfig'; import {ReIndexingSensitivity} from '../../../common/config/private/IPrivateConfig';
import {PhotoDTO} from '../../../common/entities/PhotoDTO';
import {OrientationType} from '../../../common/entities/RandomQueryDTO';
import {Brackets} from 'typeorm';
export class GalleryManager implements IGalleryManager, ISQLGalleryManager { export class GalleryManager implements IGalleryManager, ISQLGalleryManager {
@ -226,4 +229,61 @@ export class GalleryManager implements IGalleryManager, ISQLGalleryManager {
} }
async getRandomPhoto(queryFilter: RandomQuery): Promise<PhotoDTO> {
const connection = await SQLConnection.getConnection();
const photosRepository = connection.getRepository(PhotoEntity);
const query = photosRepository.createQueryBuilder('photo');
query.innerJoinAndSelect('photo.directory', 'directory');
if (queryFilter.directory) {
const directoryName = path.basename(queryFilter.directory);
const directoryParent = path.join(path.dirname(queryFilter.directory), path.sep);
query.where(new Brackets(qb => {
qb.where('directory.name = :name AND directory.path = :path', {
name: directoryName,
path: directoryParent
});
if (queryFilter.recursive) {
qb.orWhere('directory.name LIKE :text COLLATE utf8_general_ci', {text: '%' + queryFilter.directory + '%'});
}
}));
}
if (queryFilter.fromDate) {
query.andWhere('photo.metadata.creationDate >= :fromDate', {
fromDate: queryFilter.fromDate.getTime()
});
}
if (queryFilter.toDate) {
query.andWhere('photo.metadata.creationDate <= :toDate', {
toDate: queryFilter.toDate.getTime()
});
}
if (queryFilter.minResolution) {
query.andWhere('photo.metadata.size.width * photo.metadata.size.height >= :minRes', {
minRes: queryFilter.minResolution * 1000 * 1000
});
}
if (queryFilter.maxResolution) {
query.andWhere('photo.metadata.size.width * photo.metadata.size.height <= :maxRes', {
maxRes: queryFilter.maxResolution * 1000 * 1000
});
}
if (queryFilter.orientation === OrientationType.landscape) {
query.andWhere('photo.metadata.size.width >= photo.metadata.size.height');
}
if (queryFilter.orientation === OrientationType.portrait) {
query.andWhere('photo.metadata.size.width <= photo.metadata.size.height');
}
return await query.groupBy('RANDOM()').limit(1).getOne();
}
} }

View File

@ -1,6 +1,7 @@
import {DirectoryDTO} from '../../../common/entities/DirectoryDTO'; import {DirectoryDTO} from '../../../common/entities/DirectoryDTO';
import {IGalleryManager} from '../interfaces/IGalleryManager';
export interface ISQLGalleryManager { export interface ISQLGalleryManager extends IGalleryManager{
listDirectory(relativeDirectoryName: string, listDirectory(relativeDirectoryName: string,
knownLastModified?: number, knownLastModified?: number,
knownLastScanned?: number): Promise<DirectoryDTO>; knownLastScanned?: number): Promise<DirectoryDTO>;

View File

@ -1,4 +1,4 @@
import {Metadata, SharpInstance} from 'sharp'; import {Metadata, Sharp} from 'sharp';
import {Dimensions, State} from 'gm'; import {Dimensions, State} from 'gm';
import {Logger} from '../../Logger'; import {Logger} from '../../Logger';
import {ThumbnailProcessingLib} from '../../../common/config/private/IPrivateConfig'; import {ThumbnailProcessingLib} from '../../../common/config/private/IPrivateConfig';
@ -87,7 +87,7 @@ export class RendererFactory {
return async (input: RendererInput): Promise<void> => { return async (input: RendererInput): Promise<void> => {
Logger.silly('[SharpThRenderer] rendering thumbnail:' + input.imagePath); Logger.silly('[SharpThRenderer] rendering thumbnail:' + input.imagePath);
const image: SharpInstance = sharp(input.imagePath); const image: Sharp = sharp(input.imagePath);
const metadata: Metadata = await image.metadata(); const metadata: Metadata = await image.metadata();
/** /**
@ -110,9 +110,10 @@ export class RendererFactory {
} else { } else {
image image
.resize(input.size, input.size, { .resize(input.size, input.size, {
kernel: kernel kernel: kernel,
}) position: sharp.gravity.centre,
.crop(sharp.strategy.center); fit: 'cover'
});
} }
await image.jpeg().toFile(input.thPath); await image.jpeg().toFile(input.thPath);
}; };

View File

@ -84,6 +84,12 @@ export class AdminRouter {
AdminMWs.updateShareSettings, AdminMWs.updateShareSettings,
RenderingMWs.renderOK RenderingMWs.renderOK
); );
app.put('/api/settings/randomPhoto',
AuthenticationMWs.authenticate,
AuthenticationMWs.authorise(UserRoles.Admin),
AdminMWs.updateRandomPhotoSettings,
RenderingMWs.renderOK
);
app.put('/api/settings/basic', app.put('/api/settings/basic',
AuthenticationMWs.authenticate, AuthenticationMWs.authenticate,
AuthenticationMWs.authorise(UserRoles.Admin), AuthenticationMWs.authorise(UserRoles.Admin),

View File

@ -10,6 +10,7 @@ export class GalleryRouter {
this.addGetImageIcon(app); this.addGetImageIcon(app);
this.addGetImageThumbnail(app); this.addGetImageThumbnail(app);
this.addGetImage(app); this.addGetImage(app);
this.addRandom(app);
this.addDirectoryList(app); this.addDirectoryList(app);
this.addSearch(app); this.addSearch(app);
@ -38,6 +39,17 @@ export class GalleryRouter {
); );
} }
private static addRandom(app) {
app.get(['/api/gallery/random'],
AuthenticationMWs.authenticate,
AuthenticationMWs.authorise(UserRoles.Guest),
// TODO: authorize path
GalleryMWs.getRandomImage,
GalleryMWs.loadImage,
RenderingMWs.renderFile
);
}
private static addGetImageThumbnail(app) { private static addGetImageThumbnail(app) {
app.get('/api/gallery/content/:imagePath(*\.(jpg|bmp|png|gif|jpeg))/thumbnail/:size?', app.get('/api/gallery/content/:imagePath(*\.(jpg|bmp|png|gif|jpeg))/thumbnail/:size?',
AuthenticationMWs.authenticate, AuthenticationMWs.authenticate,

13
common/QueryParams.ts Normal file
View File

@ -0,0 +1,13 @@
export const QueryParams = {
gallery: {
random: {
directory: 'dir',
recursive: 'recursive',
orientation: 'orientation',
fromDate: 'fromDate',
toDate: 'toDate',
minResolution: 'fromRes',
maxResolution: 'toRes'
}
}
};

View File

@ -16,6 +16,10 @@ export module ClientConfig {
passwordProtected: boolean; passwordProtected: boolean;
} }
export interface RandomPhotoConfig {
enabled: boolean;
}
export interface MapConfig { export interface MapConfig {
enabled: boolean; enabled: boolean;
googleApiKey: string; googleApiKey: string;
@ -32,6 +36,7 @@ export module ClientConfig {
Search: SearchConfig; Search: SearchConfig;
Sharing: SharingConfig; Sharing: SharingConfig;
Map: MapConfig; Map: MapConfig;
RandomPhoto: RandomPhotoConfig;
concurrentThumbnailGenerations: number; concurrentThumbnailGenerations: number;
enableCache: boolean; enableCache: boolean;
enableOnScrollRendering: boolean; enableOnScrollRendering: boolean;
@ -73,6 +78,9 @@ export class PublicConfigClass {
enabled: true, enabled: true,
googleApiKey: '' googleApiKey: ''
}, },
RandomPhoto: {
enabled: true
},
concurrentThumbnailGenerations: 1, concurrentThumbnailGenerations: 1,
enableCache: true, enableCache: true,
enableOnScrollRendering: true, enableOnScrollRendering: true,

View File

@ -0,0 +1,13 @@
export enum OrientationType {
any = 0, portrait = 1, landscape = 2
}
export interface RandomQueryDTO {
directory?: string;
recursive?: boolean;
orientation?: OrientationType;
fromDate?: string;
toDate?: string;
minResolution?: number;
maxResolution?: number;
}

View File

@ -48,6 +48,8 @@
<app-settings-share #share [hidden]="!share.hasAvailableSettings" <app-settings-share #share [hidden]="!share.hasAvailableSettings"
[simplifiedMode]="simplifiedMode"></app-settings-share> [simplifiedMode]="simplifiedMode"></app-settings-share>
<app-settings-map #map [hidden]="!map.hasAvailableSettings" [simplifiedMode]="simplifiedMode"></app-settings-map> <app-settings-map #map [hidden]="!map.hasAvailableSettings" [simplifiedMode]="simplifiedMode"></app-settings-map>
<app-settings-random-photo #random [hidden]="!random.hasAvailableSettings"
[simplifiedMode]="simplifiedMode"></app-settings-random-photo>
<app-settings-other #other [hidden]="!other.hasAvailableSettings" <app-settings-other #other [hidden]="!other.hasAvailableSettings"
[simplifiedMode]="simplifiedMode"></app-settings-other> [simplifiedMode]="simplifiedMode"></app-settings-other>
<app-settings-indexing #indexing [hidden]="!indexing.hasAvailableSettings" <app-settings-indexing #indexing [hidden]="!indexing.hasAvailableSettings"

View File

@ -43,11 +43,12 @@ import {GalleryShareComponent} from './gallery/share/share.gallery.component';
import {ShareLoginComponent} from './sharelogin/share-login.component'; import {ShareLoginComponent} from './sharelogin/share-login.component';
import {ShareService} from './gallery/share.service'; import {ShareService} from './gallery/share.service';
import {ModalModule} from 'ngx-bootstrap/modal'; import {ModalModule} from 'ngx-bootstrap/modal';
import {BsDatepickerModule} from 'ngx-bootstrap/datepicker';
import {DatabaseSettingsComponent} from './settings/database/database.settings.component'; import {DatabaseSettingsComponent} from './settings/database/database.settings.component';
import {ToastrModule} from 'ngx-toastr'; import {ToastrModule} from 'ngx-toastr';
import {BrowserAnimationsModule} from '@angular/platform-browser/animations'; import {BrowserAnimationsModule} from '@angular/platform-browser/animations';
import {NotificationService} from './model/notification.service'; import {NotificationService} from './model/notification.service';
import {JWBootstrapSwitchModule} from 'jw-bootstrap-switch-ng2'; import {JwBootstrapSwitchNg2Module} from 'jw-bootstrap-switch-ng2';
import {ClipboardModule} from 'ngx-clipboard'; import {ClipboardModule} from 'ngx-clipboard';
import {NavigationService} from './model/navigation.service'; import {NavigationService} from './model/navigation.service';
import {InfoPanelLightboxComponent} from './gallery/lightbox/infopanel/info-panel.lightbox.gallery.component'; import {InfoPanelLightboxComponent} from './gallery/lightbox/infopanel/info-panel.lightbox.gallery.component';
@ -69,6 +70,8 @@ import {I18n, MISSING_TRANSLATION_STRATEGY} from '@ngx-translate/i18n-polyfill';
import {QueryService} from './model/query.service'; import {QueryService} from './model/query.service';
import {IconizeSortingMethod} from './pipes/IconizeSortingMethod'; import {IconizeSortingMethod} from './pipes/IconizeSortingMethod';
import {StringifySortingMethod} from './pipes/StringifySortingMethod'; import {StringifySortingMethod} from './pipes/StringifySortingMethod';
import {RandomQueryBuilderGalleryComponent} from './gallery/random-query-builder/random-query-builder.gallery.component';
import {RandomPhotoSettingsComponent} from './settings/random-photo/random-photo.settings.component';
@Injectable() @Injectable()
export class GoogleMapsConfig { export class GoogleMapsConfig {
@ -117,14 +120,15 @@ export function translationsFactory(locale: string) {
BrowserAnimationsModule, BrowserAnimationsModule,
appRoutes, appRoutes,
ClipboardModule, ClipboardModule,
JWBootstrapSwitchModule, JwBootstrapSwitchNg2Module,
TooltipModule.forRoot(), TooltipModule.forRoot(),
ToastrModule.forRoot(), ToastrModule.forRoot(),
ModalModule.forRoot(), ModalModule.forRoot(),
CollapseModule.forRoot(), CollapseModule.forRoot(),
BsDropdownModule.forRoot(), BsDropdownModule.forRoot(),
AgmCoreModule.forRoot(), AgmCoreModule.forRoot(),
SlimLoadingBarModule.forRoot() SlimLoadingBarModule.forRoot(),
BsDatepickerModule.forRoot()
], ],
declarations: [AppComponent, declarations: [AppComponent,
LoginComponent, LoginComponent,
@ -148,6 +152,7 @@ export function translationsFactory(locale: string) {
GalleryPhotoComponent, GalleryPhotoComponent,
AdminComponent, AdminComponent,
InfoPanelLightboxComponent, InfoPanelLightboxComponent,
RandomQueryBuilderGalleryComponent,
// Settings // Settings
UserMangerSettingsComponent, UserMangerSettingsComponent,
DatabaseSettingsComponent, DatabaseSettingsComponent,
@ -155,6 +160,7 @@ export function translationsFactory(locale: string) {
ThumbnailSettingsComponent, ThumbnailSettingsComponent,
SearchSettingsComponent, SearchSettingsComponent,
ShareSettingsComponent, ShareSettingsComponent,
RandomPhotoSettingsComponent,
BasicSettingsComponent, BasicSettingsComponent,
OtherSettingsComponent, OtherSettingsComponent,
IndexingSettingsComponent, IndexingSettingsComponent,
@ -185,12 +191,12 @@ export function translationsFactory(locale: string) {
deps: [LOCALE_ID] deps: [LOCALE_ID]
}, },
I18n, I18n,
/*
{provide: TRANSLATIONS, useValue: translationsFactory('en')}, {provide: TRANSLATIONS, useValue: translationsFactory('en')},
{provide: TRANSLATIONS_FORMAT, useValue: 'xlf'}, {provide: TRANSLATIONS_FORMAT, useValue: 'xlf'},
{provide: LOCALE_ID, useValue: 'en'}, {provide: LOCALE_ID, useValue: 'en'},
{provide: MISSING_TRANSLATION_STRATEGY, useValue: MissingTranslationStrategy.Ignore}, {provide: MISSING_TRANSLATION_STRATEGY, useValue: MissingTranslationStrategy.Ignore},
*/
], ],
bootstrap: [AppComponent] bootstrap: [AppComponent]
}) })

View File

@ -69,3 +69,14 @@ app-language {
padding-right: 1.0rem; padding-right: 1.0rem;
padding-left: 1.0rem; padding-left: 1.0rem;
} }
a.dropdown-item {
padding:0.3rem 1.0rem 0.3rem 0.8rem;
}
a.dropdown-item span{
padding-right: 0.8rem;
}

View File

@ -19,25 +19,38 @@
<span class="navbar-text" *ngIf="user.value"> <span class="navbar-text" *ngIf="user.value">
<span class="oi oi-person"></span> {{user.value.name}}</span> <span class="oi oi-person"></span> {{user.value.name}}</span>
</li> </li>
<li class="nav-item" *ngIf="isAdmin()">
<a style="cursor: pointer;"
class="nav-link admin-link"
[routerLink]="['/admin']">
<span class="oi oi-wrench"></span>
<span *ngIf="notificationService.notifications.length>0" class="badge">{{notificationService.notifications.length}}</span>
</a>
</li>
<li class="nav-item ml-2"> <li class="nav-item ml-2">
<app-language class="navbar-btn" isDark="true"></app-language> <app-language class="navbar-btn" isDark="true"></app-language>
</li> </li>
<li class="nav-item ml-2" *ngIf="authenticationRequired"> <div class="btn-group" dropdown placement="bottom right" container="body" >
<button class="btn btn-default navbar-btn" <button id="button-basic" dropdownToggle
style="cursor: pointer" type="button" class="btn btn-dark dropdown-toggle"
(click)="logout()"> aria-controls="dropdown-basic">
<span class="oi oi-account-logout"></span> <span class="oi oi-menu"></span>
<ng-container i18n>Logout</ng-container> <span *ngIf="isAdmin() && notificationService.notifications.length>0" class="badge">{{notificationService.notifications.length}}</span>
</button> </button>
</li> <ul id="dropdown-basic" *dropdownMenu
class="dropdown-menu dropdown-menu-right"
role="menu" aria-labelledby="button-basic" >
<ng-content select="[navbar-menu]"></ng-content>
<li role="menuitem" *ngIf="isAdmin()">
<a class="dropdown-item" href="#" [routerLink]="['/admin']">
<span class="oi oi-wrench"></span>
<span *ngIf="notificationService.notifications.length>0" class="badge">{{notificationService.notifications.length}}</span>
<ng-container i18n>Settings</ng-container>
</a>
</li>
<li role="menuitem" *ngIf="authenticationRequired">
<a class="dropdown-item" href="#" (click)="logout()" i18n>
<span class="oi oi-account-logout"></span>
<ng-container i18n>Logout</ng-container>
</a>
</li>
</ul>
</div>
</ul> </ul>

View File

@ -8,7 +8,9 @@ export class FullScreenService {
OnFullScreenChange = new Event<boolean>(); OnFullScreenChange = new Event<boolean>();
public isFullScreenEnabled(): boolean { public isFullScreenEnabled(): boolean {
return !!(document.fullscreenElement || document['mozFullScreenElement'] || document.webkitFullscreenElement); return !!(document['fullscreenElement'] ||
document['mozFullScreenElement'] ||
document['webkitFullscreenElement']);
} }
public showFullScreen(element: any) { public showFullScreen(element: any) {
@ -37,8 +39,8 @@ export class FullScreenService {
document.exitFullscreen(); document.exitFullscreen();
} else if (document['mozCancelFullScreen']) { } else if (document['mozCancelFullScreen']) {
document['mozCancelFullScreen'](); document['mozCancelFullScreen']();
} else if (document.webkitExitFullscreen) { } else if (document['webkitExitFullscreen']) {
document.webkitExitFullscreen(); document['webkitExitFullscreen']();
} }
this.OnFullScreenChange.trigger(false); this.OnFullScreenChange.trigger(false);
} }

View File

@ -2,7 +2,6 @@
<app-frame> <app-frame>
<ng-container navbar> <ng-container navbar>
<li class="nav-item" *ngIf="countDown"> <li class="nav-item" *ngIf="countDown">
<span class="navbar-text"> <span class="navbar-text">
<ng-container i18n>Link availability</ng-container> <ng-container i18n>Link availability</ng-container>
@ -20,6 +19,12 @@
</li> </li>
</ng-container> </ng-container>
<ng-container navbar-menu>
<li role="menuitem" *ngIf="showRandomPhotoBuilder">
<app-gallery-random-query-builder ></app-gallery-random-query-builder>
</li>
</ng-container>
<div body class="container-fluid" style="width: 100%; padding:0" *ngIf="_galleryService.content.value.directory"> <div body class="container-fluid" style="width: 100%; padding:0" *ngIf="_galleryService.content.value.directory">
<app-gallery-navbar [directory]="_galleryService.content.value.directory"></app-gallery-navbar> <app-gallery-navbar [directory]="_galleryService.content.value.directory"></app-gallery-navbar>

View File

@ -14,8 +14,6 @@ import {UserRoles} from '../../../common/entities/UserDTO';
import {interval} from 'rxjs'; import {interval} from 'rxjs';
import {ContentWrapper} from '../../../common/entities/ConentWrapper'; import {ContentWrapper} from '../../../common/entities/ConentWrapper';
import {PageHelper} from '../model/page.helper'; import {PageHelper} from '../model/page.helper';
import {QueryService} from '../model/query.service';
import {LightboxStates} from './lightbox/lightbox.gallery.component';
@Component({ @Component({
selector: 'app-gallery', selector: 'app-gallery',
@ -29,6 +27,8 @@ export class GalleryComponent implements OnInit, OnDestroy {
public showSearchBar = false; public showSearchBar = false;
public showShare = false; public showShare = false;
public showRandomPhotoBuilder = false;
public directories: DirectoryDTO[] = []; public directories: DirectoryDTO[] = [];
public isPhotoWithLocation = false; public isPhotoWithLocation = false;
private $counter; private $counter;
@ -142,7 +142,7 @@ export class GalleryComponent implements OnInit, OnDestroy {
} }
this.showSearchBar = Config.Client.Search.enabled && this._authService.isAuthorized(UserRoles.Guest); this.showSearchBar = Config.Client.Search.enabled && this._authService.isAuthorized(UserRoles.Guest);
this.showShare = Config.Client.Sharing.enabled && this._authService.isAuthorized(UserRoles.User); this.showShare = Config.Client.Sharing.enabled && this._authService.isAuthorized(UserRoles.User);
this.showRandomPhotoBuilder = Config.Client.RandomPhoto.enabled && this._authService.isAuthorized(UserRoles.Guest);
this.subscription.content = this._galleryService.content.subscribe(this.onContentChange); this.subscription.content = this._galleryService.content.subscribe(this.onContentChange);
this.subscription.route = this._route.params.subscribe(this.onRoute); this.subscription.route = this._route.params.subscribe(this.onRoute);

View File

@ -228,18 +228,18 @@ export class GalleryLightboxComponent implements OnDestroy, OnInit {
return; return;
} }
const event: KeyboardEvent = window.event ? <any>window.event : e; const event: KeyboardEvent = window.event ? <any>window.event : e;
switch (event.keyCode) { switch (event.key) {
case 37: case 'ArrowLeft':
if (this.activePhotoId > 0) { if (this.activePhotoId > 0) {
this.prevImage(); this.prevImage();
} }
break; break;
case 39: case 'ArrowRight':
if (this.activePhotoId < this.gridPhotoQL.length - 1) { if (this.activePhotoId < this.gridPhotoQL.length - 1) {
this.nextImage(); this.nextImage();
} }
break; break;
case 27: // escape case 'Escape': // escape
this.hide(); this.hide();
break; break;
} }

View File

@ -9,13 +9,14 @@
<agm-map <agm-map
[style.width.px]="mapDimension.width" [style.width.px]="mapDimension.width"
[style.height.px]="mapDimension.height" [style.height.px]="mapDimension.height"
[fitBounds]="latlngBounds"> [fitBounds]="true">
<agm-marker <agm-marker
*ngFor="let photo of mapPhotos" *ngFor="let photo of mapPhotos"
[latitude]="photo.latitude" [latitude]="photo.latitude"
[longitude]="photo.longitude" [longitude]="photo.longitude"
[iconUrl]="photo.iconUrl" [iconUrl]="photo.iconUrl"
(markerClick)="loadPreview(photo)"> (markerClick)="loadPreview(photo)"
[agmFitBounds]="true">
<agm-info-window> <agm-info-window>
<img *ngIf="photo.preview.thumbnail.Src" <img *ngIf="photo.preview.thumbnail.Src"
[style.width.px]="photo.preview.width" [style.width.px]="photo.preview.width"

View File

@ -29,7 +29,6 @@ export class GalleryMapLightboxComponent implements OnChanges, AfterViewInit {
@ViewChild('root') elementRef: ElementRef; @ViewChild('root') elementRef: ElementRef;
@ViewChild(AgmMap) map: AgmMap; @ViewChild(AgmMap) map: AgmMap;
public latlngBounds: LatLngBounds;
constructor(public fullScreenService: FullScreenService, constructor(public fullScreenService: FullScreenService,
@ -136,25 +135,6 @@ export class GalleryMapLightboxComponent implements OnChanges, AfterViewInit {
return obj; return obj;
}); });
this.findPhotosBounds().catch(console.error);
}
private async findPhotosBounds() {
await this.mapsAPILoader.load();
if (!window['google']) {
return;
}
this.latlngBounds = new window['google'].maps.LatLngBounds();
for (const photo of this.mapPhotos) {
this.latlngBounds.extend(new window['google'].maps.LatLng(photo.latitude, photo.longitude));
}
const clat = this.latlngBounds.getCenter().lat();
const clng = this.latlngBounds.getCenter().lng();
this.latlngBounds.extend(new window['google'].maps.LatLng(clat + 0.5, clng + 0.5));
this.latlngBounds.extend(new window['google'].maps.LatLng(clat - 0.5, clng - 0.5));
} }
@ -188,8 +168,8 @@ export class GalleryMapLightboxComponent implements OnChanges, AfterViewInit {
return; return;
} }
const event: KeyboardEvent = window.event ? <any>window.event : e; const event: KeyboardEvent = window.event ? <any>window.event : e;
switch (event.keyCode) { switch (event.key) {
case 27: // escape case 'Escape': // escape
this.hide(); this.hide();
break; break;
} }

View File

@ -8,11 +8,12 @@
[usePanning]="false" [usePanning]="false"
[draggable]="false" [draggable]="false"
[zoom]="0" [zoom]="0"
[fitBounds]="latlngBounds"> [fitBounds]="true">
<agm-marker <agm-marker
*ngFor="let photo of mapPhotos" *ngFor="let photo of mapPhotos"
[latitude]="photo.latitude" [latitude]="photo.latitude"
[longitude]="photo.longitude"> [longitude]="photo.longitude"
[agmFitBounds]="true">
</agm-marker> </agm-marker>
</agm-map> </agm-map>
<div class="overlay" (click)="click()" <div class="overlay" (click)="click()"

View File

@ -4,7 +4,7 @@ import {Dimension, IRenderable} from '../../model/IRenderable';
import {GalleryMapLightboxComponent} from './lightbox/lightbox.map.gallery.component'; import {GalleryMapLightboxComponent} from './lightbox/lightbox.map.gallery.component';
import {ThumbnailManagerService} from '../thumnailManager.service'; import {ThumbnailManagerService} from '../thumnailManager.service';
import {FullScreenService} from '../fullscreen.service'; import {FullScreenService} from '../fullscreen.service';
import {LatLngBounds, MapsAPILoader} from '@agm/core'; import {LatLngBounds, MapsAPILoader, AgmMap} from '@agm/core';
@Component({ @Component({
selector: 'app-gallery-map', selector: 'app-gallery-map',
@ -17,8 +17,7 @@ export class GalleryMapComponent implements OnChanges, IRenderable, AfterViewIni
@ViewChild(GalleryMapLightboxComponent) mapLightbox: GalleryMapLightboxComponent; @ViewChild(GalleryMapLightboxComponent) mapLightbox: GalleryMapLightboxComponent;
mapPhotos: Array<{ latitude: number, longitude: number }> = []; mapPhotos: Array<{ latitude: number, longitude: number }> = [];
public latlngBounds: LatLngBounds; @ViewChild('map') mapElement: ElementRef;
@ViewChild('map') map: ElementRef;
height = null; height = null;
@ -37,32 +36,14 @@ export class GalleryMapComponent implements OnChanges, IRenderable, AfterViewIni
}); });
this.findPhotosBounds().catch(console.error);
} }
ngAfterViewInit() { ngAfterViewInit() {
setTimeout(() => { setTimeout(() => {
this.height = this.map.nativeElement.clientHeight; this.height = this.mapElement.nativeElement.clientHeight;
}, 0); }, 0);
} }
private async findPhotosBounds() {
await this.mapsAPILoader.load();
if (!window['google']) {
return;
}
this.latlngBounds = new window['google'].maps.LatLngBounds();
for (const photo of this.mapPhotos) {
this.latlngBounds.extend(new window['google'].maps.LatLng(photo.latitude, photo.longitude));
}
const clat = this.latlngBounds.getCenter().lat();
const clng = this.latlngBounds.getCenter().lng();
this.latlngBounds.extend(new window['google'].maps.LatLng(clat + 0.5, clng + 0.5));
this.latlngBounds.extend(new window['google'].maps.LatLng(clat - 0.5, clng - 0.5));
}
click() { click() {
this.mapLightbox.show(this.getDimension()); this.mapLightbox.show(this.getDimension());
@ -70,10 +51,10 @@ export class GalleryMapComponent implements OnChanges, IRenderable, AfterViewIni
public getDimension(): Dimension { public getDimension(): Dimension {
return <Dimension>{ return <Dimension>{
top: this.map.nativeElement.offsetTop, top: this.mapElement.nativeElement.offsetTop,
left: this.map.nativeElement.offsetLeft, left: this.mapElement.nativeElement.offsetLeft,
width: this.map.nativeElement.offsetWidth, width: this.mapElement.nativeElement.offsetWidth,
height: this.map.nativeElement.offsetHeight height: this.mapElement.nativeElement.offsetHeight
}; };
} }
} }

View File

@ -0,0 +1,27 @@
.modal {
z-index: 9999;
}
.full-width {
width: 100%;
}
.row {
padding-top: 1px;
padding-bottom: 1px;
}
a.disabled {
/* Make the disabled links grayish*/
color: gray;
/* And disable the pointer events */
pointer-events: none;
}
a.dropdown-item {
padding:0.3rem 1.0rem 0.3rem 0.8rem;
}
a.dropdown-item span{
padding-right: 0.8rem;
}

View File

@ -0,0 +1,134 @@
<a class="dropdown-item {{enabled? '' : 'disabled'}}" href="#" (click)="openModal(randomModal)">
<span class="oi oi-random"></span>
<ng-container i18n>Random link</ng-container>
</a>
<ng-template #randomModal>
<!-- sharing Modal-->
<div class="modal-header">
<h5 class="modal-title" i18n>Random Link generator</h5>
<button type="button" class="close" (click)="hideModal()" data-dismiss="modal" aria-label="Close">
<span aria-hidden="true">&times;</span>
</button>
</div>
<div class="modal-body">
<div class="row">
<div class="col-7 col-sm-9">
<input id="randomLink"
name="randomLink"
placeholder="link"
class="form-control input-md"
type="text"
[ngModel]="url">
</div>
<div class="col-5 col-sm-3">
<button id="copyButton" name="copyButton"
ngxClipboard [cbContent]="url"
(cbOnSuccess)="onCopy()"
class="btn btn-primary btn-block" i18n>Copy
</button>
</div>
</div>
<hr/>
<div class="row">
<div class="col-4">
<label class="control-label" i18n>In Folder:</label>
</div>
<div class="col-8">
<input disabled type="text"
class="full-width form-control"
[ngModel]="data.directory">
</div>
</div>
<div class="row">
<div class="col-4">
<label class="control-label" i18n>Include subfolders:</label>
</div>
<div class="col-8">
<bSwitch
class="switch"
name="includeSubfolders"
[switch-on-color]="'success'"
[switch-inverse]="'inverse'"
[switch-off-text]="text.No"
[switch-on-text]="text.Yes"
[switch-handle-width]="'100'"
[switch-label-width]="'20'"
(change)="update()"
[(ngModel)]="data.recursive">
</bSwitch>
</div>
</div>
<div class="row">
<div class="col-4">
<label class="control-label" i18n>Orientation:</label>
</div>
<div class="col-4">
<select class="form-control" [(ngModel)]="data.orientation" (change)="update()" name="orientation"
required>
<option [ngValue]="OrientationType.any" i18n>Any</option>
<option [ngValue]="OrientationType.landscape" i18n>Landscape</option>
<option [ngValue]="OrientationType.portrait" i18n>Portrait</option>
</select>
</div>
</div>
<div class="row">
<div class="col-4">
<label class="control-label" i18n>Date:</label>
</div>
<div class="col-4 form-group">
<input type="text"
placeholder="from: YYYY-MM-DD"
class="form-control"
bsDatepicker
(bsValueChange)="update()"
[(ngModel)]="data.fromDate"
[bsConfig]="{ dateInputFormat: 'YYYY-MM-DD' }">
</div>
<div class="col-4 form-group">
<input type="text"
placeholder="to: YYYY-MM-DD"
class="form-control"
bsDatepicker
(bsValueChange)="update()"
[(ngModel)]="data.toDate"
[bsConfig]="{ dateInputFormat: 'YYYY-MM-DD' }">
</div>
</div>
<div class="row">
<div class="col-4">
<label class="control-label" i18n>Resolution:</label>
</div>
<div class="col-4">
<div class="input-group">
<input type="number" class="form-control"
(change)="update()"
[(ngModel)]="data.minResolution"
id="minResolution" placeholder="min" step="1" min="0">
<div class="input-group-append">
<div class="input-group-text">Mpx</div>
</div>
</div>
</div>
<div class="col-4">
<div class="input-group">
<input type="number" class="form-control"
(change)="update()"
[(ngModel)]="data.maxResolution"
id="maxResolution" placeholder="max" step="1" min="0">
<div class="input-group-append">
<div class="input-group-text">Mpx</div>
</div>
</div>
</div>
</div>
</div>
</ng-template>

View File

@ -0,0 +1,104 @@
import {Component, OnDestroy, OnInit, TemplateRef} from '@angular/core';
import {Utils} from '../../../../common/Utils';
import {GalleryService} from '../gallery.service';
import {ContentWrapper} from '../../../../common/entities/ConentWrapper';
import {Config} from '../../../../common/config/public/Config';
import {NotificationService} from '../../model/notification.service';
import {DirectoryDTO} from '../../../../common/entities/DirectoryDTO';
import {I18n} from '@ngx-translate/i18n-polyfill';
import {BsModalService} from 'ngx-bootstrap/modal';
import {BsModalRef} from 'ngx-bootstrap/modal/bs-modal-ref.service';
import {OrientationType, RandomQueryDTO} from '../../../../common/entities/RandomQueryDTO';
import {NetworkService} from '../../model/network/network.service';
@Component({
selector: 'app-gallery-random-query-builder',
templateUrl: './random-query-builder.gallery.component.html',
styleUrls: ['./random-query-builder.gallery.component.css'],
})
export class RandomQueryBuilderGalleryComponent implements OnInit, OnDestroy {
enabled = true;
url = '';
data: RandomQueryDTO = {
orientation: OrientationType.any,
directory: '',
recursive: true,
minResolution: null,
maxResolution: null,
toDate: null,
fromDate: null
};
contentSubscription = null;
OrientationType;
modalRef: BsModalRef;
text = {
Yes: 'Yes',
No: 'No'
};
constructor(public _galleryService: GalleryService,
private _notification: NotificationService,
public i18n: I18n,
private modalService: BsModalService) {
this.OrientationType = OrientationType;
this.text.Yes = i18n('Yes');
this.text.No = i18n('No');
}
ngOnInit() {
this.contentSubscription = this._galleryService.content.subscribe((content: ContentWrapper) => {
this.enabled = !!content.directory;
if (!this.enabled) {
return;
}
this.data.directory = Utils.concatUrls((<DirectoryDTO>content.directory).path, (<DirectoryDTO>content.directory).name);
});
}
ngOnDestroy() {
if (this.contentSubscription !== null) {
this.contentSubscription.unsubscribe();
}
}
update() {
setTimeout(() => {
const data = Utils.clone(this.data);
for (const key of Object.keys(data)) {
if (!data[key]) {
delete data[key];
}
}
this.url = NetworkService.buildUrl(Config.Client.publicUrl + '/api/gallery/random/', data);
}, 0);
}
openModal(template: TemplateRef<any>) {
if (!this.enabled) {
return;
}
if (this.modalRef) {
this.modalRef.hide();
}
this.modalRef = this.modalService.show(template);
document.body.style.paddingRight = '0px';
this.update();
return false;
}
onCopy() {
this._notification.success(this.i18n('Url has been copied to clipboard'));
}
public hideModal() {
this.modalRef.hide();
this.modalRef = null;
}
}

View File

@ -17,15 +17,7 @@ export class NetworkService {
private slimLoadingBarService: SlimLoadingBarService) { private slimLoadingBarService: SlimLoadingBarService) {
} }
public postJson<T>(url: string, data: any = {}): Promise<T> { public static buildUrl(url: string, data?: { [key: string]: any }) {
return this.callJson('post', url, data);
}
public putJson<T>(url: string, data: any = {}): Promise<T> {
return this.callJson('put', url, data);
}
public getJson<T>(url: string, data?: { [key: string]: any }): Promise<T> {
if (data) { if (data) {
const keys = Object.getOwnPropertyNames(data); const keys = Object.getOwnPropertyNames(data);
if (keys.length > 0) { if (keys.length > 0) {
@ -38,7 +30,19 @@ export class NetworkService {
} }
} }
} }
return this.callJson('get', url); return url;
}
public postJson<T>(url: string, data: any = {}): Promise<T> {
return this.callJson('post', url, data);
}
public putJson<T>(url: string, data: any = {}): Promise<T> {
return this.callJson('put', url, data);
}
public getJson<T>(url: string, data?: { [key: string]: any }): Promise<T> {
return this.callJson('get', NetworkService.buildUrl(url, data));
} }
public deleteJson<T>(url: string): Promise<T> { public deleteJson<T>(url: string): Promise<T> {

View File

@ -0,0 +1,3 @@
.panel-info {
text-align: center;
}

View File

@ -0,0 +1,45 @@
<form #settingsForm="ngForm">
<div class="card mb-4"
[ngClass]="settings.enabled && !_settingsService.isSupported()?'panel-warning':''">
<h5 class="card-header">
<ng-container i18n>Random Photo settings</ng-container>
<div class="switch-wrapper">
<bSwitch
class="switch"
name="enabled"
[switch-on-color]="'success'"
[switch-inverse]="'inverse'"
[switch-off-text]="text.Disabled"
[switch-on-text]="text.Enabled"
[switch-disabled]="inProgress || (!settings.enabled && !_settingsService.isSupported())"
[switch-handle-width]="'100'"
[switch-label-width]="'20'"
[(ngModel)]="settings.enabled">
</bSwitch>
</div>
</h5>
<div class="card-body">
<div [hidden]="!error" class="alert alert-danger" role="alert"><strong>Error: </strong>{{error}}</div>
<ng-container *ngIf="settings.enabled || _settingsService.isSupported()">
<div class="panel-info" i18n>
This feature enables you to generate 'random photo' urls.
That URL returns a photo random selected from your gallery.
You can use the url with 3rd party like random changing desktop background.
</div>
</ng-container>
<div class="panel-info" *ngIf="(!settings.enabled && !_settingsService.isSupported())" i18n>
Random Photo is not supported with these settings
</div>
<button class="btn btn-success float-right"
[disabled]="!settingsForm.form.valid || !changed || inProgress"
(click)="save()" i18n>Save
</button>
<button class="btn btn-default float-right"
(click)="reset()" i18n>Reset
</button>
</div>
</div>
</form>

View File

@ -0,0 +1,31 @@
import {Component} from '@angular/core';
import {SettingsComponent} from '../_abstract/abstract.settings.component';
import {AuthenticationService} from '../../model/network/authentication.service';
import {NavigationService} from '../../model/navigation.service';
import {NotificationService} from '../../model/notification.service';
import {ClientConfig} from '../../../../common/config/public/ConfigClass';
import {RandomPhotoSettingsService} from './random-photo.settings.service';
import {I18n} from '@ngx-translate/i18n-polyfill';
@Component({
selector: 'app-settings-random-photo',
templateUrl: './random-photo.settings.component.html',
styleUrls: ['./random-photo.settings.component.css',
'./../_abstract/abstract.settings.component.css'],
providers: [RandomPhotoSettingsService],
})
export class RandomPhotoSettingsComponent extends SettingsComponent<ClientConfig.RandomPhotoConfig> {
constructor(_authService: AuthenticationService,
_navigation: NavigationService,
_settingsService: RandomPhotoSettingsService,
notification: NotificationService,
i18n: I18n) {
super(i18n('Random Photo'), _authService, _navigation, _settingsService, notification, i18n, s => s.Client.RandomPhoto);
}
}

View File

@ -0,0 +1,25 @@
import {Injectable} from '@angular/core';
import {NetworkService} from '../../model/network/network.service';
import {DatabaseType} from '../../../../common/config/private/IPrivateConfig';
import {ClientConfig} from '../../../../common/config/public/ConfigClass';
import {SettingsService} from '../settings.service';
import {AbstractSettingsService} from '../_abstract/abstract.settings.service';
@Injectable()
export class RandomPhotoSettingsService extends AbstractSettingsService<ClientConfig.SharingConfig> {
constructor(private _networkService: NetworkService,
_settingsService: SettingsService) {
super(_settingsService);
}
public isSupported(): boolean {
return this._settingsService.settings.value.Server.database.type !== DatabaseType.memory;
}
public updateSettings(settings: ClientConfig.SharingConfig): Promise<void> {
return this._networkService.putJson('/settings/randomPhoto', {settings: settings});
}
}

View File

@ -33,6 +33,9 @@ export class SettingsService {
enabled: true, enabled: true,
googleApiKey: '' googleApiKey: ''
}, },
RandomPhoto: {
enabled: true
},
urlBase: '', urlBase: '',
publicUrl: '', publicUrl: '',
applicationTitle: '', applicationTitle: '',

View File

@ -16,7 +16,7 @@
"ng": "ng", "ng": "ng",
"lint": "ng lint", "lint": "ng lint",
"e2e": "ng e2e", "e2e": "ng e2e",
"run-dev": "ng build --aot -w --output-path=./dist --locale en --i18n-format xlf --i18n-file frontend/translate/messages.en.xlf --missing-translation warning", "run-dev": "ng build --aot --watch --output-path=./dist --i18n-locale en --i18n-format xlf --i18n-file frontend/translate/messages.en.xlf --i18n-missing-translation warning",
"update-translation": "gulp update-translation", "update-translation": "gulp update-translation",
"add-translation": "gulp add-translation" "add-translation": "gulp add-translation"
}, },
@ -33,46 +33,46 @@
"cookie-parser": "1.4.3", "cookie-parser": "1.4.3",
"cookie-session": "2.0.0-beta.3", "cookie-session": "2.0.0-beta.3",
"ejs": "2.6.1", "ejs": "2.6.1",
"express": "4.16.3", "express": "4.16.4",
"jimp": "0.2.28", "jimp": "0.5.4",
"locale": "0.1.0", "locale": "0.1.0",
"reflect-metadata": "0.1.12", "reflect-metadata": "0.1.12",
"sqlite3": "4.0.2", "sqlite3": "4.0.2",
"ts-exif-parser": "0.1.24", "ts-exif-parser": "0.1.24",
"ts-node-iptc": "1.0.10", "ts-node-iptc": "1.0.10",
"typeconfig": "1.0.6", "typeconfig": "1.0.6",
"typeorm": "0.2.7", "typeorm": "0.2.8",
"winston": "2.4.2" "winston": "2.4.2"
}, },
"devDependencies": { "devDependencies": {
"@agm/core": "1.0.0-beta.3", "@agm/core": "1.0.0-beta.5",
"@angular-devkit/build-angular": "0.7.1", "@angular-devkit/build-angular": "0.10.2",
"@angular-devkit/build-optimizer": "0.7.1", "@angular-devkit/build-optimizer": "0.10.2",
"@angular/animations": "6.1.0", "@angular/animations": "7.0.0",
"@angular/cli": "6.1.1", "@angular/cli": "7.0.2",
"@angular/common": "6.1.0", "@angular/common": "7.0.0",
"@angular/compiler": "6.1.0", "@angular/compiler": "7.0.0",
"@angular/compiler-cli": "6.1.0", "@angular/compiler-cli": "7.0.0",
"@angular/core": "6.1.0", "@angular/core": "7.0.0",
"@angular/forms": "6.1.0", "@angular/forms": "7.0.0",
"@angular/http": "6.1.0", "@angular/http": "7.0.0",
"@angular/language-service": "^6.1.0", "@angular/language-service": "7.0.0",
"@angular/platform-browser": "6.1.0", "@angular/platform-browser": "7.0.0",
"@angular/platform-browser-dynamic": "6.1.0", "@angular/platform-browser-dynamic": "7.0.0",
"@angular/router": "6.1.0", "@angular/router": "7.0.0",
"@ngx-translate/i18n-polyfill": "1.0.0", "@ngx-translate/i18n-polyfill": "1.0.0",
"@types/bcryptjs": "2.4.1", "@types/bcryptjs": "2.4.2",
"@types/chai": "4.1.4", "@types/chai": "4.1.6",
"@types/cookie-session": "2.0.35", "@types/cookie-session": "2.0.36",
"@types/express": "4.16.0", "@types/express": "4.16.0",
"@types/gm": "1.18.0", "@types/gm": "1.18.1",
"@types/jasmine": "2.8.8", "@types/jasmine": "2.8.9",
"@types/node": "10.5.4", "@types/node": "10.12.0",
"@types/sharp": "0.17.9", "@types/sharp": "0.21.0",
"@types/winston": "^2.3.9", "@types/winston": "2.3.9",
"bootstrap": "4.1.3", "bootstrap": "4.1.3",
"chai": "4.1.2", "chai": "4.2.0",
"codelyzer": "4.4.2", "codelyzer": "4.5.0",
"core-js": "2.5.7", "core-js": "2.5.7",
"ejs-loader": "0.3.1", "ejs-loader": "0.3.1",
"gulp": "3.9.1", "gulp": "3.9.1",
@ -81,45 +81,48 @@
"gulp-zip": "4.2.0", "gulp-zip": "4.2.0",
"hammerjs": "2.0.8", "hammerjs": "2.0.8",
"intl": "1.2.5", "intl": "1.2.5",
"jasmine-core": "3.1.0", "jasmine-core": "3.2.1",
"jasmine-spec-reporter": "4.2.1", "jasmine-spec-reporter": "4.2.1",
"jw-bootstrap-switch-ng2": "1.0.10", "jw-bootstrap-switch-ng2": "2.0.2",
"karma": "2.0.5", "karma": "3.0.0",
"karma-chrome-launcher": "2.2.0", "karma-chrome-launcher": "2.2.0",
"karma-cli": "1.0.1", "karma-cli": "1.0.1",
"karma-coverage-istanbul-reporter": "2.0.1", "karma-coverage-istanbul-reporter": "2.0.4",
"karma-jasmine": "1.1.2", "karma-jasmine": "1.1.2",
"karma-jasmine-html-reporter": "1.2.0", "karma-jasmine-html-reporter": "1.3.1",
"karma-remap-istanbul": "0.6.0", "karma-remap-istanbul": "0.6.0",
"karma-systemjs": "0.16.0", "karma-systemjs": "0.16.0",
"merge2": "1.2.2", "merge2": "1.2.3",
"mocha": "5.2.0", "mocha": "5.2.0",
"ng2-cookies": "1.0.12", "ng2-cookies": "1.0.12",
"ng2-slim-loading-bar": "4.0.0", "ng2-slim-loading-bar": "4.0.0",
"ngx-bootstrap": "3.0.1", "ngx-bootstrap": "3.0.1",
"ngx-clipboard": "11.1.1", "ngx-clipboard": "11.1.9",
"ngx-toastr": "8.10.0", "ngx-toastr": "9.1.1",
"open-iconic": "1.1.1", "open-iconic": "1.1.1",
"protractor": "5.4.0", "protractor": "5.4.1",
"remap-istanbul": "0.11.1", "remap-istanbul": "0.12.0",
"rimraf": "2.6.2", "rimraf": "2.6.2",
"run-sequence": "2.2.1", "run-sequence": "2.2.1",
"rxjs": "6.2.2", "rxjs": "6.3.3",
"rxjs-compat": "^6.2.2", "rxjs-compat": "^6.3.3",
"ts-helpers": "1.1.2", "ts-helpers": "1.1.2",
"ts-node": "7.0.0", "ts-node": "7.0.1",
"tslint": "5.11.0", "tslint": "5.11.0",
"typescript": "2.9.2", "typescript": "3.1.3",
"xlf-google-translate": "1.0.0-beta.11", "xlf-google-translate": "1.0.0-beta.11",
"zone.js": "0.8.26" "zone.js": "0.8.26"
}, },
"resolutions": {
"natives": "1.1.3"
},
"optionalDependencies": { "optionalDependencies": {
"mysql": "2.16.0", "mysql": "2.16.0",
"bcrypt": "3.0.0", "bcrypt": "3.0.2",
"gm": "1.23.1", "gm": "1.23.1",
"sharp": "0.20.5" "sharp": "0.21.0"
}, },
"engines": { "engines": {
"node": ">= 6.9 <10.0" "node": ">= 6.9 <=10.0"
} }
} }