IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)
Navigation

Inscrivez-vous gratuitement
pour pouvoir participer, suivre les réponses en temps réel, voter pour les messages, poser vos propres questions et recevoir la newsletter

  1. #1
    Membre régulier
    Homme Profil pro
    Chef de projet NTIC
    Inscrit en
    Avril 2022
    Messages
    37
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Ille et Vilaine (Bretagne)

    Informations professionnelles :
    Activité : Chef de projet NTIC
    Secteur : Industrie

    Informations forums :
    Inscription : Avril 2022
    Messages : 37
    Points : 79
    Points
    79
    Par défaut Utilisation inter-activités d'une Snackbar(à la manière d'un Toast, sur son lit de patates douces émincées)
    Bonjour à tous ! Aujourd'hui, recette !

    Pour sa réalisation, il vous faudra :
    - Android 11+
    - un Wrapper de SnackBar (fourni)
    - Un thème de SnackBar (fourni)
    - Des messages à afficher(sous forme de SnackBar)
    - Des patates douces

    Maintenant que l'on a tous les ingrédients, on peut commencer !(pour de vrai)
    Je cherche à utiliser les Snackbar un peut comme sont utilisés les Toast.makeText() : ceux si s'affichent dans le contexte de l'application, et non de l'activité. En découle le fait que lors d'un changement d'activité(via StartActivity ou simplement le fait de quitter l'application), le Toast reste affiché est n'est pas limité au contenant qui l'a lancé(En l'occurence, la plupart du temps, une activité).
    C'est fort utile car ça permet de donner un feedback à l'utilisateur en même temps que l'on change de page, MAIS les Toasts ne sont plus modifiables depuis quelques versions. Donc c'est un rectangle ovoïdal, gris neutre, avec couleur de police d'écriture noire peu visible.

    Je cherche à utiliser la personnalisation du SnackBar en recréant la fonctionnalité de persistance du Toast, j'ai donc après quelques recherches, appliqué une solution proposée sur un forum Anglophone, à savoir utiliser un snackbar wrapper :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92
    93
    94
    95
    96
    97
    98
    99
    100
    101
    102
    103
    104
    105
    106
    107
    108
    109
    110
    111
    112
    113
    114
    115
    116
    117
    118
    119
    120
    121
    122
    123
    124
    125
    126
    127
    128
    129
    130
    131
    132
    133
    134
    135
    136
    137
    138
    139
    140
    141
    142
    143
    144
    145
    146
    147
    148
    149
    150
    151
    152
    153
    package com.bic.btt_prototype;
     
    import static com.bic.btt_prototype.ToolBox.iLog;
     
    import android.content.Context;
    import android.graphics.PixelFormat;
    import android.os.IBinder;
    import android.view.ContextThemeWrapper;
    import android.view.Gravity;
    import android.view.View;
    import android.view.WindowManager;
    import android.widget.FrameLayout;
     
    import androidx.annotation.NonNull;
    import androidx.annotation.Nullable;
    import androidx.coordinatorlayout.widget.CoordinatorLayout;
    import androidx.core.view.GravityCompat;
    import androidx.core.view.ViewCompat;
     
    import com.google.android.material.snackbar.Snackbar;
     
    public class snackbarwrapper
    {
        private final CharSequence text;
        private final int duration;
        private final WindowManager windowManager;
        private final Context appplicationContext;
        @Nullable
        private Snackbar.Callback externalCallback;
        @Nullable
        private Action action;
     
        @NonNull
        public static snackbarwrapper make(@NonNull Context applicationContext, @NonNull CharSequence text, @Snackbar.Duration int duration)
        {
            iLog("snackbar.make : ");
            return new snackbarwrapper(applicationContext, text, duration);
        }
     
        private snackbarwrapper(@NonNull final Context appplicationContext, @NonNull CharSequence text, @Snackbar.Duration int duration)
        {
            this.appplicationContext = appplicationContext;
            this.windowManager = (WindowManager) appplicationContext.getSystemService(Context.WINDOW_SERVICE);
            this.text = text;
            this.duration = duration;
        }
     
        public void show()
        {
            WindowManager.LayoutParams layoutParams = createDefaultLayoutParams(WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY, null);
            windowManager.addView(new FrameLayout(appplicationContext)
            {
                @Override
                protected void onAttachedToWindow()
                {
                    super.onAttachedToWindow();
                    onRootViewAvailable(this);
                }
     
            }, layoutParams);
        }
     
        private void onRootViewAvailable(final FrameLayout rootView)
        {
            final CoordinatorLayout snackbarContainer = new CoordinatorLayout(new ContextThemeWrapper(appplicationContext, R.style.FOL_Theme_SnackbarWrapper))
            {
                @Override
                public void onAttachedToWindow()
                {
                    super.onAttachedToWindow();
                    onSnackbarContainerAttached(rootView, this);
                }
            };
            windowManager.addView(snackbarContainer, createDefaultLayoutParams(WindowManager.LayoutParams.TYPE_APPLICATION_PANEL, rootView.getWindowToken()));
        }
     
        private void onSnackbarContainerAttached(final View rootView, final CoordinatorLayout snackbarContainer)
        {
            Snackbar snackbar = Snackbar.make(snackbarContainer, text, duration);
            snackbar.setCallback(new Snackbar.Callback()
            {
                @Override
                public void onDismissed(Snackbar snackbar, int event)
                {
                    super.onDismissed(snackbar, event);
                    // Clean up (NOTE! This callback can be called multiple times)
                    if (snackbarContainer.getParent() != null && rootView.getParent() != null)
                    {
                        windowManager.removeView(snackbarContainer);
                        windowManager.removeView(rootView);
                    }
                    if (externalCallback != null)
                    {
                        externalCallback.onDismissed(snackbar, event);
                    }
                }
     
                @Override
                public void onShown(Snackbar snackbar)
                {
                    super.onShown(snackbar);
                    if (externalCallback != null)
                    {
                        externalCallback.onShown(snackbar);
                    }
                }
            });
            if (action != null)
            {
                snackbar.setAction(action.text, action.listener);
            }
            snackbar.show();
        }
     
        private WindowManager.LayoutParams createDefaultLayoutParams(int type, @Nullable IBinder windowToken)
        {
            WindowManager.LayoutParams layoutParams = new WindowManager.LayoutParams();
            layoutParams.format = PixelFormat.TRANSLUCENT;
            layoutParams.width = WindowManager.LayoutParams.MATCH_PARENT;
            layoutParams.height = WindowManager.LayoutParams.WRAP_CONTENT;
            layoutParams.gravity = GravityCompat.getAbsoluteGravity(Gravity.CENTER_HORIZONTAL | Gravity.BOTTOM, ViewCompat.LAYOUT_DIRECTION_LTR);
            layoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL;
            layoutParams.type = type;
            layoutParams.token = windowToken;
            return layoutParams;
        }
     
        @NonNull
        public snackbarwrapper setCallback(@Nullable Snackbar.Callback callback)
        {
            this.externalCallback = callback;
            return this;
        }
     
        @NonNull
        public snackbarwrapper setAction(CharSequence text, final View.OnClickListener listener)
        {
            action = new Action(text, listener);
            return this;
        }
     
        private static class Action
        {
            private final CharSequence text;
            private final View.OnClickListener listener;
     
            public Action(CharSequence text, View.OnClickListener listener)
            {
                this.text = text;
                this.listener = listener;
            }
        }
    }
    Le thème de la bar étant :
    Code XML : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
        <style name="FOL_Theme_SnackbarWrapper" parent="@style/Theme.AppCompat">
            <item name="android:textSize">19sp</item>
            <item name="android:textColor">@color/white</item>
            <item name="android:colorBackground">@color/greencheck</item>
        </style>


    Cette snackbar me retourne une belle erreur à l'affichage :
    Caused by: android.view.WindowManager$BadTokenException: Unable to add window android.view.ViewRootImpl$W@7062ad -- permission denied for window type 2038
    Et ceci pour la ligne
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
     windowManager.addView(new FrameLayout(appplicationContext)
    Édition de 14h20 :
    J'ai essayé de rajouter
    Code XML : Sélectionner tout - Visualiser dans une fenêtre à part
    <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/>
    dans le manifeste mais cela ne change rien.

    Quelqu'un aurait une petite idée ?

    (avec les patates douces, coupez les en frites de 0.8/0.8, cuisson à l'eau pendant 10/15 min, tremper dans l'amidon(Type Maizena) et ensuite passage au four ! L'amidon permet de les rendre croustillantes. Ne pas oublier le filet d’huile ! Sortir du four une fois bien dorées)

    Merci d'avance !

    Phae

  2. #2
    Membre régulier
    Homme Profil pro
    Chef de projet NTIC
    Inscrit en
    Avril 2022
    Messages
    37
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Ille et Vilaine (Bretagne)

    Informations professionnelles :
    Activité : Chef de projet NTIC
    Secteur : Industrie

    Informations forums :
    Inscription : Avril 2022
    Messages : 37
    Points : 79
    Points
    79
    Par défaut
    J'espère que vous avez apprécié la recette !

    En tout cas, finalement cette erreur venait du fait qu'il est désormais nécessaire d'autoriser l'application à s'afficher en superposition aux autres. J'ai donc rajouté ceci sur mon splashscreen :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
     
    // on vérifie si le paramètre est actif ou pas
            if (!Settings.canDrawOverlays(this)) {
    // nouvelle SnackBar classique
                Snackbar snackbar = Snackbar
                        .make(mContentView, "AUTORISER L'AFFICHAGE EN SUPERPOSITION", 5000)
                        .setAction("AUTORISER", v -> {
    // on change d'app pour les réglages du devicew
                            Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION, Uri.parse("package:" + getPackageName()));
                            startActivity(intent);
                        });
    // on affiche le tout
                snackbar.show();
            }
    Et ça marche pas dégueu !

    Bon, ma SnackBar est pas positionnée au bon endroit, mais c'est un autre problème cette fois !


    Bon appétit

    Phae

Discussions similaires

  1. utilisation des icons d'une manière dynamique
    Par LandGreen dans le forum Delphi
    Réponses: 9
    Dernier message: 15/02/2022, 12h17
  2. Empêcher l'utilisation des cookies d'une manière global en js
    Par Mzalboul dans le forum Général JavaScript
    Réponses: 0
    Dernier message: 22/02/2020, 09h30
  3. Réponses: 1
    Dernier message: 04/07/2018, 16h57
  4. Utiliser l'instance d'une activité à partir d'une autre classe
    Par Heavy Metal Hero dans le forum Android
    Réponses: 4
    Dernier message: 17/10/2013, 17h45
  5. [XL-2003] Modifier une macro pour qu'elle s'exécute sur son propre fichier
    Par Vadorblanc dans le forum Macros et VBA Excel
    Réponses: 2
    Dernier message: 20/03/2013, 19h21

Partager

Partager
  • Envoyer la discussion sur Viadeo
  • Envoyer la discussion sur Twitter
  • Envoyer la discussion sur Google
  • Envoyer la discussion sur Facebook
  • Envoyer la discussion sur Digg
  • Envoyer la discussion sur Delicious
  • Envoyer la discussion sur MySpace
  • Envoyer la discussion sur Yahoo