Set the fill of a note to an external image

I try to find a way to use external images. I use a container with a note for each author I came across in my readings. I use to add an image and some additional attributes. If I use inline images (copied to the TB file) opening the container takes a while. So I would like to switch to external images. I used the $File attribute, markdown and this code:

^value($Name)^ *^value($BirthDate)^

to include the image in my note. I like to click on the folder icon of the $File attribute to set the path of the image. Very easy to work with.

The only thing I would like to see: the background of the note in map view should show my image and not the markdown code. $Fill doesnā€™t help. Any idea how to do it?

1 Like

If I use inline images (copied to the TB file) opening the container takes a while.

While a large image would make the file larger and might make it open more slowly, Iā€™m skeptical that it would have any effect on the time needed to zoom into the container. Are you sure?

Oh - yes - I have a container with around 50 authors - each one with a small image and it takes several seconds to open the map view of the container. I have containers with many text only notes and they open much faster. This is no big issue for me - I try to avoid images or the switch to an external image. For the export it seems to be the better way to go too - Iā€™m still not using the export feature but trying to find my way to curate (how Michael would call it) my notes with TB. Itā€™s fun to discover all the hidden gems of TB :wink:

Oh ā€“ the destination has multiple images.

Iā€™d not thought of that. Each of this images has to be unpacked and scaled to draw the map; that could be quite a bit of work.

but is there a way to show an external image in the map?

Iā€™d ask ā€œbut why are you doing thisā€? If you look at these notes on $File youā€™ll see it simply links to a file. If viewed in Displayed Attributes or Get Info:attributes/General you see a folder icon. If clicked, Tinderbox asks macOS finder [sic] to open that file with the associated app.

In contrast, you want the Fill attribute to use a file path. But, had you looked at details of $Fill
you wouls see it is a String-data-type attribute. Clearly the tw are not the same! :slight_smile:

My last is not meant as a snarky comment. Rather, Iā€™m truly confused as to why one would think the above method would work.

At present, Fill attribute can only use files in the app support ā€˜fillā€™ folder. I think changes are in hand (I write as at v9.0.0_) to allow macOS aliases placed there.

Either way, you can have a calculated $Fill value but the file targetted needs to be in the right place.

So for now set $Fill, noting that you can use action (not export) code to set the value of $Fill.

Yes; if you add the image to the Fill subfolder of the Tinderbox support folder, it will be available as a fill image just like the built-in fill textures such as Linen.

BTW, I do this all the time.

Personally, I donā€™t like story primary files in the applications fill directory. Also, for $Fill to work you canā€™t have the extension on the file name. And, for Hover to work you need the extension.

So, here is what I do.

I have an attribute $MediaFIleName. This is where the primary file name goes.

I have a TBXConfig note where I store my paths to my fiels.

I use Hazel. When I add a picture to my people or logo directory I have Haze move a copy to the TBX $Fill directory.

I then use a $Edict to populate the $Fill and $Hover attributes.

$Fill=$PathFIllLogo("TBXConfig")+$MediaFileName.replace("\.\w+$",""); 
$HoverImage=$PathHover("TBXConfig")+$PathFIllLogo("TBXConfig")+$MediaFileName;`

The result:
image

2 Likes

Easy to explain: I would like to use Markup to include an external image. I donā€™t want to manually copy the path in the Finder and paste it into the note. So I simply use the $File attribute. With an empty $File attribute I can click in the folder icon and select a file. The result is a string with the path to this file and this value is what I use in the markup. Very easy to integrate an external image that way. No need to copy&paste the path.

Now I expected that the path I got from the $File attribute could be used in the $Fill attribute too. Both attributes use strings. The limitation here is the special ā€œfillā€ folder.

@satikusala thatā€™s a nice approach. Will try this too. Did you compare the speed if you have a container with many notes containing an image as fill pattern to a collection of notes using inline images?

3 Likes

@satikusala I just tested the performance: the external $Fill images are much faster. I tested with a folder containing 40 notes each with an image. With inline images opening the folder takes several seconds, with the external images the folder opens in less then a second (map view). Great. Thatā€™s the way I will go.

I donā€™t understand why you store the path to the files? Since $Fill needs the name of the image file only (without the extension). No path needed here - or did I miss something (ā€œJacques_Derridaā€ works - no need for ā€œ/Users/sampleuser/Library/Application Support/Tinderbox/fill/acques_Derridaā€)?!

And I still like to use $File as my transport media for the file name :wink:

Great. Glad it works for you.

I have a sub-folder in $Fill for people and a second one for Logos, i.e. for hover. The path for people points to people, and to get Hover to work I find the full path is the most reliable.

Hi @webline, thanks for your note. Iā€™ve found that format and size matter. You need png or jpeg. Also, if the file is too large it wonā€™t display. I donā€™t know what the limits are.

Itā€™s getting even more ā€œspecialā€ here - sorry. I just try to understand some concepts of TBX and use them for my work.
The external path for $Fill works fine now. The next thing Iā€™m trying to solve: after I change the $File attribute I would like to update the $Fill attribute with the name of the file.

This stamp ā€œSetFillImageFromFileā€ will do it:

stamp("GetFileNameOnly");
stamp("GetFileNameNoExtension");
$Fill(original)="";
$Fill(original)=$PathFillImages("pPreferences")+$ImageName(original);
$FileIsModified=false;

The first two stamps just strip the path and the type ending from the value stored in $File.
The line with $Fill(original)=""; is there because of my new issue I run into.

First step was an $Edict with:

if(!$IsPrototype){stamp("SetFillImageFromFile")}

This worked but it seems to me that this will add some overhead to my file. I want to apply the stamp only once if the value of $File has changed. It got even worse when using it as a $Rule.

So I created a new attribute called $FileIsModified and an agent with this query:

$Prototype=="pPerson" & $File=="" & $FileIsModified==false

and a simple $AgentAction: $FileIsModified=true;

Works fine - there is a delay because the Agent will not run immediately. But thatā€™s not my problem now.

Then I created a second Agent with this query:

$Prototype=="pPerson" & $File!="" & $FileIsModified==true

and the Action:

stamp("SetFillImageFromFile");$FileIsModified=false;

Now - after so many words :wink: - here is my problem: everything works. Only the $Fill attribute never gets set to the new value. The other three attributes will be set, but not $Fill. If I apply the stamp manually to the note all attributes were set including $Fill.

Any idea? All I need is a way to simulate an ā€œOnChangeā€ event for the $Fill attribute.

Is there a demo file for this? I see all the code, but running it my head, Iā€™m getting lost. :slight_smile:

One way to resolve your confusion re images is to look at this section of aTbRef, re support foldersā€”itā€™s a quick read to scan through>. You will note that storage for only two forms of image use are indicated: fills and badges.

Iā€™m not sure why these are internal and others are external; thatā€™s a question for Eastgate but Iā€™d hazard that easy re-use across many docs and possibly efficiency may be factors.

I do think that while, especially to old Mac users, in 2001 using filenames with no file extension would have not been confusing, in 2021 the position is reversed. This will be especially true for those not from a programming background. For a process to fail due to provision of a file extension for a file asset ( given the extension can easily be detected and parsed out), I sense a change is due here.

Another reflection of incremental app improvement over the years is that I donā€™t think there is documentation of what Tinderbox process can use image data, how they are specified and any format restrictions. Perhaps it is time to add this to Help along with a list of Application Support folders (default and optional). A thought, anyway :slight_smile:

@mwra here is a demo file. I solved my problem with another flag. With this flag I trigger an $Edict that set the $Fill attribute. Seems to be some kind of racing condition with the $Fillā€¦
To test my demo you need two folders with (small) image files. Both with the same images. One in the ā€œfillā€ folder of TBX the other one (#2) anywhere on your Mac. Use the $File folder icon to select any file from the folder #2 and wait. The image of the note should update after the agents ran. Then clean the $File attribute (empty) wait until the agents run and the image should disappear.

PersonsExternalImages.tbx (96.8 KB)

P.S: create new notes in the persons folder. Add the $File to this note

1 Like

Doesnā€™t work for me and the app crashes if I try to open the Inspector. In fact every TBX now crashes. Loks like Iā€™ll have to re-boot in order to get Tinderbox to work again, but unfortunately I canā€™t re-boot tonight as Iā€™m in the middle of running some long data scripts (non-Tinderbox work).

Tinderbox still unstable (gremlin in your file but got it open again) What is the real-world actual problem we are trying to solve here? I still canā€™t figure it out.

Meanwhile, stamps are useful for testing things but I doubt they were intended to be called inlined in other actions. So Iā€™ve fixed the action code to remove the inline stamp calls. I also noted that several notes have no title ($Name). See here: PersonsExternalImages-fix.tbx (101.8 KB)

Take a look at that and see if it works better (at least not crashing all the time!)

Also, although not mentioned, there seems to be an intent to use a ā€˜peopleā€™ sub-folder of fills. But Iā€™m still too lost to make much more progress.

@mwra if have no crashes at all with the file. Never had!? There are some notes with old data - referring to the external images on my Mac. Maybe thatā€™s the reason for the crashes? Here everything is rock solid. Sorry if the file gives you trouble with your Mac.

You didnā€™t get the idea behind my implementation (donā€™t have to be a bad thingā€¦). If you put the code into an $Edict - it will run all the time updating the attributes over and over again and slowing down TBX. Thatā€™s the reason for my use of the agents. The way you did it was my first approach but it slows down the file.

I donā€™t want my code in one box - if I canā€™t define my own methods and reuse them ā†’ stamps are my way to go. If there is no other way to reuse the code - why not use stamps? They do work fine.

Yes - in the fill folder of TBX there is a sub-folder ā€œpeopleā€ with the images. And yes - itā€™s a file to test stuff so there are incomplete notes and other things not finished.

Your solution is incomplete. If you empty the $File attribute and then load a new file - the $Edict will not run because $NewImageLoad=false is still set.

using only code in the $Edict will do it too - like this:

if($File=="" & $FileIsModified==false) {
  $FileIsModified=true;
  $Fill="";
} else {
  if($File!="" & $FileIsModified==true) {
    $ImageFileName=$File.replace(".*\/(.+)","$1");
    $ImageName=$ImageFileName.replace("\.\w+$","");
    $Fill(original)=$PathFillImages("/pPreferences")+$ImageName;
    $FileIsModified=false;
    $NewImageLoad=true;
  } else {
    if($NewImageLoad) {
     $Fill(original)=$PathFillImages("/pPreferences")+$ImageName;
     $NewImageLoad=false;
    }
  }
}

Easier to maintain then using the two agents and the $Edict code. Will go this way.