<template>
    <div class="bg-white p-6 rounded border border-grey-light">
        <div class="filemanager-breadcrumbs">
            /<span class="px-1">home</span><!--
            -->/<router-link :to="{ name: 'server-files' }" class="px-1">container</router-link><!--
            --><span v-for="crumb in breadcrumbs" class="inline-block">
                <span v-if="crumb.path">
                    /<router-link :to="{ name: 'server-files', params: { path: crumb.path } }" class="px-1">{{crumb.directoryName}}</router-link>
                </span>
                <span v-else>
                    /<span class="px-1 font-semibold">{{crumb.directoryName}}</span>
                </span>
            </span>
        </div>
        <div v-if="loading">
            <div class="spinner spinner-xl blue"></div>
        </div>
        <div v-else-if="!loading && errorMessage">
            <div class="alert error" v-text="errorMessage"></div>
        </div>
        <div class="filemanager" v-else>
            <div class="header">
                <div class="flex-none w-8"></div>
                <div class="flex-1">Name</div>
                <div class="flex-1 text-right">Size</div>
                <div class="flex-1 text-right">Modified</div>
                <div class="flex-none w-1/6">Actions</div>
            </div>
            <div v-if="!directories.length && !files.length">
                <p class="text-grey text-sm text-center p-6 pb-4">This directory is empty.</p>
            </div>
            <div v-else>
                <div v-for="directory in directories">
                    <file-manager-folder-row :directory="directory"/>
                </div>
                <div v-for="file in files">
                    <file-manager-file-row :file="file" :editable="editableFiles" />
                </div>
            </div>
        </div>
    </div>
</template>

<script>
    import map from 'lodash/map';
    import filter from 'lodash/filter';
    import isObject from 'lodash/isObject';
    import { mapState } from 'vuex';
    import FileManagerFileRow from '../components/FileManagerFileRow';
    import FileManagerFolderRow from '../components/FileManagerFolderRow';

    export default {
        name: 'file-manager-page',
        components: {FileManagerFolderRow, FileManagerFileRow},

        computed: {
            ...mapState('server', ['server', 'credentials']),
            ...mapState('socket', ['connected']),

            /**
             * Configure the breadcrumbs that display on the filemanager based on the directory that the
             * user is currently in.
             */
            breadcrumbs: function () {
                const directories = this.currentDirectory.replace(/^\/|\/$/, '').split('/');
                if (directories.length < 1 || !directories[0]) {
                    return [];
                }

                return map(directories, function (value, key) {
                    if (key === directories.length - 1) {
                        return {directoryName: value};
                    }

                    return {
                        directoryName: value,
                        path: directories.slice(0, key + 1).join('/'),
                    };
                });
            }
        },

        watch: {
            /**
             * When the route changes reload the directory.
             */
            '$route': function (to) {
                this.currentDirectory = to.params.path || '/';
            },

            /**
             * Watch the current directory setting and when it changes update the file listing.
             */
            currentDirectory: function () {
                this.listDirectory();
            },

            /**
             * When we reconnect to the Daemon make sure we grab a listing of all of the files
             * so that the error message disappears and we then load in a fresh listing.
             */
            connected: function () {
                if (this.connected) {
                    this.listDirectory();
                }
            }
        },

        data: function () {
            return {
                currentDirectory: this.$route.params.path || '/',
                loading: true,
                errorMessage: null,

                directories: [],
                editableFiles: [],
                files: [],
            };
        },

        mounted: function () {
            this.listDirectory();
        },

        methods: {
            /**
             * List the contents of a directory.
             */
            listDirectory: function () {
                this.loading = true;

                window.axios.get(this.route('server.files', {
                    server: this.$route.params.id,
                    directory: encodeURI(this.currentDirectory.replace(/^\/|\/$/, '')),
                }))
                    .then((response) => {
                        this.files = filter(response.data.contents, function (o) {
                            return o.file;
                        });

                        this.directories = filter(response.data.contents, function (o) {
                            return o.directory;
                        });

                        this.editableFiles = response.data.editable;
                        this.errorMessage = null;
                    })
                    .catch(err => {
                        console.error({err});
                        if (err.response.status === 404) {
                            this.errorMessage = 'The directory you requested could not be located on the server.';
                            return;
                        }

                        if (err.response.data && isObject(err.response.data.errors)) {
                            err.response.data.errors.forEach(error => {
                                this.errorMessage = error.detail;
                            });
                        }
                    })
                    .finally(() => {
                        this.loading = false;
                    });
            },
        }
    };
</script>