lochmueller/language-detection

Modern language detection middleware for TYPO3. Based on PSR-7, PSR-14 & PSR-15.

Fund package maintenance!
lochmueller
paypal.me/lochmueller

Installs: 32 941

Dependents: 1

Suggesters: 0

Security: 0

Stars: 12

Watchers: 2

Forks: 5

Open Issues: 0

Type:typo3-cms-extension

4.0.3 2024-01-17 09:19 UTC

README

Meta: start with why Latest Stable Version Total Downloads License Crowdin Average time to resolve an issue Percentage of issues still open

Compatibility: TYPO3 TYPO3 TYPO3

Quality: Test codecov Scrutinizer Code Quality PHPStan

Support: Donate contributions welcome Plant Tree

Table of Contents

  1. Why?
  2. Installation
  3. Configuration
  4. Event Structure
    1. CheckLanguageDetectionEvent
    2. DetectUserLanguagesEvent
    3. NegotiateSiteLanguageEvent
    4. BuildResponseEvent
  5. Troubleshooting
  6. Dev
  7. Contribution
  8. Licence

Why?

Language Detection should be easy & simple to integrate and powerfully in development! TYPO3 Core do not handle language detection via client information. EXT:languag_detection use a PSR-15/PSR-7 middleware/request (TYPO3 Documentation) to handle a language detection logic via PSR-14 events(TYPO3 Documentation). Very flexible! Give it a try and checkout the future of language detection!

Installation

composer require lochmueller/language-detection

Configuration

Use the site configuration module to configure the language detection. Just enable it, and it will work :) There are several configuration options for the Site configuration that handle the control events. The following screenshot show the options of the detection configuration.

Configuration

Event Structure

There are four central PSR-14 events that control the language detection. The attached list explain the different events and the default listener. The events are ordered in the execution order.

Diagram Request flow

CheckLanguageDetectionEvent

Check if the language detection should execute by the extension. You can register listeners for this event and call "disableLanguageDetection" on the event object to disable the language detection.

Default-Listener:

Name Description
BackendUserCheck Check if a backend user call the language detection and disable the redirect (respect "disableRedirectWithBackendSession" config)
BotAgentCheck Check if a bot call the language detection and disable the redirect
EnableCheck Check if the Language Detection is enabled in the current Site
FromCurrentPageCheck Check the referrer and disable the redirect if the user comes from the current site
PathCheck Check if the user call "/" and disable the redirect for other paths (respect "allowAllPaths" configuration)
WorkspacePreviewCheck Check if the page is a workspace preview and disable the redirect

DetectUserLanguagesEvent

This event collect user information to get the user languages. You can register your own detections and manipulate the data via "getUserLanguages" and "setUserLanguages".

Default-Listener:

Name Description
BrowserLanguageDetect Get the users "accept-language" languages
GeoPluginDetect Send the IP to geoplugin.net and add the language of the location to the checked languages (respect "addIpLocationToBrowserLanguage" configuration)
MaxMindDetect Use MaxMind database or webservice to get the country information

Please keep data privacy in mind in case of the "IpLanguage" Listener!

NegotiateSiteLanguageEvent

This event calculates the best matching page language for the user. If you build your own listener. Please use "setSelectedLanguage" on the event. If a language is already selected the default listener will be skipped.

Default-Listener:

Name Description
DefaultNegotiation Check the Locale and TwoLetterIso of the TYPO3 languages against the user languages of the previous event
FallbackNegotiation Handle a fallback, if there are no matches by the default negotiation

BuildResponseEvent

The last event build the middleware response. You can overwrite this step. You have to use "setResponse" to set the response.

Default-Listener:

Name Description
DefaultResponse Build the response object and respect the "redirectHttpStatusCode" config

Troubleshooting

There are missing or wrong languages in the detection process. Why?

Do you check in incognito mode? The browser will not send all languages in incognito mode. So "wrong results" are possible. Please check the request header to TYPO3 in detail. Otherwise, perhaps the DefaultNegotiation do not handle the "best fitting language" selection process for your needs?

Why the redirect not work on subpages?

The middleware is early in the middleware stack. There is no concept of links and translations (or even page UID). Furthermore, it is recommended not redirect on subpages. A user that call a subpage first bookmark the page or search in a search engine. In both cases the user already get the right language. I suggest hreflang tags so search engines get the right language of the content https://developers.google.com/search/docs/advanced/crawling/localized-versions

Dev

Run all code standards

docker run --rm -it --volume $(pwd):/app prooph/composer:8.0 -d /app code:all

Execute tests with PHP 8.0:

docker run --rm -it --volume $(pwd):/app prooph/composer:8.0 -d /app test:unit

With coverage:

docker run --rm -it --volume $(pwd):/app cicnavi/dap:80 /app/.Build/bin/phpunit -c /app/phpunit.xml --coverage-text --testdox --coverage-html=/app/var/phpunit

Run Mutation tests:

docker run --rm -it --workdir=/app/ --volume $(pwd):/app cicnavi/dap:80 /app/.Build/bin/infection -c /app/phpunit.xml

Contribution

Thanks all for the great contribution to the project!

GitHub Contributors Image

Licence

This package is Treeware. If you use it in production, then we ask that you buy the world a tree to thank us for our work. By contributing to the Treeware forest you’ll be creating employment for local families and restoring wildlife habitats.