<?php

namespace App\Http\Controllers\Admin;

use App\Album;
use App\AlbumDefaultAnonymousPermission;
use App\AlbumDefaultGroupPermission;
use App\AlbumDefaultUserPermission;
use App\AlbumRedirect;
use App\Facade\Theme;
use App\Facade\UserConfig;
use App\Group;
use App\Helpers\DbHelper;
use App\Helpers\FileHelper;
use App\Helpers\MiscHelper;
use App\Helpers\PermissionsHelper;
use App\Http\Controllers\Controller;
use App\Http\Requests;
use App\Label;
use App\Permission;
use App\Photo;
use App\Services\AlbumService;
use App\Services\PhotoService;
use App\Storage;
use App\User;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\App;
use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\DB;
use Illuminate\Support\Facades\View;

class AlbumController extends Controller
{
    public static function doesGroupHaveDefaultPermission(Group $group, Permission $permission)
    {
        return AlbumDefaultGroupPermission::where([
                'group_id' => $group->id,
                'permission_id' => $permission->id
            ])->count() > 0;
    }

    public static function doesUserHaveDefaultPermission($user, Permission $permission)
    {
        // User will be null for anonymous users
        if (is_null($user))
        {
            return AlbumDefaultAnonymousPermission::where(['permission_id' => $permission->id])->count() > 0;
        }
        else
        {
            return AlbumDefaultUserPermission::where([
                    'user_id' => $user->id,
                    'permission_id' => $permission->id
                ])->count() > 0;
        }
    }

    public function __construct()
    {
        $this->middleware('auth');
        View::share('is_admin', true);
    }

    public function analyse($id, $queue_token)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        $album = $this->loadAlbum($id, 'upload-photos');
        $photos = $album->photos()
            ->where('is_analysed', false)
            ->orderBy('created_at')
            ->get();

        if (count($photos) == 0)
        {
            return redirect(route('albums.show', ['id' => $album->id]));
        }

        return Theme::render('admin.analyse_album', ['album' => $album, 'photos' => $photos, 'queue_token' => $queue_token]);
    }

    /**
     * Show the form for creating a new resource.
     *
     * @return \Illuminate\Http\Response
     */
    public function create(Request $request)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        $albumSources = [];
        foreach (Storage::where('is_active', true)->orderBy('name')->get() as $storage)
        {
            $albumSources[$storage->id] = $storage->name;
        }

        if (count($albumSources) == 0)
        {
            $request->session()->flash('info', trans('admin.create_album_no_storage'));
            return redirect(route('storage.create'));
        }

        $albumService = new AlbumService();
        $defaultSource = Storage::where('is_default', true)->limit(1)->first();

        return Theme::render('admin.create_album', [
            'album_sources' => $albumSources,
            'default_storage_id' => (!is_null($defaultSource) ? $defaultSource->id : 0),
            'parent_albums' => $albumService->getFlattenedAlbumTree()
        ]);
    }

    public function defaultPermissions()
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        $addNewGroups = [];
        $existingGroups = [];
        foreach (Group::orderBy('name')->get() as $group)
        {
            if (AlbumDefaultGroupPermission::where('group_id', $group->id)->count() == 0)
            {
                $addNewGroups[] = $group;
            }
            else
            {
                $existingGroups[] = $group;
            }
        }

        $existingUsers = [];
        foreach (User::orderBy('name')->get() as $user)
        {
            if (AlbumDefaultUserPermission::where('user_id', $user->id)->count() > 0)
            {
                $existingUsers[] = $user;
            }
        }

        return Theme::render('admin.album_default_permissions', [
            'add_new_groups' => $addNewGroups,
            'all_permissions' => Permission::where('section', 'album')->get(),
            'existing_groups' => $existingGroups,
            'existing_users' => $existingUsers
        ]);
    }

    public function delete($id)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        $album = $this->loadAlbum($id, 'delete');

        return Theme::render('admin.delete_album', ['album' => $album]);
    }

    public function deleteRedirect(Request $request, $id, $redirectId)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        $album = $this->loadAlbum($id, 'delete');

        /** @var AlbumRedirect $redirect */
        $redirect = $album->redirects()->where('id', $redirectId)->first();
        if (is_null($redirect))
        {
            App::abort(404);
            return;
        }

        $redirect->delete();
        $request->session()->flash('success', trans('admin.delete_redirect_success_message'));
        return redirect(route('albums.show', ['id' => $id, 'tab' => 'redirects']));
    }

    /**
     * Remove the specified resource from storage.
     *
     * @param  int  $id
     * @return \Illuminate\Http\Response
     */
    public function destroy(Request $request, $id)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        $album = $this->loadAlbum($id, 'delete');

        if ($album->children()->count() > 0)
        {
            $request->session()->flash('error', trans('admin.delete_album_failed_children', ['album' => $album->name]));
            return redirect(route('albums.index'));
        }

        // Delete all the photo files
        /** @var Photo $photo */
        foreach ($album->photos as $photo)
        {
            $photoService = new PhotoService($photo);
            $photoService->delete();
        }

        $album->getAlbumSource()->deleteAlbumContents();
        $album->delete();

        $request->session()->flash('success', trans('admin.delete_album_success_message', ['album' => $album->name]));

        return redirect(route('albums.index'));
    }

    /**
     * Show the form for editing the specified resource.
     *
     * @param  int  $id
     * @return \Illuminate\Http\Response
     */
    public function edit(Request $request, $id)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        $album = $this->loadAlbum($id);

        if (!$request->session()->has('_old_input'))
        {
            $request->session()->flash('_old_input', $album->toArray());
        }

        $albumService = new AlbumService();

        return Theme::render('admin.edit_album', [
            'album' => $album,
            'parent_albums' => $albumService->getFlattenedAlbumTree()
        ]);
    }

    /**
     * Display a listing of the resource.
     *
     * @return \Illuminate\Http\Response
     */
    public function index(Request $request)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        // Only get top-level albums
        $albums = DbHelper::getAlbumsForCurrentUser(0);
        foreach ($albums as $album)
        {
            $this->loadChildAlbums($album);
        }

        return Theme::render('admin.list_albums', [
            'albums' => $albums,
            'success' => $request->session()->get('success'),
        ]);
    }

    /**
     * Show the form for editing the specified resource.
     *
     * @param  int  $id
     * @return \Illuminate\Http\Response
     */
    public function metadata(Request $request, $id)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        /** @var Album $album */
        $album = $this->loadAlbum($id);

        $photosNeededToUpdate = $album->photos()->where('metadata_version', '<', PhotoService::METADATA_VERSION)->get();

        return Theme::render('admin.album_metadata', [
            'album' => $album,
            'current_metadata' => PhotoService::METADATA_VERSION,
            'photos' => $photosNeededToUpdate,
            'queue_token' => MiscHelper::randomString()
        ]);
    }

    public function setDefaultGroupPermissions(Request $request)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        if ($request->get('action') == 'add_group' && $request->has('group_id'))
        {
            /* Add a new group to the default permission list */

            /** @var Group $group */
            $group = Group::where('id', $request->get('group_id'))->first();
            if (is_null($group))
            {
                App::abort(404);
            }

            // Link all default permissions to the group
            /** @var Permission $permission */
            foreach (Permission::where(['section' => 'album', 'is_default' => true])->get() as $permission)
            {
                $defaultPermission = new AlbumDefaultGroupPermission();
                $defaultPermission->group_id = $group->id;
                $defaultPermission->permission_id = $permission->id;
                $defaultPermission->save();
            }
        }
        else if ($request->get('action') == 'update_group_permissions')
        {
            /* Update existing group permissions for this album */
            AlbumDefaultGroupPermission::truncate();

            $permissions = $request->get('permissions');
            if (is_array($permissions))
            {
                foreach ($permissions as $groupID => $permissionIDs)
                {
                    foreach ($permissionIDs as $permissionID)
                    {
                        $defaultPermission = new AlbumDefaultGroupPermission();
                        $defaultPermission->group_id = $groupID;
                        $defaultPermission->permission_id = $permissionID;
                        $defaultPermission->save();
                    }
                }
            }
        }

        // Rebuild the permissions cache
        $helper = new PermissionsHelper();
        $helper->rebuildCache();

        return redirect(route('albums.defaultPermissions'));
    }

    public function setDefaultUserPermissions(Request $request)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        if ($request->get('action') == 'add_user' && $request->has('user_id'))
        {
            /* Add a new user to the permission list for this album */

            /** @var User $user */
            $user = User::where('id', $request->get('user_id'))->first();
            if (is_null($user))
            {
                App::abort(404);
            }

            // Link all default permissions to the group
            /** @var Permission $permission */
            foreach (Permission::where(['section' => 'album', 'is_default' => true])->get() as $permission)
            {
                $defaultPermission = new AlbumDefaultUserPermission();
                $defaultPermission->user_id = $user->id;
                $defaultPermission->permission_id = $permission->id;
                $defaultPermission->save();
            }
        }
        else if ($request->get('action') == 'update_user_permissions')
        {
            /* Update existing user and anonymous permissions for this album */
            AlbumDefaultAnonymousPermission::truncate();
            AlbumDefaultUserPermission::truncate();

            $permissions = $request->get('permissions');
            if (is_array($permissions))
            {
                if (isset($permissions['anonymous']))
                {
                    foreach ($permissions['anonymous'] as $permissionID)
                    {
                        $defaultPermission = new AlbumDefaultAnonymousPermission();
                        $defaultPermission->permission_id = $permissionID;
                        $defaultPermission->save();
                    }
                }

                foreach ($permissions as $key => $value)
                {
                    $userID = intval($key);
                    if ($userID == 0)
                    {
                        // Skip non-numeric IDs (e.g. anonymous)
                        continue;
                    }

                    foreach ($value as $permissionID)
                    {
                        $defaultPermission = new AlbumDefaultUserPermission();
                        $defaultPermission->user_id = $userID;
                        $defaultPermission->permission_id = $permissionID;
                        $defaultPermission->save();
                    }
                }
            }
        }

        // Rebuild the permissions cache
        $helper = new PermissionsHelper();
        $helper->rebuildCache();

        return redirect(route('albums.defaultPermissions'));
    }

    public function setGroupPermissions(Request $request, $id)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        /** @var Album $album */
        $album = $this->loadAlbum($id, 'change-permissions');

        if ($request->get('action') == 'add_group' && $request->has('group_id'))
        {
            /* Add a new group to the permission list for this album */

            /** @var Group $group */
            $group = Group::where('id', $request->get('group_id'))->first();
            if (is_null($group))
            {
                App::abort(404);
            }

            // Link all default permissions to the group
            /** @var Permission $permission */
            foreach (Permission::where(['section' => 'album', 'is_default' => true])->get() as $permission)
            {
                $album->groupPermissions()->attach($permission->id, [
                    'group_id' => $group->id,
                    'created_at' => new \DateTime(),
                    'updated_at' => new \DateTime()
                ]);
            }
        }
        else if ($request->get('action') == 'update_group_permissions')
        {
            /* Update existing group permissions for this album */
            $album->groupPermissions()->detach();

            $permissions = $request->get('permissions');
            if (is_array($permissions))
            {
                foreach ($permissions as $groupID => $permissionIDs)
                {
                    foreach ($permissionIDs as $permissionID)
                    {
                        $album->groupPermissions()->attach($permissionID, [
                            'group_id' => $groupID,
                            'created_at' => new \DateTime(),
                            'updated_at' => new \DateTime()
                        ]);
                    }
                }
            }
        }

        $album->save();

        // Rebuild the permissions cache
        $helper = new PermissionsHelper();
        $helper->rebuildCache();

        return redirect(route('albums.show', [$album->id, 'tab' => 'permissions']));
    }

    public function setUserPermissions(Request $request, $id)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        /** @var Album $album */
        $album = $this->loadAlbum($id, 'change-permissions');

        if ($request->get('action') == 'add_user' && $request->has('user_id'))
        {
            /* Add a new user to the permission list for this album */

            /** @var User $user */
            $user = User::where('id', $request->get('user_id'))->first();
            if (is_null($user))
            {
                App::abort(404);
            }

            // Link all default permissions to the group
            /** @var Permission $permission */
            foreach (Permission::where(['section' => 'album', 'is_default' => true])->get() as $permission)
            {
                $album->userPermissions()->attach($permission->id, [
                    'user_id' => $user->id,
                    'created_at' => new \DateTime(),
                    'updated_at' => new \DateTime()
                ]);
            }
        }
        else if ($request->get('action') == 'update_user_permissions')
        {
            /* Update existing user and anonymous permissions for this album */
            $album->anonymousPermissions()->detach();
            $album->userPermissions()->detach();

            $permissions = $request->get('permissions');
            if (is_array($permissions))
            {
                if (isset($permissions['anonymous']))
                {
                    foreach ($permissions['anonymous'] as $permissionID)
                    {
                        $album->anonymousPermissions()->attach($permissionID, [
                            'created_at' => new \DateTime(),
                            'updated_at' => new \DateTime()
                        ]);
                    }
                }

                foreach ($permissions as $key => $value)
                {
                    $userID = intval($key);
                    if ($userID == 0)
                    {
                        // Skip non-numeric IDs (e.g. anonymous)
                        continue;
                    }

                    foreach ($value as $permissionID)
                    {
                        $album->userPermissions()->attach($permissionID, [
                            'user_id' => $userID,
                            'created_at' => new \DateTime(),
                            'updated_at' => new \DateTime()
                        ]);
                    }
                }
            }
        }

        $album->save();

        // Rebuild the permissions cache
        $helper = new PermissionsHelper();
        $helper->rebuildCache();

        return redirect(route('albums.show', [$album->id, 'tab' => 'permissions']));
    }

    /**
     * Display the specified resource.
     *
     * @param  int  $id
     * @return \Illuminate\Http\Response
     */
    public function show(Request $request, $id)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        $album = $this->loadAlbum($id);
        $photos = $album->photos()
            ->orderBy(DB::raw('COALESCE(taken_at, created_at)'))
            ->paginate(UserConfig::get('items_per_page_admin'));

        // See if we can upload (need the GD extension)
        $isUploadEnabled = extension_loaded('gd');
        $fileUploadLimit = MiscHelper::convertToBytes(ini_get('upload_max_filesize')) / (1024*1024);
        $postLimit = MiscHelper::convertToBytes(ini_get('post_max_size')) / (1024*1024);
        $fileUploadOrPostLowerLimit = ($postLimit < $fileUploadLimit) ? $postLimit : $fileUploadLimit;

        $allowedAlbumViews = [];
        foreach (UserConfig::allowedAlbumViews() as $view)
        {
            $allowedAlbumViews[$view] = trans(sprintf('gallery.album_views.%s', $view));
        }

        $addNewGroups = [];
        $existingGroups = [];
        foreach (Group::orderBy('name')->get() as $group)
        {
            if ($album->groupPermissions()->where('group_id', $group->id)->count() == 0)
            {
                $addNewGroups[] = $group;
            }
            else
            {
                $existingGroups[] = $group;
            }
        }

        $existingUsers = [];
        foreach (User::orderBy('name')->get() as $user)
        {
            if ($album->userPermissions()->where('user_id', $user->id)->count() > 0)
            {
                $existingUsers[] = $user;
            }
        }

        $activeTab = $request->get('tab');

        if (!$request->session()->has('_old_input'))
        {
            $request->session()->flash('_old_input', $album->toArray());
        }

        // Get the cameras used in this album
        $cameras = $album->cameras();

        return Theme::render('admin.show_album', [
            'active_tab' => (strlen($activeTab) == 0) ? 'photos' : $activeTab,
            'album' => $album,
            'add_new_groups' => $addNewGroups,
            'all_permissions' => Permission::where('section', 'album')->get(),
            'allowed_views' => $allowedAlbumViews,
            'bulk_actions' => [
                'rotate_left' => trans('admin.photo_actions.rotate_left'),
                'rotate_right' => trans('admin.photo_actions.rotate_right'),
                '-' => '-----',
                'flip_horizontal' => trans('admin.photo_actions.flip_horizontal'),
                'flip_vertical' => trans('admin.photo_actions.flip_vertical'),
                'flip_both' => trans('admin.photo_actions.flip_both'),
                '--' => '-----',
                'change_album' => trans('admin.photo_actions.change_album'),
                'refresh_thumbnails' => trans('admin.photo_actions.refresh_thumbnails'),
                'delete' => trans('admin.photo_actions.delete')
            ],
            'cameras' => $cameras,
            'error' => $request->session()->get('error'),
            'existing_groups' => $existingGroups,
            'existing_users' => $existingUsers,
            'file_upload_limit' => $fileUploadLimit,
            'is_upload_enabled' => $isUploadEnabled,
            'labels' => Label::all(),
            'max_post_limit' => $postLimit,
            'max_post_limit_bulk' => $fileUploadOrPostLowerLimit,
            'photos' => $photos,
            'queue_token' => MiscHelper::randomString(),
            'success' => $request->session()->get('success'),
            'warning' => $request->session()->get('warning')
        ]);
    }

    /**
     * Store a newly created resource in storage.
     *
     * @param  \Illuminate\Http\Request  $request
     * @return \Illuminate\Http\Response
     */
    public function store(Requests\StoreAlbumRequest $request)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        $album = new Album();
        $album->fill($request->only(['name', 'description', 'storage_id', 'parent_album_id']));
        $album->is_permissions_inherited = (strtolower($request->get('is_permissions_inherited')) == 'on');

        if (strlen($album->parent_album_id) == 0)
        {
            $album->parent_album_id = null;
        }

        $album->default_view = UserConfig::get('default_album_view');
        $album->user_id = Auth::user()->id;

        $album->generateAlias();
        $album->generateUrlPath();
        $album->save();

        // Link the default permissions (if a public album)
        $isPrivate = (strtolower($request->get('is_private')) == 'on');
        if (!$album->is_permissions_inherited && !$isPrivate)
        {
            $defaultAlbumUserPermissions = AlbumDefaultUserPermission::all();
            $defaultAlbumGroupPermissions = AlbumDefaultGroupPermission::all();
            $defaultAnonPermissions = AlbumDefaultAnonymousPermission::all();

            /** @var AlbumDefaultAnonymousPermission $permission */
            foreach ($defaultAnonPermissions as $permission)
            {
                $album->anonymousPermissions()->attach($permission->permission_id, [
                    'created_at' => new \DateTime(),
                    'updated_at' => new \DateTime()
                ]);
            }

            /** @var AlbumDefaultGroupPermission $permission */
            foreach ($defaultAlbumGroupPermissions as $permission)
            {
                $album->groupPermissions()->attach($permission->permission_id, [
                    'group_id' => $permission->group_id,
                    'created_at' => new \DateTime(),
                    'updated_at' => new \DateTime()
                ]);
            }

            /** @var AlbumDefaultUserPermission $permission */
            foreach ($defaultAlbumUserPermissions as $permission)
            {
                $album->userPermissions()->attach($permission->permission_id, [
                    'user_id' => $permission->user_id,
                    'created_at' => new \DateTime(),
                    'updated_at' => new \DateTime()
                ]);
            }
        }

        // Rebuild the permissions cache
        $helper = new PermissionsHelper();
        $helper->rebuildCache();

        return redirect(route('albums.show', ['id' => $album->id]));
    }

    public function storeRedirect(Requests\StoreAlbumRedirectRequest $request, $id)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        $album = $this->loadAlbum($id);

        $redirect = new AlbumRedirect();
        $redirect->fill($request->only('source_url'));
        $redirect->album_id = $album->id;
        $redirect->save();

        $request->session()->flash('success', trans('admin.create_redirect_success_message'));
        return redirect(route('albums.show', ['id' => $id, 'tab' => 'redirects']));
    }

    /**
     * Update the specified resource in storage.
     *
     * @param  \Illuminate\Http\Request  $request
     * @param  int  $id
     * @return \Illuminate\Http\Response
     */
    public function update(Requests\StoreAlbumRequest $request, $id)
    {
        $this->authorizeAccessToAdminPanel('admin:manage-albums');

        $album = $this->loadAlbum($id);
        $currentParentID = $album->parent_album_id;

        $album->fill($request->only(['name', 'description', 'parent_album_id']));
        $album->is_permissions_inherited = (strtolower($request->get('is_permissions_inherited')) == 'on');

        if (strlen($album->parent_album_id) == 0)
        {
            $album->parent_album_id = null;
        }

        // These keys are optional and may or may not be in the request, depending on the page requesting it
        foreach (['storage_id', 'default_view'] as $key)
        {
            if ($request->has($key))
            {
                $album->$key = $request->get($key);
            }
        }

        // Re-generate the URL path to ensure it's correct if the parent has changed
        if ($currentParentID != $album->parent_album_id)
        {
            // Create a redirect if required
            $redirectData = [
                'album_id' => $album->id,
                'source_url' => sprintf('/%s', $album->url_path)
            ];

            if (strtolower($request->get('preserve_url_redirect')) == 'on' && AlbumRedirect::where($redirectData)->count() == 0)
            {
                $redirect = new AlbumRedirect();
                $redirect->fill($redirectData);
                $redirect->save();
            }

            // Update the URL path
            $album->generateUrlPath();
        }
        
        $album->save();

        // Rebuild the permissions cache
        $helper = new PermissionsHelper();
        $helper->rebuildCache();

        $request->session()->flash('success', trans('admin.album_saved_successfully', ['name' => $album->name]));

        return redirect(route('albums.show', ['id' => $id]));
    }

    /**
     * @param $id
     * @return Album
     */
    private function loadAlbum($id, $permission = 'edit')
    {
        $album = Album::where('id', intval($id))->first();
        if (is_null($album))
        {
            App::abort(404);
            return null;
        }

        $this->authorize($permission, $album);

        return $album;
    }

    private function loadChildAlbums(Album $album)
    {
        $album->child_albums = DbHelper::getChildAlbums($album);
        foreach ($album->child_albums as $childAlbum)
        {
            $this->loadChildAlbums($childAlbum);
        }
    }
}