5 / 5
Jan 2020

Dear developers,

some time ago we implemented a custom widget and added it to our fork:

https://github.com/asig2016/egroupware/blob/19.1_p1/api/js/etemplate/et2_widget_currency.js8
https://github.com/asig2016/egroupware/blob/19.1_p1/api/src/Etemplate/Widget/Currency.php5

The widget is working just fine from there. But in order to have to minimize the differences between our fork and the upstream branch I tried to move the widget to a custom api application.

For this I used the hook etemplate2_register_widgets as used in the import_export application:

$setup_info[‘importexport’][‘hooks’][‘etemplate2_register_widgets’] = ‘importexport_admin_prefs_sidebox_hooks::widgets’;

I created I class file and added

Etemplate\Widget::registerWidget(‘achelper_widget_currency’, array(‘currency’));

at the end.

Since I din’t find a way to include the js file to be recognized before loading of the form I included it in the

*egw:uses …

section of etemplate2.js

in the

/api/js/etemplate/

folder.

The widget appears correctly in the templates I use it, but when I submit the form it does not transfer the value to the php $content array.

Is the way I moved the widget correct, what can I do to fix this issue?

Thanks a lot in advance and

Best regards

Alex

  • created

    Jan '20
  • last reply

    Jan '20
  • 4

    replies

  • 1.1k

    views

  • 3

    users

  • 5

    links

Hmm, there are couple of things wrong :frowning:

The full class-name of the widget is according to https://github.com/asig2016/egroupware/blob/19.1_p1/api/src/Etemplate/Widget/Currency.php#L141
EGroupware\Api\Etemplate\Widget\currency (the c should be capital, but that does not matter AFAIK).
But you are not registering it and the commented call use the wrong class-name: https://github.com/asig2016/egroupware/blob/19.1_p1/api/src/Etemplate/Widget/Currency.php#L141
Best is to use Curreny::class to let PHP fill out the correct class name.

The javascript code of the jQuery plugin is included twice: https://github.com/asig2016/egroupware/blob/19.1_p1/api/js/etemplate/et2_widget_currency.js#L15
This might or might not work, I never tried.

We (@hadi.nategh, @ngegw) need to look into how an autoloaded widget from an app can load it’s JavaScript code. I expected Api\Framework::includeJS("/$app/$file"); in beforeSendToClient method of widget to work.

I also see no problem to add a currency widget to eT2 in general, or enhance our float widget to use the thousands-separator from our user-preferences.

Ralf

If everything else is right, it should be found and included automatically.

You may have to explicitly register it, eg:

// At the end of the file, after your widget class definition
et2_register_widget(Currency, [“currency”]);

The registry is firmly cached; it takes an hour to expire.

Nathan

Hello @RalfBecker,
thanks for your reply, it helped me search at the right place. The different class name currency2 at the commented out code was a desperate action while trying to find out why things where not working… Nevertheless I actually had made 2 similar mistakes:

  1. I had forgotten the “.inc.” letters in the filename of my class file, so it was not autoloaded.
  2. I made an error when using the Api\Framework::includeJS("/$app/$file") function you mentioned.

After that I can confirm that moving a widget to a custom application is straight forward. Let’s say

  1. the application is called “achelper”,
  2. the widget is called “currency”
  3. then the php class should be named achelper_widget_currency and the classfile class.achelper_widget_currency.inc.php
  4. the js file is named et2_widget_currency.js

Steps:

  1. In the setup.inc.php of the achelper application add the ‘etemplate2_register_widgets’ hook to a static function:

$setup_info[‘achelper’][‘hooks’][‘etemplate2_register_widgets’] = ‘achelper_hooks::widgets’;

  1. In the achelper_hooks class add a static function, in this case named widgets as defined in setup.inc.php returning the widget class:

/**
Returns a list of custom widgets classes for etemplate2
*/
public static function widgets()
{
return [‘achelper_widget_currency’];
}

  1. Put the js file ‘et2_widget_currency.js’ in the js folder of the achelper app:

/achelper/js/et2_widget_currency.js

  1. Include the js file in the constructor of the php class:

**
eTemplate currency textbox
Supports formatting for decimal & thousands, and currency prefix/suffix
*/
class achelper_widget_currency extends Etemplate\Widget
{
public function __construct($xml=’’) {

  Api\Framework::IncludeJS("/achelper/js/et2_widget_currency.js");
    parent::__construct($xml);

}

Including in the beforeSendToClient() function works also fine, I don’t know which is the better place…

Now my last question about this is how the et2_widget_currency.js can be included in the min.js files. I included it in the Gruntfile.js in the Egroupware root and ran “grunt” but my widget is still using the uncompressed version.

Thanks again everyone and Best Regards,
Alex

beforeSendToClient() make more sense, as it is only called when creating the dialog, not when submitting it.

Which .min.js file are you talking about?
etemplate2.min.js is only for the stock eT2 widgets.

What you can do is to bundle all JavaScript files with the app.js of your custom application, like we do it eg. for calendar or projectmanager, see the Gruntfile.js.

Ralf