Skip to content

Commit

Permalink
NEW LinkFieldController to handle FormSchema
Browse files Browse the repository at this point in the history
  • Loading branch information
emteknetnz committed Oct 26, 2023
1 parent f0a3b25 commit 7aaa11a
Show file tree
Hide file tree
Showing 17 changed files with 1,627 additions and 90 deletions.
1 change: 0 additions & 1 deletion _config.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,5 +5,4 @@

// Avoid creating global variables
call_user_func(function () {

});
2 changes: 1 addition & 1 deletion _config/config.yml
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ Name: linkfield

SilverStripe\Admin\LeftAndMain:
extensions:
- SilverStripe\LinkField\Extensions\LeftAndMain
- SilverStripe\LinkField\Extensions\LeftAndMainExtension

SilverStripe\Admin\ModalController:
extensions:
Expand Down
23 changes: 23 additions & 0 deletions _graphql/models.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
# This was copied from Elemental models.yml so has some extra stuff commented out
SilverStripe\LinkField\Models\Link:
# fields:
# id: true
# lastEdited: true
# absoluteLink: String
# title: true
# showTitle: true
# sort: true
# blockSchema: ObjectType
# obsoleteClassName: String
# isPublished: Boolean
# isLiveVersion: Boolean
# canCreate: Boolean
# canPublish: Boolean
# canUnpublish: Boolean
# canDelete: Boolean
operations:
# copyToStage: true
# readOne: true
delete: true
# publish: true
# unpublish: true
1,050 changes: 1,049 additions & 1 deletion client/dist/js/bundle.js

Large diffs are not rendered by default.

69 changes: 68 additions & 1 deletion client/dist/styles/bundle.css

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 2 additions & 0 deletions client/src/boot/registerQueries.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,11 @@
import Injector from 'lib/Injector';
import readLinkTypes from 'state/linkTypes/readLinkTypes';
import readLinkDescription from 'state/linkDescription/readLinkDescription';
import deleteLink from 'state/linkOperations/deleteLink';

const registerQueries = () => {
Injector.query.register('readLinkTypes', readLinkTypes);
Injector.query.register('readLinkDescription', readLinkDescription);
Injector.query.register('deleteLink', deleteLink);
};
export default registerQueries;
66 changes: 53 additions & 13 deletions client/src/components/LinkField/LinkField.js
Original file line number Diff line number Diff line change
@@ -1,35 +1,49 @@
import React, { Fragment, useState } from 'react';
import { compose } from 'redux';
import { bindActionCreators, compose } from 'redux';
import { connect } from 'react-redux';
import { inject, injectGraphql, loadComponent } from 'lib/Injector';
import fieldHolder from 'components/FieldHolder/FieldHolder';
import * as toastsActions from 'state/toasts/ToastsActions';
import backend from 'lib/Backend';
import Config from 'lib/Config';

const LinkField = ({ id, loading, Loading, data, LinkPicker, onChange, types, linkDescription, ...props }) => {
const LinkField = ({ loading, Loading, data, LinkPicker, onChange, types, linkDescription, submitForm, actions, ...props }) => {
if (loading) {
return <Loading />;
}

const [editing, setEditing] = useState(false);
const [newTypeKey, setNewTypeKey] = useState('');
const [linkID, setLinkID] = useState(data.ID || 0);

const onClear = (event) => {
if (typeof onChange !== 'function') {
return;
}

onChange(event, { id, value: {} });
// deleteUrl is set via LeftAndMainExtension.php
const section = Config.getSection('SilverStripe\\Admin\\LeftAndMain');
const endpoint = `${section.form.DynamicLink.deleteUrl}/${linkID}`;
backend.delete(endpoint)
.then(() => {
actions.toasts.success('Deleted link');
})
.catch(() => {
actions.toasts.error('Failed to delete link');
});

if (typeof onChange === 'function') {
onChange(event, {});
}
};

const { typeKey } = data;
const type = types[typeKey];
const modalType = newTypeKey ? types[newTypeKey] : type;

let title = data ? data.Title : '';

if (!title) {
title = data ? data.TitleRelField : '';
}

const linkProps = {
const pickerProps = {
title,
link: type ? { type, title, description: linkDescription } : undefined,
onEdit: () => { setEditing(true); },
Expand All @@ -41,15 +55,32 @@ const LinkField = ({ id, loading, Loading, data, LinkPicker, onChange, types, li
types: Object.values(types)
};

const onModalSubmit = (modalData, action, submitFn) => {
const { SecurityID, action_insert: actionInsert, ...value } = modalData;
const onModalSubmit = async (modalData, action, submitFn) => {
const { SecurityID, action_submit: actionSubmit, ...data } = modalData;

const formSchema = await submitFn();
console.log('formSchema', formSchema);

let id = 0;

// formSchema will be undefined for FileLink which uses asset-admin graphql when using InsertMediaModal rather than FormBuilderModal
if (typeof formSchema !== 'undefined') {
const match = formSchema.id.match(/\/saveForm\/([0-9]+)$/);
if (match) {
id = parseInt(match[1]);
}
}

if (typeof onChange === 'function') {
onChange(event, { id, value });
data.ID = id;
onChange(null, data);
}

setEditing(false);
setNewTypeKey('');
setLinkID(id);

actions.toasts.success('Saved link');

return Promise.resolve();
};
Expand All @@ -68,7 +99,7 @@ const LinkField = ({ id, loading, Loading, data, LinkPicker, onChange, types, li
const LinkModal = loadComponent(`LinkModal.${handlerName}`);

return <Fragment>
<LinkPicker {...linkProps} />
<LinkPicker {...pickerProps} />
<LinkModal {...modalProps} />
</Fragment>;
};
Expand All @@ -81,10 +112,19 @@ const stringifyData = (Component) => (({ data, value, ...props }) => {
return <Component dataStr={JSON.stringify(dataValue)} {...props} data={dataValue} />;
});

const mapDispatchToProps = (dispatch) => {
return {
actions: {
toasts: bindActionCreators(toastsActions, dispatch),
},
};
}

export default compose(
inject(['LinkPicker', 'Loading']),
injectGraphql('readLinkTypes'),
stringifyData,
injectGraphql('readLinkDescription'),
fieldHolder
fieldHolder,
connect(null, mapDispatchToProps)
)(LinkField);
5 changes: 4 additions & 1 deletion client/src/components/LinkModal/LinkModal.js
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,10 @@ const buildSchemaUrl = (key, data) => {
const parsedQs = qs.parse(parsedURL.query);
parsedQs.key = key;
if (data) {
parsedQs.data = JSON.stringify(data);
for (const prop of ['href', 'path', 'pathname']) {
const id = typeof data.ID !== 'undefined' ? data.ID : '0';
parsedURL[prop] += `/${id}`;
}
}
return url.format({ ...parsedURL, search: qs.stringify(parsedQs)});
}
Expand Down
5 changes: 2 additions & 3 deletions client/src/entwine/JsonField.js
Original file line number Diff line number Diff line change
Expand Up @@ -33,9 +33,9 @@ jQuery.entwine('ss', ($) => {
Root.render(<ReactField {...props} noHolder/>);
},

handleChange(event, {id, value}) {
handleChange(event, data) {
const fieldID = $(this).data('field-id');
$('#' + fieldID).val(JSON.stringify(value)).trigger('change');
$('#' + fieldID).val(JSON.stringify(data)).trigger('change');
this.refresh();
},

Expand All @@ -50,7 +50,6 @@ jQuery.entwine('ss', ($) => {
const value = dataStr ? JSON.parse(dataStr) : undefined;

return {
id: fieldID,
value,
onChange: this.handleChange.bind(this)
};
Expand Down
48 changes: 48 additions & 0 deletions client/src/state/linkOperations/deleteLink.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
/* eslint-disable */
import { graphqlTemplates } from 'lib/Injector';

const apolloConfig = {
props(
props
) {
const {
data: {
error,
deleteLinks,
loading: networkLoading,
},
} = props;
const errors = error && error.graphQLErrors &&
error.graphQLErrors.map((graphQLError) => graphQLError.message);

// const types = readLinkTypes ?
// readLinkTypes.reduce((accumulator, type) => (
// { ...accumulator, [type.key]: type }
// ), {}) :
// {};

return {
loading: networkLoading,
// types,
graphQLErrors: errors,
};
},
};

const { DELETE } = graphqlTemplates;
const query = {
apolloConfig,
templateName: DELETE,
pluralName: 'Links',
pagination: false,
params: {
ids: '[ID]'
},
args: {
root: {
ids: 'ids'
}
},
// fields: ['key', 'title', 'handlerName'],
};
export default query;
Loading

0 comments on commit 7aaa11a

Please sign in to comment.