craft\events\SiteEvent
now has a$oldPrimarySiteId
property, which will be set to the previous primary site ID (which may stil be the current site ID, if it didn’t just change).craft\helpers\Search::normalizeKeywords()
now has a$langage
argument, which can be set if the character mappings should be pulled from a different language than the current app language.craft\services\Sites::getEditableSiteIds()
andgetEditableSites()
now return the same things asgetAllSiteIds()
andgetAllSites()
when there’s only one site. (#3049)
- Fixed a bug where user verification links could get mangled when emails were parsed as Markdown, if the verification code contained two or more underscores.
- Fixed a bug where Craft was misinterpreting
X-Forwarded-For
headers as the user’s IP instead of the server’s IP. (#3036) - Fixed a bug where Craft wasn’t auto-scrolling the content container when dragging items near a window edge. (#3048)
- Fixed a PHP error that occurred when loading a Debug Toolbar panel on a page that contained serialized Checkboxes or Multi-Select field data. (#3034)
- Fixed a bug where elements’ normalized search keywords weren’t always using the correct language-specific character mappings. (#3046)
- Craft no longer shows the installer when it can’t establish a database connection if Dev Mode isn’t enabled.
- Fixed an error that occurred when deleting users from the Users index page.
- Fixed a bug where Delete User modals weren’t showing the total number of entries that will be transferred/deleted.
- Craft now includes a summary of the content that will be transferred/deleted in Delete User modals. (#875)
|date
,|time
, and|datetime
filters now support alocale
argument, for specifying which locale’s formatter should be doing the date/time formatting. (#3006)- Added
craft\base\ApplicationTrait::getIsInitialized()
. - Added
craft\base\ClonefixTrait
. - Added
craft\controllers\AssetsController::actionThumb()
. - Added
craft\controllers\UsersController::actionUserContentSummary()
. - Added
craft\controllers\UsersController::EVENT_DEFINE_CONTENT_SUMMARY
. - Added
craft\helpers\App::backtrace()
. - Added
craft\queue\jobs\PropagateElements
. - Added
craft\services\Elements::propagateElement()
.
- Editable tables now submit an empty string when they have no rows.
- Reduced the overhead when adding a new site by only resaving existing assets, categories, global sets, and tags once for the newly-created site, rather than for all sites.
- Web-based queue workers now call
craft\helpers\App::maxPowerCaptain()
before running the queue. (#3011) - The PHP Info utility no longer displays the original values for settings and only the current environment value. (#2990)
- Loosened up most of Craft’s Composer dependency constraints.
- Craft no longer publishes asset thumbnails to the
cpresources/
folder. attributes
,error
,errors
, andscenario
are now reserved field handles. (#3032)- Improved the look of Control Panel tabs.
craft\web\UrlManager::createUrl()
,createAbsoluteUrl()
, andgetMatchedElement()
now log warnings if they’re called before Craft has been fully initialized. (#3028)
- Deprecated
craft\controllers\AssetsController::actionGenerateThumb()
.
- Fixed a bug where sidebar meta info on Edit User pages was bleeding over the edge of the page’s content area.
- Fixed a bug where Table fields wouldn’t remember if they had no rows in their Default Values setting. (#2979)
- Fixed a bug where passing
timezone=false
to the|date
,|time
, and|datetime
filters would not preserve the given date’s time zone. - Fixed a bug where AM/PM strings in formatted dates weren’t respecting the casing specified by the
A
/a
character in the date format. (#3007) - Fixed a bug you could get an invalid license warning in cases where web API calls returned a 500 response code.
- Fixed a bug where cloning models and queries would lose any associated behaviors. (#2857)
- Fixed a bug where custom field params were getting forgotten when calling
getNext()
andgetPrev()
, if an element query object was passed in. (#3019) - Fixed a bug where datepickers were getting scrollbars.
- Fixed a bug where volumes’ field layouts weren’t getting deleted when volumes were deleted. (#3022)
- Fixed a bug where deleting a section or an entry type wouldn’t delete any associated entries that didn’t exist in the primary site. (#3023)
- Fixed a bug where the
svg()
Twig function could convertid
attributes within the SVG contents to invalid IDs. (#3025) - Fixed a bug where asset thumbnails wouldn’t load reliably in the Control Panel on load-balanced environments. (#3026)
- Fixed a PHP error that could occur when validating Assets fields if a file was uploaded but no longer exists at the temp location. (#3033)
- Added a
leaves
element query param that limits the selected elements to just the leaves in the structure (elements without children). - Added
craft\helpers\Db::deleteIfExists()
. - Added
craft\services\Categories::deleteGroup()
. (#3000) - Added
craft\services\Tags::deleteTagGroup()
. (#3000) - Added
craft\services\UserGroups::deleteGroup()
. (#3000)
- Improved Control Panel styling. (#2883)
- Removed
craft\services\Fields::updateFieldVersionAfterRequest()
.
- Fixed a caching bug where the Fields service could still think a field existed after it had been deleted. (#2985)
- Fixed a bug where Craft would not invalidate the dynamically-generated
craft\behaviors\ContentBehavior
andcraft\behaviors\ElementQueryBehavior
after saving/deleting a custom field, if the request didn’t end normally. (#2999) - Fixed a PHP error that could occur when saving entries with a URI format that contained certain Twig filters. (#2995)
- Fixed a bug where
{shorthand}
variables in templates rendered bycraft\web\View::renderObjectTemplate()
could end up referencing global variables, if the variable wasn’t a property of the object. (#3002) - Fixed a bug where the Find and Replace utility wasn’t updating element titles. (#2996)
- Fixed some wonky behavior if one of the custom user profile tabs was called “Account”. (#2998)
- Fixed a bug where dragging a folder on the Assets index page could have unexpected results. (#2873)
- Reduced the likelihood of SQL deadlock errors when saving elements. (#3003)
- Sort options defined by element types’
sortOptions()
/defineSortOptions()
methods can now be specified as sub-arrays withlabel
,orderBy
, andattribute
keys. - Entries and categories can now be sorted by their slugs.
- The “Cache remote images?” option in the Asset Indexes utility is now enabled by default. (#2977)
- Fixed a bug where it was not possible to order search results by search score, if the element type didn’t specify any sort options.
- Fixed a bug where clicking on “Date Created” and “Date Updated” column headers on element indexes wouldn’t update the sort order. (#2975)
- Fixed a bug where Edit Entry pages were listing more than the 10 most recent versions. (#2976)
- Fixed a SQL error that occurred when upgrading from Craft 2 to 3 via the terminal. (#1347)
- Fixed the alignment of expand/collapse toggles in asset index sidebars. (#2981)
- Fixed a bug where the “New Entry” menu on the Entries index page would not contain any options on single-site installs, running MySQL. (#2961)
- Fixed a bug where the
siteName
config setting wasn’t working as expected when set to an array. (#2968)
- Improved the output of
craft\helpers\DateTimeHelper::humanDurationFromInterval()
. - Updated Garnish to 0.1.24.
- Fixed JavaScript errors that could occur in the Control Panel on pages with Ajax requests. (#2966)
- Fixed a bug where the “New Entry” menu on the Entries index page would not contain any options on single-site installs. (#2961)
- Fixed a bug where JavaScript files registered with
craft\web\View::registerJsFile()
would be ignored if thedepends
option was set. (#2965)
- Fixed a bug where Craft wasn’t converting empty strings to
null
when saving data to non-textual columns. - Fixed a bug where Craft would show a Database Connection Error on Install requests, if it couldn’t connect to the database.
- Fixed a bug where Craft wasn’t keeping track of element queries that were executed within
{% cache %}
tags. (#2959)
- Added support for a
CRAFT_LICENSE_KEY
PHP constant, which can be set to the project’s license key, taking precedence over thelicense.key
file. - Added
craft\helpers\Stringy::getLangSpecificCharsArray()
. - Added
craft\web\View::setRegisteredAssetBundles()
. - Added
craft\web\View::setRegisteredJsFiles()
.
- Generated site URLs now always include full host info, even if the base site URL is root/protocol-relative. (#2919)
- Variables passed into
craft\web\View::renderObjectTemplate()
can now be referenced using the shorthand syntax (e.g.{foo}
). craft\helpers\StringHelper::asciiCharMap()
now has$flat
and$language
arguments.- Craft no longer saves new versions of entries when absolutely nothing changed about them in the save request. (#2923)
- Craft no longer enforces plugins’
minVersionRequired
settings if the currently-installed version begins with ` - dev-`.
- Improved the performance of element queries when a lot of values were passed into a param, such as
id
, by usingIN()
andNOT IN()
conditions when possible. (#2937) - The Asset Indexes utility no longer skips files with leading underscores. (#2943)
- Updated Garnish to 0.1.23.
- Deprecated the
customAsciiCharMappings
config setting. (Any corrections to ASCII char mappings should be submitted to Stringy.)
- Fixed a PHP error that could occur when
craft\fields\Number::normalizeValue()
was called without passing an$element
argument. (#2913) - Fixed a bug where it was not possible to fetch Matrix blocks with the
relatedTo
param if a specific custom field was specified. - Fixed a bug where
craft\helpers\UrlHelper::url()
andsiteUrl()
were not respecting the$scheme
argument for site URLs. - Fixed a bug where
{id}
tags within element URI formats weren’t getting parsed correctly on first save. (#2922) - Fixed a bug where
craft\helpers\MigrationHelper::dropAllForeignKeysToTable()
wasn’t working correctly. (#2897) - Fixed a “Craft is not defined” JavaScript error that could occur on the Forgot Password page in the Control Panel and Dev Toolbar requests.
- Fixed a bug where rotating the screen on iOS would change how the page was zoomed.
- Fixed a bug where
craft\helpers\StringHelper::toAscii()
and theCraft.asciiString()
JS method weren’t using language-specific character replacements, or any custom replacements defined by thecustomAsciiCharMappings
config setting. - Fixed a bug where the number
0
would not save in a Plain Text field. - Fixed a bug where Craft could pick the wrong current site if the primary site had a root-relative or protocol-relative URL, and another site didn’t, but was otherwise an equal match.
- Fixed a bug where Control Panel Ajax requests could cause some asset bundles and JavaScript files to be double-registered in the browser.
- Fixed a bug where the “New entry” menu on the Entries index page was including sections that weren’t available in the selected site, and they weren’t linking to Edit Entry pages for the selected site. (#2925)
- Fixed a bug where the
|date
,|time
, and|datetime
filters weren’t respecting their$timezone
arguments. (#2926) - Fixed a bug where element queries weren’t respecting the
asArray
param when callingone()
. (#2940) - Fixed a bug where the Asset Indexes utility wouldn’t work as expected if all of a volume’s assets had been deleted from the file system. (#2955)
- Fixed a SQL error that could occur when a
{% cache %}
tag had no body. (#2953)
- Added a default plugin icon to plugins without an icon in the Plugin Store.
- Added
craft\helpers\ArrayHelper::without()
andwithoutValue()
. - Added
craft\base\FieldInterface::modifyElementIndexQuery()
. - Added
craft\elements\db\ElementQueryInterface::andWith()
.
- Fixed a bug where Craft was checking the file system when determining if an asset was a GIF, when it should have just been checking the file extension.
craft\base\Plugin
now sets the default$controllerNamespace
value to the plugin class’ namespace +\controllers
or\console\controllers
, depending on whether it’s a web or console request.- Improved the contrast of success and error notices in the Control Panel to meet WCAG AA requirements. (#2885)
fieldValue
is now a protected field handle. (#2893)- Craft will no longer discard any preloaded elements when setting the
with
param on an element query, fixing a bug where disabled Matrix blocks could show up in Live Preview if any nested fields were getting eager-loaded. (#1576) - Improved memory usage when using the
{% cache %}
tag. (#2903)
- Fixed a bug where the Plugin Store was listing featured plugins (e.g. “Recently Added”) in alphabetical order rather than the API-defined order. (pixelandtonic/craftnet#83)
- Fixed a SQL error that occurred when programmatically saving a field layout, if the field’s
required
property wasn’t set. - Fixed a JavaScript error that could occur when multiple Assets fields were present on the same page.
- Fixed an error that could occur when running the
setup
command on some environments. - Fixed a PHP error that could occur when calling
craft\elements\db\ElementQuery::addOrderBy()
if$columns
normalized to an empty array. (#2896) - Fixed a bug where it wasn’t possible to access custom field values on Matrix blocks via
matrixblock
reference tags. - Fixed a bug where relational fields with only disabled elements selected would get empty table cells on element indexes. (#2910)
- Number fields now have a “Default Value” setting. (#927)
- Added the
preserveCmykColorspace
config setting, which can be set totrue
to prevent images’ color spaces from getting converted to sRGB on environments running ImageMagick.
- Error text is now orange instead of red. (#2885)
- Detail panes now have a lighter, more saturated background color.
- Fixed a bug where Craft’s default MySQL backup command would not respect the
unixSocket
database config setting. (#2794) - Fixed a bug where some SVG files were not recognized as SVG files.
- Fixed a bug where Table fields could add the wrong number of default rows if the Min Rows setting was set, and the Default Values setting had something other than one row. (#2864)
- Fixed an error that could occur when parsing asset reference tags. (craftcms/redactor#47)
- Fixed a bug where “Try” and “Buy” buttons in the Plugin Store were visible when the
allowUpdates
config setting was disabled. (#2781) - Fixed a bug where Number fields would forget their Min/Max Value settings if they were set to 0.
- Fixed a bug where entry versions could be displayed in the wrong order if multiple versions had the same creation date. (#2889)
- Fixed an error that occurred when installing Craft on a domain with an active user session.
- Fixed a bug where email verification links weren’t working for publicly-registered users if the registration form contained a Password field and the default user group granted permission to access the Control Panel.
- Login errors for locked users now factor in whether the
preventUserEnumeration
config setting is enabled.
- Added the
transformGifs
config setting, which can be set tofalse
to prevent GIFs from getting transformed or cleansed. (#2845) - Added
craft\helpers\FileHelper::isGif()
.
- Craft no longer logs warnings about missing translation files when Dev Mode isn’t enabled. (#1531)
- Added
craft\services\Deprecator::$logTarget
. (#2870) craft\services\Deprecator::log()
no longer returns anything.
- Fixed a bug where it was impossible to upload new assets to Assets fields using base64-encoded strings. (#2855)
- Fixed a bug where Assets fields would ignore all submitted asset IDs if any new assets were uploaded as well.
- Fixed a bug where SVG files that were using single quotes instead of double quotes would not be recognized as SVGs.
- Fixed a bug where translated versions of the “It looks like someone is currently performing a system update.” message contained an HTML-encoded
<br/>
tag. - Fixed a bug where changing an entry’s type could skip adding the new entry type’s tabs, if the previous entry type didn’t have any tabs. (#2859)
- Fixed warnings about missing SVG files that were logged by Control Panel requests.
- Fixed a bug where the
|date
filter would ignore date formatting characters that don’t have ICU counterparts. (#2867) - Fixed a bug where the global
currentUser
Twig variable could be set tonull
and global sets and could be missing some custom field values when a user was logged-in, if a plugin was loading Twig during or immediately after plugin instantiation. (#2866)
- Error messages about missing plugin-supplied field and volume types now show an Install button when possible.
- Added
craft\base\MissingComponentTrait::getPlaceholderHtml()
. - Added
craft\db\Migration::EVENT_AFTER_UP
andEVENT_AFTER_DOWN
events. - Added
craft\elements\Asset::getContents()
.
- Edit User pages will now warn editors when leaving the page with unsaved changes. (#2832)
- Modules are once again loaded before plugins, so they have a chance to register Twig initialization events before a plugin initializes Twig. (#2831)
craft\helpers\FileHelper::isSvg()
now returnstrue
for files with animage/svg
MIME type (missing the+xml
). (#2837)- The
svg()
Twig function now accepts assets to be passed directly into it. (#2838) - The “Save and add another” save menu option on Edit Entry and Edit Categories pages now maintain the currently-selected site. (#2844)
- PHP date patterns that are only a month name or week day name character will now format the date using the stand-alone month/week day name value. (For example,
'F'
will format a date as “Maggio” instead of “maggio”.) - Servers without the Intl extension will now use location-agnostic locale data as a fallback if locale data for the specific locale isn’t available.
- The
|date
Twig filter always goes throughcraft\i18n\Formatter::asDate()
now, unless formatting aDateInterval
object. - The Settings → Plugins page now shows “Buy now” buttons for any commercial plugins that don’t have a license key yet.
- Deprecated
craft\helpers\DateTimeHelper::translateDate()
.craft\i18n\Formatter::asDate()
should be used instead.
- Removed the
translate
argument from the|date
,|time
, and|datetime
Twig filters; the resulting formatted dates will always be translated now. (UsemyDate.format()
to avoid translations.)
- Fixed an error that could occur in the Plugin Store.
- Fixed a bug where
myDate|date('F')
was returning the short “May” translation rather than the full-length one. (#2848)
- Fields’ translation icons now reveal the chosen Translation Method in their tooltip. (#2808)
- Improved the error messages displayed when an Assets field has an invalid Upload Location setting. (#2803)
- Craft now logs errors that occur when saving and replacing assets. (#2814)
- Single sections’ entry types’ handles are now updated to match their section’s handle whenever the section is saved. (#2824)
- The Control Panel background color was lightened up a bit.
- Fixed an error that would occur on servers without the Phar PHP extension enabled.
- Fixed an error that could occur if a Matrix block was deleted by a queue job. (#2813)
- Fixed a bug where Twig could be configured to output times in UTC rather than the system timezone, if a bootstrapped module was loading Twig. (#2761)
- Fixed a SQL error that could occur when upgrading from Craft 2 to Craft 3 with an active user session.
- Fixed various SQL errors that could occur when upgrading from Craft 2 to Craft 3, if there were any lingering Craft 3 database tables from a previous upgrade attempt.
- Fixed a bug where the Clear Caches tool was deleting the
.gitignore
file insideweb/cpresources/
. (#2823) - Fixed the vertical positioning of checkboxes in the Control Panel. (#2825)
- Fixed a JavaScript error that could occur if an element type’s class name contained
\u
. (#2826)
- Added the
craft.globalSets()
template function. (#2790) - Added the
hasDescendants
element query param. (#2786) - Added
craft\elements\User::hasDashboard
.
- Sections and category groups now ignore posted Template settings for sites that don’t have URI Formats.
- Control Panel resources are once again eager-published. (#2763)
entries/save-entries
andcategories/save-category
actions now include theslug
for responses that accept JSON. (#2792)- Most
craft\services\Path
methods now have a$create
argument, which can be set tofalse
to prevent the directory from being created if it doesn’t exist yet. - Craft no longer creates directories when it just needed to clear it. (#2771)
craft\services\Config::setDotEnvVar()
now sets the environment variable for the current request, in addition to updating the.env
file.- Removed
craft\controllers\AssetsController::actionDownloadTempAsset()
. - User now must be logged in to use the Asset Preview File functionality.
- Fixed a bug where users would regain all default Dashboard widgets if all widgets were removed. (#2769)
- Fixed a bug where you would get a “not a valid language” error message when creating a new site using certain languages.
- Fixed a bug where database connection settings that were set by the
setup
command weren’t always taking effect in time for the CLI installer. (#2774) - Fixed a bug where empty Plain Text fields were getting empty string values rather than
null
. - Fixed a bug where elements within relational fields could have two thumbnails. (#2785)
- Fixed a bug where it was not possible to pass a
--table-prefix
argument to thesetup/db-creds
command. (#2791) - Fixed an error that occurred for users without permission to perform updates, if available update info wasn’t cached.
- Fixed an error that occurred when
craft\elements\Asset::sources()
was called in a console request. (#2798) - Fixed JavaScript errors that could occur on the front-end after deleting Matrix blocks. (#2799)
- Fixed an error that occurred when editing an entry if any of the entry’s revisions were created with an entry type that no longer exists.
- Fixed an error that could occur when saving an asset. (#2764)
- Fixed a bug where Craft assumed an asset was missing if there was an error when indexing it. (#2763)
- Added
craft\elements\Entry::updateTitle()
. - Added
Yii::alias()
.
- New sites’ Base URLs now default to
@web/
. - Textual custom fields now ensure that they don’t contain 4+ byte characters. (#2725)
- It is no longer expected that all of the
defaultSearchTermOptions
config setting options will be set if any of the default option values need to be overridden. (#2737) - Control Panel panes now have at least 48 pixels of bottom padding. (#2744)
- Craft now intercepts 404-ing resource requests, and publishes the resources on the fly.
- The Clear Caches utility now has a “Control Panel resources” option.
- The Clear Caches utility now sorts the cache options alphabetically.
- When enabling new sites for a section, the new sites’ content is now based on the primary site’s content, if the section was and still is enabled for the primary site. (#2748)
- Improved the responsiveness of element indexes.
Craft.BaseElementIndexView
now has aloadMoreElementsAction
setting. (#2762)
- Fixed a bug where the Clear Caches utility was not deleting template caches. (#2720)
- Fixed a bug where the Plugin Store was not displaying payment errors on checkout.
- Fixed a bug where Control Panel-defined routes that contained special regular expression characters weren’t working. (#2721)
- Fixed a bug where it was not possible to save system messages in some cases.
- Fixed a bug where static translations within dynamic entry title formats were getting translated using the current site’s language, rather than the entry’s language. (#2722)
- Fixed a bug where deprecation errors for some date formatting methods were not escaping backslashes.
- Fixed a bug where plugins’ “Last update” timestamps in the Plugin Store weren’t getting formatted correctly in Safari. (#2733)
- Fixed references to a nonexistent
Craft.eot
file in the Control Panel CSS. (#2740) - Fixed a bug where the default PostgreSQL database restore command wasn’t setting the
PGPASSWORD
environment variable. (#2741) - Fixed an error that could occur if the system time zone was not supported by the ICU library, on environments with the Intl extension loaded.
- Fixed a bug where several administrative fields had translatable icons. (#2742)
- Fixed a bug where
craft\controllers\PluginStoreController::actionSavePluginLicenseKeys()
was trying to set a plugin license key for plugins which were not installed.
- Fixed a bug assets were not getting cleansed on upload. (#2709)
- Added the
EVENT_BEFORE_DELETE_CACHES
andEVENT_AFTER_DELETE_CACHES
events tocraft\services\TemplateCaches
. - Added
craft\events\DeleteTemplateCachesEvent
.
- Craft now deletes all compiled templates whenever Craft or a plugin is updated. (#2686)
- The Plugin Store now displays commercial plugins’ renewal prices. (#2690)
- The Plugin Store no longer shows the “Upgrade Craft CMS” link if Craft is already running (and licensed to run) the Pro edition. (#2713)
- Matrix fields now set
$propagating
totrue
when saving Matrix blocks, if the owner element is propagating. craft\helpers\ArrayHelper::toArray()
no longer throws a deprecation error when a string without commas is passed to it. (#2711)- Editable tables now support an
html
column type, which will output cell values directly without encoding HTML entities. (#2716) Craft.EditableTable
instances are now accessible via.data('editable-table')
on their<table>
element. (#2694)- Updated Composer to 1.6.3. (#2707)
- Updated Garnish to 0.1.22. (#2689)
- Fixed an error that could occur in the Control Panel if any plugins with licensing issues were installed. (#2691)
- Fixed a bug on the Plugin Store’s Payment screen where the “Use a new credit card” radio option would not get selected automatically even if it was the only one available.
- Fixed a bug where
craft\web\assets\vue\VueAsset
didn’t respect theuseCompressedJs
config setting. - Fixed an error that occurred when saving a Single entry over Ajax. (#2687)
- Fixed an error that could occur when disabling a site on a Single section. (#2695)
- Fixed an error that could occur on requests without a content type on the response. (#2704)
- Fixed a bug where the
includeSubfolders
asset query param wasn’t including results in the parent folder. (#2706) - Fixed an error that could occur when querying for users eager-loaded with their photos, if any of the resulting users didn’t have a photo. (#2708)
- Fixed a bug where relational fields within Matrix fields wouldn’t save relations to elements that didn’t exist on all of the sites the owner element existed on. (#2683)
- Fixed a bug where relational fields were ignoring disabled related elements in various functions, including required field validation and value serialization.
- Fixed an error that would occur if a new custom field was created and added to an element’s field layout, and its value was accessed, all in the same request. (#2705)
- Fixed a bug where the
id
param was ignored when used on an eager-loaded elements’ criteria. (#2717) - Fixed a bug where the default restore command for MySQL wouldn’t actually restore the database. (#2714)
- Brought back and deprecated the
Craft::Personal
andCraft::Client
constants.
- Fixed a bug where elements’
getNext()
andgetPrev()
methods were modifying the element query passed into them. (#2160) - Fixed a bug where Table fields could be pre-populated with one too many rows. (#2680)
- Craft no longer sends exception messages to error templates, unless the exception is an instance of
yii\base\UserException
.
- Fixed a bug where Craft Pro installs were getting identified as Craft Solo in the Control Panel.
- The codebase has been completely rewritten and refactored to improve performance, maintainability, and extensibility.
- Craft can now be installed via Composer in addition to a zip file. (#895)
- Craft’s setup wizard is now available as a CLI tool in addition to the web-based one.
- Plugins are now loaded as Composer dependencies, and implemented as extensions of Yii modules.
- Added multi-site support.
- Added the Plugin Store, where plugins can be discovered, trialled, and purchased. (#808)
- Plugins can now be updated and removed from within the Control Panel.
- Asset sources are now called “volumes”, and plugins can supply their own volume types.
- Added the Image Editor, which can be used to rotate, crop, and flip images, as well as set focal points on them.
- Added asset previews, which can be triggered via a “Preview file” action on the Assets index, or with a
shift
+spacebar
keyboard shortcut throughout the Control Panel. - Asset editor HUDs now show image previews. (#837)
- Added the “Utilities” section to the Control Panel, replacing the Tools area of the Settings page.
- Added the Debug Toolbar, powered by the Debug Extension for Yii 2.
- Added support for Content Migrations.
- Added support for PostgreSQL.
- The Control Panel has been redesigned for better usability, readability and responsiveness.
- Renamed all “URL Format” things to “URI Format”, in the Control Panel UI and in the code.
- Added the “Propagate entries across all enabled sites?” section setting. If disabled, entries will only be associated with the site they were created on. (#2330)
- Structure sections and category groups no longer have Nested URL Format settings. (It’s still possible to achieve the same result with a single URI Format setting.)
- When an entry type is updated, Craft now re-saves all entries of that type.
- When a category is deleted, its nested categories are no longer deleted with it.
- Craft no longer re-saves all localizable elements after a new site is created; entries and Matrix blocks are skipped, and plugins that supply custom element types must now re-save their elements manually as well.
- The “New entry” and “New category” buttons on Entries and Categories index pages now load the Edit page for the currently-selected site. (#2236)
- Elements now validate that custom field values will fit within their database columns, for fields with textual or numeric column types.
- User photos are now assets. (#933)
- Assets now have a “Link” table attribute option.
- Volumes’ “Base URL” settings can now begin with
@web
, which is an alias for the root URL that Craft is running from. - Local volumes’ “File System Path” settings can now begin with
@webroot
, which is an alias for the path to the directory thatindex.php
lives in. - Global Sets’ field layouts can now have custom tabs.
- Color inputs can now be left blank.
- Color values within Table fields are now represented by
craft\fields\data\ColorData
objects. - Element titles now get a validation error if they contain any 4+ byte characters (like emoji), on servers running MySQL. (#2513)
- Lightswitch fields that don’t have a value yet will now be assigned the default field value, even for existing elements. (#2404)
- The system installer now sets the initial admin account’s preferred language to the site language selected in the installation wizard. (#2480)
- Table fields now have “Min Rows”, “Max Rows”, and “Add Row Label” settings. (#2372)
- Table fields now have “Date”, “Time”, “Lightswitch”, and “Color” column type options.
- Color fields now return a
craft\fields\data\ColorData
object, withhex
,rgb
,red
,green
,blue
,r
,g
,b
, andluma
properties. - Matrix fields now have “Manage blocks on a per-site basis”, “Min Blocks”, and “Max Blocks” settings.
- Matrix fields with only one block type, and equal values for the Min Blocks and Max Blocks settings, now hide the UI for adding and deleting blocks.
- Matrix fields with only one block type will now auto-create the minimum number of blocks required by the field, per the Min Blocks setting, for new elements. (#850)
- The
migrate/up
console command will now update the appropriate schema version in the database after successfully completing all migrations. (#1907) - Users can now set their preferred language to any supported application language. (#847)
- Users are no longer logged out when verifying a new email address on their own account. (#1421)
- Users no longer get an exception or error message if they click on an invalid/expired email verification link and are already logged in. Instead they’ll be redirected to wherever they would normally be taken immediately after logging in. (#1422)
- If anything prevents a user from being deleted, any changes that were made in preparation for deleting the user are now rolled back.
- Added
webp
as a web-safe image format. - Craft now checks if the current installation can manipulate an image instead of checking against a predefined list. (#1648, #1545)
- The
getCsrfInput()
global function has been renamed tocsrfInput()
. (getCsrfInput() still works but produces a deprecation error.) - The
{% cache %}
tag no longer includes the query string when storing the cache URL. - Added the
|timestamp
Twig filter, for formatting a date as a user-friendly timestamp. - Added the
|datetime
Twig filter, for formatting a date with a localized date+time format. - Added the
|time
Twig filter, for formatting a date with a localized time format. - Added the
|multisort
Twig filter, which duplicates an array and sorts it with craft\helpers\ArrayHelper::multisort(). - Added the
|atom
and|rss
Twig filters, for formatting dates in Atom and RSS date formats, respectively. - Added the
|column
Twig filter, for capturing the key/property values of a series of arrays/objects. - Added the
|index
Twig filter, for indexing an array of arrays/objects by one of their keys/values. - Added the
|filterByValue
Twig filter. - Added the
|duration
Twig filter, which converts aDateInterval
object into a human-readable duration. - The
t
filter now always defaults to translating the given string using thesite
category unless it is otherwise specified (e.g.myString|t('pluginhandle')
). - The
|date
filter can be passed'short'
,'medium'
,'long'
, and'full'
, which will format the date with a localized date format. - It is now possibly to customize the SQL of element queries, and there are more choices on how the data should be returned.
- Element queries are no longer limited to 100 results by default.
- The “Failed” message in the queue HUD in the Control Panel now shows the full error message as alt text. (#855)
- Added the
convertFilenamesToAscii
config setting. - Added the
preserveExifData
config setting,false
by default and requires Imagick. (#2034) - Added the
aliases
config setting, providing an easy way to define custom aliases. - Removed support for automatically determining the values for the
omitScriptNameInUrls
andusePathInfo
config settings. - It’s now possible to override Craft’s application config via
config/app.php
. - It’s now possible to override volume settings via
config/volumes.php
. - It’s now possible to override all plugins’ settings via
config/<plugin-handle>.php
. - Renamed the
runTasksAutomatically
config setting torunQueueAutomatically
. - The
translationDebugOutput
config setting will now wrap strings with@
characters if the category isapp
,$
if the category issite
, and%
for anything else. - All user-defined strings in the Control Panel (e.g. section names) are now translated using the
site
category, to prevent translation conflicts with Craft’s own Control Panel translations. - Routes can now be stored on a per-site basis, rather than per-locale.
- Web requests are now logged to
storage/logs/web.log
. - Web requests that result in 404 errors are now logged to
storage/logs/web-404s.log
. - Console requests are now logged to
storage/logs/console.log
. - Queue requests are now logged to
storage/logs/queue.log
. - Craft 3 now requires PHP 7.0.0 or later.
- Craft 3 now requires MySQL 5.5+ or PostgreSQL 9.5+.
- Craft now takes advantage of the PHP Intl extension when available.
- Craft now uses Stringy for better string processing support.
- Craft now uses Flysystem for better asset volume support.
- Craft now uses Swiftmailer for better email sending support.
- Craft now uses the Yii 2 Queue Extension for managing background tasks.
- Craft now uses the Zend Feed library for better RSS and Atom processing support.
- Updated Yii to 2.0.15.1.
- Updated Twig to 2.4.
- Updated Guzzle to 6.3.
- Many things have been deprecated. See Changes in Craft 3 for a complete list.
- Fixed a bug where a PHP session would be started on every template rendering request whether it was needed or not. (#1765)
- Craft uses OpenSSL for encryption rather than mcrypt, which is far more secure and well-maintained.