smartlabsat / sulu-image-crop-content-type
An open-source Sulu bundle providing a custom image crop content type with an integrated React component for intuitive image cropping.
Installs: 5
Dependents: 0
Suggesters: 0
Security: 0
Stars: 0
Watchers: 1
Forks: 0
Open Issues: 0
Language:JavaScript
Type:sulu-bundle
pkg:composer/smartlabsat/sulu-image-crop-content-type
Requires
- php: >=8.2
- sulu/sulu: ^2.6
Requires (Dev)
- friendsofphp/php-cs-fixer: ^3.0
- phpstan/phpstan: ^1.0
- phpstan/phpstan-symfony: ^1.0
- phpunit/phpunit: ^10.0
This package is auto-updated.
Last update: 2025-12-12 16:36:57 UTC
README
An open-source Sulu bundle providing a custom image crop content type with an integrated React component for intuitive image cropping.
🚀 Introduction
This bundle allows you to seamlessly select, crop, and manage images within the Sulu Admin. It augments the default Sulu Media handling by introducing a specialized image_crop field type, powered by a custom React component.
⚠️ Requirements
| Requirement | Version |
|---|---|
| PHP | ^8.2 |
| Sulu | ^2.6 |
| Node | ^20 |
📥 Installation
Require the bundle
composer require smartlabsat/sulu-image-crop-content-type
Create or update your frontend build
If you haven't created an admin folder yet:
mkdir -p assets/admin
Download Sulu's Admin build
bin/console sulu:admin:download-build
cd assets/admin
npm install
Link the local JS bundle from the vendor folder
npm install file:../../vendor/smartlabsat/sulu-image-crop-content-type/Resources/js
Import the bundle in /assets/admin/app.js
// /assets/admin/app.js // Add project specific javascript code and import of additional bundles here: import 'sulu-smartlabsat-image-crop-bundle';
Build
Build your admin assets
npm run build
🎨 Usage
Once everything is installed, you can use the image_crop type in your Sulu templates or pages:
<property name="composite" type="image_crop"> <params> <param name="previewCrop" value="300x"/> </params> </property>
Note: The previewCrop parameter controls the thumbnail size that appears in the admin interface (e.g., 300x). Example Template Configuration
config/templates/pages/imagecroppage.xml
<?xml version="1.0"?> <template xmlns="http://schemas.sulu.io/template/template" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://schemas.sulu.io/template/template http://schemas.sulu.io/template/template-1.0.xsd"> <key>image_crop_page</key> <view>pages/image_crop_page</view> <controller>Sulu\Bundle\WebsiteBundle\Controller\DefaultController::indexAction</controller> <meta> <title lang="en">Image Crop Page</title> </meta> <properties> <property name="title" type="text_line" colspan="12"> <meta> <title lang="en">Title</title> </meta> </property> <property name="url" type="resource_locator" colspan="12"> <tag name="sulu.rlp"/> <meta> <title lang="en">URL</title> </meta> </property> <property name="composite" type="image_crop"> <params> <param name="previewCrop" value="300x"/> </params> </property> </properties> </template>
templates/pages/image_crop_page.html.twig
{% block content %}
<!DOCTYPE html>
<html lang="de">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>Image Cropper Demo</title>
<!-- Bulma CSS -->
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bulma@0.9.4/css/bulma.min.css">
<!-- Font Awesome for Icons -->
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css"
integrity="sha512-p0p+65dAxpkrv+7c6Wbb5e9s6RO7XlOQhvfA4CTp6GAGcJZUT1kn2SZig0wYxumCAtdIsP+S3f+q1D1r5u0xig=="
crossorigin="anonymous" referrerpolicy="no-referrer"/>
<style>
.image-container {
display: flex;
justify-content: center;
}
.component-title {
display: flex;
align-items: center;
gap: 0.5rem;
}
.image-container img {
max-width: 100%;
height: auto;
}
</style>
</head>
<body>
<section class="section">
<div class="container">
<div class="level">
<div class="level-left">
<div class="level-item">
<h1 class="title component-title">
<span class="icon is-large">
<i class="fas fa-crop fa-2x"></i>
</span>
<span>{{ content.title }}</span>
</h1>
</div>
</div>
</div>
{% set composite = content.composite|default({}) %}
{% set media = composite.media|default(null) %}
{% if media %}
<div class="box">
<figure class="image image-container">
{% set cropKey = '300x' %}
<img src="{{ media.thumbnails[cropKey]|default('') }}" alt="Image Cropper Demo"/>
</figure>
</div>
{% endif %}
</div>
</section>
<script src="https://cdn.jsdelivr.net/npm/@material-ui/styles@4.11.5/index.min.js"></script>
</body>
</html>
{% endblock %}
If you haven’t defined any image crops in your project yet, please refer to Sulu Docs on Image Formats.
⚛️ React Component Overview
Below is a simplified overview of the React component that powers the cropping interface: • ImageCropContentType class • Utilizes Sulu’s SingleSelectionStore to manage the selected media • Integrates Sulu’s CropOverlay for the cropping functionality • Offers convenient Select, Crop, Edit, Refresh, and Remove actions • Automatically handles reloading the image to ensure the newest cropped version is displayed
âť“ Troubleshooting
Error: There is no field with key “image_crop” registered… If you see this error in the Sulu Admin, it usually means your Admin frontend build wasn’t run correctly. Please re-check the steps:
- Execute npm install (and the npm install file:../../vendor/...) in the assets/admin folder.
- Make sure you import 'sulu-smartlabsat-image-crop-bundle' in your app.js.
- Finally, run npm run build.
đź“„ License
Licensed under the MIT license. Feel free to use, modify, and distribute this software according to the terms of the license.
Happy Cropping! ✂️
If you have any questions, feel free to open an issue or pull request!
